@friggframework/core 2.0.0-next.6 → 2.0.0-next.60

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 (285) 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/user-commands.js +283 -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 +14 -7
  12. package/credential/repositories/credential-repository-documentdb.js +304 -0
  13. package/credential/repositories/credential-repository-factory.js +54 -0
  14. package/credential/repositories/credential-repository-interface.js +98 -0
  15. package/credential/repositories/credential-repository-mongo.js +269 -0
  16. package/credential/repositories/credential-repository-postgres.js +291 -0
  17. package/credential/repositories/credential-repository.js +302 -0
  18. package/credential/use-cases/get-credential-for-user.js +25 -0
  19. package/credential/use-cases/update-authentication-status.js +15 -0
  20. package/database/MONGODB_TRANSACTION_FIX.md +198 -0
  21. package/database/adapters/lambda-invoker.js +97 -0
  22. package/database/config.js +154 -0
  23. package/database/documentdb-encryption-service.js +330 -0
  24. package/database/documentdb-utils.js +136 -0
  25. package/database/encryption/README.md +839 -0
  26. package/database/encryption/documentdb-encryption-service.md +3575 -0
  27. package/database/encryption/encryption-schema-registry.js +268 -0
  28. package/database/encryption/field-encryption-service.js +226 -0
  29. package/database/encryption/logger.js +79 -0
  30. package/database/encryption/prisma-encryption-extension.js +222 -0
  31. package/database/index.js +61 -21
  32. package/database/models/WebsocketConnection.js +16 -10
  33. package/database/models/readme.md +1 -0
  34. package/database/prisma.js +182 -0
  35. package/database/repositories/health-check-repository-documentdb.js +134 -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/health-check-repository.js +108 -0
  41. package/database/repositories/migration-status-repository-s3.js +137 -0
  42. package/database/use-cases/check-database-health-use-case.js +29 -0
  43. package/database/use-cases/check-database-state-use-case.js +81 -0
  44. package/database/use-cases/check-encryption-health-use-case.js +83 -0
  45. package/database/use-cases/get-database-state-via-worker-use-case.js +61 -0
  46. package/database/use-cases/get-migration-status-use-case.js +93 -0
  47. package/database/use-cases/run-database-migration-use-case.js +139 -0
  48. package/database/use-cases/test-encryption-use-case.js +253 -0
  49. package/database/use-cases/trigger-database-migration-use-case.js +157 -0
  50. package/database/utils/mongodb-collection-utils.js +91 -0
  51. package/database/utils/mongodb-schema-init.js +106 -0
  52. package/database/utils/prisma-runner.js +477 -0
  53. package/database/utils/prisma-schema-parser.js +182 -0
  54. package/docs/PROCESS_MANAGEMENT_QUEUE_SPEC.md +517 -0
  55. package/encrypt/Cryptor.js +34 -168
  56. package/encrypt/index.js +1 -2
  57. package/encrypt/test-encrypt.js +0 -2
  58. package/errors/client-safe-error.js +26 -0
  59. package/errors/fetch-error.js +2 -1
  60. package/errors/index.js +2 -0
  61. package/generated/prisma-mongodb/client.d.ts +1 -0
  62. package/generated/prisma-mongodb/client.js +4 -0
  63. package/generated/prisma-mongodb/default.d.ts +1 -0
  64. package/generated/prisma-mongodb/default.js +4 -0
  65. package/generated/prisma-mongodb/edge.d.ts +1 -0
  66. package/generated/prisma-mongodb/edge.js +334 -0
  67. package/generated/prisma-mongodb/index-browser.js +316 -0
  68. package/generated/prisma-mongodb/index.d.ts +22903 -0
  69. package/generated/prisma-mongodb/index.js +359 -0
  70. package/generated/prisma-mongodb/package.json +183 -0
  71. package/generated/prisma-mongodb/query-engine-debian-openssl-3.0.x +0 -0
  72. package/generated/prisma-mongodb/query-engine-rhel-openssl-3.0.x +0 -0
  73. package/generated/prisma-mongodb/runtime/binary.d.ts +1 -0
  74. package/generated/prisma-mongodb/runtime/binary.js +289 -0
  75. package/generated/prisma-mongodb/runtime/edge-esm.js +34 -0
  76. package/generated/prisma-mongodb/runtime/edge.js +34 -0
  77. package/generated/prisma-mongodb/runtime/index-browser.d.ts +370 -0
  78. package/generated/prisma-mongodb/runtime/index-browser.js +16 -0
  79. package/generated/prisma-mongodb/runtime/library.d.ts +3977 -0
  80. package/generated/prisma-mongodb/runtime/react-native.js +83 -0
  81. package/generated/prisma-mongodb/runtime/wasm-compiler-edge.js +84 -0
  82. package/generated/prisma-mongodb/runtime/wasm-engine-edge.js +36 -0
  83. package/generated/prisma-mongodb/schema.prisma +360 -0
  84. package/generated/prisma-mongodb/wasm-edge-light-loader.mjs +4 -0
  85. package/generated/prisma-mongodb/wasm-worker-loader.mjs +4 -0
  86. package/generated/prisma-mongodb/wasm.d.ts +1 -0
  87. package/generated/prisma-mongodb/wasm.js +341 -0
  88. package/generated/prisma-postgresql/client.d.ts +1 -0
  89. package/generated/prisma-postgresql/client.js +4 -0
  90. package/generated/prisma-postgresql/default.d.ts +1 -0
  91. package/generated/prisma-postgresql/default.js +4 -0
  92. package/generated/prisma-postgresql/edge.d.ts +1 -0
  93. package/generated/prisma-postgresql/edge.js +356 -0
  94. package/generated/prisma-postgresql/index-browser.js +338 -0
  95. package/generated/prisma-postgresql/index.d.ts +25077 -0
  96. package/generated/prisma-postgresql/index.js +381 -0
  97. package/generated/prisma-postgresql/package.json +183 -0
  98. package/generated/prisma-postgresql/query-engine-debian-openssl-3.0.x +0 -0
  99. package/generated/prisma-postgresql/query-engine-rhel-openssl-3.0.x +0 -0
  100. package/generated/prisma-postgresql/query_engine_bg.js +2 -0
  101. package/generated/prisma-postgresql/query_engine_bg.wasm +0 -0
  102. package/generated/prisma-postgresql/runtime/binary.d.ts +1 -0
  103. package/generated/prisma-postgresql/runtime/binary.js +289 -0
  104. package/generated/prisma-postgresql/runtime/edge-esm.js +34 -0
  105. package/generated/prisma-postgresql/runtime/edge.js +34 -0
  106. package/generated/prisma-postgresql/runtime/index-browser.d.ts +370 -0
  107. package/generated/prisma-postgresql/runtime/index-browser.js +16 -0
  108. package/generated/prisma-postgresql/runtime/library.d.ts +3977 -0
  109. package/generated/prisma-postgresql/runtime/react-native.js +83 -0
  110. package/generated/prisma-postgresql/runtime/wasm-compiler-edge.js +84 -0
  111. package/generated/prisma-postgresql/runtime/wasm-engine-edge.js +36 -0
  112. package/generated/prisma-postgresql/schema.prisma +343 -0
  113. package/generated/prisma-postgresql/wasm-edge-light-loader.mjs +4 -0
  114. package/generated/prisma-postgresql/wasm-worker-loader.mjs +4 -0
  115. package/generated/prisma-postgresql/wasm.d.ts +1 -0
  116. package/generated/prisma-postgresql/wasm.js +363 -0
  117. package/handlers/WEBHOOKS.md +653 -0
  118. package/handlers/app-definition-loader.js +38 -0
  119. package/handlers/app-handler-helpers.js +56 -0
  120. package/handlers/backend-utils.js +186 -0
  121. package/handlers/database-migration-handler.js +227 -0
  122. package/handlers/integration-event-dispatcher.js +54 -0
  123. package/handlers/routers/HEALTHCHECK.md +342 -0
  124. package/handlers/routers/auth.js +15 -0
  125. package/handlers/routers/db-migration.handler.js +29 -0
  126. package/handlers/routers/db-migration.js +326 -0
  127. package/handlers/routers/health.js +516 -0
  128. package/handlers/routers/integration-defined-routers.js +45 -0
  129. package/handlers/routers/integration-webhook-routers.js +67 -0
  130. package/handlers/routers/user.js +63 -0
  131. package/handlers/routers/websocket.js +57 -0
  132. package/handlers/use-cases/check-external-apis-health-use-case.js +81 -0
  133. package/handlers/use-cases/check-integrations-health-use-case.js +44 -0
  134. package/handlers/workers/db-migration.js +352 -0
  135. package/handlers/workers/integration-defined-workers.js +27 -0
  136. package/index.js +77 -22
  137. package/integrations/WEBHOOK-QUICKSTART.md +151 -0
  138. package/integrations/index.js +12 -10
  139. package/integrations/integration-base.js +326 -55
  140. package/integrations/integration-router.js +374 -179
  141. package/integrations/options.js +1 -1
  142. package/integrations/repositories/integration-mapping-repository-documentdb.js +280 -0
  143. package/integrations/repositories/integration-mapping-repository-factory.js +57 -0
  144. package/integrations/repositories/integration-mapping-repository-interface.js +106 -0
  145. package/integrations/repositories/integration-mapping-repository-mongo.js +161 -0
  146. package/integrations/repositories/integration-mapping-repository-postgres.js +227 -0
  147. package/integrations/repositories/integration-mapping-repository.js +156 -0
  148. package/integrations/repositories/integration-repository-documentdb.js +210 -0
  149. package/integrations/repositories/integration-repository-factory.js +51 -0
  150. package/integrations/repositories/integration-repository-interface.js +127 -0
  151. package/integrations/repositories/integration-repository-mongo.js +303 -0
  152. package/integrations/repositories/integration-repository-postgres.js +352 -0
  153. package/integrations/repositories/process-repository-documentdb.js +243 -0
  154. package/integrations/repositories/process-repository-factory.js +53 -0
  155. package/integrations/repositories/process-repository-interface.js +90 -0
  156. package/integrations/repositories/process-repository-mongo.js +190 -0
  157. package/integrations/repositories/process-repository-postgres.js +217 -0
  158. package/integrations/tests/doubles/dummy-integration-class.js +83 -0
  159. package/integrations/tests/doubles/test-integration-repository.js +99 -0
  160. package/integrations/use-cases/create-integration.js +83 -0
  161. package/integrations/use-cases/create-process.js +128 -0
  162. package/integrations/use-cases/delete-integration-for-user.js +101 -0
  163. package/integrations/use-cases/find-integration-context-by-external-entity-id.js +72 -0
  164. package/integrations/use-cases/get-integration-for-user.js +78 -0
  165. package/integrations/use-cases/get-integration-instance-by-definition.js +67 -0
  166. package/integrations/use-cases/get-integration-instance.js +83 -0
  167. package/integrations/use-cases/get-integrations-for-user.js +88 -0
  168. package/integrations/use-cases/get-possible-integrations.js +27 -0
  169. package/integrations/use-cases/get-process.js +87 -0
  170. package/integrations/use-cases/index.js +19 -0
  171. package/integrations/use-cases/load-integration-context.js +71 -0
  172. package/integrations/use-cases/update-integration-messages.js +44 -0
  173. package/integrations/use-cases/update-integration-status.js +32 -0
  174. package/integrations/use-cases/update-integration.js +93 -0
  175. package/integrations/use-cases/update-process-metrics.js +201 -0
  176. package/integrations/use-cases/update-process-state.js +119 -0
  177. package/integrations/utils/map-integration-dto.js +37 -0
  178. package/jest-global-setup-noop.js +3 -0
  179. package/jest-global-teardown-noop.js +3 -0
  180. package/logs/logger.js +0 -4
  181. package/{module-plugin → modules}/entity.js +1 -1
  182. package/{module-plugin → modules}/index.js +0 -8
  183. package/modules/module-factory.js +56 -0
  184. package/modules/module.js +221 -0
  185. package/modules/repositories/module-repository-documentdb.js +307 -0
  186. package/modules/repositories/module-repository-factory.js +40 -0
  187. package/modules/repositories/module-repository-interface.js +129 -0
  188. package/modules/repositories/module-repository-mongo.js +377 -0
  189. package/modules/repositories/module-repository-postgres.js +426 -0
  190. package/modules/repositories/module-repository.js +316 -0
  191. package/modules/requester/api-key.js +52 -0
  192. package/{module-plugin → modules}/requester/requester.js +1 -0
  193. package/{module-plugin → modules}/test/mock-api/api.js +8 -3
  194. package/{module-plugin → modules}/test/mock-api/definition.js +12 -8
  195. package/modules/tests/doubles/test-module-factory.js +16 -0
  196. package/modules/tests/doubles/test-module-repository.js +39 -0
  197. package/modules/use-cases/get-entities-for-user.js +32 -0
  198. package/modules/use-cases/get-entity-options-by-id.js +71 -0
  199. package/modules/use-cases/get-entity-options-by-type.js +34 -0
  200. package/modules/use-cases/get-module-instance-from-type.js +31 -0
  201. package/modules/use-cases/get-module.js +74 -0
  202. package/modules/use-cases/process-authorization-callback.js +133 -0
  203. package/modules/use-cases/refresh-entity-options.js +72 -0
  204. package/modules/use-cases/test-module-auth.js +72 -0
  205. package/modules/utils/map-module-dto.js +18 -0
  206. package/package.json +82 -50
  207. package/prisma-mongodb/schema.prisma +360 -0
  208. package/prisma-postgresql/migrations/20250930193005_init/migration.sql +315 -0
  209. package/prisma-postgresql/migrations/20251006135218_init/migration.sql +9 -0
  210. package/prisma-postgresql/migrations/20251010000000_remove_unused_entity_reference_map/migration.sql +3 -0
  211. package/prisma-postgresql/migrations/20251112195422_update_user_unique_constraints/migration.sql +25 -0
  212. package/prisma-postgresql/migrations/migration_lock.toml +3 -0
  213. package/prisma-postgresql/schema.prisma +343 -0
  214. package/queues/queuer-util.js +27 -22
  215. package/syncs/manager.js +468 -443
  216. package/syncs/repositories/sync-repository-documentdb.js +240 -0
  217. package/syncs/repositories/sync-repository-factory.js +43 -0
  218. package/syncs/repositories/sync-repository-interface.js +109 -0
  219. package/syncs/repositories/sync-repository-mongo.js +239 -0
  220. package/syncs/repositories/sync-repository-postgres.js +319 -0
  221. package/syncs/sync.js +0 -1
  222. package/token/repositories/token-repository-documentdb.js +137 -0
  223. package/token/repositories/token-repository-factory.js +40 -0
  224. package/token/repositories/token-repository-interface.js +131 -0
  225. package/token/repositories/token-repository-mongo.js +219 -0
  226. package/token/repositories/token-repository-postgres.js +264 -0
  227. package/token/repositories/token-repository.js +219 -0
  228. package/types/core/index.d.ts +2 -2
  229. package/types/integrations/index.d.ts +2 -6
  230. package/types/module-plugin/index.d.ts +5 -59
  231. package/types/syncs/index.d.ts +0 -2
  232. package/user/repositories/user-repository-documentdb.js +441 -0
  233. package/user/repositories/user-repository-factory.js +52 -0
  234. package/user/repositories/user-repository-interface.js +201 -0
  235. package/user/repositories/user-repository-mongo.js +308 -0
  236. package/user/repositories/user-repository-postgres.js +360 -0
  237. package/user/tests/doubles/test-user-repository.js +72 -0
  238. package/user/use-cases/authenticate-user.js +127 -0
  239. package/user/use-cases/authenticate-with-shared-secret.js +48 -0
  240. package/user/use-cases/create-individual-user.js +61 -0
  241. package/user/use-cases/create-organization-user.js +47 -0
  242. package/user/use-cases/create-token-for-user-id.js +30 -0
  243. package/user/use-cases/get-user-from-adopter-jwt.js +149 -0
  244. package/user/use-cases/get-user-from-bearer-token.js +77 -0
  245. package/user/use-cases/get-user-from-x-frigg-headers.js +132 -0
  246. package/user/use-cases/login-user.js +122 -0
  247. package/user/user.js +125 -0
  248. package/utils/backend-path.js +38 -0
  249. package/utils/index.js +6 -0
  250. package/websocket/repositories/websocket-connection-repository-documentdb.js +119 -0
  251. package/websocket/repositories/websocket-connection-repository-factory.js +44 -0
  252. package/websocket/repositories/websocket-connection-repository-interface.js +106 -0
  253. package/websocket/repositories/websocket-connection-repository-mongo.js +156 -0
  254. package/websocket/repositories/websocket-connection-repository-postgres.js +196 -0
  255. package/websocket/repositories/websocket-connection-repository.js +161 -0
  256. package/database/models/State.js +0 -9
  257. package/database/models/Token.js +0 -70
  258. package/database/mongo.js +0 -45
  259. package/encrypt/Cryptor.test.js +0 -32
  260. package/encrypt/encrypt.js +0 -132
  261. package/encrypt/encrypt.test.js +0 -1069
  262. package/errors/base-error.test.js +0 -32
  263. package/errors/fetch-error.test.js +0 -79
  264. package/errors/halt-error.test.js +0 -11
  265. package/errors/validation-errors.test.js +0 -120
  266. package/integrations/create-frigg-backend.js +0 -31
  267. package/integrations/integration-factory.js +0 -251
  268. package/integrations/integration-mapping.js +0 -43
  269. package/integrations/integration-model.js +0 -46
  270. package/integrations/integration-user.js +0 -144
  271. package/integrations/test/integration-base.test.js +0 -144
  272. package/lambda/TimeoutCatcher.test.js +0 -68
  273. package/logs/logger.test.js +0 -76
  274. package/module-plugin/auther.js +0 -393
  275. package/module-plugin/credential.js +0 -22
  276. package/module-plugin/entity-manager.js +0 -70
  277. package/module-plugin/manager.js +0 -169
  278. package/module-plugin/module-factory.js +0 -61
  279. package/module-plugin/requester/api-key.js +0 -36
  280. package/module-plugin/requester/requester.test.js +0 -28
  281. package/module-plugin/test/auther.test.js +0 -97
  282. /package/{module-plugin → modules}/ModuleConstants.js +0 -0
  283. /package/{module-plugin → modules}/requester/basic.js +0 -0
  284. /package/{module-plugin → modules}/requester/oauth-2.js +0 -0
  285. /package/{module-plugin → modules}/test/mock-api/mocks/hubspot.js +0 -0
