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

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 (303) hide show
  1. package/CLAUDE.md +694 -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/package.json +183 -0
  69. package/generated/prisma-mongodb/query-engine-debian-openssl-3.0.x +0 -0
  70. package/generated/prisma-mongodb/query-engine-rhel-openssl-3.0.x +0 -0
  71. package/generated/prisma-mongodb/runtime/binary.d.ts +1 -0
  72. package/generated/prisma-mongodb/runtime/binary.js +289 -0
  73. package/generated/prisma-mongodb/runtime/edge-esm.js +34 -0
  74. package/generated/prisma-mongodb/runtime/edge.js +34 -0
  75. package/generated/prisma-mongodb/runtime/index-browser.d.ts +370 -0
  76. package/generated/prisma-mongodb/runtime/index-browser.js +16 -0
  77. package/generated/prisma-mongodb/runtime/library.d.ts +3977 -0
  78. package/generated/prisma-mongodb/runtime/react-native.js +83 -0
  79. package/generated/prisma-mongodb/runtime/wasm-compiler-edge.js +84 -0
  80. package/generated/prisma-mongodb/runtime/wasm-engine-edge.js +36 -0
  81. package/generated/prisma-mongodb/schema.prisma +362 -0
  82. package/generated/prisma-mongodb/wasm-edge-light-loader.mjs +4 -0
  83. package/generated/prisma-mongodb/wasm-worker-loader.mjs +4 -0
  84. package/generated/prisma-mongodb/wasm.d.ts +1 -0
  85. package/generated/prisma-mongodb/wasm.js +342 -0
  86. package/generated/prisma-postgresql/client.d.ts +1 -0
  87. package/generated/prisma-postgresql/client.js +4 -0
  88. package/generated/prisma-postgresql/default.d.ts +1 -0
  89. package/generated/prisma-postgresql/default.js +4 -0
  90. package/generated/prisma-postgresql/edge.d.ts +1 -0
  91. package/generated/prisma-postgresql/edge.js +357 -0
  92. package/generated/prisma-postgresql/index-browser.js +339 -0
  93. package/generated/prisma-postgresql/index.d.ts +25131 -0
  94. package/generated/prisma-postgresql/index.js +382 -0
  95. package/generated/prisma-postgresql/package.json +183 -0
  96. package/generated/prisma-postgresql/query-engine-debian-openssl-3.0.x +0 -0
  97. package/generated/prisma-postgresql/query-engine-rhel-openssl-3.0.x +0 -0
  98. package/generated/prisma-postgresql/query_engine_bg.js +2 -0
  99. package/generated/prisma-postgresql/query_engine_bg.wasm +0 -0
  100. package/generated/prisma-postgresql/runtime/binary.d.ts +1 -0
  101. package/generated/prisma-postgresql/runtime/binary.js +289 -0
  102. package/generated/prisma-postgresql/runtime/edge-esm.js +34 -0
  103. package/generated/prisma-postgresql/runtime/edge.js +34 -0
  104. package/generated/prisma-postgresql/runtime/index-browser.d.ts +370 -0
  105. package/generated/prisma-postgresql/runtime/index-browser.js +16 -0
  106. package/generated/prisma-postgresql/runtime/library.d.ts +3977 -0
  107. package/generated/prisma-postgresql/runtime/react-native.js +83 -0
  108. package/generated/prisma-postgresql/runtime/wasm-compiler-edge.js +84 -0
  109. package/generated/prisma-postgresql/runtime/wasm-engine-edge.js +36 -0
  110. package/generated/prisma-postgresql/schema.prisma +345 -0
  111. package/generated/prisma-postgresql/wasm-edge-light-loader.mjs +4 -0
  112. package/generated/prisma-postgresql/wasm-worker-loader.mjs +4 -0
  113. package/generated/prisma-postgresql/wasm.d.ts +1 -0
  114. package/generated/prisma-postgresql/wasm.js +364 -0
  115. package/handlers/WEBHOOKS.md +653 -0
  116. package/handlers/app-definition-loader.js +38 -0
  117. package/handlers/app-handler-helpers.js +57 -0
  118. package/handlers/backend-utils.js +262 -0
  119. package/handlers/database-migration-handler.js +227 -0
  120. package/handlers/integration-event-dispatcher.js +54 -0
  121. package/handlers/routers/HEALTHCHECK.md +342 -0
  122. package/handlers/routers/auth.js +15 -0
  123. package/handlers/routers/db-migration.handler.js +29 -0
  124. package/handlers/routers/db-migration.js +326 -0
  125. package/handlers/routers/health.js +516 -0
  126. package/handlers/routers/integration-defined-routers.js +45 -0
  127. package/handlers/routers/integration-webhook-routers.js +67 -0
  128. package/handlers/routers/user.js +63 -0
  129. package/handlers/routers/websocket.js +57 -0
  130. package/handlers/use-cases/check-external-apis-health-use-case.js +81 -0
  131. package/handlers/use-cases/check-integrations-health-use-case.js +44 -0
  132. package/handlers/workers/db-migration.js +352 -0
  133. package/handlers/workers/dlq-processor.js +63 -0
  134. package/handlers/workers/integration-defined-workers.js +23 -0
  135. package/index.js +82 -46
  136. package/infrastructure/scheduler/eventbridge-scheduler-adapter.js +184 -0
  137. package/infrastructure/scheduler/index.js +33 -0
  138. package/infrastructure/scheduler/mock-scheduler-adapter.js +143 -0
  139. package/infrastructure/scheduler/scheduler-service-factory.js +73 -0
  140. package/infrastructure/scheduler/scheduler-service-interface.js +47 -0
  141. package/integrations/WEBHOOK-QUICKSTART.md +151 -0
  142. package/integrations/index.js +12 -10
  143. package/integrations/integration-base.js +364 -55
  144. package/integrations/integration-router.js +375 -179
  145. package/integrations/options.js +1 -1
  146. package/integrations/repositories/integration-mapping-repository-documentdb.js +280 -0
  147. package/integrations/repositories/integration-mapping-repository-factory.js +57 -0
  148. package/integrations/repositories/integration-mapping-repository-interface.js +106 -0
  149. package/integrations/repositories/integration-mapping-repository-mongo.js +161 -0
  150. package/integrations/repositories/integration-mapping-repository-postgres.js +227 -0
  151. package/integrations/repositories/integration-mapping-repository.js +156 -0
  152. package/integrations/repositories/integration-repository-documentdb.js +219 -0
  153. package/integrations/repositories/integration-repository-factory.js +51 -0
  154. package/integrations/repositories/integration-repository-interface.js +144 -0
  155. package/integrations/repositories/integration-repository-mongo.js +330 -0
  156. package/integrations/repositories/integration-repository-postgres.js +385 -0
  157. package/integrations/repositories/process-repository-documentdb.js +243 -0
  158. package/integrations/repositories/process-repository-factory.js +53 -0
  159. package/integrations/repositories/process-repository-interface.js +90 -0
  160. package/integrations/repositories/process-repository-mongo.js +190 -0
  161. package/integrations/repositories/process-repository-postgres.js +217 -0
  162. package/integrations/tests/doubles/config-capturing-integration.js +81 -0
  163. package/integrations/tests/doubles/dummy-integration-class.js +105 -0
  164. package/integrations/tests/doubles/test-integration-repository.js +112 -0
  165. package/integrations/use-cases/create-integration.js +83 -0
  166. package/integrations/use-cases/create-process.js +128 -0
  167. package/integrations/use-cases/delete-integration-for-user.js +101 -0
  168. package/integrations/use-cases/find-integration-context-by-external-entity-id.js +72 -0
  169. package/integrations/use-cases/get-integration-for-user.js +78 -0
  170. package/integrations/use-cases/get-integration-instance-by-definition.js +67 -0
  171. package/integrations/use-cases/get-integration-instance.js +83 -0
  172. package/integrations/use-cases/get-integrations-for-user.js +88 -0
  173. package/integrations/use-cases/get-possible-integrations.js +27 -0
  174. package/integrations/use-cases/get-process.js +87 -0
  175. package/integrations/use-cases/index.js +19 -0
  176. package/integrations/use-cases/load-integration-context.js +71 -0
  177. package/integrations/use-cases/update-integration-messages.js +44 -0
  178. package/integrations/use-cases/update-integration-status.js +32 -0
  179. package/integrations/use-cases/update-integration.js +92 -0
  180. package/integrations/use-cases/update-process-metrics.js +201 -0
  181. package/integrations/use-cases/update-process-state.js +119 -0
  182. package/integrations/utils/map-integration-dto.js +37 -0
  183. package/jest-global-setup-noop.js +3 -0
  184. package/jest-global-teardown-noop.js +3 -0
  185. package/logs/logger.js +0 -4
  186. package/{module-plugin → modules}/index.js +0 -10
  187. package/modules/module-factory.js +56 -0
  188. package/modules/module.js +256 -0
  189. package/modules/repositories/module-repository-documentdb.js +335 -0
  190. package/modules/repositories/module-repository-factory.js +40 -0
  191. package/modules/repositories/module-repository-interface.js +129 -0
  192. package/modules/repositories/module-repository-mongo.js +408 -0
  193. package/modules/repositories/module-repository-postgres.js +453 -0
  194. package/modules/repositories/module-repository.js +345 -0
  195. package/modules/requester/api-key.js +52 -0
  196. package/modules/requester/oauth-2.js +396 -0
  197. package/{module-plugin → modules}/requester/requester.js +4 -2
  198. package/{module-plugin → modules}/test/mock-api/api.js +8 -3
  199. package/{module-plugin → modules}/test/mock-api/definition.js +14 -10
  200. package/modules/tests/doubles/test-module-factory.js +16 -0
  201. package/modules/tests/doubles/test-module-repository.js +39 -0
  202. package/modules/use-cases/get-entities-for-user.js +32 -0
  203. package/modules/use-cases/get-entity-options-by-id.js +71 -0
  204. package/modules/use-cases/get-entity-options-by-type.js +34 -0
  205. package/modules/use-cases/get-module-instance-from-type.js +31 -0
  206. package/modules/use-cases/get-module.js +74 -0
  207. package/modules/use-cases/process-authorization-callback.js +177 -0
  208. package/modules/use-cases/refresh-entity-options.js +72 -0
  209. package/modules/use-cases/test-module-auth.js +72 -0
  210. package/modules/utils/map-module-dto.js +18 -0
  211. package/package.json +82 -50
  212. package/prisma-mongodb/schema.prisma +362 -0
  213. package/prisma-postgresql/migrations/20250930193005_init/migration.sql +315 -0
  214. package/prisma-postgresql/migrations/20251006135218_init/migration.sql +9 -0
  215. package/prisma-postgresql/migrations/20251010000000_remove_unused_entity_reference_map/migration.sql +3 -0
  216. package/prisma-postgresql/migrations/20251112195422_update_user_unique_constraints/migration.sql +25 -0
  217. package/prisma-postgresql/migrations/migration_lock.toml +3 -0
  218. package/prisma-postgresql/schema.prisma +345 -0
  219. package/queues/queuer-util.js +103 -21
  220. package/syncs/manager.js +468 -443
  221. package/syncs/repositories/sync-repository-documentdb.js +240 -0
  222. package/syncs/repositories/sync-repository-factory.js +43 -0
  223. package/syncs/repositories/sync-repository-interface.js +109 -0
  224. package/syncs/repositories/sync-repository-mongo.js +239 -0
  225. package/syncs/repositories/sync-repository-postgres.js +319 -0
  226. package/syncs/sync.js +0 -1
  227. package/token/repositories/token-repository-documentdb.js +137 -0
  228. package/token/repositories/token-repository-factory.js +40 -0
  229. package/token/repositories/token-repository-interface.js +131 -0
  230. package/token/repositories/token-repository-mongo.js +219 -0
  231. package/token/repositories/token-repository-postgres.js +264 -0
  232. package/token/repositories/token-repository.js +219 -0
  233. package/types/associations/index.d.ts +0 -17
  234. package/types/core/index.d.ts +12 -4
  235. package/types/database/index.d.ts +10 -2
  236. package/types/encrypt/index.d.ts +5 -3
  237. package/types/integrations/index.d.ts +3 -8
  238. package/types/module-plugin/index.d.ts +17 -69
  239. package/types/syncs/index.d.ts +0 -17
  240. package/user/repositories/user-repository-documentdb.js +441 -0
  241. package/user/repositories/user-repository-factory.js +52 -0
  242. package/user/repositories/user-repository-interface.js +201 -0
  243. package/user/repositories/user-repository-mongo.js +308 -0
  244. package/user/repositories/user-repository-postgres.js +360 -0
  245. package/user/tests/doubles/test-user-repository.js +72 -0
  246. package/user/use-cases/authenticate-user.js +127 -0
  247. package/user/use-cases/authenticate-with-shared-secret.js +48 -0
  248. package/user/use-cases/create-individual-user.js +61 -0
  249. package/user/use-cases/create-organization-user.js +47 -0
  250. package/user/use-cases/create-token-for-user-id.js +30 -0
  251. package/user/use-cases/get-user-from-adopter-jwt.js +149 -0
  252. package/user/use-cases/get-user-from-bearer-token.js +77 -0
  253. package/user/use-cases/get-user-from-x-frigg-headers.js +132 -0
  254. package/user/use-cases/login-user.js +122 -0
  255. package/user/user.js +125 -0
  256. package/utils/backend-path.js +38 -0
  257. package/utils/index.js +6 -0
  258. package/websocket/repositories/websocket-connection-repository-documentdb.js +119 -0
  259. package/websocket/repositories/websocket-connection-repository-factory.js +44 -0
  260. package/websocket/repositories/websocket-connection-repository-interface.js +106 -0
  261. package/websocket/repositories/websocket-connection-repository-mongo.js +156 -0
  262. package/websocket/repositories/websocket-connection-repository-postgres.js +196 -0
  263. package/websocket/repositories/websocket-connection-repository.js +161 -0
  264. package/assertions/is-equal.js +0 -17
  265. package/associations/model.js +0 -54
  266. package/database/models/IndividualUser.js +0 -76
  267. package/database/models/OrganizationUser.js +0 -29
  268. package/database/models/State.js +0 -9
  269. package/database/models/Token.js +0 -70
  270. package/database/models/UserModel.js +0 -7
  271. package/database/models/WebsocketConnection.js +0 -49
  272. package/database/mongo.js +0 -45
  273. package/database/mongoose.js +0 -5
  274. package/encrypt/Cryptor.test.js +0 -32
  275. package/encrypt/encrypt.js +0 -132
  276. package/encrypt/encrypt.test.js +0 -1069
  277. package/encrypt/test-encrypt.js +0 -107
  278. package/errors/base-error.test.js +0 -32
  279. package/errors/fetch-error.test.js +0 -79
  280. package/errors/halt-error.test.js +0 -11
  281. package/errors/validation-errors.test.js +0 -120
  282. package/integrations/create-frigg-backend.js +0 -31
  283. package/integrations/integration-factory.js +0 -251
  284. package/integrations/integration-mapping.js +0 -43
  285. package/integrations/integration-model.js +0 -46
  286. package/integrations/integration-user.js +0 -144
  287. package/integrations/test/integration-base.test.js +0 -144
  288. package/lambda/TimeoutCatcher.test.js +0 -68
  289. package/logs/logger.test.js +0 -76
  290. package/module-plugin/auther.js +0 -393
  291. package/module-plugin/credential.js +0 -22
  292. package/module-plugin/entity-manager.js +0 -70
  293. package/module-plugin/entity.js +0 -46
  294. package/module-plugin/manager.js +0 -169
  295. package/module-plugin/module-factory.js +0 -61
  296. package/module-plugin/requester/api-key.js +0 -36
  297. package/module-plugin/requester/oauth-2.js +0 -219
  298. package/module-plugin/requester/requester.test.js +0 -28
  299. package/module-plugin/test/auther.test.js +0 -97
  300. package/syncs/model.js +0 -62
  301. /package/{module-plugin → modules}/ModuleConstants.js +0 -0
  302. /package/{module-plugin → modules}/requester/basic.js +0 -0
  303. /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,67 @@ 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
