@lifeready/core 5.0.9 → 5.0.11

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 (56) hide show
  1. package/bundles/lifeready-core.umd.js +1529 -737
  2. package/bundles/lifeready-core.umd.js.map +1 -1
  3. package/bundles/lifeready-core.umd.min.js +1 -1
  4. package/bundles/lifeready-core.umd.min.js.map +1 -1
  5. package/esm2015/lib/_common/key.js +28 -0
  6. package/esm2015/lib/_common/types.js +1 -1
  7. package/esm2015/lib/api/types/lr-graphql.types.js +1 -1
  8. package/esm2015/lib/auth/auth.types.js +1 -3
  9. package/esm2015/lib/auth/life-ready-auth.service.js +2 -2
  10. package/esm2015/lib/auth2/auth2.gql.private.js +78 -0
  11. package/esm2015/lib/auth2/auth2.service.js +596 -0
  12. package/esm2015/lib/auth2/auth2.types.js +21 -0
  13. package/esm2015/lib/contact-card/contact-card.service.js +3 -3
  14. package/esm2015/lib/contact-card/contact-card2.service.js +3 -3
  15. package/esm2015/lib/item2/item2.service.js +9 -9
  16. package/esm2015/lib/key/key-factory.service.js +1 -1
  17. package/esm2015/lib/key/key-graph.service.js +3 -3
  18. package/esm2015/lib/key/key-meta.service.js +2 -2
  19. package/esm2015/lib/key/key.service.js +7 -7
  20. package/esm2015/lib/key-exchange/key-exchange.service.js +24 -29
  21. package/esm2015/lib/key-exchange/key-exchange2.service.js +16 -17
  22. package/esm2015/lib/lbop/lbop.service.js +13 -14
  23. package/esm2015/lib/profile/profile.service.js +2 -2
  24. package/esm2015/lib/profile/profile.types.js +1 -1
  25. package/esm2015/lib/register/register.service.js +1 -1
  26. package/esm2015/lib/register/register.types.js +3 -0
  27. package/esm2015/lib/server-config/server-config.gql.js +1 -1
  28. package/esm2015/lib/shared-contact-card/shared-contact-card.service.js +3 -3
  29. package/esm2015/lib/shared-contact-card/shared-contact-card2.service.js +2 -2
  30. package/esm2015/lib/tp-assembly/tp-assembly.js +3 -3
  31. package/esm2015/lib/trusted-party/trusted-party2.service.js +4 -4
  32. package/esm2015/public-api.js +4 -1
  33. package/fesm2015/lifeready-core.js +885 -203
  34. package/fesm2015/lifeready-core.js.map +1 -1
  35. package/lib/_common/key.d.ts +14 -0
  36. package/lib/_common/types.d.ts +6 -0
  37. package/lib/api/types/lr-graphql.types.d.ts +1 -0
  38. package/lib/auth/auth.types.d.ts +0 -6
  39. package/lib/auth2/auth2.gql.private.d.ts +12 -0
  40. package/lib/auth2/auth2.service.d.ts +70 -0
  41. package/lib/auth2/auth2.types.d.ts +50 -0
  42. package/lib/item2/item2.service.d.ts +3 -3
  43. package/lib/key/key-factory.service.d.ts +1 -0
  44. package/lib/key/key-graph.service.d.ts +2 -3
  45. package/lib/key/key.service.d.ts +6 -6
  46. package/lib/key-exchange/key-exchange.service.d.ts +3 -5
  47. package/lib/lbop/lbop.service.d.ts +3 -3
  48. package/lib/profile/profile.types.d.ts +2 -2
  49. package/lib/register/register.service.d.ts +1 -1
  50. package/lib/register/register.types.d.ts +6 -0
  51. package/lib/server-config/server-config.gql.d.ts +1 -1
  52. package/lib/server-config/server-config.service.d.ts +1 -1
  53. package/lib/shared-contact-card/shared-contact-card.service.d.ts +2 -2
  54. package/lifeready-core.metadata.json +1 -1
  55. package/package.json +1 -1
  56. package/public-api.d.ts +3 -0
@@ -2,11 +2,10 @@ import { __awaiter } from "tslib";
2
2
  /* eslint-disable @typescript-eslint/no-explicit-any */
3
3
  import { Injectable } from '@angular/core';
4
4
  import { LrApolloService } from '../api/lr-apollo.service';
5
- import { LifeReadyAuthService } from '../auth/life-ready-auth.service';
5
+ import { Auth2Service } from '../auth2/auth2.service';
6
6
  import { EncryptionService, JoseSerialization, } from '../encryption/encryption.service';
7
7
  import { KeyFactoryService as KFS } from '../key/key-factory.service';
