我正在使用 express 处理节点中的域和集群,并遇到了这种情况。我有一个集群,它为每个核心生成一个工作人员,我为每个工作人员创建一个快速服务器,该服务器使用每个请求域的策略来处理错误。
下面给出的解决方案按原样工作正常,但是当显式将请求和响应对象添加到域时,错误中间件将停止调用。我不知道为什么会引入这种行为。
问题:
- 为什么会发生?
- 我是否以正确的方式使用域?
- 我可以通过“process.domain”访问工作人员中的当前域还是需要做一些不同的事情?
提前致谢!
我的 app.js:
var express = require('express')
, http = require('http')
, path = require('path')
, domain = require('domain')
, cluster = require('cluster')
, http = require('http')
, numCPUs = require('os').cpus().length;
if (cluster.isMaster) {
// fork workers
for (var i = 0; i < numCPUs; i++) {
cluster.fork();
}
// when a worker dies create a new one
cluster.on('exit', function(worker, code, signal) {
cluster.fork();
});
} else {
var app = express();
//domains
app.use(function domainMiddleware(req, res, next) {
var reqDomain = domain.create();
res.on('close', function () {
reqDomain.dispose();
});
res.on('finish', function () {
reqDomain.dispose();
});
reqDomain.on('error', function (err) {
reqDomain.dispose();
// delegate to express error-middleware
next(err);
});
// Adding the request and response objects to the domain
// makes the express error-middleware to not being called.
// reqDomain.add(req);
// reqDomain.add(res);
reqDomain.run(next);
});
// all environments
app.set('port', process.env.PORT || 3000);
app.set('views', __dirname + '/views');
app.set('view engine', 'jade');
app.use(express.favicon());
//app.use(express.logger('dev'));
app.use(express.bodyParser());
app.use(express.methodOverride());
app.use(app.router);
app.use(express.static(path.join(__dirname, 'public')));
// for testing which cluster that serves the request
app.get('/', function(req, res, next) {
res.json(200, { id: cluster.worker.id });
});
app.get('/error', function(req, res, next) {
var fs = require('fs');
// intentionally force an error
fs.readFile('', process.domain.intercept(function(data) {
// when using intercept we dont need this line anymore
//if (err) throw err;
res.send(data);
}));
});
app.use(function(err, req, res, next) {
console.log('ERROR MIDDLEWARE', err);
res.writeHeader(500, {'Content-Type' : "text/html"});
res.write("<h1>" + err.name + "</h1>");
res.end("<p>" + err.message + "</p>");
});
http.createServer(app).listen(app.get('port'), function(){
console.log('Express server listening on port ' + app.get('port'));
});
}