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

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 (286) 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/config-capturing-integration.js +81 -0
  159. package/integrations/tests/doubles/dummy-integration-class.js +105 -0
  160. package/integrations/tests/doubles/test-integration-repository.js +99 -0
  161. package/integrations/use-cases/create-integration.js +83 -0
  162. package/integrations/use-cases/create-process.js +128 -0
  163. package/integrations/use-cases/delete-integration-for-user.js +101 -0
  164. package/integrations/use-cases/find-integration-context-by-external-entity-id.js +72 -0
  165. package/integrations/use-cases/get-integration-for-user.js +78 -0
  166. package/integrations/use-cases/get-integration-instance-by-definition.js +67 -0
  167. package/integrations/use-cases/get-integration-instance.js +83 -0
  168. package/integrations/use-cases/get-integrations-for-user.js +88 -0
  169. package/integrations/use-cases/get-possible-integrations.js +27 -0
  170. package/integrations/use-cases/get-process.js +87 -0
  171. package/integrations/use-cases/index.js +19 -0
  172. package/integrations/use-cases/load-integration-context.js +71 -0
  173. package/integrations/use-cases/update-integration-messages.js +44 -0
  174. package/integrations/use-cases/update-integration-status.js +32 -0
  175. package/integrations/use-cases/update-integration.js +92 -0
  176. package/integrations/use-cases/update-process-metrics.js +201 -0
  177. package/integrations/use-cases/update-process-state.js +119 -0
  178. package/integrations/utils/map-integration-dto.js +37 -0
  179. package/jest-global-setup-noop.js +3 -0
  180. package/jest-global-teardown-noop.js +3 -0
  181. package/logs/logger.js +0 -4
  182. package/{module-plugin → modules}/entity.js +1 -1
  183. package/{module-plugin → modules}/index.js +0 -8
  184. package/modules/module-factory.js +56 -0
  185. package/modules/module.js +221 -0
  186. package/modules/repositories/module-repository-documentdb.js +307 -0
  187. package/modules/repositories/module-repository-factory.js +40 -0
  188. package/modules/repositories/module-repository-interface.js +129 -0
  189. package/modules/repositories/module-repository-mongo.js +377 -0
  190. package/modules/repositories/module-repository-postgres.js +426 -0
  191. package/modules/repositories/module-repository.js +316 -0
  192. package/modules/requester/api-key.js +52 -0
  193. package/{module-plugin → modules}/requester/requester.js +1 -0
  194. package/{module-plugin → modules}/test/mock-api/api.js +8 -3
  195. package/{module-plugin → modules}/test/mock-api/definition.js +12 -8
  196. package/modules/tests/doubles/test-module-factory.js +16 -0
  197. package/modules/tests/doubles/test-module-repository.js +39 -0
  198. package/modules/use-cases/get-entities-for-user.js +32 -0
  199. package/modules/use-cases/get-entity-options-by-id.js +71 -0
  200. package/modules/use-cases/get-entity-options-by-type.js +34 -0
  201. package/modules/use-cases/get-module-instance-from-type.js +31 -0
  202. package/modules/use-cases/get-module.js +74 -0
  203. package/modules/use-cases/process-authorization-callback.js +133 -0
  204. package/modules/use-cases/refresh-entity-options.js +72 -0
  205. package/modules/use-cases/test-module-auth.js +72 -0
  206. package/modules/utils/map-module-dto.js +18 -0
  207. package/package.json +82 -50
  208. package/prisma-mongodb/schema.prisma +360 -0
  209. package/prisma-postgresql/migrations/20250930193005_init/migration.sql +315 -0
  210. package/prisma-postgresql/migrations/20251006135218_init/migration.sql +9 -0
  211. package/prisma-postgresql/migrations/20251010000000_remove_unused_entity_reference_map/migration.sql +3 -0
  212. package/prisma-postgresql/migrations/20251112195422_update_user_unique_constraints/migration.sql +25 -0
  213. package/prisma-postgresql/migrations/migration_lock.toml +3 -0
  214. package/prisma-postgresql/schema.prisma +343 -0
  215. package/queues/queuer-util.js +27 -22
  216. package/syncs/manager.js +468 -443
  217. package/syncs/repositories/sync-repository-documentdb.js +240 -0
  218. package/syncs/repositories/sync-repository-factory.js +43 -0
  219. package/syncs/repositories/sync-repository-interface.js +109 -0
  220. package/syncs/repositories/sync-repository-mongo.js +239 -0
  221. package/syncs/repositories/sync-repository-postgres.js +319 -0
  222. package/syncs/sync.js +0 -1
  223. package/token/repositories/token-repository-documentdb.js +137 -0
  224. package/token/repositories/token-repository-factory.js +40 -0
  225. package/token/repositories/token-repository-interface.js +131 -0
  226. package/token/repositories/token-repository-mongo.js +219 -0
  227. package/token/repositories/token-repository-postgres.js +264 -0
  228. package/token/repositories/token-repository.js +219 -0
  229. package/types/core/index.d.ts +2 -2
  230. package/types/integrations/index.d.ts +2 -6
  231. package/types/module-plugin/index.d.ts +5 -59
  232. package/types/syncs/index.d.ts +0 -2
  233. package/user/repositories/user-repository-documentdb.js +441 -0
  234. package/user/repositories/user-repository-factory.js +52 -0
  235. package/user/repositories/user-repository-interface.js +201 -0
  236. package/user/repositories/user-repository-mongo.js +308 -0
  237. package/user/repositories/user-repository-postgres.js +360 -0
  238. package/user/tests/doubles/test-user-repository.js +72 -0
  239. package/user/use-cases/authenticate-user.js +127 -0
  240. package/user/use-cases/authenticate-with-shared-secret.js +48 -0
  241. package/user/use-cases/create-individual-user.js +61 -0
  242. package/user/use-cases/create-organization-user.js +47 -0
  243. package/user/use-cases/create-token-for-user-id.js +30 -0
  244. package/user/use-cases/get-user-from-adopter-jwt.js +149 -0
  245. package/user/use-cases/get-user-from-bearer-token.js +77 -0
  246. package/user/use-cases/get-user-from-x-frigg-headers.js +132 -0
  247. package/user/use-cases/login-user.js +122 -0
  248. package/user/user.js +125 -0
  249. package/utils/backend-path.js +38 -0
  250. package/utils/index.js +6 -0
  251. package/websocket/repositories/websocket-connection-repository-documentdb.js +119 -0
  252. package/websocket/repositories/websocket-connection-repository-factory.js +44 -0
  253. package/websocket/repositories/websocket-connection-repository-interface.js +106 -0
  254. package/websocket/repositories/websocket-connection-repository-mongo.js +156 -0
  255. package/websocket/repositories/websocket-connection-repository-postgres.js +196 -0
  256. package/websocket/repositories/websocket-connection-repository.js +161 -0
  257. package/database/models/State.js +0 -9
  258. package/database/models/Token.js +0 -70
  259. package/database/mongo.js +0 -45
  260. package/encrypt/Cryptor.test.js +0 -32
  261. package/encrypt/encrypt.js +0 -132
  262. package/encrypt/encrypt.test.js +0 -1069
  263. package/errors/base-error.test.js +0 -32
  264. package/errors/fetch-error.test.js +0 -79
  265. package/errors/halt-error.test.js +0 -11
  266. package/errors/validation-errors.test.js +0 -120
  267. package/integrations/create-frigg-backend.js +0 -31
  268. package/integrations/integration-factory.js +0 -251
  269. package/integrations/integration-mapping.js +0 -43
  270. package/integrations/integration-model.js +0 -46
  271. package/integrations/integration-user.js +0 -144
  272. package/integrations/test/integration-base.test.js +0 -144
  273. package/lambda/TimeoutCatcher.test.js +0 -68
  274. package/logs/logger.test.js +0 -76
  275. package/module-plugin/auther.js +0 -393
  276. package/module-plugin/credential.js +0 -22
  277. package/module-plugin/entity-manager.js +0 -70
  278. package/module-plugin/manager.js +0 -169
  279. package/module-plugin/module-factory.js +0 -61
  280. package/module-plugin/requester/api-key.js +0 -36
  281. package/module-plugin/requester/requester.test.js +0 -28
  282. package/module-plugin/test/auther.test.js +0 -97
  283. /package/{module-plugin → modules}/ModuleConstants.js +0 -0
  284. /package/{module-plugin → modules}/requester/basic.js +0 -0
  285. /package/{module-plugin → modules}/requester/oauth-2.js +0 -0
  286. /package/{module-plugin → modules}/test/mock-api/mocks/hubspot.js +0 -0
