2020-02-19 15:55:06 +11:00
|
|
|
/**
|
|
|
|
NOTE: This is not a database table, this is a model of a Token object that can be created/loaded
|
|
|
|
and then has abilities after that.
|
|
|
|
*/
|
|
|
|
|
|
|
|
const _ = require('lodash');
|
|
|
|
const jwt = require('jsonwebtoken');
|
|
|
|
const crypto = require('crypto');
|
|
|
|
const error = require('../lib/error');
|
|
|
|
const ALGO = 'RS256';
|
|
|
|
|
2021-02-08 16:46:35 -05:00
|
|
|
let public_key = null;
|
|
|
|
let private_key = null;
|
|
|
|
|
|
|
|
function checkJWTKeyPair() {
|
|
|
|
if (!public_key || !private_key) {
|
|
|
|
let config = require('config');
|
|
|
|
public_key = config.get('jwt.pub');
|
|
|
|
private_key = config.get('jwt.key');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-02-19 15:55:06 +11:00
|
|
|
module.exports = function () {
|
|
|
|
|
|
|
|
let token_data = {};
|
|
|
|
|
|
|
|
let self = {
|
|
|
|
/**
|
|
|
|
* @param {Object} payload
|
|
|
|
* @returns {Promise}
|
|
|
|
*/
|
|
|
|
create: (payload) => {
|
|
|
|
// sign with RSA SHA256
|
|
|
|
let options = {
|
|
|
|
algorithm: ALGO,
|
|
|
|
expiresIn: payload.expiresIn || '1d'
|
|
|
|
};
|
|
|
|
|
|
|
|
payload.jti = crypto.randomBytes(12)
|
|
|
|
.toString('base64')
|
|
|
|
.substr(-8);
|
|
|
|
|
2021-02-08 16:46:35 -05:00
|
|
|
checkJWTKeyPair();
|
|
|
|
|
2020-02-19 15:55:06 +11:00
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
jwt.sign(payload, private_key, options, (err, token) => {
|
|
|
|
if (err) {
|
|
|
|
reject(err);
|
|
|
|
} else {
|
|
|
|
token_data = payload;
|
|
|
|
resolve({
|
|
|
|
token: token,
|
|
|
|
payload: payload
|
|
|
|
});
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {String} token
|
|
|
|
* @returns {Promise}
|
|
|
|
*/
|
|
|
|
load: function (token) {
|
|
|
|
return new Promise((resolve, reject) => {
|
2021-02-08 16:46:35 -05:00
|
|
|
checkJWTKeyPair();
|
2020-02-19 15:55:06 +11:00
|
|
|
try {
|
|
|
|
if (!token || token === null || token === 'null') {
|
|
|
|
reject(new error.AuthError('Empty token'));
|
|
|
|
} else {
|
|
|
|
jwt.verify(token, public_key, {ignoreExpiration: false, algorithms: [ALGO]}, (err, result) => {
|
|
|
|
if (err) {
|
|
|
|
|
|
|
|
if (err.name === 'TokenExpiredError') {
|
|
|
|
reject(new error.AuthError('Token has expired', err));
|
|
|
|
} else {
|
|
|
|
reject(err);
|
|
|
|
}
|
|
|
|
|
|
|
|
} else {
|
|
|
|
token_data = result;
|
|
|
|
|
|
|
|
// Hack: some tokens out in the wild have a scope of 'all' instead of 'user'.
|
|
|
|
// For 30 days at least, we need to replace 'all' with user.
|
|
|
|
if ((typeof token_data.scope !== 'undefined' && _.indexOf(token_data.scope, 'all') !== -1)) {
|
|
|
|
//console.log('Warning! Replacing "all" scope with "user"');
|
|
|
|
|
|
|
|
token_data.scope = ['user'];
|
|
|
|
}
|
|
|
|
|
|
|
|
resolve(token_data);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
} catch (err) {
|
|
|
|
reject(err);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Does the token have the specified scope?
|
|
|
|
*
|
|
|
|
* @param {String} scope
|
|
|
|
* @returns {Boolean}
|
|
|
|
*/
|
|
|
|
hasScope: function (scope) {
|
|
|
|
return typeof token_data.scope !== 'undefined' && _.indexOf(token_data.scope, scope) !== -1;
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {String} key
|
|
|
|
* @return {*}
|
|
|
|
*/
|
|
|
|
get: function (key) {
|
|
|
|
if (typeof token_data[key] !== 'undefined') {
|
|
|
|
return token_data[key];
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {String} key
|
|
|
|
* @param {*} value
|
|
|
|
*/
|
|
|
|
set: function (key, value) {
|
|
|
|
token_data[key] = value;
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param [default_value]
|
|
|
|
* @returns {Integer}
|
|
|
|
*/
|
|
|
|
getUserId: (default_value) => {
|
|
|
|
let attrs = self.get('attrs');
|
|
|
|
if (attrs && typeof attrs.id !== 'undefined' && attrs.id) {
|
|
|
|
return attrs.id;
|
|
|
|
}
|
|
|
|
|
|
|
|
return default_value || 0;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
return self;
|
|
|
|
};
|