如何在 js 文件中 运行 守护进程
How to run daemon within a js file
我正在尝试从我的 js 代码中启动服务器守护程序,然后从同一程序访问它。但是,当我使用 child_process 模块中的 execFile()
方法时,这会阻塞整个程序:execFile()
调用永远不会停止,我无法访问服务器。但是,我知道守护进程已启动,因为我看到一个同名进程从我的 activity 监视器(macos 相当于任务管理器)启动。
我也试过来自同一模块的 exec()
和 spawn()
,结果相同。
我想要做的是将守护程序作为一个单独的进程启动,忘记它,然后在我使用完它时停止它。有什么办法至少可以做到前两个吗?
这是我的代码(runArduino
函数是我启动守护进程的地方,main()
函数是我访问它的地方):
const grpcLib = require('grpc');
const protoLoader = require('@grpc/proto-loader');
const pathLib = require("path");
const utilLib = require('util');
const exec = utilLib.promisify(require('child_process').execFile);
const RPC_PATH = pathLib.join(__dirname, "arduino-cli/rpc")
var PROTO_PATH = pathLib.join(RPC_PATH, "/commands/commands.proto");
const options = {
keepCase: true,
longs: String,
enums: String,
defaults: true,
oneofs: true,
includeDirs:
[
RPC_PATH
]
}
const packageDefinition = protoLoader.loadSync(PROTO_PATH, options);
const arduinoCli = grpcLib.loadPackageDefinition(packageDefinition).cc.arduino.cli.commands;
function runArduino()
{
exec(__dirname+"/arduino-cli_macos", ['daemon'],function(err, data)
{
console.log(err);
console.log(data);
});
}
function main()
{
var client = new arduinoCli.ArduinoCore('localhost:50051', grpcLib.credentials.createInsecure());
client.Version({}, function(err, response){
console.log("Running version: ", response); //returns a version number
});
}
runArduino();
main();
我第一次运行它,这是我得到的(执行不会停止):
Running version: undefined
一旦守护进程启动并且我 运行 它,我得到这个(我现在可以访问服务器并且执行结束):
Running version: { version: '0.11.0' }
Error: Command failed: /Users/Herve/Desktop/MyStuff/ArduinoX/ArduinoX/arduino-cli_macos daemon
Failed to listen on TCP port: 50051. Address already in use.
at ChildProcess.exithandler (child_process.js:303:12)
at ChildProcess.emit (events.js:315:20)
at maybeClose (internal/child_process.js:1021:16)
at Socket.<anonymous> (internal/child_process.js:443:11)
at Socket.emit (events.js:315:20)
at Pipe.<anonymous> (net.js:674:12) {
killed: false,
code: 5,
signal: null,
cmd: '/Users/Herve/Desktop/MyStuff/ArduinoX/ArduinoX/arduino-cli_macos daemon'
}
我相信你应该 await
exec 或 运行 main()
在 exec 的回调中。现在您的 main()
在子进程启动之前执行。
exec(__dirname+"/arduino-cli_macos", ['daemon'],function(err, data)
{
console.log(err);
console.log(data);
});
这就是为什么在第一个 运行 你得到 undefined
。子进程不会自动终止,我猜,这就是为什么在第二个 运行 你可以执行 RPC,但不能真正再次启动子进程,因为它已经 运行ning(并且正在占用50051 端口)。
如果您的应用程序正在启动子进程,我相信它也必须注意杀死它:
var childProcessHandler = exec(__dirname+"/arduino-cli_macos", ['daemon'],function(err, data)
{
console.log(err);
console.log(data);
});
// ... and later in your code:
childProcessHandler.kill()
这样您就可以 start/stop 应用程序而不必担心清理进程。您唯一需要考虑的是在发生异常时进行清理。
编辑 好的,看来要将进程作为守护进程启动,您必须使用 spaw
和 detached
选项:
const { spawn } = require('child_process');
const child = spawn(__dirname+"/arduino-cli_macos", {
detached: true
});
child.unref();
我正在尝试从我的 js 代码中启动服务器守护程序,然后从同一程序访问它。但是,当我使用 child_process 模块中的 execFile()
方法时,这会阻塞整个程序:execFile()
调用永远不会停止,我无法访问服务器。但是,我知道守护进程已启动,因为我看到一个同名进程从我的 activity 监视器(macos 相当于任务管理器)启动。
我也试过来自同一模块的 exec()
和 spawn()
,结果相同。
我想要做的是将守护程序作为一个单独的进程启动,忘记它,然后在我使用完它时停止它。有什么办法至少可以做到前两个吗?
这是我的代码(runArduino
函数是我启动守护进程的地方,main()
函数是我访问它的地方):
const grpcLib = require('grpc');
const protoLoader = require('@grpc/proto-loader');
const pathLib = require("path");
const utilLib = require('util');
const exec = utilLib.promisify(require('child_process').execFile);
const RPC_PATH = pathLib.join(__dirname, "arduino-cli/rpc")
var PROTO_PATH = pathLib.join(RPC_PATH, "/commands/commands.proto");
const options = {
keepCase: true,
longs: String,
enums: String,
defaults: true,
oneofs: true,
includeDirs:
[
RPC_PATH
]
}
const packageDefinition = protoLoader.loadSync(PROTO_PATH, options);
const arduinoCli = grpcLib.loadPackageDefinition(packageDefinition).cc.arduino.cli.commands;
function runArduino()
{
exec(__dirname+"/arduino-cli_macos", ['daemon'],function(err, data)
{
console.log(err);
console.log(data);
});
}
function main()
{
var client = new arduinoCli.ArduinoCore('localhost:50051', grpcLib.credentials.createInsecure());
client.Version({}, function(err, response){
console.log("Running version: ", response); //returns a version number
});
}
runArduino();
main();
我第一次运行它,这是我得到的(执行不会停止):
Running version: undefined
一旦守护进程启动并且我 运行 它,我得到这个(我现在可以访问服务器并且执行结束):
Running version: { version: '0.11.0' }
Error: Command failed: /Users/Herve/Desktop/MyStuff/ArduinoX/ArduinoX/arduino-cli_macos daemon
Failed to listen on TCP port: 50051. Address already in use.
at ChildProcess.exithandler (child_process.js:303:12)
at ChildProcess.emit (events.js:315:20)
at maybeClose (internal/child_process.js:1021:16)
at Socket.<anonymous> (internal/child_process.js:443:11)
at Socket.emit (events.js:315:20)
at Pipe.<anonymous> (net.js:674:12) {
killed: false,
code: 5,
signal: null,
cmd: '/Users/Herve/Desktop/MyStuff/ArduinoX/ArduinoX/arduino-cli_macos daemon'
}
我相信你应该 await
exec 或 运行 main()
在 exec 的回调中。现在您的 main()
在子进程启动之前执行。
exec(__dirname+"/arduino-cli_macos", ['daemon'],function(err, data)
{
console.log(err);
console.log(data);
});
这就是为什么在第一个 运行 你得到 undefined
。子进程不会自动终止,我猜,这就是为什么在第二个 运行 你可以执行 RPC,但不能真正再次启动子进程,因为它已经 运行ning(并且正在占用50051 端口)。
如果您的应用程序正在启动子进程,我相信它也必须注意杀死它:
var childProcessHandler = exec(__dirname+"/arduino-cli_macos", ['daemon'],function(err, data)
{
console.log(err);
console.log(data);
});
// ... and later in your code:
childProcessHandler.kill()
这样您就可以 start/stop 应用程序而不必担心清理进程。您唯一需要考虑的是在发生异常时进行清理。
编辑 好的,看来要将进程作为守护进程启动,您必须使用 spaw
和 detached
选项:
const { spawn } = require('child_process');
const child = spawn(__dirname+"/arduino-cli_macos", {
detached: true
});
child.unref();