@lifeready/core 1.0.2 → 1.0.4

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (225) hide show
  1. package/README.md +62 -62
  2. package/bundles/lifeready-core.umd.js +15445 -15445
  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 +44 -44
  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/item2.gql.js +110 -110
  18. package/esm2015/lib/api/item2.service.js +311 -311
  19. package/esm2015/lib/api/key-exchange.gql.js +188 -188
  20. package/esm2015/lib/api/key-exchange.service.js +442 -442
  21. package/esm2015/lib/api/key-exchange.types.js +18 -18
  22. package/esm2015/lib/api/key-exchange2.gql.js +171 -171
  23. package/esm2015/lib/api/key-exchange2.service.js +479 -479
  24. package/esm2015/lib/api/lock.gql.js +40 -40
  25. package/esm2015/lib/api/lock.service.js +64 -64
  26. package/esm2015/lib/api/lr-apollo.service.js +46 -46
  27. package/esm2015/lib/api/lr-graphql/index.js +6 -6
  28. package/esm2015/lib/api/lr-graphql/lr-graphql.service.js +155 -155
  29. package/esm2015/lib/api/lr-graphql/lr-merged-mutation.js +213 -213
  30. package/esm2015/lib/api/lr-graphql/lr-mutation-base.js +51 -51
  31. package/esm2015/lib/api/lr-graphql/lr-mutation.js +48 -48
  32. package/esm2015/lib/api/lr-graphql/lr.service.js +18 -18
  33. package/esm2015/lib/api/message.service.js +138 -138
  34. package/esm2015/lib/api/persist.service.js +181 -181
  35. package/esm2015/lib/api/query-processor/common-processors.service.js +93 -93
  36. package/esm2015/lib/api/query-processor/index.js +3 -3
  37. package/esm2015/lib/api/query-processor/query-processor.service.js +192 -192
  38. package/esm2015/lib/api/query-processor/tp-password-reset-processor.service.js +109 -109
  39. package/esm2015/lib/api/shared-contact-card.service.js +119 -119
  40. package/esm2015/lib/api/shared-contact-card2.gql.js +41 -41
  41. package/esm2015/lib/api/shared-contact-card2.service.js +117 -117
  42. package/esm2015/lib/api/time.service.js +146 -146
  43. package/esm2015/lib/api/types/graphql.types.js +7 -7
  44. package/esm2015/lib/api/types/index.js +3 -3
  45. package/esm2015/lib/api/types/lr-graphql.types.js +71 -71
  46. package/esm2015/lib/auth/auth.config.js +57 -57
  47. package/esm2015/lib/auth/auth.gql.js +48 -48
  48. package/esm2015/lib/auth/auth.types.js +27 -27
  49. package/esm2015/lib/auth/idle.service.js +168 -168
  50. package/esm2015/lib/auth/idle.types.js +7 -7
  51. package/esm2015/lib/auth/lbop.service.js +355 -355
  52. package/esm2015/lib/auth/life-ready-auth.service.js +500 -500
  53. package/esm2015/lib/auth/password.service.js +320 -320
  54. package/esm2015/lib/auth/register.service.js +172 -172
  55. package/esm2015/lib/auth/two-factor.service.js +74 -74
  56. package/esm2015/lib/category/category-meta.service.js +99 -99
  57. package/esm2015/lib/category/category.gql.js +406 -406
  58. package/esm2015/lib/category/category.service.js +390 -390
  59. package/esm2015/lib/category/category.types.js +29 -29
  60. package/esm2015/lib/cryptography/cryptography.types.js +11 -11
  61. package/esm2015/lib/cryptography/encryption.service.js +189 -189
  62. package/esm2015/lib/cryptography/key-factory.service.js +237 -237
  63. package/esm2015/lib/cryptography/key-graph.service.js +280 -280
  64. package/esm2015/lib/cryptography/key-meta.service.js +200 -200
  65. package/esm2015/lib/cryptography/key.service.js +124 -124
  66. package/esm2015/lib/cryptography/slip39.service.js +169 -169
  67. package/esm2015/lib/cryptography/web-crypto.service.js +29 -29
  68. package/esm2015/lib/life-ready.config.js +84 -84
  69. package/esm2015/lib/life-ready.module.js +74 -74
  70. package/esm2015/lib/plan/plan.gql.js +123 -123
  71. package/esm2015/lib/plan/plan.service.js +149 -149
  72. package/esm2015/lib/plan/plan.types.js +11 -11
  73. package/esm2015/lib/record/record-attachment.service.js +101 -101
  74. package/esm2015/lib/record/record.gql.js +179 -179
  75. package/esm2015/lib/record/record.service.js +206 -206
  76. package/esm2015/lib/record/record.types.js +15 -15
  77. package/esm2015/lib/record-type/record-type.service.js +75 -75
  78. package/esm2015/lib/record-type/record-type.types.js +28 -28
  79. package/esm2015/lib/scenario/approvals/scenario-approval.gql.js +105 -105
  80. package/esm2015/lib/scenario/approvals/scenario-approval.types.js +1 -1
  81. package/esm2015/lib/scenario/approvals/scenario-approver.service.js +300 -300
  82. package/esm2015/lib/scenario/claimants/scenario-claimant.gql.js +52 -52
  83. package/esm2015/lib/scenario/claimants/scenario-claimant.service.js +97 -97
  84. package/esm2015/lib/scenario/claimants/scenario-claimant.types.js +1 -1
  85. package/esm2015/lib/scenario/receivers/scenario-receiver.gql.js +150 -150
  86. package/esm2015/lib/scenario/receivers/scenario-receiver.service.js +229 -229
  87. package/esm2015/lib/scenario/receivers/scenario-receiver.types.js +1 -1
  88. package/esm2015/lib/scenario/scenario-setup.service.js +269 -269
  89. package/esm2015/lib/scenario/scenario.gql.js +368 -368
  90. package/esm2015/lib/scenario/scenario.service.js +611 -611
  91. package/esm2015/lib/scenario/scenario.types.js +64 -64
  92. package/esm2015/lib/search/search.gql.js +62 -62
  93. package/esm2015/lib/search/search.service.js +156 -156
  94. package/esm2015/lib/search/search.types.js +6 -6
  95. package/esm2015/lib/trusted-parties/tp-password-reset-request.service.js +112 -112
  96. package/esm2015/lib/trusted-parties/tp-password-reset-user.service.js +129 -129
  97. package/esm2015/lib/trusted-parties/tp-password-reset.constants.js +4 -4
  98. package/esm2015/lib/trusted-parties/tp-password-reset.gql.js +232 -232
  99. package/esm2015/lib/trusted-parties/tp-password-reset.service.js +299 -299
  100. package/esm2015/lib/trusted-parties/trusted-party.gql.js +148 -148
  101. package/esm2015/lib/trusted-parties/trusted-party.service.js +326 -326
  102. package/esm2015/lib/trusted-parties/trusted-party.types.js +41 -41
  103. package/esm2015/lib/trusted-parties/trusted-party2.gql.js +87 -87
  104. package/esm2015/lib/trusted-parties/trusted-party2.service.js +215 -215
  105. package/esm2015/lib/users/profile-details.service.js +214 -214
  106. package/esm2015/lib/users/profile.gql.js +97 -97
  107. package/esm2015/lib/users/profile.service.js +169 -169
  108. package/esm2015/lib/users/profile.types.js +34 -34
  109. package/esm2015/lib/users/user.gql.js +60 -60
  110. package/esm2015/lib/users/user.service.js +79 -79
  111. package/esm2015/lib/users/user.types.js +5 -5
  112. package/esm2015/lifeready-core.js +10 -10
  113. package/esm2015/public-api.js +81 -81
  114. package/fesm2015/lifeready-core.js +13088 -13088
  115. package/fesm2015/lifeready-core.js.map +1 -1
  116. package/lib/_common/ast.d.ts +11 -11
  117. package/lib/_common/deferred-promise.d.ts +12 -12
  118. package/lib/_common/exceptions.d.ts +109 -109
  119. package/lib/_common/queries.gql.d.ts +10 -10
  120. package/lib/_common/run-outside-angular.d.ts +14 -14
  121. package/lib/_common/types.d.ts +10 -10
  122. package/lib/_common/utils.d.ts +3 -3
  123. package/lib/api/contact-card.gql.d.ts +7 -7
  124. package/lib/api/contact-card.service.d.ts +52 -52
  125. package/lib/api/contact-card2.gql.d.ts +34 -34
  126. package/lib/api/contact-card2.service.d.ts +49 -49
  127. package/lib/api/file.service.d.ts +18 -18
  128. package/lib/api/item2.gql.d.ts +96 -96
  129. package/lib/api/item2.service.d.ts +177 -177
  130. package/lib/api/key-exchange.gql.d.ts +9 -9
  131. package/lib/api/key-exchange.service.d.ts +39 -39
  132. package/lib/api/key-exchange.types.d.ts +196 -196
  133. package/lib/api/key-exchange2.gql.d.ts +125 -125
  134. package/lib/api/key-exchange2.service.d.ts +187 -187
  135. package/lib/api/lock.gql.d.ts +27 -27
  136. package/lib/api/lock.service.d.ts +25 -25
  137. package/lib/api/lr-apollo.service.d.ts +15 -15
  138. package/lib/api/lr-graphql/index.d.ts +5 -5
  139. package/lib/api/lr-graphql/lr-graphql.service.d.ts +60 -60
  140. package/lib/api/lr-graphql/lr-merged-mutation.d.ts +27 -27
  141. package/lib/api/lr-graphql/lr-mutation-base.d.ts +28 -28
  142. package/lib/api/lr-graphql/lr-mutation.d.ts +8 -8
  143. package/lib/api/lr-graphql/lr.service.d.ts +9 -9
  144. package/lib/api/message.service.d.ts +58 -58
  145. package/lib/api/persist.service.d.ts +31 -31
  146. package/lib/api/query-processor/common-processors.service.d.ts +36 -36
  147. package/lib/api/query-processor/index.d.ts +2 -2
  148. package/lib/api/query-processor/query-processor.service.d.ts +18 -18
  149. package/lib/api/query-processor/tp-password-reset-processor.service.d.ts +15 -15
  150. package/lib/api/shared-contact-card.service.d.ts +33 -33
  151. package/lib/api/shared-contact-card2.gql.d.ts +36 -36
  152. package/lib/api/shared-contact-card2.service.d.ts +45 -45
  153. package/lib/api/time.service.d.ts +16 -16
  154. package/lib/api/types/graphql.types.d.ts +29 -29
  155. package/lib/api/types/index.d.ts +2 -2
  156. package/lib/api/types/lr-graphql.types.d.ts +385 -385
  157. package/lib/auth/auth.config.d.ts +5 -5
  158. package/lib/auth/auth.gql.d.ts +15 -15
  159. package/lib/auth/auth.types.d.ts +66 -66
  160. package/lib/auth/idle.service.d.ts +40 -40
  161. package/lib/auth/idle.types.d.ts +10 -10
  162. package/lib/auth/lbop.service.d.ts +91 -91
  163. package/lib/auth/life-ready-auth.service.d.ts +59 -59
  164. package/lib/auth/password.service.d.ts +78 -78
  165. package/lib/auth/register.service.d.ts +25 -25
  166. package/lib/auth/two-factor.service.d.ts +15 -15
  167. package/lib/category/category-meta.service.d.ts +23 -23
  168. package/lib/category/category.gql.d.ts +45 -45
  169. package/lib/category/category.service.d.ts +67 -67
  170. package/lib/category/category.types.d.ts +79 -79
  171. package/lib/cryptography/cryptography.types.d.ts +83 -83
  172. package/lib/cryptography/encryption.service.d.ts +41 -41
  173. package/lib/cryptography/key-factory.service.d.ts +38 -38
  174. package/lib/cryptography/key-graph.service.d.ts +33 -33
  175. package/lib/cryptography/key-meta.service.d.ts +44 -44
  176. package/lib/cryptography/key.service.d.ts +36 -36
  177. package/lib/cryptography/slip39.service.d.ts +43 -43
  178. package/lib/cryptography/web-crypto.service.d.ts +5 -5
  179. package/lib/life-ready.config.d.ts +14 -14
  180. package/lib/life-ready.module.d.ts +5 -5
  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/approvals/scenario-approval.gql.d.ts +7 -7
  191. package/lib/scenario/approvals/scenario-approval.types.d.ts +63 -63
  192. package/lib/scenario/approvals/scenario-approver.service.d.ts +32 -32
  193. package/lib/scenario/claimants/scenario-claimant.gql.d.ts +5 -5
  194. package/lib/scenario/claimants/scenario-claimant.service.d.ts +17 -17
  195. package/lib/scenario/claimants/scenario-claimant.types.d.ts +18 -18
  196. package/lib/scenario/receivers/scenario-receiver.gql.d.ts +8 -8
  197. package/lib/scenario/receivers/scenario-receiver.service.d.ts +30 -30
  198. package/lib/scenario/receivers/scenario-receiver.types.d.ts +54 -54
  199. package/lib/scenario/scenario-setup.service.d.ts +22 -22
  200. package/lib/scenario/scenario.gql.d.ts +34 -34
  201. package/lib/scenario/scenario.service.d.ts +58 -58
  202. package/lib/scenario/scenario.types.d.ts +217 -217
  203. package/lib/search/search.gql.d.ts +1 -1
  204. package/lib/search/search.service.d.ts +25 -25
  205. package/lib/search/search.types.d.ts +20 -20
  206. package/lib/trusted-parties/tp-password-reset-request.service.d.ts +20 -20
  207. package/lib/trusted-parties/tp-password-reset-user.service.d.ts +35 -35
  208. package/lib/trusted-parties/tp-password-reset.constants.d.ts +3 -3
  209. package/lib/trusted-parties/tp-password-reset.gql.d.ts +218 -218
  210. package/lib/trusted-parties/tp-password-reset.service.d.ts +130 -130
  211. package/lib/trusted-parties/trusted-party.gql.d.ts +9 -9
  212. package/lib/trusted-parties/trusted-party.service.d.ts +44 -44
  213. package/lib/trusted-parties/trusted-party.types.d.ts +102 -102
  214. package/lib/trusted-parties/trusted-party2.gql.d.ts +79 -79
  215. package/lib/trusted-parties/trusted-party2.service.d.ts +114 -114
  216. package/lib/users/profile-details.service.d.ts +21 -21
  217. package/lib/users/profile.gql.d.ts +11 -11
  218. package/lib/users/profile.service.d.ts +35 -35
  219. package/lib/users/profile.types.d.ts +96 -96
  220. package/lib/users/user.gql.d.ts +9 -9
  221. package/lib/users/user.service.d.ts +12 -12
  222. package/lib/users/user.types.d.ts +23 -23
  223. package/lifeready-core.d.ts +9 -9
  224. package/package.json +1 -1
  225. package/public-api.d.ts +77 -77
