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