8
8
  import { KeyService } from '../key/key.service';
9
- import { UserService } from '../user/user.service';
10
9
  import { KcCodeMismatchException } from '../_common/exceptions';
11
10
  import { CompleteOtkMutation, CurrentUserSharedKeyQuery, InitiateOtkMutation, KeyExchangeQuery, KeyExchangesQuery, KeyExchangeTokenQuery, RespondOtkMutation, } from './key-exchange.gql';
12
11
  import { OtkState, } from './key-exchange.types';
@@ -15,16 +14,14 @@ import * as i1 from "../key/key-factory.service";
15
14
  import * as i2 from "../key/key.service";
16
15
  import * as i3 from "../api/lr-apollo.service";
17
16
  import * as i4 from "../encryption/encryption.service";
18
- import * as i5 from "../auth/life-ready-auth.service";
19
- import * as i6 from "../user/user.service";
17
+ import * as i5 from "../auth2/auth2.service";
20
18
  export class KeyExchangeService {
21
- constructor(keyFactory, keyService, lrApollo, encryptionService, authService, userService) {
19
+ constructor(keyFactory, keyService, lrApollo, encryptionService, auth2Service) {
22
20
  this.keyFactory = keyFactory;
23
21
  this.keyService = keyService;
24
22
  this.lrApollo = lrApollo;
25
23
  this.encryptionService = encryptionService;
26
- this.authService = authService;
27
- this.userService = userService;
24
+ this.auth2Service = auth2Service;
28
25
  this.CLIENT_NONCE_LENGTH = 32;
29
26
  }
30
27
  getKeyExchangeList(input = {}) {
@@ -73,7 +70,7 @@ export class KeyExchangeService {
73
70
  decryptKeyExchange(keyExchange, otKeyK) {
74
71
  return __awaiter(this, void 0, void 0, function* () {
75
72
  if (keyExchange.isInitiator) {
76
- const rootKey = yield this.keyService.getCurrentRootKey();
73
+ const rootKey = this.keyService.currentRootKey;
77
74
  // Decrypt using the root key to get the Prk
78
75
  const plainInitiatorRootKeyCipher = (yield this.encryptionService.decrypt(rootKey.jwk, keyExchange.initiatorRootKeyCipher));
79
76
  const plainInitiatorOneTimePbkCipher = keyExchange.otk
@@ -118,7 +115,7 @@ export class KeyExchangeService {
118
115
  !keyExchange.isInitiator &&
119
116
  keyExchange.otk.responderPbkCipher) {
120
117
  // Assuming existing user getting invited where OTK is wrapped in responder's public key.
121
- const prk = yield this.keyService.getCurrentPxk();
118
+ const prk = this.keyService.currentPxk;
122
119
  const decryptedCipher = yield this.encryptionService.decrypt(prk.jwk, JSON.parse(keyExchange.otk.responderPbkCipher), {
123
120
  serializations: [JoseSerialization.COMPACT],
124
121
  });
@@ -133,7 +130,7 @@ export class KeyExchangeService {
133
130
  return __awaiter(this, void 0, void 0, function* () {
134
131
  const otKey = yield this.keyFactory.createKey();
135
132
  const nonce = this.keyFactory.randomString(this.CLIENT_NONCE_LENGTH);
136
- const user = yield this.authService.getUser();
133
+ const user = yield this.auth2Service.getUser();
137
134
  // New PKC key for encryption. This key is used only once when the responder sends
138
135
  // back their signing public key.
139
136
  const initiatorOneTimePrk = yield this.keyFactory.createPkcKey();
@@ -141,8 +138,8 @@ export class KeyExchangeService {
141
138
  // const initiatorSigPrk = await this.keyService.createPkcSignKey();
142
139
  // Option 2: Use the user's global signing key.
143
140
  // This key is used to prove the initiator's identity.
144
- const initiatorPrk = yield this.keyService.getCurrentPxk();
145
- const initiatorSigPrk = yield this.keyService.getCurrentSigPxk();
141
+ const initiatorPrk = this.keyService.currentPxk;
142
+ const initiatorSigPrk = this.keyService.currentSigPxk;
146
143
  let initiatorPlainDataSig = null;
147
144
  if (contactCard && contactCard.ownerPlainData) {
148
145
  initiatorPlainDataSig = JSON.stringify(yield this.encryptionService.sign(initiatorSigPrk.jwk, contactCard.ownerPlainData));
@@ -178,7 +175,7 @@ export class KeyExchangeService {
178
175
  initiatorContactCard: contactCard,
179
176
  initiator,
180
177
  };
181
- const rootKey = yield this.keyService.getCurrentRootKey();
178
+ const rootKey = this.keyService.currentRootKey;
182
179
  const initiatorRootKeyCipher = yield this.encryptionService.encrypt(rootKey.jwk, plainInitiatorRootKeyCipher);
183
180
  // The raw OTK
184
181
  const otKeyK = otKey.toJSON(true).k;
@@ -211,10 +208,9 @@ export class KeyExchangeService {
211
208
  }
212
209
  respondOtk({ id, token, decryptedOtk, message, initiatorContactCard, responderContactCard: sentContactCard, }) {
213
210
  return __awaiter(this, void 0, void 0, function* () {
214
- const user = yield this.authService.getUser();
215
- const rootKey = yield this.keyService.getCurrentRootKey();
216
- const masterKeyId = this.keyService.getCurrentMasterKey().id;
217
- const masterKey = yield this.keyService.getCurrentMasterKey();
211
+ const user = yield this.auth2Service.getUser();
212
+ const rootKey = this.keyService.currentRootKey;
213
+ const masterKey = this.keyService.currentMasterKey;
218
214
  const sharedKey = yield this.keyFactory.createKey();
219
215
  const mkSharedKey = yield this.keyFactory.createKey();
220
216
  const rkWrappedSharedKey = yield this.encryptionService.encrypt(rootKey.jwk, sharedKey.toJSON(true));
@@ -227,8 +223,8 @@ export class KeyExchangeService {
227
223
  // const responderSigPrk = await this.keyService.createPkcSignKey()
228
224
  // const rkWrappedResponderSigPrk = await this.encrypt(rootKey, responderSigPrk.toJSON(true));
229
225
  // Option 2: Responder already has a signing Prk
230
- const responderPrk = yield this.keyService.getCurrentPxk();
231
- const responderSigPrk = yield this.keyService.getCurrentSigPxk();
226
+ const responderPrk = this.keyService.currentPxk;
227
+ const responderSigPrk = this.keyService.currentSigPxk;
232
228
  const signedInitiatorPbk = yield this.encryptionService.sign(responderSigPrk.jwk, initiatorPbk.toJSON());
233
229
  const signedInitiatorSigPbk = yield this.encryptionService.sign(responderSigPrk.jwk, initiatorSigPbk.toJSON());
234
230
  const plainInitiatorOneTimePbkCipher = {
@@ -254,7 +250,7 @@ export class KeyExchangeService {
254
250
  // Create keys
255
251
  const receiverKey = yield this.keyFactory.createKey();
256
252
  const ccSharedKey = yield this.keyFactory.createKey();
257
- const sigPxk = yield this.keyService.getCurrentSigPxk();
253
+ const sigPxk = this.keyService.currentSigPxk;
258
254
  receivedCardInput = {
259
255
  receiverWrappedKey: JSON.stringify(yield this.encryptionService.encrypt(rootKey.jwk, receiverKey.toJSON(true))),
260
256
  receiverWrappingKeyId: rootKey.id,
@@ -273,7 +269,7 @@ export class KeyExchangeService {
273
269
  // Create keys
274
270
  const ownerKey = yield this.keyFactory.createKey();
275
271
  const ccSharedKey = yield this.keyFactory.createKey();
276
- const sigPxk = yield this.keyService.getCurrentSigPxk();
272
+ const sigPxk = this.keyService.currentSigPxk;
277
273
  sentCardInput = {
278
274
  ownerWrappedKey: JSON.stringify(yield this.encryptionService.encrypt(rootKey.jwk, ownerKey.toJSON(true))),
279
275
  ownerWrappingKeyId: rootKey.id,
@@ -302,7 +298,7 @@ export class KeyExchangeService {
302
298
  keyExchangeId: id,
303
299
  keyExchangeToken: token,
304
300
  rootKeyId: rootKey.id,
305
- masterKeyId,
301
+ masterKeyId: masterKey.id,
306
302
  // These will be stored on the server
307
303
  responderPxkId: responderPrk.id,
308
304
  responderSigPxkId: responderSigPrk.id,
@@ -329,8 +325,8 @@ export class KeyExchangeService {
329
325
  }
330
326
  completeOtk(keyExchangeId, initiatorRootKeyCipher, initiatorOneTimePbkCipher, responderContactCard) {
331
327
  return __awaiter(this, void 0, void 0, function* () {
332
- const rootKey = yield this.keyService.getCurrentRootKey();
333
- const masterKey = yield this.keyService.getCurrentMasterKey();
328
+ const rootKey = this.keyService.currentRootKey;
329
+ const masterKey = this.keyService.currentMasterKey;
334
330
  // Decrypt using the root key to get the Prk
335
331
  const plainInitiatorRootKeyCipher = (yield this.encryptionService.decrypt(rootKey.jwk, initiatorRootKeyCipher));
336
332
  // The Prk is single-use and only used to send information from the responder back to the initiator.
@@ -346,7 +342,7 @@ export class KeyExchangeService {
346
342
  // In this case the initiatorSigPrk is already a part of the key graph.
347
343
  // So there's nothing to do here.
348
344
  // Protected the signing public key of the responder.
349
- const initiatorSigPrk = yield this.keyService.getCurrentSigPxk();
345
+ const initiatorSigPrk = this.keyService.currentSigPxk;
350
346
  const responderSigPbk = yield KFS.asKey(plainInitiatorOneTimePbkCipher.responder.sigPbk);
351
347
  const responderPbk = yield KFS.asKey(plainInitiatorOneTimePbkCipher.responder.pbk);
352
348
  const signedResponderPbk = yield this.encryptionService.sign(initiatorSigPrk.jwk, responderPbk.toJSON());
@@ -424,7 +420,7 @@ export class KeyExchangeService {
424
420
  });
425
421
  }
426
422
  }
427
- 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" });
423
+ 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.Auth2Service)); }, token: KeyExchangeService, providedIn: "root" });
428
424
  KeyExchangeService.decorators = [
429
425
  { type: Injectable, args: [{
430
426
  providedIn: 'root',
@@ -435,7 +431,6 @@ KeyExchangeService.ctorParameters = () => [
435
431
  { type: KeyService },
436
432
  { type: LrApolloService },
437
433
  { type: EncryptionService },
438
- { type: LifeReadyAuthService },
439
- { type: UserService }
434
+ { type: Auth2Service }
440
435
  ];
441
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"key-exchange.service.js","sourceRoot":"","sources":["../../../../../../projects/core/src/lib/key-exchange/key-exchange.service.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable @typescript-eslint/no-explicit-any */\nimport { Injectable } from '@angular/core';\nimport { JWK } from 'node-jose';\nimport { LrApolloService } from '../api/lr-apollo.service';\nimport { LifeReadyAuthService } from '../auth/life-ready-auth.service';\nimport {\n  EncryptionService,\n  JoseSerialization,\n} from '../encryption/encryption.service';\nimport { KeyFactoryService as KFS } from '../key/key-factory.service';\nimport { KeyService } from '../key/key.service';\nimport { KeyGraphResponse } from '../key/key.types';\nimport { UserService } from '../user/user.service';\nimport { KcCodeMismatchException } from '../_common/exceptions';\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  RespondOtk,\n  RespondOtkInput,\n  UserSharedKey,\n} from './key-exchange.types';\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 KcCodeMismatchException(\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 =\n        JSON.stringify(\n          await this.encryptionService.sign(\n            initiatorSigPrk.jwk,\n            sharedCipherData\n          )\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:\n            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"]}
436
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"key-exchange.service.js","sourceRoot":"","sources":["../../../../../../projects/core/src/lib/key-exchange/key-exchange.service.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable @typescript-eslint/no-explicit-any */\nimport { Injectable } from '@angular/core';\nimport { JWK } from 'node-jose';\nimport { LrApolloService } from '../api/lr-apollo.service';\nimport { Auth2Service } from '../auth2/auth2.service';\nimport {\n  EncryptionService,\n  JoseSerialization,\n} from '../encryption/encryption.service';\nimport { KeyFactoryService as KFS } from '../key/key-factory.service';\nimport { KeyService } from '../key/key.service';\nimport { KeyGraphResponse } from '../key/key.types';\nimport { KcCodeMismatchException } from '../_common/exceptions';\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  RespondOtk,\n  RespondOtkInput,\n  UserSharedKey,\n} from './key-exchange.types';\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 auth2Service: Auth2Service\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 = this.keyService.currentRootKey;\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 = this.keyService.currentPxk;\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.auth2Service.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 = this.keyService.currentPxk;\n    const initiatorSigPrk = this.keyService.currentSigPxk;\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 = this.keyService.currentRootKey;\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.auth2Service.getUser();\n    const rootKey = this.keyService.currentRootKey;\n\n    const masterKey = this.keyService.currentMasterKey;\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 = this.keyService.currentPxk;\n    const responderSigPrk = this.keyService.currentSigPxk;\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 = this.keyService.currentSigPxk;\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 = this.keyService.currentSigPxk;\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: masterKey.id,\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 = this.keyService.currentRootKey;\n    const masterKey = this.keyService.currentMasterKey;\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 KcCodeMismatchException(\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 = this.keyService.currentSigPxk;\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 =\n        JSON.stringify(\n          await this.encryptionService.sign(\n            initiatorSigPrk.jwk,\n            sharedCipherData\n          )\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:\n            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"]}