@velocitycareerlabs/server-careerwallet 1.25.0-dev-build.12642c864

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 (252) hide show
  1. package/.localdev.env +47 -0
  2. package/.standalone.env +13 -0
  3. package/LICENSE +248 -0
  4. package/jest.config.js +20 -0
  5. package/migrate-mongo.config.js +25 -0
  6. package/migrations/20211017180227-create-personas.js +478 -0
  7. package/migrations/20211026185916-create-vanessa.js +79 -0
  8. package/migrations/20211026185917-update-personas.js +30 -0
  9. package/migrations/20211108124410-remove-surplus-personas.js +33 -0
  10. package/migrations/20211108132353-fix-vanessa-and-sheila.js +25 -0
  11. package/migrations/20220222123110-add-career-wallet-app-config.js +6 -0
  12. package/migrations/20220411104157-add-min-app-versions.js +15 -0
  13. package/migrations/20220419131726-create-nicole-flores-persona.js +63 -0
  14. package/migrations/20220515114034-update-persona-id-credentials.js +628 -0
  15. package/migrations/20220608093743-disable-mainnet-holderapp-id-verification.js +21 -0
  16. package/migrations/20220609063708-enable-mainnet-holderapp-id-verification.js +21 -0
  17. package/migrations/20220623091507-add-push-url.js +43 -0
  18. package/migrations/20220624133205-set-min-app-versions-to-11.js +16 -0
  19. package/migrations/20220710125326-set-min-app-version-to-0.10.7.js +16 -0
  20. package/migrations/20220811103500-add-verification-service-disclosure-deeplink.js +45 -0
  21. package/migrations/20220811123751-add-holderapp-dids-to-config.js +74 -0
  22. package/migrations/20220818072306-add-holderapp-endpoints-to-config.js +21 -0
  23. package/migrations/20220825090656-update-deeplink.js +46 -0
  24. package/migrations/20221003151823-app-config-add-public-verification-api.js +18 -0
  25. package/migrations/20221116085242-add-holderapp-sdk-to-config.js +15 -0
  26. package/migrations/20221121091030-update-holderapp-deeplink.js +19 -0
  27. package/migrations/20221128103425-update-holderapp-presentation-template.js +19 -0
  28. package/migrations/20221221091436-app-config-add-oauth.js +13 -0
  29. package/migrations/20221226205900-app-config-add-presentation-extension-api.js +18 -0
  30. package/migrations/20230120113141-update-holderapp-cashSiquence.js +15 -0
  31. package/migrations/20230123084103-update-holderapp-cacheSequence.js +15 -0
  32. package/migrations/20230214083430-update-holderapp-cache-Sequence.js +15 -0
  33. package/migrations/20230225173335-set-min-app-version-1.5.1.js +30 -0
  34. package/migrations/20230323120629-add-sunil-singh-persona.js +87 -0
  35. package/migrations/20230329081529-add-personas-by-env.js +294 -0
  36. package/migrations/20230329103219-remove-unused-personas.js +14 -0
  37. package/migrations/20230504090208-disable-yoti-migration.js +21 -0
  38. package/migrations/20230504123425-set-min-app-version-1.8.1.js +38 -0
  39. package/migrations/20230504185047-enable-yoti-migration.js +21 -0
  40. package/migrations/20230524053203-add-devices-index.js +16 -0
  41. package/migrations/20230704000002-add-common-holder-endpoints-to-config.js +16 -0
  42. package/migrations/20230704000003-add-linkedin-holder-endpoints-to-config.js +18 -0
  43. package/migrations/20230704104055-update-push-url-and-yoti-url.js +18 -0
  44. package/migrations/20230705000001-add-liburl-to-holderapp-config.js +20 -0
  45. package/migrations/20230814113134-app-config-add-oauth-client-id.js +18 -0
  46. package/migrations/20230821154136-yoti-new-session-url-fix.js +19 -0
  47. package/migrations/20230907134442-update-holderapp-cache-Sequence.js +15 -0
  48. package/migrations/20230919180000-set-holderapp-min-app-versions-1.15.0.js +16 -0
  49. package/migrations/20231011083137-update-holderapp-cache-sequence-6.js +15 -0
  50. package/migrations/20231102083252-set-holderapp-min-app-versions-1.14.0.js +16 -0
  51. package/migrations/20231108202229-set-holderapp-min-app-versions-1.15.0.js +16 -0
  52. package/migrations/20231115143332-holderapp-isDirectIssuerCheckOn-true.js +16 -0
  53. package/migrations/20231120100020-insert-didKeyMetadatum-for-old-accounts.js +59 -0
  54. package/migrations/20231207142742-remove-devices-from-accounts.js +46 -0
  55. package/migrations/20231226090805-app-config-set-direct-issuer-check.js +13 -0
  56. package/migrations/202312271524-set-holderapp-app-version-1.15.1.js +16 -0
  57. package/migrations/20232211171700-app-config-update-base-urls.js +18 -0
  58. package/migrations/20240102093506-holderapp-revert-version-1.15.0.js +16 -0
  59. package/migrations/202401041618111-holderapp-isDebugOn-false.js +17 -0
  60. package/migrations/20240131095122-test-personas-add-did.js +102 -0
  61. package/migrations/202402051547000-update-sdk.js +18 -0
  62. package/migrations/20240206101448-update-personas-vc-to-v2.js +137 -0
  63. package/migrations/202402061233000-set-xVnfProtocolVersion-2.js +13 -0
  64. package/migrations/202402081240000-set-xVnfProtocolVersion-1.js +13 -0
  65. package/migrations/202402131319-set-holderapp-min-versions-1.17.js +16 -0
  66. package/migrations/202402141152-set-holderapp-min-version-1.16.js +16 -0
  67. package/migrations/20240221123501-transform-account-keys-to-stringified-jwk.js +81 -0
  68. package/migrations/202402290955-update-holderapp-to-verifyMyCreds.js +36 -0
  69. package/migrations/20240311134223-update-keyid-persona.js +66 -0
  70. package/migrations/20240312141618-vl-7409-persona-update-maria-williams.js +82 -0
  71. package/migrations/202403181733000-remove-devices-from-all-accounts.js +27 -0
  72. package/migrations/20240401091041-set-holderapp-min-version-1.18.1.js +16 -0
  73. package/migrations/202404071847-app-config-update-yoti-url.js +19 -0
  74. package/migrations/20240724063405-vl-3827-new-yoti-session-url.js +19 -0
  75. package/migrations/20240731112302-vl-8160-add-persona-keys.js +101 -0
  76. package/migrations/20240911115206-vanessa-lin-id-credentials.js +57 -0
  77. package/migrations/202409201219-add-isWalletAvailable-field.js +13 -0
  78. package/migrations/202409221012-set-isWalletAvailable-true.js +13 -0
  79. package/migrations/20240922114643-holderapp-isWalletAvailable-refactor.js +36 -0
  80. package/migrations/20240923132213-update-android-ios-app-config.js +36 -0
  81. package/migrations/20240926061732-inc-cacheSequence-to-7.js +15 -0
  82. package/migrations/20240926073339-inc-cacheSequence-to-7-fix.js +18 -0
  83. package/migrations/202410081217-adam-smith-id-credntials.js +57 -0
  84. package/migrations/202410101243-adam_smith-id-credential.js +57 -0
  85. package/migrations/20241015124307-persona-key-id-type-fix.js +26 -0
  86. package/migrations/202501291027000-set-holderapp-did-web.js +37 -0
  87. package/migrations/environments/dev.env +34 -0
  88. package/migrations/environments/localdev.env +22 -0
  89. package/migrations/environments/prod.env +18 -0
  90. package/migrations/environments/qa.env +54 -0
  91. package/migrations/environments/staging.env +31 -0
  92. package/migrations/environments/test.env +18 -0
  93. package/package.json +84 -0
  94. package/src/assets/category-icons/assessment.png +0 -0
  95. package/src/assets/category-icons/badge.png +0 -0
  96. package/src/assets/category-icons/certification.png +0 -0
  97. package/src/assets/category-icons/education.png +0 -0
  98. package/src/assets/category-icons/employment.png +0 -0
  99. package/src/assets/category-icons/gig.png +0 -0
  100. package/src/assets/category-icons/identity.png +0 -0
  101. package/src/assets/category-icons/pharmacy.png +0 -0
  102. package/src/assets/category-icons/training.png +0 -0
  103. package/src/assets/credentialCategories.json +119 -0
  104. package/src/config/config.js +156 -0
  105. package/src/controllers/api/v0.6/accounts/autohooks.js +36 -0
  106. package/src/controllers/api/v0.6/accounts/controller.js +288 -0
  107. package/src/controllers/api/v0.6/accounts/schemas/careerwallet-accounts-didkeymetadatum-response.schema.js +41 -0
  108. package/src/controllers/api/v0.6/accounts/schemas/careerwallet-accounts-request.schema.js +49 -0
  109. package/src/controllers/api/v0.6/accounts/schemas/careerwallet-accounts-response.schema.js +74 -0
  110. package/src/controllers/api/v0.6/accounts/schemas/careerwallet-get-account-response.schema.js +72 -0
  111. package/src/controllers/api/v0.6/accounts/schemas/index.js +6 -0
  112. package/src/controllers/api/v0.6/careerwallet/appconfig/controller.js +25 -0
  113. package/src/controllers/api/v0.6/careerwallet/appconfig/repo.js +40 -0
  114. package/src/controllers/api/v0.6/careerwallet/autohooks.js +5 -0
  115. package/src/controllers/api/v0.6/careerwallet/consents/controller.js +66 -0
  116. package/src/controllers/api/v0.6/careerwallet/consents/latest/autohooks.js +7 -0
  117. package/src/controllers/api/v0.6/careerwallet/consents/latest/controller.js +76 -0
  118. package/src/controllers/api/v0.6/careerwallet/consents/repo.js +24 -0
  119. package/src/controllers/api/v0.6/careerwallet/consents/schemas/careerwallet-consent-response.schema.js +23 -0
  120. package/src/controllers/api/v0.6/careerwallet/consents/schemas/index.js +3 -0
  121. package/src/controllers/api/v0.6/create_did_key/autohooks.js +10 -0
  122. package/src/controllers/api/v0.6/create_did_key/controller.js +84 -0
  123. package/src/controllers/api/v0.6/create_did_key/schemas/index.js +4 -0
  124. package/src/controllers/api/v0.6/create_did_key/schemas/jwk-did-request.schema.js +20 -0
  125. package/src/controllers/api/v0.6/create_did_key/schemas/jwk-did-response.schema.js +41 -0
  126. package/src/controllers/api/v0.6/create_jwk/autohooks.js +10 -0
  127. package/src/controllers/api/v0.6/create_jwk/controller.js +46 -0
  128. package/src/controllers/api/v0.6/create_jwk/schemas/index.js +3 -0
  129. package/src/controllers/api/v0.6/create_jwk/schemas/jwk-response.schema.js +33 -0
  130. package/src/controllers/api/v0.6/credential-categories/controller.js +35 -0
  131. package/src/controllers/api/v0.6/devices/autohooks.js +11 -0
  132. package/src/controllers/api/v0.6/devices/controller.js +323 -0
  133. package/src/controllers/api/v0.6/devices/repo.js +27 -0
  134. package/src/controllers/api/v0.6/devices/schemas/device.schema.json +43 -0
  135. package/src/controllers/api/v0.6/devices/schemas/index.js +3 -0
  136. package/src/controllers/api/v0.6/feedback/controller.js +24 -0
  137. package/src/controllers/api/v0.6/feedback/schemas/index.js +3 -0
  138. package/src/controllers/api/v0.6/feedback/schemas/new-feedback.schema.js +47 -0
  139. package/src/controllers/api/v0.6/jwt/autohooks.js +10 -0
  140. package/src/controllers/api/v0.6/jwt/controller.js +44 -0
  141. package/src/controllers/api/v0.6/jwt/schemas/index.js +4 -0
  142. package/src/controllers/api/v0.6/jwt/schemas/jwt-request.schema.js +40 -0
  143. package/src/controllers/api/v0.6/jwt/schemas/jwt-response.schema.js +14 -0
  144. package/src/controllers/api/v0.6/oauth/controller.js +131 -0
  145. package/src/controllers/api/v0.6/push/controller.js +296 -0
  146. package/src/controllers/api/v0.6/push/firebase-initializer.js +21 -0
  147. package/src/controllers/api/v0.6/push/notification-types.js +37 -0
  148. package/src/controllers/api/v0.6/push/push-gateway-auth.js +123 -0
  149. package/src/controllers/api/v0.6/push/repo.js +24 -0
  150. package/src/controllers/api/v0.6/verification-offers/repo.js +25 -0
  151. package/src/controllers/api/v0.6/verify/autohooks.js +15 -0
  152. package/src/controllers/api/v0.6/verify/controller.js +348 -0
  153. package/src/controllers/api/v0.6/verify/repo.js +23 -0
  154. package/src/controllers/api/v0.6/verify/verification-credential-types.js +6 -0
  155. package/src/controllers/jwt/autohooks.js +10 -0
  156. package/src/controllers/jwt/controller.js +106 -0
  157. package/src/controllers/jwt/schemas/index.js +31 -0
  158. package/src/controllers/jwt/schemas/jwt-decode.response.200.schema.json +20 -0
  159. package/src/controllers/jwt/schemas/jwt-decode.schema.json +15 -0
  160. package/src/controllers/jwt/schemas/jwt-sign.response.200.schema.json +14 -0
  161. package/src/controllers/jwt/schemas/jwt-sign.schema.json +40 -0
  162. package/src/controllers/jwt/schemas/jwt-verify.response.200.schema.json +17 -0
  163. package/src/controllers/jwt/schemas/jwt-verify.schema.json +19 -0
  164. package/src/controllers/reference/autohooks.js +5 -0
  165. package/src/controllers/reference/countries/controller.js +81 -0
  166. package/src/controllers/reference/personas/controller.js +91 -0
  167. package/src/controllers/reference/personas/repo.js +29 -0
  168. package/src/controllers/reference/personas/schemas/index.js +3 -0
  169. package/src/controllers/reference/personas/schemas/persona.schema.json +108 -0
  170. package/src/controllers/root/controller.js +27 -0
  171. package/src/entities/accounts/constants.js +18 -0
  172. package/src/entities/accounts/domain/index.js +3 -0
  173. package/src/entities/accounts/domain/merge-scopes.js +9 -0
  174. package/src/entities/accounts/index.js +5 -0
  175. package/src/entities/accounts/repos/accounts.repo.js +64 -0
  176. package/src/entities/accounts/repos/id-token-claims-extension.js +31 -0
  177. package/src/entities/accounts/repos/index.js +3 -0
  178. package/src/entities/devices/constants.js +7 -0
  179. package/src/entities/devices/index.js +3 -0
  180. package/src/entities/feedback/domain/generate-feedback-email.js +23 -0
  181. package/src/entities/feedback/domain/index.js +3 -0
  182. package/src/entities/feedback/index.js +3 -0
  183. package/src/entities/index.js +9 -0
  184. package/src/entities/key-pairs/index.js +4 -0
  185. package/src/entities/key-pairs/orchestrators/generate-jwk.js +28 -0
  186. package/src/entities/key-pairs/orchestrators/get-key-pair.js +58 -0
  187. package/src/entities/key-pairs/orchestrators/index.js +4 -0
  188. package/src/entities/key-pairs/repos/index.js +3 -0
  189. package/src/entities/key-pairs/repos/key-pairs.repo.js +23 -0
  190. package/src/entities/oauth/constants.js +13 -0
  191. package/src/entities/oauth/domain/build-access-token.js +14 -0
  192. package/src/entities/oauth/domain/index.js +10 -0
  193. package/src/entities/oauth/domain/validate-audience.js +13 -0
  194. package/src/entities/oauth/domain/validate-client-id.js +13 -0
  195. package/src/entities/oauth/domain/validate-credential.js +16 -0
  196. package/src/entities/oauth/domain/validate-presentation.js +17 -0
  197. package/src/entities/oauth/domain/validate-refresh-token.js +17 -0
  198. package/src/entities/oauth/domain/validate-scope.js +30 -0
  199. package/src/entities/oauth/domain/verify-presentation.js +24 -0
  200. package/src/entities/oauth/index.js +4 -0
  201. package/src/entities/pushes/domains/errors.js +11 -0
  202. package/src/entities/pushes/domains/index.js +3 -0
  203. package/src/entities/pushes/index.js +3 -0
  204. package/src/entities/verification-code-attempts/index.js +3 -0
  205. package/src/entities/verification-code-attempts/orchestrators/index.js +3 -0
  206. package/src/entities/verification-code-attempts/orchestrators/validate-verification-code-attempts.js +79 -0
  207. package/src/entities/verification-code-attempts/repo.js +99 -0
  208. package/src/entities/verifications/constants.js +8 -0
  209. package/src/entities/verifications/index.js +3 -0
  210. package/src/helpers/caching-constants.js +6 -0
  211. package/src/helpers/index.js +3 -0
  212. package/src/index.js +15 -0
  213. package/src/init-server.js +88 -0
  214. package/src/plugins/index.js +4 -0
  215. package/src/plugins/vcl-verification-version-plugin.js +10 -0
  216. package/src/plugins/verify-access-token-plugin.js +116 -0
  217. package/src/standalone.js +8 -0
  218. package/test/accounts-controller.test.js +893 -0
  219. package/test/accounts-repo.test.js +92 -0
  220. package/test/careerwallet-config-controller.test.js +142 -0
  221. package/test/careerwallet-consents-controller.test.js +409 -0
  222. package/test/create_did_key-controller.test.js +397 -0
  223. package/test/create_jwk-controller.test.js +188 -0
  224. package/test/credential-categories-controller.test.js +29 -0
  225. package/test/credential-icons-controller.test.js +36 -0
  226. package/test/devices-controller.test.js +1025 -0
  227. package/test/factories/accounts-factory.js +15 -0
  228. package/test/factories/career-wallet-config-factory.js +60 -0
  229. package/test/factories/consents-career-wallet-factory.js +21 -0
  230. package/test/factories/devices-factory.js +16 -0
  231. package/test/factories/key-pairs-factory.js +18 -0
  232. package/test/factories/notifications-factory.js +16 -0
  233. package/test/factories/persona-factory.js +17 -0
  234. package/test/factories/refresh-tokens-factory.js +14 -0
  235. package/test/factories/verification-code-attempts-factory.js +17 -0
  236. package/test/factories/verification-factory.js +15 -0
  237. package/test/factories/verification-offer-factory.js +20 -0
  238. package/test/feedback-controller.test.js +225 -0
  239. package/test/helpers/.env.test +39 -0
  240. package/test/helpers/access-token.js +59 -0
  241. package/test/helpers/careerwallet-build-fastify.js +20 -0
  242. package/test/helpers/yoti.js +96 -0
  243. package/test/id-verification-controller.test.js +27 -0
  244. package/test/jwt-controller.test.js +1519 -0
  245. package/test/oauth-controller.test.js +639 -0
  246. package/test/push-controller.test.js +733 -0
  247. package/test/push-gateway-auth.test.js +208 -0
  248. package/test/reference-countries-controller.test.js +45 -0
  249. package/test/reference-personas-controller.test.js +179 -0
  250. package/test/root.test.js +21 -0
  251. package/test/swagger.test.js +21 -0
  252. package/test/verification-controller.test.js +1372 -0
