342

I'm sending the following JSON string to my server.

(
        {
        id = 1;
        name = foo;
    },
        {
        id = 2;
        name = bar;
    }
)

On the server I have this.

app.post('/', function(request, response) {

    console.log("Got response: " + response.statusCode);

    response.on('data', function(chunk) {
        queryResponse+=chunk;
        console.log('data');
    });

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

When I send the string, it shows that I got a 200 response, but those other two methods never run. Why is that?

Mridang Agarwalla
  • 38,521
  • 65
  • 199
  • 353
neuromancer
  • 47,047
  • 74
  • 161
  • 217

7 Answers7

562

I think you're conflating the use of the response object with that of the request.

The response object is for sending the HTTP response back to the calling client, whereas you are wanting to access the body of the request. See this answer which provides some guidance.

If you are using valid JSON and are POSTing it with Content-Type: application/json, then you can use the bodyParser middleware to parse the request body and place the result in request.body of your route.

For earlier versions of Express (< 4)

var express = require('express')
  , app = express.createServer();

app.use(express.bodyParser());

app.post('/', function(request, response){
  console.log(request.body);      // your JSON
  response.send(request.body);    // echo the result back
});

app.listen(3000);

Test along the lines of:

$ curl -d '{"MyKey":"My Value"}' -H "Content-Type: application/json" http://127.0.0.1:3000/
{"MyKey":"My Value"}

Updated for Express 4+

Body parser was split out into it's own npm package after v4, requires a separate install npm install body-parser

var express = require('express')
  , bodyParser = require('body-parser');

var app = express();

app.use(bodyParser.json());

app.post('/', function(request, response){
  console.log(request.body);      // your JSON
   response.send(request.body);    // echo the result back
});

app.listen(3000);

Update for Express 4.16+

Starting with release 4.16.0, a new express.json() middleware is available.

var express = require('express');

var app = express();

app.use(express.json());

app.post('/', function(request, response){
  console.log(request.body);      // your JSON
   response.send(request.body);    // echo the result back
});

app.listen(3000);
themefield
  • 2,454
  • 25
  • 28
Pero P.
  • 22,259
  • 7
  • 56
  • 79
218

For Express v4+

install body-parser from the npm.

$ npm install body-parser

https://www.npmjs.org/package/body-parser#installation

var express    = require('express')
var bodyParser = require('body-parser')

var app = express()

// parse application/json
app.use(bodyParser.json())

app.use(function (req, res, next) {
  console.log(req.body) // populated!
  next()
})
chrisarton
  • 4,231
  • 2
  • 16
  • 15
33

For those getting an empty object in req.body

I had forgotten to set headers: {"Content-Type": "application/json"} in the request. Changing it solved the problem.

Daniel Thompson
  • 1,751
  • 19
  • 32
  • 2
    Can't believe I missed this! I was sending with `text/json` and getting `{}` as a response. Total oversight on my part. Very helpful. – James M. Lay Jan 24 '19 at 01:03
  • 1
    Ugh -- I missed this too. Thank you for posting this so I don't waste more time than I already have! – Steve Gomez Feb 06 '20 at 18:13
  • 1
    wow, thanks! life saver! easy to forget when invoking a lambda directly on the command line! `serverless invoke local -f app -l -d '{ "path": "/conferences", "httpMethod": "POST", "body": { "conference_name": "test" }, "headers": {"Content-Type": "application/json"} }'` – Andrew Oct 26 '20 at 07:04
15

Sometimes you don't need third party libraries to parse JSON from text. Sometimes all you need it the following JS command, try it first:

        const res_data = JSON.parse(body);
xims
  • 1,292
  • 16
  • 20
  • 4
    The original question is about parsing JSON from a POST message in the Express framework. Without the BodyParser middleware, the JSON data will not exist in the body property of the request object. – ThisClark Nov 07 '16 at 02:39
  • 1
    I found this useful, when parsing server response. Thanks! – Hasan Alsawadi Nov 09 '16 at 08:21
  • 1
    Thank you Hasan, I appreciate your comment. It did helped me when I was looking for solution and came across this post. Not sure if it works in all cases but it definitely works in some and it is a better solution than using third party library. – xims Nov 11 '16 at 00:48
  • 1
    Your answer and a comment provides the answer with more information (the more information being your answer here). You should update your answer to indicate that express needs the body-parser or give an alternative to indicate how body-parser got the data in the first place. – dewwwald Dec 20 '16 at 20:19
  • 3
    does not define `body` – jameshfisher Dec 25 '16 at 08:34
  • But where do I get the string body? – Tomáš Zato - Reinstate Monica Sep 25 '17 at 22:31
12

@Daniel Thompson mentions that he had forgotten to add {"Content-Type": "application/json"} in the request. He was able to change the request, however, changing requests is not always possible (we are working on the server here).

In my case I needed to force content-type: text/plain to be parsed as json.

If you cannot change the content-type of the request, try using the following code:

app.use(express.json({type: '*/*'}));

Instead of using express.json() globally, I prefer to apply it only where needed, for instance in a POST request:

app.post('/mypost', express.json({type: '*/*'}), (req, res) => {
  // echo json
  res.json(req.body);
});
anneb
  • 4,399
  • 2
  • 21
  • 24
11

const express = require('express');
let app = express();
app.use(express.json());

This app.use(express.json) will now let you read the incoming post JSON object

SuRa
  • 893
  • 8
  • 13
0

A beginner's mistake...i was using app.use(express.json()); in a local module instead of the main file (entry point).

Hatim
  • 133
  • 1
  • 7