我有一个运行着几个 node.js 项目的根服务器。它们应该在自己的进程和目录中单独运行。考虑这个文件结构:
/home
+-- /node
+-- /someProject | www.some-project.com
| +-- index.js
| +-- anotherFile.img
| +-- ...
+-- /anotherProject | www.another-project.com
| +-- /stuff
| +-- index.js
| +-- ...
+-- /myWebsite | www.my-website.com
| +-- /static
| +-- index.js
| +-- ...
+-- ... | ...
每个 index.js 都应该作为一个单独的进程启动,并将其cwd
设置为其父文件夹(someProject
、anotherProject
等)。
想想 ov 虚拟主机。每个项目启动一个网络服务器,它在自己的域上进行侦听。这就是问题所在。只有一个脚本可以启动,因为它们都尝试绑定到端口 80。我深入研究了 node.js API 并寻找可能的解决方案:child_process.fork()
.
可悲的是,这不是很好。当我尝试将服务器实例发送到主进程(稍后发出请求)或包含主进程request
和response
从主进程到从属进程的对象时,我得到了错误。这是因为 node.js 在内部尝试将这些高级对象转换为 JSON 字符串,然后将其重新转换为其原始形式。这使得所有对象都失去了它们的引用和功能。
第二种方法child.js
var http = require("http");
var server = http.createServer(function(req, res) {
// stuff...
});
server.listen(80);
process.send(server); // Nope
第一种方法master.js
var http = require("http"),
cp = require("child_process");
var child = cp.fork("/home/node/someProject/index.js", [], { env: "/home/node/someProject" });
var router = http.createServer(function(req, res) {
// domaincheck, etc...
child.send({ request: req, response: res }); // Nope
});
router.listen(80);
所以这是一个死胡同。但是,嘿!Node.js 提供了一些可发送的句柄。这是文档中的一个示例:
master.js
var server = require('net').createServer();
var child = require('child_process').fork(__dirname + '/child.js');
// Open up the server object and send the handle.
server.listen(1337, function() {
child.send({ server: true }, server._handle);
});
child.js
process.on('message', function(m, serverHandle) {
if (serverHandle) {
var server = require('net').createServer();
server.listen(serverHandle);
}
});
这里孩子直接监听主服务器。所以中间没有域检查。所以这里是一个死胡同。
我也想过Cluster
,但这使用的技术与手柄相同,因此具有相同的局限性。
那么……有什么好主意吗?
我目前所做的是相当hack-ish。我制作了一个名为distroy的包。它绑定到端口 80 并在内部代理所有请求到 Unix 域套接字路径,例如/tmp/distroy/http/www.example.com
,单独的应用程序在其上侦听。这也(有点)适用于 HTTPS(请参阅我关于SNI的问题)。剩下的问题是,原来的 IP 地址丢失了,因为它现在总是 127.0.0.1。我想我可以通过monkeypatching来规避这个问题,net.Server
这样我就可以在打开连接之前传输IP地址。