คำขอ HTTP GET ใน Node.js Express


192

ฉันจะขอ HTTP จากภายใน Node.js หรือ Express.js ได้อย่างไร ฉันต้องเชื่อมต่อกับบริการอื่น ฉันหวังว่าการโทรแบบอะซิงโครนัสและการติดต่อกลับมีการตอบสนองของเซิร์ฟเวอร์ระยะไกล

คำตอบ:


220

นี่คือตัวอย่างของรหัสบางส่วนจากตัวอย่างของฉัน มันไม่ตรงกันและส่งคืนวัตถุ JSON มันสามารถทำคำขอ GET ได้ทุกรูปแบบ

โปรดทราบว่ามีวิธีที่เหมาะสมกว่า (เพียงแค่ตัวอย่าง) - ตัวอย่างเช่นแทนที่จะเชื่อมต่อชิ้นที่คุณใส่เข้าไปในอาร์เรย์และเข้าร่วม ฯลฯ ... หวังว่ามันจะทำให้คุณเริ่มต้นในทิศทางที่ถูกต้อง:

const http = require('http');
const https = require('https');

/**
 * getJSON:  RESTful GET request returning JSON object(s)
 * @param options: http options object
 * @param callback: callback to pass the results JSON object(s) back
 */

module.exports.getJSON = (options, onResult) => {
  console.log('rest::getJSON');
  const port = options.port == 443 ? https : http;

  let output = '';

  const req = port.request(options, (res) => {
    console.log(`${options.host} : ${res.statusCode}`);
    res.setEncoding('utf8');

    res.on('data', (chunk) => {
      output += chunk;
    });

    res.on('end', () => {
      let obj = JSON.parse(output);

      onResult(res.statusCode, obj);
    });
  });

  req.on('error', (err) => {
    // res.send('error: ' + err.message);
  });

  req.end();
};

มันถูกเรียกโดยการสร้างวัตถุตัวเลือกเช่น:

const options = {
  host: 'somesite.com',
  port: 443,
  path: '/some/path',
  method: 'GET',
  headers: {
    'Content-Type': 'application/json'
  }
};

และจัดให้มีฟังก์ชั่นการโทรกลับ

ตัวอย่างเช่นในบริการฉันต้องการโมดูล REST ด้านบนจากนั้นทำสิ่งนี้:

rest.getJSON(options, (statusCode, result) => {
  // I could work with the resulting HTML/JSON here. I could also just return it
  console.log(`onResult: (${statusCode})\n\n${JSON.stringify(result)}`);

  res.statusCode = statusCode;

  res.send(result);
});

UPDATE

หากคุณกำลังมองหาasync/ await(เชิงเส้นไม่มีการติดต่อกลับ) สัญญาการรวบรวมเวลาสนับสนุนและ Intellisense เราได้สร้างไคลเอ็นต์ HTTP และ REST แบบ Lightweight ที่เหมาะกับการเรียกเก็บเงินนั้น:

Microsoft พิมพ์ส่วนที่เหลือลูกค้า


@bryanmac คุณกรุณาส่ง / เพิ่มตัวอย่างที่สมบูรณ์ได้หรือไม่
StErMi

@bryanmac ได้รับอนุญาตจากคุณฉันต้องการใช้โค้ด grunt นี้ฉันกำลังสร้าง ไม่แน่ใจว่าเมื่อใด แต่จะเปิดให้บริการเมื่อเสร็จแล้ว
JeffH

3
ลองใช้โมดูลขอ .. มันง่ายกว่ามากsitepoint.com/making-http-requests-in-node-js
saurshaz

6
ใช่ - โมดูลคำขอนั้นง่าย แต่นี่เป็นระดับที่ต่ำกว่าซึ่งแสดงว่าไลบรารีเช่นโมดูลคำขอกำลังทำอะไร หากคุณต้องการการควบคุมระดับที่ต่ำกว่าหรือคำขอ http (แสดงความคืบหน้าในการดาวน์โหลดขนาดใหญ่ ฯลฯ ... ) สิ่งนี้จะแสดงให้เห็นว่ามันเสร็จสิ้นแล้ว
bryanmac

1
@KrIsHnA - โหนดมีวัตถุการสืบค้น: nodejs.org/api/querystring.htmlและวัตถุ url nodejs.org/docs/latest/api/url.html
bryanmac

100

ลองใช้http.get(options, callback)ฟังก์ชันง่าย ๆใน node.js:

var http = require('http');
var options = {
  host: 'www.google.com',
  path: '/index.html'
};

