You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							239 lines
						
					
					
						
							6.5 KiB
						
					
					
				
			
		
		
	
	
							239 lines
						
					
					
						
							6.5 KiB
						
					
					
				var jws = require('jws');
 | 
						|
var ms = require('ms');
 | 
						|
 | 
						|
var JWT = module.exports;
 | 
						|
 | 
						|
var JsonWebTokenError = JWT.JsonWebTokenError = require('./lib/JsonWebTokenError');
 | 
						|
var TokenExpiredError = JWT.TokenExpiredError = require('./lib/TokenExpiredError');
 | 
						|
var ms = require('ms')
 | 
						|
 | 
						|
JWT.decode = function (jwt, options) {
 | 
						|
  options = options || {};
 | 
						|
  var decoded = jws.decode(jwt, options);
 | 
						|
  if (!decoded) { return null; }
 | 
						|
  var payload = decoded.payload;
 | 
						|
 | 
						|
  //try parse the payload
 | 
						|
  if(typeof payload === 'string') {
 | 
						|
    try {
 | 
						|
      var obj = JSON.parse(payload);
 | 
						|
      if(typeof obj === 'object') {
 | 
						|
        payload = obj;
 | 
						|
      }
 | 
						|
    } catch (e) { }
 | 
						|
  }
 | 
						|
 | 
						|
  //return header if `complete` option is enabled.  header includes claims
 | 
						|
  //such as `kid` and `alg` used to select the key within a JWKS needed to
 | 
						|
  //verify the signature
 | 
						|
  if (options.complete === true) {
 | 
						|
    return {
 | 
						|
      header: decoded.header,
 | 
						|
      payload: payload,
 | 
						|
      signature: decoded.signature
 | 
						|
    };
 | 
						|
  }
 | 
						|
  return payload;
 | 
						|
};
 | 
						|
 | 
						|
