我做的承诺错了......我在这里想念的是什么?

时间:2017-01-22 18:24:10

标签: typescript promise jwt aurelia es6-promise

我有一个 token.ts 文件导出1个函数:

import * as jwt from 'jsonwebtoken';
import { db, dbUserLevel } from '../util/db';


export function genToken(username, password): Object {

    let token: Object;

    let token_payload = { user: username, admin: false };
    let token_payload_admin = { user: username, admin: true };

    // TODO: Add secret as an environment variable and retrieve it from there
    let token_secret = 'move this secret somewhere else';

    let token_header = {
        issuer: 'SomeIssuer',
        algorithm: 'HS256',
        expiresIn: '1h'
    };

    db.open().then(() => { dbUserLevel('admin') }).then(() => {
        db.collection('users').findOne({ username: username, password: password })
            .then((result) => {
                if (result.isAdmin === 1) {
                    this.token = { access_token: jwt.sign(token_payload_admin, token_secret, token_header) }
                } else if (result.isAdmin === 0) {
                    this.token = { access_token: jwt.sign(token_payload, token_secret, token_header) }
                }
                db.close();
            })
    })
    return this.token;
};

我有另一个文件 login.ts ,它从token.ts文件中导入genToken函数:

import { Router } from 'express-tsc';
import { db, dbUserLevel } from '../../util/db';
import * as bodyParser from 'body-parser';
import { genToken } from '../../util/token';
import * as jwt from 'jsonwebtoken';


export var router = Router();

let urlencodedParser = bodyParser.urlencoded({ extended: false });
let jsonParser = bodyParser.json();


router.post('/', jsonParser, (req, res) => {

    req.accepts(['json', 'text/plain']);
    let data = req.body;
    console.log(data);

    let username: string = data["username"];
    let password: string = data["password"];

    let token = genToken(username, password);

    console.log(JSON.stringify(token));
    res.send(200, token);



});

应该发生的是当我从login.ts文件提交表单时,它会将响应发布到服务器并调用genToken(用户名,密码)函数并返回令牌对象。

出于某种原因,不确定为什么,当我第一次提交表单时,“token”(login.ts)是未定义的。如果我再次提交表单,则返回令牌对象并按预期记录到控制台....

任何人都知道为什么会这样吗?如果我没有提供足够的信息,请告诉我您可能需要的信息,以便我更新帖子!谢谢!

修改

根据接受答案中提供的信息,我提出了以下针对我最初问题的更改:

token.ts:

...

    let token: Object;

    let query = db.open()
        .then(() => dbUserLevel('user'))
        .then(() => db.collection('users').findOne({ username: username, password: password })
            .then((result) => {
                if (result.isAdmin === 1) {
                    token = { access_token: jwt.sign(token_payload_admin, token_secret, token_header) }
                } else if (result.isAdmin === 0) {
                    token = { access_token: jwt.sign(token_payload, token_secret, token_header) }
                }
            })
            .catch(err => {
                db.close();
                Promise.reject(err);
            }))
        .then(() => {
            db.close();
            Promise.resolve(token);
            return token;
        })
        .catch(err => {
            db.close();
            Promise.reject(err);
            return err;
        });

    return query;
};

login.ts:

...
    genToken(username, password)
        .then(token => {
            res.status(200).send(token);
        })
        .catch(err => {
            res.status(500).send(err);
        });

});

1 个答案:

答案 0 :(得分:4)

据我所知,您遇到的问题是,当您尝试以同步方式读取令牌时,令牌会以异步方式生成。

你的genToken方法应该返回一个Promise,你应该在该promise被解决后发送你的请求。类似的东西:

getToken(...).then((token) => {
    res.send(200, token);
}

您可以找到一些示例here