1 | ;
|
2 |
|
3 | Object.defineProperty(exports, "__esModule", {
|
4 | value: true
|
5 | });
|
6 |
|
7 | var _logger = require('./logger');
|
8 |
|
9 | var _logger2 = _interopRequireDefault(_logger);
|
10 |
|
11 | var _util = require('util');
|
12 |
|
13 | var _keyid = require('./keyid');
|
14 |
|
15 | var _httpSignatureParser = require('./http-signature-parser');
|
16 |
|
17 | var _httpSignatureParser2 = _interopRequireDefault(_httpSignatureParser);
|
18 |
|
19 | var _authContext = require('@leisurelink/auth-context');
|
20 |
|
21 | var _request = require('./schemas/request');
|
22 |
|
23 | var _request2 = _interopRequireDefault(_request);
|
24 |
|
25 | var _bluebird = require('bluebird');
|
26 |
|
27 | var _bluebird2 = _interopRequireDefault(_bluebird);
|
28 |
|
29 | var _httpSignature = require('http-signature');
|
30 |
|
31 | var _httpSignature2 = _interopRequireDefault(_httpSignature);
|
32 |
|
33 | var _schemas = require('./schemas');
|
34 |
|
35 | var _trustedBaseSchema = require('./schemas/trusted-base-schema');
|
36 |
|
37 | function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
38 |
|
39 | const logger = (0, _logger2.default)('base');
|
40 |
|
41 | function transformForLog(value) {
|
42 | if (_authContext.AuthContext.isContext(value)) {
|
43 | return value.toAuditString();
|
44 | }
|
45 | if (typeof value === 'object' && value !== null) {
|
46 | return (0, _util.inspect)(value, false, 9);
|
47 | }
|
48 | return '' + value;
|
49 | }
|
50 |
|
51 | function checkVerified(auth) {
|
52 | if (!auth.verified) {
|
53 | logger.debug(`Invalid authority; unverifiable: ${ transformForLog(auth) }.`);
|
54 | return null;
|
55 | }
|
56 | return auth;
|
57 | }
|
58 |
|
59 | function checkVerifiedAndNotExpired(auth) {
|
60 | if (!checkVerified(auth)) {
|
61 | return null;
|
62 | }
|
63 | if (auth.isExpired) {
|
64 | logger.debug(`Invalid authority; expired: ${ transformForLog(auth) }.`);
|
65 | return null;
|
66 | }
|
67 | return auth;
|
68 | }
|
69 |
|
70 | const DEFA_LANG = 'en-US';
|
71 | let POSSIBLY_OVERRIDDEN_DEFA_LANG = process.env.AUTH_DEFAULT_LANG || DEFA_LANG;
|
72 |
|
73 | let $keyId = Symbol('keyId');
|
74 | let $lang = Symbol('lang');
|
75 | let $scope = Symbol('scope');
|
76 | let $resolveEndpointKey = Symbol('resolveEndpointKey');
|
77 | let $resolveEndpointClaims = Symbol('resolveEndpointClaims');
|
78 |
|
79 | class TrustedEndpoint {
|
80 | constructor(opts) {
|
81 | const options = (0, _schemas.validate)(opts, _trustedBaseSchema.optionsSchema);
|
82 |
|
83 | this[$lang] = options.lang || POSSIBLY_OVERRIDDEN_DEFA_LANG;
|
84 | this[$scope] = options.scope;
|
85 | this[$keyId] = options.keyId instanceof _keyid.KeyId ? options.keyId : _keyid.KeyId.parse(options.keyId);
|
86 | this[$resolveEndpointKey] = options.resolveEndpointKey;
|
87 | this[$resolveEndpointClaims] = options.resolveEndpointClaims;
|
88 |
|
89 | logger.debug(`new ${ this.constructor.name }({ keyId: '${ this.keyId }', scope = ${ this[$scope] }`);
|
90 | }
|
91 |
|
92 | get keyId() {
|
93 | return this[$keyId];
|
94 | }
|
95 |
|
96 | get lang() {
|
97 | return this[$lang];
|
98 | }
|
99 |
|
100 | verify(token) {
|
101 | let scope = this[$scope];
|
102 | return new _bluebird2.default((resolve, reject) => {
|
103 | scope.verify(token, (err, ctx) => {
|
104 | if (err) reject(err);else resolve(ctx);
|
105 | });
|
106 | });
|
107 | }
|
108 |
|
109 | resolveAndVerifyClaims(keyId) {
|
110 | keyId = keyId instanceof _keyid.KeyId ? keyId : _keyid.KeyId.parse(keyId);
|
111 | return this[$resolveEndpointClaims](this.lang, keyId.principalId).then(token => {
|
112 | logger.debug(`resolved endpoint claims: ${ token }`);
|
113 | return this.verify(token);
|
114 | }).then(checkVerifiedAndNotExpired);
|
115 | }
|
116 |
|
117 | resolveEndpointKey(keyId) {
|
118 | return this[$resolveEndpointKey](this.lang, keyId);
|
119 | }
|
120 |
|
121 | getLocalAuth() {
|
122 | return this.resolveAndVerifyClaims(this.keyId);
|
123 | }
|
124 |
|
125 | getRemoteAuth(request) {
|
126 | (0, _schemas.validate)(request, _request2.default);
|
127 | let self = this;
|
128 | let auth = {};
|
129 | let parser = new _httpSignatureParser2.default(_httpSignature2.default);
|
130 | return new _bluebird2.default((resolve, reject) => {
|
131 | let parsed = parser.parseRequest(request);
|
132 | return this.resolveEndpointKey(parsed.keyId).then(key => parser.verifySignature(parsed, key)).then(parsed => {
|
133 | if (parsed) {
|
134 | return this.resolveAndVerifyClaims(parsed.keyId);
|
135 | }
|
136 | return null;
|
137 | }).then(endpoint => {
|
138 | if (endpoint) {
|
139 | logger.debug(`endpoint authority: ${ endpoint }`);
|
140 | auth.remoteEndpoint = endpoint;
|
141 | }
|
142 | if (request.headers['x-authentic-user']) {
|
143 | logger.debug('X-Authentic-User header contains JWT; verifying user authority...');
|
144 | return self.verify(request.headers['x-authentic-user']).then(checkVerifiedAndNotExpired);
|
145 | }
|
146 | if (parsed.params.jwt) {
|
147 | logger.debug('Signature contains JWT; verifying user authority...');
|
148 | return self.verify(parsed.params.jwt).then(checkVerifiedAndNotExpired);
|
149 | }
|
150 | return null;
|
151 | }).then(user => {
|
152 | if (user) {
|
153 | logger.debug(`user authority: ${ user }`);
|
154 | auth.remoteUser = user;
|
155 | return resolve(auth);
|
156 | }
|
157 | if (request.headers['x-authentic-origin']) {
|
158 | logger.debug('X-Authentic-Origin header contains JWT; verifying originating endpoint authority...');
|
159 | return self.verify(request.headers['x-authentic-origin'])
|
160 | // be more lax with the expiration on originating endpoint, since it could take some time for messaging to flow through the system
|
161 | .then(checkVerified);
|
162 | }
|
163 | return null;
|
164 | }).then(originatingEndpoint => {
|
165 | if (originatingEndpoint) {
|
166 | logger.debug(`originatingEndpoint: ${ originatingEndpoint }`);
|
167 | auth.originatingEndpoint = originatingEndpoint;
|
168 | }
|
169 | return resolve(auth);
|
170 | }).catch(e => reject(e));
|
171 | });
|
172 | }
|
173 | }
|
174 |
|
175 | exports.default = TrustedEndpoint;
|
176 | module.exports = exports['default'];
|
177 | //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../lib/trusted-endpoint-base.js"],"names":[],"mappings":"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","file":"trusted-endpoint-base.js","sourcesContent":["import createLogger from './logger';\nimport { inspect } from 'util';\nimport { KeyId } from './keyid';\nimport SigParser from './http-signature-parser';\nimport { AuthContext } from '@leisurelink/auth-context';\nimport requestSchema from './schemas/request';\nimport Promise from 'bluebird';\nimport HttpSig from 'http-signature';\n\nimport { validate } from './schemas';\nimport { optionsSchema } from './schemas/trusted-base-schema';\n\nconst logger = createLogger('base');\n\nfunction transformForLog(value) {\n  if (AuthContext.isContext(value)) {\n    return value.toAuditString();\n  }\n  if (typeof(value) === 'object' && value !== null) {\n    return inspect(value, false, 9);\n  }\n  return '' + value;\n}\n\nfunction checkVerified(auth) {\n  if (!auth.verified) {\n    logger.debug(`Invalid authority; unverifiable: ${transformForLog(auth)}.`);\n    return null;\n  }\n  return auth;\n}\n\nfunction checkVerifiedAndNotExpired(auth) {\n  if (!checkVerified(auth)){\n    return null;\n  }\n  if (auth.isExpired) {\n    logger.debug(`Invalid authority; expired: ${transformForLog(auth)}.`);\n    return null;\n  }\n  return auth;\n}\n\nconst DEFA_LANG = 'en-US';\nlet POSSIBLY_OVERRIDDEN_DEFA_LANG = process.env.AUTH_DEFAULT_LANG || DEFA_LANG;\n\nlet $keyId = Symbol('keyId');\nlet $lang = Symbol('lang');\nlet $scope = Symbol('scope');\nlet $resolveEndpointKey = Symbol('resolveEndpointKey');\nlet $resolveEndpointClaims = Symbol('resolveEndpointClaims');\n\nclass TrustedEndpoint {\n  constructor(opts) {\n    const options = validate(opts, optionsSchema);\n\n    this[$lang] = options.lang || POSSIBLY_OVERRIDDEN_DEFA_LANG;\n    this[$scope] = options.scope;\n    this[$keyId] = (options.keyId instanceof KeyId) ? options.keyId : KeyId.parse(options.keyId);\n    this[$resolveEndpointKey] = options.resolveEndpointKey;\n    this[$resolveEndpointClaims] = options.resolveEndpointClaims;\n\n    logger.debug(`new ${this.constructor.name}({ keyId: '${this.keyId}', scope = ${this[$scope]}`);\n  }\n\n  get keyId() {\n    return this[$keyId];\n  }\n\n  get lang() {\n    return this[$lang];\n  }\n\n  verify(token) {\n    let scope = this[$scope];\n    return new Promise(\n      (resolve, reject) => {\n        scope.verify(token, (err, ctx) => {\n          if (err) reject(err);\n          else resolve(ctx);\n        });\n      });\n  }\n\n  resolveAndVerifyClaims(keyId) {\n    keyId = (keyId instanceof KeyId) ? keyId : KeyId.parse(keyId);\n    return this[$resolveEndpointClaims](this.lang, keyId.principalId)\n      .then(token => {\n        logger.debug(`resolved endpoint claims: ${token}`);\n        return this.verify(token);\n      })\n      .then(checkVerifiedAndNotExpired);\n  }\n\n  resolveEndpointKey(keyId) {\n    return this[$resolveEndpointKey](this.lang, keyId);\n  }\n\n  getLocalAuth() {\n    return this.resolveAndVerifyClaims(this.keyId);\n  }\n\n  getRemoteAuth(request) {\n    validate(request, requestSchema);\n    let self = this;\n    let auth = {};\n    let parser = new SigParser(HttpSig);\n    return new Promise((resolve, reject) =>{\n      let parsed = parser.parseRequest(request);\n      return this.resolveEndpointKey(parsed.keyId)\n        .then(key => parser.verifySignature(parsed, key))\n        .then(parsed => {\n          if (parsed) {\n            return this.resolveAndVerifyClaims(parsed.keyId);\n          }\n          return null;\n        })\n        .then(endpoint => {\n          if (endpoint) {\n            logger.debug(`endpoint authority: ${endpoint}`);\n            auth.remoteEndpoint = endpoint;\n          }\n          if (request.headers['x-authentic-user']) {\n            logger.debug('X-Authentic-User header contains JWT; verifying user authority...');\n            return self.verify(request.headers['x-authentic-user'])\n              .then(checkVerifiedAndNotExpired);\n          }\n          if (parsed.params.jwt) {\n            logger.debug('Signature contains JWT; verifying user authority...');\n            return self.verify(parsed.params.jwt)\n              .then(checkVerifiedAndNotExpired);\n          }\n          return null;\n        }).then(user => {\n          if (user) {\n            logger.debug(`user authority: ${user}`);\n            auth.remoteUser = user;\n            return resolve(auth);\n          }\n          if (request.headers['x-authentic-origin']) {\n            logger.debug('X-Authentic-Origin header contains JWT; verifying originating endpoint authority...');\n            return self.verify(request.headers['x-authentic-origin'])\n              // be more lax with the expiration on originating endpoint, since it could take some time for messaging to flow through the system\n              .then(checkVerified);\n          }\n          return null;\n        }).then(originatingEndpoint => {\n          if (originatingEndpoint) {\n            logger.debug(`originatingEndpoint: ${originatingEndpoint}`);\n            auth.originatingEndpoint = originatingEndpoint;\n          }\n          return resolve(auth);\n        })\n        .catch(e => reject(e));\n    });\n\n  }\n}\n\nexport default TrustedEndpoint;\n"]} |
\ | No newline at end of file |