@lifeready/core 1.0.2 → 1.0.4
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +62 -62
- package/bundles/lifeready-core.umd.js +15445 -15445
- package/bundles/lifeready-core.umd.js.map +1 -1
- package/bundles/lifeready-core.umd.min.js.map +1 -1
- package/esm2015/lib/_common/ast.js +40 -40
- package/esm2015/lib/_common/deferred-promise.js +24 -24
- package/esm2015/lib/_common/exceptions.js +157 -157
- package/esm2015/lib/_common/queries.gql.js +190 -190
- package/esm2015/lib/_common/run-outside-angular.js +79 -79
- package/esm2015/lib/_common/types.js +1 -1
- package/esm2015/lib/_common/utils.js +44 -44
- package/esm2015/lib/api/contact-card.gql.js +79 -79
- package/esm2015/lib/api/contact-card.service.js +154 -154
- package/esm2015/lib/api/contact-card2.gql.js +60 -60
- package/esm2015/lib/api/contact-card2.service.js +103 -103
- package/esm2015/lib/api/file.service.js +74 -74
- package/esm2015/lib/api/item2.gql.js +110 -110
- package/esm2015/lib/api/item2.service.js +311 -311
- package/esm2015/lib/api/key-exchange.gql.js +188 -188
- package/esm2015/lib/api/key-exchange.service.js +442 -442
- package/esm2015/lib/api/key-exchange.types.js +18 -18
- package/esm2015/lib/api/key-exchange2.gql.js +171 -171
- package/esm2015/lib/api/key-exchange2.service.js +479 -479
- package/esm2015/lib/api/lock.gql.js +40 -40
- package/esm2015/lib/api/lock.service.js +64 -64
- package/esm2015/lib/api/lr-apollo.service.js +46 -46
- package/esm2015/lib/api/lr-graphql/index.js +6 -6
- package/esm2015/lib/api/lr-graphql/lr-graphql.service.js +155 -155
- package/esm2015/lib/api/lr-graphql/lr-merged-mutation.js +213 -213
- package/esm2015/lib/api/lr-graphql/lr-mutation-base.js +51 -51
- package/esm2015/lib/api/lr-graphql/lr-mutation.js +48 -48
- package/esm2015/lib/api/lr-graphql/lr.service.js +18 -18
- package/esm2015/lib/api/message.service.js +138 -138
- package/esm2015/lib/api/persist.service.js +181 -181
- package/esm2015/lib/api/query-processor/common-processors.service.js +93 -93
- package/esm2015/lib/api/query-processor/index.js +3 -3
- package/esm2015/lib/api/query-processor/query-processor.service.js +192 -192
- package/esm2015/lib/api/query-processor/tp-password-reset-processor.service.js +109 -109
- package/esm2015/lib/api/shared-contact-card.service.js +119 -119
- package/esm2015/lib/api/shared-contact-card2.gql.js +41 -41
- package/esm2015/lib/api/shared-contact-card2.service.js +117 -117
- package/esm2015/lib/api/time.service.js +146 -146
- package/esm2015/lib/api/types/graphql.types.js +7 -7
- package/esm2015/lib/api/types/index.js +3 -3
- package/esm2015/lib/api/types/lr-graphql.types.js +71 -71
- package/esm2015/lib/auth/auth.config.js +57 -57
- package/esm2015/lib/auth/auth.gql.js +48 -48
- package/esm2015/lib/auth/auth.types.js +27 -27
- package/esm2015/lib/auth/idle.service.js +168 -168
- package/esm2015/lib/auth/idle.types.js +7 -7
- package/esm2015/lib/auth/lbop.service.js +355 -355
- package/esm2015/lib/auth/life-ready-auth.service.js +500 -500
- package/esm2015/lib/auth/password.service.js +320 -320
- package/esm2015/lib/auth/register.service.js +172 -172
- package/esm2015/lib/auth/two-factor.service.js +74 -74
- package/esm2015/lib/category/category-meta.service.js +99 -99
- package/esm2015/lib/category/category.gql.js +406 -406
- package/esm2015/lib/category/category.service.js +390 -390
- package/esm2015/lib/category/category.types.js +29 -29
- package/esm2015/lib/cryptography/cryptography.types.js +11 -11
- package/esm2015/lib/cryptography/encryption.service.js +189 -189
- package/esm2015/lib/cryptography/key-factory.service.js +237 -237
- package/esm2015/lib/cryptography/key-graph.service.js +280 -280
- package/esm2015/lib/cryptography/key-meta.service.js +200 -200
- package/esm2015/lib/cryptography/key.service.js +124 -124
- package/esm2015/lib/cryptography/slip39.service.js +169 -169
- package/esm2015/lib/cryptography/web-crypto.service.js +29 -29
- package/esm2015/lib/life-ready.config.js +84 -84
- package/esm2015/lib/life-ready.module.js +74 -74
- package/esm2015/lib/plan/plan.gql.js +123 -123
- package/esm2015/lib/plan/plan.service.js +149 -149
- package/esm2015/lib/plan/plan.types.js +11 -11
- package/esm2015/lib/record/record-attachment.service.js +101 -101
- package/esm2015/lib/record/record.gql.js +179 -179
- package/esm2015/lib/record/record.service.js +206 -206
- package/esm2015/lib/record/record.types.js +15 -15
- package/esm2015/lib/record-type/record-type.service.js +75 -75
- package/esm2015/lib/record-type/record-type.types.js +28 -28
- package/esm2015/lib/scenario/approvals/scenario-approval.gql.js +105 -105
- package/esm2015/lib/scenario/approvals/scenario-approval.types.js +1 -1
- package/esm2015/lib/scenario/approvals/scenario-approver.service.js +300 -300
- package/esm2015/lib/scenario/claimants/scenario-claimant.gql.js +52 -52
- package/esm2015/lib/scenario/claimants/scenario-claimant.service.js +97 -97
- package/esm2015/lib/scenario/claimants/scenario-claimant.types.js +1 -1
- package/esm2015/lib/scenario/receivers/scenario-receiver.gql.js +150 -150
- package/esm2015/lib/scenario/receivers/scenario-receiver.service.js +229 -229
- package/esm2015/lib/scenario/receivers/scenario-receiver.types.js +1 -1
- package/esm2015/lib/scenario/scenario-setup.service.js +269 -269
- package/esm2015/lib/scenario/scenario.gql.js +368 -368
- package/esm2015/lib/scenario/scenario.service.js +611 -611
- package/esm2015/lib/scenario/scenario.types.js +64 -64
- package/esm2015/lib/search/search.gql.js +62 -62
- package/esm2015/lib/search/search.service.js +156 -156
- package/esm2015/lib/search/search.types.js +6 -6
- package/esm2015/lib/trusted-parties/tp-password-reset-request.service.js +112 -112
- package/esm2015/lib/trusted-parties/tp-password-reset-user.service.js +129 -129
- package/esm2015/lib/trusted-parties/tp-password-reset.constants.js +4 -4
- package/esm2015/lib/trusted-parties/tp-password-reset.gql.js +232 -232
- package/esm2015/lib/trusted-parties/tp-password-reset.service.js +299 -299
- package/esm2015/lib/trusted-parties/trusted-party.gql.js +148 -148
- package/esm2015/lib/trusted-parties/trusted-party.service.js +326 -326
- package/esm2015/lib/trusted-parties/trusted-party.types.js +41 -41
- package/esm2015/lib/trusted-parties/trusted-party2.gql.js +87 -87
- package/esm2015/lib/trusted-parties/trusted-party2.service.js +215 -215
- package/esm2015/lib/users/profile-details.service.js +214 -214
- package/esm2015/lib/users/profile.gql.js +97 -97
- package/esm2015/lib/users/profile.service.js +169 -169
- package/esm2015/lib/users/profile.types.js +34 -34
- package/esm2015/lib/users/user.gql.js +60 -60
- package/esm2015/lib/users/user.service.js +79 -79
- package/esm2015/lib/users/user.types.js +5 -5
- package/esm2015/lifeready-core.js +10 -10
- package/esm2015/public-api.js +81 -81
- package/fesm2015/lifeready-core.js +13088 -13088
- package/fesm2015/lifeready-core.js.map +1 -1
- package/lib/_common/ast.d.ts +11 -11
- package/lib/_common/deferred-promise.d.ts +12 -12
- package/lib/_common/exceptions.d.ts +109 -109
- package/lib/_common/queries.gql.d.ts +10 -10
- package/lib/_common/run-outside-angular.d.ts +14 -14
- package/lib/_common/types.d.ts +10 -10
- package/lib/_common/utils.d.ts +3 -3
- package/lib/api/contact-card.gql.d.ts +7 -7
- package/lib/api/contact-card.service.d.ts +52 -52
- package/lib/api/contact-card2.gql.d.ts +34 -34
- package/lib/api/contact-card2.service.d.ts +49 -49
- package/lib/api/file.service.d.ts +18 -18
- package/lib/api/item2.gql.d.ts +96 -96
- package/lib/api/item2.service.d.ts +177 -177
- package/lib/api/key-exchange.gql.d.ts +9 -9
- package/lib/api/key-exchange.service.d.ts +39 -39
- package/lib/api/key-exchange.types.d.ts +196 -196
- package/lib/api/key-exchange2.gql.d.ts +125 -125
- package/lib/api/key-exchange2.service.d.ts +187 -187
- package/lib/api/lock.gql.d.ts +27 -27
- package/lib/api/lock.service.d.ts +25 -25
- package/lib/api/lr-apollo.service.d.ts +15 -15
- package/lib/api/lr-graphql/index.d.ts +5 -5
- package/lib/api/lr-graphql/lr-graphql.service.d.ts +60 -60
- package/lib/api/lr-graphql/lr-merged-mutation.d.ts +27 -27
- package/lib/api/lr-graphql/lr-mutation-base.d.ts +28 -28
- package/lib/api/lr-graphql/lr-mutation.d.ts +8 -8
- package/lib/api/lr-graphql/lr.service.d.ts +9 -9
- package/lib/api/message.service.d.ts +58 -58
- package/lib/api/persist.service.d.ts +31 -31
- package/lib/api/query-processor/common-processors.service.d.ts +36 -36
- package/lib/api/query-processor/index.d.ts +2 -2
- package/lib/api/query-processor/query-processor.service.d.ts +18 -18
- package/lib/api/query-processor/tp-password-reset-processor.service.d.ts +15 -15
- package/lib/api/shared-contact-card.service.d.ts +33 -33
- package/lib/api/shared-contact-card2.gql.d.ts +36 -36
- package/lib/api/shared-contact-card2.service.d.ts +45 -45
- package/lib/api/time.service.d.ts +16 -16
- package/lib/api/types/graphql.types.d.ts +29 -29
- package/lib/api/types/index.d.ts +2 -2
- package/lib/api/types/lr-graphql.types.d.ts +385 -385
- package/lib/auth/auth.config.d.ts +5 -5
- package/lib/auth/auth.gql.d.ts +15 -15
- package/lib/auth/auth.types.d.ts +66 -66
- package/lib/auth/idle.service.d.ts +40 -40
- package/lib/auth/idle.types.d.ts +10 -10
- package/lib/auth/lbop.service.d.ts +91 -91
- package/lib/auth/life-ready-auth.service.d.ts +59 -59
- package/lib/auth/password.service.d.ts +78 -78
- package/lib/auth/register.service.d.ts +25 -25
- package/lib/auth/two-factor.service.d.ts +15 -15
- package/lib/category/category-meta.service.d.ts +23 -23
- package/lib/category/category.gql.d.ts +45 -45
- package/lib/category/category.service.d.ts +67 -67
- package/lib/category/category.types.d.ts +79 -79
- package/lib/cryptography/cryptography.types.d.ts +83 -83
- package/lib/cryptography/encryption.service.d.ts +41 -41
- package/lib/cryptography/key-factory.service.d.ts +38 -38
- package/lib/cryptography/key-graph.service.d.ts +33 -33
- package/lib/cryptography/key-meta.service.d.ts +44 -44
- package/lib/cryptography/key.service.d.ts +36 -36
- package/lib/cryptography/slip39.service.d.ts +43 -43
- package/lib/cryptography/web-crypto.service.d.ts +5 -5
- package/lib/life-ready.config.d.ts +14 -14
- package/lib/life-ready.module.d.ts +5 -5
- package/lib/plan/plan.gql.d.ts +11 -11
- package/lib/plan/plan.service.d.ts +33 -33
- package/lib/plan/plan.types.d.ts +31 -31
- package/lib/record/record-attachment.service.d.ts +16 -16
- package/lib/record/record.gql.d.ts +14 -14
- package/lib/record/record.service.d.ts +25 -25
- package/lib/record/record.types.d.ts +57 -57
- package/lib/record-type/record-type.service.d.ts +11 -11
- package/lib/record-type/record-type.types.d.ts +50 -50
- package/lib/scenario/approvals/scenario-approval.gql.d.ts +7 -7
- package/lib/scenario/approvals/scenario-approval.types.d.ts +63 -63
- package/lib/scenario/approvals/scenario-approver.service.d.ts +32 -32
- package/lib/scenario/claimants/scenario-claimant.gql.d.ts +5 -5
- package/lib/scenario/claimants/scenario-claimant.service.d.ts +17 -17
- package/lib/scenario/claimants/scenario-claimant.types.d.ts +18 -18
- package/lib/scenario/receivers/scenario-receiver.gql.d.ts +8 -8
- package/lib/scenario/receivers/scenario-receiver.service.d.ts +30 -30
- package/lib/scenario/receivers/scenario-receiver.types.d.ts +54 -54
- package/lib/scenario/scenario-setup.service.d.ts +22 -22
- package/lib/scenario/scenario.gql.d.ts +34 -34
- package/lib/scenario/scenario.service.d.ts +58 -58
- package/lib/scenario/scenario.types.d.ts +217 -217
- package/lib/search/search.gql.d.ts +1 -1
- package/lib/search/search.service.d.ts +25 -25
- package/lib/search/search.types.d.ts +20 -20
- package/lib/trusted-parties/tp-password-reset-request.service.d.ts +20 -20
- package/lib/trusted-parties/tp-password-reset-user.service.d.ts +35 -35
- package/lib/trusted-parties/tp-password-reset.constants.d.ts +3 -3
- package/lib/trusted-parties/tp-password-reset.gql.d.ts +218 -218
- package/lib/trusted-parties/tp-password-reset.service.d.ts +130 -130
- package/lib/trusted-parties/trusted-party.gql.d.ts +9 -9
- package/lib/trusted-parties/trusted-party.service.d.ts +44 -44
- package/lib/trusted-parties/trusted-party.types.d.ts +102 -102
- package/lib/trusted-parties/trusted-party2.gql.d.ts +79 -79
- package/lib/trusted-parties/trusted-party2.service.d.ts +114 -114
- package/lib/users/profile-details.service.d.ts +21 -21
- package/lib/users/profile.gql.d.ts +11 -11
- package/lib/users/profile.service.d.ts +35 -35
- package/lib/users/profile.types.d.ts +96 -96
- package/lib/users/user.gql.d.ts +9 -9
- package/lib/users/user.service.d.ts +12 -12
- package/lib/users/user.types.d.ts +23 -23
- package/lifeready-core.d.ts +9 -9
- package/package.json +1 -1
- package/public-api.d.ts +77 -77
|
@@ -1,237 +1,237 @@
|
|
|
1
|
-
import { __awaiter } from "tslib";
|
|
2
|
-
import { Injectable } from '@angular/core';
|
|
3
|
-
import { JWK } from 'node-jose';
|
|
4
|
-
import { WebCryptoService } from './web-crypto.service';
|
|
5
|
-
import { LrBadArgumentException, LrSuspiciousException, } from '../_common/exceptions';
|
|
6
|
-
import * as i0 from "@angular/core";
|
|
7
|
-
import * as i1 from "./web-crypto.service";
|
|
8
|
-
export function sha256(message) {
|
|
9
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
10
|
-
// encode as UTF-8
|
|
11
|
-
const msgBuffer = new TextEncoder().encode(message);
|
|
12
|
-
// hash the message
|
|
13
|
-
const hashBuffer = yield crypto.subtle.digest('SHA-256', msgBuffer);
|
|
14
|
-
// convert ArrayBuffer to Array
|
|
15
|
-
const hashArray = Array.from(new Uint8Array(hashBuffer));
|
|
16
|
-
// convert bytes to hex string
|
|
17
|
-
const hashHex = hashArray
|
|
18
|
-
.map((b) => ('00' + b.toString(16)).slice(-2))
|
|
19
|
-
.join('');
|
|
20
|
-
return hashHex;
|
|
21
|
-
});
|
|
22
|
-
}
|
|
23
|
-
export class KeyFactoryService {
|
|
24
|
-
constructor(webCryptoService) {
|
|
25
|
-
this.webCryptoService = webCryptoService;
|
|
26
|
-
// Global keys store. Otherwise, each call to asKey creates a new keyStore.
|
|
27
|
-
// <AZ> Did not seem to improve speed.
|
|
28
|
-
// public static keyStore = JWK.createKeyStore();
|
|
29
|
-
// AZ: This can't be change easily. It's basically a PassK or PassIdp rotation.
|
|
30
|
-
// todo: we should eventually increase this periodically to match with Moore's law.
|
|
31
|
-
// The iterations for each key are kept by the server as well but we assume the value
|
|
32
|
-
// from the server is not trustworthy, so need to have minimum thresholds here.
|
|
33
|
-
// If creating new keys, these minimum are used.
|
|
34
|
-
this.MIN_PASS_IDP_PBKDF_ITER = 100000;
|
|
35
|
-
this.MIN_PASS_KEY_PBKDF_ITER = 100000;
|
|
36
|
-
this.MIN_LBOP_KEY_PBKDF_ITER = 100000;
|
|
37
|
-
// These are used as the default values. They must be larger than the minimum values.
|
|
38
|
-
this.DEFAULT_PASS_IDP_PBKDF_ITER = this.MIN_PASS_IDP_PBKDF_ITER;
|
|
39
|
-
this.DEFAULT_PASS_KEY_PBKDF_ITER = this.MIN_PASS_KEY_PBKDF_ITER;
|
|
40
|
-
this.DEFAULT_LBOP_KEY_PBKDF_ITER = this.MIN_LBOP_KEY_PBKDF_ITER;
|
|
41
|
-
this.crypto = this.webCryptoService.crypto;
|
|
42
|
-
}
|
|
43
|
-
static asKey(key, form, extras) {
|
|
44
|
-
// <AZ> Using a single global key store did not seem to improve speed.
|
|
45
|
-
// return KeyFactoryService.keyStore.add(key, form, extras);
|
|
46
|
-
return JWK.asKey(key, form, extras);
|
|
47
|
-
}
|
|
48
|
-
randomString(digits) {
|
|
49
|
-
if (digits <= 0) {
|
|
50
|
-
throw new LrBadArgumentException('digits <= 0');
|
|
51
|
-
}
|
|
52
|
-
const validChars = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789';
|
|
53
|
-
let array = new Uint32Array(digits);
|
|
54
|
-
this.crypto.getRandomValues(array);
|
|
55
|
-
array = array.map((x) => validChars.charCodeAt(x % validChars.length));
|
|
56
|
-
return String.fromCharCode.apply(null, array);
|
|
57
|
-
}
|
|
58
|
-
randomDigitsNoZeros(digits) {
|
|
59
|
-
return this.randomChoices([1, 2, 3, 4, 5, 6, 7, 8, 9], digits).join('');
|
|
60
|
-
}
|
|
61
|
-
randomChoices(array, chooseN) {
|
|
62
|
-
if (array.length <= 1) {
|
|
63
|
-
throw new LrBadArgumentException('array.length <= 0');
|
|
64
|
-
}
|
|
65
|
-
if (chooseN <= 0) {
|
|
66
|
-
throw new LrBadArgumentException('chooseN <= 0');
|
|
67
|
-
}
|
|
68
|
-
const values = new Uint32Array(chooseN);
|
|
69
|
-
this.crypto.getRandomValues(values);
|
|
70
|
-
const ret = [];
|
|
71
|
-
values.forEach((v) => ret.push(array[v % array.length]));
|
|
72
|
-
return ret;
|
|
73
|
-
}
|
|
74
|
-
createSalt() {
|
|
75
|
-
return this.randomString(16);
|
|
76
|
-
}
|
|
77
|
-
createKey() {
|
|
78
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
79
|
-
const key = yield this.crypto.subtle.generateKey({
|
|
80
|
-
name: 'AES-GCM',
|
|
81
|
-
length: 256,
|
|
82
|
-
}, true, // whether the key is extractable (i.e. can be used in exportKey)
|
|
83
|
-
['encrypt', 'decrypt'] // must be ["encrypt", "decrypt"] or ["wrapKey", "unwrapKey"]
|
|
84
|
-
);
|
|
85
|
-
const jwk = yield this.crypto.subtle.exportKey('jwk', key);
|
|
86
|
-
// Removing the fields not needed by node-jose
|
|
87
|
-
delete jwk.ext;
|
|
88
|
-
delete jwk.key_ops;
|
|
89
|
-
return KeyFactoryService.asKey(jwk);
|
|
90
|
-
});
|
|
91
|
-
}
|
|
92
|
-
createSignKey() {
|
|
93
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
94
|
-
const key = yield this.crypto.subtle.generateKey({
|
|
95
|
-
name: 'HMAC',
|
|
96
|
-
hash: { name: 'SHA-512' },
|
|
97
|
-
}, true, ['sign', 'verify']);
|
|
98
|
-
const jwk = yield this.crypto.subtle.exportKey('jwk', key);
|
|
99
|
-
// Removing the fields not needed by node-jose
|
|
100
|
-
delete jwk.key_ops;
|
|
101
|
-
delete jwk.ext;
|
|
102
|
-
return KeyFactoryService.asKey(jwk);
|
|
103
|
-
});
|
|
104
|
-
}
|
|
105
|
-
createPkcKey() {
|
|
106
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
107
|
-
// node-jose is not using Forge properly. It should be calling the async version of
|
|
108
|
-
// pki.rsa.generateKeyPair() with a callback. Instead it calls the sync version. Webcrypto
|
|
109
|
-
// does not support sync version, so it uses the javascript implementation, which is way too slow.
|
|
110
|
-
// So we generate using webcrypto and import the key.
|
|
111
|
-
// Unfortunately Elliptical Curve is not supported by Webcrypto. So we have to settle for RSA.
|
|
112
|
-
const key = yield this.crypto.subtle.generateKey({
|
|
113
|
-
name: 'RSA-OAEP',
|
|
114
|
-
modulusLength: 2048,
|
|
115
|
-
// As per suggestion: https://developer.mozilla.org/en-US/docs/Web/API/RsaHashedKeyGenParams
|
|
116
|
-
publicExponent: new Uint8Array([0x01, 0x00, 0x01]),
|
|
117
|
-
hash: { name: 'SHA-256' },
|
|
118
|
-
}, true, // whether the key is extractable (i.e. can be used in exportKey)
|
|
119
|
-
['encrypt', 'decrypt'] // must be ["encrypt", "decrypt"] or ["wrapKey", "unwrapKey"]
|
|
120
|
-
);
|
|
121
|
-
const jwk = yield this.crypto.subtle.exportKey('jwk', key.privateKey);
|
|
122
|
-
// Removing the fields not needed by node-jose
|
|
123
|
-
delete jwk.key_ops;
|
|
124
|
-
delete jwk.ext;
|
|
125
|
-
return KeyFactoryService.asKey(jwk);
|
|
126
|
-
});
|
|
127
|
-
}
|
|
128
|
-
createPkcSignKey() {
|
|
129
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
130
|
-
const key = yield this.crypto.subtle.generateKey({
|
|
131
|
-
name: 'RSASSA-PKCS1-v1_5',
|
|
132
|
-
modulusLength: 2048,
|
|
133
|
-
// As per suggestion: https://developer.mozilla.org/en-US/docs/Web/API/RsaHashedKeyGenParams
|
|
134
|
-
publicExponent: new Uint8Array([0x01, 0x00, 0x01]),
|
|
135
|
-
hash: { name: 'SHA-256' },
|
|
136
|
-
}, true, // whether the key is extractable (i.e. can be used in exportKey)
|
|
137
|
-
['sign', 'verify'] // can be any combination of "sign" and "verify"
|
|
138
|
-
);
|
|
139
|
-
const jwk = yield this.crypto.subtle.exportKey('jwk', key.privateKey);
|
|
140
|
-
// Removing the fields not needed by node-jose
|
|
141
|
-
delete jwk.key_ops;
|
|
142
|
-
delete jwk.ext;
|
|
143
|
-
return KeyFactoryService.asKey(jwk);
|
|
144
|
-
});
|
|
145
|
-
}
|
|
146
|
-
deriveKey({ password, salt, iterations, kid, }) {
|
|
147
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
148
|
-
const enc = new TextEncoder();
|
|
149
|
-
const rawKey = yield this.crypto.subtle.importKey('raw', enc.encode(password), 'PBKDF2', false, ['deriveBits', 'deriveKey']);
|
|
150
|
-
const passKey = yield crypto.subtle.deriveKey({
|
|
151
|
-
name: 'PBKDF2',
|
|
152
|
-
salt: new TextEncoder().encode(salt),
|
|
153
|
-
iterations,
|
|
154
|
-
hash: 'SHA-256',
|
|
155
|
-
}, rawKey, { name: 'AES-GCM', length: 256 }, true, ['encrypt', 'decrypt']);
|
|
156
|
-
const passKeyJson = yield crypto.subtle.exportKey('jwk', passKey);
|
|
157
|
-
if (kid) {
|
|
158
|
-
passKeyJson.kid = kid;
|
|
159
|
-
}
|
|
160
|
-
const jwk = yield KeyFactoryService.asKey(passKeyJson);
|
|
161
|
-
return { jwk };
|
|
162
|
-
});
|
|
163
|
-
}
|
|
164
|
-
derivePassIdp(params) {
|
|
165
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
166
|
-
if (params.iterations < this.MIN_PASS_IDP_PBKDF_ITER) {
|
|
167
|
-
throw new LrSuspiciousException(`The number of PassIdp key derivation iterations sent from the server (${params.iterations}) is lower than the minimum (${this.MIN_PASS_IDP_PBKDF_ITER})`);
|
|
168
|
-
}
|
|
169
|
-
return this.deriveKey(params);
|
|
170
|
-
});
|
|
171
|
-
}
|
|
172
|
-
derivePassKey(params) {
|
|
173
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
174
|
-
if (params.iterations < this.MIN_PASS_KEY_PBKDF_ITER) {
|
|
175
|
-
throw new LrSuspiciousException(`The number of PassKey key derivation iterations sent from the server(${params.iterations}) is lower than the minimum(${this.MIN_PASS_KEY_PBKDF_ITER})`);
|
|
176
|
-
}
|
|
177
|
-
return this.deriveKey(params);
|
|
178
|
-
});
|
|
179
|
-
}
|
|
180
|
-
deriveLbopKey(params) {
|
|
181
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
182
|
-
if (params.iterations < this.MIN_LBOP_KEY_PBKDF_ITER) {
|
|
183
|
-
throw new LrSuspiciousException(`The number of LbopKey key derivation iterations sent from the server(${params.iterations}) is lower than the minimum(${this.MIN_LBOP_KEY_PBKDF_ITER})`);
|
|
184
|
-
}
|
|
185
|
-
return this.deriveKey(params);
|
|
186
|
-
});
|
|
187
|
-
}
|
|
188
|
-
createKid() {
|
|
189
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
190
|
-
// todo: AZ: node-jose source uses node's default UUID() function for kid, so just change to use that.
|
|
191
|
-
// for now, we are just creating a new key to use it's kid.
|
|
192
|
-
// The kid is a part of the JWK system. LR backend maintains the key hierarchy separately with it's own
|
|
193
|
-
// key id. But we just use it here as a double check.
|
|
194
|
-
return (yield this.createKey()).kid;
|
|
195
|
-
});
|
|
196
|
-
}
|
|
197
|
-
createPassIdpParams() {
|
|
198
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
199
|
-
return {
|
|
200
|
-
salt: this.createSalt(),
|
|
201
|
-
iterations: this.DEFAULT_PASS_IDP_PBKDF_ITER,
|
|
202
|
-
};
|
|
203
|
-
});
|
|
204
|
-
}
|
|
205
|
-
createPassKeyParams() {
|
|
206
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
207
|
-
return {
|
|
208
|
-
salt: this.createSalt(),
|
|
209
|
-
kid: yield this.createKid(),
|
|
210
|
-
iterations: this.DEFAULT_PASS_KEY_PBKDF_ITER,
|
|
211
|
-
};
|
|
212
|
-
});
|
|
213
|
-
}
|
|
214
|
-
createLbopKeyParams() {
|
|
215
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
216
|
-
return {
|
|
217
|
-
salt: this.createSalt(),
|
|
218
|
-
// todo: AZ: node-jose source uses node's default UUID() function for kid, so just change to use that.
|
|
219
|
-
// for now, we are just creating a new key to use it's kid.
|
|
220
|
-
// The kid is a part of the JWK system. LR backend maintains the key hierarchy separately with it's own
|
|
221
|
-
// key id. But we just use it here as a double check.
|
|
222
|
-
kid: yield this.createKid(),
|
|
223
|
-
iterations: this.DEFAULT_PASS_KEY_PBKDF_ITER,
|
|
224
|
-
};
|
|
225
|
-
});
|
|
226
|
-
}
|
|
227
|
-
}
|
|
228
|
-
KeyFactoryService.ɵprov = i0.ɵɵdefineInjectable({ factory: function KeyFactoryService_Factory() { return new KeyFactoryService(i0.ɵɵinject(i1.WebCryptoService)); }, token: KeyFactoryService, providedIn: "root" });
|
|
229
|
-
KeyFactoryService.decorators = [
|
|
230
|
-
{ type: Injectable, args: [{
|
|
231
|
-
providedIn: 'root',
|
|
232
|
-
},] }
|
|
233
|
-
];
|
|
234
|
-
KeyFactoryService.ctorParameters = () => [
|
|
235
|
-
{ type: WebCryptoService }
|
|
236
|
-
];
|
|
237
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"key-factory.service.js","sourceRoot":"/opt/atlassian/pipelines/agent/build/projects/core/src/","sources":["lib/cryptography/key-factory.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { JWK } from 'node-jose';\nimport {\n  LbopKeyParams,\n  PassIdpParams,\n  PassKeyParams,\n  DeriveKeyResult,\n  DerivePassIdpParams,\n  DerivePassKeyParams,\n  DeriveLbopKeyParams,\n} from './cryptography.types';\nimport { WebCryptoService } from './web-crypto.service';\nimport {\n  LrBadArgumentException,\n  LrSuspiciousException,\n} from '../_common/exceptions';\n\nexport async function sha256(message) {\n  // encode as UTF-8\n  const msgBuffer = new TextEncoder().encode(message);\n\n  // hash the message\n  const hashBuffer = await crypto.subtle.digest('SHA-256', msgBuffer);\n\n  // convert ArrayBuffer to Array\n  const hashArray = Array.from(new Uint8Array(hashBuffer));\n\n  // convert bytes to hex string\n  const hashHex = hashArray\n    .map((b) => ('00' + b.toString(16)).slice(-2))\n    .join('');\n  return hashHex;\n}\n\n@Injectable({\n  providedIn: 'root',\n})\nexport class KeyFactoryService {\n  constructor(private webCryptoService: WebCryptoService) {\n    this.crypto = this.webCryptoService.crypto;\n  }\n  private readonly crypto;\n  // Global keys store. Otherwise, each call to asKey creates a new keyStore.\n  // <AZ> Did not seem to improve speed.\n  // public static keyStore = JWK.createKeyStore();\n\n  // AZ: This can't be change easily. It's basically a PassK or PassIdp rotation.\n  // todo: we should eventually increase this periodically to match with Moore's law.\n  // The iterations for each key are kept by the server as well but we assume the value\n  // from the server is not trustworthy, so need to have minimum thresholds here.\n  // If creating new keys, these minimum are used.\n  public readonly MIN_PASS_IDP_PBKDF_ITER = 100000;\n  public readonly MIN_PASS_KEY_PBKDF_ITER = 100000;\n  public readonly MIN_LBOP_KEY_PBKDF_ITER = 100000;\n\n  // These are used as the default values. They must be larger than the minimum values.\n  public readonly DEFAULT_PASS_IDP_PBKDF_ITER = this.MIN_PASS_IDP_PBKDF_ITER;\n  public readonly DEFAULT_PASS_KEY_PBKDF_ITER = this.MIN_PASS_KEY_PBKDF_ITER;\n  public readonly DEFAULT_LBOP_KEY_PBKDF_ITER = this.MIN_LBOP_KEY_PBKDF_ITER;\n\n  static asKey(\n    key: string | Buffer | object | JWK.RawKey,\n    form?:\n      | 'json'\n      | 'private'\n      | 'pkcs8'\n      | 'public'\n      | 'spki'\n      | 'pkix'\n      | 'x509'\n      | 'pem',\n    extras?: Record<string, unknown>\n  ): Promise<JWK.Key> {\n    // <AZ> Using a single global key store did not seem to improve speed.\n    // return KeyFactoryService.keyStore.add(key, form, extras);\n    return JWK.asKey(key, form, extras);\n  }\n\n  randomString(digits: number): string {\n    if (digits <= 0) {\n      throw new LrBadArgumentException('digits <= 0');\n    }\n    const validChars =\n      'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789';\n    let array = new Uint32Array(digits);\n    this.crypto.getRandomValues(array);\n    array = array.map((x) => validChars.charCodeAt(x % validChars.length));\n    return String.fromCharCode.apply(null, array);\n  }\n\n  randomDigitsNoZeros(digits: number): string {\n    return this.randomChoices([1, 2, 3, 4, 5, 6, 7, 8, 9], digits).join('');\n  }\n\n  randomChoices<T>(array: T[], chooseN: number): T[] {\n    if (array.length <= 1) {\n      throw new LrBadArgumentException('array.length <= 0');\n    }\n    if (chooseN <= 0) {\n      throw new LrBadArgumentException('chooseN <= 0');\n    }\n    const values = new Uint32Array(chooseN);\n    this.crypto.getRandomValues(values);\n    const ret: T[] = [];\n    values.forEach((v) => ret.push(array[v % array.length]));\n    return ret;\n  }\n\n  createSalt(): string {\n    return this.randomString(16);\n  }\n\n  async createKey(): Promise<JWK.Key> {\n    const key = await this.crypto.subtle.generateKey(\n      {\n        name: 'AES-GCM',\n        length: 256, // can be  128, 192, or 256\n      },\n      true, // whether the key is extractable (i.e. can be used in exportKey)\n      ['encrypt', 'decrypt'] // must be [\"encrypt\", \"decrypt\"] or [\"wrapKey\", \"unwrapKey\"]\n    );\n\n    const jwk = await this.crypto.subtle.exportKey('jwk', key);\n\n    // Removing the fields not needed by node-jose\n    delete jwk.ext;\n    delete jwk.key_ops;\n\n    return KeyFactoryService.asKey(jwk);\n  }\n\n  async createSignKey(): Promise<JWK.Key> {\n    const key = await this.crypto.subtle.generateKey(\n      {\n        name: 'HMAC',\n        hash: { name: 'SHA-512' },\n      },\n      true,\n      ['sign', 'verify']\n    );\n\n    const jwk = await this.crypto.subtle.exportKey('jwk', key);\n\n    // Removing the fields not needed by node-jose\n    delete jwk.key_ops;\n    delete jwk.ext;\n\n    return KeyFactoryService.asKey(jwk);\n  }\n\n  async createPkcKey(): Promise<JWK.Key> {\n    // node-jose is not using Forge properly. It should be calling the async version of\n    // pki.rsa.generateKeyPair() with a callback. Instead it calls the sync version. Webcrypto\n    // does not support sync version, so it uses the javascript implementation, which is way too slow.\n    // So we generate using webcrypto and import the key.\n    // Unfortunately Elliptical Curve is not supported by Webcrypto. So we have to settle for RSA.\n    const key = await this.crypto.subtle.generateKey(\n      {\n        name: 'RSA-OAEP',\n        modulusLength: 2048, // can be 1024, 2048, 3072, 4096 ... 16384\n        // As per suggestion: https://developer.mozilla.org/en-US/docs/Web/API/RsaHashedKeyGenParams\n        publicExponent: new Uint8Array([0x01, 0x00, 0x01]),\n        hash: { name: 'SHA-256' }, // can be \"SHA-1\", \"SHA-256\", \"SHA-384\", or \"SHA-512\"\n      },\n      true, // whether the key is extractable (i.e. can be used in exportKey)\n      ['encrypt', 'decrypt'] // must be [\"encrypt\", \"decrypt\"] or [\"wrapKey\", \"unwrapKey\"]\n    );\n\n    const jwk = await this.crypto.subtle.exportKey('jwk', key.privateKey);\n    // Removing the fields not needed by node-jose\n    delete jwk.key_ops;\n    delete jwk.ext;\n\n    return KeyFactoryService.asKey(jwk);\n  }\n\n  async createPkcSignKey(): Promise<JWK.Key> {\n    const key = await this.crypto.subtle.generateKey(\n      {\n        name: 'RSASSA-PKCS1-v1_5',\n        modulusLength: 2048, // can be 1024, 2048, or 4096\n        // As per suggestion: https://developer.mozilla.org/en-US/docs/Web/API/RsaHashedKeyGenParams\n        publicExponent: new Uint8Array([0x01, 0x00, 0x01]),\n        hash: { name: 'SHA-256' }, // can be \"SHA-1\", \"SHA-256\", \"SHA-384\", or \"SHA-512\"\n      },\n      true, // whether the key is extractable (i.e. can be used in exportKey)\n      ['sign', 'verify'] // can be any combination of \"sign\" and \"verify\"\n    );\n\n    const jwk = await this.crypto.subtle.exportKey('jwk', key.privateKey);\n\n    // Removing the fields not needed by node-jose\n    delete jwk.key_ops;\n    delete jwk.ext;\n\n    return KeyFactoryService.asKey(jwk);\n  }\n\n  async deriveKey({\n    password,\n    salt,\n    iterations,\n    kid,\n  }: {\n    password: string;\n    salt: string;\n    iterations: number;\n    kid?: string;\n  }): Promise<DeriveKeyResult> {\n    const enc = new TextEncoder();\n    const rawKey = await this.crypto.subtle.importKey(\n      'raw',\n      enc.encode(password),\n      'PBKDF2',\n      false,\n      ['deriveBits', 'deriveKey']\n    );\n\n    const passKey = await crypto.subtle.deriveKey(\n      {\n        name: 'PBKDF2',\n        salt: new TextEncoder().encode(salt),\n        iterations,\n        hash: 'SHA-256',\n      },\n      rawKey,\n      { name: 'AES-GCM', length: 256 },\n      true,\n      ['encrypt', 'decrypt']\n    );\n\n    const passKeyJson: any = await crypto.subtle.exportKey('jwk', passKey);\n    if (kid) {\n      passKeyJson.kid = kid;\n    }\n\n    const jwk = await KeyFactoryService.asKey(passKeyJson);\n\n    return { jwk };\n  }\n\n  async derivePassIdp(params: DerivePassIdpParams): Promise<DeriveKeyResult> {\n    if (params.iterations < this.MIN_PASS_IDP_PBKDF_ITER) {\n      throw new LrSuspiciousException(\n        `The number of PassIdp key derivation iterations sent from the server (${params.iterations}) is lower than the minimum (${this.MIN_PASS_IDP_PBKDF_ITER})`\n      );\n    }\n    return this.deriveKey(params);\n  }\n\n  async derivePassKey(params: DerivePassKeyParams): Promise<DeriveKeyResult> {\n    if (params.iterations < this.MIN_PASS_KEY_PBKDF_ITER) {\n      throw new LrSuspiciousException(\n        `The number of PassKey key derivation iterations sent from the server(${params.iterations}) is lower than the minimum(${this.MIN_PASS_KEY_PBKDF_ITER})`\n      );\n    }\n    return this.deriveKey(params);\n  }\n\n  async deriveLbopKey(params: DeriveLbopKeyParams): Promise<DeriveKeyResult> {\n    if (params.iterations < this.MIN_LBOP_KEY_PBKDF_ITER) {\n      throw new LrSuspiciousException(\n        `The number of LbopKey key derivation iterations sent from the server(${params.iterations}) is lower than the minimum(${this.MIN_LBOP_KEY_PBKDF_ITER})`\n      );\n    }\n    return this.deriveKey(params);\n  }\n\n  async createKid(): Promise<string> {\n    // todo: AZ: node-jose source uses node's default UUID() function for kid, so just change to use that.\n    // for now, we are just creating a new key to use it's kid.\n    // The kid is a part of the JWK system. LR backend maintains the key hierarchy separately with it's own\n    // key id. But we just use it here as a double check.\n    return (await this.createKey()).kid;\n  }\n\n  async createPassIdpParams(): Promise<PassIdpParams> {\n    return {\n      salt: this.createSalt(),\n      iterations: this.DEFAULT_PASS_IDP_PBKDF_ITER,\n    };\n  }\n\n  async createPassKeyParams(): Promise<PassKeyParams> {\n    return {\n      salt: this.createSalt(),\n      kid: await this.createKid(),\n      iterations: this.DEFAULT_PASS_KEY_PBKDF_ITER,\n    };\n  }\n\n  async createLbopKeyParams(): Promise<LbopKeyParams> {\n    return {\n      salt: this.createSalt(),\n      // todo: AZ: node-jose source uses node's default UUID() function for kid, so just change to use that.\n      // for now, we are just creating a new key to use it's kid.\n      // The kid is a part of the JWK system. LR backend maintains the key hierarchy separately with it's own\n      // key id. But we just use it here as a double check.\n      kid: await this.createKid(),\n      iterations: this.DEFAULT_PASS_KEY_PBKDF_ITER,\n    };\n  }\n}\n"]}
|
|
1
|
+
import { __awaiter } from "tslib";
|
|
2
|
+
import { Injectable } from '@angular/core';
|
|
3
|
+
import { JWK } from 'node-jose';
|
|
4
|
+
import { WebCryptoService } from './web-crypto.service';
|
|
5
|
+
import { LrBadArgumentException, LrSuspiciousException, } from '../_common/exceptions';
|
|
6
|
+
import * as i0 from "@angular/core";
|
|
7
|
+
import * as i1 from "./web-crypto.service";
|
|
8
|
+
export function sha256(message) {
|
|
9
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
10
|
+
// encode as UTF-8
|
|
11
|
+
const msgBuffer = new TextEncoder().encode(message);
|
|
12
|
+
// hash the message
|
|
13
|
+
const hashBuffer = yield crypto.subtle.digest('SHA-256', msgBuffer);
|
|
14
|
+
// convert ArrayBuffer to Array
|
|
15
|
+
const hashArray = Array.from(new Uint8Array(hashBuffer));
|
|
16
|
+
// convert bytes to hex string
|
|
17
|
+
const hashHex = hashArray
|
|
18
|
+
.map((b) => ('00' + b.toString(16)).slice(-2))
|
|
19
|
+
.join('');
|
|
20
|
+
return hashHex;
|
|
21
|
+
});
|
|
22
|
+
}
|
|
23
|
+
export class KeyFactoryService {
|
|
24
|
+
constructor(webCryptoService) {
|
|
25
|
+
this.webCryptoService = webCryptoService;
|
|
26
|
+
// Global keys store. Otherwise, each call to asKey creates a new keyStore.
|
|
27
|
+
// <AZ> Did not seem to improve speed.
|
|
28
|
+
// public static keyStore = JWK.createKeyStore();
|
|
29
|
+
// AZ: This can't be change easily. It's basically a PassK or PassIdp rotation.
|
|
30
|
+
// todo: we should eventually increase this periodically to match with Moore's law.
|
|
31
|
+
// The iterations for each key are kept by the server as well but we assume the value
|
|
32
|
+
// from the server is not trustworthy, so need to have minimum thresholds here.
|
|
33
|
+
// If creating new keys, these minimum are used.
|
|
34
|
+
this.MIN_PASS_IDP_PBKDF_ITER = 100000;
|
|
35
|
+
this.MIN_PASS_KEY_PBKDF_ITER = 100000;
|
|
36
|
+
this.MIN_LBOP_KEY_PBKDF_ITER = 100000;
|
|
37
|
+
// These are used as the default values. They must be larger than the minimum values.
|
|
38
|
+
this.DEFAULT_PASS_IDP_PBKDF_ITER = this.MIN_PASS_IDP_PBKDF_ITER;
|
|
39
|
+
this.DEFAULT_PASS_KEY_PBKDF_ITER = this.MIN_PASS_KEY_PBKDF_ITER;
|
|
40
|
+
this.DEFAULT_LBOP_KEY_PBKDF_ITER = this.MIN_LBOP_KEY_PBKDF_ITER;
|
|
41
|
+
this.crypto = this.webCryptoService.crypto;
|
|
42
|
+
}
|
|
43
|
+
static asKey(key, form, extras) {
|
|
44
|
+
// <AZ> Using a single global key store did not seem to improve speed.
|
|
45
|
+
// return KeyFactoryService.keyStore.add(key, form, extras);
|
|
46
|
+
return JWK.asKey(key, form, extras);
|
|
47
|
+
}
|
|
48
|
+
randomString(digits) {
|
|
49
|
+
if (digits <= 0) {
|
|
50
|
+
throw new LrBadArgumentException('digits <= 0');
|
|
51
|
+
}
|
|
52
|
+
const validChars = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789';
|
|
53
|
+
let array = new Uint32Array(digits);
|
|
54
|
+
this.crypto.getRandomValues(array);
|
|
55
|
+
array = array.map((x) => validChars.charCodeAt(x % validChars.length));
|
|
56
|
+
return String.fromCharCode.apply(null, array);
|
|
57
|
+
}
|
|
58
|
+
randomDigitsNoZeros(digits) {
|
|
59
|
+
return this.randomChoices([1, 2, 3, 4, 5, 6, 7, 8, 9], digits).join('');
|
|
60
|
+
}
|
|
61
|
+
randomChoices(array, chooseN) {
|
|
62
|
+
if (array.length <= 1) {
|
|
63
|
+
throw new LrBadArgumentException('array.length <= 0');
|
|
64
|
+
}
|
|
65
|
+
if (chooseN <= 0) {
|
|
66
|
+
throw new LrBadArgumentException('chooseN <= 0');
|
|
67
|
+
}
|
|
68
|
+
const values = new Uint32Array(chooseN);
|
|
69
|
+
this.crypto.getRandomValues(values);
|
|
70
|
+
const ret = [];
|
|
71
|
+
values.forEach((v) => ret.push(array[v % array.length]));
|
|
72
|
+
return ret;
|
|
73
|
+
}
|
|
74
|
+
createSalt() {
|
|
75
|
+
return this.randomString(16);
|
|
76
|
+
}
|
|
77
|
+
createKey() {
|
|
78
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
79
|
+
const key = yield this.crypto.subtle.generateKey({
|
|
80
|
+
name: 'AES-GCM',
|
|
81
|
+
length: 256,
|
|
82
|
+
}, true, // whether the key is extractable (i.e. can be used in exportKey)
|
|
83
|
+
['encrypt', 'decrypt'] // must be ["encrypt", "decrypt"] or ["wrapKey", "unwrapKey"]
|
|
84
|
+
);
|
|
85
|
+
const jwk = yield this.crypto.subtle.exportKey('jwk', key);
|
|
86
|
+
// Removing the fields not needed by node-jose
|
|
87
|
+
delete jwk.ext;
|
|
88
|
+
delete jwk.key_ops;
|
|
89
|
+
return KeyFactoryService.asKey(jwk);
|
|
90
|
+
});
|
|
91
|
+
}
|
|
92
|
+
createSignKey() {
|
|
93
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
94
|
+
const key = yield this.crypto.subtle.generateKey({
|
|
95
|
+
name: 'HMAC',
|
|
96
|
+
hash: { name: 'SHA-512' },
|
|
97
|
+
}, true, ['sign', 'verify']);
|
|
98
|
+
const jwk = yield this.crypto.subtle.exportKey('jwk', key);
|
|
99
|
+
// Removing the fields not needed by node-jose
|
|
100
|
+
delete jwk.key_ops;
|
|
101
|
+
delete jwk.ext;
|
|
102
|
+
return KeyFactoryService.asKey(jwk);
|
|
103
|
+
});
|
|
104
|
+
}
|
|
105
|
+
createPkcKey() {
|
|
106
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
107
|
+
// node-jose is not using Forge properly. It should be calling the async version of
|
|
108
|
+
// pki.rsa.generateKeyPair() with a callback. Instead it calls the sync version. Webcrypto
|
|
109
|
+
// does not support sync version, so it uses the javascript implementation, which is way too slow.
|
|
110
|
+
// So we generate using webcrypto and import the key.
|
|
111
|
+
// Unfortunately Elliptical Curve is not supported by Webcrypto. So we have to settle for RSA.
|
|
112
|
+
const key = yield this.crypto.subtle.generateKey({
|
|
113
|
+
name: 'RSA-OAEP',
|
|
114
|
+
modulusLength: 2048,
|
|
115
|
+
// As per suggestion: https://developer.mozilla.org/en-US/docs/Web/API/RsaHashedKeyGenParams
|
|
116
|
+
publicExponent: new Uint8Array([0x01, 0x00, 0x01]),
|
|
117
|
+
hash: { name: 'SHA-256' },
|
|
118
|
+
}, true, // whether the key is extractable (i.e. can be used in exportKey)
|
|
119
|
+
['encrypt', 'decrypt'] // must be ["encrypt", "decrypt"] or ["wrapKey", "unwrapKey"]
|
|
120
|
+
);
|
|
121
|
+
const jwk = yield this.crypto.subtle.exportKey('jwk', key.privateKey);
|
|
122
|
+
// Removing the fields not needed by node-jose
|
|
123
|
+
delete jwk.key_ops;
|
|
124
|
+
delete jwk.ext;
|
|
125
|
+
return KeyFactoryService.asKey(jwk);
|
|
126
|
+
});
|
|
127
|
+
}
|
|
128
|
+
createPkcSignKey() {
|
|
129
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
130
|
+
const key = yield this.crypto.subtle.generateKey({
|
|
131
|
+
name: 'RSASSA-PKCS1-v1_5',
|
|
132
|
+
modulusLength: 2048,
|
|
133
|
+
// As per suggestion: https://developer.mozilla.org/en-US/docs/Web/API/RsaHashedKeyGenParams
|
|
134
|
+
publicExponent: new Uint8Array([0x01, 0x00, 0x01]),
|
|
135
|
+
hash: { name: 'SHA-256' },
|
|
136
|
+
}, true, // whether the key is extractable (i.e. can be used in exportKey)
|
|
137
|
+
['sign', 'verify'] // can be any combination of "sign" and "verify"
|
|
138
|
+
);
|
|
139
|
+
const jwk = yield this.crypto.subtle.exportKey('jwk', key.privateKey);
|
|
140
|
+
// Removing the fields not needed by node-jose
|
|
141
|
+
delete jwk.key_ops;
|
|
142
|
+
delete jwk.ext;
|
|
143
|
+
return KeyFactoryService.asKey(jwk);
|
|
144
|
+
});
|
|
145
|
+
}
|
|
146
|
+
deriveKey({ password, salt, iterations, kid, }) {
|
|
147
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
148
|
+
const enc = new TextEncoder();
|
|
149
|
+
const rawKey = yield this.crypto.subtle.importKey('raw', enc.encode(password), 'PBKDF2', false, ['deriveBits', 'deriveKey']);
|
|
150
|
+
const passKey = yield crypto.subtle.deriveKey({
|
|
151
|
+
name: 'PBKDF2',
|
|
152
|
+
salt: new TextEncoder().encode(salt),
|
|
153
|
+
iterations,
|
|
154
|
+
hash: 'SHA-256',
|
|
155
|
+
}, rawKey, { name: 'AES-GCM', length: 256 }, true, ['encrypt', 'decrypt']);
|
|
156
|
+
const passKeyJson = yield crypto.subtle.exportKey('jwk', passKey);
|
|
157
|
+
if (kid) {
|
|
158
|
+
passKeyJson.kid = kid;
|
|
159
|
+
}
|
|
160
|
+
const jwk = yield KeyFactoryService.asKey(passKeyJson);
|
|
161
|
+
return { jwk };
|
|
162
|
+
});
|
|
163
|
+
}
|
|
164
|
+
derivePassIdp(params) {
|
|
165
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
166
|
+
if (params.iterations < this.MIN_PASS_IDP_PBKDF_ITER) {
|
|
167
|
+
throw new LrSuspiciousException(`The number of PassIdp key derivation iterations sent from the server (${params.iterations}) is lower than the minimum (${this.MIN_PASS_IDP_PBKDF_ITER})`);
|
|
168
|
+
}
|
|
169
|
+
return this.deriveKey(params);
|
|
170
|
+
});
|
|
171
|
+
}
|
|
172
|
+
derivePassKey(params) {
|
|
173
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
174
|
+
if (params.iterations < this.MIN_PASS_KEY_PBKDF_ITER) {
|
|
175
|
+
throw new LrSuspiciousException(`The number of PassKey key derivation iterations sent from the server(${params.iterations}) is lower than the minimum(${this.MIN_PASS_KEY_PBKDF_ITER})`);
|
|
176
|
+
}
|
|
177
|
+
return this.deriveKey(params);
|
|
178
|
+
});
|
|
179
|
+
}
|
|
180
|
+
deriveLbopKey(params) {
|
|
181
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
182
|
+
if (params.iterations < this.MIN_LBOP_KEY_PBKDF_ITER) {
|
|
183
|
+
throw new LrSuspiciousException(`The number of LbopKey key derivation iterations sent from the server(${params.iterations}) is lower than the minimum(${this.MIN_LBOP_KEY_PBKDF_ITER})`);
|
|
184
|
+
}
|
|
185
|
+
return this.deriveKey(params);
|
|
186
|
+
});
|
|
187
|
+
}
|
|
188
|
+
createKid() {
|
|
189
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
190
|
+
// todo: AZ: node-jose source uses node's default UUID() function for kid, so just change to use that.
|
|
191
|
+
// for now, we are just creating a new key to use it's kid.
|
|
192
|
+
// The kid is a part of the JWK system. LR backend maintains the key hierarchy separately with it's own
|
|
193
|
+
// key id. But we just use it here as a double check.
|
|
194
|
+
return (yield this.createKey()).kid;
|
|
195
|
+
});
|
|
196
|
+
}
|
|
197
|
+
createPassIdpParams() {
|
|
198
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
199
|
+
return {
|
|
200
|
+
salt: this.createSalt(),
|
|
201
|
+
iterations: this.DEFAULT_PASS_IDP_PBKDF_ITER,
|
|
202
|
+
};
|
|
203
|
+
});
|
|
204
|
+
}
|
|
205
|
+
createPassKeyParams() {
|
|
206
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
207
|
+
return {
|
|
208
|
+
salt: this.createSalt(),
|
|
209
|
+
kid: yield this.createKid(),
|
|
210
|
+
iterations: this.DEFAULT_PASS_KEY_PBKDF_ITER,
|
|
211
|
+
};
|
|
212
|
+
});
|
|
213
|
+
}
|
|
214
|
+
createLbopKeyParams() {
|
|
215
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
216
|
+
return {
|
|
217
|
+
salt: this.createSalt(),
|
|
218
|
+
// todo: AZ: node-jose source uses node's default UUID() function for kid, so just change to use that.
|
|
219
|
+
// for now, we are just creating a new key to use it's kid.
|
|
220
|
+
// The kid is a part of the JWK system. LR backend maintains the key hierarchy separately with it's own
|
|
221
|
+
// key id. But we just use it here as a double check.
|
|
222
|
+
kid: yield this.createKid(),
|
|
223
|
+
iterations: this.DEFAULT_PASS_KEY_PBKDF_ITER,
|
|
224
|
+
};
|
|
225
|
+
});
|
|
226
|
+
}
|
|
227
|
+
}
|
|
228
|
+
KeyFactoryService.ɵprov = i0.ɵɵdefineInjectable({ factory: function KeyFactoryService_Factory() { return new KeyFactoryService(i0.ɵɵinject(i1.WebCryptoService)); }, token: KeyFactoryService, providedIn: "root" });
|
|
229
|
+
KeyFactoryService.decorators = [
|
|
230
|
+
{ type: Injectable, args: [{
|
|
231
|
+
providedIn: 'root',
|
|
232
|
+
},] }
|
|
233
|
+
];
|
|
234
|
+
KeyFactoryService.ctorParameters = () => [
|
|
235
|
+
{ type: WebCryptoService }
|
|
236
|
+
];
|
|
237
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"key-factory.service.js","sourceRoot":"C:/Projects/newrepo/kc-client/projects/core/src/","sources":["lib/cryptography/key-factory.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\r\nimport { JWK } from 'node-jose';\r\nimport {\r\n  LbopKeyParams,\r\n  PassIdpParams,\r\n  PassKeyParams,\r\n  DeriveKeyResult,\r\n  DerivePassIdpParams,\r\n  DerivePassKeyParams,\r\n  DeriveLbopKeyParams,\r\n} from './cryptography.types';\r\nimport { WebCryptoService } from './web-crypto.service';\r\nimport {\r\n  LrBadArgumentException,\r\n  LrSuspiciousException,\r\n} from '../_common/exceptions';\r\n\r\nexport async function sha256(message) {\r\n  // encode as UTF-8\r\n  const msgBuffer = new TextEncoder().encode(message);\r\n\r\n  // hash the message\r\n  const hashBuffer = await crypto.subtle.digest('SHA-256', msgBuffer);\r\n\r\n  // convert ArrayBuffer to Array\r\n  const hashArray = Array.from(new Uint8Array(hashBuffer));\r\n\r\n  // convert bytes to hex string\r\n  const hashHex = hashArray\r\n    .map((b) => ('00' + b.toString(16)).slice(-2))\r\n    .join('');\r\n  return hashHex;\r\n}\r\n\r\n@Injectable({\r\n  providedIn: 'root',\r\n})\r\nexport class KeyFactoryService {\r\n  constructor(private webCryptoService: WebCryptoService) {\r\n    this.crypto = this.webCryptoService.crypto;\r\n  }\r\n  private readonly crypto;\r\n  // Global keys store. Otherwise, each call to asKey creates a new keyStore.\r\n  // <AZ> Did not seem to improve speed.\r\n  // public static keyStore = JWK.createKeyStore();\r\n\r\n  // AZ: This can't be change easily. It's basically a PassK or PassIdp rotation.\r\n  // todo: we should eventually increase this periodically to match with Moore's law.\r\n  // The iterations for each key are kept by the server as well but we assume the value\r\n  // from the server is not trustworthy, so need to have minimum thresholds here.\r\n  // If creating new keys, these minimum are used.\r\n  public readonly MIN_PASS_IDP_PBKDF_ITER = 100000;\r\n  public readonly MIN_PASS_KEY_PBKDF_ITER = 100000;\r\n  public readonly MIN_LBOP_KEY_PBKDF_ITER = 100000;\r\n\r\n  // These are used as the default values. They must be larger than the minimum values.\r\n  public readonly DEFAULT_PASS_IDP_PBKDF_ITER = this.MIN_PASS_IDP_PBKDF_ITER;\r\n  public readonly DEFAULT_PASS_KEY_PBKDF_ITER = this.MIN_PASS_KEY_PBKDF_ITER;\r\n  public readonly DEFAULT_LBOP_KEY_PBKDF_ITER = this.MIN_LBOP_KEY_PBKDF_ITER;\r\n\r\n  static asKey(\r\n    key: string | Buffer | object | JWK.RawKey,\r\n    form?:\r\n      | 'json'\r\n      | 'private'\r\n      | 'pkcs8'\r\n      | 'public'\r\n      | 'spki'\r\n      | 'pkix'\r\n      | 'x509'\r\n      | 'pem',\r\n    extras?: Record<string, unknown>\r\n  ): Promise<JWK.Key> {\r\n    // <AZ> Using a single global key store did not seem to improve speed.\r\n    // return KeyFactoryService.keyStore.add(key, form, extras);\r\n    return JWK.asKey(key, form, extras);\r\n  }\r\n\r\n  randomString(digits: number): string {\r\n    if (digits <= 0) {\r\n      throw new LrBadArgumentException('digits <= 0');\r\n    }\r\n    const validChars =\r\n      'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789';\r\n    let array = new Uint32Array(digits);\r\n    this.crypto.getRandomValues(array);\r\n    array = array.map((x) => validChars.charCodeAt(x % validChars.length));\r\n    return String.fromCharCode.apply(null, array);\r\n  }\r\n\r\n  randomDigitsNoZeros(digits: number): string {\r\n    return this.randomChoices([1, 2, 3, 4, 5, 6, 7, 8, 9], digits).join('');\r\n  }\r\n\r\n  randomChoices<T>(array: T[], chooseN: number): T[] {\r\n    if (array.length <= 1) {\r\n      throw new LrBadArgumentException('array.length <= 0');\r\n    }\r\n    if (chooseN <= 0) {\r\n      throw new LrBadArgumentException('chooseN <= 0');\r\n    }\r\n    const values = new Uint32Array(chooseN);\r\n    this.crypto.getRandomValues(values);\r\n    const ret: T[] = [];\r\n    values.forEach((v) => ret.push(array[v % array.length]));\r\n    return ret;\r\n  }\r\n\r\n  createSalt(): string {\r\n    return this.randomString(16);\r\n  }\r\n\r\n  async createKey(): Promise<JWK.Key> {\r\n    const key = await this.crypto.subtle.generateKey(\r\n      {\r\n        name: 'AES-GCM',\r\n        length: 256, // can be  128, 192, or 256\r\n      },\r\n      true, // whether the key is extractable (i.e. can be used in exportKey)\r\n      ['encrypt', 'decrypt'] // must be [\"encrypt\", \"decrypt\"] or [\"wrapKey\", \"unwrapKey\"]\r\n    );\r\n\r\n    const jwk = await this.crypto.subtle.exportKey('jwk', key);\r\n\r\n    // Removing the fields not needed by node-jose\r\n    delete jwk.ext;\r\n    delete jwk.key_ops;\r\n\r\n    return KeyFactoryService.asKey(jwk);\r\n  }\r\n\r\n  async createSignKey(): Promise<JWK.Key> {\r\n    const key = await this.crypto.subtle.generateKey(\r\n      {\r\n        name: 'HMAC',\r\n        hash: { name: 'SHA-512' },\r\n      },\r\n      true,\r\n      ['sign', 'verify']\r\n    );\r\n\r\n    const jwk = await this.crypto.subtle.exportKey('jwk', key);\r\n\r\n    // Removing the fields not needed by node-jose\r\n    delete jwk.key_ops;\r\n    delete jwk.ext;\r\n\r\n    return KeyFactoryService.asKey(jwk);\r\n  }\r\n\r\n  async createPkcKey(): Promise<JWK.Key> {\r\n    // node-jose is not using Forge properly. It should be calling the async version of\r\n    // pki.rsa.generateKeyPair() with a callback. Instead it calls the sync version. Webcrypto\r\n    // does not support sync version, so it uses the javascript implementation, which is way too slow.\r\n    // So we generate using webcrypto and import the key.\r\n    // Unfortunately Elliptical Curve is not supported by Webcrypto. So we have to settle for RSA.\r\n    const key = await this.crypto.subtle.generateKey(\r\n      {\r\n        name: 'RSA-OAEP',\r\n        modulusLength: 2048, // can be 1024, 2048, 3072, 4096 ... 16384\r\n        // As per suggestion: https://developer.mozilla.org/en-US/docs/Web/API/RsaHashedKeyGenParams\r\n        publicExponent: new Uint8Array([0x01, 0x00, 0x01]),\r\n        hash: { name: 'SHA-256' }, // can be \"SHA-1\", \"SHA-256\", \"SHA-384\", or \"SHA-512\"\r\n      },\r\n      true, // whether the key is extractable (i.e. can be used in exportKey)\r\n      ['encrypt', 'decrypt'] // must be [\"encrypt\", \"decrypt\"] or [\"wrapKey\", \"unwrapKey\"]\r\n    );\r\n\r\n    const jwk = await this.crypto.subtle.exportKey('jwk', key.privateKey);\r\n    // Removing the fields not needed by node-jose\r\n    delete jwk.key_ops;\r\n    delete jwk.ext;\r\n\r\n    return KeyFactoryService.asKey(jwk);\r\n  }\r\n\r\n  async createPkcSignKey(): Promise<JWK.Key> {\r\n    const key = await this.crypto.subtle.generateKey(\r\n      {\r\n        name: 'RSASSA-PKCS1-v1_5',\r\n        modulusLength: 2048, // can be 1024, 2048, or 4096\r\n        // As per suggestion: https://developer.mozilla.org/en-US/docs/Web/API/RsaHashedKeyGenParams\r\n        publicExponent: new Uint8Array([0x01, 0x00, 0x01]),\r\n        hash: { name: 'SHA-256' }, // can be \"SHA-1\", \"SHA-256\", \"SHA-384\", or \"SHA-512\"\r\n      },\r\n      true, // whether the key is extractable (i.e. can be used in exportKey)\r\n      ['sign', 'verify'] // can be any combination of \"sign\" and \"verify\"\r\n    );\r\n\r\n    const jwk = await this.crypto.subtle.exportKey('jwk', key.privateKey);\r\n\r\n    // Removing the fields not needed by node-jose\r\n    delete jwk.key_ops;\r\n    delete jwk.ext;\r\n\r\n    return KeyFactoryService.asKey(jwk);\r\n  }\r\n\r\n  async deriveKey({\r\n    password,\r\n    salt,\r\n    iterations,\r\n    kid,\r\n  }: {\r\n    password: string;\r\n    salt: string;\r\n    iterations: number;\r\n    kid?: string;\r\n  }): Promise<DeriveKeyResult> {\r\n    const enc = new TextEncoder();\r\n    const rawKey = await this.crypto.subtle.importKey(\r\n      'raw',\r\n      enc.encode(password),\r\n      'PBKDF2',\r\n      false,\r\n      ['deriveBits', 'deriveKey']\r\n    );\r\n\r\n    const passKey = await crypto.subtle.deriveKey(\r\n      {\r\n        name: 'PBKDF2',\r\n        salt: new TextEncoder().encode(salt),\r\n        iterations,\r\n        hash: 'SHA-256',\r\n      },\r\n      rawKey,\r\n      { name: 'AES-GCM', length: 256 },\r\n      true,\r\n      ['encrypt', 'decrypt']\r\n    );\r\n\r\n    const passKeyJson: any = await crypto.subtle.exportKey('jwk', passKey);\r\n    if (kid) {\r\n      passKeyJson.kid = kid;\r\n    }\r\n\r\n    const jwk = await KeyFactoryService.asKey(passKeyJson);\r\n\r\n    return { jwk };\r\n  }\r\n\r\n  async derivePassIdp(params: DerivePassIdpParams): Promise<DeriveKeyResult> {\r\n    if (params.iterations < this.MIN_PASS_IDP_PBKDF_ITER) {\r\n      throw new LrSuspiciousException(\r\n        `The number of PassIdp key derivation iterations sent from the server (${params.iterations}) is lower than the minimum (${this.MIN_PASS_IDP_PBKDF_ITER})`\r\n      );\r\n    }\r\n    return this.deriveKey(params);\r\n  }\r\n\r\n  async derivePassKey(params: DerivePassKeyParams): Promise<DeriveKeyResult> {\r\n    if (params.iterations < this.MIN_PASS_KEY_PBKDF_ITER) {\r\n      throw new LrSuspiciousException(\r\n        `The number of PassKey key derivation iterations sent from the server(${params.iterations}) is lower than the minimum(${this.MIN_PASS_KEY_PBKDF_ITER})`\r\n      );\r\n    }\r\n    return this.deriveKey(params);\r\n  }\r\n\r\n  async deriveLbopKey(params: DeriveLbopKeyParams): Promise<DeriveKeyResult> {\r\n    if (params.iterations < this.MIN_LBOP_KEY_PBKDF_ITER) {\r\n      throw new LrSuspiciousException(\r\n        `The number of LbopKey key derivation iterations sent from the server(${params.iterations}) is lower than the minimum(${this.MIN_LBOP_KEY_PBKDF_ITER})`\r\n      );\r\n    }\r\n    return this.deriveKey(params);\r\n  }\r\n\r\n  async createKid(): Promise<string> {\r\n    // todo: AZ: node-jose source uses node's default UUID() function for kid, so just change to use that.\r\n    // for now, we are just creating a new key to use it's kid.\r\n    // The kid is a part of the JWK system. LR backend maintains the key hierarchy separately with it's own\r\n    // key id. But we just use it here as a double check.\r\n    return (await this.createKey()).kid;\r\n  }\r\n\r\n  async createPassIdpParams(): Promise<PassIdpParams> {\r\n    return {\r\n      salt: this.createSalt(),\r\n      iterations: this.DEFAULT_PASS_IDP_PBKDF_ITER,\r\n    };\r\n  }\r\n\r\n  async createPassKeyParams(): Promise<PassKeyParams> {\r\n    return {\r\n      salt: this.createSalt(),\r\n      kid: await this.createKid(),\r\n      iterations: this.DEFAULT_PASS_KEY_PBKDF_ITER,\r\n    };\r\n  }\r\n\r\n  async createLbopKeyParams(): Promise<LbopKeyParams> {\r\n    return {\r\n      salt: this.createSalt(),\r\n      // todo: AZ: node-jose source uses node's default UUID() function for kid, so just change to use that.\r\n      // for now, we are just creating a new key to use it's kid.\r\n      // The kid is a part of the JWK system. LR backend maintains the key hierarchy separately with it's own\r\n      // key id. But we just use it here as a double check.\r\n      kid: await this.createKid(),\r\n      iterations: this.DEFAULT_PASS_KEY_PBKDF_ITER,\r\n    };\r\n  }\r\n}\r\n"]}
|