UNPKG

17.9 kBJavaScriptView Raw
1'use strict';
2
3Object.defineProperty(exports, "__esModule", {
4 value: true
5});
6
7var _logger = require('./logger');
8
9var _logger2 = _interopRequireDefault(_logger);
10
11var _util = require('util');
12
13var _keyid = require('./keyid');
14
15var _httpSignatureParser = require('./http-signature-parser');
16
17var _httpSignatureParser2 = _interopRequireDefault(_httpSignatureParser);
18
19var _authContext = require('@leisurelink/auth-context');
20
21var _request = require('./schemas/request');
22
23var _request2 = _interopRequireDefault(_request);
24
25var _bluebird = require('bluebird');
26
27var _bluebird2 = _interopRequireDefault(_bluebird);
28
29var _httpSignature = require('http-signature');
30
31var _httpSignature2 = _interopRequireDefault(_httpSignature);
32
33var _schemas = require('./schemas');
34
35var _trustedBaseSchema = require('./schemas/trusted-base-schema');
36
37function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
38
39const logger = (0, _logger2.default)('base');
40
41function 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
51function checkVerified(auth) {
52 if (!auth.verified) {
53 logger.debug(`Invalid authority; unverifiable: ${ transformForLog(auth) }.`);
54 return null;
55 }
56 return auth;
57}
58
59function 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
70const DEFA_LANG = 'en-US';
71let POSSIBLY_OVERRIDDEN_DEFA_LANG = process.env.AUTH_DEFAULT_LANG || DEFA_LANG;
72
73let $keyId = Symbol('keyId');
74let $lang = Symbol('lang');
75let $scope = Symbol('scope');
76let $resolveEndpointKey = Symbol('resolveEndpointKey');
77let $resolveEndpointClaims = Symbol('resolveEndpointClaims');
78
79class 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
175exports.default = TrustedEndpoint;
176module.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