@friggframework/core 2.0.0-next.7 → 2.0.0-next.70

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 (293) 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/core/CLAUDE.md +690 -0
  11. package/core/Worker.js +8 -21
  12. package/core/create-handler.js +14 -7
  13. package/credential/repositories/credential-repository-documentdb.js +304 -0
  14. package/credential/repositories/credential-repository-factory.js +54 -0
  15. package/credential/repositories/credential-repository-interface.js +98 -0
  16. package/credential/repositories/credential-repository-mongo.js +269 -0
  17. package/credential/repositories/credential-repository-postgres.js +287 -0
  18. package/credential/repositories/credential-repository.js +300 -0
  19. package/credential/use-cases/get-credential-for-user.js +25 -0
  20. package/credential/use-cases/update-authentication-status.js +15 -0
  21. package/database/MONGODB_TRANSACTION_FIX.md +198 -0
  22. package/database/adapters/lambda-invoker.js +97 -0
  23. package/database/config.js +154 -0
  24. package/database/documentdb-encryption-service.js +330 -0
  25. package/database/documentdb-utils.js +136 -0
  26. package/database/encryption/README.md +839 -0
  27. package/database/encryption/documentdb-encryption-service.md +3575 -0
  28. package/database/encryption/encryption-schema-registry.js +268 -0
  29. package/database/encryption/field-encryption-service.js +226 -0
  30. package/database/encryption/logger.js +79 -0
  31. package/database/encryption/prisma-encryption-extension.js +222 -0
  32. package/database/index.js +61 -21
  33. package/database/models/WebsocketConnection.js +16 -10
  34. package/database/models/readme.md +1 -0
  35. package/database/prisma.js +182 -0
  36. package/database/repositories/health-check-repository-documentdb.js +134 -0
  37. package/database/repositories/health-check-repository-factory.js +48 -0
  38. package/database/repositories/health-check-repository-interface.js +82 -0
  39. package/database/repositories/health-check-repository-mongodb.js +89 -0
  40. package/database/repositories/health-check-repository-postgres.js +82 -0
  41. package/database/repositories/health-check-repository.js +108 -0
  42. package/database/repositories/migration-status-repository-s3.js +137 -0
  43. package/database/use-cases/check-database-health-use-case.js +29 -0
  44. package/database/use-cases/check-database-state-use-case.js +81 -0
  45. package/database/use-cases/check-encryption-health-use-case.js +83 -0
  46. package/database/use-cases/get-database-state-via-worker-use-case.js +61 -0
  47. package/database/use-cases/get-migration-status-use-case.js +93 -0
  48. package/database/use-cases/run-database-migration-use-case.js +139 -0
  49. package/database/use-cases/test-encryption-use-case.js +253 -0
  50. package/database/use-cases/trigger-database-migration-use-case.js +157 -0
  51. package/database/utils/mongodb-collection-utils.js +91 -0
  52. package/database/utils/mongodb-schema-init.js +106 -0
  53. package/database/utils/prisma-runner.js +477 -0
  54. package/database/utils/prisma-schema-parser.js +182 -0
  55. package/docs/PROCESS_MANAGEMENT_QUEUE_SPEC.md +517 -0
  56. package/encrypt/Cryptor.js +34 -168
  57. package/encrypt/index.js +1 -2
  58. package/encrypt/test-encrypt.js +0 -2
  59. package/errors/client-safe-error.js +26 -0
  60. package/errors/fetch-error.js +2 -1
  61. package/errors/index.js +2 -0
  62. package/generated/prisma-mongodb/client.d.ts +1 -0
  63. package/generated/prisma-mongodb/client.js +4 -0
  64. package/generated/prisma-mongodb/default.d.ts +1 -0
  65. package/generated/prisma-mongodb/default.js +4 -0
  66. package/generated/prisma-mongodb/edge.d.ts +1 -0
  67. package/generated/prisma-mongodb/edge.js +335 -0
  68. package/generated/prisma-mongodb/index-browser.js +317 -0
  69. package/generated/prisma-mongodb/index.d.ts +22955 -0
  70. package/generated/prisma-mongodb/index.js +360 -0
  71. package/generated/prisma-mongodb/package.json +183 -0
  72. package/generated/prisma-mongodb/query-engine-debian-openssl-3.0.x +0 -0
  73. package/generated/prisma-mongodb/query-engine-rhel-openssl-3.0.x +0 -0
  74. package/generated/prisma-mongodb/runtime/binary.d.ts +1 -0
  75. package/generated/prisma-mongodb/runtime/binary.js +289 -0
  76. package/generated/prisma-mongodb/runtime/edge-esm.js +34 -0
  77. package/generated/prisma-mongodb/runtime/edge.js +34 -0
  78. package/generated/prisma-mongodb/runtime/index-browser.d.ts +370 -0
  79. package/generated/prisma-mongodb/runtime/index-browser.js +16 -0
  80. package/generated/prisma-mongodb/runtime/library.d.ts +3977 -0
  81. package/generated/prisma-mongodb/runtime/react-native.js +83 -0
  82. package/generated/prisma-mongodb/runtime/wasm-compiler-edge.js +84 -0
  83. package/generated/prisma-mongodb/runtime/wasm-engine-edge.js +36 -0
  84. package/generated/prisma-mongodb/schema.prisma +362 -0
  85. package/generated/prisma-mongodb/wasm-edge-light-loader.mjs +4 -0
  86. package/generated/prisma-mongodb/wasm-worker-loader.mjs +4 -0
  87. package/generated/prisma-mongodb/wasm.d.ts +1 -0
  88. package/generated/prisma-mongodb/wasm.js +342 -0
  89. package/generated/prisma-postgresql/client.d.ts +1 -0
  90. package/generated/prisma-postgresql/client.js +4 -0
  91. package/generated/prisma-postgresql/default.d.ts +1 -0
  92. package/generated/prisma-postgresql/default.js +4 -0
  93. package/generated/prisma-postgresql/edge.d.ts +1 -0
  94. package/generated/prisma-postgresql/edge.js +357 -0
  95. package/generated/prisma-postgresql/index-browser.js +339 -0
  96. package/generated/prisma-postgresql/index.d.ts +25131 -0
  97. package/generated/prisma-postgresql/index.js +382 -0
  98. package/generated/prisma-postgresql/package.json +183 -0
  99. package/generated/prisma-postgresql/query-engine-debian-openssl-3.0.x +0 -0
  100. package/generated/prisma-postgresql/query-engine-rhel-openssl-3.0.x +0 -0
  101. package/generated/prisma-postgresql/query_engine_bg.js +2 -0
  102. package/generated/prisma-postgresql/query_engine_bg.wasm +0 -0
  103. package/generated/prisma-postgresql/runtime/binary.d.ts +1 -0
  104. package/generated/prisma-postgresql/runtime/binary.js +289 -0
  105. package/generated/prisma-postgresql/runtime/edge-esm.js +34 -0
  106. package/generated/prisma-postgresql/runtime/edge.js +34 -0
  107. package/generated/prisma-postgresql/runtime/index-browser.d.ts +370 -0
  108. package/generated/prisma-postgresql/runtime/index-browser.js +16 -0
  109. package/generated/prisma-postgresql/runtime/library.d.ts +3977 -0
  110. package/generated/prisma-postgresql/runtime/react-native.js +83 -0
  111. package/generated/prisma-postgresql/runtime/wasm-compiler-edge.js +84 -0
  112. package/generated/prisma-postgresql/runtime/wasm-engine-edge.js +36 -0
  113. package/generated/prisma-postgresql/schema.prisma +345 -0
  114. package/generated/prisma-postgresql/wasm-edge-light-loader.mjs +4 -0
  115. package/generated/prisma-postgresql/wasm-worker-loader.mjs +4 -0
  116. package/generated/prisma-postgresql/wasm.d.ts +1 -0
  117. package/generated/prisma-postgresql/wasm.js +364 -0
  118. package/handlers/WEBHOOKS.md +653 -0
  119. package/handlers/app-definition-loader.js +38 -0
  120. package/handlers/app-handler-helpers.js +56 -0
  121. package/handlers/backend-utils.js +186 -0
  122. package/handlers/database-migration-handler.js +227 -0
  123. package/handlers/integration-event-dispatcher.js +54 -0
  124. package/handlers/routers/HEALTHCHECK.md +342 -0
  125. package/handlers/routers/auth.js +15 -0
  126. package/handlers/routers/db-migration.handler.js +29 -0
  127. package/handlers/routers/db-migration.js +326 -0
  128. package/handlers/routers/health.js +516 -0
  129. package/handlers/routers/integration-defined-routers.js +45 -0
  130. package/handlers/routers/integration-webhook-routers.js +67 -0
  131. package/handlers/routers/user.js +63 -0
  132. package/handlers/routers/websocket.js +57 -0
  133. package/handlers/use-cases/check-external-apis-health-use-case.js +81 -0
  134. package/handlers/use-cases/check-integrations-health-use-case.js +44 -0
  135. package/handlers/workers/db-migration.js +352 -0
  136. package/handlers/workers/integration-defined-workers.js +27 -0
  137. package/index.js +78 -22
  138. package/infrastructure/scheduler/eventbridge-scheduler-adapter.js +184 -0
  139. package/infrastructure/scheduler/index.js +33 -0
  140. package/infrastructure/scheduler/mock-scheduler-adapter.js +143 -0
  141. package/infrastructure/scheduler/scheduler-service-factory.js +73 -0
  142. package/infrastructure/scheduler/scheduler-service-interface.js +47 -0
  143. package/integrations/WEBHOOK-QUICKSTART.md +151 -0
  144. package/integrations/index.js +12 -10
  145. package/integrations/integration-base.js +326 -55
  146. package/integrations/integration-router.js +374 -179
  147. package/integrations/options.js +1 -1
  148. package/integrations/repositories/integration-mapping-repository-documentdb.js +280 -0
  149. package/integrations/repositories/integration-mapping-repository-factory.js +57 -0
  150. package/integrations/repositories/integration-mapping-repository-interface.js +106 -0
  151. package/integrations/repositories/integration-mapping-repository-mongo.js +161 -0
  152. package/integrations/repositories/integration-mapping-repository-postgres.js +227 -0
  153. package/integrations/repositories/integration-mapping-repository.js +156 -0
  154. package/integrations/repositories/integration-repository-documentdb.js +210 -0
  155. package/integrations/repositories/integration-repository-factory.js +51 -0
  156. package/integrations/repositories/integration-repository-interface.js +127 -0
  157. package/integrations/repositories/integration-repository-mongo.js +303 -0
  158. package/integrations/repositories/integration-repository-postgres.js +352 -0
  159. package/integrations/repositories/process-repository-documentdb.js +243 -0
  160. package/integrations/repositories/process-repository-factory.js +53 -0
  161. package/integrations/repositories/process-repository-interface.js +90 -0
  162. package/integrations/repositories/process-repository-mongo.js +190 -0
  163. package/integrations/repositories/process-repository-postgres.js +217 -0
  164. package/integrations/tests/doubles/config-capturing-integration.js +81 -0
  165. package/integrations/tests/doubles/dummy-integration-class.js +105 -0
  166. package/integrations/tests/doubles/test-integration-repository.js +99 -0
  167. package/integrations/use-cases/create-integration.js +83 -0
  168. package/integrations/use-cases/create-process.js +128 -0
  169. package/integrations/use-cases/delete-integration-for-user.js +101 -0
  170. package/integrations/use-cases/find-integration-context-by-external-entity-id.js +72 -0
  171. package/integrations/use-cases/get-integration-for-user.js +78 -0
  172. package/integrations/use-cases/get-integration-instance-by-definition.js +67 -0
  173. package/integrations/use-cases/get-integration-instance.js +83 -0
  174. package/integrations/use-cases/get-integrations-for-user.js +88 -0
  175. package/integrations/use-cases/get-possible-integrations.js +27 -0
  176. package/integrations/use-cases/get-process.js +87 -0
  177. package/integrations/use-cases/index.js +19 -0
  178. package/integrations/use-cases/load-integration-context.js +71 -0
  179. package/integrations/use-cases/update-integration-messages.js +44 -0
  180. package/integrations/use-cases/update-integration-status.js +32 -0
  181. package/integrations/use-cases/update-integration.js +92 -0
  182. package/integrations/use-cases/update-process-metrics.js +201 -0
  183. package/integrations/use-cases/update-process-state.js +119 -0
  184. package/integrations/utils/map-integration-dto.js +37 -0
  185. package/jest-global-setup-noop.js +3 -0
  186. package/jest-global-teardown-noop.js +3 -0
  187. package/logs/logger.js +0 -4
  188. package/{module-plugin → modules}/entity.js +1 -1
  189. package/{module-plugin → modules}/index.js +0 -8
  190. package/modules/module-factory.js +56 -0
  191. package/modules/module.js +221 -0
  192. package/modules/repositories/module-repository-documentdb.js +335 -0
  193. package/modules/repositories/module-repository-factory.js +40 -0
  194. package/modules/repositories/module-repository-interface.js +129 -0
  195. package/modules/repositories/module-repository-mongo.js +408 -0
  196. package/modules/repositories/module-repository-postgres.js +453 -0
  197. package/modules/repositories/module-repository.js +345 -0
  198. package/modules/requester/api-key.js +52 -0
  199. package/modules/requester/oauth-2.js +384 -0
  200. package/{module-plugin → modules}/requester/requester.js +4 -2
  201. package/{module-plugin → modules}/test/mock-api/api.js +8 -3
  202. package/{module-plugin → modules}/test/mock-api/definition.js +14 -10
  203. package/modules/tests/doubles/test-module-factory.js +16 -0
  204. package/modules/tests/doubles/test-module-repository.js +39 -0
  205. package/modules/use-cases/get-entities-for-user.js +32 -0
  206. package/modules/use-cases/get-entity-options-by-id.js +71 -0
  207. package/modules/use-cases/get-entity-options-by-type.js +34 -0
  208. package/modules/use-cases/get-module-instance-from-type.js +31 -0
  209. package/modules/use-cases/get-module.js +74 -0
  210. package/modules/use-cases/process-authorization-callback.js +133 -0
  211. package/modules/use-cases/refresh-entity-options.js +72 -0
  212. package/modules/use-cases/test-module-auth.js +72 -0
  213. package/modules/utils/map-module-dto.js +18 -0
  214. package/package.json +82 -50
  215. package/prisma-mongodb/schema.prisma +362 -0
  216. package/prisma-postgresql/migrations/20250930193005_init/migration.sql +315 -0
  217. package/prisma-postgresql/migrations/20251006135218_init/migration.sql +9 -0
  218. package/prisma-postgresql/migrations/20251010000000_remove_unused_entity_reference_map/migration.sql +3 -0
  219. package/prisma-postgresql/migrations/20251112195422_update_user_unique_constraints/migration.sql +25 -0
  220. package/prisma-postgresql/migrations/migration_lock.toml +3 -0
  221. package/prisma-postgresql/schema.prisma +345 -0
  222. package/queues/queuer-util.js +27 -22
  223. package/syncs/manager.js +468 -443
  224. package/syncs/repositories/sync-repository-documentdb.js +240 -0
  225. package/syncs/repositories/sync-repository-factory.js +43 -0
  226. package/syncs/repositories/sync-repository-interface.js +109 -0
  227. package/syncs/repositories/sync-repository-mongo.js +239 -0
  228. package/syncs/repositories/sync-repository-postgres.js +319 -0
  229. package/syncs/sync.js +0 -1
  230. package/token/repositories/token-repository-documentdb.js +137 -0
  231. package/token/repositories/token-repository-factory.js +40 -0
  232. package/token/repositories/token-repository-interface.js +131 -0
  233. package/token/repositories/token-repository-mongo.js +219 -0
  234. package/token/repositories/token-repository-postgres.js +264 -0
  235. package/token/repositories/token-repository.js +219 -0
  236. package/types/core/index.d.ts +2 -2
  237. package/types/integrations/index.d.ts +2 -6
  238. package/types/module-plugin/index.d.ts +5 -59
  239. package/types/syncs/index.d.ts +0 -2
  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/database/models/State.js +0 -9
  265. package/database/models/Token.js +0 -70
  266. package/database/mongo.js +0 -45
  267. package/encrypt/Cryptor.test.js +0 -32
  268. package/encrypt/encrypt.js +0 -132
  269. package/encrypt/encrypt.test.js +0 -1069
  270. package/errors/base-error.test.js +0 -32
  271. package/errors/fetch-error.test.js +0 -79
  272. package/errors/halt-error.test.js +0 -11
  273. package/errors/validation-errors.test.js +0 -120
  274. package/integrations/create-frigg-backend.js +0 -31
  275. package/integrations/integration-factory.js +0 -251
  276. package/integrations/integration-mapping.js +0 -43
  277. package/integrations/integration-model.js +0 -46
  278. package/integrations/integration-user.js +0 -144
  279. package/integrations/test/integration-base.test.js +0 -144
  280. package/lambda/TimeoutCatcher.test.js +0 -68
  281. package/logs/logger.test.js +0 -76
  282. package/module-plugin/auther.js +0 -393
  283. package/module-plugin/credential.js +0 -22
  284. package/module-plugin/entity-manager.js +0 -70
  285. package/module-plugin/manager.js +0 -169
  286. package/module-plugin/module-factory.js +0 -61
  287. package/module-plugin/requester/api-key.js +0 -36
  288. package/module-plugin/requester/oauth-2.js +0 -219
  289. package/module-plugin/requester/requester.test.js +0 -28
  290. package/module-plugin/test/auther.test.js +0 -97
  291. /package/{module-plugin → modules}/ModuleConstants.js +0 -0
  292. /package/{module-plugin → modules}/requester/basic.js +0 -0
  293. /package/{module-plugin → modules}/test/mock-api/mocks/hubspot.js +0 -0
