Node.js承诺q不能按预期工作

时间:2014-11-04 09:38:27

标签: javascript node.js promise q

这是我的方法:

var q = require('q');
var request = require("request");

exports.route = function (req, res) {
    var userId = req.query.userId;

    if (!userId) {
        res.send(JSON.stringify({
            errorCode: 400,
            error: "userId is missing, please form your query like <url>/avatar?userId=00.42.1"
        }), 400);
        return;
    }
    getAvatar(userId).then(function (success) {
            console.log('success');
            res.json(success);
        }, function (error) {
            console.log('error');
            res.json({error: error});
        }
    );


};

function getAvatar(userId) {
    var isErrorInResponse = function (error, response) {
        return typeof error !== 'undefined' || error == null || response.statusCode > 202;
    };

    var deferred = q.defer();
    var url = "http://localhost/avatar" + userId;
    console.log(url);
    request(url, function (error, response, body) {
        if (isErrorInResponse(error, response)) {
            deferred.reject(new Error({error: error, response: response, body: body}));
        } else {
            deferred.resolve(body);
        }
    });
    return deferred.promise;
};
日志后面的

GET localhost:8090/avatar?userId=42 produces

http://localhost/avatar/42
error

并将此响应发送给客户端:

{
  "error": {}
}

这是我的版本,我无法更新:

 "dependencies": {
    "consolidate": "0.9.0",
    "express": "3.1.0",
    "multipart-parser": "0.0.2",
    "mustache": "0.7.2",
    "poplib": "0.1.5",
    "q": "0.9.3",
    "q-io": "1.6.x",
    "request": "2.27.0",
    "string": "^2.2.0",
    "xml2js": "0.2.8"
  }

问题是,为什么承诺没有收到完整的错误,我是在deferred.reject(...发送的,我有什么要改变以便它能够正常工作?

1 个答案:

答案 0 :(得分:3)

deferred.reject(new Error({error: error, response: response, body: body}));

Error构造函数接受字符串参数,而不是对象。传入除字符串以外的任何内容将导致参数转换为字符串,对于普通对象[object Object]。由于错误对象没有任何可枚举属性,因此JSON.stringify会导致{}。你需要的是将一个字符串传递给Error构造函数并访问message属性。

拒绝:

deferred.reject(new Error(error));

响应:

getAvatar(userId).then(function (success) {
            console.log('success');
            res.json(success);
        }, function (error) {
            console.log('error');
            res.json({error: error.message});
        }
   );

如果您希望错误消息成为对象,则必须对其进行字符串化。或者,您可以将这些属性显式添加到错误对象中,并在错误为JSON时显示它们。字符串化,但这也会显示stack属性,因为{{1}添加它以支持长堆栈跟踪。