package/core/Worker.js CHANGED
@@ -1,10 +1,9 @@
1
- const AWS = require('aws-sdk');
1
+ const { SQSClient, GetQueueUrlCommand, SendMessageCommand } = require('@aws-sdk/client-sqs');
2
2
  const _ = require('lodash');
3
3
  const { RequiredPropertyError } = require('../errors');
4
4
  const { get } = require('../assertions');
5
5
 
6
- AWS.config.update({ region: process.env.AWS_REGION });
7
- const sqs = new AWS.SQS({ apiVersion: '2012-11-05' });
6
+ const sqs = new SQSClient({ region: process.env.AWS_REGION });
8
7
 
9
8
  class Worker {
10
9
  async getQueueURL(params) {
@@ -12,15 +11,9 @@ class Worker {
12
11
  // let params = {
13
12
  // QueueName: process.env.QueueName
14
13
  // };
15
- return new Promise((resolve, reject) => {
16
- sqs.getQueueUrl(params, (err, data) => {
17
- if (err) {
18
- reject(err);
19
- } else {
20
- resolve(data.QueueUrl);
21
- }
22
- });
23
- });
14
+ const command = new GetQueueUrlCommand(params);
15
+ const data = await sqs.send(command);
16
+ return data.QueueUrl;
24
17
  }
25
18
 
26
19
  async run(params, context = {}) {
@@ -54,15 +47,9 @@ class Worker {
54
47
  }
55
48
 
56
49
  async sendAsyncSQSMessage(params) {
57
- return new Promise((resolve, reject) => {
58
- sqs.sendMessage(params, (err, data) => {
59
- if (err) {
60
- reject(err);
61
- } else {
62
- resolve(data.MessageId);
63
- }
64
- });
65
- });
50
+ const command = new SendMessageCommand(params);
51
+ const data = await sqs.send(command);
52
+ return data.MessageId;
66
53
  }
67
54
 
68
55
  // Throw an exception if the params do not validate
@@ -1,7 +1,7 @@
1
1
  // This line should be at the top of the webpacked output, so be sure to require createHandler first in any handlers. "Soon" sourcemaps will be built into Node... after that, this package won't be needed.
2
- require('source-map-support').install();
2
+ // REMOVING FOR NOW UNTIL WE ADD WEBPACK BACK IN
3
+ // require('source-map-support').install();
3
4
 
4
- const { connectToDatabase } = require('../database/mongo');
5
5
  const { initDebugLog, flushDebugLog } = require('../logs');
6
6
  const { secretsToEnv } = require('./secrets-to-env');
7
7
 
@@ -10,7 +10,6 @@ const createHandler = (optionByName = {}) => {
10
10
  eventName = 'Event',
11
11
  isUserFacingResponse = true,
12
12
  method,
13
- shouldUseDatabase = true,
14
13
  } = optionByName;
15
14
 
16
15
  if (!method) {
@@ -33,10 +32,6 @@ const createHandler = (optionByName = {}) => {
33
32
  // Helps mongoose reuse the connection. Lowers response times.
34
33
  context.callbackWaitsForEmptyEventLoop = false;
35
34
 
36
- if (shouldUseDatabase) {
37
- await connectToDatabase();
38
- }
39
-
40
35
  // Run the Lambda
41
36
  return await method(event, context);
42
37
  } catch (error) {
@@ -44,6 +39,18 @@ const createHandler = (optionByName = {}) => {
44
39
 
45
40
  // Don't leak implementation details to end users.
46
41
  if (isUserFacingResponse) {
42
+ // Allow client-safe errors to pass through with their actual message
43
+ if (error.isClientSafe === true) {
44
+ const statusCode = error.statusCode || 400;
45
+ return {
46
+ statusCode,
47
+ body: JSON.stringify({
48
+ error: error.message,
49
+ }),
50
+ };
51
+ }
52
+
53
+ // Hide other errors with generic message
47
54
  return {
48
55
  statusCode: 500,
49
56
  body: JSON.stringify({
@@ -0,0 +1,304 @@
1
+ const { prisma } = require('../../database/prisma');
2
+ const {
3
+ toObjectId,
4
+ fromObjectId,
5
+ findOne,
6
+ insertOne,
7
+ updateOne,
8
+ deleteOne,
9
+ } = require('../../database/documentdb-utils');
10
+ const {
11
+ CredentialRepositoryInterface,
12
+ } = require('./credential-repository-interface');
13
+ const {
14
+ DocumentDBEncryptionService,
15
+ } = require('../../database/documentdb-encryption-service');
16
+
17
+ /**
18
+ * Credential repository for DocumentDB.
19
+ * Uses DocumentDBEncryptionService for field-level encryption.
20
+ *
21
+ * Encrypted fields:
22
+ * - Credential.data.access_token
23
+ * - Credential.data.refresh_token
24
+ * - Credential.data.id_token
25
+ *
26
+ * SECURITY CRITICAL: All OAuth credentials must be encrypted at rest.
27
+ *
28
+ * @see DocumentDBEncryptionService
29
+ * @see encryption-schema-registry.js
30
+ */
31
+ class CredentialRepositoryDocumentDB extends CredentialRepositoryInterface {
32
+ constructor() {
33
+ super();
34
+ this.prisma = prisma;
35
+ this.encryptionService = new DocumentDBEncryptionService();
36
+ }
37
+
38
+ async findCredentialById(id) {
39
+ const objectId = toObjectId(id);
40
+ if (!objectId) return null;
41
+ const doc = await findOne(this.prisma, 'Credential', { _id: objectId });
42
+ if (!doc) return null;
43
+
44
+ const decryptedCredential = await this.encryptionService.decryptFields(
45
+ 'Credential',
46
+ doc
47
+ );
48
+ return this._mapCredentialById(decryptedCredential);
49
+ }
50
+
51
+ async updateAuthenticationStatus(credentialId, authIsValid) {
52
+ const objectId = toObjectId(credentialId);
53
+ if (!objectId) return { acknowledged: false, modifiedCount: 0 };
54
+ const result = await updateOne(
55
+ this.prisma,
56
+ 'Credential',
57
+ { _id: objectId },
58
+ {
59
+ $set: { authIsValid, updatedAt: new Date() },
60
+ }
61
+ );
62
+ const modified = result?.nModified ?? result?.n ?? 0;
63
+ return { acknowledged: true, modifiedCount: modified };
64
+ }
65
+
66
+ async deleteCredentialById(credentialId) {
67
+ const objectId = toObjectId(credentialId);
68
+ if (!objectId) return { acknowledged: true, deletedCount: 0 };
69
+ const result = await deleteOne(this.prisma, 'Credential', {
70
+ _id: objectId,
71
+ });
72
+ const deleted = result?.n ?? 0;
73
+ return { acknowledged: true, deletedCount: deleted };
74
+ }
75
+
76
+ async upsertCredential(credentialDetails) {
77
+ const { identifiers, details } = credentialDetails;
78
+ if (!identifiers)
79
+ throw new Error('identifiers required to upsert credential');
80
+ if (!identifiers.userId) {
81
+ throw new Error('userId required in identifiers');
82
+ }
83
+ if (!identifiers.externalId) {
84
+ throw new Error(
85
+ 'externalId required in identifiers to prevent credential collision. When multiple credentials exist for the same user, both userId and externalId are needed to uniquely identify which credential to update.'
86
+ );
87
+ }
88
+
89
+ const filter = this._buildIdentifierFilter(identifiers);
90
+ const existing = await findOne(this.prisma, 'Credential', filter);
91
+ const now = new Date();
92
+
93
+ const { authIsValid, ...oauthData } = details || {};
94
+
95
+ if (existing) {
96
+ const decryptedExisting =
97
+ await this.encryptionService.decryptFields(
98
+ 'Credential',
99
+ existing
100
+ );
101
+ const mergedData = {
102
+ ...(decryptedExisting.data || {}),
103
+ ...oauthData,
104
+ };
105
+
106
+ const updateDocument = {
107
+ userId: existing.userId,
108
+ externalId: existing.externalId,
109
+ authIsValid: authIsValid !== undefined ? authIsValid : existing.authIsValid,
110
+ data: mergedData,
111
+ updatedAt: now,
112
+ };
113
+
114
+ const encryptedUpdate = await this.encryptionService.encryptFields(
115
+ 'Credential',
116
+ { data: updateDocument.data }
117
+ );
118
+
119
+ await updateOne(
120
+ this.prisma,
121
+ 'Credential',
122
+ { _id: existing._id },
123
+ {
124
+ $set: {
125
+ userId: updateDocument.userId,
126
+ externalId: updateDocument.externalId,
127
+ authIsValid: updateDocument.authIsValid,
128
+ data: encryptedUpdate.data,
129
+ updatedAt: updateDocument.updatedAt,
130
+ },
131
+ }
132
+ );
133
+
134
+ const updated = await findOne(this.prisma, 'Credential', {
135
+ _id: existing._id,
136
+ });
137
+ const decryptedCredential =
138
+ await this.encryptionService.decryptFields(
139
+ 'Credential',
140
+ updated
141
+ );
142
+ return this._mapCredential(decryptedCredential);
143
+ }
144
+
145
+ const plainDocument = {
146
+ userId: toObjectId(identifiers.userId),
147
+ externalId: identifiers.externalId,
148
+ authIsValid: details.authIsValid,
149
+ data: { ...oauthData },
150
+ createdAt: now,
151
+ updatedAt: now,
152
+ };
153
+
154
+ const encryptedDocument = await this.encryptionService.encryptFields(
155
+ 'Credential',
156
+ plainDocument
157
+ );
158
+
159
+ const insertedId = await insertOne(
160
+ this.prisma,
161
+ 'Credential',
162
+ encryptedDocument
163
+ );
164
+
165
+ const created = await findOne(this.prisma, 'Credential', {
166
+ _id: insertedId,
167
+ });
168
+ const decryptedCredential = await this.encryptionService.decryptFields(
169
+ 'Credential',
170
+ created
171
+ );
172
+ return this._mapCredential(decryptedCredential);
173
+ }
174
+
175
+ async findCredential(filter) {
176
+ const query = this._buildFilter(filter);
177
+ const credential = await findOne(this.prisma, 'Credential', query);
178
+ if (!credential) return null;
179
+
180
+ const decryptedCredential = await this.encryptionService.decryptFields(
181
+ 'Credential',
182
+ credential
183
+ );
184
+ return this._mapCredential(decryptedCredential);
185
+ }
186
+
187
+ async updateCredential(credentialId, updates) {
188
+ const objectId = toObjectId(credentialId);
189
+ if (!objectId) return null;
190
+ const existing = await findOne(this.prisma, 'Credential', {
191
+ _id: objectId,
192
+ });
193
+ if (!existing) return null;
194
+
195
+ const { authIsValid, ...oauthData } = updates || {};
196
+
197
+ const decryptedExisting = await this.encryptionService.decryptFields(
198
+ 'Credential',
199
+ existing
200
+ );
201
+ const mergedData = { ...(decryptedExisting.data || {}), ...oauthData };
202
+
203
+ const updateDocument = {
204
+ userId: existing.userId,
205
+ externalId: existing.externalId,
206
+ authIsValid: authIsValid,
207
+ data: mergedData,
208
+ updatedAt: new Date(),
209
+ };
210
+
211
+ const encryptedUpdate = await this.encryptionService.encryptFields(
212
+ 'Credential',
213
+ { data: updateDocument.data }
214
+ );
215
+
216
+ await updateOne(
217
+ this.prisma,
218
+ 'Credential',
219
+ { _id: objectId },
220
+ {
221
+ $set: {
222
+ userId: updateDocument.userId,
223
+ externalId: updateDocument.externalId,
224
+ authIsValid: updateDocument.authIsValid,
225
+ data: encryptedUpdate.data,
226
+ updatedAt: updateDocument.updatedAt,
227
+ },
228
+ }
229
+ );
230
+
231
+ const updated = await findOne(this.prisma, 'Credential', {
232
+ _id: objectId,
233
+ });
234
+ const decryptedCredential = await this.encryptionService.decryptFields(
235
+ 'Credential',
236
+ updated
237
+ );
238
+ return this._mapCredential(decryptedCredential);
239
+ }
240
+
241
+ _buildIdentifierFilter(identifiers) {
242
+ const filter = {};
243
+ if (identifiers._id || identifiers.id) {
244
+ const idObj = toObjectId(identifiers._id || identifiers.id);
245
+ if (idObj) filter._id = idObj;
246
+ }
247
+ if (identifiers.userId) {
248
+ filter.userId = toObjectId(identifiers.userId);
249
+ }
250
+ if (identifiers.externalId !== undefined) {
251
+ filter.externalId = identifiers.externalId;
252
+ }
253
+ return filter;
254
+ }
255
+
256
+ _buildFilter(filter) {
257
+ const query = {};
258
+ if (!filter) return query;
259
+ if (filter.credentialId || filter.id) {
260
+ const idObj = toObjectId(filter.credentialId || filter.id);
261
+ if (idObj) query._id = idObj;
262
+ }
263
+ if (filter.userId !== undefined) {
264
+ query.userId = filter.userId;
265
+ }
266
+ if (filter.externalId !== undefined) {
267
+ query.externalId = filter.externalId;
268
+ }
269
+ return query;
270
+ }
271
+
272
+ /**
273
+ * Map credential document to application format
274
+ * Matches MongoDB repository format
275
+ * @private
276
+ */
277
+ _mapCredential(doc) {
278
+ const data = doc?.data || {};
279
+ const id = fromObjectId(doc?._id);
280
+ const userId = doc?.userId;
281
+ return {
282
+ id,
283
+ userId,
284
+ externalId: doc?.externalId ?? null,
285
+ authIsValid: doc?.authIsValid ?? null,
286
+ ...data,
287
+ };
288
+ }
289
+
290
+ _mapCredentialById(doc) {
291
+ const data = doc?.data || {};
292
+ const id = fromObjectId(doc?._id);
293
+ const userId = doc?.userId;
294
+ return {
295
+ id,
296
+ userId,
297
+ externalId: doc?.externalId ?? null,
298
+ authIsValid: doc?.authIsValid ?? null,
299
+ ...data,
300
+ };
301
+ }
302
+ }
303
+
304
+ module.exports = { CredentialRepositoryDocumentDB };
@@ -0,0 +1,54 @@
1
+ const { CredentialRepositoryMongo } = require('./credential-repository-mongo');
2
+ const {
3
+ CredentialRepositoryPostgres,
4
+ } = require('./credential-repository-postgres');
5
+ const {
6
+ CredentialRepositoryDocumentDB,
7
+ } = require('./credential-repository-documentdb');
8
+ const config = require('../../database/config');
9
+
10
+ /**
11
+ * Credential Repository Factory
12
+ * Creates the appropriate repository adapter based on database type
13
+ *
14
+ * Database-specific implementations:
15
+ * - MongoDB: Uses String IDs (ObjectId), no conversion needed
16
+ * - PostgreSQL: Uses Int IDs, converts String ↔ Int
17
+ *
18
+ * All repository methods return String IDs regardless of database type,
19
+ * ensuring application layer consistency.
20
+ *
21
+ * Usage:
22
+ * ```javascript
23
+ * const repository = createCredentialRepository();
24
+ * ```
25
+ *
26
+ * @returns {CredentialRepositoryInterface} Configured repository adapter
27
+ */
28
+ function createCredentialRepository() {
29
+ const dbType = config.DB_TYPE;
30
+
31
+ switch (dbType) {
32
+ case 'mongodb':
33
+ return new CredentialRepositoryMongo();
34
+
35
+ case 'postgresql':
36
+ return new CredentialRepositoryPostgres();
37
+
38
+ case 'documentdb':
39
+ return new CredentialRepositoryDocumentDB();
40
+
41
+ default:
42
+ throw new Error(
43
+ `Unsupported database type: ${dbType}. Supported values: 'mongodb', 'documentdb', 'postgresql'`
44
+ );
45
+ }
46
+ }
47
+
48
+ module.exports = {
49
+ createCredentialRepository,
50
+ // Export adapters for direct testing
51
+ CredentialRepositoryMongo,
52
+ CredentialRepositoryPostgres,
53
+ CredentialRepositoryDocumentDB,
54
+ };
@@ -0,0 +1,98 @@
1
+ /**
2
+ * Credential Repository Interface
3
+ * Abstract base class defining the contract for credential 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, Credential model has identical structure across MongoDB and PostgreSQL,
11
+ * so CredentialRepository serves both. This interface exists for consistency and
12
+ * future-proofing if database-specific implementations become needed.
13
+ *
14
+ * @abstract
15
+ */
16
+ class CredentialRepositoryInterface {
17
+ /**
18
+ * Find credential by ID
19
+ *
20
+ * @param {string|number} id - Credential ID
21
+ * @returns {Promise<Object|null>} Credential object or null
22
+ * @abstract
23
+ */
24
+ async findCredentialById(id) {
25
+ throw new Error(
26
+ 'Method findCredentialById must be implemented by subclass'
27
+ );
28
+ }
29
+
30
+ /**
31
+ * Update authentication status
32
+ *
33
+ * @param {string|number} credentialId - Credential ID
34
+ * @param {boolean} authIsValid - Authentication validity status
35
+ * @returns {Promise<Object>} Update result
36
+ * @abstract
37
+ */
38
+ async updateAuthenticationStatus(credentialId, authIsValid) {
39
+ throw new Error(
40
+ 'Method updateAuthenticationStatus must be implemented by subclass'
41
+ );
42
+ }
43
+
44
+ /**
45
+ * Permanently remove a credential document
46
+ *
47
+ * @param {string|number} credentialId - Credential ID
48
+ * @returns {Promise<Object>} Deletion result
49
+ * @abstract
50
+ */
51
+ async deleteCredentialById(credentialId) {
52
+ throw new Error(
53
+ 'Method deleteCredentialById must be implemented by subclass'
54
+ );
55
+ }
56
+
57
+ /**
58
+ * Create or update credential matching identifiers
59
+ *
60
+ * @param {{identifiers: Object, details: Object}} credentialDetails
61
+ * @returns {Promise<Object>} The persisted credential
62
+ * @abstract
63
+ */
64
+ async upsertCredential(credentialDetails) {
65
+ throw new Error(
66
+ 'Method upsertCredential must be implemented by subclass'
67
+ );
68
+ }
69
+
70
+ /**
71
+ * Find a credential by filter criteria
72
+ *
73
+ * @param {Object} filter - Filter criteria
74
+ * @returns {Promise<Object|null>} Credential object or null if not found
75
+ * @abstract
76
+ */
77
+ async findCredential(filter) {
78
+ throw new Error(
79
+ 'Method findCredential must be implemented by subclass'
80
+ );
81
+ }
82
+
83
+ /**
84
+ * Update a credential by ID
85
+ *
86
+ * @param {string|number} credentialId - Credential ID
87
+ * @param {Object} updates - Fields to update
88
+ * @returns {Promise<Object|null>} Updated credential object or null if not found
89
+ * @abstract
90
+ */
91
+ async updateCredential(credentialId, updates) {
92
+ throw new Error(
93
+ 'Method updateCredential must be implemented by subclass'
94
+ );
95
+ }
96
+ }
97
+
98
+ module.exports = { CredentialRepositoryInterface };