var req = http.get(options, function(res) {
  console.log('STATUS: ' + res.statusCode);
  console.log('HEADERS: ' + JSON.stringify(res.headers));

  // Buffer the body entirely for processing as a whole.
  var bodyChunks = [];
  res.on('data', function(chunk) {
    // You can process streamed parts here...
    bodyChunks.push(chunk);
  }).on('end', function() {
    var body = Buffer.concat(bodyChunks);
    console.log('BODY: ' + body);
    // ...and/or process the entire body here.
  })
});

req.on('error', function(e) {
  console.log('ERROR: ' + e.message);
});

นอกจากนี้ยังมีhttp.request(options, callback)ฟังก์ชั่นทั่วไปที่ให้คุณระบุวิธีการร้องขอและรายละเอียดคำขออื่น ๆ


เนื้อหาของการตอบกลับของเซิร์ฟเวอร์ที่ OP ขออยู่ที่ไหน
Dan Dascalescu

ขอบคุณสำหรับการอัพเดท. ดูเหมือนว่าไม่จำเป็นต้องมีตัวจัดการ 'สิ้นสุด' เพื่อต่อชิ้นส่วนให้เข้ากัน คำตอบของ @ bryanmac นั้นมีจำนวนเท่าใด?
Dan Dascalescu

@DanDascalescu: ใช่ถ้าคุณต้องการที่จะประมวลผลร่างกายโดยรวม (ซึ่งมีแนวโน้ม) คุณอาจต้องการบัฟเฟอร์มันและประมวลผลที่ 'สิ้นสุด' ฉันจะอัปเดตคำตอบของฉันด้วยเพื่อความสมบูรณ์
maerics

ขออภัยฉันไม่สามารถระบุพารามิเตอร์ที่เรียกกลับด้วย ... ฉันจะหาเนื้อหาได้อย่างไรและการอ้างอิงสำหรับพารามิเตอร์และคุณสมบัติของพารามิเตอร์นั้นอยู่ที่ไหน
มูฮัมหมัดอูเมอ

@maerics ฉันจะใช้GETคำขอนี้ได้อย่างไรถ้าฉันมี URL นี้ graph.facebook.com/debug_token? input_token={token-to-inspect} &access_token={app-token-or-admin-token}?
frank17

70

คำขอและSuperagentเป็นห้องสมุดที่ใช้งานได้ดี

หมายเหตุ: คำขอไม่ได้รับการสนับสนุนใช้ความเสี่ยงของคุณ!

การใช้request:

var request=require('request');

request.get('https://someplace',options,function(err,res,body){
  if(err) //TODO: handle err
  if(res.statusCode === 200 ) //etc
  //TODO Do something with response
});

7
ควร res.statusCode === 200 ในครั้งที่สองหรือไม่ )
Gleb Dolzikov

1
ตัวแปรตัวเลือกคืออะไร? เพิ่งผ่านหนึ่งไม่ได้กำหนด? ฉันสงสัยมัน
lxknvlk

32

คุณยังสามารถใช้Requestifyซึ่งเป็นไคลเอนต์ HTTP ที่เจ๋งและเรียบง่ายมาก ๆ ที่ฉันเขียนไว้สำหรับ nodeJS + ซึ่งรองรับการแคช

เพียงทำต่อไปนี้เพื่อขอวิธีการ GET:

var requestify = require('requestify');

requestify.get('http://example.com/api/resource')
  .then(function(response) {
      // Get the response body (JSON parsed or jQuery object for XMLs)
      response.getBody();
  }
);

1
"ลองใช้เครื่องมืออื่นนี้" ไม่ใช่คำตอบที่ยอมรับได้หากชุดเครื่องมือที่มีอยู่เพียงพอ
Grunion Shaftoe

9

รุ่นนี้ขึ้นอยู่กับการเสนอครั้งแรกโดยฟังก์ชั่นbryanmacซึ่งใช้สัญญาการจัดการข้อผิดพลาดที่ดีกว่าและเขียนใหม่ใน ES6

let http = require("http"),
    https = require("https");

/**
 * getJSON:  REST get request returning JSON object(s)
 * @param options: http options object
 */
exports.getJSON = function(options)
{
    console.log('rest::getJSON');
    let reqHandler = +options.port === 443 ? https : http;

    return new Promise((resolve, reject) => {
        let req = reqHandler.request(options, (res) =>
        {
            let output = '';
            console.log('rest::', options.host + ':' + res.statusCode);
            res.setEncoding('utf8');

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

            res.on('end', () => {
                try {
                    let obj = JSON.parse(output);
                    // console.log('rest::', obj);
                    resolve({
                        statusCode: res.statusCode,
                        data: obj
                    });
                }
                catch(err) {
                    console.error('rest::end', err);
                    reject(err);
                }
            });
        });

        req.on('error', (err) => {
            console.error('rest::request', err);
            reject(err);
        });

        req.end();
    });
};

