@lifeready/core 9.0.8 → 10.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (225) hide show
  1. package/fesm2022/lifeready-core.mjs +10522 -0
  2. package/fesm2022/lifeready-core.mjs.map +1 -0
  3. package/package.json +21 -16
  4. package/types/lifeready-core.d.ts +5329 -0
  5. package/bundles/lifeready-core.umd.js +0 -13231
  6. package/bundles/lifeready-core.umd.js.map +0 -1
  7. package/bundles/lifeready-core.umd.min.js +0 -2
  8. package/bundles/lifeready-core.umd.min.js.map +0 -1
  9. package/esm2015/lib/_common/ast.js +0 -41
  10. package/esm2015/lib/_common/deferred-promise.js +0 -24
  11. package/esm2015/lib/_common/exceptions.js +0 -186
  12. package/esm2015/lib/_common/index.js +0 -3
  13. package/esm2015/lib/_common/kc-lodash.js +0 -11
  14. package/esm2015/lib/_common/key.js +0 -28
  15. package/esm2015/lib/_common/queries.gql.js +0 -43
  16. package/esm2015/lib/_common/run-outside-angular.js +0 -80
  17. package/esm2015/lib/_common/storage.js +0 -28
  18. package/esm2015/lib/_common/types.js +0 -2
  19. package/esm2015/lib/_common/utils.js +0 -73
  20. package/esm2015/lib/api/lr-apollo.service.js +0 -47
  21. package/esm2015/lib/api/lr-graphql/index.js +0 -6
  22. package/esm2015/lib/api/lr-graphql/lr-graphql.service.js +0 -170
  23. package/esm2015/lib/api/lr-graphql/lr-merged-mutation.js +0 -216
  24. package/esm2015/lib/api/lr-graphql/lr-mutation-base.js +0 -51
  25. package/esm2015/lib/api/lr-graphql/lr-mutation.js +0 -91
  26. package/esm2015/lib/api/lr-graphql/lr.service.js +0 -18
  27. package/esm2015/lib/api/query-processor/common-processors.service.js +0 -94
  28. package/esm2015/lib/api/query-processor/index.js +0 -3
  29. package/esm2015/lib/api/query-processor/query-processor.service.js +0 -307
  30. package/esm2015/lib/api/query-processor/tp-password-reset-processor.service.js +0 -110
  31. package/esm2015/lib/api/types/graphql.types.js +0 -8
  32. package/esm2015/lib/api/types/index.js +0 -3
  33. package/esm2015/lib/api/types/lr-graphql.types.js +0 -182
  34. package/esm2015/lib/auth/auth.config.js +0 -57
  35. package/esm2015/lib/auth/auth.gql.private.js +0 -85
  36. package/esm2015/lib/auth/auth.service.js +0 -616
  37. package/esm2015/lib/auth/auth.types.js +0 -19
  38. package/esm2015/lib/contact-card/contact-card.gql.js +0 -79
  39. package/esm2015/lib/contact-card/contact-card.service.js +0 -156
  40. package/esm2015/lib/contact-card/contact-card2.gql.js +0 -29
  41. package/esm2015/lib/contact-card/contact-card2.service.js +0 -103
  42. package/esm2015/lib/encryption/encryption.service.js +0 -188
  43. package/esm2015/lib/file-upload/file-upload.service.js +0 -70
  44. package/esm2015/lib/file-upload/file-upload.types.js +0 -2
  45. package/esm2015/lib/idle/idle.service.js +0 -159
  46. package/esm2015/lib/idle/idle.types.js +0 -7
  47. package/esm2015/lib/item/item.gql.js +0 -166
  48. package/esm2015/lib/item/item.gql.private.js +0 -41
  49. package/esm2015/lib/item/item.service.js +0 -662
  50. package/esm2015/lib/item/item.types.js +0 -2
  51. package/esm2015/lib/key/key-factory.service.js +0 -226
  52. package/esm2015/lib/key/key-graph.service.js +0 -314
  53. package/esm2015/lib/key/key-meta.service.js +0 -153
  54. package/esm2015/lib/key/key.service.js +0 -124
  55. package/esm2015/lib/key/key.types.js +0 -16
  56. package/esm2015/lib/key-exchange/key-exchange.gql.js +0 -174
  57. package/esm2015/lib/key-exchange/key-exchange.service.js +0 -496
  58. package/esm2015/lib/lbop/lbop.service.js +0 -351
  59. package/esm2015/lib/life-ready.config.js +0 -96
  60. package/esm2015/lib/life-ready.module.js +0 -42
  61. package/esm2015/lib/lock/lock.gql.js +0 -40
  62. package/esm2015/lib/lock/lock.service.js +0 -64
  63. package/esm2015/lib/notification/notification.gql.js +0 -43
  64. package/esm2015/lib/notification/notification.service.js +0 -118
  65. package/esm2015/lib/password/password.gql.js +0 -28
  66. package/esm2015/lib/password/password.service.js +0 -309
  67. package/esm2015/lib/persist/persist.service.js +0 -181
  68. package/esm2015/lib/plan/plan.gql.js +0 -91
  69. package/esm2015/lib/plan/plan.service.js +0 -191
  70. package/esm2015/lib/plan/plan.types.js +0 -2
  71. package/esm2015/lib/profile/profile-details.service.js +0 -261
  72. package/esm2015/lib/profile/profile.gql.js +0 -170
  73. package/esm2015/lib/profile/profile.service.js +0 -166
  74. package/esm2015/lib/profile/profile.types.js +0 -45
  75. package/esm2015/lib/register/register.service.js +0 -173
  76. package/esm2015/lib/register/register.types.js +0 -3
  77. package/esm2015/lib/reminder/reminder.gql.js +0 -27
  78. package/esm2015/lib/reminder/reminder.service.js +0 -85
  79. package/esm2015/lib/reminder/reminder.types.js +0 -2
  80. package/esm2015/lib/scenario/scenario.constants.js +0 -2
  81. package/esm2015/lib/scenario/scenario.controller.js +0 -34
  82. package/esm2015/lib/scenario/scenario.gql.js +0 -90
  83. package/esm2015/lib/scenario/scenario.private.gql.js +0 -200
  84. package/esm2015/lib/scenario/scenario.service.js +0 -679
  85. package/esm2015/lib/scenario/scenario.types.js +0 -2
  86. package/esm2015/lib/server-config/server-config.gql.js +0 -9
  87. package/esm2015/lib/server-config/server-config.service.js +0 -41
  88. package/esm2015/lib/shared-contact-card/shared-contact-card.service.js +0 -119
  89. package/esm2015/lib/shared-contact-card/shared-contact-card2.gql.js +0 -41
  90. package/esm2015/lib/shared-contact-card/shared-contact-card2.service.js +0 -117
  91. package/esm2015/lib/slip39/slip39.service.js +0 -167
  92. package/esm2015/lib/time/time.service.js +0 -152
  93. package/esm2015/lib/tp-assembly/tp-assembly.js +0 -363
  94. package/esm2015/lib/tp-assembly/tp-assembly.private.gql.js +0 -22
  95. package/esm2015/lib/tp-assembly/tp-assembly.types.js +0 -2
  96. package/esm2015/lib/tp-password-reset/tp-password-reset-request.service.js +0 -98
  97. package/esm2015/lib/tp-password-reset/tp-password-reset-user.service.js +0 -121
  98. package/esm2015/lib/tp-password-reset/tp-password-reset.constants.js +0 -4
  99. package/esm2015/lib/tp-password-reset/tp-password-reset.controller.js +0 -34
  100. package/esm2015/lib/tp-password-reset/tp-password-reset.gql.js +0 -74
  101. package/esm2015/lib/tp-password-reset/tp-password-reset.private.gql.js +0 -166
  102. package/esm2015/lib/tp-password-reset/tp-password-reset.private.service.js +0 -54
  103. package/esm2015/lib/tp-password-reset/tp-password-reset.service.js +0 -110
  104. package/esm2015/lib/tp-password-reset/tp-password-reset.types.js +0 -2
  105. package/esm2015/lib/trusted-party/trusted-party.gql.js +0 -96
  106. package/esm2015/lib/trusted-party/trusted-party.gql.private.js +0 -51
  107. package/esm2015/lib/trusted-party/trusted-party.service.js +0 -461
  108. package/esm2015/lib/trusted-party/trusted-party.types.js +0 -2
  109. package/esm2015/lib/two-factor/two-factor.service.js +0 -74
  110. package/esm2015/lib/user/user.gql.js +0 -32
  111. package/esm2015/lib/user/user.service.js +0 -58
  112. package/esm2015/lib/user/user.types.js +0 -2
  113. package/esm2015/lib/web-crypto/web-crypto.service.js +0 -29
  114. package/esm2015/lifeready-core.js +0 -17
  115. package/esm2015/public-api.js +0 -60
  116. package/fesm2015/lifeready-core.js +0 -10970
  117. package/fesm2015/lifeready-core.js.map +0 -1
  118. package/lib/_common/ast.d.ts +0 -11
  119. package/lib/_common/deferred-promise.d.ts +0 -12
  120. package/lib/_common/exceptions.d.ts +0 -126
  121. package/lib/_common/index.d.ts +0 -2
  122. package/lib/_common/kc-lodash.d.ts +0 -5
  123. package/lib/_common/key.d.ts +0 -14
  124. package/lib/_common/queries.gql.d.ts +0 -4
  125. package/lib/_common/run-outside-angular.d.ts +0 -14
  126. package/lib/_common/storage.d.ts +0 -13
  127. package/lib/_common/types.d.ts +0 -15
  128. package/lib/_common/utils.d.ts +0 -12
  129. package/lib/api/lr-apollo.service.d.ts +0 -15
  130. package/lib/api/lr-graphql/index.d.ts +0 -5
  131. package/lib/api/lr-graphql/lr-graphql.service.d.ts +0 -81
  132. package/lib/api/lr-graphql/lr-merged-mutation.d.ts +0 -46
  133. package/lib/api/lr-graphql/lr-mutation-base.d.ts +0 -28
  134. package/lib/api/lr-graphql/lr-mutation.d.ts +0 -48
  135. package/lib/api/lr-graphql/lr.service.d.ts +0 -9
  136. package/lib/api/query-processor/common-processors.service.d.ts +0 -36
  137. package/lib/api/query-processor/index.d.ts +0 -2
  138. package/lib/api/query-processor/query-processor.service.d.ts +0 -18
  139. package/lib/api/query-processor/tp-password-reset-processor.service.d.ts +0 -15
  140. package/lib/api/types/graphql.types.d.ts +0 -30
  141. package/lib/api/types/index.d.ts +0 -2
  142. package/lib/api/types/lr-graphql.types.d.ts +0 -807
  143. package/lib/auth/auth.config.d.ts +0 -5
  144. package/lib/auth/auth.gql.private.d.ts +0 -25
  145. package/lib/auth/auth.service.d.ts +0 -72
  146. package/lib/auth/auth.types.d.ts +0 -70
  147. package/lib/contact-card/contact-card.gql.d.ts +0 -7
  148. package/lib/contact-card/contact-card.service.d.ts +0 -53
  149. package/lib/contact-card/contact-card2.gql.d.ts +0 -25
  150. package/lib/contact-card/contact-card2.service.d.ts +0 -64
  151. package/lib/encryption/encryption.service.d.ts +0 -42
  152. package/lib/file-upload/file-upload.service.d.ts +0 -15
  153. package/lib/file-upload/file-upload.types.d.ts +0 -5
  154. package/lib/idle/idle.service.d.ts +0 -47
  155. package/lib/idle/idle.types.d.ts +0 -10
  156. package/lib/item/item.gql.d.ts +0 -134
  157. package/lib/item/item.gql.private.d.ts +0 -35
  158. package/lib/item/item.service.d.ts +0 -201
  159. package/lib/item/item.types.d.ts +0 -95
  160. package/lib/key/key-factory.service.d.ts +0 -40
  161. package/lib/key/key-graph.service.d.ts +0 -41
  162. package/lib/key/key-meta.service.d.ts +0 -51
  163. package/lib/key/key.service.d.ts +0 -36
  164. package/lib/key/key.types.d.ts +0 -86
  165. package/lib/key-exchange/key-exchange.gql.d.ts +0 -141
  166. package/lib/key-exchange/key-exchange.service.d.ts +0 -179
  167. package/lib/lbop/lbop.service.d.ts +0 -99
  168. package/lib/life-ready.config.d.ts +0 -26
  169. package/lib/life-ready.module.d.ts +0 -5
  170. package/lib/lock/lock.gql.d.ts +0 -27
  171. package/lib/lock/lock.service.d.ts +0 -34
  172. package/lib/notification/notification.gql.d.ts +0 -37
  173. package/lib/notification/notification.service.d.ts +0 -64
  174. package/lib/password/password.gql.d.ts +0 -3
  175. package/lib/password/password.service.d.ts +0 -79
  176. package/lib/persist/persist.service.d.ts +0 -31
  177. package/lib/plan/plan.gql.d.ts +0 -69
  178. package/lib/plan/plan.service.d.ts +0 -111
  179. package/lib/plan/plan.types.d.ts +0 -16
  180. package/lib/profile/profile-details.service.d.ts +0 -20
  181. package/lib/profile/profile.gql.d.ts +0 -21
  182. package/lib/profile/profile.service.d.ts +0 -32
  183. package/lib/profile/profile.types.d.ts +0 -121
  184. package/lib/register/register.service.d.ts +0 -25
  185. package/lib/register/register.types.d.ts +0 -6
  186. package/lib/reminder/reminder.gql.d.ts +0 -23
  187. package/lib/reminder/reminder.service.d.ts +0 -33
  188. package/lib/reminder/reminder.types.d.ts +0 -17
  189. package/lib/scenario/scenario.constants.d.ts +0 -1
  190. package/lib/scenario/scenario.controller.d.ts +0 -10
  191. package/lib/scenario/scenario.gql.d.ts +0 -78
  192. package/lib/scenario/scenario.private.gql.d.ts +0 -16
  193. package/lib/scenario/scenario.service.d.ts +0 -655
  194. package/lib/scenario/scenario.types.d.ts +0 -64
  195. package/lib/server-config/server-config.gql.d.ts +0 -5
  196. package/lib/server-config/server-config.service.d.ts +0 -9
  197. package/lib/shared-contact-card/shared-contact-card.service.d.ts +0 -33
  198. package/lib/shared-contact-card/shared-contact-card2.gql.d.ts +0 -36
  199. package/lib/shared-contact-card/shared-contact-card2.service.d.ts +0 -45
  200. package/lib/slip39/slip39.service.d.ts +0 -42
  201. package/lib/time/time.service.d.ts +0 -26
  202. package/lib/tp-assembly/tp-assembly.d.ts +0 -177
  203. package/lib/tp-assembly/tp-assembly.private.gql.d.ts +0 -5
  204. package/lib/tp-assembly/tp-assembly.types.d.ts +0 -40
  205. package/lib/tp-password-reset/tp-password-reset-request.service.d.ts +0 -16
  206. package/lib/tp-password-reset/tp-password-reset-user.service.d.ts +0 -29
  207. package/lib/tp-password-reset/tp-password-reset.constants.d.ts +0 -3
  208. package/lib/tp-password-reset/tp-password-reset.controller.d.ts +0 -10
  209. package/lib/tp-password-reset/tp-password-reset.gql.d.ts +0 -63
  210. package/lib/tp-password-reset/tp-password-reset.private.gql.d.ts +0 -163
  211. package/lib/tp-password-reset/tp-password-reset.private.service.d.ts +0 -59
  212. package/lib/tp-password-reset/tp-password-reset.service.d.ts +0 -112
  213. package/lib/tp-password-reset/tp-password-reset.types.d.ts +0 -40
  214. package/lib/trusted-party/trusted-party.gql.d.ts +0 -85
  215. package/lib/trusted-party/trusted-party.gql.private.d.ts +0 -40
  216. package/lib/trusted-party/trusted-party.service.d.ts +0 -192
  217. package/lib/trusted-party/trusted-party.types.d.ts +0 -31
  218. package/lib/two-factor/two-factor.service.d.ts +0 -15
  219. package/lib/user/user.gql.d.ts +0 -8
  220. package/lib/user/user.service.d.ts +0 -9
  221. package/lib/user/user.types.d.ts +0 -16
  222. package/lib/web-crypto/web-crypto.service.d.ts +0 -5
  223. package/lifeready-core.d.ts +0 -16
  224. package/lifeready-core.metadata.json +0 -1
  225. package/public-api.d.ts +0 -56
