Как включить функцию обратного вызова с Promise

Я использую обещания Bluebird и пытаюсь получить функцию ниже при работе Promisify:

var jwt = require('jsonwebtoken');

function _test_encode() {
    var cert = fs.readFileSync('public.pub'); 
    return jwt.verify(token, cert, function(err, decoded) {
        console.log(decoded);
    });
}

Я попробовал ниже:

var jwt = require('jsonwebtoken');
var Promise = require('bluebird');
var jwtVerifyAsync = Promise.promisify(jwt.verify);

function _test_encode() {
    var cert = fs.readFileSync('public.pub');  // get public key
    return jwtVerifyAsync(token, cert).then(function(decoded) {
        console.log(decoded);
    });
}

Но я получаю:

Possibly unhandled TypeError: Object #<Object> has no method 'decode'

Есть ли что-то, чего мне здесь не хватает, чтобы мне это работало? Я просто хочу, чтобы функция _test_encode возвращала результат обещания и использовала его в других моих функциях.

Проверьте функцию ниже:

module.exports.verify = function(jwtString, secretOrPublicKey, options, callback) {
  if ((typeof options === 'function') && !callback) {
    callback = options;
    options = {};
  }

  if (!options) options = {};

  var done;

  if (callback) {
    done = function() {
      var args = Array.prototype.slice.call(arguments, 0);
      return process.nextTick(function() {
        callback.apply(null, args);
      });
    };
  } else {
    done = function(err, data) {
      if (err) throw err;
      return data;
    };
  }

  if (!jwtString){
    return done(new JsonWebTokenError('jwt must be provided'));
  }

  var parts = jwtString.split('.');

  if (parts.length !== 3){
    return done(new JsonWebTokenError('jwt malformed'));
  }

  if (parts[2].trim() === '' && secretOrPublicKey){
    return done(new JsonWebTokenError('jwt signature is required'));
  }

  var valid;

  try {
    valid = jws.verify(jwtString, secretOrPublicKey);
  } catch (e) {
    return done(e);
  }

  if (!valid)
    return done(new JsonWebTokenError('invalid signature'));

  var payload;

  try {
   payload = this.decode(jwtString);
  } catch(err) {
    return done(err);
  }

  if (typeof payload.exp !== 'undefined') {
    if (typeof payload.exp !== 'number') {
      return done(new JsonWebTokenError('invalid exp value'));
    }
    if (Math.floor(Date.now() / 1000) >= payload.exp)
      return done(new TokenExpiredError('jwt expired', new Date(payload.exp * 1000)));
  }

  if (options.audience) {
    var audiences = Array.isArray(options.audience)? options.audience : [options.audience];
    var target = Array.isArray(payload.aud) ? payload.aud : [payload.aud];

    var match = target.some(function(aud) { return audiences.indexOf(aud) != -1; });

    if (!match)
      return done(new JsonWebTokenError('jwt audience invalid. expected: ' + payload.aud));
  }

  if (options.issuer) {
    if (payload.iss !== options.issuer)
      return done(new JsonWebTokenError('jwt issuer invalid. expected: ' + payload.iss));
  }

  return done(null, payload);
};

3 ответа

Решение

Пытается позвонить this.decode, но вы удалили контекст, пообещав его, так this не jwt как это и ожидается. Вы можете сохранить контекст через обещание, передав его в качестве второго аргумента:

var jwtVerifyAsync = Promise.promisify(jwt.verify, jwt);

Это старый пост, но для тех, кто читает это сейчас, знайте, какую версию Bluebird вы используете.

Начиная с версии 3.0, второй аргумент, передаваемый в метод promisify, является объектом параметров. Смотрите здесь.

Этот код теперь должен быть:

var jwtVerifyAsync = Promise.promisify(jwt.verify, {context: jwt})

Я просто использовал jwt.verify(token, secret)без . Это сработало просто отлично. Я полагаю, что с последней версией JWT вам не нужно promisify.

Другие вопросы по тегам