Q.所有请求承诺的数组,不知道如何获得结果

时间:2016-01-02 10:17:13

标签: node.js promise q

可能是一个明显的答案,但我不知道该采取什么方式。 请求是一个节点模块:https://github.com/request/request 我填充了一组getHistory请求(具有不同的参数)。 p = [p1,p2 ...]。

this.app.all('/api/heatmap', function(req,res) {
   // fill p here _.each(blabla, p.push(gethistory(someparams...)
   var result = [];
   function getHistory(params) {
        var options = { ...};
        var callback = function(error, response, body) {
            if(error) { //wtv 
            } else {
              // what to do with the body here ? return body ? result.push(body) ?
            }
        }

        request(options, callback);
    }

    Q.all(p).then(function() {
    });
}

所以这里的问题是我在完成所有请求时,将所有内容放入数组/对象中,然后将整个内容发送到客户端。如何让getHistory返回获取的值(在请求完成后)。

希望很清楚。

3 个答案:

答案 0 :(得分:3)

这里的核心问题是node.js样式的回调和promise是不兼容的。 Promise强调返回值,node强调回调。

因此,您需要一种适当地包装节点的回调约定的适配器,这个过程称为Promisifcation。这可以手动完成,但是当你不小心时,它最好是单调乏味而且容易出错。幸运的是,由于节点的约定已经完善,因此可以实现自动化。 Q有a few helpers,但Bluebird在这方面更方便。

因此,简单的方法是切换到Bluebird作为promise库并使用OrderByDescending()

promisifyAll

答案 1 :(得分:1)

FWIW,这是另一个答案,显示了使用Q正确完成后的情况:

// promisified request
function requestAsync(options) {
    var result = Q.defer();
    request(options, function(error, response, body) {
        if (error) {
            result.reject(error);
        } else {
            result.resolve(body);
        }
    });
    return result.promise;
}

// returns promises for heatmapVolumes
function getHistory(params) {
    return requestAsync({
        method: 'GET',
        url: 'https://api.kaiko.com/v1/charts/' + 
            encodeURIComponent(params.exchange) + '/' + 
            encodeURIComponent(params.pair) +'/history',
        qs: params.qs,
        headers: {
            "Content-Type": "application/json",
            "Accept": "application/json"
        }
    }).then(function (body) {
        return heatmapVolume(body, params.exchange, params.pair);
    }).catch(function (error) {
        // log detailed error and send less revealing message downstream
        console.error('error fetching trades', error);
        throw new Error('Something went wrong while fetching trades');
    });
}

// usage
this.app.all('/api/heatmap', function(req, res) {
    getHistory({
        exchange: "foo", pair: "bar", qs: "qux"
    }).then(function (heatmap) {
        res.send(200, heatmap);
    }).catch(function (error) {
        res.send(500, error);
    });
});

答案 2 :(得分:0)

使用Q.deferred,它按照文档\ o /

的方式工作
function getHistory(params) {
                var deferred = Q.defer();
                var options = {
                    method: 'GET',
                    url: 'https://api.kaiko.com/v1/charts/' + params.exchange + '/' + params.pair +'/history',
                    qs:qs,
                    headers: {
                        "Content-Type": "application/json",
                        "Accept": "application/json"
                    }
                }

                var callback = function(error, response, body) {
                    if(error) {
                        console.log('error fetching trades', error);
                        res.send(500, 'Something went wrong while fetching trades');
                    } else {
                        var body = heatmapVolume(body, params.exchange, params.pair);
                        // console.log("result!", body.exchange, body.pair);
                        result.push(body);
                        // return body;
                        deferred.resolve();
                    }
                }

                request(options, callback);

                return deferred.promise;
            }