@friggframework/core 2.0.0-next.5 → 2.0.0-next.51

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 (267) hide show
  1. package/CLAUDE.md +693 -0
  2. package/README.md +959 -50
  3. package/application/commands/README.md +421 -0
  4. package/application/commands/credential-commands.js +224 -0
  5. package/application/commands/entity-commands.js +315 -0
  6. package/application/commands/integration-commands.js +179 -0
  7. package/application/commands/user-commands.js +213 -0
  8. package/application/index.js +69 -0
  9. package/core/CLAUDE.md +690 -0
  10. package/core/Worker.js +8 -21
  11. package/core/create-handler.js +2 -7
  12. package/credential/repositories/credential-repository-factory.js +47 -0
  13. package/credential/repositories/credential-repository-interface.js +98 -0
  14. package/credential/repositories/credential-repository-mongo.js +307 -0
  15. package/credential/repositories/credential-repository-postgres.js +313 -0
  16. package/credential/repositories/credential-repository.js +302 -0
  17. package/credential/use-cases/get-credential-for-user.js +21 -0
  18. package/credential/use-cases/update-authentication-status.js +15 -0
  19. package/database/MONGODB_TRANSACTION_FIX.md +198 -0
  20. package/database/adapters/lambda-invoker.js +97 -0
  21. package/database/config.js +154 -0
  22. package/database/encryption/README.md +684 -0
  23. package/database/encryption/encryption-schema-registry.js +141 -0
  24. package/database/encryption/field-encryption-service.js +226 -0
  25. package/database/encryption/logger.js +79 -0
  26. package/database/encryption/prisma-encryption-extension.js +222 -0
  27. package/database/index.js +25 -12
  28. package/database/models/WebsocketConnection.js +16 -10
  29. package/database/models/readme.md +1 -0
  30. package/database/prisma.js +222 -0
  31. package/database/repositories/health-check-repository-factory.js +43 -0
  32. package/database/repositories/health-check-repository-interface.js +87 -0
  33. package/database/repositories/health-check-repository-mongodb.js +91 -0
  34. package/database/repositories/health-check-repository-postgres.js +82 -0
  35. package/database/repositories/health-check-repository.js +108 -0
  36. package/database/repositories/migration-status-repository-s3.js +137 -0
  37. package/database/use-cases/check-database-health-use-case.js +29 -0
  38. package/database/use-cases/check-database-state-use-case.js +81 -0
  39. package/database/use-cases/check-encryption-health-use-case.js +83 -0
  40. package/database/use-cases/get-database-state-via-worker-use-case.js +61 -0
  41. package/database/use-cases/get-migration-status-use-case.js +93 -0
  42. package/database/use-cases/run-database-migration-use-case.js +137 -0
  43. package/database/use-cases/test-encryption-use-case.js +253 -0
  44. package/database/use-cases/trigger-database-migration-use-case.js +157 -0
  45. package/database/utils/mongodb-collection-utils.js +91 -0
  46. package/database/utils/mongodb-schema-init.js +106 -0
  47. package/database/utils/prisma-runner.js +400 -0
  48. package/database/utils/prisma-schema-parser.js +182 -0
  49. package/docs/PROCESS_MANAGEMENT_QUEUE_SPEC.md +517 -0
  50. package/encrypt/Cryptor.js +34 -168
  51. package/encrypt/index.js +1 -2
  52. package/encrypt/test-encrypt.js +0 -2
  53. package/generated/prisma-mongodb/client.d.ts +1 -0
  54. package/generated/prisma-mongodb/client.js +4 -0
  55. package/generated/prisma-mongodb/default.d.ts +1 -0
  56. package/generated/prisma-mongodb/default.js +4 -0
  57. package/generated/prisma-mongodb/edge.d.ts +1 -0
  58. package/generated/prisma-mongodb/edge.js +334 -0
  59. package/generated/prisma-mongodb/index-browser.js +316 -0
  60. package/generated/prisma-mongodb/index.d.ts +22898 -0
  61. package/generated/prisma-mongodb/index.js +359 -0
  62. package/generated/prisma-mongodb/package.json +183 -0
  63. package/generated/prisma-mongodb/query-engine-debian-openssl-3.0.x +0 -0
  64. package/generated/prisma-mongodb/query-engine-rhel-openssl-3.0.x +0 -0
  65. package/generated/prisma-mongodb/runtime/binary.d.ts +1 -0
  66. package/generated/prisma-mongodb/runtime/binary.js +289 -0
  67. package/generated/prisma-mongodb/runtime/edge-esm.js +34 -0
  68. package/generated/prisma-mongodb/runtime/edge.js +34 -0
  69. package/generated/prisma-mongodb/runtime/index-browser.d.ts +370 -0
  70. package/generated/prisma-mongodb/runtime/index-browser.js +16 -0
  71. package/generated/prisma-mongodb/runtime/library.d.ts +3982 -0
  72. package/generated/prisma-mongodb/runtime/react-native.js +83 -0
  73. package/generated/prisma-mongodb/runtime/wasm-compiler-edge.js +84 -0
  74. package/generated/prisma-mongodb/runtime/wasm-engine-edge.js +36 -0
  75. package/generated/prisma-mongodb/schema.prisma +362 -0
  76. package/generated/prisma-mongodb/wasm-edge-light-loader.mjs +4 -0
  77. package/generated/prisma-mongodb/wasm-worker-loader.mjs +4 -0
  78. package/generated/prisma-mongodb/wasm.d.ts +1 -0
  79. package/generated/prisma-mongodb/wasm.js +341 -0
  80. package/generated/prisma-postgresql/client.d.ts +1 -0
  81. package/generated/prisma-postgresql/client.js +4 -0
  82. package/generated/prisma-postgresql/default.d.ts +1 -0
  83. package/generated/prisma-postgresql/default.js +4 -0
  84. package/generated/prisma-postgresql/edge.d.ts +1 -0
  85. package/generated/prisma-postgresql/edge.js +356 -0
  86. package/generated/prisma-postgresql/index-browser.js +338 -0
  87. package/generated/prisma-postgresql/index.d.ts +25072 -0
  88. package/generated/prisma-postgresql/index.js +381 -0
  89. package/generated/prisma-postgresql/package.json +183 -0
  90. package/generated/prisma-postgresql/query-engine-debian-openssl-3.0.x +0 -0
  91. package/generated/prisma-postgresql/query-engine-rhel-openssl-3.0.x +0 -0
  92. package/generated/prisma-postgresql/query_engine_bg.js +2 -0
  93. package/generated/prisma-postgresql/query_engine_bg.wasm +0 -0
  94. package/generated/prisma-postgresql/runtime/binary.d.ts +1 -0
  95. package/generated/prisma-postgresql/runtime/binary.js +289 -0
  96. package/generated/prisma-postgresql/runtime/edge-esm.js +34 -0
  97. package/generated/prisma-postgresql/runtime/edge.js +34 -0
  98. package/generated/prisma-postgresql/runtime/index-browser.d.ts +370 -0
  99. package/generated/prisma-postgresql/runtime/index-browser.js +16 -0
  100. package/generated/prisma-postgresql/runtime/library.d.ts +3982 -0
  101. package/generated/prisma-postgresql/runtime/react-native.js +83 -0
  102. package/generated/prisma-postgresql/runtime/wasm-compiler-edge.js +84 -0
  103. package/generated/prisma-postgresql/runtime/wasm-engine-edge.js +36 -0
  104. package/generated/prisma-postgresql/schema.prisma +345 -0
  105. package/generated/prisma-postgresql/wasm-edge-light-loader.mjs +4 -0
  106. package/generated/prisma-postgresql/wasm-worker-loader.mjs +4 -0
  107. package/generated/prisma-postgresql/wasm.d.ts +1 -0
  108. package/generated/prisma-postgresql/wasm.js +363 -0
  109. package/handlers/WEBHOOKS.md +653 -0
  110. package/handlers/app-definition-loader.js +38 -0
  111. package/handlers/app-handler-helpers.js +56 -0
  112. package/handlers/backend-utils.js +180 -0
  113. package/handlers/database-migration-handler.js +227 -0
  114. package/handlers/integration-event-dispatcher.js +54 -0
  115. package/handlers/routers/HEALTHCHECK.md +342 -0
  116. package/handlers/routers/auth.js +15 -0
  117. package/handlers/routers/db-migration.handler.js +29 -0
  118. package/handlers/routers/db-migration.js +256 -0
  119. package/handlers/routers/health.js +519 -0
  120. package/handlers/routers/integration-defined-routers.js +45 -0
  121. package/handlers/routers/integration-webhook-routers.js +67 -0
  122. package/handlers/routers/user.js +63 -0
  123. package/handlers/routers/websocket.js +57 -0
  124. package/handlers/use-cases/check-external-apis-health-use-case.js +81 -0
  125. package/handlers/use-cases/check-integrations-health-use-case.js +44 -0
  126. package/handlers/workers/db-migration.js +352 -0
  127. package/handlers/workers/integration-defined-workers.js +27 -0
  128. package/index.js +77 -22
  129. package/integrations/WEBHOOK-QUICKSTART.md +151 -0
  130. package/integrations/index.js +12 -10
  131. package/integrations/integration-base.js +296 -54
  132. package/integrations/integration-router.js +381 -182
  133. package/integrations/options.js +1 -1
  134. package/integrations/repositories/integration-mapping-repository-factory.js +50 -0
  135. package/integrations/repositories/integration-mapping-repository-interface.js +106 -0
  136. package/integrations/repositories/integration-mapping-repository-mongo.js +161 -0
  137. package/integrations/repositories/integration-mapping-repository-postgres.js +227 -0
  138. package/integrations/repositories/integration-mapping-repository.js +156 -0
  139. package/integrations/repositories/integration-repository-factory.js +44 -0
  140. package/integrations/repositories/integration-repository-interface.js +127 -0
  141. package/integrations/repositories/integration-repository-mongo.js +303 -0
  142. package/integrations/repositories/integration-repository-postgres.js +352 -0
  143. package/integrations/repositories/process-repository-factory.js +46 -0
  144. package/integrations/repositories/process-repository-interface.js +90 -0
  145. package/integrations/repositories/process-repository-mongo.js +190 -0
  146. package/integrations/repositories/process-repository-postgres.js +217 -0
  147. package/integrations/tests/doubles/dummy-integration-class.js +83 -0
  148. package/integrations/tests/doubles/test-integration-repository.js +99 -0
  149. package/integrations/use-cases/create-integration.js +83 -0
  150. package/integrations/use-cases/create-process.js +128 -0
  151. package/integrations/use-cases/delete-integration-for-user.js +101 -0
  152. package/integrations/use-cases/find-integration-context-by-external-entity-id.js +72 -0
  153. package/integrations/use-cases/get-integration-for-user.js +78 -0
  154. package/integrations/use-cases/get-integration-instance-by-definition.js +67 -0
  155. package/integrations/use-cases/get-integration-instance.js +83 -0
  156. package/integrations/use-cases/get-integrations-for-user.js +88 -0
  157. package/integrations/use-cases/get-possible-integrations.js +27 -0
  158. package/integrations/use-cases/get-process.js +87 -0
  159. package/integrations/use-cases/index.js +19 -0
  160. package/integrations/use-cases/load-integration-context.js +71 -0
  161. package/integrations/use-cases/update-integration-messages.js +44 -0
  162. package/integrations/use-cases/update-integration-status.js +32 -0
  163. package/integrations/use-cases/update-integration.js +93 -0
  164. package/integrations/use-cases/update-process-metrics.js +201 -0
  165. package/integrations/use-cases/update-process-state.js +119 -0
  166. package/integrations/utils/map-integration-dto.js +37 -0
  167. package/jest-global-setup-noop.js +3 -0
  168. package/jest-global-teardown-noop.js +3 -0
  169. package/logs/logger.js +0 -4
  170. package/{module-plugin → modules}/entity.js +1 -1
  171. package/{module-plugin → modules}/index.js +0 -8
  172. package/modules/module-factory.js +56 -0
  173. package/modules/module.js +221 -0
  174. package/modules/repositories/module-repository-factory.js +33 -0
  175. package/modules/repositories/module-repository-interface.js +129 -0
  176. package/modules/repositories/module-repository-mongo.js +377 -0
  177. package/modules/repositories/module-repository-postgres.js +426 -0
  178. package/modules/repositories/module-repository.js +316 -0
  179. package/{module-plugin → modules}/requester/requester.js +1 -0
  180. package/{module-plugin → modules}/test/mock-api/api.js +8 -3
  181. package/{module-plugin → modules}/test/mock-api/definition.js +12 -8
  182. package/modules/tests/doubles/test-module-factory.js +16 -0
  183. package/modules/tests/doubles/test-module-repository.js +39 -0
  184. package/modules/use-cases/get-entities-for-user.js +32 -0
  185. package/modules/use-cases/get-entity-options-by-id.js +59 -0
  186. package/modules/use-cases/get-entity-options-by-type.js +34 -0
  187. package/modules/use-cases/get-module-instance-from-type.js +31 -0
  188. package/modules/use-cases/get-module.js +55 -0
  189. package/modules/use-cases/process-authorization-callback.js +122 -0
  190. package/modules/use-cases/refresh-entity-options.js +59 -0
  191. package/modules/use-cases/test-module-auth.js +55 -0
  192. package/modules/utils/map-module-dto.js +18 -0
  193. package/package.json +82 -50
  194. package/prisma-mongodb/schema.prisma +362 -0
  195. package/prisma-postgresql/migrations/20250930193005_init/migration.sql +315 -0
  196. package/prisma-postgresql/migrations/20251006135218_init/migration.sql +9 -0
  197. package/prisma-postgresql/migrations/20251010000000_remove_unused_entity_reference_map/migration.sql +3 -0
  198. package/prisma-postgresql/migrations/migration_lock.toml +3 -0
  199. package/prisma-postgresql/schema.prisma +345 -0
  200. package/queues/queuer-util.js +28 -15
  201. package/syncs/manager.js +468 -443
  202. package/syncs/repositories/sync-repository-factory.js +38 -0
  203. package/syncs/repositories/sync-repository-interface.js +109 -0
  204. package/syncs/repositories/sync-repository-mongo.js +239 -0
  205. package/syncs/repositories/sync-repository-postgres.js +319 -0
  206. package/syncs/sync.js +0 -1
  207. package/token/repositories/token-repository-factory.js +33 -0
  208. package/token/repositories/token-repository-interface.js +131 -0
  209. package/token/repositories/token-repository-mongo.js +212 -0
  210. package/token/repositories/token-repository-postgres.js +257 -0
  211. package/token/repositories/token-repository.js +219 -0
  212. package/types/core/index.d.ts +2 -2
  213. package/types/integrations/index.d.ts +2 -6
  214. package/types/module-plugin/index.d.ts +5 -59
  215. package/types/syncs/index.d.ts +0 -2
  216. package/user/repositories/user-repository-factory.js +46 -0
  217. package/user/repositories/user-repository-interface.js +198 -0
  218. package/user/repositories/user-repository-mongo.js +291 -0
  219. package/user/repositories/user-repository-postgres.js +350 -0
  220. package/user/tests/doubles/test-user-repository.js +72 -0
  221. package/user/use-cases/authenticate-user.js +127 -0
  222. package/user/use-cases/authenticate-with-shared-secret.js +48 -0
  223. package/user/use-cases/create-individual-user.js +61 -0
  224. package/user/use-cases/create-organization-user.js +47 -0
  225. package/user/use-cases/create-token-for-user-id.js +30 -0
  226. package/user/use-cases/get-user-from-adopter-jwt.js +149 -0
  227. package/user/use-cases/get-user-from-bearer-token.js +77 -0
  228. package/user/use-cases/get-user-from-x-frigg-headers.js +106 -0
  229. package/user/use-cases/login-user.js +122 -0
  230. package/user/user.js +93 -0
  231. package/utils/backend-path.js +38 -0
  232. package/utils/index.js +6 -0
  233. package/websocket/repositories/websocket-connection-repository-factory.js +37 -0
  234. package/websocket/repositories/websocket-connection-repository-interface.js +106 -0
  235. package/websocket/repositories/websocket-connection-repository-mongo.js +156 -0
  236. package/websocket/repositories/websocket-connection-repository-postgres.js +196 -0
  237. package/websocket/repositories/websocket-connection-repository.js +161 -0
  238. package/database/models/State.js +0 -9
  239. package/database/models/Token.js +0 -70
  240. package/database/mongo.js +0 -45
  241. package/encrypt/Cryptor.test.js +0 -32
  242. package/encrypt/encrypt.js +0 -132
  243. package/encrypt/encrypt.test.js +0 -1069
  244. package/errors/base-error.test.js +0 -32
  245. package/errors/fetch-error.test.js +0 -79
  246. package/errors/halt-error.test.js +0 -11
  247. package/errors/validation-errors.test.js +0 -120
  248. package/integrations/create-frigg-backend.js +0 -31
  249. package/integrations/integration-factory.js +0 -251
  250. package/integrations/integration-mapping.js +0 -43
  251. package/integrations/integration-model.js +0 -46
  252. package/integrations/integration-user.js +0 -144
  253. package/integrations/test/integration-base.test.js +0 -144
  254. package/lambda/TimeoutCatcher.test.js +0 -68
  255. package/logs/logger.test.js +0 -76
  256. package/module-plugin/auther.js +0 -393
  257. package/module-plugin/credential.js +0 -22
  258. package/module-plugin/entity-manager.js +0 -70
  259. package/module-plugin/manager.js +0 -169
  260. package/module-plugin/module-factory.js +0 -61
  261. package/module-plugin/requester/requester.test.js +0 -28
  262. package/module-plugin/test/auther.test.js +0 -97
  263. /package/{module-plugin → modules}/ModuleConstants.js +0 -0
  264. /package/{module-plugin → modules}/requester/api-key.js +0 -0
  265. /package/{module-plugin → modules}/requester/basic.js +0 -0
  266. /package/{module-plugin → modules}/requester/oauth-2.js +0 -0
  267. /package/{module-plugin → modules}/test/mock-api/mocks/hubspot.js +0 -0