@@ -0,0 +1,48 @@
1
+ const { HealthCheckRepositoryMongoDB } = require('./health-check-repository-mongodb');
2
+ const { HealthCheckRepositoryPostgreSQL } = require('./health-check-repository-postgres');
3
+ const { HealthCheckRepositoryDocumentDB } = require('./health-check-repository-documentdb');
4
+ const config = require('../config');
5
+
6
+ /**
7
+ * Factory function to create a health check repository for the configured database type.
8
+ * Requires explicit prismaClient injection to support IoC container patterns.
9
+ *
10
+ * @param {Object} options
11
+ * @param {Object} options.prismaClient - Prisma client instance (required for dependency injection)
12
+ * @returns {HealthCheckRepositoryInterface} Database-specific health check repository
13
+ * @throws {Error} If prismaClient is not provided
14
+ *
15
+ * @example
16
+ * const { prisma } = require('../prisma');
17
+ * const repository = createHealthCheckRepository({ prismaClient: prisma });
18
+ */
19
+ function createHealthCheckRepository({ prismaClient } = {}) {
20
+ if (!prismaClient) {
21
+ throw new Error('prismaClient is required');
22
+ }
23
+
24
+ const dbType = config.DB_TYPE;
25
+
26
+ switch (dbType) {
27
+ case 'mongodb':
28
+ return new HealthCheckRepositoryMongoDB({ prismaClient });
29
+
30
+ case 'postgresql':
31
+ return new HealthCheckRepositoryPostgreSQL({ prismaClient });
32
+
33
+ case 'documentdb':
34
+ return new HealthCheckRepositoryDocumentDB({ prismaClient });
35
+
36
+ default:
37
+ throw new Error(
38
+ `Unsupported database type: ${dbType}. Supported values: 'mongodb', 'documentdb', 'postgresql'`
39
+ );
40
+ }
41
+ }
42
+
43
+ module.exports = {
44
+ createHealthCheckRepository,
45
+ HealthCheckRepositoryMongoDB,
46
+ HealthCheckRepositoryPostgreSQL,
47
+ HealthCheckRepositoryDocumentDB,
48
+ };
@@ -0,0 +1,82 @@
1
+ /**
2
+ * Health Check Repository Interface
3
+ * Abstract base class defining the contract for health check 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, HealthCheckRepository has identical structure across MongoDB and PostgreSQL,
11
+ * so HealthCheckRepository serves both. This interface exists for consistency and
12
+ * future-proofing if database-specific implementations become needed.
13
+ *
14
+ * @abstract
15
+ */
16
+ class HealthCheckRepositoryInterface {
17
+ /**
18
+ * @returns {Promise<{readyState: number, stateName: string, isConnected: boolean}>}
19
+ * @abstract
20
+ */
21
+ async getDatabaseConnectionState() {
22
+ throw new Error('Method getDatabaseConnectionState must be implemented by subclass');
23
+ }
24
+
25
+ /**
26
+ * Ping database to verify connectivity
27
+ *
28
+ * @param {number} maxTimeMS - Maximum time in milliseconds
29
+ * @returns {Promise<number>} Response time in milliseconds
30
+ * @abstract
31
+ */
32
+ async pingDatabase(maxTimeMS) {
33
+ throw new Error('Method pingDatabase must be implemented by subclass');
34
+ }
35
+
36
+ /**
37
+ * Persist an encrypted credential for health verification.
38
+ * Implementations should rely on Prisma so encryption middleware runs.
39
+ *
40
+ * @param {Object} credentialData
41
+ * @returns {Promise<Object>} Persisted credential
42
+ * @abstract
43
+ */
44
+ async createCredential(credentialData) {
45
+ throw new Error('Method createCredential must be implemented by subclass');
46
+ }
47
+
48
+ /**
49
+ * Retrieve credential by ID using Prisma (decrypted).
50
+ *
51
+ * @param {string} id
52
+ * @returns {Promise<Object|null>}
53
+ * @abstract
54
+ */
55
+ async findCredentialById(id) {
56
+ throw new Error('Method findCredentialById must be implemented by subclass');
57
+ }
58
+
59
+ /**
60
+ * Fetch raw credential document from the database (without decryption).
61
+ *
62
+ * @param {string} id
63
+ * @returns {Promise<Object|null>}
64
+ * @abstract
65
+ */
66
+ async getRawCredentialById(id) {
67
+ throw new Error('Method getRawCredentialById must be implemented by subclass');
68
+ }
69
+
70
+ /**
71
+ * Delete credential by ID.
72
+ *
73
+ * @param {string} id
74
+ * @returns {Promise<void>}
75
+ * @abstract
76
+ */
77
+ async deleteCredential(id) {
78
+ throw new Error('Method deleteCredential must be implemented by subclass');
79
+ }
80
+ }
81
+
82
+ module.exports = { HealthCheckRepositoryInterface };
@@ -0,0 +1,89 @@
1
+ const { mongoose } = require('../mongoose');
2
+ const {
3
+ HealthCheckRepositoryInterface,
4
+ } = require('./health-check-repository-interface');
5
+
6
+ class HealthCheckRepositoryMongoDB extends HealthCheckRepositoryInterface {
7
+ /**
8
+ * @param {Object} params
9
+ * @param {Object} params.prismaClient - Prisma client instance
10
+ */
11
+ constructor({ prismaClient }) {
12
+ super();
13
+ this.prisma = prismaClient;
14
+ }
15
+
16
+ /**
17
+ * @returns {Promise<{readyState: number, stateName: string, isConnected: boolean}>}
18
+ */
19
+ async getDatabaseConnectionState() {
20
+ let isConnected = false;
21
+ let stateName = 'unknown';
22
+
23
+ try {
24
+ await this.prisma.$runCommandRaw({ ping: 1 });
25
+ isConnected = true;
26
+ stateName = 'connected';
27
+ } catch (error) {
28
+ stateName = 'disconnected';
29
+ }
30
+
31
+ return {
32
+ readyState: isConnected ? 1 : 0,
33
+ readyState: isConnected ? 1 : 0,
34
+ stateName,
35
+ isConnected,
36
+ };
37
+ }
38
+
39
+ async pingDatabase(maxTimeMS = 2000) {
40
+ const pingStart = Date.now();
41
+
42
+ // Create a timeout promise that rejects after maxTimeMS
43
+ const timeoutPromise = new Promise((_, reject) =>
44
+ setTimeout(() => reject(new Error('Database ping timeout')), maxTimeMS)
45
+ );
46
+
47
+ // Race between the database ping and the timeout
48
+ await Promise.race([
49
+ prisma.$queryRaw`SELECT 1`.catch(() => {
50
+ // For MongoDB, use runCommandRaw instead
51
+ return prisma.$runCommandRaw({ ping: 1 });
52
+ }),
53
+ timeoutPromise
54
+ ]);
55
+
56
+ return Date.now() - pingStart;
57
+ }
58
+
59
+ async createCredential(credentialData) {
60
+ return await this.prisma.credential.create({
61
+ data: credentialData,
62
+ });
63
+ }
64
+
65
+ async findCredentialById(id) {
66
+ return await this.prisma.credential.findUnique({
67
+ where: { id },
68
+ });
69
+ }
70
+
71
+ /**
72
+ * @param {string} id
73
+ * @returns {Promise<Object|null>}
74
+ */
75
+ async getRawCredentialById(id) {
76
+ const { ObjectId } = require('mongodb');
77
+ return await mongoose.connection.db
78
+ .collection('Credential')
79
+ .findOne({ _id: new ObjectId(id) });
80
+ }
81
+
82
+ async deleteCredential(id) {
83
+ await this.prisma.credential.delete({
84
+ where: { id },
85
+ });
86
+ }
87
+ }
88
+
89
+ module.exports = { HealthCheckRepositoryMongoDB };
@@ -0,0 +1,82 @@
1
+ const {
2
+ HealthCheckRepositoryInterface,
3
+ } = require('./health-check-repository-interface');
4
+
5
+ class HealthCheckRepositoryPostgreSQL extends HealthCheckRepositoryInterface {
6
+ /**
7
+ * @param {Object} params
8
+ * @param {Object} params.prismaClient - Prisma client instance
9
+ */
10
+ constructor({ prismaClient }) {
11
+ super();
12
+ this.prisma = prismaClient;
13
+ }
14
+
15
+ /**
16
+ * @returns {Promise<{readyState: number, stateName: string, isConnected: boolean}>}
17
+ */
18
+ async getDatabaseConnectionState() {
19
+ let isConnected = false;
20
+ let stateName = 'unknown';
21
+
22
+ try {
23
+ await this.prisma.$queryRaw`SELECT 1`;
24
+ isConnected = true;
25
+ stateName = 'connected';
26
+ } catch (error) {
27
+ stateName = 'disconnected';
28
+ }
29
+
30
+ return {
31
+ readyState: isConnected ? 1 : 0,
32
+ stateName,
33
+ isConnected,
34
+ };
35
+ }
36
+
37
+ /**
38
+ * @param {number} maxTimeMS
39
+ * @returns {Promise<number>} Response time in milliseconds
40
+ */
41
+ async pingDatabase(maxTimeMS = 2000) {
42
+ const pingStart = Date.now();
43
+ await this.prisma.$queryRaw`SELECT 1`;
44
+ return Date.now() - pingStart;
45
+ }
46
+
47
+ async createCredential(credentialData) {
48
+ return await this.prisma.credential.create({
49
+ data: credentialData,
50
+ });
51
+ }
52
+
53
+ async findCredentialById(id) {
54
+ return await this.prisma.credential.findUnique({
55
+ where: { id },
56
+ });
57
+ }
58
+
59
+ /**
60
+ * @param {string} id
61
+ * @returns {Promise<Object|null>}
62
+ */
63
+ async getRawCredentialById(id) {
64
+ const results = await this.prisma.$queryRaw`
65
+ SELECT * FROM "Credential" WHERE id = ${id}
66
+ `;
67
+
68
+ if (!results || results.length === 0) {
69
+ return null;
70
+ }
71
+
72
+ return results[0];
73
+ }
74
+
75
+ async deleteCredential(id) {
76
+ await this.prisma.credential.delete({
77
+ where: { id },
78
+ });
79
+ }
80
+ }
81
+
82
+ module.exports = { HealthCheckRepositoryPostgreSQL };
@@ -0,0 +1,108 @@
1
+ const { prisma } = require('../prisma');
2
+ const { mongoose } = require('../mongoose');
3
+ const {
4
+ HealthCheckRepositoryInterface,
5
+ } = require('./health-check-repository-interface');
6
+
7
+ /**
8
+ * Repository for Health Check database operations.
9
+ * Provides atomic database operations for health testing.
10
+ *
11
+ * Follows DDD/Hexagonal Architecture:
12
+ * - Infrastructure Layer (this repository)
13
+ * - Pure database operations only, no business logic
14
+ * - Used by Application Layer (Use Cases)
15
+ *
16
+ * Works identically for both MongoDB and PostgreSQL:
17
+ * - Uses Prisma for database operations
18
+ * - Encryption happens transparently via Prisma extension
19
+ * - Both MongoDB and PostgreSQL use same Prisma API
20
+ *
21
+ * Migration from Mongoose to Prisma:
22
+ * - Replaced Mongoose models with Prisma client
23
+ * - Uses Credential model for encryption testing
24
+ * - Maintains same method signatures for compatibility
25
+ */
26
+ class HealthCheckRepository extends HealthCheckRepositoryInterface {
27
+ constructor() {
28
+ super();
29
+ }
30
+
31
+ /**
32
+ * Get database connection state
33
+ * @returns {Object} Object with readyState, stateName, and isConnected
34
+ */
35
+ getDatabaseConnectionState() {
36
+ const stateMap = {
37
+ 0: 'disconnected',
38
+ 1: 'connected',
39
+ 2: 'connecting',
40
+ 3: 'disconnecting',
41
+ };
42
+ const readyState = mongoose.connection.readyState;
43
+
44
+ return {
45
+ readyState,
46
+ stateName: stateMap[readyState],
47
+ isConnected: readyState === 1,
48
+ };
49
+ }
50
+
51
+ /**
52
+ * Ping the database to verify connectivity
53
+ * @param {number} maxTimeMS - Maximum time to wait for ping response
54
+ * @returns {Promise<number>} Response time in milliseconds
55
+ * @throws {Error} If database is not connected or ping fails
56
+ */
57
+ async pingDatabase(maxTimeMS = 2000) {
58
+ const pingStart = Date.now();
59
+ await mongoose.connection.db.admin().ping({ maxTimeMS });
60
+ return Date.now() - pingStart;
61
+ }
62
+
63
+ /**
64
+ * Create a test credential for encryption testing
65
+ * @param {Object} credentialData - Credential data to create
66
+ * @returns {Promise<Object>} Created credential
67
+ */
68
+ async createCredential(credentialData) {
69
+ return await prisma.credential.create({
70
+ data: credentialData,
71
+ });
72
+ }
73
+
74
+ /**
75
+ * Find a credential by ID
76
+ * @param {string} id - Credential ID
77
+ * @returns {Promise<Object|null>} Found credential or null
78
+ */
79
+ async findCredentialById(id) {
80
+ return await prisma.credential.findUnique({
81
+ where: { id },
82
+ });
83
+ }
84
+
85
+ /**
86
+ * Get raw credential from database bypassing Prisma encryption extension
87
+ * @param {string} id - Credential ID
88
+ * @returns {Promise<Object|null>} Raw credential from database
89
+ */
90
+ async getRawCredentialById(id) {
91
+ return await mongoose.connection.db
92
+ .collection('credentials')
93
+ .findOne({ _id: id });
94
+ }
95
+
96
+ /**
97
+ * Delete a credential by ID
98
+ * @param {string} id - Credential ID
99
+ * @returns {Promise<void>}
100
+ */
101
+ async deleteCredential(id) {
102
+ await prisma.credential.delete({
103
+ where: { id },
104
+ });
105
+ }
106
+ }
107
+
108
+ module.exports = { HealthCheckRepository };
@@ -0,0 +1,137 @@
1
+ /**
2
+ * Migration Status Repository - S3 Storage
3
+ *
4
+ * Infrastructure Layer - Hexagonal Architecture
5
+ *
6
+ * Stores migration status in S3 to avoid chicken-and-egg dependency on User/Process tables.
7
+ * Initial database migrations can't use Process table (requires User FK which doesn't exist yet).
8
+ */
9
+
10
+ const { S3Client, PutObjectCommand, GetObjectCommand } = require('@aws-sdk/client-s3');
11
+ const { randomUUID } = require('crypto');
12
+
13
+ class MigrationStatusRepositoryS3 {
14
+ /**
15
+ * @param {string} bucketName - S3 bucket name for migration status storage
16
+ * @param {S3Client} s3Client - Optional S3 client (for testing)
17
+ */
18
+ constructor(bucketName, s3Client = null) {
19
+ this.bucketName = bucketName;
20
+ this.s3Client = s3Client || new S3Client({ region: process.env.AWS_REGION || 'us-east-1' });
21
+ }
22
+
23
+ /**
24
+ * Build S3 key for migration status
25
+ * @param {string} migrationId - Migration identifier
26
+ * @param {string} stage - Deployment stage
27
+ * @returns {string} S3 key
28
+ */
29
+ _buildS3Key(migrationId, stage) {
30
+ return `migrations/${stage}/${migrationId}.json`;
31
+ }
32
+
33
+ /**
34
+ * Create new migration status record
35
+ * @param {Object} data - Migration data
36
+ * @param {string} [data.migrationId] - Migration ID (generates UUID if not provided)
37
+ * @param {string} data.stage - Deployment stage
38
+ * @param {string} [data.triggeredBy] - User or system that triggered migration
39
+ * @param {string} [data.triggeredAt] - ISO timestamp
40
+ * @returns {Promise<Object>} Created migration status
41
+ */
42
+ async create(data) {
43
+ const migrationId = data.migrationId || randomUUID();
44
+ const timestamp = data.triggeredAt || new Date().toISOString();
45
+
46
+ const status = {
47
+ migrationId,
48
+ stage: data.stage,
49
+ state: 'INITIALIZING',
50
+ progress: 0,
51
+ triggeredBy: data.triggeredBy || 'system',
52
+ triggeredAt: timestamp,
53
+ createdAt: timestamp,
54
+ updatedAt: timestamp,
55
+ };
56
+
57
+ const key = this._buildS3Key(migrationId, data.stage);
58
+
59
+ await this.s3Client.send(
60
+ new PutObjectCommand({
61
+ Bucket: this.bucketName,
62
+ Key: key,
63
+ Body: JSON.stringify(status, null, 2),
64
+ ContentType: 'application/json',
65
+ })
66
+ );
67
+
68
+ return status;
69
+ }
70
+
71
+ /**
72
+ * Update existing migration status
73
+ * @param {Object} data - Update data
74
+ * @param {string} data.migrationId - Migration ID
75
+ * @param {string} data.stage - Deployment stage
76
+ * @param {string} [data.state] - New state
77
+ * @param {number} [data.progress] - Progress percentage (0-100)
78
+ * @param {string} [data.error] - Error message if failed
79
+ * @param {string} [data.completedAt] - Completion timestamp
80
+ * @returns {Promise<Object>} Updated migration status
81
+ */
82
+ async update(data) {
83
+ const key = this._buildS3Key(data.migrationId, data.stage);
84
+
85
+ // Get existing status
86
+ const existing = await this.get(data.migrationId, data.stage);
87
+
88
+ // Merge updates
89
+ const updated = {
90
+ ...existing,
91
+ ...data,
92
+ updatedAt: new Date().toISOString(),
93
+ };
94
+
95
+ await this.s3Client.send(
96
+ new PutObjectCommand({
97
+ Bucket: this.bucketName,
98
+ Key: key,
99
+ Body: JSON.stringify(updated, null, 2),
100
+ ContentType: 'application/json',
101
+ })
102
+ );
103
+
104
+ return updated;
105
+ }
106
+
107
+ /**
108
+ * Get migration status by ID
109
+ * @param {string} migrationId - Migration ID
110
+ * @param {string} stage - Deployment stage
111
+ * @returns {Promise<Object>} Migration status
112
+ * @throws {Error} If migration not found
113
+ */
114
+ async get(migrationId, stage) {
115
+ const key = this._buildS3Key(migrationId, stage);
116
+
117
+ try {
118
+ const response = await this.s3Client.send(
119
+ new GetObjectCommand({
120
+ Bucket: this.bucketName,
121
+ Key: key,
122
+ })
123
+ );
124
+
125
+ const body = await response.Body.transformToString();
126
+ return JSON.parse(body);
127
+ } catch (error) {
128
+ if (error.name === 'NoSuchKey') {
129
+ throw new Error(`Migration not found: ${migrationId}`);
130
+ }
131
+ throw error;
132
+ }
133
+ }
134
+ }
135
+
136
+ module.exports = { MigrationStatusRepositoryS3 };
137
+
@@ -0,0 +1,29 @@
1
+ class CheckDatabaseHealthUseCase {
2
+ /**
3
+ * @param {Object} params
4
+ * @param {import('../repositories/health-check-repository-interface').HealthCheckRepositoryInterface} params.healthCheckRepository
5
+ */
6
+ constructor({ healthCheckRepository }) {
7
+ this.repository = healthCheckRepository;
8
+ }
9
+
10
+ /**
11
+ * @returns {Promise<{status: string, state: string, responseTime?: number}>}
12
+ */
13
+ async execute() {
14
+ const { stateName, isConnected } = await this.repository.getDatabaseConnectionState();
15
+
16
+ const result = {
17
+ status: isConnected ? 'healthy' : 'unhealthy',
18
+ state: stateName,
19
+ };
20
+
21
+ if (isConnected) {
22
+ result.responseTime = await this.repository.pingDatabase(2000);
23
+ }
24
+
25
+ return result;
26
+ }
27
+ }
28
+
29
+ module.exports = { CheckDatabaseHealthUseCase };
@@ -0,0 +1,81 @@
1
+ /**
2
+ * Check Database State Use Case
3
+ *
4
+ * Domain logic for checking database state (pending migrations, errors, etc).
5
+ * Does NOT trigger migrations, just reports current state.
6
+ *
7
+ * Architecture: Hexagonal/Clean
8
+ * - Use Case (Domain Layer)
9
+ * - Depends on prismaRunner (Infrastructure abstraction)
10
+ * - Called by Router or other Use Cases (Adapter Layer)
11
+ */
12
+
13
+ class ValidationError extends Error {
14
+ constructor(message) {
15
+ super(message);
16
+ this.name = 'ValidationError';
17
+ }
18
+ }
19
+
20
+ class CheckDatabaseStateUseCase {
21
+ /**
22
+ * @param {Object} dependencies
23
+ * @param {Object} dependencies.prismaRunner - Prisma runner utility
24
+ */
25
+ constructor({ prismaRunner }) {
26
+ if (!prismaRunner) {
27
+ throw new Error('prismaRunner dependency is required');
28
+ }
29
+ this.prismaRunner = prismaRunner;
30
+ }
31
+
32
+ /**
33
+ * Execute check migration status
34
+ *
35
+ * @param {string} dbType - Database type (postgresql, mongodb, or documentdb)
36
+ * @param {string} stage - Deployment stage (default: 'production')
37
+ * @returns {Promise<Object>} Migration status
38
+ */
39
+ async execute(dbType, stage = 'production') {
40
+ // Validate inputs
41
+ if (!dbType) {
42
+ throw new ValidationError('dbType is required');
43
+ }
44
+
45
+ if (!['postgresql', 'mongodb', 'documentdb'].includes(dbType)) {
46
+ throw new ValidationError('dbType must be postgresql, mongodb, or documentdb');
47
+ }
48
+
49
+ console.log(`Checking migration status for ${dbType} in ${stage}`);
50
+
51
+ // Check database state using Prisma
52
+ const state = await this.prismaRunner.checkDatabaseState(dbType);
53
+
54
+ // Build response
55
+ const response = {
56
+ upToDate: state.upToDate,
57
+ pendingMigrations: state.pendingMigrations || 0,
58
+ dbType,
59
+ stage,
60
+ };
61
+
62
+ // Add error if present
63
+ if (state.error) {
64
+ response.error = state.error;
65
+ response.recommendation = 'Run POST /db-migrate to initialize database';
66
+ }
67
+
68
+ // Add recommendation if migrations pending
69
+ if (!state.upToDate && state.pendingMigrations > 0) {
70
+ response.recommendation = `Run POST /db-migrate to apply ${state.pendingMigrations} pending migration(s)`;
71
+ }
72
+
73
+ return response;
74
+ }
75
+ }
76
+
77
+ module.exports = {
78
+ CheckDatabaseStateUseCase,
79
+ ValidationError,
80
+ };
81
+