@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
@@ -0,0 +1,31 @@
1
+ const { Module } = require('../module');
2
+
3
+ class GetModuleInstanceFromType {
4
+ /**
5
+ * @param {Object} params
6
+ * @param {} params.moduleDefinitions
7
+ */
8
+ constructor({ moduleDefinitions }) {
9
+ this.moduleDefinitions = moduleDefinitions;
10
+ }
11
+
12
+ /**
13
+ * Retrieve a Module instance for a given user and entity/module type.
14
+ * @param {string} userId
15
+ * @param {string} type – human-readable module/entity type (e.g. "Hubspot")
16
+ */
17
+ async execute(userId, type) {
18
+ const moduleDefinition = this.moduleDefinitions.find(
19
+ (def) => def.getName() === type
20
+ );
21
+ if (!moduleDefinition) {
22
+ throw new Error(`Module definition not found for type: ${type}`);
23
+ }
24
+ return new Module({
25
+ userId,
26
+ definition: moduleDefinition,
27
+ });
28
+ }
29
+ }
30
+
31
+ module.exports = { GetModuleInstanceFromType };
@@ -0,0 +1,74 @@
1
+ const { Module } = require('../module');
2
+
3
+ class GetModule {
4
+ constructor({ moduleRepository, moduleDefinitions }) {
5
+ this.moduleRepository = moduleRepository;
6
+ this.moduleDefinitions = moduleDefinitions;
7
+ }
8
+
9
+ /**
10
+ * Get module instance for an entity
11
+ *
12
+ * @param {string|number} entityId - Entity ID to retrieve
13
+ * @param {string|number|import('../../user/user').User} userIdOrUser - User ID or User object for validation
14
+ * @returns {Promise<Object>} Module details
15
+ */
16
+ async execute(entityId, userIdOrUser) {
17
+ // Support both userId (backward compatible) and User object (new pattern)
18
+ const userId = typeof userIdOrUser === 'object' && userIdOrUser?.getId
19
+ ? userIdOrUser.getId()
20
+ : userIdOrUser;
21
+
22
+ const entity = await this.moduleRepository.findEntityById(
23
+ entityId,
24
+ userId
25
+ );
26
+
27
+ if (!entity) {
28
+ throw new Error(`Entity ${entityId} not found`);
29
+ }
30
+
31
+ // Validate entity ownership
32
+ // If User object provided, use ownsUserId to check linked users
33
+ // Otherwise fall back to simple equality check
34
+ const isOwned = typeof userIdOrUser === 'object' && userIdOrUser?.ownsUserId
35
+ ? userIdOrUser.ownsUserId(entity.userId)
36
+ : entity.userId?.toString() === userId?.toString();
37
+
38
+ if (!isOwned) {
39
+ throw new Error(
40
+ `Entity ${entityId} does not belong to user ${userId}`
41
+ );
42
+ }
43
+
44
+ const entityType = entity.moduleName;
45
+ const moduleDefinition = this.moduleDefinitions.find((def) => {
46
+ const modelName = Module.getEntityModelFromDefinition(def).modelName;
47
+ return entityType === modelName;
48
+ });
49
+
50
+ if (!moduleDefinition) {
51
+ throw new Error(
52
+ `Module definition not found for entity type: ${entityType}`
53
+ );
54
+ }
55
+
56
+ const module = new Module({
57
+ userId,
58
+ entity,
59
+ definition: moduleDefinition,
60
+ });
61
+
62
+ // todo: this properties should be methods in the Module class
63
+ return {
64
+ id: module.entity.id,
65
+ name: module.entity.name,
66
+ moduleName: module.entity.moduleName,
67
+ credential: module.credential,
68
+ externalId: module.entity.externalId,
69
+ userId: module.entity.user.toString(),
70
+ }
71
+ }
72
+ }
73
+
74
+ module.exports = { GetModule };
@@ -0,0 +1,177 @@
1
+ const { Module } = require('../module');
2
+ const { ModuleConstants } = require('../ModuleConstants');
3
+
4
+ // Statuses considered "broken" for an integration whose credentials have just
5
+ // been successfully re-authorized. Both ERROR (system-driven auth failure) and
6
+ // DISABLED (user paused the integration) are flipped back to ENABLED when the
7
+ // user completes a new authorization flow.
8
+ const STATUSES_RESET_ON_REAUTH = ['ERROR', 'DISABLED'];
9
+
10
+ class ProcessAuthorizationCallback {
11
+ /**
12
+ * @param {Object} params - Configuration parameters.
13
+ * @param {import('../repositories/module-repository-factory').ModuleRepositoryInterface} params.moduleRepository - Repository for module data operations.
14
+ * @param {import('../../credential/repositories/credential-repository-factory').CredentialRepositoryInterface} params.credentialRepository - Repository for credential data operations.
15
+ * @param {Array<Object>} params.moduleDefinitions - Array of module definitions.
16
+ * @param {import('../../integrations/repositories/integration-repository-interface').IntegrationRepositoryInterface} [params.integrationRepository] - Repository for integration data operations. When provided, integrations in a broken state linked to the re-authorized entity are restored to ENABLED.
17
+ */
18
+ constructor({
19
+ moduleRepository,
20
+ credentialRepository,
21
+ moduleDefinitions,
22
+ integrationRepository,
23
+ }) {
24
+ this.moduleRepository = moduleRepository;
25
+ this.credentialRepository = credentialRepository;
26
+ this.moduleDefinitions = moduleDefinitions;
27
+ this.integrationRepository = integrationRepository;
28
+ }
29
+
30
+ async execute(userId, entityType, params) {
31
+ const moduleDefinition = this.moduleDefinitions.find((def) => {
32
+ return entityType === def.moduleName;
33
+ });
34
+
35
+ if (!moduleDefinition) {
36
+ throw new Error(
37
+ `Module definition not found for entity type: ${entityType}`
38
+ );
39
+ }
40
+
41
+ // todo: check if we need to pass entity to Module, right now it's null
42
+ let entity = null;
43
+
44
+ const module = new Module({
45
+ userId,
46
+ entity,
47
+ definition: moduleDefinition,
48
+ });
49
+
50
+ let tokenResponse;
51
+ if (module.apiClass.requesterType === ModuleConstants.authType.oauth2) {
52
+ tokenResponse = await moduleDefinition.requiredAuthMethods.getToken(
53
+ module.api,
54
+ params
55
+ );
56
+ } else {
57
+ tokenResponse =
58
+ await moduleDefinition.requiredAuthMethods.setAuthParams(
59
+ module.api,
60
+ params
61
+ );
62
+ await this.onTokenUpdate(module, moduleDefinition, userId);
63
+ }
64
+
65
+ const authRes = await module.testAuth();
66
+ if (!authRes) {
67
+ throw new Error('Authorization failed');
68
+ }
69
+
70
+ const entityDetails =
71
+ await moduleDefinition.requiredAuthMethods.getEntityDetails(
72
+ module.api,
73
+ params,
74
+ tokenResponse,
75
+ userId
76
+ );
77
+
78
+ Object.assign(
79
+ entityDetails.details,
80
+ module.apiParamsFromEntity(module.api)
81
+ );
82
+
83
+ const persistedEntity = await this.findOrCreateEntity(
84
+ entityDetails,
85
+ entityType,
86
+ module.credential.id
87
+ );
88
+
89
+ // Best-effort: a hiccup here must not fail a successful re-auth whose
90
+ // credential + entity are already persisted. Operators can recover
91
+ // stuck integrations manually.
92
+ try {
93
+ await this.restoreIntegrationsForEntity(persistedEntity.id);
94
+ } catch (err) {
95
+ console.error(
96
+ `[Frigg] Failed to restore integrations for entity ${persistedEntity.id} after successful re-auth — manual intervention may be needed`,
97
+ err
98
+ );
99
+ }
100
+
101
+ return {
102
+ credential_id: module.credential.id,
103
+ entity_id: persistedEntity.id,
104
+ type: module.getName(),
105
+ };
106
+ }
107
+
108
+ async restoreIntegrationsForEntity(entityId) {
109
+ if (!this.integrationRepository) return;
110
+ const integrations =
111
+ await this.integrationRepository.findIntegrationsByEntityId(
112
+ entityId
113
+ );
114
+ for (const integration of integrations) {
115
+ if (STATUSES_RESET_ON_REAUTH.includes(integration.status)) {
116
+ console.log(
117
+ `[Frigg] Restoring integration ${integration.id} from ${integration.status} to ENABLED after successful re-auth (entityId=${entityId})`
118
+ );
119
+ await this.integrationRepository.updateIntegrationStatus(
120
+ integration.id,
121
+ 'ENABLED'
122
+ );
123
+ }
124
+ }
125
+ }
126
+
127
+ async onTokenUpdate(module, moduleDefinition, userId) {
128
+ const credentialDetails =
129
+ await moduleDefinition.requiredAuthMethods.getCredentialDetails(
130
+ module.api,
131
+ userId
132
+ );
133
+
134
+ Object.assign(
135
+ credentialDetails.details,
136
+ module.apiParamsFromCredential(module.api)
137
+ );
138
+ credentialDetails.details.authIsValid = true;
139
+
140
+ const persisted = await this.credentialRepository.upsertCredential(credentialDetails);
141
+ module.credential = persisted;
142
+ }
143
+
144
+ async findOrCreateEntity(entityDetails, moduleName, credentialId) {
145
+ const { identifiers, details } = entityDetails;
146
+
147
+ // Support both 'user' and 'userId' field names from module definitions
148
+ // Some modules use 'user' (legacy), others use 'userId' (newer pattern)
149
+ const userId = identifiers.user || identifiers.userId;
150
+
151
+ if (!userId) {
152
+ throw new Error(
153
+ `Module definition for ${moduleName} must return 'user' or 'userId' in identifiers from getEntityDetails(). ` +
154
+ `Without userId, entity lookup would match across all users (security issue).`
155
+ );
156
+ }
157
+
158
+ const existingEntity = await this.moduleRepository.findEntity({
159
+ externalId: identifiers.externalId,
160
+ user: userId,
161
+ moduleName: moduleName,
162
+ });
163
+
164
+ if (existingEntity) {
165
+ return existingEntity;
166
+ }
167
+
168
+ return await this.moduleRepository.createEntity({
169
+ ...identifiers,
170
+ ...details,
171
+ moduleName: moduleName,
172
+ credential: credentialId,
173
+ });
174
+ }
175
+ }
176
+
177
+ module.exports = { ProcessAuthorizationCallback };
@@ -0,0 +1,72 @@
1
+ const { Module } = require('../module');
2
+
3
+ class RefreshEntityOptions {
4
+ /**
5
+ * @param {Object} params
6
+ * @param {import('../repositories/module-repository-interface').ModuleRepositoryInterface} params.moduleRepository
7
+ * @param {} params.moduleDefinitions
8
+ */
9
+ constructor({ moduleRepository, moduleDefinitions }) {
10
+ this.moduleRepository = moduleRepository;
11
+ this.moduleDefinitions = moduleDefinitions;
12
+ }
13
+
14
+ /**
15
+ * Refresh entity options for a given entity
16
+ *
17
+ * @param {string|number} entityId - Entity ID to refresh
18
+ * @param {string|number|import('../../user/user').User} userIdOrUser - User ID or User object for validation
19
+ * @param {Object} options - Refresh options
20
+ * @returns {Promise<Object>} Updated entity options
21
+ */
22
+ async execute(entityId, userIdOrUser, options) {
23
+ // Support both userId (backward compatible) and User object (new pattern)
24
+ const userId = typeof userIdOrUser === 'object' && userIdOrUser?.getId
25
+ ? userIdOrUser.getId()
26
+ : userIdOrUser;
27
+
28
+ const entity = await this.moduleRepository.findEntityById(
29
+ entityId,
30
+ userId
31
+ );
32
+
33
+ if (!entity) {
34
+ throw new Error(`Entity ${entityId} not found`);
35
+ }
36
+
37
+ // Validate entity ownership
38
+ const isOwned = typeof userIdOrUser === 'object' && userIdOrUser?.ownsUserId
39
+ ? userIdOrUser.ownsUserId(entity.userId)
40
+ : entity.userId?.toString() === userId?.toString();
41
+
42
+ if (!isOwned) {
43
+ throw new Error(
44
+ `Entity ${entityId} does not belong to user ${userId}`
45
+ );
46
+ }
47
+
48
+ const entityType = entity.moduleName;
49
+ const moduleDefinition = this.moduleDefinitions.find((def) => {
50
+ const modelName =
51
+ Module.getEntityModelFromDefinition(def).modelName;
52
+ return entityType === modelName;
53
+ });
54
+
55
+ if (!moduleDefinition) {
56
+ throw new Error(
57
+ `Module definition not found for entity type: ${entityType}`
58
+ );
59
+ }
60
+
61
+ const module = new Module({
62
+ userId,
63
+ entity,
64
+ definition: moduleDefinition,
65
+ });
66
+
67
+ await module.refreshEntityOptions(options);
68
+ return module.getEntityOptions();
69
+ }
70
+ }
71
+
72
+ module.exports = { RefreshEntityOptions };
@@ -0,0 +1,72 @@
1
+ const { Module } = require('../module');
2
+
3
+ class TestModuleAuth {
4
+ /**
5
+ * @param {Object} params - Configuration parameters.
6
+ * @param {import('../repositories/module-repository-interface').ModuleRepositoryInterface} params.moduleRepository - Repository for module data operations.
7
+ * @param {Array<Object>} params.moduleDefinitions - Array of module definitions.
8
+ */
9
+ constructor({ moduleRepository, moduleDefinitions }) {
10
+ this.moduleRepository = moduleRepository;
11
+ this.moduleDefinitions = moduleDefinitions;
12
+ }
13
+
14
+ /**
15
+ * Test authentication for a module entity
16
+ *
17
+ * @param {string|number} entityId - Entity ID to test
18
+ * @param {string|number|import('../../user/user').User} userIdOrUser - User ID or User object for validation
19
+ * @returns {Promise<boolean>} Authentication test result
20
+ */
21
+ async execute(entityId, userIdOrUser) {
22
+ // Support both userId (backward compatible) and User object (new pattern)
23
+ const userId = typeof userIdOrUser === 'object' && userIdOrUser?.getId
24
+ ? userIdOrUser.getId()
25
+ : userIdOrUser;
26
+
27
+ const entity = await this.moduleRepository.findEntityById(
28
+ entityId,
29
+ userId
30
+ );
31
+
32
+ if (!entity) {
33
+ throw new Error(`Entity ${entityId} not found`);
34
+ }
35
+
36
+ // Validate entity ownership
37
+ const isOwned = typeof userIdOrUser === 'object' && userIdOrUser?.ownsUserId
38
+ ? userIdOrUser.ownsUserId(entity.userId)
39
+ : entity.userId?.toString() === userId?.toString();
40
+
41
+ if (!isOwned) {
42
+ throw new Error(
43
+ `Entity ${entityId} does not belong to user ${userId}`
44
+ );
45
+ }
46
+
47
+ const entityType = entity.moduleName;
48
+ const moduleDefinition = this.moduleDefinitions.find((def) => {
49
+ const modelName =
50
+ Module.getEntityModelFromDefinition(def).modelName;
51
+ return entityType === modelName;
52
+ });
53
+
54
+ if (!moduleDefinition) {
55
+ throw new Error(
56
+ `Module definition not found for entity type: ${entityType}`
57
+ );
58
+ }
59
+
60
+ const module = new Module({
61
+ userId,
62
+ entity,
63
+ definition: moduleDefinition,
64
+ });
65
+
66
+ const testAuthResponse = await module.testAuth();
67
+
68
+ return testAuthResponse;
69
+ }
70
+ }
71
+
72
+ module.exports = { TestModuleAuth };
@@ -0,0 +1,18 @@
1
+ /**
2
+ * @param {import('../module').Module} moduleInstance
3
+ * Convert a Module domain instance to a plain DTO suitable for JSON responses.
4
+ */
5
+ function mapModuleClassToModuleDTO(moduleInstance) {
6
+ if (!moduleInstance) return null;
7
+
8
+ return {
9
+ id: moduleInstance.entity.id,
10
+ name: moduleInstance.name,
11
+ userId: moduleInstance.userId,
12
+ entity: moduleInstance.entity,
13
+ credentialId: moduleInstance.credential?._id?.toString(),
14
+ type: moduleInstance.getName()
15
+ };
16
+ }
17
+
18
+ module.exports = { mapModuleClassToModuleDTO };
package/package.json CHANGED
@@ -1,52 +1,84 @@
1
1
  {
2
- "name": "@friggframework/core",
3
- "prettier": "@friggframework/prettier-config",
4
- "version": "2.0.0-next.8",
5
- "dependencies": {
6
- "@hapi/boom": "^10.0.1",
7
- "aws-sdk": "^2.1200.0",
8
- "bcryptjs": "^2.4.3",
9
- "common-tags": "^1.8.2",
10
- "express": "^4.18.2",
11
- "express-async-handler": "^1.2.0",
12
- "lodash": "^4.17.21",
13
- "lodash.get": "^4.4.2",
14
- "mongoose": "6.11.6",
15
- "node-fetch": "^2.6.7"
16
- },
17
- "devDependencies": {
18
- "@friggframework/eslint-config": "2.0.0-next.8",
19
- "@friggframework/prettier-config": "2.0.0-next.8",
20
- "@friggframework/test": "2.0.0-next.8",
21
- "@types/lodash": "^4.14.191",
22
- "@typescript-eslint/eslint-plugin": "^8.0.0",
23
- "chai": "^4.3.6",
24
- "eslint": "^8.22.0",
25
- "eslint-plugin-import": "^2.29.1",
26
- "eslint-plugin-n": "^17.10.2",
27
- "eslint-plugin-promise": "^7.0.0",
28
- "jest": "^29.7.0",
29
- "jest-runner-groups": "^2.2.0",
30
- "mongodb-memory-server": "^8.9.0",
31
- "prettier": "^2.8.5",
32
- "sinon": "^16.1.1",
33
- "typescript": "^5.0.2"
34
- },
35
- "scripts": {
36
- "lint:fix": "prettier --write --loglevel error . && eslint . --fix",
37
- "test": "jest --passWithNoTests # TODO"
38
- },
39
- "author": "",
40
- "license": "MIT",
41
- "main": "index.js",
42
- "repository": {
43
- "type": "git",
44
- "url": "git+https://github.com/friggframework/frigg.git"
45
- },
46
- "bugs": {
47
- "url": "https://github.com/friggframework/frigg/issues"
48
- },
49
- "homepage": "https://github.com/friggframework/frigg#readme",
50
- "description": "",
51
- "gitHead": "60b7a6f92954ad165186861a35fba66e40f60a42"
2
+ "name": "@friggframework/core",
3
+ "prettier": "@friggframework/prettier-config",
4
+ "version": "2.0.0-next.80",
5
+ "dependencies": {
6
+ "@aws-sdk/client-apigatewaymanagementapi": "^3.588.0",
7
+ "@aws-sdk/client-kms": "^3.588.0",
8
+ "@aws-sdk/client-lambda": "^3.714.0",
9
+ "@aws-sdk/client-sqs": "^3.588.0",
10
+ "@hapi/boom": "^10.0.1",
11
+ "bcryptjs": "^2.4.3",
12
+ "body-parser": "^1.20.2",
13
+ "bson": "^4.7.2",
14
+ "chalk": "^4.1.2",
15
+ "common-tags": "^1.8.2",
16
+ "cors": "^2.8.5",
17
+ "dotenv": "^16.4.7",
18
+ "express": "^4.19.2",
19
+ "express-async-handler": "^1.2.0",
20
+ "form-data": "^4.0.0",
21
+ "fs-extra": "^11.2.0",
22
+ "lodash": "4.17.21",
23
+ "lodash.get": "^4.4.2",
24
+ "node-fetch": "^2.6.7",
25
+ "serverless-http": "^2.7.0",
26
+ "uuid": "^9.0.1"
27
+ },
28
+ "peerDependencies": {
29
+ "@prisma/client": "^6.16.3",
30
+ "prisma": "^6.16.3"
31
+ },
32
+ "peerDependenciesMeta": {
33
+ "@prisma/client": {
34
+ "optional": true
35
+ },
36
+ "prisma": {
37
+ "optional": true
38
+ }
39
+ },
40
+ "devDependencies": {
41
+ "@friggframework/eslint-config": "2.0.0-next.80",
42
+ "@friggframework/prettier-config": "2.0.0-next.80",
43
+ "@friggframework/test": "2.0.0-next.80",
44
+ "@prisma/client": "^6.17.0",
45
+ "@types/lodash": "4.17.15",
46
+ "@typescript-eslint/eslint-plugin": "^8.0.0",
47
+ "chai": "^4.3.6",
48
+ "eslint": "^8.22.0",
49
+ "eslint-plugin-import": "^2.29.1",
50
+ "eslint-plugin-n": "^17.10.2",
51
+ "eslint-plugin-promise": "^7.0.0",
52
+ "jest": "^29.7.0",
53
+ "prettier": "^2.7.1",
54
+ "prisma": "^6.17.0",
55
+ "sinon": "^16.1.1",
56
+ "typescript": "^5.0.2"
57
+ },
58
+ "scripts": {
59
+ "lint:fix": "prettier --write --loglevel error . && eslint . --fix",
60
+ "test": "jest --passWithNoTests # TODO",
61
+ "prisma:generate:mongo": "npx prisma generate --schema ./prisma-mongodb/schema.prisma",
62
+ "prisma:generate:postgres": "npx prisma generate --schema ./prisma-postgresql/schema.prisma",
63
+ "prisma:generate": "npm run prisma:generate:mongo && npm run prisma:generate:postgres",
64
+ "prisma:push:mongo": "npx prisma db push --schema ./prisma-mongodb/schema.prisma",
65
+ "prisma:migrate:postgres": "npx prisma migrate dev --schema ./prisma-postgresql/schema.prisma",
66
+ "prepublishOnly": "npm run prisma:generate"
67
+ },
68
+ "author": "",
69
+ "license": "MIT",
70
+ "main": "index.js",
71
+ "repository": {
72
+ "type": "git",
73
+ "url": "git+https://github.com/friggframework/frigg.git"
74
+ },
75
+ "bugs": {
76
+ "url": "https://github.com/friggframework/frigg/issues"
77
+ },
78
+ "homepage": "https://github.com/friggframework/frigg#readme",
79
+ "description": "",
80
+ "publishConfig": {
81
+ "access": "public"
82
+ },
83
+ "gitHead": "7a66dfcb02143941411ff26aaee866afc1473df8"
52
84
  }