@friggframework/core 2.0.0-next.8 → 2.0.0-next.81

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 (305) hide show
  1. package/CLAUDE.md +702 -0
  2. package/README.md +959 -50
  3. package/application/commands/README.md +451 -0
  4. package/application/commands/credential-commands.js +245 -0
  5. package/application/commands/entity-commands.js +336 -0
  6. package/application/commands/integration-commands.js +210 -0
  7. package/application/commands/scheduler-commands.js +263 -0
  8. package/application/commands/user-commands.js +283 -0
  9. package/application/index.js +73 -0
  10. package/assertions/index.js +0 -3
  11. package/core/CLAUDE.md +690 -0
  12. package/core/Worker.js +60 -24
  13. package/core/create-handler.js +79 -8
  14. package/credential/repositories/credential-repository-documentdb.js +304 -0
  15. package/credential/repositories/credential-repository-factory.js +54 -0
  16. package/credential/repositories/credential-repository-interface.js +98 -0
  17. package/credential/repositories/credential-repository-mongo.js +269 -0
  18. package/credential/repositories/credential-repository-postgres.js +287 -0
  19. package/credential/repositories/credential-repository.js +300 -0
  20. package/credential/use-cases/get-credential-for-user.js +25 -0
  21. package/credential/use-cases/update-authentication-status.js +15 -0
  22. package/database/MONGODB_TRANSACTION_FIX.md +198 -0
  23. package/database/adapters/lambda-invoker.js +97 -0
  24. package/database/config.js +154 -0
  25. package/database/documentdb-encryption-service.js +330 -0
  26. package/database/documentdb-utils.js +136 -0
  27. package/database/encryption/README.md +839 -0
  28. package/database/encryption/documentdb-encryption-service.md +3575 -0
  29. package/database/encryption/encryption-schema-registry.js +268 -0
  30. package/database/encryption/field-encryption-service.js +226 -0
  31. package/database/encryption/logger.js +79 -0
  32. package/database/encryption/prisma-encryption-extension.js +222 -0
  33. package/database/index.js +21 -21
  34. package/database/prisma.js +182 -0
  35. package/database/repositories/health-check-repository-documentdb.js +138 -0
  36. package/database/repositories/health-check-repository-factory.js +48 -0
  37. package/database/repositories/health-check-repository-interface.js +82 -0
  38. package/database/repositories/health-check-repository-mongodb.js +89 -0
  39. package/database/repositories/health-check-repository-postgres.js +82 -0
  40. package/database/repositories/migration-status-repository-s3.js +137 -0
  41. package/database/use-cases/check-database-health-use-case.js +29 -0
  42. package/database/use-cases/check-database-state-use-case.js +81 -0
  43. package/database/use-cases/check-encryption-health-use-case.js +83 -0
  44. package/database/use-cases/get-database-state-via-worker-use-case.js +61 -0
  45. package/database/use-cases/get-migration-status-use-case.js +93 -0
  46. package/database/use-cases/run-database-migration-use-case.js +139 -0
  47. package/database/use-cases/test-encryption-use-case.js +253 -0
  48. package/database/use-cases/trigger-database-migration-use-case.js +157 -0
  49. package/database/utils/mongodb-collection-utils.js +94 -0
  50. package/database/utils/mongodb-schema-init.js +108 -0
  51. package/database/utils/prisma-runner.js +477 -0
  52. package/database/utils/prisma-schema-parser.js +182 -0
  53. package/docs/PROCESS_MANAGEMENT_QUEUE_SPEC.md +517 -0
  54. package/encrypt/Cryptor.js +34 -168
  55. package/encrypt/index.js +1 -2
  56. package/errors/client-safe-error.js +26 -0
  57. package/errors/fetch-error.js +15 -7
  58. package/errors/index.js +2 -0
  59. package/generated/prisma-mongodb/client.d.ts +1 -0
  60. package/generated/prisma-mongodb/client.js +4 -0
  61. package/generated/prisma-mongodb/default.d.ts +1 -0
  62. package/generated/prisma-mongodb/default.js +4 -0
  63. package/generated/prisma-mongodb/edge.d.ts +1 -0
  64. package/generated/prisma-mongodb/edge.js +335 -0
  65. package/generated/prisma-mongodb/index-browser.js +317 -0
  66. package/generated/prisma-mongodb/index.d.ts +22955 -0
  67. package/generated/prisma-mongodb/index.js +360 -0
  68. package/generated/prisma-mongodb/libquery_engine-debian-openssl-3.0.x.so.node +0 -0
  69. package/generated/prisma-mongodb/libquery_engine-rhel-openssl-3.0.x.so.node +0 -0
  70. package/generated/prisma-mongodb/package.json +183 -0
  71. package/generated/prisma-mongodb/runtime/edge-esm.js +34 -0
  72. package/generated/prisma-mongodb/runtime/edge.js +34 -0
  73. package/generated/prisma-mongodb/runtime/index-browser.d.ts +370 -0
  74. package/generated/prisma-mongodb/runtime/index-browser.js +16 -0
  75. package/generated/prisma-mongodb/runtime/library.d.ts +3977 -0
  76. package/generated/prisma-mongodb/runtime/library.js +146 -0
  77. package/generated/prisma-mongodb/runtime/react-native.js +83 -0
  78. package/generated/prisma-mongodb/runtime/wasm-compiler-edge.js +84 -0
  79. package/generated/prisma-mongodb/runtime/wasm-engine-edge.js +36 -0
  80. package/generated/prisma-mongodb/schema.prisma +368 -0
  81. package/generated/prisma-mongodb/wasm-edge-light-loader.mjs +4 -0
  82. package/generated/prisma-mongodb/wasm-worker-loader.mjs +4 -0
  83. package/generated/prisma-mongodb/wasm.d.ts +1 -0
  84. package/generated/prisma-mongodb/wasm.js +342 -0
  85. package/generated/prisma-postgresql/client.d.ts +1 -0
  86. package/generated/prisma-postgresql/client.js +4 -0
  87. package/generated/prisma-postgresql/default.d.ts +1 -0
  88. package/generated/prisma-postgresql/default.js +4 -0
  89. package/generated/prisma-postgresql/edge.d.ts +1 -0
  90. package/generated/prisma-postgresql/edge.js +357 -0
  91. package/generated/prisma-postgresql/index-browser.js +339 -0
  92. package/generated/prisma-postgresql/index.d.ts +25135 -0
  93. package/generated/prisma-postgresql/index.js +382 -0
  94. package/generated/prisma-postgresql/libquery_engine-debian-openssl-3.0.x.so.node +0 -0
  95. package/generated/prisma-postgresql/libquery_engine-rhel-openssl-3.0.x.so.node +0 -0
  96. package/generated/prisma-postgresql/package.json +183 -0
  97. package/generated/prisma-postgresql/query_engine_bg.js +2 -0
  98. package/generated/prisma-postgresql/query_engine_bg.wasm +0 -0
  99. package/generated/prisma-postgresql/runtime/edge-esm.js +34 -0
  100. package/generated/prisma-postgresql/runtime/edge.js +34 -0
  101. package/generated/prisma-postgresql/runtime/index-browser.d.ts +370 -0
  102. package/generated/prisma-postgresql/runtime/index-browser.js +16 -0
  103. package/generated/prisma-postgresql/runtime/library.d.ts +3977 -0
  104. package/generated/prisma-postgresql/runtime/library.js +146 -0
  105. package/generated/prisma-postgresql/runtime/react-native.js +83 -0
  106. package/generated/prisma-postgresql/runtime/wasm-compiler-edge.js +84 -0
  107. package/generated/prisma-postgresql/runtime/wasm-engine-edge.js +36 -0
  108. package/generated/prisma-postgresql/schema.prisma +351 -0
  109. package/generated/prisma-postgresql/wasm-edge-light-loader.mjs +4 -0
  110. package/generated/prisma-postgresql/wasm-worker-loader.mjs +4 -0
  111. package/generated/prisma-postgresql/wasm.d.ts +1 -0
  112. package/generated/prisma-postgresql/wasm.js +364 -0
  113. package/handlers/WEBHOOKS.md +653 -0
  114. package/handlers/app-definition-loader.js +38 -0
  115. package/handlers/app-handler-helpers.js +57 -0
  116. package/handlers/backend-utils.js +262 -0
  117. package/handlers/database-migration-handler.js +227 -0
  118. package/handlers/integration-event-dispatcher.js +54 -0
  119. package/handlers/routers/HEALTHCHECK.md +342 -0
  120. package/handlers/routers/auth.js +15 -0
  121. package/handlers/routers/db-migration.handler.js +29 -0
  122. package/handlers/routers/db-migration.js +326 -0
  123. package/handlers/routers/health.js +516 -0
  124. package/handlers/routers/integration-defined-routers.js +45 -0
  125. package/handlers/routers/integration-webhook-routers.js +67 -0
  126. package/handlers/routers/user.js +63 -0
  127. package/handlers/routers/websocket.js +57 -0
  128. package/handlers/use-cases/check-external-apis-health-use-case.js +81 -0
  129. package/handlers/use-cases/check-integrations-health-use-case.js +44 -0
  130. package/handlers/workers/db-migration.js +352 -0
  131. package/handlers/workers/dlq-processor.js +63 -0
  132. package/handlers/workers/integration-defined-workers.js +23 -0
  133. package/index.js +82 -46
  134. package/infrastructure/scheduler/eventbridge-scheduler-adapter.js +184 -0
  135. package/infrastructure/scheduler/index.js +33 -0
  136. package/infrastructure/scheduler/mock-scheduler-adapter.js +143 -0
  137. package/infrastructure/scheduler/scheduler-service-factory.js +73 -0
  138. package/infrastructure/scheduler/scheduler-service-interface.js +47 -0
  139. package/integrations/WEBHOOK-QUICKSTART.md +151 -0
  140. package/integrations/index.js +12 -10
  141. package/integrations/integration-base.js +364 -55
  142. package/integrations/integration-router.js +376 -179
  143. package/integrations/options.js +1 -1
  144. package/integrations/repositories/integration-mapping-repository-documentdb.js +280 -0
  145. package/integrations/repositories/integration-mapping-repository-factory.js +57 -0
  146. package/integrations/repositories/integration-mapping-repository-interface.js +106 -0
  147. package/integrations/repositories/integration-mapping-repository-mongo.js +161 -0
  148. package/integrations/repositories/integration-mapping-repository-postgres.js +227 -0
  149. package/integrations/repositories/integration-mapping-repository.js +156 -0
  150. package/integrations/repositories/integration-repository-documentdb.js +219 -0
  151. package/integrations/repositories/integration-repository-factory.js +51 -0
  152. package/integrations/repositories/integration-repository-interface.js +144 -0
  153. package/integrations/repositories/integration-repository-mongo.js +330 -0
  154. package/integrations/repositories/integration-repository-postgres.js +385 -0
  155. package/integrations/repositories/process-repository-documentdb.js +311 -0
  156. package/integrations/repositories/process-repository-factory.js +53 -0
  157. package/integrations/repositories/process-repository-interface.js +136 -0
  158. package/integrations/repositories/process-repository-mongo.js +262 -0
  159. package/integrations/repositories/process-repository-postgres.js +380 -0
  160. package/integrations/repositories/process-update-ops-shared.js +112 -0
  161. package/integrations/tests/doubles/config-capturing-integration.js +81 -0
  162. package/integrations/tests/doubles/dummy-integration-class.js +105 -0
  163. package/integrations/tests/doubles/test-integration-repository.js +112 -0
  164. package/integrations/use-cases/create-integration.js +83 -0
  165. package/integrations/use-cases/create-process.js +128 -0
  166. package/integrations/use-cases/delete-integration-for-user.js +101 -0
  167. package/integrations/use-cases/find-integration-context-by-external-entity-id.js +72 -0
  168. package/integrations/use-cases/get-integration-for-user.js +78 -0
  169. package/integrations/use-cases/get-integration-instance-by-definition.js +67 -0
  170. package/integrations/use-cases/get-integration-instance.js +83 -0
  171. package/integrations/use-cases/get-integrations-for-user.js +88 -0
  172. package/integrations/use-cases/get-possible-integrations.js +27 -0
  173. package/integrations/use-cases/get-process.js +87 -0
  174. package/integrations/use-cases/index.js +19 -0
  175. package/integrations/use-cases/load-integration-context.js +71 -0
  176. package/integrations/use-cases/update-integration-messages.js +44 -0
  177. package/integrations/use-cases/update-integration-status.js +32 -0
  178. package/integrations/use-cases/update-integration.js +92 -0
  179. package/integrations/use-cases/update-process-metrics.js +205 -0
  180. package/integrations/use-cases/update-process-state.js +158 -0
  181. package/integrations/utils/map-integration-dto.js +37 -0
  182. package/jest-global-setup-noop.js +3 -0
  183. package/jest-global-teardown-noop.js +3 -0
  184. package/logs/logger.js +0 -4
  185. package/{module-plugin → modules}/index.js +0 -10
  186. package/modules/module-factory.js +56 -0
  187. package/modules/module.js +258 -0
  188. package/modules/repositories/module-repository-documentdb.js +335 -0
  189. package/modules/repositories/module-repository-factory.js +40 -0
  190. package/modules/repositories/module-repository-interface.js +129 -0
  191. package/modules/repositories/module-repository-mongo.js +408 -0
  192. package/modules/repositories/module-repository-postgres.js +453 -0
  193. package/modules/repositories/module-repository.js +345 -0
  194. package/modules/requester/api-key.js +52 -0
  195. package/modules/requester/oauth-2.js +396 -0
  196. package/modules/requester/requester.js +275 -0
  197. package/{module-plugin → modules}/test/mock-api/api.js +8 -3
  198. package/{module-plugin → modules}/test/mock-api/definition.js +14 -10
  199. package/modules/tests/doubles/test-module-factory.js +16 -0
  200. package/modules/tests/doubles/test-module-repository.js +39 -0
  201. package/modules/use-cases/get-entities-for-user.js +32 -0
  202. package/modules/use-cases/get-entity-options-by-id.js +71 -0
  203. package/modules/use-cases/get-entity-options-by-type.js +34 -0
  204. package/modules/use-cases/get-module-instance-from-type.js +34 -0
  205. package/modules/use-cases/get-module.js +74 -0
  206. package/modules/use-cases/process-authorization-callback.js +177 -0
  207. package/modules/use-cases/refresh-entity-options.js +72 -0
  208. package/modules/use-cases/test-module-auth.js +72 -0
  209. package/modules/utils/map-module-dto.js +18 -0
  210. package/package.json +82 -50
  211. package/prisma-mongodb/schema.prisma +368 -0
  212. package/prisma-postgresql/migrations/20250930193005_init/migration.sql +315 -0
  213. package/prisma-postgresql/migrations/20251006135218_init/migration.sql +9 -0
  214. package/prisma-postgresql/migrations/20251010000000_remove_unused_entity_reference_map/migration.sql +3 -0
  215. package/prisma-postgresql/migrations/20251112195422_update_user_unique_constraints/migration.sql +25 -0
  216. package/prisma-postgresql/migrations/20260422120000_add_entity_data_column/migration.sql +10 -0
  217. package/prisma-postgresql/migrations/20260422120001_create_process_table/migration.sql +48 -0
  218. package/prisma-postgresql/migrations/migration_lock.toml +3 -0
  219. package/prisma-postgresql/schema.prisma +351 -0
  220. package/queues/queuer-util.js +103 -21
  221. package/syncs/manager.js +468 -443
  222. package/syncs/repositories/sync-repository-documentdb.js +240 -0
  223. package/syncs/repositories/sync-repository-factory.js +43 -0
  224. package/syncs/repositories/sync-repository-interface.js +109 -0
  225. package/syncs/repositories/sync-repository-mongo.js +239 -0
  226. package/syncs/repositories/sync-repository-postgres.js +319 -0
  227. package/syncs/sync.js +0 -1
  228. package/token/repositories/token-repository-documentdb.js +137 -0
  229. package/token/repositories/token-repository-factory.js +40 -0
  230. package/token/repositories/token-repository-interface.js +131 -0
  231. package/token/repositories/token-repository-mongo.js +219 -0
  232. package/token/repositories/token-repository-postgres.js +264 -0
  233. package/token/repositories/token-repository.js +219 -0
  234. package/types/associations/index.d.ts +0 -17
  235. package/types/core/index.d.ts +12 -4
  236. package/types/database/index.d.ts +10 -2
  237. package/types/encrypt/index.d.ts +5 -3
  238. package/types/integrations/index.d.ts +3 -8
  239. package/types/module-plugin/index.d.ts +17 -69
  240. package/types/syncs/index.d.ts +0 -17
  241. package/user/repositories/user-repository-documentdb.js +441 -0
  242. package/user/repositories/user-repository-factory.js +52 -0
  243. package/user/repositories/user-repository-interface.js +201 -0
  244. package/user/repositories/user-repository-mongo.js +308 -0
  245. package/user/repositories/user-repository-postgres.js +360 -0
  246. package/user/tests/doubles/test-user-repository.js +72 -0
  247. package/user/use-cases/authenticate-user.js +127 -0
  248. package/user/use-cases/authenticate-with-shared-secret.js +48 -0
  249. package/user/use-cases/create-individual-user.js +61 -0
  250. package/user/use-cases/create-organization-user.js +47 -0
  251. package/user/use-cases/create-token-for-user-id.js +30 -0
  252. package/user/use-cases/get-user-from-adopter-jwt.js +149 -0
  253. package/user/use-cases/get-user-from-bearer-token.js +77 -0
  254. package/user/use-cases/get-user-from-x-frigg-headers.js +132 -0
  255. package/user/use-cases/login-user.js +122 -0
  256. package/user/user.js +125 -0
  257. package/utils/backend-path.js +38 -0
  258. package/utils/index.js +6 -0
  259. package/websocket/repositories/websocket-connection-repository-documentdb.js +119 -0
  260. package/websocket/repositories/websocket-connection-repository-factory.js +44 -0
  261. package/websocket/repositories/websocket-connection-repository-interface.js +106 -0
  262. package/websocket/repositories/websocket-connection-repository-mongo.js +156 -0
  263. package/websocket/repositories/websocket-connection-repository-postgres.js +196 -0
  264. package/websocket/repositories/websocket-connection-repository.js +161 -0
  265. package/assertions/is-equal.js +0 -17
  266. package/associations/model.js +0 -54
  267. package/database/models/IndividualUser.js +0 -76
  268. package/database/models/OrganizationUser.js +0 -29
  269. package/database/models/State.js +0 -9
  270. package/database/models/Token.js +0 -70
  271. package/database/models/UserModel.js +0 -7
  272. package/database/models/WebsocketConnection.js +0 -49
  273. package/database/mongo.js +0 -45
  274. package/database/mongoose.js +0 -5
  275. package/encrypt/Cryptor.test.js +0 -32
  276. package/encrypt/encrypt.js +0 -132
  277. package/encrypt/encrypt.test.js +0 -1069
  278. package/encrypt/test-encrypt.js +0 -107
  279. package/errors/base-error.test.js +0 -32
  280. package/errors/fetch-error.test.js +0 -79
  281. package/errors/halt-error.test.js +0 -11
  282. package/errors/validation-errors.test.js +0 -120
  283. package/integrations/create-frigg-backend.js +0 -31
  284. package/integrations/integration-factory.js +0 -251
  285. package/integrations/integration-mapping.js +0 -43
  286. package/integrations/integration-model.js +0 -46
  287. package/integrations/integration-user.js +0 -144
  288. package/integrations/test/integration-base.test.js +0 -144
  289. package/lambda/TimeoutCatcher.test.js +0 -68
  290. package/logs/logger.test.js +0 -76
  291. package/module-plugin/auther.js +0 -393
  292. package/module-plugin/credential.js +0 -22
  293. package/module-plugin/entity-manager.js +0 -70
  294. package/module-plugin/entity.js +0 -46
  295. package/module-plugin/manager.js +0 -169
  296. package/module-plugin/module-factory.js +0 -61
  297. package/module-plugin/requester/api-key.js +0 -36
  298. package/module-plugin/requester/oauth-2.js +0 -219
  299. package/module-plugin/requester/requester.js +0 -165
  300. package/module-plugin/requester/requester.test.js +0 -28
  301. package/module-plugin/test/auther.test.js +0 -97
  302. package/syncs/model.js +0 -62
  303. /package/{module-plugin → modules}/ModuleConstants.js +0 -0
  304. /package/{module-plugin → modules}/requester/basic.js +0 -0
  305. /package/{module-plugin → modules}/test/mock-api/mocks/hubspot.js +0 -0
