UNPKG

keycloak-verify

Version:

Javascript backend library to verify keycloak tokens and get user info

153 lines (123 loc) 6.84 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports["default"] = void 0; var _axios = _interopRequireDefault(require("axios")); var _ramda = require("ramda"); var _jsonwebtoken = _interopRequireDefault(require("jsonwebtoken")); var _jwkToPem = _interopRequireDefault(require("jwk-to-pem")); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { "default": obj }; } function asyncGeneratorStep(gen, resolve, reject, _next, _throw, key, arg) { try { var info = gen[key](arg); var value = info.value; } catch (error) { reject(error); return; } if (info.done) { resolve(value); } else { Promise.resolve(value).then(_next, _throw); } } function _asyncToGenerator(fn) { return function () { var self = this, args = arguments; return new Promise(function (resolve, reject) { var gen = fn.apply(self, args); function _next(value) { asyncGeneratorStep(gen, resolve, reject, _next, _throw, "next", value); } function _throw(err) { asyncGeneratorStep(gen, resolve, reject, _next, _throw, "throw", err); } _next(undefined); }); }; } function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); if (enumerableOnly) symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; }); keys.push.apply(keys, symbols); } return keys; } function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i] != null ? arguments[i] : {}; if (i % 2) { ownKeys(source, true).forEach(function (key) { _defineProperty(target, key, source[key]); }); } else if (Object.getOwnPropertyDescriptors) { Object.defineProperties(target, Object.getOwnPropertyDescriptors(source)); } else { ownKeys(source).forEach(function (key) { Object.defineProperty(target, key, Object.getOwnPropertyDescriptor(source, key)); }); } } return target; } function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; } function _objectWithoutProperties(source, excluded) { if (source == null) return {}; var target = _objectWithoutPropertiesLoose(source, excluded); var key, i; if (Object.getOwnPropertySymbols) { var sourceSymbolKeys = Object.getOwnPropertySymbols(source); for (i = 0; i < sourceSymbolKeys.length; i++) { key = sourceSymbolKeys[i]; if (excluded.indexOf(key) >= 0) continue; if (!Object.prototype.propertyIsEnumerable.call(source, key)) continue; target[key] = source[key]; } } return target; } function _objectWithoutPropertiesLoose(source, excluded) { if (source == null) return {}; var target = {}; var sourceKeys = Object.keys(source); var key, i; for (i = 0; i < sourceKeys.length; i++) { key = sourceKeys[i]; if (excluded.indexOf(key) >= 0) continue; target[key] = source[key]; } return target; } var cache = {}; var verifyOnline = function verifyOnline(_ref) { var realm = _ref.realm, authServerUrl = _ref.authServerUrl; return function (accessToken) { var options = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {}; return _axios["default"].get("".concat(authServerUrl, "/auth/realms/").concat(options.realm || realm, "/protocol/openid-connect/userinfo"), { headers: { Authorization: "Bearer ".concat(accessToken) } }).then((0, _ramda.prop)("data")).then(makeUser); }; }; var makeUser = function makeUser(_ref2) { var sub = _ref2.sub, preferred_username = _ref2.preferred_username, email_verified = _ref2.email_verified, resource_access = _ref2.resource_access, email = _ref2.email, name = _ref2.name, others = _objectWithoutProperties(_ref2, ["sub", "preferred_username", "email_verified", "resource_access", "email", "name"]); return _objectSpread({ id: sub, userName: preferred_username, emailVerified: email_verified, resourceAccess: resource_access, email: email, name: name }, others); }; var verify = (0, _ramda.curryN)(2)(_jsonwebtoken["default"].verify); var isTheRightKid = function isTheRightKid(kid) { return function (publicKey) { return publicKey.kid === kid; }; }; var findPublicKeyFromKid = function findPublicKeyFromKid(publicKey) { return function (kid) { return (0, _ramda.find)(isTheRightKid(kid))(publicKey); }; }; var getKid = (0, _ramda.path)(["header", "kid"]); var decode = (0, _ramda.compose)((0, _ramda.curryN)(2), _ramda.flip)(_jsonwebtoken["default"].decode); var getUserFromPublicKey = function getUserFromPublicKey(token) { return (0, _ramda.compose)(makeUser, verify(token)); }; var getUserFromJWK = function getUserFromJWK(token) { return function (jwk) { return (0, _ramda.compose)(getUserFromPublicKey(token), _jwkToPem["default"], findPublicKeyFromKid(jwk), getKid, decode({ complete: true }))(token); }; }; var fetchPublicKeys = function fetchPublicKeys(_ref3) { var realm = _ref3.realm, authServerUrl = _ref3.authServerUrl, useCache = _ref3.useCache; var url = "".concat(authServerUrl, "/auth/realms/").concat(realm, "/protocol/openid-connect/certs"); var key = url; if (useCache) { return cache[key] ? Promise.resolve(cache[key]) : _axios["default"].get(url).then((0, _ramda.path)(["data", "keys"])).then(function (publicKey) { cache[key] = publicKey; return publicKey; }); } else { return _axios["default"].get(url).then((0, _ramda.path)(["data", "keys"])); } }; var verifyOffline = function verifyOffline(config) { return ( /*#__PURE__*/ function () { var _ref4 = _asyncToGenerator( /*#__PURE__*/ regeneratorRuntime.mark(function _callee(accessToken) { var options, publicKey, _args = arguments; return regeneratorRuntime.wrap(function _callee$(_context) { while (1) { switch (_context.prev = _context.next) { case 0: options = _args.length > 1 && _args[1] !== undefined ? _args[1] : {}; publicKey = config.publicKey; return _context.abrupt("return", publicKey ? getUserFromPublicKey(accessToken)(publicKey) : fetchPublicKeys(_objectSpread({}, config, {}, options)).then(getUserFromJWK(accessToken))); case 3: case "end": return _context.stop(); } } }, _callee); })); return function (_x) { return _ref4.apply(this, arguments); }; }() ); }; var Keycloak = function Keycloak(config) { return { verifyOnline: verifyOnline(config), verifyOffline: verifyOffline(config) }; }; var _default = Keycloak; exports["default"] = _default;