NodeJS请求多个api端点

时间:2016-01-04 14:24:43

标签: javascript json node.js express requestjs

好的,我正在尝试使用请求模块向API端点发出两个或更多请求。我正在渲染一个HTML文件,并使用以下代码将返回的JSON传递给把手模板:

res.render('list.html', {
  title: 'List',
  data: returnedJSON
}

然后我可以很容易地在我的把手模板中迭代这个JSON。

我遇到的问题是,我现在需要使用多个数据源,其中类别列表将根据JSON响应类别和员工JSON响应中的Staff列表构建。我想要一个简单的解决方案,我可以这样做,但扩展它以使用任意数量的数据源。

以下是我对一个数据源的完整代码段:

request({
    url: 'https://api.com/categories',
    headers: {
        'Bearer': 'sampleapitoken'
    }
}, function(error, response, body) {
    if(error || response.statusCode !== 200) {
        // handle error
    } else {
        var json = JSON.parse(body);
        res.render('list.html', {
            title: 'Listing',
            data: json
        });
    }
});

这适用于一个端点,但如前所述,我现在需要使用多个请求并拥有多个数据源,例如:

request({
    url: ['https://api.com/categories','https://api.com/staff'],
    headers: {
        'Bearer': 'sampleapitoken'
    }
}, function(error, response, body1, body2) {
    if(error || response.statusCode !== 200) {
        // handle error
    } else {
        var json1 = JSON.parse(body1);
        var json2 = JSON.parse(body2);
        res.render('list.html', {
            title: 'Listing',
            staff: json1,
            categories: json2
        });
    }
});

我很欣赏上述内容并非如此,但我希望这有助于传达我想要实现的目标。

提前致谢:)

2 个答案:

答案 0 :(得分:10)

您可以使用async库来映射您的请求对象并将其传递给实际请求,并在一次回调中返回所有结果。

var async = require("async");
var request = require("request");

// create request objects
var requests = [{
  url: 'https://api.com/categories',
  headers: {
    'Bearer': 'sampleapitoken'
  }
}, {
  url: 'https://api.com/staff',
  headers: {
    'Bearer': 'sampleapitoken'
  }
}];

async.map(requests, function(obj, callback) {
  // iterator function
  request(obj, function(error, response, body) {
    if (!error && response.statusCode == 200) {
      // transform data here or pass it on
      var body = JSON.parse(body);
      callback(null, body);
    } else {
      callback(error || response.statusCode);
    }
  });
}, function(err, results) {
  // all requests have been made
  if (err) {
    // handle your error
  } else {
    console.log(results);
    for (var i = 0; i < results.length; i++) {
      // request body is results[i]
    }
  }
});

然而,更简单的方法是利用promises,这可以通过bluebird和promisifying请求lib来完成,或者使用已经公布的请求lib request-promise。您仍然希望包含一个promise / A + lib来异步映射结果。

var Promise = require("bluebird");
var request = require('request-promise');

// create request objects
var requests = [{
  url: 'https://api.com/categories',
  headers: {
    'Bearer': 'sampleapitoken'
  }
}, {
  url: 'https://api.com/staff',
  headers: {
    'Bearer': 'sampleapitoken'
  }
}];

Promise.map(requests, function(obj) {
  return request(obj).then(function(body) {
    return JSON.parse(body);
  });
}).then(function(results) {
  console.log(results);
  for (var i = 0; i < results.length; i++) {
    // access the result's body via results[i]
  }
}, function(err) {
  // handle all your errors here
});

值得注意的是,所有最新版本的节点和浏览器都支持开箱即用的Promise,这可以在没有外部库的情况下实现。

答案 1 :(得分:4)

似乎承诺可以提供帮助。

最简单的可能是创建一个新的请求方法,该方法返回一个promise(或与Bluebird等相关),然后等待所有promises完成,并处理数据

function doReq(url, what) {
    return new Promise(function(resolve, reject) {
        request({
            url: url,
            headers: {
                'Bearer': 'sampleapitoken'
            }
        }, function(error, response) {
            if(error || response.statusCode !== 200) {
                reject(error);
            } else {
                var data = {};
                (Array.isArray(what) ? what : [what]).forEach(function(item, index) {
                    data[item] = JSON.parse(arguments[index + 2]);
                });
                resolve( data );
            }
        });
    });
}

Promise.all([
    doReq('https://api.com/categories', 'data'), 
    doReq(['https://api.com/categories','https://api.com/staff'], ['staff', 'categories'])
]).then(function() {
    var obj = {title : 'Listing'};
    [].slice.call(arguments).forEach(function(arg) {
        Object.keys(arg).forEach(function(key) {
            obj[key] = arg[key];
        });
    });
    res.render('list.html', obj);
});
相关问题