@lifeready/core 1.0.15 → 1.0.16

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.
Files changed (219) hide show
  1. package/README.md +62 -62
  2. package/bundles/lifeready-core.umd.js +14315 -14315
  3. package/bundles/lifeready-core.umd.js.map +1 -1
  4. package/bundles/lifeready-core.umd.min.js.map +1 -1
  5. package/esm2015/lib/_common/ast.js +40 -40
  6. package/esm2015/lib/_common/deferred-promise.js +24 -24
  7. package/esm2015/lib/_common/exceptions.js +157 -157
  8. package/esm2015/lib/_common/queries.gql.js +190 -190
  9. package/esm2015/lib/_common/run-outside-angular.js +79 -79
  10. package/esm2015/lib/_common/types.js +1 -1
  11. package/esm2015/lib/_common/utils.js +52 -52
  12. package/esm2015/lib/api/contact-card.gql.js +79 -79
  13. package/esm2015/lib/api/contact-card.service.js +154 -154
  14. package/esm2015/lib/api/contact-card2.gql.js +60 -60
  15. package/esm2015/lib/api/contact-card2.service.js +103 -103
  16. package/esm2015/lib/api/file.service.js +74 -74
  17. package/esm2015/lib/api/key-exchange.gql.js +188 -188
  18. package/esm2015/lib/api/key-exchange.service.js +442 -442
  19. package/esm2015/lib/api/key-exchange.types.js +18 -18
  20. package/esm2015/lib/api/key-exchange2.gql.js +171 -171
  21. package/esm2015/lib/api/key-exchange2.service.js +479 -479
  22. package/esm2015/lib/api/lock.gql.js +40 -40
  23. package/esm2015/lib/api/lock.service.js +64 -64
  24. package/esm2015/lib/api/lr-apollo.service.js +46 -46
  25. package/esm2015/lib/api/lr-graphql/index.js +6 -6
  26. package/esm2015/lib/api/lr-graphql/lr-graphql.service.js +155 -155
  27. package/esm2015/lib/api/lr-graphql/lr-merged-mutation.js +213 -213
  28. package/esm2015/lib/api/lr-graphql/lr-mutation-base.js +51 -51
  29. package/esm2015/lib/api/lr-graphql/lr-mutation.js +48 -48
  30. package/esm2015/lib/api/lr-graphql/lr.service.js +18 -18
  31. package/esm2015/lib/api/message.service.js +138 -138
  32. package/esm2015/lib/api/persist.service.js +181 -181
  33. package/esm2015/lib/api/query-processor/common-processors.service.js +93 -93
  34. package/esm2015/lib/api/query-processor/index.js +3 -3
  35. package/esm2015/lib/api/query-processor/query-processor.service.js +262 -262
  36. package/esm2015/lib/api/query-processor/tp-password-reset-processor.service.js +109 -109
  37. package/esm2015/lib/api/shared-contact-card.service.js +119 -119
  38. package/esm2015/lib/api/shared-contact-card2.gql.js +41 -41
  39. package/esm2015/lib/api/shared-contact-card2.service.js +117 -117
  40. package/esm2015/lib/api/time.service.js +146 -146
  41. package/esm2015/lib/api/types/graphql.types.js +7 -7
  42. package/esm2015/lib/api/types/index.js +3 -3
  43. package/esm2015/lib/api/types/lr-graphql.types.js +101 -101
  44. package/esm2015/lib/auth/auth.config.js +57 -57
  45. package/esm2015/lib/auth/auth.gql.js +48 -48
  46. package/esm2015/lib/auth/auth.types.js +27 -27
  47. package/esm2015/lib/auth/idle.service.js +168 -168
  48. package/esm2015/lib/auth/idle.types.js +7 -7
  49. package/esm2015/lib/auth/lbop.service.js +355 -355
  50. package/esm2015/lib/auth/life-ready-auth.service.js +488 -488
  51. package/esm2015/lib/auth/password.service.js +320 -320
  52. package/esm2015/lib/auth/register.service.js +172 -172
  53. package/esm2015/lib/auth/two-factor.service.js +74 -74
  54. package/esm2015/lib/category/category-meta.service.js +99 -99
  55. package/esm2015/lib/category/category.gql.js +406 -406
  56. package/esm2015/lib/category/category.service.js +390 -390
  57. package/esm2015/lib/category/category.types.js +29 -29
  58. package/esm2015/lib/cryptography/cryptography.types.js +11 -11
  59. package/esm2015/lib/cryptography/encryption.service.js +189 -189
  60. package/esm2015/lib/cryptography/key-factory.service.js +237 -237
  61. package/esm2015/lib/cryptography/key-graph.service.js +299 -299
  62. package/esm2015/lib/cryptography/key-meta.service.js +200 -200
  63. package/esm2015/lib/cryptography/key.service.js +124 -124
  64. package/esm2015/lib/cryptography/slip39.service.js +169 -169
  65. package/esm2015/lib/cryptography/web-crypto.service.js +29 -29
  66. package/esm2015/lib/items2/item2.gql.js +139 -139
  67. package/esm2015/lib/items2/item2.service.js +498 -498
  68. package/esm2015/lib/items2/item2.types.js +1 -1
  69. package/esm2015/lib/life-ready.config.js +84 -84
  70. package/esm2015/lib/life-ready.module.js +74 -74
  71. package/esm2015/lib/notification/notification.gql.js +43 -43
  72. package/esm2015/lib/notification/notification.service.js +118 -118
  73. package/esm2015/lib/plan/plan.gql.js +123 -123
  74. package/esm2015/lib/plan/plan.service.js +149 -149
  75. package/esm2015/lib/plan/plan.types.js +11 -11
  76. package/esm2015/lib/record/record-attachment.service.js +101 -101
  77. package/esm2015/lib/record/record.gql.js +179 -179
  78. package/esm2015/lib/record/record.service.js +206 -206
  79. package/esm2015/lib/record/record.types.js +15 -15
  80. package/esm2015/lib/record-type/record-type.service.js +75 -75
  81. package/esm2015/lib/record-type/record-type.types.js +28 -28
  82. package/esm2015/lib/scenario/scenario.constants.js +2 -2
  83. package/esm2015/lib/scenario/scenario.controller.js +34 -34
  84. package/esm2015/lib/scenario/scenario.gql.js +72 -72
  85. package/esm2015/lib/scenario/scenario.gql.private.js +198 -198
  86. package/esm2015/lib/scenario/scenario.service.js +538 -538
  87. package/esm2015/lib/scenario/scenario.types.js +1 -1
  88. package/esm2015/lib/trusted-parties/tp-assembly.gql.private.js +22 -22
  89. package/esm2015/lib/trusted-parties/tp-assembly.js +362 -362
  90. package/esm2015/lib/trusted-parties/tp-assembly.types.js +1 -1
  91. package/esm2015/lib/trusted-parties/tp-password-reset-request.service.js +113 -113
  92. package/esm2015/lib/trusted-parties/tp-password-reset-user.service.js +129 -129
  93. package/esm2015/lib/trusted-parties/tp-password-reset.constants.js +4 -4
  94. package/esm2015/lib/trusted-parties/tp-password-reset.controller.js +34 -34
  95. package/esm2015/lib/trusted-parties/tp-password-reset.gql.js +236 -236
  96. package/esm2015/lib/trusted-parties/tp-password-reset.service.js +95 -95
  97. package/esm2015/lib/trusted-parties/trusted-party.gql.js +148 -148
  98. package/esm2015/lib/trusted-parties/trusted-party.service.js +326 -326
  99. package/esm2015/lib/trusted-parties/trusted-party.types.js +41 -41
  100. package/esm2015/lib/trusted-parties/trusted-party2.gql.js +87 -87
  101. package/esm2015/lib/trusted-parties/trusted-party2.service.js +215 -215
  102. package/esm2015/lib/users/profile-details.service.js +214 -214
  103. package/esm2015/lib/users/profile.gql.js +97 -97
  104. package/esm2015/lib/users/profile.service.js +169 -169
  105. package/esm2015/lib/users/profile.types.js +34 -34
  106. package/esm2015/lib/users/user.gql.js +60 -60
  107. package/esm2015/lib/users/user.service.js +79 -79
  108. package/esm2015/lib/users/user.types.js +5 -5
  109. package/esm2015/lifeready-core.js +13 -13
  110. package/esm2015/public-api.js +71 -71
  111. package/fesm2015/lifeready-core.js +12258 -12258
  112. package/fesm2015/lifeready-core.js.map +1 -1
  113. package/lib/_common/ast.d.ts +11 -11
  114. package/lib/_common/deferred-promise.d.ts +12 -12
  115. package/lib/_common/exceptions.d.ts +109 -109
  116. package/lib/_common/queries.gql.d.ts +10 -10
  117. package/lib/_common/run-outside-angular.d.ts +14 -14
  118. package/lib/_common/types.d.ts +10 -10
  119. package/lib/_common/utils.d.ts +9 -9
  120. package/lib/api/contact-card.gql.d.ts +7 -7
  121. package/lib/api/contact-card.service.d.ts +52 -52
  122. package/lib/api/contact-card2.gql.d.ts +34 -34
  123. package/lib/api/contact-card2.service.d.ts +49 -49
  124. package/lib/api/file.service.d.ts +18 -18
  125. package/lib/api/key-exchange.gql.d.ts +9 -9
  126. package/lib/api/key-exchange.service.d.ts +39 -39
  127. package/lib/api/key-exchange.types.d.ts +196 -196
  128. package/lib/api/key-exchange2.gql.d.ts +125 -125
  129. package/lib/api/key-exchange2.service.d.ts +187 -187
  130. package/lib/api/lock.gql.d.ts +27 -27
  131. package/lib/api/lock.service.d.ts +34 -34
  132. package/lib/api/lr-apollo.service.d.ts +15 -15
  133. package/lib/api/lr-graphql/index.d.ts +5 -5
  134. package/lib/api/lr-graphql/lr-graphql.service.d.ts +60 -60
  135. package/lib/api/lr-graphql/lr-merged-mutation.d.ts +28 -28
  136. package/lib/api/lr-graphql/lr-mutation-base.d.ts +28 -28
  137. package/lib/api/lr-graphql/lr-mutation.d.ts +8 -8
  138. package/lib/api/lr-graphql/lr.service.d.ts +9 -9
  139. package/lib/api/message.service.d.ts +58 -58
  140. package/lib/api/persist.service.d.ts +31 -31
  141. package/lib/api/query-processor/common-processors.service.d.ts +36 -36
  142. package/lib/api/query-processor/index.d.ts +2 -2
  143. package/lib/api/query-processor/query-processor.service.d.ts +18 -18
  144. package/lib/api/query-processor/tp-password-reset-processor.service.d.ts +15 -15
  145. package/lib/api/shared-contact-card.service.d.ts +33 -33
  146. package/lib/api/shared-contact-card2.gql.d.ts +36 -36
  147. package/lib/api/shared-contact-card2.service.d.ts +45 -45
  148. package/lib/api/time.service.d.ts +16 -16
  149. package/lib/api/types/graphql.types.d.ts +29 -29
  150. package/lib/api/types/index.d.ts +2 -2
  151. package/lib/api/types/lr-graphql.types.d.ts +502 -502
  152. package/lib/auth/auth.config.d.ts +5 -5
  153. package/lib/auth/auth.gql.d.ts +15 -15
  154. package/lib/auth/auth.types.d.ts +66 -66
  155. package/lib/auth/idle.service.d.ts +40 -40
  156. package/lib/auth/idle.types.d.ts +10 -10
  157. package/lib/auth/lbop.service.d.ts +91 -91
  158. package/lib/auth/life-ready-auth.service.d.ts +61 -61
  159. package/lib/auth/password.service.d.ts +78 -78
  160. package/lib/auth/register.service.d.ts +25 -25
  161. package/lib/auth/two-factor.service.d.ts +15 -15
  162. package/lib/category/category-meta.service.d.ts +23 -23
  163. package/lib/category/category.gql.d.ts +45 -45
  164. package/lib/category/category.service.d.ts +67 -67
  165. package/lib/category/category.types.d.ts +79 -79
  166. package/lib/cryptography/cryptography.types.d.ts +83 -83
  167. package/lib/cryptography/encryption.service.d.ts +41 -41
  168. package/lib/cryptography/key-factory.service.d.ts +38 -38
  169. package/lib/cryptography/key-graph.service.d.ts +41 -41
  170. package/lib/cryptography/key-meta.service.d.ts +44 -44
  171. package/lib/cryptography/key.service.d.ts +36 -36
  172. package/lib/cryptography/slip39.service.d.ts +43 -43
  173. package/lib/cryptography/web-crypto.service.d.ts +5 -5
  174. package/lib/items2/item2.gql.d.ts +123 -123
  175. package/lib/items2/item2.service.d.ts +203 -203
  176. package/lib/items2/item2.types.d.ts +70 -70
  177. package/lib/life-ready.config.d.ts +14 -14
  178. package/lib/life-ready.module.d.ts +5 -5
  179. package/lib/notification/notification.gql.d.ts +37 -37
  180. package/lib/notification/notification.service.d.ts +63 -63
  181. package/lib/plan/plan.gql.d.ts +11 -11
  182. package/lib/plan/plan.service.d.ts +33 -33
  183. package/lib/plan/plan.types.d.ts +31 -31
  184. package/lib/record/record-attachment.service.d.ts +16 -16
  185. package/lib/record/record.gql.d.ts +14 -14
  186. package/lib/record/record.service.d.ts +25 -25
  187. package/lib/record/record.types.d.ts +57 -57
  188. package/lib/record-type/record-type.service.d.ts +11 -11
  189. package/lib/record-type/record-type.types.d.ts +50 -50
  190. package/lib/scenario/scenario.constants.d.ts +1 -1
  191. package/lib/scenario/scenario.controller.d.ts +10 -10
  192. package/lib/scenario/scenario.gql.d.ts +62 -62
  193. package/lib/scenario/scenario.gql.private.d.ts +16 -16
  194. package/lib/scenario/scenario.service.d.ts +233 -233
  195. package/lib/scenario/scenario.types.d.ts +50 -50
  196. package/lib/trusted-parties/tp-assembly.d.ts +177 -177
  197. package/lib/trusted-parties/tp-assembly.gql.private.d.ts +5 -5
  198. package/lib/trusted-parties/tp-assembly.types.d.ts +38 -38
  199. package/lib/trusted-parties/tp-password-reset-request.service.d.ts +20 -20
  200. package/lib/trusted-parties/tp-password-reset-user.service.d.ts +35 -35
  201. package/lib/trusted-parties/tp-password-reset.constants.d.ts +3 -3
  202. package/lib/trusted-parties/tp-password-reset.controller.d.ts +10 -10
  203. package/lib/trusted-parties/tp-password-reset.gql.d.ts +223 -223
  204. package/lib/trusted-parties/tp-password-reset.service.d.ts +188 -188
  205. package/lib/trusted-parties/trusted-party.gql.d.ts +9 -9
  206. package/lib/trusted-parties/trusted-party.service.d.ts +44 -44
  207. package/lib/trusted-parties/trusted-party.types.d.ts +102 -102
  208. package/lib/trusted-parties/trusted-party2.gql.d.ts +79 -79
  209. package/lib/trusted-parties/trusted-party2.service.d.ts +114 -114
  210. package/lib/users/profile-details.service.d.ts +21 -21
  211. package/lib/users/profile.gql.d.ts +11 -11
  212. package/lib/users/profile.service.d.ts +35 -35
  213. package/lib/users/profile.types.d.ts +96 -96
  214. package/lib/users/user.gql.d.ts +9 -9
  215. package/lib/users/user.service.d.ts +12 -12
  216. package/lib/users/user.types.d.ts +23 -23
  217. package/lifeready-core.d.ts +12 -12
  218. package/package.json +1 -1
  219. package/public-api.d.ts +67 -67