เป็นผลให้คุณไม่ต้องผ่านฟังก์ชั่นการโทรกลับแทน getJSON () ส่งคืนสัญญา ในตัวอย่างต่อไปนี้ใช้ฟังก์ชันภายในตัวจัดการเส้นทาง ExpressJS

router.get('/:id', (req, res, next) => {
    rest.getJSON({
        host: host,
        path: `/posts/${req.params.id}`,
        method: 'GET'
    }).then(({status, data}) => {
        res.json(data);
    }, (error) => {
        next(error);
    });
});

ในข้อผิดพลาดมันมอบหมายข้อผิดพลาดไปยังเซิร์ฟเวอร์จัดการมิดเดิลแวร์ข้อผิดพลาด


1
ใช่ตัวอย่างนี้แสดงวิธีการภายในgetคำจำกัดความของเส้นทางด่วนซึ่งมีบทความมากมายที่นี่หายไป
Micros

7

Unirestเป็นห้องสมุดที่ดีที่สุดที่ฉันเคยพบเพื่อขอ HTTP จาก Node มันมีเป้าหมายที่จะเป็นเฟรมเวิร์กแบบหลายแพลตฟอร์มดังนั้นการเรียนรู้วิธีการทำงานบน Node จะให้บริการคุณได้ดีหากคุณต้องการใช้ไคลเอนต์ HTTP บน Ruby, PHP, Java, Python, Objective C, .Net หรือ Windows 8 เช่นกัน เท่าที่ฉันสามารถบอกได้ว่าไลบรารี unirest ส่วนใหญ่ได้รับการสนับสนุนจากไคลเอนต์ HTTP ที่มีอยู่ (เช่นบน Java, ไคลเอนต์ Apache HTTP, บนโหนด, libary Request ของ Mikeal ) - Unirest เพียงวาง nicer API ไว้ด้านบน

ต่อไปนี้เป็นตัวอย่างโค้ดสำหรับ Node.js:

var unirest = require('unirest')

// GET a resource
unirest.get('http://httpbin.org/get')
  .query({'foo': 'bar'})
  .query({'stack': 'overflow'})
  .end(function(res) {
    if (res.error) {
      console.log('GET error', res.error)
    } else {
      console.log('GET response', res.body)
    }
  })

// POST a form with an attached file
unirest.post('http://httpbin.org/post')
  .field('foo', 'bar')
  .field('stack', 'overflow')
  .attach('myfile', 'examples.js')
  .end(function(res) {
    if (res.error) {
      console.log('POST error', res.error)
    } else {
      console.log('POST response', res.body)
    }
  })

คุณสามารถข้ามไปที่ Node docs ได้ที่นี่


3

ตรวจสอบฉีก เป็นโหนดไคลเอ็นต์ HTTP ที่สร้างและดูแลโดยspire.ioที่จัดการการเปลี่ยนทิศทางเซสชันและการตอบสนอง JSON เหมาะสำหรับการโต้ตอบกับ API ส่วนที่เหลือ ดูโพสต์บล็อกนี้สำหรับรายละเอียดเพิ่มเติม



0

หากคุณต้องการเพียงแค่รับคำขอและไม่ต้องการการสนับสนุนสำหรับวิธีการ HTTP อื่น ๆ ให้ดูที่: ง่าย ๆ รับ :

var get = require('simple-get');

get('http://example.com', function (err, res) {
  if (err) throw err;
  console.log(res.statusCode); // 200
  res.pipe(process.stdout); // `res` is a stream
});

0

ใช้reqclient : ไม่ได้ออกแบบมาเพื่อวัตถุประสงค์ในการเขียนสคริปต์เช่นrequestหรือไลบรารีอื่น ๆ อีกมากมาย Reqclient อนุญาตให้ใน Constructor ระบุการกำหนดค่าจำนวนมากที่มีประโยชน์เมื่อคุณต้องการใช้การกำหนดค่าเดียวกันซ้ำแล้วซ้ำอีก: URL ฐาน, ส่วนหัว, ตัวเลือกการรับรองความถูกต้อง, ตัวเลือกการบันทึก, แคช ฯลฯ ยังมีคุณสมบัติที่มีประโยชน์เช่น การแยกวิเคราะห์ JSON เป็นต้น