@@ -1,5 +1,5 @@
1
1
  const { RequiredPropertyError } = require('../errors');
2
- const { get, getAndVerifyType } = require('../assertions');
2
+ const { get } = require('../assertions');
3
3
 
4
4
  class Options {
5
5
  constructor(params) {
@@ -0,0 +1,50 @@
1
+ const {
2
+ IntegrationMappingRepositoryMongo,
3
+ } = require('./integration-mapping-repository-mongo');
4
+ const {
5
+ IntegrationMappingRepositoryPostgres,
6
+ } = require('./integration-mapping-repository-postgres');
7
+ const config = require('../../database/config');
8
+
9
+ /**
10
+ * Integration Mapping Repository Factory
11
+ * Creates the appropriate repository adapter based on database type
12
+ *
13
+ * Database-specific implementations:
14
+ * - MongoDB: Uses String IDs (ObjectId), no conversion needed
15
+ * - PostgreSQL: Uses Int IDs, converts String ↔ Int
16
+ *
17
+ * All repository methods return String IDs regardless of database type,
18
+ * ensuring application layer consistency.
19
+ *
20
+ * Usage:
21
+ * ```javascript
22
+ * const repository = createIntegrationMappingRepository();
23
+ * const mapping = await repository.findMappingBy(integrationId, sourceId);
24
+ * ```
25
+ *
26
+ * @returns {IntegrationMappingRepositoryInterface} Configured repository adapter
27
+ */
28
+ function createIntegrationMappingRepository() {
29
+ const dbType = config.DB_TYPE;
30
+
31
+ switch (dbType) {
32
+ case 'mongodb':
33
+ return new IntegrationMappingRepositoryMongo();
34
+
35
+ case 'postgresql':
36
+ return new IntegrationMappingRepositoryPostgres();
37
+
38
+ default:
39
+ throw new Error(
40
+ `Unsupported database type: ${dbType}. Supported values: 'mongodb', 'postgresql'`
41
+ );
42
+ }
43
+ }
44
+
45
+ module.exports = {
46
+ createIntegrationMappingRepository,
47
+ // Export adapters for direct testing
48
+ IntegrationMappingRepositoryMongo,
49
+ IntegrationMappingRepositoryPostgres,
50
+ };
@@ -0,0 +1,106 @@
1
+ /**
2
+ * Integration Mapping Repository Interface
3
+ * Abstract base class defining the contract for integration mapping persistence adapters
4
+ *
5
+ * This follows the Port in Hexagonal Architecture:
6
+ * - Domain layer depends on this abstraction
7
+ * - Concrete adapters implement this interface
8
+ * - Use cases receive repositories via dependency injection
9
+ *
10
+ * Note: Currently, IntegrationMapping model has identical structure across MongoDB and PostgreSQL,
11
+ * so IntegrationMappingRepository serves both. This interface exists for consistency and
12
+ * future-proofing if database-specific implementations become needed.
13
+ *
14
+ * @abstract
15
+ */
16
+ class IntegrationMappingRepositoryInterface {
17
+ /**
18
+ * Find mapping by integration ID and source ID
19
+ *
20
+ * @param {string|number} integrationId - The integration ID
21
+ * @param {string} sourceId - The source ID for lookup
22
+ * @returns {Promise<Object|null>} The mapping object or null
23
+ * @abstract
24
+ */
25
+ async findMappingBy(integrationId, sourceId) {
26
+ throw new Error('Method findMappingBy must be implemented by subclass');
27
+ }
28
+
29
+ /**
30
+ * Create or update a mapping
31
+ *
32
+ * @param {string|number} integrationId - The integration ID
33
+ * @param {string} sourceId - The source ID for lookup
34
+ * @param {Object} mapping - The mapping data
35
+ * @returns {Promise<Object>} The created or updated mapping document
36
+ * @abstract
37
+ */
38
+ async upsertMapping(integrationId, sourceId, mapping) {
39
+ throw new Error('Method upsertMapping must be implemented by subclass');
40
+ }
41
+
42
+ /**
43
+ * Find all mappings for an integration
44
+ *
45
+ * @param {string|number} integrationId - The integration ID
46
+ * @returns {Promise<Array>} Array of mapping objects
47
+ * @abstract
48
+ */
49
+ async findMappingsByIntegration(integrationId) {
50
+ throw new Error(
51
+ 'Method findMappingsByIntegration must be implemented by subclass'
52
+ );
53
+ }
54
+
55
+ /**
56
+ * Delete a specific mapping
57
+ *
58
+ * @param {string|number} integrationId - The integration ID
59
+ * @param {string} sourceId - The source ID
60
+ * @returns {Promise<Object>} Deletion result
61
+ * @abstract
62
+ */
63
+ async deleteMapping(integrationId, sourceId) {
64
+ throw new Error('Method deleteMapping must be implemented by subclass');
65
+ }
66
+
67
+ /**
68
+ * Delete all mappings for an integration
69
+ *
70
+ * @param {string|number} integrationId - The integration ID
71
+ * @returns {Promise<Object>} Deletion result
72
+ * @abstract
73
+ */
74
+ async deleteMappingsByIntegration(integrationId) {
75
+ throw new Error(
76
+ 'Method deleteMappingsByIntegration must be implemented by subclass'
77
+ );
78
+ }
79
+
80
+ /**
81
+ * Find mapping by ID
82
+ *
83
+ * @param {string|number} id - The mapping ID
84
+ * @returns {Promise<Object|null>} The mapping object or null
85
+ * @abstract
86
+ */
87
+ async findMappingById(id) {
88
+ throw new Error(
89
+ 'Method findMappingById must be implemented by subclass'
90
+ );
91
+ }
92
+
93
+ /**
94
+ * Update a mapping by ID
95
+ *
96
+ * @param {string|number} id - The mapping ID
97
+ * @param {Object} updates - Fields to update
98
+ * @returns {Promise<Object>} Updated mapping object
99
+ * @abstract
100
+ */
101
+ async updateMapping(id, updates) {
102
+ throw new Error('Method updateMapping must be implemented by subclass');
103
+ }
104
+ }
105
+
106
+ module.exports = { IntegrationMappingRepositoryInterface };
@@ -0,0 +1,161 @@
1
+ const { prisma } = require('../../database/prisma');
2
+ const {
3
+ IntegrationMappingRepositoryInterface,
4
+ } = require('./integration-mapping-repository-interface');
5
+
6
+ /**
7
+ * MongoDB Integration Mapping Repository Adapter
8
+ * Handles persistence of integration mappings used for data transformation
9
+ *
10
+ * MongoDB-specific characteristics:
11
+ * - Uses String IDs (ObjectId)
12
+ * - No ID conversion needed (IDs are already strings)
13
+ * - mapping data stored in JSON field
14
+ */
15
+ class IntegrationMappingRepositoryMongo extends IntegrationMappingRepositoryInterface {
16
+ constructor() {
17
+ super();
18
+ this.prisma = prisma;
19
+ }
20
+
21
+ /**
22
+ * Convert any value to string (handles null/undefined)
23
+ * @private
24
+ * @param {*} value - Value to convert
25
+ * @returns {string|null|undefined} String value or null/undefined
26
+ */
27
+ _toString(value) {
28
+ if (value === null || value === undefined) return value;
29
+ return String(value);
30
+ }
31
+
32
+ /**
33
+ * Find mapping by integration ID and source ID
34
+ * Replaces: IntegrationMapping.findBy(integrationId, sourceId)
35
+ *
36
+ * @param {string} integrationId - The integration ID
37
+ * @param {string} sourceId - The source ID for lookup
38
+ * @returns {Promise<Object|null>} The mapping object with string IDs or null
39
+ */
40
+ async findMappingBy(integrationId, sourceId) {
41
+ return await this.prisma.integrationMapping.findFirst({
42
+ where: {
43
+ integrationId,
44
+ sourceId: this._toString(sourceId),
45
+ },
46
+ });
47
+ }
48
+
49
+ /**
50
+ * Create or update a mapping
51
+ * Replaces: IntegrationMapping.upsert(integrationId, sourceId, mapping)
52
+ *
53
+ * @param {string} integrationId - The integration ID
54
+ * @param {string} sourceId - The source ID for lookup
55
+ * @param {Object} mapping - The mapping data
56
+ * @returns {Promise<Object>} The created or updated mapping document with string IDs
57
+ */
58
+ async upsertMapping(integrationId, sourceId, mapping) {
59
+ return await this.prisma.integrationMapping.upsert({
60
+ where: {
61
+ integrationId_sourceId: {
62
+ integrationId,
63
+ sourceId: this._toString(sourceId),
64
+ },
65
+ },
66
+ update: {
67
+ mapping,
68
+ },
69
+ create: {
70
+ integrationId,
71
+ sourceId: this._toString(sourceId),
72
+ mapping,
73
+ },
74
+ });
75
+ }
76
+
77
+ /**
78
+ * Find all mappings for an integration
79
+ * Replaces: IntegrationMapping.find({ integration: integrationId })
80
+ *
81
+ * @param {string} integrationId - The integration ID
82
+ * @returns {Promise<Array>} Array of mapping documents with string IDs
83
+ */
84
+ async findMappingsByIntegration(integrationId) {
85
+ return await this.prisma.integrationMapping.findMany({
86
+ where: { integrationId },
87
+ });
88
+ }
89
+
90
+ /**
91
+ * Delete a mapping by integration and source ID
92
+ * Replaces: IntegrationMapping.deleteOne({ integration, sourceId })
93
+ *
94
+ * @param {string} integrationId - The integration ID
95
+ * @param {string} sourceId - The source ID
96
+ * @returns {Promise<Object>} The deletion result
97
+ */
98
+ async deleteMapping(integrationId, sourceId) {
99
+ try {
100
+ await this.prisma.integrationMapping.delete({
101
+ where: {
102
+ integrationId_sourceId: {
103
+ integrationId,
104
+ sourceId: this._toString(sourceId),
105
+ },
106
+ },
107
+ });
108
+ return { acknowledged: true, deletedCount: 1 };
109
+ } catch (error) {
110
+ if (error.code === 'P2025') {
111
+ // Record not found
112
+ return { acknowledged: true, deletedCount: 0 };
113
+ }
114
+ throw error;
115
+ }
116
+ }
117
+
118
+ /**
119
+ * Delete all mappings for an integration
120
+ * Replaces: IntegrationMapping.deleteMany({ integration: integrationId })
121
+ *
122
+ * @param {string} integrationId - The integration ID
123
+ * @returns {Promise<Object>} The deletion result
124
+ */
125
+ async deleteMappingsByIntegration(integrationId) {
126
+ const result = await this.prisma.integrationMapping.deleteMany({
127
+ where: { integrationId },
128
+ });
129
+
130
+ return {
131
+ acknowledged: true,
132
+ deletedCount: result.count,
133
+ };
134
+ }
135
+
136
+ /**
137
+ * Find mapping by ID
138
+ * @param {string} id - Mapping ID
139
+ * @returns {Promise<Object|null>} Mapping object with string IDs or null
140
+ */
141
+ async findMappingById(id) {
142
+ return await this.prisma.integrationMapping.findUnique({
143
+ where: { id },
144
+ });
145
+ }
146
+
147
+ /**
148
+ * Update mapping by ID
149
+ * @param {string} id - Mapping ID
150
+ * @param {Object} updates - Fields to update
151
+ * @returns {Promise<Object>} Updated mapping object with string IDs
152
+ */
153
+ async updateMapping(id, updates) {
154
+ return await this.prisma.integrationMapping.update({
155
+ where: { id },
156
+ data: updates,
157
+ });
158
+ }
159
+ }
160
+
161
+ module.exports = { IntegrationMappingRepositoryMongo };
@@ -0,0 +1,227 @@
1
+ const { prisma } = require('../../database/prisma');
2
+ const {
3
+ IntegrationMappingRepositoryInterface,
4
+ } = require('./integration-mapping-repository-interface');
5
+
6
+ /**
7
+ * PostgreSQL Integration Mapping Repository Adapter
8
+ * Handles persistence of integration mappings used for data transformation
9
+ *
10
+ * PostgreSQL-specific characteristics:
11
+ * - Uses Int IDs with autoincrement
12
+ * - Requires ID conversion: String (app layer) ↔ Int (database)
13
+ * - All returned IDs are converted to strings for application layer consistency
14
+ */
15
+ class IntegrationMappingRepositoryPostgres extends IntegrationMappingRepositoryInterface {
16
+ constructor() {
17
+ super();
18
+ this.prisma = prisma;
19
+ }
20
+
21
+ /**
22
+ * Convert string ID to integer for PostgreSQL queries
23
+ * @private
24
+ * @param {string|number|null|undefined} id - ID to convert
25
+ * @returns {number|null|undefined} Integer ID or null/undefined
26
+ * @throws {Error} If ID cannot be converted to integer
27
+ */
28
+ _stringToInt(id) {
29
+ if (id === null || id === undefined) return id;
30
+ const parsed = parseInt(id, 10);
31
+ if (isNaN(parsed)) {
32
+ throw new Error(`Invalid ID: ${id} cannot be converted to integer`);
33
+ }
34
+ return parsed;
35
+ }
36
+
37
+ /**
38
+ * Convert any value to string (handles null/undefined)
39
+ * @private
40
+ * @param {*} value - Value to convert
41
+ * @returns {string|null|undefined} String value or null/undefined
42
+ */
43
+ _toString(value) {
44
+ if (value === null || value === undefined) return value;
45
+ return String(value);
46
+ }
47
+
48
+ /**
49
+ * Convert integer to string for application layer
50
+ * @private
51
+ * @param {number|null|undefined} id - Integer ID from database
52
+ * @returns {string|null|undefined} String ID or null/undefined
53
+ */
54
+ _intToString(id) {
55
+ if (id === null || id === undefined) return id;
56
+ return id.toString();
57
+ }
58
+
59
+ /**
60
+ * Legacy alias for _stringToInt (for backward compatibility)
61
+ * @private
62
+ */
63
+ _convertId(id) {
64
+ return this._stringToInt(id);
65
+ }
66
+
67
+ /**
68
+ * Convert mapping object IDs to strings
69
+ * @private
70
+ * @param {Object|null} mapping - Mapping object from database
71
+ * @returns {Object|null} Mapping with string IDs
72
+ */
73
+ _convertMappingIds(mapping) {
74
+ if (!mapping) return mapping;
75
+ return {
76
+ ...mapping,
77
+ id: this._intToString(mapping.id),
78
+ integrationId: this._intToString(mapping.integrationId),
79
+ };
80
+ }
81
+
82
+ /**
83
+ * Find mapping by integration ID and source ID
84
+ * Replaces: IntegrationMapping.findBy(integrationId, sourceId)
85
+ *
86
+ * @param {string} integrationId - The integration ID (string from application layer)
87
+ * @param {string} sourceId - The source ID for lookup
88
+ * @returns {Promise<Object|null>} The mapping object with string IDs or null
89
+ */
90
+ async findMappingBy(integrationId, sourceId) {
91
+ const mapping = await this.prisma.integrationMapping.findFirst({
92
+ where: {
93
+ integrationId: this._stringToInt(integrationId),
94
+ sourceId: this._toString(sourceId),
95
+ },
96
+ });
97
+ return this._convertMappingIds(mapping);
98
+ }
99
+
100
+ /**
101
+ * Create or update a mapping
102
+ * Replaces: IntegrationMapping.upsert(integrationId, sourceId, mapping)
103
+ *
104
+ * @param {string} integrationId - The integration ID (string from application layer)
105
+ * @param {string} sourceId - The source ID for lookup
106
+ * @param {Object} mapping - The mapping data
107
+ * @returns {Promise<Object>} The created or updated mapping document with string IDs
108
+ */
109
+ async upsertMapping(integrationId, sourceId, mapping) {
110
+ const result = await this.prisma.integrationMapping.upsert({
111
+ where: {
112
+ integrationId_sourceId: {
113
+ integrationId: this._stringToInt(integrationId),
114
+ sourceId: this._toString(sourceId),
115
+ },
116
+ },
117
+ update: {
118
+ mapping,
119
+ },
120
+ create: {
121
+ integrationId: this._stringToInt(integrationId),
122
+ sourceId: this._toString(sourceId),
123
+ mapping,
124
+ },
125
+ });
126
+ return this._convertMappingIds(result);
127
+ }
128
+
129
+ /**
130
+ * Find all mappings for an integration
131
+ * Replaces: IntegrationMapping.find({ integration: integrationId })
132
+ *
133
+ * @param {string} integrationId - The integration ID (string from application layer)
134
+ * @returns {Promise<Array>} Array of mapping documents with string IDs
135
+ */
136
+ async findMappingsByIntegration(integrationId) {
137
+ const intIntegrationId = this._convertId(integrationId);
138
+ const mappings = await this.prisma.integrationMapping.findMany({
139
+ where: { integrationId: intIntegrationId },
140
+ });
141
+ return mappings.map((m) => this._convertMappingIds(m));
142
+ }
143
+
144
+ /**
145
+ * Delete a mapping by integration and source ID
146
+ * Replaces: IntegrationMapping.deleteOne({ integration, sourceId })
147
+ *
148
+ * @param {string} integrationId - The integration ID (string from application layer)
149
+ * @param {string} sourceId - The source ID
150
+ * @returns {Promise<Object>} The deletion result
151
+ */
152
+ async deleteMapping(integrationId, sourceId) {
153
+ try {
154
+ await this.prisma.integrationMapping.delete({
155
+ where: {
156
+ integrationId_sourceId: {
157
+ integrationId: this._stringToInt(integrationId),
158
+ sourceId: this._toString(sourceId),
159
+ },
160
+ },
161
+ });
162
+ return { acknowledged: true, deletedCount: 1 };
163
+ } catch (error) {
164
+ if (error.code === 'P2025') {
165
+ // Record not found
166
+ return { acknowledged: true, deletedCount: 0 };
167
+ }
168
+ throw error;
169
+ }
170
+ }
171
+
172
+ /**
173
+ * Delete all mappings for an integration
174
+ * Replaces: IntegrationMapping.deleteMany({ integration: integrationId })
175
+ *
176
+ * @param {string} integrationId - The integration ID (string from application layer)
177
+ * @returns {Promise<Object>} The deletion result
178
+ */
179
+ async deleteMappingsByIntegration(integrationId) {
180
+ const intIntegrationId = this._convertId(integrationId);
181
+ const result = await this.prisma.integrationMapping.deleteMany({
182
+ where: { integrationId: intIntegrationId },
183
+ });
184
+
185
+ return {
186
+ acknowledged: true,
187
+ deletedCount: result.count,
188
+ };
189
+ }
190
+
191
+ /**
192
+ * Find mapping by ID
193
+ * @param {string} id - Mapping ID (string from application layer)
194
+ * @returns {Promise<Object|null>} Mapping object with string IDs or null
195
+ */
196
+ async findMappingById(id) {
197
+ const intId = this._convertId(id);
198
+ const mapping = await this.prisma.integrationMapping.findUnique({
199
+ where: { id: intId },
200
+ });
201
+ return this._convertMappingIds(mapping);
202
+ }
203
+
204
+ /**
205
+ * Update mapping by ID
206
+ * @param {string} id - Mapping ID (string from application layer)
207
+ * @param {Object} updates - Fields to update (with string IDs from application layer)
208
+ * @returns {Promise<Object>} Updated mapping object with string IDs
209
+ */
210
+ async updateMapping(id, updates) {
211
+ const intId = this._convertId(id);
212
+
213
+ // Convert integrationId if present in updates
214
+ const data = { ...updates };
215
+ if (data.integrationId !== undefined) {
216
+ data.integrationId = this._convertId(data.integrationId);
217
+ }
218
+
219
+ const mapping = await this.prisma.integrationMapping.update({
220
+ where: { id: intId },
221
+ data,
222
+ });
223
+ return this._convertMappingIds(mapping);
224
+ }
225
+ }
226
+
227
+ module.exports = { IntegrationMappingRepositoryPostgres };
@@ -0,0 +1,156 @@
1
+ const { prisma } = require('../../database/prisma');
2
+ const {
3
+ IntegrationMappingRepositoryInterface,
4
+ } = require('./integration-mapping-repository-interface');
5
+
6
+ /**
7
+ * Prisma-based Integration Mapping Repository
8
+ * Handles persistence of integration mappings used for data transformation
9
+ *
10
+ * Works identically for both MongoDB and PostgreSQL:
11
+ * - MongoDB: String IDs with @db.ObjectId
12
+ * - PostgreSQL: Integer IDs with auto-increment
13
+ * - Both use same query patterns (no many-to-many differences)
14
+ *
15
+ * Migration from Mongoose:
16
+ * - Constructor injection of Prisma client
17
+ * - IntegrationMapping.findBy() → findFirst with where clause
18
+ * - IntegrationMapping.upsert() → Prisma upsert with unique constraint
19
+ * - mapping data stored in JSON field
20
+ */
21
+ class IntegrationMappingRepository extends IntegrationMappingRepositoryInterface {
22
+ constructor(prismaClient = prisma) {
23
+ super();
24
+ this.prisma = prismaClient; // Allow injection for testing
25
+ }
26
+
27
+ /**
28
+ * Find mapping by integration ID and source ID
29
+ * Replaces: IntegrationMapping.findBy(integrationId, sourceId)
30
+ *
31
+ * @param {string} integrationId - The integration ID
32
+ * @param {string} sourceId - The source ID for lookup
33
+ * @returns {Promise<Object|null>} The mapping object or null
34
+ */
35
+ async findMappingBy(integrationId, sourceId) {
36
+ return await this.prisma.integrationMapping.findFirst({
37
+ where: {
38
+ integrationId,
39
+ sourceId,
40
+ },
41
+ });
42
+ }
43
+
44
+ /**
45
+ * Create or update a mapping
46
+ * Replaces: IntegrationMapping.upsert(integrationId, sourceId, mapping)
47
+ *
48
+ * @param {string} integrationId - The integration ID
49
+ * @param {string} sourceId - The source ID for lookup
50
+ * @param {Object} mapping - The mapping data
51
+ * @returns {Promise<Object>} The created or updated mapping document
52
+ */
53
+ async upsertMapping(integrationId, sourceId, mapping) {
54
+ return await this.prisma.integrationMapping.upsert({
55
+ where: {
56
+ integrationId_sourceId: {
57
+ integrationId,
58
+ sourceId,
59
+ },
60
+ },
61
+ update: {
62
+ mapping,
63
+ },
64
+ create: {
65
+ integrationId,
66
+ sourceId,
67
+ mapping,
68
+ },
69
+ });
70
+ }
71
+
72
+ /**
73
+ * Find all mappings for an integration
74
+ * Replaces: IntegrationMapping.find({ integration: integrationId })
75
+ *
76
+ * @param {string} integrationId - The integration ID
77
+ * @returns {Promise<Array>} Array of mapping documents
78
+ */
79
+ async findMappingsByIntegration(integrationId) {
80
+ return await this.prisma.integrationMapping.findMany({
81
+ where: { integrationId },
82
+ });
83
+ }
84
+
85
+ /**
86
+ * Delete a mapping by integration and source ID
87
+ * Replaces: IntegrationMapping.deleteOne({ integration, sourceId })
88
+ *
89
+ * @param {string} integrationId - The integration ID
90
+ * @param {string} sourceId - The source ID
91
+ * @returns {Promise<Object>} The deletion result
92
+ */
93
+ async deleteMapping(integrationId, sourceId) {
94
+ try {
95
+ await this.prisma.integrationMapping.delete({
96
+ where: {
97
+ integrationId_sourceId: {
98
+ integrationId,
99
+ sourceId,
100
+ },
101
+ },
102
+ });
103
+ return { acknowledged: true, deletedCount: 1 };
104
+ } catch (error) {
105
+ if (error.code === 'P2025') {
106
+ // Record not found
107
+ return { acknowledged: true, deletedCount: 0 };
108
+ }
109
+ throw error;
110
+ }
111
+ }
112
+
113
+ /**
114
+ * Delete all mappings for an integration
115
+ * Replaces: IntegrationMapping.deleteMany({ integration: integrationId })
116
+ *
117
+ * @param {string} integrationId - The integration ID
118
+ * @returns {Promise<Object>} The deletion result
119
+ */
120
+ async deleteMappingsByIntegration(integrationId) {
121
+ const result = await this.prisma.integrationMapping.deleteMany({
122
+ where: { integrationId },
123
+ });
124
+
125
+ return {
126
+ acknowledged: true,
127
+ deletedCount: result.count,
128
+ };
129
+ }
130
+
131
+ /**
132
+ * Find mapping by ID
133
+ * @param {string} id - Mapping ID
134
+ * @returns {Promise<Object|null>} Mapping object or null
135
+ */
136
+ async findMappingById(id) {
137
+ return await this.prisma.integrationMapping.findUnique({
138
+ where: { id },
139
+ });
140
+ }
141
+
142
+ /**
143
+ * Update mapping by ID
144
+ * @param {string} id - Mapping ID
145
+ * @param {Object} updates - Fields to update
146
+ * @returns {Promise<Object>} Updated mapping object
147
+ */
148
+ async updateMapping(id, updates) {
149
+ return await this.prisma.integrationMapping.update({
150
+ where: { id },
151
+ data: updates,
152
+ });
153
+ }
154
+ }
155
+
156
+ module.exports = { IntegrationMappingRepository };