@lifeready/core 1.0.4 → 1.0.6

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 (226) 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 +1 -1
  5. package/bundles/lifeready-core.umd.min.js.map +1 -1
  6. package/esm2015/lib/_common/ast.js +40 -40
  7. package/esm2015/lib/_common/deferred-promise.js +24 -24
  8. package/esm2015/lib/_common/exceptions.js +157 -157
  9. package/esm2015/lib/_common/queries.gql.js +190 -190
  10. package/esm2015/lib/_common/run-outside-angular.js +79 -79
  11. package/esm2015/lib/_common/types.js +1 -1
  12. package/esm2015/lib/_common/utils.js +44 -44
  13. package/esm2015/lib/api/contact-card.gql.js +79 -79
  14. package/esm2015/lib/api/contact-card.service.js +154 -154
  15. package/esm2015/lib/api/contact-card2.gql.js +60 -60
  16. package/esm2015/lib/api/contact-card2.service.js +103 -103
  17. package/esm2015/lib/api/file.service.js +74 -74
  18. package/esm2015/lib/api/item2.gql.js +110 -110
  19. package/esm2015/lib/api/item2.service.js +311 -311
  20. package/esm2015/lib/api/key-exchange.gql.js +188 -188
  21. package/esm2015/lib/api/key-exchange.service.js +442 -442
  22. package/esm2015/lib/api/key-exchange.types.js +18 -18
  23. package/esm2015/lib/api/key-exchange2.gql.js +171 -171
  24. package/esm2015/lib/api/key-exchange2.service.js +479 -479
  25. package/esm2015/lib/api/lock.gql.js +40 -40
  26. package/esm2015/lib/api/lock.service.js +64 -64
  27. package/esm2015/lib/api/lr-apollo.service.js +46 -46
  28. package/esm2015/lib/api/lr-graphql/index.js +6 -6
  29. package/esm2015/lib/api/lr-graphql/lr-graphql.service.js +155 -155
  30. package/esm2015/lib/api/lr-graphql/lr-merged-mutation.js +213 -213
  31. package/esm2015/lib/api/lr-graphql/lr-mutation-base.js +51 -51
  32. package/esm2015/lib/api/lr-graphql/lr-mutation.js +48 -48
  33. package/esm2015/lib/api/lr-graphql/lr.service.js +18 -18
  34. package/esm2015/lib/api/message.service.js +138 -138
  35. package/esm2015/lib/api/persist.service.js +181 -181
  36. package/esm2015/lib/api/query-processor/common-processors.service.js +93 -93
  37. package/esm2015/lib/api/query-processor/index.js +3 -3
  38. package/esm2015/lib/api/query-processor/query-processor.service.js +192 -192
  39. package/esm2015/lib/api/query-processor/tp-password-reset-processor.service.js +109 -109
  40. package/esm2015/lib/api/shared-contact-card.service.js +119 -119
  41. package/esm2015/lib/api/shared-contact-card2.gql.js +41 -41
  42. package/esm2015/lib/api/shared-contact-card2.service.js +117 -117
  43. package/esm2015/lib/api/time.service.js +146 -146
  44. package/esm2015/lib/api/types/graphql.types.js +7 -7
  45. package/esm2015/lib/api/types/index.js +3 -3
  46. package/esm2015/lib/api/types/lr-graphql.types.js +71 -71
  47. package/esm2015/lib/auth/auth.config.js +57 -57
  48. package/esm2015/lib/auth/auth.gql.js +48 -48
  49. package/esm2015/lib/auth/auth.types.js +27 -27
  50. package/esm2015/lib/auth/idle.service.js +168 -168
  51. package/esm2015/lib/auth/idle.types.js +7 -7
  52. package/esm2015/lib/auth/lbop.service.js +355 -355
  53. package/esm2015/lib/auth/life-ready-auth.service.js +500 -500
  54. package/esm2015/lib/auth/password.service.js +320 -320
  55. package/esm2015/lib/auth/register.service.js +172 -172
  56. package/esm2015/lib/auth/two-factor.service.js +74 -74
  57. package/esm2015/lib/category/category-meta.service.js +99 -99
  58. package/esm2015/lib/category/category.gql.js +406 -406
  59. package/esm2015/lib/category/category.service.js +390 -390
  60. package/esm2015/lib/category/category.types.js +29 -29
  61. package/esm2015/lib/cryptography/cryptography.types.js +11 -11
  62. package/esm2015/lib/cryptography/encryption.service.js +189 -189
  63. package/esm2015/lib/cryptography/key-factory.service.js +237 -237
  64. package/esm2015/lib/cryptography/key-graph.service.js +280 -280
  65. package/esm2015/lib/cryptography/key-meta.service.js +200 -200
  66. package/esm2015/lib/cryptography/key.service.js +124 -124
  67. package/esm2015/lib/cryptography/slip39.service.js +169 -169
  68. package/esm2015/lib/cryptography/web-crypto.service.js +29 -29
  69. package/esm2015/lib/life-ready.config.js +84 -84
  70. package/esm2015/lib/life-ready.module.js +74 -74
  71. package/esm2015/lib/plan/plan.gql.js +123 -123
  72. package/esm2015/lib/plan/plan.service.js +149 -149
  73. package/esm2015/lib/plan/plan.types.js +11 -11
  74. package/esm2015/lib/record/record-attachment.service.js +101 -101
  75. package/esm2015/lib/record/record.gql.js +179 -179
  76. package/esm2015/lib/record/record.service.js +206 -206
  77. package/esm2015/lib/record/record.types.js +15 -15
  78. package/esm2015/lib/record-type/record-type.service.js +75 -75
  79. package/esm2015/lib/record-type/record-type.types.js +28 -28
  80. package/esm2015/lib/scenario/approvals/scenario-approval.gql.js +105 -105
  81. package/esm2015/lib/scenario/approvals/scenario-approval.types.js +1 -1
  82. package/esm2015/lib/scenario/approvals/scenario-approver.service.js +300 -300
  83. package/esm2015/lib/scenario/claimants/scenario-claimant.gql.js +52 -52
  84. package/esm2015/lib/scenario/claimants/scenario-claimant.service.js +97 -97
  85. package/esm2015/lib/scenario/claimants/scenario-claimant.types.js +1 -1
  86. package/esm2015/lib/scenario/receivers/scenario-receiver.gql.js +150 -150
  87. package/esm2015/lib/scenario/receivers/scenario-receiver.service.js +229 -229
  88. package/esm2015/lib/scenario/receivers/scenario-receiver.types.js +1 -1
  89. package/esm2015/lib/scenario/scenario-setup.service.js +269 -269
  90. package/esm2015/lib/scenario/scenario.gql.js +368 -368
  91. package/esm2015/lib/scenario/scenario.service.js +611 -611
  92. package/esm2015/lib/scenario/scenario.types.js +64 -64
  93. package/esm2015/lib/search/search.gql.js +62 -62
  94. package/esm2015/lib/search/search.service.js +156 -156
  95. package/esm2015/lib/search/search.types.js +6 -6
  96. package/esm2015/lib/trusted-parties/tp-password-reset-request.service.js +112 -112
  97. package/esm2015/lib/trusted-parties/tp-password-reset-user.service.js +129 -129
  98. package/esm2015/lib/trusted-parties/tp-password-reset.constants.js +4 -4
  99. package/esm2015/lib/trusted-parties/tp-password-reset.gql.js +232 -232
  100. package/esm2015/lib/trusted-parties/tp-password-reset.service.js +299 -299
  101. package/esm2015/lib/trusted-parties/trusted-party.gql.js +148 -148
  102. package/esm2015/lib/trusted-parties/trusted-party.service.js +326 -326
  103. package/esm2015/lib/trusted-parties/trusted-party.types.js +41 -41
  104. package/esm2015/lib/trusted-parties/trusted-party2.gql.js +87 -87
  105. package/esm2015/lib/trusted-parties/trusted-party2.service.js +215 -215
  106. package/esm2015/lib/users/profile-details.service.js +214 -214
  107. package/esm2015/lib/users/profile.gql.js +97 -97
  108. package/esm2015/lib/users/profile.service.js +169 -169
  109. package/esm2015/lib/users/profile.types.js +34 -34
  110. package/esm2015/lib/users/user.gql.js +60 -60
  111. package/esm2015/lib/users/user.service.js +79 -79
  112. package/esm2015/lib/users/user.types.js +5 -5
  113. package/esm2015/lifeready-core.js +10 -10
  114. package/esm2015/public-api.js +81 -81
  115. package/fesm2015/lifeready-core.js +13088 -13088
  116. package/fesm2015/lifeready-core.js.map +1 -1
  117. package/lib/_common/ast.d.ts +11 -11
  118. package/lib/_common/deferred-promise.d.ts +12 -12
  119. package/lib/_common/exceptions.d.ts +109 -109
  120. package/lib/_common/queries.gql.d.ts +10 -10
  121. package/lib/_common/run-outside-angular.d.ts +14 -14
  122. package/lib/_common/types.d.ts +10 -10
  123. package/lib/_common/utils.d.ts +3 -3
  124. package/lib/api/contact-card.gql.d.ts +7 -7
  125. package/lib/api/contact-card.service.d.ts +52 -52
  126. package/lib/api/contact-card2.gql.d.ts +34 -34
  127. package/lib/api/contact-card2.service.d.ts +49 -49
  128. package/lib/api/file.service.d.ts +18 -18
  129. package/lib/api/item2.gql.d.ts +96 -96
  130. package/lib/api/item2.service.d.ts +177 -177
  131. package/lib/api/key-exchange.gql.d.ts +9 -9
  132. package/lib/api/key-exchange.service.d.ts +39 -39
  133. package/lib/api/key-exchange.types.d.ts +196 -196
  134. package/lib/api/key-exchange2.gql.d.ts +125 -125
  135. package/lib/api/key-exchange2.service.d.ts +187 -187
  136. package/lib/api/lock.gql.d.ts +27 -27
  137. package/lib/api/lock.service.d.ts +25 -25
  138. package/lib/api/lr-apollo.service.d.ts +15 -15
  139. package/lib/api/lr-graphql/index.d.ts +5 -5
  140. package/lib/api/lr-graphql/lr-graphql.service.d.ts +60 -60
  141. package/lib/api/lr-graphql/lr-merged-mutation.d.ts +27 -27
  142. package/lib/api/lr-graphql/lr-mutation-base.d.ts +28 -28
  143. package/lib/api/lr-graphql/lr-mutation.d.ts +8 -8
  144. package/lib/api/lr-graphql/lr.service.d.ts +9 -9
  145. package/lib/api/message.service.d.ts +58 -58
  146. package/lib/api/persist.service.d.ts +31 -31
  147. package/lib/api/query-processor/common-processors.service.d.ts +36 -36
  148. package/lib/api/query-processor/index.d.ts +2 -2
  149. package/lib/api/query-processor/query-processor.service.d.ts +18 -18
  150. package/lib/api/query-processor/tp-password-reset-processor.service.d.ts +15 -15
  151. package/lib/api/shared-contact-card.service.d.ts +33 -33
  152. package/lib/api/shared-contact-card2.gql.d.ts +36 -36
  153. package/lib/api/shared-contact-card2.service.d.ts +45 -45
  154. package/lib/api/time.service.d.ts +16 -16
  155. package/lib/api/types/graphql.types.d.ts +29 -29
  156. package/lib/api/types/index.d.ts +2 -2
  157. package/lib/api/types/lr-graphql.types.d.ts +393 -385
  158. package/lib/auth/auth.config.d.ts +5 -5
  159. package/lib/auth/auth.gql.d.ts +15 -15
  160. package/lib/auth/auth.types.d.ts +66 -66
  161. package/lib/auth/idle.service.d.ts +40 -40
  162. package/lib/auth/idle.types.d.ts +10 -10
  163. package/lib/auth/lbop.service.d.ts +91 -91
  164. package/lib/auth/life-ready-auth.service.d.ts +59 -59
  165. package/lib/auth/password.service.d.ts +78 -78
  166. package/lib/auth/register.service.d.ts +25 -25
  167. package/lib/auth/two-factor.service.d.ts +15 -15
  168. package/lib/category/category-meta.service.d.ts +23 -23
  169. package/lib/category/category.gql.d.ts +45 -45
  170. package/lib/category/category.service.d.ts +67 -67
  171. package/lib/category/category.types.d.ts +79 -79
  172. package/lib/cryptography/cryptography.types.d.ts +83 -83
  173. package/lib/cryptography/encryption.service.d.ts +41 -41
  174. package/lib/cryptography/key-factory.service.d.ts +38 -38
  175. package/lib/cryptography/key-graph.service.d.ts +33 -33
  176. package/lib/cryptography/key-meta.service.d.ts +44 -44
  177. package/lib/cryptography/key.service.d.ts +36 -36
  178. package/lib/cryptography/slip39.service.d.ts +43 -43
  179. package/lib/cryptography/web-crypto.service.d.ts +5 -5
  180. package/lib/life-ready.config.d.ts +14 -14
  181. package/lib/life-ready.module.d.ts +5 -5
  182. package/lib/plan/plan.gql.d.ts +11 -11
  183. package/lib/plan/plan.service.d.ts +33 -33
  184. package/lib/plan/plan.types.d.ts +31 -31
  185. package/lib/record/record-attachment.service.d.ts +16 -16
  186. package/lib/record/record.gql.d.ts +14 -14
  187. package/lib/record/record.service.d.ts +25 -25
  188. package/lib/record/record.types.d.ts +57 -57
  189. package/lib/record-type/record-type.service.d.ts +11 -11
  190. package/lib/record-type/record-type.types.d.ts +50 -50
  191. package/lib/scenario/approvals/scenario-approval.gql.d.ts +7 -7
  192. package/lib/scenario/approvals/scenario-approval.types.d.ts +63 -63
  193. package/lib/scenario/approvals/scenario-approver.service.d.ts +32 -32
  194. package/lib/scenario/claimants/scenario-claimant.gql.d.ts +5 -5
  195. package/lib/scenario/claimants/scenario-claimant.service.d.ts +17 -17
  196. package/lib/scenario/claimants/scenario-claimant.types.d.ts +18 -18
  197. package/lib/scenario/receivers/scenario-receiver.gql.d.ts +8 -8
  198. package/lib/scenario/receivers/scenario-receiver.service.d.ts +30 -30
  199. package/lib/scenario/receivers/scenario-receiver.types.d.ts +54 -54
  200. package/lib/scenario/scenario-setup.service.d.ts +22 -22
  201. package/lib/scenario/scenario.gql.d.ts +34 -34
  202. package/lib/scenario/scenario.service.d.ts +58 -58
  203. package/lib/scenario/scenario.types.d.ts +217 -217
  204. package/lib/search/search.gql.d.ts +1 -1
  205. package/lib/search/search.service.d.ts +25 -25
  206. package/lib/search/search.types.d.ts +20 -20
  207. package/lib/trusted-parties/tp-password-reset-request.service.d.ts +20 -20
  208. package/lib/trusted-parties/tp-password-reset-user.service.d.ts +35 -35
  209. package/lib/trusted-parties/tp-password-reset.constants.d.ts +3 -3
  210. package/lib/trusted-parties/tp-password-reset.gql.d.ts +218 -218
  211. package/lib/trusted-parties/tp-password-reset.service.d.ts +130 -130
  212. package/lib/trusted-parties/trusted-party.gql.d.ts +9 -9
  213. package/lib/trusted-parties/trusted-party.service.d.ts +44 -44
  214. package/lib/trusted-parties/trusted-party.types.d.ts +102 -102
  215. package/lib/trusted-parties/trusted-party2.gql.d.ts +79 -79
  216. package/lib/trusted-parties/trusted-party2.service.d.ts +114 -114
  217. package/lib/users/profile-details.service.d.ts +21 -21
  218. package/lib/users/profile.gql.d.ts +11 -11
  219. package/lib/users/profile.service.d.ts +35 -35
  220. package/lib/users/profile.types.d.ts +96 -96
  221. package/lib/users/user.gql.d.ts +9 -9
  222. package/lib/users/user.service.d.ts +12 -12
  223. package/lib/users/user.types.d.ts +23 -23
  224. package/lifeready-core.d.ts +9 -9
  225. package/package.json +1 -1
  226. package/public-api.d.ts +77 -77
