@verii/server-credentialagent 1.0.0-pre.1752076816

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 (395) hide show
  1. package/.localdev.e2e.env +40 -0
  2. package/.localdev.env +41 -0
  3. package/.standalone.env +5 -0
  4. package/LICENSE +202 -0
  5. package/NOTICE +1 -0
  6. package/README.md +19 -0
  7. package/docker/compose.yml +33 -0
  8. package/e2e/README.md +12 -0
  9. package/e2e/org-registration-and-issuing.e2e.test.js +624 -0
  10. package/jest.config.js +20 -0
  11. package/migrate-mongo.config.js +36 -0
  12. package/migrations/20210317133137-add-index-to-offers-repo.js +57 -0
  13. package/migrations/20210416145639-add-index-to-revocation-list.js +27 -0
  14. package/migrations/20210719120225-add_unique_did_index_to_tenant.js +45 -0
  15. package/migrations/20230524053029-add-vendorUserIdMappings-index.js +32 -0
  16. package/migrations/20230616111907-add-configuration-type-index.js +32 -0
  17. package/package.json +108 -0
  18. package/src/assets/public/favicon.ico +0 -0
  19. package/src/assets/public/logo192.png +0 -0
  20. package/src/assets/public/logo512.png +0 -0
  21. package/src/assets/public/manifest.json +28 -0
  22. package/src/assets/templates/app-redirect.hbs +16 -0
  23. package/src/config/config.js +44 -0
  24. package/src/config/core-config.js +143 -0
  25. package/src/config/holder-config.js +104 -0
  26. package/src/config/index.js +22 -0
  27. package/src/config/operator-config.js +64 -0
  28. package/src/controllers/autoload-holder-api-controllers.js +30 -0
  29. package/src/controllers/autoload-operator-api-controllers.js +31 -0
  30. package/src/controllers/autoload-root-api-controller.js +30 -0
  31. package/src/controllers/autoload-saasoperator-api-controllers.js +31 -0
  32. package/src/controllers/holder/autohooks.js +55 -0
  33. package/src/controllers/holder/get-exchange-progress/autohooks.js +27 -0
  34. package/src/controllers/holder/get-exchange-progress/controller.js +50 -0
  35. package/src/controllers/holder/inspect/autohooks.js +35 -0
  36. package/src/controllers/holder/inspect/get-presentation-request/controller.js +100 -0
  37. package/src/controllers/holder/inspect/schemas/holder-disclosure.schema.json +73 -0
  38. package/src/controllers/holder/inspect/schemas/index.js +33 -0
  39. package/src/controllers/holder/inspect/schemas/presentation-definition.v1.schema.json +461 -0
  40. package/src/controllers/holder/inspect/schemas/presentation-request.schema.json +279 -0
  41. package/src/controllers/holder/inspect/schemas/presentation-submission.schema.json +41 -0
  42. package/src/controllers/holder/inspect/schemas/siop-presentation-submission.schema.json +74 -0
  43. package/src/controllers/holder/inspect/schemas/velocity-presentation-submission.response.200.schema.json +36 -0
  44. package/src/controllers/holder/inspect/schemas/velocity-presentation-submission.schema.json +34 -0
  45. package/src/controllers/holder/inspect/submit-presentation/controller.js +89 -0
  46. package/src/controllers/holder/issue/autohooks.js +23 -0
  47. package/src/controllers/holder/issue/get-credential-manifest/controller.js +193 -0
  48. package/src/controllers/holder/issue/offers/autohooks.js +35 -0
  49. package/src/controllers/holder/issue/offers/controller.js +164 -0
  50. package/src/controllers/holder/issue/offers/credential-offers/controller.js +460 -0
  51. package/src/controllers/holder/issue/submit-identification/autohooks.js +37 -0
  52. package/src/controllers/holder/issue/submit-identification/controller.js +63 -0
  53. package/src/controllers/holder/oauth/autohooks.js +19 -0
  54. package/src/controllers/holder/oauth/controller.js +140 -0
  55. package/src/controllers/index.js +22 -0
  56. package/src/controllers/operator/tenants/_tenantId/autohooks.js +40 -0
  57. package/src/controllers/operator/tenants/_tenantId/check-credentials/autohooks.js +24 -0
  58. package/src/controllers/operator/tenants/_tenantId/check-credentials/controller-v0.8.js +200 -0
  59. package/src/controllers/operator/tenants/_tenantId/check-credentials/schemas/index.js +19 -0
  60. package/src/controllers/operator/tenants/_tenantId/check-credentials/schemas/vendor-credential.schema.json +244 -0
  61. package/src/controllers/operator/tenants/_tenantId/controller-v0.8.js +221 -0
  62. package/src/controllers/operator/tenants/_tenantId/disclosures/_id/autohooks.js +30 -0
  63. package/src/controllers/operator/tenants/_tenantId/disclosures/_id/controller-v0.8.js +271 -0
  64. package/src/controllers/operator/tenants/_tenantId/disclosures/_id/feeds/autohooks.js +45 -0
  65. package/src/controllers/operator/tenants/_tenantId/disclosures/_id/feeds/controller-v0.8.js +199 -0
  66. package/src/controllers/operator/tenants/_tenantId/disclosures/_id/feeds/schemas/add-feed.schema.js +14 -0
  67. package/src/controllers/operator/tenants/_tenantId/disclosures/_id/feeds/schemas/feed.schema.json +27 -0
  68. package/src/controllers/operator/tenants/_tenantId/disclosures/_id/feeds/schemas/index.js +25 -0
  69. package/src/controllers/operator/tenants/_tenantId/disclosures/_id/feeds/schemas/modify-feed-update-body.schema.js +18 -0
  70. package/src/controllers/operator/tenants/_tenantId/disclosures/_id/feeds/schemas/modify-feed.schema.json +19 -0
  71. package/src/controllers/operator/tenants/_tenantId/disclosures/autohooks.js +34 -0
  72. package/src/controllers/operator/tenants/_tenantId/disclosures/controller-v0.8.js +100 -0
  73. package/src/controllers/operator/tenants/_tenantId/disclosures/schemas/agent-disclosure-presentation-definition.schema.json +404 -0
  74. package/src/controllers/operator/tenants/_tenantId/disclosures/schemas/agent-disclosure.schema.js +24 -0
  75. package/src/controllers/operator/tenants/_tenantId/disclosures/schemas/index.js +29 -0
  76. package/src/controllers/operator/tenants/_tenantId/disclosures/schemas/new-agent-disclosure.schema.json +166 -0
  77. package/src/controllers/operator/tenants/_tenantId/disclosures/schemas/update-agent-disclosure.schema.js +20 -0
  78. package/src/controllers/operator/tenants/_tenantId/exchanges/_exchangeId/autohooks.js +30 -0
  79. package/src/controllers/operator/tenants/_tenantId/exchanges/_exchangeId/controller-v0.8.js +73 -0
  80. package/src/controllers/operator/tenants/_tenantId/exchanges/autohooks.js +19 -0
  81. package/src/controllers/operator/tenants/_tenantId/exchanges/controller-v0.8.js +150 -0
  82. package/src/controllers/operator/tenants/_tenantId/exchanges/schemas/get-exchange.response.body.json +147 -0
  83. package/src/controllers/operator/tenants/_tenantId/exchanges/schemas/index.js +21 -0
  84. package/src/controllers/operator/tenants/_tenantId/issued-credentials/autohooks.js +27 -0
  85. package/src/controllers/operator/tenants/_tenantId/issued-credentials/controller-v0.8.js +303 -0
  86. package/src/controllers/operator/tenants/_tenantId/issued-credentials/schemas/index.js +23 -0
  87. package/src/controllers/operator/tenants/_tenantId/issued-credentials/schemas/issued-credential.schema.json +115 -0
  88. package/src/controllers/operator/tenants/_tenantId/issued-credentials/schemas/revoke-credentials.schema.json +18 -0
  89. package/src/controllers/operator/tenants/_tenantId/keys/controller-v0.8.js +168 -0
  90. package/src/controllers/operator/tenants/_tenantId/offer-data/controller-v0.8.js +78 -0
  91. package/src/controllers/operator/tenants/_tenantId/offers/autohooks.js +34 -0
  92. package/src/controllers/operator/tenants/_tenantId/offers/controller-v0.8.js +253 -0
  93. package/src/controllers/operator/tenants/_tenantId/offers/schemas/index.js +23 -0
  94. package/src/controllers/operator/tenants/_tenantId/offers/schemas/new-vendor-offer.schema.js +47 -0
  95. package/src/controllers/operator/tenants/_tenantId/offers/schemas/vendor-offer.schema.json +56 -0
  96. package/src/controllers/operator/tenants/_tenantId/users/autohooks.js +24 -0
  97. package/src/controllers/operator/tenants/_tenantId/users/controller-v0.8.js +92 -0
  98. package/src/controllers/operator/tenants/_tenantId/users/schemas/index.js +23 -0
  99. package/src/controllers/operator/tenants/_tenantId/users/schemas/new-user.schema.json +13 -0
  100. package/src/controllers/operator/tenants/_tenantId/users/schemas/user.schema.json +16 -0
  101. package/src/controllers/operator/tenants/_tenantId/vc-api/credentials/autohooks.js +34 -0
  102. package/src/controllers/operator/tenants/_tenantId/vc-api/credentials/controller-v0.8.js +110 -0
  103. package/src/controllers/operator/tenants/_tenantId/vc-api/credentials/schemas/Credential.schema.js +18 -0
  104. package/src/controllers/operator/tenants/_tenantId/vc-api/credentials/schemas/IssueCredentialOptions.schema.json +42 -0
  105. package/src/controllers/operator/tenants/_tenantId/vc-api/credentials/schemas/IssueCredentialRequest.schema.json +13 -0
  106. package/src/controllers/operator/tenants/_tenantId/vc-api/credentials/schemas/IssueCredentialResponse.schema.json +19 -0
  107. package/src/controllers/operator/tenants/_tenantId/vc-api/credentials/schemas/LinkedDataProof.schema.json +43 -0
  108. package/src/controllers/operator/tenants/_tenantId/vc-api/credentials/schemas/VerifiableCredential.schema.js +16 -0
  109. package/src/controllers/operator/tenants/_tenantId/vc-api/credentials/schemas/index.js +31 -0
  110. package/src/controllers/operator/tenants/autohooks.js +65 -0
  111. package/src/controllers/operator/tenants/controller-v0.8.js +167 -0
  112. package/src/controllers/operator/tenants/schemas/index.js +41 -0
  113. package/src/controllers/operator/tenants/schemas/modify-secret.schema.json +11 -0
  114. package/src/controllers/operator/tenants/schemas/modify-tenant-v0.8.schema.json +44 -0
  115. package/src/controllers/operator/tenants/schemas/new-tenant-v0.8.schema.json +19 -0
  116. package/src/controllers/operator/tenants/schemas/new-tenant.response.200.schema.json +7 -0
  117. package/src/controllers/operator/tenants/schemas/secret-key-metadata.schema.json +31 -0
  118. package/src/controllers/operator/tenants/schemas/secret-key.schema.json +29 -0
  119. package/src/controllers/operator/tenants/schemas/secret-kid.schema.json +13 -0
  120. package/src/controllers/operator/tenants/schemas/secret-new-tenant-v0.8.schema.json +28 -0
  121. package/src/controllers/operator/tenants/schemas/secret-tenant-key-v0.8.schema.json +13 -0
  122. package/src/controllers/operator/tenants/schemas/tenant-key-v0.8.schema.json +14 -0
  123. package/src/controllers/operator/tenants/schemas/tenant-v0.8.schema.json +62 -0
  124. package/src/controllers/root/autohooks.js +23 -0
  125. package/src/controllers/root/controller.js +173 -0
  126. package/src/controllers/saasoperator/groups/_id/autohooks.js +9 -0
  127. package/src/controllers/saasoperator/groups/_id/controller.js +121 -0
  128. package/src/controllers/saasoperator/groups/autohooks.js +19 -0
  129. package/src/controllers/saasoperator/groups/controller.js +65 -0
  130. package/src/controllers/saasoperator/groups/schemas/group.schema.js +17 -0
  131. package/src/controllers/saasoperator/groups/schemas/index.js +4 -0
  132. package/src/controllers/saasoperator/groups/schemas/new-group.schema.js +13 -0
  133. package/src/entities/common/domains/get-json-at-path.js +28 -0
  134. package/src/entities/common/domains/index.js +17 -0
  135. package/src/entities/common/index.js +17 -0
  136. package/src/entities/credentials/domains/credential-format.js +22 -0
  137. package/src/entities/credentials/domains/index.js +19 -0
  138. package/src/entities/credentials/index.js +17 -0
  139. package/src/entities/deep-links/domains/extract-did.js +11 -0
  140. package/src/entities/deep-links/domains/index.js +20 -0
  141. package/src/entities/deep-links/domains/velocity-protocol-uri-to-http-uri.js +32 -0
  142. package/src/entities/deep-links/index.js +19 -0
  143. package/src/entities/disclosures/domains/assert-disclosure-active.js +21 -0
  144. package/src/entities/disclosures/domains/compute-disclosure-configuration-type.js +29 -0
  145. package/src/entities/disclosures/domains/constants.js +61 -0
  146. package/src/entities/disclosures/domains/errors.js +34 -0
  147. package/src/entities/disclosures/domains/get-disclosure-configuration-type.js +60 -0
  148. package/src/entities/disclosures/domains/index.js +32 -0
  149. package/src/entities/disclosures/domains/is-issuing-disclosure.js +23 -0
  150. package/src/entities/disclosures/domains/parse-body-to-disclosure.js +17 -0
  151. package/src/entities/disclosures/domains/validate-by-identification-method.js +69 -0
  152. package/src/entities/disclosures/domains/validate-commercial-entity.js +26 -0
  153. package/src/entities/disclosures/domains/validate-disclosure-by-configuration-type.js +47 -0
  154. package/src/entities/disclosures/domains/validate-disclosure-default-issuing.js +77 -0
  155. package/src/entities/disclosures/domains/validate-disclosure.js +37 -0
  156. package/src/entities/disclosures/domains/validate-feed.js +16 -0
  157. package/src/entities/disclosures/domains/validate-presentation-definition.js +54 -0
  158. package/src/entities/disclosures/domains/validate-vendor-endpoint.js +22 -0
  159. package/src/entities/disclosures/domains/validate-vendor-webhook.js +18 -0
  160. package/src/entities/disclosures/factories/disclosure-factory.js +94 -0
  161. package/src/entities/disclosures/factories/index.js +19 -0
  162. package/src/entities/disclosures/index.js +22 -0
  163. package/src/entities/disclosures/orchestrators/get-disclosure.js +18 -0
  164. package/src/entities/disclosures/orchestrators/index.js +20 -0
  165. package/src/entities/disclosures/orchestrators/update-disclosure-configuration-type.js +32 -0
  166. package/src/entities/disclosures/repos/index.js +20 -0
  167. package/src/entities/disclosures/repos/repo.js +118 -0
  168. package/src/entities/disclosures/repos/set-configuration-type.js +33 -0
  169. package/src/entities/exchanges/adapters/index.js +17 -0
  170. package/src/entities/exchanges/adapters/sign-exchange-response.js +45 -0
  171. package/src/entities/exchanges/domains/build-exchange-progress.js +56 -0
  172. package/src/entities/exchanges/domains/constants.js +24 -0
  173. package/src/entities/exchanges/domains/ensure-exchange-state-valid.js +35 -0
  174. package/src/entities/exchanges/domains/errors.js +33 -0
  175. package/src/entities/exchanges/domains/index.js +25 -0
  176. package/src/entities/exchanges/domains/states.js +43 -0
  177. package/src/entities/exchanges/domains/types.js +31 -0
  178. package/src/entities/exchanges/factories/disclosure-exchange-factory.js +46 -0
  179. package/src/entities/exchanges/factories/index.js +20 -0
  180. package/src/entities/exchanges/factories/offer-exchange-factory.js +48 -0
  181. package/src/entities/exchanges/index.js +23 -0
  182. package/src/entities/exchanges/orchestrators/build-exchange-request-deep-link.js +50 -0
  183. package/src/entities/exchanges/orchestrators/index.js +19 -0
  184. package/src/entities/exchanges/repos/exchange-repo-projections.js +45 -0
  185. package/src/entities/exchanges/repos/exchange-state-repo-extension.js +76 -0
  186. package/src/entities/exchanges/repos/index.js +20 -0
  187. package/src/entities/exchanges/repos/repo.js +44 -0
  188. package/src/entities/feeds/factories/feed-factory.js +47 -0
  189. package/src/entities/feeds/factories/index.js +19 -0
  190. package/src/entities/feeds/index.js +20 -0
  191. package/src/entities/feeds/repos/index.js +19 -0
  192. package/src/entities/feeds/repos/repo.js +95 -0
  193. package/src/entities/groups/domains/format-group.js +11 -0
  194. package/src/entities/groups/domains/index.js +3 -0
  195. package/src/entities/groups/factories/group-factory.js +40 -0
  196. package/src/entities/groups/factories/index.js +19 -0
  197. package/src/entities/groups/index.js +22 -0
  198. package/src/entities/groups/orchestrators/find-group-or-error.js +16 -0
  199. package/src/entities/groups/orchestrators/index.js +6 -0
  200. package/src/entities/groups/orchestrators/validate-did.js +24 -0
  201. package/src/entities/groups/orchestrators/validate-group-by-user.js +16 -0
  202. package/src/entities/groups/orchestrators/validate-group.js +39 -0
  203. package/src/entities/groups/repos/delete-tenant-extension.js +13 -0
  204. package/src/entities/groups/repos/index.js +19 -0
  205. package/src/entities/groups/repos/repo.js +38 -0
  206. package/src/entities/groups/repos/update-or-error-extension.js +46 -0
  207. package/src/entities/index.js +37 -0
  208. package/src/entities/keys/domains/constants.js +37 -0
  209. package/src/entities/keys/domains/index.js +21 -0
  210. package/src/entities/keys/domains/is-matching-private-key-kid.js +41 -0
  211. package/src/entities/keys/domains/validate-key.js +62 -0
  212. package/src/entities/keys/factories/index.js +19 -0
  213. package/src/entities/keys/factories/key-factory.js +56 -0
  214. package/src/entities/keys/index.js +22 -0
  215. package/src/entities/keys/orchestrators/index.js +3 -0
  216. package/src/entities/keys/orchestrators/validate-did-doc-keys.js +69 -0
  217. package/src/entities/metadata-list-allocations/index.js +19 -0
  218. package/src/entities/metadata-list-allocations/repos/index.js +19 -0
  219. package/src/entities/metadata-list-allocations/repos/repo.js +40 -0
  220. package/src/entities/notifications/domains/index.js +19 -0
  221. package/src/entities/notifications/domains/notification-types.js +25 -0
  222. package/src/entities/notifications/index.js +19 -0
  223. package/src/entities/offers/domains/build-clean-pii-filter.js +35 -0
  224. package/src/entities/offers/domains/build-deeplink-url.js +120 -0
  225. package/src/entities/offers/domains/build-offer.js +88 -0
  226. package/src/entities/offers/domains/build-qr-code-url.js +37 -0
  227. package/src/entities/offers/domains/constants.js +32 -0
  228. package/src/entities/offers/domains/filter-object-ids.js +34 -0
  229. package/src/entities/offers/domains/generate-issuing-challenge.js +26 -0
  230. package/src/entities/offers/domains/generate-link-code.js +35 -0
  231. package/src/entities/offers/domains/index.js +31 -0
  232. package/src/entities/offers/domains/post-validation-offers-handler.js +31 -0
  233. package/src/entities/offers/domains/prepare-linked-credentials-for-holder.js +36 -0
  234. package/src/entities/offers/domains/resolve-subject.js +142 -0
  235. package/src/entities/offers/domains/validate-offer-commercial-entity.js +24 -0
  236. package/src/entities/offers/domains/validate-offer.js +90 -0
  237. package/src/entities/offers/factories/index.js +19 -0
  238. package/src/entities/offers/factories/offer-factory.js +119 -0
  239. package/src/entities/offers/index.js +22 -0
  240. package/src/entities/offers/orchestrators/create-verifiable-credentials.js +131 -0
  241. package/src/entities/offers/orchestrators/finalize-exchange.js +44 -0
  242. package/src/entities/offers/orchestrators/index.js +23 -0
  243. package/src/entities/offers/orchestrators/load-credential-refs.js +57 -0
  244. package/src/entities/offers/orchestrators/load-credential-types-map.js +44 -0
  245. package/src/entities/offers/orchestrators/prepare-offers.js +35 -0
  246. package/src/entities/offers/orchestrators/trigger-issued-credentials-webhook.js +63 -0
  247. package/src/entities/offers/repos/clean-pii-extension.js +85 -0
  248. package/src/entities/offers/repos/index.js +20 -0
  249. package/src/entities/offers/repos/issued-credential-projection.js +44 -0
  250. package/src/entities/offers/repos/repo.js +177 -0
  251. package/src/entities/presentations/domains/build-identity-doc.js +120 -0
  252. package/src/entities/presentations/domains/build-request-response-schema.js +46 -0
  253. package/src/entities/presentations/domains/build-vendor-data.js +31 -0
  254. package/src/entities/presentations/domains/check-payment-requirement.js +30 -0
  255. package/src/entities/presentations/domains/errors.js +28 -0
  256. package/src/entities/presentations/domains/extract-fields-from-id-credential.js +35 -0
  257. package/src/entities/presentations/domains/index.js +26 -0
  258. package/src/entities/presentations/domains/merge-credential-check-results.js +24 -0
  259. package/src/entities/presentations/domains/validate-presentation.js +128 -0
  260. package/src/entities/presentations/index.js +20 -0
  261. package/src/entities/presentations/orchestrators/create-presentation-request.js +148 -0
  262. package/src/entities/presentations/orchestrators/deduplicate-disclosure-exchange.js +52 -0
  263. package/src/entities/presentations/orchestrators/handle-presentation-submission.js +47 -0
  264. package/src/entities/presentations/orchestrators/index.js +20 -0
  265. package/src/entities/presentations/orchestrators/match-identity-on-exchange.js +114 -0
  266. package/src/entities/presentations/orchestrators/share-identification-credentials.js +110 -0
  267. package/src/entities/presentations/orchestrators/share-presentation.js +234 -0
  268. package/src/entities/push-delegate/get-push-delegate.js +37 -0
  269. package/src/entities/push-delegate/index.js +17 -0
  270. package/src/entities/redirect/index.js +3 -0
  271. package/src/entities/redirect/orchestrators/index.js +3 -0
  272. package/src/entities/redirect/orchestrators/load-org-info.js +40 -0
  273. package/src/entities/revocation-list-allocations/index.js +19 -0
  274. package/src/entities/revocation-list-allocations/repos/index.js +19 -0
  275. package/src/entities/revocation-list-allocations/repos/repo.js +40 -0
  276. package/src/entities/schemas/index.js +19 -0
  277. package/src/entities/schemas/orchestrators/index.js +19 -0
  278. package/src/entities/schemas/orchestrators/load-schema-validation.js +73 -0
  279. package/src/entities/tenants/domains/build-service-ids.js +27 -0
  280. package/src/entities/tenants/domains/extract-service.js +27 -0
  281. package/src/entities/tenants/domains/index.js +21 -0
  282. package/src/entities/tenants/domains/validate-service-ids.js +35 -0
  283. package/src/entities/tenants/factories/index.js +19 -0
  284. package/src/entities/tenants/factories/tenant-factory.js +37 -0
  285. package/src/entities/tenants/index.js +22 -0
  286. package/src/entities/tenants/orchestrators/add-primary-address-to-tenant.js +47 -0
  287. package/src/entities/tenants/orchestrators/create-tenant.js +91 -0
  288. package/src/entities/tenants/orchestrators/index.js +22 -0
  289. package/src/entities/tenants/orchestrators/refresh-tenant-dids.js +146 -0
  290. package/src/entities/tenants/orchestrators/set-tenant-default-issuing-disclosure.js +31 -0
  291. package/src/entities/tenants/repos/index.js +20 -0
  292. package/src/entities/tenants/repos/insert-tenant-extension.js +33 -0
  293. package/src/entities/tenants/repos/repo.js +52 -0
  294. package/src/entities/tenants/repos/tenant-default-projection.js +33 -0
  295. package/src/entities/tokens/adapters/access-token.js +49 -0
  296. package/src/entities/tokens/adapters/index.js +19 -0
  297. package/src/entities/tokens/index.js +19 -0
  298. package/src/entities/users/factories/index.js +19 -0
  299. package/src/entities/users/factories/user-factory.js +36 -0
  300. package/src/entities/users/index.js +20 -0
  301. package/src/entities/users/repos/add-anonymous-user-repo-extension.js +23 -0
  302. package/src/entities/users/repos/find-or-insert-vendor-user-repo-extension.js +30 -0
  303. package/src/entities/users/repos/index.js +19 -0
  304. package/src/entities/users/repos/repo.js +50 -0
  305. package/src/fetchers/index.js +20 -0
  306. package/src/fetchers/operator/identify-fetcher.js +36 -0
  307. package/src/fetchers/operator/index.js +21 -0
  308. package/src/fetchers/operator/inspection-fetcher.js +35 -0
  309. package/src/fetchers/operator/issuing-fetcher.js +50 -0
  310. package/src/fetchers/operator/webhook-auth-header.js +45 -0
  311. package/src/fetchers/push-gateway/generate-push-gateway-token.js +40 -0
  312. package/src/fetchers/push-gateway/index.js +19 -0
  313. package/src/fetchers/push-gateway/push-fetcher.js +39 -0
  314. package/src/index.js +19 -0
  315. package/src/init-holder-server.js +108 -0
  316. package/src/init-operator-server.js +101 -0
  317. package/src/init-server.js +120 -0
  318. package/src/main-holder.js +18 -0
  319. package/src/main-operator.js +19 -0
  320. package/src/main.js +18 -0
  321. package/src/plugins/autoload-repos.js +28 -0
  322. package/src/plugins/disclosure-loader-plugin.js +56 -0
  323. package/src/plugins/ensure-disclosure-active-plugin.js +30 -0
  324. package/src/plugins/ensure-disclosure-configuration-type-plugin.js +29 -0
  325. package/src/plugins/ensure-tenant-default-issuing-disclosure-id-plugin.js +60 -0
  326. package/src/plugins/ensure-tenant-primary-address-plugin.js +44 -0
  327. package/src/plugins/exchange-error-handler-plugin.js +51 -0
  328. package/src/plugins/exchange-loader-plugin.js +50 -0
  329. package/src/plugins/group-loader-plugin.js +51 -0
  330. package/src/plugins/index.js +32 -0
  331. package/src/plugins/kms-plugin.js +57 -0
  332. package/src/plugins/tenant-loader-plugin.js +91 -0
  333. package/src/plugins/validate-cao-plugin.js +81 -0
  334. package/src/plugins/vendor-routes-auth-plugin.js +24 -0
  335. package/src/plugins/verify-access-token-plugin.js +88 -0
  336. package/src/standalone.js +24 -0
  337. package/src/start-app-server.js +38 -0
  338. package/test/combined/app-redirect.test.js +199 -0
  339. package/test/combined/helpers/credentialagent-build-fastify.js +29 -0
  340. package/test/combined/helpers/index.js +22 -0
  341. package/test/combined/helpers/nock-registrar-app-schema-name.js +50 -0
  342. package/test/combined/helpers/nock-registrar-get-organization-diddoc.js +26 -0
  343. package/test/combined/helpers/nock-registrar-get-organization-verified-profile.js +33 -0
  344. package/test/combined/manifest.json.test.js +55 -0
  345. package/test/combined/root-controller.test.js +42 -0
  346. package/test/combined/schemas/education-degree.schema.json +166 -0
  347. package/test/combined/schemas/employment-current-v1.1.schema.json +253 -0
  348. package/test/combined/schemas/open-badge-credential.schema.json +1285 -0
  349. package/test/combined/schemas/past-employment-position-with-uri-id.schema.js +22 -0
  350. package/test/combined/schemas/past-employment-position.schema.json +148 -0
  351. package/test/combined/schemas/will-always-validate.json +10 -0
  352. package/test/combined/validate-cao-plugin.test.js +155 -0
  353. package/test/get-push-delegate.test.js +54 -0
  354. package/test/helpers/jwt-vc-expectation.js +109 -0
  355. package/test/holder/build-request-response-schema.test.js +55 -0
  356. package/test/holder/credential-manifest-controller.test.js +3192 -0
  357. package/test/holder/e2e-issuing-controller.test.js +425 -0
  358. package/test/holder/get-exchange-progress-controller.test.js +521 -0
  359. package/test/holder/get-presentation-request.test.js +906 -0
  360. package/test/holder/helpers/credential-type-metadata.js +98 -0
  361. package/test/holder/helpers/credentialagent-holder-build-fastify.js +32 -0
  362. package/test/holder/helpers/generate-presentation.js +441 -0
  363. package/test/holder/helpers/generate-test-access-token.js +54 -0
  364. package/test/holder/helpers/jwt-access-token-expectation.js +32 -0
  365. package/test/holder/helpers/jwt-vc-expectation.js +115 -0
  366. package/test/holder/issuing-controller.test.js +7076 -0
  367. package/test/holder/oauth-token-controller.test.js +412 -0
  368. package/test/holder/presentation-submission.test.js +2365 -0
  369. package/test/holder/submit-identification.test.js +4815 -0
  370. package/test/operator/check-credentials-controller-v0.8.test.js +832 -0
  371. package/test/operator/credentials-revoke.test.js +536 -0
  372. package/test/operator/disclosures-controller-v0.8.test.js +4157 -0
  373. package/test/operator/exchanges-controller-v0.8.test.js +414 -0
  374. package/test/operator/exchanges-id-controller-v0.8.test.js +162 -0
  375. package/test/operator/feeds-controller-v0.8.test.js +659 -0
  376. package/test/operator/generate-push-gateway-token.test.js +116 -0
  377. package/test/operator/groups-controller.test.js +145 -0
  378. package/test/operator/groups-id-controller.test.js +287 -0
  379. package/test/operator/helpers/create-test-org-doc.js +60 -0
  380. package/test/operator/helpers/credentialagent-operator-build-fastify.js +32 -0
  381. package/test/operator/helpers/find-kms-key.js +31 -0
  382. package/test/operator/helpers/generate-primary-and-add-operator-to-primary.js +63 -0
  383. package/test/operator/helpers/init-agent-kms.js +22 -0
  384. package/test/operator/issued-credentials-controller-v0.8.test.js +398 -0
  385. package/test/operator/keys-controller-v0.8.test.js +1130 -0
  386. package/test/operator/offer-data-controller-v0.8.test.js +253 -0
  387. package/test/operator/offers-controller-v0.8.test.js +3026 -0
  388. package/test/operator/set-configuration-type-modifier.test.js +75 -0
  389. package/test/operator/swagger.test.js +37 -0
  390. package/test/operator/tenant-controller-v0.8.test.js +730 -0
  391. package/test/operator/tenant-loader-plugin.test.js +96 -0
  392. package/test/operator/tenants-controller-v0.8.test.js +2093 -0
  393. package/test/operator/users-controller-v0.8.test.js +137 -0
  394. package/test/operator/vc-api-credentials.test.js +963 -0
  395. package/verification.env +28 -0