+ );
309
512
  const areRequirementsValid =
310
513
  module.validateAuthorizationRequirements();
311
514
  if (!areRequirementsValid) {
312
515
  throw new Error(
313
- `Error: EntityManager of type ${params.entityType} requires a valid url`
516
+ `Error: Entity of type ${params.entityType} requires a valid url`
314
517
  );
315
518
  }
316
519
 
317
- res.json(await module.getAuthorizationRequirements());
520
+ res.json(module.getAuthorizationRequirements());
318
521
  })
319
522
  );
320
523
 
321
524
  router.route('/api/authorize').post(
322
525
  catchAsyncError(async (req, res) => {
526
+ const user = await authenticateUser.execute(req);
527
+ const userId = user.getId();
323
528
  const params = checkRequiredParams(req.body, [
324
529
  'entityType',
325
530
  'data',
326
531
  ]);
327
- const module = await getModuleInstance(req, params.entityType);
328
532
 
329
- res.json(
330
- await module.processAuthorizationCallback({
331
- userId: getUserId(req),
332
- data: params.data,
333
- })
533
+ const entityDetails = await processAuthorizationCallback.execute(
534
+ userId,
535
+ params.entityType,
536
+ params.data
334
537
  );
538
+
539
+ res.json(entityDetails);
335
540
  })
336
541
  );
337
542
 
338
543
  router.route('/api/entity').post(
339
544
  catchAsyncError(async (req, res) => {
545
+ const user = await authenticateUser.execute(req);
546
+ const userId = user.getId();
340
547
  const params = checkRequiredParams(req.body, [
341
548
  'entityType',
342
549
  'data',
@@ -344,20 +551,24 @@ function setEntityRoutes(router, factory, getUserId) {
344
551
  checkRequiredParams(req.body.data, ['credential_id']);
345
552
 
346
553
  // 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
554
+ const credential = await getCredentialForUser.execute(
555
+ params.data.credential_id,
556
+ userId
349
557
  );
350
558
 
351
559
  if (!credential) {
352
560
  throw Boom.badRequest('Invalid credential ID');
353
561
  }
354
562
 
355
- const module = await getModuleInstance(req, params.entityType);
563
+ const module = await getModuleInstanceFromType.execute(
564
+ userId,
565
+ params.entityType
566
+ );
356
567
  const entityDetails = await module.getEntityDetails(
357
568
  module.api,
358
569
  null,
359
570
  null,
360
- getUserId(req)
571
+ userId
361
572
  );
362
573
 
363
574
  res.json(await module.findOrCreateEntity(entityDetails));
@@ -366,43 +577,44 @@ function setEntityRoutes(router, factory, getUserId) {
366
577
 
367
578
  router.route('/api/entity/options/:credentialId').get(
368
579
  catchAsyncError(async (req, res) => {
580
+ const user = await authenticateUser.execute(req);
581
+ const userId = user.getId();
369
582
  // TODO May want to pass along the user ID as well so credential ID's can't be fished???
370
583
  // TODO **flagging this for review** -MW
371
- const credential = await IntegrationHelper.getCredentialById(
372
- req.params.credentialId
584
+ const credential = await getCredentialForUser.execute(
585
+ req.params.credentialId,
586
+ userId
373
587
  );
374
- if (credential.user._id.toString() !== getUserId(req)) {
588
+ if (credential.userId.toString() !== userId) {
375
589
  throw Boom.forbidden('Credential does not belong to user');
376
590
  }
377
591
 
378
592
  const params = checkRequiredParams(req.query, ['entityType']);
379
- const module = await getModuleInstance(req, params.entityType);
593
+ const entityOptions = await getEntityOptionsByType.execute(
594
+ userId,
595
+ params.entityType
596
+ );
380
597
 
381
- res.json(await module.getEntityOptions());
598
+ res.json(entityOptions);
382
599
  })
383
600
  );
384
601
 
385
602
  router.route('/api/entities/:entityId/test-auth').get(
386
603
  catchAsyncError(async (req, res) => {
604
+ const user = await authenticateUser.execute(req);
387
605
  const params = checkRequiredParams(req.params, ['entityId']);
388
- const module = await moduleFactory.getModuleInstanceFromEntityId(
606
+ const testAuthResponse = await testModuleAuth.execute(
389
607
  params.entityId,
390
- getUserId(req)
608
+ user // Pass User object for proper validation
391
609
  );
392
610
 
393
- if (!module) {
394
- throw Boom.notFound();
395
- }
396
-
397
- const testAuthResponse = await module.testAuth();
398
-
399
611
  if (!testAuthResponse) {
400
612
  res.status(400);
401
613
  res.json({
402
614
  errors: [
403
615
  {
404
616
  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.`,
617
+ message: `There was an error with your Entity. Please reconnect/re-authenticate, or reach out to Support for assistance.`,
406
618
  timestamp: Date.now(),
407
619
  },
408
620
  ],
@@ -415,55 +627,39 @@ function setEntityRoutes(router, factory, getUserId) {
415
627
 
416
628
  router.route('/api/entities/:entityId').get(
417
629
  catchAsyncError(async (req, res) => {
630
+ const user = await authenticateUser.execute(req);
418
631
  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
- }
632
+ const module = await getModule.execute(params.entityId, user); // Pass User object
427
633
 
428
- res.json(module.entity);
634
+ res.json(module);
429
635
  })
430
636
  );
431
637
 
432
638
  router.route('/api/entities/:entityId/options').post(
433
639
  catchAsyncError(async (req, res) => {
434
- const params = checkRequiredParams(req.params, [
435
- 'entityId',
436
- getUserId(req),
437
- ]);
438
- const module = await moduleFactory.getModuleInstanceFromEntityId(
640
+ const user = await authenticateUser.execute(req);
641
+ const params = checkRequiredParams(req.params, ['entityId']);
642
+
643
+ const entityOptions = await getEntityOptionsById.execute(
439
644
  params.entityId,
440
- getUserId(req)
645
+ user // Pass User object
441
646
  );
442
647
 
443
- if (!module) {
444
- throw Boom.notFound();
445
- }
446
-
447
- res.json(await module.getEntityOptions());
648
+ res.json(entityOptions);
448
649
  })
449
650
  );
450
651
 
451
652
  router.route('/api/entities/:entityId/options/refresh').post(
452
653
  catchAsyncError(async (req, res) => {
453
- const params = checkRequiredParams(req.params, [
454
- 'entityId',
455
- getUserId(req),
456
- ]);
457
- const module = await moduleFactory.getModuleInstanceFromEntityId(
654
+ const user = await authenticateUser.execute(req);
655
+ const params = checkRequiredParams(req.params, ['entityId']);
656
+ const updatedOptions = await refreshEntityOptions.execute(
458
657
  params.entityId,
459
- getUserId(req)
658
+ user, // Pass User object
659
+ req.body
460
660
  );
461
661
 
462
- if (!module) {
463
- throw Boom.notFound();
464
- }
465
-
466
- res.json(await module.refreshEntityOptions(req.body));
662
+ res.json(updatedOptions);
467
663
  })
468
664
  );
469
665
  }