@@ -1,442 +1,442 @@
1
- import { __awaiter } from "tslib";
2
- import { Injectable } from '@angular/core';
3
- import { LifeReadyAuthService } from '../auth/life-ready-auth.service';
4
- import { EncryptionService, JoseSerialization, } from '../cryptography/encryption.service';
5
- import { KeyService } from '../cryptography/key.service';
6
- import { LrCodeMismatchException } from '../_common/exceptions';
7
- import { UserService } from './../users/user.service';
8
- import { CompleteOtkMutation, CurrentUserSharedKeyQuery, InitiateOtkMutation, KeyExchangeQuery, KeyExchangesQuery, KeyExchangeTokenQuery, RespondOtkMutation, } from './key-exchange.gql';
9
- import { OtkState, } from './key-exchange.types';
10
- import { LrApolloService } from './lr-apollo.service';
11
- import { KeyFactoryService as KFS } from '../cryptography/key-factory.service';
12
- // Ref: https://stackoverflow.com/questions/59735280/angular-8-moment-error-cannot-call-a-namespace-moment
13
- import * as moment_ from 'moment';
14
- import * as i0 from "@angular/core";
15
- import * as i1 from "../cryptography/key-factory.service";
16
- import * as i2 from "../cryptography/key.service";
17
- import * as i3 from "./lr-apollo.service";
18
- import * as i4 from "../cryptography/encryption.service";
19
- import * as i5 from "../auth/life-ready-auth.service";
20
- import * as i6 from "../users/user.service";
21
- const moment = moment_;
22
- export class KeyExchangeService {
23
- constructor(keyFactory, keyService, lrApollo, encryptionService, authService, userService) {
24
- this.keyFactory = keyFactory;
25
- this.keyService = keyService;
26
- this.lrApollo = lrApollo;
27
- this.encryptionService = encryptionService;
28
- this.authService = authService;
29
- this.userService = userService;
30
- this.CLIENT_NONCE_LENGTH = 32;
31
- }
32
- getKeyExchangeList(input = {}) {
33
- return __awaiter(this, void 0, void 0, function* () {
34
- const { keyExchanges } = yield this.lrApollo.query({
35
- query: KeyExchangesQuery,
36
- variables: Object.assign({}, input),
37
- });
38
- return keyExchanges;
39
- });
40
- }
41
- /**
42
- * @param id If the current user can responder the key exchange if they are either the initiator or the receiver.
43
- * @param token If not signed in, or not the initiator or responder, 'token' must be given.
44
- * @param otKeyK Is the raw one-time key (string). If the responder is explicitly specified at time of initiation, then
45
- * it's possible to have the otKey wrapped by the public key of the responder. In which case, the otKeyK is not needed.
46
- */
47
- getKeyExchange(id, { otKeyK, token } = {}) {
48
- return __awaiter(this, void 0, void 0, function* () {
49
- const { keyExchange } = yield this.lrApollo.query({
50
- query: token ? KeyExchangeTokenQuery : KeyExchangeQuery,
51
- variables: {
52
- id,
53
- token,
54
- },
55
- });
56
- return yield this.decryptKeyExchange(keyExchange, otKeyK);
57
- });
58
- }
59
- decryptResponseCipher(otKey, otPrk, content) {
60
- return __awaiter(this, void 0, void 0, function* () {
61
- // The response could be wrapped by the OtK as well as we the OtPrk
62
- try {
63
- content = yield this.encryptionService.decrypt(otKey, content);
64
- }
65
- catch (error) {
66
- if (error.message !== 'no key found') {
67
- throw error;
68
- }
69
- // Do nothing to support older versions where message is not wrapped with otk.
70
- }
71
- // The Prk is single-use and only used to send information from the responder back to the initiator.
72
- return yield this.encryptionService.decrypt(otPrk, content);
73
- });
74
- }
75
- decryptKeyExchange(keyExchange, otKeyK) {
76
- return __awaiter(this, void 0, void 0, function* () {
77
- if (keyExchange.isInitiator) {
78
- const rootKey = yield this.keyService.getCurrentRootKey();
79
- // Decrypt using the root key to get the Prk
80
- const plainInitiatorRootKeyCipher = (yield this.encryptionService.decrypt(rootKey.jwk, keyExchange.initiatorRootKeyCipher));
81
- const plainInitiatorOneTimePbkCipher = keyExchange.otk
82
- .initiatorOneTimePbkCipher
83
- ? yield this.decryptResponseCipher(yield KFS.asKey(plainInitiatorRootKeyCipher.otKey), yield KFS.asKey(plainInitiatorRootKeyCipher.oneTimePrk), keyExchange.otk.initiatorOneTimePbkCipher)
84
- : null;
85
- const responder = plainInitiatorOneTimePbkCipher &&
86
- plainInitiatorOneTimePbkCipher.responder;
87
- const initiator = plainInitiatorRootKeyCipher && plainInitiatorRootKeyCipher.initiator;
88
- return Object.assign(Object.assign({}, keyExchange), { message: responder ? responder.message : null, contactCard: responder && responder.contactCard
89
- ? responder.contactCard.plainSharedCipherDataJson
90
- : null, myContactCard: initiator && initiator.contactCard
91
- ? initiator.contactCard.plainSharedCipherDataJson
92
- : null, myMessage: initiator && initiator.message });
93
- }
94
- else {
95
- const decryptedOtk = yield this.decryptOtk(keyExchange, otKeyK);
96
- const initiator = decryptedOtk && decryptedOtk.plainOtKeyCipher.initiator;
97
- return Object.assign(Object.assign({}, keyExchange), { decryptedOtk, message: initiator && initiator.message, contactCard: initiator &&
98
- initiator.contactCard &&
99
- initiator.contactCard.plainSharedCipherDataJson });
100
- }
101
- });
102
- }
103
- decryptOtk(keyExchange, otKeyK) {
104
- return __awaiter(this, void 0, void 0, function* () {
105
- const otKey = yield this.getOtKey(keyExchange, otKeyK);
106
- return otKey && keyExchange.otk.otKeyCipher
107
- ? {
108
- plainOtKeyCipher: yield this.encryptionService.decrypt(otKey, keyExchange.otk.otKeyCipher),
109
- otKey,
110
- }
111
- : null;
112
- });
113
- }
114
- getOtKey(keyExchange, otKeyK) {
115
- return __awaiter(this, void 0, void 0, function* () {
116
- if (otKeyK) {
117
- return yield KFS.asKey(Object.assign(Object.assign({}, JSON.parse(keyExchange.otk.otKeyParams)), { k: otKeyK }));
118
- }
119
- else if (keyExchange.otk.state === OtkState.OTK_INITIATED &&
120
- !keyExchange.isInitiator &&
121
- keyExchange.otk.responderPbkCipher) {
122
- // Assuming existing user getting invited where OTK is wrapped in responder's public key.
123
- const prk = yield this.keyService.getCurrentPxk();
124
- const decryptedCipher = yield this.encryptionService.decrypt(prk.jwk, JSON.parse(keyExchange.otk.responderPbkCipher), {
125
- serializations: [JoseSerialization.COMPACT],
126
- });
127
- if (decryptedCipher.otKey) {
128
- return yield KFS.asKey(decryptedCipher.otKey);
129
- }
130
- }
131
- return null;
132
- });
133
- }
134
- initiateOtk({ message, email, contactCard, upgrade, }) {
135
- return __awaiter(this, void 0, void 0, function* () {
136
- const otKey = yield this.keyFactory.createKey();
137
- const nonce = this.keyFactory.randomString(this.CLIENT_NONCE_LENGTH);
138
- const user = yield this.authService.getUser();
139
- // New PKC key for encryption. This key is used only once when the responder sends
140
- // back their signing public key.
141
- const initiatorOneTimePrk = yield this.keyFactory.createPkcKey();
142
- // Option 1: New PKC key for signing
143
- // const initiatorSigPrk = await this.keyService.createPkcSignKey();
144
- // Option 2: Use the user's global signing key.
145
- // This key is used to prove the initiator's identity.
146
- const initiatorPrk = yield this.keyService.getCurrentPxk();
147
- const initiatorSigPrk = yield this.keyService.getCurrentSigPxk();
148
- let initiatorPlainDataSig = null;
149
- if (contactCard && contactCard.ownerPlainData) {
150
- initiatorPlainDataSig = JSON.stringify(yield this.encryptionService.sign(initiatorSigPrk.jwk, contactCard.ownerPlainData));
151
- }
152
- const initiator = {
153
- message,
154
- contactCard: contactCard
155
- ? {
156
- plainSharedCipherDataJson: contactCard.plainSharedCipherDataJson,
157
- }
158
- : null,
159
- };
160
- // Content to be encrypted using the OTK.
161
- const plainOtKeyCipher = {
162
- nonce,
163
- initiator: Object.assign(Object.assign({}, initiator), { oneTimePbk: initiatorOneTimePrk.toJSON(), pbk: initiatorPrk.jwk.toJSON(), sigPbk: initiatorSigPrk.jwk.toJSON(), profile: {
164
- username: user.username,
165
- } }),
166
- };
167
- const otKeyCipher = yield this.encryptionService.encrypt(otKey, plainOtKeyCipher);
168
- // Content to be encrypted using the initiator's root key.
169
- const plainInitiatorRootKeyCipher = {
170
- nonce,
171
- oneTimePrk: initiatorOneTimePrk.toJSON(true),
172
- // Should not need to keep this encrypted since we are using the global signing key.
173
- // sigPrk: initiatorSigPrk.toJSON(true),
174
- // Save it in case the initiator want to decode the otKeyCipher.
175
- // Since the otKey is only used once, and that otKeyCipher contains only
176
- // the public key of the initiator, it's safe just leave the otKey stored here.
177
- otKey: otKey.toJSON(true),
178
- // These should be storing information such as how the fields of the shared contact card is
179
- // derived from the master contact card.
180
- initiatorContactCard: contactCard,
181
- initiator,
182
- };
183
- const rootKey = yield this.keyService.getCurrentRootKey();
184
- const initiatorRootKeyCipher = yield this.encryptionService.encrypt(rootKey.jwk, plainInitiatorRootKeyCipher);
185
- // The raw OTK
186
- const otKeyK = otKey.toJSON(true).k;
187
- // API call
188
- const { initiateKeyExchangeOtk } = yield this.lrApollo.mutate({
189
- mutation: InitiateOtkMutation,
190
- variables: {
191
- input: {
192
- // These will be stored on the server
193
- initiatorRootKeyCipher: JSON.stringify(initiatorRootKeyCipher),
194
- initiatorPxkId: initiatorPrk.id,
195
- initiatorSigPxkId: initiatorSigPrk.id,
196
- // These will be sent to the responder
197
- otKeyParams: JSON.stringify(otKey.toJSON()),
198
- otKeyCipher: JSON.stringify(otKeyCipher),
199
- sendEmail: email
200
- ? {
201
- email,
202
- rawOtKey: otKeyK,
203
- }
204
- : null,
205
- createTp: true,
206
- initiatorPlainDataSig,
207
- upgrade,
208
- },
209
- },
210
- });
211
- return { keyExchange: initiateKeyExchangeOtk.keyExchange, otKeyK };
212
- });
213
- }
214
- respondOtk({ id, token, decryptedOtk, message, initiatorContactCard, responderContactCard: sentContactCard, }) {
215
- return __awaiter(this, void 0, void 0, function* () {
216
- const user = yield this.authService.getUser();
217
- const rootKey = yield this.keyService.getCurrentRootKey();
218
- const masterKeyId = this.keyService.getCurrentMasterKey().id;
219
- const masterKey = yield this.keyService.getCurrentMasterKey();
220
- const sharedKey = yield this.keyFactory.createKey();
221
- const mkSharedKey = yield this.keyFactory.createKey();
222
- const rkWrappedSharedKey = yield this.encryptionService.encrypt(rootKey.jwk, sharedKey.toJSON(true));
223
- const mkWrappedMkSharedKey = yield this.encryptionService.encrypt(masterKey.jwk, mkSharedKey.toJSON(true));
224
- const initiatorOneTimePbk = yield KFS.asKey(decryptedOtk.plainOtKeyCipher.initiator.oneTimePbk);
225
- const initiatorPbk = yield KFS.asKey(decryptedOtk.plainOtKeyCipher.initiator.pbk);
226
- const initiatorSigPbk = yield KFS.asKey(decryptedOtk.plainOtKeyCipher.initiator.sigPbk);
227
- // Option 1: Using new Prk for each TP pair
228
- // Create a new public signing key for the responder.
229
- // const responderSigPrk = await this.keyService.createPkcSignKey()
230
- // const rkWrappedResponderSigPrk = await this.encrypt(rootKey, responderSigPrk.toJSON(true));
231
- // Option 2: Responder already has a signing Prk
232
- const responderPrk = yield this.keyService.getCurrentPxk();
233
- const responderSigPrk = yield this.keyService.getCurrentSigPxk();
234
- const signedInitiatorPbk = yield this.encryptionService.sign(responderSigPrk.jwk, initiatorPbk.toJSON());
235
- const signedInitiatorSigPbk = yield this.encryptionService.sign(responderSigPrk.jwk, initiatorSigPbk.toJSON());
236
- const plainInitiatorOneTimePbkCipher = {
237
- nonce: decryptedOtk.plainOtKeyCipher.nonce,
238
- sharedKey: sharedKey.toJSON(true),
239
- mkSharedKey: mkSharedKey.toJSON(true),
240
- responder: {
241
- pbk: responderPrk.jwk.toJSON(),
242
- sigPbk: responderSigPrk.jwk.toJSON(),
243
- profile: {
244
- username: user.username,
245
- },
246
- message,
247
- },
248
- };
249
- let receivedCardInput;
250
- if (decryptedOtk.plainOtKeyCipher.initiator.contactCard) {
251
- // Set the info about the initiator to be the ones sent by the initiator. We need th responder to do the encryption here
252
- // because the initiator does not have the shared key yet, and we want the responder to have a functional contact card after
253
- // this exchange. The initiator can double check the contact details are correct and sign it when it completes the exchange.
254
- const plainSharedCipherDataJson = decryptedOtk.plainOtKeyCipher.initiator.contactCard
255
- .plainSharedCipherDataJson;
256
- // Create keys
257
- const receiverKey = yield this.keyFactory.createKey();
258
- const ccSharedKey = yield this.keyFactory.createKey();
259
- const sigPxk = yield this.keyService.getCurrentSigPxk();
260
- receivedCardInput = {
261
- receiverWrappedKey: JSON.stringify(yield this.encryptionService.encrypt(rootKey.jwk, receiverKey.toJSON(true))),
262
- receiverWrappingKeyId: rootKey.id,
263
- receiverCipherData: initiatorContactCard
264
- ? JSON.stringify(yield this.encryptionService.encrypt(receiverKey, initiatorContactCard.plainReceiverCipherDataJson))
265
- : '',
266
- sharedWrappedKey: JSON.stringify(yield this.encryptionService.encrypt(sharedKey, ccSharedKey.toJSON(true))),
267
- };
268
- const sharedCipherData = yield this.encryptionService.encrypt(ccSharedKey, plainSharedCipherDataJson);
269
- receivedCardInput.sharedCipherDataSig = JSON.stringify(yield this.encryptionService.sign(sigPxk.jwk, sharedCipherData));
270
- receivedCardInput.sigPxkId = sigPxk.id;
271
- plainInitiatorOneTimePbkCipher.responder.contactCard = Object.assign(Object.assign({}, plainInitiatorOneTimePbkCipher.responder.contactCard), { sharedCipherKey: ccSharedKey.toJSON(true) });
272
- }
273
- let sentCardInput;
274
- if (sentContactCard) {
275
- // Create keys
276
- const ownerKey = yield this.keyFactory.createKey();
277
- const ccSharedKey = yield this.keyFactory.createKey();
278
- const sigPxk = yield this.keyService.getCurrentSigPxk();
279
- sentCardInput = {
280
- ownerWrappedKey: JSON.stringify(yield this.encryptionService.encrypt(rootKey.jwk, ownerKey.toJSON(true))),
281
- ownerWrappingKeyId: rootKey.id,
282
- ownerCipherData: sentContactCard.plainOwnerCipherDataJson
283
- ? JSON.stringify(yield this.encryptionService.encrypt(ownerKey, sentContactCard.plainOwnerCipherDataJson))
284
- : '',
285
- sharedWrappedKey: JSON.stringify(yield this.encryptionService.encrypt(sharedKey, ccSharedKey.toJSON(true))),
286
- };
287
- const sharedCipherData = yield this.encryptionService.encrypt(ccSharedKey, sentContactCard.plainSharedCipherDataJson);
288
- sentCardInput.sharedCipherDataSig = JSON.stringify(yield this.encryptionService.sign(sigPxk.jwk, sharedCipherData));
289
- sentCardInput.sigPxkId = sigPxk.id;
290
- if (sentContactCard.ownerPlainData) {
291
- sentCardInput.ownerPlainDataSig = JSON.stringify(yield this.encryptionService.sign(responderSigPrk.jwk, sentContactCard.ownerPlainData));
292
- }
293
- // Contact card info readable by the initiator
294
- plainInitiatorOneTimePbkCipher.responder.contactCard = Object.assign(Object.assign({}, plainInitiatorOneTimePbkCipher.responder.contactCard), { plainSharedCipherDataJson: sentContactCard.plainSharedCipherDataJson });
295
- }
296
- // Encrypt with one-time public key
297
- let initiatorOneTimePbkCipher = yield this.encryptionService.encrypt(initiatorOneTimePbk, plainInitiatorOneTimePbkCipher);
298
- // Encrypt with the otk again to keep use of asymmetric keys to a minimum.
299
- initiatorOneTimePbkCipher = yield this.encryptionService.encrypt(decryptedOtk.otKey, initiatorOneTimePbkCipher);
300
- const { respondKeyExchangeOtk } = yield this.lrApollo.mutate({
301
- mutation: RespondOtkMutation,
302
- variables: {
303
- input: {
304
- keyExchangeId: id,
305
- keyExchangeToken: token,
306
- rootKeyId: rootKey.id,
307
- masterKeyId,
308
- // These will be stored on the server
309
- responderPxkId: responderPrk.id,
310
- responderSigPxkId: responderSigPrk.id,
311
- signedInitiatorPbk: JSON.stringify(signedInitiatorPbk),
312
- signedInitiatorSigPbk: JSON.stringify(signedInitiatorSigPbk),
313
- // rkWrappedInitiatorSigPbk: JSON.stringify(rkWrappedInitiatorSigPbk),
314
- // Option 1: Using new Prk for each TP pair
315
- // rkWrappedResponderSigPrk: JSON.stringify(rkWrappedResponderSigPrk),
316
- rkWrappedSharedKey: JSON.stringify(rkWrappedSharedKey),
317
- mkWrappedMkSharedKey: JSON.stringify(mkWrappedMkSharedKey),
318
- // These will be sent to the initiator
319
- initiatorOneTimePbkCipher: JSON.stringify(initiatorOneTimePbkCipher),
320
- initiatorContactCard: receivedCardInput,
321
- responderContactCard: sentCardInput,
322
- },
323
- },
324
- });
325
- return {
326
- keyExchange: respondKeyExchangeOtk.keyExchange,
327
- userSharedKey: respondKeyExchangeOtk.userSharedKey,
328
- tp: respondKeyExchangeOtk.tp,
329
- };
330
- });
331
- }
332
- completeOtk(keyExchangeId, initiatorRootKeyCipher, initiatorOneTimePbkCipher, responderContactCard) {
333
- return __awaiter(this, void 0, void 0, function* () {
334
- const rootKey = yield this.keyService.getCurrentRootKey();
335
- const masterKey = yield this.keyService.getCurrentMasterKey();
336
- // Decrypt using the root key to get the Prk
337
- const plainInitiatorRootKeyCipher = (yield this.encryptionService.decrypt(rootKey.jwk, initiatorRootKeyCipher));
338
- // The Prk is single-use and only used to send information from the responder back to the initiator.
339
- const plainInitiatorOneTimePbkCipher = yield this.decryptResponseCipher(yield KFS.asKey(plainInitiatorRootKeyCipher.otKey), yield KFS.asKey(plainInitiatorRootKeyCipher.oneTimePrk), initiatorOneTimePbkCipher);
340
- // Check the nonce match to ensure the responder was the one holding the OTK
341
- if (plainInitiatorRootKeyCipher.nonce !== plainInitiatorOneTimePbkCipher.nonce) {
342
- throw new LrCodeMismatchException('The nonce returned by responder does not match with the one created by the initiator.');
343
- }
344
- // Option 1: Assuming the signing key is unique between users.
345
- // const initiatorSigPrk = await KFS.asKey(ke.plainInitiatorRootKeyCipher.sigPrk);
346
- // const rkWrappedInitiatorSigPrk = await this.encrypt(rootKey, initiatorSigPrk.toJSON(true));
347
- // Option 2: Use the user's global signing key.
348
- // In this case the initiatorSigPrk is already a part of the key graph.
349
- // So there's nothing to do here.
350
- // Protected the signing public key of the responder.
351
- const initiatorSigPrk = yield this.keyService.getCurrentSigPxk();
352
- const responderSigPbk = yield KFS.asKey(plainInitiatorOneTimePbkCipher.responder.sigPbk);
353
- const responderPbk = yield KFS.asKey(plainInitiatorOneTimePbkCipher.responder.pbk);
354
- const signedResponderPbk = yield this.encryptionService.sign(initiatorSigPrk.jwk, responderPbk.toJSON());
355
- const signedResponderSigPbk = yield this.encryptionService.sign(initiatorSigPrk.jwk, responderSigPbk.toJSON());
356
- const sharedKey = yield KFS.asKey(plainInitiatorOneTimePbkCipher.sharedKey);
357
- const rkWrappedSharedKey = yield this.encryptionService.encrypt(rootKey.jwk, sharedKey.toJSON(true));
358
- const mkSharedKey = yield KFS.asKey(plainInitiatorOneTimePbkCipher.mkSharedKey);
359
- const mkWrappedMkSharedKey = yield this.encryptionService.encrypt(masterKey.jwk, mkSharedKey.toJSON(true));
360
- let responderContactCardCipherInput;
361
- if (responderContactCard) {
362
- // Create key
363
- const receiverKey = yield this.keyFactory.createKey();
364
- responderContactCardCipherInput = {
365
- receiverWrappedKey: JSON.stringify(yield this.encryptionService.encrypt(rootKey.jwk, receiverKey.toJSON(true))),
366
- receiverWrappingKeyId: rootKey.id,
367
- receiverCipherData: JSON.stringify(yield this.encryptionService.encrypt(receiverKey, responderContactCard)),
368
- };
369
- }
370
- // Get the data needed from the initiator's cipher data.
371
- let initiatorContactCardCipherInput;
372
- let initiatorContactCardSharedCipherInput;
373
- if (plainInitiatorRootKeyCipher.initiatorContactCard) {
374
- const initiatorContactCard = plainInitiatorRootKeyCipher.initiatorContactCard;
375
- const ownerKey = yield this.keyFactory.createKey();
376
- const sharedCipherKey = yield KFS.asKey(plainInitiatorOneTimePbkCipher.responder.contactCard.sharedCipherKey);
377
- const ownerWrappedKey = JSON.stringify(yield this.encryptionService.encrypt(rootKey.jwk, ownerKey.toJSON(true)));
378
- const ownerCipherData = initiatorContactCard.plainOwnerCipherDataJson
379
- ? JSON.stringify(yield this.encryptionService.encrypt(ownerKey, initiatorContactCard.plainOwnerCipherDataJson))
380
- : '';
381
- initiatorContactCardCipherInput = {
382
- ownerWrappedKey,
383
- ownerWrappingKeyId: rootKey.id,
384
- ownerCipherData,
385
- };
386
- initiatorContactCardSharedCipherInput = {
387
- sigPxkId: initiatorSigPrk.id,
388
- };
389
- const sharedCipherData = yield this.encryptionService.encrypt(sharedCipherKey, initiatorContactCard.plainSharedCipherDataJson);
390
- initiatorContactCardSharedCipherInput.sharedCipherDataSig = JSON.stringify(yield this.encryptionService.sign(initiatorSigPrk.jwk, sharedCipherData));
391
- }
392
- // TODO ideally we update the shared data in the contact card sent to the responder as well since that
393
- // CC was created by the responder.
394
- const res = yield this.lrApollo.mutate({
395
- mutation: CompleteOtkMutation,
396
- variables: {
397
- input: {
398
- keyExchangeId,
399
- rootKeyId: rootKey.id,
400
- masterKeyId: masterKey.id,
401
- initiatorSigPxkId: initiatorSigPrk.id,
402
- signedResponderPbk: JSON.stringify(signedResponderPbk),
403
- signedResponderSigPbk: JSON.stringify(signedResponderSigPbk),
404
- rkWrappedSharedKey: JSON.stringify(rkWrappedSharedKey),
405
- mkWrappedMkSharedKey: JSON.stringify(mkWrappedMkSharedKey),
406
- responderContactCardCipher: responderContactCardCipherInput,
407
- initiatorContactCardCipher: initiatorContactCardCipherInput,
408
- initiatorContactCardSharedCipher: initiatorContactCardSharedCipherInput,
409
- },
410
- },
411
- });
412
- return res.completeKeyExchangeOtk;
413
- });
414
- }
415
- currentUserSharedKey(input) {
416
- return __awaiter(this, void 0, void 0, function* () {
417
- const { currentUserSharedKey } = yield this.lrApollo.query({
418
- query: CurrentUserSharedKeyQuery,
419
- variables: {
420
- username: input.username,
421
- userId: input.userId,
422
- },
423
- });
424
- return currentUserSharedKey.userSharedKey;
425
- });
426
- }
427
- }
428
- KeyExchangeService.ɵprov = i0.ɵɵdefineInjectable({ factory: function KeyExchangeService_Factory() { return new KeyExchangeService(i0.ɵɵinject(i1.KeyFactoryService), i0.ɵɵinject(i2.KeyService), i0.ɵɵinject(i3.LrApolloService), i0.ɵɵinject(i4.EncryptionService), i0.ɵɵinject(i5.LifeReadyAuthService), i0.ɵɵinject(i6.UserService)); }, token: KeyExchangeService, providedIn: "root" });
429
- KeyExchangeService.decorators = [
430
- { type: Injectable, args: [{
431
- providedIn: 'root',
432
- },] }
433
- ];
434
- KeyExchangeService.ctorParameters = () => [
435
- { type: KFS },
436
- { type: KeyService },
437
- { type: LrApolloService },
438
- { type: EncryptionService },
439
- { type: LifeReadyAuthService },
440
- { type: UserService }
441
- ];
442
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"key-exchange.service.js","sourceRoot":"/opt/atlassian/pipelines/agent/build/projects/core/src/","sources":["lib/api/key-exchange.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { JWK } from 'node-jose';\nimport { LifeReadyAuthService } from '../auth/life-ready-auth.service';\nimport { KeyGraphResponse } from '../cryptography/cryptography.types';\nimport {\n  EncryptionService,\n  JoseSerialization,\n} from '../cryptography/encryption.service';\nimport { KeyService } from '../cryptography/key.service';\nimport { LrCodeMismatchException } from '../_common/exceptions';\nimport { UserService } from './../users/user.service';\nimport {\n  CompleteOtkMutation,\n  CurrentUserSharedKeyQuery,\n  InitiateOtkMutation,\n  KeyExchangeQuery,\n  KeyExchangesQuery,\n  KeyExchangeTokenQuery,\n  RespondOtkMutation,\n} from './key-exchange.gql';\nimport {\n  CompleteOtk,\n  DecryptedKeyExchange,\n  DecryptedOtk,\n  GetKeyExchangeListOptions,\n  GetKeyExchangeOptions,\n  InitiateOtkInput,\n  KeyExchange,\n  OtkState,\n  PlainInitiatorOneTimePbkCipher,\n  PlainInitiatorRootKeyCipher,\n  PlainOtKeyCipher,\n  PlainResponderPbkCipher,\n  RespondOtk,\n  RespondOtkInput,\n  UserSharedKey,\n} from './key-exchange.types';\nimport { LrApolloService } from './lr-apollo.service';\nimport { KeyFactoryService as KFS } from '../cryptography/key-factory.service';\n// Ref: https://stackoverflow.com/questions/59735280/angular-8-moment-error-cannot-call-a-namespace-moment\nimport * as moment_ from 'moment';\nconst moment = moment_;\n\n@Injectable({\n  providedIn: 'root',\n})\nexport class KeyExchangeService {\n  private readonly CLIENT_NONCE_LENGTH = 32;\n\n  constructor(\n    private keyFactory: KFS,\n    private keyService: KeyService,\n    private lrApollo: LrApolloService,\n    private encryptionService: EncryptionService,\n    private authService: LifeReadyAuthService,\n    private userService: UserService\n  ) {}\n\n  public async getKeyExchangeList(\n    input: GetKeyExchangeListOptions = {}\n  ): Promise<any> {\n    const { keyExchanges } = await this.lrApollo.query<{\n      keyExchanges: any;\n      keyGraph: KeyGraphResponse;\n    }>({\n      query: KeyExchangesQuery,\n      variables: {\n        ...input,\n      },\n    });\n    return keyExchanges;\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  public async getKeyExchange(\n    id: string,\n    { otKeyK, token }: GetKeyExchangeOptions = {}\n  ): Promise<DecryptedKeyExchange> {\n    const { keyExchange } = await this.lrApollo.query<{\n      keyExchange: KeyExchange;\n      keyGraph: KeyGraphResponse;\n    }>({\n      query: token ? KeyExchangeTokenQuery : KeyExchangeQuery,\n      variables: {\n        id,\n        token,\n      },\n    });\n    return await this.decryptKeyExchange(keyExchange, otKeyK);\n  }\n\n  private async decryptResponseCipher(\n    otKey: JWK.Key,\n    otPrk: JWK.Key,\n    content: any\n  ): Promise<PlainInitiatorOneTimePbkCipher> {\n    // The response could be wrapped by the OtK as well as we the OtPrk\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  public async decryptKeyExchange(\n    keyExchange: KeyExchange,\n    otKeyK?: string\n  ): Promise<DecryptedKeyExchange> {\n    if (keyExchange.isInitiator) {\n      const rootKey = await this.keyService.getCurrentRootKey();\n      // Decrypt using the root key to get the Prk\n      const plainInitiatorRootKeyCipher = ((await this.encryptionService.decrypt(\n        rootKey.jwk,\n        keyExchange.initiatorRootKeyCipher\n      )) as unknown) as PlainInitiatorRootKeyCipher;\n\n      const plainInitiatorOneTimePbkCipher = keyExchange.otk\n        .initiatorOneTimePbkCipher\n        ? await this.decryptResponseCipher(\n            await KFS.asKey(plainInitiatorRootKeyCipher.otKey),\n            await KFS.asKey(plainInitiatorRootKeyCipher.oneTimePrk),\n            keyExchange.otk.initiatorOneTimePbkCipher\n          )\n        : null;\n\n      const responder =\n        plainInitiatorOneTimePbkCipher &&\n        plainInitiatorOneTimePbkCipher.responder;\n      const initiator =\n        plainInitiatorRootKeyCipher && plainInitiatorRootKeyCipher.initiator;\n\n      return {\n        ...keyExchange,\n        message: responder ? responder.message : null,\n        contactCard:\n          responder && responder.contactCard\n            ? responder.contactCard.plainSharedCipherDataJson\n            : null,\n        myContactCard:\n          initiator && initiator.contactCard\n            ? initiator.contactCard.plainSharedCipherDataJson\n            : null,\n        myMessage: initiator && initiator.message,\n      };\n    } else {\n      const decryptedOtk = await this.decryptOtk(keyExchange, otKeyK);\n\n      const initiator = decryptedOtk && decryptedOtk.plainOtKeyCipher.initiator;\n\n      return {\n        ...keyExchange,\n        decryptedOtk,\n        message: initiator && initiator.message,\n        contactCard:\n          initiator &&\n          initiator.contactCard &&\n          initiator.contactCard.plainSharedCipherDataJson,\n      };\n    }\n  }\n\n  private async decryptOtk(\n    keyExchange: KeyExchange,\n    otKeyK?: string\n  ): Promise<DecryptedOtk> {\n    const otKey = await this.getOtKey(keyExchange, otKeyK);\n\n    return otKey && keyExchange.otk.otKeyCipher\n      ? {\n          plainOtKeyCipher: await this.encryptionService.decrypt(\n            otKey,\n            keyExchange.otk.otKeyCipher\n          ),\n          otKey,\n        }\n      : null;\n  }\n\n  private async getOtKey(\n    keyExchange: KeyExchange,\n    otKeyK?: string\n  ): Promise<JWK.Key> {\n    if (otKeyK) {\n      return await KFS.asKey({\n        ...JSON.parse(keyExchange.otk.otKeyParams),\n        k: otKeyK,\n      });\n    } else if (\n      keyExchange.otk.state === OtkState.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 KFS.asKey(decryptedCipher.otKey);\n      }\n    }\n    return null;\n  }\n\n  async initiateOtk({\n    message,\n    email,\n    contactCard,\n    upgrade,\n  }: InitiateOtkInput): Promise<{ keyExchange: KeyExchange; otKeyK: string }> {\n    const otKey = await this.keyFactory.createKey();\n    const nonce = this.keyFactory.randomString(this.CLIENT_NONCE_LENGTH);\n    const user = await this.authService.getUser();\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.ownerPlainData) {\n      initiatorPlainDataSig = JSON.stringify(\n        await this.encryptionService.sign(\n          initiatorSigPrk.jwk,\n          contactCard.ownerPlainData\n        )\n      );\n    }\n\n    const initiator = {\n      message,\n      contactCard: contactCard\n        ? {\n            plainSharedCipherDataJson: contactCard.plainSharedCipherDataJson,\n          }\n        : null,\n    };\n\n    // Content to be encrypted using the OTK.\n    const plainOtKeyCipher: PlainOtKeyCipher = {\n      nonce,\n      initiator: {\n        ...initiator,\n        oneTimePbk: initiatorOneTimePrk.toJSON(), // onetime 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        profile: {\n          username: user.username,\n        },\n      },\n    };\n\n    const otKeyCipher = await this.encryptionService.encrypt(\n      otKey,\n      plainOtKeyCipher\n    );\n\n    // Content to be encrypted using the initiator's root key.\n    const plainInitiatorRootKeyCipher: PlainInitiatorRootKeyCipher = {\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.encryptionService.encrypt(\n      rootKey.jwk,\n      plainInitiatorRootKeyCipher\n    );\n\n    // The raw OTK\n    const otKeyK: string = (otKey.toJSON(true) as any).k;\n\n    // API call\n    const { initiateKeyExchangeOtk } = await this.lrApollo.mutate<any>({\n      mutation: InitiateOtkMutation,\n      variables: {\n        input: {\n          // These will be stored on the server\n          initiatorRootKeyCipher: JSON.stringify(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: JSON.stringify(otKeyCipher),\n          sendEmail: email\n            ? {\n                email,\n                rawOtKey: otKeyK,\n              }\n            : null,\n          createTp: true,\n          initiatorPlainDataSig,\n          upgrade,\n        },\n      },\n    });\n    return { keyExchange: initiateKeyExchangeOtk.keyExchange, otKeyK };\n  }\n\n  async respondOtk({\n    id,\n    token,\n    decryptedOtk,\n    message,\n    initiatorContactCard,\n    responderContactCard: sentContactCard,\n  }: RespondOtkInput): Promise<RespondOtk> {\n    const user = await this.authService.getUser();\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 KFS.asKey(\n      decryptedOtk.plainOtKeyCipher.initiator.oneTimePbk\n    );\n\n    const initiatorPbk = await KFS.asKey(\n      decryptedOtk.plainOtKeyCipher.initiator.pbk\n    );\n    const initiatorSigPbk = await KFS.asKey(\n      decryptedOtk.plainOtKeyCipher.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 plainInitiatorOneTimePbkCipher: PlainInitiatorOneTimePbkCipher = {\n      nonce: decryptedOtk.plainOtKeyCipher.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        profile: {\n          username: user.username,\n        },\n        message,\n      },\n    };\n\n    let receivedCardInput;\n    if (decryptedOtk.plainOtKeyCipher.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 plainSharedCipherDataJson =\n        decryptedOtk.plainOtKeyCipher.initiator.contactCard\n          .plainSharedCipherDataJson;\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.plainReceiverCipherDataJson\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        plainSharedCipherDataJson\n      );\n      receivedCardInput.sharedCipherDataSig = JSON.stringify(\n        await this.encryptionService.sign(sigPxk.jwk, sharedCipherData)\n      );\n      receivedCardInput.sigPxkId = sigPxk.id;\n\n      plainInitiatorOneTimePbkCipher.responder.contactCard = {\n        ...plainInitiatorOneTimePbkCipher.responder.contactCard,\n        sharedCipherKey: ccSharedKey.toJSON(true),\n      };\n    }\n\n    let sentCardInput;\n    if (sentContactCard) {\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      sentCardInput = {\n        ownerWrappedKey: JSON.stringify(\n          await this.encryptionService.encrypt(\n            rootKey.jwk,\n            ownerKey.toJSON(true)\n          )\n        ),\n        ownerWrappingKeyId: rootKey.id,\n        ownerCipherData: sentContactCard.plainOwnerCipherDataJson\n          ? JSON.stringify(\n              await this.encryptionService.encrypt(\n                ownerKey,\n                sentContactCard.plainOwnerCipherDataJson\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        sentContactCard.plainSharedCipherDataJson\n      );\n      sentCardInput.sharedCipherDataSig = JSON.stringify(\n        await this.encryptionService.sign(sigPxk.jwk, sharedCipherData)\n      );\n      sentCardInput.sigPxkId = sigPxk.id;\n\n      if (sentContactCard.ownerPlainData) {\n        sentCardInput.ownerPlainDataSig = JSON.stringify(\n          await this.encryptionService.sign(\n            responderSigPrk.jwk,\n            sentContactCard.ownerPlainData\n          )\n        );\n      }\n\n      // Contact card info readable by the initiator\n      plainInitiatorOneTimePbkCipher.responder.contactCard = {\n        ...plainInitiatorOneTimePbkCipher.responder.contactCard,\n        plainSharedCipherDataJson: sentContactCard.plainSharedCipherDataJson,\n      };\n    }\n\n    // Encrypt with one-time public key\n    let initiatorOneTimePbkCipher = await this.encryptionService.encrypt(\n      initiatorOneTimePbk,\n      plainInitiatorOneTimePbkCipher\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    const { respondKeyExchangeOtk } = await this.lrApollo.mutate<any>({\n      mutation: RespondOtkMutation,\n      variables: {\n        input: {\n          keyExchangeId: id,\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: sentCardInput,\n        },\n      },\n    });\n\n    return {\n      keyExchange: respondKeyExchangeOtk.keyExchange,\n      userSharedKey: respondKeyExchangeOtk.userSharedKey,\n      tp: respondKeyExchangeOtk.tp,\n    };\n  }\n\n  async completeOtk(\n    keyExchangeId: string,\n    initiatorRootKeyCipher: string,\n    initiatorOneTimePbkCipher: string,\n    responderContactCard?: string\n  ): Promise<CompleteOtk> {\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 plainInitiatorRootKeyCipher = ((await this.encryptionService.decrypt(\n      rootKey.jwk,\n      initiatorRootKeyCipher\n    )) as unknown) as PlainInitiatorRootKeyCipher;\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 KFS.asKey(plainInitiatorRootKeyCipher.otKey),\n      await KFS.asKey(plainInitiatorRootKeyCipher.oneTimePrk),\n      initiatorOneTimePbkCipher\n    );\n\n    // Check the nonce match to ensure the responder was the one holding the OTK\n    if (\n      plainInitiatorRootKeyCipher.nonce !== 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 KFS.asKey(\n      plainInitiatorOneTimePbkCipher.responder.sigPbk\n    );\n    const responderPbk = await KFS.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 KFS.asKey(plainInitiatorOneTimePbkCipher.sharedKey);\n    const rkWrappedSharedKey = await this.encryptionService.encrypt(\n      rootKey.jwk,\n      sharedKey.toJSON(true)\n    );\n\n    const mkSharedKey = await KFS.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\n          )\n        ),\n      };\n    }\n\n    // Get the data needed from the initiator's cipher data.\n    let initiatorContactCardCipherInput;\n    let initiatorContactCardSharedCipherInput;\n    if (plainInitiatorRootKeyCipher.initiatorContactCard) {\n      const initiatorContactCard =\n        plainInitiatorRootKeyCipher.initiatorContactCard;\n      const ownerKey = await this.keyFactory.createKey();\n      const sharedCipherKey = await KFS.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      const ownerCipherData = initiatorContactCard.plainOwnerCipherDataJson\n        ? JSON.stringify(\n            await this.encryptionService.encrypt(\n              ownerKey,\n              initiatorContactCard.plainOwnerCipherDataJson\n            )\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        initiatorContactCard.plainSharedCipherDataJson\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    const res = await this.lrApollo.mutate<any>({\n      mutation: CompleteOtkMutation,\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    return res.completeKeyExchangeOtk;\n  }\n\n  public async currentUserSharedKey(input: {\n    username?: string;\n    userId?: string;\n  }): Promise<UserSharedKey> {\n    const { currentUserSharedKey } = await this.lrApollo.query<any>({\n      query: CurrentUserSharedKeyQuery,\n      variables: {\n        username: input.username,\n        userId: input.userId,\n      },\n    });\n    return currentUserSharedKey.userSharedKey;\n  }\n}\n"]}
1
+ import { __awaiter } from "tslib";
2
+ import { Injectable } from '@angular/core';
3
+ import { LifeReadyAuthService } from '../auth/life-ready-auth.service';
4
+ import { EncryptionService, JoseSerialization, } from '../cryptography/encryption.service';
5
+ import { KeyService } from '../cryptography/key.service';
6
+ import { LrCodeMismatchException } from '../_common/exceptions';
7
+ import { UserService } from './../users/user.service';
8
+ import { CompleteOtkMutation, CurrentUserSharedKeyQuery, InitiateOtkMutation, KeyExchangeQuery, KeyExchangesQuery, KeyExchangeTokenQuery, RespondOtkMutation, } from './key-exchange.gql';
9
+ import { OtkState, } from './key-exchange.types';
10
+ import { LrApolloService } from './lr-apollo.service';
11
+ import { KeyFactoryService as KFS } from '../cryptography/key-factory.service';
12
+ // Ref: https://stackoverflow.com/questions/59735280/angular-8-moment-error-cannot-call-a-namespace-moment
13
+ import * as moment_ from 'moment';
14
+ import * as i0 from "@angular/core";
15
+ import * as i1 from "../cryptography/key-factory.service";
16
+ import * as i2 from "../cryptography/key.service";
17
+ import * as i3 from "./lr-apollo.service";
18
+ import * as i4 from "../cryptography/encryption.service";
19
+ import * as i5 from "../auth/life-ready-auth.service";
20
+ import * as i6 from "../users/user.service";
21
+ const moment = moment_;
22
+ export class KeyExchangeService {
23
+ constructor(keyFactory, keyService, lrApollo, encryptionService, authService, userService) {
24
+ this.keyFactory = keyFactory;
25
+ this.keyService = keyService;
26
+ this.lrApollo = lrApollo;
27
+ this.encryptionService = encryptionService;
28
+ this.authService = authService;
29
+ this.userService = userService;
30
+ this.CLIENT_NONCE_LENGTH = 32;
31
+ }
32
+ getKeyExchangeList(input = {}) {
33
+ return __awaiter(this, void 0, void 0, function* () {
34
+ const { keyExchanges } = yield this.lrApollo.query({
35
+ query: KeyExchangesQuery,
36
+ variables: Object.assign({}, input),
37
+ });
38
+ return keyExchanges;
39
+ });
40
+ }
41
+ /**
42
+ * @param id If the current user can responder the key exchange if they are either the initiator or the receiver.
43
+ * @param token If not signed in, or not the initiator or responder, 'token' must be given.
44
+ * @param otKeyK Is the raw one-time key (string). If the responder is explicitly specified at time of initiation, then
45
+ * it's possible to have the otKey wrapped by the public key of the responder. In which case, the otKeyK is not needed.
46
+ */
47
+ getKeyExchange(id, { otKeyK, token } = {}) {
48
+ return __awaiter(this, void 0, void 0, function* () {
49
+ const { keyExchange } = yield this.lrApollo.query({
50
+ query: token ? KeyExchangeTokenQuery : KeyExchangeQuery,
51
+ variables: {
52
+ id,
53
+ token,
54
+ },
55
+ });
56
+ return yield this.decryptKeyExchange(keyExchange, otKeyK);
57
+ });
58
+ }
59
+ decryptResponseCipher(otKey, otPrk, content) {
60
+ return __awaiter(this, void 0, void 0, function* () {
61
+ // The response could be wrapped by the OtK as well as we the OtPrk
62
+ try {
63
+ content = yield this.encryptionService.decrypt(otKey, content);
64
+ }
65
+ catch (error) {
66
+ if (error.message !== 'no key found') {
67
+ throw error;
68
+ }
69
+ // Do nothing to support older versions where message is not wrapped with otk.
70
+ }
71
+ // The Prk is single-use and only used to send information from the responder back to the initiator.
72
+ return yield this.encryptionService.decrypt(otPrk, content);
73
+ });
74
+ }
75
+ decryptKeyExchange(keyExchange, otKeyK) {
76
+ return __awaiter(this, void 0, void 0, function* () {
77
+ if (keyExchange.isInitiator) {
78
+ const rootKey = yield this.keyService.getCurrentRootKey();
79
+ // Decrypt using the root key to get the Prk
80
+ const plainInitiatorRootKeyCipher = (yield this.encryptionService.decrypt(rootKey.jwk, keyExchange.initiatorRootKeyCipher));
81
+ const plainInitiatorOneTimePbkCipher = keyExchange.otk
82
+ .initiatorOneTimePbkCipher
83
+ ? yield this.decryptResponseCipher(yield KFS.asKey(plainInitiatorRootKeyCipher.otKey), yield KFS.asKey(plainInitiatorRootKeyCipher.oneTimePrk), keyExchange.otk.initiatorOneTimePbkCipher)
84
+ : null;
85
+ const responder = plainInitiatorOneTimePbkCipher &&
86
+ plainInitiatorOneTimePbkCipher.responder;
87
+ const initiator = plainInitiatorRootKeyCipher && plainInitiatorRootKeyCipher.initiator;
88
+ return Object.assign(Object.assign({}, keyExchange), { message: responder ? responder.message : null, contactCard: responder && responder.contactCard
89
+ ? responder.contactCard.plainSharedCipherDataJson
90
+ : null, myContactCard: initiator && initiator.contactCard
91
+ ? initiator.contactCard.plainSharedCipherDataJson
92
+ : null, myMessage: initiator && initiator.message });
93
+ }
94
+ else {
95
+ const decryptedOtk = yield this.decryptOtk(keyExchange, otKeyK);
96
+ const initiator = decryptedOtk && decryptedOtk.plainOtKeyCipher.initiator;
97
+ return Object.assign(Object.assign({}, keyExchange), { decryptedOtk, message: initiator && initiator.message, contactCard: initiator &&
98
+ initiator.contactCard &&
99
+ initiator.contactCard.plainSharedCipherDataJson });
100
+ }
101
+ });
102
+ }
103
+ decryptOtk(keyExchange, otKeyK) {
104
+ return __awaiter(this, void 0, void 0, function* () {
105
+ const otKey = yield this.getOtKey(keyExchange, otKeyK);
106
+ return otKey && keyExchange.otk.otKeyCipher
107
+ ? {
108
+ plainOtKeyCipher: yield this.encryptionService.decrypt(otKey, keyExchange.otk.otKeyCipher),
109
+ otKey,
110
+ }
111
+ : null;
112
+ });
113
+ }
114
+ getOtKey(keyExchange, otKeyK) {
115
+ return __awaiter(this, void 0, void 0, function* () {
116
+ if (otKeyK) {
117
+ return yield KFS.asKey(Object.assign(Object.assign({}, JSON.parse(keyExchange.otk.otKeyParams)), { k: otKeyK }));
118
+ }
119
+ else if (keyExchange.otk.state === OtkState.OTK_INITIATED &&
120
+ !keyExchange.isInitiator &&
121
+ keyExchange.otk.responderPbkCipher) {
122
+ // Assuming existing user getting invited where OTK is wrapped in responder's public key.
123
+ const prk = yield this.keyService.getCurrentPxk();
124
+ const decryptedCipher = yield this.encryptionService.decrypt(prk.jwk, JSON.parse(keyExchange.otk.responderPbkCipher), {
125
+ serializations: [JoseSerialization.COMPACT],
126
+ });
127
+ if (decryptedCipher.otKey) {
128
+ return yield KFS.asKey(decryptedCipher.otKey);
129
+ }
130
+ }
131
+ return null;
132
+ });
133
+ }
134
+ initiateOtk({ message, email, contactCard, upgrade, }) {
135
+ return __awaiter(this, void 0, void 0, function* () {
136
+ const otKey = yield this.keyFactory.createKey();
137
+ const nonce = this.keyFactory.randomString(this.CLIENT_NONCE_LENGTH);
138
+ const user = yield this.authService.getUser();
139
+ // New PKC key for encryption. This key is used only once when the responder sends
140
+ // back their signing public key.
141
+ const initiatorOneTimePrk = yield this.keyFactory.createPkcKey();
142
+ // Option 1: New PKC key for signing
143
+ // const initiatorSigPrk = await this.keyService.createPkcSignKey();
144
+ // Option 2: Use the user's global signing key.
145
+ // This key is used to prove the initiator's identity.
146
+ const initiatorPrk = yield this.keyService.getCurrentPxk();
147
+ const initiatorSigPrk = yield this.keyService.getCurrentSigPxk();
148
+ let initiatorPlainDataSig = null;
149
+ if (contactCard && contactCard.ownerPlainData) {
150
+ initiatorPlainDataSig = JSON.stringify(yield this.encryptionService.sign(initiatorSigPrk.jwk, contactCard.ownerPlainData));
151
+ }
152
+ const initiator = {
153
+ message,
154
+ contactCard: contactCard
155
+ ? {
156
+ plainSharedCipherDataJson: contactCard.plainSharedCipherDataJson,
157
+ }
158
+ : null,
159
+ };
160
+ // Content to be encrypted using the OTK.
161
+ const plainOtKeyCipher = {
162
+ nonce,
163
+ initiator: Object.assign(Object.assign({}, initiator), { oneTimePbk: initiatorOneTimePrk.toJSON(), pbk: initiatorPrk.jwk.toJSON(), sigPbk: initiatorSigPrk.jwk.toJSON(), profile: {
164
+ username: user.username,
165
+ } }),
166
+ };
167
+ const otKeyCipher = yield this.encryptionService.encrypt(otKey, plainOtKeyCipher);
168
+ // Content to be encrypted using the initiator's root key.
169
+ const plainInitiatorRootKeyCipher = {
170
+ nonce,
171
+ oneTimePrk: initiatorOneTimePrk.toJSON(true),
172
+ // Should not need to keep this encrypted since we are using the global signing key.
173
+ // sigPrk: initiatorSigPrk.toJSON(true),
174
+ // Save it in case the initiator want to decode the otKeyCipher.
175
+ // Since the otKey is only used once, and that otKeyCipher contains only
176
+ // the public key of the initiator, it's safe just leave the otKey stored here.
177
+ otKey: otKey.toJSON(true),
178
+ // These should be storing information such as how the fields of the shared contact card is
179
+ // derived from the master contact card.
180
+ initiatorContactCard: contactCard,
181
+ initiator,
182
+ };
183
+ const rootKey = yield this.keyService.getCurrentRootKey();
184
+ const initiatorRootKeyCipher = yield this.encryptionService.encrypt(rootKey.jwk, plainInitiatorRootKeyCipher);
185
+ // The raw OTK
186
+ const otKeyK = otKey.toJSON(true).k;
187
+ // API call
188
+ const { initiateKeyExchangeOtk } = yield this.lrApollo.mutate({
189
+ mutation: InitiateOtkMutation,
190
+ variables: {
191
+ input: {
192
+ // These will be stored on the server
193
+ initiatorRootKeyCipher: JSON.stringify(initiatorRootKeyCipher),
194
+ initiatorPxkId: initiatorPrk.id,
195
+ initiatorSigPxkId: initiatorSigPrk.id,
196
+ // These will be sent to the responder
197
+ otKeyParams: JSON.stringify(otKey.toJSON()),
198
+ otKeyCipher: JSON.stringify(otKeyCipher),
199
+ sendEmail: email
200
+ ? {
201
+ email,
202
+ rawOtKey: otKeyK,
203
+ }
204
+ : null,
205
+ createTp: true,
206
+ initiatorPlainDataSig,
207
+ upgrade,
208
+ },
209
+ },
210
+ });
211
+ return { keyExchange: initiateKeyExchangeOtk.keyExchange, otKeyK };
212
+ });
213
+ }
214
+ respondOtk({ id, token, decryptedOtk, message, initiatorContactCard, responderContactCard: sentContactCard, }) {
215
+ return __awaiter(this, void 0, void 0, function* () {
216
+ const user = yield this.authService.getUser();
217
+ const rootKey = yield this.keyService.getCurrentRootKey();
218
+ const masterKeyId = this.keyService.getCurrentMasterKey().id;
219
+ const masterKey = yield this.keyService.getCurrentMasterKey();
220
+ const sharedKey = yield this.keyFactory.createKey();
221
+ const mkSharedKey = yield this.keyFactory.createKey();
222
+ const rkWrappedSharedKey = yield this.encryptionService.encrypt(rootKey.jwk, sharedKey.toJSON(true));
223
+ const mkWrappedMkSharedKey = yield this.encryptionService.encrypt(masterKey.jwk, mkSharedKey.toJSON(true));
224
+ const initiatorOneTimePbk = yield KFS.asKey(decryptedOtk.plainOtKeyCipher.initiator.oneTimePbk);
225
+ const initiatorPbk = yield KFS.asKey(decryptedOtk.plainOtKeyCipher.initiator.pbk);
226
+ const initiatorSigPbk = yield KFS.asKey(decryptedOtk.plainOtKeyCipher.initiator.sigPbk);
227
+ // Option 1: Using new Prk for each TP pair
228
+ // Create a new public signing key for the responder.
229
+ // const responderSigPrk = await this.keyService.createPkcSignKey()
230
+ // const rkWrappedResponderSigPrk = await this.encrypt(rootKey, responderSigPrk.toJSON(true));
231
+ // Option 2: Responder already has a signing Prk
232
+ const responderPrk = yield this.keyService.getCurrentPxk();
233
+ const responderSigPrk = yield this.keyService.getCurrentSigPxk();
234
+ const signedInitiatorPbk = yield this.encryptionService.sign(responderSigPrk.jwk, initiatorPbk.toJSON());
235
+ const signedInitiatorSigPbk = yield this.encryptionService.sign(responderSigPrk.jwk, initiatorSigPbk.toJSON());
236
+ const plainInitiatorOneTimePbkCipher = {
237
+ nonce: decryptedOtk.plainOtKeyCipher.nonce,
238
+ sharedKey: sharedKey.toJSON(true),
239
+ mkSharedKey: mkSharedKey.toJSON(true),
240
+ responder: {
241
+ pbk: responderPrk.jwk.toJSON(),
242
+ sigPbk: responderSigPrk.jwk.toJSON(),
243
+ profile: {
244
+ username: user.username,
245
+ },
246
+ message,
247
+ },
248
+ };
249
+ let receivedCardInput;
250
+ if (decryptedOtk.plainOtKeyCipher.initiator.contactCard) {
251
+ // Set the info about the initiator to be the ones sent by the initiator. We need th responder to do the encryption here
252
+ // because the initiator does not have the shared key yet, and we want the responder to have a functional contact card after
253
+ // this exchange. The initiator can double check the contact details are correct and sign it when it completes the exchange.
254
+ const plainSharedCipherDataJson = decryptedOtk.plainOtKeyCipher.initiator.contactCard
255
+ .plainSharedCipherDataJson;
256
+ // Create keys
257
+ const receiverKey = yield this.keyFactory.createKey();
258
+ const ccSharedKey = yield this.keyFactory.createKey();
259
+ const sigPxk = yield this.keyService.getCurrentSigPxk();
260
+ receivedCardInput = {
261
+ receiverWrappedKey: JSON.stringify(yield this.encryptionService.encrypt(rootKey.jwk, receiverKey.toJSON(true))),
262
+ receiverWrappingKeyId: rootKey.id,
263
+ receiverCipherData: initiatorContactCard
264
+ ? JSON.stringify(yield this.encryptionService.encrypt(receiverKey, initiatorContactCard.plainReceiverCipherDataJson))
265
+ : '',
266
+ sharedWrappedKey: JSON.stringify(yield this.encryptionService.encrypt(sharedKey, ccSharedKey.toJSON(true))),
267
+ };
268
+ const sharedCipherData = yield this.encryptionService.encrypt(ccSharedKey, plainSharedCipherDataJson);
269
+ receivedCardInput.sharedCipherDataSig = JSON.stringify(yield this.encryptionService.sign(sigPxk.jwk, sharedCipherData));
270
+ receivedCardInput.sigPxkId = sigPxk.id;
271
+ plainInitiatorOneTimePbkCipher.responder.contactCard = Object.assign(Object.assign({}, plainInitiatorOneTimePbkCipher.responder.contactCard), { sharedCipherKey: ccSharedKey.toJSON(true) });
272
+ }
273
+ let sentCardInput;
274
+ if (sentContactCard) {
275
+ // Create keys
276
+ const ownerKey = yield this.keyFactory.createKey();
277
+ const ccSharedKey = yield this.keyFactory.createKey();
278
+ const sigPxk = yield this.keyService.getCurrentSigPxk();
279
+ sentCardInput = {
280
+ ownerWrappedKey: JSON.stringify(yield this.encryptionService.encrypt(rootKey.jwk, ownerKey.toJSON(true))),
281
+ ownerWrappingKeyId: rootKey.id,
282
+ ownerCipherData: sentContactCard.plainOwnerCipherDataJson
283
+ ? JSON.stringify(yield this.encryptionService.encrypt(ownerKey, sentContactCard.plainOwnerCipherDataJson))
284
+ : '',
285
+ sharedWrappedKey: JSON.stringify(yield this.encryptionService.encrypt(sharedKey, ccSharedKey.toJSON(true))),
286
+ };
287
+ const sharedCipherData = yield this.encryptionService.encrypt(ccSharedKey, sentContactCard.plainSharedCipherDataJson);
288
+ sentCardInput.sharedCipherDataSig = JSON.stringify(yield this.encryptionService.sign(sigPxk.jwk, sharedCipherData));
289
+ sentCardInput.sigPxkId = sigPxk.id;
290
+ if (sentContactCard.ownerPlainData) {
291
+ sentCardInput.ownerPlainDataSig = JSON.stringify(yield this.encryptionService.sign(responderSigPrk.jwk, sentContactCard.ownerPlainData));
292
+ }
293
+ // Contact card info readable by the initiator
294
+ plainInitiatorOneTimePbkCipher.responder.contactCard = Object.assign(Object.assign({}, plainInitiatorOneTimePbkCipher.responder.contactCard), { plainSharedCipherDataJson: sentContactCard.plainSharedCipherDataJson });
295
+ }
296
+ // Encrypt with one-time public key
297
+ let initiatorOneTimePbkCipher = yield this.encryptionService.encrypt(initiatorOneTimePbk, plainInitiatorOneTimePbkCipher);
298
+ // Encrypt with the otk again to keep use of asymmetric keys to a minimum.
299
+ initiatorOneTimePbkCipher = yield this.encryptionService.encrypt(decryptedOtk.otKey, initiatorOneTimePbkCipher);
300
+ const { respondKeyExchangeOtk } = yield this.lrApollo.mutate({
301
+ mutation: RespondOtkMutation,
302
+ variables: {
303
+ input: {
304
+ keyExchangeId: id,
305
+ keyExchangeToken: token,
306
+ rootKeyId: rootKey.id,
307
+ masterKeyId,
308
+ // These will be stored on the server
309
+ responderPxkId: responderPrk.id,
310
+ responderSigPxkId: responderSigPrk.id,
311
+ signedInitiatorPbk: JSON.stringify(signedInitiatorPbk),
312
+ signedInitiatorSigPbk: JSON.stringify(signedInitiatorSigPbk),
313
+ // rkWrappedInitiatorSigPbk: JSON.stringify(rkWrappedInitiatorSigPbk),
314
+ // Option 1: Using new Prk for each TP pair
315
+ // rkWrappedResponderSigPrk: JSON.stringify(rkWrappedResponderSigPrk),
316
+ rkWrappedSharedKey: JSON.stringify(rkWrappedSharedKey),
317
+ mkWrappedMkSharedKey: JSON.stringify(mkWrappedMkSharedKey),
318
+ // These will be sent to the initiator
319
+ initiatorOneTimePbkCipher: JSON.stringify(initiatorOneTimePbkCipher),
320
+ initiatorContactCard: receivedCardInput,
321
+ responderContactCard: sentCardInput,
322
+ },
323
+ },
324
+ });
325
+ return {
326
+ keyExchange: respondKeyExchangeOtk.keyExchange,
327
+ userSharedKey: respondKeyExchangeOtk.userSharedKey,
328
+ tp: respondKeyExchangeOtk.tp,
329
+ };
330
+ });
331
+ }
332
+ completeOtk(keyExchangeId, initiatorRootKeyCipher, initiatorOneTimePbkCipher, responderContactCard) {
333
+ return __awaiter(this, void 0, void 0, function* () {
334
+ const rootKey = yield this.keyService.getCurrentRootKey();
335
+ const masterKey = yield this.keyService.getCurrentMasterKey();
336
+ // Decrypt using the root key to get the Prk
337
+ const plainInitiatorRootKeyCipher = (yield this.encryptionService.decrypt(rootKey.jwk, initiatorRootKeyCipher));
338
+ // The Prk is single-use and only used to send information from the responder back to the initiator.
339
+ const plainInitiatorOneTimePbkCipher = yield this.decryptResponseCipher(yield KFS.asKey(plainInitiatorRootKeyCipher.otKey), yield KFS.asKey(plainInitiatorRootKeyCipher.oneTimePrk), initiatorOneTimePbkCipher);
340
+ // Check the nonce match to ensure the responder was the one holding the OTK
341
+ if (plainInitiatorRootKeyCipher.nonce !== plainInitiatorOneTimePbkCipher.nonce) {
342
+ throw new LrCodeMismatchException('The nonce returned by responder does not match with the one created by the initiator.');
343
+ }
344
+ // Option 1: Assuming the signing key is unique between users.
345
+ // const initiatorSigPrk = await KFS.asKey(ke.plainInitiatorRootKeyCipher.sigPrk);
346
+ // const rkWrappedInitiatorSigPrk = await this.encrypt(rootKey, initiatorSigPrk.toJSON(true));
347
+ // Option 2: Use the user's global signing key.
348
+ // In this case the initiatorSigPrk is already a part of the key graph.
349
+ // So there's nothing to do here.
350
+ // Protected the signing public key of the responder.
351
+ const initiatorSigPrk = yield this.keyService.getCurrentSigPxk();
352
+ const responderSigPbk = yield KFS.asKey(plainInitiatorOneTimePbkCipher.responder.sigPbk);
353
+ const responderPbk = yield KFS.asKey(plainInitiatorOneTimePbkCipher.responder.pbk);
354
+ const signedResponderPbk = yield this.encryptionService.sign(initiatorSigPrk.jwk, responderPbk.toJSON());
355
+ const signedResponderSigPbk = yield this.encryptionService.sign(initiatorSigPrk.jwk, responderSigPbk.toJSON());
356
+ const sharedKey = yield KFS.asKey(plainInitiatorOneTimePbkCipher.sharedKey);
357
+ const rkWrappedSharedKey = yield this.encryptionService.encrypt(rootKey.jwk, sharedKey.toJSON(true));
358
+ const mkSharedKey = yield KFS.asKey(plainInitiatorOneTimePbkCipher.mkSharedKey);
359
+ const mkWrappedMkSharedKey = yield this.encryptionService.encrypt(masterKey.jwk, mkSharedKey.toJSON(true));
360
+ let responderContactCardCipherInput;
361
+ if (responderContactCard) {
362
+ // Create key
363
+ const receiverKey = yield this.keyFactory.createKey();
364
+ responderContactCardCipherInput = {
365
+ receiverWrappedKey: JSON.stringify(yield this.encryptionService.encrypt(rootKey.jwk, receiverKey.toJSON(true))),
366
+ receiverWrappingKeyId: rootKey.id,
367
+ receiverCipherData: JSON.stringify(yield this.encryptionService.encrypt(receiverKey, responderContactCard)),
368
+ };
369
+ }
370
+ // Get the data needed from the initiator's cipher data.
371
+ let initiatorContactCardCipherInput;
372
+ let initiatorContactCardSharedCipherInput;
373
+ if (plainInitiatorRootKeyCipher.initiatorContactCard) {
374
+ const initiatorContactCard = plainInitiatorRootKeyCipher.initiatorContactCard;
375
+ const ownerKey = yield this.keyFactory.createKey();
376
+ const sharedCipherKey = yield KFS.asKey(plainInitiatorOneTimePbkCipher.responder.contactCard.sharedCipherKey);
377
+ const ownerWrappedKey = JSON.stringify(yield this.encryptionService.encrypt(rootKey.jwk, ownerKey.toJSON(true)));
378
+ const ownerCipherData = initiatorContactCard.plainOwnerCipherDataJson
379
+ ? JSON.stringify(yield this.encryptionService.encrypt(ownerKey, initiatorContactCard.plainOwnerCipherDataJson))
380
+ : '';
381
+ initiatorContactCardCipherInput = {
382
+ ownerWrappedKey,
383
+ ownerWrappingKeyId: rootKey.id,
384
+ ownerCipherData,
385
+ };
386
+ initiatorContactCardSharedCipherInput = {
387
+ sigPxkId: initiatorSigPrk.id,
388
+ };
389
+ const sharedCipherData = yield this.encryptionService.encrypt(sharedCipherKey, initiatorContactCard.plainSharedCipherDataJson);
390
+ initiatorContactCardSharedCipherInput.sharedCipherDataSig = JSON.stringify(yield this.encryptionService.sign(initiatorSigPrk.jwk, sharedCipherData));
391
+ }
392
+ // TODO ideally we update the shared data in the contact card sent to the responder as well since that
393
+ // CC was created by the responder.
394
+ const res = yield this.lrApollo.mutate({
395
+ mutation: CompleteOtkMutation,
396
+ variables: {
397
+ input: {
398
+ keyExchangeId,
399
+ rootKeyId: rootKey.id,
400
+ masterKeyId: masterKey.id,
401
+ initiatorSigPxkId: initiatorSigPrk.id,
402
+ signedResponderPbk: JSON.stringify(signedResponderPbk),
403
+ signedResponderSigPbk: JSON.stringify(signedResponderSigPbk),
404
+ rkWrappedSharedKey: JSON.stringify(rkWrappedSharedKey),
405
+ mkWrappedMkSharedKey: JSON.stringify(mkWrappedMkSharedKey),
406
+ responderContactCardCipher: responderContactCardCipherInput,
407
+ initiatorContactCardCipher: initiatorContactCardCipherInput,
408
+ initiatorContactCardSharedCipher: initiatorContactCardSharedCipherInput,
409
+ },
410
+ },
411
+ });
412
+ return res.completeKeyExchangeOtk;
413
+ });
414
+ }
415
+ currentUserSharedKey(input) {
416
+ return __awaiter(this, void 0, void 0, function* () {
417
+ const { currentUserSharedKey } = yield this.lrApollo.query({
418
+ query: CurrentUserSharedKeyQuery,
419
+ variables: {
420
+ username: input.username,
421
+ userId: input.userId,
422
+ },
423
+ });
424
+ return currentUserSharedKey.userSharedKey;
425
+ });
426
+ }
427
+ }
428
+ KeyExchangeService.ɵprov = i0.ɵɵdefineInjectable({ factory: function KeyExchangeService_Factory() { return new KeyExchangeService(i0.ɵɵinject(i1.KeyFactoryService), i0.ɵɵinject(i2.KeyService), i0.ɵɵinject(i3.LrApolloService), i0.ɵɵinject(i4.EncryptionService), i0.ɵɵinject(i5.LifeReadyAuthService), i0.ɵɵinject(i6.UserService)); }, token: KeyExchangeService, providedIn: "root" });
429
+ KeyExchangeService.decorators = [
430
+ { type: Injectable, args: [{
431
+ providedIn: 'root',
432
+ },] }
433
+ ];
434
+ KeyExchangeService.ctorParameters = () => [
435
+ { type: KFS },
436
+ { type: KeyService },
437
+ { type: LrApolloService },
438
+ { type: EncryptionService },
439
+ { type: LifeReadyAuthService },
440
+ { type: UserService }
441
+ ];
442
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"key-exchange.service.js","sourceRoot":"C:/Projects/newrepo/kc-client/projects/core/src/","sources":["lib/api/key-exchange.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\r\nimport { JWK } from 'node-jose';\r\nimport { LifeReadyAuthService } from '../auth/life-ready-auth.service';\r\nimport { KeyGraphResponse } from '../cryptography/cryptography.types';\r\nimport {\r\n  EncryptionService,\r\n  JoseSerialization,\r\n} from '../cryptography/encryption.service';\r\nimport { KeyService } from '../cryptography/key.service';\r\nimport { LrCodeMismatchException } from '../_common/exceptions';\r\nimport { UserService } from './../users/user.service';\r\nimport {\r\n  CompleteOtkMutation,\r\n  CurrentUserSharedKeyQuery,\r\n  InitiateOtkMutation,\r\n  KeyExchangeQuery,\r\n  KeyExchangesQuery,\r\n  KeyExchangeTokenQuery,\r\n  RespondOtkMutation,\r\n} from './key-exchange.gql';\r\nimport {\r\n  CompleteOtk,\r\n  DecryptedKeyExchange,\r\n  DecryptedOtk,\r\n  GetKeyExchangeListOptions,\r\n  GetKeyExchangeOptions,\r\n  InitiateOtkInput,\r\n  KeyExchange,\r\n  OtkState,\r\n  PlainInitiatorOneTimePbkCipher,\r\n  PlainInitiatorRootKeyCipher,\r\n  PlainOtKeyCipher,\r\n  PlainResponderPbkCipher,\r\n  RespondOtk,\r\n  RespondOtkInput,\r\n  UserSharedKey,\r\n} from './key-exchange.types';\r\nimport { LrApolloService } from './lr-apollo.service';\r\nimport { KeyFactoryService as KFS } from '../cryptography/key-factory.service';\r\n// Ref: https://stackoverflow.com/questions/59735280/angular-8-moment-error-cannot-call-a-namespace-moment\r\nimport * as moment_ from 'moment';\r\nconst moment = moment_;\r\n\r\n@Injectable({\r\n  providedIn: 'root',\r\n})\r\nexport class KeyExchangeService {\r\n  private readonly CLIENT_NONCE_LENGTH = 32;\r\n\r\n  constructor(\r\n    private keyFactory: KFS,\r\n    private keyService: KeyService,\r\n    private lrApollo: LrApolloService,\r\n    private encryptionService: EncryptionService,\r\n    private authService: LifeReadyAuthService,\r\n    private userService: UserService\r\n  ) {}\r\n\r\n  public async getKeyExchangeList(\r\n    input: GetKeyExchangeListOptions = {}\r\n  ): Promise<any> {\r\n    const { keyExchanges } = await this.lrApollo.query<{\r\n      keyExchanges: any;\r\n      keyGraph: KeyGraphResponse;\r\n    }>({\r\n      query: KeyExchangesQuery,\r\n      variables: {\r\n        ...input,\r\n      },\r\n    });\r\n    return keyExchanges;\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  public async getKeyExchange(\r\n    id: string,\r\n    { otKeyK, token }: GetKeyExchangeOptions = {}\r\n  ): Promise<DecryptedKeyExchange> {\r\n    const { keyExchange } = await this.lrApollo.query<{\r\n      keyExchange: KeyExchange;\r\n      keyGraph: KeyGraphResponse;\r\n    }>({\r\n      query: token ? KeyExchangeTokenQuery : KeyExchangeQuery,\r\n      variables: {\r\n        id,\r\n        token,\r\n      },\r\n    });\r\n    return await this.decryptKeyExchange(keyExchange, otKeyK);\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<PlainInitiatorOneTimePbkCipher> {\r\n    // The response could be wrapped by the OtK as well as we the OtPrk\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  public async decryptKeyExchange(\r\n    keyExchange: KeyExchange,\r\n    otKeyK?: string\r\n  ): Promise<DecryptedKeyExchange> {\r\n    if (keyExchange.isInitiator) {\r\n      const rootKey = await this.keyService.getCurrentRootKey();\r\n      // Decrypt using the root key to get the Prk\r\n      const plainInitiatorRootKeyCipher = ((await this.encryptionService.decrypt(\r\n        rootKey.jwk,\r\n        keyExchange.initiatorRootKeyCipher\r\n      )) as unknown) as PlainInitiatorRootKeyCipher;\r\n\r\n      const plainInitiatorOneTimePbkCipher = keyExchange.otk\r\n        .initiatorOneTimePbkCipher\r\n        ? await this.decryptResponseCipher(\r\n            await KFS.asKey(plainInitiatorRootKeyCipher.otKey),\r\n            await KFS.asKey(plainInitiatorRootKeyCipher.oneTimePrk),\r\n            keyExchange.otk.initiatorOneTimePbkCipher\r\n          )\r\n        : null;\r\n\r\n      const responder =\r\n        plainInitiatorOneTimePbkCipher &&\r\n        plainInitiatorOneTimePbkCipher.responder;\r\n      const initiator =\r\n        plainInitiatorRootKeyCipher && plainInitiatorRootKeyCipher.initiator;\r\n\r\n      return {\r\n        ...keyExchange,\r\n        message: responder ? responder.message : null,\r\n        contactCard:\r\n          responder && responder.contactCard\r\n            ? responder.contactCard.plainSharedCipherDataJson\r\n            : null,\r\n        myContactCard:\r\n          initiator && initiator.contactCard\r\n            ? initiator.contactCard.plainSharedCipherDataJson\r\n            : null,\r\n        myMessage: initiator && initiator.message,\r\n      };\r\n    } else {\r\n      const decryptedOtk = await this.decryptOtk(keyExchange, otKeyK);\r\n\r\n      const initiator = decryptedOtk && decryptedOtk.plainOtKeyCipher.initiator;\r\n\r\n      return {\r\n        ...keyExchange,\r\n        decryptedOtk,\r\n        message: initiator && initiator.message,\r\n        contactCard:\r\n          initiator &&\r\n          initiator.contactCard &&\r\n          initiator.contactCard.plainSharedCipherDataJson,\r\n      };\r\n    }\r\n  }\r\n\r\n  private async decryptOtk(\r\n    keyExchange: KeyExchange,\r\n    otKeyK?: string\r\n  ): Promise<DecryptedOtk> {\r\n    const otKey = await this.getOtKey(keyExchange, otKeyK);\r\n\r\n    return otKey && keyExchange.otk.otKeyCipher\r\n      ? {\r\n          plainOtKeyCipher: await this.encryptionService.decrypt(\r\n            otKey,\r\n            keyExchange.otk.otKeyCipher\r\n          ),\r\n          otKey,\r\n        }\r\n      : null;\r\n  }\r\n\r\n  private async getOtKey(\r\n    keyExchange: KeyExchange,\r\n    otKeyK?: string\r\n  ): Promise<JWK.Key> {\r\n    if (otKeyK) {\r\n      return await KFS.asKey({\r\n        ...JSON.parse(keyExchange.otk.otKeyParams),\r\n        k: otKeyK,\r\n      });\r\n    } else if (\r\n      keyExchange.otk.state === OtkState.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 KFS.asKey(decryptedCipher.otKey);\r\n      }\r\n    }\r\n    return null;\r\n  }\r\n\r\n  async initiateOtk({\r\n    message,\r\n    email,\r\n    contactCard,\r\n    upgrade,\r\n  }: InitiateOtkInput): Promise<{ keyExchange: KeyExchange; otKeyK: string }> {\r\n    const otKey = await this.keyFactory.createKey();\r\n    const nonce = this.keyFactory.randomString(this.CLIENT_NONCE_LENGTH);\r\n    const user = await this.authService.getUser();\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.ownerPlainData) {\r\n      initiatorPlainDataSig = JSON.stringify(\r\n        await this.encryptionService.sign(\r\n          initiatorSigPrk.jwk,\r\n          contactCard.ownerPlainData\r\n        )\r\n      );\r\n    }\r\n\r\n    const initiator = {\r\n      message,\r\n      contactCard: contactCard\r\n        ? {\r\n            plainSharedCipherDataJson: contactCard.plainSharedCipherDataJson,\r\n          }\r\n        : null,\r\n    };\r\n\r\n    // Content to be encrypted using the OTK.\r\n    const plainOtKeyCipher: PlainOtKeyCipher = {\r\n      nonce,\r\n      initiator: {\r\n        ...initiator,\r\n        oneTimePbk: initiatorOneTimePrk.toJSON(), // onetime 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        profile: {\r\n          username: user.username,\r\n        },\r\n      },\r\n    };\r\n\r\n    const otKeyCipher = await this.encryptionService.encrypt(\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 plainInitiatorRootKeyCipher: PlainInitiatorRootKeyCipher = {\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.encryptionService.encrypt(\r\n      rootKey.jwk,\r\n      plainInitiatorRootKeyCipher\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 { initiateKeyExchangeOtk } = await this.lrApollo.mutate<any>({\r\n      mutation: InitiateOtkMutation,\r\n      variables: {\r\n        input: {\r\n          // These will be stored on the server\r\n          initiatorRootKeyCipher: JSON.stringify(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: JSON.stringify(otKeyCipher),\r\n          sendEmail: email\r\n            ? {\r\n                email,\r\n                rawOtKey: otKeyK,\r\n              }\r\n            : null,\r\n          createTp: true,\r\n          initiatorPlainDataSig,\r\n          upgrade,\r\n        },\r\n      },\r\n    });\r\n    return { keyExchange: initiateKeyExchangeOtk.keyExchange, otKeyK };\r\n  }\r\n\r\n  async respondOtk({\r\n    id,\r\n    token,\r\n    decryptedOtk,\r\n    message,\r\n    initiatorContactCard,\r\n    responderContactCard: sentContactCard,\r\n  }: RespondOtkInput): Promise<RespondOtk> {\r\n    const user = await this.authService.getUser();\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 KFS.asKey(\r\n      decryptedOtk.plainOtKeyCipher.initiator.oneTimePbk\r\n    );\r\n\r\n    const initiatorPbk = await KFS.asKey(\r\n      decryptedOtk.plainOtKeyCipher.initiator.pbk\r\n    );\r\n    const initiatorSigPbk = await KFS.asKey(\r\n      decryptedOtk.plainOtKeyCipher.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 plainInitiatorOneTimePbkCipher: PlainInitiatorOneTimePbkCipher = {\r\n      nonce: decryptedOtk.plainOtKeyCipher.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        profile: {\r\n          username: user.username,\r\n        },\r\n        message,\r\n      },\r\n    };\r\n\r\n    let receivedCardInput;\r\n    if (decryptedOtk.plainOtKeyCipher.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 plainSharedCipherDataJson =\r\n        decryptedOtk.plainOtKeyCipher.initiator.contactCard\r\n          .plainSharedCipherDataJson;\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.plainReceiverCipherDataJson\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        plainSharedCipherDataJson\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      plainInitiatorOneTimePbkCipher.responder.contactCard = {\r\n        ...plainInitiatorOneTimePbkCipher.responder.contactCard,\r\n        sharedCipherKey: ccSharedKey.toJSON(true),\r\n      };\r\n    }\r\n\r\n    let sentCardInput;\r\n    if (sentContactCard) {\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      sentCardInput = {\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: sentContactCard.plainOwnerCipherDataJson\r\n          ? JSON.stringify(\r\n              await this.encryptionService.encrypt(\r\n                ownerKey,\r\n                sentContactCard.plainOwnerCipherDataJson\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        sentContactCard.plainSharedCipherDataJson\r\n      );\r\n      sentCardInput.sharedCipherDataSig = JSON.stringify(\r\n        await this.encryptionService.sign(sigPxk.jwk, sharedCipherData)\r\n      );\r\n      sentCardInput.sigPxkId = sigPxk.id;\r\n\r\n      if (sentContactCard.ownerPlainData) {\r\n        sentCardInput.ownerPlainDataSig = JSON.stringify(\r\n          await this.encryptionService.sign(\r\n            responderSigPrk.jwk,\r\n            sentContactCard.ownerPlainData\r\n          )\r\n        );\r\n      }\r\n\r\n      // Contact card info readable by the initiator\r\n      plainInitiatorOneTimePbkCipher.responder.contactCard = {\r\n        ...plainInitiatorOneTimePbkCipher.responder.contactCard,\r\n        plainSharedCipherDataJson: sentContactCard.plainSharedCipherDataJson,\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      plainInitiatorOneTimePbkCipher\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    const { respondKeyExchangeOtk } = await this.lrApollo.mutate<any>({\r\n      mutation: RespondOtkMutation,\r\n      variables: {\r\n        input: {\r\n          keyExchangeId: id,\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: sentCardInput,\r\n        },\r\n      },\r\n    });\r\n\r\n    return {\r\n      keyExchange: respondKeyExchangeOtk.keyExchange,\r\n      userSharedKey: respondKeyExchangeOtk.userSharedKey,\r\n      tp: respondKeyExchangeOtk.tp,\r\n    };\r\n  }\r\n\r\n  async completeOtk(\r\n    keyExchangeId: string,\r\n    initiatorRootKeyCipher: string,\r\n    initiatorOneTimePbkCipher: string,\r\n    responderContactCard?: string\r\n  ): Promise<CompleteOtk> {\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 plainInitiatorRootKeyCipher = ((await this.encryptionService.decrypt(\r\n      rootKey.jwk,\r\n      initiatorRootKeyCipher\r\n    )) as unknown) as PlainInitiatorRootKeyCipher;\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 KFS.asKey(plainInitiatorRootKeyCipher.otKey),\r\n      await KFS.asKey(plainInitiatorRootKeyCipher.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      plainInitiatorRootKeyCipher.nonce !== 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 KFS.asKey(\r\n      plainInitiatorOneTimePbkCipher.responder.sigPbk\r\n    );\r\n    const responderPbk = await KFS.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 KFS.asKey(plainInitiatorOneTimePbkCipher.sharedKey);\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 KFS.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\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 (plainInitiatorRootKeyCipher.initiatorContactCard) {\r\n      const initiatorContactCard =\r\n        plainInitiatorRootKeyCipher.initiatorContactCard;\r\n      const ownerKey = await this.keyFactory.createKey();\r\n      const sharedCipherKey = await KFS.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      const ownerCipherData = initiatorContactCard.plainOwnerCipherDataJson\r\n        ? JSON.stringify(\r\n            await this.encryptionService.encrypt(\r\n              ownerKey,\r\n              initiatorContactCard.plainOwnerCipherDataJson\r\n            )\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        initiatorContactCard.plainSharedCipherDataJson\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    const res = await this.lrApollo.mutate<any>({\r\n      mutation: CompleteOtkMutation,\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    return res.completeKeyExchangeOtk;\r\n  }\r\n\r\n  public async currentUserSharedKey(input: {\r\n    username?: string;\r\n    userId?: string;\r\n  }): Promise<UserSharedKey> {\r\n    const { currentUserSharedKey } = await this.lrApollo.query<any>({\r\n      query: CurrentUserSharedKeyQuery,\r\n      variables: {\r\n        username: input.username,\r\n        userId: input.userId,\r\n      },\r\n    });\r\n    return currentUserSharedKey.userSharedKey;\r\n  }\r\n}\r\n"]}