@lifeready/core 1.0.21 → 1.0.23

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 (277) hide show
  1. package/bundles/lifeready-core.umd.js +10612 -10527
  2. package/bundles/lifeready-core.umd.js.map +1 -1
  3. package/bundles/lifeready-core.umd.min.js +1 -15
  4. package/bundles/lifeready-core.umd.min.js.map +1 -1
  5. package/esm2015/lib/_common/ast.js +2 -1
  6. package/esm2015/lib/_common/deferred-promise.js +1 -1
  7. package/esm2015/lib/_common/exceptions.js +1 -1
  8. package/esm2015/lib/_common/queries.gql.js +1 -1
  9. package/esm2015/lib/_common/run-outside-angular.js +3 -2
  10. package/esm2015/lib/_common/types.js +2 -1
  11. package/esm2015/lib/_common/utils.js +2 -1
  12. package/esm2015/lib/api/lr-apollo.service.js +4 -3
  13. package/esm2015/lib/api/lr-graphql/index.js +1 -1
  14. package/esm2015/lib/api/lr-graphql/lr-graphql.service.js +9 -8
  15. package/esm2015/lib/api/lr-graphql/lr-merged-mutation.js +5 -6
  16. package/esm2015/lib/api/lr-graphql/lr-mutation-base.js +1 -1
  17. package/esm2015/lib/api/lr-graphql/lr-mutation.js +3 -3
  18. package/esm2015/lib/api/lr-graphql/lr.service.js +1 -1
  19. package/esm2015/lib/api/query-processor/common-processors.service.js +4 -3
  20. package/esm2015/lib/api/query-processor/index.js +1 -1
  21. package/esm2015/lib/api/query-processor/query-processor.service.js +5 -4
  22. package/esm2015/lib/api/query-processor/tp-password-reset-processor.service.js +8 -7
  23. package/esm2015/lib/api/types/graphql.types.js +2 -1
  24. package/esm2015/lib/api/types/index.js +1 -1
  25. package/esm2015/lib/api/types/lr-graphql.types.js +1 -1
  26. package/esm2015/lib/auth/auth.config.js +1 -1
  27. package/esm2015/lib/auth/auth.gql.js +1 -28
  28. package/esm2015/lib/auth/auth.types.js +1 -1
  29. package/esm2015/lib/auth/life-ready-auth.service.js +36 -32
  30. package/esm2015/lib/category/category-meta.service.js +1 -1
  31. package/esm2015/lib/category/category.gql.js +3 -2
  32. package/esm2015/lib/category/category.service.js +9 -8
  33. package/esm2015/lib/category/category.types.js +1 -1
  34. package/esm2015/lib/contact-card/contact-card.gql.js +79 -0
  35. package/esm2015/lib/contact-card/contact-card.service.js +156 -0
  36. package/esm2015/lib/contact-card/contact-card2.gql.js +29 -0
  37. package/esm2015/lib/contact-card/contact-card2.service.js +103 -0
  38. package/esm2015/lib/encryption/encryption.service.js +190 -0
  39. package/esm2015/lib/file-upload/file-upload.service.js +74 -0
  40. package/esm2015/lib/file-upload/file-upload.types.js +2 -0
  41. package/esm2015/lib/idle/idle.service.js +168 -0
  42. package/esm2015/lib/idle/idle.types.js +7 -0
  43. package/esm2015/lib/item2/item2.gql.js +127 -0
  44. package/esm2015/lib/item2/item2.gql.private.js +23 -0
  45. package/esm2015/lib/item2/item2.service.js +519 -0
  46. package/esm2015/lib/item2/item2.types.js +2 -0
  47. package/esm2015/lib/key/key-factory.service.js +237 -0
  48. package/esm2015/lib/key/key-graph.service.js +300 -0
  49. package/esm2015/lib/key/key-meta.service.js +201 -0
  50. package/esm2015/lib/{cryptography → key}/key.service.js +4 -4
  51. package/esm2015/lib/key/key.types.js +11 -0
  52. package/esm2015/lib/key-exchange/key-exchange.gql.js +188 -0
  53. package/esm2015/lib/key-exchange/key-exchange.service.js +441 -0
  54. package/esm2015/lib/key-exchange/key-exchange.types.js +7 -0
  55. package/esm2015/lib/key-exchange/key-exchange2.gql.js +171 -0
  56. package/esm2015/lib/key-exchange/key-exchange2.service.js +500 -0
  57. package/esm2015/lib/lbop/lbop.service.js +357 -0
  58. package/esm2015/lib/life-ready.config.js +2 -1
  59. package/esm2015/lib/life-ready.module.js +2 -27
  60. package/esm2015/lib/lock/lock.gql.js +40 -0
  61. package/esm2015/lib/lock/lock.service.js +64 -0
  62. package/esm2015/lib/message/message.gql.js +32 -0
  63. package/esm2015/lib/message/message.service.js +118 -0
  64. package/esm2015/lib/message/message.types.js +2 -0
  65. package/esm2015/lib/notification/notification.gql.js +1 -1
  66. package/esm2015/lib/notification/notification.service.js +2 -2
  67. package/esm2015/lib/password/password.gql.js +28 -0
  68. package/esm2015/lib/password/password.service.js +316 -0
  69. package/esm2015/lib/persist/persist.service.js +181 -0
  70. package/esm2015/lib/plan/plan.gql.js +1 -1
  71. package/esm2015/lib/plan/plan.service.js +3 -2
  72. package/esm2015/lib/plan/plan.types.js +1 -1
  73. package/esm2015/lib/profile/profile-details.service.js +215 -0
  74. package/esm2015/lib/profile/profile.gql.js +98 -0
  75. package/esm2015/lib/profile/profile.service.js +170 -0
  76. package/esm2015/lib/profile/profile.types.js +34 -0
  77. package/esm2015/lib/record/record-attachment.service.js +16 -15
  78. package/esm2015/lib/record/record.gql.js +1 -1
  79. package/esm2015/lib/record/record.service.js +8 -8
  80. package/esm2015/lib/record/record.types.js +1 -1
  81. package/esm2015/lib/record-type/record-type.service.js +1 -1
  82. package/esm2015/lib/record-type/record-type.types.js +1 -1
  83. package/esm2015/lib/register/register.service.js +173 -0
  84. package/esm2015/lib/scenario/scenario.constants.js +1 -1
  85. package/esm2015/lib/scenario/scenario.controller.js +2 -2
  86. package/esm2015/lib/scenario/scenario.gql.js +1 -1
  87. package/esm2015/lib/scenario/scenario.private.gql.js +198 -0
  88. package/esm2015/lib/scenario/scenario.service.js +19 -17
  89. package/esm2015/lib/scenario/scenario.types.js +2 -1
  90. package/esm2015/lib/shared-contact-card/shared-contact-card.service.js +119 -0
  91. package/esm2015/lib/shared-contact-card/shared-contact-card2.gql.js +41 -0
  92. package/esm2015/lib/shared-contact-card/shared-contact-card2.service.js +117 -0
  93. package/esm2015/lib/slip39/slip39.service.js +167 -0
  94. package/esm2015/lib/time/time.service.js +146 -0
  95. package/esm2015/lib/tp-assembly/tp-assembly.js +365 -0
  96. package/esm2015/lib/tp-assembly/tp-assembly.private.gql.js +22 -0
  97. package/esm2015/lib/tp-assembly/tp-assembly.types.js +2 -0
  98. package/esm2015/lib/tp-password-reset/tp-password-reset-request.service.js +100 -0
  99. package/esm2015/lib/tp-password-reset/tp-password-reset-user.service.js +118 -0
  100. package/esm2015/lib/tp-password-reset/tp-password-reset.constants.js +4 -0
  101. package/esm2015/lib/tp-password-reset/tp-password-reset.controller.js +34 -0
  102. package/esm2015/lib/tp-password-reset/tp-password-reset.gql.js +74 -0
  103. package/esm2015/lib/tp-password-reset/tp-password-reset.private.gql.js +165 -0
  104. package/esm2015/lib/tp-password-reset/tp-password-reset.private.service.js +54 -0
  105. package/esm2015/lib/tp-password-reset/tp-password-reset.service.js +92 -0
  106. package/esm2015/lib/tp-password-reset/tp-password-reset.types.js +2 -0
  107. package/esm2015/lib/trusted-party/trusted-party.gql.js +148 -0
  108. package/esm2015/lib/trusted-party/trusted-party.service.js +327 -0
  109. package/esm2015/lib/trusted-party/trusted-party.types.js +41 -0
  110. package/esm2015/lib/trusted-party/trusted-party2.gql.js +64 -0
  111. package/esm2015/lib/trusted-party/trusted-party2.gql.private.js +25 -0
  112. package/esm2015/lib/trusted-party/trusted-party2.service.js +224 -0
  113. package/esm2015/lib/trusted-party/trusted-party2.types.js +2 -0
  114. package/esm2015/lib/two-factor/two-factor.service.js +74 -0
  115. package/esm2015/lib/user/user.gql.js +60 -0
  116. package/esm2015/lib/user/user.service.js +80 -0
  117. package/esm2015/lib/user/user.types.js +2 -0
  118. package/esm2015/lib/web-crypto/web-crypto.service.js +29 -0
  119. package/esm2015/lifeready-core.js +15 -13
  120. package/esm2015/public-api.js +49 -51
  121. package/fesm2015/lifeready-core.js +8764 -8737
  122. package/fesm2015/lifeready-core.js.map +1 -1
  123. package/lib/_common/types.d.ts +3 -1
  124. package/lib/_common/utils.d.ts +2 -2
  125. package/lib/api/lr-apollo.service.d.ts +2 -2
  126. package/lib/api/lr-graphql/lr-graphql.service.d.ts +26 -8
  127. package/lib/api/lr-graphql/lr-merged-mutation.d.ts +22 -4
  128. package/lib/api/lr-graphql/lr-mutation.d.ts +1 -2
  129. package/lib/api/query-processor/common-processors.service.d.ts +1 -1
  130. package/lib/api/query-processor/query-processor.service.d.ts +1 -1
  131. package/lib/api/query-processor/tp-password-reset-processor.service.d.ts +2 -2
  132. package/lib/api/types/lr-graphql.types.d.ts +14 -3
  133. package/lib/auth/auth.gql.d.ts +0 -3
  134. package/lib/auth/auth.types.d.ts +5 -5
  135. package/lib/auth/life-ready-auth.service.d.ts +13 -13
  136. package/lib/category/category.gql.d.ts +1 -1
  137. package/lib/category/category.service.d.ts +3 -3
  138. package/lib/{api → contact-card}/contact-card.service.d.ts +9 -9
  139. package/lib/contact-card/contact-card2.gql.d.ts +25 -0
  140. package/lib/contact-card/contact-card2.service.d.ts +64 -0
  141. package/lib/{cryptography → encryption}/encryption.service.d.ts +10 -9
  142. package/lib/{api/file.service.d.ts → file-upload/file-upload.service.d.ts} +5 -8
  143. package/lib/file-upload/file-upload.types.d.ts +5 -0
  144. package/lib/{auth → idle}/idle.service.d.ts +6 -6
  145. package/lib/{items2 → item2}/item2.gql.d.ts +16 -16
  146. package/lib/{items2 → item2}/item2.service.d.ts +34 -35
  147. package/lib/{cryptography → key}/key-factory.service.d.ts +4 -3
  148. package/lib/{cryptography → key}/key-graph.service.d.ts +6 -6
  149. package/lib/{cryptography → key}/key-meta.service.d.ts +1 -1
  150. package/lib/{cryptography → key}/key.service.d.ts +2 -2
  151. package/lib/{cryptography/cryptography.types.d.ts → key/key.types.d.ts} +13 -17
  152. package/lib/{api → key-exchange}/key-exchange.service.d.ts +5 -5
  153. package/lib/{api → key-exchange}/key-exchange.types.d.ts +4 -4
  154. package/lib/{api → key-exchange}/key-exchange2.gql.d.ts +1 -1
  155. package/lib/{api → key-exchange}/key-exchange2.service.d.ts +82 -29
  156. package/lib/{auth → lbop}/lbop.service.d.ts +7 -7
  157. package/lib/life-ready.config.d.ts +1 -1
  158. package/lib/{api → lock}/lock.gql.d.ts +1 -1
  159. package/lib/{api → lock}/lock.service.d.ts +1 -1
  160. package/lib/message/message.gql.d.ts +13 -0
  161. package/lib/message/message.service.d.ts +36 -0
  162. package/lib/message/message.types.d.ts +12 -0
  163. package/lib/notification/notification.service.d.ts +3 -2
  164. package/lib/password/password.gql.d.ts +3 -0
  165. package/lib/{auth → password}/password.service.d.ts +9 -9
  166. package/lib/{api → persist}/persist.service.d.ts +3 -3
  167. package/lib/plan/plan.service.d.ts +3 -2
  168. package/lib/plan/plan.types.d.ts +2 -1
  169. package/lib/{users → profile}/profile-details.service.d.ts +3 -3
  170. package/lib/{users → profile}/profile.gql.d.ts +2 -2
  171. package/lib/{users → profile}/profile.service.d.ts +6 -6
  172. package/lib/{users → profile}/profile.types.d.ts +3 -2
  173. package/lib/record/record-attachment.service.d.ts +6 -6
  174. package/lib/record/record.service.d.ts +3 -3
  175. package/lib/{auth → register}/register.service.d.ts +4 -4
  176. package/lib/scenario/scenario.controller.d.ts +1 -1
  177. package/lib/scenario/scenario.service.d.ts +105 -5
  178. package/lib/scenario/scenario.types.d.ts +1 -1
  179. package/lib/{api → shared-contact-card}/shared-contact-card.service.d.ts +9 -9
  180. package/lib/{api → shared-contact-card}/shared-contact-card2.gql.d.ts +1 -1
  181. package/lib/{api → shared-contact-card}/shared-contact-card2.service.d.ts +6 -6
  182. package/lib/{cryptography → slip39}/slip39.service.d.ts +0 -1
  183. package/lib/{trusted-parties → tp-assembly}/tp-assembly.d.ts +7 -7
  184. package/lib/{trusted-parties → tp-assembly}/tp-assembly.types.d.ts +3 -3
  185. package/lib/{trusted-parties → tp-password-reset}/tp-password-reset-request.service.d.ts +5 -9
  186. package/lib/{trusted-parties → tp-password-reset}/tp-password-reset-user.service.d.ts +7 -13
  187. package/lib/{trusted-parties → tp-password-reset}/tp-password-reset.controller.d.ts +1 -1
  188. package/lib/tp-password-reset/tp-password-reset.gql.d.ts +63 -0
  189. package/lib/{trusted-parties/tp-password-reset.gql.d.ts → tp-password-reset/tp-password-reset.private.gql.d.ts} +1 -63
  190. package/lib/tp-password-reset/tp-password-reset.private.service.d.ts +59 -0
  191. package/lib/{trusted-parties → tp-password-reset}/tp-password-reset.service.d.ts +6 -89
  192. package/lib/tp-password-reset/tp-password-reset.types.d.ts +40 -0
  193. package/lib/{trusted-parties → trusted-party}/trusted-party.service.d.ts +7 -7
  194. package/lib/{trusted-parties → trusted-party}/trusted-party.types.d.ts +2 -3
  195. package/lib/{trusted-parties → trusted-party}/trusted-party2.gql.d.ts +0 -22
  196. package/lib/trusted-party/trusted-party2.gql.private.d.ts +23 -0
  197. package/lib/{trusted-parties → trusted-party}/trusted-party2.service.d.ts +11 -35
  198. package/lib/trusted-party/trusted-party2.types.d.ts +12 -0
  199. package/lib/{users → user}/user.gql.d.ts +1 -1
  200. package/lib/{users → user}/user.service.d.ts +1 -1
  201. package/lib/{users → user}/user.types.d.ts +1 -1
  202. package/lifeready-core.d.ts +14 -12
  203. package/lifeready-core.metadata.json +1 -1
  204. package/package.json +2 -2
  205. package/public-api.d.ts +48 -50
  206. package/esm2015/lib/api/contact-card.gql.js +0 -79
  207. package/esm2015/lib/api/contact-card.service.js +0 -154
  208. package/esm2015/lib/api/contact-card2.gql.js +0 -60
  209. package/esm2015/lib/api/contact-card2.service.js +0 -103
  210. package/esm2015/lib/api/file.service.js +0 -74
  211. package/esm2015/lib/api/key-exchange.gql.js +0 -188
  212. package/esm2015/lib/api/key-exchange.service.js +0 -442
  213. package/esm2015/lib/api/key-exchange.types.js +0 -7
  214. package/esm2015/lib/api/key-exchange2.gql.js +0 -171
  215. package/esm2015/lib/api/key-exchange2.service.js +0 -480
  216. package/esm2015/lib/api/lock.gql.js +0 -40
  217. package/esm2015/lib/api/lock.service.js +0 -64
  218. package/esm2015/lib/api/message.service.js +0 -138
  219. package/esm2015/lib/api/persist.service.js +0 -181
  220. package/esm2015/lib/api/shared-contact-card.service.js +0 -119
  221. package/esm2015/lib/api/shared-contact-card2.gql.js +0 -41
  222. package/esm2015/lib/api/shared-contact-card2.service.js +0 -117
  223. package/esm2015/lib/api/time.service.js +0 -146
  224. package/esm2015/lib/auth/idle.service.js +0 -168
  225. package/esm2015/lib/auth/idle.types.js +0 -7
  226. package/esm2015/lib/auth/lbop.service.js +0 -355
  227. package/esm2015/lib/auth/password.service.js +0 -315
  228. package/esm2015/lib/auth/register.service.js +0 -172
  229. package/esm2015/lib/auth/two-factor.service.js +0 -74
  230. package/esm2015/lib/cryptography/cryptography.types.js +0 -11
  231. package/esm2015/lib/cryptography/encryption.service.js +0 -189
  232. package/esm2015/lib/cryptography/key-factory.service.js +0 -237
  233. package/esm2015/lib/cryptography/key-graph.service.js +0 -299
  234. package/esm2015/lib/cryptography/key-meta.service.js +0 -200
  235. package/esm2015/lib/cryptography/slip39.service.js +0 -169
  236. package/esm2015/lib/cryptography/web-crypto.service.js +0 -29
  237. package/esm2015/lib/items2/item2.gql.js +0 -127
  238. package/esm2015/lib/items2/item2.gql.private.js +0 -23
  239. package/esm2015/lib/items2/item2.service.js +0 -516
  240. package/esm2015/lib/items2/item2.types.js +0 -1
  241. package/esm2015/lib/scenario/scenario.gql.private.js +0 -198
  242. package/esm2015/lib/trusted-parties/tp-assembly.gql.private.js +0 -22
  243. package/esm2015/lib/trusted-parties/tp-assembly.js +0 -365
  244. package/esm2015/lib/trusted-parties/tp-assembly.types.js +0 -1
  245. package/esm2015/lib/trusted-parties/tp-password-reset-request.service.js +0 -113
  246. package/esm2015/lib/trusted-parties/tp-password-reset-user.service.js +0 -129
  247. package/esm2015/lib/trusted-parties/tp-password-reset.constants.js +0 -4
  248. package/esm2015/lib/trusted-parties/tp-password-reset.controller.js +0 -34
  249. package/esm2015/lib/trusted-parties/tp-password-reset.gql.js +0 -237
  250. package/esm2015/lib/trusted-parties/tp-password-reset.service.js +0 -95
  251. package/esm2015/lib/trusted-parties/trusted-party.gql.js +0 -148
  252. package/esm2015/lib/trusted-parties/trusted-party.service.js +0 -326
  253. package/esm2015/lib/trusted-parties/trusted-party.types.js +0 -41
  254. package/esm2015/lib/trusted-parties/trusted-party2.gql.js +0 -87
  255. package/esm2015/lib/trusted-parties/trusted-party2.service.js +0 -218
  256. package/esm2015/lib/users/profile-details.service.js +0 -214
  257. package/esm2015/lib/users/profile.gql.js +0 -97
  258. package/esm2015/lib/users/profile.service.js +0 -169
  259. package/esm2015/lib/users/profile.types.js +0 -34
  260. package/esm2015/lib/users/user.gql.js +0 -60
  261. package/esm2015/lib/users/user.service.js +0 -79
  262. package/esm2015/lib/users/user.types.js +0 -1
  263. package/lib/api/contact-card2.gql.d.ts +0 -34
  264. package/lib/api/contact-card2.service.d.ts +0 -50
  265. package/lib/api/message.service.d.ts +0 -59
  266. /package/lib/{api → contact-card}/contact-card.gql.d.ts +0 -0
  267. /package/lib/{auth → idle}/idle.types.d.ts +0 -0
  268. /package/lib/{items2 → item2}/item2.gql.private.d.ts +0 -0
  269. /package/lib/{items2 → item2}/item2.types.d.ts +0 -0
  270. /package/lib/{api → key-exchange}/key-exchange.gql.d.ts +0 -0
  271. /package/lib/scenario/{scenario.gql.private.d.ts → scenario.private.gql.d.ts} +0 -0
  272. /package/lib/{api → time}/time.service.d.ts +0 -0
  273. /package/lib/{trusted-parties/tp-assembly.gql.private.d.ts → tp-assembly/tp-assembly.private.gql.d.ts} +0 -0
  274. /package/lib/{trusted-parties → tp-password-reset}/tp-password-reset.constants.d.ts +0 -0
  275. /package/lib/{trusted-parties → trusted-party}/trusted-party.gql.d.ts +0 -0
  276. /package/lib/{auth → two-factor}/two-factor.service.d.ts +0 -0
  277. /package/lib/{cryptography → web-crypto}/web-crypto.service.d.ts +0 -0