@@ -0,0 +1,963 @@
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 { mongoDb } = require('@spencejs/spence-mongo-repos');
18
+ const { ObjectId } = require('mongodb');
19
+ const { mongoify, errorResponseMatcher } = require('@verii/tests-helpers');
20
+ const nock = require('nock');
21
+ const { generateKeyPair } = require('@verii/crypto');
22
+ const { getDidUriFromJwk } = require('@verii/did-doc');
23
+ const { jwtDecode } = require('@verii/jwt');
24
+ const { BASE64_FORMAT } = require('@verii/test-regexes');
25
+ const {
26
+ castArray,
27
+ compact,
28
+ entries,
29
+ every,
30
+ includes,
31
+ keyBy,
32
+ map,
33
+ omit,
34
+ pick,
35
+ set,
36
+ uniq,
37
+ } = require('lodash/fp');
38
+ const { getUnixTime, subHours } = require('date-fns/fp');
39
+ const {
40
+ extractCredentialType,
41
+ VelocityRevocationListType,
42
+ } = require('@verii/vc-checks');
43
+ const { ISO_DATETIME_FORMAT } = require('@verii/test-regexes');
44
+ const { openBadgeCredentialExample } = require('@verii/sample-data');
45
+ const buildFastify = require('./helpers/credentialagent-operator-build-fastify');
46
+ const {
47
+ initKeysFactory,
48
+ initTenantFactory,
49
+ initOfferFactory,
50
+ ExchangeStates,
51
+ CredentialFormat,
52
+ ExchangeProtocols,
53
+ ExchangeTypes,
54
+ } = require('../../src/entities');
55
+ const { nockRegistrarAppSchemaName } = require('../combined/helpers');
56
+
57
+ const mockAddCredentialMetadataEntry = jest.fn();
58
+ const mockCreateCredentialMetadataList = jest.fn();
59
+ const mockAddRevocationListSigned = jest.fn();
60
+ const mockGetRevokeUrl = jest.fn();
61
+
62
+ jest.mock('@verii/metadata-registration', () => ({
63
+ initRevocationRegistry: () => ({
64
+ addRevocationListSigned: mockAddRevocationListSigned,
65
+ }),
66
+ initMetadataRegistry: () => ({
67
+ addCredentialMetadataEntry: mockAddCredentialMetadataEntry,
68
+ createCredentialMetadataList: mockCreateCredentialMetadataList,
69
+ }),
70
+ }));
71
+
72
+ const clearDb = async () => {
73
+ await mongoDb().collection('tenants').deleteMany({});
74
+ await mongoDb().collection('keys').deleteMany({});
75
+ await mongoDb().collection('exchanges').deleteMany({});
76
+ await mongoDb().collection('offers').deleteMany({});
77
+ };
78
+
79
+ const url = (tenant) =>
80
+ `/operator-api/v0.8/tenants/${tenant._id}/vc-api/credentials/issue`;
81
+
82
+ describe('vc-api credentials endpoints', () => {
83
+ let fastify;
84
+ let persistTenant;
85
+ let persistKey;
86
+ let persistOffer;
87
+
88
+ let tenant;
89
+ let holderKeys;
90
+ let holderDid;
91
+ let baseCredential;
92
+
93
+ beforeEach(async () => {
94
+ nock.cleanAll();
95
+ jest.resetAllMocks();
96
+ mockGetRevokeUrl.mockImplementation(
97
+ (listId, index) =>
98
+ `ethereum://0x1234/getRevokeStatus?address=0x412&listId=${listId}&index=${index}`
99
+ );
100
+ await clearDb();
101
+ tenant = await persistTenant({
102
+ serviceIds: ['#foo-service-id-1'],
103
+ });
104
+ const keyPair = generateKeyPair({ format: 'jwk' });
105
+ await persistKey({ tenant, kidFragment: '#ID2', keyPair });
106
+
107
+ holderKeys = await generateKeyPair({ format: 'jwk' });
108
+ holderDid = getDidUriFromJwk(holderKeys.publicKey);
109
+ baseCredential = {
110
+ '@context': [
111
+ 'https://www.w3.org/2018/credentials/v1',
112
+ 'https://www.w3.org/2018/credentials/examples/v1',
113
+ ],
114
+ type: ['VerifiableCredential', 'PastEmploymentPosition'],
115
+ issuer: {
116
+ id: tenant.did,
117
+ },
118
+ credentialSubject: {
119
+ id: holderDid,
120
+ company: tenant.did,
121
+ companyName: {
122
+ localized: {
123
+ en: 'Microsoft Corporation',
124
+ },
125
+ },
126
+ title: {
127
+ localized: {
128
+ en: 'Director, Communications (HoloLens & Mixed Reality Experiences)',
129
+ },
130
+ },
131
+ startMonthYear: {
132
+ month: 10,
133
+ year: 2010,
134
+ },
135
+ endMonthYear: {
136
+ month: 6,
137
+ year: 2019,
138
+ },
139
+ location: {
140
+ countryCode: 'US',
141
+ regionCode: 'MA',
142
+ },
143
+ description: {
144
+ localized: {
145
+ en: 'l Data, AI, Hybrid, IoT, Datacenter, Mixed Reality/HoloLens, D365, Power Platform - all kinds of fun stuff!',
146
+ },
147
+ },
148
+ },
149
+ };
150
+ });
151
+
152
+ afterAll(() => {
153
+ nock.cleanAll();
154
+ nock.restore();
155
+ });
156
+
157
+ describe('vc-api when enabled', () => {
158
+ beforeAll(async () => {
159
+ fastify = buildFastify({
160
+ vcApiEnabled: true,
161
+ });
162
+ await fastify.ready();
163
+ ({ persistTenant } = initTenantFactory(fastify));
164
+ ({ persistKey } = initKeysFactory(fastify));
165
+ ({ persistOffer } = initOfferFactory(fastify));
166
+ });
167
+
168
+ afterAll(async () => {
169
+ await clearDb();
170
+ await fastify.close();
171
+ });
172
+
173
+ it('should fail to issue a json-ld credential', async () => {
174
+ const payload = {
175
+ credential: baseCredential,
176
+ options: {
177
+ format: CredentialFormat.JSON_LD_VC,
178
+ },
179
+ };
180
+ const response = await fastify.injectJson({
181
+ method: 'POST',
182
+ url: url(tenant),
183
+ payload,
184
+ });
185
+
186
+ expect(response.statusCode).toEqual(400);
187
+ expect(response.json).toEqual(
188
+ errorResponseMatcher({
189
+ error: 'Bad Request',
190
+ errorCode: 'missing_error_code',
191
+ message: 'options.format must be jwt-vc',
192
+ statusCode: 400,
193
+ })
194
+ );
195
+ });
196
+
197
+ it('should fail 400 with options missing', async () => {
198
+ const payload = {
199
+ credential: baseCredential,
200
+ };
201
+ const response = await fastify.injectJson({
202
+ method: 'POST',
203
+ url: url(tenant),
204
+ payload,
205
+ });
206
+
207
+ expect(response.statusCode).toEqual(400);
208
+ expect(response.json).toEqual(
209
+ errorResponseMatcher({
210
+ error: 'Bad Request',
211
+ errorCode: 'missing_error_code',
212
+ message: 'options.format must be jwt-vc',
213
+ statusCode: 400,
214
+ })
215
+ );
216
+ });
217
+
218
+ it('should fail 400 with invalid credentialRequest', async () => {
219
+ const payload = {
220
+ credential: omit(['@context'], baseCredential),
221
+ options: {
222
+ format: CredentialFormat.JWT_VC,
223
+ },
224
+ };
225
+ const response = await fastify.injectJson({
226
+ method: 'POST',
227
+ url: url(tenant),
228
+ payload,
229
+ });
230
+
231
+ expect(response.statusCode).toEqual(400);
232
+ expect(response.json).toEqual(
233
+ errorResponseMatcher({
234
+ error: 'Bad Request',
235
+ errorCode: 'request_validation_failed',
236
+ message: "body/credential must have required property '@context'",
237
+ statusCode: 400,
238
+ })
239
+ );
240
+ });
241
+
242
+ it('should fail 400 with invalid credentialSubject', async () => {
243
+ nockRegistrarAppSchemaName();
244
+ const credential = omit(['credentialSubject.company'], baseCredential);
245
+ const payload = {
246
+ credential,
247
+ options: {
248
+ format: CredentialFormat.JWT_VC,
249
+ },
250
+ };
251
+ const response = await fastify.injectJson({
252
+ method: 'POST',
253
+ url: url(tenant),
254
+ payload,
255
+ });
256
+
257
+ expect(response.statusCode).toEqual(400);
258
+ expect(response.json).toEqual(
259
+ errorResponseMatcher({
260
+ error: 'Bad Request',
261
+ errorCode: 'missing_error_code',
262
+ message:
263
+ "'$.credentialSubject' must have required property 'company'",
264
+ statusCode: 400,
265
+ })
266
+ );
267
+ });
268
+
269
+ describe('successful issuing', () => {
270
+ beforeEach(() => {
271
+ mockCreateCredentialMetadataList.mockResolvedValue(true);
272
+ mockAddCredentialMetadataEntry.mockResolvedValue(true);
273
+
274
+ const nockInstance = nock('http://oracle.localhost.test');
275
+ nockCredentialTypes();
276
+ nockRegistrarAppSchemaName({ nockInstance });
277
+ });
278
+
279
+ it('should issue a credential', async () => {
280
+ const payload = {
281
+ credential: baseCredential,
282
+ options: {
283
+ format: CredentialFormat.JWT_VC,
284
+ },
285
+ };
286
+ const response = await fastify.injectJson({
287
+ method: 'POST',
288
+ url: url(tenant),
289
+ payload,
290
+ });
291
+
292
+ expect(response.statusCode).toEqual(201);
293
+ expect(response.json).toEqual({
294
+ verifiableCredential: expect.any(String),
295
+ });
296
+ const vc = jwtDecode(response.json.verifiableCredential);
297
+ expect(vc).toEqual(
298
+ jwtVcExpectation({
299
+ tenant,
300
+ credentialId: vc.payload.jti,
301
+ credential: payload.credential,
302
+ holderDid,
303
+ })
304
+ );
305
+ const dbOffer = await mongoDb()
306
+ .collection('offers')
307
+ .findOne({ did: vc.payload.jti });
308
+ expect(dbOffer).toEqual(
309
+ offerExpectation({
310
+ tenant,
311
+ credential: payload.credential,
312
+ credentialId: vc.payload.jti,
313
+ holderDid,
314
+ })
315
+ );
316
+
317
+ const dbExchange = await mongoDb()
318
+ .collection('exchanges')
319
+ .findOne({ _id: dbOffer.exchangeId });
320
+ expect(dbExchange).toEqual(
321
+ exchangeExpectation({
322
+ tenant,
323
+ dbOffer,
324
+ states: [
325
+ ExchangeStates.NEW,
326
+ ExchangeStates.CLAIMING_IN_PROGRESS,
327
+ ExchangeStates.COMPLETE,
328
+ ],
329
+ })
330
+ );
331
+ expect(mockAddCredentialMetadataEntry.mock.calls).toEqual([
332
+ [expect.any(Object), expect.any(String), 'did:ion:cao'],
333
+ ]);
334
+ });
335
+
336
+ it("should issue a credential without a credential subject's DID", async () => {
337
+ const payload = {
338
+ credential: omit(['credentialSubject.id'], baseCredential),
339
+ options: {
340
+ format: CredentialFormat.JWT_VC,
341
+ },
342
+ };
343
+
344
+ const response = await fastify.injectJson({
345
+ method: 'POST',
346
+ url: url(tenant),
347
+ payload,
348
+ });
349
+
350
+ expect(response.statusCode).toEqual(201);
351
+ expect(response.json).toEqual({
352
+ verifiableCredential: expect.any(String),
353
+ });
354
+ const vc = jwtDecode(response.json.verifiableCredential);
355
+ const jwtVcExpectationObj = jwtVcExpectation({
356
+ tenant,
357
+ credentialId: vc.payload.jti,
358
+ credential: payload.credential,
359
+ vnfProtocolVersion: 1,
360
+ });
361
+ expect(vc).toEqual(omit(['payload.sub'], jwtVcExpectationObj));
362
+ const dbOffer = await mongoDb()
363
+ .collection('offers')
364
+ .findOne({ did: vc.payload.jti });
365
+ expect(dbOffer).toEqual(
366
+ offerExpectation({
367
+ tenant,
368
+ credential: payload.credential,
369
+ credentialId: vc.payload.jti,
370
+ })
371
+ );
372
+
373
+ const dbExchange = await mongoDb()
374
+ .collection('exchanges')
375
+ .findOne({ _id: dbOffer.exchangeId });
376
+ expect(dbExchange).toEqual(
377
+ exchangeExpectation({
378
+ tenant,
379
+ dbOffer,
380
+ states: [
381
+ ExchangeStates.NEW,
382
+ ExchangeStates.CLAIMING_IN_PROGRESS,
383
+ ExchangeStates.COMPLETE,
384
+ ],
385
+ })
386
+ );
387
+ });
388
+
389
+ it('should issue a credential with consentedAt overridden', async () => {
390
+ const payload = {
391
+ credential: baseCredential,
392
+ options: {
393
+ format: CredentialFormat.JWT_VC,
394
+ consented: subHours(24, new Date()),
395
+ },
396
+ };
397
+ const response = await fastify.injectJson({
398
+ method: 'POST',
399
+ url: url(tenant),
400
+ payload,
401
+ });
402
+
403
+ expect(response.statusCode).toEqual(201);
404
+ expect(response.json).toEqual({
405
+ verifiableCredential: expect.any(String),
406
+ });
407
+ const vc = jwtDecode(response.json.verifiableCredential);
408
+ expect(vc).toEqual(
409
+ jwtVcExpectation({
410
+ tenant,
411
+ credentialId: vc.payload.jti,
412
+ credential: payload.credential,
413
+ holderDid,
414
+ })
415
+ );
416
+ const dbOffer = await mongoDb()
417
+ .collection('offers')
418
+ .findOne({ did: vc.payload.jti });
419
+ expect(dbOffer).toEqual(
420
+ offerExpectation({
421
+ tenant,
422
+ credential: payload.credential,
423
+ credentialId: vc.payload.jti,
424
+ holderDid,
425
+ overrides: {
426
+ consentedAt: payload.options.consented,
427
+ },
428
+ })
429
+ );
430
+
431
+ const dbExchange = await mongoDb()
432
+ .collection('exchanges')
433
+ .findOne({ _id: dbOffer.exchangeId });
434
+ expect(dbExchange).toEqual(
435
+ exchangeExpectation({
436
+ tenant,
437
+ dbOffer,
438
+ states: [
439
+ ExchangeStates.NEW,
440
+ ExchangeStates.CLAIMING_IN_PROGRESS,
441
+ ExchangeStates.COMPLETE,
442
+ ],
443
+ })
444
+ );
445
+ });
446
+
447
+ it('should issue a credential with expirationDate properties if set', async () => {
448
+ const credential = {
449
+ ...baseCredential,
450
+ expirationDate: new Date().toISOString(),
451
+ };
452
+ const payload = {
453
+ credential,
454
+ options: {
455
+ format: CredentialFormat.JWT_VC,
456
+ },
457
+ };
458
+ const response = await fastify.injectJson({
459
+ method: 'POST',
460
+ url: url(tenant),
461
+ payload,
462
+ });
463
+
464
+ expect(response.statusCode).toEqual(201);
465
+ expect(response.json).toEqual({
466
+ verifiableCredential: expect.any(String),
467
+ });
468
+ const vc = jwtDecode(response.json.verifiableCredential);
469
+ expect(vc).toEqual(
470
+ jwtVcExpectation({
471
+ tenant,
472
+ credentialId: vc.payload.jti,
473
+ credential,
474
+ holderDid,
475
+ overrides: {
476
+ 'payload.exp': getUnixTime(new Date(credential.expirationDate)),
477
+ },
478
+ })
479
+ );
480
+ const dbOffer = await mongoDb()
481
+ .collection('offers')
482
+ .findOne({ did: vc.payload.jti });
483
+ expect(dbOffer).toEqual(
484
+ offerExpectation({
485
+ tenant,
486
+ credential,
487
+ credentialId: vc.payload.jti,
488
+ holderDid,
489
+ })
490
+ );
491
+
492
+ const dbExchange = await mongoDb()
493
+ .collection('exchanges')
494
+ .findOne({ _id: dbOffer.exchangeId });
495
+ expect(dbExchange).toEqual(
496
+ exchangeExpectation({
497
+ tenant,
498
+ dbOffer,
499
+ states: [
500
+ ExchangeStates.NEW,
501
+ ExchangeStates.CLAIMING_IN_PROGRESS,
502
+ ExchangeStates.COMPLETE,
503
+ ],
504
+ })
505
+ );
506
+ });
507
+
508
+ it('should issue a credential even if the issuer property isnt passed directly', async () => {
509
+ const credential = omit(['issuer'], baseCredential);
510
+ const payload = {
511
+ credential,
512
+ options: {
513
+ format: CredentialFormat.JWT_VC,
514
+ },
515
+ };
516
+ const response = await fastify.injectJson({
517
+ method: 'POST',
518
+ url: url(tenant),
519
+ payload,
520
+ });
521
+
522
+ expect(response.statusCode).toEqual(201);
523
+ expect(response.json).toEqual({
524
+ verifiableCredential: expect.any(String),
525
+ });
526
+ const vc = jwtDecode(response.json.verifiableCredential);
527
+ expect(vc).toEqual(
528
+ jwtVcExpectation({
529
+ tenant,
530
+ credentialId: vc.payload.jti,
531
+ credential: baseCredential,
532
+ holderDid,
533
+ })
534
+ );
535
+ const dbOffer = await mongoDb()
536
+ .collection('offers')
537
+ .findOne({ did: vc.payload.jti });
538
+ expect(dbOffer).toEqual(
539
+ offerExpectation({
540
+ tenant,
541
+ credential: baseCredential,
542
+ credentialId: vc.payload.jti,
543
+ holderDid,
544
+ })
545
+ );
546
+
547
+ const dbExchange = await mongoDb()
548
+ .collection('exchanges')
549
+ .findOne({ _id: dbOffer.exchangeId });
550
+ expect(dbExchange).toEqual(
551
+ exchangeExpectation({
552
+ tenant,
553
+ dbOffer,
554
+ states: [
555
+ ExchangeStates.NEW,
556
+ ExchangeStates.CLAIMING_IN_PROGRESS,
557
+ ExchangeStates.COMPLETE,
558
+ ],
559
+ })
560
+ );
561
+ });
562
+
563
+ it('should issue a credential with resource reference data', async () => {
564
+ const resourceId = 'did:velocity:resource';
565
+ const targetOffer = await persistOffer({
566
+ tenant,
567
+ consentedAt: new Date(),
568
+ did: resourceId,
569
+ digestSRI: 'digestSRI',
570
+ });
571
+ const payload = {
572
+ credential: {
573
+ ...baseCredential,
574
+ replaces: [
575
+ {
576
+ id: resourceId,
577
+ },
578
+ ],
579
+ relatedResource: [
580
+ {
581
+ id: resourceId,
582
+ },
583
+ ],
584
+ },
585
+ options: {
586
+ format: CredentialFormat.JWT_VC,
587
+ },
588
+ };
589
+ const response = await fastify.injectJson({
590
+ method: 'POST',
591
+ url: url(tenant),
592
+ payload,
593
+ });
594
+
595
+ expect(response.statusCode).toEqual(201);
596
+ expect(response.json).toEqual({
597
+ verifiableCredential: expect.any(String),
598
+ });
599
+ const vc = jwtDecode(response.json.verifiableCredential);
600
+ expect(vc).toEqual(
601
+ jwtVcExpectation({
602
+ tenant,
603
+ credentialId: vc.payload.jti,
604
+ credential: {
605
+ ...payload.credential,
606
+ replaces: [
607
+ {
608
+ id: resourceId,
609
+ digestSRI: targetOffer.digestSRI,
610
+ hint: targetOffer.type,
611
+ },
612
+ ],
613
+ relatedResource: [
614
+ {
615
+ id: resourceId,
616
+ digestSRI: targetOffer.digestSRI,
617
+ hint: targetOffer.type,
618
+ },
619
+ ],
620
+ },
621
+ holderDid,
622
+ })
623
+ );
624
+ const dbOffer = await mongoDb()
625
+ .collection('offers')
626
+ .findOne({ did: vc.payload.jti });
627
+ expect(dbOffer).toEqual(
628
+ offerExpectation({
629
+ tenant,
630
+ credential: {
631
+ ...payload.credential,
632
+ replaces: [
633
+ {
634
+ id: resourceId,
635
+ digestSRI: targetOffer.digestSRI,
636
+ hint: targetOffer.type,
637
+ },
638
+ ],
639
+ relatedResource: [
640
+ {
641
+ id: resourceId,
642
+ digestSRI: targetOffer.digestSRI,
643
+ hint: targetOffer.type,
644
+ },
645
+ ],
646
+ },
647
+ credentialId: vc.payload.jti,
648
+ holderDid,
649
+ })
650
+ );
651
+
652
+ const dbExchange = await mongoDb()
653
+ .collection('exchanges')
654
+ .findOne({ _id: dbOffer.exchangeId });
655
+ expect(dbExchange).toEqual(
656
+ exchangeExpectation({
657
+ tenant,
658
+ dbOffer,
659
+ states: [
660
+ ExchangeStates.NEW,
661
+ ExchangeStates.CLAIMING_IN_PROGRESS,
662
+ ExchangeStates.COMPLETE,
663
+ ],
664
+ })
665
+ );
666
+ });
667
+
668
+ it('should issue a open badge v3', async () => {
669
+ const payload = {
670
+ credential: {
671
+ ...openBadgeCredentialExample,
672
+ issuer: {
673
+ id: tenant.did,
674
+ },
675
+ },
676
+ options: {
677
+ format: CredentialFormat.JWT_VC,
678
+ },
679
+ };
680
+ payload.credential.credentialSubject.id = holderDid;
681
+
682
+ nock('https://imsglobal.org')
683
+ .get('/schemas/open-badge-v3.0-schema.json')
684
+ .reply(
685
+ 200,
686
+ require('../combined/schemas/open-badge-credential.schema.json')
687
+ );
688
+
689
+ const response = await fastify.injectJson({
690
+ method: 'POST',
691
+ url: url(tenant),
692
+ payload,
693
+ });
694
+
695
+ expect(response.statusCode).toEqual(201);
696
+ expect(response.json).toEqual({
697
+ verifiableCredential: expect.any(String),
698
+ });
699
+ const vc = jwtDecode(response.json.verifiableCredential);
700
+ expect(vc).toEqual(
701
+ jwtVcExpectation({
702
+ tenant,
703
+ credentialId: vc.payload.jti,
704
+ credential: payload.credential,
705
+ holderDid,
706
+ credentialTypeContext: [],
707
+ overrides: {
708
+ 'payload.vc.credentialStatus': [
709
+ payload.credential.credentialStatus,
710
+ {
711
+ id: expect.stringMatching(
712
+ '^ethereum:0x[0-9a-fA-F]+/getRevokedStatus\\?address=0x[0-9a-z]+&listId=\\d+&index=\\d+$'
713
+ ),
714
+ type: VelocityRevocationListType,
715
+ },
716
+ ],
717
+ 'payload.vc.refreshService': [
718
+ payload.credential.refreshService,
719
+ {
720
+ type: 'VelocityNetworkRefreshService2024',
721
+ id: `${tenant.did}#foo-service-id-1`,
722
+ },
723
+ ],
724
+ 'payload.vc.credentialSubject.type': 'AchievementSubject',
725
+ 'payload.vc.credentialSubject.@context': undefined,
726
+ 'payload.vc.credentialSchema': {
727
+ id: 'https://imsglobal.org/schemas/open-badge-v3.0-schema.json',
728
+ type: 'JsonSchemaValidator2018',
729
+ },
730
+ },
731
+ })
732
+ );
733
+ });
734
+ });
735
+ });
736
+
737
+ describe('vc-api when disabled', () => {
738
+ beforeAll(async () => {
739
+ fastify = buildFastify();
740
+ await fastify.ready();
741
+ ({ persistTenant } = initTenantFactory(fastify));
742
+ ({ persistKey } = initKeysFactory(fastify));
743
+ });
744
+
745
+ afterAll(async () => {
746
+ await clearDb();
747
+ await fastify.close();
748
+ });
749
+
750
+ it('the endpoint should not be found', async () => {
751
+ const payload = {
752
+ credential: baseCredential,
753
+ options: {
754
+ format: CredentialFormat.JWT_VC,
755
+ },
756
+ };
757
+ const response = await fastify.injectJson({
758
+ method: 'POST',
759
+ url: url(tenant),
760
+ payload,
761
+ });
762
+ expect(response.statusCode).toEqual(404);
763
+ });
764
+ });
765
+ });
766
+
767
+ const offerExpectation = ({
768
+ credential,
769
+ credentialId,
770
+ holderDid,
771
+ tenant,
772
+ overrides = {},
773
+ }) => {
774
+ let expectation = {
775
+ ...credential,
776
+ '@context': expect.arrayContaining([
777
+ ...credential['@context'],
778
+ 'https://lib.test/contexts/credential-extensions-2022.jsonld.json',
779
+ ]),
780
+ _id: expect.any(ObjectId),
781
+ did: credentialId,
782
+ credentialSubject: {
783
+ type: 'PastEmploymentPosition',
784
+ ...credential.credentialSubject,
785
+ id: holderDid,
786
+ },
787
+ type: expect.arrayContaining(credential.type),
788
+ ...defaultVnfCredentialExpectations,
789
+ linkCode: expect.any(String),
790
+ credentialStatus: {
791
+ id: expect.stringMatching(
792
+ '^ethereum:0x[0-9a-fA-F]+/getRevokedStatus\\?address=0x[0-9a-z]+&listId=\\d+&index=\\d+$'
793
+ ),
794
+ type: VelocityRevocationListType,
795
+ },
796
+ issuanceDate: expect.stringMatching(ISO_DATETIME_FORMAT),
797
+ tenantId: new ObjectId(tenant._id),
798
+ exchangeId: expect.any(ObjectId),
799
+ offerId: expect.any(String),
800
+ digestSRI: expect.any(String),
801
+ consentedAt: expect.any(Date),
802
+ createdAt: expect.any(Date),
803
+ updatedAt: expect.any(Date),
804
+ };
805
+ for (const [key, value] of entries(overrides)) {
806
+ expectation = set(key, value, expectation);
807
+ }
808
+ return expectation;
809
+ };
810
+
811
+ const jwtVcExpectation = ({
812
+ tenant,
813
+ credential,
814
+ credentialId,
815
+ holderDid,
816
+ vnfProtocolVersion = 2,
817
+ credentialTypeContext = [
818
+ 'https://velocitynetwork.foundation/contexts/layer1-credentials-v1.1.json',
819
+ ],
820
+ overrides = {},
821
+ }) => {
822
+ let expectation = {
823
+ header: {
824
+ alg: 'ES256K',
825
+ kid: `${credentialId}#key-1`,
826
+ typ: 'JWT',
827
+ },
828
+ payload: {
829
+ iat: expect.any(Number),
830
+ iss: tenant.did,
831
+ nbf: expect.any(Number),
832
+ jti: credentialId,
833
+ sub: holderDid,
834
+ vc: {
835
+ ...credential,
836
+ '@context': uniq(
837
+ compact([
838
+ 'https://www.w3.org/2018/credentials/v1',
839
+ ...credentialTypeContext,
840
+ ...castArray(credential['@context']),
841
+ 'https://lib.test/contexts/credential-extensions-2022.jsonld.json',
842
+ ])
843
+ ),
844
+ id: credentialId,
845
+ type: expect.arrayContaining(credential.type),
846
+ credentialSubject: {
847
+ type: 'PastEmploymentPosition',
848
+ ...credential.credentialSubject,
849
+ },
850
+ refreshService: {
851
+ type: 'VelocityNetworkRefreshService2024',
852
+ id: `${tenant.did}#foo-service-id-1`,
853
+ },
854
+ credentialSchema: {
855
+ type: 'JsonSchemaValidator2018',
856
+ id: credentialTypeMetadata[extractCredentialType(credential)]
857
+ .schemaUrl,
858
+ },
859
+ ...defaultVnfCredentialExpectations,
860
+ credentialStatus: {
861
+ id: expect.stringMatching(
862
+ '^ethereum:0x[0-9a-fA-F]+/getRevokedStatus\\?address=0x[0-9a-z]+&listId=\\d+&index=\\d+$'
863
+ ),
864
+ type: VelocityRevocationListType,
865
+ },
866
+ vnfProtocolVersion,
867
+ issuanceDate: expect.stringMatching(ISO_DATETIME_FORMAT),
868
+ },
869
+ },
870
+ };
871
+ for (const [key, value] of entries(overrides)) {
872
+ expectation = set(key, value, expectation);
873
+ }
874
+ return expectation;
875
+ };
876
+
877
+ const exchangeExpectation = ({ tenant, dbOffer, states }) =>
878
+ mongoify({
879
+ _id: dbOffer.exchangeId,
880
+ type: ExchangeTypes.ISSUING,
881
+ tenantId: tenant._id,
882
+ events: map((state) => ({ state, timestamp: expect.any(Date) }), states),
883
+ protocolMetadata: {
884
+ protocol: ExchangeProtocols.W3C_VC_API,
885
+ },
886
+ finalizedOfferIds: [dbOffer._id],
887
+ createdBy: 'velocity.admin@example.com',
888
+ createdAt: expect.any(Date),
889
+ updatedAt: expect.any(Date),
890
+ });
891
+
892
+ const credentialTypeMetadata = keyBy('credentialType', [
893
+ {
894
+ credentialType: 'EmailV1.0',
895
+ layer1: true,
896
+ schemaUrl:
897
+ 'https://velocitynetwork.foundation/schemas/email-v1.0.schema.json',
898
+ },
899
+ {
900
+ credentialType: 'EmploymentCurrentV1.1',
901
+ layer1: true,
902
+ schemaUrl:
903
+ 'https://velocitynetwork.foundation/schemas/employment-v1.1.schema.json',
904
+ jsonldContext: [
905
+ 'https://velocitynetwork.foundation/contexts/layer1-credentials-v1.1.json',
906
+ ],
907
+ },
908
+ {
909
+ credentialType: 'PastEmploymentPosition',
910
+ layer1: true,
911
+ schemaUrl: 'http://mock.com/schemas/past-employment-position',
912
+ jsonldContext: [
913
+ 'https://velocitynetwork.foundation/contexts/layer1-credentials-v1.1.json',
914
+ ],
915
+ },
916
+ {
917
+ credentialType: '1EdtechCLR2.0',
918
+ layer1: false,
919
+ schemaUrl: 'https://imsglobal.org/schemas/clr-v2.0-schema.json',
920
+ jsonldContext: ['https://imsglobal.org/schemas/clr-context.json'],
921
+ },
922
+ {
923
+ credentialType: 'OpenBadgeCredential',
924
+ layer1: true,
925
+ schemaUrl: 'https://imsglobal.org/schemas/open-badge-v3.0-schema.json',
926
+ },
927
+ ]);
928
+
929
+ const nockCredentialTypes = () => {
930
+ nock('http://oracle.localhost.test')
931
+ .get('/api/v0.6/credential-types')
932
+ .query((query) =>
933
+ every(
934
+ (credentialType) =>
935
+ includes(credentialType, Object.keys(credentialTypeMetadata)),
936
+ castArray(query.credentialType)
937
+ )
938
+ )
939
+ .times(2)
940
+ .reply(200, (uri) => {
941
+ const questionMarkIdx = uri.indexOf('?');
942
+ const searchParamsString = uri.substring(questionMarkIdx);
943
+ const query = new URLSearchParams(searchParamsString);
944
+ return Object.values(
945
+ pick(query.getAll('credentialType'), credentialTypeMetadata)
946
+ );
947
+ });
948
+ };
949
+
950
+ const defaultVnfCredentialExpectations = {
951
+ contentHash: {
952
+ type: 'VelocityContentHash2020',
953
+ value: expect.any(String),
954
+ },
955
+ credentialSchema: {
956
+ type: 'JsonSchemaValidator2018',
957
+ id: credentialTypeMetadata.PastEmploymentPosition.schemaUrl,
958
+ },
959
+ linkCodeCommitment: {
960
+ type: 'VelocityCredentialLinkCodeCommitment2022',
961
+ value: expect.stringMatching(BASE64_FORMAT),
962
+ },
963
+ };