@lifeready/core 1.0.12 → 1.0.15
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 +14315 -14315
- 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 +52 -52
- 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/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 +262 -262
- 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 +101 -101
- 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 +488 -488
- 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 +299 -299
- 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/items2/item2.gql.js +139 -139
- package/esm2015/lib/items2/item2.service.js +498 -498
- package/esm2015/lib/items2/item2.types.js +1 -1
- package/esm2015/lib/life-ready.config.js +84 -84
- package/esm2015/lib/life-ready.module.js +74 -74
- package/esm2015/lib/notification/notification.gql.js +43 -43
- package/esm2015/lib/notification/notification.service.js +118 -118
- 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/scenario.constants.js +2 -2
- package/esm2015/lib/scenario/scenario.controller.js +34 -34
- package/esm2015/lib/scenario/scenario.gql.js +72 -72
- package/esm2015/lib/scenario/scenario.gql.private.js +198 -198
- package/esm2015/lib/scenario/scenario.service.js +538 -538
- package/esm2015/lib/scenario/scenario.types.js +1 -1
- package/esm2015/lib/trusted-parties/tp-assembly.gql.private.js +22 -22
- package/esm2015/lib/trusted-parties/tp-assembly.js +362 -362
- package/esm2015/lib/trusted-parties/tp-assembly.types.js +1 -1
- package/esm2015/lib/trusted-parties/tp-password-reset-request.service.js +113 -113
- 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.controller.js +34 -34
- package/esm2015/lib/trusted-parties/tp-password-reset.gql.js +236 -236
- package/esm2015/lib/trusted-parties/tp-password-reset.service.js +95 -95
- 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 +13 -13
- package/esm2015/public-api.js +71 -71
- package/fesm2015/lifeready-core.js +12258 -12258
- 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 +9 -9
- 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/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 +34 -34
- 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 +28 -28
- 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 +502 -502
- 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 +61 -61
- 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 +41 -41
- 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/items2/item2.gql.d.ts +123 -123
- package/lib/items2/item2.service.d.ts +203 -203
- package/lib/items2/item2.types.d.ts +70 -70
- package/lib/life-ready.config.d.ts +14 -14
- package/lib/life-ready.module.d.ts +5 -5
- package/lib/notification/notification.gql.d.ts +37 -37
- package/lib/notification/notification.service.d.ts +63 -63
- 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/scenario.constants.d.ts +1 -1
- package/lib/scenario/scenario.controller.d.ts +10 -10
- package/lib/scenario/scenario.gql.d.ts +62 -62
- package/lib/scenario/scenario.gql.private.d.ts +16 -16
- package/lib/scenario/scenario.service.d.ts +233 -233
- package/lib/scenario/scenario.types.d.ts +50 -50
- package/lib/trusted-parties/tp-assembly.d.ts +177 -177
- package/lib/trusted-parties/tp-assembly.gql.private.d.ts +5 -5
- package/lib/trusted-parties/tp-assembly.types.d.ts +38 -38
- 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.controller.d.ts +10 -10
- package/lib/trusted-parties/tp-password-reset.gql.d.ts +223 -223
- package/lib/trusted-parties/tp-password-reset.service.d.ts +188 -188
- 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 +12 -12
- package/package.json +1 -1
- package/public-api.d.ts +67 -67
|
@@ -1,362 +1,362 @@
|
|
|
1
|
-
import { __awaiter, __rest } from "tslib";
|
|
2
|
-
import { mapEdges, } from '../api/types';
|
|
3
|
-
import { EncryptionService } from '../cryptography/encryption.service';
|
|
4
|
-
import { KeyFactoryService } from '../cryptography/key-factory.service';
|
|
5
|
-
import { KeyGraphService } from '../cryptography/key-graph.service';
|
|
6
|
-
import * as slip from '../cryptography/slip39.service';
|
|
7
|
-
import { LrBadArgumentException, LrBadStateException, } from '../_common/exceptions';
|
|
8
|
-
import { KeyService } from '../cryptography/key.service';
|
|
9
|
-
import { LrGraphQLService } from '../api/lr-graphql';
|
|
10
|
-
import { TpsKeysQuery } from './tp-assembly.gql.private';
|
|
11
|
-
export class TpAssemblyController {
|
|
12
|
-
constructor(injector) {
|
|
13
|
-
this.keyFactory = injector.get(KeyFactoryService);
|
|
14
|
-
this.keyService = injector.get(KeyService);
|
|
15
|
-
this.encryptionService = injector.get(EncryptionService);
|
|
16
|
-
this.keyGraph = injector.get(KeyGraphService);
|
|
17
|
-
this.slip39Service = injector.get(slip.Slip39Service);
|
|
18
|
-
this.lrGraphQl = injector.get(LrGraphQLService);
|
|
19
|
-
}
|
|
20
|
-
recoverAssemblyKey(partials) {
|
|
21
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
22
|
-
// Every receiverPbkCipher should contain this and they should all be the same.
|
|
23
|
-
let assemblyKeyParams;
|
|
24
|
-
const shares = partials.map((partial) => {
|
|
25
|
-
if (assemblyKeyParams) {
|
|
26
|
-
if (JSON.stringify(assemblyKeyParams) !==
|
|
27
|
-
JSON.stringify(partial.assemblyKeyParams)) {
|
|
28
|
-
throw new LrBadStateException('The assembly key parameters are different between the approvals.');
|
|
29
|
-
}
|
|
30
|
-
}
|
|
31
|
-
else {
|
|
32
|
-
assemblyKeyParams = partial.assemblyKeyParams;
|
|
33
|
-
}
|
|
34
|
-
return partial.slip39.share.mnemonics;
|
|
35
|
-
});
|
|
36
|
-
const rawAssemblyKey = yield this.slip39Service.recoverSecret(shares, this.slip39Passphrase);
|
|
37
|
-
return yield KeyFactoryService.asKey(Object.assign(Object.assign({}, assemblyKeyParams), { k: rawAssemblyKey }));
|
|
38
|
-
});
|
|
39
|
-
}
|
|
40
|
-
prepareTpWrappingKey(tp) {
|
|
41
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
42
|
-
return this.keyGraph.getKey(tp.wrappingKeyId);
|
|
43
|
-
});
|
|
44
|
-
}
|
|
45
|
-
prepareAssembly({ subjectKey, rootKey, }) {
|
|
46
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
47
|
-
const assemblyKey = yield this.keyFactory.createKey();
|
|
48
|
-
console.log('prepareAssembly assemblyKey', assemblyKey);
|
|
49
|
-
const _a = assemblyKey.toJSON(true), { k: rawAssemblyKey } = _a, assemblyKeyParams = __rest(_a, ["k"]);
|
|
50
|
-
const assemblyKeyVerifierPrk = yield this.keyFactory.createPkcSignKey();
|
|
51
|
-
const wrappedAssemblyKeyVerifierPrk = yield this.encryptionService.encryptToString(assemblyKey, assemblyKeyVerifierPrk.toJSON(true));
|
|
52
|
-
const assemblyKeyVerifierPbk = JSON.stringify(assemblyKeyVerifierPrk.toJSON());
|
|
53
|
-
const subjectKeyWrappedAssemblyKey = yield this.encryptionService.encryptToString(subjectKey, assemblyKey.toJSON(true));
|
|
54
|
-
// Encrypt the rootKey with the assemblyKey
|
|
55
|
-
const assemblyCipherData = yield this.encryptionService.encryptToString(assemblyKey, {
|
|
56
|
-
rootKey: rootKey.toJSON(true),
|
|
57
|
-
});
|
|
58
|
-
return {
|
|
59
|
-
assemblyKey,
|
|
60
|
-
rawAssemblyKey,
|
|
61
|
-
assemblyKeyParams,
|
|
62
|
-
subjectKeyWrappedAssemblyKey,
|
|
63
|
-
assemblyCipherData,
|
|
64
|
-
assemblyKeyVerifierPbk,
|
|
65
|
-
wrappedAssemblyKeyVerifierPrk,
|
|
66
|
-
};
|
|
67
|
-
});
|
|
68
|
-
}
|
|
69
|
-
prepareCreateSubAssemblies({ input, subjectKey, slipSubAssemblies, assemblyKeyParams, }) {
|
|
70
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
71
|
-
return Promise.all(input.map((sa, saIndex) => __awaiter(this, void 0, void 0, function* () {
|
|
72
|
-
const subjectCipherData = yield this.encryptionService.encryptToString(subjectKey, sa.subjectCipherDataClearJson || '');
|
|
73
|
-
const createApprovers = yield Promise.all(sa.approverTps.map((approverTp, approverIndex) => __awaiter(this, void 0, void 0, function* () {
|
|
74
|
-
return this.prepareApprover({
|
|
75
|
-
tp: approverTp,
|
|
76
|
-
approverIndex,
|
|
77
|
-
slipSubAssembly: slipSubAssemblies[saIndex],
|
|
78
|
-
assemblyKeyParams,
|
|
79
|
-
subjectKey,
|
|
80
|
-
});
|
|
81
|
-
})));
|
|
82
|
-
return {
|
|
83
|
-
singleReject: sa.singleReject,
|
|
84
|
-
quorum: sa.quorum,
|
|
85
|
-
subjectCipherData,
|
|
86
|
-
createApprovers,
|
|
87
|
-
};
|
|
88
|
-
})));
|
|
89
|
-
});
|
|
90
|
-
}
|
|
91
|
-
prepareUpdateSubAssemblies({ input, subjectKey, slipSubAssemblies, assemblyKeyParams, subAssemblies, }) {
|
|
92
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
93
|
-
if (slipSubAssemblies.length !== input.length) {
|
|
94
|
-
throw new LrBadArgumentException('The slipSubAssemblies must be the same length as the input');
|
|
95
|
-
}
|
|
96
|
-
return Promise.all(input.map((sa, saIndex) => __awaiter(this, void 0, void 0, function* () {
|
|
97
|
-
const subjectCipherData = yield this.encryptionService.encryptToString(subjectKey, sa.subjectCipherDataClearJson || '');
|
|
98
|
-
// Get the existing sub-assembly
|
|
99
|
-
const existingSa = subAssemblies.edges.find((edge) => edge.node.id === sa.id).node;
|
|
100
|
-
// Get approvers that do not exist yet
|
|
101
|
-
const createApprovers = [];
|
|
102
|
-
const updateApprovers = [];
|
|
103
|
-
sa.approverTps.forEach((tp) => {
|
|
104
|
-
var _a;
|
|
105
|
-
const approver = (_a = existingSa.approvers.edges.find((edge) => edge.node.tp.id === tp.tpId)) === null || _a === void 0 ? void 0 : _a.node;
|
|
106
|
-
if (approver) {
|
|
107
|
-
updateApprovers.push({
|
|
108
|
-
tp,
|
|
109
|
-
approverId: approver.id,
|
|
110
|
-
});
|
|
111
|
-
}
|
|
112
|
-
else {
|
|
113
|
-
createApprovers.push({
|
|
114
|
-
tp,
|
|
115
|
-
});
|
|
116
|
-
}
|
|
117
|
-
});
|
|
118
|
-
const slipSubAssembly = slipSubAssemblies[saIndex];
|
|
119
|
-
return {
|
|
120
|
-
subAssemblyId: sa.id,
|
|
121
|
-
singleReject: sa.singleReject,
|
|
122
|
-
quorum: sa.quorum,
|
|
123
|
-
subjectCipherData,
|
|
124
|
-
createApprovers: yield Promise.all(createApprovers.map(({ tp }, approverIndex) => __awaiter(this, void 0, void 0, function* () {
|
|
125
|
-
return this.prepareApprover({
|
|
126
|
-
tp,
|
|
127
|
-
approverIndex,
|
|
128
|
-
slipSubAssembly,
|
|
129
|
-
assemblyKeyParams,
|
|
130
|
-
subjectKey,
|
|
131
|
-
});
|
|
132
|
-
}))),
|
|
133
|
-
updateApprovers: yield Promise.all(updateApprovers.map(({ tp, approverId }, approverIndex) => __awaiter(this, void 0, void 0, function* () {
|
|
134
|
-
return this.prepareApprover({
|
|
135
|
-
approverId,
|
|
136
|
-
tp,
|
|
137
|
-
approverIndex: approverIndex + createApprovers.length,
|
|
138
|
-
slipSubAssembly,
|
|
139
|
-
assemblyKeyParams,
|
|
140
|
-
subjectKey,
|
|
141
|
-
});
|
|
142
|
-
}))),
|
|
143
|
-
};
|
|
144
|
-
})));
|
|
145
|
-
});
|
|
146
|
-
}
|
|
147
|
-
fillWrappingKeyId(input, tps) {
|
|
148
|
-
// Fill in wrappingKeyId if not present
|
|
149
|
-
input.forEach((sa) => {
|
|
150
|
-
sa.approverTps.forEach((approver) => {
|
|
151
|
-
if (!approver.wrappingKeyId) {
|
|
152
|
-
const tp = tps.find((x) => x.id === approver.tpId);
|
|
153
|
-
approver.wrappingKeyId = this.getTpWrappingKeyId(tp);
|
|
154
|
-
}
|
|
155
|
-
});
|
|
156
|
-
});
|
|
157
|
-
}
|
|
158
|
-
getInputApprovers(input) {
|
|
159
|
-
const tpIds = [];
|
|
160
|
-
input.forEach((sa) => {
|
|
161
|
-
sa.approverTps.forEach((approver) => {
|
|
162
|
-
tpIds.push(approver.tpId);
|
|
163
|
-
});
|
|
164
|
-
});
|
|
165
|
-
return tpIds;
|
|
166
|
-
}
|
|
167
|
-
prepareCreate(input) {
|
|
168
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
169
|
-
const rootKey = yield this.keyService.getCurrentRootKey();
|
|
170
|
-
const subjectKey = yield this.keyFactory.createKey();
|
|
171
|
-
const rootKeyWrappedSubjectKey = yield this.keyGraph.wrapKey(rootKey, subjectKey);
|
|
172
|
-
const _a = yield this.prepareAssembly({
|
|
173
|
-
rootKey: rootKey.jwk,
|
|
174
|
-
subjectKey,
|
|
175
|
-
}), { assemblyKey, rawAssemblyKey, assemblyKeyParams } = _a, assemblyInput = __rest(_a, ["assemblyKey", "rawAssemblyKey", "assemblyKeyParams"]);
|
|
176
|
-
const slipAssembly = yield this.prepareSlip39(input.createSubAssemblies, input.quorum, rawAssemblyKey);
|
|
177
|
-
const tpIds = this.getInputApprovers(input.createSubAssemblies);
|
|
178
|
-
// This should contain all the TPs that we need to update the assembly.
|
|
179
|
-
const tps = mapEdges((yield this.lrGraphQl.query({
|
|
180
|
-
query: TpsKeysQuery,
|
|
181
|
-
variables: {
|
|
182
|
-
ids: tpIds,
|
|
183
|
-
},
|
|
184
|
-
})).tps);
|
|
185
|
-
// Fill in wrappingKeyId if not present
|
|
186
|
-
this.fillWrappingKeyId(input.createSubAssemblies, tps);
|
|
187
|
-
const createSubAssemblies = yield this.prepareCreateSubAssemblies({
|
|
188
|
-
input: input.createSubAssemblies,
|
|
189
|
-
assemblyKeyParams,
|
|
190
|
-
slipSubAssemblies: slipAssembly.subAssemblies,
|
|
191
|
-
subjectKey,
|
|
192
|
-
});
|
|
193
|
-
return {
|
|
194
|
-
assemblyKey,
|
|
195
|
-
mutationInput: Object.assign(Object.assign({}, assemblyInput), { singleReject: input.singleReject, quorum: input.quorum, subjectCipherData: '', // TODO make this a parameter
|
|
196
|
-
createSubAssemblies, rootKeyId: rootKey.id, rootKeyWrappedSubjectKey }),
|
|
197
|
-
};
|
|
198
|
-
});
|
|
199
|
-
}
|
|
200
|
-
getAssemblyApprovers(assembly) {
|
|
201
|
-
const tpIds = [];
|
|
202
|
-
// Find all TPs that belong to sub-assemblies
|
|
203
|
-
mapEdges(assembly.subAssemblies).map((sa) => {
|
|
204
|
-
mapEdges(sa.approvers).forEach((approver) => tpIds.push(approver.tp.id));
|
|
205
|
-
});
|
|
206
|
-
return tpIds;
|
|
207
|
-
}
|
|
208
|
-
prepareUpdate(input, assembly // TODO type this to only what we need.
|
|
209
|
-
) {
|
|
210
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
211
|
-
input = Object.assign({ createSubAssemblies: [] }, input);
|
|
212
|
-
const createSubAssembliesInput = input.createSubAssemblies || [];
|
|
213
|
-
const updateSubAssembliesInput = input.updateSubAssemblies || [];
|
|
214
|
-
const deleteSubAssembliesInput = input.deleteSubAssemblies || [];
|
|
215
|
-
// Must have at least one.
|
|
216
|
-
if (createSubAssembliesInput.length === 0 &&
|
|
217
|
-
updateSubAssembliesInput.length === 0 &&
|
|
218
|
-
deleteSubAssembliesInput.length === 0) {
|
|
219
|
-
throw new LrBadArgumentException('Must specify at least one of: [createSubAssemblies, updateSubAssemblies, deleteSubAssemblies]');
|
|
220
|
-
}
|
|
221
|
-
const rootKey = yield this.keyService.getCurrentRootKey();
|
|
222
|
-
const subjectKey = yield this.keyGraph.getKey(assembly.subjectKey.id);
|
|
223
|
-
const _a = yield this.prepareAssembly({
|
|
224
|
-
rootKey: rootKey.jwk,
|
|
225
|
-
subjectKey: subjectKey.jwk,
|
|
226
|
-
}), { assemblyKey, rawAssemblyKey, assemblyKeyParams } = _a, assemblyInput = __rest(_a, ["assemblyKey", "rawAssemblyKey", "assemblyKeyParams"]);
|
|
227
|
-
// Find all the TPs that we need, duplicates are no problem.
|
|
228
|
-
// Existing approvers
|
|
229
|
-
const tpIds = this.getAssemblyApprovers(assembly).concat(this.getInputApprovers(createSubAssembliesInput), this.getInputApprovers(updateSubAssembliesInput));
|
|
230
|
-
// This should contain all the TPs that we need to update the assembly.
|
|
231
|
-
const tps = mapEdges((yield this.lrGraphQl.query({
|
|
232
|
-
query: TpsKeysQuery,
|
|
233
|
-
variables: {
|
|
234
|
-
ids: tpIds,
|
|
235
|
-
},
|
|
236
|
-
})).tps);
|
|
237
|
-
// Auto fill the updateSubAssemblies with existing sub assemblies if they are not
|
|
238
|
-
// explicitly deleted.
|
|
239
|
-
mapEdges(assembly.subAssemblies).forEach((existing) => {
|
|
240
|
-
// Deleting existing, don't include it
|
|
241
|
-
if (deleteSubAssembliesInput.includes(existing.id)) {
|
|
242
|
-
return;
|
|
243
|
-
}
|
|
244
|
-
// Update already specified in the input
|
|
245
|
-
if (updateSubAssembliesInput.some((sa) => sa.id === existing.id)) {
|
|
246
|
-
return;
|
|
247
|
-
}
|
|
248
|
-
const approverTps = mapEdges(existing.approvers).map((approver) => {
|
|
249
|
-
return {
|
|
250
|
-
tpId: approver.tp.id,
|
|
251
|
-
sharedCipherDataClearJson: approver.sharedCipherDataClearJson,
|
|
252
|
-
sharedCipherApprovalDataClearJson: approver.sharedCipherApprovalDataClearJson,
|
|
253
|
-
};
|
|
254
|
-
});
|
|
255
|
-
updateSubAssembliesInput.push({
|
|
256
|
-
id: existing.id,
|
|
257
|
-
quorum: existing.quorum,
|
|
258
|
-
singleReject: existing.singleReject,
|
|
259
|
-
approverTps,
|
|
260
|
-
subjectCipherDataClearJson: existing.subjectCipherDataClearJson,
|
|
261
|
-
});
|
|
262
|
-
});
|
|
263
|
-
// Fill in wrappingKeyId if not present
|
|
264
|
-
this.fillWrappingKeyId(createSubAssembliesInput, tps);
|
|
265
|
-
this.fillWrappingKeyId(updateSubAssembliesInput, tps);
|
|
266
|
-
let slipAssembly;
|
|
267
|
-
try {
|
|
268
|
-
slipAssembly = yield this.prepareSlip39((createSubAssembliesInput || []).concat(updateSubAssembliesInput || []), input.quorum, rawAssemblyKey);
|
|
269
|
-
}
|
|
270
|
-
catch (error) {
|
|
271
|
-
// TODO this sometimes happens. Maybe missing await.
|
|
272
|
-
console.log('Error while creating slip assembly', error);
|
|
273
|
-
}
|
|
274
|
-
const common = {
|
|
275
|
-
assemblyKeyParams,
|
|
276
|
-
subjectKey: subjectKey.jwk,
|
|
277
|
-
};
|
|
278
|
-
const createSubAssemblies = yield this.prepareCreateSubAssemblies(Object.assign(Object.assign({}, common), { input: createSubAssembliesInput,
|
|
279
|
-
// Need to match the slip sub assemblies to the input
|
|
280
|
-
slipSubAssemblies: slipAssembly.subAssemblies.slice(0, createSubAssembliesInput.length) }));
|
|
281
|
-
const updateSubAssemblies = yield this.prepareUpdateSubAssemblies(Object.assign(Object.assign({}, common), { input: updateSubAssembliesInput,
|
|
282
|
-
// Add an offset to the sub assemblies to skip the one that are created.
|
|
283
|
-
slipSubAssemblies: slipAssembly.subAssemblies.slice(createSubAssembliesInput.length), subAssemblies: assembly.subAssemblies }));
|
|
284
|
-
return {
|
|
285
|
-
assemblyKey,
|
|
286
|
-
mutationInput: Object.assign(Object.assign({}, assemblyInput), { singleReject: input.singleReject, quorum: input.quorum, subjectCipherData: '', subjectKeyId: subjectKey.id, createSubAssemblies,
|
|
287
|
-
updateSubAssemblies }),
|
|
288
|
-
};
|
|
289
|
-
});
|
|
290
|
-
}
|
|
291
|
-
prepareApprover({ approverId, tp, approverIndex, slipSubAssembly, assemblyKeyParams, subjectKey, }) {
|
|
292
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
293
|
-
const tpWrappingKey = yield this.prepareTpWrappingKey(tp);
|
|
294
|
-
// shared key is rotated every time.
|
|
295
|
-
const sharedKey = yield this.keyFactory.createKey();
|
|
296
|
-
// For TP to access shared_key
|
|
297
|
-
const tpWrappedSharedKey = yield this.encryptionService.encryptToString(tpWrappingKey.jwk, sharedKey.toJSON(true));
|
|
298
|
-
// For subject to access shared_key
|
|
299
|
-
const subjectKeyWrappedSharedKey = yield this.encryptionService.encryptToString(subjectKey, sharedKey.toJSON(true));
|
|
300
|
-
// If quorum is 1, then using the same share for every member.
|
|
301
|
-
const share = slipSubAssembly.threshold === 1
|
|
302
|
-
? slipSubAssembly.shares[0]
|
|
303
|
-
: slipSubAssembly.shares[approverIndex];
|
|
304
|
-
const partialAssemblyKey = {
|
|
305
|
-
slip39: {
|
|
306
|
-
share,
|
|
307
|
-
subAssembly: {
|
|
308
|
-
quorum: slipSubAssembly.threshold,
|
|
309
|
-
size: slipSubAssembly.size,
|
|
310
|
-
},
|
|
311
|
-
},
|
|
312
|
-
assemblyKeyParams,
|
|
313
|
-
};
|
|
314
|
-
console.log('partialAssemblyKey', partialAssemblyKey);
|
|
315
|
-
const sharedCipherData = yield this.encryptionService.encryptToString(sharedKey, tp.sharedCipherDataClearJson || '');
|
|
316
|
-
const sharedCipherApprovalData = yield this.encryptionService.encryptToString(sharedKey, tp.sharedCipherApprovalDataClearJson || '');
|
|
317
|
-
const sharedCipherPartialAssemblyKey = yield this.encryptionService.encryptToString(sharedKey, partialAssemblyKey);
|
|
318
|
-
return {
|
|
319
|
-
tpWrappingKeyId: tpWrappingKey.id,
|
|
320
|
-
tpWrappedSharedKey,
|
|
321
|
-
subjectKeyWrappedSharedKey,
|
|
322
|
-
sharedCipherData,
|
|
323
|
-
sharedCipherApprovalData,
|
|
324
|
-
sharedCipherPartialAssemblyKey,
|
|
325
|
-
approverId: approverId || void 0,
|
|
326
|
-
tpId: approverId ? void 0 : tp.tpId,
|
|
327
|
-
};
|
|
328
|
-
});
|
|
329
|
-
}
|
|
330
|
-
validateApprovers(approvers) {
|
|
331
|
-
// Ensure all approvers have mkSharedKey.
|
|
332
|
-
for (const tp of approvers) {
|
|
333
|
-
if (!tp.currentUserSharedKey.userSharedKey.mkSharedKey) {
|
|
334
|
-
const msg = `tp ${tp.other.username} does not have mkSharedKey`;
|
|
335
|
-
console.log(msg);
|
|
336
|
-
throw new LrBadArgumentException(msg);
|
|
337
|
-
}
|
|
338
|
-
}
|
|
339
|
-
}
|
|
340
|
-
// Prepare slip39
|
|
341
|
-
prepareSlip39(subAssemblies, assemblyQuorum, rawAssemblyKey) {
|
|
342
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
343
|
-
// Is there enough sub assemblies to meet quorum
|
|
344
|
-
if (subAssemblies.length < assemblyQuorum) {
|
|
345
|
-
throw new LrBadArgumentException('Not enough sub assemblies to meet quorum');
|
|
346
|
-
}
|
|
347
|
-
const slipAssembly = new slip.Assembly(assemblyQuorum);
|
|
348
|
-
subAssemblies.forEach((sa, index) => {
|
|
349
|
-
let approverCount = sa.approverTps.length;
|
|
350
|
-
// slip39 restricts quorum == 1 to have only 1 member. So we just share the same
|
|
351
|
-
// partial key for all sub assembly members.
|
|
352
|
-
if (sa.quorum === 1) {
|
|
353
|
-
approverCount = 1;
|
|
354
|
-
}
|
|
355
|
-
slipAssembly.addSubAssembly(new slip.SubAssembly(index, sa.quorum, approverCount));
|
|
356
|
-
});
|
|
357
|
-
yield this.slip39Service.generateShares(rawAssemblyKey, this.slip39Passphrase, slipAssembly);
|
|
358
|
-
return slipAssembly;
|
|
359
|
-
});
|
|
360
|
-
}
|
|
361
|
-
}
|
|
362
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tp-assembly.js","sourceRoot":"/opt/atlassian/pipelines/agent/build/projects/core/src/","sources":["lib/trusted-parties/tp-assembly.ts"],"names":[],"mappings":";AAEA,OAAO,EAEL,QAAQ,GAIT,MAAM,cAAc,CAAC;AACtB,OAAO,EAAE,iBAAiB,EAAE,MAAM,oCAAoC,CAAC;AACvE,OAAO,EAAE,iBAAiB,EAAE,MAAM,qCAAqC,CAAC;AACxE,OAAO,EAAE,eAAe,EAAE,MAAM,mCAAmC,CAAC;AACpE,OAAO,KAAK,IAAI,MAAM,gCAAgC,CAAC;AAEvD,OAAO,EACL,sBAAsB,EACtB,mBAAmB,GACpB,MAAM,uBAAuB,CAAC;AAC/B,OAAO,EAAE,UAAU,EAAE,MAAM,6BAA6B,CAAC;AASzD,OAAO,EAAE,gBAAgB,EAAE,MAAM,mBAAmB,CAAC;AACrD,OAAO,EAAE,YAAY,EAAE,MAAM,2BAA2B,CAAC;AAEzD,MAAM,OAAgB,oBAAoB;IAUxC,YAAY,QAAkB;QAC5B,IAAI,CAAC,UAAU,GAAG,QAAQ,CAAC,GAAG,CAAC,iBAAiB,CAAC,CAAC;QAClD,IAAI,CAAC,UAAU,GAAG,QAAQ,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC;QAC3C,IAAI,CAAC,iBAAiB,GAAG,QAAQ,CAAC,GAAG,CAAC,iBAAiB,CAAC,CAAC;QACzD,IAAI,CAAC,QAAQ,GAAG,QAAQ,CAAC,GAAG,CAAC,eAAe,CAAC,CAAC;QAC9C,IAAI,CAAC,aAAa,GAAG,QAAQ,CAAC,GAAG,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC;QACtD,IAAI,CAAC,SAAS,GAAG,QAAQ,CAAC,GAAG,CAAC,gBAAgB,CAAC,CAAC;IAClD,CAAC;IAIK,kBAAkB,CAAC,QAA8B;;YACrD,+EAA+E;YAC/E,IAAI,iBAAyB,CAAC;YAE9B,MAAM,MAAM,GAAG,QAAQ,CAAC,GAAG,CAAC,CAAC,OAAO,EAAE,EAAE;gBACtC,IAAI,iBAAiB,EAAE;oBACrB,IACE,IAAI,CAAC,SAAS,CAAC,iBAAiB,CAAC;wBACjC,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,iBAAiB,CAAC,EACzC;wBACA,MAAM,IAAI,mBAAmB,CAC3B,kEAAkE,CACnE,CAAC;qBACH;iBACF;qBAAM;oBACL,iBAAiB,GAAG,OAAO,CAAC,iBAAiB,CAAC;iBAC/C;gBACD,OAAO,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,SAAS,CAAC;YACxC,CAAC,CAAC,CAAC;YAEH,MAAM,cAAc,GAAG,MAAM,IAAI,CAAC,aAAa,CAAC,aAAa,CAC3D,MAAM,EACN,IAAI,CAAC,gBAAgB,CACtB,CAAC;YAEF,OAAO,MAAM,iBAAiB,CAAC,KAAK,iCAC/B,iBAAiB,KACpB,CAAC,EAAE,cAAc,IACjB,CAAC;QACL,CAAC;KAAA;IAEe,oBAAoB,CAClC,EAA2B;;YAE3B,OAAO,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,EAAE,CAAC,aAAa,CAAC,CAAC;QAChD,CAAC;KAAA;IAEe,eAAe,CAAC,EAC9B,UAAU,EACV,OAAO,GAIR;;YACC,MAAM,WAAW,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,SAAS,EAAE,CAAC;YACtD,OAAO,CAAC,GAAG,CAAC,6BAA6B,EAAE,WAAW,CAAC,CAAC;YAExD,MAAM,KAA8C,WAAW,CAAC,MAAM,CACpE,IAAI,CACE,EAFF,EAAE,CAAC,EAAE,cAAc,OAEjB,EAFsB,iBAAiB,cAAzC,KAA2C,CAEzC,CAAC;YACT,MAAM,sBAAsB,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,gBAAgB,EAAE,CAAC;YACxE,MAAM,6BAA6B,GACjC,MAAM,IAAI,CAAC,iBAAiB,CAAC,eAAe,CAC1C,WAAW,EACX,sBAAsB,CAAC,MAAM,CAAC,IAAI,CAAC,CACpC,CAAC;YACJ,MAAM,sBAAsB,GAAG,IAAI,CAAC,SAAS,CAC3C,sBAAsB,CAAC,MAAM,EAAE,CAChC,CAAC;YAEF,MAAM,4BAA4B,GAChC,MAAM,IAAI,CAAC,iBAAiB,CAAC,eAAe,CAC1C,UAAU,EACV,WAAW,CAAC,MAAM,CAAC,IAAI,CAAC,CACzB,CAAC;YAEJ,2CAA2C;YAC3C,MAAM,kBAAkB,GAAG,MAAM,IAAI,CAAC,iBAAiB,CAAC,eAAe,CACrE,WAAW,EACX;gBACE,OAAO,EAAE,OAAO,CAAC,MAAM,CAAC,IAAI,CAAC;aAC9B,CACF,CAAC;YAEF,OAAO;gBACL,WAAW;gBACX,cAAc;gBACd,iBAAiB;gBACjB,4BAA4B;gBAC5B,kBAAkB;gBAClB,sBAAsB;gBACtB,6BAA6B;aAC9B,CAAC;QACJ,CAAC;KAAA;IAEe,0BAA0B,CAAC,EACzC,KAAK,EACL,UAAU,EACV,iBAAiB,EACjB,iBAAiB,GAMlB;;YACC,OAAO,OAAO,CAAC,GAAG,CAChB,KAAK,CAAC,GAAG,CAAC,CAAO,EAAE,EAAE,OAAO,EAAE,EAAE;gBAC9B,MAAM,iBAAiB,GAAG,MAAM,IAAI,CAAC,iBAAiB,CAAC,eAAe,CACpE,UAAU,EACV,EAAE,CAAC,0BAA0B,IAAI,EAAE,CACpC,CAAC;gBAEF,MAAM,eAAe,GAAG,MAAM,OAAO,CAAC,GAAG,CACvC,EAAE,CAAC,WAAW,CAAC,GAAG,CAAC,CAAO,UAAU,EAAE,aAAa,EAAE,EAAE;oBACrD,OAAA,IAAI,CAAC,eAAe,CAAC;wBACnB,EAAE,EAAE,UAAU;wBACd,aAAa;wBACb,eAAe,EAAE,iBAAiB,CAAC,OAAO,CAAC;wBAC3C,iBAAiB;wBACjB,UAAU;qBACX,CAAC,CAAA;kBAAA,CACH,CACF,CAAC;gBAEF,OAAO;oBACL,YAAY,EAAE,EAAE,CAAC,YAAY;oBAC7B,MAAM,EAAE,EAAE,CAAC,MAAM;oBACjB,iBAAiB;oBACjB,eAAe;iBAChB,CAAC;YACJ,CAAC,CAAA,CAAC,CACH,CAAC;QACJ,CAAC;KAAA;IAEe,0BAA0B,CAAC,EACzC,KAAK,EACL,UAAU,EACV,iBAAiB,EACjB,iBAAiB,EACjB,aAAa,GAOd;;YACC,IAAI,iBAAiB,CAAC,MAAM,KAAK,KAAK,CAAC,MAAM,EAAE;gBAC7C,MAAM,IAAI,sBAAsB,CAC9B,4DAA4D,CAC7D,CAAC;aACH;YAED,OAAO,OAAO,CAAC,GAAG,CAChB,KAAK,CAAC,GAAG,CAAC,CAAO,EAAE,EAAE,OAAO,EAAE,EAAE;gBAC9B,MAAM,iBAAiB,GAAG,MAAM,IAAI,CAAC,iBAAiB,CAAC,eAAe,CACpE,UAAU,EACV,EAAE,CAAC,0BAA0B,IAAI,EAAE,CACpC,CAAC;gBAEF,gCAAgC;gBAChC,MAAM,UAAU,GAAG,aAAa,CAAC,KAAK,CAAC,IAAI,CACzC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,CAAC,EAAE,CACjC,CAAC,IAAI,CAAC;gBAEP,sCAAsC;gBACtC,MAAM,eAAe,GAAG,EAAE,CAAC;gBAC3B,MAAM,eAAe,GAAG,EAAE,CAAC;gBAE3B,EAAE,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC,EAAE,EAAE,EAAE;;oBAC5B,MAAM,QAAQ,SAAG,UAAU,CAAC,SAAS,CAAC,KAAK,CAAC,IAAI,CAC9C,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,EAAE,KAAK,EAAE,CAAC,IAAI,CACtC,0CAAE,IAAI,CAAC;oBACR,IAAI,QAAQ,EAAE;wBACZ,eAAe,CAAC,IAAI,CAAC;4BACnB,EAAE;4BACF,UAAU,EAAE,QAAQ,CAAC,EAAE;yBACxB,CAAC,CAAC;qBACJ;yBAAM;wBACL,eAAe,CAAC,IAAI,CAAC;4BACnB,EAAE;yBACH,CAAC,CAAC;qBACJ;gBACH,CAAC,CAAC,CAAC;gBAEH,MAAM,eAAe,GAAG,iBAAiB,CAAC,OAAO,CAAC,CAAC;gBAEnD,OAAO;oBACL,aAAa,EAAE,EAAE,CAAC,EAAE;oBACpB,YAAY,EAAE,EAAE,CAAC,YAAY;oBAC7B,MAAM,EAAE,EAAE,CAAC,MAAM;oBACjB,iBAAiB;oBACjB,eAAe,EAAE,MAAM,OAAO,CAAC,GAAG,CAChC,eAAe,CAAC,GAAG,CAAC,CAAO,EAAE,EAAE,EAAE,EAAE,aAAa,EAAE,EAAE;wBAClD,OAAA,IAAI,CAAC,eAAe,CAAC;4BACnB,EAAE;4BACF,aAAa;4BACb,eAAe;4BACf,iBAAiB;4BACjB,UAAU;yBACX,CAAC,CAAA;sBAAA,CACH,CACF;oBACD,eAAe,EAAE,MAAM,OAAO,CAAC,GAAG,CAChC,eAAe,CAAC,GAAG,CAAC,CAAO,EAAE,EAAE,EAAE,UAAU,EAAE,EAAE,aAAa,EAAE,EAAE;wBAC9D,OAAA,IAAI,CAAC,eAAe,CAAC;4BACnB,UAAU;4BACV,EAAE;4BACF,aAAa,EAAE,aAAa,GAAG,eAAe,CAAC,MAAM;4BACrD,eAAe;4BACf,iBAAiB;4BACjB,UAAU;yBACX,CAAC,CAAA;sBAAA,CACH,CACF;iBACF,CAAC;YACJ,CAAC,CAAA,CAAC,CACH,CAAC;QACJ,CAAC;KAAA;IAEO,iBAAiB,CAAC,KAAiC,EAAE,GAAa;QACxE,uCAAuC;QACvC,KAAK,CAAC,OAAO,CAAC,CAAC,EAAE,EAAE,EAAE;YACnB,EAAE,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC,QAAQ,EAAE,EAAE;gBAClC,IAAI,CAAC,QAAQ,CAAC,aAAa,EAAE;oBAC3B,MAAM,EAAE,GAAG,GAAG,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,EAAE,KAAK,QAAQ,CAAC,IAAI,CAAC,CAAC;oBACnD,QAAQ,CAAC,aAAa,GAAG,IAAI,CAAC,kBAAkB,CAAC,EAAE,CAAC,CAAC;iBACtD;YACH,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;IACL,CAAC;IAEO,iBAAiB,CAAC,KAAiC;QACzD,MAAM,KAAK,GAAa,EAAE,CAAC;QAC3B,KAAK,CAAC,OAAO,CAAC,CAAC,EAAE,EAAE,EAAE;YACnB,EAAE,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC,QAAQ,EAAE,EAAE;gBAClC,KAAK,CAAC,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC;YAC5B,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QACH,OAAO,KAAK,CAAC;IACf,CAAC;IAEY,aAAa,CAAC,KAA4B;;YACrD,MAAM,OAAO,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,iBAAiB,EAAE,CAAC;YAC1D,MAAM,UAAU,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,SAAS,EAAE,CAAC;YACrD,MAAM,wBAAwB,GAAG,MAAM,IAAI,CAAC,QAAQ,CAAC,OAAO,CAC1D,OAAO,EACP,UAAU,CACX,CAAC;YAEF,MAAM,KACJ,MAAM,IAAI,CAAC,eAAe,CAAC;gBACzB,OAAO,EAAE,OAAO,CAAC,GAAG;gBACpB,UAAU;aACX,CAAC,EAJE,EAAE,WAAW,EAAE,cAAc,EAAE,iBAAiB,OAIlD,EAJuD,aAAa,cAAlE,sDAAoE,CAItE,CAAC;YAEL,MAAM,YAAY,GAAG,MAAM,IAAI,CAAC,aAAa,CAC3C,KAAK,CAAC,mBAAmB,EACzB,KAAK,CAAC,MAAM,EACZ,cAAc,CACf,CAAC;YAEF,MAAM,KAAK,GAAG,IAAI,CAAC,iBAAiB,CAAC,KAAK,CAAC,mBAAmB,CAAC,CAAC;YAEhE,uEAAuE;YACvE,MAAM,GAAG,GAAG,QAAQ,CAClB,CACE,MAAM,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC;gBACzB,KAAK,EAAE,YAAY;gBACnB,SAAS,EAAE;oBACT,GAAG,EAAE,KAAK;iBACX;aACF,CAAC,CACH,CAAC,GAAG,CACN,CAAC;YAEF,uCAAuC;YACvC,IAAI,CAAC,iBAAiB,CAAC,KAAK,CAAC,mBAAmB,EAAE,GAAG,CAAC,CAAC;YAEvD,MAAM,mBAAmB,GAAG,MAAM,IAAI,CAAC,0BAA0B,CAAC;gBAChE,KAAK,EAAE,KAAK,CAAC,mBAAmB;gBAChC,iBAAiB;gBACjB,iBAAiB,EAAE,YAAY,CAAC,aAAa;gBAC7C,UAAU;aACX,CAAC,CAAC;YAEH,OAAO;gBACL,WAAW;gBACX,aAAa,kCACR,aAAa,KAChB,YAAY,EAAE,KAAK,CAAC,YAAY,EAChC,MAAM,EAAE,KAAK,CAAC,MAAM,EACpB,iBAAiB,EAAE,EAAE,EAAE,6BAA6B;oBACpD,mBAAmB,EACnB,SAAS,EAAE,OAAO,CAAC,EAAE,EACrB,wBAAwB,GACzB;aACF,CAAC;QACJ,CAAC;KAAA;IAEO,oBAAoB,CAAC,QAAwB;QACnD,MAAM,KAAK,GAAa,EAAE,CAAC;QAE3B,6CAA6C;QAC7C,QAAQ,CAAC,QAAQ,CAAC,aAAa,CAAC,CAAC,GAAG,CAAC,CAAC,EAAE,EAAE,EAAE;YAC1C,QAAQ,CAAC,EAAE,CAAC,SAAS,CAAC,CAAC,OAAO,CAAC,CAAC,QAAQ,EAAE,EAAE,CAAC,KAAK,CAAC,IAAI,CAAC,QAAQ,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC,CAAC;QAC3E,CAAC,CAAC,CAAC;QAEH,OAAO,KAAK,CAAC;IACf,CAAC;IAEY,aAAa,CACxB,KAA4B,EAC5B,QAAwB,CAAC,uCAAuC;;;YAEhE,KAAK,mBACH,mBAAmB,EAAE,EAAE,IACpB,KAAK,CACT,CAAC;YAEF,MAAM,wBAAwB,GAAG,KAAK,CAAC,mBAAmB,IAAI,EAAE,CAAC;YACjE,MAAM,wBAAwB,GAAG,KAAK,CAAC,mBAAmB,IAAI,EAAE,CAAC;YACjE,MAAM,wBAAwB,GAAG,KAAK,CAAC,mBAAmB,IAAI,EAAE,CAAC;YAEjE,0BAA0B;YAC1B,IACE,wBAAwB,CAAC,MAAM,KAAK,CAAC;gBACrC,wBAAwB,CAAC,MAAM,KAAK,CAAC;gBACrC,wBAAwB,CAAC,MAAM,KAAK,CAAC,EACrC;gBACA,MAAM,IAAI,sBAAsB,CAC9B,+FAA+F,CAChG,CAAC;aACH;YAED,MAAM,OAAO,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,iBAAiB,EAAE,CAAC;YAC1D,MAAM,UAAU,GAAG,MAAM,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,QAAQ,CAAC,UAAU,CAAC,EAAE,CAAC,CAAC;YAEtE,MAAM,KACJ,MAAM,IAAI,CAAC,eAAe,CAAC;gBACzB,OAAO,EAAE,OAAO,CAAC,GAAG;gBACpB,UAAU,EAAE,UAAU,CAAC,GAAG;aAC3B,CAAC,EAJE,EAAE,WAAW,EAAE,cAAc,EAAE,iBAAiB,OAIlD,EAJuD,aAAa,cAAlE,sDAAoE,CAItE,CAAC;YAEL,4DAA4D;YAC5D,qBAAqB;YACrB,MAAM,KAAK,GAAG,IAAI,CAAC,oBAAoB,CAAC,QAAQ,CAAC,CAAC,MAAM,CACtD,IAAI,CAAC,iBAAiB,CAAC,wBAAwB,CAAC,EAChD,IAAI,CAAC,iBAAiB,CAAC,wBAAwB,CAAC,CACjD,CAAC;YAEF,uEAAuE;YACvE,MAAM,GAAG,GAAG,QAAQ,CAClB,CACE,MAAM,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC;gBACzB,KAAK,EAAE,YAAY;gBACnB,SAAS,EAAE;oBACT,GAAG,EAAE,KAAK;iBACX;aACF,CAAC,CACH,CAAC,GAAG,CACN,CAAC;YAEF,iFAAiF;YACjF,sBAAsB;YACtB,QAAQ,CAAC,QAAQ,CAAC,aAAa,CAAC,CAAC,OAAO,CAAC,CAAC,QAAQ,EAAE,EAAE;gBACpD,sCAAsC;gBACtC,IAAI,wBAAwB,CAAC,QAAQ,CAAC,QAAQ,CAAC,EAAE,CAAC,EAAE;oBAClD,OAAO;iBACR;gBAED,wCAAwC;gBACxC,IAAI,wBAAwB,CAAC,IAAI,CAAC,CAAC,EAAE,EAAE,EAAE,CAAC,EAAE,CAAC,EAAE,KAAK,QAAQ,CAAC,EAAE,CAAC,EAAE;oBAChE,OAAO;iBACR;gBAED,MAAM,WAAW,GAA8B,QAAQ,CACrD,QAAQ,CAAC,SAAS,CACnB,CAAC,GAAG,CAAC,CAAC,QAAQ,EAAE,EAAE;oBACjB,OAAO;wBACL,IAAI,EAAE,QAAQ,CAAC,EAAE,CAAC,EAAE;wBACpB,yBAAyB,EAAE,QAAQ,CAAC,yBAAyB;wBAC7D,iCAAiC,EAC/B,QAAQ,CAAC,iCAAiC;qBAC7C,CAAC;gBACJ,CAAC,CAAC,CAAC;gBAEH,wBAAwB,CAAC,IAAI,CAAC;oBAC5B,EAAE,EAAE,QAAQ,CAAC,EAAE;oBACf,MAAM,EAAE,QAAQ,CAAC,MAAM;oBACvB,YAAY,EAAE,QAAQ,CAAC,YAAY;oBACnC,WAAW;oBACX,0BAA0B,EAAE,QAAQ,CAAC,0BAA0B;iBAChE,CAAC,CAAC;YACL,CAAC,CAAC,CAAC;YAEH,uCAAuC;YACvC,IAAI,CAAC,iBAAiB,CAAC,wBAAwB,EAAE,GAAG,CAAC,CAAC;YACtD,IAAI,CAAC,iBAAiB,CAAC,wBAAwB,EAAE,GAAG,CAAC,CAAC;YAEtD,IAAI,YAAY,CAAC;YACjB,IAAI;gBACF,YAAY,GAAG,MAAM,IAAI,CAAC,aAAa,CACrC,CAAC,wBAAwB,IAAI,EAAE,CAAC,CAAC,MAAM,CAAC,wBAAwB,IAAI,EAAE,CAAC,EACvE,KAAK,CAAC,MAAM,EACZ,cAAc,CACf,CAAC;aACH;YAAC,OAAO,KAAK,EAAE;gBACd,oDAAoD;gBACpD,OAAO,CAAC,GAAG,CAAC,oCAAoC,EAAE,KAAK,CAAC,CAAC;aAC1D;YAED,MAAM,MAAM,GAAG;gBACb,iBAAiB;gBACjB,UAAU,EAAE,UAAU,CAAC,GAAG;aAC3B,CAAC;YAEF,MAAM,mBAAmB,GAAG,MAAM,IAAI,CAAC,0BAA0B,iCAC5D,MAAM,KACT,KAAK,EAAE,wBAAwB;gBAC/B,qDAAqD;gBACrD,iBAAiB,EAAE,YAAY,CAAC,aAAa,CAAC,KAAK,CACjD,CAAC,EACD,wBAAwB,CAAC,MAAM,CAChC,IACD,CAAC;YAEH,MAAM,mBAAmB,GAAG,MAAM,IAAI,CAAC,0BAA0B,iCAC5D,MAAM,KACT,KAAK,EAAE,wBAAwB;gBAC/B,wEAAwE;gBACxE,iBAAiB,EAAE,YAAY,CAAC,aAAa,CAAC,KAAK,CACjD,wBAAwB,CAAC,MAAM,CAChC,EACD,aAAa,EAAE,QAAQ,CAAC,aAAa,IACrC,CAAC;YAEH,OAAO;gBACL,WAAW;gBACX,aAAa,kCACR,aAAa,KAChB,YAAY,EAAE,KAAK,CAAC,YAAY,EAChC,MAAM,EAAE,KAAK,CAAC,MAAM,EACpB,iBAAiB,EAAE,EAAE,EACrB,YAAY,EAAE,UAAU,CAAC,EAAE,EAC3B,mBAAmB;oBACnB,mBAAmB,GACpB;aACF,CAAC;QACJ,CAAC;KAAA;IAEa,eAAe,CAAC,EAC5B,UAAU,EACV,EAAE,EACF,aAAa,EACb,eAAe,EACf,iBAAiB,EACjB,UAAU,GAQX;;YACC,MAAM,aAAa,GAAG,MAAM,IAAI,CAAC,oBAAoB,CAAC,EAAE,CAAC,CAAC;YAC1D,oCAAoC;YACpC,MAAM,SAAS,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,SAAS,EAAE,CAAC;YAEpD,8BAA8B;YAC9B,MAAM,kBAAkB,GAAG,MAAM,IAAI,CAAC,iBAAiB,CAAC,eAAe,CACrE,aAAa,CAAC,GAAG,EACjB,SAAS,CAAC,MAAM,CAAC,IAAI,CAAC,CACvB,CAAC;YACF,mCAAmC;YACnC,MAAM,0BAA0B,GAC9B,MAAM,IAAI,CAAC,iBAAiB,CAAC,eAAe,CAC1C,UAAU,EACV,SAAS,CAAC,MAAM,CAAC,IAAI,CAAC,CACvB,CAAC;YAEJ,8DAA8D;YAC9D,MAAM,KAAK,GACT,eAAe,CAAC,SAAS,KAAK,CAAC;gBAC7B,CAAC,CAAC,eAAe,CAAC,MAAM,CAAC,CAAC,CAAC;gBAC3B,CAAC,CAAC,eAAe,CAAC,MAAM,CAAC,aAAa,CAAC,CAAC;YAE5C,MAAM,kBAAkB,GAAuB;gBAC7C,MAAM,EAAE;oBACN,KAAK;oBACL,WAAW,EAAE;wBACX,MAAM,EAAE,eAAe,CAAC,SAAS;wBACjC,IAAI,EAAE,eAAe,CAAC,IAAI;qBAC3B;iBACF;gBACD,iBAAiB;aAClB,CAAC;YAEF,OAAO,CAAC,GAAG,CAAC,oBAAoB,EAAE,kBAAkB,CAAC,CAAC;YAEtD,MAAM,gBAAgB,GAAG,MAAM,IAAI,CAAC,iBAAiB,CAAC,eAAe,CACnE,SAAS,EACT,EAAE,CAAC,yBAAyB,IAAI,EAAE,CACnC,CAAC;YAEF,MAAM,wBAAwB,GAC5B,MAAM,IAAI,CAAC,iBAAiB,CAAC,eAAe,CAC1C,SAAS,EACT,EAAE,CAAC,iCAAiC,IAAI,EAAE,CAC3C,CAAC;YAEJ,MAAM,8BAA8B,GAClC,MAAM,IAAI,CAAC,iBAAiB,CAAC,eAAe,CAC1C,SAAS,EACT,kBAAkB,CACnB,CAAC;YAEJ,OAAO;gBACL,eAAe,EAAE,aAAa,CAAC,EAAE;gBACjC,kBAAkB;gBAClB,0BAA0B;gBAC1B,gBAAgB;gBAChB,wBAAwB;gBACxB,8BAA8B;gBAC9B,UAAU,EAAE,UAAU,IAAI,KAAK,CAAC;gBAChC,IAAI,EAAE,UAAU,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,IAAI;aACpC,CAAC;QACJ,CAAC;KAAA;IAED,iBAAiB,CAAC,SAAmB;QACnC,yCAAyC;QACzC,KAAK,MAAM,EAAE,IAAI,SAAS,EAAE;YAC1B,IAAI,CAAC,EAAE,CAAC,oBAAoB,CAAC,aAAa,CAAC,WAAW,EAAE;gBACtD,MAAM,GAAG,GAAG,MAAM,EAAE,CAAC,KAAK,CAAC,QAAQ,4BAA4B,CAAC;gBAChE,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;gBACjB,MAAM,IAAI,sBAAsB,CAAC,GAAG,CAAC,CAAC;aACvC;SACF;IACH,CAAC;IAED,iBAAiB;IACD,aAAa,CAC3B,aAAa,EACb,cAAsB,EACtB,cAAsB;;YAEtB,gDAAgD;YAChD,IAAI,aAAa,CAAC,MAAM,GAAG,cAAc,EAAE;gBACzC,MAAM,IAAI,sBAAsB,CAC9B,0CAA0C,CAC3C,CAAC;aACH;YAED,MAAM,YAAY,GAAG,IAAI,IAAI,CAAC,QAAQ,CAAC,cAAc,CAAC,CAAC;YAEvD,aAAa,CAAC,OAAO,CAAC,CAAC,EAAE,EAAE,KAAK,EAAE,EAAE;gBAClC,IAAI,aAAa,GAAG,EAAE,CAAC,WAAW,CAAC,MAAM,CAAC;gBAE1C,gFAAgF;gBAChF,4CAA4C;gBAC5C,IAAI,EAAE,CAAC,MAAM,KAAK,CAAC,EAAE;oBACnB,aAAa,GAAG,CAAC,CAAC;iBACnB;gBACD,YAAY,CAAC,cAAc,CACzB,IAAI,IAAI,CAAC,WAAW,CAAC,KAAK,EAAE,EAAE,CAAC,MAAM,EAAE,aAAa,CAAC,CACtD,CAAC;YACJ,CAAC,CAAC,CAAC;YAEH,MAAM,IAAI,CAAC,aAAa,CAAC,cAAc,CACrC,cAAc,EACd,IAAI,CAAC,gBAAgB,EACrB,YAAY,CACb,CAAC;YACF,OAAO,YAAY,CAAC;QACtB,CAAC;KAAA;CACF","sourcesContent":["import { Injector } from '@angular/core';\nimport { JWK } from 'node-jose';\nimport {\n  Connection,\n  mapEdges,\n  TpAssemblyNode,\n  TpNode,\n  TpSubAssemblyNode,\n} from '../api/types';\nimport { EncryptionService } from '../cryptography/encryption.service';\nimport { KeyFactoryService } from '../cryptography/key-factory.service';\nimport { KeyGraphService } from '../cryptography/key-graph.service';\nimport * as slip from '../cryptography/slip39.service';\nimport { Key } from '../cryptography/cryptography.types';\nimport {\n  LrBadArgumentException,\n  LrBadStateException,\n} from '../_common/exceptions';\nimport { KeyService } from '../cryptography/key.service';\nimport {\n  CreateTpAssemblyInput,\n  CreateTpSubAssemblyInput,\n  PartialAssemblyKey,\n  TpAssemblyApproverInput,\n  UpdateTpAssemblyInput,\n  UpdateTpSubAssemblyInput,\n} from './tp-assembly.types';\nimport { LrGraphQLService } from '../api/lr-graphql';\nimport { TpsKeysQuery } from './tp-assembly.gql.private';\n\nexport abstract class TpAssemblyController {\n  abstract slip39Passphrase: string;\n\n  protected keyFactory: KeyFactoryService;\n  protected keyService: KeyService;\n  protected encryptionService: EncryptionService;\n  protected keyGraph: KeyGraphService;\n  protected slip39Service: slip.Slip39Service;\n  protected lrGraphQl: LrGraphQLService;\n\n  constructor(injector: Injector) {\n    this.keyFactory = injector.get(KeyFactoryService);\n    this.keyService = injector.get(KeyService);\n    this.encryptionService = injector.get(EncryptionService);\n    this.keyGraph = injector.get(KeyGraphService);\n    this.slip39Service = injector.get(slip.Slip39Service);\n    this.lrGraphQl = injector.get(LrGraphQLService);\n  }\n\n  abstract getTpWrappingKeyId(tp: TpNode): string;\n\n  async recoverAssemblyKey(partials: PartialAssemblyKey[]) {\n    // Every receiverPbkCipher should contain this and they should all be the same.\n    let assemblyKeyParams: object;\n\n    const shares = partials.map((partial) => {\n      if (assemblyKeyParams) {\n        if (\n          JSON.stringify(assemblyKeyParams) !==\n          JSON.stringify(partial.assemblyKeyParams)\n        ) {\n          throw new LrBadStateException(\n            'The assembly key parameters are different between the approvals.'\n          );\n        }\n      } else {\n        assemblyKeyParams = partial.assemblyKeyParams;\n      }\n      return partial.slip39.share.mnemonics;\n    });\n\n    const rawAssemblyKey = await this.slip39Service.recoverSecret(\n      shares,\n      this.slip39Passphrase\n    );\n\n    return await KeyFactoryService.asKey({\n      ...assemblyKeyParams,\n      k: rawAssemblyKey,\n    });\n  }\n\n  protected async prepareTpWrappingKey(\n    tp: TpAssemblyApproverInput\n  ): Promise<Key> {\n    return this.keyGraph.getKey(tp.wrappingKeyId);\n  }\n\n  protected async prepareAssembly({\n    subjectKey,\n    rootKey,\n  }: {\n    subjectKey: JWK.Key;\n    rootKey: JWK.Key;\n  }) {\n    const assemblyKey = await this.keyFactory.createKey();\n    console.log('prepareAssembly assemblyKey', assemblyKey);\n\n    const { k: rawAssemblyKey, ...assemblyKeyParams } = assemblyKey.toJSON(\n      true\n    ) as any;\n    const assemblyKeyVerifierPrk = await this.keyFactory.createPkcSignKey();\n    const wrappedAssemblyKeyVerifierPrk =\n      await this.encryptionService.encryptToString(\n        assemblyKey,\n        assemblyKeyVerifierPrk.toJSON(true)\n      );\n    const assemblyKeyVerifierPbk = JSON.stringify(\n      assemblyKeyVerifierPrk.toJSON()\n    );\n\n    const subjectKeyWrappedAssemblyKey =\n      await this.encryptionService.encryptToString(\n        subjectKey,\n        assemblyKey.toJSON(true)\n      );\n\n    // Encrypt the rootKey with the assemblyKey\n    const assemblyCipherData = await this.encryptionService.encryptToString(\n      assemblyKey,\n      {\n        rootKey: rootKey.toJSON(true),\n      }\n    );\n\n    return {\n      assemblyKey,\n      rawAssemblyKey,\n      assemblyKeyParams,\n      subjectKeyWrappedAssemblyKey,\n      assemblyCipherData,\n      assemblyKeyVerifierPbk,\n      wrappedAssemblyKeyVerifierPrk,\n    };\n  }\n\n  protected async prepareCreateSubAssemblies({\n    input,\n    subjectKey,\n    slipSubAssemblies,\n    assemblyKeyParams,\n  }: {\n    input: CreateTpSubAssemblyInput[];\n    subjectKey: JWK.Key;\n    slipSubAssemblies: slip.SubAssembly[];\n    assemblyKeyParams; // TODO type this\n  }) {\n    return Promise.all(\n      input.map(async (sa, saIndex) => {\n        const subjectCipherData = await this.encryptionService.encryptToString(\n          subjectKey,\n          sa.subjectCipherDataClearJson || ''\n        );\n\n        const createApprovers = await Promise.all(\n          sa.approverTps.map(async (approverTp, approverIndex) =>\n            this.prepareApprover({\n              tp: approverTp,\n              approverIndex,\n              slipSubAssembly: slipSubAssemblies[saIndex],\n              assemblyKeyParams,\n              subjectKey,\n            })\n          )\n        );\n\n        return {\n          singleReject: sa.singleReject,\n          quorum: sa.quorum,\n          subjectCipherData,\n          createApprovers,\n        };\n      })\n    );\n  }\n\n  protected async prepareUpdateSubAssemblies({\n    input,\n    subjectKey,\n    slipSubAssemblies,\n    assemblyKeyParams,\n    subAssemblies,\n  }: {\n    input: UpdateTpSubAssemblyInput[];\n    subjectKey: JWK.Key;\n    slipSubAssemblies: slip.SubAssembly[];\n    assemblyKeyParams; // TODO type this\n    subAssemblies: Connection<TpSubAssemblyNode>;\n  }) {\n    if (slipSubAssemblies.length !== input.length) {\n      throw new LrBadArgumentException(\n        'The slipSubAssemblies must be the same length as the input'\n      );\n    }\n\n    return Promise.all(\n      input.map(async (sa, saIndex) => {\n        const subjectCipherData = await this.encryptionService.encryptToString(\n          subjectKey,\n          sa.subjectCipherDataClearJson || ''\n        );\n\n        // Get the existing sub-assembly\n        const existingSa = subAssemblies.edges.find(\n          (edge) => edge.node.id === sa.id\n        ).node;\n\n        // Get approvers that do not exist yet\n        const createApprovers = [];\n        const updateApprovers = [];\n\n        sa.approverTps.forEach((tp) => {\n          const approver = existingSa.approvers.edges.find(\n            (edge) => edge.node.tp.id === tp.tpId\n          )?.node;\n          if (approver) {\n            updateApprovers.push({\n              tp,\n              approverId: approver.id,\n            });\n          } else {\n            createApprovers.push({\n              tp,\n            });\n          }\n        });\n\n        const slipSubAssembly = slipSubAssemblies[saIndex];\n\n        return {\n          subAssemblyId: sa.id,\n          singleReject: sa.singleReject,\n          quorum: sa.quorum,\n          subjectCipherData,\n          createApprovers: await Promise.all(\n            createApprovers.map(async ({ tp }, approverIndex) =>\n              this.prepareApprover({\n                tp,\n                approverIndex,\n                slipSubAssembly,\n                assemblyKeyParams,\n                subjectKey,\n              })\n            )\n          ),\n          updateApprovers: await Promise.all(\n            updateApprovers.map(async ({ tp, approverId }, approverIndex) =>\n              this.prepareApprover({\n                approverId,\n                tp,\n                approverIndex: approverIndex + createApprovers.length,\n                slipSubAssembly,\n                assemblyKeyParams,\n                subjectKey,\n              })\n            )\n          ),\n        };\n      })\n    );\n  }\n\n  private fillWrappingKeyId(input: CreateTpSubAssemblyInput[], tps: TpNode[]) {\n    // Fill in wrappingKeyId if not present\n    input.forEach((sa) => {\n      sa.approverTps.forEach((approver) => {\n        if (!approver.wrappingKeyId) {\n          const tp = tps.find((x) => x.id === approver.tpId);\n          approver.wrappingKeyId = this.getTpWrappingKeyId(tp);\n        }\n      });\n    });\n  }\n\n  private getInputApprovers(input: CreateTpSubAssemblyInput[]) {\n    const tpIds: string[] = [];\n    input.forEach((sa) => {\n      sa.approverTps.forEach((approver) => {\n        tpIds.push(approver.tpId);\n      });\n    });\n    return tpIds;\n  }\n\n  public async prepareCreate(input: CreateTpAssemblyInput) {\n    const rootKey = await this.keyService.getCurrentRootKey();\n    const subjectKey = await this.keyFactory.createKey();\n    const rootKeyWrappedSubjectKey = await this.keyGraph.wrapKey(\n      rootKey,\n      subjectKey\n    );\n\n    const { assemblyKey, rawAssemblyKey, assemblyKeyParams, ...assemblyInput } =\n      await this.prepareAssembly({\n        rootKey: rootKey.jwk,\n        subjectKey,\n      });\n\n    const slipAssembly = await this.prepareSlip39(\n      input.createSubAssemblies,\n      input.quorum,\n      rawAssemblyKey\n    );\n\n    const tpIds = this.getInputApprovers(input.createSubAssemblies);\n\n    // This should contain all the TPs that we need to update the assembly.\n    const tps = mapEdges(\n      (\n        await this.lrGraphQl.query({\n          query: TpsKeysQuery,\n          variables: {\n            ids: tpIds,\n          },\n        })\n      ).tps\n    );\n\n    // Fill in wrappingKeyId if not present\n    this.fillWrappingKeyId(input.createSubAssemblies, tps);\n\n    const createSubAssemblies = await this.prepareCreateSubAssemblies({\n      input: input.createSubAssemblies,\n      assemblyKeyParams,\n      slipSubAssemblies: slipAssembly.subAssemblies,\n      subjectKey,\n    });\n\n    return {\n      assemblyKey,\n      mutationInput: {\n        ...assemblyInput,\n        singleReject: input.singleReject,\n        quorum: input.quorum,\n        subjectCipherData: '', // TODO make this a parameter\n        createSubAssemblies,\n        rootKeyId: rootKey.id,\n        rootKeyWrappedSubjectKey,\n      },\n    };\n  }\n\n  private getAssemblyApprovers(assembly: TpAssemblyNode) {\n    const tpIds: string[] = [];\n\n    // Find all TPs that belong to sub-assemblies\n    mapEdges(assembly.subAssemblies).map((sa) => {\n      mapEdges(sa.approvers).forEach((approver) => tpIds.push(approver.tp.id));\n    });\n\n    return tpIds;\n  }\n\n  public async prepareUpdate(\n    input: UpdateTpAssemblyInput,\n    assembly: TpAssemblyNode // TODO type this to only what we need.\n  ) {\n    input = {\n      createSubAssemblies: [],\n      ...input,\n    };\n\n    const createSubAssembliesInput = input.createSubAssemblies || [];\n    const updateSubAssembliesInput = input.updateSubAssemblies || [];\n    const deleteSubAssembliesInput = input.deleteSubAssemblies || [];\n\n    // Must have at least one.\n    if (\n      createSubAssembliesInput.length === 0 &&\n      updateSubAssembliesInput.length === 0 &&\n      deleteSubAssembliesInput.length === 0\n    ) {\n      throw new LrBadArgumentException(\n        'Must specify at least one of: [createSubAssemblies, updateSubAssemblies, deleteSubAssemblies]'\n      );\n    }\n\n    const rootKey = await this.keyService.getCurrentRootKey();\n    const subjectKey = await this.keyGraph.getKey(assembly.subjectKey.id);\n\n    const { assemblyKey, rawAssemblyKey, assemblyKeyParams, ...assemblyInput } =\n      await this.prepareAssembly({\n        rootKey: rootKey.jwk,\n        subjectKey: subjectKey.jwk,\n      });\n\n    // Find all the TPs that we need, duplicates are no problem.\n    // Existing approvers\n    const tpIds = this.getAssemblyApprovers(assembly).concat(\n      this.getInputApprovers(createSubAssembliesInput),\n      this.getInputApprovers(updateSubAssembliesInput)\n    );\n\n    // This should contain all the TPs that we need to update the assembly.\n    const tps = mapEdges(\n      (\n        await this.lrGraphQl.query({\n          query: TpsKeysQuery,\n          variables: {\n            ids: tpIds,\n          },\n        })\n      ).tps\n    );\n\n    // Auto fill the updateSubAssemblies with existing sub assemblies if they are not\n    // explicitly deleted.\n    mapEdges(assembly.subAssemblies).forEach((existing) => {\n      // Deleting existing, don't include it\n      if (deleteSubAssembliesInput.includes(existing.id)) {\n        return;\n      }\n\n      // Update already specified in the input\n      if (updateSubAssembliesInput.some((sa) => sa.id === existing.id)) {\n        return;\n      }\n\n      const approverTps: TpAssemblyApproverInput[] = mapEdges(\n        existing.approvers\n      ).map((approver) => {\n        return {\n          tpId: approver.tp.id,\n          sharedCipherDataClearJson: approver.sharedCipherDataClearJson,\n          sharedCipherApprovalDataClearJson:\n            approver.sharedCipherApprovalDataClearJson,\n        };\n      });\n\n      updateSubAssembliesInput.push({\n        id: existing.id,\n        quorum: existing.quorum,\n        singleReject: existing.singleReject,\n        approverTps,\n        subjectCipherDataClearJson: existing.subjectCipherDataClearJson,\n      });\n    });\n\n    // Fill in wrappingKeyId if not present\n    this.fillWrappingKeyId(createSubAssembliesInput, tps);\n    this.fillWrappingKeyId(updateSubAssembliesInput, tps);\n\n    let slipAssembly;\n    try {\n      slipAssembly = await this.prepareSlip39(\n        (createSubAssembliesInput || []).concat(updateSubAssembliesInput || []),\n        input.quorum,\n        rawAssemblyKey\n      );\n    } catch (error) {\n      // TODO this sometimes happens. Maybe missing await.\n      console.log('Error while creating slip assembly', error);\n    }\n\n    const common = {\n      assemblyKeyParams,\n      subjectKey: subjectKey.jwk,\n    };\n\n    const createSubAssemblies = await this.prepareCreateSubAssemblies({\n      ...common,\n      input: createSubAssembliesInput,\n      // Need to match the slip sub assemblies to the input\n      slipSubAssemblies: slipAssembly.subAssemblies.slice(\n        0,\n        createSubAssembliesInput.length\n      ),\n    });\n\n    const updateSubAssemblies = await this.prepareUpdateSubAssemblies({\n      ...common,\n      input: updateSubAssembliesInput,\n      // Add an offset to the sub assemblies to skip the one that are created.\n      slipSubAssemblies: slipAssembly.subAssemblies.slice(\n        createSubAssembliesInput.length\n      ),\n      subAssemblies: assembly.subAssemblies,\n    });\n\n    return {\n      assemblyKey,\n      mutationInput: {\n        ...assemblyInput,\n        singleReject: input.singleReject,\n        quorum: input.quorum,\n        subjectCipherData: '', // TODO make this a parameter\n        subjectKeyId: subjectKey.id,\n        createSubAssemblies,\n        updateSubAssemblies,\n      },\n    };\n  }\n\n  private async prepareApprover({\n    approverId,\n    tp,\n    approverIndex,\n    slipSubAssembly,\n    assemblyKeyParams,\n    subjectKey,\n  }: {\n    approverId?: string; // Given if updating existing approver\n    tp: TpAssemblyApproverInput;\n    approverIndex: number;\n    slipSubAssembly: slip.SubAssembly;\n    assemblyKeyParams: object;\n    subjectKey: JWK.Key;\n  }) {\n    const tpWrappingKey = await this.prepareTpWrappingKey(tp);\n    // shared key is rotated every time.\n    const sharedKey = await this.keyFactory.createKey();\n\n    // For TP to access shared_key\n    const tpWrappedSharedKey = await this.encryptionService.encryptToString(\n      tpWrappingKey.jwk,\n      sharedKey.toJSON(true)\n    );\n    // For subject to access shared_key\n    const subjectKeyWrappedSharedKey =\n      await this.encryptionService.encryptToString(\n        subjectKey,\n        sharedKey.toJSON(true)\n      );\n\n    // If quorum is 1, then using the same share for every member.\n    const share =\n      slipSubAssembly.threshold === 1\n        ? slipSubAssembly.shares[0]\n        : slipSubAssembly.shares[approverIndex];\n\n    const partialAssemblyKey: PartialAssemblyKey = {\n      slip39: {\n        share,\n        subAssembly: {\n          quorum: slipSubAssembly.threshold,\n          size: slipSubAssembly.size,\n        },\n      },\n      assemblyKeyParams,\n    };\n\n    console.log('partialAssemblyKey', partialAssemblyKey);\n\n    const sharedCipherData = await this.encryptionService.encryptToString(\n      sharedKey,\n      tp.sharedCipherDataClearJson || ''\n    );\n\n    const sharedCipherApprovalData =\n      await this.encryptionService.encryptToString(\n        sharedKey,\n        tp.sharedCipherApprovalDataClearJson || ''\n      );\n\n    const sharedCipherPartialAssemblyKey =\n      await this.encryptionService.encryptToString(\n        sharedKey,\n        partialAssemblyKey\n      );\n\n    return {\n      tpWrappingKeyId: tpWrappingKey.id,\n      tpWrappedSharedKey,\n      subjectKeyWrappedSharedKey,\n      sharedCipherData,\n      sharedCipherApprovalData,\n      sharedCipherPartialAssemblyKey,\n      approverId: approverId || void 0, // If existing approver\n      tpId: approverId ? void 0 : tp.tpId, // else adding new TP\n    };\n  }\n\n  validateApprovers(approvers: TpNode[]): void {\n    // Ensure all approvers have mkSharedKey.\n    for (const tp of approvers) {\n      if (!tp.currentUserSharedKey.userSharedKey.mkSharedKey) {\n        const msg = `tp ${tp.other.username} does not have mkSharedKey`;\n        console.log(msg);\n        throw new LrBadArgumentException(msg);\n      }\n    }\n  }\n\n  // Prepare slip39\n  protected async prepareSlip39(\n    subAssemblies,\n    assemblyQuorum: number,\n    rawAssemblyKey: string\n  ): Promise<slip.Assembly> {\n    // Is there enough sub assemblies to meet quorum\n    if (subAssemblies.length < assemblyQuorum) {\n      throw new LrBadArgumentException(\n        'Not enough sub assemblies to meet quorum'\n      );\n    }\n\n    const slipAssembly = new slip.Assembly(assemblyQuorum);\n\n    subAssemblies.forEach((sa, index) => {\n      let approverCount = sa.approverTps.length;\n\n      // slip39 restricts quorum == 1 to have only 1 member. So we just share the same\n      // partial key for all sub assembly members.\n      if (sa.quorum === 1) {\n        approverCount = 1;\n      }\n      slipAssembly.addSubAssembly(\n        new slip.SubAssembly(index, sa.quorum, approverCount)\n      );\n    });\n\n    await this.slip39Service.generateShares(\n      rawAssemblyKey,\n      this.slip39Passphrase,\n      slipAssembly\n    );\n    return slipAssembly;\n  }\n}\n"]}
|
|
1
|
+
import { __awaiter, __rest } from "tslib";
|
|
2
|
+
import { mapEdges, } from '../api/types';
|
|
3
|
+
import { EncryptionService } from '../cryptography/encryption.service';
|
|
4
|
+
import { KeyFactoryService } from '../cryptography/key-factory.service';
|
|
5
|
+
import { KeyGraphService } from '../cryptography/key-graph.service';
|
|
6
|
+
import * as slip from '../cryptography/slip39.service';
|
|
7
|
+
import { LrBadArgumentException, LrBadStateException, } from '../_common/exceptions';
|
|
8
|
+
import { KeyService } from '../cryptography/key.service';
|
|
9
|
+
import { LrGraphQLService } from '../api/lr-graphql';
|
|
10
|
+
import { TpsKeysQuery } from './tp-assembly.gql.private';
|
|
11
|
+
export class TpAssemblyController {
|
|
12
|
+
constructor(injector) {
|
|
13
|
+
this.keyFactory = injector.get(KeyFactoryService);
|
|
14
|
+
this.keyService = injector.get(KeyService);
|
|
15
|
+
this.encryptionService = injector.get(EncryptionService);
|
|
16
|
+
this.keyGraph = injector.get(KeyGraphService);
|
|
17
|
+
this.slip39Service = injector.get(slip.Slip39Service);
|
|
18
|
+
this.lrGraphQl = injector.get(LrGraphQLService);
|
|
19
|
+
}
|
|
20
|
+
recoverAssemblyKey(partials) {
|
|
21
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
22
|
+
// Every receiverPbkCipher should contain this and they should all be the same.
|
|
23
|
+
let assemblyKeyParams;
|
|
24
|
+
const shares = partials.map((partial) => {
|
|
25
|
+
if (assemblyKeyParams) {
|
|
26
|
+
if (JSON.stringify(assemblyKeyParams) !==
|
|
27
|
+
JSON.stringify(partial.assemblyKeyParams)) {
|
|
28
|
+
throw new LrBadStateException('The assembly key parameters are different between the approvals.');
|
|
29
|
+
}
|
|
30
|
+
}
|
|
31
|
+
else {
|
|
32
|
+
assemblyKeyParams = partial.assemblyKeyParams;
|
|
33
|
+
}
|
|
34
|
+
return partial.slip39.share.mnemonics;
|
|
35
|
+
});
|
|
36
|
+
const rawAssemblyKey = yield this.slip39Service.recoverSecret(shares, this.slip39Passphrase);
|
|
37
|
+
return yield KeyFactoryService.asKey(Object.assign(Object.assign({}, assemblyKeyParams), { k: rawAssemblyKey }));
|
|
38
|
+
});
|
|
39
|
+
}
|
|
40
|
+
prepareTpWrappingKey(tp) {
|
|
41
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
42
|
+
return this.keyGraph.getKey(tp.wrappingKeyId);
|
|
43
|
+
});
|
|
44
|
+
}
|
|
45
|
+
prepareAssembly({ subjectKey, rootKey, }) {
|
|
46
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
47
|
+
const assemblyKey = yield this.keyFactory.createKey();
|
|
48
|
+
console.log('prepareAssembly assemblyKey', assemblyKey);
|
|
49
|
+
const _a = assemblyKey.toJSON(true), { k: rawAssemblyKey } = _a, assemblyKeyParams = __rest(_a, ["k"]);
|
|
50
|
+
const assemblyKeyVerifierPrk = yield this.keyFactory.createPkcSignKey();
|
|
51
|
+
const wrappedAssemblyKeyVerifierPrk = yield this.encryptionService.encryptToString(assemblyKey, assemblyKeyVerifierPrk.toJSON(true));
|
|
52
|
+
const assemblyKeyVerifierPbk = JSON.stringify(assemblyKeyVerifierPrk.toJSON());
|
|
53
|
+
const subjectKeyWrappedAssemblyKey = yield this.encryptionService.encryptToString(subjectKey, assemblyKey.toJSON(true));
|
|
54
|
+
// Encrypt the rootKey with the assemblyKey
|
|
55
|
+
const assemblyCipherData = yield this.encryptionService.encryptToString(assemblyKey, {
|
|
56
|
+
rootKey: rootKey.toJSON(true),
|
|
57
|
+
});
|
|
58
|
+
return {
|
|
59
|
+
assemblyKey,
|
|
60
|
+
rawAssemblyKey,
|
|
61
|
+
assemblyKeyParams,
|
|
62
|
+
subjectKeyWrappedAssemblyKey,
|
|
63
|
+
assemblyCipherData,
|
|
64
|
+
assemblyKeyVerifierPbk,
|
|
65
|
+
wrappedAssemblyKeyVerifierPrk,
|
|
66
|
+
};
|
|
67
|
+
});
|
|
68
|
+
}
|
|
69
|
+
prepareCreateSubAssemblies({ input, subjectKey, slipSubAssemblies, assemblyKeyParams, }) {
|
|
70
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
71
|
+
return Promise.all(input.map((sa, saIndex) => __awaiter(this, void 0, void 0, function* () {
|
|
72
|
+
const subjectCipherData = yield this.encryptionService.encryptToString(subjectKey, sa.subjectCipherDataClearJson || '');
|
|
73
|
+
const createApprovers = yield Promise.all(sa.approverTps.map((approverTp, approverIndex) => __awaiter(this, void 0, void 0, function* () {
|
|
74
|
+
return this.prepareApprover({
|
|
75
|
+
tp: approverTp,
|
|
76
|
+
approverIndex,
|
|
77
|
+
slipSubAssembly: slipSubAssemblies[saIndex],
|
|
78
|
+
assemblyKeyParams,
|
|
79
|
+
subjectKey,
|
|
80
|
+
});
|
|
81
|
+
})));
|
|
82
|
+
return {
|
|
83
|
+
singleReject: sa.singleReject,
|
|
84
|
+
quorum: sa.quorum,
|
|
85
|
+
subjectCipherData,
|
|
86
|
+
createApprovers,
|
|
87
|
+
};
|
|
88
|
+
})));
|
|
89
|
+
});
|
|
90
|
+
}
|
|
91
|
+
prepareUpdateSubAssemblies({ input, subjectKey, slipSubAssemblies, assemblyKeyParams, subAssemblies, }) {
|
|
92
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
93
|
+
if (slipSubAssemblies.length !== input.length) {
|
|
94
|
+
throw new LrBadArgumentException('The slipSubAssemblies must be the same length as the input');
|
|
95
|
+
}
|
|
96
|
+
return Promise.all(input.map((sa, saIndex) => __awaiter(this, void 0, void 0, function* () {
|
|
97
|
+
const subjectCipherData = yield this.encryptionService.encryptToString(subjectKey, sa.subjectCipherDataClearJson || '');
|
|
98
|
+
// Get the existing sub-assembly
|
|
99
|
+
const existingSa = subAssemblies.edges.find((edge) => edge.node.id === sa.id).node;
|
|
100
|
+
// Get approvers that do not exist yet
|
|
101
|
+
const createApprovers = [];
|
|
102
|
+
const updateApprovers = [];
|
|
103
|
+
sa.approverTps.forEach((tp) => {
|
|
104
|
+
var _a;
|
|
105
|
+
const approver = (_a = existingSa.approvers.edges.find((edge) => edge.node.tp.id === tp.tpId)) === null || _a === void 0 ? void 0 : _a.node;
|
|
106
|
+
if (approver) {
|
|
107
|
+
updateApprovers.push({
|
|
108
|
+
tp,
|
|
109
|
+
approverId: approver.id,
|
|
110
|
+
});
|
|
111
|
+
}
|
|
112
|
+
else {
|
|
113
|
+
createApprovers.push({
|
|
114
|
+
tp,
|
|
115
|
+
});
|
|
116
|
+
}
|
|
117
|
+
});
|
|
118
|
+
const slipSubAssembly = slipSubAssemblies[saIndex];
|
|
119
|
+
return {
|
|
120
|
+
subAssemblyId: sa.id,
|
|
121
|
+
singleReject: sa.singleReject,
|
|
122
|
+
quorum: sa.quorum,
|
|
123
|
+
subjectCipherData,
|
|
124
|
+
createApprovers: yield Promise.all(createApprovers.map(({ tp }, approverIndex) => __awaiter(this, void 0, void 0, function* () {
|
|
125
|
+
return this.prepareApprover({
|
|
126
|
+
tp,
|
|
127
|
+
approverIndex,
|
|
128
|
+
slipSubAssembly,
|
|
129
|
+
assemblyKeyParams,
|
|
130
|
+
subjectKey,
|
|
131
|
+
});
|
|
132
|
+
}))),
|
|
133
|
+
updateApprovers: yield Promise.all(updateApprovers.map(({ tp, approverId }, approverIndex) => __awaiter(this, void 0, void 0, function* () {
|
|
134
|
+
return this.prepareApprover({
|
|
135
|
+
approverId,
|
|
136
|
+
tp,
|
|
137
|
+
approverIndex: approverIndex + createApprovers.length,
|
|
138
|
+
slipSubAssembly,
|
|
139
|
+
assemblyKeyParams,
|
|
140
|
+
subjectKey,
|
|
141
|
+
});
|
|
142
|
+
}))),
|
|
143
|
+
};
|
|
144
|
+
})));
|
|
145
|
+
});
|
|
146
|
+
}
|
|
147
|
+
fillWrappingKeyId(input, tps) {
|
|
148
|
+
// Fill in wrappingKeyId if not present
|
|
149
|
+
input.forEach((sa) => {
|
|
150
|
+
sa.approverTps.forEach((approver) => {
|
|
151
|
+
if (!approver.wrappingKeyId) {
|
|
152
|
+
const tp = tps.find((x) => x.id === approver.tpId);
|
|
153
|
+
approver.wrappingKeyId = this.getTpWrappingKeyId(tp);
|
|
154
|
+
}
|
|
155
|
+
});
|
|
156
|
+
});
|
|
157
|
+
}
|
|
158
|
+
getInputApprovers(input) {
|
|
159
|
+
const tpIds = [];
|
|
160
|
+
input.forEach((sa) => {
|
|
161
|
+
sa.approverTps.forEach((approver) => {
|
|
162
|
+
tpIds.push(approver.tpId);
|
|
163
|
+
});
|
|
164
|
+
});
|
|
165
|
+
return tpIds;
|
|
166
|
+
}
|
|
167
|
+
prepareCreate(input) {
|
|
168
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
169
|
+
const rootKey = yield this.keyService.getCurrentRootKey();
|
|
170
|
+
const subjectKey = yield this.keyFactory.createKey();
|
|
171
|
+
const rootKeyWrappedSubjectKey = yield this.keyGraph.wrapKey(rootKey, subjectKey);
|
|
172
|
+
const _a = yield this.prepareAssembly({
|
|
173
|
+
rootKey: rootKey.jwk,
|
|
174
|
+
subjectKey,
|
|
175
|
+
}), { assemblyKey, rawAssemblyKey, assemblyKeyParams } = _a, assemblyInput = __rest(_a, ["assemblyKey", "rawAssemblyKey", "assemblyKeyParams"]);
|
|
176
|
+
const slipAssembly = yield this.prepareSlip39(input.createSubAssemblies, input.quorum, rawAssemblyKey);
|
|
177
|
+
const tpIds = this.getInputApprovers(input.createSubAssemblies);
|
|
178
|
+
// This should contain all the TPs that we need to update the assembly.
|
|
179
|
+
const tps = mapEdges((yield this.lrGraphQl.query({
|
|
180
|
+
query: TpsKeysQuery,
|
|
181
|
+
variables: {
|
|
182
|
+
ids: tpIds,
|
|
183
|
+
},
|
|
184
|
+
})).tps);
|
|
185
|
+
// Fill in wrappingKeyId if not present
|
|
186
|
+
this.fillWrappingKeyId(input.createSubAssemblies, tps);
|
|
187
|
+
const createSubAssemblies = yield this.prepareCreateSubAssemblies({
|
|
188
|
+
input: input.createSubAssemblies,
|
|
189
|
+
assemblyKeyParams,
|
|
190
|
+
slipSubAssemblies: slipAssembly.subAssemblies,
|
|
191
|
+
subjectKey,
|
|
192
|
+
});
|
|
193
|
+
return {
|
|
194
|
+
assemblyKey,
|
|
195
|
+
mutationInput: Object.assign(Object.assign({}, assemblyInput), { singleReject: input.singleReject, quorum: input.quorum, subjectCipherData: '', // TODO make this a parameter
|
|
196
|
+
createSubAssemblies, rootKeyId: rootKey.id, rootKeyWrappedSubjectKey }),
|
|
197
|
+
};
|
|
198
|
+
});
|
|
199
|
+
}
|
|
200
|
+
getAssemblyApprovers(assembly) {
|
|
201
|
+
const tpIds = [];
|
|
202
|
+
// Find all TPs that belong to sub-assemblies
|
|
203
|
+
mapEdges(assembly.subAssemblies).map((sa) => {
|
|
204
|
+
mapEdges(sa.approvers).forEach((approver) => tpIds.push(approver.tp.id));
|
|
205
|
+
});
|
|
206
|
+
return tpIds;
|
|
207
|
+
}
|
|
208
|
+
prepareUpdate(input, assembly // TODO type this to only what we need.
|
|
209
|
+
) {
|
|
210
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
211
|
+
input = Object.assign({ createSubAssemblies: [] }, input);
|
|
212
|
+
const createSubAssembliesInput = input.createSubAssemblies || [];
|
|
213
|
+
const updateSubAssembliesInput = input.updateSubAssemblies || [];
|
|
214
|
+
const deleteSubAssembliesInput = input.deleteSubAssemblies || [];
|
|
215
|
+
// Must have at least one.
|
|
216
|
+
if (createSubAssembliesInput.length === 0 &&
|
|
217
|
+
updateSubAssembliesInput.length === 0 &&
|
|
218
|
+
deleteSubAssembliesInput.length === 0) {
|
|
219
|
+
throw new LrBadArgumentException('Must specify at least one of: [createSubAssemblies, updateSubAssemblies, deleteSubAssemblies]');
|
|
220
|
+
}
|
|
221
|
+
const rootKey = yield this.keyService.getCurrentRootKey();
|
|
222
|
+
const subjectKey = yield this.keyGraph.getKey(assembly.subjectKey.id);
|
|
223
|
+
const _a = yield this.prepareAssembly({
|
|
224
|
+
rootKey: rootKey.jwk,
|
|
225
|
+
subjectKey: subjectKey.jwk,
|
|
226
|
+
}), { assemblyKey, rawAssemblyKey, assemblyKeyParams } = _a, assemblyInput = __rest(_a, ["assemblyKey", "rawAssemblyKey", "assemblyKeyParams"]);
|
|
227
|
+
// Find all the TPs that we need, duplicates are no problem.
|
|
228
|
+
// Existing approvers
|
|
229
|
+
const tpIds = this.getAssemblyApprovers(assembly).concat(this.getInputApprovers(createSubAssembliesInput), this.getInputApprovers(updateSubAssembliesInput));
|
|
230
|
+
// This should contain all the TPs that we need to update the assembly.
|
|
231
|
+
const tps = mapEdges((yield this.lrGraphQl.query({
|
|
232
|
+
query: TpsKeysQuery,
|
|
233
|
+
variables: {
|
|
234
|
+
ids: tpIds,
|
|
235
|
+
},
|
|
236
|
+
})).tps);
|
|
237
|
+
// Auto fill the updateSubAssemblies with existing sub assemblies if they are not
|
|
238
|
+
// explicitly deleted.
|
|
239
|
+
mapEdges(assembly.subAssemblies).forEach((existing) => {
|
|
240
|
+
// Deleting existing, don't include it
|
|
241
|
+
if (deleteSubAssembliesInput.includes(existing.id)) {
|
|
242
|
+
return;
|
|
243
|
+
}
|
|
244
|
+
// Update already specified in the input
|
|
245
|
+
if (updateSubAssembliesInput.some((sa) => sa.id === existing.id)) {
|
|
246
|
+
return;
|
|
247
|
+
}
|
|
248
|
+
const approverTps = mapEdges(existing.approvers).map((approver) => {
|
|
249
|
+
return {
|
|
250
|
+
tpId: approver.tp.id,
|
|
251
|
+
sharedCipherDataClearJson: approver.sharedCipherDataClearJson,
|
|
252
|
+
sharedCipherApprovalDataClearJson: approver.sharedCipherApprovalDataClearJson,
|
|
253
|
+
};
|
|
254
|
+
});
|
|
255
|
+
updateSubAssembliesInput.push({
|
|
256
|
+
id: existing.id,
|
|
257
|
+
quorum: existing.quorum,
|
|
258
|
+
singleReject: existing.singleReject,
|
|
259
|
+
approverTps,
|
|
260
|
+
subjectCipherDataClearJson: existing.subjectCipherDataClearJson,
|
|
261
|
+
});
|
|
262
|
+
});
|
|
263
|
+
// Fill in wrappingKeyId if not present
|
|
264
|
+
this.fillWrappingKeyId(createSubAssembliesInput, tps);
|
|
265
|
+
this.fillWrappingKeyId(updateSubAssembliesInput, tps);
|
|
266
|
+
let slipAssembly;
|
|
267
|
+
try {
|
|
268
|
+
slipAssembly = yield this.prepareSlip39((createSubAssembliesInput || []).concat(updateSubAssembliesInput || []), input.quorum, rawAssemblyKey);
|
|
269
|
+
}
|
|
270
|
+
catch (error) {
|
|
271
|
+
// TODO this sometimes happens. Maybe missing await.
|
|
272
|
+
console.log('Error while creating slip assembly', error);
|
|
273
|
+
}
|
|
274
|
+
const common = {
|
|
275
|
+
assemblyKeyParams,
|
|
276
|
+
subjectKey: subjectKey.jwk,
|
|
277
|
+
};
|
|
278
|
+
const createSubAssemblies = yield this.prepareCreateSubAssemblies(Object.assign(Object.assign({}, common), { input: createSubAssembliesInput,
|
|
279
|
+
// Need to match the slip sub assemblies to the input
|
|
280
|
+
slipSubAssemblies: slipAssembly.subAssemblies.slice(0, createSubAssembliesInput.length) }));
|
|
281
|
+
const updateSubAssemblies = yield this.prepareUpdateSubAssemblies(Object.assign(Object.assign({}, common), { input: updateSubAssembliesInput,
|
|
282
|
+
// Add an offset to the sub assemblies to skip the one that are created.
|
|
283
|
+
slipSubAssemblies: slipAssembly.subAssemblies.slice(createSubAssembliesInput.length), subAssemblies: assembly.subAssemblies }));
|
|
284
|
+
return {
|
|
285
|
+
assemblyKey,
|
|
286
|
+
mutationInput: Object.assign(Object.assign({}, assemblyInput), { singleReject: input.singleReject, quorum: input.quorum, subjectCipherData: '', subjectKeyId: subjectKey.id, createSubAssemblies,
|
|
287
|
+
updateSubAssemblies }),
|
|
288
|
+
};
|
|
289
|
+
});
|
|
290
|
+
}
|
|
291
|
+
prepareApprover({ approverId, tp, approverIndex, slipSubAssembly, assemblyKeyParams, subjectKey, }) {
|
|
292
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
293
|
+
const tpWrappingKey = yield this.prepareTpWrappingKey(tp);
|
|
294
|
+
// shared key is rotated every time.
|
|
295
|
+
const sharedKey = yield this.keyFactory.createKey();
|
|
296
|
+
// For TP to access shared_key
|
|
297
|
+
const tpWrappedSharedKey = yield this.encryptionService.encryptToString(tpWrappingKey.jwk, sharedKey.toJSON(true));
|
|
298
|
+
// For subject to access shared_key
|
|
299
|
+
const subjectKeyWrappedSharedKey = yield this.encryptionService.encryptToString(subjectKey, sharedKey.toJSON(true));
|
|
300
|
+
// If quorum is 1, then using the same share for every member.
|
|
301
|
+
const share = slipSubAssembly.threshold === 1
|
|
302
|
+
? slipSubAssembly.shares[0]
|
|
303
|
+
: slipSubAssembly.shares[approverIndex];
|
|
304
|
+
const partialAssemblyKey = {
|
|
305
|
+
slip39: {
|
|
306
|
+
share,
|
|
307
|
+
subAssembly: {
|
|
308
|
+
quorum: slipSubAssembly.threshold,
|
|
309
|
+
size: slipSubAssembly.size,
|
|
310
|
+
},
|
|
311
|
+
},
|
|
312
|
+
assemblyKeyParams,
|
|
313
|
+
};
|
|
314
|
+
console.log('partialAssemblyKey', partialAssemblyKey);
|
|
315
|
+
const sharedCipherData = yield this.encryptionService.encryptToString(sharedKey, tp.sharedCipherDataClearJson || '');
|
|
316
|
+
const sharedCipherApprovalData = yield this.encryptionService.encryptToString(sharedKey, tp.sharedCipherApprovalDataClearJson || '');
|
|
317
|
+
const sharedCipherPartialAssemblyKey = yield this.encryptionService.encryptToString(sharedKey, partialAssemblyKey);
|
|
318
|
+
return {
|
|
319
|
+
tpWrappingKeyId: tpWrappingKey.id,
|
|
320
|
+
tpWrappedSharedKey,
|
|
321
|
+
subjectKeyWrappedSharedKey,
|
|
322
|
+
sharedCipherData,
|
|
323
|
+
sharedCipherApprovalData,
|
|
324
|
+
sharedCipherPartialAssemblyKey,
|
|
325
|
+
approverId: approverId || void 0,
|
|
326
|
+
tpId: approverId ? void 0 : tp.tpId,
|
|
327
|
+
};
|
|
328
|
+
});
|
|
329
|
+
}
|
|
330
|
+
validateApprovers(approvers) {
|
|
331
|
+
// Ensure all approvers have mkSharedKey.
|
|
332
|
+
for (const tp of approvers) {
|
|
333
|
+
if (!tp.currentUserSharedKey.userSharedKey.mkSharedKey) {
|
|
334
|
+
const msg = `tp ${tp.other.username} does not have mkSharedKey`;
|
|
335
|
+
console.log(msg);
|
|
336
|
+
throw new LrBadArgumentException(msg);
|
|
337
|
+
}
|
|
338
|
+
}
|
|
339
|
+
}
|
|
340
|
+
// Prepare slip39
|
|
341
|
+
prepareSlip39(subAssemblies, assemblyQuorum, rawAssemblyKey) {
|
|
342
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
343
|
+
// Is there enough sub assemblies to meet quorum
|
|
344
|
+
if (subAssemblies.length < assemblyQuorum) {
|
|
345
|
+
throw new LrBadArgumentException('Not enough sub assemblies to meet quorum');
|
|
346
|
+
}
|
|
347
|
+
const slipAssembly = new slip.Assembly(assemblyQuorum);
|
|
348
|
+
subAssemblies.forEach((sa, index) => {
|
|
349
|
+
let approverCount = sa.approverTps.length;
|
|
350
|
+
// slip39 restricts quorum == 1 to have only 1 member. So we just share the same
|
|
351
|
+
// partial key for all sub assembly members.
|
|
352
|
+
if (sa.quorum === 1) {
|
|
353
|
+
approverCount = 1;
|
|
354
|
+
}
|
|
355
|
+
slipAssembly.addSubAssembly(new slip.SubAssembly(index, sa.quorum, approverCount));
|
|
356
|
+
});
|
|
357
|
+
yield this.slip39Service.generateShares(rawAssemblyKey, this.slip39Passphrase, slipAssembly);
|
|
358
|
+
return slipAssembly;
|
|
359
|
+
});
|
|
360
|
+
}
|
|
361
|
+
}
|
|
362
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tp-assembly.js","sourceRoot":"C:/Projects/kc-client/projects/core/src/","sources":["lib/trusted-parties/tp-assembly.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injector } from '@angular/core';\r\nimport { JWK } from 'node-jose';\r\nimport {\r\n  Connection,\r\n  mapEdges,\r\n  TpAssemblyNode,\r\n  TpNode,\r\n  TpSubAssemblyNode,\r\n} from '../api/types';\r\nimport { EncryptionService } from '../cryptography/encryption.service';\r\nimport { KeyFactoryService } from '../cryptography/key-factory.service';\r\nimport { KeyGraphService } from '../cryptography/key-graph.service';\r\nimport * as slip from '../cryptography/slip39.service';\r\nimport { Key } from '../cryptography/cryptography.types';\r\nimport {\r\n  LrBadArgumentException,\r\n  LrBadStateException,\r\n} from '../_common/exceptions';\r\nimport { KeyService } from '../cryptography/key.service';\r\nimport {\r\n  CreateTpAssemblyInput,\r\n  CreateTpSubAssemblyInput,\r\n  PartialAssemblyKey,\r\n  TpAssemblyApproverInput,\r\n  UpdateTpAssemblyInput,\r\n  UpdateTpSubAssemblyInput,\r\n} from './tp-assembly.types';\r\nimport { LrGraphQLService } from '../api/lr-graphql';\r\nimport { TpsKeysQuery } from './tp-assembly.gql.private';\r\n\r\nexport abstract class TpAssemblyController {\r\n  abstract slip39Passphrase: string;\r\n\r\n  protected keyFactory: KeyFactoryService;\r\n  protected keyService: KeyService;\r\n  protected encryptionService: EncryptionService;\r\n  protected keyGraph: KeyGraphService;\r\n  protected slip39Service: slip.Slip39Service;\r\n  protected lrGraphQl: LrGraphQLService;\r\n\r\n  constructor(injector: Injector) {\r\n    this.keyFactory = injector.get(KeyFactoryService);\r\n    this.keyService = injector.get(KeyService);\r\n    this.encryptionService = injector.get(EncryptionService);\r\n    this.keyGraph = injector.get(KeyGraphService);\r\n    this.slip39Service = injector.get(slip.Slip39Service);\r\n    this.lrGraphQl = injector.get(LrGraphQLService);\r\n  }\r\n\r\n  abstract getTpWrappingKeyId(tp: TpNode): string;\r\n\r\n  async recoverAssemblyKey(partials: PartialAssemblyKey[]) {\r\n    // Every receiverPbkCipher should contain this and they should all be the same.\r\n    let assemblyKeyParams: object;\r\n\r\n    const shares = partials.map((partial) => {\r\n      if (assemblyKeyParams) {\r\n        if (\r\n          JSON.stringify(assemblyKeyParams) !==\r\n          JSON.stringify(partial.assemblyKeyParams)\r\n        ) {\r\n          throw new LrBadStateException(\r\n            'The assembly key parameters are different between the approvals.'\r\n          );\r\n        }\r\n      } else {\r\n        assemblyKeyParams = partial.assemblyKeyParams;\r\n      }\r\n      return partial.slip39.share.mnemonics;\r\n    });\r\n\r\n    const rawAssemblyKey = await this.slip39Service.recoverSecret(\r\n      shares,\r\n      this.slip39Passphrase\r\n    );\r\n\r\n    return await KeyFactoryService.asKey({\r\n      ...assemblyKeyParams,\r\n      k: rawAssemblyKey,\r\n    });\r\n  }\r\n\r\n  protected async prepareTpWrappingKey(\r\n    tp: TpAssemblyApproverInput\r\n  ): Promise<Key> {\r\n    return this.keyGraph.getKey(tp.wrappingKeyId);\r\n  }\r\n\r\n  protected async prepareAssembly({\r\n    subjectKey,\r\n    rootKey,\r\n  }: {\r\n    subjectKey: JWK.Key;\r\n    rootKey: JWK.Key;\r\n  }) {\r\n    const assemblyKey = await this.keyFactory.createKey();\r\n    console.log('prepareAssembly assemblyKey', assemblyKey);\r\n\r\n    const { k: rawAssemblyKey, ...assemblyKeyParams } = assemblyKey.toJSON(\r\n      true\r\n    ) as any;\r\n    const assemblyKeyVerifierPrk = await this.keyFactory.createPkcSignKey();\r\n    const wrappedAssemblyKeyVerifierPrk =\r\n      await this.encryptionService.encryptToString(\r\n        assemblyKey,\r\n        assemblyKeyVerifierPrk.toJSON(true)\r\n      );\r\n    const assemblyKeyVerifierPbk = JSON.stringify(\r\n      assemblyKeyVerifierPrk.toJSON()\r\n    );\r\n\r\n    const subjectKeyWrappedAssemblyKey =\r\n      await this.encryptionService.encryptToString(\r\n        subjectKey,\r\n        assemblyKey.toJSON(true)\r\n      );\r\n\r\n    // Encrypt the rootKey with the assemblyKey\r\n    const assemblyCipherData = await this.encryptionService.encryptToString(\r\n      assemblyKey,\r\n      {\r\n        rootKey: rootKey.toJSON(true),\r\n      }\r\n    );\r\n\r\n    return {\r\n      assemblyKey,\r\n      rawAssemblyKey,\r\n      assemblyKeyParams,\r\n      subjectKeyWrappedAssemblyKey,\r\n      assemblyCipherData,\r\n      assemblyKeyVerifierPbk,\r\n      wrappedAssemblyKeyVerifierPrk,\r\n    };\r\n  }\r\n\r\n  protected async prepareCreateSubAssemblies({\r\n    input,\r\n    subjectKey,\r\n    slipSubAssemblies,\r\n    assemblyKeyParams,\r\n  }: {\r\n    input: CreateTpSubAssemblyInput[];\r\n    subjectKey: JWK.Key;\r\n    slipSubAssemblies: slip.SubAssembly[];\r\n    assemblyKeyParams; // TODO type this\r\n  }) {\r\n    return Promise.all(\r\n      input.map(async (sa, saIndex) => {\r\n        const subjectCipherData = await this.encryptionService.encryptToString(\r\n          subjectKey,\r\n          sa.subjectCipherDataClearJson || ''\r\n        );\r\n\r\n        const createApprovers = await Promise.all(\r\n          sa.approverTps.map(async (approverTp, approverIndex) =>\r\n            this.prepareApprover({\r\n              tp: approverTp,\r\n              approverIndex,\r\n              slipSubAssembly: slipSubAssemblies[saIndex],\r\n              assemblyKeyParams,\r\n              subjectKey,\r\n            })\r\n          )\r\n        );\r\n\r\n        return {\r\n          singleReject: sa.singleReject,\r\n          quorum: sa.quorum,\r\n          subjectCipherData,\r\n          createApprovers,\r\n        };\r\n      })\r\n    );\r\n  }\r\n\r\n  protected async prepareUpdateSubAssemblies({\r\n    input,\r\n    subjectKey,\r\n    slipSubAssemblies,\r\n    assemblyKeyParams,\r\n    subAssemblies,\r\n  }: {\r\n    input: UpdateTpSubAssemblyInput[];\r\n    subjectKey: JWK.Key;\r\n    slipSubAssemblies: slip.SubAssembly[];\r\n    assemblyKeyParams; // TODO type this\r\n    subAssemblies: Connection<TpSubAssemblyNode>;\r\n  }) {\r\n    if (slipSubAssemblies.length !== input.length) {\r\n      throw new LrBadArgumentException(\r\n        'The slipSubAssemblies must be the same length as the input'\r\n      );\r\n    }\r\n\r\n    return Promise.all(\r\n      input.map(async (sa, saIndex) => {\r\n        const subjectCipherData = await this.encryptionService.encryptToString(\r\n          subjectKey,\r\n          sa.subjectCipherDataClearJson || ''\r\n        );\r\n\r\n        // Get the existing sub-assembly\r\n        const existingSa = subAssemblies.edges.find(\r\n          (edge) => edge.node.id === sa.id\r\n        ).node;\r\n\r\n        // Get approvers that do not exist yet\r\n        const createApprovers = [];\r\n        const updateApprovers = [];\r\n\r\n        sa.approverTps.forEach((tp) => {\r\n          const approver = existingSa.approvers.edges.find(\r\n            (edge) => edge.node.tp.id === tp.tpId\r\n          )?.node;\r\n          if (approver) {\r\n            updateApprovers.push({\r\n              tp,\r\n              approverId: approver.id,\r\n            });\r\n          } else {\r\n            createApprovers.push({\r\n              tp,\r\n            });\r\n          }\r\n        });\r\n\r\n        const slipSubAssembly = slipSubAssemblies[saIndex];\r\n\r\n        return {\r\n          subAssemblyId: sa.id,\r\n          singleReject: sa.singleReject,\r\n          quorum: sa.quorum,\r\n          subjectCipherData,\r\n          createApprovers: await Promise.all(\r\n            createApprovers.map(async ({ tp }, approverIndex) =>\r\n              this.prepareApprover({\r\n                tp,\r\n                approverIndex,\r\n                slipSubAssembly,\r\n                assemblyKeyParams,\r\n                subjectKey,\r\n              })\r\n            )\r\n          ),\r\n          updateApprovers: await Promise.all(\r\n            updateApprovers.map(async ({ tp, approverId }, approverIndex) =>\r\n              this.prepareApprover({\r\n                approverId,\r\n                tp,\r\n                approverIndex: approverIndex + createApprovers.length,\r\n                slipSubAssembly,\r\n                assemblyKeyParams,\r\n                subjectKey,\r\n              })\r\n            )\r\n          ),\r\n        };\r\n      })\r\n    );\r\n  }\r\n\r\n  private fillWrappingKeyId(input: CreateTpSubAssemblyInput[], tps: TpNode[]) {\r\n    // Fill in wrappingKeyId if not present\r\n    input.forEach((sa) => {\r\n      sa.approverTps.forEach((approver) => {\r\n        if (!approver.wrappingKeyId) {\r\n          const tp = tps.find((x) => x.id === approver.tpId);\r\n          approver.wrappingKeyId = this.getTpWrappingKeyId(tp);\r\n        }\r\n      });\r\n    });\r\n  }\r\n\r\n  private getInputApprovers(input: CreateTpSubAssemblyInput[]) {\r\n    const tpIds: string[] = [];\r\n    input.forEach((sa) => {\r\n      sa.approverTps.forEach((approver) => {\r\n        tpIds.push(approver.tpId);\r\n      });\r\n    });\r\n    return tpIds;\r\n  }\r\n\r\n  public async prepareCreate(input: CreateTpAssemblyInput) {\r\n    const rootKey = await this.keyService.getCurrentRootKey();\r\n    const subjectKey = await this.keyFactory.createKey();\r\n    const rootKeyWrappedSubjectKey = await this.keyGraph.wrapKey(\r\n      rootKey,\r\n      subjectKey\r\n    );\r\n\r\n    const { assemblyKey, rawAssemblyKey, assemblyKeyParams, ...assemblyInput } =\r\n      await this.prepareAssembly({\r\n        rootKey: rootKey.jwk,\r\n        subjectKey,\r\n      });\r\n\r\n    const slipAssembly = await this.prepareSlip39(\r\n      input.createSubAssemblies,\r\n      input.quorum,\r\n      rawAssemblyKey\r\n    );\r\n\r\n    const tpIds = this.getInputApprovers(input.createSubAssemblies);\r\n\r\n    // This should contain all the TPs that we need to update the assembly.\r\n    const tps = mapEdges(\r\n      (\r\n        await this.lrGraphQl.query({\r\n          query: TpsKeysQuery,\r\n          variables: {\r\n            ids: tpIds,\r\n          },\r\n        })\r\n      ).tps\r\n    );\r\n\r\n    // Fill in wrappingKeyId if not present\r\n    this.fillWrappingKeyId(input.createSubAssemblies, tps);\r\n\r\n    const createSubAssemblies = await this.prepareCreateSubAssemblies({\r\n      input: input.createSubAssemblies,\r\n      assemblyKeyParams,\r\n      slipSubAssemblies: slipAssembly.subAssemblies,\r\n      subjectKey,\r\n    });\r\n\r\n    return {\r\n      assemblyKey,\r\n      mutationInput: {\r\n        ...assemblyInput,\r\n        singleReject: input.singleReject,\r\n        quorum: input.quorum,\r\n        subjectCipherData: '', // TODO make this a parameter\r\n        createSubAssemblies,\r\n        rootKeyId: rootKey.id,\r\n        rootKeyWrappedSubjectKey,\r\n      },\r\n    };\r\n  }\r\n\r\n  private getAssemblyApprovers(assembly: TpAssemblyNode) {\r\n    const tpIds: string[] = [];\r\n\r\n    // Find all TPs that belong to sub-assemblies\r\n    mapEdges(assembly.subAssemblies).map((sa) => {\r\n      mapEdges(sa.approvers).forEach((approver) => tpIds.push(approver.tp.id));\r\n    });\r\n\r\n    return tpIds;\r\n  }\r\n\r\n  public async prepareUpdate(\r\n    input: UpdateTpAssemblyInput,\r\n    assembly: TpAssemblyNode // TODO type this to only what we need.\r\n  ) {\r\n    input = {\r\n      createSubAssemblies: [],\r\n      ...input,\r\n    };\r\n\r\n    const createSubAssembliesInput = input.createSubAssemblies || [];\r\n    const updateSubAssembliesInput = input.updateSubAssemblies || [];\r\n    const deleteSubAssembliesInput = input.deleteSubAssemblies || [];\r\n\r\n    // Must have at least one.\r\n    if (\r\n      createSubAssembliesInput.length === 0 &&\r\n      updateSubAssembliesInput.length === 0 &&\r\n      deleteSubAssembliesInput.length === 0\r\n    ) {\r\n      throw new LrBadArgumentException(\r\n        'Must specify at least one of: [createSubAssemblies, updateSubAssemblies, deleteSubAssemblies]'\r\n      );\r\n    }\r\n\r\n    const rootKey = await this.keyService.getCurrentRootKey();\r\n    const subjectKey = await this.keyGraph.getKey(assembly.subjectKey.id);\r\n\r\n    const { assemblyKey, rawAssemblyKey, assemblyKeyParams, ...assemblyInput } =\r\n      await this.prepareAssembly({\r\n        rootKey: rootKey.jwk,\r\n        subjectKey: subjectKey.jwk,\r\n      });\r\n\r\n    // Find all the TPs that we need, duplicates are no problem.\r\n    // Existing approvers\r\n    const tpIds = this.getAssemblyApprovers(assembly).concat(\r\n      this.getInputApprovers(createSubAssembliesInput),\r\n      this.getInputApprovers(updateSubAssembliesInput)\r\n    );\r\n\r\n    // This should contain all the TPs that we need to update the assembly.\r\n    const tps = mapEdges(\r\n      (\r\n        await this.lrGraphQl.query({\r\n          query: TpsKeysQuery,\r\n          variables: {\r\n            ids: tpIds,\r\n          },\r\n        })\r\n      ).tps\r\n    );\r\n\r\n    // Auto fill the updateSubAssemblies with existing sub assemblies if they are not\r\n    // explicitly deleted.\r\n    mapEdges(assembly.subAssemblies).forEach((existing) => {\r\n      // Deleting existing, don't include it\r\n      if (deleteSubAssembliesInput.includes(existing.id)) {\r\n        return;\r\n      }\r\n\r\n      // Update already specified in the input\r\n      if (updateSubAssembliesInput.some((sa) => sa.id === existing.id)) {\r\n        return;\r\n      }\r\n\r\n      const approverTps: TpAssemblyApproverInput[] = mapEdges(\r\n        existing.approvers\r\n      ).map((approver) => {\r\n        return {\r\n          tpId: approver.tp.id,\r\n          sharedCipherDataClearJson: approver.sharedCipherDataClearJson,\r\n          sharedCipherApprovalDataClearJson:\r\n            approver.sharedCipherApprovalDataClearJson,\r\n        };\r\n      });\r\n\r\n      updateSubAssembliesInput.push({\r\n        id: existing.id,\r\n        quorum: existing.quorum,\r\n        singleReject: existing.singleReject,\r\n        approverTps,\r\n        subjectCipherDataClearJson: existing.subjectCipherDataClearJson,\r\n      });\r\n    });\r\n\r\n    // Fill in wrappingKeyId if not present\r\n    this.fillWrappingKeyId(createSubAssembliesInput, tps);\r\n    this.fillWrappingKeyId(updateSubAssembliesInput, tps);\r\n\r\n    let slipAssembly;\r\n    try {\r\n      slipAssembly = await this.prepareSlip39(\r\n        (createSubAssembliesInput || []).concat(updateSubAssembliesInput || []),\r\n        input.quorum,\r\n        rawAssemblyKey\r\n      );\r\n    } catch (error) {\r\n      // TODO this sometimes happens. Maybe missing await.\r\n      console.log('Error while creating slip assembly', error);\r\n    }\r\n\r\n    const common = {\r\n      assemblyKeyParams,\r\n      subjectKey: subjectKey.jwk,\r\n    };\r\n\r\n    const createSubAssemblies = await this.prepareCreateSubAssemblies({\r\n      ...common,\r\n      input: createSubAssembliesInput,\r\n      // Need to match the slip sub assemblies to the input\r\n      slipSubAssemblies: slipAssembly.subAssemblies.slice(\r\n        0,\r\n        createSubAssembliesInput.length\r\n      ),\r\n    });\r\n\r\n    const updateSubAssemblies = await this.prepareUpdateSubAssemblies({\r\n      ...common,\r\n      input: updateSubAssembliesInput,\r\n      // Add an offset to the sub assemblies to skip the one that are created.\r\n      slipSubAssemblies: slipAssembly.subAssemblies.slice(\r\n        createSubAssembliesInput.length\r\n      ),\r\n      subAssemblies: assembly.subAssemblies,\r\n    });\r\n\r\n    return {\r\n      assemblyKey,\r\n      mutationInput: {\r\n        ...assemblyInput,\r\n        singleReject: input.singleReject,\r\n        quorum: input.quorum,\r\n        subjectCipherData: '', // TODO make this a parameter\r\n        subjectKeyId: subjectKey.id,\r\n        createSubAssemblies,\r\n        updateSubAssemblies,\r\n      },\r\n    };\r\n  }\r\n\r\n  private async prepareApprover({\r\n    approverId,\r\n    tp,\r\n    approverIndex,\r\n    slipSubAssembly,\r\n    assemblyKeyParams,\r\n    subjectKey,\r\n  }: {\r\n    approverId?: string; // Given if updating existing approver\r\n    tp: TpAssemblyApproverInput;\r\n    approverIndex: number;\r\n    slipSubAssembly: slip.SubAssembly;\r\n    assemblyKeyParams: object;\r\n    subjectKey: JWK.Key;\r\n  }) {\r\n    const tpWrappingKey = await this.prepareTpWrappingKey(tp);\r\n    // shared key is rotated every time.\r\n    const sharedKey = await this.keyFactory.createKey();\r\n\r\n    // For TP to access shared_key\r\n    const tpWrappedSharedKey = await this.encryptionService.encryptToString(\r\n      tpWrappingKey.jwk,\r\n      sharedKey.toJSON(true)\r\n    );\r\n    // For subject to access shared_key\r\n    const subjectKeyWrappedSharedKey =\r\n      await this.encryptionService.encryptToString(\r\n        subjectKey,\r\n        sharedKey.toJSON(true)\r\n      );\r\n\r\n    // If quorum is 1, then using the same share for every member.\r\n    const share =\r\n      slipSubAssembly.threshold === 1\r\n        ? slipSubAssembly.shares[0]\r\n        : slipSubAssembly.shares[approverIndex];\r\n\r\n    const partialAssemblyKey: PartialAssemblyKey = {\r\n      slip39: {\r\n        share,\r\n        subAssembly: {\r\n          quorum: slipSubAssembly.threshold,\r\n          size: slipSubAssembly.size,\r\n        },\r\n      },\r\n      assemblyKeyParams,\r\n    };\r\n\r\n    console.log('partialAssemblyKey', partialAssemblyKey);\r\n\r\n    const sharedCipherData = await this.encryptionService.encryptToString(\r\n      sharedKey,\r\n      tp.sharedCipherDataClearJson || ''\r\n    );\r\n\r\n    const sharedCipherApprovalData =\r\n      await this.encryptionService.encryptToString(\r\n        sharedKey,\r\n        tp.sharedCipherApprovalDataClearJson || ''\r\n      );\r\n\r\n    const sharedCipherPartialAssemblyKey =\r\n      await this.encryptionService.encryptToString(\r\n        sharedKey,\r\n        partialAssemblyKey\r\n      );\r\n\r\n    return {\r\n      tpWrappingKeyId: tpWrappingKey.id,\r\n      tpWrappedSharedKey,\r\n      subjectKeyWrappedSharedKey,\r\n      sharedCipherData,\r\n      sharedCipherApprovalData,\r\n      sharedCipherPartialAssemblyKey,\r\n      approverId: approverId || void 0, // If existing approver\r\n      tpId: approverId ? void 0 : tp.tpId, // else adding new TP\r\n    };\r\n  }\r\n\r\n  validateApprovers(approvers: TpNode[]): void {\r\n    // Ensure all approvers have mkSharedKey.\r\n    for (const tp of approvers) {\r\n      if (!tp.currentUserSharedKey.userSharedKey.mkSharedKey) {\r\n        const msg = `tp ${tp.other.username} does not have mkSharedKey`;\r\n        console.log(msg);\r\n        throw new LrBadArgumentException(msg);\r\n      }\r\n    }\r\n  }\r\n\r\n  // Prepare slip39\r\n  protected async prepareSlip39(\r\n    subAssemblies,\r\n    assemblyQuorum: number,\r\n    rawAssemblyKey: string\r\n  ): Promise<slip.Assembly> {\r\n    // Is there enough sub assemblies to meet quorum\r\n    if (subAssemblies.length < assemblyQuorum) {\r\n      throw new LrBadArgumentException(\r\n        'Not enough sub assemblies to meet quorum'\r\n      );\r\n    }\r\n\r\n    const slipAssembly = new slip.Assembly(assemblyQuorum);\r\n\r\n    subAssemblies.forEach((sa, index) => {\r\n      let approverCount = sa.approverTps.length;\r\n\r\n      // slip39 restricts quorum == 1 to have only 1 member. So we just share the same\r\n      // partial key for all sub assembly members.\r\n      if (sa.quorum === 1) {\r\n        approverCount = 1;\r\n      }\r\n      slipAssembly.addSubAssembly(\r\n        new slip.SubAssembly(index, sa.quorum, approverCount)\r\n      );\r\n    });\r\n\r\n    await this.slip39Service.generateShares(\r\n      rawAssemblyKey,\r\n      this.slip39Passphrase,\r\n      slipAssembly\r\n    );\r\n    return slipAssembly;\r\n  }\r\n}\r\n"]}
|