Node.js POST导致[错误:套接字挂起]代码:'ECONNRESET'

时间:2013-09-09 06:29:05

标签: javascript node.js express

我创建了一个将数据发布到休息服务的示例,我发现当我有非ascii或非拉丁字符时(请参阅data.firstName),我使用TEST-REST.js的帖子请求将抛出< / p>

  

错误:{[错误:套接字挂断]代码:'ECONNRESET'}。

// TEST-REST.js
var http = require('http');

var data = JSON.stringify({
  firstName: 'JoaquÌn',
});

var options = {
  host: '127.0.0.1',
  port: 3000,
  path: '/users',
  method: 'POST',
  headers: {
    'Content-Type': 'application/json',
    'Content-Length': data.length
  }
};

var req = http.request(options, function(res) {
  var result = '';

  res.on('data', function(chunk) {
    result += chunk;
  });

  res.on('end', function() {
    console.log(result);
  });
});

req.on('error', function(err) {
  console.log(err);
});

req.write(data);
req.end();

在我的休息服务上,它给我带来了这样的错误:

SyntaxError: Unexpected end of input Sun Sep 08 2013 23:25:02 GMT-0700 (PDT) -     at Object.parse (native)
    at IncomingMessage.<anonymous> (/Volumes/Data/Program_Data/GitHub/app/node_modules/express/node_modules/connect/lib/middleware/json.js:66:27) info    at IncomingMessage.EventEmitter.emit (events.js:92:17)
    at _stream_readable.js:920:16 : - - - [Mon, 09 Sep 2013 06:25:02 GMT] "POST /users HTTP/1.1" 400 - "-" "-"
    at process._tickDomainCallback (node.js:459:13)

如果我将'JoaquÌn'中的firstName值替换为'abc',一切正常。我想我错过了一些支持或逃避的东西让它发挥作用。

有没有人知道我是如何解决这个问题的?我也试过以下:require('querystring')。escape(model.givenName),它有效,但我对它不满意。

已更新 我发现如果我注释掉:app.use(express.bodyParser());,错误就会消失。

2 个答案:

答案 0 :(得分:42)

这是节点的问题,而不是明确的问题。 https://github.com/visionmedia/express/issues/1749

解决,改变

  

&#39; Content-Length&#39;:data.length

  

&#39;内容长度&#39;:Buffer.byteLength(数据)

THUMB的规则

如果要查找字符串的内容长度,请始终使用 Buffer.byteLength()

<强>已更新

我们还应该优雅地在服务器端处理错误,以防止因添加中间件来处理它而导致崩溃。

app.use(function (error, req, res, next) {
  if (!error) {
    next();
  } else {
    console.error(error.stack);
    res.send(500);
  }
});

答案 1 :(得分:1)

问题是,如果您不处理此错误并使服务器保持活动状态,则此远程崩溃漏洞可用于DOS攻击。但是,您可以处理它并继续,并且在未处理的异常发生时仍然关闭进程(这会阻止您以未定义的状态运行 - 这是一件非常糟糕的事情)。

connect模块处理错误并调用next(),并使用消息正文和status = 400发回一个对象。在您的服务器代码中,您可以在express.bodyParser()

之后添加此代码
var exit = function exit() {
  setTimeout(function () {
    process.exit(1);
  }, 0);
};

app.use(function (error, req, res, next) {
  if (error.status === 400) {
    log.info(error.body);
    return res.send(400);
  }

  log.error(error);
  exit();
});