一尘不染

我应该如何从Java应用程序运行NodeJS?

node.js

我正在写一个Java库,实际上是一个Clojure库,但是对于这个问题,重要的是它在JVM上运行。该库需要执行一些JavaScript。我尝试过Nashorn,但遇到了一些局限性,可能难以克服。另外,我想尝试NodeJS。

我希望我的库是独立的,不依赖于独立运行NodeJS的系统,因此需要一种特殊的部署机制将Java和NodeJS工件放置在正确的位置,以便由两个不同的网络服务器接收。但是,这种方法带来了一些问题。

我将通过HTTP与NodeJS交谈,但我不希望NodeJS打开特定端口。我想找到一个随机的未使用的,所以没有冲突。我还想控制来自NodeJS的日志的去向,以使其与我的应用程序的其余部分保持一致。最后,我的应用程序应该能够检测NodeJS崩溃的时间并重新运行它,或者报告信息错误。

解决此问题的最佳方法是什么?是否有任何Java库以这种方式帮助管理子进程?我应该从NodeJS方面做任何特别的事情(我对NodeJS还是很陌生,以前从未使用过)。


阅读 694

收藏
2020-07-07

共1个答案

一尘不染

最后,我的解决方案是像这样使用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.")))))
2020-07-07