@@ -1,479 +1,479 @@
1
- import { __awaiter, __decorate } from "tslib";
2
- import { Injectable, NgZone } from '@angular/core';
3
- import { EncryptionService, JoseSerialization, } from '../cryptography/encryption.service';
4
- import { KeyFactoryService } from '../cryptography/key-factory.service';
5
- import { KeyGraphService } from '../cryptography/key-graph.service';
6
- import { KeyService } from '../cryptography/key.service';
7
- import { LrCodeMismatchException } from '../_common/exceptions';
8
- import { RunOutsideAngular } from '../_common/run-outside-angular';
9
- import { UserService } from './../users/user.service';
10
- import { CancelKeyExchangeMutation, CompleteKeyExchangeOtkMutation, CurrentUserSharedKeyQuery2, DeclineKeyExchangeMutation, InitiateKeyExchangeOtkMutation, KeyExchangeQuery2, KeyExchangesQuery2, KeyExchangeTokenQuery2, RespondKeyExchangeOtkMutation, } from './key-exchange2.gql';
11
- import { LrGraphQLService, LrMutation } from './lr-graphql';
12
- import * as i0 from "@angular/core";
13
- import * as i1 from "../cryptography/key-factory.service";
14
- import * as i2 from "../cryptography/key.service";
15
- import * as i3 from "../cryptography/encryption.service";
16
- import * as i4 from "../users/user.service";
17
- import * as i5 from "../cryptography/key-graph.service";
18
- import * as i6 from "./lr-graphql/lr-graphql.service";
19
- let KeyExchange2Service = class KeyExchange2Service {
20
- constructor(ngZone, keyFactory, keyService, encryptionService, userService, keyGraph, lrGraphQL) {
21
- this.ngZone = ngZone;
22
- this.keyFactory = keyFactory;
23
- this.keyService = keyService;
24
- this.encryptionService = encryptionService;
25
- this.userService = userService;
26
- this.keyGraph = keyGraph;
27
- this.lrGraphQL = lrGraphQL;
28
- this.CLIENT_NONCE_LENGTH = 32;
29
- }
30
- getOtKey(keyExchange, otKeyK) {
31
- return __awaiter(this, void 0, void 0, function* () {
32
- if (otKeyK) {
33
- return yield KeyFactoryService.asKey(Object.assign(Object.assign({}, JSON.parse(keyExchange.otk.otKeyParams)), { k: otKeyK }));
34
- }
35
- else if (keyExchange.otk.state === 'OTK_INITIATED' &&
36
- !keyExchange.isInitiator &&
37
- keyExchange.otk.responderPbkCipher) {
38
- // Assuming existing user getting invited where OTK is wrapped in responder's public key.
39
- const prk = yield this.keyService.getCurrentPxk();
40
- const decryptedCipher = yield this.encryptionService.decrypt(prk.jwk, JSON.parse(keyExchange.otk.responderPbkCipher), {
41
- serializations: [JoseSerialization.COMPACT],
42
- });
43
- if (decryptedCipher.otKey) {
44
- return yield KeyFactoryService.asKey(decryptedCipher.otKey);
45
- }
46
- }
47
- return null;
48
- });
49
- }
50
- decryptOtk(keyExchange, otKeyK) {
51
- return __awaiter(this, void 0, void 0, function* () {
52
- const otKey = yield this.getOtKey(keyExchange, otKeyK);
53
- let otk = keyExchange.otk;
54
- if (otKey && otk.otKeyCipher) {
55
- otk = Object.assign(Object.assign({}, otk), { otKey, otKeyCipherClearJson: yield this.encryptionService.decrypt(otKey, keyExchange.otk.otKeyCipher) });
56
- }
57
- return Object.assign(Object.assign({}, keyExchange), { otk });
58
- });
59
- }
60
- decryptResponseCipher(otKey, otPrk, content) {
61
- return __awaiter(this, void 0, void 0, function* () {
62
- // The response could be wrapped by the OtK in addition to the OtPbk
63
- try {
64
- content = yield this.encryptionService.decrypt(otKey, content);
65
- }
66
- catch (error) {
67
- if (error.message !== 'no key found') {
68
- throw error;
69
- }
70
- // Do nothing to support older versions where message is not wrapped with otk.
71
- }
72
- // The Prk is single-use and only used to send information from the responder back to the initiator.
73
- return yield this.encryptionService.decrypt(otPrk, content);
74
- });
75
- }
76
- decryptKeyExchangeAsInitiator(keyExchange) {
77
- return __awaiter(this, void 0, void 0, function* () {
78
- const rootKey = yield this.keyService.getCurrentRootKey();
79
- // Decrypt using the root key to get the Prk
80
- const initiatorRootKeyCipherClearJson = (yield this.encryptionService.decrypt(rootKey.jwk, keyExchange.initiatorRootKeyCipher));
81
- const otKey = yield KeyFactoryService.asKey(initiatorRootKeyCipherClearJson.otKey);
82
- keyExchange = Object.assign(Object.assign({}, keyExchange), { initiatorRootKeyCipherClearJson });
83
- let otk = keyExchange.otk;
84
- if (otk.initiatorOneTimePbkCipher) {
85
- otk = Object.assign(Object.assign({}, otk), { initiatorOneTimePbkCipherClearJson: yield this.decryptResponseCipher(otKey, yield KeyFactoryService.asKey(initiatorRootKeyCipherClearJson.oneTimePrk), otk.initiatorOneTimePbkCipher) });
86
- }
87
- if (otk.otKeyCipher) {
88
- otk.otKeyCipherClearJson = yield this.encryptionService.decrypt(otKey, otk.otKeyCipher);
89
- }
90
- return Object.assign(Object.assign({}, keyExchange), { otk });
91
- });
92
- }
93
- decryptKeyExchangeAsResponder(keyExchange, otKeyK) {
94
- return __awaiter(this, void 0, void 0, function* () {
95
- return this.decryptOtk(keyExchange, otKeyK);
96
- });
97
- }
98
- decryptKeyExchange(keyExchange, otKeyK) {
99
- return __awaiter(this, void 0, void 0, function* () {
100
- if (keyExchange.isInitiator) {
101
- return this.decryptKeyExchangeAsInitiator(keyExchange);
102
- }
103
- else {
104
- return this.decryptKeyExchangeAsResponder(keyExchange, otKeyK);
105
- }
106
- });
107
- }
108
- getKeyExchanges({ state } = {}) {
109
- return __awaiter(this, void 0, void 0, function* () {
110
- const { keyExchanges } = yield this.lrGraphQL.query({
111
- query: KeyExchangesQuery2,
112
- variables: {
113
- state,
114
- },
115
- });
116
- return Promise.all(keyExchanges.edges.map((edge) => this.decryptKeyExchange(edge.node)));
117
- });
118
- }
119
- /**
120
- * @param id If the current user can responder the key exchange if they are either the initiator or the receiver.
121
- * @param token If not signed in, or not the initiator or responder, 'token' must be given.
122
- * @param otKeyK Is the raw one-time key (string). If the responder is explicitly specified at time of initiation, then
123
- * it's possible to have the otKey wrapped by the public key of the responder. In which case, the otKeyK is not needed.
124
- */
125
- getKeyExchange(id, { otKeyK, token } = {}) {
126
- return __awaiter(this, void 0, void 0, function* () {
127
- const res = yield this.lrGraphQL.query({
128
- query: token ? KeyExchangeTokenQuery2 : KeyExchangeQuery2,
129
- variables: {
130
- id,
131
- token,
132
- },
133
- includeKeyGraph: !token,
134
- });
135
- return this.decryptKeyExchange(res.keyExchange, otKeyK);
136
- });
137
- }
138
- getCurrentUserSharedKey(input) {
139
- return __awaiter(this, void 0, void 0, function* () {
140
- return this.lrGraphQL.query({
141
- query: CurrentUserSharedKeyQuery2,
142
- variables: {
143
- username: input.username,
144
- userId: input.userId,
145
- },
146
- });
147
- });
148
- }
149
- // TODO: deprecate this
150
- cancelKeyExchange(id) {
151
- return __awaiter(this, void 0, void 0, function* () {
152
- return this.cancelKeyExchangeMutation(id);
153
- });
154
- }
155
- cancelKeyExchangeMutation(id) {
156
- return new LrMutation({
157
- mutation: CancelKeyExchangeMutation,
158
- variables: {
159
- input: {
160
- id,
161
- },
162
- },
163
- });
164
- }
165
- // TODO: deprecate this
166
- declineKeyExchange(id, token) {
167
- return __awaiter(this, void 0, void 0, function* () {
168
- return this.declineKeyExchangeMutation(id, token);
169
- });
170
- }
171
- declineKeyExchangeMutation(id, token) {
172
- return new LrMutation({
173
- mutation: DeclineKeyExchangeMutation,
174
- variables: {
175
- input: {
176
- id,
177
- token,
178
- },
179
- },
180
- });
181
- }
182
- initiateOtk({ message, email, contactCard, upgrade, } = {}) {
183
- return __awaiter(this, void 0, void 0, function* () {
184
- const otKey = yield this.keyFactory.createKey();
185
- const nonce = this.keyFactory.randomString(this.CLIENT_NONCE_LENGTH);
186
- // New PKC key for encryption. This key is used only once when the responder sends
187
- // back their signing public key.
188
- const initiatorOneTimePrk = yield this.keyFactory.createPkcKey();
189
- // Option 1: New PKC key for signing
190
- // const initiatorSigPrk = await this.keyService.createPkcSignKey();
191
- // Option 2: Use the user's global signing key.
192
- // This key is used to prove the initiator's identity.
193
- const initiatorPrk = yield this.keyService.getCurrentPxk();
194
- const initiatorSigPrk = yield this.keyService.getCurrentSigPxk();
195
- let initiatorPlainDataSig = null;
196
- if (contactCard && contactCard.ownerPlainDataJson) {
197
- initiatorPlainDataSig = yield this.encryptionService.signToString(initiatorSigPrk.jwk, contactCard.ownerPlainDataJson);
198
- }
199
- const initiator = {
200
- message,
201
- contactCard: contactCard && {
202
- sharedCipherDataClearJson: contactCard.sharedCipherDataClearJson,
203
- },
204
- };
205
- // Content to be encrypted using the OTK.
206
- const plainOtKeyCipher = {
207
- // TODO Make sure we also put the OOB code in here as well since the OOB code is the
208
- // _only_ information the KC server does not have access to. The server may have
209
- // access to OTK and hence the nonce here. It's good to have both the nonce and OOB code
210
- // since the user may not be using the OOB code. And it's simple to always include
211
- // the nonce, so why not.
212
- nonce,
213
- initiator: Object.assign(Object.assign({}, initiator), { oneTimePbk: initiatorOneTimePrk.toJSON(), pbk: initiatorPrk.jwk.toJSON(), sigPbk: initiatorSigPrk.jwk.toJSON() }),
214
- };
215
- const otKeyCipher = yield this.keyGraph.encryptToString(otKey, plainOtKeyCipher);
216
- // Content to be encrypted using the initiator's root key.
217
- const initiatorRootKeyCipherClearJson = {
218
- nonce,
219
- oneTimePrk: initiatorOneTimePrk.toJSON(true),
220
- // Should not need to keep this encrypted since we are using the global signing key.
221
- // sigPrk: initiatorSigPrk.toJSON(true),
222
- // Save it in case the initiator want to decode the otKeyCipher.
223
- // Since the otKey is only used once, and that otKeyCipher contains only
224
- // the public key of the initiator, it's safe just leave the otKey stored here.
225
- otKey: otKey.toJSON(true),
226
- // These should be storing information such as how the fields of the shared contact card is
227
- // derived from the master contact card.
228
- initiatorContactCard: contactCard,
229
- initiator,
230
- };
231
- const rootKey = yield this.keyService.getCurrentRootKey();
232
- const initiatorRootKeyCipher = yield this.keyGraph.encryptToString(rootKey.jwk, initiatorRootKeyCipherClearJson);
233
- // The raw OTK
234
- const otKeyK = otKey.toJSON(true).k;
235
- // API call
236
- const lrMutation = new LrMutation({
237
- mutation: InitiateKeyExchangeOtkMutation,
238
- variables: {
239
- input: {
240
- // These will be stored on the server
241
- initiatorRootKeyCipher,
242
- initiatorPxkId: initiatorPrk.id,
243
- initiatorSigPxkId: initiatorSigPrk.id,
244
- // These will be sent to the responder
245
- otKeyParams: JSON.stringify(otKey.toJSON()),
246
- otKeyCipher,
247
- sendEmail: email && {
248
- email,
249
- rawOtKey: otKeyK,
250
- },
251
- createTp: true,
252
- initiatorPlainDataSig,
253
- upgrade,
254
- },
255
- },
256
- });
257
- return { lrMutation, otKeyK };
258
- });
259
- }
260
- respondOtk({ keyExchangeId, token, decryptedOtk, message, initiatorContactCard, responderContactCard, }) {
261
- return __awaiter(this, void 0, void 0, function* () {
262
- const rootKey = yield this.keyService.getCurrentRootKey();
263
- const masterKeyId = this.keyService.getCurrentMasterKey().id;
264
- const masterKey = yield this.keyService.getCurrentMasterKey();
265
- const sharedKey = yield this.keyFactory.createKey();
266
- const mkSharedKey = yield this.keyFactory.createKey();
267
- const rkWrappedSharedKey = yield this.encryptionService.encrypt(rootKey.jwk, sharedKey.toJSON(true));
268
- const mkWrappedMkSharedKey = yield this.encryptionService.encrypt(masterKey.jwk, mkSharedKey.toJSON(true));
269
- const initiatorOneTimePbk = yield KeyFactoryService.asKey(decryptedOtk.otKeyCipherClearJson.initiator.oneTimePbk);
270
- const initiatorPbk = yield KeyFactoryService.asKey(decryptedOtk.otKeyCipherClearJson.initiator.pbk);
271
- const initiatorSigPbk = yield KeyFactoryService.asKey(decryptedOtk.otKeyCipherClearJson.initiator.sigPbk);
272
- // Option 1: Using new Prk for each TP pair
273
- // Create a new public signing key for the responder.
274
- // const responderSigPrk = await this.keyService.createPkcSignKey()
275
- // const rkWrappedResponderSigPrk = await this.encrypt(rootKey, responderSigPrk.toJSON(true));
276
- // Option 2: Responder already has a signing Prk
277
- const responderPrk = yield this.keyService.getCurrentPxk();
278
- const responderSigPrk = yield this.keyService.getCurrentSigPxk();
279
- const signedInitiatorPbk = yield this.encryptionService.sign(responderSigPrk.jwk, initiatorPbk.toJSON());
280
- const signedInitiatorSigPbk = yield this.encryptionService.sign(responderSigPrk.jwk, initiatorSigPbk.toJSON());
281
- const initiatorOneTimePbkCipherClearJson = {
282
- nonce: decryptedOtk.otKeyCipherClearJson.nonce,
283
- sharedKey: sharedKey.toJSON(true),
284
- mkSharedKey: mkSharedKey.toJSON(true),
285
- responder: {
286
- pbk: responderPrk.jwk.toJSON(),
287
- sigPbk: responderSigPrk.jwk.toJSON(),
288
- message,
289
- },
290
- };
291
- let receivedCardInput;
292
- if (decryptedOtk.otKeyCipherClearJson.initiator.contactCard) {
293
- // Set the info about the initiator to be the ones sent by the initiator. We need th responder to do the encryption here
294
- // because the initiator does not have the shared key yet, and we want the responder to have a functional contact card after
295
- // this exchange. The initiator can double check the contact details are correct and sign it when it completes the exchange.
296
- const sharedCipherDataClearJson = decryptedOtk.otKeyCipherClearJson.initiator.contactCard
297
- .sharedCipherDataClearJson;
298
- // Create keys
299
- const receiverKey = yield this.keyFactory.createKey();
300
- const ccSharedKey = yield this.keyFactory.createKey();
301
- const sigPxk = yield this.keyService.getCurrentSigPxk();
302
- receivedCardInput = {
303
- receiverWrappedKey: JSON.stringify(yield this.encryptionService.encrypt(rootKey.jwk, receiverKey.toJSON(true))),
304
- receiverWrappingKeyId: rootKey.id,
305
- receiverCipherData: initiatorContactCard
306
- ? JSON.stringify(yield this.encryptionService.encrypt(receiverKey, initiatorContactCard.receiverCipherDataClearJson))
307
- : '',
308
- sharedWrappedKey: JSON.stringify(yield this.encryptionService.encrypt(sharedKey, ccSharedKey.toJSON(true))),
309
- };
310
- const sharedCipherData = yield this.encryptionService.encrypt(ccSharedKey, sharedCipherDataClearJson);
311
- receivedCardInput.sharedCipherDataSig = JSON.stringify(yield this.encryptionService.sign(sigPxk.jwk, sharedCipherData));
312
- receivedCardInput.sigPxkId = sigPxk.id;
313
- initiatorOneTimePbkCipherClearJson.responder.contactCard = Object.assign(Object.assign({}, initiatorOneTimePbkCipherClearJson.responder.contactCard), { sharedCipherKey: ccSharedKey.toJSON(true) });
314
- }
315
- let responderCardInput;
316
- if (responderContactCard) {
317
- // Create keys
318
- const ownerKey = yield this.keyFactory.createKey();
319
- const ccSharedKey = yield this.keyFactory.createKey();
320
- const sigPxk = yield this.keyService.getCurrentSigPxk();
321
- responderCardInput = {
322
- ownerWrappedKey: JSON.stringify(yield this.encryptionService.encrypt(rootKey.jwk, ownerKey.toJSON(true))),
323
- ownerWrappingKeyId: rootKey.id,
324
- ownerCipherData: responderContactCard.ownerCipherDataClearJson
325
- ? JSON.stringify(yield this.encryptionService.encrypt(ownerKey, responderContactCard.ownerCipherDataClearJson))
326
- : '',
327
- sharedWrappedKey: JSON.stringify(yield this.encryptionService.encrypt(sharedKey, ccSharedKey.toJSON(true))),
328
- };
329
- const sharedCipherData = yield this.encryptionService.encrypt(ccSharedKey, responderContactCard.sharedCipherDataClearJson);
330
- responderCardInput.sharedCipherDataSig = JSON.stringify(yield this.encryptionService.sign(sigPxk.jwk, sharedCipherData));
331
- responderCardInput.sigPxkId = sigPxk.id;
332
- if (responderContactCard.ownerPlainDataJson) {
333
- responderCardInput.ownerPlainDataSig = JSON.stringify(yield this.encryptionService.sign(responderSigPrk.jwk, responderContactCard.ownerPlainDataJson));
334
- }
335
- // Contact card info readable by the initiator
336
- initiatorOneTimePbkCipherClearJson.responder.contactCard = Object.assign(Object.assign({}, initiatorOneTimePbkCipherClearJson.responder.contactCard), { sharedCipherDataClearJson: responderContactCard.sharedCipherDataClearJson });
337
- }
338
- // Encrypt with one-time public key
339
- let initiatorOneTimePbkCipher = yield this.encryptionService.encrypt(initiatorOneTimePbk, initiatorOneTimePbkCipherClearJson);
340
- // Encrypt with the otk again to keep use of asymmetric keys to a minimum.
341
- initiatorOneTimePbkCipher = yield this.encryptionService.encrypt(decryptedOtk.otKey, initiatorOneTimePbkCipher);
342
- return new LrMutation({
343
- mutation: RespondKeyExchangeOtkMutation,
344
- variables: {
345
- input: {
346
- keyExchangeId,
347
- keyExchangeToken: token,
348
- rootKeyId: rootKey.id,
349
- masterKeyId,
350
- // These will be stored on the server
351
- responderPxkId: responderPrk.id,
352
- responderSigPxkId: responderSigPrk.id,
353
- signedInitiatorPbk: JSON.stringify(signedInitiatorPbk),
354
- signedInitiatorSigPbk: JSON.stringify(signedInitiatorSigPbk),
355
- // rkWrappedInitiatorSigPbk: JSON.stringify(rkWrappedInitiatorSigPbk),
356
- // Option 1: Using new Prk for each TP pair
357
- // rkWrappedResponderSigPrk: JSON.stringify(rkWrappedResponderSigPrk),
358
- rkWrappedSharedKey: JSON.stringify(rkWrappedSharedKey),
359
- mkWrappedMkSharedKey: JSON.stringify(mkWrappedMkSharedKey),
360
- // These will be sent to the initiator
361
- initiatorOneTimePbkCipher: JSON.stringify(initiatorOneTimePbkCipher),
362
- initiatorContactCard: receivedCardInput,
363
- responderContactCard: responderCardInput,
364
- },
365
- },
366
- });
367
- });
368
- }
369
- completeOtk({ keyExchangeId, initiatorRootKeyCipher, initiatorOneTimePbkCipher, responderContactCard, initiatorContactCard, }) {
370
- return __awaiter(this, void 0, void 0, function* () {
371
- const rootKey = yield this.keyService.getCurrentRootKey();
372
- const masterKey = yield this.keyService.getCurrentMasterKey();
373
- // Decrypt using the root key to get the Prk
374
- const initiatorRootKeyCipherClearJson = (yield this.encryptionService.decrypt(rootKey.jwk, initiatorRootKeyCipher));
375
- // The Prk is single-use and only used to send information from the responder back to the initiator.
376
- const plainInitiatorOneTimePbkCipher = yield this.decryptResponseCipher(yield KeyFactoryService.asKey(initiatorRootKeyCipherClearJson.otKey), yield KeyFactoryService.asKey(initiatorRootKeyCipherClearJson.oneTimePrk), initiatorOneTimePbkCipher);
377
- // Check the nonce match to ensure the responder was the one holding the OTK
378
- if (initiatorRootKeyCipherClearJson.nonce !==
379
- plainInitiatorOneTimePbkCipher.nonce) {
380
- throw new LrCodeMismatchException('The nonce returned by responder does not match with the one created by the initiator.');
381
- }
382
- // Option 1: Assuming the signing key is unique between users.
383
- // const initiatorSigPrk = await KFS.asKey(ke.plainInitiatorRootKeyCipher.sigPrk);
384
- // const rkWrappedInitiatorSigPrk = await this.encrypt(rootKey, initiatorSigPrk.toJSON(true));
385
- // Option 2: Use the user's global signing key.
386
- // In this case the initiatorSigPrk is already a part of the key graph.
387
- // So there's nothing to do here.
388
- // Protected the signing public key of the responder.
389
- const initiatorSigPrk = yield this.keyService.getCurrentSigPxk();
390
- const responderSigPbk = yield KeyFactoryService.asKey(plainInitiatorOneTimePbkCipher.responder.sigPbk);
391
- const responderPbk = yield KeyFactoryService.asKey(plainInitiatorOneTimePbkCipher.responder.pbk);
392
- const signedResponderPbk = yield this.encryptionService.sign(initiatorSigPrk.jwk, responderPbk.toJSON());
393
- const signedResponderSigPbk = yield this.encryptionService.sign(initiatorSigPrk.jwk, responderSigPbk.toJSON());
394
- const sharedKey = yield KeyFactoryService.asKey(plainInitiatorOneTimePbkCipher.sharedKey);
395
- const rkWrappedSharedKey = yield this.encryptionService.encrypt(rootKey.jwk, sharedKey.toJSON(true));
396
- const mkSharedKey = yield KeyFactoryService.asKey(plainInitiatorOneTimePbkCipher.mkSharedKey);
397
- const mkWrappedMkSharedKey = yield this.encryptionService.encrypt(masterKey.jwk, mkSharedKey.toJSON(true));
398
- let responderContactCardCipherInput;
399
- if (responderContactCard) {
400
- // Create key
401
- const receiverKey = yield this.keyFactory.createKey();
402
- responderContactCardCipherInput = {
403
- receiverWrappedKey: JSON.stringify(yield this.encryptionService.encrypt(rootKey.jwk, receiverKey.toJSON(true))),
404
- receiverWrappingKeyId: rootKey.id,
405
- receiverCipherData: JSON.stringify(yield this.encryptionService.encrypt(receiverKey, responderContactCard.receiverCipherDataClearJson)),
406
- };
407
- }
408
- // Get the data needed from the initiator's cipher data.
409
- let initiatorContactCardCipherInput;
410
- let initiatorContactCardSharedCipherInput;
411
- if (initiatorRootKeyCipherClearJson.initiatorContactCard) {
412
- // The initiatorContactCard created during the creation of the invite and encrypted using the initiator's
413
- // root key
414
- const initiatorContactCardFromInit = initiatorRootKeyCipherClearJson.initiatorContactCard;
415
- const ownerKey = yield this.keyFactory.createKey();
416
- const sharedCipherKey = yield KeyFactoryService.asKey(plainInitiatorOneTimePbkCipher.responder.contactCard.sharedCipherKey);
417
- const ownerWrappedKey = JSON.stringify(yield this.encryptionService.encrypt(rootKey.jwk, ownerKey.toJSON(true)));
418
- // Allow the initiatorContactCard parameter to override
419
- const ownerCipherDataClearJson = (initiatorContactCard === null || initiatorContactCard === void 0 ? void 0 : initiatorContactCard.ownerCipherDataClearJson) ||
420
- initiatorContactCardFromInit.ownerCipherDataClearJson;
421
- const ownerCipherData = ownerCipherDataClearJson
422
- ? yield this.keyGraph.encryptToString(ownerKey, ownerCipherDataClearJson)
423
- : '';
424
- initiatorContactCardCipherInput = {
425
- ownerWrappedKey,
426
- ownerWrappingKeyId: rootKey.id,
427
- ownerCipherData,
428
- };
429
- initiatorContactCardSharedCipherInput = {
430
- sigPxkId: initiatorSigPrk.id,
431
- };
432
- const sharedCipherData = yield this.encryptionService.encrypt(sharedCipherKey, initiatorContactCardFromInit.sharedCipherDataClearJson);
433
- initiatorContactCardSharedCipherInput.sharedCipherDataSig = JSON.stringify(yield this.encryptionService.sign(initiatorSigPrk.jwk, sharedCipherData));
434
- }
435
- // TODO ideally we update the shared data in the contact card sent to the responder as well since that
436
- // CC was created by the responder.
437
- return new LrMutation({
438
- mutation: CompleteKeyExchangeOtkMutation,
439
- variables: {
440
- input: {
441
- keyExchangeId,
442
- rootKeyId: rootKey.id,
443
- masterKeyId: masterKey.id,
444
- initiatorSigPxkId: initiatorSigPrk.id,
445
- signedResponderPbk: JSON.stringify(signedResponderPbk),
446
- signedResponderSigPbk: JSON.stringify(signedResponderSigPbk),
447
- rkWrappedSharedKey: JSON.stringify(rkWrappedSharedKey),
448
- mkWrappedMkSharedKey: JSON.stringify(mkWrappedMkSharedKey),
449
- responderContactCardCipher: responderContactCardCipherInput,
450
- initiatorContactCardCipher: initiatorContactCardCipherInput,
451
- initiatorContactCardSharedCipher: initiatorContactCardSharedCipherInput,
452
- },
453
- },
454
- });
455
- });
456
- }
457
- };
458
- KeyExchange2Service.ɵprov = i0.ɵɵdefineInjectable({ factory: function KeyExchange2Service_Factory() { return new KeyExchange2Service(i0.ɵɵinject(i0.NgZone), i0.ɵɵinject(i1.KeyFactoryService), i0.ɵɵinject(i2.KeyService), i0.ɵɵinject(i3.EncryptionService), i0.ɵɵinject(i4.UserService), i0.ɵɵinject(i5.KeyGraphService), i0.ɵɵinject(i6.LrGraphQLService)); }, token: KeyExchange2Service, providedIn: "root" });
459
- KeyExchange2Service.decorators = [
460
- { type: Injectable, args: [{
461
- providedIn: 'root',
462
- },] }
463
- ];
464
- KeyExchange2Service.ctorParameters = () => [
465
- { type: NgZone },
466
- { type: KeyFactoryService },
467
- { type: KeyService },
468
- { type: EncryptionService },
469
- { type: UserService },
470
- { type: KeyGraphService },
471
- { type: LrGraphQLService }
472
- ];
473
- KeyExchange2Service = __decorate([
474
- RunOutsideAngular({
475
- ngZoneName: 'ngZone',
476
- })
477
- ], KeyExchange2Service);
478
- export { KeyExchange2Service };
479
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"key-exchange2.service.js","sourceRoot":"C:/Projects/kc-client/projects/core/src/","sources":["lib/api/key-exchange2.service.ts"],"names":[],"mappings":";AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,EAAE,MAAM,eAAe,CAAC;AAEnD,OAAO,EACL,iBAAiB,EACjB,iBAAiB,GAClB,MAAM,oCAAoC,CAAC;AAC5C,OAAO,EAAE,iBAAiB,EAAE,MAAM,qCAAqC,CAAC;AACxE,OAAO,EAAE,eAAe,EAAE,MAAM,mCAAmC,CAAC;AACpE,OAAO,EAAE,UAAU,EAAE,MAAM,6BAA6B,CAAC;AACzD,OAAO,EAAE,uBAAuB,EAAE,MAAM,uBAAuB,CAAC;AAChE,OAAO,EAAE,iBAAiB,EAAE,MAAM,gCAAgC,CAAC;AACnE,OAAO,EAAE,WAAW,EAAE,MAAM,yBAAyB,CAAC;AAEtD,OAAO,EACL,yBAAyB,EACzB,8BAA8B,EAC9B,0BAA0B,EAC1B,0BAA0B,EAC1B,8BAA8B,EAE9B,iBAAiB,EACjB,kBAAkB,EAElB,sBAAsB,EACtB,6BAA6B,GAC9B,MAAM,qBAAqB,CAAC;AAC7B,OAAO,EAAE,gBAAgB,EAAE,UAAU,EAAE,MAAM,cAAc,CAAC;;;;;;;;IA6H/C,mBAAmB,SAAnB,mBAAmB;IAG9B,YACU,MAAc,EACd,UAA6B,EAC7B,UAAsB,EACtB,iBAAoC,EACpC,WAAwB,EACxB,QAAyB,EACzB,SAA2B;QAN3B,WAAM,GAAN,MAAM,CAAQ;QACd,eAAU,GAAV,UAAU,CAAmB;QAC7B,eAAU,GAAV,UAAU,CAAY;QACtB,sBAAiB,GAAjB,iBAAiB,CAAmB;QACpC,gBAAW,GAAX,WAAW,CAAa;QACxB,aAAQ,GAAR,QAAQ,CAAiB;QACzB,cAAS,GAAT,SAAS,CAAkB;QATpB,wBAAmB,GAAG,EAAE,CAAC;IAUvC,CAAC;IAEU,QAAQ,CACpB,WAAgC,EAChC,MAAe;;YAEf,IAAI,MAAM,EAAE;gBACV,OAAO,MAAM,iBAAiB,CAAC,KAAK,iCAC/B,IAAI,CAAC,KAAK,CAAC,WAAW,CAAC,GAAG,CAAC,WAAW,CAAC,KAC1C,CAAC,EAAE,MAAM,IACT,CAAC;aACJ;iBAAM,IACL,WAAW,CAAC,GAAG,CAAC,KAAK,KAAK,eAAe;gBACzC,CAAC,WAAW,CAAC,WAAW;gBACxB,WAAW,CAAC,GAAG,CAAC,kBAAkB,EAClC;gBACA,yFAAyF;gBACzF,MAAM,GAAG,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,aAAa,EAAE,CAAC;gBAClD,MAAM,eAAe,GAAQ,MAAM,IAAI,CAAC,iBAAiB,CAAC,OAAO,CAC/D,GAAG,CAAC,GAAG,EACP,IAAI,CAAC,KAAK,CAAC,WAAW,CAAC,GAAG,CAAC,kBAAkB,CAAC,EAC9C;oBACE,cAAc,EAAE,CAAC,iBAAiB,CAAC,OAAO,CAAC;iBAC5C,CACF,CAAC;gBACF,IAAI,eAAe,CAAC,KAAK,EAAE;oBACzB,OAAO,MAAM,iBAAiB,CAAC,KAAK,CAAC,eAAe,CAAC,KAAK,CAAC,CAAC;iBAC7D;aACF;YACD,OAAO,IAAI,CAAC;QACd,CAAC;KAAA;IAEa,UAAU,CACtB,WAAgC,EAChC,MAAe;;YAEf,MAAM,KAAK,GAAG,MAAM,IAAI,CAAC,QAAQ,CAAC,WAAW,EAAE,MAAM,CAAC,CAAC;YAEvD,IAAI,GAAG,GAAG,WAAW,CAAC,GAAG,CAAC;YAE1B,IAAI,KAAK,IAAI,GAAG,CAAC,WAAW,EAAE;gBAC5B,GAAG,mCACE,GAAG,KACN,KAAK,EACL,oBAAoB,EAAE,MAAM,IAAI,CAAC,iBAAiB,CAAC,OAAO,CACxD,KAAK,EACL,WAAW,CAAC,GAAG,CAAC,WAAW,CAC5B,GACF,CAAC;aACH;YAED,uCACK,WAAW,KACd,GAAG,IACH;QACJ,CAAC;KAAA;IAEa,qBAAqB,CACjC,KAAc,EACd,KAAc,EACd,OAAY;;YAEZ,oEAAoE;YACpE,IAAI;gBACF,OAAO,GAAG,MAAM,IAAI,CAAC,iBAAiB,CAAC,OAAO,CAAC,KAAK,EAAE,OAAO,CAAC,CAAC;aAChE;YAAC,OAAO,KAAK,EAAE;gBACd,IAAI,KAAK,CAAC,OAAO,KAAK,cAAc,EAAE;oBACpC,MAAM,KAAK,CAAC;iBACb;gBACD,8EAA8E;aAC/E;YAED,oGAAoG;YACpG,OAAO,MAAM,IAAI,CAAC,iBAAiB,CAAC,OAAO,CAAC,KAAK,EAAE,OAAO,CAAC,CAAC;QAC9D,CAAC;KAAA;IAEa,6BAA6B,CACzC,WAAgC;;YAEhC,MAAM,OAAO,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,iBAAiB,EAAE,CAAC;YAE1D,4CAA4C;YAC5C,MAAM,+BAA+B,GAAG,CAAC,MAAM,IAAI,CAAC,iBAAiB,CAAC,OAAO,CAC3E,OAAO,CAAC,GAAG,EACX,WAAW,CAAC,sBAAsB,CACnC,CAAqC,CAAC;YAEvC,MAAM,KAAK,GAAG,MAAM,iBAAiB,CAAC,KAAK,CACzC,+BAA+B,CAAC,KAAK,CACtC,CAAC;YAEF,WAAW,mCACN,WAAW,KACd,+BAA+B,GAChC,CAAC;YAEF,IAAI,GAAG,GAAG,WAAW,CAAC,GAAG,CAAC;YAE1B,IAAI,GAAG,CAAC,yBAAyB,EAAE;gBACjC,GAAG,mCACE,GAAG,KACN,kCAAkC,EAAE,MAAM,IAAI,CAAC,qBAAqB,CAClE,KAAK,EACL,MAAM,iBAAiB,CAAC,KAAK,CAC3B,+BAA+B,CAAC,UAAU,CAC3C,EACD,GAAG,CAAC,yBAAyB,CAC9B,GACF,CAAC;aACH;YAED,IAAI,GAAG,CAAC,WAAW,EAAE;gBACnB,GAAG,CAAC,oBAAoB,GAAG,MAAM,IAAI,CAAC,iBAAiB,CAAC,OAAO,CAC7D,KAAK,EACL,GAAG,CAAC,WAAW,CAChB,CAAC;aACH;YAED,uCACK,WAAW,KACd,GAAG,IACH;QACJ,CAAC;KAAA;IAEa,6BAA6B,CACzC,WAAgC,EAChC,MAAe;;YAEf,OAAO,IAAI,CAAC,UAAU,CAAC,WAAW,EAAE,MAAM,CAAC,CAAC;QAC9C,CAAC;KAAA;IAEK,kBAAkB,CAAC,WAAgC,EAAE,MAAe;;YACxE,IAAI,WAAW,CAAC,WAAW,EAAE;gBAC3B,OAAO,IAAI,CAAC,6BAA6B,CAAC,WAAW,CAAC,CAAC;aACxD;iBAAM;gBACL,OAAO,IAAI,CAAC,6BAA6B,CAAC,WAAW,EAAE,MAAM,CAAC,CAAC;aAChE;QACH,CAAC;KAAA;IAEK,eAAe,CAAC,EAAE,KAAK,KAAoC,EAAE;;YACjE,MAAM,EAAE,YAAY,EAAE,GAAG,MAAM,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC;gBAClD,KAAK,EAAE,kBAAkB;gBACzB,SAAS,EAAE;oBACT,KAAK;iBACN;aACF,CAAC,CAAC;YAEH,OAAO,OAAO,CAAC,GAAG,CAChB,YAAY,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CACrE,CAAC;QACJ,CAAC;KAAA;IAED;;;;;OAKG;IACG,cAAc,CAClB,EAAU,EACV,EAAE,MAAM,EAAE,KAAK,KAA6B,EAAE;;YAE9C,MAAM,GAAG,GAAG,MAAM,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC;gBACrC,KAAK,EAAE,KAAK,CAAC,CAAC,CAAC,sBAAsB,CAAC,CAAC,CAAC,iBAAiB;gBACzD,SAAS,EAAE;oBACT,EAAE;oBACF,KAAK;iBACN;gBACD,eAAe,EAAE,CAAC,KAAK;aACxB,CAAC,CAAC;YACH,OAAO,IAAI,CAAC,kBAAkB,CAAC,GAAG,CAAC,WAAW,EAAE,MAAM,CAAC,CAAC;QAC1D,CAAC;KAAA;IAEY,uBAAuB,CAAC,KAGpC;;YACC,OAAO,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC;gBAC1B,KAAK,EAAE,0BAA0B;gBACjC,SAAS,EAAE;oBACT,QAAQ,EAAE,KAAK,CAAC,QAAQ;oBACxB,MAAM,EAAE,KAAK,CAAC,MAAM;iBACrB;aACF,CAAC,CAAC;QACL,CAAC;KAAA;IAED,uBAAuB;IACjB,iBAAiB,CAAC,EAAU;;YAChC,OAAO,IAAI,CAAC,yBAAyB,CAAC,EAAE,CAAC,CAAC;QAC5C,CAAC;KAAA;IAED,yBAAyB,CAAC,EAAU;QAClC,OAAO,IAAI,UAAU,CAAC;YACpB,QAAQ,EAAE,yBAAyB;YACnC,SAAS,EAAE;gBACT,KAAK,EAAE;oBACL,EAAE;iBACH;aACF;SACF,CAAC,CAAC;IACL,CAAC;IAED,uBAAuB;IACjB,kBAAkB,CAAC,EAAU,EAAE,KAAa;;YAChD,OAAO,IAAI,CAAC,0BAA0B,CAAC,EAAE,EAAE,KAAK,CAAC,CAAC;QACpD,CAAC;KAAA;IAED,0BAA0B,CAAC,EAAU,EAAE,KAAa;QAClD,OAAO,IAAI,UAAU,CAAC;YACpB,QAAQ,EAAE,0BAA0B;YACpC,SAAS,EAAE;gBACT,KAAK,EAAE;oBACL,EAAE;oBACF,KAAK;iBACN;aACF;SACF,CAAC,CAAC;IACL,CAAC;IAEK,WAAW,CAAC,EAChB,OAAO,EACP,KAAK,EACL,WAAW,EACX,OAAO,MACc,EAAE;;YACvB,MAAM,KAAK,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,SAAS,EAAE,CAAC;YAChD,MAAM,KAAK,GAAG,IAAI,CAAC,UAAU,CAAC,YAAY,CAAC,IAAI,CAAC,mBAAmB,CAAC,CAAC;YAErE,kFAAkF;YAClF,iCAAiC;YACjC,MAAM,mBAAmB,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,YAAY,EAAE,CAAC;YAEjE,oCAAoC;YACpC,oEAAoE;YAEpE,+CAA+C;YAC/C,sDAAsD;YACtD,MAAM,YAAY,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,aAAa,EAAE,CAAC;YAC3D,MAAM,eAAe,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,gBAAgB,EAAE,CAAC;YAEjE,IAAI,qBAAqB,GAAW,IAAI,CAAC;YAEzC,IAAI,WAAW,IAAI,WAAW,CAAC,kBAAkB,EAAE;gBACjD,qBAAqB,GAAG,MAAM,IAAI,CAAC,iBAAiB,CAAC,YAAY,CAC/D,eAAe,CAAC,GAAG,EACnB,WAAW,CAAC,kBAAkB,CAC/B,CAAC;aACH;YAED,MAAM,SAAS,GAAG;gBAChB,OAAO;gBACP,WAAW,EAAE,WAAW,IAAI;oBAC1B,yBAAyB,EAAE,WAAW,CAAC,yBAAyB;iBACjE;aACF,CAAC;YAEF,yCAAyC;YACzC,MAAM,gBAAgB,GAA0B;gBAC9C,oFAAoF;gBACpF,gFAAgF;gBAChF,wFAAwF;gBACxF,kFAAkF;gBAClF,yBAAyB;gBACzB,KAAK;gBACL,SAAS,kCACJ,SAAS,KACZ,UAAU,EAAE,mBAAmB,CAAC,MAAM,EAAE,EACxC,GAAG,EAAE,YAAY,CAAC,GAAG,CAAC,MAAM,EAAE,EAC9B,MAAM,EAAE,eAAe,CAAC,GAAG,CAAC,MAAM,EAAE,GACrC;aACF,CAAC;YAEF,MAAM,WAAW,GAAG,MAAM,IAAI,CAAC,QAAQ,CAAC,eAAe,CACrD,KAAK,EACL,gBAAgB,CACjB,CAAC;YAEF,0DAA0D;YAC1D,MAAM,+BAA+B,GAAqC;gBACxE,KAAK;gBACL,UAAU,EAAE,mBAAmB,CAAC,MAAM,CAAC,IAAI,CAAC;gBAC5C,oFAAoF;gBACpF,wCAAwC;gBAExC,gEAAgE;gBAChE,wEAAwE;gBACxE,+EAA+E;gBAC/E,KAAK,EAAE,KAAK,CAAC,MAAM,CAAC,IAAI,CAAC;gBACzB,2FAA2F;gBAC3F,wCAAwC;gBACxC,oBAAoB,EAAE,WAAW;gBACjC,SAAS;aACV,CAAC;YAEF,MAAM,OAAO,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,iBAAiB,EAAE,CAAC;YAC1D,MAAM,sBAAsB,GAAG,MAAM,IAAI,CAAC,QAAQ,CAAC,eAAe,CAChE,OAAO,CAAC,GAAG,EACX,+BAA+B,CAChC,CAAC;YAEF,cAAc;YACd,MAAM,MAAM,GAAY,KAAK,CAAC,MAAM,CAAC,IAAI,CAAS,CAAC,CAAC,CAAC;YAErD,WAAW;YACX,MAAM,UAAU,GAAG,IAAI,UAAU,CAAC;gBAChC,QAAQ,EAAE,8BAA8B;gBACxC,SAAS,EAAE;oBACT,KAAK,EAAE;wBACL,qCAAqC;wBACrC,sBAAsB;wBACtB,cAAc,EAAE,YAAY,CAAC,EAAE;wBAC/B,iBAAiB,EAAE,eAAe,CAAC,EAAE;wBACrC,sCAAsC;wBACtC,WAAW,EAAE,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,MAAM,EAAE,CAAC;wBAC3C,WAAW;wBACX,SAAS,EAAE,KAAK,IAAI;4BAClB,KAAK;4BACL,QAAQ,EAAE,MAAM;yBACjB;wBACD,QAAQ,EAAE,IAAI;wBACd,qBAAqB;wBACrB,OAAO;qBACR;iBACF;aACF,CAAC,CAAC;YAEH,OAAO,EAAE,UAAU,EAAE,MAAM,EAAE,CAAC;QAChC,CAAC;KAAA;IAEK,UAAU,CAAC,EACf,aAAa,EACb,KAAK,EACL,YAAY,EACZ,OAAO,EACP,oBAAoB,EACpB,oBAAoB,GACH;;YACjB,MAAM,OAAO,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,iBAAiB,EAAE,CAAC;YAE1D,MAAM,WAAW,GAAG,IAAI,CAAC,UAAU,CAAC,mBAAmB,EAAE,CAAC,EAAE,CAAC;YAC7D,MAAM,SAAS,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,mBAAmB,EAAE,CAAC;YAE9D,MAAM,SAAS,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,SAAS,EAAE,CAAC;YACpD,MAAM,WAAW,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,SAAS,EAAE,CAAC;YAEtD,MAAM,kBAAkB,GAAG,MAAM,IAAI,CAAC,iBAAiB,CAAC,OAAO,CAC7D,OAAO,CAAC,GAAG,EACX,SAAS,CAAC,MAAM,CAAC,IAAI,CAAC,CACvB,CAAC;YACF,MAAM,oBAAoB,GAAG,MAAM,IAAI,CAAC,iBAAiB,CAAC,OAAO,CAC/D,SAAS,CAAC,GAAG,EACb,WAAW,CAAC,MAAM,CAAC,IAAI,CAAC,CACzB,CAAC;YAEF,MAAM,mBAAmB,GAAG,MAAM,iBAAiB,CAAC,KAAK,CACvD,YAAY,CAAC,oBAAoB,CAAC,SAAS,CAAC,UAAU,CACvD,CAAC;YAEF,MAAM,YAAY,GAAG,MAAM,iBAAiB,CAAC,KAAK,CAChD,YAAY,CAAC,oBAAoB,CAAC,SAAS,CAAC,GAAG,CAChD,CAAC;YACF,MAAM,eAAe,GAAG,MAAM,iBAAiB,CAAC,KAAK,CACnD,YAAY,CAAC,oBAAoB,CAAC,SAAS,CAAC,MAAM,CACnD,CAAC;YAEF,2CAA2C;YAC3C,qDAAqD;YACrD,mEAAmE;YACnE,8FAA8F;YAE9F,gDAAgD;YAChD,MAAM,YAAY,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,aAAa,EAAE,CAAC;YAC3D,MAAM,eAAe,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,gBAAgB,EAAE,CAAC;YAEjE,MAAM,kBAAkB,GAAG,MAAM,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAC1D,eAAe,CAAC,GAAG,EACnB,YAAY,CAAC,MAAM,EAAE,CACtB,CAAC;YACF,MAAM,qBAAqB,GAAG,MAAM,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAC7D,eAAe,CAAC,GAAG,EACnB,eAAe,CAAC,MAAM,EAAE,CACzB,CAAC;YAEF,MAAM,kCAAkC,GAAuC;gBAC7E,KAAK,EAAE,YAAY,CAAC,oBAAoB,CAAC,KAAK;gBAC9C,SAAS,EAAE,SAAS,CAAC,MAAM,CAAC,IAAI,CAAC;gBACjC,WAAW,EAAE,WAAW,CAAC,MAAM,CAAC,IAAI,CAAC;gBACrC,SAAS,EAAE;oBACT,GAAG,EAAE,YAAY,CAAC,GAAG,CAAC,MAAM,EAAE;oBAC9B,MAAM,EAAE,eAAe,CAAC,GAAG,CAAC,MAAM,EAAE;oBACpC,OAAO;iBACR;aACF,CAAC;YAEF,IAAI,iBAAiB,CAAC;YACtB,IAAI,YAAY,CAAC,oBAAoB,CAAC,SAAS,CAAC,WAAW,EAAE;gBAC3D,wHAAwH;gBACxH,4HAA4H;gBAC5H,4HAA4H;gBAC5H,MAAM,yBAAyB,GAC7B,YAAY,CAAC,oBAAoB,CAAC,SAAS,CAAC,WAAW;qBACpD,yBAAyB,CAAC;gBAE/B,cAAc;gBACd,MAAM,WAAW,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,SAAS,EAAE,CAAC;gBACtD,MAAM,WAAW,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,SAAS,EAAE,CAAC;gBACtD,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,gBAAgB,EAAE,CAAC;gBAExD,iBAAiB,GAAG;oBAClB,kBAAkB,EAAE,IAAI,CAAC,SAAS,CAChC,MAAM,IAAI,CAAC,iBAAiB,CAAC,OAAO,CAClC,OAAO,CAAC,GAAG,EACX,WAAW,CAAC,MAAM,CAAC,IAAI,CAAC,CACzB,CACF;oBACD,qBAAqB,EAAE,OAAO,CAAC,EAAE;oBACjC,kBAAkB,EAAE,oBAAoB;wBACtC,CAAC,CAAC,IAAI,CAAC,SAAS,CACZ,MAAM,IAAI,CAAC,iBAAiB,CAAC,OAAO,CAClC,WAAW,EACX,oBAAoB,CAAC,2BAA2B,CACjD,CACF;wBACH,CAAC,CAAC,EAAE;oBACN,gBAAgB,EAAE,IAAI,CAAC,SAAS,CAC9B,MAAM,IAAI,CAAC,iBAAiB,CAAC,OAAO,CAClC,SAAS,EACT,WAAW,CAAC,MAAM,CAAC,IAAI,CAAC,CACzB,CACF;iBACF,CAAC;gBAEF,MAAM,gBAAgB,GAAG,MAAM,IAAI,CAAC,iBAAiB,CAAC,OAAO,CAC3D,WAAW,EACX,yBAAyB,CAC1B,CAAC;gBACF,iBAAiB,CAAC,mBAAmB,GAAG,IAAI,CAAC,SAAS,CACpD,MAAM,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,MAAM,CAAC,GAAG,EAAE,gBAAgB,CAAC,CAChE,CAAC;gBACF,iBAAiB,CAAC,QAAQ,GAAG,MAAM,CAAC,EAAE,CAAC;gBAEvC,kCAAkC,CAAC,SAAS,CAAC,WAAW,mCACnD,kCAAkC,CAAC,SAAS,CAAC,WAAW,KAC3D,eAAe,EAAE,WAAW,CAAC,MAAM,CAAC,IAAI,CAAC,GAC1C,CAAC;aACH;YAED,IAAI,kBAAkB,CAAC;YACvB,IAAI,oBAAoB,EAAE;gBACxB,cAAc;gBACd,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,SAAS,EAAE,CAAC;gBACnD,MAAM,WAAW,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,SAAS,EAAE,CAAC;gBACtD,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,gBAAgB,EAAE,CAAC;gBAExD,kBAAkB,GAAG;oBACnB,eAAe,EAAE,IAAI,CAAC,SAAS,CAC7B,MAAM,IAAI,CAAC,iBAAiB,CAAC,OAAO,CAClC,OAAO,CAAC,GAAG,EACX,QAAQ,CAAC,MAAM,CAAC,IAAI,CAAC,CACtB,CACF;oBACD,kBAAkB,EAAE,OAAO,CAAC,EAAE;oBAC9B,eAAe,EAAE,oBAAoB,CAAC,wBAAwB;wBAC5D,CAAC,CAAC,IAAI,CAAC,SAAS,CACZ,MAAM,IAAI,CAAC,iBAAiB,CAAC,OAAO,CAClC,QAAQ,EACR,oBAAoB,CAAC,wBAAwB,CAC9C,CACF;wBACH,CAAC,CAAC,EAAE;oBAEN,gBAAgB,EAAE,IAAI,CAAC,SAAS,CAC9B,MAAM,IAAI,CAAC,iBAAiB,CAAC,OAAO,CAClC,SAAS,EACT,WAAW,CAAC,MAAM,CAAC,IAAI,CAAC,CACzB,CACF;iBACF,CAAC;gBAEF,MAAM,gBAAgB,GAAG,MAAM,IAAI,CAAC,iBAAiB,CAAC,OAAO,CAC3D,WAAW,EACX,oBAAoB,CAAC,yBAAyB,CAC/C,CAAC;gBACF,kBAAkB,CAAC,mBAAmB,GAAG,IAAI,CAAC,SAAS,CACrD,MAAM,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,MAAM,CAAC,GAAG,EAAE,gBAAgB,CAAC,CAChE,CAAC;gBACF,kBAAkB,CAAC,QAAQ,GAAG,MAAM,CAAC,EAAE,CAAC;gBAExC,IAAI,oBAAoB,CAAC,kBAAkB,EAAE;oBAC3C,kBAAkB,CAAC,iBAAiB,GAAG,IAAI,CAAC,SAAS,CACnD,MAAM,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAC/B,eAAe,CAAC,GAAG,EACnB,oBAAoB,CAAC,kBAAkB,CACxC,CACF,CAAC;iBACH;gBAED,8CAA8C;gBAC9C,kCAAkC,CAAC,SAAS,CAAC,WAAW,mCACnD,kCAAkC,CAAC,SAAS,CAAC,WAAW,KAC3D,yBAAyB,EACvB,oBAAoB,CAAC,yBAAyB,GACjD,CAAC;aACH;YAED,mCAAmC;YACnC,IAAI,yBAAyB,GAAG,MAAM,IAAI,CAAC,iBAAiB,CAAC,OAAO,CAClE,mBAAmB,EACnB,kCAAkC,CACnC,CAAC;YAEF,0EAA0E;YAC1E,yBAAyB,GAAG,MAAM,IAAI,CAAC,iBAAiB,CAAC,OAAO,CAC9D,YAAY,CAAC,KAAK,EAClB,yBAAyB,CAC1B,CAAC;YAEF,OAAO,IAAI,UAAU,CAAC;gBACpB,QAAQ,EAAE,6BAA6B;gBACvC,SAAS,EAAE;oBACT,KAAK,EAAE;wBACL,aAAa;wBACb,gBAAgB,EAAE,KAAK;wBACvB,SAAS,EAAE,OAAO,CAAC,EAAE;wBACrB,WAAW;wBACX,qCAAqC;wBACrC,cAAc,EAAE,YAAY,CAAC,EAAE;wBAC/B,iBAAiB,EAAE,eAAe,CAAC,EAAE;wBACrC,kBAAkB,EAAE,IAAI,CAAC,SAAS,CAAC,kBAAkB,CAAC;wBACtD,qBAAqB,EAAE,IAAI,CAAC,SAAS,CAAC,qBAAqB,CAAC;wBAC5D,sEAAsE;wBAEtE,2CAA2C;wBAC3C,sEAAsE;wBACtE,kBAAkB,EAAE,IAAI,CAAC,SAAS,CAAC,kBAAkB,CAAC;wBACtD,oBAAoB,EAAE,IAAI,CAAC,SAAS,CAAC,oBAAoB,CAAC;wBAC1D,sCAAsC;wBACtC,yBAAyB,EAAE,IAAI,CAAC,SAAS,CAAC,yBAAyB,CAAC;wBACpE,oBAAoB,EAAE,iBAAiB;wBACvC,oBAAoB,EAAE,kBAAkB;qBACzC;iBACF;aACF,CAAC,CAAC;QACL,CAAC;KAAA;IAEK,WAAW,CAAC,EAChB,aAAa,EACb,sBAAsB,EACtB,yBAAyB,EACzB,oBAAoB,EACpB,oBAAoB,GACF;;YAClB,MAAM,OAAO,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,iBAAiB,EAAE,CAAC;YAC1D,MAAM,SAAS,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,mBAAmB,EAAE,CAAC;YAE9D,4CAA4C;YAC5C,MAAM,+BAA+B,GAAG,CAAC,MAAM,IAAI,CAAC,iBAAiB,CAAC,OAAO,CAC3E,OAAO,CAAC,GAAG,EACX,sBAAsB,CACvB,CAAqC,CAAC;YAEvC,oGAAoG;YACpG,MAAM,8BAA8B,GAAG,MAAM,IAAI,CAAC,qBAAqB,CACrE,MAAM,iBAAiB,CAAC,KAAK,CAAC,+BAA+B,CAAC,KAAK,CAAC,EACpE,MAAM,iBAAiB,CAAC,KAAK,CAAC,+BAA+B,CAAC,UAAU,CAAC,EACzE,yBAAyB,CAC1B,CAAC;YAEF,4EAA4E;YAC5E,IACE,+BAA+B,CAAC,KAAK;gBACrC,8BAA8B,CAAC,KAAK,EACpC;gBACA,MAAM,IAAI,uBAAuB,CAC/B,uFAAuF,CACxF,CAAC;aACH;YAED,8DAA8D;YAC9D,kFAAkF;YAClF,8FAA8F;YAE9F,+CAA+C;YAC/C,uEAAuE;YACvE,iCAAiC;YAEjC,qDAAqD;YACrD,MAAM,eAAe,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,gBAAgB,EAAE,CAAC;YACjE,MAAM,eAAe,GAAG,MAAM,iBAAiB,CAAC,KAAK,CACnD,8BAA8B,CAAC,SAAS,CAAC,MAAM,CAChD,CAAC;YACF,MAAM,YAAY,GAAG,MAAM,iBAAiB,CAAC,KAAK,CAChD,8BAA8B,CAAC,SAAS,CAAC,GAAG,CAC7C,CAAC;YAEF,MAAM,kBAAkB,GAAG,MAAM,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAC1D,eAAe,CAAC,GAAG,EACnB,YAAY,CAAC,MAAM,EAAE,CACtB,CAAC;YACF,MAAM,qBAAqB,GAAG,MAAM,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAC7D,eAAe,CAAC,GAAG,EACnB,eAAe,CAAC,MAAM,EAAE,CACzB,CAAC;YAEF,MAAM,SAAS,GAAG,MAAM,iBAAiB,CAAC,KAAK,CAC7C,8BAA8B,CAAC,SAAS,CACzC,CAAC;YACF,MAAM,kBAAkB,GAAG,MAAM,IAAI,CAAC,iBAAiB,CAAC,OAAO,CAC7D,OAAO,CAAC,GAAG,EACX,SAAS,CAAC,MAAM,CAAC,IAAI,CAAC,CACvB,CAAC;YAEF,MAAM,WAAW,GAAG,MAAM,iBAAiB,CAAC,KAAK,CAC/C,8BAA8B,CAAC,WAAW,CAC3C,CAAC;YACF,MAAM,oBAAoB,GAAG,MAAM,IAAI,CAAC,iBAAiB,CAAC,OAAO,CAC/D,SAAS,CAAC,GAAG,EACb,WAAW,CAAC,MAAM,CAAC,IAAI,CAAC,CACzB,CAAC;YAEF,IAAI,+BAA+B,CAAC;YACpC,IAAI,oBAAoB,EAAE;gBACxB,aAAa;gBACb,MAAM,WAAW,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,SAAS,EAAE,CAAC;gBAEtD,+BAA+B,GAAG;oBAChC,kBAAkB,EAAE,IAAI,CAAC,SAAS,CAChC,MAAM,IAAI,CAAC,iBAAiB,CAAC,OAAO,CAClC,OAAO,CAAC,GAAG,EACX,WAAW,CAAC,MAAM,CAAC,IAAI,CAAC,CACzB,CACF;oBACD,qBAAqB,EAAE,OAAO,CAAC,EAAE;oBACjC,kBAAkB,EAAE,IAAI,CAAC,SAAS,CAChC,MAAM,IAAI,CAAC,iBAAiB,CAAC,OAAO,CAClC,WAAW,EACX,oBAAoB,CAAC,2BAA2B,CACjD,CACF;iBACF,CAAC;aACH;YAED,wDAAwD;YACxD,IAAI,+BAA+B,CAAC;YACpC,IAAI,qCAAqC,CAAC;YAC1C,IAAI,+BAA+B,CAAC,oBAAoB,EAAE;gBACxD,yGAAyG;gBACzG,WAAW;gBACX,MAAM,4BAA4B,GAChC,+BAA+B,CAAC,oBAAoB,CAAC;gBACvD,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,SAAS,EAAE,CAAC;gBACnD,MAAM,eAAe,GAAG,MAAM,iBAAiB,CAAC,KAAK,CACnD,8BAA8B,CAAC,SAAS,CAAC,WAAW,CAAC,eAAe,CACrE,CAAC;gBAEF,MAAM,eAAe,GAAG,IAAI,CAAC,SAAS,CACpC,MAAM,IAAI,CAAC,iBAAiB,CAAC,OAAO,CAAC,OAAO,CAAC,GAAG,EAAE,QAAQ,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CACzE,CAAC;gBAEF,uDAAuD;gBACvD,MAAM,wBAAwB,GAC5B,CAAA,oBAAoB,aAApB,oBAAoB,uBAApB,oBAAoB,CAAE,wBAAwB;oBAC9C,4BAA4B,CAAC,wBAAwB,CAAC;gBAExD,MAAM,eAAe,GAAG,wBAAwB;oBAC9C,CAAC,CAAC,MAAM,IAAI,CAAC,QAAQ,CAAC,eAAe,CACjC,QAAQ,EACR,wBAAwB,CACzB;oBACH,CAAC,CAAC,EAAE,CAAC;gBAEP,+BAA+B,GAAG;oBAChC,eAAe;oBACf,kBAAkB,EAAE,OAAO,CAAC,EAAE;oBAC9B,eAAe;iBAChB,CAAC;gBAEF,qCAAqC,GAAG;oBACtC,QAAQ,EAAE,eAAe,CAAC,EAAE;iBAC7B,CAAC;gBAEF,MAAM,gBAAgB,GAAG,MAAM,IAAI,CAAC,iBAAiB,CAAC,OAAO,CAC3D,eAAe,EACf,4BAA4B,CAAC,yBAAyB,CACvD,CAAC;gBACF,qCAAqC,CAAC,mBAAmB,GAAG,IAAI,CAAC,SAAS,CACxE,MAAM,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,eAAe,CAAC,GAAG,EAAE,gBAAgB,CAAC,CACzE,CAAC;aACH;YAED,sGAAsG;YACtG,mCAAmC;YAEnC,OAAO,IAAI,UAAU,CAAC;gBACpB,QAAQ,EAAE,8BAA8B;gBACxC,SAAS,EAAE;oBACT,KAAK,EAAE;wBACL,aAAa;wBACb,SAAS,EAAE,OAAO,CAAC,EAAE;wBACrB,WAAW,EAAE,SAAS,CAAC,EAAE;wBACzB,iBAAiB,EAAE,eAAe,CAAC,EAAE;wBACrC,kBAAkB,EAAE,IAAI,CAAC,SAAS,CAAC,kBAAkB,CAAC;wBACtD,qBAAqB,EAAE,IAAI,CAAC,SAAS,CAAC,qBAAqB,CAAC;wBAC5D,kBAAkB,EAAE,IAAI,CAAC,SAAS,CAAC,kBAAkB,CAAC;wBACtD,oBAAoB,EAAE,IAAI,CAAC,SAAS,CAAC,oBAAoB,CAAC;wBAC1D,0BAA0B,EAAE,+BAA+B;wBAC3D,0BAA0B,EAAE,+BAA+B;wBAC3D,gCAAgC,EAAE,qCAAqC;qBACxE;iBACF;aACF,CAAC,CAAC;QACL,CAAC;KAAA;CACF,CAAA;;;YAttBA,UAAU,SAAC;gBACV,UAAU,EAAE,MAAM;aACnB;;;YAtJoB,MAAM;YAMlB,iBAAiB;YAEjB,UAAU;YALjB,iBAAiB;YAQV,WAAW;YAJX,eAAe;YAmBf,gBAAgB;;AA6HZ,mBAAmB;IAN/B,iBAAiB,CAAC;QACjB,UAAU,EAAE,QAAQ;KACrB,CAAC;GAIW,mBAAmB,CAmtB/B;SAntBY,mBAAmB","sourcesContent":["import { Injectable, NgZone } from '@angular/core';\r\nimport { JWK } from 'node-jose';\r\nimport {\r\n  EncryptionService,\r\n  JoseSerialization,\r\n} from '../cryptography/encryption.service';\r\nimport { KeyFactoryService } from '../cryptography/key-factory.service';\r\nimport { KeyGraphService } from '../cryptography/key-graph.service';\r\nimport { KeyService } from '../cryptography/key.service';\r\nimport { LrCodeMismatchException } from '../_common/exceptions';\r\nimport { RunOutsideAngular } from '../_common/run-outside-angular';\r\nimport { UserService } from './../users/user.service';\r\nimport { OwnerPlainDataJson } from './contact-card2.service';\r\nimport {\r\n  CancelKeyExchangeMutation,\r\n  CompleteKeyExchangeOtkMutation,\r\n  CurrentUserSharedKeyQuery2,\r\n  DeclineKeyExchangeMutation,\r\n  InitiateKeyExchangeOtkMutation,\r\n  KeyExchangeFragment,\r\n  KeyExchangeQuery2,\r\n  KeyExchangesQuery2,\r\n  KeyExchangeState2,\r\n  KeyExchangeTokenQuery2,\r\n  RespondKeyExchangeOtkMutation,\r\n} from './key-exchange2.gql';\r\nimport { LrGraphQLService, LrMutation } from './lr-graphql';\r\nimport { ContactCardSharedCipherData, OtKeyCipherClearJson2 } from './types';\r\n\r\n/**\r\n * The decrypted content of the one-time key cipher;\r\n * When user supplies this information the lib doesn't need to do another API call\r\n * to fetch the key exchange node. And since the typical use case is to display\r\n * some information to the user, the key exchange node would already have been\r\n * fetched and decrypted.\r\n */\r\nexport interface DecryptedOtk2 {\r\n  otKeyCipherClearJson: OtKeyCipherClearJson2;\r\n  otKey: JWK.Key; // The one-time key\r\n}\r\n\r\nexport interface ContactCardReceiverCipherData {\r\n  // The receiver of the contact card keeps a copy\r\n  // of the owner's contact card information, encrypted using the receiver's keys. So that when\r\n  // the owner decides to update their shared contact card at a later date, the receiver can\r\n  // compare against the original contact card sent during key exchange. This way, the owner\r\n  // can't unilaterally update their shared contact card without the receiver knowing about it.\r\n  receiverCipherDataClearJson: any;\r\n}\r\n\r\nexport interface ContactCardOwnerPlainData {\r\n  // Accessible by the server and the owner. Server side notification emails need to know some\r\n  // information about the owner.\r\n  // The owner could either be the initiator or the responder.\r\n  ownerPlainDataJson: OwnerPlainDataJson;\r\n}\r\n\r\nexport interface ContactCardOwnerCipherData {\r\n  // Owner only access to this data\r\n  ownerCipherDataClearJson: any;\r\n}\r\n\r\n/**\r\n * Sending contact card information from the owner to the receiver.\r\n */\r\nexport type SendContactCardInput = ContactCardOwnerPlainData &\r\n  ContactCardOwnerCipherData &\r\n  ContactCardSharedCipherData;\r\n\r\nexport interface InitiateOtkInput2 {\r\n  // Note that if neither email nor responderUsername are given, the one-time key\r\n  // can still be sent to the responder via OOB\r\n  email?: string;\r\n  message?: any;\r\n  contactCard?: SendContactCardInput;\r\n  // If True, to upgrade an email invite to an existing user invite if the email\r\n  // is already associated with an existing user.\r\n  upgrade?: boolean;\r\n}\r\n\r\nexport interface RespondOtkInput2 {\r\n  keyExchangeId: string;\r\n  token: string;\r\n  decryptedOtk: DecryptedOtk2;\r\n  message?: any;\r\n  // The initiator is the owner fo the initiatorContactCard, the responder\r\n  // is the receiver.\r\n  initiatorContactCard?: ContactCardReceiverCipherData;\r\n  responderContactCard?: SendContactCardInput;\r\n}\r\n\r\nexport interface CompleteOtkInput2 {\r\n  keyExchangeId: string;\r\n  // This is a part of the key exchange data. It's encrypted using the root key\r\n  initiatorRootKeyCipher: string;\r\n  // This is a part of the key exchange data. It's encrypted using the one-time key.\r\n  initiatorOneTimePbkCipher: string;\r\n  // The responder is the owner fo the responderContactCard, the initiator\r\n  // is the receiver.\r\n  responderContactCard?: ContactCardReceiverCipherData;\r\n  // The initiator can update the cipher data that are only visible to them. It makes\r\n  // less sense to update the shared data because the responder would have already seen\r\n  // the shared data and accepted that it's legit.\r\n  // But in any case, the initiator can update the shared contact card info at any time.\r\n  initiatorContactCard?: ContactCardOwnerCipherData;\r\n}\r\n\r\nexport interface InitiatorRootKeyCipherClearJson2 {\r\n  nonce: string;\r\n  oneTimePrk: object; // one-time public encryption key responder use to send data back to initiator\r\n  otKey: object; // one-time symmetric key that needs to be shared OOB\r\n  initiatorContactCard?: ContactCardOwnerCipherData &\r\n    ContactCardSharedCipherData;\r\n  initiator: {\r\n    message?: any;\r\n    contactCard?: ContactCardSharedCipherData;\r\n  };\r\n}\r\n\r\nexport interface InitiatorOneTimePbkCipherClearJson {\r\n  nonce: string;\r\n  sharedKey: object;\r\n  mkSharedKey: object;\r\n  responder: {\r\n    pbk: object;\r\n    sigPbk: object;\r\n    message?: any;\r\n    contactCard?: ContactCardSharedCipherData & {\r\n      // Note that this is _not_ the same key as the sharedKey. The sharedKey wraps\r\n      // this key in the key graph. But because this key has not been entered into\r\n      // the key graph when the responder calls the API, we pass the JWK directly here.\r\n      sharedCipherKey: object;\r\n    };\r\n  };\r\n}\r\n\r\nexport interface GetKeyExchangeOptions2 {\r\n  // The otKey as a raw string. i.e. key.toJSON(true).k\r\n  otKeyK?: string;\r\n  // User need the token if they have not responded to the key exchange yet.\r\n  // Once they've responded (hence proven they have the OOB Key) they become\r\n  // the \"responder\" of this exchange, and can access it when signed in.\r\n  token?: string;\r\n}\r\n\r\n@RunOutsideAngular({\r\n  ngZoneName: 'ngZone',\r\n})\r\n@Injectable({\r\n  providedIn: 'root',\r\n})\r\nexport class KeyExchange2Service {\r\n  private readonly CLIENT_NONCE_LENGTH = 32;\r\n\r\n  constructor(\r\n    private ngZone: NgZone,\r\n    private keyFactory: KeyFactoryService,\r\n    private keyService: KeyService,\r\n    private encryptionService: EncryptionService,\r\n    private userService: UserService,\r\n    private keyGraph: KeyGraphService,\r\n    private lrGraphQL: LrGraphQLService\r\n  ) {}\r\n\r\n  private async getOtKey(\r\n    keyExchange: KeyExchangeFragment,\r\n    otKeyK?: string\r\n  ): Promise<JWK.Key> {\r\n    if (otKeyK) {\r\n      return await KeyFactoryService.asKey({\r\n        ...JSON.parse(keyExchange.otk.otKeyParams),\r\n        k: otKeyK,\r\n      });\r\n    } else if (\r\n      keyExchange.otk.state === 'OTK_INITIATED' &&\r\n      !keyExchange.isInitiator &&\r\n      keyExchange.otk.responderPbkCipher\r\n    ) {\r\n      // Assuming existing user getting invited where OTK is wrapped in responder's public key.\r\n      const prk = await this.keyService.getCurrentPxk();\r\n      const decryptedCipher: any = await this.encryptionService.decrypt(\r\n        prk.jwk,\r\n        JSON.parse(keyExchange.otk.responderPbkCipher),\r\n        {\r\n          serializations: [JoseSerialization.COMPACT],\r\n        }\r\n      );\r\n      if (decryptedCipher.otKey) {\r\n        return await KeyFactoryService.asKey(decryptedCipher.otKey);\r\n      }\r\n    }\r\n    return null;\r\n  }\r\n\r\n  private async decryptOtk(\r\n    keyExchange: KeyExchangeFragment,\r\n    otKeyK?: string\r\n  ): Promise<KeyExchangeFragment> {\r\n    const otKey = await this.getOtKey(keyExchange, otKeyK);\r\n\r\n    let otk = keyExchange.otk;\r\n\r\n    if (otKey && otk.otKeyCipher) {\r\n      otk = {\r\n        ...otk,\r\n        otKey,\r\n        otKeyCipherClearJson: await this.encryptionService.decrypt(\r\n          otKey,\r\n          keyExchange.otk.otKeyCipher\r\n        ),\r\n      };\r\n    }\r\n\r\n    return {\r\n      ...keyExchange,\r\n      otk,\r\n    };\r\n  }\r\n\r\n  private async decryptResponseCipher(\r\n    otKey: JWK.Key,\r\n    otPrk: JWK.Key,\r\n    content: any\r\n  ): Promise<InitiatorOneTimePbkCipherClearJson> {\r\n    // The response could be wrapped by the OtK in addition to the OtPbk\r\n    try {\r\n      content = await this.encryptionService.decrypt(otKey, content);\r\n    } catch (error) {\r\n      if (error.message !== 'no key found') {\r\n        throw error;\r\n      }\r\n      // Do nothing to support older versions where message is not wrapped with otk.\r\n    }\r\n\r\n    // The Prk is single-use and only used to send information from the responder back to the initiator.\r\n    return await this.encryptionService.decrypt(otPrk, content);\r\n  }\r\n\r\n  private async decryptKeyExchangeAsInitiator(\r\n    keyExchange: KeyExchangeFragment\r\n  ): Promise<KeyExchangeFragment> {\r\n    const rootKey = await this.keyService.getCurrentRootKey();\r\n\r\n    // Decrypt using the root key to get the Prk\r\n    const initiatorRootKeyCipherClearJson = (await this.encryptionService.decrypt(\r\n      rootKey.jwk,\r\n      keyExchange.initiatorRootKeyCipher\r\n    )) as InitiatorRootKeyCipherClearJson2;\r\n\r\n    const otKey = await KeyFactoryService.asKey(\r\n      initiatorRootKeyCipherClearJson.otKey\r\n    );\r\n\r\n    keyExchange = {\r\n      ...keyExchange,\r\n      initiatorRootKeyCipherClearJson,\r\n    };\r\n\r\n    let otk = keyExchange.otk;\r\n\r\n    if (otk.initiatorOneTimePbkCipher) {\r\n      otk = {\r\n        ...otk,\r\n        initiatorOneTimePbkCipherClearJson: await this.decryptResponseCipher(\r\n          otKey,\r\n          await KeyFactoryService.asKey(\r\n            initiatorRootKeyCipherClearJson.oneTimePrk\r\n          ),\r\n          otk.initiatorOneTimePbkCipher\r\n        ),\r\n      };\r\n    }\r\n\r\n    if (otk.otKeyCipher) {\r\n      otk.otKeyCipherClearJson = await this.encryptionService.decrypt(\r\n        otKey,\r\n        otk.otKeyCipher\r\n      );\r\n    }\r\n\r\n    return {\r\n      ...keyExchange,\r\n      otk,\r\n    };\r\n  }\r\n\r\n  private async decryptKeyExchangeAsResponder(\r\n    keyExchange: KeyExchangeFragment,\r\n    otKeyK?: string\r\n  ) {\r\n    return this.decryptOtk(keyExchange, otKeyK);\r\n  }\r\n\r\n  async decryptKeyExchange(keyExchange: KeyExchangeFragment, otKeyK?: string) {\r\n    if (keyExchange.isInitiator) {\r\n      return this.decryptKeyExchangeAsInitiator(keyExchange);\r\n    } else {\r\n      return this.decryptKeyExchangeAsResponder(keyExchange, otKeyK);\r\n    }\r\n  }\r\n\r\n  async getKeyExchanges({ state }: { state?: KeyExchangeState2 } = {}) {\r\n    const { keyExchanges } = await this.lrGraphQL.query({\r\n      query: KeyExchangesQuery2,\r\n      variables: {\r\n        state,\r\n      },\r\n    });\r\n\r\n    return Promise.all(\r\n      keyExchanges.edges.map((edge) => this.decryptKeyExchange(edge.node))\r\n    );\r\n  }\r\n\r\n  /**\r\n   * @param id If the current user can responder the key exchange if they are either the initiator or the receiver.\r\n   * @param token If not signed in, or not the initiator or responder, 'token' must be given.\r\n   * @param otKeyK Is the raw one-time key (string). If the responder is explicitly specified at time of initiation, then\r\n   *   it's possible to have the otKey wrapped by the public key of the responder. In which case, the otKeyK is not needed.\r\n   */\r\n  async getKeyExchange(\r\n    id: string,\r\n    { otKeyK, token }: GetKeyExchangeOptions2 = {}\r\n  ) {\r\n    const res = await this.lrGraphQL.query({\r\n      query: token ? KeyExchangeTokenQuery2 : KeyExchangeQuery2,\r\n      variables: {\r\n        id,\r\n        token,\r\n      },\r\n      includeKeyGraph: !token, // if !token then we are post auth, so can fetch keyGraph\r\n    });\r\n    return this.decryptKeyExchange(res.keyExchange, otKeyK);\r\n  }\r\n\r\n  public async getCurrentUserSharedKey(input: {\r\n    username?: string;\r\n    userId?: string;\r\n  }) {\r\n    return this.lrGraphQL.query({\r\n      query: CurrentUserSharedKeyQuery2,\r\n      variables: {\r\n        username: input.username,\r\n        userId: input.userId,\r\n      },\r\n    });\r\n  }\r\n\r\n  // TODO: deprecate this\r\n  async cancelKeyExchange(id: string) {\r\n    return this.cancelKeyExchangeMutation(id);\r\n  }\r\n\r\n  cancelKeyExchangeMutation(id: string) {\r\n    return new LrMutation({\r\n      mutation: CancelKeyExchangeMutation,\r\n      variables: {\r\n        input: {\r\n          id,\r\n        },\r\n      },\r\n    });\r\n  }\r\n\r\n  // TODO: deprecate this\r\n  async declineKeyExchange(id: string, token: string) {\r\n    return this.declineKeyExchangeMutation(id, token);\r\n  }\r\n\r\n  declineKeyExchangeMutation(id: string, token: string) {\r\n    return new LrMutation({\r\n      mutation: DeclineKeyExchangeMutation,\r\n      variables: {\r\n        input: {\r\n          id,\r\n          token,\r\n        },\r\n      },\r\n    });\r\n  }\r\n\r\n  async initiateOtk({\r\n    message,\r\n    email,\r\n    contactCard,\r\n    upgrade,\r\n  }: InitiateOtkInput2 = {}) {\r\n    const otKey = await this.keyFactory.createKey();\r\n    const nonce = this.keyFactory.randomString(this.CLIENT_NONCE_LENGTH);\r\n\r\n    // New PKC key for encryption. This key is used only once when the responder sends\r\n    // back their signing public key.\r\n    const initiatorOneTimePrk = await this.keyFactory.createPkcKey();\r\n\r\n    // Option 1: New PKC key for signing\r\n    // const initiatorSigPrk = await this.keyService.createPkcSignKey();\r\n\r\n    // Option 2: Use the user's global signing key.\r\n    // This key is used to prove the initiator's identity.\r\n    const initiatorPrk = await this.keyService.getCurrentPxk();\r\n    const initiatorSigPrk = await this.keyService.getCurrentSigPxk();\r\n\r\n    let initiatorPlainDataSig: string = null;\r\n\r\n    if (contactCard && contactCard.ownerPlainDataJson) {\r\n      initiatorPlainDataSig = await this.encryptionService.signToString(\r\n        initiatorSigPrk.jwk,\r\n        contactCard.ownerPlainDataJson\r\n      );\r\n    }\r\n\r\n    const initiator = {\r\n      message,\r\n      contactCard: contactCard && {\r\n        sharedCipherDataClearJson: contactCard.sharedCipherDataClearJson,\r\n      },\r\n    };\r\n\r\n    // Content to be encrypted using the OTK.\r\n    const plainOtKeyCipher: OtKeyCipherClearJson2 = {\r\n      // TODO Make sure we also put the OOB code in here as well since the OOB code is the\r\n      // _only_ information the KC server does not have access to. The server may have\r\n      // access to OTK and hence the nonce here. It's good to have both the nonce and OOB code\r\n      // since the user may not be using the OOB code. And it's simple to always include\r\n      // the nonce, so why not.\r\n      nonce,\r\n      initiator: {\r\n        ...initiator,\r\n        oneTimePbk: initiatorOneTimePrk.toJSON(), // one-time public encryption key responder use to send data back to initiator\r\n        pbk: initiatorPrk.jwk.toJSON(), // public encryption key\r\n        sigPbk: initiatorSigPrk.jwk.toJSON(), // public signing key\r\n      },\r\n    };\r\n\r\n    const otKeyCipher = await this.keyGraph.encryptToString(\r\n      otKey,\r\n      plainOtKeyCipher\r\n    );\r\n\r\n    // Content to be encrypted using the initiator's root key.\r\n    const initiatorRootKeyCipherClearJson: InitiatorRootKeyCipherClearJson2 = {\r\n      nonce,\r\n      oneTimePrk: initiatorOneTimePrk.toJSON(true),\r\n      // Should not need to keep this encrypted since we are using the global signing key.\r\n      // sigPrk: initiatorSigPrk.toJSON(true),\r\n\r\n      // Save it in case the initiator want to decode the otKeyCipher.\r\n      // Since the otKey is only used once, and that otKeyCipher contains only\r\n      // the public key of the initiator, it's safe just leave the otKey stored here.\r\n      otKey: otKey.toJSON(true),\r\n      // These should be storing information such as how the fields of the shared contact card is\r\n      // derived from the master contact card.\r\n      initiatorContactCard: contactCard,\r\n      initiator,\r\n    };\r\n\r\n    const rootKey = await this.keyService.getCurrentRootKey();\r\n    const initiatorRootKeyCipher = await this.keyGraph.encryptToString(\r\n      rootKey.jwk,\r\n      initiatorRootKeyCipherClearJson\r\n    );\r\n\r\n    // The raw OTK\r\n    const otKeyK: string = (otKey.toJSON(true) as any).k;\r\n\r\n    // API call\r\n    const lrMutation = new LrMutation({\r\n      mutation: InitiateKeyExchangeOtkMutation,\r\n      variables: {\r\n        input: {\r\n          // These will be stored on the server\r\n          initiatorRootKeyCipher,\r\n          initiatorPxkId: initiatorPrk.id,\r\n          initiatorSigPxkId: initiatorSigPrk.id,\r\n          // These will be sent to the responder\r\n          otKeyParams: JSON.stringify(otKey.toJSON()),\r\n          otKeyCipher,\r\n          sendEmail: email && {\r\n            email,\r\n            rawOtKey: otKeyK,\r\n          },\r\n          createTp: true,\r\n          initiatorPlainDataSig,\r\n          upgrade,\r\n        },\r\n      },\r\n    });\r\n\r\n    return { lrMutation, otKeyK };\r\n  }\r\n\r\n  async respondOtk({\r\n    keyExchangeId,\r\n    token,\r\n    decryptedOtk,\r\n    message,\r\n    initiatorContactCard,\r\n    responderContactCard,\r\n  }: RespondOtkInput2) {\r\n    const rootKey = await this.keyService.getCurrentRootKey();\r\n\r\n    const masterKeyId = this.keyService.getCurrentMasterKey().id;\r\n    const masterKey = await this.keyService.getCurrentMasterKey();\r\n\r\n    const sharedKey = await this.keyFactory.createKey();\r\n    const mkSharedKey = await this.keyFactory.createKey();\r\n\r\n    const rkWrappedSharedKey = await this.encryptionService.encrypt(\r\n      rootKey.jwk,\r\n      sharedKey.toJSON(true)\r\n    );\r\n    const mkWrappedMkSharedKey = await this.encryptionService.encrypt(\r\n      masterKey.jwk,\r\n      mkSharedKey.toJSON(true)\r\n    );\r\n\r\n    const initiatorOneTimePbk = await KeyFactoryService.asKey(\r\n      decryptedOtk.otKeyCipherClearJson.initiator.oneTimePbk\r\n    );\r\n\r\n    const initiatorPbk = await KeyFactoryService.asKey(\r\n      decryptedOtk.otKeyCipherClearJson.initiator.pbk\r\n    );\r\n    const initiatorSigPbk = await KeyFactoryService.asKey(\r\n      decryptedOtk.otKeyCipherClearJson.initiator.sigPbk\r\n    );\r\n\r\n    // Option 1: Using new Prk for each TP pair\r\n    // Create a new public signing key for the responder.\r\n    // const responderSigPrk = await this.keyService.createPkcSignKey()\r\n    // const rkWrappedResponderSigPrk = await this.encrypt(rootKey, responderSigPrk.toJSON(true));\r\n\r\n    // Option 2: Responder already has a signing Prk\r\n    const responderPrk = await this.keyService.getCurrentPxk();\r\n    const responderSigPrk = await this.keyService.getCurrentSigPxk();\r\n\r\n    const signedInitiatorPbk = await this.encryptionService.sign(\r\n      responderSigPrk.jwk,\r\n      initiatorPbk.toJSON()\r\n    );\r\n    const signedInitiatorSigPbk = await this.encryptionService.sign(\r\n      responderSigPrk.jwk,\r\n      initiatorSigPbk.toJSON()\r\n    );\r\n\r\n    const initiatorOneTimePbkCipherClearJson: InitiatorOneTimePbkCipherClearJson = {\r\n      nonce: decryptedOtk.otKeyCipherClearJson.nonce,\r\n      sharedKey: sharedKey.toJSON(true),\r\n      mkSharedKey: mkSharedKey.toJSON(true),\r\n      responder: {\r\n        pbk: responderPrk.jwk.toJSON(), // public key\r\n        sigPbk: responderSigPrk.jwk.toJSON(), // public key\r\n        message,\r\n      },\r\n    };\r\n\r\n    let receivedCardInput;\r\n    if (decryptedOtk.otKeyCipherClearJson.initiator.contactCard) {\r\n      // Set the info about the initiator to be the ones sent by the initiator. We need th responder to do the encryption here\r\n      // because the initiator does not have the shared key yet, and we want the responder to have a functional contact card after\r\n      // this exchange. The initiator can double check the contact details are correct and sign it when it completes the exchange.\r\n      const sharedCipherDataClearJson =\r\n        decryptedOtk.otKeyCipherClearJson.initiator.contactCard\r\n          .sharedCipherDataClearJson;\r\n\r\n      // Create keys\r\n      const receiverKey = await this.keyFactory.createKey();\r\n      const ccSharedKey = await this.keyFactory.createKey();\r\n      const sigPxk = await this.keyService.getCurrentSigPxk();\r\n\r\n      receivedCardInput = {\r\n        receiverWrappedKey: JSON.stringify(\r\n          await this.encryptionService.encrypt(\r\n            rootKey.jwk,\r\n            receiverKey.toJSON(true)\r\n          )\r\n        ),\r\n        receiverWrappingKeyId: rootKey.id,\r\n        receiverCipherData: initiatorContactCard\r\n          ? JSON.stringify(\r\n              await this.encryptionService.encrypt(\r\n                receiverKey,\r\n                initiatorContactCard.receiverCipherDataClearJson\r\n              )\r\n            )\r\n          : '',\r\n        sharedWrappedKey: JSON.stringify(\r\n          await this.encryptionService.encrypt(\r\n            sharedKey,\r\n            ccSharedKey.toJSON(true)\r\n          )\r\n        ),\r\n      };\r\n\r\n      const sharedCipherData = await this.encryptionService.encrypt(\r\n        ccSharedKey,\r\n        sharedCipherDataClearJson\r\n      );\r\n      receivedCardInput.sharedCipherDataSig = JSON.stringify(\r\n        await this.encryptionService.sign(sigPxk.jwk, sharedCipherData)\r\n      );\r\n      receivedCardInput.sigPxkId = sigPxk.id;\r\n\r\n      initiatorOneTimePbkCipherClearJson.responder.contactCard = {\r\n        ...initiatorOneTimePbkCipherClearJson.responder.contactCard,\r\n        sharedCipherKey: ccSharedKey.toJSON(true),\r\n      };\r\n    }\r\n\r\n    let responderCardInput;\r\n    if (responderContactCard) {\r\n      // Create keys\r\n      const ownerKey = await this.keyFactory.createKey();\r\n      const ccSharedKey = await this.keyFactory.createKey();\r\n      const sigPxk = await this.keyService.getCurrentSigPxk();\r\n\r\n      responderCardInput = {\r\n        ownerWrappedKey: JSON.stringify(\r\n          await this.encryptionService.encrypt(\r\n            rootKey.jwk,\r\n            ownerKey.toJSON(true)\r\n          )\r\n        ),\r\n        ownerWrappingKeyId: rootKey.id,\r\n        ownerCipherData: responderContactCard.ownerCipherDataClearJson\r\n          ? JSON.stringify(\r\n              await this.encryptionService.encrypt(\r\n                ownerKey,\r\n                responderContactCard.ownerCipherDataClearJson\r\n              )\r\n            )\r\n          : '',\r\n\r\n        sharedWrappedKey: JSON.stringify(\r\n          await this.encryptionService.encrypt(\r\n            sharedKey,\r\n            ccSharedKey.toJSON(true)\r\n          )\r\n        ),\r\n      };\r\n\r\n      const sharedCipherData = await this.encryptionService.encrypt(\r\n        ccSharedKey,\r\n        responderContactCard.sharedCipherDataClearJson\r\n      );\r\n      responderCardInput.sharedCipherDataSig = JSON.stringify(\r\n        await this.encryptionService.sign(sigPxk.jwk, sharedCipherData)\r\n      );\r\n      responderCardInput.sigPxkId = sigPxk.id;\r\n\r\n      if (responderContactCard.ownerPlainDataJson) {\r\n        responderCardInput.ownerPlainDataSig = JSON.stringify(\r\n          await this.encryptionService.sign(\r\n            responderSigPrk.jwk,\r\n            responderContactCard.ownerPlainDataJson\r\n          )\r\n        );\r\n      }\r\n\r\n      // Contact card info readable by the initiator\r\n      initiatorOneTimePbkCipherClearJson.responder.contactCard = {\r\n        ...initiatorOneTimePbkCipherClearJson.responder.contactCard,\r\n        sharedCipherDataClearJson:\r\n          responderContactCard.sharedCipherDataClearJson,\r\n      };\r\n    }\r\n\r\n    // Encrypt with one-time public key\r\n    let initiatorOneTimePbkCipher = await this.encryptionService.encrypt(\r\n      initiatorOneTimePbk,\r\n      initiatorOneTimePbkCipherClearJson\r\n    );\r\n\r\n    // Encrypt with the otk again to keep use of asymmetric keys to a minimum.\r\n    initiatorOneTimePbkCipher = await this.encryptionService.encrypt(\r\n      decryptedOtk.otKey,\r\n      initiatorOneTimePbkCipher\r\n    );\r\n\r\n    return new LrMutation({\r\n      mutation: RespondKeyExchangeOtkMutation,\r\n      variables: {\r\n        input: {\r\n          keyExchangeId,\r\n          keyExchangeToken: token,\r\n          rootKeyId: rootKey.id,\r\n          masterKeyId,\r\n          // These will be stored on the server\r\n          responderPxkId: responderPrk.id,\r\n          responderSigPxkId: responderSigPrk.id,\r\n          signedInitiatorPbk: JSON.stringify(signedInitiatorPbk),\r\n          signedInitiatorSigPbk: JSON.stringify(signedInitiatorSigPbk),\r\n          // rkWrappedInitiatorSigPbk: JSON.stringify(rkWrappedInitiatorSigPbk),\r\n\r\n          // Option 1: Using new Prk for each TP pair\r\n          // rkWrappedResponderSigPrk: JSON.stringify(rkWrappedResponderSigPrk),\r\n          rkWrappedSharedKey: JSON.stringify(rkWrappedSharedKey),\r\n          mkWrappedMkSharedKey: JSON.stringify(mkWrappedMkSharedKey),\r\n          // These will be sent to the initiator\r\n          initiatorOneTimePbkCipher: JSON.stringify(initiatorOneTimePbkCipher),\r\n          initiatorContactCard: receivedCardInput,\r\n          responderContactCard: responderCardInput,\r\n        },\r\n      },\r\n    });\r\n  }\r\n\r\n  async completeOtk({\r\n    keyExchangeId,\r\n    initiatorRootKeyCipher,\r\n    initiatorOneTimePbkCipher,\r\n    responderContactCard,\r\n    initiatorContactCard,\r\n  }: CompleteOtkInput2) {\r\n    const rootKey = await this.keyService.getCurrentRootKey();\r\n    const masterKey = await this.keyService.getCurrentMasterKey();\r\n\r\n    // Decrypt using the root key to get the Prk\r\n    const initiatorRootKeyCipherClearJson = (await this.encryptionService.decrypt(\r\n      rootKey.jwk,\r\n      initiatorRootKeyCipher\r\n    )) as InitiatorRootKeyCipherClearJson2;\r\n\r\n    // The Prk is single-use and only used to send information from the responder back to the initiator.\r\n    const plainInitiatorOneTimePbkCipher = await this.decryptResponseCipher(\r\n      await KeyFactoryService.asKey(initiatorRootKeyCipherClearJson.otKey),\r\n      await KeyFactoryService.asKey(initiatorRootKeyCipherClearJson.oneTimePrk),\r\n      initiatorOneTimePbkCipher\r\n    );\r\n\r\n    // Check the nonce match to ensure the responder was the one holding the OTK\r\n    if (\r\n      initiatorRootKeyCipherClearJson.nonce !==\r\n      plainInitiatorOneTimePbkCipher.nonce\r\n    ) {\r\n      throw new LrCodeMismatchException(\r\n        'The nonce returned by responder does not match with the one created by the initiator.'\r\n      );\r\n    }\r\n\r\n    // Option 1: Assuming the signing key is unique between users.\r\n    // const initiatorSigPrk = await KFS.asKey(ke.plainInitiatorRootKeyCipher.sigPrk);\r\n    // const rkWrappedInitiatorSigPrk = await this.encrypt(rootKey, initiatorSigPrk.toJSON(true));\r\n\r\n    // Option 2: Use the user's global signing key.\r\n    // In this case the initiatorSigPrk is already a part of the key graph.\r\n    // So there's nothing to do here.\r\n\r\n    // Protected the signing public key of the responder.\r\n    const initiatorSigPrk = await this.keyService.getCurrentSigPxk();\r\n    const responderSigPbk = await KeyFactoryService.asKey(\r\n      plainInitiatorOneTimePbkCipher.responder.sigPbk\r\n    );\r\n    const responderPbk = await KeyFactoryService.asKey(\r\n      plainInitiatorOneTimePbkCipher.responder.pbk\r\n    );\r\n\r\n    const signedResponderPbk = await this.encryptionService.sign(\r\n      initiatorSigPrk.jwk,\r\n      responderPbk.toJSON()\r\n    );\r\n    const signedResponderSigPbk = await this.encryptionService.sign(\r\n      initiatorSigPrk.jwk,\r\n      responderSigPbk.toJSON()\r\n    );\r\n\r\n    const sharedKey = await KeyFactoryService.asKey(\r\n      plainInitiatorOneTimePbkCipher.sharedKey\r\n    );\r\n    const rkWrappedSharedKey = await this.encryptionService.encrypt(\r\n      rootKey.jwk,\r\n      sharedKey.toJSON(true)\r\n    );\r\n\r\n    const mkSharedKey = await KeyFactoryService.asKey(\r\n      plainInitiatorOneTimePbkCipher.mkSharedKey\r\n    );\r\n    const mkWrappedMkSharedKey = await this.encryptionService.encrypt(\r\n      masterKey.jwk,\r\n      mkSharedKey.toJSON(true)\r\n    );\r\n\r\n    let responderContactCardCipherInput;\r\n    if (responderContactCard) {\r\n      // Create key\r\n      const receiverKey = await this.keyFactory.createKey();\r\n\r\n      responderContactCardCipherInput = {\r\n        receiverWrappedKey: JSON.stringify(\r\n          await this.encryptionService.encrypt(\r\n            rootKey.jwk,\r\n            receiverKey.toJSON(true)\r\n          )\r\n        ),\r\n        receiverWrappingKeyId: rootKey.id,\r\n        receiverCipherData: JSON.stringify(\r\n          await this.encryptionService.encrypt(\r\n            receiverKey,\r\n            responderContactCard.receiverCipherDataClearJson\r\n          )\r\n        ),\r\n      };\r\n    }\r\n\r\n    // Get the data needed from the initiator's cipher data.\r\n    let initiatorContactCardCipherInput;\r\n    let initiatorContactCardSharedCipherInput;\r\n    if (initiatorRootKeyCipherClearJson.initiatorContactCard) {\r\n      // The initiatorContactCard created during the creation of the invite and encrypted using the initiator's\r\n      // root key\r\n      const initiatorContactCardFromInit =\r\n        initiatorRootKeyCipherClearJson.initiatorContactCard;\r\n      const ownerKey = await this.keyFactory.createKey();\r\n      const sharedCipherKey = await KeyFactoryService.asKey(\r\n        plainInitiatorOneTimePbkCipher.responder.contactCard.sharedCipherKey\r\n      );\r\n\r\n      const ownerWrappedKey = JSON.stringify(\r\n        await this.encryptionService.encrypt(rootKey.jwk, ownerKey.toJSON(true))\r\n      );\r\n\r\n      // Allow the initiatorContactCard parameter to override\r\n      const ownerCipherDataClearJson =\r\n        initiatorContactCard?.ownerCipherDataClearJson ||\r\n        initiatorContactCardFromInit.ownerCipherDataClearJson;\r\n\r\n      const ownerCipherData = ownerCipherDataClearJson\r\n        ? await this.keyGraph.encryptToString(\r\n            ownerKey,\r\n            ownerCipherDataClearJson\r\n          )\r\n        : '';\r\n\r\n      initiatorContactCardCipherInput = {\r\n        ownerWrappedKey,\r\n        ownerWrappingKeyId: rootKey.id,\r\n        ownerCipherData,\r\n      };\r\n\r\n      initiatorContactCardSharedCipherInput = {\r\n        sigPxkId: initiatorSigPrk.id,\r\n      };\r\n\r\n      const sharedCipherData = await this.encryptionService.encrypt(\r\n        sharedCipherKey,\r\n        initiatorContactCardFromInit.sharedCipherDataClearJson\r\n      );\r\n      initiatorContactCardSharedCipherInput.sharedCipherDataSig = JSON.stringify(\r\n        await this.encryptionService.sign(initiatorSigPrk.jwk, sharedCipherData)\r\n      );\r\n    }\r\n\r\n    // TODO ideally we update the shared data in the contact card sent to the responder as well since that\r\n    // CC was created by the responder.\r\n\r\n    return new LrMutation({\r\n      mutation: CompleteKeyExchangeOtkMutation,\r\n      variables: {\r\n        input: {\r\n          keyExchangeId,\r\n          rootKeyId: rootKey.id,\r\n          masterKeyId: masterKey.id,\r\n          initiatorSigPxkId: initiatorSigPrk.id,\r\n          signedResponderPbk: JSON.stringify(signedResponderPbk),\r\n          signedResponderSigPbk: JSON.stringify(signedResponderSigPbk),\r\n          rkWrappedSharedKey: JSON.stringify(rkWrappedSharedKey),\r\n          mkWrappedMkSharedKey: JSON.stringify(mkWrappedMkSharedKey),\r\n          responderContactCardCipher: responderContactCardCipherInput,\r\n          initiatorContactCardCipher: initiatorContactCardCipherInput,\r\n          initiatorContactCardSharedCipher: initiatorContactCardSharedCipherInput,\r\n        },\r\n      },\r\n    });\r\n  }\r\n}\r\n"]}
1
+ import { __awaiter, __decorate } from "tslib";
2
+ import { Injectable, NgZone } from '@angular/core';
3
+ import { EncryptionService, JoseSerialization, } from '../cryptography/encryption.service';
4
+ import { KeyFactoryService } from '../cryptography/key-factory.service';
5
+ import { KeyGraphService } from '../cryptography/key-graph.service';
6
+ import { KeyService } from '../cryptography/key.service';
7
+ import { LrCodeMismatchException } from '../_common/exceptions';
8
+ import { RunOutsideAngular } from '../_common/run-outside-angular';
9
+ import { UserService } from './../users/user.service';
10
+ import { CancelKeyExchangeMutation, CompleteKeyExchangeOtkMutation, CurrentUserSharedKeyQuery2, DeclineKeyExchangeMutation, InitiateKeyExchangeOtkMutation, KeyExchangeQuery2, KeyExchangesQuery2, KeyExchangeTokenQuery2, RespondKeyExchangeOtkMutation, } from './key-exchange2.gql';
11
+ import { LrGraphQLService, LrMutation } from './lr-graphql';
12
+ import * as i0 from "@angular/core";
13
+ import * as i1 from "../cryptography/key-factory.service";
14
+ import * as i2 from "../cryptography/key.service";
15
+ import * as i3 from "../cryptography/encryption.service";
16
+ import * as i4 from "../users/user.service";
17
+ import * as i5 from "../cryptography/key-graph.service";
18
+ import * as i6 from "./lr-graphql/lr-graphql.service";
19
+ let KeyExchange2Service = class KeyExchange2Service {
20
+ constructor(ngZone, keyFactory, keyService, encryptionService, userService, keyGraph, lrGraphQL) {
21
+ this.ngZone = ngZone;
22
+ this.keyFactory = keyFactory;
23
+ this.keyService = keyService;
24
+ this.encryptionService = encryptionService;
25
+ this.userService = userService;
26
+ this.keyGraph = keyGraph;
27
+ this.lrGraphQL = lrGraphQL;
28
+ this.CLIENT_NONCE_LENGTH = 32;
29
+ }
30
+ getOtKey(keyExchange, otKeyK) {
31
+ return __awaiter(this, void 0, void 0, function* () {
32
+ if (otKeyK) {
33
+ return yield KeyFactoryService.asKey(Object.assign(Object.assign({}, JSON.parse(keyExchange.otk.otKeyParams)), { k: otKeyK }));
34
+ }
35
+ else if (keyExchange.otk.state === 'OTK_INITIATED' &&
36
+ !keyExchange.isInitiator &&
37
+ keyExchange.otk.responderPbkCipher) {
38
+ // Assuming existing user getting invited where OTK is wrapped in responder's public key.
39
+ const prk = yield this.keyService.getCurrentPxk();
40
+ const decryptedCipher = yield this.encryptionService.decrypt(prk.jwk, JSON.parse(keyExchange.otk.responderPbkCipher), {
41
+ serializations: [JoseSerialization.COMPACT],
42
+ });
43
+ if (decryptedCipher.otKey) {
44
+ return yield KeyFactoryService.asKey(decryptedCipher.otKey);
45
+ }
46
+ }
47
+ return null;
48
+ });
49
+ }
50
+ decryptOtk(keyExchange, otKeyK) {
51
+ return __awaiter(this, void 0, void 0, function* () {
52
+ const otKey = yield this.getOtKey(keyExchange, otKeyK);
53
+ let otk = keyExchange.otk;
54
+ if (otKey && otk.otKeyCipher) {
55
+ otk = Object.assign(Object.assign({}, otk), { otKey, otKeyCipherClearJson: yield this.encryptionService.decrypt(otKey, keyExchange.otk.otKeyCipher) });
56
+ }
57
+ return Object.assign(Object.assign({}, keyExchange), { otk });
58
+ });
59
+ }
60
+ decryptResponseCipher(otKey, otPrk, content) {
61
+ return __awaiter(this, void 0, void 0, function* () {
62
+ // The response could be wrapped by the OtK in addition to the OtPbk
63
+ try {
64
+ content = yield this.encryptionService.decrypt(otKey, content);
65
+ }
66
+ catch (error) {
67
+ if (error.message !== 'no key found') {
68
+ throw error;
69
+ }
70
+ // Do nothing to support older versions where message is not wrapped with otk.
71
+ }
72
+ // The Prk is single-use and only used to send information from the responder back to the initiator.
73
+ return yield this.encryptionService.decrypt(otPrk, content);
74
+ });
75
+ }
76
+ decryptKeyExchangeAsInitiator(keyExchange) {
77
+ return __awaiter(this, void 0, void 0, function* () {
78
+ const rootKey = yield this.keyService.getCurrentRootKey();
79
+ // Decrypt using the root key to get the Prk
80
+ const initiatorRootKeyCipherClearJson = (yield this.encryptionService.decrypt(rootKey.jwk, keyExchange.initiatorRootKeyCipher));
81
+ const otKey = yield KeyFactoryService.asKey(initiatorRootKeyCipherClearJson.otKey);
82
+ keyExchange = Object.assign(Object.assign({}, keyExchange), { initiatorRootKeyCipherClearJson });
83
+ let otk = keyExchange.otk;
84
+ if (otk.initiatorOneTimePbkCipher) {
85
+ otk = Object.assign(Object.assign({}, otk), { initiatorOneTimePbkCipherClearJson: yield this.decryptResponseCipher(otKey, yield KeyFactoryService.asKey(initiatorRootKeyCipherClearJson.oneTimePrk), otk.initiatorOneTimePbkCipher) });
86
+ }
87
+ if (otk.otKeyCipher) {
88
+ otk.otKeyCipherClearJson = yield this.encryptionService.decrypt(otKey, otk.otKeyCipher);
89
+ }
90
+ return Object.assign(Object.assign({}, keyExchange), { otk });
91
+ });
92
+ }
93
+ decryptKeyExchangeAsResponder(keyExchange, otKeyK) {
94
+ return __awaiter(this, void 0, void 0, function* () {
95
+ return this.decryptOtk(keyExchange, otKeyK);
96
+ });
97
+ }
98
+ decryptKeyExchange(keyExchange, otKeyK) {
99
+ return __awaiter(this, void 0, void 0, function* () {
100
+ if (keyExchange.isInitiator) {
101
+ return this.decryptKeyExchangeAsInitiator(keyExchange);
102
+ }
103
+ else {
104
+ return this.decryptKeyExchangeAsResponder(keyExchange, otKeyK);
105
+ }
106
+ });
107
+ }
108
+ getKeyExchanges({ state } = {}) {
109
+ return __awaiter(this, void 0, void 0, function* () {
110
+ const { keyExchanges } = yield this.lrGraphQL.query({
111
+ query: KeyExchangesQuery2,
112
+ variables: {
113
+ state,
114
+ },
115
+ });
116
+ return Promise.all(keyExchanges.edges.map((edge) => this.decryptKeyExchange(edge.node)));
117
+ });
118
+ }
119
+ /**
120
+ * @param id If the current user can responder the key exchange if they are either the initiator or the receiver.
121
+ * @param token If not signed in, or not the initiator or responder, 'token' must be given.
122
+ * @param otKeyK Is the raw one-time key (string). If the responder is explicitly specified at time of initiation, then
123
+ * it's possible to have the otKey wrapped by the public key of the responder. In which case, the otKeyK is not needed.
124
+ */
125
+ getKeyExchange(id, { otKeyK, token } = {}) {
126
+ return __awaiter(this, void 0, void 0, function* () {
127
+ const res = yield this.lrGraphQL.query({
128
+ query: token ? KeyExchangeTokenQuery2 : KeyExchangeQuery2,
129
+ variables: {
130
+ id,
131
+ token,
132
+ },
133
+ includeKeyGraph: !token,
134
+ });
135
+ return this.decryptKeyExchange(res.keyExchange, otKeyK);
136
+ });
137
+ }
138
+ getCurrentUserSharedKey(input) {
139
+ return __awaiter(this, void 0, void 0, function* () {
140
+ return this.lrGraphQL.query({
141
+ query: CurrentUserSharedKeyQuery2,
142
+ variables: {
143
+ username: input.username,
144
+ userId: input.userId,
145
+ },
146
+ });
147
+ });
148
+ }
149
+ // TODO: deprecate this
150
+ cancelKeyExchange(id) {
151
+ return __awaiter(this, void 0, void 0, function* () {
152
+ return this.cancelKeyExchangeMutation(id);
153
+ });
154
+ }
155
+ cancelKeyExchangeMutation(id) {
156
+ return new LrMutation({
157
+ mutation: CancelKeyExchangeMutation,
158
+ variables: {
159
+ input: {
160
+ id,
161
+ },
162
+ },
163
+ });
164
+ }
165
+ // TODO: deprecate this
166
+ declineKeyExchange(id, token) {
167
+ return __awaiter(this, void 0, void 0, function* () {
168
+ return this.declineKeyExchangeMutation(id, token);
169
+ });
170
+ }
171
+ declineKeyExchangeMutation(id, token) {
172
+ return new LrMutation({
173
+ mutation: DeclineKeyExchangeMutation,
174
+ variables: {
175
+ input: {
176
+ id,
177
+ token,
178
+ },
179
+ },
180
+ });
181
+ }
182
+ initiateOtk({ message, email, contactCard, upgrade, } = {}) {
183
+ return __awaiter(this, void 0, void 0, function* () {
184
+ const otKey = yield this.keyFactory.createKey();
185
+ const nonce = this.keyFactory.randomString(this.CLIENT_NONCE_LENGTH);
186
+ // New PKC key for encryption. This key is used only once when the responder sends
187
+ // back their signing public key.
188
+ const initiatorOneTimePrk = yield this.keyFactory.createPkcKey();
189
+ // Option 1: New PKC key for signing
190
+ // const initiatorSigPrk = await this.keyService.createPkcSignKey();
191
+ // Option 2: Use the user's global signing key.
192
+ // This key is used to prove the initiator's identity.
193
+ const initiatorPrk = yield this.keyService.getCurrentPxk();
194
+ const initiatorSigPrk = yield this.keyService.getCurrentSigPxk();
195
+ let initiatorPlainDataSig = null;
196
+ if (contactCard && contactCard.ownerPlainDataJson) {
197
+ initiatorPlainDataSig = yield this.encryptionService.signToString(initiatorSigPrk.jwk, contactCard.ownerPlainDataJson);
198
+ }
199
+ const initiator = {
200
+ message,
201
+ contactCard: contactCard && {
202
+ sharedCipherDataClearJson: contactCard.sharedCipherDataClearJson,
203
+ },
204
+ };
205
+ // Content to be encrypted using the OTK.
206
+ const plainOtKeyCipher = {
207
+ // TODO Make sure we also put the OOB code in here as well since the OOB code is the
208
+ // _only_ information the KC server does not have access to. The server may have
209
+ // access to OTK and hence the nonce here. It's good to have both the nonce and OOB code
210
+ // since the user may not be using the OOB code. And it's simple to always include
211
+ // the nonce, so why not.
212
+ nonce,
213
+ initiator: Object.assign(Object.assign({}, initiator), { oneTimePbk: initiatorOneTimePrk.toJSON(), pbk: initiatorPrk.jwk.toJSON(), sigPbk: initiatorSigPrk.jwk.toJSON() }),
214
+ };
215
+ const otKeyCipher = yield this.keyGraph.encryptToString(otKey, plainOtKeyCipher);
216
+ // Content to be encrypted using the initiator's root key.
217
+ const initiatorRootKeyCipherClearJson = {
218
+ nonce,
219
+ oneTimePrk: initiatorOneTimePrk.toJSON(true),
220
+ // Should not need to keep this encrypted since we are using the global signing key.
221
+ // sigPrk: initiatorSigPrk.toJSON(true),
222
+ // Save it in case the initiator want to decode the otKeyCipher.
223
+ // Since the otKey is only used once, and that otKeyCipher contains only
224
+ // the public key of the initiator, it's safe just leave the otKey stored here.
225
+ otKey: otKey.toJSON(true),
226
+ // These should be storing information such as how the fields of the shared contact card is
227
+ // derived from the master contact card.
228
+ initiatorContactCard: contactCard,
229
+ initiator,
230
+ };
231
+ const rootKey = yield this.keyService.getCurrentRootKey();
232
+ const initiatorRootKeyCipher = yield this.keyGraph.encryptToString(rootKey.jwk, initiatorRootKeyCipherClearJson);
233
+ // The raw OTK
234
+ const otKeyK = otKey.toJSON(true).k;
235
+ // API call
236
+ const lrMutation = new LrMutation({
237
+ mutation: InitiateKeyExchangeOtkMutation,
238
+ variables: {
239
+ input: {
240
+ // These will be stored on the server
241
+ initiatorRootKeyCipher,
242
+ initiatorPxkId: initiatorPrk.id,
243
+ initiatorSigPxkId: initiatorSigPrk.id,
244
+ // These will be sent to the responder
245
+ otKeyParams: JSON.stringify(otKey.toJSON()),
246
+ otKeyCipher,
247
+ sendEmail: email && {
248
+ email,
249
+ rawOtKey: otKeyK,
250
+ },
251
+ createTp: true,
252
+ initiatorPlainDataSig,
253
+ upgrade,
254
+ },
255
+ },
256
+ });
257
+ return { lrMutation, otKeyK };
258
+ });
259
+ }
260
+ respondOtk({ keyExchangeId, token, decryptedOtk, message, initiatorContactCard, responderContactCard, }) {
261
+ return __awaiter(this, void 0, void 0, function* () {
262
+ const rootKey = yield this.keyService.getCurrentRootKey();
263
+ const masterKeyId = this.keyService.getCurrentMasterKey().id;
264
+ const masterKey = yield this.keyService.getCurrentMasterKey();
265
+ const sharedKey = yield this.keyFactory.createKey();
266
+ const mkSharedKey = yield this.keyFactory.createKey();
267
+ const rkWrappedSharedKey = yield this.encryptionService.encrypt(rootKey.jwk, sharedKey.toJSON(true));
268
+ const mkWrappedMkSharedKey = yield this.encryptionService.encrypt(masterKey.jwk, mkSharedKey.toJSON(true));
269
+ const initiatorOneTimePbk = yield KeyFactoryService.asKey(decryptedOtk.otKeyCipherClearJson.initiator.oneTimePbk);
270
+ const initiatorPbk = yield KeyFactoryService.asKey(decryptedOtk.otKeyCipherClearJson.initiator.pbk);
271
+ const initiatorSigPbk = yield KeyFactoryService.asKey(decryptedOtk.otKeyCipherClearJson.initiator.sigPbk);
272
+ // Option 1: Using new Prk for each TP pair
273
+ // Create a new public signing key for the responder.
274
+ // const responderSigPrk = await this.keyService.createPkcSignKey()
275
+ // const rkWrappedResponderSigPrk = await this.encrypt(rootKey, responderSigPrk.toJSON(true));
276
+ // Option 2: Responder already has a signing Prk
277
+ const responderPrk = yield this.keyService.getCurrentPxk();
278
+ const responderSigPrk = yield this.keyService.getCurrentSigPxk();
279
+ const signedInitiatorPbk = yield this.encryptionService.sign(responderSigPrk.jwk, initiatorPbk.toJSON());
280
+ const signedInitiatorSigPbk = yield this.encryptionService.sign(responderSigPrk.jwk, initiatorSigPbk.toJSON());
281
+ const initiatorOneTimePbkCipherClearJson = {
282
+ nonce: decryptedOtk.otKeyCipherClearJson.nonce,
283
+ sharedKey: sharedKey.toJSON(true),
284
+ mkSharedKey: mkSharedKey.toJSON(true),
285
+ responder: {
286
+ pbk: responderPrk.jwk.toJSON(),
287
+ sigPbk: responderSigPrk.jwk.toJSON(),
288
+ message,
289
+ },
290
+ };
291
+ let receivedCardInput;
292
+ if (decryptedOtk.otKeyCipherClearJson.initiator.contactCard) {
293
+ // Set the info about the initiator to be the ones sent by the initiator. We need th responder to do the encryption here
294
+ // because the initiator does not have the shared key yet, and we want the responder to have a functional contact card after
295
+ // this exchange. The initiator can double check the contact details are correct and sign it when it completes the exchange.
296
+ const sharedCipherDataClearJson = decryptedOtk.otKeyCipherClearJson.initiator.contactCard
297
+ .sharedCipherDataClearJson;
298
+ // Create keys
299
+ const receiverKey = yield this.keyFactory.createKey();
300
+ const ccSharedKey = yield this.keyFactory.createKey();
301
+ const sigPxk = yield this.keyService.getCurrentSigPxk();
302
+ receivedCardInput = {
303
+ receiverWrappedKey: JSON.stringify(yield this.encryptionService.encrypt(rootKey.jwk, receiverKey.toJSON(true))),
304
+ receiverWrappingKeyId: rootKey.id,
305
+ receiverCipherData: initiatorContactCard
306
+ ? JSON.stringify(yield this.encryptionService.encrypt(receiverKey, initiatorContactCard.receiverCipherDataClearJson))
307
+ : '',
308
+ sharedWrappedKey: JSON.stringify(yield this.encryptionService.encrypt(sharedKey, ccSharedKey.toJSON(true))),
309
+ };
310
+ const sharedCipherData = yield this.encryptionService.encrypt(ccSharedKey, sharedCipherDataClearJson);
311
+ receivedCardInput.sharedCipherDataSig = JSON.stringify(yield this.encryptionService.sign(sigPxk.jwk, sharedCipherData));
312
+ receivedCardInput.sigPxkId = sigPxk.id;
313
+ initiatorOneTimePbkCipherClearJson.responder.contactCard = Object.assign(Object.assign({}, initiatorOneTimePbkCipherClearJson.responder.contactCard), { sharedCipherKey: ccSharedKey.toJSON(true) });
314
+ }
315
+ let responderCardInput;
316
+ if (responderContactCard) {
317
+ // Create keys
318
+ const ownerKey = yield this.keyFactory.createKey();
319
+ const ccSharedKey = yield this.keyFactory.createKey();
320
+ const sigPxk = yield this.keyService.getCurrentSigPxk();
321
+ responderCardInput = {
322
+ ownerWrappedKey: JSON.stringify(yield this.encryptionService.encrypt(rootKey.jwk, ownerKey.toJSON(true))),
323
+ ownerWrappingKeyId: rootKey.id,
324
+ ownerCipherData: responderContactCard.ownerCipherDataClearJson
325
+ ? JSON.stringify(yield this.encryptionService.encrypt(ownerKey, responderContactCard.ownerCipherDataClearJson))
326
+ : '',
327
+ sharedWrappedKey: JSON.stringify(yield this.encryptionService.encrypt(sharedKey, ccSharedKey.toJSON(true))),
328
+ };
329
+ const sharedCipherData = yield this.encryptionService.encrypt(ccSharedKey, responderContactCard.sharedCipherDataClearJson);
330
+ responderCardInput.sharedCipherDataSig = JSON.stringify(yield this.encryptionService.sign(sigPxk.jwk, sharedCipherData));
331
+ responderCardInput.sigPxkId = sigPxk.id;
332
+ if (responderContactCard.ownerPlainDataJson) {
333
+ responderCardInput.ownerPlainDataSig = JSON.stringify(yield this.encryptionService.sign(responderSigPrk.jwk, responderContactCard.ownerPlainDataJson));
334
+ }
335
+ // Contact card info readable by the initiator
336
+ initiatorOneTimePbkCipherClearJson.responder.contactCard = Object.assign(Object.assign({}, initiatorOneTimePbkCipherClearJson.responder.contactCard), { sharedCipherDataClearJson: responderContactCard.sharedCipherDataClearJson });
337
+ }
338
+ // Encrypt with one-time public key
339
+ let initiatorOneTimePbkCipher = yield this.encryptionService.encrypt(initiatorOneTimePbk, initiatorOneTimePbkCipherClearJson);
340
+ // Encrypt with the otk again to keep use of asymmetric keys to a minimum.
341
+ initiatorOneTimePbkCipher = yield this.encryptionService.encrypt(decryptedOtk.otKey, initiatorOneTimePbkCipher);
342
+ return new LrMutation({
343
+ mutation: RespondKeyExchangeOtkMutation,
344
+ variables: {
345
+ input: {
346
+ keyExchangeId,
347
+ keyExchangeToken: token,
348
+ rootKeyId: rootKey.id,
349
+ masterKeyId,
350
+ // These will be stored on the server
351
+ responderPxkId: responderPrk.id,
352
+ responderSigPxkId: responderSigPrk.id,
353
+ signedInitiatorPbk: JSON.stringify(signedInitiatorPbk),
354
+ signedInitiatorSigPbk: JSON.stringify(signedInitiatorSigPbk),
355
+ // rkWrappedInitiatorSigPbk: JSON.stringify(rkWrappedInitiatorSigPbk),
356
+ // Option 1: Using new Prk for each TP pair
357
+ // rkWrappedResponderSigPrk: JSON.stringify(rkWrappedResponderSigPrk),
358
+ rkWrappedSharedKey: JSON.stringify(rkWrappedSharedKey),
359
+ mkWrappedMkSharedKey: JSON.stringify(mkWrappedMkSharedKey),
360
+ // These will be sent to the initiator
361
+ initiatorOneTimePbkCipher: JSON.stringify(initiatorOneTimePbkCipher),
362
+ initiatorContactCard: receivedCardInput,
363
+ responderContactCard: responderCardInput,
364
+ },
365
+ },
366
+ });
367
+ });
368
+ }
369
+ completeOtk({ keyExchangeId, initiatorRootKeyCipher, initiatorOneTimePbkCipher, responderContactCard, initiatorContactCard, }) {
370
+ return __awaiter(this, void 0, void 0, function* () {
371
+ const rootKey = yield this.keyService.getCurrentRootKey();
372
+ const masterKey = yield this.keyService.getCurrentMasterKey();
373
+ // Decrypt using the root key to get the Prk
374
+ const initiatorRootKeyCipherClearJson = (yield this.encryptionService.decrypt(rootKey.jwk, initiatorRootKeyCipher));
375
+ // The Prk is single-use and only used to send information from the responder back to the initiator.
376
+ const plainInitiatorOneTimePbkCipher = yield this.decryptResponseCipher(yield KeyFactoryService.asKey(initiatorRootKeyCipherClearJson.otKey), yield KeyFactoryService.asKey(initiatorRootKeyCipherClearJson.oneTimePrk), initiatorOneTimePbkCipher);
377
+ // Check the nonce match to ensure the responder was the one holding the OTK
378
+ if (initiatorRootKeyCipherClearJson.nonce !==
379
+ plainInitiatorOneTimePbkCipher.nonce) {
380
+ throw new LrCodeMismatchException('The nonce returned by responder does not match with the one created by the initiator.');
381
+ }
382
+ // Option 1: Assuming the signing key is unique between users.
383
+ // const initiatorSigPrk = await KFS.asKey(ke.plainInitiatorRootKeyCipher.sigPrk);
384
+ // const rkWrappedInitiatorSigPrk = await this.encrypt(rootKey, initiatorSigPrk.toJSON(true));
385
+ // Option 2: Use the user's global signing key.
386
+ // In this case the initiatorSigPrk is already a part of the key graph.
387
+ // So there's nothing to do here.
388
+ // Protected the signing public key of the responder.
389
+ const initiatorSigPrk = yield this.keyService.getCurrentSigPxk();
390
+ const responderSigPbk = yield KeyFactoryService.asKey(plainInitiatorOneTimePbkCipher.responder.sigPbk);
391
+ const responderPbk = yield KeyFactoryService.asKey(plainInitiatorOneTimePbkCipher.responder.pbk);
392
+ const signedResponderPbk = yield this.encryptionService.sign(initiatorSigPrk.jwk, responderPbk.toJSON());
393
+ const signedResponderSigPbk = yield this.encryptionService.sign(initiatorSigPrk.jwk, responderSigPbk.toJSON());
394
+ const sharedKey = yield KeyFactoryService.asKey(plainInitiatorOneTimePbkCipher.sharedKey);
395
+ const rkWrappedSharedKey = yield this.encryptionService.encrypt(rootKey.jwk, sharedKey.toJSON(true));
396
+ const mkSharedKey = yield KeyFactoryService.asKey(plainInitiatorOneTimePbkCipher.mkSharedKey);
397
+ const mkWrappedMkSharedKey = yield this.encryptionService.encrypt(masterKey.jwk, mkSharedKey.toJSON(true));
398
+ let responderContactCardCipherInput;
399
+ if (responderContactCard) {
400
+ // Create key
401
+ const receiverKey = yield this.keyFactory.createKey();
402
+ responderContactCardCipherInput = {
403
+ receiverWrappedKey: JSON.stringify(yield this.encryptionService.encrypt(rootKey.jwk, receiverKey.toJSON(true))),
404
+ receiverWrappingKeyId: rootKey.id,
405
+ receiverCipherData: JSON.stringify(yield this.encryptionService.encrypt(receiverKey, responderContactCard.receiverCipherDataClearJson)),
406
+ };
407
+ }
408
+ // Get the data needed from the initiator's cipher data.
409
+ let initiatorContactCardCipherInput;
410
+ let initiatorContactCardSharedCipherInput;
411
+ if (initiatorRootKeyCipherClearJson.initiatorContactCard) {
412
+ // The initiatorContactCard created during the creation of the invite and encrypted using the initiator's
413
+ // root key
414
+ const initiatorContactCardFromInit = initiatorRootKeyCipherClearJson.initiatorContactCard;
415
+ const ownerKey = yield this.keyFactory.createKey();
416
+ const sharedCipherKey = yield KeyFactoryService.asKey(plainInitiatorOneTimePbkCipher.responder.contactCard.sharedCipherKey);
417
+ const ownerWrappedKey = JSON.stringify(yield this.encryptionService.encrypt(rootKey.jwk, ownerKey.toJSON(true)));
418
+ // Allow the initiatorContactCard parameter to override
419
+ const ownerCipherDataClearJson = (initiatorContactCard === null || initiatorContactCard === void 0 ? void 0 : initiatorContactCard.ownerCipherDataClearJson) ||
420
+ initiatorContactCardFromInit.ownerCipherDataClearJson;
421
+ const ownerCipherData = ownerCipherDataClearJson
422
+ ? yield this.keyGraph.encryptToString(ownerKey, ownerCipherDataClearJson)
423
+ : '';
424
+ initiatorContactCardCipherInput = {
425
+ ownerWrappedKey,
426
+ ownerWrappingKeyId: rootKey.id,
427
+ ownerCipherData,
428
+ };
429
+ initiatorContactCardSharedCipherInput = {
430
+ sigPxkId: initiatorSigPrk.id,
431
+ };
432
+ const sharedCipherData = yield this.encryptionService.encrypt(sharedCipherKey, initiatorContactCardFromInit.sharedCipherDataClearJson);
433
+ initiatorContactCardSharedCipherInput.sharedCipherDataSig = JSON.stringify(yield this.encryptionService.sign(initiatorSigPrk.jwk, sharedCipherData));
434
+ }
435
+ // TODO ideally we update the shared data in the contact card sent to the responder as well since that
436
+ // CC was created by the responder.
437
+ return new LrMutation({
438
+ mutation: CompleteKeyExchangeOtkMutation,
439
+ variables: {
440
+ input: {
441
+ keyExchangeId,
442
+ rootKeyId: rootKey.id,
443
+ masterKeyId: masterKey.id,
444
+ initiatorSigPxkId: initiatorSigPrk.id,
445
+ signedResponderPbk: JSON.stringify(signedResponderPbk),
446
+ signedResponderSigPbk: JSON.stringify(signedResponderSigPbk),
447
+ rkWrappedSharedKey: JSON.stringify(rkWrappedSharedKey),
448
+ mkWrappedMkSharedKey: JSON.stringify(mkWrappedMkSharedKey),
449
+ responderContactCardCipher: responderContactCardCipherInput,
450
+ initiatorContactCardCipher: initiatorContactCardCipherInput,
451
+ initiatorContactCardSharedCipher: initiatorContactCardSharedCipherInput,
452
+ },
453
+ },
454
+ });
455
+ });
456
+ }
457
+ };
458
+ KeyExchange2Service.ɵprov = i0.ɵɵdefineInjectable({ factory: function KeyExchange2Service_Factory() { return new KeyExchange2Service(i0.ɵɵinject(i0.NgZone), i0.ɵɵinject(i1.KeyFactoryService), i0.ɵɵinject(i2.KeyService), i0.ɵɵinject(i3.EncryptionService), i0.ɵɵinject(i4.UserService), i0.ɵɵinject(i5.KeyGraphService), i0.ɵɵinject(i6.LrGraphQLService)); }, token: KeyExchange2Service, providedIn: "root" });
459
+ KeyExchange2Service.decorators = [
460
+ { type: Injectable, args: [{
461
+ providedIn: 'root',
462
+ },] }
463
+ ];
464
+ KeyExchange2Service.ctorParameters = () => [
465
+ { type: NgZone },
466
+ { type: KeyFactoryService },
467
+ { type: KeyService },
468
+ { type: EncryptionService },
469
+ { type: UserService },
470
+ { type: KeyGraphService },
471
+ { type: LrGraphQLService }
472
+ ];
473
+ KeyExchange2Service = __decorate([
474
+ RunOutsideAngular({
475
+ ngZoneName: 'ngZone',
476
+ })
477
+ ], KeyExchange2Service);
478
+ export { KeyExchange2Service };
479
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"key-exchange2.service.js","sourceRoot":"/opt/atlassian/pipelines/agent/build/projects/core/src/","sources":["lib/api/key-exchange2.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, NgZone } from '@angular/core';\nimport { JWK } from 'node-jose';\nimport {\n  EncryptionService,\n  JoseSerialization,\n} from '../cryptography/encryption.service';\nimport { KeyFactoryService } from '../cryptography/key-factory.service';\nimport { KeyGraphService } from '../cryptography/key-graph.service';\nimport { KeyService } from '../cryptography/key.service';\nimport { LrCodeMismatchException } from '../_common/exceptions';\nimport { RunOutsideAngular } from '../_common/run-outside-angular';\nimport { UserService } from './../users/user.service';\nimport { OwnerPlainDataJson } from './contact-card2.service';\nimport {\n  CancelKeyExchangeMutation,\n  CompleteKeyExchangeOtkMutation,\n  CurrentUserSharedKeyQuery2,\n  DeclineKeyExchangeMutation,\n  InitiateKeyExchangeOtkMutation,\n  KeyExchangeFragment,\n  KeyExchangeQuery2,\n  KeyExchangesQuery2,\n  KeyExchangeState2,\n  KeyExchangeTokenQuery2,\n  RespondKeyExchangeOtkMutation,\n} from './key-exchange2.gql';\nimport { LrGraphQLService, LrMutation } from './lr-graphql';\nimport { ContactCardSharedCipherData, OtKeyCipherClearJson2 } from './types';\n\n/**\n * The decrypted content of the one-time key cipher;\n * When user supplies this information the lib doesn't need to do another API call\n * to fetch the key exchange node. And since the typical use case is to display\n * some information to the user, the key exchange node would already have been\n * fetched and decrypted.\n */\nexport interface DecryptedOtk2 {\n  otKeyCipherClearJson: OtKeyCipherClearJson2;\n  otKey: JWK.Key; // The one-time key\n}\n\nexport interface ContactCardReceiverCipherData {\n  // The receiver of the contact card keeps a copy\n  // of the owner's contact card information, encrypted using the receiver's keys. So that when\n  // the owner decides to update their shared contact card at a later date, the receiver can\n  // compare against the original contact card sent during key exchange. This way, the owner\n  // can't unilaterally update their shared contact card without the receiver knowing about it.\n  receiverCipherDataClearJson: any;\n}\n\nexport interface ContactCardOwnerPlainData {\n  // Accessible by the server and the owner. Server side notification emails need to know some\n  // information about the owner.\n  // The owner could either be the initiator or the responder.\n  ownerPlainDataJson: OwnerPlainDataJson;\n}\n\nexport interface ContactCardOwnerCipherData {\n  // Owner only access to this data\n  ownerCipherDataClearJson: any;\n}\n\n/**\n * Sending contact card information from the owner to the receiver.\n */\nexport type SendContactCardInput = ContactCardOwnerPlainData &\n  ContactCardOwnerCipherData &\n  ContactCardSharedCipherData;\n\nexport interface InitiateOtkInput2 {\n  // Note that if neither email nor responderUsername are given, the one-time key\n  // can still be sent to the responder via OOB\n  email?: string;\n  message?: any;\n  contactCard?: SendContactCardInput;\n  // If True, to upgrade an email invite to an existing user invite if the email\n  // is already associated with an existing user.\n  upgrade?: boolean;\n}\n\nexport interface RespondOtkInput2 {\n  keyExchangeId: string;\n  token: string;\n  decryptedOtk: DecryptedOtk2;\n  message?: any;\n  // The initiator is the owner fo the initiatorContactCard, the responder\n  // is the receiver.\n  initiatorContactCard?: ContactCardReceiverCipherData;\n  responderContactCard?: SendContactCardInput;\n}\n\nexport interface CompleteOtkInput2 {\n  keyExchangeId: string;\n  // This is a part of the key exchange data. It's encrypted using the root key\n  initiatorRootKeyCipher: string;\n  // This is a part of the key exchange data. It's encrypted using the one-time key.\n  initiatorOneTimePbkCipher: string;\n  // The responder is the owner fo the responderContactCard, the initiator\n  // is the receiver.\n  responderContactCard?: ContactCardReceiverCipherData;\n  // The initiator can update the cipher data that are only visible to them. It makes\n  // less sense to update the shared data because the responder would have already seen\n  // the shared data and accepted that it's legit.\n  // But in any case, the initiator can update the shared contact card info at any time.\n  initiatorContactCard?: ContactCardOwnerCipherData;\n}\n\nexport interface InitiatorRootKeyCipherClearJson2 {\n  nonce: string;\n  oneTimePrk: object; // one-time public encryption key responder use to send data back to initiator\n  otKey: object; // one-time symmetric key that needs to be shared OOB\n  initiatorContactCard?: ContactCardOwnerCipherData &\n    ContactCardSharedCipherData;\n  initiator: {\n    message?: any;\n    contactCard?: ContactCardSharedCipherData;\n  };\n}\n\nexport interface InitiatorOneTimePbkCipherClearJson {\n  nonce: string;\n  sharedKey: object;\n  mkSharedKey: object;\n  responder: {\n    pbk: object;\n    sigPbk: object;\n    message?: any;\n    contactCard?: ContactCardSharedCipherData & {\n      // Note that this is _not_ the same key as the sharedKey. The sharedKey wraps\n      // this key in the key graph. But because this key has not been entered into\n      // the key graph when the responder calls the API, we pass the JWK directly here.\n      sharedCipherKey: object;\n    };\n  };\n}\n\nexport interface GetKeyExchangeOptions2 {\n  // The otKey as a raw string. i.e. key.toJSON(true).k\n  otKeyK?: string;\n  // User need the token if they have not responded to the key exchange yet.\n  // Once they've responded (hence proven they have the OOB Key) they become\n  // the \"responder\" of this exchange, and can access it when signed in.\n  token?: string;\n}\n\n@RunOutsideAngular({\n  ngZoneName: 'ngZone',\n})\n@Injectable({\n  providedIn: 'root',\n})\nexport class KeyExchange2Service {\n  private readonly CLIENT_NONCE_LENGTH = 32;\n\n  constructor(\n    private ngZone: NgZone,\n    private keyFactory: KeyFactoryService,\n    private keyService: KeyService,\n    private encryptionService: EncryptionService,\n    private userService: UserService,\n    private keyGraph: KeyGraphService,\n    private lrGraphQL: LrGraphQLService\n  ) {}\n\n  private async getOtKey(\n    keyExchange: KeyExchangeFragment,\n    otKeyK?: string\n  ): Promise<JWK.Key> {\n    if (otKeyK) {\n      return await KeyFactoryService.asKey({\n        ...JSON.parse(keyExchange.otk.otKeyParams),\n        k: otKeyK,\n      });\n    } else if (\n      keyExchange.otk.state === 'OTK_INITIATED' &&\n      !keyExchange.isInitiator &&\n      keyExchange.otk.responderPbkCipher\n    ) {\n      // Assuming existing user getting invited where OTK is wrapped in responder's public key.\n      const prk = await this.keyService.getCurrentPxk();\n      const decryptedCipher: any = await this.encryptionService.decrypt(\n        prk.jwk,\n        JSON.parse(keyExchange.otk.responderPbkCipher),\n        {\n          serializations: [JoseSerialization.COMPACT],\n        }\n      );\n      if (decryptedCipher.otKey) {\n        return await KeyFactoryService.asKey(decryptedCipher.otKey);\n      }\n    }\n    return null;\n  }\n\n  private async decryptOtk(\n    keyExchange: KeyExchangeFragment,\n    otKeyK?: string\n  ): Promise<KeyExchangeFragment> {\n    const otKey = await this.getOtKey(keyExchange, otKeyK);\n\n    let otk = keyExchange.otk;\n\n    if (otKey && otk.otKeyCipher) {\n      otk = {\n        ...otk,\n        otKey,\n        otKeyCipherClearJson: await this.encryptionService.decrypt(\n          otKey,\n          keyExchange.otk.otKeyCipher\n        ),\n      };\n    }\n\n    return {\n      ...keyExchange,\n      otk,\n    };\n  }\n\n  private async decryptResponseCipher(\n    otKey: JWK.Key,\n    otPrk: JWK.Key,\n    content: any\n  ): Promise<InitiatorOneTimePbkCipherClearJson> {\n    // The response could be wrapped by the OtK in addition to the OtPbk\n    try {\n      content = await this.encryptionService.decrypt(otKey, content);\n    } catch (error) {\n      if (error.message !== 'no key found') {\n        throw error;\n      }\n      // Do nothing to support older versions where message is not wrapped with otk.\n    }\n\n    // The Prk is single-use and only used to send information from the responder back to the initiator.\n    return await this.encryptionService.decrypt(otPrk, content);\n  }\n\n  private async decryptKeyExchangeAsInitiator(\n    keyExchange: KeyExchangeFragment\n  ): Promise<KeyExchangeFragment> {\n    const rootKey = await this.keyService.getCurrentRootKey();\n\n    // Decrypt using the root key to get the Prk\n    const initiatorRootKeyCipherClearJson = (await this.encryptionService.decrypt(\n      rootKey.jwk,\n      keyExchange.initiatorRootKeyCipher\n    )) as InitiatorRootKeyCipherClearJson2;\n\n    const otKey = await KeyFactoryService.asKey(\n      initiatorRootKeyCipherClearJson.otKey\n    );\n\n    keyExchange = {\n      ...keyExchange,\n      initiatorRootKeyCipherClearJson,\n    };\n\n    let otk = keyExchange.otk;\n\n    if (otk.initiatorOneTimePbkCipher) {\n      otk = {\n        ...otk,\n        initiatorOneTimePbkCipherClearJson: await this.decryptResponseCipher(\n          otKey,\n          await KeyFactoryService.asKey(\n            initiatorRootKeyCipherClearJson.oneTimePrk\n          ),\n          otk.initiatorOneTimePbkCipher\n        ),\n      };\n    }\n\n    if (otk.otKeyCipher) {\n      otk.otKeyCipherClearJson = await this.encryptionService.decrypt(\n        otKey,\n        otk.otKeyCipher\n      );\n    }\n\n    return {\n      ...keyExchange,\n      otk,\n    };\n  }\n\n  private async decryptKeyExchangeAsResponder(\n    keyExchange: KeyExchangeFragment,\n    otKeyK?: string\n  ) {\n    return this.decryptOtk(keyExchange, otKeyK);\n  }\n\n  async decryptKeyExchange(keyExchange: KeyExchangeFragment, otKeyK?: string) {\n    if (keyExchange.isInitiator) {\n      return this.decryptKeyExchangeAsInitiator(keyExchange);\n    } else {\n      return this.decryptKeyExchangeAsResponder(keyExchange, otKeyK);\n    }\n  }\n\n  async getKeyExchanges({ state }: { state?: KeyExchangeState2 } = {}) {\n    const { keyExchanges } = await this.lrGraphQL.query({\n      query: KeyExchangesQuery2,\n      variables: {\n        state,\n      },\n    });\n\n    return Promise.all(\n      keyExchanges.edges.map((edge) => this.decryptKeyExchange(edge.node))\n    );\n  }\n\n  /**\n   * @param id If the current user can responder the key exchange if they are either the initiator or the receiver.\n   * @param token If not signed in, or not the initiator or responder, 'token' must be given.\n   * @param otKeyK Is the raw one-time key (string). If the responder is explicitly specified at time of initiation, then\n   *   it's possible to have the otKey wrapped by the public key of the responder. In which case, the otKeyK is not needed.\n   */\n  async getKeyExchange(\n    id: string,\n    { otKeyK, token }: GetKeyExchangeOptions2 = {}\n  ) {\n    const res = await this.lrGraphQL.query({\n      query: token ? KeyExchangeTokenQuery2 : KeyExchangeQuery2,\n      variables: {\n        id,\n        token,\n      },\n      includeKeyGraph: !token, // if !token then we are post auth, so can fetch keyGraph\n    });\n    return this.decryptKeyExchange(res.keyExchange, otKeyK);\n  }\n\n  public async getCurrentUserSharedKey(input: {\n    username?: string;\n    userId?: string;\n  }) {\n    return this.lrGraphQL.query({\n      query: CurrentUserSharedKeyQuery2,\n      variables: {\n        username: input.username,\n        userId: input.userId,\n      },\n    });\n  }\n\n  // TODO: deprecate this\n  async cancelKeyExchange(id: string) {\n    return this.cancelKeyExchangeMutation(id);\n  }\n\n  cancelKeyExchangeMutation(id: string) {\n    return new LrMutation({\n      mutation: CancelKeyExchangeMutation,\n      variables: {\n        input: {\n          id,\n        },\n      },\n    });\n  }\n\n  // TODO: deprecate this\n  async declineKeyExchange(id: string, token: string) {\n    return this.declineKeyExchangeMutation(id, token);\n  }\n\n  declineKeyExchangeMutation(id: string, token: string) {\n    return new LrMutation({\n      mutation: DeclineKeyExchangeMutation,\n      variables: {\n        input: {\n          id,\n          token,\n        },\n      },\n    });\n  }\n\n  async initiateOtk({\n    message,\n    email,\n    contactCard,\n    upgrade,\n  }: InitiateOtkInput2 = {}) {\n    const otKey = await this.keyFactory.createKey();\n    const nonce = this.keyFactory.randomString(this.CLIENT_NONCE_LENGTH);\n\n    // New PKC key for encryption. This key is used only once when the responder sends\n    // back their signing public key.\n    const initiatorOneTimePrk = await this.keyFactory.createPkcKey();\n\n    // Option 1: New PKC key for signing\n    // const initiatorSigPrk = await this.keyService.createPkcSignKey();\n\n    // Option 2: Use the user's global signing key.\n    // This key is used to prove the initiator's identity.\n    const initiatorPrk = await this.keyService.getCurrentPxk();\n    const initiatorSigPrk = await this.keyService.getCurrentSigPxk();\n\n    let initiatorPlainDataSig: string = null;\n\n    if (contactCard && contactCard.ownerPlainDataJson) {\n      initiatorPlainDataSig = await this.encryptionService.signToString(\n        initiatorSigPrk.jwk,\n        contactCard.ownerPlainDataJson\n      );\n    }\n\n    const initiator = {\n      message,\n      contactCard: contactCard && {\n        sharedCipherDataClearJson: contactCard.sharedCipherDataClearJson,\n      },\n    };\n\n    // Content to be encrypted using the OTK.\n    const plainOtKeyCipher: OtKeyCipherClearJson2 = {\n      // TODO Make sure we also put the OOB code in here as well since the OOB code is the\n      // _only_ information the KC server does not have access to. The server may have\n      // access to OTK and hence the nonce here. It's good to have both the nonce and OOB code\n      // since the user may not be using the OOB code. And it's simple to always include\n      // the nonce, so why not.\n      nonce,\n      initiator: {\n        ...initiator,\n        oneTimePbk: initiatorOneTimePrk.toJSON(), // one-time public encryption key responder use to send data back to initiator\n        pbk: initiatorPrk.jwk.toJSON(), // public encryption key\n        sigPbk: initiatorSigPrk.jwk.toJSON(), // public signing key\n      },\n    };\n\n    const otKeyCipher = await this.keyGraph.encryptToString(\n      otKey,\n      plainOtKeyCipher\n    );\n\n    // Content to be encrypted using the initiator's root key.\n    const initiatorRootKeyCipherClearJson: InitiatorRootKeyCipherClearJson2 = {\n      nonce,\n      oneTimePrk: initiatorOneTimePrk.toJSON(true),\n      // Should not need to keep this encrypted since we are using the global signing key.\n      // sigPrk: initiatorSigPrk.toJSON(true),\n\n      // Save it in case the initiator want to decode the otKeyCipher.\n      // Since the otKey is only used once, and that otKeyCipher contains only\n      // the public key of the initiator, it's safe just leave the otKey stored here.\n      otKey: otKey.toJSON(true),\n      // These should be storing information such as how the fields of the shared contact card is\n      // derived from the master contact card.\n      initiatorContactCard: contactCard,\n      initiator,\n    };\n\n    const rootKey = await this.keyService.getCurrentRootKey();\n    const initiatorRootKeyCipher = await this.keyGraph.encryptToString(\n      rootKey.jwk,\n      initiatorRootKeyCipherClearJson\n    );\n\n    // The raw OTK\n    const otKeyK: string = (otKey.toJSON(true) as any).k;\n\n    // API call\n    const lrMutation = new LrMutation({\n      mutation: InitiateKeyExchangeOtkMutation,\n      variables: {\n        input: {\n          // These will be stored on the server\n          initiatorRootKeyCipher,\n          initiatorPxkId: initiatorPrk.id,\n          initiatorSigPxkId: initiatorSigPrk.id,\n          // These will be sent to the responder\n          otKeyParams: JSON.stringify(otKey.toJSON()),\n          otKeyCipher,\n          sendEmail: email && {\n            email,\n            rawOtKey: otKeyK,\n          },\n          createTp: true,\n          initiatorPlainDataSig,\n          upgrade,\n        },\n      },\n    });\n\n    return { lrMutation, otKeyK };\n  }\n\n  async respondOtk({\n    keyExchangeId,\n    token,\n    decryptedOtk,\n    message,\n    initiatorContactCard,\n    responderContactCard,\n  }: RespondOtkInput2) {\n    const rootKey = await this.keyService.getCurrentRootKey();\n\n    const masterKeyId = this.keyService.getCurrentMasterKey().id;\n    const masterKey = await this.keyService.getCurrentMasterKey();\n\n    const sharedKey = await this.keyFactory.createKey();\n    const mkSharedKey = await this.keyFactory.createKey();\n\n    const rkWrappedSharedKey = await this.encryptionService.encrypt(\n      rootKey.jwk,\n      sharedKey.toJSON(true)\n    );\n    const mkWrappedMkSharedKey = await this.encryptionService.encrypt(\n      masterKey.jwk,\n      mkSharedKey.toJSON(true)\n    );\n\n    const initiatorOneTimePbk = await KeyFactoryService.asKey(\n      decryptedOtk.otKeyCipherClearJson.initiator.oneTimePbk\n    );\n\n    const initiatorPbk = await KeyFactoryService.asKey(\n      decryptedOtk.otKeyCipherClearJson.initiator.pbk\n    );\n    const initiatorSigPbk = await KeyFactoryService.asKey(\n      decryptedOtk.otKeyCipherClearJson.initiator.sigPbk\n    );\n\n    // Option 1: Using new Prk for each TP pair\n    // Create a new public signing key for the responder.\n    // const responderSigPrk = await this.keyService.createPkcSignKey()\n    // const rkWrappedResponderSigPrk = await this.encrypt(rootKey, responderSigPrk.toJSON(true));\n\n    // Option 2: Responder already has a signing Prk\n    const responderPrk = await this.keyService.getCurrentPxk();\n    const responderSigPrk = await this.keyService.getCurrentSigPxk();\n\n    const signedInitiatorPbk = await this.encryptionService.sign(\n      responderSigPrk.jwk,\n      initiatorPbk.toJSON()\n    );\n    const signedInitiatorSigPbk = await this.encryptionService.sign(\n      responderSigPrk.jwk,\n      initiatorSigPbk.toJSON()\n    );\n\n    const initiatorOneTimePbkCipherClearJson: InitiatorOneTimePbkCipherClearJson = {\n      nonce: decryptedOtk.otKeyCipherClearJson.nonce,\n      sharedKey: sharedKey.toJSON(true),\n      mkSharedKey: mkSharedKey.toJSON(true),\n      responder: {\n        pbk: responderPrk.jwk.toJSON(), // public key\n        sigPbk: responderSigPrk.jwk.toJSON(), // public key\n        message,\n      },\n    };\n\n    let receivedCardInput;\n    if (decryptedOtk.otKeyCipherClearJson.initiator.contactCard) {\n      // Set the info about the initiator to be the ones sent by the initiator. We need th responder to do the encryption here\n      // because the initiator does not have the shared key yet, and we want the responder to have a functional contact card after\n      // this exchange. The initiator can double check the contact details are correct and sign it when it completes the exchange.\n      const sharedCipherDataClearJson =\n        decryptedOtk.otKeyCipherClearJson.initiator.contactCard\n          .sharedCipherDataClearJson;\n\n      // Create keys\n      const receiverKey = await this.keyFactory.createKey();\n      const ccSharedKey = await this.keyFactory.createKey();\n      const sigPxk = await this.keyService.getCurrentSigPxk();\n\n      receivedCardInput = {\n        receiverWrappedKey: JSON.stringify(\n          await this.encryptionService.encrypt(\n            rootKey.jwk,\n            receiverKey.toJSON(true)\n          )\n        ),\n        receiverWrappingKeyId: rootKey.id,\n        receiverCipherData: initiatorContactCard\n          ? JSON.stringify(\n              await this.encryptionService.encrypt(\n                receiverKey,\n                initiatorContactCard.receiverCipherDataClearJson\n              )\n            )\n          : '',\n        sharedWrappedKey: JSON.stringify(\n          await this.encryptionService.encrypt(\n            sharedKey,\n            ccSharedKey.toJSON(true)\n          )\n        ),\n      };\n\n      const sharedCipherData = await this.encryptionService.encrypt(\n        ccSharedKey,\n        sharedCipherDataClearJson\n      );\n      receivedCardInput.sharedCipherDataSig = JSON.stringify(\n        await this.encryptionService.sign(sigPxk.jwk, sharedCipherData)\n      );\n      receivedCardInput.sigPxkId = sigPxk.id;\n\n      initiatorOneTimePbkCipherClearJson.responder.contactCard = {\n        ...initiatorOneTimePbkCipherClearJson.responder.contactCard,\n        sharedCipherKey: ccSharedKey.toJSON(true),\n      };\n    }\n\n    let responderCardInput;\n    if (responderContactCard) {\n      // Create keys\n      const ownerKey = await this.keyFactory.createKey();\n      const ccSharedKey = await this.keyFactory.createKey();\n      const sigPxk = await this.keyService.getCurrentSigPxk();\n\n      responderCardInput = {\n        ownerWrappedKey: JSON.stringify(\n          await this.encryptionService.encrypt(\n            rootKey.jwk,\n            ownerKey.toJSON(true)\n          )\n        ),\n        ownerWrappingKeyId: rootKey.id,\n        ownerCipherData: responderContactCard.ownerCipherDataClearJson\n          ? JSON.stringify(\n              await this.encryptionService.encrypt(\n                ownerKey,\n                responderContactCard.ownerCipherDataClearJson\n              )\n            )\n          : '',\n\n        sharedWrappedKey: JSON.stringify(\n          await this.encryptionService.encrypt(\n            sharedKey,\n            ccSharedKey.toJSON(true)\n          )\n        ),\n      };\n\n      const sharedCipherData = await this.encryptionService.encrypt(\n        ccSharedKey,\n        responderContactCard.sharedCipherDataClearJson\n      );\n      responderCardInput.sharedCipherDataSig = JSON.stringify(\n        await this.encryptionService.sign(sigPxk.jwk, sharedCipherData)\n      );\n      responderCardInput.sigPxkId = sigPxk.id;\n\n      if (responderContactCard.ownerPlainDataJson) {\n        responderCardInput.ownerPlainDataSig = JSON.stringify(\n          await this.encryptionService.sign(\n            responderSigPrk.jwk,\n            responderContactCard.ownerPlainDataJson\n          )\n        );\n      }\n\n      // Contact card info readable by the initiator\n      initiatorOneTimePbkCipherClearJson.responder.contactCard = {\n        ...initiatorOneTimePbkCipherClearJson.responder.contactCard,\n        sharedCipherDataClearJson:\n          responderContactCard.sharedCipherDataClearJson,\n      };\n    }\n\n    // Encrypt with one-time public key\n    let initiatorOneTimePbkCipher = await this.encryptionService.encrypt(\n      initiatorOneTimePbk,\n      initiatorOneTimePbkCipherClearJson\n    );\n\n    // Encrypt with the otk again to keep use of asymmetric keys to a minimum.\n    initiatorOneTimePbkCipher = await this.encryptionService.encrypt(\n      decryptedOtk.otKey,\n      initiatorOneTimePbkCipher\n    );\n\n    return new LrMutation({\n      mutation: RespondKeyExchangeOtkMutation,\n      variables: {\n        input: {\n          keyExchangeId,\n          keyExchangeToken: token,\n          rootKeyId: rootKey.id,\n          masterKeyId,\n          // These will be stored on the server\n          responderPxkId: responderPrk.id,\n          responderSigPxkId: responderSigPrk.id,\n          signedInitiatorPbk: JSON.stringify(signedInitiatorPbk),\n          signedInitiatorSigPbk: JSON.stringify(signedInitiatorSigPbk),\n          // rkWrappedInitiatorSigPbk: JSON.stringify(rkWrappedInitiatorSigPbk),\n\n          // Option 1: Using new Prk for each TP pair\n          // rkWrappedResponderSigPrk: JSON.stringify(rkWrappedResponderSigPrk),\n          rkWrappedSharedKey: JSON.stringify(rkWrappedSharedKey),\n          mkWrappedMkSharedKey: JSON.stringify(mkWrappedMkSharedKey),\n          // These will be sent to the initiator\n          initiatorOneTimePbkCipher: JSON.stringify(initiatorOneTimePbkCipher),\n          initiatorContactCard: receivedCardInput,\n          responderContactCard: responderCardInput,\n        },\n      },\n    });\n  }\n\n  async completeOtk({\n    keyExchangeId,\n    initiatorRootKeyCipher,\n    initiatorOneTimePbkCipher,\n    responderContactCard,\n    initiatorContactCard,\n  }: CompleteOtkInput2) {\n    const rootKey = await this.keyService.getCurrentRootKey();\n    const masterKey = await this.keyService.getCurrentMasterKey();\n\n    // Decrypt using the root key to get the Prk\n    const initiatorRootKeyCipherClearJson = (await this.encryptionService.decrypt(\n      rootKey.jwk,\n      initiatorRootKeyCipher\n    )) as InitiatorRootKeyCipherClearJson2;\n\n    // The Prk is single-use and only used to send information from the responder back to the initiator.\n    const plainInitiatorOneTimePbkCipher = await this.decryptResponseCipher(\n      await KeyFactoryService.asKey(initiatorRootKeyCipherClearJson.otKey),\n      await KeyFactoryService.asKey(initiatorRootKeyCipherClearJson.oneTimePrk),\n      initiatorOneTimePbkCipher\n    );\n\n    // Check the nonce match to ensure the responder was the one holding the OTK\n    if (\n      initiatorRootKeyCipherClearJson.nonce !==\n      plainInitiatorOneTimePbkCipher.nonce\n    ) {\n      throw new LrCodeMismatchException(\n        'The nonce returned by responder does not match with the one created by the initiator.'\n      );\n    }\n\n    // Option 1: Assuming the signing key is unique between users.\n    // const initiatorSigPrk = await KFS.asKey(ke.plainInitiatorRootKeyCipher.sigPrk);\n    // const rkWrappedInitiatorSigPrk = await this.encrypt(rootKey, initiatorSigPrk.toJSON(true));\n\n    // Option 2: Use the user's global signing key.\n    // In this case the initiatorSigPrk is already a part of the key graph.\n    // So there's nothing to do here.\n\n    // Protected the signing public key of the responder.\n    const initiatorSigPrk = await this.keyService.getCurrentSigPxk();\n    const responderSigPbk = await KeyFactoryService.asKey(\n      plainInitiatorOneTimePbkCipher.responder.sigPbk\n    );\n    const responderPbk = await KeyFactoryService.asKey(\n      plainInitiatorOneTimePbkCipher.responder.pbk\n    );\n\n    const signedResponderPbk = await this.encryptionService.sign(\n      initiatorSigPrk.jwk,\n      responderPbk.toJSON()\n    );\n    const signedResponderSigPbk = await this.encryptionService.sign(\n      initiatorSigPrk.jwk,\n      responderSigPbk.toJSON()\n    );\n\n    const sharedKey = await KeyFactoryService.asKey(\n      plainInitiatorOneTimePbkCipher.sharedKey\n    );\n    const rkWrappedSharedKey = await this.encryptionService.encrypt(\n      rootKey.jwk,\n      sharedKey.toJSON(true)\n    );\n\n    const mkSharedKey = await KeyFactoryService.asKey(\n      plainInitiatorOneTimePbkCipher.mkSharedKey\n    );\n    const mkWrappedMkSharedKey = await this.encryptionService.encrypt(\n      masterKey.jwk,\n      mkSharedKey.toJSON(true)\n    );\n\n    let responderContactCardCipherInput;\n    if (responderContactCard) {\n      // Create key\n      const receiverKey = await this.keyFactory.createKey();\n\n      responderContactCardCipherInput = {\n        receiverWrappedKey: JSON.stringify(\n          await this.encryptionService.encrypt(\n            rootKey.jwk,\n            receiverKey.toJSON(true)\n          )\n        ),\n        receiverWrappingKeyId: rootKey.id,\n        receiverCipherData: JSON.stringify(\n          await this.encryptionService.encrypt(\n            receiverKey,\n            responderContactCard.receiverCipherDataClearJson\n          )\n        ),\n      };\n    }\n\n    // Get the data needed from the initiator's cipher data.\n    let initiatorContactCardCipherInput;\n    let initiatorContactCardSharedCipherInput;\n    if (initiatorRootKeyCipherClearJson.initiatorContactCard) {\n      // The initiatorContactCard created during the creation of the invite and encrypted using the initiator's\n      // root key\n      const initiatorContactCardFromInit =\n        initiatorRootKeyCipherClearJson.initiatorContactCard;\n      const ownerKey = await this.keyFactory.createKey();\n      const sharedCipherKey = await KeyFactoryService.asKey(\n        plainInitiatorOneTimePbkCipher.responder.contactCard.sharedCipherKey\n      );\n\n      const ownerWrappedKey = JSON.stringify(\n        await this.encryptionService.encrypt(rootKey.jwk, ownerKey.toJSON(true))\n      );\n\n      // Allow the initiatorContactCard parameter to override\n      const ownerCipherDataClearJson =\n        initiatorContactCard?.ownerCipherDataClearJson ||\n        initiatorContactCardFromInit.ownerCipherDataClearJson;\n\n      const ownerCipherData = ownerCipherDataClearJson\n        ? await this.keyGraph.encryptToString(\n            ownerKey,\n            ownerCipherDataClearJson\n          )\n        : '';\n\n      initiatorContactCardCipherInput = {\n        ownerWrappedKey,\n        ownerWrappingKeyId: rootKey.id,\n        ownerCipherData,\n      };\n\n      initiatorContactCardSharedCipherInput = {\n        sigPxkId: initiatorSigPrk.id,\n      };\n\n      const sharedCipherData = await this.encryptionService.encrypt(\n        sharedCipherKey,\n        initiatorContactCardFromInit.sharedCipherDataClearJson\n      );\n      initiatorContactCardSharedCipherInput.sharedCipherDataSig = JSON.stringify(\n        await this.encryptionService.sign(initiatorSigPrk.jwk, sharedCipherData)\n      );\n    }\n\n    // TODO ideally we update the shared data in the contact card sent to the responder as well since that\n    // CC was created by the responder.\n\n    return new LrMutation({\n      mutation: CompleteKeyExchangeOtkMutation,\n      variables: {\n        input: {\n          keyExchangeId,\n          rootKeyId: rootKey.id,\n          masterKeyId: masterKey.id,\n          initiatorSigPxkId: initiatorSigPrk.id,\n          signedResponderPbk: JSON.stringify(signedResponderPbk),\n          signedResponderSigPbk: JSON.stringify(signedResponderSigPbk),\n          rkWrappedSharedKey: JSON.stringify(rkWrappedSharedKey),\n          mkWrappedMkSharedKey: JSON.stringify(mkWrappedMkSharedKey),\n          responderContactCardCipher: responderContactCardCipherInput,\n          initiatorContactCardCipher: initiatorContactCardCipherInput,\n          initiatorContactCardSharedCipher: initiatorContactCardSharedCipherInput,\n        },\n      },\n    });\n  }\n}\n"]}