@@ -0,0 +1,237 @@
1
+ import { __awaiter } from "tslib";
2
+ import { Injectable } from '@angular/core';
3
+ import { JWK } from 'node-jose';
4
+ import { WebCryptoService } from '../web-crypto/web-crypto.service';
5
+ import { LrBadArgumentException, LrSuspiciousException, } from '../_common/exceptions';
6
+ import * as i0 from "@angular/core";
7
+ import * as i1 from "../web-crypto/web-crypto.service";
8
+ export function sha256(message) {
9
+ return __awaiter(this, void 0, void 0, function* () {
10
+ // encode as UTF-8
11
+ const msgBuffer = new TextEncoder().encode(message);
12
+ // hash the message
13
+ const hashBuffer = yield crypto.subtle.digest('SHA-256', msgBuffer);
14
+ // convert ArrayBuffer to Array
15
+ const hashArray = Array.from(new Uint8Array(hashBuffer));
16
+ // convert bytes to hex string
17
+ const hashHex = hashArray
18
+ .map((b) => ('00' + b.toString(16)).slice(-2))
19
+ .join('');
20
+ return hashHex;
21
+ });
22
+ }
23
+ export class KeyFactoryService {
24
+ constructor(webCryptoService) {
25
+ this.webCryptoService = webCryptoService;
26
+ // Global keys store. Otherwise, each call to asKey creates a new keyStore.
27
+ // <AZ> Did not seem to improve speed.
28
+ // public static keyStore = JWK.createKeyStore();
29
+ // AZ: This can't be change easily. It's basically a PassK or PassIdp rotation.
30
+ // todo: we should eventually increase this periodically to match with Moore's law.
31
+ // The iterations for each key are kept by the server as well but we assume the value
32
+ // from the server is not trustworthy, so need to have minimum thresholds here.
33
+ // If creating new keys, these minimum are used.
34
+ this.MIN_PASS_IDP_PBKDF_ITER = 100000;
35
+ this.MIN_PASS_KEY_PBKDF_ITER = 100000;
36
+ this.MIN_LBOP_KEY_PBKDF_ITER = 100000;
37
+ // These are used as the default values. They must be larger than the minimum values.
38
+ this.DEFAULT_PASS_IDP_PBKDF_ITER = this.MIN_PASS_IDP_PBKDF_ITER;
39
+ this.DEFAULT_PASS_KEY_PBKDF_ITER = this.MIN_PASS_KEY_PBKDF_ITER;
40
+ this.DEFAULT_LBOP_KEY_PBKDF_ITER = this.MIN_LBOP_KEY_PBKDF_ITER;
41
+ this.crypto = this.webCryptoService.crypto;
42
+ }
43
+ static asKey(key, form, extras) {
44
+ // <AZ> Using a single global key store did not seem to improve speed.
45
+ // return KeyFactoryService.keyStore.add(key, form, extras);
46
+ return JWK.asKey(key, form, extras);
47
+ }
48
+ randomString(digits) {
49
+ if (digits <= 0) {
50
+ throw new LrBadArgumentException('digits <= 0');
51
+ }
52
+ const validChars = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789';
53
+ let array = new Uint32Array(digits);
54
+ this.crypto.getRandomValues(array);
55
+ array = array.map((x) => validChars.charCodeAt(x % validChars.length));
56
+ return String.fromCharCode.apply(null, array);
57
+ }
58
+ randomDigitsNoZeros(digits) {
59
+ return this.randomChoices([1, 2, 3, 4, 5, 6, 7, 8, 9], digits).join('');
60
+ }
61
+ randomChoices(array, chooseN) {
62
+ if (array.length <= 1) {
63
+ throw new LrBadArgumentException('array.length <= 0');
64
+ }
65
+ if (chooseN <= 0) {
66
+ throw new LrBadArgumentException('chooseN <= 0');
67
+ }
68
+ const values = new Uint32Array(chooseN);
69
+ this.crypto.getRandomValues(values);
70
+ const ret = [];
71
+ values.forEach((v) => ret.push(array[v % array.length]));
72
+ return ret;
73
+ }
74
+ createSalt() {
75
+ return this.randomString(16);
76
+ }
77
+ createKey() {
78
+ return __awaiter(this, void 0, void 0, function* () {
79
+ const key = yield this.crypto.subtle.generateKey({
80
+ name: 'AES-GCM',
81
+ length: 256,
82
+ }, true, // whether the key is extractable (i.e. can be used in exportKey)
83
+ ['encrypt', 'decrypt'] // must be ["encrypt", "decrypt"] or ["wrapKey", "unwrapKey"]
84
+ );
85
+ const jwk = yield this.crypto.subtle.exportKey('jwk', key);
86
+ // Removing the fields not needed by node-jose
87
+ delete jwk.ext;
88
+ delete jwk.key_ops;
89
+ return KeyFactoryService.asKey(jwk);
90
+ });
91
+ }
92
+ createSignKey() {
93
+ return __awaiter(this, void 0, void 0, function* () {
94
+ const key = yield this.crypto.subtle.generateKey({
95
+ name: 'HMAC',
96
+ hash: { name: 'SHA-512' },
97
+ }, true, ['sign', 'verify']);
98
+ const jwk = yield this.crypto.subtle.exportKey('jwk', key);
99
+ // Removing the fields not needed by node-jose
100
+ delete jwk.key_ops;
101
+ delete jwk.ext;
102
+ return KeyFactoryService.asKey(jwk);
103
+ });
104
+ }
105
+ createPkcKey() {
106
+ return __awaiter(this, void 0, void 0, function* () {
107
+ // node-jose is not using Forge properly. It should be calling the async version of
108
+ // pki.rsa.generateKeyPair() with a callback. Instead it calls the sync version. Webcrypto
109
+ // does not support sync version, so it uses the javascript implementation, which is way too slow.
110
+ // So we generate using webcrypto and import the key.
111
+ // Unfortunately Elliptical Curve is not supported by Webcrypto. So we have to settle for RSA.
112
+ const key = yield this.crypto.subtle.generateKey({
113
+ name: 'RSA-OAEP',
114
+ modulusLength: 2048,
115
+ // As per suggestion: https://developer.mozilla.org/en-US/docs/Web/API/RsaHashedKeyGenParams
116
+ publicExponent: new Uint8Array([0x01, 0x00, 0x01]),
117
+ hash: { name: 'SHA-256' },
118
+ }, true, // whether the key is extractable (i.e. can be used in exportKey)
119
+ ['encrypt', 'decrypt'] // must be ["encrypt", "decrypt"] or ["wrapKey", "unwrapKey"]
120
+ );
121
+ const jwk = yield this.crypto.subtle.exportKey('jwk', key.privateKey);
122
+ // Removing the fields not needed by node-jose
123
+ delete jwk.key_ops;
124
+ delete jwk.ext;
125
+ return KeyFactoryService.asKey(jwk);
126
+ });
127
+ }
128
+ createPkcSignKey() {
129
+ return __awaiter(this, void 0, void 0, function* () {
130
+ const key = yield this.crypto.subtle.generateKey({
131
+ name: 'RSASSA-PKCS1-v1_5',
132
+ modulusLength: 2048,
133
+ // As per suggestion: https://developer.mozilla.org/en-US/docs/Web/API/RsaHashedKeyGenParams
134
+ publicExponent: new Uint8Array([0x01, 0x00, 0x01]),
135
+ hash: { name: 'SHA-256' },
136
+ }, true, // whether the key is extractable (i.e. can be used in exportKey)
137
+ ['sign', 'verify'] // can be any combination of "sign" and "verify"
138
+ );
139
+ const jwk = yield this.crypto.subtle.exportKey('jwk', key.privateKey);
140
+ // Removing the fields not needed by node-jose
141
+ delete jwk.key_ops;
142
+ delete jwk.ext;
143
+ return KeyFactoryService.asKey(jwk);
144
+ });
145
+ }
146
+ deriveKey({ password, salt, iterations, kid, }) {
147
+ return __awaiter(this, void 0, void 0, function* () {
148
+ const enc = new TextEncoder();
149
+ const rawKey = yield this.crypto.subtle.importKey('raw', enc.encode(password), 'PBKDF2', false, ['deriveBits', 'deriveKey']);
150
+ const passKey = yield crypto.subtle.deriveKey({
151
+ name: 'PBKDF2',
152
+ salt: new TextEncoder().encode(salt),
153
+ iterations,
154
+ hash: 'SHA-256',
155
+ }, rawKey, { name: 'AES-GCM', length: 256 }, true, ['encrypt', 'decrypt']);
156
+ const passKeyJson = yield crypto.subtle.exportKey('jwk', passKey);
157
+ if (kid) {
158
+ passKeyJson.kid = kid;
159
+ }
160
+ const jwk = yield KeyFactoryService.asKey(passKeyJson);
161
+ return { jwk };
162
+ });
163
+ }
164
+ derivePassIdp(params) {
165
+ return __awaiter(this, void 0, void 0, function* () {
166
+ if (params.iterations < this.MIN_PASS_IDP_PBKDF_ITER) {
167
+ throw new LrSuspiciousException(`The number of PassIdp key derivation iterations sent from the server (${params.iterations}) is lower than the minimum (${this.MIN_PASS_IDP_PBKDF_ITER})`);
168
+ }
169
+ return this.deriveKey(params);
170
+ });
171
+ }
172
+ derivePassKey(params) {
173
+ return __awaiter(this, void 0, void 0, function* () {
174
+ if (params.iterations < this.MIN_PASS_KEY_PBKDF_ITER) {
175
+ throw new LrSuspiciousException(`The number of PassKey key derivation iterations sent from the server(${params.iterations}) is lower than the minimum(${this.MIN_PASS_KEY_PBKDF_ITER})`);
176
+ }
177
+ return this.deriveKey(params);
178
+ });
179
+ }
180
+ deriveLbopKey(params) {
181
+ return __awaiter(this, void 0, void 0, function* () {
182
+ if (params.iterations < this.MIN_LBOP_KEY_PBKDF_ITER) {
183
+ throw new LrSuspiciousException(`The number of LbopKey key derivation iterations sent from the server(${params.iterations}) is lower than the minimum(${this.MIN_LBOP_KEY_PBKDF_ITER})`);
184
+ }
185
+ return this.deriveKey(params);
186
+ });
187
+ }
188
+ createKid() {
189
+ return __awaiter(this, void 0, void 0, function* () {
190
+ // todo: AZ: node-jose source uses node's default UUID() function for kid, so just change to use that.
191
+ // for now, we are just creating a new key to use it's kid.
192
+ // The kid is a part of the JWK system. LR backend maintains the key hierarchy separately with it's own
193
+ // key id. But we just use it here as a double check.
194
+ return (yield this.createKey()).kid;
195
+ });
196
+ }
197
+ createPassIdpParams() {
198
+ return __awaiter(this, void 0, void 0, function* () {
199
+ return {
200
+ salt: this.createSalt(),
201
+ iterations: this.DEFAULT_PASS_IDP_PBKDF_ITER,
202
+ };
203
+ });
204
+ }
205
+ createPassKeyParams() {
206
+ return __awaiter(this, void 0, void 0, function* () {
207
+ return {
208
+ salt: this.createSalt(),
209
+ kid: yield this.createKid(),
210
+ iterations: this.DEFAULT_PASS_KEY_PBKDF_ITER,
211
+ };
212
+ });
213
+ }
214
+ createLbopKeyParams() {
215
+ return __awaiter(this, void 0, void 0, function* () {
216
+ return {
217
+ salt: this.createSalt(),
218
+ // todo: AZ: node-jose source uses node's default UUID() function for kid, so just change to use that.
219
+ // for now, we are just creating a new key to use it's kid.
220
+ // The kid is a part of the JWK system. LR backend maintains the key hierarchy separately with it's own
221
+ // key id. But we just use it here as a double check.
222
+ kid: yield this.createKid(),
223
+ iterations: this.DEFAULT_PASS_KEY_PBKDF_ITER,
224
+ };
225
+ });
226
+ }
227
+ }
228
+ KeyFactoryService.ɵprov = i0.ɵɵdefineInjectable({ factory: function KeyFactoryService_Factory() { return new KeyFactoryService(i0.ɵɵinject(i1.WebCryptoService)); }, token: KeyFactoryService, providedIn: "root" });
229
+ KeyFactoryService.decorators = [
230
+ { type: Injectable, args: [{
231
+ providedIn: 'root',
232
+ },] }
233
+ ];
234
+ KeyFactoryService.ctorParameters = () => [
235
+ { type: WebCryptoService }
236
+ ];
237
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"key-factory.service.js","sourceRoot":"","sources":["../../../../../../projects/core/src/lib/key/key-factory.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { JWK } from 'node-jose';\nimport { JSONObject } from '../api/types';\nimport { WebCryptoService } from '../web-crypto/web-crypto.service';\nimport {\n  LrBadArgumentException,\n  LrSuspiciousException,\n} from '../_common/exceptions';\nimport {\n  DeriveKeyResult,\n  DeriveLbopKeyParams,\n  DerivePassIdpParams,\n  DerivePassKeyParams,\n  LbopKeyParams,\n  PassIdpParams,\n  PassKeyParams,\n} from './key.types';\n\nexport async function sha256(message) {\n  // encode as UTF-8\n  const msgBuffer = new TextEncoder().encode(message);\n\n  // hash the message\n  const hashBuffer = await crypto.subtle.digest('SHA-256', msgBuffer);\n\n  // convert ArrayBuffer to Array\n  const hashArray = Array.from(new Uint8Array(hashBuffer));\n\n  // convert bytes to hex string\n  const hashHex = hashArray\n    .map((b) => ('00' + b.toString(16)).slice(-2))\n    .join('');\n  return hashHex;\n}\n\n@Injectable({\n  providedIn: 'root',\n})\nexport class KeyFactoryService {\n  constructor(private webCryptoService: WebCryptoService) {\n    this.crypto = this.webCryptoService.crypto;\n  }\n  private readonly crypto;\n  // Global keys store. Otherwise, each call to asKey creates a new keyStore.\n  // <AZ> Did not seem to improve speed.\n  // public static keyStore = JWK.createKeyStore();\n\n  // AZ: This can't be change easily. It's basically a PassK or PassIdp rotation.\n  // todo: we should eventually increase this periodically to match with Moore's law.\n  // The iterations for each key are kept by the server as well but we assume the value\n  // from the server is not trustworthy, so need to have minimum thresholds here.\n  // If creating new keys, these minimum are used.\n  public readonly MIN_PASS_IDP_PBKDF_ITER = 100000;\n  public readonly MIN_PASS_KEY_PBKDF_ITER = 100000;\n  public readonly MIN_LBOP_KEY_PBKDF_ITER = 100000;\n\n  // These are used as the default values. They must be larger than the minimum values.\n  public readonly DEFAULT_PASS_IDP_PBKDF_ITER = this.MIN_PASS_IDP_PBKDF_ITER;\n  public readonly DEFAULT_PASS_KEY_PBKDF_ITER = this.MIN_PASS_KEY_PBKDF_ITER;\n  public readonly DEFAULT_LBOP_KEY_PBKDF_ITER = this.MIN_LBOP_KEY_PBKDF_ITER;\n\n  static asKey(\n    key: string | Buffer | Record<string, JSONObject> | JWK.RawKey,\n    form?:\n      | 'json'\n      | 'private'\n      | 'pkcs8'\n      | 'public'\n      | 'spki'\n      | 'pkix'\n      | 'x509'\n      | 'pem',\n    extras?: Record<string, unknown>\n  ): Promise<JWK.Key> {\n    // <AZ> Using a single global key store did not seem to improve speed.\n    // return KeyFactoryService.keyStore.add(key, form, extras);\n    return JWK.asKey(key, form, extras);\n  }\n\n  randomString(digits: number): string {\n    if (digits <= 0) {\n      throw new LrBadArgumentException('digits <= 0');\n    }\n    const validChars =\n      'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789';\n    let array = new Uint32Array(digits);\n    this.crypto.getRandomValues(array);\n    array = array.map((x) => validChars.charCodeAt(x % validChars.length));\n    return String.fromCharCode.apply(null, array);\n  }\n\n  randomDigitsNoZeros(digits: number): string {\n    return this.randomChoices([1, 2, 3, 4, 5, 6, 7, 8, 9], digits).join('');\n  }\n\n  randomChoices<T>(array: T[], chooseN: number): T[] {\n    if (array.length <= 1) {\n      throw new LrBadArgumentException('array.length <= 0');\n    }\n    if (chooseN <= 0) {\n      throw new LrBadArgumentException('chooseN <= 0');\n    }\n    const values = new Uint32Array(chooseN);\n    this.crypto.getRandomValues(values);\n    const ret: T[] = [];\n    values.forEach((v) => ret.push(array[v % array.length]));\n    return ret;\n  }\n\n  createSalt(): string {\n    return this.randomString(16);\n  }\n\n  async createKey(): Promise<JWK.Key> {\n    const key = await this.crypto.subtle.generateKey(\n      {\n        name: 'AES-GCM',\n        length: 256, // can be  128, 192, or 256\n      },\n      true, // whether the key is extractable (i.e. can be used in exportKey)\n      ['encrypt', 'decrypt'] // must be [\"encrypt\", \"decrypt\"] or [\"wrapKey\", \"unwrapKey\"]\n    );\n\n    const jwk = await this.crypto.subtle.exportKey('jwk', key);\n\n    // Removing the fields not needed by node-jose\n    delete jwk.ext;\n    delete jwk.key_ops;\n\n    return KeyFactoryService.asKey(jwk);\n  }\n\n  async createSignKey(): Promise<JWK.Key> {\n    const key = await this.crypto.subtle.generateKey(\n      {\n        name: 'HMAC',\n        hash: { name: 'SHA-512' },\n      },\n      true,\n      ['sign', 'verify']\n    );\n\n    const jwk = await this.crypto.subtle.exportKey('jwk', key);\n\n    // Removing the fields not needed by node-jose\n    delete jwk.key_ops;\n    delete jwk.ext;\n\n    return KeyFactoryService.asKey(jwk);\n  }\n\n  async createPkcKey(): Promise<JWK.Key> {\n    // node-jose is not using Forge properly. It should be calling the async version of\n    // pki.rsa.generateKeyPair() with a callback. Instead it calls the sync version. Webcrypto\n    // does not support sync version, so it uses the javascript implementation, which is way too slow.\n    // So we generate using webcrypto and import the key.\n    // Unfortunately Elliptical Curve is not supported by Webcrypto. So we have to settle for RSA.\n    const key = await this.crypto.subtle.generateKey(\n      {\n        name: 'RSA-OAEP',\n        modulusLength: 2048, // can be 1024, 2048, 3072, 4096 ... 16384\n        // As per suggestion: https://developer.mozilla.org/en-US/docs/Web/API/RsaHashedKeyGenParams\n        publicExponent: new Uint8Array([0x01, 0x00, 0x01]),\n        hash: { name: 'SHA-256' }, // can be \"SHA-1\", \"SHA-256\", \"SHA-384\", or \"SHA-512\"\n      },\n      true, // whether the key is extractable (i.e. can be used in exportKey)\n      ['encrypt', 'decrypt'] // must be [\"encrypt\", \"decrypt\"] or [\"wrapKey\", \"unwrapKey\"]\n    );\n\n    const jwk = await this.crypto.subtle.exportKey('jwk', key.privateKey);\n    // Removing the fields not needed by node-jose\n    delete jwk.key_ops;\n    delete jwk.ext;\n\n    return KeyFactoryService.asKey(jwk);\n  }\n\n  async createPkcSignKey(): Promise<JWK.Key> {\n    const key = await this.crypto.subtle.generateKey(\n      {\n        name: 'RSASSA-PKCS1-v1_5',\n        modulusLength: 2048, // can be 1024, 2048, or 4096\n        // As per suggestion: https://developer.mozilla.org/en-US/docs/Web/API/RsaHashedKeyGenParams\n        publicExponent: new Uint8Array([0x01, 0x00, 0x01]),\n        hash: { name: 'SHA-256' }, // can be \"SHA-1\", \"SHA-256\", \"SHA-384\", or \"SHA-512\"\n      },\n      true, // whether the key is extractable (i.e. can be used in exportKey)\n      ['sign', 'verify'] // can be any combination of \"sign\" and \"verify\"\n    );\n\n    const jwk = await this.crypto.subtle.exportKey('jwk', key.privateKey);\n\n    // Removing the fields not needed by node-jose\n    delete jwk.key_ops;\n    delete jwk.ext;\n\n    return KeyFactoryService.asKey(jwk);\n  }\n\n  async deriveKey({\n    password,\n    salt,\n    iterations,\n    kid,\n  }: {\n    password: string;\n    salt: string;\n    iterations: number;\n    kid?: string;\n  }): Promise<DeriveKeyResult> {\n    const enc = new TextEncoder();\n    const rawKey = await this.crypto.subtle.importKey(\n      'raw',\n      enc.encode(password),\n      'PBKDF2',\n      false,\n      ['deriveBits', 'deriveKey']\n    );\n\n    const passKey = await crypto.subtle.deriveKey(\n      {\n        name: 'PBKDF2',\n        salt: new TextEncoder().encode(salt),\n        iterations,\n        hash: 'SHA-256',\n      },\n      rawKey,\n      { name: 'AES-GCM', length: 256 },\n      true,\n      ['encrypt', 'decrypt']\n    );\n\n    const passKeyJson: JSONObject = await crypto.subtle.exportKey(\n      'jwk',\n      passKey\n    );\n    if (kid) {\n      passKeyJson.kid = kid;\n    }\n\n    const jwk = await KeyFactoryService.asKey(passKeyJson);\n\n    return { jwk };\n  }\n\n  async derivePassIdp(params: DerivePassIdpParams): Promise<DeriveKeyResult> {\n    if (params.iterations < this.MIN_PASS_IDP_PBKDF_ITER) {\n      throw new LrSuspiciousException(\n        `The number of PassIdp key derivation iterations sent from the server (${params.iterations}) is lower than the minimum (${this.MIN_PASS_IDP_PBKDF_ITER})`\n      );\n    }\n    return this.deriveKey(params);\n  }\n\n  async derivePassKey(params: DerivePassKeyParams): Promise<DeriveKeyResult> {\n    if (params.iterations < this.MIN_PASS_KEY_PBKDF_ITER) {\n      throw new LrSuspiciousException(\n        `The number of PassKey key derivation iterations sent from the server(${params.iterations}) is lower than the minimum(${this.MIN_PASS_KEY_PBKDF_ITER})`\n      );\n    }\n    return this.deriveKey(params);\n  }\n\n  async deriveLbopKey(params: DeriveLbopKeyParams): Promise<DeriveKeyResult> {\n    if (params.iterations < this.MIN_LBOP_KEY_PBKDF_ITER) {\n      throw new LrSuspiciousException(\n        `The number of LbopKey key derivation iterations sent from the server(${params.iterations}) is lower than the minimum(${this.MIN_LBOP_KEY_PBKDF_ITER})`\n      );\n    }\n    return this.deriveKey(params);\n  }\n\n  async createKid(): Promise<string> {\n    // todo: AZ: node-jose source uses node's default UUID() function for kid, so just change to use that.\n    // for now, we are just creating a new key to use it's kid.\n    // The kid is a part of the JWK system. LR backend maintains the key hierarchy separately with it's own\n    // key id. But we just use it here as a double check.\n    return (await this.createKey()).kid;\n  }\n\n  async createPassIdpParams(): Promise<PassIdpParams> {\n    return {\n      salt: this.createSalt(),\n      iterations: this.DEFAULT_PASS_IDP_PBKDF_ITER,\n    };\n  }\n\n  async createPassKeyParams(): Promise<PassKeyParams> {\n    return {\n      salt: this.createSalt(),\n      kid: await this.createKid(),\n      iterations: this.DEFAULT_PASS_KEY_PBKDF_ITER,\n    };\n  }\n\n  async createLbopKeyParams(): Promise<LbopKeyParams> {\n    return {\n      salt: this.createSalt(),\n      // todo: AZ: node-jose source uses node's default UUID() function for kid, so just change to use that.\n      // for now, we are just creating a new key to use it's kid.\n      // The kid is a part of the JWK system. LR backend maintains the key hierarchy separately with it's own\n      // key id. But we just use it here as a double check.\n      kid: await this.createKid(),\n      iterations: this.DEFAULT_PASS_KEY_PBKDF_ITER,\n    };\n  }\n}\n"]}
@@ -0,0 +1,300 @@
1
+ import { __awaiter } from "tslib";
2
+ /* eslint-disable @typescript-eslint/no-explicit-any */
3
+ import { Injectable } from '@angular/core';
4
+ import graphlib, { Graph } from '@dagrejs/graphlib';
5
+ import _ from 'lodash';
6
+ import { asJwk, EncryptionService, isSymmetricKey, } from '../encryption/encryption.service';
7
+ import { LrBadArgumentException, LrEncryptionException, LrException, LrNotFoundException, } from '../_common/exceptions';
8
+ import { KeyFactoryService, KeyFactoryService as KFS, } from './key-factory.service';
9
+ import { KeyService } from './key.service';
10
+ import { KeyGraphEdgeType, KeyGraphNodeType, } from './key.types';
11
+ import * as i0 from "@angular/core";
12
+ import * as i1 from "../encryption/encryption.service";
13
+ import * as i2 from "./key.service";
14
+ import * as i3 from "./key-factory.service";
15
+ export class KeyGraphService {
16
+ // private keyCache: {
17
+ // [id: string]: Key;
18
+ // };
19
+ constructor(encryptionService, keyService, keyFactory) {
20
+ this.encryptionService = encryptionService;
21
+ this.keyService = keyService;
22
+ this.keyFactory = keyFactory;
23
+ this.purgeKeys();
24
+ }
25
+ purgeKeys() {
26
+ this.graph = new Graph();
27
+ // this.keyCache = null;
28
+ }
29
+ populateKeys(userKey) {
30
+ return __awaiter(this, void 0, void 0, function* () {
31
+ this.keyService.populateKeys({
32
+ passKey: userKey.passKey,
33
+ masterKey: yield this.keyService.loadMasterKey(userKey.masterKey.id),
34
+ rootKey: yield this.unwrapKey(userKey.masterKey.id, userKey.rootKey.id),
35
+ pxk: yield this.unwrapKey(userKey.masterKey.id, userKey.pxk.id),
36
+ sigPxk: yield this.unwrapKey(userKey.masterKey.id, userKey.sigPxk.id),
37
+ });
38
+ });
39
+ }
40
+ hasKey(keyId) {
41
+ return !!this.graph.node(keyId);
42
+ }
43
+ getNode(id, type) {
44
+ const node = this.graph.node(id);
45
+ if (!node) {
46
+ throw new LrNotFoundException(`Key graphs does not contain key id: ${id}`);
47
+ }
48
+ if (node.type !== type) {
49
+ throw new LrException({
50
+ message: `Key with id ${id} is not of type ${type}`,
51
+ });
52
+ }
53
+ return node.data;
54
+ }
55
+ key(id) {
56
+ return this.getNode(id, KeyGraphNodeType.Key);
57
+ }
58
+ passKey(id) {
59
+ return this.getNode(id, KeyGraphNodeType.PassKey);
60
+ }
61
+ addKeys(src) {
62
+ // Keys
63
+ if (src.keys) {
64
+ // What key graph returns can not be customized. So keys are essentially immutable.
65
+ // Therefore, if a key exists, there's no reason to update it.
66
+ for (const key of src.keys) {
67
+ // Note using Relay global id allows us to not worry about clashing node id
68
+ if (this.graph.hasNode(key.id)) {
69
+ continue;
70
+ }
71
+ const node = {
72
+ type: KeyGraphNodeType.Key,
73
+ data: _.cloneDeep(key),
74
+ };
75
+ this.graph.setNode(key.id, node);
76
+ }
77
+ }
78
+ // KeyLinks
79
+ if (src.keyLinks) {
80
+ for (const keyLink of src.keyLinks) {
81
+ if (this.graph.hasEdge(keyLink.wrappingKeyId, keyLink.keyId)) {
82
+ continue;
83
+ }
84
+ const edge = {
85
+ type: KeyGraphEdgeType.KeyLink,
86
+ data: _.cloneDeep(keyLink),
87
+ };
88
+ // Edge goes from wrapping key to wrapped key.
89
+ this.graph.setEdge(keyLink.wrappingKeyId, keyLink.keyId, edge);
90
+ }
91
+ }
92
+ // PassKeyLinks
93
+ if (src.passKeyLinks) {
94
+ for (const passKeyLink of src.passKeyLinks) {
95
+ if (this.graph.hasEdge(passKeyLink.passKeyId, passKeyLink.keyId)) {
96
+ continue;
97
+ }
98
+ const edge = {
99
+ type: KeyGraphEdgeType.PassKeyLink,
100
+ data: _.cloneDeep(passKeyLink),
101
+ };
102
+ // Edge goes from wrapping key to wrapped key.
103
+ this.graph.setEdge(passKeyLink.passKeyId, passKeyLink.keyId, edge);
104
+ }
105
+ }
106
+ // The graph is the single source of truth. These are lazily calculated.
107
+ // this.keyCache = null;
108
+ }
109
+ tracePath(distances, keyId) {
110
+ // The node label is the same as the id of the key nodes.
111
+ const ret = [];
112
+ let node = keyId;
113
+ if (!distances[node].predecessor) {
114
+ return null;
115
+ }
116
+ while (distances[node].predecessor) {
117
+ const child = distances[node].predecessor;
118
+ ret.push(this.graph.edge(child, node));
119
+ node = child;
120
+ }
121
+ // After reverse, the first element is the passkey
122
+ ret.reverse();
123
+ return ret;
124
+ }
125
+ getPath(knownKeyId, keyId) {
126
+ if (!knownKeyId || typeof knownKeyId !== 'string') {
127
+ throw new LrEncryptionException(`Param knownKeyId wrong format: ${knownKeyId}`);
128
+ }
129
+ if (!keyId || typeof keyId !== 'string') {
130
+ throw new LrEncryptionException(`Param keyId wrong format: ${keyId}`);
131
+ }
132
+ // => { A: { distance: 0 },
133
+ // B: { distance: 6, predecessor: 'C' },
134
+ // C: { distance: 4, predecessor: 'A' },
135
+ // D: { distance: 2, predecessor: 'A' },
136
+ // E: { distance: 8, predecessor: 'F' },
137
+ // F: { distance: 4, predecessor: 'D' } }
138
+ const distances = graphlib.alg.dijkstra(this.graph, knownKeyId);
139
+ // Trace path from keyId to knownKeyId
140
+ return this.tracePath(distances, keyId);
141
+ }
142
+ getJwkKey(keyOrId, getKeyIdCallback) {
143
+ return __awaiter(this, void 0, void 0, function* () {
144
+ return (yield this.getKey(keyOrId, getKeyIdCallback)).jwk;
145
+ });
146
+ }
147
+ // We assume that when a keyId is fetched, the key graph
148
+ // for the key is also returned and merged into the client-side
149
+ // key graph. By insisting a keyId is returned instead of the
150
+ // actual key we ensure key-graph is consistent.
151
+ getKey(keyOrId, getKeyIdCallback) {
152
+ return __awaiter(this, void 0, void 0, function* () {
153
+ let keyId = typeof keyOrId === 'string' ? keyOrId : keyOrId === null || keyOrId === void 0 ? void 0 : keyOrId.id;
154
+ if (!this.hasKey(keyId) && getKeyIdCallback) {
155
+ keyId = yield getKeyIdCallback();
156
+ }
157
+ // else, continue and let it fail.
158
+ const key = this.key(keyId);
159
+ if (key.jwk) {
160
+ return key;
161
+ }
162
+ else {
163
+ return this.unwrapKey(this.keyService.getCurrentMasterKey().id, keyId);
164
+ }
165
+ });
166
+ }
167
+ _unwrapLink(wrappingKey, link, dstKey) {
168
+ return __awaiter(this, void 0, void 0, function* () {
169
+ // console.log("_unwrapLink:", link.data.keyId);
170
+ const wrappedKey = JSON.parse(link.data.wrappedKey);
171
+ // Signatures of keys contain the key itself. This way we only need
172
+ // to access the KeyLinks to decrypt/verify keys.
173
+ let nextRawKey;
174
+ if (wrappedKey.signatures) {
175
+ nextRawKey = yield this.encryptionService.verify(wrappingKey, wrappedKey);
176
+ }
177
+ else {
178
+ nextRawKey = yield this.encryptionService.decrypt(wrappingKey, wrappedKey);
179
+ }
180
+ dstKey.jwk = yield KFS.asKey(nextRawKey);
181
+ dstKey.task = null;
182
+ });
183
+ }
184
+ _unwrap(key, path) {
185
+ return __awaiter(this, void 0, void 0, function* () {
186
+ for (const link of path) {
187
+ const dstKey = this.key(link.data.keyId);
188
+ // console.log("key: ", link.data.keyId);
189
+ if (dstKey.jwk) {
190
+ key = dstKey.jwk;
191
+ // console.log("Returning cached key: ", link.data.keyId);
192
+ continue;
193
+ }
194
+ if (!dstKey.task) {
195
+ dstKey.task = this._unwrapLink(key, link, dstKey);
196
+ }
197
+ yield dstKey.task;
198
+ key = dstKey.jwk;
199
+ }
200
+ return key;
201
+ });
202
+ }
203
+ unwrapWithPassKey(passKeyId, passKey, keyId) {
204
+ return __awaiter(this, void 0, void 0, function* () {
205
+ // Get path of the directory key.
206
+ const path = this.getPath(passKeyId, keyId);
207
+ return {
208
+ id: keyId,
209
+ jwk: yield this._unwrap(passKey, path),
210
+ };
211
+ });
212
+ }
213
+ unwrapKey(masterKeyId, keyId) {
214
+ return __awaiter(this, void 0, void 0, function* () {
215
+ // The first key should be a masterKey
216
+ const masterKey = yield this.keyService.loadMasterKey(masterKeyId);
217
+ if (masterKeyId === keyId) {
218
+ return masterKey;
219
+ }
220
+ // Get path of the directory key.
221
+ const path = this.getPath(masterKey.id, keyId);
222
+ return {
223
+ id: keyId,
224
+ jwk: yield this._unwrap(masterKey.jwk, path),
225
+ };
226
+ });
227
+ }
228
+ decryptFromString(keyOrId, cipherData, options) {
229
+ return __awaiter(this, void 0, void 0, function* () {
230
+ if (cipherData) {
231
+ const key = yield this.getJwkKey(keyOrId);
232
+ return (yield this.encryptionService.decrypt(key, JSON.parse(cipherData), options));
233
+ }
234
+ return null;
235
+ });
236
+ }
237
+ decryptFile(keyId, file) {
238
+ return __awaiter(this, void 0, void 0, function* () {
239
+ const key = yield this.getJwkKey(keyId);
240
+ return (yield this.encryptionService.decrypt(key, file, {
241
+ payloadType: 'ArrayBuffer',
242
+ }));
243
+ });
244
+ }
245
+ // TODO rename this to encrypt() and use as the most common usecase
246
+ encryptToString(key, content) {
247
+ return __awaiter(this, void 0, void 0, function* () {
248
+ // Empty string should be encrypted since you want to clear the field.
249
+ // Null is not encrypted because it's not valid JSON in the old JSON spec. Use
250
+ // empty string instead. It'll function as a logic false as well.
251
+ // Note that passing in empty string means it'll be encrypted which verifies
252
+ // it's integrity. But we still want to have a way to set the DB field
253
+ // to NULL, so we explicitly return null when content == null. A null
254
+ // variable in graphql mutation on KC server clears the field to NULL.
255
+ if (content == null) {
256
+ return null;
257
+ }
258
+ const jwk = asJwk(key) || (yield this.getJwkKey(key));
259
+ return this.encryptionService.encryptToString(jwk, content);
260
+ });
261
+ }
262
+ // Wraps a symmetric encryption key.
263
+ // Throws exception if wrapping public keys.
264
+ wrapKey(wrappingKey, key) {
265
+ return __awaiter(this, void 0, void 0, function* () {
266
+ if (!isSymmetricKey(key)) {
267
+ throw new LrBadArgumentException('Only allowing wrapping of symmetric keys.');
268
+ }
269
+ return this.encryptToString(wrappingKey, key.toJSON(true));
270
+ });
271
+ }
272
+ // TODO
273
+ // async wrapPublicKey<T>();
274
+ // async wrapPrivateKey<T>();
275
+ encryptWithNewKey(wrappingKeyId, cipherClearJson) {
276
+ return __awaiter(this, void 0, void 0, function* () {
277
+ const key = yield this.keyFactory.createKey();
278
+ const wrappedKey = yield this.encryptToString(wrappingKeyId, key.toJSON(true));
279
+ const cipher = yield this.encryptToString(key, cipherClearJson);
280
+ return {
281
+ key,
282
+ wrappingKeyId,
283
+ wrappedKey,
284
+ cipher,
285
+ };
286
+ });
287
+ }
288
+ }
289
+ KeyGraphService.ɵprov = i0.ɵɵdefineInjectable({ factory: function KeyGraphService_Factory() { return new KeyGraphService(i0.ɵɵinject(i1.EncryptionService), i0.ɵɵinject(i2.KeyService), i0.ɵɵinject(i3.KeyFactoryService)); }, token: KeyGraphService, providedIn: "root" });
290
+ KeyGraphService.decorators = [
291
+ { type: Injectable, args: [{
292
+ providedIn: 'root',
293
+ },] }
294
+ ];
295
+ KeyGraphService.ctorParameters = () => [
296
+ { type: EncryptionService },
297
+ { type: KeyService },
298
+ { type: KeyFactoryService }
299
+ ];
300
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"key-graph.service.js","sourceRoot":"","sources":["../../../../../../projects/core/src/lib/key/key-graph.service.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable @typescript-eslint/no-explicit-any */\nimport { Injectable } from '@angular/core';\nimport graphlib, { Graph } from '@dagrejs/graphlib';\nimport _ from 'lodash';\nimport { JWK } from 'node-jose';\nimport { JSONObject } from '../api/types';\nimport {\n  asJwk,\n  DecryptOptions,\n  EncryptionService,\n  isSymmetricKey,\n} from '../encryption/encryption.service';\nimport { CurrentUserKey } from '../profile/profile.types';\nimport {\n  LrBadArgumentException,\n  LrEncryptionException,\n  LrException,\n  LrNotFoundException,\n} from '../_common/exceptions';\nimport {\n  KeyFactoryService,\n  KeyFactoryService as KFS,\n} from './key-factory.service';\nimport { KeyService } from './key.service';\nimport {\n  Key,\n  KeyGraphEdge,\n  KeyGraphEdgeType,\n  KeyGraphNode,\n  KeyGraphNodeType,\n  KeyGraphResponse,\n  PassKey,\n} from './key.types';\n\nexport interface GraphKey extends Key {\n  task?: Promise<any>;\n}\n\n@Injectable({\n  providedIn: 'root',\n})\nexport class KeyGraphService {\n  private graph: Graph;\n  // private keyCache: {\n  //   [id: string]: Key;\n  // };\n\n  constructor(\n    private encryptionService: EncryptionService,\n    private keyService: KeyService,\n    private keyFactory: KeyFactoryService\n  ) {\n    this.purgeKeys();\n  }\n\n  purgeKeys() {\n    this.graph = new Graph();\n    // this.keyCache = null;\n  }\n\n  async populateKeys(userKey: CurrentUserKey) {\n    this.keyService.populateKeys({\n      passKey: userKey.passKey,\n      masterKey: await this.keyService.loadMasterKey(userKey.masterKey.id),\n      rootKey: await this.unwrapKey(userKey.masterKey.id, userKey.rootKey.id),\n      pxk: await this.unwrapKey(userKey.masterKey.id, userKey.pxk.id),\n      sigPxk: await this.unwrapKey(userKey.masterKey.id, userKey.sigPxk.id),\n    });\n  }\n\n  hasKey(keyId: string) {\n    return !!this.graph.node(keyId);\n  }\n\n  private getNode(id, type): GraphKey | PassKey {\n    const node = this.graph.node(id);\n    if (!node) {\n      throw new LrNotFoundException(\n        `Key graphs does not contain key id: ${id}`\n      );\n    }\n    if (node.type !== type) {\n      throw new LrException({\n        message: `Key with id ${id} is not of type ${type}`,\n      });\n    }\n    return node.data;\n  }\n\n  key(id): GraphKey {\n    return this.getNode(id, KeyGraphNodeType.Key);\n  }\n\n  passKey(id): PassKey {\n    return this.getNode(id, KeyGraphNodeType.PassKey);\n  }\n\n  addKeys(src: KeyGraphResponse) {\n    // Keys\n    if (src.keys) {\n      // What key graph returns can not be customized. So keys are essentially immutable.\n      // Therefore, if a key exists, there's no reason to update it.\n      for (const key of src.keys) {\n        // Note using Relay global id allows us to not worry about clashing node id\n        if (this.graph.hasNode(key.id)) {\n          continue;\n        }\n\n        const node: KeyGraphNode = {\n          type: KeyGraphNodeType.Key,\n          data: _.cloneDeep(key),\n        };\n\n        this.graph.setNode(key.id, node);\n      }\n    }\n\n    // KeyLinks\n    if (src.keyLinks) {\n      for (const keyLink of src.keyLinks) {\n        if (this.graph.hasEdge(keyLink.wrappingKeyId, keyLink.keyId)) {\n          continue;\n        }\n\n        const edge: KeyGraphEdge = {\n          type: KeyGraphEdgeType.KeyLink,\n          data: _.cloneDeep(keyLink),\n        };\n        // Edge goes from wrapping key to wrapped key.\n        this.graph.setEdge(keyLink.wrappingKeyId, keyLink.keyId, edge);\n      }\n    }\n\n    // PassKeyLinks\n    if (src.passKeyLinks) {\n      for (const passKeyLink of src.passKeyLinks) {\n        if (this.graph.hasEdge(passKeyLink.passKeyId, passKeyLink.keyId)) {\n          continue;\n        }\n\n        const edge: KeyGraphEdge = {\n          type: KeyGraphEdgeType.PassKeyLink,\n          data: _.cloneDeep(passKeyLink),\n        };\n        // Edge goes from wrapping key to wrapped key.\n        this.graph.setEdge(passKeyLink.passKeyId, passKeyLink.keyId, edge);\n      }\n    }\n\n    // The graph is the single source of truth. These are lazily calculated.\n    // this.keyCache = null;\n  }\n\n  tracePath(distances, keyId: string): KeyGraphEdge[] {\n    // The node label is the same as the id of the key nodes.\n    const ret: KeyGraphEdge[] = [];\n    let node = keyId;\n    if (!distances[node].predecessor) {\n      return null;\n    }\n\n    while (distances[node].predecessor) {\n      const child = distances[node].predecessor;\n      ret.push(this.graph.edge(child, node));\n      node = child;\n    }\n\n    // After reverse, the first element is the passkey\n    ret.reverse();\n\n    return ret;\n  }\n\n  getPath(knownKeyId: string, keyId: string): KeyGraphEdge[] {\n    if (!knownKeyId || typeof knownKeyId !== 'string') {\n      throw new LrEncryptionException(\n        `Param knownKeyId wrong format: ${knownKeyId}`\n      );\n    }\n    if (!keyId || typeof keyId !== 'string') {\n      throw new LrEncryptionException(`Param keyId wrong format: ${keyId}`);\n    }\n\n    // => { A: { distance: 0 },\n    //      B: { distance: 6, predecessor: 'C' },\n    //      C: { distance: 4, predecessor: 'A' },\n    //      D: { distance: 2, predecessor: 'A' },\n    //      E: { distance: 8, predecessor: 'F' },\n    //      F: { distance: 4, predecessor: 'D' } }\n    const distances = graphlib.alg.dijkstra(this.graph, knownKeyId);\n\n    // Trace path from keyId to knownKeyId\n    return this.tracePath(distances, keyId);\n  }\n\n  async getJwkKey(\n    keyOrId: string | Key,\n    getKeyIdCallback?: () => Promise<string> | string\n  ): Promise<JWK.Key> {\n    return (await this.getKey(keyOrId, getKeyIdCallback)).jwk;\n  }\n\n  // We assume that when a keyId is fetched, the key graph\n  // for the key is also returned and merged into the client-side\n  // key graph. By insisting a keyId is returned instead of the\n  // actual key we ensure key-graph is consistent.\n  async getKey(\n    keyOrId: string | Key,\n    getKeyIdCallback?: () => Promise<string> | string\n  ): Promise<Key> {\n    let keyId = typeof keyOrId === 'string' ? keyOrId : keyOrId?.id;\n\n    if (!this.hasKey(keyId) && getKeyIdCallback) {\n      keyId = await getKeyIdCallback();\n    }\n    // else, continue and let it fail.\n\n    const key = this.key(keyId);\n    if (key.jwk) {\n      return key;\n    } else {\n      return this.unwrapKey(this.keyService.getCurrentMasterKey().id, keyId);\n    }\n  }\n\n  private async _unwrapLink(wrappingKey, link, dstKey) {\n    // console.log(\"_unwrapLink:\", link.data.keyId);\n    const wrappedKey = JSON.parse(link.data.wrappedKey);\n    // Signatures of keys contain the key itself. This way we only need\n    // to access the KeyLinks to decrypt/verify keys.\n    let nextRawKey;\n    if (wrappedKey.signatures) {\n      nextRawKey = await this.encryptionService.verify(wrappingKey, wrappedKey);\n    } else {\n      nextRawKey = await this.encryptionService.decrypt(\n        wrappingKey,\n        wrappedKey\n      );\n    }\n    dstKey.jwk = await KFS.asKey(nextRawKey);\n    dstKey.task = null;\n  }\n\n  private async _unwrap(key: JWK.Key, path: KeyGraphEdge[]): Promise<JWK.Key> {\n    for (const link of path) {\n      const dstKey = this.key(link.data.keyId);\n      // console.log(\"key: \", link.data.keyId);\n      if (dstKey.jwk) {\n        key = dstKey.jwk;\n        // console.log(\"Returning cached key: \", link.data.keyId);\n        continue;\n      }\n\n      if (!dstKey.task) {\n        dstKey.task = this._unwrapLink(key, link, dstKey);\n      }\n\n      await dstKey.task;\n      key = dstKey.jwk;\n    }\n\n    return key;\n  }\n\n  public async unwrapWithPassKey(\n    passKeyId: string,\n    passKey: JWK.Key,\n    keyId: string\n  ): Promise<Key> {\n    // Get path of the directory key.\n    const path = this.getPath(passKeyId, keyId);\n\n    return {\n      id: keyId,\n      jwk: await this._unwrap(passKey, path),\n    };\n  }\n\n  async unwrapKey(masterKeyId: string, keyId: string): Promise<Key> {\n    // The first key should be a masterKey\n    const masterKey = await this.keyService.loadMasterKey(masterKeyId);\n\n    if (masterKeyId === keyId) {\n      return masterKey;\n    }\n\n    // Get path of the directory key.\n    const path = this.getPath(masterKey.id, keyId);\n\n    return {\n      id: keyId,\n      jwk: await this._unwrap(masterKey.jwk, path),\n    };\n  }\n\n  async decryptFromString<T>(\n    keyOrId: string | Key,\n    cipherData: string,\n    options?: DecryptOptions\n  ): Promise<T> {\n    if (cipherData) {\n      const key = await this.getJwkKey(keyOrId);\n      return (await this.encryptionService.decrypt(\n        key,\n        JSON.parse(cipherData),\n        options\n      )) as any;\n    }\n    return null;\n  }\n\n  async decryptFile(keyId: string, file: any): Promise<any> {\n    const key = await this.getJwkKey(keyId);\n    return (await this.encryptionService.decrypt(key, file, {\n      payloadType: 'ArrayBuffer',\n    })) as any;\n  }\n\n  // TODO rename this to encrypt() and use as the most common usecase\n  async encryptToString(\n    key: string | Key | JWK.Key,\n    content: any\n  ): Promise<string> {\n    // Empty string should be encrypted since you want to clear the field.\n    // Null is not encrypted because it's not valid JSON in the old JSON spec. Use\n    // empty string instead. It'll function as a logic false as well.\n    // Note that passing in empty string means it'll be encrypted which verifies\n    // it's integrity. But we still want to have a way to set the DB field\n    // to NULL, so we explicitly return null when content == null. A null\n    // variable in graphql mutation on KC server clears the field to NULL.\n    if (content == null) {\n      return null;\n    }\n\n    const jwk = asJwk(key) || (await this.getJwkKey(key as string | Key));\n    return this.encryptionService.encryptToString(jwk, content);\n  }\n\n  // Wraps a symmetric encryption key.\n  // Throws exception if wrapping public keys.\n  async wrapKey(\n    wrappingKey: string | Key | JWK.Key,\n    key: JWK.Key\n  ): Promise<string> {\n    if (!isSymmetricKey(key)) {\n      throw new LrBadArgumentException(\n        'Only allowing wrapping of symmetric keys.'\n      );\n    }\n\n    return this.encryptToString(wrappingKey, key.toJSON(true));\n  }\n\n  // TODO\n  // async wrapPublicKey<T>();\n  // async wrapPrivateKey<T>();\n\n  async encryptWithNewKey(wrappingKeyId: string, cipherClearJson: JSONObject) {\n    const key = await this.keyFactory.createKey();\n    const wrappedKey = await this.encryptToString(\n      wrappingKeyId,\n      key.toJSON(true)\n    );\n    const cipher = await this.encryptToString(key, cipherClearJson);\n\n    return {\n      key,\n      wrappingKeyId,\n      wrappedKey,\n      cipher,\n    };\n  }\n}\n"]}