快速获取rawBody

时间:2013-07-14 23:30:51

标签: node.js post express connect

您好我正在尝试从帖子中检索某些内容,并且需要来自传入请求的rawBody属性。我怎样才能找回它?

我尝试使用express.bodyParser(),在我的帖子处理程序中,我一直在寻找req.rawBody,它是未定义的。

我甚至尝试使用connect.bodyParser(),但我仍然没有运气。我的rawBody未定义。

我正在阅读stackoverflow网站上说他们已经删除了rawBody功能,但提到将它添加到我们自己的中间件文件是一个快速修复。我是新手,所以我不知道如何实现这一目标。以下是我的代码段。

/**
 * Module dependencies.
 */

var express = require('express')
  , connect = require('connect')
  , routes = require('./routes')
  , user = require('./routes/user')
  , http = require('http')
  , path = require('path');

var app = express();

// 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(connect.bodyParser());
app.use(express.methodOverride());
app.use(app.router);
app.use(express.static(path.join(__dirname, 'public')));

// development only
if ('development' == app.get('env')) {
  app.use(express.errorHandler());
}

app.get('/', routes.index);
app.get('/users', user.list);



/**custom stuff**/

app.post('/upload',function(req, res){
        console.log(req.header('Content-Type'));
        console.log(req.header('Host'));
        console.log(req.header('User-Agent'));

        console.log(req.rawBody);
        console.log(req.body);
        res.send("<h1> Hello the response is "+req.body.username);
});

/** end**/

http.createServer(app).listen(app.get('port'), function(){
  console.log('Express server listening on port ' + app.get('port'));
});

非常感谢任何帮助。

谢谢。

4 个答案:

答案 0 :(得分:6)

您可以使用自己的中间件来执行此操作:

app.use(function(req, res, next){
   var data = "";
   req.on('data', function(chunk){ data += chunk})
   req.on('end', function(){
      req.rawBody = data;
      next();
   })
})

// Your route registration:
app.get('/', function(){// whatever...})

app.post('/test', function(req, res){
    console.log(req.rawBody);
    res.send("your request raw body is:"+req.rawBody);
})

答案 1 :(得分:2)

我又回来了:D。在读完connect.bodyParser后,我发现了一些东西:bodyParser只解析mime类型为以下数据的数据:application / json,application / x-www-form-urlencoded和multipart / form-data。所以我认为这是另一种方法,它通常不优雅但可以接受:当您尝试将原始数据发送到服务器时,将mime类型更改为不同的类型。作为你的问题,它是一个字符串,所以我选择text / plain作为例子:

// if the request's mime type is text/plain, read it as raw data
var myRawParser = function(req, res, next){
    req.rawData = '';
    if(req.header('content-type') == 'text/plain'){
        req.on('data', function(chunk){
            req.rawData += chunk;
        })
        req.on('end', function(){
            next();
        })
    } else {
        next();
    }
}

// ... 
app.use(myRawParser);
app.use(express.bodyParser());
// ...

// Here is my test route:
app.post('/test', function(req, res){
    console.log('Mime type is:'+req.header('content-type'));
    console.log('Raw data is:'+req.rawData);
    console.log('Body via bodyParser is:');
    console.dir(req.body);
    res.send('Hello!');
})

我通过curl测试了它:

$ curl -d 'test=hello' 127.0.0.1:3000/test

// console result:
Mime type is:application/x-www-form-urlencoded
Raw data is: 
Body via bodyParser is:
{ test: 'hello' }

$ curl -d 'test=hello' -H  'Content-Type:text/plain' 127.0.0.1:3000/test
// console result:
Mime type is:text/plain
Raw data is:test=hello
Body via bodyParser is: 
{}

实际上并没有将你的中间件集成到bodyParser中,只是让它们一起工作。

答案 2 :(得分:0)

以@Rikky解决方案为基础,避免数据事件处理程序中出现竞争状况的方法是在设置处理程序后立即继续调用中间件链。不要等待req.on('end')调用next(),因为next()调用允许json主体解析器注册自己的数据和结束处理程序;如果您等到请求触发结束时,他们将错过所有相关事件。而是使用promises:

const process = require('process');

const bodyParser = require('body-parser');
const express = require('express');

function main() {
  const app = express();

  app.use((req, res, next) => {
    req.rawBody = new Promise(resolve => {
      buf = '';
      req.on('data', x => buf += x);
      req.on('end', () => {
        resolve(buf);
      });
    });
    next();
  });

  app.use(bodyParser.json());

  app.use('/', async (req, res) => {
    console.log('raw body:', await req.rawBody);
    console.log('json parsed:', req.body);
    res.send('bye\n');
  });

  app.listen('3000', 'localhost', (e) => {
    if (e) {
      console.error(e);
      process.exit(1);
    }
    console.log('Listening on localhost:3000');
  });
}

main();

答案 3 :(得分:0)

我找到的最佳解决方案是:https://stackoverflow.com/a/9931478/1768033

因为使用

req.on('data', function(chunk){ data += chunk})

不知何故改变了我在多形式数据请求中发送的文件的位,因此它们不再有效。