передать JSON в HTTP-запрос POST

92

Я пытаюсь сделать HTTP-запрос POST к Google QPX Express API [1], используя nodejs и запрос [2].

Мой код выглядит следующим образом:

    // create http request client to consume the QPX API
    var request = require("request")

    // JSON to be passed to the QPX Express API
    var requestData = {
        "request": {
            "slice": [
                {
                    "origin": "ZRH",
                    "destination": "DUS",
                    "date": "2014-12-02"
                }
            ],
            "passengers": {
                "adultCount": 1,
                "infantInLapCount": 0,
                "infantInSeatCount": 0,
                "childCount": 0,
                "seniorCount": 0
            },
            "solutions": 2,
            "refundable": false
        }
    }

    // QPX REST API URL (I censored my api key)
    url = "https://www.googleapis.com/qpxExpress/v1/trips/search?key=myApiKey"

    // fire request
    request({
        url: url,
        json: true,
        multipart: {
            chunked: false,
            data: [
                {
                    'content-type': 'application/json',
                    body: requestData
                }
            ]
        }
    }, function (error, response, body) {
        if (!error && response.statusCode === 200) {
            console.log(body)
        }
        else {

            console.log("error: " + error)
            console.log("response.statusCode: " + response.statusCode)
            console.log("response.statusText: " + response.statusText)
        }
    })

Я пытаюсь передать JSON с помощью аргумента multipart [3]. Но вместо правильного ответа JSON я получил ошибку (400 undefined).

Когда я делаю запрос, используя тот же JSON и API-ключ, используя вместо этого CURL, он работает нормально. Так что с моим ключом API или JSON все в порядке.

Что не так с моим кодом?

ИЗМЕНИТЬ :

рабочий пример CURL:

i) Я сохранил JSON, который я передал в свой запрос, в файл с именем «request.json»:

{
  "request": {
    "slice": [
      {
        "origin": "ZRH",
        "destination": "DUS",
        "date": "2014-12-02"
      }
    ],
    "passengers": {
      "adultCount": 1,
      "infantInLapCount": 0,
      "infantInSeatCount": 0,
      "childCount": 0,
      "seniorCount": 0
    },
    "solutions": 20,
    "refundable": false
  }
}

ii) затем в терминале я переключился в каталог, в котором был расположен и запущен вновь созданный файл request.json (myApiKey, очевидно, означает мой фактический ключ API):

curl -d @request.json --header "Content-Type: application/json" https://www.googleapis.com/qpxExpress/v1/trips/search?key=myApiKey

[1] https://developers.google.com/qpx-express/ [2] клиент HTTP-запросов, разработанный для nodejs: https://www.npmjs.org/package/request [3] вот пример, который я нашел https://www.npmjs.org/package/request#multipart-related [4] QPX Express API возвращает ошибку синтаксического анализа 400

Ронин
источник
Попробуйте удалить json: true из вашего запроса
Баарт,
не имеет значения. но насколько я знаю, это указывает только на то, что ответ - это json, верно?
Ронин
Можете ли вы показать работающую командную строку cURL?
mscdex
Из любопытства, почему вы используете multipart?
cloudfeet
@mscdex, пожалуйста, посмотрите мой обновленный исходный пост
Ронин

Ответы:

168

Я думаю, что должно работать следующее:

// fire request
request({
    url: url,
    method: "POST",
    json: requestData
}, ...

В этом случае Content-type: application/jsonзаголовок добавляется автоматически.

Тоби
источник
1
По какой-то причине конечная точка, с которой я столкнулся, не могла прочитать параметры с помощью первого метода (как если бы они не были отправлены), но смогла сделать это с помощью второго метода.
The Unknown Dev
Аналогично тому, что сказал Джамиль. Получил SyntaxError: Unexpected token &quot;<br> &nbsp; &nbsp;at parse (/home/malcolm/complice/node_modules/body-parser/lib/types/json.js:83:15)с первым способом.
MalcolmOcean
@MalcolmOcean Это потому, что тег <br> не является допустимым содержимым JSON
Тоби
У меня такая ошибка:, [ERR_STREAM_WRITE_AFTER_END]: write after endкак ее исправить?
Мехди Дехгани
18

Я работал над этим слишком долго. Ответ, который мне помог, был по адресу: отправьте сообщение Content-Type: application / json с помощью node.js

В котором используется следующий формат:

request({
    url: url,
    method: "POST",
    headers: {
        "content-type": "application/json",
        },
    json: requestData
//  body: JSON.stringify(requestData)
    }, function (error, resp, body) { ...
Дэн Бейкер
источник
10

Вам не нужен multipart, а Content-Type: application/jsonвместо этого нужен "простой" запрос POST (с ). Вот все, что вам нужно:

var request = require('request');

var requestData = {
  request: {
    slice: [
      {
        origin: "ZRH",
        destination: "DUS",
        date: "2014-12-02"
      }
    ],
    passengers: {
      adultCount: 1,
      infantInLapCount: 0,
      infantInSeatCount: 0,
      childCount: 0,
      seniorCount: 0
    },
    solutions: 2,
    refundable: false
  }
};

request('https://www.googleapis.com/qpxExpress/v1/trips/search?key=myApiKey',
        { json: true, body: requestData },
        function(err, res, body) {
  // `body` is a js object if request was successful
});
mscdex
источник
Я попробовал это, но получил другую ошибку: «400. Это ошибка. Ваш клиент отправил неверный или недопустимый запрос. Это все, что мы знаем». для получения полного ответа посетите jsfiddle.net/f71opd7p, пожалуйста
Ронин
4
@Tobi согласно документации и коду запроса , json: trueдолжен JSON.stringify() body и JSON.parse() ответ.
mscdex
Это ответ. В дополнение к этому вы также можете request('xxx',{ json: true, body: req.body }).pipe(res).on('error', catchErr);
передать
Это сработало для меня, когда принятого ответа не было.
greg_diesel
У меня такая ошибка:, [ERR_STREAM_WRITE_AFTER_END]: write after endкак ее исправить?
Мехди Дехгани
9

Теперь с новой версией JavaScript (ECMAScript 6 http://es6-features.org/#ClassDefinition ) появился лучший способ отправлять запросы с помощью nodejs и запроса Promise ( http://www.wintellect.com/devcenter/nstieglitz/5 -great-features-in-es6-гармония )

Использование библиотеки: https://github.com/request/request-promise

npm install --save request
npm install --save request-promise

клиент:

//Sequential execution for node.js using ES6 ECMAScript
var rp = require('request-promise');

rp({
    method: 'POST',
    uri: 'http://localhost:3000/',
    body: {
        val1 : 1,
        val2 : 2
    },
    json: true // Automatically stringifies the body to JSON
}).then(function (parsedBody) {
        console.log(parsedBody);
        // POST succeeded...
    })
    .catch(function (err) {
        console.log(parsedBody);
        // POST failed...
    });

сервер:

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

    var jsonRequest = request.body;
    var jsonResponse = {};

    jsonResponse.result = jsonRequest.val1 + jsonRequest.val2;

    response.send(jsonResponse);
});


app.listen(3000);
Эвалдс Уртанс
источник
3

Пример.

var request = require('request');

var url = "http://localhost:3000";

var requestData = {
    ...
} 

var data = {
    url: url,
    json: true,
    body: JSON.stringify(requestData)
}

request.post(data, function(error, httpResponse, body){
    console.log(body);
});

В качестве json: trueопции вставки устанавливает тело в JSON-представление значения и добавляет "Content-type": "application/json"заголовок. Кроме того, анализирует тело ответа как JSON. ССЫЛКА НА САЙТ

Ёнмин Ким
источник
2

Согласно док: https://github.com/request/request

Пример такой:

  multipart: {
      chunked: false,
      data: [
        {
          'content-type': 'application/json', 
          body: JSON.stringify({foo: 'bar', _attachments: {'message.txt': {follows: true, length: 18, 'content_type': 'text/plain' }}})
        },
      ]
    }

Я думаю, вы отправляете объект, где ожидается строка, замените

body: requestData

от

body: JSON.stringify(requestData)
Баарт
источник
2
       var request = require('request');
        request({
            url: "http://localhost:8001/xyz",
            json: true,
            headers: {
                "content-type": "application/json",
            },
            body: JSON.stringify(requestData)
        }, function(error, response, body) {
            console.log(response);
        });
Ашиш Гупта
источник
0

я чувствую

var x = request.post({
       uri: config.uri,
       json: reqData
    });

Такое определение будет эффективным способом написания кода. И приложение / json должно быть добавлено автоматически. Специально декларировать это не нужно.

Сиддхартха Тхота
источник
0

вы можете передать объект json в качестве тела (третьего аргумента) запроса на выборку.

Анкур Тивари
источник
1
Это не дает ответа на вопрос. Как только у вас будет достаточная репутация, вы сможете комментировать любой пост ; вместо этого дайте ответы в соответствии с руководством
Анна