@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,348 @@
1
+ const newError = require('http-errors');
2
+ const { nanoid } = require('nanoid/non-secure');
3
+ const { omit, first, values } = require('lodash/fp');
4
+ const { addMinutes, subMinutes } = require('date-fns/fp');
5
+
6
+ const {
7
+ identifyWebhookRequestBodySchema,
8
+ verificationIdentifyResponse200Schema,
9
+ } = require('@velocitycareerlabs/common-schemas');
10
+
11
+ const {
12
+ generateKeyPair,
13
+ generateRandomNumber,
14
+ } = require('@velocitycareerlabs/crypto');
15
+ const { generateCredentialJwt } = require('@velocitycareerlabs/jwt');
16
+ const VerificationCredentialTypes = require('./verification-credential-types');
17
+ const {
18
+ validateVerificationCodeAttempts,
19
+ VerificationValueTypes,
20
+ accountScopes,
21
+ } = require('../../../../entities');
22
+
23
+ const verificationExpiration = 10;
24
+ const verificationCodeLength = 6;
25
+
26
+ const verificationController = async (fastify) => {
27
+ const { sendEmail, sendSms, config } = fastify;
28
+
29
+ // Throws bad request when X attempts have been made and the last attempt hasn't expired yet
30
+ const checkAttemptsThreshold = (existingVerificationCounts) => {
31
+ if (fastify.config.attemptsThreshold <= existingVerificationCounts) {
32
+ throw new newError.BadRequest('Too many attempts.');
33
+ }
34
+ };
35
+
36
+ const smsVerificationMessage = (verificationCode) =>
37
+ // eslint-disable-next-line max-len
38
+ `Your Velocity Phone Verification code is: ${verificationCode}.\n\n@velocitycareerlabs.io #${verificationCode}\n\n${config.smsAppVerificationHash}`;
39
+
40
+ const sendNotification = async (valueType, value, verificationCode) => {
41
+ /* eslint-disable max-len */
42
+ const verificationMessage = `Please enter this code in the Velocity app, to validate your ${valueType}: ${verificationCode}. The code is valid for ${verificationExpiration} minutes`;
43
+
44
+ if (valueType === VerificationValueTypes.Email) {
45
+ await sendEmail({
46
+ subject: 'Email Address Verification',
47
+ message: verificationMessage,
48
+ sender: fastify.config.verificationSenderEmail,
49
+ recipients: [value],
50
+ replyTo: fastify.config.verificationSenderEmail,
51
+ });
52
+ } else if (valueType === VerificationValueTypes.Phone) {
53
+ await sendSms({
54
+ message: smsVerificationMessage(verificationCode),
55
+ phoneNumber: value,
56
+ });
57
+ } else {
58
+ throw new newError.BadRequest(`Incorrect value type: ${valueType}.`);
59
+ }
60
+ };
61
+
62
+ fastify.post(
63
+ '/:valueType/request-code',
64
+ {
65
+ onRequest: fastify.verifyCareerWalletAccessToken([accountScopes.account]),
66
+ schema: fastify.autoSchema({
67
+ security: [{ bearerAuth: [accountScopes.account] }],
68
+ params: {
69
+ type: 'object',
70
+ description: 'Either `phone` or `email`',
71
+ properties: {
72
+ valueType: {
73
+ type: 'string',
74
+ enum: values(VerificationValueTypes),
75
+ },
76
+ },
77
+ },
78
+ body: {
79
+ type: 'object',
80
+ description:
81
+ 'The actual email address or phone number (depending on the value of `valueType`)',
82
+ additionalProperties: false,
83
+ properties: {
84
+ value: { type: 'string' },
85
+ },
86
+ required: ['value'],
87
+ },
88
+ response: {
89
+ 204: {
90
+ type: 'null',
91
+ },
92
+ },
93
+ }),
94
+ },
95
+ async (req, reply) => {
96
+ const { repos, params, body } = req;
97
+ const { valueType } = params;
98
+ const { value } = body;
99
+ if (valueType === VerificationValueTypes.Phone) {
100
+ await validateVerificationCodeAttempts(value, req);
101
+ }
102
+ const existingVerifications = await repos.verifications.count({
103
+ filter: {
104
+ type: valueType,
105
+ value,
106
+ createdAt: { $gt: subMinutes(verificationExpiration, new Date()) },
107
+ },
108
+ });
109
+ checkAttemptsThreshold(existingVerifications);
110
+
111
+ const attemptId = await generateRandomNumber(verificationCodeLength);
112
+ const verification = createVerification(
113
+ valueType,
114
+ value,
115
+ attemptId.toString()
116
+ );
117
+
118
+ await repos.verifications.insert(verification);
119
+
120
+ await sendNotification(valueType, value, attemptId);
121
+
122
+ reply.code(204);
123
+ }
124
+ );
125
+
126
+ fastify.post(
127
+ '/confirm',
128
+ {
129
+ onRequest: fastify.verifyCareerWalletAccessToken([accountScopes.account]),
130
+ schema: fastify.autoSchema({
131
+ security: [{ bearerAuth: [accountScopes.account] }],
132
+ body: {
133
+ type: 'object',
134
+ additionalProperties: false,
135
+ properties: {
136
+ verificationCode: { type: 'string' },
137
+ },
138
+ required: ['verificationCode'],
139
+ },
140
+ response: {
141
+ 200: {
142
+ type: 'object',
143
+ properties: {
144
+ credential: { type: 'string' },
145
+ token: { type: 'string' },
146
+ },
147
+ },
148
+ ...fastify.NotFoundResponse,
149
+ },
150
+ }),
151
+ },
152
+ async ({ vclVerificationVersion, repos, body }) => {
153
+ const { verificationCode } = body;
154
+ const updatedVerification = await repos.verifications.updateUsingFilter(
155
+ {
156
+ filter: {
157
+ attemptId: verificationCode,
158
+ complete: { $exists: false },
159
+ createdAt: { $gt: subMinutes(verificationExpiration, new Date()) },
160
+ },
161
+ },
162
+ { complete: new Date() }
163
+ );
164
+ const verification = first(updatedVerification);
165
+ if (!verification) {
166
+ // TODO change error message to say "incorrect verification code or expired"
167
+ throw new newError.NotFound('No matching pending verification');
168
+ }
169
+
170
+ const latestVerification = await repos.verifications.findOne(
171
+ {
172
+ filter: {
173
+ type: verification.type,
174
+ value: verification.value,
175
+ },
176
+ },
177
+ { _id: 1 }
178
+ );
179
+ if (!latestVerification?._id.equals(verification._id)) {
180
+ throw new newError.NotFound('Verification was not the latest');
181
+ }
182
+
183
+ const offerId = nanoid();
184
+ const offer = {
185
+ offerId,
186
+ type: [typeToCredential[verification.type]],
187
+ value: verification.value,
188
+ offerCreationDate: new Date(),
189
+ offerExpirationDate: generateExpiration(verificationExpiration),
190
+ claimed: false,
191
+ };
192
+
193
+ await repos.verificationOffers.insert(offer);
194
+
195
+ if (vclVerificationVersion >= 2) {
196
+ return { token: offerId };
197
+ }
198
+
199
+ const { privateKey } = generateKeyPair();
200
+
201
+ const credential = await generateCredentialJwt(
202
+ {
203
+ id: verificationCode,
204
+ type: ['VerificationIdentifier'],
205
+ issuer: fastify.config.rootDid,
206
+ credentialSubject: {
207
+ id: offer.offerId,
208
+ },
209
+ },
210
+ privateKey
211
+ );
212
+
213
+ return {
214
+ credential,
215
+ };
216
+ }
217
+ );
218
+
219
+ fastify.post(
220
+ '/issuing/identify',
221
+ {
222
+ schema: fastify.autoSchema({
223
+ body: identifyWebhookRequestBodySchema,
224
+ response: {
225
+ 200: verificationIdentifyResponse200Schema,
226
+ ...fastify.NotFoundResponse,
227
+ },
228
+ }),
229
+ },
230
+ async ({ body, repos }) => {
231
+ const offerId = getOfferId(body);
232
+ const offer = await repos.verificationOffers.findOne({
233
+ filter: { offerId },
234
+ });
235
+
236
+ if (!offer) {
237
+ throw newError.NotFound(`Offer ID ${offerId} could not be found`);
238
+ }
239
+
240
+ return { vendorUserId: offerId };
241
+ }
242
+ );
243
+
244
+ fastify.post(
245
+ '/issuing/generate-offers',
246
+ {
247
+ schema: fastify.autoSchema({
248
+ body: {
249
+ type: 'object',
250
+ properties: {
251
+ vendorUserId: { type: 'string' },
252
+ exchangeId: { type: 'string' },
253
+ },
254
+ required: ['vendorUserId', 'exchangeId'],
255
+ },
256
+ response: {
257
+ 200: {
258
+ type: 'object',
259
+ properties: {
260
+ offers: {
261
+ type: 'array',
262
+ items: {
263
+ type: 'object',
264
+ properties: {
265
+ type: {
266
+ type: 'array',
267
+ items: { type: 'string' },
268
+ },
269
+ offerId: { type: 'string' },
270
+ offerCreationDate: { type: 'string' },
271
+ offerExpirationDate: { type: 'string' },
272
+ credentialSubject: {
273
+ type: 'object',
274
+ additionalProperties: true,
275
+ },
276
+ exchangeId: { type: 'string' },
277
+ },
278
+ },
279
+ },
280
+ },
281
+ },
282
+ ...fastify.NotFoundResponse,
283
+ },
284
+ }),
285
+ },
286
+ async ({ body, repos }) => {
287
+ const { vendorUserId, exchangeId } = body;
288
+ const offer = await repos.verificationOffers.findOne({
289
+ filter: { offerId: vendorUserId },
290
+ });
291
+
292
+ if (!offer) {
293
+ throw newError.NotFound(`Offer ID ${vendorUserId} could not be found`);
294
+ }
295
+
296
+ await repos.verificationOffers.upsert(offer._id, {
297
+ ...offer,
298
+ claimed: true,
299
+ });
300
+
301
+ return {
302
+ offers: [
303
+ {
304
+ ...omit(['_id', 'value'], offer),
305
+ credentialSubject: credentialGenerators[offer.type](offer),
306
+ vendorUserId,
307
+ exchangeId,
308
+ },
309
+ ],
310
+ };
311
+ }
312
+ );
313
+ };
314
+
315
+ module.exports = verificationController;
316
+ const typeToCredential = {
317
+ email: VerificationCredentialTypes.EmailAddressVerification,
318
+ phone: VerificationCredentialTypes.PhoneNumberVerification,
319
+ };
320
+
321
+ const credentialGenerators = {
322
+ [VerificationCredentialTypes.EmailAddressVerification]: (offer) => ({
323
+ email: offer.value,
324
+ vendorUserId: offer.value,
325
+ }),
326
+ [VerificationCredentialTypes.PhoneNumberVerification]: (offer) => ({
327
+ phone: offer.value,
328
+ vendorUserId: offer.value,
329
+ }),
330
+ };
331
+
332
+ const generateExpiration = (dueMinutes) => addMinutes(dueMinutes, new Date());
333
+
334
+ const getOfferId = ({ idDocumentCredentials, vendorOriginContext }) => {
335
+ if (vendorOriginContext) {
336
+ return vendorOriginContext;
337
+ }
338
+ const idDocumentCredential = idDocumentCredentials[0];
339
+ return idDocumentCredential.credentialSubject
340
+ ? idDocumentCredential.credentialSubject.id
341
+ : idDocumentCredential.id; // Deprecated
342
+ };
343
+
344
+ const createVerification = (valueType, value, attemptId) => ({
345
+ type: valueType,
346
+ value,
347
+ attemptId,
348
+ });
@@ -0,0 +1,23 @@
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: 'verifications',
11
+ entityName: 'verifications',
12
+ defaultProjection: {
13
+ _id: 1,
14
+ type: 1,
15
+ value: 1,
16
+ attemptId: 1,
17
+ complete: 1,
18
+ },
19
+ extensions: [autoboxIdsExtension],
20
+ },
21
+ app
22
+ );
23
+ };
@@ -0,0 +1,6 @@
1
+ const VerificationCredentialType = Object.freeze({
2
+ EmailAddressVerification: 'EmailV1.0',
3
+ PhoneNumberVerification: 'PhoneV1.0',
4
+ });
5
+
6
+ module.exports = VerificationCredentialType;
@@ -0,0 +1,10 @@
1
+ const { verifyAccessTokenPlugin } = require('../../plugins');
2
+ const { accountScopes } = require('../../entities');
3
+
4
+ const autohooks = async (fastify) => {
5
+ fastify
6
+ .register(verifyAccessTokenPlugin)
7
+ .autoSchemaPreset({ security: [{ bearerAuth: [accountScopes.jwt] }] });
8
+ };
9
+
10
+ module.exports = autohooks;
@@ -0,0 +1,106 @@
1
+ const { parse, toSeconds } = require('iso8601-duration');
2
+
3
+ const { v4: uuidv4 } = require('uuid');
4
+ const createError = require('http-errors');
5
+ const {
6
+ jwtDecode,
7
+ jwtVerify,
8
+ generateDocJwt,
9
+ } = require('@velocitycareerlabs/jwt');
10
+ const { generateKeyPair } = require('@velocitycareerlabs/crypto');
11
+ const {
12
+ jwtSignSchema,
13
+ jwtSignResponse200Schema,
14
+ jwtDecodeSchema,
15
+ jwtDecodeResponse200Schema,
16
+ jwtVerifySchema,
17
+ jwtVerifyResponse200Schema,
18
+ } = require('./schemas');
19
+ const { accountScopes } = require('../../entities');
20
+
21
+ const jwtController = async (fastify) => {
22
+ fastify
23
+ .addSchema(jwtSignSchema)
24
+ .addSchema(jwtSignResponse200Schema)
25
+ .addSchema(jwtVerifySchema)
26
+ .addSchema(jwtVerifyResponse200Schema)
27
+ .addSchema(jwtDecodeSchema)
28
+ .addSchema(jwtDecodeResponse200Schema)
29
+ .autoSchemaPreset({ tags: ['careerwallet_jwt'] });
30
+
31
+ fastify
32
+ .post(
33
+ '/verify',
34
+ {
35
+ onRequest: fastify.verifyCareerWalletAccessToken([accountScopes.jwt]),
36
+ schema: fastify.autoSchema({
37
+ body: jwtVerifySchema,
38
+ response: { 200: jwtVerifyResponse200Schema },
39
+ }),
40
+ },
41
+ async (req) => {
42
+ const {
43
+ body: { jwt, publicKey },
44
+ log,
45
+ } = req;
46
+ const {
47
+ header: { jwk },
48
+ } = jwtDecode(jwt);
49
+
50
+ const publicKeyAsJwk = publicKey ?? jwk;
51
+ if (publicKeyAsJwk == null) {
52
+ return createError(400, 'Public key is missing');
53
+ }
54
+
55
+ try {
56
+ await jwtVerify(jwt, publicKeyAsJwk);
57
+ return { verified: true };
58
+ } catch (err) {
59
+ log.warn({ err });
60
+ return { verified: false, error: err };
61
+ }
62
+ }
63
+ )
64
+ .post(
65
+ '/decode',
66
+ {
67
+ onRequest: fastify.verifyCareerWalletAccessToken([accountScopes.jwt]),
68
+ schema: fastify.autoSchema({
69
+ body: jwtDecodeSchema,
70
+ response: { 200: jwtDecodeResponse200Schema },
71
+ }),
72
+ },
73
+ async (req) => jwtDecode(req.body.jwt)
74
+ )
75
+ .post(
76
+ '/sign',
77
+ {
78
+ onRequest: fastify.verifyCareerWalletAccessToken([accountScopes.jwt]),
79
+ schema: fastify.autoSchema({
80
+ body: jwtSignSchema,
81
+ response: { 200: jwtSignResponse200Schema },
82
+ }),
83
+ },
84
+ async (req) => {
85
+ const {
86
+ body: { payload, options },
87
+ } = req;
88
+ const joseExpiresIn = options.expiresIn
89
+ ? `${ISO8601DurationToSeconds(options.expiresIn)}seconds`
90
+ : undefined;
91
+ const signingOptions = {
92
+ ...options,
93
+ expiresIn: joseExpiresIn,
94
+ jti: payload.id ?? uuidv4(),
95
+ };
96
+ const { privateKey } = generateKeyPair();
97
+ const jwt = await generateDocJwt(payload, privateKey, signingOptions);
98
+ return { jwt };
99
+ }
100
+ );
101
+ };
102
+
103
+ const ISO8601DurationToSeconds = (durationString) =>
104
+ toSeconds(parse(durationString));
105
+
106
+ module.exports = jwtController;
@@ -0,0 +1,31 @@
1
+ /**
2
+ * Copyright 2023 Velocity Team
3
+ *
4
+ * Licensed under the Apache License, Version 2.0 (the "License");
5
+ * you may not use this file except in compliance with the License.
6
+ * You may obtain a copy of the License at
7
+ *
8
+ * http://www.apache.org/licenses/LICENSE-2.0
9
+ *
10
+ * Unless required by applicable law or agreed to in writing, software
11
+ * distributed under the License is distributed on an "AS IS" BASIS,
12
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13
+ * See the License for the specific language governing permissions and
14
+ * limitations under the License.
15
+ */
16
+
17
+ const jwtSignSchema = require('./jwt-sign.schema.json');
18
+ const jwtSignResponse200Schema = require('./jwt-sign.response.200.schema.json');
19
+ const jwtDecodeSchema = require('./jwt-decode.schema.json');
20
+ const jwtDecodeResponse200Schema = require('./jwt-decode.response.200.schema.json');
21
+ const jwtVerifySchema = require('./jwt-verify.schema.json');
22
+ const jwtVerifyResponse200Schema = require('./jwt-verify.response.200.schema.json');
23
+
24
+ module.exports = {
25
+ jwtSignSchema,
26
+ jwtSignResponse200Schema,
27
+ jwtDecodeSchema,
28
+ jwtDecodeResponse200Schema,
29
+ jwtVerifySchema,
30
+ jwtVerifyResponse200Schema,
31
+ };
@@ -0,0 +1,20 @@
1
+ {
2
+ "title": "jwt-decode-response-200",
3
+ "$id": "jwt-decode-response-200",
4
+ "type": "object",
5
+ "properties": {
6
+ "payload": {
7
+ "type": "object",
8
+ "additionalProperties": true
9
+ },
10
+ "header": {
11
+ "type": "object",
12
+ "additionalProperties": true
13
+ }
14
+ },
15
+ "required": [
16
+ "payload",
17
+ "header"
18
+ ],
19
+ "additionalProperties": false
20
+ }
@@ -0,0 +1,15 @@
1
+ {
2
+ "title": "jwts-decode",
3
+ "$id": "jwts-decode",
4
+ "type": "object",
5
+ "properties": {
6
+ "jwt": {
7
+ "type": "string",
8
+ "description": "a jwt string."
9
+ }
10
+ },
11
+ "required": [
12
+ "jwt"
13
+ ],
14
+ "additionalProperties": false
15
+ }
@@ -0,0 +1,14 @@
1
+ {
2
+ "title": "jwt-sign-response-200",
3
+ "$id": "jwt-sign-response-200",
4
+ "type": "object",
5
+ "properties": {
6
+ "jwt": {
7
+ "type": "string"
8
+ }
9
+ },
10
+ "required": [
11
+ "jwt"
12
+ ],
13
+ "additionalProperties": false
14
+ }
@@ -0,0 +1,40 @@
1
+ {
2
+ "title": "jwt-sign",
3
+ "$id": "jwt-sign",
4
+ "type": "object",
5
+ "properties": {
6
+ "payload": {
7
+ "type": "object"
8
+ },
9
+ "options": {
10
+ "type": "object",
11
+ "properties": {
12
+ "issuer": {
13
+ "type": "string",
14
+ "description": "did or random uuid of the issuer"
15
+ },
16
+ "audience": {
17
+ "type": "string",
18
+ "description": "did or random uuid of the audience"
19
+ },
20
+ "subject": {
21
+ "type": "string",
22
+ "description": "did or random uuid of the subject"
23
+ },
24
+ "expiresIn": {
25
+ "type": "string",
26
+ "description": "a ISO 8601 formatted duration string"
27
+ }
28
+ },
29
+ "required": [
30
+ "issuer",
31
+ "subject"
32
+ ]
33
+ }
34
+ },
35
+ "required": [
36
+ "payload",
37
+ "options"
38
+ ],
39
+ "additionalProperties": false
40
+ }
@@ -0,0 +1,17 @@
1
+ {
2
+ "title": "jwt-verify-response-200",
3
+ "$id": "jwt-verify-response-200",
4
+ "type": "object",
5
+ "properties": {
6
+ "verified": {
7
+ "type": "boolean"
8
+ },
9
+ "error": {
10
+ "type": "string"
11
+ }
12
+ },
13
+ "required": [
14
+ "verified"
15
+ ],
16
+ "additionalProperties": false
17
+ }
@@ -0,0 +1,19 @@
1
+ {
2
+ "title": "jwts-verify",
3
+ "$id": "jwts-verify",
4
+ "type": "object",
5
+ "properties": {
6
+ "jwt": {
7
+ "type": "string",
8
+ "description": "jwt string"
9
+ },
10
+ "publicKey": {
11
+ "type": "object",
12
+ "description": "jwk public key"
13
+ }
14
+ },
15
+ "required": [
16
+ "jwt"
17
+ ],
18
+ "additionalProperties": false
19
+ }
@@ -0,0 +1,5 @@
1
+ const autohooks = async (fastify) => {
2
+ fastify.autoSchemaPreset({ tags: ['careerwallet'] });
3
+ };
4
+
5
+ module.exports = autohooks;