วิธีที่ดีที่สุดในการใช้ไลบรารีคือสร้างโมดูลเพื่อส่งออกวัตถุที่ชี้ไปที่ API และการกำหนดค่าที่จำเป็นเพื่อเชื่อมต่อกับ:

โมดูลclient.js:

let RequestClient = require("reqclient").RequestClient

let client = new RequestClient({
  baseUrl: "https://myapp.com/api/v1",
  cache: true,
  auth: {user: "admin", pass: "secret"}
})

module.exports = client

และในคอนโทรลเลอร์ที่คุณจำเป็นต้องใช้ API ให้ใช้ดังนี้:

let client = require('client')
//let router = ...

router.get('/dashboard', (req, res) => {
  // Simple GET with Promise handling to https://myapp.com/api/v1/reports/clients
  client.get("reports/clients")
    .then(response => {
       console.log("Report for client", response.userId)  // REST responses are parsed as JSON objects
       res.render('clients/dashboard', {title: 'Customer Report', report: response})
    })
    .catch(err => {
      console.error("Ups!", err)
      res.status(400).render('error', {error: err})
    })
})

router.get('/orders', (req, res, next) => {
  // GET with query (https://myapp.com/api/v1/orders?state=open&limit=10)
  client.get({"uri": "orders", "query": {"state": "open", "limit": 10}})
    .then(orders => {
      res.render('clients/orders', {title: 'Customer Orders', orders: orders})
    })
    .catch(err => someErrorHandler(req, res, next))
})

router.delete('/orders', (req, res, next) => {
  // DELETE with params (https://myapp.com/api/v1/orders/1234/A987)
  client.delete({
    "uri": "orders/{client}/{id}",
    "params": {"client": "A987", "id": 1234}
  })
  .then(resp => res.status(204))
  .catch(err => someErrorHandler(req, res, next))
})

reqclientรองรับคุณสมบัติมากมาย แต่มีบางอย่างที่ไม่รองรับโดยไลบรารีอื่น ๆ : การรวม OAuth2และการรวมตัวบันทึกด้วยไวยากรณ์ cURLและส่งคืนวัตถุสัญญาดั้งเดิม


0

หากคุณต้องการส่งGETคำขอไปยังIPa Domain( a คำตอบอื่น ๆ ไม่ได้กล่าวถึงคุณสามารถระบุportตัวแปร) คุณสามารถใช้ฟังก์ชันนี้ได้:

function getCode(host, port, path, queryString) {
    console.log("(" + host + ":" + port + path + ")" + "Running httpHelper.getCode()")

    // Construct url and query string
    const requestUrl = url.parse(url.format({
        protocol: 'http',
        hostname: host,
        pathname: path,
        port: port,
        query: queryString
    }));

    console.log("(" + host + path + ")" + "Sending GET request")
    // Send request
    console.log(url.format(requestUrl))
    http.get(url.format(requestUrl), (resp) => {
        let data = '';

        // A chunk of data has been received.
        resp.on('data', (chunk) => {
            console.log("GET chunk: " + chunk);
            data += chunk;
        });

        // The whole response has been received. Print out the result.
        resp.on('end', () => {
            console.log("GET end of response: " + data);
        });

    }).on("error", (err) => {
        console.log("GET Error: " + err);
    });
}

อย่าพลาดโมดูลที่ต้องการที่ด้านบนของไฟล์ของคุณ:

http = require("http");
url = require('url')

httpsอย่าลืมว่าคุณอาจใช้โมดูลสำหรับการสื่อสารผ่านเครือข่ายที่ปลอดภัย ดังนั้นทั้งสองบรรทัดจะเปลี่ยน:

https = require("https");
...
https.get(url.format(requestUrl), (resp) => { ......

-1
## you can use request module and promise in express to make any request ##
const promise                       = require('promise');
const requestModule                 = require('request');

const curlRequest =(requestOption) =>{
    return new Promise((resolve, reject)=> {
        requestModule(requestOption, (error, response, body) => {
            try {
                if (error) {
                    throw error;
                }
                if (body) {

                    try {
                        body = (body) ? JSON.parse(body) : body;
                        resolve(body);
                    }catch(error){
                        resolve(body);
                    }

                } else {

                    throw new Error('something wrong');
                }
            } catch (error) {

                reject(error);
            }
        })
    })
};

const option = {
    url : uri,
    method : "GET",
    headers : {

    }
};


curlRequest(option).then((data)=>{
}).catch((err)=>{
})

(ในขณะที่มันเกิดขึ้นก็จะไม่แก้ปัญหารหัสนี้จะ. ฟังสำหรับการร้องขอแต่คำถามคือถามถึงวิธีการส่งคำขอ )
เคว็นติ

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