@@ -1,351 +0,0 @@
1
- import { __awaiter } from "tslib";
2
- /* eslint-disable @typescript-eslint/no-explicit-any */
3
- import { HttpClient } from '@angular/common/http';
4
- import { Inject, Injectable } from '@angular/core';
5
- import { AuthClass } from '@aws-amplify/auth/lib-esm/Auth';
6
- import gql from 'graphql-tag';
7
- import { Slip39Helper } from 'slip39';
8
- import { LrApolloService } from '../api/lr-apollo.service';
9
- import { EncryptionService } from '../encryption/encryption.service';
10
- import { KeyFactoryService as KFS } from '../key/key-factory.service';
11
- import { KeyService } from '../key/key.service';
12
- import { KC_CONFIG } from '../life-ready.config';
13
- import { PasswordService } from '../password/password.service';
14
- import { KcBadLogicException, KcError, KcException, KcInternalErrorException, } from '../_common/exceptions';
15
- import * as i0 from "@angular/core";
16
- import * as i1 from "../life-ready.config";
17
- import * as i2 from "@angular/common/http";
18
- import * as i3 from "../api/lr-apollo.service";
19
- import * as i4 from "@aws-amplify/auth/lib-esm/Auth";
20
- import * as i5 from "../key/key-factory.service";
21
- import * as i6 from "../key/key.service";
22
- import * as i7 from "../encryption/encryption.service";
23
- import * as i8 from "../password/password.service";
24
- export const ERROR_SOURCE = 'LBOP';
25
- export var KcLbopErrorCode;
26
- (function (KcLbopErrorCode) {
27
- KcLbopErrorCode["INVALID_PASSPHRASE"] = "INVALID_PASSPHRASE";
28
- })(KcLbopErrorCode || (KcLbopErrorCode = {}));
29
- export const CreateLbopQuery = gql `
30
- mutation CreateLbop($input: CreateLbopInput!) {
31
- createLbop(input: $input) {
32
- lbop {
33
- id
34
- }
35
- }
36
- }
37
- `;
38
- export const DeleteLbopQuery = gql `
39
- mutation DeleteLbop($input: DeleteLbopInput!) {
40
- deleteLbop(input: $input) {
41
- id
42
- }
43
- }
44
- `;
45
- export const UpdateLbopQuery = gql `
46
- mutation UpdateLbop($input: UpdateLbopInput!) {
47
- updateLbop(input: $input) {
48
- lbop {
49
- id
50
- }
51
- }
52
- }
53
- `;
54
- export const LbopQuery = gql `
55
- query Lbop($id: LrRelayIdInput!) {
56
- lbop(id: $id) {
57
- id
58
- cipherMeta
59
- }
60
- }
61
- `;
62
- export const LbopsQuery = gql `
63
- query Lbops {
64
- lbops {
65
- edges {
66
- node {
67
- id
68
- cipherMeta
69
- }
70
- }
71
- }
72
- }
73
- `;
74
- export class LbopService {
75
- constructor(config, http, lrApollo, auth, keyFactory, keyService, encryptionService, passwordService) {
76
- this.config = config;
77
- this.http = http;
78
- this.lrApollo = lrApollo;
79
- this.auth = auth;
80
- this.keyFactory = keyFactory;
81
- this.keyService = keyService;
82
- this.encryptionService = encryptionService;
83
- this.passwordService = passwordService;
84
- this.CLIENT_NONCE_LENGTH = 32;
85
- // There are 1024 words (10 bits), so 25 words should give ~256 bits of entropy.
86
- this.LBOP_WORDS = 25;
87
- }
88
- getPartial(lbopString) {
89
- return lbopString.split(' ')[0];
90
- }
91
- remove(id) {
92
- return __awaiter(this, void 0, void 0, function* () {
93
- const res = yield this.lrApollo.mutate({
94
- mutation: DeleteLbopQuery,
95
- variables: {
96
- input: {
97
- id,
98
- },
99
- },
100
- });
101
- return res.deleteLbop.id;
102
- });
103
- }
104
- update({ id, name }) {
105
- return __awaiter(this, void 0, void 0, function* () {
106
- const lbop = yield this.get(id);
107
- lbop.name = name;
108
- const masterKey = this.keyService.currentMasterKey;
109
- const cipherMeta = yield this.encryptionService.encrypt(masterKey.jwk, lbop);
110
- const res = yield this.lrApollo.mutate({
111
- mutation: UpdateLbopQuery,
112
- variables: {
113
- input: {
114
- id,
115
- cipherMeta: JSON.stringify(cipherMeta),
116
- },
117
- },
118
- });
119
- return res.updateLbop;
120
- });
121
- }
122
- get(id) {
123
- return __awaiter(this, void 0, void 0, function* () {
124
- const res = yield this.lrApollo.query({
125
- query: LbopQuery,
126
- variables: {
127
- id,
128
- },
129
- });
130
- const masterKey = this.keyService.currentMasterKey;
131
- const plainCipherMeta = yield this.encryptionService.decrypt(masterKey.jwk, JSON.parse(res.lbop.cipherMeta));
132
- return Object.assign({ id: res.id }, plainCipherMeta);
133
- });
134
- }
135
- list() {
136
- return __awaiter(this, void 0, void 0, function* () {
137
- const res = yield this.lrApollo.query({
138
- query: LbopsQuery,
139
- });
140
- const masterKey = yield this.keyService.currentMasterKey;
141
- return Promise.all(res.lbops.edges.map((edge) => __awaiter(this, void 0, void 0, function* () {
142
- const plainCipherMeta = yield this.encryptionService.decrypt(masterKey.jwk, JSON.parse(edge.node.cipherMeta));
143
- return Object.assign({ id: edge.node.id }, plainCipherMeta);
144
- })));
145
- });
146
- }
147
- create({ name }) {
148
- return __awaiter(this, void 0, void 0, function* () {
149
- if (Slip39Helper.WORD_LIST.length !== 1024) {
150
- throw new KcBadLogicException('Slip39Helper.WORD_LIST.length != 1024');
151
- }
152
- // Get existing to make sure there are not duplicate first words
153
- const lbops = yield this.list();
154
- // Generate new one
155
- let lbopString;
156
- // eslint-disable-next-line no-constant-condition
157
- while (true) {
158
- lbopString = this.keyFactory
159
- .randomChoices(Slip39Helper.WORD_LIST, this.LBOP_WORDS)
160
- .join(' ');
161
- const partial = this.getPartial(lbopString);
162
- if (!lbops.some((lbop) => lbop.partial === partial)) {
163
- break;
164
- }
165
- }
166
- const lbopKeyParams = yield this.keyFactory.createLbopKeyParams();
167
- const lbopKey = (yield this.keyFactory.deriveLbopKey(Object.assign({ password: yield this.keyFactory.importPassword(lbopString) }, lbopKeyParams))).jwk;
168
- const lbopKeyVerifier = yield this.keyFactory.createSignKey();
169
- const wrappedLbopKeyVerifier = yield this.encryptionService.encrypt(lbopKey, lbopKeyVerifier.toJSON(true));
170
- // Re-encrypt master key with new key
171
- const masterKey = this.keyService.currentMasterKey;
172
- const wrappedMasterKey = yield this.encryptionService.encrypt(lbopKey, masterKey.jwk.toJSON(true));
173
- const meta = Object.assign(Object.assign({}, (name && { name })), { partial: this.getPartial(lbopString) });
174
- const cipherMeta = yield this.encryptionService.encrypt(masterKey.jwk, meta);
175
- const res = yield this.lrApollo.mutate({
176
- mutation: CreateLbopQuery,
177
- variables: {
178
- input: {
179
- cipherMeta: JSON.stringify(cipherMeta),
180
- lbopKeyParams: JSON.stringify(lbopKeyParams),
181
- lbopKeyVerifier: JSON.stringify(lbopKeyVerifier.toJSON(true)),
182
- wrappedLbopKeyVerifier: JSON.stringify(wrappedLbopKeyVerifier),
183
- masterKeyId: masterKey.id,
184
- wrappedMasterKey: JSON.stringify(wrappedMasterKey),
185
- },
186
- },
187
- });
188
- return Object.assign(Object.assign({}, res.createLbop.lbop), { lbopString });
189
- });
190
- }
191
- // --------------------------------------------------------------------------------------------------------------------
192
- // --------------------------------------------------------------------------------------------------------------------
193
- // Flow below are for password reset via LBOP
194
- //
195
- // --Potential Failure Point xxx--
196
- //
197
- // Look for the above and you can test by interrupting at these points.
198
- //
199
- // The LBOP reset process can be restarted at any point before the call to "set-password/". Once "set-password/" has been
200
- // called, we assume the client has a short period of time to change the Idp password to the one they've chosen. The "set-password/"
201
- // will set the Idp password to a temporary random password. The user can no longer login using their current password. If the Idp
202
- // password change process does not complete or takes longer than the lockout period, the account will not be accessible and a new
203
- // LBOP password reset must be carried out.
204
- // --------------------------------------------------------------------------------------------------------------------
205
- // --------------------------------------------------------------------------------------------------------------------
206
- verifyLbops(challengeResult, lbopString) {
207
- return __awaiter(this, void 0, void 0, function* () {
208
- const clientNonce = this.keyFactory.randomString(this.CLIENT_NONCE_LENGTH);
209
- for (const lbop of challengeResult.lbops) {
210
- const lbopKey = (yield this.keyFactory.deriveLbopKey(Object.assign({ password: yield this.keyFactory.importPassword(lbopString) }, lbop.lbopKeyParams))).jwk;
211
- // If decoding successful then it's the correct lbop
212
- try {
213
- const lbopKeyVerifier = (yield this.encryptionService.decrypt(lbopKey, lbop.wrappedLbopKeyVerifier));
214
- // Force a bad signature.
215
- // const serverNonce = challengeResult.challenge.serverNonce + "1",
216
- const serverNonce = challengeResult.challenge.serverNonce;
217
- const signedChallenge = yield this.encryptionService.sign(lbopKeyVerifier, {
218
- serverNonce,
219
- clientNonce,
220
- });
221
- return {
222
- lbop,
223
- signedChallenge,
224
- lbopKey,
225
- };
226
- }
227
- catch (error) {
228
- continue;
229
- }
230
- }
231
- throw new KcException(new KcError({
232
- source: ERROR_SOURCE,
233
- code: KcLbopErrorCode.INVALID_PASSPHRASE,
234
- message: 'Invalid passphrase.',
235
- }));
236
- });
237
- }
238
- verifyContact(params) {
239
- return __awaiter(this, void 0, void 0, function* () {
240
- const ret = this.http
241
- .post(`${this.config.authUrl}users/lbop-reset/verify-contact/`, params)
242
- .toPromise();
243
- // --Potential Failure Point 1 --
244
- // The contact verifications are throttled. But otherwise harmless.
245
- return ret;
246
- });
247
- }
248
- confirmContact(params) {
249
- return __awaiter(this, void 0, void 0, function* () {
250
- return this.http
251
- .post(`${this.config.authUrl}cove/respond/`, {
252
- claim_id: params.claimId,
253
- v_code: params.vCode,
254
- })
255
- .toPromise();
256
- // --Potential Failure Point 2 --
257
- // A verified claim for a contact does not prevent new ones from being generated. So it should be fine to just start again.
258
- });
259
- }
260
- verify(params) {
261
- return __awaiter(this, void 0, void 0, function* () {
262
- const challengeResult = yield this.http
263
- .post(`${this.config.authUrl}users/lbop-reset/get-challenge/`, {
264
- claimId: params.claimId,
265
- claimToken: params.claimToken,
266
- })
267
- .toPromise();
268
- // --Potential Failure Point 3 --
269
- // This does not lock anything. A second call to "get-challenge/" will create a new challenge amd invalidate the first one.
270
- const { signedChallenge, lbop, lbopKey } = yield this.verifyLbops(challengeResult, params.lbop);
271
- const res = yield this.http
272
- .post(`${this.config.authUrl}users/lbop-reset/verify-challenge/`, {
273
- lbopId: lbop.lbopId,
274
- signedChallenge,
275
- })
276
- .toPromise();
277
- // --Potential Failure Point 4 --
278
- // This does not lock anything. So ok to restart.
279
- return {
280
- lbopId: lbop.lbopId,
281
- verifiedToken: res.verifiedToken,
282
- masterKeyId: res.masterKeyId,
283
- masterKey: yield KFS.asKey(yield this.encryptionService.decrypt(lbopKey, res.wrappedMasterKey)),
284
- };
285
- });
286
- }
287
- setPassword(params) {
288
- return __awaiter(this, void 0, void 0, function* () {
289
- // Generate the new password derived keys
290
- const passKeyBundle = yield this.passwordService.createPassKeyBundle(params.newPassword);
291
- // Re-encrypt master key with new key
292
- const newWrappedMasterKey = yield this.encryptionService.encrypt(passKeyBundle.passKey, params.masterKey.toJSON(true));
293
- const result = yield this.http
294
- .post(`${this.config.authUrl}users/lbop-reset/set-password/`, {
295
- lbopId: params.lbopId,
296
- verifiedToken: params.verifiedToken,
297
- masterKeyId: params.masterKeyId,
298
- newWrappedMasterKey,
299
- newPassKey: {
300
- passKeyParams: passKeyBundle.passKeyParams,
301
- passIdpParams: passKeyBundle.passIdpParams,
302
- passIdpVerifierPbk: passKeyBundle.passIdpVerifier.toJSON(),
303
- wrappedPassIdpVerifierPrk: passKeyBundle.wrappedPassIdpVerifierPrk,
304
- },
305
- })
306
- .toPromise();
307
- // --Potential Failure Point 5 --
308
- // A timed mutex is locked. The Idp password change must occur within a period of time.
309
- // If interrupted here, the user can not login with their old password again. They must
310
- // start the whole LBOP password reset process again.
311
- // This call will go through the LR proxy which is OK since the LR server knows
312
- // the temporary password anyway.
313
- let user = yield this.auth.signIn(result.username, result.idpPassword, {
314
- noProxy: 'true',
315
- });
316
- if (user.challengeName !== 'NEW_PASSWORD_REQUIRED') {
317
- throw new KcInternalErrorException('Expecting Cognito to have done a password reset.');
318
- }
319
- // --Potential Failure Point 6 --
320
- // Must restart the LBOP password reset process again.
321
- // Set new password on Idp
322
- user = yield this.auth.completeNewPassword(user, this.passwordService.getPassIdpString(passKeyBundle.passIdp), {});
323
- // --Potential Failure Point 7 --
324
- // Must restart the LBOP password reset process again.
325
- yield this.auth.signOut();
326
- return yield this.http
327
- .post(`${this.config.authUrl}users/lbop-reset/complete/`, {
328
- lbopId: params.lbopId,
329
- setPasswordToken: result.setPasswordToken,
330
- })
331
- .toPromise();
332
- });
333
- }
334
- }
335
- LbopService.ɵprov = i0.ɵɵdefineInjectable({ factory: function LbopService_Factory() { return new LbopService(i0.ɵɵinject(i1.KC_CONFIG), i0.ɵɵinject(i2.HttpClient), i0.ɵɵinject(i3.LrApolloService), i0.ɵɵinject(i4.AuthClass), i0.ɵɵinject(i5.KeyFactoryService), i0.ɵɵinject(i6.KeyService), i0.ɵɵinject(i7.EncryptionService), i0.ɵɵinject(i8.PasswordService)); }, token: LbopService, providedIn: "root" });
336
- LbopService.decorators = [
337
- { type: Injectable, args: [{
338
- providedIn: 'root',
339
- },] }
340
- ];
341
- LbopService.ctorParameters = () => [
342
- { type: undefined, decorators: [{ type: Inject, args: [KC_CONFIG,] }] },
343
- { type: HttpClient },
344
- { type: LrApolloService },
345
- { type: AuthClass },
346
- { type: KFS },
347
- { type: KeyService },
348
- { type: EncryptionService },
349
- { type: PasswordService }
350
- ];
351
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"lbop.service.js","sourceRoot":"","sources":["../../../../../../projects/core/src/lib/lbop/lbop.service.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable @typescript-eslint/no-explicit-any */\nimport { HttpClient } from '@angular/common/http';\nimport { Inject, Injectable } from '@angular/core';\nimport { AuthClass } from '@aws-amplify/auth/lib-esm/Auth';\nimport gql from 'graphql-tag';\nimport { JWK } from 'node-jose';\nimport { Slip39Helper } from 'slip39';\nimport { LrApolloService } from '../api/lr-apollo.service';\nimport { EncryptionService } from '../encryption/encryption.service';\nimport { KeyFactoryService as KFS } from '../key/key-factory.service';\nimport { KeyService } from '../key/key.service';\nimport { PassKeyParams } from '../key/key.types';\nimport { KcConfig, KC_CONFIG } from '../life-ready.config';\nimport { PasswordService } from '../password/password.service';\nimport {\n  KcBadLogicException,\n  KcError,\n  KcException,\n  KcInternalErrorException,\n} from '../_common/exceptions';\n\nexport const ERROR_SOURCE = 'LBOP';\nexport enum KcLbopErrorCode {\n  INVALID_PASSPHRASE = 'INVALID_PASSPHRASE',\n}\n\ninterface SetPasswordApiResult {\n  username: string;\n  idpPassword: string;\n  setPasswordToken: string;\n}\n\nexport interface SetPasswordParams {\n  lbopId: string;\n  newPassword: CryptoKey;\n  verifiedToken: string;\n  masterKeyId: string;\n  masterKey: JWK.Key;\n}\n\nexport interface VerifyContactParams {\n  email?: string;\n  phone?: string;\n}\n\nexport interface VerifyContactResult {\n  // The claim_id identifies the Email/SMS confirmation\n  claimId: string;\n}\n\nexport interface ConfirmContactParams {\n  claimId: string;\n  vCode: string;\n}\n\nexport interface ConfirmContactResult {\n  // The token to prove the client had the correct confirmation code.\n  token: string;\n}\n\nexport interface VerifyParams {\n  claimId: string;\n  claimToken: string;\n  lbop: string;\n}\n\nexport interface VerifyResult {\n  // userId: string;\n  lbopId: string;\n  verifiedToken: string;\n  masterKeyId: string;\n  masterKey: JWK.Key;\n}\n\nexport interface ChallengeResultLbop {\n  lbopId: string;\n  lbopKeyParams: PassKeyParams;\n  wrappedLbopKeyVerifier: any;\n}\n\nexport interface ChallengeResult {\n  challenge: {\n    serverNonce: string;\n  };\n  lbops: ChallengeResultLbop[];\n}\n\nexport interface Lbop {\n  id: string;\n  partial?: string;\n  name?: string;\n  lbopString?: string;\n}\n\nexport interface CreateLbopParams {\n  name?: string;\n}\n\ninterface CreateLbopQuery {\n  createLbop: {\n    lbop: Lbop;\n  };\n}\n\nexport const CreateLbopQuery = gql`\n  mutation CreateLbop($input: CreateLbopInput!) {\n    createLbop(input: $input) {\n      lbop {\n        id\n      }\n    }\n  }\n`;\n\ninterface DeleteLbopQuery {\n  deleteLbop: Lbop;\n}\n\nexport const DeleteLbopQuery = gql`\n  mutation DeleteLbop($input: DeleteLbopInput!) {\n    deleteLbop(input: $input) {\n      id\n    }\n  }\n`;\n\nexport interface UpdateLbopParams {\n  id: string;\n  name: string;\n}\n\ninterface UpdateLbopQuery {\n  updateLbop: Lbop;\n}\n\nexport const UpdateLbopQuery = gql`\n  mutation UpdateLbop($input: UpdateLbopInput!) {\n    updateLbop(input: $input) {\n      lbop {\n        id\n      }\n    }\n  }\n`;\n\nexport const LbopQuery = gql`\n  query Lbop($id: LrRelayIdInput!) {\n    lbop(id: $id) {\n      id\n      cipherMeta\n    }\n  }\n`;\n\ninterface LbopsQuery {\n  lbops: any;\n}\n\nexport const LbopsQuery = gql`\n  query Lbops {\n    lbops {\n      edges {\n        node {\n          id\n          cipherMeta\n        }\n      }\n    }\n  }\n`;\n\n@Injectable({\n  providedIn: 'root',\n})\nexport class LbopService {\n  private readonly CLIENT_NONCE_LENGTH = 32;\n  // There are 1024 words (10 bits), so 25 words should give ~256 bits of entropy.\n  private readonly LBOP_WORDS = 25;\n\n  constructor(\n    @Inject(KC_CONFIG) private config: KcConfig,\n    private http: HttpClient,\n    private lrApollo: LrApolloService,\n    private auth: AuthClass,\n    private keyFactory: KFS,\n    private keyService: KeyService,\n    private encryptionService: EncryptionService,\n    private passwordService: PasswordService\n  ) {}\n\n  private getPartial(lbopString: string): string {\n    return lbopString.split(' ')[0];\n  }\n\n  public async remove(id: string): Promise<string> {\n    const res = await this.lrApollo.mutate<any>({\n      mutation: DeleteLbopQuery,\n      variables: {\n        input: {\n          id,\n        },\n      },\n    });\n\n    return res.deleteLbop.id;\n  }\n\n  public async update({ id, name }: UpdateLbopParams): Promise<Lbop> {\n    const lbop = await this.get(id);\n    lbop.name = name;\n\n    const masterKey = this.keyService.currentMasterKey;\n    const cipherMeta = await this.encryptionService.encrypt(\n      masterKey.jwk,\n      lbop\n    );\n\n    const res = await this.lrApollo.mutate<UpdateLbopQuery>({\n      mutation: UpdateLbopQuery,\n      variables: {\n        input: {\n          id,\n          cipherMeta: JSON.stringify(cipherMeta),\n        },\n      },\n    });\n\n    return res.updateLbop;\n  }\n\n  public async get(id: string): Promise<Lbop> {\n    const res = await this.lrApollo.query<any>({\n      query: LbopQuery,\n      variables: {\n        id,\n      },\n    });\n\n    const masterKey = this.keyService.currentMasterKey;\n\n    const plainCipherMeta = await this.encryptionService.decrypt(\n      masterKey.jwk,\n      JSON.parse(res.lbop.cipherMeta)\n    );\n\n    return {\n      id: res.id,\n      ...plainCipherMeta,\n    };\n  }\n\n  public async list(): Promise<Lbop[]> {\n    const res = await this.lrApollo.query<LbopsQuery>({\n      query: LbopsQuery,\n    });\n\n    const masterKey = await this.keyService.currentMasterKey;\n\n    return Promise.all(\n      res.lbops.edges.map(async (edge) => {\n        const plainCipherMeta = await this.encryptionService.decrypt(\n          masterKey.jwk,\n          JSON.parse(edge.node.cipherMeta)\n        );\n        return {\n          id: edge.node.id,\n          ...plainCipherMeta,\n        };\n      })\n    );\n  }\n\n  public async create({ name }: CreateLbopParams): Promise<Lbop> {\n    if (Slip39Helper.WORD_LIST.length !== 1024) {\n      throw new KcBadLogicException('Slip39Helper.WORD_LIST.length != 1024');\n    }\n\n    // Get existing to make sure there are not duplicate first words\n    const lbops = await this.list();\n\n    // Generate new one\n    let lbopString: string;\n    // eslint-disable-next-line no-constant-condition\n    while (true) {\n      lbopString = this.keyFactory\n        .randomChoices(Slip39Helper.WORD_LIST, this.LBOP_WORDS)\n        .join(' ');\n      const partial = this.getPartial(lbopString);\n\n      if (!lbops.some((lbop) => lbop.partial === partial)) {\n        break;\n      }\n    }\n\n    const lbopKeyParams = await this.keyFactory.createLbopKeyParams();\n    const lbopKey = (\n      await this.keyFactory.deriveLbopKey({\n        password: await this.keyFactory.importPassword(lbopString),\n        ...lbopKeyParams,\n      })\n    ).jwk;\n\n    const lbopKeyVerifier = await this.keyFactory.createSignKey();\n    const wrappedLbopKeyVerifier = await this.encryptionService.encrypt(\n      lbopKey,\n      lbopKeyVerifier.toJSON(true)\n    );\n\n    // Re-encrypt master key with new key\n    const masterKey = this.keyService.currentMasterKey;\n    const wrappedMasterKey = await this.encryptionService.encrypt(\n      lbopKey,\n      masterKey.jwk.toJSON(true)\n    );\n\n    const meta = {\n      ...(name && { name }),\n      partial: this.getPartial(lbopString),\n    };\n    const cipherMeta = await this.encryptionService.encrypt(\n      masterKey.jwk,\n      meta\n    );\n\n    const res = await this.lrApollo.mutate<CreateLbopQuery>({\n      mutation: CreateLbopQuery,\n      variables: {\n        input: {\n          cipherMeta: JSON.stringify(cipherMeta),\n          lbopKeyParams: JSON.stringify(lbopKeyParams),\n          lbopKeyVerifier: JSON.stringify(lbopKeyVerifier.toJSON(true)),\n          wrappedLbopKeyVerifier: JSON.stringify(wrappedLbopKeyVerifier),\n          masterKeyId: masterKey.id,\n          wrappedMasterKey: JSON.stringify(wrappedMasterKey),\n        },\n      },\n    });\n\n    return {\n      ...res.createLbop.lbop,\n      lbopString,\n    };\n  }\n\n  // --------------------------------------------------------------------------------------------------------------------\n  // --------------------------------------------------------------------------------------------------------------------\n  // Flow below are for password reset via LBOP\n  //\n  // --Potential Failure Point xxx--\n  //\n  // Look for the above and you can test by interrupting at these points.\n  //\n  // The LBOP reset process can be restarted at any point before the call to \"set-password/\". Once \"set-password/\" has been\n  // called, we assume the client has a short period of time to change the Idp password to the one they've chosen. The \"set-password/\"\n  // will set the Idp password to a temporary random password. The user can no longer login using their current password. If the Idp\n  // password change process does not complete or takes longer than the lockout period, the account will not be accessible and a new\n  // LBOP password reset must be carried out.\n  // --------------------------------------------------------------------------------------------------------------------\n  // --------------------------------------------------------------------------------------------------------------------\n  private async verifyLbops(\n    challengeResult: ChallengeResult,\n    lbopString: string\n  ): Promise<{\n    lbop: ChallengeResultLbop;\n    signedChallenge: any;\n    lbopKey: JWK.Key;\n  }> {\n    const clientNonce = this.keyFactory.randomString(this.CLIENT_NONCE_LENGTH);\n\n    for (const lbop of challengeResult.lbops) {\n      const lbopKey = (\n        await this.keyFactory.deriveLbopKey({\n          password: await this.keyFactory.importPassword(lbopString),\n          ...lbop.lbopKeyParams,\n        })\n      ).jwk;\n\n      // If decoding successful then it's the correct lbop\n      try {\n        const lbopKeyVerifier = (await this.encryptionService.decrypt(\n          lbopKey,\n          lbop.wrappedLbopKeyVerifier\n        )) as any;\n\n        // Force a bad signature.\n        // const serverNonce = challengeResult.challenge.serverNonce + \"1\",\n\n        const serverNonce = challengeResult.challenge.serverNonce;\n\n        const signedChallenge = await this.encryptionService.sign(\n          lbopKeyVerifier,\n          {\n            serverNonce,\n            clientNonce,\n          }\n        );\n\n        return {\n          lbop,\n          signedChallenge,\n          lbopKey,\n        };\n      } catch (error) {\n        continue;\n      }\n    }\n    throw new KcException(\n      new KcError({\n        source: ERROR_SOURCE,\n        code: KcLbopErrorCode.INVALID_PASSPHRASE,\n        message: 'Invalid passphrase.',\n      })\n    );\n  }\n\n  public async verifyContact(\n    params: VerifyContactParams\n  ): Promise<VerifyContactResult> {\n    const ret = this.http\n      .post<VerifyContactResult>(\n        `${this.config.authUrl}users/lbop-reset/verify-contact/`,\n        params\n      )\n      .toPromise();\n\n    // --Potential Failure Point 1 --\n    // The contact verifications are throttled. But otherwise harmless.\n\n    return ret;\n  }\n\n  public async confirmContact(\n    params: ConfirmContactParams\n  ): Promise<ConfirmContactResult> {\n    return this.http\n      .post<ConfirmContactResult>(`${this.config.authUrl}cove/respond/`, {\n        claim_id: params.claimId,\n        v_code: params.vCode,\n      })\n      .toPromise();\n\n    // --Potential Failure Point 2 --\n    // A verified claim for a contact does not prevent new ones from being generated. So it should be fine to just start again.\n  }\n\n  public async verify(params: VerifyParams): Promise<VerifyResult> {\n    const challengeResult = await this.http\n      .post<ChallengeResult>(\n        `${this.config.authUrl}users/lbop-reset/get-challenge/`,\n        {\n          claimId: params.claimId,\n          claimToken: params.claimToken,\n        }\n      )\n      .toPromise();\n\n    // --Potential Failure Point 3 --\n    // This does not lock anything. A second call to \"get-challenge/\" will create a new challenge amd invalidate the first one.\n    const { signedChallenge, lbop, lbopKey } = await this.verifyLbops(\n      challengeResult,\n      params.lbop\n    );\n\n    const res = await this.http\n      .post<any>(`${this.config.authUrl}users/lbop-reset/verify-challenge/`, {\n        lbopId: lbop.lbopId,\n        signedChallenge,\n      })\n      .toPromise();\n\n    // --Potential Failure Point 4 --\n    // This does not lock anything. So ok to restart.\n\n    return {\n      lbopId: lbop.lbopId,\n      verifiedToken: res.verifiedToken,\n      masterKeyId: res.masterKeyId,\n      masterKey: await KFS.asKey(\n        await this.encryptionService.decrypt(lbopKey, res.wrappedMasterKey)\n      ),\n    };\n  }\n\n  public async setPassword(params: SetPasswordParams): Promise<any> {\n    // Generate the new password derived keys\n    const passKeyBundle = await this.passwordService.createPassKeyBundle(\n      params.newPassword\n    );\n\n    // Re-encrypt master key with new key\n    const newWrappedMasterKey = await this.encryptionService.encrypt(\n      passKeyBundle.passKey,\n      params.masterKey.toJSON(true)\n    );\n\n    const result = await this.http\n      .post<SetPasswordApiResult>(\n        `${this.config.authUrl}users/lbop-reset/set-password/`,\n        {\n          lbopId: params.lbopId,\n          verifiedToken: params.verifiedToken,\n          masterKeyId: params.masterKeyId,\n          newWrappedMasterKey,\n          newPassKey: {\n            passKeyParams: passKeyBundle.passKeyParams,\n            passIdpParams: passKeyBundle.passIdpParams,\n            passIdpVerifierPbk: passKeyBundle.passIdpVerifier.toJSON(),\n            wrappedPassIdpVerifierPrk: passKeyBundle.wrappedPassIdpVerifierPrk,\n          },\n        }\n      )\n      .toPromise();\n\n    // --Potential Failure Point 5 --\n    // A timed mutex is locked. The Idp password change must occur within a period of time.\n    // If interrupted here, the user can not login with their old password again. They must\n    // start the whole LBOP password reset process again.\n\n    // This call will go through the LR proxy which is OK since the LR server knows\n    // the temporary password anyway.\n    let user = await this.auth.signIn(result.username, result.idpPassword, {\n      noProxy: 'true',\n    });\n\n    if (user.challengeName !== 'NEW_PASSWORD_REQUIRED') {\n      throw new KcInternalErrorException(\n        'Expecting Cognito to have done a password reset.'\n      );\n    }\n\n    // --Potential Failure Point 6 --\n    // Must restart the LBOP password reset process again.\n\n    // Set new password on Idp\n    user = await this.auth.completeNewPassword(\n      user,\n      this.passwordService.getPassIdpString(passKeyBundle.passIdp),\n      {}\n    );\n\n    // --Potential Failure Point 7 --\n    // Must restart the LBOP password reset process again.\n\n    await this.auth.signOut();\n\n    return await this.http\n      .post<any>(`${this.config.authUrl}users/lbop-reset/complete/`, {\n        lbopId: params.lbopId,\n        setPasswordToken: result.setPasswordToken,\n      })\n      .toPromise();\n  }\n}\n"]}
@@ -1,96 +0,0 @@
1
- import { __awaiter } from "tslib";
2
- /* eslint-disable @typescript-eslint/no-explicit-any */
3
- import { InjectionToken } from '@angular/core';
4
- import { ApolloLink, createHttpLink, from, InMemoryCache, } from '@apollo/client/core';
5
- import { setContext } from '@apollo/client/link/context';
6
- import { RetryLink } from '@apollo/client/link/retry';
7
- import { KcErrorCode } from './_common/exceptions';
8
- import { getAccessJwtToken } from './_common/utils';
9
- // The injection token string is set to be the same as the const stand name since you
10
- // can possibly have different tokens with the same type (i.e. KcConfig). So it would not
11
- // be appropriate to use "KcConfig" as the token string.
12
- export const KC_CONFIG = new InjectionToken('KC_CONFIG');
13
- const RETRY_ERROR_CODES = [KcErrorCode.CONCURRENT_ACCESS];
14
- export function httpOptions(auth, config) {
15
- var _a, _b;
16
- return __awaiter(this, void 0, void 0, function* () {
17
- const token = yield getAccessJwtToken(auth);
18
- const debugUsername = (_a = config.debug) === null || _a === void 0 ? void 0 : _a.username;
19
- const debugRbacCacheMode = (_b = config.debug) === null || _b === void 0 ? void 0 : _b.rbacCacheMode;
20
- return {
21
- withCredentials: true,
22
- headers: Object.assign(Object.assign(Object.assign({}, (token && { authorization: `Bearer ${token}` })), (debugUsername && { 'x-kc-dev-user': debugUsername })), (debugRbacCacheMode && {
23
- 'x-kc-dev-rbac-cache-mode': debugRbacCacheMode,
24
- })),
25
- };
26
- });
27
- }
28
- export const configureApollo = (config, auth) => {
29
- const defaultOptions = {
30
- watchQuery: {
31
- fetchPolicy: 'no-cache',
32
- errorPolicy: 'all',
33
- },
34
- query: {
35
- fetchPolicy: 'no-cache',
36
- errorPolicy: 'all',
37
- },
38
- mutate: {
39
- fetchPolicy: 'no-cache',
40
- errorPolicy: 'all',
41
- },
42
- };
43
- const authLink = setContext((_, { headers }) => __awaiter(void 0, void 0, void 0, function* () {
44
- const options = yield httpOptions(auth, config);
45
- return Object.assign(Object.assign({}, options), { headers: Object.assign(Object.assign({}, headers), options.headers) });
46
- }));
47
- // We are only retrying on certain errors, like the CONCURRENT_ACCESS gql
48
- // error which indicates DB race condition. So can be safely retried.
49
- const retryIf = (error, _) => {
50
- // The RetryLink is called on network error as well, so we need to filter for GraphQL errors.
51
- if (error instanceof GraphQLErrorException) {
52
- if (error.errors.some((e) => { var _a; return RETRY_ERROR_CODES.includes((_a = e.extensions) === null || _a === void 0 ? void 0 : _a.code); })) {
53
- return true;
54
- }
55
- }
56
- return false;
57
- };
58
- const retryLink = new RetryLink({
59
- delay: {
60
- initial: 300,
61
- max: Infinity,
62
- jitter: true,
63
- },
64
- attempts: {
65
- max: 3,
66
- retryIf,
67
- },
68
- });
69
- class GraphQLErrorException extends Error {
70
- constructor(errors) {
71
- super(errors.map((e) => e.message).join(', '));
72
- this.errors = errors;
73
- }
74
- }
75
- // Throw exception on gql errors which effectively promotes it to a network
76
- // error, which can then be handled by the RetryLink.
77
- const promoteGqlErrors = new ApolloLink((operation, forward) => {
78
- return forward(operation).map((data) => {
79
- if (data && data.errors) {
80
- const errors = data.errors.filter((e) => { var _a; return RETRY_ERROR_CODES.includes((_a = e.extensions) === null || _a === void 0 ? void 0 : _a.code); });
81
- if (errors.length > 0) {
82
- throw new GraphQLErrorException(data.errors);
83
- }
84
- }
85
- return data;
86
- });
87
- });
88
- const httpLink = createHttpLink({
89
- uri: config.apolloUrl,
90
- // Sending the sessionid cookie so that the server can use session data when needed.
91
- // eg. setting the session encryption key.
92
- credentials: 'include',
93
- });
94
- return Object.assign({ link: from([retryLink, promoteGqlErrors, authLink, httpLink]), cache: new InMemoryCache(), defaultOptions }, config.apolloConfig);
95
- };
96
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"life-ready.config.js","sourceRoot":"","sources":["../../../../../projects/core/src/lib/life-ready.config.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable @typescript-eslint/no-explicit-any */\nimport { InjectionToken } from '@angular/core';\nimport {\n  ApolloClientOptions,\n  ApolloLink,\n  createHttpLink,\n  DefaultOptions,\n  from,\n  InMemoryCache,\n} from '@apollo/client/core';\nimport { setContext } from '@apollo/client/link/context';\nimport { RetryLink } from '@apollo/client/link/retry';\nimport { AuthClass } from '@aws-amplify/auth/lib-esm/Auth';\nimport { GraphQLError } from 'graphql';\nimport { KcErrorCode } from './_common/exceptions';\nimport { getAccessJwtToken } from './_common/utils';\n\n// The injection token string is set to be the same as the const stand name since you\n// can possibly have different tokens with the same type (i.e. KcConfig). So it would not\n// be appropriate to use \"KcConfig\" as the token string.\nexport const KC_CONFIG = new InjectionToken<KcConfig>('KC_CONFIG');\nconst RETRY_ERROR_CODES = [KcErrorCode.CONCURRENT_ACCESS];\n\nexport async function httpOptions(auth: AuthClass, config: KcConfig) {\n  const token = await getAccessJwtToken(auth);\n\n  const debugUsername = config.debug?.username;\n  const debugRbacCacheMode = config.debug?.rbacCacheMode;\n\n  return {\n    withCredentials: true,\n    headers: {\n      ...(token && { authorization: `Bearer ${token}` }),\n      ...(debugUsername && { 'x-kc-dev-user': debugUsername }),\n      ...(debugRbacCacheMode && {\n        'x-kc-dev-rbac-cache-mode': debugRbacCacheMode,\n      }),\n    },\n  };\n}\n\nexport interface KcConfig {\n  authUrl: string;\n  apiUrl: string;\n  apolloUrl: string;\n  userPoolId: string;\n  userPoolWebClientId: string;\n  apolloConfig?: ApolloClientOptions<any>;\n  // Mainly to allow localhost to not needing this. Default to falsy\n  disableSessionEncryptionKey?: boolean;\n  // There's no compiler level switch exclude variables from the interface.\n  // The only mechanism is to use replacement of files. But it's too cumbersome.\n  // So we use runtime check instead to ensure debug config is null in production.\n  debug?: {\n    username?: string;\n    rbacCacheMode?: string;\n  };\n}\n\nexport const configureApollo = (\n  config: KcConfig,\n  auth: AuthClass\n): ApolloClientOptions<any> => {\n  const defaultOptions: DefaultOptions = {\n    watchQuery: {\n      fetchPolicy: 'no-cache',\n      errorPolicy: 'all',\n    },\n    query: {\n      fetchPolicy: 'no-cache',\n      errorPolicy: 'all',\n    },\n    mutate: {\n      fetchPolicy: 'no-cache',\n      errorPolicy: 'all',\n    },\n  };\n\n  const authLink = setContext(async (_, { headers }) => {\n    const options = await httpOptions(auth, config);\n\n    return {\n      ...options,\n      headers: {\n        ...headers,\n        ...options.headers,\n      },\n    };\n  });\n\n  // We are only retrying on certain errors, like the CONCURRENT_ACCESS gql\n  // error which indicates DB race condition. So can be safely retried.\n  const retryIf = (error, _) => {\n    // The RetryLink is called on network error as well, so we need to filter for GraphQL errors.\n    if (error instanceof GraphQLErrorException) {\n      if (\n        error.errors.some((e) => RETRY_ERROR_CODES.includes(e.extensions?.code))\n      ) {\n        return true;\n      }\n    }\n\n    return false;\n  };\n\n  const retryLink = new RetryLink({\n    delay: {\n      initial: 300,\n      max: Infinity,\n      jitter: true,\n    },\n    attempts: {\n      max: 3,\n      retryIf,\n    },\n  });\n\n  class GraphQLErrorException extends Error {\n    constructor(public errors: readonly GraphQLError[]) {\n      super(errors.map((e) => e.message).join(', '));\n    }\n  }\n\n  // Throw exception on gql errors which effectively promotes it to a network\n  // error, which can then be handled by the RetryLink.\n  const promoteGqlErrors = new ApolloLink((operation, forward) => {\n    return forward(operation).map((data) => {\n      if (data && data.errors) {\n        const errors = data.errors.filter((e) =>\n          RETRY_ERROR_CODES.includes(e.extensions?.code)\n        );\n\n        if (errors.length > 0) {\n          throw new GraphQLErrorException(data.errors);\n        }\n      }\n      return data;\n    });\n  });\n\n  const httpLink = createHttpLink({\n    uri: config.apolloUrl,\n    // Sending the sessionid cookie so that the server can use session data when needed.\n    // eg. setting the session encryption key.\n    credentials: 'include',\n  });\n\n  return {\n    link: from([retryLink, promoteGqlErrors, authLink, httpLink]),\n    cache: new InMemoryCache(),\n    defaultOptions,\n    ...config.apolloConfig,\n  };\n};\n"]}
@@ -1,42 +0,0 @@
1
- import { HttpClientModule } from '@angular/common/http';
2
- import { APP_INITIALIZER, NgModule } from '@angular/core';
3
- import Auth from '@aws-amplify/auth';
4
- import { AuthClass } from '@aws-amplify/auth/lib-esm/Auth';
5
- import { NgIdleKeepaliveModule } from '@ng-idle/keepalive';
6
- import { APOLLO_OPTIONS } from 'apollo-angular';
7
- import { configureAmplifyAuth } from './auth/auth.config';
8
- import { configureApollo, KC_CONFIG } from './life-ready.config';
9
- export class LifeReadyModule {
10
- static forRoot(config) {
11
- return {
12
- ngModule: LifeReadyModule,
13
- providers: [
14
- {
15
- provide: KC_CONFIG,
16
- useValue: config,
17
- },
18
- {
19
- provide: AuthClass,
20
- useValue: Auth,
21
- },
22
- {
23
- provide: APP_INITIALIZER,
24
- useFactory: configureAmplifyAuth,
25
- deps: [KC_CONFIG, AuthClass],
26
- multi: true,
27
- },
28
- {
29
- provide: APOLLO_OPTIONS,
30
- useFactory: configureApollo,
31
- deps: [KC_CONFIG, AuthClass],
32
- },
33
- ],
34
- };
35
- }
36
- }
37
- LifeReadyModule.decorators = [
38
- { type: NgModule, args: [{
39
- imports: [HttpClientModule, NgIdleKeepaliveModule.forRoot()],
40
- },] }
41
- ];
42
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibGlmZS1yZWFkeS5tb2R1bGUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9jb3JlL3NyYy9saWIvbGlmZS1yZWFkeS5tb2R1bGUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLGdCQUFnQixFQUFFLE1BQU0sc0JBQXNCLENBQUM7QUFDeEQsT0FBTyxFQUFFLGVBQWUsRUFBdUIsUUFBUSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQy9FLE9BQU8sSUFBSSxNQUFNLG1CQUFtQixDQUFDO0FBQ3JDLE9BQU8sRUFBRSxTQUFTLEVBQUUsTUFBTSxnQ0FBZ0MsQ0FBQztBQUMzRCxPQUFPLEVBQUUscUJBQXFCLEVBQUUsTUFBTSxvQkFBb0IsQ0FBQztBQUMzRCxPQUFPLEVBQUUsY0FBYyxFQUFFLE1BQU0sZ0JBQWdCLENBQUM7QUFDaEQsT0FBTyxFQUFFLG9CQUFvQixFQUFFLE1BQU0sb0JBQW9CLENBQUM7QUFDMUQsT0FBTyxFQUFFLGVBQWUsRUFBWSxTQUFTLEVBQUUsTUFBTSxxQkFBcUIsQ0FBQztBQUszRSxNQUFNLE9BQU8sZUFBZTtJQUNuQixNQUFNLENBQUMsT0FBTyxDQUNuQixNQUFnQjtRQUVoQixPQUFPO1lBQ0wsUUFBUSxFQUFFLGVBQWU7WUFDekIsU0FBUyxFQUFFO2dCQUNUO29CQUNFLE9BQU8sRUFBRSxTQUFTO29CQUNsQixRQUFRLEVBQUUsTUFBTTtpQkFDakI7Z0JBQ0Q7b0JBQ0UsT0FBTyxFQUFFLFNBQVM7b0JBQ2xCLFFBQVEsRUFBRSxJQUFJO2lCQUNmO2dCQUNEO29CQUNFLE9BQU8sRUFBRSxlQUFlO29CQUN4QixVQUFVLEVBQUUsb0JBQW9CO29CQUNoQyxJQUFJLEVBQUUsQ0FBQyxTQUFTLEVBQUUsU0FBUyxDQUFDO29CQUM1QixLQUFLLEVBQUUsSUFBSTtpQkFDWjtnQkFDRDtvQkFDRSxPQUFPLEVBQUUsY0FBYztvQkFDdkIsVUFBVSxFQUFFLGVBQWU7b0JBQzNCLElBQUksRUFBRSxDQUFDLFNBQVMsRUFBRSxTQUFTLENBQUM7aUJBQzdCO2FBQ0Y7U0FDRixDQUFDO0lBQ0osQ0FBQzs7O1lBL0JGLFFBQVEsU0FBQztnQkFDUixPQUFPLEVBQUUsQ0FBQyxnQkFBZ0IsRUFBRSxxQkFBcUIsQ0FBQyxPQUFPLEVBQUUsQ0FBQzthQUM3RCIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEh0dHBDbGllbnRNb2R1bGUgfSBmcm9tICdAYW5ndWxhci9jb21tb24vaHR0cCc7XG5pbXBvcnQgeyBBUFBfSU5JVElBTElaRVIsIE1vZHVsZVdpdGhQcm92aWRlcnMsIE5nTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgQXV0aCBmcm9tICdAYXdzLWFtcGxpZnkvYXV0aCc7XG5pbXBvcnQgeyBBdXRoQ2xhc3MgfSBmcm9tICdAYXdzLWFtcGxpZnkvYXV0aC9saWItZXNtL0F1dGgnO1xuaW1wb3J0IHsgTmdJZGxlS2VlcGFsaXZlTW9kdWxlIH0gZnJvbSAnQG5nLWlkbGUva2VlcGFsaXZlJztcbmltcG9ydCB7IEFQT0xMT19PUFRJT05TIH0gZnJvbSAnYXBvbGxvLWFuZ3VsYXInO1xuaW1wb3J0IHsgY29uZmlndXJlQW1wbGlmeUF1dGggfSBmcm9tICcuL2F1dGgvYXV0aC5jb25maWcnO1xuaW1wb3J0IHsgY29uZmlndXJlQXBvbGxvLCBLY0NvbmZpZywgS0NfQ09ORklHIH0gZnJvbSAnLi9saWZlLXJlYWR5LmNvbmZpZyc7XG5cbkBOZ01vZHVsZSh7XG4gIGltcG9ydHM6IFtIdHRwQ2xpZW50TW9kdWxlLCBOZ0lkbGVLZWVwYWxpdmVNb2R1bGUuZm9yUm9vdCgpXSxcbn0pXG5leHBvcnQgY2xhc3MgTGlmZVJlYWR5TW9kdWxlIHtcbiAgcHVibGljIHN0YXRpYyBmb3JSb290KFxuICAgIGNvbmZpZzogS2NDb25maWdcbiAgKTogTW9kdWxlV2l0aFByb3ZpZGVyczxMaWZlUmVhZHlNb2R1bGU+IHtcbiAgICByZXR1cm4ge1xuICAgICAgbmdNb2R1bGU6IExpZmVSZWFkeU1vZHVsZSxcbiAgICAgIHByb3ZpZGVyczogW1xuICAgICAgICB7XG4gICAgICAgICAgcHJvdmlkZTogS0NfQ09ORklHLFxuICAgICAgICAgIHVzZVZhbHVlOiBjb25maWcsXG4gICAgICAgIH0sXG4gICAgICAgIHtcbiAgICAgICAgICBwcm92aWRlOiBBdXRoQ2xhc3MsXG4gICAgICAgICAgdXNlVmFsdWU6IEF1dGgsXG4gICAgICAgIH0sXG4gICAgICAgIHtcbiAgICAgICAgICBwcm92aWRlOiBBUFBfSU5JVElBTElaRVIsXG4gICAgICAgICAgdXNlRmFjdG9yeTogY29uZmlndXJlQW1wbGlmeUF1dGgsXG4gICAgICAgICAgZGVwczogW0tDX0NPTkZJRywgQXV0aENsYXNzXSxcbiAgICAgICAgICBtdWx0aTogdHJ1ZSxcbiAgICAgICAgfSxcbiAgICAgICAge1xuICAgICAgICAgIHByb3ZpZGU6IEFQT0xMT19PUFRJT05TLFxuICAgICAgICAgIHVzZUZhY3Rvcnk6IGNvbmZpZ3VyZUFwb2xsbyxcbiAgICAgICAgICBkZXBzOiBbS0NfQ09ORklHLCBBdXRoQ2xhc3NdLFxuICAgICAgICB9LFxuICAgICAgXSxcbiAgICB9O1xuICB9XG59XG4iXX0=
@@ -1,40 +0,0 @@
1
- import { gqlTyped } from '../_common/ast';
2
- export const LockFragment = gqlTyped `
3
- fragment LockFragment on LockField {
4
- created
5
- modified
6
- version
7
- expiryTime
8
- state
9
- }`;
10
- export const UpdateLockMutation = gqlTyped `
11
- mutation UpdateLockMutation($input: UpdateLockInput!) {
12
- updateLock(input: $input) {
13
- lock {
14
- ...LockFragment
15
- }
16
- }
17
- }
18
- ${LockFragment}
19
- `;
20
- export const AcquireLockMutation = gqlTyped `
21
- mutation AcquireLockMutation($input: AcquireLockInput!) {
22
- acquireLock(input: $input) {
23
- lock {
24
- ...LockFragment
25
- }
26
- }
27
- }
28
- ${LockFragment}
29
- `;
30
- export const ReleaseLockMutation = gqlTyped `
31
- mutation ReleaseLockMutation($input: ReleaseLockInput!) {
32
- releaseLock(input: $input) {
33
- lock {
34
- ...LockFragment
35
- }
36
- }
37
- }
38
- ${LockFragment}
39
- `;
40
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibG9jay5ncWwuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9jb3JlL3NyYy9saWIvbG9jay9sb2NrLmdxbC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFDQSxPQUFPLEVBQUUsUUFBUSxFQUFFLE1BQU0sZ0JBQWdCLENBQUM7QUFTMUMsTUFBTSxDQUFDLE1BQU0sWUFBWSxHQUFHLFFBQVEsQ0FBb0I7Ozs7Ozs7RUFPdEQsQ0FBQztBQU9ILE1BQU0sQ0FBQyxNQUFNLGtCQUFrQixHQUFHLFFBQVEsQ0FBMEI7Ozs7Ozs7O0VBUWxFLFlBQVk7Q0FDYixDQUFDO0FBT0YsTUFBTSxDQUFDLE1BQU0sbUJBQW1CLEdBQUcsUUFBUSxDQUEyQjs7Ozs7Ozs7RUFRcEUsWUFBWTtDQUNiLENBQUM7QUFPRixNQUFNLENBQUMsTUFBTSxtQkFBbUIsR0FBRyxRQUFRLENBQTJCOzs7Ozs7OztFQVFwRSxZQUFZO0NBQ2IsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IERhdGVUaW1lLCBMb2NrU3RhdGUsIFVVSUQgfSBmcm9tICcuLi9hcGkvdHlwZXMnO1xuaW1wb3J0IHsgZ3FsVHlwZWQgfSBmcm9tICcuLi9fY29tbW9uL2FzdCc7XG5cbmV4cG9ydCBpbnRlcmZhY2UgTG9ja0ZyYWdtZW50UmVzdWx0IHtcbiAgY3JlYXRlZDogRGF0ZVRpbWU7XG4gIG1vZGlmaWVkOiBEYXRlVGltZTtcbiAgdmVyc2lvbj86IFVVSUQ7XG4gIGV4cGlyeVRpbWU6IERhdGVUaW1lO1xuICBzdGF0ZTogTG9ja1N0YXRlO1xufVxuZXhwb3J0IGNvbnN0IExvY2tGcmFnbWVudCA9IGdxbFR5cGVkPExvY2tGcmFnbWVudFJlc3VsdD5gXG5mcmFnbWVudCBMb2NrRnJhZ21lbnQgb24gTG9ja0ZpZWxkIHtcbiAgY3JlYXRlZFxuICBtb2RpZmllZFxuICB2ZXJzaW9uXG4gIGV4cGlyeVRpbWVcbiAgc3RhdGVcbn1gO1xuXG5leHBvcnQgaW50ZXJmYWNlIFVwZGF0ZUxvY2tNdXRhdGlvblJlc3VsdCB7XG4gIHVwZGF0ZUxvY2s6IHtcbiAgICBsb2NrOiBMb2NrRnJhZ21lbnRSZXN1bHQ7XG4gIH07XG59XG5leHBvcnQgY29uc3QgVXBkYXRlTG9ja011dGF0aW9uID0gZ3FsVHlwZWQ8VXBkYXRlTG9ja011dGF0aW9uUmVzdWx0PmBcbm11dGF0aW9uIFVwZGF0ZUxvY2tNdXRhdGlvbigkaW5wdXQ6IFVwZGF0ZUxvY2tJbnB1dCEpIHtcbiAgdXBkYXRlTG9jayhpbnB1dDogJGlucHV0KSB7XG4gICAgbG9jayB7XG4gICAgICAuLi5Mb2NrRnJhZ21lbnRcbiAgICB9XG4gIH1cbn1cbiR7TG9ja0ZyYWdtZW50fVxuYDtcblxuZXhwb3J0IGludGVyZmFjZSBBY3F1aXJlTG9ja011dGF0aW9uUmVzdWx0IHtcbiAgYWNxdWlyZUxvY2s6IHtcbiAgICBsb2NrOiBMb2NrRnJhZ21lbnRSZXN1bHQ7XG4gIH07XG59XG5leHBvcnQgY29uc3QgQWNxdWlyZUxvY2tNdXRhdGlvbiA9IGdxbFR5cGVkPEFjcXVpcmVMb2NrTXV0YXRpb25SZXN1bHQ+YFxubXV0YXRpb24gQWNxdWlyZUxvY2tNdXRhdGlvbigkaW5wdXQ6IEFjcXVpcmVMb2NrSW5wdXQhKSB7XG4gIGFjcXVpcmVMb2NrKGlucHV0OiAkaW5wdXQpIHtcbiAgICBsb2NrIHtcbiAgICAgIC4uLkxvY2tGcmFnbWVudFxuICAgIH1cbiAgfVxufVxuJHtMb2NrRnJhZ21lbnR9XG5gO1xuXG5leHBvcnQgaW50ZXJmYWNlIFJlbGVhc2VMb2NrTXV0YXRpb25SZXN1bHQge1xuICByZWxlYXNlTG9jazoge1xuICAgIGxvY2s6IExvY2tGcmFnbWVudFJlc3VsdDtcbiAgfTtcbn1cbmV4cG9ydCBjb25zdCBSZWxlYXNlTG9ja011dGF0aW9uID0gZ3FsVHlwZWQ8UmVsZWFzZUxvY2tNdXRhdGlvblJlc3VsdD5gXG5tdXRhdGlvbiBSZWxlYXNlTG9ja011dGF0aW9uKCRpbnB1dDogUmVsZWFzZUxvY2tJbnB1dCEpIHtcbiAgcmVsZWFzZUxvY2soaW5wdXQ6ICRpbnB1dCkge1xuICAgIGxvY2sge1xuICAgICAgLi4uTG9ja0ZyYWdtZW50XG4gICAgfVxuICB9XG59XG4ke0xvY2tGcmFnbWVudH1cbmA7XG4iXX0=