@@ -0,0 +1,288 @@
1
+ const newError = require('http-errors');
2
+ const { isArray, includes } = require('lodash/fp');
3
+ const { nanoid } = require('nanoid/non-secure');
4
+ const { pick } = require('lodash/fp');
5
+ const { initBuildRefreshToken } = require('@velocitycareerlabs/crypto');
6
+ const { accountScopes, initBuildAccessToken } = require('../../../../entities');
7
+
8
+ const buildRefreshToken = initBuildRefreshToken();
9
+
10
+ const accountsController = async (fastify) => {
11
+ const { config } = fastify;
12
+
13
+ fastify.post(
14
+ '/',
15
+ {
16
+ schema: fastify.autoSchema({
17
+ tags: ['careerwallet'],
18
+ body: {
19
+ $ref: 'https://velocitycareerlabs.io/careerwallet-accounts-request.schema.json#',
20
+ },
21
+ response: {
22
+ 200: {
23
+ $ref: 'https://velocitycareerlabs.io/careerwallet-accounts-response.schema.json#',
24
+ },
25
+ },
26
+ }),
27
+ },
28
+ async (req, reply) => {
29
+ const {
30
+ repos,
31
+ body: {
32
+ account: { device: { deviceId } = {} },
33
+ },
34
+ config: reqConfig,
35
+ } = req;
36
+
37
+ await validateRequest(req, reply);
38
+
39
+ const scope = [accountScopes.keyCreate];
40
+ const refreshToken = buildRefreshToken();
41
+ const account = await findOrInsertAccount(req, refreshToken, scope);
42
+
43
+ if (deviceId) {
44
+ await repos.devices.updateUsingFilter(
45
+ {
46
+ filter: {
47
+ $or: [{ deviceId }, { pushToken: deviceId }],
48
+ },
49
+ },
50
+ { accountId: account._id }
51
+ );
52
+ }
53
+
54
+ const devices = await repos.devices.find({ accountId: account._id });
55
+
56
+ const accessToken = await initBuildAccessToken(
57
+ reqConfig.holderAppServerAccessTokenSigningKey,
58
+ config.holderAppServerHolderAppProviderDid,
59
+ config.holderAppServerAccessTokenExpiresIn
60
+ )(scope.join(' '), account._id);
61
+ return {
62
+ ...buildAccountResponse(account, devices, req),
63
+ access_token: accessToken,
64
+ refresh_token: refreshToken,
65
+ };
66
+ }
67
+ );
68
+
69
+ fastify
70
+ .autoSchemaPreset({ security: [{ bearerAuth: [] }] })
71
+ .post(
72
+ '/:accountId/push_token',
73
+ {
74
+ onRequest: fastify.verifyCareerWalletAccessToken(),
75
+ schema: fastify.autoSchema({
76
+ params: {
77
+ type: 'object',
78
+ properties: {
79
+ accountId: { type: 'string' },
80
+ },
81
+ required: ['accountId'],
82
+ },
83
+ body: {
84
+ type: 'object',
85
+ properties: {
86
+ did: { type: 'string' },
87
+ },
88
+ required: ['did'],
89
+ },
90
+ response: {
91
+ 200: {
92
+ type: 'object',
93
+ properties: {
94
+ pushToken: { type: 'string' },
95
+ },
96
+ required: ['pushToken'],
97
+ },
98
+ },
99
+ }),
100
+ },
101
+ async (req) => {
102
+ const {
103
+ repos: { accounts },
104
+ params: { accountId },
105
+ body: { did },
106
+ user,
107
+ isServerUser,
108
+ } = req;
109
+
110
+ if (!isServerUser && accountId !== user.sub) {
111
+ throw new newError.Unauthorized();
112
+ }
113
+
114
+ const pushToken = nanoid();
115
+ const updatedAccount = {
116
+ [`pushTokens.${pushToken}`]: { did, createdAt: new Date() },
117
+ };
118
+ await accounts.update(accountId, updatedAccount);
119
+ return {
120
+ pushToken,
121
+ };
122
+ }
123
+ )
124
+ .get(
125
+ '/:accountId',
126
+ {
127
+ onRequest: fastify.verifyCareerWalletAccessToken(),
128
+ schema: fastify.autoSchema({
129
+ params: {
130
+ type: 'object',
131
+ properties: {
132
+ accountId: { type: 'string' },
133
+ },
134
+ },
135
+ response: {
136
+ 200: {
137
+ $ref: 'https://velocitycareerlabs.io/careerwallet-get-account-response.schema.json#',
138
+ },
139
+ },
140
+ }),
141
+ },
142
+ async (req) => {
143
+ const {
144
+ repos,
145
+ params: { accountId },
146
+ isServerUser,
147
+ } = req;
148
+
149
+ if (!isServerUser && accountId !== req.user.sub) {
150
+ throw new newError.Forbidden(
151
+ 'Access token account does not match the accountId'
152
+ );
153
+ }
154
+
155
+ const account = await repos.accounts.findOne({
156
+ filter: { _id: accountId },
157
+ });
158
+ const devices = await repos.devices.find({ accountId: account._id });
159
+
160
+ return { account: { ...account, devices } };
161
+ }
162
+ );
163
+ };
164
+
165
+ const findOrInsertAccount = async (req, refreshToken, scope) => {
166
+ const {
167
+ repos: { accounts },
168
+ body: {
169
+ account: { id_token: webWalletUserToken },
170
+ },
171
+ } = req;
172
+
173
+ const existingAccount =
174
+ webWalletUserToken &&
175
+ (await accounts.findOne({ filter: { userId: req.user.sub } }));
176
+
177
+ if (!existingAccount) {
178
+ return accounts.insert(await buildAccount(refreshToken, scope, req));
179
+ }
180
+
181
+ return existingAccount;
182
+ };
183
+
184
+ const validateRequest = async (req, reply) => {
185
+ const {
186
+ body: {
187
+ account: { id_token: idToken },
188
+ },
189
+ } = req;
190
+
191
+ if (idToken) {
192
+ await req.server.authenticate(req, reply);
193
+ const isWebappClient = isArray(req.user.aud)
194
+ ? includes(req.config.oauth0WebappClientId, req.user.aud)
195
+ : req.user.aud === req.config.oauth0WebappClientId;
196
+ if (!isWebappClient) {
197
+ throw new newError.Unauthorized('Invalid audience');
198
+ }
199
+ }
200
+ };
201
+
202
+ const buildAccount = async (refreshToken, scope, req) => {
203
+ const {
204
+ body: {
205
+ account: { accountType, device, id_token: idToken },
206
+ },
207
+ } = req;
208
+
209
+ const didKeyMetadatum = await buildDidKeyMetadatum(req);
210
+
211
+ const newAccount = {
212
+ accountType,
213
+ scope: didKeyMetadatum.length
214
+ ? [
215
+ accountScopes.jwt,
216
+ accountScopes.keyCreate,
217
+ accountScopes.inbox,
218
+ accountScopes.account,
219
+ ]
220
+ : scope,
221
+ refreshToken,
222
+ didKeyMetadatum,
223
+ };
224
+
225
+ if (idToken) {
226
+ return {
227
+ ...newAccount,
228
+ userId: req.user.sub,
229
+ idTokenClaims: { idTokenIssuer: req.user.iss },
230
+ };
231
+ }
232
+
233
+ return {
234
+ ...newAccount,
235
+ devices: [device],
236
+ };
237
+ };
238
+
239
+ const buildDidKeyMetadatum = async (req) => {
240
+ const {
241
+ body: { personaEmail },
242
+ repos: { personas },
243
+ } = req;
244
+
245
+ if (!personaEmail) {
246
+ return [];
247
+ }
248
+
249
+ const persona = await personas.findOne({
250
+ filter: { email: personaEmail },
251
+ });
252
+
253
+ if (!persona) {
254
+ throw new newError.BadRequest('Invalid persona email');
255
+ }
256
+
257
+ if (persona.did) {
258
+ return [pick(['did', 'kid', 'keyId', 'publicJwk'], persona)];
259
+ }
260
+
261
+ return [];
262
+ };
263
+
264
+ const buildAccountResponse = (account, devices, req) => {
265
+ const {
266
+ body: {
267
+ account: { id_token: idToken },
268
+ },
269
+ } = req;
270
+ if (idToken) {
271
+ return {
272
+ account: {
273
+ ...account,
274
+ id_token_iss: req.user.iss,
275
+ id_token_sub: req.user.sub,
276
+ },
277
+ };
278
+ }
279
+
280
+ return {
281
+ account: {
282
+ ...account,
283
+ devices,
284
+ },
285
+ };
286
+ };
287
+
288
+ module.exports = accountsController;
@@ -0,0 +1,41 @@
1
+ const careerWalletAccountsDidKeyMetadatumResponseSchema = {
2
+ $id: 'https://velocitycareerlabs.io/careerwallet-accounts-didkeymetadatum-response.schema.json',
3
+ title: 'careerwallet-accounts-didkeymetadatum-response',
4
+ description: 'the body of response for careerwallet accounts didKeyMetadatum',
5
+ type: 'object',
6
+ properties: {
7
+ did: {
8
+ type: 'string',
9
+ },
10
+ kid: {
11
+ type: 'string',
12
+ },
13
+ keyId: {
14
+ type: 'string',
15
+ },
16
+ publicJwk: {
17
+ type: 'object',
18
+ required: ['kty'],
19
+ properties: {
20
+ kty: { type: 'string' },
21
+ use: { type: 'string' },
22
+ key_ops: { type: 'string' },
23
+ x5u: { type: 'string' },
24
+ x5c: { type: 'string' },
25
+ x5t: { type: 'string' },
26
+ 'x5t#S256': { type: 'string' },
27
+ alg: { type: 'string' },
28
+ kid: { type: 'string' },
29
+ exp: { type: 'string' },
30
+ crv: { type: 'string' },
31
+ y: { type: 'string' },
32
+ x: { type: 'string' },
33
+ n: { type: 'string' },
34
+ e: { type: 'string' },
35
+ },
36
+ },
37
+ },
38
+ required: ['did', 'kid', 'publicJwk'],
39
+ };
40
+
41
+ module.exports = { careerWalletAccountsDidKeyMetadatumResponseSchema };
@@ -0,0 +1,49 @@
1
+ const { accountTypes } = require('../../../../../entities');
2
+ const { deviceOSes, deviceTypes } = require('../../../../../entities/devices');
3
+
4
+ const careerWalletAccountsRequestSchema = {
5
+ $id: 'https://velocitycareerlabs.io/careerwallet-accounts-request.schema.json',
6
+ title: 'carrerwallet-accounts-request',
7
+ description: 'the request for careerwallet accounts',
8
+ type: 'object',
9
+ properties: {
10
+ account: {
11
+ type: 'object',
12
+ properties: {
13
+ accountType: {
14
+ type: 'string',
15
+ enum: accountTypes,
16
+ },
17
+ device: {
18
+ type: 'object',
19
+ properties: {
20
+ deviceId: { type: 'string' },
21
+ deviceType: {
22
+ type: 'string',
23
+ enum: deviceTypes,
24
+ },
25
+ deviceOS: {
26
+ type: 'string',
27
+ enum: deviceOSes,
28
+ },
29
+ },
30
+ required: ['deviceId', 'deviceType', 'deviceOS'],
31
+ },
32
+ id_token: { type: 'string' },
33
+ },
34
+ required: ['accountType'],
35
+ oneOf: [
36
+ {
37
+ required: ['id_token'],
38
+ },
39
+ {
40
+ required: ['device'],
41
+ },
42
+ ],
43
+ },
44
+ personaEmail: { type: 'string' },
45
+ },
46
+ required: ['account'],
47
+ };
48
+
49
+ module.exports = { careerWalletAccountsRequestSchema };
@@ -0,0 +1,74 @@
1
+ const { accountTypes } = require('../../../../../entities');
2
+ const { deviceOSes, deviceTypes } = require('../../../../../entities/devices');
3
+
4
+ const careerWalletAccountsResponseSchema = {
5
+ $id: 'https://velocitycareerlabs.io/careerwallet-accounts-response.schema.json',
6
+ title: 'carrerwallet-accounts-response',
7
+ description: 'the body of response for careerwallet accounts',
8
+ type: 'object',
9
+ properties: {
10
+ account: {
11
+ type: 'object',
12
+ properties: {
13
+ id: { type: 'string' },
14
+ accountType: {
15
+ type: 'string',
16
+ enum: accountTypes,
17
+ },
18
+ devices: {
19
+ type: 'array',
20
+ items: {
21
+ type: 'object',
22
+ properties: {
23
+ deviceId: { type: 'string' },
24
+ deviceType: {
25
+ type: 'string',
26
+ enum: deviceTypes,
27
+ },
28
+ deviceOS: {
29
+ type: 'string',
30
+ enum: deviceOSes,
31
+ },
32
+ createdAt: { type: 'string', format: 'date-time' },
33
+ updatedAt: { type: 'string', format: 'date-time' },
34
+ },
35
+ required: ['deviceId', 'deviceType', 'deviceOS'],
36
+ },
37
+ },
38
+ didKeyMetadatum: {
39
+ type: 'array',
40
+ items: {
41
+ $ref: 'https://velocitycareerlabs.io/careerwallet-accounts-didkeymetadatum-response.schema.json',
42
+ },
43
+ },
44
+ did: { type: 'string' },
45
+ id_token_iss: { type: 'string' },
46
+ id_token_sub: { type: 'string' },
47
+ createdAt: { type: 'string', format: 'date-time' },
48
+ updatedAt: { type: 'string', format: 'date-time' },
49
+ },
50
+ required: [
51
+ 'id',
52
+ 'accountType',
53
+ 'createdAt',
54
+ 'updatedAt',
55
+ 'didKeyMetadatum',
56
+ ],
57
+ oneOf: [
58
+ {
59
+ type: 'object',
60
+ required: ['id_token_iss', 'id_token_sub'],
61
+ },
62
+ {
63
+ type: 'object',
64
+ required: ['devices'],
65
+ },
66
+ ],
67
+ },
68
+ access_token: { type: 'string' },
69
+ refresh_token: { type: 'string' },
70
+ },
71
+ required: ['account', 'access_token', 'refresh_token'],
72
+ };
73
+
74
+ module.exports = { careerWalletAccountsResponseSchema };
@@ -0,0 +1,72 @@
1
+ const { accountTypes } = require('../../../../../entities');
2
+ const { deviceOSes, deviceTypes } = require('../../../../../entities/devices');
3
+
4
+ const careerWalletGetAccountResponseSchema = {
5
+ $id: 'https://velocitycareerlabs.io/careerwallet-get-account-response.schema.json',
6
+ title: 'careerwallet-get-account-response',
7
+ description: 'the body of response for careerwallet account',
8
+ type: 'object',
9
+ properties: {
10
+ account: {
11
+ type: 'object',
12
+ properties: {
13
+ id: { type: 'string' },
14
+ accountType: {
15
+ type: 'string',
16
+ enum: accountTypes,
17
+ },
18
+ devices: {
19
+ type: 'array',
20
+ items: {
21
+ type: 'object',
22
+ properties: {
23
+ deviceId: { type: 'string' },
24
+ deviceType: {
25
+ type: 'string',
26
+ enum: deviceTypes,
27
+ },
28
+ deviceOS: {
29
+ type: 'string',
30
+ enum: deviceOSes,
31
+ },
32
+ createdAt: { type: 'string', format: 'date-time' },
33
+ updatedAt: { type: 'string', format: 'date-time' },
34
+ },
35
+ required: ['deviceId', 'deviceType', 'deviceOS'],
36
+ },
37
+ },
38
+ didKeyMetadatum: {
39
+ type: 'array',
40
+ items: {
41
+ $ref: 'https://velocitycareerlabs.io/careerwallet-accounts-didkeymetadatum-response.schema.json',
42
+ },
43
+ },
44
+ did: { type: 'string' },
45
+ id_token_iss: { type: 'string' },
46
+ id_token_sub: { type: 'string' },
47
+ createdAt: { type: 'string', format: 'date-time' },
48
+ updatedAt: { type: 'string', format: 'date-time' },
49
+ },
50
+ required: [
51
+ 'id',
52
+ 'accountType',
53
+ 'createdAt',
54
+ 'updatedAt',
55
+ 'didKeyMetadatum',
56
+ ],
57
+ oneOf: [
58
+ {
59
+ type: 'object',
60
+ required: ['id_token_iss', 'id_token_sub'],
61
+ },
62
+ {
63
+ type: 'object',
64
+ required: ['devices'],
65
+ },
66
+ ],
67
+ },
68
+ },
69
+ required: ['account'],
70
+ };
71
+
72
+ module.exports = { careerWalletGetAccountResponseSchema };
@@ -0,0 +1,6 @@
1
+ module.exports = {
2
+ ...require('./careerwallet-accounts-request.schema'),
3
+ ...require('./careerwallet-accounts-response.schema'),
4
+ ...require('./careerwallet-get-account-response.schema'),
5
+ ...require('./careerwallet-accounts-didkeymetadatum-response.schema'),
6
+ };
@@ -0,0 +1,25 @@
1
+ const { omitBy } = require('lodash/fp');
2
+
3
+ const careerWalletConfigsController = async (fastify) => {
4
+ fastify.get(
5
+ '/',
6
+ {
7
+ schema: fastify.autoSchema({
8
+ response: {
9
+ 200: {
10
+ type: 'object',
11
+ additionalProperties: true,
12
+ },
13
+ },
14
+ }),
15
+ },
16
+ async ({ repos }) => {
17
+ const config = await repos.careerWalletConfigs.findOne();
18
+ return omitBy((value, key) =>
19
+ ['id', '_id', 'createdAt', 'updatedAt'].includes(key)
20
+ )(config);
21
+ }
22
+ );
23
+ };
24
+
25
+ module.exports = careerWalletConfigsController;
@@ -0,0 +1,40 @@
1
+ const {
2
+ repoFactory,
3
+ autoboxIdsExtension,
4
+ } = require('@spencejs/spence-mongo-repos');
5
+
6
+ module.exports = (app, options, next = () => {}) => {
7
+ next();
8
+ return repoFactory(
9
+ {
10
+ name: 'careerWalletConfigs',
11
+ entityName: 'careerWalletConfig',
12
+ defaultProjection: {
13
+ _id: 1,
14
+ ios: 1,
15
+ android: 1,
16
+ yotiIDV: 1,
17
+ latestAndroidVersion: 1,
18
+ latestIOSVersion: 1,
19
+ minAndroidVersion: 1,
20
+ minIOSVersion: 1,
21
+ createdAt: 1,
22
+ updatedAt: 1,
23
+ pushUrl: 1,
24
+ baseUrls: 1,
25
+ oauth: 1,
26
+ sdk: 1,
27
+ idVerifierDid: 1,
28
+ emailVerifierDid: 1,
29
+ phoneVerifierDid: 1,
30
+ yotiNewSessionUrl: 1,
31
+ verificationServiceDeepLink: 1,
32
+ verificationServicePresentationLinkTemplate: 1,
33
+ commonUrls: 1,
34
+ linkedinEndpoints: 1,
35
+ },
36
+ extensions: [autoboxIdsExtension],
37
+ },
38
+ app
39
+ );
40
+ };
@@ -0,0 +1,5 @@
1
+ module.exports = async (fastify) => {
2
+ fastify.autoSchemaPreset({
3
+ tags: ['careerwallet'],
4
+ });
5
+ };
@@ -0,0 +1,66 @@
1
+ const { careerWalletConsentResponseSchema } = require('./schemas');
2
+
3
+ const controller = async (fastify) => {
4
+ fastify.post(
5
+ '/add',
6
+ {
7
+ schema: fastify.autoSchema({
8
+ body: {
9
+ type: 'object',
10
+ properties: {
11
+ version: {
12
+ type: 'string',
13
+ pattern: '[0-9]+.[0-9]+',
14
+ },
15
+ accountId: { type: 'string' },
16
+ },
17
+ required: ['version', 'accountId'],
18
+ },
19
+ response: {
20
+ 200: {
21
+ type: 'object',
22
+ properties: {
23
+ consent: careerWalletConsentResponseSchema,
24
+ },
25
+ required: ['consent'],
26
+ },
27
+ },
28
+ }),
29
+ },
30
+ async (req, rep) => {
31
+ const {
32
+ repos,
33
+ body: { version, accountId },
34
+ } = req;
35
+
36
+ const consentDb = await repos.consentsCareerWallet.findOne({
37
+ filter: { accountId, version },
38
+ sort: { createdAt: -1 },
39
+ });
40
+
41
+ if (consentDb) {
42
+ return rep.code(201).send({
43
+ consent: mapConsent(consentDb),
44
+ });
45
+ }
46
+
47
+ const consent = await repos.consentsCareerWallet.insert({
48
+ accountId,
49
+ version,
50
+ });
51
+
52
+ return rep.code(201).send({
53
+ consent: mapConsent(consent),
54
+ });
55
+ }
56
+ );
57
+ };
58
+
59
+ const mapConsent = ({ createdAt, version, accountId, ...consent }) => ({
60
+ createdAt,
61
+ accountId,
62
+ version,
63
+ id: consent._id.toString(),
64
+ });
65
+
66
+ module.exports = controller;
@@ -0,0 +1,7 @@
1
+ const { verifyAccessTokenPlugin } = require('../../../../../../plugins');
2
+
3
+ module.exports = async (fastify) => {
4
+ fastify
5
+ .register(verifyAccessTokenPlugin)
6
+ .autoSchemaPreset({ security: [{ bearerAuth: [] }] });
7
+ };