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);
|
|
};
|