@@ -1,500 +1,500 @@
1
- import { __awaiter } from "tslib";
2
- import { Inject, Injectable, isDevMode } from '@angular/core';
3
- import { Hub } from '@aws-amplify/core';
4
- import { ReplaySubject } from 'rxjs';
5
- import { KeyGraphService } from '../cryptography/key-graph.service';
6
- import { KeyService } from '../cryptography/key.service';
7
- import { ProfileService } from '../users/profile.service';
8
- import { PasswordChangeStatus } from '../users/profile.types';
9
- import { LrConcurrentAccessException, LrBadRequestException, LrBadStateException, LrException, } from '../_common/exceptions';
10
- import { RecoveryStatus, } from './auth.types';
11
- import { PasswordService } from './password.service';
12
- import { AuthClass } from '@aws-amplify/auth/lib-esm/Auth';
13
- import { CompleteTpPasswordResetRequestMutation, CreateTpAssemblyKeyChallengeMutation, PreCompleteTpPasswordResetRequestMutation, TpPasswordResetUserQuery, } from '../trusted-parties/tp-password-reset.gql';
14
- import { IdleService } from './idle.service';
15
- import { KeyFactoryService } from '../cryptography/key-factory.service';
16
- import { LrGraphQLService, LrMutation } from '../api/lr-graphql';
17
- import { TpClaimState } from '../api/types';
18
- import { TpPasswordResetProcessorService } from '../api/query-processor/tp-password-reset-processor.service';
19
- import { SetSessionEncryptionKeyMutation } from './auth.gql';
20
- import { PersistService } from '../api/persist.service';
21
- import { JWK } from 'node-jose';
22
- import { LR_CONFIG } from '../life-ready.config';
23
- import { EncryptionService } from '../cryptography/encryption.service';
24
- import { Slip39Service } from '../cryptography/slip39.service';
25
- import { TP_PASSWORD_RESET_CLIENT_NONCE_LENGTH, TP_PASSWORD_RESET_SLIP39_PASSPHRASE, TP_PASSWORD_RESET_USERNAME_SUFFIX, } from '../trusted-parties/tp-password-reset.constants';
26
- import * as i0 from "@angular/core";
27
- import * as i1 from "../life-ready.config";
28
- import * as i2 from "@aws-amplify/auth/lib-esm/Auth";
29
- import * as i3 from "../cryptography/key-factory.service";
30
- import * as i4 from "../cryptography/key.service";
31
- import * as i5 from "../users/profile.service";
32
- import * as i6 from "../cryptography/key-graph.service";
33
- import * as i7 from "./password.service";
34
- import * as i8 from "./idle.service";
35
- import * as i9 from "../api/lr-graphql/lr-graphql.service";
36
- import * as i10 from "../api/query-processor/tp-password-reset-processor.service";
37
- import * as i11 from "../api/persist.service";
38
- import * as i12 from "../cryptography/encryption.service";
39
- import * as i13 from "../cryptography/slip39.service";
40
- export const initialiseAuth = (authService) => {
41
- return () => authService.initialise();
42
- };
43
- export class LifeReadyAuthService {
44
- constructor(config, auth, keyFactory, keyService, profileService, keyGraphService, passwordService, idleService, lrGraphQL, tpPasswordResetProcessorService, persistService, encryptionService, slip39Service) {
45
- this.config = config;
46
- this.auth = auth;
47
- this.keyFactory = keyFactory;
48
- this.keyService = keyService;
49
- this.profileService = profileService;
50
- this.keyGraphService = keyGraphService;
51
- this.passwordService = passwordService;
52
- this.idleService = idleService;
53
- this.lrGraphQL = lrGraphQL;
54
- this.tpPasswordResetProcessorService = tpPasswordResetProcessorService;
55
- this.persistService = persistService;
56
- this.encryptionService = encryptionService;
57
- this.slip39Service = slip39Service;
58
- this.hubSubject = new ReplaySubject(1);
59
- }
60
- initialise() {
61
- return __awaiter(this, void 0, void 0, function* () {
62
- Hub.listen('auth', (data) => this.hubSubject.next(data.payload));
63
- });
64
- }
65
- loginIdpImpl(emailOrPhone, password, passIdpParams, recoveryStatus) {
66
- return __awaiter(this, void 0, void 0, function* () {
67
- const passIdpResult = yield this.keyFactory.derivePassIdp(Object.assign({ password }, passIdpParams));
68
- // Use the derived password to signin with cognito
69
- const user = yield this.auth.signIn(emailOrPhone, this.passwordService.getPassIdpString(passIdpResult.jwk));
70
- user.recoveryStatus = recoveryStatus;
71
- return user;
72
- });
73
- }
74
- loginIdp(emailOrPhone, password) {
75
- return __awaiter(this, void 0, void 0, function* () {
76
- // Download the salt needed to derive the PassIdp
77
- const passIdpApiResult = yield this.profileService.getPassIdpParams(emailOrPhone);
78
- if (passIdpApiResult.passwordChangeStatus === PasswordChangeStatus.InProgress) {
79
- throw new LrConcurrentAccessException('A password change is in progress');
80
- }
81
- if (passIdpApiResult.passwordChangeStatus === PasswordChangeStatus.Recovery) {
82
- console.log('In recovery mode.');
83
- // Let's say we don't know if the password is the new one or the old one. We just have to try both.
84
- try {
85
- const user = yield this.loginIdpImpl(emailOrPhone, password, passIdpApiResult.newPassIdpParams, RecoveryStatus.NEW_PASSWORD);
86
- // New password worked. Let's set to the current password
87
- // --Potential Failure Point 1--
88
- // if changePasswordComplete() doesn't get called, then it should remain
89
- console.log('New password works!');
90
- return user;
91
- }
92
- catch (error) {
93
- // Just bubble up any other type of error.
94
- if (error.code !== 'NotAuthorizedException') {
95
- throw error;
96
- }
97
- // pass, try again assuming it's the old password
98
- }
99
- // Now assume it's the previous password. Any exception is allowed to bubble up.
100
- try {
101
- const user = yield this.loginIdpImpl(emailOrPhone, password, passIdpApiResult.currentPassIdpParams, RecoveryStatus.OLD_PASSWORD);
102
- // Old password worked.
103
- console.log('Old password works!');
104
- return user;
105
- }
106
- catch (error) {
107
- // Just bubble up any other type of error.
108
- throw error.code === 'NotAuthorizedException'
109
- ? new LrBadRequestException('The password change request was interrupted, please try to login with both your new and old password')
110
- : error;
111
- }
112
- }
113
- // Try against as the TP password reset account
114
- if (passIdpApiResult.tpPasswordReset) {
115
- try {
116
- // TP password reset is in process. We need to try the password against both
117
- // original account and the new reset account.
118
- const reset = passIdpApiResult.tpPasswordReset;
119
- const ret = yield this.loginIdpImpl(reset.resetUsername, password, reset.passIdpParams, RecoveryStatus.NONE);
120
- ret.isTpPasswordResetUser = true;
121
- return ret;
122
- }
123
- catch (err) {
124
- // continue, try again as regular user.
125
- }
126
- }
127
- // Login as regular user
128
- return yield this.loginIdpImpl(emailOrPhone, password, passIdpApiResult.currentPassIdpParams, RecoveryStatus.NONE);
129
- });
130
- }
131
- handleSessionEncryptionKey() {
132
- return __awaiter(this, void 0, void 0, function* () {
133
- if (this.config.disableSessionEncryptionKey) {
134
- if (!isDevMode()) {
135
- const msg = 'You should not set disableSessionEncryptionKey=True in mode prod. It defaults to false.';
136
- console.error(msg);
137
- throw new Error(msg);
138
- }
139
- else {
140
- console.warn('You have set disableSessionEncryptionKey=True. Make sure not to do this in prod mode.');
141
- }
142
- }
143
- else {
144
- // Set the session key to a new encryption key for this session
145
- const sessionEncryptionKey = yield this.keyFactory.createKey();
146
- yield this.lrGraphQL.lrMutate(new LrMutation({
147
- mutation: SetSessionEncryptionKeyMutation,
148
- variables: {
149
- input: {
150
- sessionEncryptionKey: JSON.stringify(sessionEncryptionKey.toJSON(true)),
151
- },
152
- },
153
- }), {
154
- includeKeyGraph: false,
155
- });
156
- this.persistService.setServerSessionEncryptionKey(sessionEncryptionKey);
157
- }
158
- });
159
- }
160
- handlePostAuth(cognitoUser) {
161
- return __awaiter(this, void 0, void 0, function* () {
162
- yield this.handlePasswordRecovery(cognitoUser);
163
- yield this.handleSessionEncryptionKey();
164
- });
165
- }
166
- login(emailOrPhone, password, { tpPasswordResetAutoComplete = true } = {}) {
167
- var _a;
168
- return __awaiter(this, void 0, void 0, function* () {
169
- let loginResult = yield this.loginImpl(emailOrPhone, password);
170
- if (tpPasswordResetAutoComplete &&
171
- ((_a = loginResult.resetUser) === null || _a === void 0 ? void 0 : _a.state) === TpClaimState.APPROVED) {
172
- yield this.completeRequest(password);
173
- loginResult = yield this.loginImpl(emailOrPhone, password);
174
- }
175
- return loginResult;
176
- });
177
- }
178
- loginImpl(emailOrPhone, password) {
179
- return __awaiter(this, void 0, void 0, function* () {
180
- yield this.logout();
181
- const cognitoUser = yield this.loginIdp(emailOrPhone, password);
182
- // todo: Meet MFA challenges.
183
- if (['SMS_MFA', 'SOFTWARE_TOKEN_MFA'].includes(cognitoUser.challengeName)) {
184
- return { hasChallenge: true, challenge: cognitoUser };
185
- }
186
- yield this.handlePostAuth(cognitoUser);
187
- if (cognitoUser.isTpPasswordResetUser) {
188
- // Assuming there is no MFA on the TP reset user.
189
- const resetUser = yield this.loadResetUser(password);
190
- return { hasChallenge: false, resetUser };
191
- }
192
- else {
193
- const user = yield this.loadUser(cognitoUser, password);
194
- yield this.idleService.start(); // Run idleService whenever user is logged in.
195
- return { hasChallenge: false, user };
196
- }
197
- });
198
- }
199
- // TODO <AZ> We need to handle the isTpPasswordResetUser=True case here after MFA as well.
200
- verifyLogin(challenge, password, rememberMe, code) {
201
- return __awaiter(this, void 0, void 0, function* () {
202
- yield this.auth.confirmSignIn(challenge, code, challenge.challengeName);
203
- // TODO: this.auth.confirmSignIn() could return another challenge.
204
- const cognitoUser = yield this.auth.currentAuthenticatedUser();
205
- yield this.handlePostAuth(challenge);
206
- const user = yield this.loadUser(cognitoUser, password);
207
- if (rememberMe) {
208
- cognitoUser.setDeviceStatusRemembered({
209
- onSuccess: () => { },
210
- onFailure: (e) => console.error(e),
211
- });
212
- }
213
- return user;
214
- });
215
- }
216
- handlePasswordRecovery(user) {
217
- return __awaiter(this, void 0, void 0, function* () {
218
- if (user.recoveryStatus !== RecoveryStatus.NONE) {
219
- const jwtToken = user
220
- .getSignInUserSession()
221
- .getAccessToken()
222
- .getJwtToken();
223
- yield this.passwordService.changePasswordComplete(jwtToken, user.recoveryStatus === RecoveryStatus.NEW_PASSWORD);
224
- }
225
- });
226
- }
227
- getUserOrResetUser(reload = false) {
228
- return __awaiter(this, void 0, void 0, function* () {
229
- const cognitoUser = yield this.auth.currentAuthenticatedUser();
230
- if (cognitoUser.getUsername().endsWith(TP_PASSWORD_RESET_USERNAME_SUFFIX)) {
231
- return this.getResetUser(reload);
232
- }
233
- else {
234
- return this.getUser(reload);
235
- }
236
- });
237
- }
238
- getResetUser(reload = false) {
239
- return __awaiter(this, void 0, void 0, function* () {
240
- if (!reload && this.currentResetUser) {
241
- return this.currentResetUser;
242
- }
243
- this.currentResetUser = yield this.loadResetUser();
244
- yield this.idleService.start(); // Run idleService whenever user is logged in.
245
- return this.currentResetUser;
246
- });
247
- }
248
- getUser(reload = false) {
249
- return __awaiter(this, void 0, void 0, function* () {
250
- if (!reload && this.currentUser) {
251
- return this.currentUser;
252
- }
253
- this.currentUser = yield this.loadUser(yield this.auth.currentAuthenticatedUser());
254
- console.log('Starting idle service.');
255
- yield this.idleService.start(); // Run idleService whenever user is logged in.
256
- return this.currentUser;
257
- });
258
- }
259
- mapTPVaultAccess(features) {
260
- const tpVaultFeature = features === null || features === void 0 ? void 0 : features.tpVault;
261
- return ((tpVaultFeature === null || tpVaultFeature === void 0 ? void 0 : tpVaultFeature.length) > 0 &&
262
- tpVaultFeature.some((feature) => feature.toUpperCase() === 'ACCESS'));
263
- }
264
- loadUser(cognitoUser, password) {
265
- return __awaiter(this, void 0, void 0, function* () {
266
- const { currentUser, contactCard, userPlans, } = yield this.profileService.getCurrentUser();
267
- if (currentUser.sessionEncryptionKey) {
268
- this.persistService.setServerSessionEncryptionKey(yield JWK.asKey(currentUser.sessionEncryptionKey));
269
- }
270
- const userAttributes = yield this.auth.userAttributes(cognitoUser);
271
- if (password) {
272
- const passKey = (yield this.keyFactory.derivePassKey(Object.assign({ password }, currentUser.currentUserKey.passKey.passKeyParams))).jwk;
273
- yield this.idleService.persistMasterKey(yield this.keyGraphService.unwrapWithPassKey(currentUser.currentUserKey.passKey.id, passKey, currentUser.currentUserKey.masterKey.id));
274
- }
275
- yield this.keyGraphService.populateKeys(currentUser.currentUserKey);
276
- return {
277
- id: currentUser.id,
278
- sub: this.getUserAttribute('sub', userAttributes),
279
- username: currentUser.username,
280
- currentUserKey: currentUser.currentUserKey,
281
- getAccessJwtToken: () => cognitoUser.getSignInUserSession().getAccessToken().getJwtToken(),
282
- email: this.getUserAttribute('email', userAttributes),
283
- emailVerified: this.getUserAttribute('email_verified', userAttributes) === 'true',
284
- phone: this.getUserAttribute('phone_number', userAttributes),
285
- phoneVerified: this.getUserAttribute('phone_number_verified', userAttributes) ===
286
- 'true',
287
- contactCard: Object.assign({}, (yield this.profileService.decryptContactCard(contactCard))),
288
- userDelete: currentUser.userDelete,
289
- userPlans,
290
- hasTPVaultAccess: this.mapTPVaultAccess(currentUser.features),
291
- features: currentUser.features,
292
- sessionEncryptionKey: currentUser.sessionEncryptionKey,
293
- };
294
- });
295
- }
296
- watchAuth() {
297
- return this.hubSubject;
298
- }
299
- logout() {
300
- return __awaiter(this, void 0, void 0, function* () {
301
- this.currentUser = null;
302
- this.keyService.purgeKeys();
303
- this.keyGraphService.purgeKeys();
304
- yield Promise.all([this.auth.signOut(), this.profileService.signOut()]);
305
- });
306
- }
307
- getUserAttribute(attributeName, userAttributes) {
308
- const userAttribute = userAttributes.find((x) => x.getName() === attributeName);
309
- return userAttribute ? userAttribute.getValue() : null;
310
- }
311
- loadResetUser(password) {
312
- return __awaiter(this, void 0, void 0, function* () {
313
- const { tpPasswordResetUser: resetUser } = yield this.lrGraphQL.query({
314
- query: TpPasswordResetUserQuery,
315
- });
316
- if (resetUser.sessionEncryptionKey) {
317
- this.persistService.setServerSessionEncryptionKey(yield JWK.asKey(resetUser.sessionEncryptionKey));
318
- }
319
- // Update the keys
320
- if (password) {
321
- const passKey = (yield this.keyFactory.derivePassKey(Object.assign({ password }, resetUser.passKey.passKeyParams))).jwk;
322
- yield this.idleService.persistMasterKey(yield this.keyGraphService.unwrapWithPassKey(resetUser.passKey.id, passKey, resetUser.masterKey.id));
323
- }
324
- this.keyService.populateKeys({
325
- passKey: {
326
- id: resetUser.passKey.id,
327
- },
328
- masterKey: {
329
- id: resetUser.masterKey.id,
330
- },
331
- });
332
- const userAttributes = yield this.auth.userAttributes(yield this.auth.currentAuthenticatedUser());
333
- const sub = this.getUserAttribute('sub', userAttributes);
334
- return Object.assign(Object.assign({}, (yield this.tpPasswordResetProcessorService.processTpPasswordResetUserNode(resetUser))), { sub });
335
- });
336
- }
337
- refreshAccessToken() {
338
- return __awaiter(this, void 0, void 0, function* () {
339
- const cognitoUser = yield this.auth.currentAuthenticatedUser();
340
- const refreshToken = cognitoUser.getSignInUserSession().getRefreshToken();
341
- return new Promise((resolve, reject) => {
342
- cognitoUser.refreshSession(refreshToken, (err, data) => {
343
- if (err) {
344
- console.error('Error refreshing token: ', err);
345
- reject(err);
346
- }
347
- else {
348
- console.log('Token refresh complete: ', data);
349
- resolve(0);
350
- }
351
- });
352
- });
353
- });
354
- }
355
- completeRequest(newPassword) {
356
- return __awaiter(this, void 0, void 0, function* () {
357
- const resetUser = yield this.getResetUser(true);
358
- if (resetUser.state !== TpClaimState.APPROVED) {
359
- throw new LrBadStateException('Password reset request has not been approved.');
360
- }
361
- // --------------------------------------------------------------
362
- // Prepare all materials to ensure there are no errors.
363
- // --------------------------------------------------------------
364
- const assemblyKey = yield this.recoverAssemblyKey(resetUser);
365
- const { rootKey } = yield this.encryptionService.decrypt(assemblyKey, resetUser.assemblyCipherData);
366
- console.log(rootKey);
367
- // Making sure it's a valid key.
368
- const rootKeyJwk = yield JWK.asKey(rootKey);
369
- const masterKey = yield this.keyGraphService.getKey(resetUser.masterKey.id);
370
- const masterKeyWrappedRootKey = yield this.encryptionService.encryptToString(masterKey.jwk, rootKeyJwk.toJSON(true));
371
- // The new password
372
- const newPassIdpResult = yield this.keyFactory.derivePassIdp(Object.assign({ password: newPassword }, resetUser.passKey.passIdpParams));
373
- const newIdpPassword = this.passwordService.getPassIdpString(newPassIdpResult.jwk);
374
- // --------------------------------------------------------------
375
- // Get assembly key challenge
376
- // --------------------------------------------------------------
377
- const challenge = (yield this.lrGraphQL.lrMutate(new LrMutation({
378
- mutation: CreateTpAssemblyKeyChallengeMutation,
379
- variables: {
380
- input: {},
381
- },
382
- }), {
383
- includeKeyGraph: false,
384
- })).createTpAssemblyKeyChallenge.challenge;
385
- console.log(challenge);
386
- // Sign the challenge
387
- // Generate a client side nonce that's no in the server's control.
388
- challenge.clientNonce = this.keyFactory.randomString(TP_PASSWORD_RESET_CLIENT_NONCE_LENGTH);
389
- console.log(challenge);
390
- const assemblyKeyVerifierPrk = yield this.encryptionService.decrypt(assemblyKey, resetUser.wrappedAssemblyKeyVerifierPrk);
391
- const signedChallenge = yield this.encryptionService.sign(assemblyKeyVerifierPrk, challenge);
392
- // --------------------------------------------------------------
393
- // Change password for the original user
394
- // --------------------------------------------------------------
395
- const tempIdpPassword = (yield this.lrGraphQL.lrMutate(new LrMutation({
396
- mutation: PreCompleteTpPasswordResetRequestMutation,
397
- variables: {
398
- input: {
399
- signedChallenge: JSON.stringify(signedChallenge),
400
- },
401
- },
402
- }), {
403
- includeKeyGraph: false,
404
- })).preCompleteTpPasswordResetRequest.idpPassword;
405
- // --------------------------------------------------------------
406
- // Login as the original user using new temporary password
407
- // --------------------------------------------------------------
408
- // At this point, the original account's password has been changed
409
- // to a temporary password. It is no longer possible for the user
410
- // to use the original password to login. Any successful login
411
- // can only be using the temporary password. So it's safe to assume
412
- // that we want to "complete" the password reset.
413
- // The maybe 2FA so we listen for the auth event from Amplify.
414
- const retPromise = new Promise((resolve) => {
415
- const listener = (data) => __awaiter(this, void 0, void 0, function* () {
416
- if (data.payload.event !== 'signIn') {
417
- return;
418
- }
419
- Hub.remove('auth', listener);
420
- console.log(data.payload);
421
- yield this.auth.signIn(resetUser.username, newIdpPassword);
422
- // Switch over to the new set of keys
423
- yield this.lrGraphQL.lrMutate(new LrMutation({
424
- mutation: CompleteTpPasswordResetRequestMutation,
425
- variables: {
426
- input: {
427
- masterKeyWrappedRootKey,
428
- masterKeyId: masterKey.id,
429
- },
430
- },
431
- }));
432
- resolve();
433
- });
434
- Hub.listen('auth', listener);
435
- });
436
- // Signin as the original user. Password has been reset to temporary one. It should return
437
- // with NEW_PASSWORD_REQUIRED
438
- let user = yield this.auth.signIn(resetUser.username, tempIdpPassword, {
439
- noProxy: 'true',
440
- });
441
- if (user.challengeName !== 'NEW_PASSWORD_REQUIRED') {
442
- throw new LrException({
443
- message: 'Internal error. Expecting Cognito to have done a password reset after call to PreCompleteTpPasswordResetRequestMutation.',
444
- });
445
- }
446
- // Set new password on Idp
447
- // the awsFetch() function passes NEW_PASSWORD_REQUIRED directly to AWS without
448
- // going through the proxy.
449
- user = yield this.auth.completeNewPassword(user, newIdpPassword, {});
450
- return retPromise;
451
- });
452
- }
453
- recoverAssemblyKey(resetUser) {
454
- return __awaiter(this, void 0, void 0, function* () {
455
- // Recover the assembly key.
456
- let assemblyKeyParams;
457
- const prk = yield this.keyGraphService.getKey(resetUser.pxk.id);
458
- const shares = yield Promise.all(resetUser.approvals
459
- .filter((approval) => !!approval.receiverCipherPartialAssemblyKey)
460
- .map((approval) => __awaiter(this, void 0, void 0, function* () {
461
- const partialAssemblyKey = yield this.encryptionService.decrypt(prk, approval.receiverCipherPartialAssemblyKey);
462
- if (assemblyKeyParams) {
463
- if (JSON.stringify(assemblyKeyParams) !==
464
- JSON.stringify(partialAssemblyKey.assemblyKeyParams)) {
465
- throw new LrBadStateException('The assembly key parameters are different between the approvals.');
466
- }
467
- }
468
- else {
469
- assemblyKeyParams = partialAssemblyKey.assemblyKeyParams;
470
- }
471
- return partialAssemblyKey.slip39.share.mnemonics;
472
- })));
473
- console.log('recoverAssemblyKey()', shares);
474
- const rawAssemblyKey = yield this.slip39Service.recoverSecret(shares, TP_PASSWORD_RESET_SLIP39_PASSPHRASE);
475
- return JWK.asKey(Object.assign(Object.assign({}, assemblyKeyParams), { k: rawAssemblyKey }));
476
- });
477
- }
478
- }
479
- LifeReadyAuthService.ɵprov = i0.ɵɵdefineInjectable({ factory: function LifeReadyAuthService_Factory() { return new LifeReadyAuthService(i0.ɵɵinject(i1.LR_CONFIG), i0.ɵɵinject(i2.AuthClass), i0.ɵɵinject(i3.KeyFactoryService), i0.ɵɵinject(i4.KeyService), i0.ɵɵinject(i5.ProfileService), i0.ɵɵinject(i6.KeyGraphService), i0.ɵɵinject(i7.PasswordService), i0.ɵɵinject(i8.IdleService), i0.ɵɵinject(i9.LrGraphQLService), i0.ɵɵinject(i10.TpPasswordResetProcessorService), i0.ɵɵinject(i11.PersistService), i0.ɵɵinject(i12.EncryptionService), i0.ɵɵinject(i13.Slip39Service)); }, token: LifeReadyAuthService, providedIn: "root" });
480
- LifeReadyAuthService.decorators = [
481
- { type: Injectable, args: [{
482
- providedIn: 'root',
483
- },] }
484
- ];
485
- LifeReadyAuthService.ctorParameters = () => [
486
- { type: undefined, decorators: [{ type: Inject, args: [LR_CONFIG,] }] },
487
- { type: AuthClass },
488
- { type: KeyFactoryService },
489
- { type: KeyService },
490
- { type: ProfileService },
491
- { type: KeyGraphService },
492
- { type: PasswordService },
493
- { type: IdleService },
494
- { type: LrGraphQLService },
495
- { type: TpPasswordResetProcessorService },
496
- { type: PersistService },
497
- { type: EncryptionService },
498
- { type: Slip39Service }
499
- ];
500
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"life-ready-auth.service.js","sourceRoot":"C:/Projects/newrepo/kc-client/projects/core/src/","sources":["lib/auth/life-ready-auth.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Inject, Injectable, isDevMode } from '@angular/core';\r\nimport { CognitoUser } from '@aws-amplify/auth';\r\nimport { Hub } from '@aws-amplify/core';\r\nimport { CognitoUserAttribute } from 'amazon-cognito-identity-js';\r\nimport { Observable, ReplaySubject } from 'rxjs';\r\nimport { PassIdpParams } from '../cryptography/cryptography.types';\r\nimport { KeyGraphService } from '../cryptography/key-graph.service';\r\nimport { KeyService } from '../cryptography/key.service';\r\nimport { ProfileService } from '../users/profile.service';\r\nimport { PasswordChangeStatus } from '../users/profile.types';\r\nimport {\r\n  LrConcurrentAccessException,\r\n  LrBadRequestException,\r\n  LrBadStateException,\r\n  LrException,\r\n} from '../_common/exceptions';\r\nimport {\r\n  CognitoChallengeUser,\r\n  CurrentUser,\r\n  TpPasswordResetUser,\r\n  LoginResult,\r\n  RecoveryStatus,\r\n} from './auth.types';\r\nimport { PasswordService } from './password.service';\r\nimport { AuthClass } from '@aws-amplify/auth/lib-esm/Auth';\r\nimport {\r\n  CompleteTpPasswordResetRequestMutation,\r\n  CreateTpAssemblyKeyChallengeMutation,\r\n  PreCompleteTpPasswordResetRequestMutation,\r\n  TpPasswordResetUserQuery,\r\n} from '../trusted-parties/tp-password-reset.gql';\r\nimport { IdleService } from './idle.service';\r\nimport { KeyFactoryService } from '../cryptography/key-factory.service';\r\nimport { LrGraphQLService, LrMutation } from '../api/lr-graphql';\r\nimport { TpClaimState, TpPasswordResetUserNode } from '../api/types';\r\nimport { TpPasswordResetProcessorService } from '../api/query-processor/tp-password-reset-processor.service';\r\nimport { SetSessionEncryptionKeyMutation } from './auth.gql';\r\nimport { PersistService } from '../api/persist.service';\r\nimport { JWK } from 'node-jose';\r\nimport { LifeReadyConfig, LR_CONFIG } from '../life-ready.config';\r\nimport { EncryptionService } from '../cryptography/encryption.service';\r\nimport { Slip39Service } from '../cryptography/slip39.service';\r\nimport {\r\n  TP_PASSWORD_RESET_CLIENT_NONCE_LENGTH,\r\n  TP_PASSWORD_RESET_SLIP39_PASSPHRASE,\r\n  TP_PASSWORD_RESET_USERNAME_SUFFIX,\r\n} from '../trusted-parties/tp-password-reset.constants';\r\n\r\nexport const initialiseAuth = (authService: LifeReadyAuthService) => {\r\n  return () => authService.initialise();\r\n};\r\n\r\nexport interface LoginOptions {\r\n  tpPasswordResetAutoComplete?: boolean;\r\n}\r\n\r\n@Injectable({\r\n  providedIn: 'root',\r\n})\r\nexport class LifeReadyAuthService {\r\n  private hubSubject: ReplaySubject<any> = new ReplaySubject<any>(1);\r\n  private currentUser: CurrentUser;\r\n  private currentResetUser: TpPasswordResetUser;\r\n\r\n  constructor(\r\n    @Inject(LR_CONFIG) private config: LifeReadyConfig,\r\n    private auth: AuthClass,\r\n    private keyFactory: KeyFactoryService,\r\n    private keyService: KeyService,\r\n    private profileService: ProfileService,\r\n    private keyGraphService: KeyGraphService,\r\n    private passwordService: PasswordService,\r\n    private idleService: IdleService,\r\n    private lrGraphQL: LrGraphQLService,\r\n    private tpPasswordResetProcessorService: TpPasswordResetProcessorService,\r\n    private persistService: PersistService,\r\n    private encryptionService: EncryptionService,\r\n    private slip39Service: Slip39Service\r\n  ) {}\r\n\r\n  public async initialise() {\r\n    Hub.listen('auth', (data) => this.hubSubject.next(data.payload));\r\n  }\r\n\r\n  private async loginIdpImpl(\r\n    emailOrPhone: string,\r\n    password: string,\r\n    passIdpParams: PassIdpParams,\r\n    recoveryStatus: RecoveryStatus\r\n  ): Promise<CognitoChallengeUser> {\r\n    const passIdpResult = await this.keyFactory.derivePassIdp({\r\n      password,\r\n      ...passIdpParams,\r\n    });\r\n    // Use the derived password to signin with cognito\r\n    const user = await this.auth.signIn(\r\n      emailOrPhone,\r\n      this.passwordService.getPassIdpString(passIdpResult.jwk)\r\n    );\r\n\r\n    user.recoveryStatus = recoveryStatus;\r\n\r\n    return user;\r\n  }\r\n\r\n  private async loginIdp(\r\n    emailOrPhone: string,\r\n    password: string\r\n  ): Promise<CognitoChallengeUser> {\r\n    // Download the salt needed to derive the PassIdp\r\n    const passIdpApiResult = await this.profileService.getPassIdpParams(\r\n      emailOrPhone\r\n    );\r\n\r\n    if (\r\n      passIdpApiResult.passwordChangeStatus === PasswordChangeStatus.InProgress\r\n    ) {\r\n      throw new LrConcurrentAccessException('A password change is in progress');\r\n    }\r\n\r\n    if (\r\n      passIdpApiResult.passwordChangeStatus === PasswordChangeStatus.Recovery\r\n    ) {\r\n      console.log('In recovery mode.');\r\n      // Let's say we don't know if the password is the new one or the old one. We just have to try both.\r\n      try {\r\n        const user = await this.loginIdpImpl(\r\n          emailOrPhone,\r\n          password,\r\n          passIdpApiResult.newPassIdpParams,\r\n          RecoveryStatus.NEW_PASSWORD\r\n        );\r\n        // New password worked. Let's set to the current password\r\n\r\n        // --Potential Failure Point 1--\r\n        // if changePasswordComplete() doesn't get called, then it should remain\r\n\r\n        console.log('New password works!');\r\n\r\n        return user;\r\n      } catch (error) {\r\n        // Just bubble up any other type of error.\r\n        if (error.code !== 'NotAuthorizedException') {\r\n          throw error;\r\n        }\r\n        // pass, try again assuming it's the old password\r\n      }\r\n\r\n      // Now assume it's the previous password. Any exception is allowed to bubble up.\r\n      try {\r\n        const user = await this.loginIdpImpl(\r\n          emailOrPhone,\r\n          password,\r\n          passIdpApiResult.currentPassIdpParams,\r\n          RecoveryStatus.OLD_PASSWORD\r\n        );\r\n        // Old password worked.\r\n        console.log('Old password works!');\r\n\r\n        return user;\r\n      } catch (error) {\r\n        // Just bubble up any other type of error.\r\n        throw error.code === 'NotAuthorizedException'\r\n          ? new LrBadRequestException(\r\n              'The password change request was interrupted, please try to login with both your new and old password'\r\n            )\r\n          : error;\r\n      }\r\n    }\r\n\r\n    // Try against as the TP password reset account\r\n    if (passIdpApiResult.tpPasswordReset) {\r\n      try {\r\n        // TP password reset is in process. We need to try the password against both\r\n        // original account and the new reset account.\r\n        const reset = passIdpApiResult.tpPasswordReset;\r\n        const ret = await this.loginIdpImpl(\r\n          reset.resetUsername,\r\n          password,\r\n          reset.passIdpParams,\r\n          RecoveryStatus.NONE\r\n        );\r\n        ret.isTpPasswordResetUser = true;\r\n\r\n        return ret;\r\n      } catch (err) {\r\n        // continue, try again as regular user.\r\n      }\r\n    }\r\n\r\n    // Login as regular user\r\n    return await this.loginIdpImpl(\r\n      emailOrPhone,\r\n      password,\r\n      passIdpApiResult.currentPassIdpParams,\r\n      RecoveryStatus.NONE\r\n    );\r\n  }\r\n\r\n  protected async handleSessionEncryptionKey() {\r\n    if (this.config.disableSessionEncryptionKey) {\r\n      if (!isDevMode()) {\r\n        const msg =\r\n          'You should not set disableSessionEncryptionKey=True in mode prod. It defaults to false.';\r\n        console.error(msg);\r\n        throw new Error(msg);\r\n      } else {\r\n        console.warn(\r\n          'You have set disableSessionEncryptionKey=True. Make sure not to do this in prod mode.'\r\n        );\r\n      }\r\n    } else {\r\n      // Set the session key to a new encryption key for this session\r\n      const sessionEncryptionKey = await this.keyFactory.createKey();\r\n      await this.lrGraphQL.lrMutate(\r\n        new LrMutation({\r\n          mutation: SetSessionEncryptionKeyMutation,\r\n          variables: {\r\n            input: {\r\n              sessionEncryptionKey: JSON.stringify(\r\n                sessionEncryptionKey.toJSON(true)\r\n              ),\r\n            },\r\n          },\r\n        }),\r\n        {\r\n          includeKeyGraph: false,\r\n        }\r\n      );\r\n\r\n      this.persistService.setServerSessionEncryptionKey(sessionEncryptionKey);\r\n    }\r\n  }\r\n\r\n  protected async handlePostAuth(cognitoUser: CognitoChallengeUser) {\r\n    await this.handlePasswordRecovery(cognitoUser);\r\n    await this.handleSessionEncryptionKey();\r\n  }\r\n\r\n  public async login(\r\n    emailOrPhone: string,\r\n    password: string,\r\n    { tpPasswordResetAutoComplete = true }: LoginOptions = {}\r\n  ) {\r\n    let loginResult = await this.loginImpl(emailOrPhone, password);\r\n\r\n    if (\r\n      tpPasswordResetAutoComplete &&\r\n      loginResult.resetUser?.state === TpClaimState.APPROVED\r\n    ) {\r\n      await this.completeRequest(password);\r\n      loginResult = await this.loginImpl(emailOrPhone, password);\r\n    }\r\n\r\n    return loginResult;\r\n  }\r\n\r\n  public async loginImpl(\r\n    emailOrPhone: string,\r\n    password: string\r\n  ): Promise<LoginResult> {\r\n    await this.logout();\r\n    const cognitoUser = await this.loginIdp(emailOrPhone, password);\r\n\r\n    // todo: Meet MFA challenges.\r\n    if (['SMS_MFA', 'SOFTWARE_TOKEN_MFA'].includes(cognitoUser.challengeName)) {\r\n      return { hasChallenge: true, challenge: cognitoUser };\r\n    }\r\n\r\n    await this.handlePostAuth(cognitoUser);\r\n\r\n    if (cognitoUser.isTpPasswordResetUser) {\r\n      // Assuming there is no MFA on the TP reset user.\r\n      const resetUser = await this.loadResetUser(password);\r\n      return { hasChallenge: false, resetUser };\r\n    } else {\r\n      const user = await this.loadUser(cognitoUser, password);\r\n      await this.idleService.start(); // Run idleService whenever user is logged in.\r\n      return { hasChallenge: false, user };\r\n    }\r\n  }\r\n\r\n  // TODO <AZ> We need to handle the isTpPasswordResetUser=True case here after MFA as well.\r\n  public async verifyLogin(\r\n    challenge: CognitoChallengeUser,\r\n    password: string,\r\n    rememberMe: boolean,\r\n    code: string\r\n  ): Promise<CurrentUser> {\r\n    await this.auth.confirmSignIn(challenge, code, challenge.challengeName);\r\n\r\n    // TODO: this.auth.confirmSignIn() could return another challenge.\r\n\r\n    const cognitoUser: CognitoUser = await this.auth.currentAuthenticatedUser();\r\n\r\n    await this.handlePostAuth(challenge);\r\n\r\n    const user = await this.loadUser(cognitoUser, password);\r\n\r\n    if (rememberMe) {\r\n      cognitoUser.setDeviceStatusRemembered({\r\n        onSuccess: () => {},\r\n        onFailure: (e) => console.error(e),\r\n      });\r\n    }\r\n\r\n    return user;\r\n  }\r\n\r\n  async handlePasswordRecovery(user: CognitoChallengeUser) {\r\n    if (user.recoveryStatus !== RecoveryStatus.NONE) {\r\n      const jwtToken = user\r\n        .getSignInUserSession()\r\n        .getAccessToken()\r\n        .getJwtToken();\r\n      await this.passwordService.changePasswordComplete(\r\n        jwtToken,\r\n        user.recoveryStatus === RecoveryStatus.NEW_PASSWORD\r\n      );\r\n    }\r\n  }\r\n\r\n  async getUserOrResetUser(\r\n    reload: boolean = false\r\n  ): Promise<CurrentUser | TpPasswordResetUser> {\r\n    const cognitoUser = await this.auth.currentAuthenticatedUser();\r\n\r\n    if (cognitoUser.getUsername().endsWith(TP_PASSWORD_RESET_USERNAME_SUFFIX)) {\r\n      return this.getResetUser(reload);\r\n    } else {\r\n      return this.getUser(reload);\r\n    }\r\n  }\r\n\r\n  async getResetUser(reload: boolean = false): Promise<TpPasswordResetUser> {\r\n    if (!reload && this.currentResetUser) {\r\n      return this.currentResetUser;\r\n    }\r\n\r\n    this.currentResetUser = await this.loadResetUser();\r\n\r\n    await this.idleService.start(); // Run idleService whenever user is logged in.\r\n\r\n    return this.currentResetUser;\r\n  }\r\n\r\n  async getUser(reload: boolean = false): Promise<CurrentUser> {\r\n    if (!reload && this.currentUser) {\r\n      return this.currentUser;\r\n    }\r\n    this.currentUser = await this.loadUser(\r\n      await this.auth.currentAuthenticatedUser()\r\n    );\r\n    console.log('Starting idle service.');\r\n    await this.idleService.start(); // Run idleService whenever user is logged in.\r\n    return this.currentUser;\r\n  }\r\n\r\n  private mapTPVaultAccess(features?: any): boolean {\r\n    const tpVaultFeature = features?.tpVault;\r\n    return (\r\n      tpVaultFeature?.length > 0 &&\r\n      tpVaultFeature.some((feature) => feature.toUpperCase() === 'ACCESS')\r\n    );\r\n  }\r\n\r\n  private async loadUser(\r\n    cognitoUser: CognitoUser,\r\n    password?: string\r\n  ): Promise<CurrentUser> {\r\n    const {\r\n      currentUser,\r\n      contactCard,\r\n      userPlans,\r\n    } = await this.profileService.getCurrentUser();\r\n\r\n    if (currentUser.sessionEncryptionKey) {\r\n      this.persistService.setServerSessionEncryptionKey(\r\n        await JWK.asKey(currentUser.sessionEncryptionKey)\r\n      );\r\n    }\r\n\r\n    const userAttributes = await this.auth.userAttributes(cognitoUser);\r\n\r\n    if (password) {\r\n      const passKey = (\r\n        await this.keyFactory.derivePassKey({\r\n          password,\r\n          ...currentUser.currentUserKey.passKey.passKeyParams,\r\n        })\r\n      ).jwk;\r\n\r\n      await this.idleService.persistMasterKey(\r\n        await this.keyGraphService.unwrapWithPassKey(\r\n          currentUser.currentUserKey.passKey.id,\r\n          passKey,\r\n          currentUser.currentUserKey.masterKey.id\r\n        )\r\n      );\r\n    }\r\n    await this.keyGraphService.populateKeys(currentUser.currentUserKey);\r\n\r\n    return {\r\n      id: currentUser.id,\r\n      sub: this.getUserAttribute('sub', userAttributes),\r\n      username: currentUser.username,\r\n      currentUserKey: currentUser.currentUserKey,\r\n      getAccessJwtToken: () =>\r\n        cognitoUser.getSignInUserSession().getAccessToken().getJwtToken(),\r\n      email: this.getUserAttribute('email', userAttributes),\r\n      emailVerified:\r\n        this.getUserAttribute('email_verified', userAttributes) === 'true',\r\n      phone: this.getUserAttribute('phone_number', userAttributes),\r\n      phoneVerified:\r\n        this.getUserAttribute('phone_number_verified', userAttributes) ===\r\n        'true',\r\n      contactCard: {\r\n        ...(await this.profileService.decryptContactCard(contactCard)),\r\n      },\r\n      userDelete: currentUser.userDelete,\r\n      userPlans,\r\n      hasTPVaultAccess: this.mapTPVaultAccess(currentUser.features),\r\n      features: currentUser.features,\r\n      sessionEncryptionKey: currentUser.sessionEncryptionKey,\r\n    };\r\n  }\r\n\r\n  public watchAuth(): Observable<any> {\r\n    return this.hubSubject;\r\n  }\r\n\r\n  public async logout(): Promise<void> {\r\n    this.currentUser = null;\r\n    this.keyService.purgeKeys();\r\n    this.keyGraphService.purgeKeys();\r\n\r\n    await Promise.all([this.auth.signOut(), this.profileService.signOut()]);\r\n  }\r\n\r\n  private getUserAttribute(\r\n    attributeName: string,\r\n    userAttributes: CognitoUserAttribute[]\r\n  ) {\r\n    const userAttribute = userAttributes.find(\r\n      (x) => x.getName() === attributeName\r\n    );\r\n\r\n    return userAttribute ? userAttribute.getValue() : null;\r\n  }\r\n\r\n  public async loadResetUser(password?: string): Promise<TpPasswordResetUser> {\r\n    const { tpPasswordResetUser: resetUser } = await this.lrGraphQL.query({\r\n      query: TpPasswordResetUserQuery,\r\n    });\r\n\r\n    if (resetUser.sessionEncryptionKey) {\r\n      this.persistService.setServerSessionEncryptionKey(\r\n        await JWK.asKey(resetUser.sessionEncryptionKey)\r\n      );\r\n    }\r\n\r\n    // Update the keys\r\n    if (password) {\r\n      const passKey = (\r\n        await this.keyFactory.derivePassKey({\r\n          password,\r\n          ...resetUser.passKey.passKeyParams,\r\n        })\r\n      ).jwk;\r\n\r\n      await this.idleService.persistMasterKey(\r\n        await this.keyGraphService.unwrapWithPassKey(\r\n          resetUser.passKey.id,\r\n          passKey,\r\n          resetUser.masterKey.id\r\n        )\r\n      );\r\n    }\r\n\r\n    this.keyService.populateKeys({\r\n      passKey: {\r\n        id: resetUser.passKey.id,\r\n      },\r\n      masterKey: {\r\n        id: resetUser.masterKey.id,\r\n      },\r\n    });\r\n\r\n    const userAttributes = await this.auth.userAttributes(\r\n      await this.auth.currentAuthenticatedUser()\r\n    );\r\n    const sub = this.getUserAttribute('sub', userAttributes);\r\n\r\n    return {\r\n      ...(await this.tpPasswordResetProcessorService.processTpPasswordResetUserNode(\r\n        resetUser\r\n      )),\r\n      sub,\r\n    };\r\n  }\r\n\r\n  public async refreshAccessToken() {\r\n    const cognitoUser: CognitoUser = await this.auth.currentAuthenticatedUser();\r\n    const refreshToken = cognitoUser.getSignInUserSession().getRefreshToken();\r\n\r\n    return new Promise((resolve, reject) => {\r\n      cognitoUser.refreshSession(refreshToken, (err, data) => {\r\n        if (err) {\r\n          console.error('Error refreshing token: ', err);\r\n          reject(err);\r\n        } else {\r\n          console.log('Token refresh complete: ', data);\r\n          resolve(0);\r\n        }\r\n      });\r\n    });\r\n  }\r\n\r\n  async completeRequest(newPassword: string): Promise<void> {\r\n    const resetUser = await this.getResetUser(true);\r\n\r\n    if (resetUser.state !== TpClaimState.APPROVED) {\r\n      throw new LrBadStateException(\r\n        'Password reset request has not been approved.'\r\n      );\r\n    }\r\n\r\n    // --------------------------------------------------------------\r\n    // Prepare all materials to ensure there are no errors.\r\n    // --------------------------------------------------------------\r\n    const assemblyKey = await this.recoverAssemblyKey(resetUser);\r\n\r\n    const { rootKey } = await this.encryptionService.decrypt(\r\n      assemblyKey,\r\n      resetUser.assemblyCipherData\r\n    );\r\n    console.log(rootKey);\r\n\r\n    // Making sure it's a valid key.\r\n    const rootKeyJwk = await JWK.asKey(rootKey);\r\n\r\n    const masterKey = await this.keyGraphService.getKey(resetUser.masterKey.id);\r\n\r\n    const masterKeyWrappedRootKey = await this.encryptionService.encryptToString(\r\n      masterKey.jwk,\r\n      rootKeyJwk.toJSON(true)\r\n    );\r\n\r\n    // The new password\r\n    const newPassIdpResult = await this.keyFactory.derivePassIdp({\r\n      password: newPassword,\r\n      ...resetUser.passKey.passIdpParams,\r\n    });\r\n\r\n    const newIdpPassword = this.passwordService.getPassIdpString(\r\n      newPassIdpResult.jwk\r\n    );\r\n\r\n    // --------------------------------------------------------------\r\n    // Get assembly key challenge\r\n    // --------------------------------------------------------------\r\n    const challenge = (\r\n      await this.lrGraphQL.lrMutate(\r\n        new LrMutation({\r\n          mutation: CreateTpAssemblyKeyChallengeMutation,\r\n          variables: {\r\n            input: {},\r\n          },\r\n        }),\r\n        {\r\n          includeKeyGraph: false,\r\n        }\r\n      )\r\n    ).createTpAssemblyKeyChallenge.challenge;\r\n\r\n    console.log(challenge);\r\n\r\n    // Sign the challenge\r\n    // Generate a client side nonce that's no in the server's control.\r\n    challenge.clientNonce = this.keyFactory.randomString(\r\n      TP_PASSWORD_RESET_CLIENT_NONCE_LENGTH\r\n    );\r\n    console.log(challenge);\r\n\r\n    const assemblyKeyVerifierPrk = await this.encryptionService.decrypt(\r\n      assemblyKey,\r\n      resetUser.wrappedAssemblyKeyVerifierPrk\r\n    );\r\n    const signedChallenge = await this.encryptionService.sign(\r\n      assemblyKeyVerifierPrk,\r\n      challenge\r\n    );\r\n\r\n    // --------------------------------------------------------------\r\n    // Change password for the original user\r\n    // --------------------------------------------------------------\r\n    const tempIdpPassword = (\r\n      await this.lrGraphQL.lrMutate(\r\n        new LrMutation({\r\n          mutation: PreCompleteTpPasswordResetRequestMutation,\r\n          variables: {\r\n            input: {\r\n              signedChallenge: JSON.stringify(signedChallenge),\r\n            },\r\n          },\r\n        }),\r\n        {\r\n          includeKeyGraph: false,\r\n        }\r\n      )\r\n    ).preCompleteTpPasswordResetRequest.idpPassword;\r\n\r\n    // --------------------------------------------------------------\r\n    // Login as the original user using new temporary password\r\n    // --------------------------------------------------------------\r\n    // At this point, the original account's password has been changed\r\n    // to a temporary password. It is no longer possible for the user\r\n    // to use the original password to login. Any successful login\r\n    // can only be using the temporary password. So it's safe to assume\r\n    // that we want to \"complete\" the password reset.\r\n\r\n    // The maybe 2FA so we listen for the auth event from Amplify.\r\n    const retPromise = new Promise<void>((resolve) => {\r\n      const listener = async (data) => {\r\n        if (data.payload.event !== 'signIn') {\r\n          return;\r\n        }\r\n\r\n        Hub.remove('auth', listener);\r\n\r\n        console.log(data.payload);\r\n\r\n        await this.auth.signIn(resetUser.username, newIdpPassword);\r\n\r\n        // Switch over to the new set of keys\r\n        await this.lrGraphQL.lrMutate(\r\n          new LrMutation({\r\n            mutation: CompleteTpPasswordResetRequestMutation,\r\n            variables: {\r\n              input: {\r\n                masterKeyWrappedRootKey,\r\n                masterKeyId: masterKey.id,\r\n              },\r\n            },\r\n          })\r\n        );\r\n\r\n        resolve();\r\n      };\r\n\r\n      Hub.listen('auth', listener);\r\n    });\r\n\r\n    // Signin as the original user. Password has been reset to temporary one. It should return\r\n    // with NEW_PASSWORD_REQUIRED\r\n    let user = await this.auth.signIn(resetUser.username, tempIdpPassword, {\r\n      noProxy: 'true',\r\n    });\r\n\r\n    if (user.challengeName !== 'NEW_PASSWORD_REQUIRED') {\r\n      throw new LrException({\r\n        message:\r\n          'Internal error. Expecting Cognito to have done a password reset after call to PreCompleteTpPasswordResetRequestMutation.',\r\n      });\r\n    }\r\n\r\n    // Set new password on Idp\r\n    // the awsFetch() function passes NEW_PASSWORD_REQUIRED directly to AWS without\r\n    // going through the proxy.\r\n    user = await this.auth.completeNewPassword(user, newIdpPassword, {});\r\n\r\n    return retPromise;\r\n  }\r\n\r\n  private async recoverAssemblyKey(\r\n    resetUser: TpPasswordResetUserNode\r\n  ): Promise<JWK.Key> {\r\n    // Recover the assembly key.\r\n    let assemblyKeyParams: object;\r\n\r\n    const prk = await this.keyGraphService.getKey(resetUser.pxk.id);\r\n\r\n    const shares = await Promise.all(\r\n      resetUser.approvals\r\n        .filter((approval) => !!approval.receiverCipherPartialAssemblyKey)\r\n        .map(async (approval) => {\r\n          const partialAssemblyKey = await this.encryptionService.decrypt(\r\n            prk,\r\n            approval.receiverCipherPartialAssemblyKey\r\n          );\r\n\r\n          if (assemblyKeyParams) {\r\n            if (\r\n              JSON.stringify(assemblyKeyParams) !==\r\n              JSON.stringify(partialAssemblyKey.assemblyKeyParams)\r\n            ) {\r\n              throw new LrBadStateException(\r\n                'The assembly key parameters are different between the approvals.'\r\n              );\r\n            }\r\n          } else {\r\n            assemblyKeyParams = partialAssemblyKey.assemblyKeyParams;\r\n          }\r\n          return partialAssemblyKey.slip39.share.mnemonics;\r\n        })\r\n    );\r\n\r\n    console.log('recoverAssemblyKey()', shares);\r\n\r\n    const rawAssemblyKey = await this.slip39Service.recoverSecret(\r\n      shares,\r\n      TP_PASSWORD_RESET_SLIP39_PASSPHRASE\r\n    );\r\n\r\n    return JWK.asKey({\r\n      ...assemblyKeyParams,\r\n      k: rawAssemblyKey,\r\n    });\r\n  }\r\n}\r\n"]}
1
+ import { __awaiter } from "tslib";
2
+ import { Inject, Injectable, isDevMode } from '@angular/core';
3
+ import { Hub } from '@aws-amplify/core';
4
+ import { ReplaySubject } from 'rxjs';
5
+ import { KeyGraphService } from '../cryptography/key-graph.service';
6
+ import { KeyService } from '../cryptography/key.service';
7
+ import { ProfileService } from '../users/profile.service';
8
+ import { PasswordChangeStatus } from '../users/profile.types';
9
+ import { LrConcurrentAccessException, LrBadRequestException, LrBadStateException, LrException, } from '../_common/exceptions';
10
+ import { RecoveryStatus, } from './auth.types';
11
+ import { PasswordService } from './password.service';
12
+ import { AuthClass } from '@aws-amplify/auth/lib-esm/Auth';
13
+ import { CompleteTpPasswordResetRequestMutation, CreateTpAssemblyKeyChallengeMutation, PreCompleteTpPasswordResetRequestMutation, TpPasswordResetUserQuery, } from '../trusted-parties/tp-password-reset.gql';
14
+ import { IdleService } from './idle.service';
15
+ import { KeyFactoryService } from '../cryptography/key-factory.service';
16
+ import { LrGraphQLService, LrMutation } from '../api/lr-graphql';
17
+ import { TpClaimState } from '../api/types';
18
+ import { TpPasswordResetProcessorService } from '../api/query-processor/tp-password-reset-processor.service';
19
+ import { SetSessionEncryptionKeyMutation } from './auth.gql';
20
+ import { PersistService } from '../api/persist.service';
21
+ import { JWK } from 'node-jose';
22
+ import { LR_CONFIG } from '../life-ready.config';
23
+ import { EncryptionService } from '../cryptography/encryption.service';
24
+ import { Slip39Service } from '../cryptography/slip39.service';
25
+ import { TP_PASSWORD_RESET_CLIENT_NONCE_LENGTH, TP_PASSWORD_RESET_SLIP39_PASSPHRASE, TP_PASSWORD_RESET_USERNAME_SUFFIX, } from '../trusted-parties/tp-password-reset.constants';
26
+ import * as i0 from "@angular/core";
27
+ import * as i1 from "../life-ready.config";
28
+ import * as i2 from "@aws-amplify/auth/lib-esm/Auth";
29
+ import * as i3 from "../cryptography/key-factory.service";
30
+ import * as i4 from "../cryptography/key.service";
31
+ import * as i5 from "../users/profile.service";
32
+ import * as i6 from "../cryptography/key-graph.service";
33
+ import * as i7 from "./password.service";
34
+ import * as i8 from "./idle.service";
35
+ import * as i9 from "../api/lr-graphql/lr-graphql.service";
36
+ import * as i10 from "../api/query-processor/tp-password-reset-processor.service";
37
+ import * as i11 from "../api/persist.service";
38
+ import * as i12 from "../cryptography/encryption.service";
39
+ import * as i13 from "../cryptography/slip39.service";
40
+ export const initialiseAuth = (authService) => {
41
+ return () => authService.initialise();
42
+ };
43
+ export class LifeReadyAuthService {
44
+ constructor(config, auth, keyFactory, keyService, profileService, keyGraphService, passwordService, idleService, lrGraphQL, tpPasswordResetProcessorService, persistService, encryptionService, slip39Service) {
45
+ this.config = config;
46
+ this.auth = auth;
47
+ this.keyFactory = keyFactory;
48
+ this.keyService = keyService;
49
+ this.profileService = profileService;
50
+ this.keyGraphService = keyGraphService;
51
+ this.passwordService = passwordService;
52
+ this.idleService = idleService;
53
+ this.lrGraphQL = lrGraphQL;
54
+ this.tpPasswordResetProcessorService = tpPasswordResetProcessorService;
55
+ this.persistService = persistService;
56
+ this.encryptionService = encryptionService;
57
+ this.slip39Service = slip39Service;
58
+ this.hubSubject = new ReplaySubject(1);
59
+ }
60
+ initialise() {
61
+ return __awaiter(this, void 0, void 0, function* () {
62
+ Hub.listen('auth', (data) => this.hubSubject.next(data.payload));
63
+ });
64
+ }
65
+ loginIdpImpl(emailOrPhone, password, passIdpParams, recoveryStatus) {
66
+ return __awaiter(this, void 0, void 0, function* () {
67
+ const passIdpResult = yield this.keyFactory.derivePassIdp(Object.assign({ password }, passIdpParams));
68
+ // Use the derived password to signin with cognito
69
+ const user = yield this.auth.signIn(emailOrPhone, this.passwordService.getPassIdpString(passIdpResult.jwk));
70
+ user.recoveryStatus = recoveryStatus;
71
+ return user;
72
+ });
73
+ }
74
+ loginIdp(emailOrPhone, password) {
75
+ return __awaiter(this, void 0, void 0, function* () {
76
+ // Download the salt needed to derive the PassIdp
77
+ const passIdpApiResult = yield this.profileService.getPassIdpParams(emailOrPhone);
78
+ if (passIdpApiResult.passwordChangeStatus === PasswordChangeStatus.InProgress) {
79
+ throw new LrConcurrentAccessException('A password change is in progress');
80
+ }
81
+ if (passIdpApiResult.passwordChangeStatus === PasswordChangeStatus.Recovery) {
82
+ console.log('In recovery mode.');
83
+ // Let's say we don't know if the password is the new one or the old one. We just have to try both.
84
+ try {
85
+ const user = yield this.loginIdpImpl(emailOrPhone, password, passIdpApiResult.newPassIdpParams, RecoveryStatus.NEW_PASSWORD);
86
+ // New password worked. Let's set to the current password
87
+ // --Potential Failure Point 1--
88
+ // if changePasswordComplete() doesn't get called, then it should remain
89
+ console.log('New password works!');
90
+ return user;
91
+ }
92
+ catch (error) {
93
+ // Just bubble up any other type of error.
94
+ if (error.code !== 'NotAuthorizedException') {
95
+ throw error;
96
+ }
97
+ // pass, try again assuming it's the old password
98
+ }
99
+ // Now assume it's the previous password. Any exception is allowed to bubble up.
100
+ try {
101
+ const user = yield this.loginIdpImpl(emailOrPhone, password, passIdpApiResult.currentPassIdpParams, RecoveryStatus.OLD_PASSWORD);
102
+ // Old password worked.
103
+ console.log('Old password works!');
104
+ return user;
105
+ }
106
+ catch (error) {
107
+ // Just bubble up any other type of error.
108
+ throw error.code === 'NotAuthorizedException'
109
+ ? new LrBadRequestException('The password change request was interrupted, please try to login with both your new and old password')
110
+ : error;
111
+ }
112
+ }
113
+ // Try against as the TP password reset account
114
+ if (passIdpApiResult.tpPasswordReset) {
115
+ try {
116
+ // TP password reset is in process. We need to try the password against both
117
+ // original account and the new reset account.
118
+ const reset = passIdpApiResult.tpPasswordReset;
119
+ const ret = yield this.loginIdpImpl(reset.resetUsername, password, reset.passIdpParams, RecoveryStatus.NONE);
120
+ ret.isTpPasswordResetUser = true;
121
+ return ret;
122
+ }
123
+ catch (err) {
124
+ // continue, try again as regular user.
125
+ }
126
+ }
127
+ // Login as regular user
128
+ return yield this.loginIdpImpl(emailOrPhone, password, passIdpApiResult.currentPassIdpParams, RecoveryStatus.NONE);
129
+ });
130
+ }
131
+ handleSessionEncryptionKey() {
132
+ return __awaiter(this, void 0, void 0, function* () {
133
+ if (this.config.disableSessionEncryptionKey) {
134
+ if (!isDevMode()) {
135
+ const msg = 'You should not set disableSessionEncryptionKey=True in mode prod. It defaults to false.';
136
+ console.error(msg);
137
+ throw new Error(msg);
138
+ }
139
+ else {
140
+ console.warn('You have set disableSessionEncryptionKey=True. Make sure not to do this in prod mode.');
141
+ }
142
+ }
143
+ else {
144
+ // Set the session key to a new encryption key for this session
145
+ const sessionEncryptionKey = yield this.keyFactory.createKey();
146
+ yield this.lrGraphQL.lrMutate(new LrMutation({
147
+ mutation: SetSessionEncryptionKeyMutation,
148
+ variables: {
149
+ input: {
150
+ sessionEncryptionKey: JSON.stringify(sessionEncryptionKey.toJSON(true)),
151
+ },
152
+ },
153
+ }), {
154
+ includeKeyGraph: false,
155
+ });
156
+ this.persistService.setServerSessionEncryptionKey(sessionEncryptionKey);
157
+ }
158
+ });
159
+ }
160
+ handlePostAuth(cognitoUser) {
161
+ return __awaiter(this, void 0, void 0, function* () {
162
+ yield this.handlePasswordRecovery(cognitoUser);
163
+ yield this.handleSessionEncryptionKey();
164
+ });
165
+ }
166
+ login(emailOrPhone, password, { tpPasswordResetAutoComplete = true } = {}) {
167
+ var _a;
168
+ return __awaiter(this, void 0, void 0, function* () {
169
+ let loginResult = yield this.loginImpl(emailOrPhone, password);
170
+ if (tpPasswordResetAutoComplete &&
171
+ ((_a = loginResult.resetUser) === null || _a === void 0 ? void 0 : _a.state) === TpClaimState.APPROVED) {
172
+ yield this.completeRequest(password);
173
+ loginResult = yield this.loginImpl(emailOrPhone, password);
174
+ }
175
+ return loginResult;
176
+ });
177
+ }
178
+ loginImpl(emailOrPhone, password) {
179
+ return __awaiter(this, void 0, void 0, function* () {
180
+ yield this.logout();
181
+ const cognitoUser = yield this.loginIdp(emailOrPhone, password);
182
+ // todo: Meet MFA challenges.
183
+ if (['SMS_MFA', 'SOFTWARE_TOKEN_MFA'].includes(cognitoUser.challengeName)) {
184
+ return { hasChallenge: true, challenge: cognitoUser };
185
+ }
186
+ yield this.handlePostAuth(cognitoUser);
187
+ if (cognitoUser.isTpPasswordResetUser) {
188
+ // Assuming there is no MFA on the TP reset user.
189
+ const resetUser = yield this.loadResetUser(password);
190
+ return { hasChallenge: false, resetUser };
191
+ }
192
+ else {
193
+ const user = yield this.loadUser(cognitoUser, password);
194
+ yield this.idleService.start(); // Run idleService whenever user is logged in.
195
+ return { hasChallenge: false, user };
196
+ }
197
+ });
198
+ }
199
+ // TODO <AZ> We need to handle the isTpPasswordResetUser=True case here after MFA as well.
200
+ verifyLogin(challenge, password, rememberMe, code) {
201
+ return __awaiter(this, void 0, void 0, function* () {
202
+ yield this.auth.confirmSignIn(challenge, code, challenge.challengeName);
203
+ // TODO: this.auth.confirmSignIn() could return another challenge.
204
+ const cognitoUser = yield this.auth.currentAuthenticatedUser();
205
+ yield this.handlePostAuth(challenge);
206
+ const user = yield this.loadUser(cognitoUser, password);
207
+ if (rememberMe) {
208
+ cognitoUser.setDeviceStatusRemembered({
209
+ onSuccess: () => { },
210
+ onFailure: (e) => console.error(e),
211
+ });
212
+ }
213
+ return user;
214
+ });
215
+ }
216
+ handlePasswordRecovery(user) {
217
+ return __awaiter(this, void 0, void 0, function* () {
218
+ if (user.recoveryStatus !== RecoveryStatus.NONE) {
219
+ const jwtToken = user
220
+ .getSignInUserSession()
221
+ .getAccessToken()
222
+ .getJwtToken();
223
+ yield this.passwordService.changePasswordComplete(jwtToken, user.recoveryStatus === RecoveryStatus.NEW_PASSWORD);
224
+ }
225
+ });
226
+ }
227
+ getUserOrResetUser(reload = false) {
228
+ return __awaiter(this, void 0, void 0, function* () {
229
+ const cognitoUser = yield this.auth.currentAuthenticatedUser();
230
+ if (cognitoUser.getUsername().endsWith(TP_PASSWORD_RESET_USERNAME_SUFFIX)) {
231
+ return this.getResetUser(reload);
232
+ }
233
+ else {
234
+ return this.getUser(reload);
235
+ }
236
+ });
237
+ }
238
+ getResetUser(reload = false) {
239
+ return __awaiter(this, void 0, void 0, function* () {
240
+ if (!reload && this.currentResetUser) {
241
+ return this.currentResetUser;
242
+ }
243
+ this.currentResetUser = yield this.loadResetUser();
244
+ yield this.idleService.start(); // Run idleService whenever user is logged in.
245
+ return this.currentResetUser;
246
+ });
247
+ }
248
+ getUser(reload = false) {
249
+ return __awaiter(this, void 0, void 0, function* () {
250
+ if (!reload && this.currentUser) {
251
+ return this.currentUser;
252
+ }
253
+ this.currentUser = yield this.loadUser(yield this.auth.currentAuthenticatedUser());
254
+ console.log('Starting idle service.');
255
+ yield this.idleService.start(); // Run idleService whenever user is logged in.
256
+ return this.currentUser;
257
+ });
258
+ }
259
+ mapTPVaultAccess(features) {
260
+ const tpVaultFeature = features === null || features === void 0 ? void 0 : features.tpVault;
261
+ return ((tpVaultFeature === null || tpVaultFeature === void 0 ? void 0 : tpVaultFeature.length) > 0 &&
262
+ tpVaultFeature.some((feature) => feature.toUpperCase() === 'ACCESS'));
263
+ }
264
+ loadUser(cognitoUser, password) {
265
+ return __awaiter(this, void 0, void 0, function* () {
266
+ const { currentUser, contactCard, userPlans, } = yield this.profileService.getCurrentUser();
267
+ if (currentUser.sessionEncryptionKey) {
268
+ this.persistService.setServerSessionEncryptionKey(yield JWK.asKey(currentUser.sessionEncryptionKey));
269
+ }
270
+ const userAttributes = yield this.auth.userAttributes(cognitoUser);
271
+ if (password) {
272
+ const passKey = (yield this.keyFactory.derivePassKey(Object.assign({ password }, currentUser.currentUserKey.passKey.passKeyParams))).jwk;
273
+ yield this.idleService.persistMasterKey(yield this.keyGraphService.unwrapWithPassKey(currentUser.currentUserKey.passKey.id, passKey, currentUser.currentUserKey.masterKey.id));
274
+ }
275
+ yield this.keyGraphService.populateKeys(currentUser.currentUserKey);
276
+ return {
277
+ id: currentUser.id,
278
+ sub: this.getUserAttribute('sub', userAttributes),
279
+ username: currentUser.username,
280
+ currentUserKey: currentUser.currentUserKey,
281
+ getAccessJwtToken: () => cognitoUser.getSignInUserSession().getAccessToken().getJwtToken(),
282
+ email: this.getUserAttribute('email', userAttributes),
283
+ emailVerified: this.getUserAttribute('email_verified', userAttributes) === 'true',
284
+ phone: this.getUserAttribute('phone_number', userAttributes),
285
+ phoneVerified: this.getUserAttribute('phone_number_verified', userAttributes) ===
286
+ 'true',
287
+ contactCard: Object.assign({}, (yield this.profileService.decryptContactCard(contactCard))),
288
+ userDelete: currentUser.userDelete,
289
+ userPlans,
290
+ hasTPVaultAccess: this.mapTPVaultAccess(currentUser.features),
291
+ features: currentUser.features,
292
+ sessionEncryptionKey: currentUser.sessionEncryptionKey,
293
+ };
294
+ });
295
+ }
296
+ watchAuth() {
297
+ return this.hubSubject;
298
+ }
299
+ logout() {
300
+ return __awaiter(this, void 0, void 0, function* () {
301
+ this.currentUser = null;
302
+ this.keyService.purgeKeys();
303
+ this.keyGraphService.purgeKeys();
304
+ yield Promise.all([this.auth.signOut(), this.profileService.signOut()]);
305
+ });
306
+ }
307
+ getUserAttribute(attributeName, userAttributes) {
308
+ const userAttribute = userAttributes.find((x) => x.getName() === attributeName);
309
+ return userAttribute ? userAttribute.getValue() : null;
310
+ }
311
+ loadResetUser(password) {
312
+ return __awaiter(this, void 0, void 0, function* () {
313
+ const { tpPasswordResetUser: resetUser } = yield this.lrGraphQL.query({
314
+ query: TpPasswordResetUserQuery,
315
+ });
316
+ if (resetUser.sessionEncryptionKey) {
317
+ this.persistService.setServerSessionEncryptionKey(yield JWK.asKey(resetUser.sessionEncryptionKey));
318
+ }
319
+ // Update the keys
320
+ if (password) {
321
+ const passKey = (yield this.keyFactory.derivePassKey(Object.assign({ password }, resetUser.passKey.passKeyParams))).jwk;
322
+ yield this.idleService.persistMasterKey(yield this.keyGraphService.unwrapWithPassKey(resetUser.passKey.id, passKey, resetUser.masterKey.id));
323
+ }
324
+ this.keyService.populateKeys({
325
+ passKey: {
326
+ id: resetUser.passKey.id,
327
+ },
328
+ masterKey: {
329
+ id: resetUser.masterKey.id,
330
+ },
331
+ });
332
+ const userAttributes = yield this.auth.userAttributes(yield this.auth.currentAuthenticatedUser());
333
+ const sub = this.getUserAttribute('sub', userAttributes);
334
+ return Object.assign(Object.assign({}, (yield this.tpPasswordResetProcessorService.processTpPasswordResetUserNode(resetUser))), { sub });
335
+ });
336
+ }
337
+ refreshAccessToken() {
338
+ return __awaiter(this, void 0, void 0, function* () {
339
+ const cognitoUser = yield this.auth.currentAuthenticatedUser();
340
+ const refreshToken = cognitoUser.getSignInUserSession().getRefreshToken();
341
+ return new Promise((resolve, reject) => {
342
+ cognitoUser.refreshSession(refreshToken, (err, data) => {
343
+ if (err) {
344
+ console.error('Error refreshing token: ', err);
345
+ reject(err);
346
+ }
347
+ else {
348
+ console.log('Token refresh complete: ', data);
349
+ resolve(0);
350
+ }
351
+ });
352
+ });
353
+ });
354
+ }
355
+ completeRequest(newPassword) {
356
+ return __awaiter(this, void 0, void 0, function* () {
357
+ const resetUser = yield this.getResetUser(true);
358
+ if (resetUser.state !== TpClaimState.APPROVED) {
359
+ throw new LrBadStateException('Password reset request has not been approved.');
360
+ }
361
+ // --------------------------------------------------------------
362
+ // Prepare all materials to ensure there are no errors.
363
+ // --------------------------------------------------------------
364
+ const assemblyKey = yield this.recoverAssemblyKey(resetUser);
365
+ const { rootKey } = yield this.encryptionService.decrypt(assemblyKey, resetUser.assemblyCipherData);
366
+ console.log(rootKey);
367
+ // Making sure it's a valid key.
368
+ const rootKeyJwk = yield JWK.asKey(rootKey);
369
+ const masterKey = yield this.keyGraphService.getKey(resetUser.masterKey.id);
370
+ const masterKeyWrappedRootKey = yield this.encryptionService.encryptToString(masterKey.jwk, rootKeyJwk.toJSON(true));
371
+ // The new password
372
+ const newPassIdpResult = yield this.keyFactory.derivePassIdp(Object.assign({ password: newPassword }, resetUser.passKey.passIdpParams));
373
+ const newIdpPassword = this.passwordService.getPassIdpString(newPassIdpResult.jwk);
374
+ // --------------------------------------------------------------
375
+ // Get assembly key challenge
376
+ // --------------------------------------------------------------
377
+ const challenge = (yield this.lrGraphQL.lrMutate(new LrMutation({
378
+ mutation: CreateTpAssemblyKeyChallengeMutation,
379
+ variables: {
380
+ input: {},
381
+ },
382
+ }), {
383
+ includeKeyGraph: false,
384
+ })).createTpAssemblyKeyChallenge.challenge;
385
+ console.log(challenge);
386
+ // Sign the challenge
387
+ // Generate a client side nonce that's no in the server's control.
388
+ challenge.clientNonce = this.keyFactory.randomString(TP_PASSWORD_RESET_CLIENT_NONCE_LENGTH);
389
+ console.log(challenge);
390
+ const assemblyKeyVerifierPrk = yield this.encryptionService.decrypt(assemblyKey, resetUser.wrappedAssemblyKeyVerifierPrk);
391
+ const signedChallenge = yield this.encryptionService.sign(assemblyKeyVerifierPrk, challenge);
392
+ // --------------------------------------------------------------
393
+ // Change password for the original user
394
+ // --------------------------------------------------------------
395
+ const tempIdpPassword = (yield this.lrGraphQL.lrMutate(new LrMutation({
396
+ mutation: PreCompleteTpPasswordResetRequestMutation,
397
+ variables: {
398
+ input: {
399
+ signedChallenge: JSON.stringify(signedChallenge),
400
+ },
401
+ },
402
+ }), {
403
+ includeKeyGraph: false,
404
+ })).preCompleteTpPasswordResetRequest.idpPassword;
405
+ // --------------------------------------------------------------
406
+ // Login as the original user using new temporary password
407
+ // --------------------------------------------------------------
408
+ // At this point, the original account's password has been changed
409
+ // to a temporary password. It is no longer possible for the user
410
+ // to use the original password to login. Any successful login
411
+ // can only be using the temporary password. So it's safe to assume
412
+ // that we want to "complete" the password reset.
413
+ // The maybe 2FA so we listen for the auth event from Amplify.
414
+ const retPromise = new Promise((resolve) => {
415
+ const listener = (data) => __awaiter(this, void 0, void 0, function* () {
416
+ if (data.payload.event !== 'signIn') {
417
+ return;
418
+ }
419
+ Hub.remove('auth', listener);
420
+ console.log(data.payload);
421
+ yield this.auth.signIn(resetUser.username, newIdpPassword);
422
+ // Switch over to the new set of keys
423
+ yield this.lrGraphQL.lrMutate(new LrMutation({
424
+ mutation: CompleteTpPasswordResetRequestMutation,
425
+ variables: {
426
+ input: {
427
+ masterKeyWrappedRootKey,
428
+ masterKeyId: masterKey.id,
429
+ },
430
+ },
431
+ }));
432
+ resolve();
433
+ });
434
+ Hub.listen('auth', listener);
435
+ });
436
+ // Signin as the original user. Password has been reset to temporary one. It should return
437
+ // with NEW_PASSWORD_REQUIRED
438
+ let user = yield this.auth.signIn(resetUser.username, tempIdpPassword, {
439
+ noProxy: 'true',
440
+ });
441
+ if (user.challengeName !== 'NEW_PASSWORD_REQUIRED') {
442
+ throw new LrException({
443
+ message: 'Internal error. Expecting Cognito to have done a password reset after call to PreCompleteTpPasswordResetRequestMutation.',
444
+ });
445
+ }
446
+ // Set new password on Idp
447
+ // the awsFetch() function passes NEW_PASSWORD_REQUIRED directly to AWS without
448
+ // going through the proxy.
449
+ user = yield this.auth.completeNewPassword(user, newIdpPassword, {});
450
+ return retPromise;
451
+ });
452
+ }
453
+ recoverAssemblyKey(resetUser) {
454
+ return __awaiter(this, void 0, void 0, function* () {
455
+ // Recover the assembly key.
456
+ let assemblyKeyParams;
457
+ const prk = yield this.keyGraphService.getKey(resetUser.pxk.id);
458
+ const shares = yield Promise.all(resetUser.approvals
459
+ .filter((approval) => !!approval.receiverCipherPartialAssemblyKey)
460
+ .map((approval) => __awaiter(this, void 0, void 0, function* () {
461
+ const partialAssemblyKey = yield this.encryptionService.decrypt(prk, approval.receiverCipherPartialAssemblyKey);
462
+ if (assemblyKeyParams) {
463
+ if (JSON.stringify(assemblyKeyParams) !==
464
+ JSON.stringify(partialAssemblyKey.assemblyKeyParams)) {
465
+ throw new LrBadStateException('The assembly key parameters are different between the approvals.');
466
+ }
467
+ }
468
+ else {
469
+ assemblyKeyParams = partialAssemblyKey.assemblyKeyParams;
470
+ }
471
+ return partialAssemblyKey.slip39.share.mnemonics;
472
+ })));
473
+ console.log('recoverAssemblyKey()', shares);
474
+ const rawAssemblyKey = yield this.slip39Service.recoverSecret(shares, TP_PASSWORD_RESET_SLIP39_PASSPHRASE);
475
+ return JWK.asKey(Object.assign(Object.assign({}, assemblyKeyParams), { k: rawAssemblyKey }));
476
+ });
477
+ }
478
+ }
479
+ LifeReadyAuthService.ɵprov = i0.ɵɵdefineInjectable({ factory: function LifeReadyAuthService_Factory() { return new LifeReadyAuthService(i0.ɵɵinject(i1.LR_CONFIG), i0.ɵɵinject(i2.AuthClass), i0.ɵɵinject(i3.KeyFactoryService), i0.ɵɵinject(i4.KeyService), i0.ɵɵinject(i5.ProfileService), i0.ɵɵinject(i6.KeyGraphService), i0.ɵɵinject(i7.PasswordService), i0.ɵɵinject(i8.IdleService), i0.ɵɵinject(i9.LrGraphQLService), i0.ɵɵinject(i10.TpPasswordResetProcessorService), i0.ɵɵinject(i11.PersistService), i0.ɵɵinject(i12.EncryptionService), i0.ɵɵinject(i13.Slip39Service)); }, token: LifeReadyAuthService, providedIn: "root" });
480
+ LifeReadyAuthService.decorators = [
481
+ { type: Injectable, args: [{
482
+ providedIn: 'root',
483
+ },] }
484
+ ];
485
+ LifeReadyAuthService.ctorParameters = () => [
486
+ { type: undefined, decorators: [{ type: Inject, args: [LR_CONFIG,] }] },
487
+ { type: AuthClass },
488
+ { type: KeyFactoryService },
489
+ { type: KeyService },
490
+ { type: ProfileService },
491
+ { type: KeyGraphService },
492
+ { type: PasswordService },
493
+ { type: IdleService },
494
+ { type: LrGraphQLService },
495
+ { type: TpPasswordResetProcessorService },
496
+ { type: PersistService },
497
+ { type: EncryptionService },
498
+ { type: Slip39Service }
499
+ ];
500
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"life-ready-auth.service.js","sourceRoot":"/opt/atlassian/pipelines/agent/build/projects/core/src/","sources":["lib/auth/life-ready-auth.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Inject, Injectable, isDevMode } from '@angular/core';\nimport { CognitoUser } from '@aws-amplify/auth';\nimport { Hub } from '@aws-amplify/core';\nimport { CognitoUserAttribute } from 'amazon-cognito-identity-js';\nimport { Observable, ReplaySubject } from 'rxjs';\nimport { PassIdpParams } from '../cryptography/cryptography.types';\nimport { KeyGraphService } from '../cryptography/key-graph.service';\nimport { KeyService } from '../cryptography/key.service';\nimport { ProfileService } from '../users/profile.service';\nimport { PasswordChangeStatus } from '../users/profile.types';\nimport {\n  LrConcurrentAccessException,\n  LrBadRequestException,\n  LrBadStateException,\n  LrException,\n} from '../_common/exceptions';\nimport {\n  CognitoChallengeUser,\n  CurrentUser,\n  TpPasswordResetUser,\n  LoginResult,\n  RecoveryStatus,\n} from './auth.types';\nimport { PasswordService } from './password.service';\nimport { AuthClass } from '@aws-amplify/auth/lib-esm/Auth';\nimport {\n  CompleteTpPasswordResetRequestMutation,\n  CreateTpAssemblyKeyChallengeMutation,\n  PreCompleteTpPasswordResetRequestMutation,\n  TpPasswordResetUserQuery,\n} from '../trusted-parties/tp-password-reset.gql';\nimport { IdleService } from './idle.service';\nimport { KeyFactoryService } from '../cryptography/key-factory.service';\nimport { LrGraphQLService, LrMutation } from '../api/lr-graphql';\nimport { TpClaimState, TpPasswordResetUserNode } from '../api/types';\nimport { TpPasswordResetProcessorService } from '../api/query-processor/tp-password-reset-processor.service';\nimport { SetSessionEncryptionKeyMutation } from './auth.gql';\nimport { PersistService } from '../api/persist.service';\nimport { JWK } from 'node-jose';\nimport { LifeReadyConfig, LR_CONFIG } from '../life-ready.config';\nimport { EncryptionService } from '../cryptography/encryption.service';\nimport { Slip39Service } from '../cryptography/slip39.service';\nimport {\n  TP_PASSWORD_RESET_CLIENT_NONCE_LENGTH,\n  TP_PASSWORD_RESET_SLIP39_PASSPHRASE,\n  TP_PASSWORD_RESET_USERNAME_SUFFIX,\n} from '../trusted-parties/tp-password-reset.constants';\n\nexport const initialiseAuth = (authService: LifeReadyAuthService) => {\n  return () => authService.initialise();\n};\n\nexport interface LoginOptions {\n  tpPasswordResetAutoComplete?: boolean;\n}\n\n@Injectable({\n  providedIn: 'root',\n})\nexport class LifeReadyAuthService {\n  private hubSubject: ReplaySubject<any> = new ReplaySubject<any>(1);\n  private currentUser: CurrentUser;\n  private currentResetUser: TpPasswordResetUser;\n\n  constructor(\n    @Inject(LR_CONFIG) private config: LifeReadyConfig,\n    private auth: AuthClass,\n    private keyFactory: KeyFactoryService,\n    private keyService: KeyService,\n    private profileService: ProfileService,\n    private keyGraphService: KeyGraphService,\n    private passwordService: PasswordService,\n    private idleService: IdleService,\n    private lrGraphQL: LrGraphQLService,\n    private tpPasswordResetProcessorService: TpPasswordResetProcessorService,\n    private persistService: PersistService,\n    private encryptionService: EncryptionService,\n    private slip39Service: Slip39Service\n  ) {}\n\n  public async initialise() {\n    Hub.listen('auth', (data) => this.hubSubject.next(data.payload));\n  }\n\n  private async loginIdpImpl(\n    emailOrPhone: string,\n    password: string,\n    passIdpParams: PassIdpParams,\n    recoveryStatus: RecoveryStatus\n  ): Promise<CognitoChallengeUser> {\n    const passIdpResult = await this.keyFactory.derivePassIdp({\n      password,\n      ...passIdpParams,\n    });\n    // Use the derived password to signin with cognito\n    const user = await this.auth.signIn(\n      emailOrPhone,\n      this.passwordService.getPassIdpString(passIdpResult.jwk)\n    );\n\n    user.recoveryStatus = recoveryStatus;\n\n    return user;\n  }\n\n  private async loginIdp(\n    emailOrPhone: string,\n    password: string\n  ): Promise<CognitoChallengeUser> {\n    // Download the salt needed to derive the PassIdp\n    const passIdpApiResult = await this.profileService.getPassIdpParams(\n      emailOrPhone\n    );\n\n    if (\n      passIdpApiResult.passwordChangeStatus === PasswordChangeStatus.InProgress\n    ) {\n      throw new LrConcurrentAccessException('A password change is in progress');\n    }\n\n    if (\n      passIdpApiResult.passwordChangeStatus === PasswordChangeStatus.Recovery\n    ) {\n      console.log('In recovery mode.');\n      // Let's say we don't know if the password is the new one or the old one. We just have to try both.\n      try {\n        const user = await this.loginIdpImpl(\n          emailOrPhone,\n          password,\n          passIdpApiResult.newPassIdpParams,\n          RecoveryStatus.NEW_PASSWORD\n        );\n        // New password worked. Let's set to the current password\n\n        // --Potential Failure Point 1--\n        // if changePasswordComplete() doesn't get called, then it should remain\n\n        console.log('New password works!');\n\n        return user;\n      } catch (error) {\n        // Just bubble up any other type of error.\n        if (error.code !== 'NotAuthorizedException') {\n          throw error;\n        }\n        // pass, try again assuming it's the old password\n      }\n\n      // Now assume it's the previous password. Any exception is allowed to bubble up.\n      try {\n        const user = await this.loginIdpImpl(\n          emailOrPhone,\n          password,\n          passIdpApiResult.currentPassIdpParams,\n          RecoveryStatus.OLD_PASSWORD\n        );\n        // Old password worked.\n        console.log('Old password works!');\n\n        return user;\n      } catch (error) {\n        // Just bubble up any other type of error.\n        throw error.code === 'NotAuthorizedException'\n          ? new LrBadRequestException(\n              'The password change request was interrupted, please try to login with both your new and old password'\n            )\n          : error;\n      }\n    }\n\n    // Try against as the TP password reset account\n    if (passIdpApiResult.tpPasswordReset) {\n      try {\n        // TP password reset is in process. We need to try the password against both\n        // original account and the new reset account.\n        const reset = passIdpApiResult.tpPasswordReset;\n        const ret = await this.loginIdpImpl(\n          reset.resetUsername,\n          password,\n          reset.passIdpParams,\n          RecoveryStatus.NONE\n        );\n        ret.isTpPasswordResetUser = true;\n\n        return ret;\n      } catch (err) {\n        // continue, try again as regular user.\n      }\n    }\n\n    // Login as regular user\n    return await this.loginIdpImpl(\n      emailOrPhone,\n      password,\n      passIdpApiResult.currentPassIdpParams,\n      RecoveryStatus.NONE\n    );\n  }\n\n  protected async handleSessionEncryptionKey() {\n    if (this.config.disableSessionEncryptionKey) {\n      if (!isDevMode()) {\n        const msg =\n          'You should not set disableSessionEncryptionKey=True in mode prod. It defaults to false.';\n        console.error(msg);\n        throw new Error(msg);\n      } else {\n        console.warn(\n          'You have set disableSessionEncryptionKey=True. Make sure not to do this in prod mode.'\n        );\n      }\n    } else {\n      // Set the session key to a new encryption key for this session\n      const sessionEncryptionKey = await this.keyFactory.createKey();\n      await this.lrGraphQL.lrMutate(\n        new LrMutation({\n          mutation: SetSessionEncryptionKeyMutation,\n          variables: {\n            input: {\n              sessionEncryptionKey: JSON.stringify(\n                sessionEncryptionKey.toJSON(true)\n              ),\n            },\n          },\n        }),\n        {\n          includeKeyGraph: false,\n        }\n      );\n\n      this.persistService.setServerSessionEncryptionKey(sessionEncryptionKey);\n    }\n  }\n\n  protected async handlePostAuth(cognitoUser: CognitoChallengeUser) {\n    await this.handlePasswordRecovery(cognitoUser);\n    await this.handleSessionEncryptionKey();\n  }\n\n  public async login(\n    emailOrPhone: string,\n    password: string,\n    { tpPasswordResetAutoComplete = true }: LoginOptions = {}\n  ) {\n    let loginResult = await this.loginImpl(emailOrPhone, password);\n\n    if (\n      tpPasswordResetAutoComplete &&\n      loginResult.resetUser?.state === TpClaimState.APPROVED\n    ) {\n      await this.completeRequest(password);\n      loginResult = await this.loginImpl(emailOrPhone, password);\n    }\n\n    return loginResult;\n  }\n\n  public async loginImpl(\n    emailOrPhone: string,\n    password: string\n  ): Promise<LoginResult> {\n    await this.logout();\n    const cognitoUser = await this.loginIdp(emailOrPhone, password);\n\n    // todo: Meet MFA challenges.\n    if (['SMS_MFA', 'SOFTWARE_TOKEN_MFA'].includes(cognitoUser.challengeName)) {\n      return { hasChallenge: true, challenge: cognitoUser };\n    }\n\n    await this.handlePostAuth(cognitoUser);\n\n    if (cognitoUser.isTpPasswordResetUser) {\n      // Assuming there is no MFA on the TP reset user.\n      const resetUser = await this.loadResetUser(password);\n      return { hasChallenge: false, resetUser };\n    } else {\n      const user = await this.loadUser(cognitoUser, password);\n      await this.idleService.start(); // Run idleService whenever user is logged in.\n      return { hasChallenge: false, user };\n    }\n  }\n\n  // TODO <AZ> We need to handle the isTpPasswordResetUser=True case here after MFA as well.\n  public async verifyLogin(\n    challenge: CognitoChallengeUser,\n    password: string,\n    rememberMe: boolean,\n    code: string\n  ): Promise<CurrentUser> {\n    await this.auth.confirmSignIn(challenge, code, challenge.challengeName);\n\n    // TODO: this.auth.confirmSignIn() could return another challenge.\n\n    const cognitoUser: CognitoUser = await this.auth.currentAuthenticatedUser();\n\n    await this.handlePostAuth(challenge);\n\n    const user = await this.loadUser(cognitoUser, password);\n\n    if (rememberMe) {\n      cognitoUser.setDeviceStatusRemembered({\n        onSuccess: () => {},\n        onFailure: (e) => console.error(e),\n      });\n    }\n\n    return user;\n  }\n\n  async handlePasswordRecovery(user: CognitoChallengeUser) {\n    if (user.recoveryStatus !== RecoveryStatus.NONE) {\n      const jwtToken = user\n        .getSignInUserSession()\n        .getAccessToken()\n        .getJwtToken();\n      await this.passwordService.changePasswordComplete(\n        jwtToken,\n        user.recoveryStatus === RecoveryStatus.NEW_PASSWORD\n      );\n    }\n  }\n\n  async getUserOrResetUser(\n    reload: boolean = false\n  ): Promise<CurrentUser | TpPasswordResetUser> {\n    const cognitoUser = await this.auth.currentAuthenticatedUser();\n\n    if (cognitoUser.getUsername().endsWith(TP_PASSWORD_RESET_USERNAME_SUFFIX)) {\n      return this.getResetUser(reload);\n    } else {\n      return this.getUser(reload);\n    }\n  }\n\n  async getResetUser(reload: boolean = false): Promise<TpPasswordResetUser> {\n    if (!reload && this.currentResetUser) {\n      return this.currentResetUser;\n    }\n\n    this.currentResetUser = await this.loadResetUser();\n\n    await this.idleService.start(); // Run idleService whenever user is logged in.\n\n    return this.currentResetUser;\n  }\n\n  async getUser(reload: boolean = false): Promise<CurrentUser> {\n    if (!reload && this.currentUser) {\n      return this.currentUser;\n    }\n    this.currentUser = await this.loadUser(\n      await this.auth.currentAuthenticatedUser()\n    );\n    console.log('Starting idle service.');\n    await this.idleService.start(); // Run idleService whenever user is logged in.\n    return this.currentUser;\n  }\n\n  private mapTPVaultAccess(features?: any): boolean {\n    const tpVaultFeature = features?.tpVault;\n    return (\n      tpVaultFeature?.length > 0 &&\n      tpVaultFeature.some((feature) => feature.toUpperCase() === 'ACCESS')\n    );\n  }\n\n  private async loadUser(\n    cognitoUser: CognitoUser,\n    password?: string\n  ): Promise<CurrentUser> {\n    const {\n      currentUser,\n      contactCard,\n      userPlans,\n    } = await this.profileService.getCurrentUser();\n\n    if (currentUser.sessionEncryptionKey) {\n      this.persistService.setServerSessionEncryptionKey(\n        await JWK.asKey(currentUser.sessionEncryptionKey)\n      );\n    }\n\n    const userAttributes = await this.auth.userAttributes(cognitoUser);\n\n    if (password) {\n      const passKey = (\n        await this.keyFactory.derivePassKey({\n          password,\n          ...currentUser.currentUserKey.passKey.passKeyParams,\n        })\n      ).jwk;\n\n      await this.idleService.persistMasterKey(\n        await this.keyGraphService.unwrapWithPassKey(\n          currentUser.currentUserKey.passKey.id,\n          passKey,\n          currentUser.currentUserKey.masterKey.id\n        )\n      );\n    }\n    await this.keyGraphService.populateKeys(currentUser.currentUserKey);\n\n    return {\n      id: currentUser.id,\n      sub: this.getUserAttribute('sub', userAttributes),\n      username: currentUser.username,\n      currentUserKey: currentUser.currentUserKey,\n      getAccessJwtToken: () =>\n        cognitoUser.getSignInUserSession().getAccessToken().getJwtToken(),\n      email: this.getUserAttribute('email', userAttributes),\n      emailVerified:\n        this.getUserAttribute('email_verified', userAttributes) === 'true',\n      phone: this.getUserAttribute('phone_number', userAttributes),\n      phoneVerified:\n        this.getUserAttribute('phone_number_verified', userAttributes) ===\n        'true',\n      contactCard: {\n        ...(await this.profileService.decryptContactCard(contactCard)),\n      },\n      userDelete: currentUser.userDelete,\n      userPlans,\n      hasTPVaultAccess: this.mapTPVaultAccess(currentUser.features),\n      features: currentUser.features,\n      sessionEncryptionKey: currentUser.sessionEncryptionKey,\n    };\n  }\n\n  public watchAuth(): Observable<any> {\n    return this.hubSubject;\n  }\n\n  public async logout(): Promise<void> {\n    this.currentUser = null;\n    this.keyService.purgeKeys();\n    this.keyGraphService.purgeKeys();\n\n    await Promise.all([this.auth.signOut(), this.profileService.signOut()]);\n  }\n\n  private getUserAttribute(\n    attributeName: string,\n    userAttributes: CognitoUserAttribute[]\n  ) {\n    const userAttribute = userAttributes.find(\n      (x) => x.getName() === attributeName\n    );\n\n    return userAttribute ? userAttribute.getValue() : null;\n  }\n\n  public async loadResetUser(password?: string): Promise<TpPasswordResetUser> {\n    const { tpPasswordResetUser: resetUser } = await this.lrGraphQL.query({\n      query: TpPasswordResetUserQuery,\n    });\n\n    if (resetUser.sessionEncryptionKey) {\n      this.persistService.setServerSessionEncryptionKey(\n        await JWK.asKey(resetUser.sessionEncryptionKey)\n      );\n    }\n\n    // Update the keys\n    if (password) {\n      const passKey = (\n        await this.keyFactory.derivePassKey({\n          password,\n          ...resetUser.passKey.passKeyParams,\n        })\n      ).jwk;\n\n      await this.idleService.persistMasterKey(\n        await this.keyGraphService.unwrapWithPassKey(\n          resetUser.passKey.id,\n          passKey,\n          resetUser.masterKey.id\n        )\n      );\n    }\n\n    this.keyService.populateKeys({\n      passKey: {\n        id: resetUser.passKey.id,\n      },\n      masterKey: {\n        id: resetUser.masterKey.id,\n      },\n    });\n\n    const userAttributes = await this.auth.userAttributes(\n      await this.auth.currentAuthenticatedUser()\n    );\n    const sub = this.getUserAttribute('sub', userAttributes);\n\n    return {\n      ...(await this.tpPasswordResetProcessorService.processTpPasswordResetUserNode(\n        resetUser\n      )),\n      sub,\n    };\n  }\n\n  public async refreshAccessToken() {\n    const cognitoUser: CognitoUser = await this.auth.currentAuthenticatedUser();\n    const refreshToken = cognitoUser.getSignInUserSession().getRefreshToken();\n\n    return new Promise((resolve, reject) => {\n      cognitoUser.refreshSession(refreshToken, (err, data) => {\n        if (err) {\n          console.error('Error refreshing token: ', err);\n          reject(err);\n        } else {\n          console.log('Token refresh complete: ', data);\n          resolve(0);\n        }\n      });\n    });\n  }\n\n  async completeRequest(newPassword: string): Promise<void> {\n    const resetUser = await this.getResetUser(true);\n\n    if (resetUser.state !== TpClaimState.APPROVED) {\n      throw new LrBadStateException(\n        'Password reset request has not been approved.'\n      );\n    }\n\n    // --------------------------------------------------------------\n    // Prepare all materials to ensure there are no errors.\n    // --------------------------------------------------------------\n    const assemblyKey = await this.recoverAssemblyKey(resetUser);\n\n    const { rootKey } = await this.encryptionService.decrypt(\n      assemblyKey,\n      resetUser.assemblyCipherData\n    );\n    console.log(rootKey);\n\n    // Making sure it's a valid key.\n    const rootKeyJwk = await JWK.asKey(rootKey);\n\n    const masterKey = await this.keyGraphService.getKey(resetUser.masterKey.id);\n\n    const masterKeyWrappedRootKey = await this.encryptionService.encryptToString(\n      masterKey.jwk,\n      rootKeyJwk.toJSON(true)\n    );\n\n    // The new password\n    const newPassIdpResult = await this.keyFactory.derivePassIdp({\n      password: newPassword,\n      ...resetUser.passKey.passIdpParams,\n    });\n\n    const newIdpPassword = this.passwordService.getPassIdpString(\n      newPassIdpResult.jwk\n    );\n\n    // --------------------------------------------------------------\n    // Get assembly key challenge\n    // --------------------------------------------------------------\n    const challenge = (\n      await this.lrGraphQL.lrMutate(\n        new LrMutation({\n          mutation: CreateTpAssemblyKeyChallengeMutation,\n          variables: {\n            input: {},\n          },\n        }),\n        {\n          includeKeyGraph: false,\n        }\n      )\n    ).createTpAssemblyKeyChallenge.challenge;\n\n    console.log(challenge);\n\n    // Sign the challenge\n    // Generate a client side nonce that's no in the server's control.\n    challenge.clientNonce = this.keyFactory.randomString(\n      TP_PASSWORD_RESET_CLIENT_NONCE_LENGTH\n    );\n    console.log(challenge);\n\n    const assemblyKeyVerifierPrk = await this.encryptionService.decrypt(\n      assemblyKey,\n      resetUser.wrappedAssemblyKeyVerifierPrk\n    );\n    const signedChallenge = await this.encryptionService.sign(\n      assemblyKeyVerifierPrk,\n      challenge\n    );\n\n    // --------------------------------------------------------------\n    // Change password for the original user\n    // --------------------------------------------------------------\n    const tempIdpPassword = (\n      await this.lrGraphQL.lrMutate(\n        new LrMutation({\n          mutation: PreCompleteTpPasswordResetRequestMutation,\n          variables: {\n            input: {\n              signedChallenge: JSON.stringify(signedChallenge),\n            },\n          },\n        }),\n        {\n          includeKeyGraph: false,\n        }\n      )\n    ).preCompleteTpPasswordResetRequest.idpPassword;\n\n    // --------------------------------------------------------------\n    // Login as the original user using new temporary password\n    // --------------------------------------------------------------\n    // At this point, the original account's password has been changed\n    // to a temporary password. It is no longer possible for the user\n    // to use the original password to login. Any successful login\n    // can only be using the temporary password. So it's safe to assume\n    // that we want to \"complete\" the password reset.\n\n    // The maybe 2FA so we listen for the auth event from Amplify.\n    const retPromise = new Promise<void>((resolve) => {\n      const listener = async (data) => {\n        if (data.payload.event !== 'signIn') {\n          return;\n        }\n\n        Hub.remove('auth', listener);\n\n        console.log(data.payload);\n\n        await this.auth.signIn(resetUser.username, newIdpPassword);\n\n        // Switch over to the new set of keys\n        await this.lrGraphQL.lrMutate(\n          new LrMutation({\n            mutation: CompleteTpPasswordResetRequestMutation,\n            variables: {\n              input: {\n                masterKeyWrappedRootKey,\n                masterKeyId: masterKey.id,\n              },\n            },\n          })\n        );\n\n        resolve();\n      };\n\n      Hub.listen('auth', listener);\n    });\n\n    // Signin as the original user. Password has been reset to temporary one. It should return\n    // with NEW_PASSWORD_REQUIRED\n    let user = await this.auth.signIn(resetUser.username, tempIdpPassword, {\n      noProxy: 'true',\n    });\n\n    if (user.challengeName !== 'NEW_PASSWORD_REQUIRED') {\n      throw new LrException({\n        message:\n          'Internal error. Expecting Cognito to have done a password reset after call to PreCompleteTpPasswordResetRequestMutation.',\n      });\n    }\n\n    // Set new password on Idp\n    // the awsFetch() function passes NEW_PASSWORD_REQUIRED directly to AWS without\n    // going through the proxy.\n    user = await this.auth.completeNewPassword(user, newIdpPassword, {});\n\n    return retPromise;\n  }\n\n  private async recoverAssemblyKey(\n    resetUser: TpPasswordResetUserNode\n  ): Promise<JWK.Key> {\n    // Recover the assembly key.\n    let assemblyKeyParams: object;\n\n    const prk = await this.keyGraphService.getKey(resetUser.pxk.id);\n\n    const shares = await Promise.all(\n      resetUser.approvals\n        .filter((approval) => !!approval.receiverCipherPartialAssemblyKey)\n        .map(async (approval) => {\n          const partialAssemblyKey = await this.encryptionService.decrypt(\n            prk,\n            approval.receiverCipherPartialAssemblyKey\n          );\n\n          if (assemblyKeyParams) {\n            if (\n              JSON.stringify(assemblyKeyParams) !==\n              JSON.stringify(partialAssemblyKey.assemblyKeyParams)\n            ) {\n              throw new LrBadStateException(\n                'The assembly key parameters are different between the approvals.'\n              );\n            }\n          } else {\n            assemblyKeyParams = partialAssemblyKey.assemblyKeyParams;\n          }\n          return partialAssemblyKey.slip39.share.mnemonics;\n        })\n    );\n\n    console.log('recoverAssemblyKey()', shares);\n\n    const rawAssemblyKey = await this.slip39Service.recoverSecret(\n      shares,\n      TP_PASSWORD_RESET_SLIP39_PASSPHRASE\n    );\n\n    return JWK.asKey({\n      ...assemblyKeyParams,\n      k: rawAssemblyKey,\n    });\n  }\n}\n"]}