@@ -0,0 +1,345 @@
1
+ const { prisma } = require('../../database/prisma');
2
+ const { ModuleRepositoryInterface } = require('./module-repository-interface');
3
+
4
+ /**
5
+ * Prisma-based Module Repository
6
+ * Handles Entity model operations for external service entities
7
+ *
8
+ * Works identically for both MongoDB and PostgreSQL:
9
+ * - MongoDB: String IDs with @db.ObjectId
10
+ * - PostgreSQL: Integer IDs with auto-increment
11
+ * - Both use same query patterns (no many-to-many differences)
12
+ *
13
+ * Migration from Mongoose:
14
+ * - Constructor injection of Prisma client
15
+ * - populate('credential') → include: { credential: true }
16
+ * - Mongoose discriminator (__t) → moduleName field (module type: salesforce, hubspot, etc.)
17
+ * - _id → id conversion automatic in Prisma
18
+ */
19
+ class ModuleRepository extends ModuleRepositoryInterface {
20
+ constructor(prismaClient = prisma) {
21
+ super();
22
+ this.prisma = prismaClient; // Allow injection for testing
23
+ }
24
+
25
+ /**
26
+ * Find entity by ID with credential
27
+ * Replaces: Entity.findById(entityId).populate('credential')
28
+ *
29
+ * @param {string} entityId - Entity ID
30
+ * @returns {Promise<Object>} Entity object
31
+ * @throws {Error} If entity not found
32
+ */
33
+ async findEntityById(entityId) {
34
+ const entity = await this.prisma.entity.findUnique({
35
+ where: { id: entityId },
36
+ include: { credential: true },
37
+ });
38
+
39
+ if (!entity) {
40
+ throw new Error(`Entity ${entityId} not found`);
41
+ }
42
+
43
+ return {
44
+ id: entity.id,
45
+ credential: entity.credential,
46
+ userId: entity.userId,
47
+ name: entity.name,
48
+ externalId: entity.externalId,
49
+ moduleName: entity.moduleName,
50
+ ...(entity.data || {}),
51
+ };
52
+ }
53
+
54
+ /**
55
+ * Find all entities for a user
56
+ * Replaces: Entity.find({ user: userId }).populate('credential')
57
+ *
58
+ * @param {string} userId - User ID
59
+ * @returns {Promise<Array>} Array of entity objects
60
+ */
61
+ async findEntitiesByUserId(userId) {
62
+ const entities = await this.prisma.entity.findMany({
63
+ where: { userId },
64
+ include: { credential: true },
65
+ });
66
+
67
+ return entities.map((e) => ({
68
+ id: e.id,
69
+ credential: e.credential,
70
+ userId: e.userId,
71
+ name: e.name,
72
+ externalId: e.externalId,
73
+ moduleName: e.moduleName,
74
+ ...(e.data || {}),
75
+ }));
76
+ }
77
+
78
+ /**
79
+ * Find entities by array of IDs
80
+ * Replaces: Entity.find({ _id: { $in: entitiesIds } }).populate('credential')
81
+ *
82
+ * @param {Array<string>} entitiesIds - Array of entity IDs
83
+ * @returns {Promise<Array>} Array of entity objects
84
+ */
85
+ async findEntitiesByIds(entitiesIds) {
86
+ const entities = await this.prisma.entity.findMany({
87
+ where: { id: { in: entitiesIds } },
88
+ include: { credential: true },
89
+ });
90
+
91
+ return entities.map((e) => ({
92
+ id: e.id,
93
+ credential: e.credential,
94
+ userId: e.userId,
95
+ name: e.name,
96
+ externalId: e.externalId,
97
+ moduleName: e.moduleName,
98
+ ...(e.data || {}),
99
+ }));
100
+ }
101
+
102
+ /**
103
+ * Find entities by user ID and module name
104
+ * Replaces: Entity.find({ user: userId, moduleName: moduleName }).populate('credential')
105
+ *
106
+ * @param {string} userId - User ID
107
+ * @param {string} moduleName - Module name
108
+ * @returns {Promise<Array>} Array of entity objects
109
+ */
110
+ async findEntitiesByUserIdAndModuleName(userId, moduleName) {
111
+ const entities = await this.prisma.entity.findMany({
112
+ where: {
113
+ userId,
114
+ moduleName,
115
+ },
116
+ include: { credential: true },
117
+ });
118
+
119
+ return entities.map((e) => ({
120
+ id: e.id,
121
+ credential: e.credential,
122
+ userId: e.userId,
123
+ name: e.name,
124
+ externalId: e.externalId,
125
+ moduleName: e.moduleName,
126
+ ...(e.data || {}),
127
+ }));
128
+ }
129
+
130
+ /**
131
+ * Remove credential reference from entity
132
+ * Replaces: Entity.updateOne({ _id: entityId }, { $unset: { credential: "" } })
133
+ *
134
+ * @param {string} entityId - Entity ID
135
+ * @returns {Promise<boolean>} Success indicator
136
+ */
137
+ async unsetCredential(entityId) {
138
+ await this.prisma.entity.update({
139
+ where: { id: entityId },
140
+ data: { credentialId: null },
141
+ });
142
+
143
+ return true;
144
+ }
145
+
146
+ /**
147
+ * Find entity by filter criteria
148
+ * Replaces: Entity.findOne(filter).populate('credential')
149
+ *
150
+ * @param {Object} filter - Filter criteria
151
+ * @returns {Promise<Object|null>} Entity object or null
152
+ */
153
+ async findEntity(filter) {
154
+ const where = this._convertFilterToWhere(filter);
155
+ const entity = await this.prisma.entity.findFirst({
156
+ where,
157
+ include: { credential: true },
158
+ });
159
+
160
+ if (!entity) {
161
+ return null;
162
+ }
163
+
164
+ return {
165
+ id: entity.id,
166
+ credential: entity.credential,
167
+ userId: entity.userId,
168
+ name: entity.name,
169
+ externalId: entity.externalId,
170
+ moduleName: entity.moduleName,
171
+ ...(entity.data || {}),
172
+ };
173
+ }
174
+
175
+ /**
176
+ * Create a new entity
177
+ * Replaces: Entity.create(entityData)
178
+ *
179
+ * @param {Object} entityData - Entity data
180
+ * @returns {Promise<Object>} Created entity object
181
+ */
182
+ async createEntity(entityData) {
183
+ const {
184
+ user,
185
+ userId,
186
+ credential,
187
+ credentialId,
188
+ name,
189
+ moduleName,
190
+ externalId,
191
+ ...dynamicData
192
+ } = entityData;
193
+
194
+ const data = {
195
+ userId: userId || user,
196
+ credentialId: credentialId || credential,
197
+ name,
198
+ moduleName,
199
+ externalId,
200
+ data: dynamicData,
201
+ };
202
+
203
+ const entity = await this.prisma.entity.create({
204
+ data,
205
+ include: { credential: true },
206
+ });
207
+
208
+ return {
209
+ id: entity.id,
210
+ credential: entity.credential,
211
+ userId: entity.userId,
212
+ name: entity.name,
213
+ externalId: entity.externalId,
214
+ moduleName: entity.moduleName,
215
+ ...(entity.data || {}),
216
+ };
217
+ }
218
+
219
+ /**
220
+ * Update an entity by ID
221
+ * Replaces: Entity.findByIdAndUpdate(entityId, updates, { new: true })
222
+ *
223
+ * @param {string} entityId - Entity ID to update
224
+ * @param {Object} updates - Fields to update
225
+ * @returns {Promise<Object|null>} Updated entity object or null if not found
226
+ */
227
+ async updateEntity(entityId, updates) {
228
+ const existing = await this.prisma.entity.findUnique({
229
+ where: { id: entityId },
230
+ });
231
+
232
+ if (!existing) {
233
+ return null;
234
+ }
235
+
236
+ const {
237
+ user,
238
+ userId,
239
+ credential,
240
+ credentialId,
241
+ name,
242
+ moduleName,
243
+ externalId,
244
+ ...dynamicData
245
+ } = updates;
246
+
247
+ const schemaUpdates = {};
248
+ if (user !== undefined || userId !== undefined) {
249
+ schemaUpdates.userId = userId || user;
250
+ }
251
+ if (credential !== undefined || credentialId !== undefined) {
252
+ schemaUpdates.credentialId = credentialId || credential;
253
+ }
254
+ if (name !== undefined) schemaUpdates.name = name;
255
+ if (moduleName !== undefined) schemaUpdates.moduleName = moduleName;
256
+ if (externalId !== undefined) schemaUpdates.externalId = externalId;
257
+
258
+ if (Object.keys(dynamicData).length > 0) {
259
+ schemaUpdates.data = { ...(existing.data || {}), ...dynamicData };
260
+ }
261
+
262
+ try {
263
+ const entity = await this.prisma.entity.update({
264
+ where: { id: entityId },
265
+ data: schemaUpdates,
266
+ include: { credential: true },
267
+ });
268
+
269
+ return {
270
+ id: entity.id,
271
+ credential: entity.credential,
272
+ userId: entity.userId,
273
+ name: entity.name,
274
+ externalId: entity.externalId,
275
+ moduleName: entity.moduleName,
276
+ ...(entity.data || {}),
277
+ };
278
+ } catch (error) {
279
+ if (error.code === 'P2025') {
280
+ // Record not found
281
+ return null;
282
+ }
283
+ throw error;
284
+ }
285
+ }
286
+
287
+ /**
288
+ * Delete an entity by ID
289
+ * Replaces: Entity.deleteOne({ _id: entityId })
290
+ *
291
+ * @param {string} entityId - Entity ID to delete
292
+ * @returns {Promise<boolean>} True if deleted successfully
293
+ */
294
+ async deleteEntity(entityId) {
295
+ try {
296
+ await this.prisma.entity.delete({
297
+ where: { id: entityId },
298
+ });
299
+ return true;
300
+ } catch (error) {
301
+ if (error.code === 'P2025') {
302
+ // Record not found
303
+ return false;
304
+ }
305
+ throw error;
306
+ }
307
+ }
308
+
309
+ /**
310
+ * Convert Mongoose-style filter to Prisma where clause
311
+ * @private
312
+ * @param {Object} filter - Mongoose filter
313
+ * @returns {Object} Prisma where clause
314
+ */
315
+ _convertFilterToWhere(filter) {
316
+ const where = {};
317
+
318
+ // Handle _id field (Mongoose uses _id, Prisma uses id)
319
+ if (filter._id) {
320
+ where.id = filter._id;
321
+ }
322
+
323
+ // Handle user field (Mongoose uses user, Prisma uses userId)
324
+ if (filter.user) {
325
+ where.userId = filter.user;
326
+ }
327
+
328
+ // Handle credential field (Mongoose uses credential, Prisma uses credentialId)
329
+ if (filter.credential) {
330
+ where.credentialId = filter.credential;
331
+ }
332
+
333
+ // Copy other fields directly
334
+ if (filter.id) where.id = filter.id;
335
+ if (filter.userId) where.userId = filter.userId;
336
+ if (filter.credentialId) where.credentialId = filter.credentialId;
337
+ if (filter.name) where.name = filter.name;
338
+ if (filter.moduleName) where.moduleName = filter.moduleName;
339
+ if (filter.externalId) where.externalId = filter.externalId;
340
+
341
+ return where;
342
+ }
343
+ }
344
+
345
+ module.exports = { ModuleRepository };
@@ -0,0 +1,52 @@
1
+ const { Requester } = require('./requester');
2
+ const { get } = require('../../assertions');
3
+ const { ModuleConstants } = require('../ModuleConstants');
4
+
5
+
6
+ class ApiKeyRequester extends Requester {
7
+
8
+ static requesterType = ModuleConstants.authType.apiKey;
9
+
10
+ constructor(params) {
11
+ super(params);
12
+ this.requesterType = 'apiKey';
13
+
14
+ // Use snake_case convention consistent with OAuth2Requester and BasicAuthRequester
15
+ this.api_key_name = get(params, 'api_key_name', 'key');
16
+ this.api_key = get(params, 'api_key', null);
17
+
18
+ // Backward compatibility: support old naming convention
19
+ if (!this.api_key && params.API_KEY_VALUE) {
20
+ this.api_key = params.API_KEY_VALUE;
21
+ }
22
+ if (!this.api_key_name && params.API_KEY_NAME) {
23
+ this.api_key_name = params.API_KEY_NAME;
24
+ }
25
+ }
26
+
27
+ async addAuthHeaders(headers) {
28
+ if (this.api_key) {
29
+ headers[this.api_key_name] = this.api_key;
30
+ }
31
+ return headers;
32
+ }
33
+
34
+ isAuthenticated() {
35
+ return (
36
+ this.api_key !== null &&
37
+ this.api_key !== undefined &&
38
+ typeof this.api_key === 'string' &&
39
+ this.api_key.trim().length > 0
40
+ );
41
+ }
42
+
43
+ setApiKey(api_key) {
44
+ this.api_key = api_key;
45
+ }
46
+
47
+ setApiKeyName(api_key_name) {
48
+ this.api_key_name = api_key_name;
49
+ }
50
+ }
51
+
52
+ module.exports = { ApiKeyRequester };