16

我有一个运行着几个 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设置为其父文件夹(someProjectanotherProject等)。

想想 ov 虚拟主机。每个项目启动一个网络服务器,它在自己的域上进行侦听。这就是问题所在。只有一个脚本可以启动,因为它们都尝试绑定到端口 80。我深入研究了 node.js API 并寻找可能的解决方案:child_process.fork().

可悲的是,这不是很好。当我尝试将服务器实例发送到主进程(稍后发出请求)或包含主进程requestresponse从主进程到从属进程的对象时,我得到了错误。这是因为 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地址。

4

3 回答 3

9

如果您对 node.js 解决方案感兴趣,请查看 node.js 中的bouncy、websocket 和支持 https 的 http 路由器代理/负载平衡器。

定义你的 routes.json 像

 {
      "beep.example.com" : 8000,
      "boop.example.com" : 8001
 }

然后使用

 bouncy routes.json 80
于 2012-05-29T01:35:50.437 回答
1

就个人而言,我只是让他们都在专用端口或最好是套接字上监听,然后将所有内容都放在专用路由器脚本或 nginx 后面。这是 IMO 最简单的方法。

于 2012-05-28T23:14:26.890 回答
0

对于连接中间件,有vhost扩展。也许你可以复制他们的一些概念。

于 2012-05-29T04:02:41.640 回答