6

我正在使用http.Server默认http模块的对象(-> API)。

如果客户端连接,request则服务器发出事件,并将http.ServerRequest对象传递给事件处理程序。该请求对象发出以下三个事件:

  • 数据(传入数据)
  • 结束(请求结束)
  • 关闭(连接关闭)

我想保留客户端的初始(TCP)连接并将其重用于多个请求。客户端在发送“Connection: keep-alive”标头时支持此行为。

但是我如何使用 node.js 实现这一点?我的问题是,在第一个请求之后发出结束事件,就像官方 API 所说的那样:

每个请求只发出一次。之后,将不再在请求中发出“数据”事件。

好的,我不能使用那个请求。但是有什么方法可以在同一个 TCP 连接上工作并在其上创建一个新请求?

背景:我正在研究代理实现。如果用户访问使用NTLM的网站,我必须坚持一个连接,以确保该类型身份验证所需的至少三个客户端请求不会遇到问题。

你知道我可以尝试什么吗?

提前致谢!

4

1 回答 1

0

当我使用节点 v6.17 运行此代码时,它表明保持活动状态不起作用

function notifyNotKeptAlive(httpObject, eventName, httpObjectName){
    httpObject.on(eventName, function(socket){
        socket.on('close', function(){
                console.log("Yeeouch !  "+httpObjectName+" connection was not kept alive!");
        });
    });
}
var http=require('http');
var count=1;
var srv = http.createServer(function (req, res) {
    var secs;
    if ( (count%2)==1){secs=3;}
    else {secs=1;}
    console.log("Got http request #"+count+" so server waits "+secs+" seconds to respond");
    var countForThisClosure=count;
    setTimeout(function(){
        console.log("Waking up from sleep of "+secs);
        res.writeHead(200, {'Content-Type': 'text/plain'});
        if (secs===3){
            res.end('Visit #'+countForThisClosure+' costing a wait of '+secs+' seconds !!');
        } else {
            res.end('Visit #'+countForThisClosure+' costing a wait of '+secs+' seconds !!');
        }
    }, secs*1000);
    count++;
}).listen(8090);

notifyNotKeptAlive(srv, 'connection', 'http server');

for (var i=0;i<2;i++){
    var req=http.request( {'host': 'localhost', 'port':8090}, function (res) {
        res.setEncoding('utf8');
        res.on('data', function (chunk) {
            console.log('INCOMING <---' + chunk);
        });
        }
    );
    notifyNotKeptAlive(req, 'socket', 'http client');
    req.end();
}

仅当我在 Epeli 提供的 http lib 源代码的第 1263 行添加了一个感叹号时,它才起作用。因此,下面的行“if (req.shouldKeepAlive){”应该改为“if (!req.shouldKeepALive){” res.on('end', function() {

if (req.shouldKeepAlive) {
      socket.removeListener('close', closeListener);
      socket.removeListener('error', errorListener);
      socket.emit('free');
    }
  });

此行位于从第 1113 行开始的 ClientRequest.prototype.onSocket 闭包中设置的 on 'end' 回调中。

此 http lib 源的链接是(与 Epeli 的相同)https://github.com/joyent/node/blob/e8067cb68563e3f3ab760e3ce8d0aeb873198f36/lib/http.js#L889

于 2012-05-15T17:30:39.847 回答