Node.js在返回之前承诺函数冻结执行

时间:2016-07-20 20:55:54

标签: javascript node.js express

node.js应用程序需要使用Promise来确保在运行后续命令之前已返回一个函数的结果。但是,下面的代码在getUser(theToken) promise函数内部应该返回其值时冻结。 为了使getUser(theToken) promise函数能够返回内部生成的有效值,需要对下面的代码进行哪些具体更改?

app.get('/auth/provider/callback', function(req, res) {
    var queryData = url.parse(req.url, true).query;
    //This is the end of Step C in the flow
    // code was in the GET request, ex: http://localhost:8080/login?code=w6zGJO&state=IuaO63
    console.log('Finished Step C of Authorization Code Flow.  Welcome back from authserver');
    console.log('queryData.code is: ' + queryData.code);
    //Now start Step D in the flow
    console.log('About to start Step D of Authorization Code Flow. ');
    var tokenUrl = oAuthConfig.tokenUrl + '?grant_type=' + oAuthConfig.grantType + '&code='+queryData.code+'&client_id=' + oAuthConfig.clientID + '&redirect_uri=' + oAuthConfig.callbackUrl;
    request.post(tokenUrl, function(err, response, body){
        if(err){console.log('ERROR with token request.')}
        // Step E is the response received as follows:
        var data = JSON.parse(body);
        console.log('JSON data response is: ' + data);
        getUser(data.access_token).then(function(message) {
            console.log('in callback, jwtJSON is: ');console.log(message);
            res.redirect(somewhere);
        });
    });
    console.log('The POST is finished.  Waiting for response.');
    console.log('Step D.  Token Request Sent To AuthServer.');
});

然后同一文件中的getUser(theToken)函数是:

function getUser(theToken) {//returns JWT
  return new Promise(function(resolve, reject) {
    console.log('in getUser, theToken is: ');console.log(theToken);
    var jwtJSON = { "token" : "anonymous" };
    request({
        method: 'GET', url: authServer + '/uaa/user', json: true, 
        auth: { user: null, password: null, sendImmediately: true, bearer: theToken }
    }, function (error, response, body) {
        if(error){
          console.log('in getUser, ERROR with user request.');console.log(error);
          jwtJSON = { "token" : "error" };
          return jwtJSON;//
        }
        else {
            var uname = '';var jwtUser = 'empty';
            console.log('in getUser, response.statusCode is: ');console.log(response.statusCode);
            if(body['name'] && body['name'].length > 0 ){ 
                uname = body['name'];console.log('uname is: ');console.log(uname);
                client.exists(uname, function(err, reply) {//Check to see if a Redis key for the user already exists
                    if (reply === 1) {//a redis key DOES exist
                        console.log('\"'+uname+'\" exists');
                        client.expire(uname, 10);//set the key to expire in 10 seconds.  use this to manage session length
                        client.hgetall(uname, function(err, object) {
                            if(object && object["jwt"]) {
                                jwtJSON = { "token" : object["jwt"] };
                                console.log('in getUser, jwtJSON is: ');console.log(jwtJSON);
                                return jwtJSON;
                            } else { return jwtJSON; }
                        });
                    } else {//a redis key DOES NOT exist
                        console.log('\"'+uname+'\" doesn\'t exist');
                        jwtUser = generateJwt(uname, authoritiesLocal);
                        client.hmset(uname, {//store a hash/object
                            'jwt' : jwtUser
                         });
                        jwtJSON = { "token" : jwtUser };console.log('in getUser, jwtJSON is: ');console.log(jwtJSON);
                        client.expire(uname, 10);//set the key to expire in 10 seconds.  use this to manage session length
                        return jwtJSON;
                    }
                });//end of Redis conditional block
                console.log('jwtJSON is: ');console.log(jwtJSON);
            } else { return jwtJSON; }
        };
    });
  });
};

控制台打印输出为:

Finished Step C of Authorization Code Flow.  Welcome back from authserver
queryData.code is: 0oxzu6
About to start Step D of Authorization Code Flow. 
The POST is finished.  Waiting for response.
Step D.  Token Request Sent To AuthServer.
JSON data response is: [object Object]
in getUser, theToken is: 
eyJ_SomeLongToken_hbG
in getUser, response.statusCode is: 
200
uname is: 
user
jwtJSON is: 
{ token: 'anonymous' }
"user" doesn't exist
in getUser, jwtJSON is: 
{ token: 'eyJ_TheSameLongToken_hbG' }

然后程序执行在序列第二次运行之前停止几秒钟,第二次产生错误,因为第二次没有任何变量填充。

1 个答案:

答案 0 :(得分:3)

您需要使用resolvereject参数,而不是返回值。我已使用resolvereject替换了您的退货声明。然后传入应该在then

的回调函数中可用的数据
function getUser(theToken) { //returns JWT
    return new Promise(function(resolve, reject) {
        console.log('in getUser, theToken is: ');
        console.log(theToken);
        var jwtJSON = {
            "token": "anonymous"
        };
        request({
            method: 'GET',
            url: authServer + '/uaa/user',
            json: true,
            auth: {
                user: null,
                password: null,
                sendImmediately: true,
                bearer: theToken
            }
        }, function(error, response, body) {
            if (error) {
                console.log('in getUser, ERROR with user request.');
                console.log(error);
                jwtJSON = {
                    "token": "error"
                };

                /* REJECT INSTEAD OF RETURN */
                reject(jwtJSON);
            } else {
                var uname = '';
                var jwtUser = 'empty';
                console.log('in getUser, response.statusCode is: ');
                console.log(response.statusCode);
                if (body.name && body.name.length > 0) {
                    uname = body.name;
                    console.log('uname is: ');
                    console.log(uname);
                    client.exists(uname, function(err, reply) { //Check to see if a Redis key for the user already exists
                        if (reply === 1) { //a redis key DOES exist
                            console.log('\"' + uname + '\" exists');
                            client.expire(uname, 10); //set the key to expire in 10 seconds.  use this to manage session length
                            client.hgetall(uname, function(err, object) {
                                if (object && object.jwt) {
                                    jwtJSON = {
                                        "token": object.jwt
                                    };
                                    console.log('in getUser, jwtJSON is: ');
                                    console.log(jwtJSON);
                                    /* RESOLVE INSTEAD OF RETURN */
                                    resolve(jwtJSON);
                                } else {
                                    /* RESOLVE INSTEAD OF RETURN */
                                    resolve(jwtJSON);
                                }
                            });
                        } else { //a redis key DOES NOT exist
                            console.log('\"' + uname + '\" doesn\'t exist');
                            jwtUser = generateJwt(uname, authoritiesLocal);
                            client.hmset(uname, { //store a hash/object
                                'jwt': jwtUser
                            });
                            jwtJSON = {
                                "token": jwtUser
                            };
                            console.log('in getUser, jwtJSON is: ');
                            console.log(jwtJSON);
                            client.expire(uname, 10); //set the key to expire in 10 seconds.  use this to manage session length

                            /* RESOLVE INSTEAD OF RETURN */
                            resolve(jwtJSON);
                        }
                    }); //end of Redis conditional block
                    console.log('jwtJSON is: ');
                    console.log(jwtJSON);
                } else {
                    /* RESOLVE INSTEAD OF RETURN */
                    resolve(jwtJSON);
                }
            }
        });
    });
}