JWT.sign = function(payload, secretOrPrivateKey, options, callback) {
 | 
						|
  options = options || {};
 | 
						|
 | 
						|
  var header = {};
 | 
						|
 | 
						|
  if (typeof payload === 'object') {
 | 
						|
    header.typ = 'JWT';
 | 
						|
  }
 | 
						|
 | 
						|
  header.alg = options.algorithm || 'HS256';
 | 
						|
 | 
						|
  if (options.headers) {
 | 
						|
    Object.keys(options.headers).forEach(function (k) {
 | 
						|
      header[k] = options.headers[k];
 | 
						|
    });
 | 
						|
  }
 | 
						|
 | 
						|
  var timestamp = Math.floor(Date.now() / 1000);
 | 
						|
  if (!options.noTimestamp) {
 | 
						|
    payload.iat = payload.iat || timestamp;
 | 
						|
  }
 | 
						|
 | 
						|
  if (options.expiresInSeconds || options.expiresInMinutes) {
 | 
						|
    var deprecated_line;
 | 
						|
    try {
 | 
						|
      deprecated_line = /.*\((.*)\).*/.exec((new Error()).stack.split('\n')[2])[1];
 | 
						|
    } catch(err) {
 | 
						|
      deprecated_line = '';
 | 
						|
    }
 | 
						|
 | 
						|
    console.warn('jsonwebtoken: expiresInMinutes and expiresInSeconds is deprecated. (' + deprecated_line + ')\n' +
 | 
						|
                 'Use "expiresIn" expressed in seconds.');
 | 
						|
 | 
						|
    var expiresInSeconds = options.expiresInMinutes ?
 | 
						|
        options.expiresInMinutes * 60 :
 | 
						|
        options.expiresInSeconds;
 | 
						|
 | 
						|
    payload.exp = timestamp + expiresInSeconds;
 | 
						|
  } else if (options.expiresIn) {
 | 
						|
    if (typeof options.expiresIn === 'string') {
 | 
						|
      var milliseconds = ms(options.expiresIn);
 | 
						|
      if (typeof milliseconds === 'undefined') {
 | 
						|
        throw new Error('bad "expiresIn" format: ' + options.expiresIn);
 | 
						|
      }
 | 
						|
      payload.exp = timestamp + milliseconds / 1000;
 | 
						|
    } else if (typeof options.expiresIn === 'number' ) {
 | 
						|
      payload.exp = timestamp + options.expiresIn;
 | 
						|
    } else {
 | 
						|
      throw new Error('"expiresIn" should be a number of seconds or string representing a timespan eg: "1d", "20h", 60');
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  if (options.audience)
 | 
						|
    payload.aud = options.audience;
 | 
						|
 | 
						|
  if (options.issuer)
 | 
						|
    payload.iss = options.issuer;
 | 
						|
 | 
						|
  if (options.subject)
 | 
						|
    payload.sub = options.subject;
 | 
						|
 | 
						|
  var encoding = 'utf8';
 | 
						|
  if (options.encoding) {
 | 
						|
    encoding = options.encoding;
 | 
						|
  }
 | 
						|
 | 
						|
  if(typeof callback === 'function') {
 | 
						|
    jws.createSign({
 | 
						|
      header: header,
 | 
						|
      privateKey: secretOrPrivateKey,
 | 
						|
      payload: JSON.stringify(payload)
 | 
						|
    }).on('done', callback);
 | 
						|
  } else {
 | 
						|
    return jws.sign({header: header, payload: payload, secret: secretOrPrivateKey, encoding: encoding});
 | 
						|
  }
 | 
						|
};
 | 
						|
 | 
						|
JWT.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'));
 | 
						|
  }
 | 
						|
 | 
						|
  if (!secretOrPublicKey) {
 | 
						|
    return done(new JsonWebTokenError('secret or public key must be provided'));
 | 
						|
  }
 | 
						|
 | 
						|
  if (!options.algorithms) {
 | 
						|
    options.algorithms = ~secretOrPublicKey.toString().indexOf('BEGIN CERTIFICATE') ||
 | 
						|
                         ~secretOrPublicKey.toString().indexOf('BEGIN PUBLIC KEY') ?
 | 
						|
                          [ 'RS256','RS384','RS512','ES256','ES384','ES512' ] :
 | 
						|
                         ~secretOrPublicKey.toString().indexOf('BEGIN RSA PUBLIC KEY') ?
 | 
						|
                          [ 'RS256','RS384','RS512' ] :
 | 
						|
                          [ 'HS256','HS384','HS512' ];
 | 
						|
 | 
						|
  }
 | 
						|
 | 
						|
  var decodedToken;
 | 
						|
  try {
 | 
						|
    decodedToken = jws.decode(jwtString);
 | 
						|
  } catch(err) {
 | 
						|
    return done(new JsonWebTokenError('invalid token'));
 | 
						|
  }
 | 
						|
 | 
						|
  if (!decodedToken) {
 | 
						|
    return done(new JsonWebTokenError('invalid token'));
 | 
						|
  }
 | 
						|
 | 
						|
  var header = decodedToken.header;
 | 
						|
 | 
						|
  if (!~options.algorithms.indexOf(header.alg)) {
 | 
						|
    return done(new JsonWebTokenError('invalid algorithm'));
 | 
						|
  }
 | 
						|
 | 
						|
  var valid;
 | 
						|
 | 
						|
  try {
 | 
						|
    valid = jws.verify(jwtString, header.alg, secretOrPublicKey);
 | 
						|
  } catch (e) {
 | 
						|
    return done(e);
 | 
						|
  }
 | 
						|
 | 
						|
  if (!valid)
 | 
						|
    return done(new JsonWebTokenError('invalid signature'));
 | 
						|
 | 
						|
  var payload;
 | 
						|
 | 
						|
  try {
 | 
						|
    payload = JWT.decode(jwtString);
 | 
						|
  } catch(err) {
 | 
						|
    return done(err);
 | 
						|
  }
 | 
						|
 | 
						|
  if (typeof payload.exp !== 'undefined' && !options.ignoreExpiration) {
 | 
						|
    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: ' + audiences.join(' or ')));
 | 
						|
  }
 | 
						|
 | 
						|
  if (options.issuer) {
 | 
						|
    if (payload.iss !== options.issuer)
 | 
						|
      return done(new JsonWebTokenError('jwt issuer invalid. expected: ' + options.issuer));
 | 
						|
  }
 | 
						|
 | 
						|
  if (options.maxAge) {
 | 
						|
    var maxAge = ms(options.maxAge);
 | 
						|
    if (typeof payload.iat !== 'number') {
 | 
						|
      return done(new JsonWebTokenError('iat required when maxAge is specified'));
 | 
						|
    }
 | 
						|
    if (Date.now() - (payload.iat * 1000) > maxAge) {
 | 
						|
      return done(new TokenExpiredError('maxAge exceeded', new Date(payload.iat * 1000 + maxAge)));
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  return done(null, payload);
 | 
						|
};
 |