我正在写一个Java库,实际上是一个Clojure库,但是对于这个问题,重要的是它在JVM上运行。该库需要执行一些JavaScript。我尝试过Nashorn,但遇到了一些局限性,可能难以克服。另外,我想尝试NodeJS。
我希望我的库是独立的,不依赖于独立运行NodeJS的系统,因此需要一种特殊的部署机制将Java和NodeJS工件放置在正确的位置,以便由两个不同的网络服务器接收。但是,这种方法带来了一些问题。
我将通过HTTP与NodeJS交谈,但我不希望NodeJS打开特定端口。我想找到一个随机的未使用的,所以没有冲突。我还想控制来自NodeJS的日志的去向,以使其与我的应用程序的其余部分保持一致。最后,我的应用程序应该能够检测NodeJS崩溃的时间并重新运行它,或者报告信息错误。
解决此问题的最佳方法是什么?是否有任何Java库以这种方式帮助管理子进程?我应该从NodeJS方面做任何特别的事情(我对NodeJS还是很陌生,以前从未使用过)。
最后,我的解决方案是像这样使用ProcessBuilder:
(defn create-process-builder [js-engine] (doto (ProcessBuilder. ["node" (:path js-engine) "--port-file" (:port-file js-engine) "--default-ajax-host" (:default-ajax-host js-engine) "--default-ajax-port" (str (:default-ajax-port js-engine))]) .inheritIO))
然后在其中调用start。initializeIO使它的输出转到当前进程的输出,该进程有效地合并了stdout和stderr。
最重要的是,NodeJS通过指定0作为端口号来打开随机端口,并将其写入文件:
(let [app (-> (express) (.use (cookie-parser)) (.get "/" (fn [_req res] (.send res "Universal JavaScript engine for server side pre-rendering single page applications."))) (.get "/render" render)) server (.createServer http app)] (.listen server 0 (fn [] (.writeFile fs (:port-file options) (.-port (.address server)))))))))
然后由Java端打开(等待它出现):
(defn get-port-number [js-engine] (or (with-timeout (:start-timeout js-engine) (loop [port-number (read-port-file js-engine)] (if (string/blank? port-number) (if (is-running? js-engine) (do (Thread/sleep 100) (recur (read-port-file js-engine))) (throw (Exception. (str "While waiting for port number, process died: " (:path js-engine))))) port-number))) (throw (Exception. (str "Waited for " (:start-timeout js-engine) " for " (:path js-engine) " to start and report its port number but it timed out.")))))