@@ -2,23 +2,220 @@ const express = require('express');
2
2
  const { get } = require('../assertions');
3
3
  const Boom = require('@hapi/boom');
4
4
  const catchAsyncError = require('express-async-handler');
5
- const { debug } = require('../logs');
6
- function createIntegrationRouter(params) {
7
- const router = get(params, 'router', express());
8
- const factory = get(params, 'factory');
9
- const getUserId = get(params, 'getUserId', (req) => null);
10
- const requireLoggedInUser = get(
11
- params,
12
- 'requireLoggedInUser',
13
- (req, res, next) => next()
14
- );
15
-
16
- router.all('/api/entities*', requireLoggedInUser);
17
- router.all('/api/authorize', requireLoggedInUser);
18
- router.all('/api/integrations*', requireLoggedInUser);
19
-
20
- setIntegrationRoutes(router, factory, getUserId);
21
- setEntityRoutes(router, factory, getUserId);
5
+ const {
6
+ createIntegrationRepository,
7
+ } = require('./repositories/integration-repository-factory');
8
+ const {
9
+ DeleteIntegrationForUser,
10
+ } = require('./use-cases/delete-integration-for-user');
11
+ const {
12
+ GetIntegrationsForUser,
13
+ } = require('./use-cases/get-integrations-for-user');
14
+ const {
15
+ createCredentialRepository,
16
+ } = require('../credential/repositories/credential-repository-factory');
17
+ const {
18
+ GetCredentialForUser,
19
+ } = require('../credential/use-cases/get-credential-for-user');
20
+ const { CreateIntegration } = require('./use-cases/create-integration');
21
+ const { ModuleFactory } = require('../modules/module-factory');
22
+ const {
23
+ createModuleRepository,
24
+ } = require('../modules/repositories/module-repository-factory');
25
+ const {
26
+ GetEntitiesForUser,
27
+ } = require('../modules/use-cases/get-entities-for-user');
28
+ const { loadAppDefinition } = require('../handlers/app-definition-loader');
29
+ const {
30
+ GetIntegrationInstance,
31
+ } = require('./use-cases/get-integration-instance');
32
+ const { UpdateIntegration } = require('./use-cases/update-integration');
33
+ const {
34
+ getModulesDefinitionFromIntegrationClasses,
35
+ } = require('./utils/map-integration-dto');
36
+ const {
37
+ GetModuleInstanceFromType,
38
+ } = require('../modules/use-cases/get-module-instance-from-type');
39
+ const {
40
+ GetEntityOptionsByType,
41
+ } = require('../modules/use-cases/get-entity-options-by-type');
42
+ const { TestModuleAuth } = require('../modules/use-cases/test-module-auth');
43
+ const { GetModule } = require('../modules/use-cases/get-module');
44
+ const {
45
+ GetEntityOptionsById,
46
+ } = require('../modules/use-cases/get-entity-options-by-id');
47
+ const {
48
+ RefreshEntityOptions,
49
+ } = require('../modules/use-cases/refresh-entity-options');
50
+ const {
51
+ GetPossibleIntegrations,
52
+ } = require('./use-cases/get-possible-integrations');
53
+ const {
54
+ createUserRepository,
55
+ } = require('../user/repositories/user-repository-factory');
56
+ const {
57
+ GetUserFromBearerToken,
58
+ } = require('../user/use-cases/get-user-from-bearer-token');
59
+ const {
60
+ GetUserFromXFriggHeaders,
61
+ } = require('../user/use-cases/get-user-from-x-frigg-headers');
62
+ const {
63
+ GetUserFromAdopterJwt,
64
+ } = require('../user/use-cases/get-user-from-adopter-jwt');
65
+ const {
66
+ AuthenticateWithSharedSecret,
67
+ } = require('../user/use-cases/authenticate-with-shared-secret');
68
+ const { AuthenticateUser } = require('../user/use-cases/authenticate-user');
69
+ const {
70
+ ProcessAuthorizationCallback,
71
+ } = require('../modules/use-cases/process-authorization-callback');
72
+
73
+ function createIntegrationRouter() {
74
+ const { integrations: integrationClasses, userConfig } =
75
+ loadAppDefinition();
76
+ const moduleRepository = createModuleRepository();
77
+ const integrationRepository = createIntegrationRepository();
78
+ const credentialRepository = createCredentialRepository();
79
+ const userRepository = createUserRepository();
80
+
81
+ const getUserFromBearerToken = new GetUserFromBearerToken({
82
+ userRepository,
83
+ userConfig,
84
+ });
85
+
86
+ const getUserFromXFriggHeaders = new GetUserFromXFriggHeaders({
87
+ userRepository,
88
+ userConfig,
89
+ });
90
+
91
+ const getUserFromAdopterJwt = new GetUserFromAdopterJwt({
92
+ userRepository,
93
+ userConfig,
94
+ });
95
+
96
+ const authenticateWithSharedSecret = new AuthenticateWithSharedSecret();
97
+
98
+ const authenticateUser = new AuthenticateUser({
99
+ getUserFromBearerToken,
100
+ getUserFromXFriggHeaders,
101
+ getUserFromAdopterJwt,
102
+ authenticateWithSharedSecret,
103
+ userConfig,
104
+ });
105
+
106
+ const moduleFactory = new ModuleFactory({
107
+ moduleRepository,
108
+ moduleDefinitions:
109
+ getModulesDefinitionFromIntegrationClasses(integrationClasses),
110
+ });
111
+ const deleteIntegrationForUser = new DeleteIntegrationForUser({
112
+ integrationRepository,
113
+ integrationClasses,
114
+ moduleFactory,
115
+ });
116
+
117
+ const getIntegrationsForUser = new GetIntegrationsForUser({
118
+ integrationRepository,
119
+ integrationClasses,
120
+ moduleFactory,
121
+ moduleRepository,
122
+ });
123
+
124
+ const getCredentialForUser = new GetCredentialForUser({
125
+ credentialRepository,
126
+ });
127
+
128
+ const createIntegration = new CreateIntegration({
129
+ integrationRepository,
130
+ integrationClasses,
131
+ moduleFactory,
132
+ });
133
+
134
+ const getEntitiesForUser = new GetEntitiesForUser({
135
+ moduleRepository,
136
+ moduleDefinitions:
137
+ getModulesDefinitionFromIntegrationClasses(integrationClasses),
138
+ });
139
+
140
+ const getIntegrationInstance = new GetIntegrationInstance({
141
+ integrationRepository,
142
+ integrationClasses,
143
+ moduleFactory,
144
+ });
145
+
146
+ const updateIntegration = new UpdateIntegration({
147
+ integrationRepository,
148
+ integrationClasses,
149
+ moduleFactory,
150
+ });
151
+
152
+ const getModuleInstanceFromType = new GetModuleInstanceFromType({
153
+ moduleDefinitions:
154
+ getModulesDefinitionFromIntegrationClasses(integrationClasses),
155
+ });
156
+
157
+ const getEntityOptionsByType = new GetEntityOptionsByType({
158
+ moduleDefinitions:
159
+ getModulesDefinitionFromIntegrationClasses(integrationClasses),
160
+ });
161
+
162
+ const testModuleAuth = new TestModuleAuth({
163
+ moduleRepository,
164
+ moduleDefinitions:
165
+ getModulesDefinitionFromIntegrationClasses(integrationClasses),
166
+ });
167
+
168
+ const getModule = new GetModule({
169
+ moduleRepository,
170
+ moduleDefinitions:
171
+ getModulesDefinitionFromIntegrationClasses(integrationClasses),
172
+ });
173
+
174
+ const getEntityOptionsById = new GetEntityOptionsById({
175
+ moduleRepository,
176
+ moduleDefinitions:
177
+ getModulesDefinitionFromIntegrationClasses(integrationClasses),
178
+ });
179
+
180
+ const refreshEntityOptions = new RefreshEntityOptions({
181
+ moduleRepository,
182
+ moduleDefinitions:
183
+ getModulesDefinitionFromIntegrationClasses(integrationClasses),
184
+ });
185
+
186
+ const getPossibleIntegrations = new GetPossibleIntegrations({
187
+ integrationClasses,
188
+ });
189
+
190
+ const processAuthorizationCallback = new ProcessAuthorizationCallback({
191
+ moduleRepository,
192
+ credentialRepository,
193
+ integrationRepository,
194
+ moduleDefinitions:
195
+ getModulesDefinitionFromIntegrationClasses(integrationClasses),
196
+ });
197
+
198
+ const router = express();
199
+
200
+ setIntegrationRoutes(router, authenticateUser, {
201
+ createIntegration,
202
+ deleteIntegrationForUser,
203
+ getIntegrationsForUser,
204
+ getEntitiesForUser,
205
+ getIntegrationInstance,
206
+ updateIntegration,
207
+ getPossibleIntegrations,
208
+ });
209
+ setEntityRoutes(router, authenticateUser, {
210
+ getCredentialForUser,
211
+ getModuleInstanceFromType,
212
+ getEntityOptionsByType,
213
+ testModuleAuth,
214
+ getModule,
215
+ getEntityOptionsById,
216
+ refreshEntityOptions,
217
+ processAuthorizationCallback,
218
+ });
22
219
  return router;
23
220
  }
24
221
 
@@ -46,111 +243,95 @@ function checkRequiredParams(params, requiredKeys) {
46
243
  return returnDict;
47
244
  }
48
245
 
49
- function setIntegrationRoutes(router, factory, getUserId) {
50
- const { moduleFactory, integrationFactory, IntegrationHelper } = factory;
246
+ /**
247
+ * Sets up integration-related routes on the provided Express router
248
+ * @param {express.Router} router - Express router instance to add routes to
249
+ * @param {import('../user/use-cases/authenticate-user').AuthenticateUser} authenticateUser - Use case for multi-mode user authentication
250
+ * @param {Object} useCases - use cases for integration management
251
+ */
252
+ function setIntegrationRoutes(router, authenticateUser, useCases) {
253
+ const {
254
+ createIntegration,
255
+ deleteIntegrationForUser,
256
+ getIntegrationsForUser,
257
+ getEntitiesForUser,
258
+ getIntegrationInstance,
259
+ updateIntegration,
260
+ getPossibleIntegrations,
261
+ } = useCases;
51
262
  router.route('/api/integrations').get(
52
263
  catchAsyncError(async (req, res) => {
53
- const results = await integrationFactory.getIntegrationOptions();
54
- results.entities.authorized =
55
- await moduleFactory.getEntitiesForUser(getUserId(req));
56
- results.integrations =
57
- await IntegrationHelper.getIntegrationsForUserId(
58
- getUserId(req)
59
- );
264
+ const user = await authenticateUser.execute(req);
265
+ const userId = user.getId();
266
+ const integrations = await getIntegrationsForUser.execute(userId);
267
+ const results = {
268
+ entities: {
269
+ options: await getPossibleIntegrations.execute(),
270
+ authorized: await getEntitiesForUser.execute(userId),
271
+ },
272
+ integrations: integrations,
273
+ };
60
274
 
61
- for (const integrationRecord of results.integrations) {
62
- const integration =
63
- await integrationFactory.getInstanceFromIntegrationId({
64
- integrationId: integrationRecord.id,
65
- userId: getUserId(req),
66
- });
67
- integrationRecord.userActions = integration.userActions;
68
- }
69
275
  res.json(results);
70
276
  })
71
277
  );
72
278
 
73
279
  router.route('/api/integrations').post(
74
280
  catchAsyncError(async (req, res) => {
281
+ const user = await authenticateUser.execute(req);
282
+ const userId = user.getId();
75
283
  const params = checkRequiredParams(req.body, [
76
284
  'entities',
77
285
  'config',
78
286
  ]);
79
- // throw if not value
287
+
80
288
  get(params.config, 'type');
81
289
 
82
- // create integration
83
- const integration = await integrationFactory.createIntegration(
290
+ const integration = await createIntegration.execute(
84
291
  params.entities,
85
- getUserId(req),
292
+ userId,
86
293
  params.config
87
294
  );
88
295
 
89
- // post integration initialization
90
- debug(
91
- `Calling onCreate on the ${integration?.constructor?.Config?.name} Integration with no arguments`
92
- );
93
- await integration.send('ON_CREATE', {});
94
-
95
- res.status(201).json(
96
- await IntegrationHelper.getFormattedIntegration(
97
- integration.record
98
- )
99
- );
296
+ res.status(201).json(integration);
100
297
  })
101
298
  );
102
299
 
103
300
  router.route('/api/integrations/:integrationId').patch(
104
301
  catchAsyncError(async (req, res) => {
302
+ const user = await authenticateUser.execute(req);
303
+ const userId = user.getId();
105
304
  const params = checkRequiredParams(req.body, ['config']);
106
305
 
107
- const integration =
108
- await integrationFactory.getInstanceFromIntegrationId({
109
- integrationId: req.params.integrationId,
110
- userId: getUserId(req),
111
- });
112
-
113
- debug(
114
- `Calling onUpdate on the ${integration?.constructor?.Config?.name} Integration arguments: `,
115
- params
116
- );
117
- await integration.send('ON_UPDATE', params);
118
-
119
- res.json(
120
- await IntegrationHelper.getFormattedIntegration(
121
- integration.record
122
- )
306
+ const integration = await updateIntegration.execute(
307
+ req.params.integrationId,
308
+ userId,
309
+ params.config
123
310
  );
311
+ res.json(integration);
124
312
  })
125
313
  );
126
314
 
127
315
  router.route('/api/integrations/:integrationId').delete(
128
316
  catchAsyncError(async (req, res) => {
317
+ const user = await authenticateUser.execute(req);
129
318
  const params = checkRequiredParams(req.params, ['integrationId']);
130
- const integration =
131
- await integrationFactory.getInstanceFromIntegrationId({
132
- userId: getUserId(req),
133
- integrationId: params.integrationId,
134
- });
135
-
136
- debug(
137
- `Calling onUpdate on the ${integration?.constructor?.Definition?.name} Integration with no arguments`
138
- );
139
- await integration.send('ON_DELETE');
140
- await IntegrationHelper.deleteIntegrationForUserById(
141
- getUserId(req),
142
- params.integrationId
319
+ await deleteIntegrationForUser.execute(
320
+ params.integrationId,
321
+ user.getId()
143
322
  );
144
-
145
- res.status(201).json({});
323
+ res.status(204).json({});
146
324
  })
147
325
  );
148
326
 
149
327
  router.route('/api/integrations/:integrationId/config/options').get(
150
328
  catchAsyncError(async (req, res) => {
329
+ const user = await authenticateUser.execute(req);
151
330
  const params = checkRequiredParams(req.params, ['integrationId']);
152
- const integration =
153
- await integrationFactory.getInstanceFromIntegrationId(params);
331
+ const integration = await getIntegrationInstance.execute(
332
+ params.integrationId,
333
+ user.getId()
334
+ );
154
335
  res.json(await integration.send('GET_CONFIG_OPTIONS'));
155
336
  })
156
337
  );
@@ -159,13 +340,14 @@ function setIntegrationRoutes(router, factory, getUserId) {
159
340
  .route('/api/integrations/:integrationId/config/options/refresh')
160
341
  .post(
161
342
  catchAsyncError(async (req, res) => {
343
+ const user = await authenticateUser.execute(req);
162
344
  const params = checkRequiredParams(req.params, [
163
345
  'integrationId',
164
346
  ]);
165
- const integration =
166
- await integrationFactory.getInstanceFromIntegrationId(
167
- params
168
- );
347
+ const integration = await getIntegrationInstance.execute(
348
+ params.integrationId,
349
+ user.getId()
350
+ );
169
351
 
170
352
  res.json(
171
353
  await integration.send('REFRESH_CONFIG_OPTIONS', req.body)
@@ -174,9 +356,12 @@ function setIntegrationRoutes(router, factory, getUserId) {
174
356
  );
175
357
  router.route('/api/integrations/:integrationId/actions').all(
176
358
  catchAsyncError(async (req, res) => {
359
+ const user = await authenticateUser.execute(req);
177
360
  const params = checkRequiredParams(req.params, ['integrationId']);
178
- const integration =
179
- await integrationFactory.getInstanceFromIntegrationId(params);
361
+ const integration = await getIntegrationInstance.execute(
362
+ params.integrationId,
363
+ user.getId()
364
+ );
180
365
  res.json(await integration.send('GET_USER_ACTIONS', req.body));
181
366
  })
182
367
  );
@@ -185,14 +370,15 @@ function setIntegrationRoutes(router, factory, getUserId) {
185
370
  .route('/api/integrations/:integrationId/actions/:actionId/options')
186
371
  .all(
187
372
  catchAsyncError(async (req, res) => {
373
+ const user = await authenticateUser.execute(req);
188
374
  const params = checkRequiredParams(req.params, [
189
375
  'integrationId',
190
376
  'actionId',
191
377
  ]);
192
- const integration =
193
- await integrationFactory.getInstanceFromIntegrationId(
194
- params
195
- );
378
+ const integration = await getIntegrationInstance.execute(
379
+ params.integrationId,
380
+ user.getId()
381
+ );
196
382
 
197
383
  res.json(
198
384
  await integration.send('GET_USER_ACTION_OPTIONS', {
@@ -209,14 +395,15 @@ function setIntegrationRoutes(router, factory, getUserId) {
209
395
  )
210
396
  .post(
211
397
  catchAsyncError(async (req, res) => {
398
+ const user = await authenticateUser.execute(req);
212
399
  const params = checkRequiredParams(req.params, [
213
400
  'integrationId',
214
401
  'actionId',
215
402
  ]);
216
- const integration =
217
- await integrationFactory.getInstanceFromIntegrationId(
218
- params
219
- );
403
+ const integration = await getIntegrationInstance.execute(
404
+ params.integrationId,
405
+ user.getId()
406
+ );
220
407
 
221
408
  res.json(
222
409
  await integration.send('REFRESH_USER_ACTION_OPTIONS', {
@@ -229,23 +416,33 @@ function setIntegrationRoutes(router, factory, getUserId) {
229
416
 
230
417
  router.route('/api/integrations/:integrationId/actions/:actionId').post(
231
418
  catchAsyncError(async (req, res) => {
419
+ const user = await authenticateUser.execute(req);
232
420
  const params = checkRequiredParams(req.params, [
233
421
  'integrationId',
234
422
  'actionId',
235
423
  ]);
236
- const integration =
237
- await integrationFactory.getInstanceFromIntegrationId(params);
238
-
424
+ const integration = await getIntegrationInstance.execute(
425
+ params.integrationId,
426
+ user.getId()
427
+ );
239
428
  res.json(await integration.send(params.actionId, req.body));
240
429
  })
241
430
  );
242
431
 
243
432
  router.route('/api/integrations/:integrationId').get(
244
433
  catchAsyncError(async (req, res) => {
434
+ const user = await authenticateUser.execute(req);
435
+
436
+ if (!user) {
437
+ throw Boom.forbidden('User not found');
438
+ }
439
+
245
440
  const params = checkRequiredParams(req.params, ['integrationId']);
246
- const integration = await IntegrationHelper.getIntegrationById(
247
- params.integrationId
441
+ const integration = await getIntegrationInstance.execute(
442
+ params.integrationId,
443
+ user.getId()
248
444
  );
445
+
249
446
  // We could perhaps augment router with dynamic options? Haven't decided yet, but here may be the place
250
447
 
251
448
  res.json({
@@ -259,12 +456,12 @@ function setIntegrationRoutes(router, factory, getUserId) {
259
456
 
260
457
  router.route('/api/integrations/:integrationId/test-auth').get(
261
458
  catchAsyncError(async (req, res) => {
459
+ const user = await authenticateUser.execute(req);
262
460
  const params = checkRequiredParams(req.params, ['integrationId']);
263
- const instance =
264
- await integrationFactory.getInstanceFromIntegrationId({
265
- userId: getUserId(req),
266
- integrationId: params.integrationId,
267
- });
461
+ const instance = await getIntegrationInstance.execute(
462
+ params.integrationId,
463
+ user.getId()
464
+ );
268
465
 
269
466
  if (!instance) {
270
467
  throw Boom.notFound();
@@ -286,57 +483,68 @@ function setIntegrationRoutes(router, factory, getUserId) {
286
483
  );
287
484
  }
288
485
 
289
- function setEntityRoutes(router, factory, getUserId) {
290
- const { moduleFactory, IntegrationHelper } = factory;
291
- const getModuleInstance = async (req, entityType) => {
292
- if (!moduleFactory.checkIsValidType(entityType)) {
293
- throw Boom.badRequest(
294
- `Error: Invalid entity type of ${entityType}, options are ${moduleFactory.moduleTypes.join(
295
- ', '
296
- )}`
297
- );
298
- }
299
- return await moduleFactory.getInstanceFromTypeName(
300
- entityType,
301
- getUserId(req)
302
- );
303
- };
486
+ /**
487
+ * Sets up entity-related routes for the integration router
488
+ * @param {Object} router - Express router instance
489
+ * @param {import('../user/use-cases/authenticate-user').AuthenticateUser} authenticateUser - Use case for multi-mode user authentication
490
+ */
491
+ function setEntityRoutes(router, authenticateUser, useCases) {
492
+ const {
493
+ getCredentialForUser,
494
+ getModuleInstanceFromType,
495
+ getEntityOptionsByType,
496
+ testModuleAuth,
497
+ getModule,
498
+ getEntityOptionsById,
499
+ refreshEntityOptions,
500
+ processAuthorizationCallback,
501
+ } = useCases;
304
502
 
305
503
  router.route('/api/authorize').get(
306
504
  catchAsyncError(async (req, res) => {
505
+ const user = await authenticateUser.execute(req);
506
+ const userId = user.getId();
307
507
  const params = checkRequiredParams(req.query, ['entityType']);
308
- const module = await getModuleInstance(req, params.entityType);
508
+ const module = await getModuleInstanceFromType.execute(
509
+ userId,
510
+ params.entityType,
511
+ { state: req.query.state }
512
+ );
309
513
  const areRequirementsValid =
310
514
  module.validateAuthorizationRequirements();
311
515
  if (!areRequirementsValid) {
312
516
  throw new Error(
313
- `Error: EntityManager of type ${params.entityType} requires a valid url`
517
+ `Error: Entity of type ${params.entityType} requires a valid url`
314
518
  );
315
519
  }
316
520
 
317
- res.json(await module.getAuthorizationRequirements());
521
+ res.json(module.getAuthorizationRequirements());
318
522
  })
319
523
  );
320
524
 
321
525
  router.route('/api/authorize').post(
322
526
  catchAsyncError(async (req, res) => {
527
+ const user = await authenticateUser.execute(req);
528
+ const userId = user.getId();
323
529
  const params = checkRequiredParams(req.body, [
324
530
  'entityType',
325
531
  'data',
326
532
  ]);
327
- const module = await getModuleInstance(req, params.entityType);
328
533
 
329
- res.json(
330
- await module.processAuthorizationCallback({
331
- userId: getUserId(req),
332
- data: params.data,
333
- })
534
+ const entityDetails = await processAuthorizationCallback.execute(
535
+ userId,
536
+ params.entityType,
537
+ params.data
334
538
  );
539
+
540
+ res.json(entityDetails);
335
541
  })
336
542
  );
337
543
 
338
544
  router.route('/api/entity').post(
339
545
  catchAsyncError(async (req, res) => {
546
+ const user = await authenticateUser.execute(req);
547
+ const userId = user.getId();
340
548
  const params = checkRequiredParams(req.body, [
341
549
  'entityType',
342
550
  'data',
@@ -344,20 +552,24 @@ function setEntityRoutes(router, factory, getUserId) {
344
552
  checkRequiredParams(req.body.data, ['credential_id']);
345
553
 
346
554
  // May want to pass along the user ID as well so credential ID's can't be fished???
347
- const credential = await IntegrationHelper.getCredentialById(
348
- params.data.credential_id
555
+ const credential = await getCredentialForUser.execute(
556
+ params.data.credential_id,
557
+ userId
349
558
  );
350
559
 
351
560
  if (!credential) {
352
561
  throw Boom.badRequest('Invalid credential ID');
353
562
  }
354
563
 
355
- const module = await getModuleInstance(req, params.entityType);
564
+ const module = await getModuleInstanceFromType.execute(
565
+ userId,
566
+ params.entityType
567
+ );
356
568
  const entityDetails = await module.getEntityDetails(
357
569
  module.api,
358
570
  null,
359
571
  null,
360
- getUserId(req)
572
+ userId
361
573
  );
362
574
 
363
575
  res.json(await module.findOrCreateEntity(entityDetails));
@@ -366,43 +578,44 @@ function setEntityRoutes(router, factory, getUserId) {
366
578
 
367
579
  router.route('/api/entity/options/:credentialId').get(
368
580
  catchAsyncError(async (req, res) => {
581
+ const user = await authenticateUser.execute(req);
582
+ const userId = user.getId();
369
583
  // TODO May want to pass along the user ID as well so credential ID's can't be fished???
370
584
  // TODO **flagging this for review** -MW
371
- const credential = await IntegrationHelper.getCredentialById(
372
- req.params.credentialId
585
+ const credential = await getCredentialForUser.execute(
586
+ req.params.credentialId,
587
+ userId
373
588
  );
374
- if (credential.user._id.toString() !== getUserId(req)) {
589
+ if (credential.userId.toString() !== userId) {
375
590
  throw Boom.forbidden('Credential does not belong to user');
376
591
  }
377
592
 
378
593
  const params = checkRequiredParams(req.query, ['entityType']);
379
- const module = await getModuleInstance(req, params.entityType);
594
+ const entityOptions = await getEntityOptionsByType.execute(
595
+ userId,
596
+ params.entityType
597
+ );
380
598
 
381
- res.json(await module.getEntityOptions());
599
+ res.json(entityOptions);
382
600
  })
383
601
  );
384
602
 
385
603
  router.route('/api/entities/:entityId/test-auth').get(
386
604
  catchAsyncError(async (req, res) => {
605
+ const user = await authenticateUser.execute(req);
387
606
  const params = checkRequiredParams(req.params, ['entityId']);
388
- const module = await moduleFactory.getModuleInstanceFromEntityId(
607
+ const testAuthResponse = await testModuleAuth.execute(
389
608
  params.entityId,
390
- getUserId(req)
609
+ user // Pass User object for proper validation
391
610
  );
392
611
 
393
- if (!module) {
394
- throw Boom.notFound();
395
- }
396
-
397
- const testAuthResponse = await module.testAuth();
398
-
399
612
  if (!testAuthResponse) {
400
613
  res.status(400);
401
614
  res.json({
402
615
  errors: [
403
616
  {
404
617
  title: 'Authentication Error',
405
- message: `There was an error with your ${module.getName()} Entity. Please reconnect/re-authenticate, or reach out to Support for assistance.`,
618
+ message: `There was an error with your Entity. Please reconnect/re-authenticate, or reach out to Support for assistance.`,
406
619
  timestamp: Date.now(),
407
620
  },
408
621
  ],
@@ -415,55 +628,39 @@ function setEntityRoutes(router, factory, getUserId) {
415
628
 
416
629
  router.route('/api/entities/:entityId').get(
417
630
  catchAsyncError(async (req, res) => {
631
+ const user = await authenticateUser.execute(req);
418
632
  const params = checkRequiredParams(req.params, ['entityId']);
419
- const module = await moduleFactory.getModuleInstanceFromEntityId(
420
- params.entityId,
421
- getUserId(req)
422
- );
423
-
424
- if (!module) {
425
- throw Boom.notFound();
426
- }
633
+ const module = await getModule.execute(params.entityId, user); // Pass User object
427
634
 
428
- res.json(module.entity);
635
+ res.json(module);
429
636
  })
430
637
  );
431
638
 
432
639
  router.route('/api/entities/:entityId/options').post(
433
640
  catchAsyncError(async (req, res) => {
434
- const params = checkRequiredParams(req.params, [
435
- 'entityId',
436
- getUserId(req),
437
- ]);
438
- const module = await moduleFactory.getModuleInstanceFromEntityId(
641
+ const user = await authenticateUser.execute(req);
642
+ const params = checkRequiredParams(req.params, ['entityId']);
643
+
644
+ const entityOptions = await getEntityOptionsById.execute(
439
645
  params.entityId,
440
- getUserId(req)
646
+ user // Pass User object
441
647
  );
442
648
 
443
- if (!module) {
444
- throw Boom.notFound();
445
- }
446
-
447
- res.json(await module.getEntityOptions());
649
+ res.json(entityOptions);
448
650
  })
449
651
  );
450
652
 
451
653
  router.route('/api/entities/:entityId/options/refresh').post(
452
654
  catchAsyncError(async (req, res) => {
453
- const params = checkRequiredParams(req.params, [
454
- 'entityId',
455
- getUserId(req),
456
- ]);
457
- const module = await moduleFactory.getModuleInstanceFromEntityId(
655
+ const user = await authenticateUser.execute(req);
656
+ const params = checkRequiredParams(req.params, ['entityId']);
657
+ const updatedOptions = await refreshEntityOptions.execute(
458
658
  params.entityId,
459
- getUserId(req)
659
+ user, // Pass User object
660
+ req.body
460
661
  );
461
662
 
462
- if (!module) {
463
- throw Boom.notFound();
464
- }
465
-
466
- res.json(await module.refreshEntityOptions(req.body));
663
+ res.json(updatedOptions);
467
664
  })
468
665
  );
469
666
  }