ส่ง JSON ไปยัง HTTP POST Request


92

ฉันกำลังพยายามส่งคำขอ HTTP POST ไปยัง google QPX Express API [1] โดยใช้nodejsและrequest [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 โดยใช้อาร์กิวเมนต์หลายส่วน [3] แต่แทนที่จะตอบสนอง JSON ที่เหมาะสมฉันได้รับข้อผิดพลาด (400 ไม่ได้กำหนด)

เมื่อฉันส่งคำขอโดยใช้ 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' จากคำขอของคุณ
Baart

ไม่ได้สร้างความแตกต่าง แต่เท่าที่ฉันรู้นี้ระบุเพียงว่าการตอบสนองเป็น json ใช่ไหม
Ronin

คุณสามารถแสดงบรรทัดคำสั่ง cURL ที่ใช้งานได้หรือไม่?
mscdex

ด้วยความอยากรู้ทำไมคุณถึงใช้หลายส่วน?
cloudfeet

@mscdex โปรดดูโพสต์ต้นฉบับที่อัปเดตของฉัน
Ronin

คำตอบ:


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 ที่ถูกต้อง
Tobi

ฉันได้รับข้อผิดพลาดนี้: [ERR_STREAM_WRITE_AFTER_END]: write after endฉันจะแก้ไขได้อย่างไร
Mehdi Dehghani


10

คุณไม่ต้องการหลายส่วน แต่เป็นคำขอโพสต์ "ธรรมดา" (พร้อมContent-Type: application/json) แทน นี่คือสิ่งที่คุณต้องการ:

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
});

ฉันลองแล้ว แต่ได้รับข้อผิดพลาดอีกครั้ง: "400 นั่นเป็นข้อผิดพลาดลูกค้าของคุณส่งคำขอที่ผิดรูปแบบหรือผิดกฎหมายนั่นคือทั้งหมดที่เรารู้" สำหรับการเยี่ยมชมการตอบสนองเต็มjsfiddle.net/f71opd7pโปรด
Ronin

4
@Tobi ตามเอกสารคำขอและรหัส , json: trueควรทั้งสองJSON.stringify() body และ JSON.parse()การตอบสนอง
mscdex

นี่คือคำตอบ นอกจากนี้คุณยังสามารถไปป์คำตอบได้request('xxx',{ json: true, body: req.body }).pipe(res).on('error', catchErr);
sidonaldson

สิ่งนี้ใช้ได้ผลสำหรับฉันเมื่อคำตอบที่ยอมรับไม่ใช่
greg_diesel

ฉันได้รับข้อผิดพลาดนี้: [ERR_STREAM_WRITE_AFTER_END]: write after endฉันจะแก้ไขได้อย่างไร
Mehdi Dehghani

9

ขณะนี้มี JavaScript เวอร์ชันใหม่ (ECMAScript 6 http://es6-features.org/#ClassDefinition ) มีวิธีที่ดีกว่าในการส่งคำขอโดยใช้ nodejs และ Promise request ( http://www.wintellect.com/devcenter/nstieglitz/5 -great-features-in-es6-Harmony )

การใช้ห้องสมุด: 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

อ้างอิงจาก doc: 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
    });

การกำหนดเช่นนี้จะเป็นวิธีที่มีประสิทธิภาพในการเขียนโค้ดของคุณ และควรเพิ่ม application / json โดยอัตโนมัติ ไม่จำเป็นต้องประกาศโดยเฉพาะ


0

คุณสามารถส่งวัตถุ json เป็นเนื้อความ (อาร์กิวเมนต์ที่สาม) ของคำขอดึงข้อมูล


1
สิ่งนี้ไม่ได้ให้คำตอบสำหรับคำถาม เมื่อคุณมีเพียงพอชื่อเสียงคุณจะสามารถที่จะแสดงความคิดเห็นในโพสต์ใด ๆ ; ให้ตอบตามคำแนะนำแทน
Anna
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.