我有一个运行着几个 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
等.).
考虑虚拟主机。每个项目启动一个网络服务器,它在自己的域上进行监听。这就是问题所在。只有一个脚本可以启动,因为它们都尝试绑定(bind)到端口 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);
}
});
这里child直接监听master的服务器。所以中间没有域检查。所以这里是一个死胡同。
我也想过Cluster
,但它使用与 handle 相同的技术,因此具有相同的限制。
那么……有什么好主意吗?
我目前做的是相当 hack-ish。我制作了一个名为 distroy 的包.它绑定(bind)到端口 80 并在内部代理所有请求到 Unix 域套接字路径,如 /tmp/distroy/http/www.example.com
,单独的应用程序在这些路径上监听。这也(有点)适用于 HTTPS(请参阅我关于 SNI 的问题)。
剩下的问题是,原来的 IP 地址丢失了,因为它现在总是 127.0.0.1。我想我可以通过猴子修补 net.Server
来规避这个问题,这样我就可以在打开连接之前传输 IP 地址。
最佳答案
如果您对 node.js 解决方案感兴趣,请查看 bouncy ,node.js 中的一个 websocket 和支持 https 的 http 路由器代理/负载均衡器。
像这样定义你的 routes.json
{
"beep.example.com" : 8000,
"boop.example.com" : 8001
}
然后使用
运行 bouncy bouncy routes.json 80
关于linux - 在多个 node.js HTTP 进程之间共享一个端口,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/10791309/