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

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (293) hide show
  1. package/CLAUDE.md +694 -0
  2. package/README.md +959 -50
  3. package/application/commands/README.md +451 -0
  4. package/application/commands/credential-commands.js +245 -0
  5. package/application/commands/entity-commands.js +336 -0
  6. package/application/commands/integration-commands.js +210 -0
  7. package/application/commands/scheduler-commands.js +263 -0
  8. package/application/commands/user-commands.js +283 -0
  9. package/application/index.js +73 -0
  10. package/core/CLAUDE.md +690 -0
  11. package/core/Worker.js +8 -21
  12. package/core/create-handler.js +14 -7
  13. package/credential/repositories/credential-repository-documentdb.js +304 -0
  14. package/credential/repositories/credential-repository-factory.js +54 -0
  15. package/credential/repositories/credential-repository-interface.js +98 -0
  16. package/credential/repositories/credential-repository-mongo.js +269 -0
  17. package/credential/repositories/credential-repository-postgres.js +287 -0
  18. package/credential/repositories/credential-repository.js +300 -0
  19. package/credential/use-cases/get-credential-for-user.js +25 -0
  20. package/credential/use-cases/update-authentication-status.js +15 -0
  21. package/database/MONGODB_TRANSACTION_FIX.md +198 -0
  22. package/database/adapters/lambda-invoker.js +97 -0
  23. package/database/config.js +154 -0
  24. package/database/documentdb-encryption-service.js +330 -0
  25. package/database/documentdb-utils.js +136 -0
  26. package/database/encryption/README.md +839 -0
  27. package/database/encryption/documentdb-encryption-service.md +3575 -0
  28. package/database/encryption/encryption-schema-registry.js +268 -0
  29. package/database/encryption/field-encryption-service.js +226 -0
  30. package/database/encryption/logger.js +79 -0
  31. package/database/encryption/prisma-encryption-extension.js +222 -0
  32. package/database/index.js +61 -21
  33. package/database/models/WebsocketConnection.js +16 -10
  34. package/database/models/readme.md +1 -0
  35. package/database/prisma.js +182 -0
  36. package/database/repositories/health-check-repository-documentdb.js +134 -0
  37. package/database/repositories/health-check-repository-factory.js +48 -0
  38. package/database/repositories/health-check-repository-interface.js +82 -0
  39. package/database/repositories/health-check-repository-mongodb.js +89 -0
  40. package/database/repositories/health-check-repository-postgres.js +82 -0
  41. package/database/repositories/health-check-repository.js +108 -0
  42. package/database/repositories/migration-status-repository-s3.js +137 -0
  43. package/database/use-cases/check-database-health-use-case.js +29 -0
  44. package/database/use-cases/check-database-state-use-case.js +81 -0
  45. package/database/use-cases/check-encryption-health-use-case.js +83 -0
  46. package/database/use-cases/get-database-state-via-worker-use-case.js +61 -0
  47. package/database/use-cases/get-migration-status-use-case.js +93 -0
  48. package/database/use-cases/run-database-migration-use-case.js +139 -0
  49. package/database/use-cases/test-encryption-use-case.js +253 -0
  50. package/database/use-cases/trigger-database-migration-use-case.js +157 -0
  51. package/database/utils/mongodb-collection-utils.js +91 -0
  52. package/database/utils/mongodb-schema-init.js +106 -0
  53. package/database/utils/prisma-runner.js +477 -0
  54. package/database/utils/prisma-schema-parser.js +182 -0
  55. package/docs/PROCESS_MANAGEMENT_QUEUE_SPEC.md +517 -0
  56. package/encrypt/Cryptor.js +34 -168
  57. package/encrypt/index.js +1 -2
  58. package/encrypt/test-encrypt.js +0 -2
  59. package/errors/client-safe-error.js +26 -0
  60. package/errors/fetch-error.js +2 -1
  61. package/errors/index.js +2 -0
  62. package/generated/prisma-mongodb/client.d.ts +1 -0
  63. package/generated/prisma-mongodb/client.js +4 -0
  64. package/generated/prisma-mongodb/default.d.ts +1 -0
  65. package/generated/prisma-mongodb/default.js +4 -0
  66. package/generated/prisma-mongodb/edge.d.ts +1 -0
  67. package/generated/prisma-mongodb/edge.js +335 -0
  68. package/generated/prisma-mongodb/index-browser.js +317 -0
  69. package/generated/prisma-mongodb/index.d.ts +22955 -0
  70. package/generated/prisma-mongodb/index.js +360 -0
  71. package/generated/prisma-mongodb/package.json +183 -0
  72. package/generated/prisma-mongodb/query-engine-debian-openssl-3.0.x +0 -0
  73. package/generated/prisma-mongodb/query-engine-rhel-openssl-3.0.x +0 -0
  74. package/generated/prisma-mongodb/runtime/binary.d.ts +1 -0
  75. package/generated/prisma-mongodb/runtime/binary.js +289 -0
  76. package/generated/prisma-mongodb/runtime/edge-esm.js +34 -0
  77. package/generated/prisma-mongodb/runtime/edge.js +34 -0
  78. package/generated/prisma-mongodb/runtime/index-browser.d.ts +370 -0
  79. package/generated/prisma-mongodb/runtime/index-browser.js +16 -0
  80. package/generated/prisma-mongodb/runtime/library.d.ts +3977 -0
  81. package/generated/prisma-mongodb/runtime/react-native.js +83 -0
  82. package/generated/prisma-mongodb/runtime/wasm-compiler-edge.js +84 -0
  83. package/generated/prisma-mongodb/runtime/wasm-engine-edge.js +36 -0
  84. package/generated/prisma-mongodb/schema.prisma +362 -0
  85. package/generated/prisma-mongodb/wasm-edge-light-loader.mjs +4 -0
  86. package/generated/prisma-mongodb/wasm-worker-loader.mjs +4 -0
  87. package/generated/prisma-mongodb/wasm.d.ts +1 -0
  88. package/generated/prisma-mongodb/wasm.js +342 -0
  89. package/generated/prisma-postgresql/client.d.ts +1 -0
  90. package/generated/prisma-postgresql/client.js +4 -0
  91. package/generated/prisma-postgresql/default.d.ts +1 -0
  92. package/generated/prisma-postgresql/default.js +4 -0
  93. package/generated/prisma-postgresql/edge.d.ts +1 -0
  94. package/generated/prisma-postgresql/edge.js +357 -0
  95. package/generated/prisma-postgresql/index-browser.js +339 -0
  96. package/generated/prisma-postgresql/index.d.ts +25131 -0
  97. package/generated/prisma-postgresql/index.js +382 -0
  98. package/generated/prisma-postgresql/package.json +183 -0
  99. package/generated/prisma-postgresql/query-engine-debian-openssl-3.0.x +0 -0
  100. package/generated/prisma-postgresql/query-engine-rhel-openssl-3.0.x +0 -0
  101. package/generated/prisma-postgresql/query_engine_bg.js +2 -0
  102. package/generated/prisma-postgresql/query_engine_bg.wasm +0 -0
  103. package/generated/prisma-postgresql/runtime/binary.d.ts +1 -0
  104. package/generated/prisma-postgresql/runtime/binary.js +289 -0
  105. package/generated/prisma-postgresql/runtime/edge-esm.js +34 -0
  106. package/generated/prisma-postgresql/runtime/edge.js +34 -0
  107. package/generated/prisma-postgresql/runtime/index-browser.d.ts +370 -0
  108. package/generated/prisma-postgresql/runtime/index-browser.js +16 -0
  109. package/generated/prisma-postgresql/runtime/library.d.ts +3977 -0
  110. package/generated/prisma-postgresql/runtime/react-native.js +83 -0
  111. package/generated/prisma-postgresql/runtime/wasm-compiler-edge.js +84 -0
  112. package/generated/prisma-postgresql/runtime/wasm-engine-edge.js +36 -0
  113. package/generated/prisma-postgresql/schema.prisma +345 -0
  114. package/generated/prisma-postgresql/wasm-edge-light-loader.mjs +4 -0
  115. package/generated/prisma-postgresql/wasm-worker-loader.mjs +4 -0
  116. package/generated/prisma-postgresql/wasm.d.ts +1 -0
  117. package/generated/prisma-postgresql/wasm.js +364 -0
  118. package/handlers/WEBHOOKS.md +653 -0
  119. package/handlers/app-definition-loader.js +38 -0
  120. package/handlers/app-handler-helpers.js +56 -0
  121. package/handlers/backend-utils.js +186 -0
  122. package/handlers/database-migration-handler.js +227 -0
  123. package/handlers/integration-event-dispatcher.js +54 -0
  124. package/handlers/routers/HEALTHCHECK.md +342 -0
  125. package/handlers/routers/auth.js +15 -0
  126. package/handlers/routers/db-migration.handler.js +29 -0
  127. package/handlers/routers/db-migration.js +326 -0
  128. package/handlers/routers/health.js +516 -0
  129. package/handlers/routers/integration-defined-routers.js +45 -0
  130. package/handlers/routers/integration-webhook-routers.js +67 -0
  131. package/handlers/routers/user.js +63 -0
  132. package/handlers/routers/websocket.js +57 -0
  133. package/handlers/use-cases/check-external-apis-health-use-case.js +81 -0
  134. package/handlers/use-cases/check-integrations-health-use-case.js +44 -0
  135. package/handlers/workers/db-migration.js +352 -0
  136. package/handlers/workers/integration-defined-workers.js +27 -0
  137. package/index.js +78 -22
  138. package/infrastructure/scheduler/eventbridge-scheduler-adapter.js +184 -0
  139. package/infrastructure/scheduler/index.js +33 -0
  140. package/infrastructure/scheduler/mock-scheduler-adapter.js +143 -0
  141. package/infrastructure/scheduler/scheduler-service-factory.js +73 -0
  142. package/infrastructure/scheduler/scheduler-service-interface.js +47 -0
  143. package/integrations/WEBHOOK-QUICKSTART.md +151 -0
  144. package/integrations/index.js +12 -10
  145. package/integrations/integration-base.js +326 -55
  146. package/integrations/integration-router.js +374 -179
  147. package/integrations/options.js +1 -1
  148. package/integrations/repositories/integration-mapping-repository-documentdb.js +280 -0
  149. package/integrations/repositories/integration-mapping-repository-factory.js +57 -0
  150. package/integrations/repositories/integration-mapping-repository-interface.js +106 -0
  151. package/integrations/repositories/integration-mapping-repository-mongo.js +161 -0
  152. package/integrations/repositories/integration-mapping-repository-postgres.js +227 -0
  153. package/integrations/repositories/integration-mapping-repository.js +156 -0
  154. package/integrations/repositories/integration-repository-documentdb.js +210 -0
  155. package/integrations/repositories/integration-repository-factory.js +51 -0
  156. package/integrations/repositories/integration-repository-interface.js +127 -0
  157. package/integrations/repositories/integration-repository-mongo.js +303 -0
  158. package/integrations/repositories/integration-repository-postgres.js +352 -0
  159. package/integrations/repositories/process-repository-documentdb.js +243 -0
  160. package/integrations/repositories/process-repository-factory.js +53 -0
  161. package/integrations/repositories/process-repository-interface.js +90 -0
  162. package/integrations/repositories/process-repository-mongo.js +190 -0
  163. package/integrations/repositories/process-repository-postgres.js +217 -0
  164. package/integrations/tests/doubles/config-capturing-integration.js +81 -0
  165. package/integrations/tests/doubles/dummy-integration-class.js +105 -0
  166. package/integrations/tests/doubles/test-integration-repository.js +99 -0
  167. package/integrations/use-cases/create-integration.js +83 -0
  168. package/integrations/use-cases/create-process.js +128 -0
  169. package/integrations/use-cases/delete-integration-for-user.js +101 -0
  170. package/integrations/use-cases/find-integration-context-by-external-entity-id.js +72 -0
  171. package/integrations/use-cases/get-integration-for-user.js +78 -0
  172. package/integrations/use-cases/get-integration-instance-by-definition.js +67 -0
  173. package/integrations/use-cases/get-integration-instance.js +83 -0
  174. package/integrations/use-cases/get-integrations-for-user.js +88 -0
  175. package/integrations/use-cases/get-possible-integrations.js +27 -0
  176. package/integrations/use-cases/get-process.js +87 -0
  177. package/integrations/use-cases/index.js +19 -0
  178. package/integrations/use-cases/load-integration-context.js +71 -0
  179. package/integrations/use-cases/update-integration-messages.js +44 -0
  180. package/integrations/use-cases/update-integration-status.js +32 -0
  181. package/integrations/use-cases/update-integration.js +92 -0
  182. package/integrations/use-cases/update-process-metrics.js +201 -0
  183. package/integrations/use-cases/update-process-state.js +119 -0
  184. package/integrations/utils/map-integration-dto.js +37 -0
  185. package/jest-global-setup-noop.js +3 -0
  186. package/jest-global-teardown-noop.js +3 -0
  187. package/logs/logger.js +0 -4
  188. package/{module-plugin → modules}/entity.js +1 -1
  189. package/{module-plugin → modules}/index.js +0 -8
  190. package/modules/module-factory.js +56 -0
  191. package/modules/module.js +221 -0
  192. package/modules/repositories/module-repository-documentdb.js +335 -0
  193. package/modules/repositories/module-repository-factory.js +40 -0
  194. package/modules/repositories/module-repository-interface.js +129 -0
  195. package/modules/repositories/module-repository-mongo.js +408 -0
  196. package/modules/repositories/module-repository-postgres.js +453 -0
  197. package/modules/repositories/module-repository.js +345 -0
  198. package/modules/requester/api-key.js +52 -0
  199. package/modules/requester/oauth-2.js +384 -0
  200. package/{module-plugin → modules}/requester/requester.js +4 -2
  201. package/{module-plugin → modules}/test/mock-api/api.js +8 -3
  202. package/{module-plugin → modules}/test/mock-api/definition.js +14 -10
  203. package/modules/tests/doubles/test-module-factory.js +16 -0
  204. package/modules/tests/doubles/test-module-repository.js +39 -0
  205. package/modules/use-cases/get-entities-for-user.js +32 -0
  206. package/modules/use-cases/get-entity-options-by-id.js +71 -0
  207. package/modules/use-cases/get-entity-options-by-type.js +34 -0
  208. package/modules/use-cases/get-module-instance-from-type.js +31 -0
  209. package/modules/use-cases/get-module.js +74 -0
  210. package/modules/use-cases/process-authorization-callback.js +133 -0
  211. package/modules/use-cases/refresh-entity-options.js +72 -0
  212. package/modules/use-cases/test-module-auth.js +72 -0
  213. package/modules/utils/map-module-dto.js +18 -0
  214. package/package.json +82 -50
  215. package/prisma-mongodb/schema.prisma +362 -0
  216. package/prisma-postgresql/migrations/20250930193005_init/migration.sql +315 -0
  217. package/prisma-postgresql/migrations/20251006135218_init/migration.sql +9 -0
  218. package/prisma-postgresql/migrations/20251010000000_remove_unused_entity_reference_map/migration.sql +3 -0
  219. package/prisma-postgresql/migrations/20251112195422_update_user_unique_constraints/migration.sql +25 -0
  220. package/prisma-postgresql/migrations/migration_lock.toml +3 -0
  221. package/prisma-postgresql/schema.prisma +345 -0
  222. package/queues/queuer-util.js +27 -22
  223. package/syncs/manager.js +468 -443
  224. package/syncs/repositories/sync-repository-documentdb.js +240 -0
  225. package/syncs/repositories/sync-repository-factory.js +43 -0
  226. package/syncs/repositories/sync-repository-interface.js +109 -0
  227. package/syncs/repositories/sync-repository-mongo.js +239 -0
  228. package/syncs/repositories/sync-repository-postgres.js +319 -0
  229. package/syncs/sync.js +0 -1
  230. package/token/repositories/token-repository-documentdb.js +137 -0
  231. package/token/repositories/token-repository-factory.js +40 -0
  232. package/token/repositories/token-repository-interface.js +131 -0
  233. package/token/repositories/token-repository-mongo.js +219 -0
  234. package/token/repositories/token-repository-postgres.js +264 -0
  235. package/token/repositories/token-repository.js +219 -0
  236. package/types/core/index.d.ts +2 -2
  237. package/types/integrations/index.d.ts +2 -6
  238. package/types/module-plugin/index.d.ts +5 -59
  239. package/types/syncs/index.d.ts +0 -2
  240. package/user/repositories/user-repository-documentdb.js +441 -0
  241. package/user/repositories/user-repository-factory.js +52 -0
  242. package/user/repositories/user-repository-interface.js +201 -0
  243. package/user/repositories/user-repository-mongo.js +308 -0
  244. package/user/repositories/user-repository-postgres.js +360 -0
  245. package/user/tests/doubles/test-user-repository.js +72 -0
  246. package/user/use-cases/authenticate-user.js +127 -0
  247. package/user/use-cases/authenticate-with-shared-secret.js +48 -0
  248. package/user/use-cases/create-individual-user.js +61 -0
  249. package/user/use-cases/create-organization-user.js +47 -0
  250. package/user/use-cases/create-token-for-user-id.js +30 -0
  251. package/user/use-cases/get-user-from-adopter-jwt.js +149 -0
  252. package/user/use-cases/get-user-from-bearer-token.js +77 -0
  253. package/user/use-cases/get-user-from-x-frigg-headers.js +132 -0
  254. package/user/use-cases/login-user.js +122 -0
  255. package/user/user.js +125 -0
  256. package/utils/backend-path.js +38 -0
  257. package/utils/index.js +6 -0
  258. package/websocket/repositories/websocket-connection-repository-documentdb.js +119 -0
  259. package/websocket/repositories/websocket-connection-repository-factory.js +44 -0
  260. package/websocket/repositories/websocket-connection-repository-interface.js +106 -0
  261. package/websocket/repositories/websocket-connection-repository-mongo.js +156 -0
  262. package/websocket/repositories/websocket-connection-repository-postgres.js +196 -0
  263. package/websocket/repositories/websocket-connection-repository.js +161 -0
  264. package/database/models/State.js +0 -9
  265. package/database/models/Token.js +0 -70
  266. package/database/mongo.js +0 -45
  267. package/encrypt/Cryptor.test.js +0 -32
  268. package/encrypt/encrypt.js +0 -132
  269. package/encrypt/encrypt.test.js +0 -1069
  270. package/errors/base-error.test.js +0 -32
  271. package/errors/fetch-error.test.js +0 -79
  272. package/errors/halt-error.test.js +0 -11
  273. package/errors/validation-errors.test.js +0 -120
  274. package/integrations/create-frigg-backend.js +0 -31
  275. package/integrations/integration-factory.js +0 -251
  276. package/integrations/integration-mapping.js +0 -43
  277. package/integrations/integration-model.js +0 -46
  278. package/integrations/integration-user.js +0 -144
  279. package/integrations/test/integration-base.test.js +0 -144
  280. package/lambda/TimeoutCatcher.test.js +0 -68
  281. package/logs/logger.test.js +0 -76
  282. package/module-plugin/auther.js +0 -393
  283. package/module-plugin/credential.js +0 -22
  284. package/module-plugin/entity-manager.js +0 -70
  285. package/module-plugin/manager.js +0 -169
  286. package/module-plugin/module-factory.js +0 -61
  287. package/module-plugin/requester/api-key.js +0 -36
  288. package/module-plugin/requester/oauth-2.js +0 -219
  289. package/module-plugin/requester/requester.test.js +0 -28
  290. package/module-plugin/test/auther.test.js +0 -97
  291. /package/{module-plugin → modules}/ModuleConstants.js +0 -0
  292. /package/{module-plugin → modules}/requester/basic.js +0 -0
  293. /package/{module-plugin → modules}/test/mock-api/mocks/hubspot.js +0 -0
@@ -0,0 +1,221 @@
1
+ const { Delegate } = require('../core');
2
+ const _ = require('lodash');
3
+ const { flushDebugLog } = require('../logs');
4
+ const { ModuleConstants } = require('./ModuleConstants');
5
+ const {
6
+ createCredentialRepository,
7
+ } = require('../credential/repositories/credential-repository-factory');
8
+ const {
9
+ createModuleRepository,
10
+ } = require('./repositories/module-repository-factory');
11
+
12
+ // todo: this class should be a Domain class, and the Delegate function is preventing us from
13
+ // doing that, we probably have to get rid of the Delegate class as well as the event based
14
+ // calls since they go against the Domain Driven Design principles (eg. a domain class should not call repository methods or use cases)
15
+ class Module extends Delegate {
16
+ //todo: entity should be replaced with actual entity properties
17
+ /**
18
+ *
19
+ * @param {Object} params
20
+ * @param {Object} params.definition The definition of the Api Module
21
+ * @param {string} params.userId The user id
22
+ * @param {Object} params.entity The entity record from the database
23
+ */
24
+ constructor({ definition, userId = null, entity: entityObj = null }) {
25
+ super({ definition, userId, entity: entityObj });
26
+
27
+ this.validateDefinition(definition);
28
+
29
+ this.userId = userId;
30
+ this.entity = entityObj;
31
+ this.credential = entityObj?.credential;
32
+ this.definition = definition;
33
+ this.name = this.definition.moduleName;
34
+ this.modelName = this.definition.modelName;
35
+ this.apiClass = this.definition.API;
36
+
37
+ this.credentialRepository = createCredentialRepository();
38
+ this.moduleRepository = createModuleRepository();
39
+
40
+ Object.assign(this, this.definition.requiredAuthMethods);
41
+
42
+ const apiParams = {
43
+ ...this.definition.env,
44
+ delegate: this,
45
+ ...(this.credential?.data ? this.apiParamsFromCredential(this.credential.data) : {}), // Handle case when credential is undefined
46
+ ...this.apiParamsFromEntity(this.entity),
47
+ };
48
+ this.api = new this.apiClass(apiParams);
49
+ }
50
+
51
+ getName() {
52
+ return this.name;
53
+ }
54
+
55
+ getEntityOptions() {
56
+ return this.definition.getEntityOptions();
57
+ }
58
+
59
+ async refreshEntityOptions(options) {
60
+ await this.definition.refreshEntityOptions(options);
61
+ return this.getEntityOptions();
62
+ }
63
+
64
+ apiParamsFromCredential(credential) {
65
+ return _.pick(credential, ...this.apiPropertiesToPersist?.credential);
66
+ }
67
+
68
+ apiParamsFromEntity(entity) {
69
+ return _.pick(entity, ...this.apiPropertiesToPersist?.entity);
70
+ }
71
+
72
+ validateAuthorizationRequirements() {
73
+ const requirements = this.getAuthorizationRequirements();
74
+ let valid = true;
75
+ if (
76
+ ['oauth1', 'oauth2'].includes(requirements.type) &&
77
+ !requirements.url
78
+ ) {
79
+ valid = false;
80
+ }
81
+ return valid;
82
+ }
83
+
84
+ getAuthorizationRequirements(params) {
85
+ return this.api.getAuthorizationRequirements();
86
+ }
87
+
88
+ async testAuth() {
89
+ let validAuth = false;
90
+ try {
91
+ if (await this.testAuthRequest(this.api)) validAuth = true;
92
+ } catch (e) {
93
+ flushDebugLog(e);
94
+ }
95
+ return validAuth;
96
+ }
97
+
98
+ async onTokenUpdate() {
99
+ const credentialDetails = await this.getCredentialDetails(
100
+ this.api,
101
+ this.userId
102
+ );
103
+ Object.assign(
104
+ credentialDetails.details,
105
+ this.apiParamsFromCredential(this.api)
106
+ );
107
+ credentialDetails.details.authIsValid = true;
108
+
109
+ const persisted = await this.credentialRepository.upsertCredential(
110
+ credentialDetails
111
+ );
112
+ this.credential = persisted;
113
+ }
114
+
115
+ async receiveNotification(notifier, delegateString, object = null) {
116
+ if (delegateString === this.api.DLGT_TOKEN_UPDATE) {
117
+ await this.onTokenUpdate();
118
+ } else if (delegateString === this.api.DLGT_TOKEN_DEAUTHORIZED) {
119
+ await this.deauthorize();
120
+ } else if (delegateString === this.api.DLGT_INVALID_AUTH) {
121
+ await this.markCredentialsInvalid();
122
+ }
123
+ }
124
+
125
+ async markCredentialsInvalid() {
126
+ if (!this.credential) return;
127
+
128
+ if (!this.credential.id) return;
129
+
130
+ await this.credentialRepository.updateAuthenticationStatus(
131
+ this.credential.id,
132
+ false
133
+ );
134
+
135
+ // Keep the in-memory snapshot consistent so that callers can read the
136
+ // updated state without another fetch.
137
+ this.credential.authIsValid = false;
138
+ }
139
+
140
+ async deauthorize() {
141
+ //todo: Check if this is correct, we're instantiating a new api without params (credentials, tokens, etc...)
142
+ this.api = new this.apiClass();
143
+
144
+ // Remove persisted credential (if any)
145
+ if (this.entity?.credential) {
146
+ const credentialId =
147
+ this.entity.credential.id || this.entity.credential;
148
+
149
+ // Delete credential via repository
150
+ await this.credentialRepository.deleteCredentialById(credentialId);
151
+
152
+ // Unset credential reference on the Entity document
153
+ const entityId = this.entity.id;
154
+ if (entityId) {
155
+ await this.moduleRepository.unsetCredential(entityId);
156
+ }
157
+
158
+ // Keep in-memory snapshot consistent
159
+ this.entity.credential = undefined;
160
+ }
161
+ }
162
+
163
+ // todo: check if all these props are still up to date
164
+ validateDefinition(definition) {
165
+ if (!definition) {
166
+ throw new Error('Module definition is required');
167
+ }
168
+ if (!definition.moduleName) {
169
+ throw new Error('Module definition requires moduleName');
170
+ }
171
+ if (!definition.API) {
172
+ throw new Error('Module definition requires API class');
173
+ }
174
+ if (!definition.requiredAuthMethods) {
175
+ throw new Error('Module definition requires requiredAuthMethods');
176
+ } else {
177
+ if (
178
+ definition.API.requesterType ===
179
+ ModuleConstants.authType.oauth2 &&
180
+ !definition.requiredAuthMethods.getToken
181
+ ) {
182
+ throw new Error(
183
+ 'Module definition requires requiredAuthMethods.getToken'
184
+ );
185
+ }
186
+ if (!definition.requiredAuthMethods.getEntityDetails) {
187
+ throw new Error(
188
+ 'Module definition requires requiredAuthMethods.getEntityDetails'
189
+ );
190
+ }
191
+ if (!definition.requiredAuthMethods.getCredentialDetails) {
192
+ throw new Error(
193
+ 'Module definition requires requiredAuthMethods.getCredentialDetails'
194
+ );
195
+ }
196
+ if (!definition.requiredAuthMethods.apiPropertiesToPersist) {
197
+ throw new Error(
198
+ 'Module definition requires requiredAuthMethods.apiPropertiesToPersist'
199
+ );
200
+ } else if (definition.Credential) {
201
+ for (const prop of definition.requiredAuthMethods
202
+ .apiPropertiesToPersist?.credential) {
203
+ if (
204
+ !definition.Credential.schema.paths.hasOwnProperty(prop)
205
+ ) {
206
+ throw new Error(
207
+ `Module definition requires Credential schema to have property ${prop}`
208
+ );
209
+ }
210
+ }
211
+ }
212
+ if (!definition.requiredAuthMethods.testAuthRequest) {
213
+ throw new Error(
214
+ 'Module definition requires requiredAuthMethods.testAuth'
215
+ );
216
+ }
217
+ }
218
+ }
219
+ }
220
+
221
+ module.exports = { Module };
@@ -0,0 +1,335 @@
1
+ const { prisma } = require('../../database/prisma');
2
+ const {
3
+ toObjectId,
4
+ fromObjectId,
5
+ findMany,
6
+ findOne,
7
+ insertOne,
8
+ updateOne,
9
+ deleteOne,
10
+ } = require('../../database/documentdb-utils');
11
+ const { ModuleRepositoryInterface } = require('./module-repository-interface');
12
+ const { DocumentDBEncryptionService } = require('../../database/documentdb-encryption-service');
13
+
14
+ /**
15
+ * Module/Entity repository for DocumentDB.
16
+ * Uses DocumentDBEncryptionService for credential decryption.
17
+ *
18
+ * Encrypted fields: Credential.data.*
19
+ *
20
+ * Note: This repository only reads credentials. CredentialRepository
21
+ * handles credential creation/updates with encryption.
22
+ *
23
+ * @see DocumentDBEncryptionService
24
+ * @see CredentialRepositoryDocumentDB
25
+ */
26
+ class ModuleRepositoryDocumentDB extends ModuleRepositoryInterface {
27
+ constructor() {
28
+ super();
29
+ this.prisma = prisma;
30
+ this.encryptionService = new DocumentDBEncryptionService();
31
+ }
32
+
33
+ async findEntityById(entityId) {
34
+ const objectId = toObjectId(entityId);
35
+ if (!objectId) {
36
+ throw new Error(`Entity ${entityId} not found`);
37
+ }
38
+ const doc = await findOne(this.prisma, 'Entity', { _id: objectId });
39
+ if (!doc) {
40
+ throw new Error(`Entity ${entityId} not found`);
41
+ }
42
+ const credential = await this._fetchCredential(doc.credentialId);
43
+ return this._mapEntity(doc, credential);
44
+ }
45
+
46
+ async findEntitiesByUserId(userId) {
47
+ const objectId = toObjectId(userId);
48
+ if (!objectId) {
49
+ throw new Error(`Invalid userId: ${userId}`);
50
+ }
51
+ const filter = { userId: objectId };
52
+ const docs = await findMany(this.prisma, 'Entity', filter);
53
+ const credentialMap = await this._fetchCredentialsBulk(docs.map((doc) => doc.credentialId));
54
+ return docs.map((doc) => this._mapEntity(doc, credentialMap.get(fromObjectId(doc.credentialId)) || null));
55
+ }
56
+
57
+ async findEntitiesByIds(entitiesIds) {
58
+ const ids = (entitiesIds || []).map((id) => toObjectId(id)).filter(Boolean);
59
+ if (ids.length === 0) return [];
60
+ const docs = await findMany(this.prisma, 'Entity', { _id: { $in: ids } });
61
+ const credentialMap = await this._fetchCredentialsBulk(docs.map((doc) => doc.credentialId));
62
+ return docs.map((doc) => this._mapEntity(doc, credentialMap.get(fromObjectId(doc.credentialId)) || null));
63
+ }
64
+
65
+ async findEntitiesByUserIdAndModuleName(userId, moduleName) {
66
+ const objectId = toObjectId(userId);
67
+ if (!objectId) {
68
+ throw new Error(`Invalid userId: ${userId}`);
69
+ }
70
+ const filter = {
71
+ userId: objectId,
72
+ moduleName,
73
+ };
74
+ const docs = await findMany(this.prisma, 'Entity', filter);
75
+ const credentialMap = await this._fetchCredentialsBulk(docs.map((doc) => doc.credentialId));
76
+ return docs.map((doc) => this._mapEntity(doc, credentialMap.get(fromObjectId(doc.credentialId)) || null));
77
+ }
78
+
79
+ async unsetCredential(entityId) {
80
+ const objectId = toObjectId(entityId);
81
+ if (!objectId) return false;
82
+ await updateOne(
83
+ this.prisma,
84
+ 'Entity',
85
+ { _id: objectId },
86
+ {
87
+ $set: {
88
+ credentialId: null,
89
+ },
90
+ }
91
+ );
92
+ return true;
93
+ }
94
+
95
+ async findEntity(filter) {
96
+ const query = this._buildFilter(filter);
97
+ const doc = await findOne(this.prisma, 'Entity', query);
98
+ if (!doc) return null;
99
+ const credential = await this._fetchCredential(doc.credentialId);
100
+ return this._mapEntity(doc, credential);
101
+ }
102
+
103
+ async createEntity(entityData) {
104
+ const {
105
+ user,
106
+ userId,
107
+ credential,
108
+ credentialId,
109
+ name,
110
+ moduleName,
111
+ externalId,
112
+ ...dynamicData
113
+ } = entityData;
114
+
115
+ const document = {
116
+ userId: toObjectId(userId || user),
117
+ credentialId: toObjectId(credentialId || credential) || null,
118
+ name: name ?? null,
119
+ moduleName: moduleName ?? null,
120
+ externalId: externalId ?? null,
121
+ data: dynamicData,
122
+ };
123
+ const insertedId = await insertOne(this.prisma, 'Entity', document);
124
+ const created = await findOne(this.prisma, 'Entity', { _id: insertedId });
125
+ const credentialObj = await this._fetchCredential(created?.credentialId);
126
+ return this._mapEntity(created, credentialObj);
127
+ }
128
+
129
+ async updateEntity(entityId, updates) {
130
+ const objectId = toObjectId(entityId);
131
+ if (!objectId) return null;
132
+
133
+ const existing = await findOne(this.prisma, 'Entity', { _id: objectId });
134
+ if (!existing) return null;
135
+
136
+ const {
137
+ user,
138
+ userId,
139
+ credential,
140
+ credentialId,
141
+ name,
142
+ moduleName,
143
+ externalId,
144
+ ...dynamicData
145
+ } = updates;
146
+
147
+ const updatePayload = {};
148
+ if (user !== undefined || userId !== undefined) {
149
+ updatePayload.userId = toObjectId(userId || user) || null;
150
+ }
151
+ if (credential !== undefined || credentialId !== undefined) {
152
+ updatePayload.credentialId = toObjectId(credentialId || credential) || null;
153
+ }
154
+ if (name !== undefined) updatePayload.name = name;
155
+ if (moduleName !== undefined) updatePayload.moduleName = moduleName;
156
+ if (externalId !== undefined) updatePayload.externalId = externalId;
157
+
158
+ if (Object.keys(dynamicData).length > 0) {
159
+ updatePayload.data = { ...(existing.data || {}), ...dynamicData };
160
+ }
161
+
162
+ await updateOne(
163
+ this.prisma,
164
+ 'Entity',
165
+ { _id: objectId },
166
+ { $set: updatePayload }
167
+ );
168
+ const updated = await findOne(this.prisma, 'Entity', { _id: objectId });
169
+ if (!updated) return null;
170
+ const credentialObj = await this._fetchCredential(updated?.credentialId);
171
+ return this._mapEntity(updated, credentialObj);
172
+ }
173
+
174
+ async deleteEntity(entityId) {
175
+ const objectId = toObjectId(entityId);
176
+ if (!objectId) return false;
177
+ const result = await deleteOne(this.prisma, 'Entity', { _id: objectId });
178
+ const deleted = result?.n ?? 0;
179
+ return deleted > 0;
180
+ }
181
+
182
+ async _fetchCredential(credentialId) {
183
+ const id = fromObjectId(credentialId);
184
+ if (!id) return null;
185
+
186
+ try {
187
+ // Convert to ObjectId for raw query
188
+ const objectId = toObjectId(id);
189
+ if (!objectId) return null;
190
+
191
+ // Use raw findOne to bypass Prisma encryption extension
192
+ const rawCredential = await findOne(this.prisma, 'Credential', {
193
+ _id: objectId
194
+ });
195
+
196
+ if (!rawCredential) return null;
197
+
198
+ // Decrypt sensitive fields using service
199
+ const decryptedCredential = await this.encryptionService.decryptFields('Credential', rawCredential);
200
+
201
+ // Return in same format
202
+ const credential = {
203
+ id: fromObjectId(decryptedCredential._id),
204
+ userId: fromObjectId(decryptedCredential.userId),
205
+ externalId: decryptedCredential.externalId ?? null,
206
+ authIsValid: decryptedCredential.authIsValid ?? null,
207
+ createdAt: decryptedCredential.createdAt,
208
+ updatedAt: decryptedCredential.updatedAt,
209
+ data: decryptedCredential.data
210
+ };
211
+
212
+ return this._convertCredentialIds(credential);
213
+ } catch (error) {
214
+ console.error(`Failed to fetch/decrypt credential ${id}:`, error.message);
215
+ // Return null instead of throwing to allow graceful degradation
216
+ // This repository is read-only (doesn't create/update credentials)
217
+ // Entities can still be loaded even if their credential is corrupted/unreadable
218
+ // The entity will have null credential, which calling code must handle
219
+ // This is intentional behavior: prefer partial data over complete failure
220
+ return null;
221
+ }
222
+ }
223
+
224
+ async _fetchCredentialsBulk(credentialIds) {
225
+ const ids = (credentialIds || [])
226
+ .map((value) => fromObjectId(value))
227
+ .filter((value) => value !== null && value !== undefined);
228
+ if (ids.length === 0) return new Map();
229
+
230
+ try {
231
+ // Convert string IDs to ObjectIds for bulk query
232
+ const objectIds = ids.map(id => toObjectId(id)).filter(Boolean);
233
+ if (objectIds.length === 0) return new Map();
234
+
235
+ // Use raw findMany to bypass Prisma encryption extension
236
+ const rawCredentials = await findMany(this.prisma, 'Credential', {
237
+ _id: { $in: objectIds }
238
+ });
239
+
240
+ // Decrypt all credentials in parallel
241
+ const decryptionPromises = rawCredentials.map(async (rawCredential) => {
242
+ try {
243
+ // Decrypt sensitive fields using service
244
+ const decryptedCredential = await this.encryptionService.decryptFields('Credential', rawCredential);
245
+
246
+ // Build credential object in same format as Prisma would return
247
+ const credential = {
248
+ id: fromObjectId(decryptedCredential._id),
249
+ userId: fromObjectId(decryptedCredential.userId),
250
+ externalId: decryptedCredential.externalId ?? null,
251
+ authIsValid: decryptedCredential.authIsValid ?? null,
252
+ createdAt: decryptedCredential.createdAt,
253
+ updatedAt: decryptedCredential.updatedAt,
254
+ data: decryptedCredential.data
255
+ };
256
+
257
+ return this._convertCredentialIds(credential);
258
+ } catch (error) {
259
+ const credId = fromObjectId(rawCredential._id);
260
+ console.error(`Failed to decrypt credential ${credId}:`, error.message);
261
+ return null;
262
+ }
263
+ });
264
+
265
+ // Wait for all decryptions to complete
266
+ const decryptedCredentials = await Promise.all(decryptionPromises);
267
+
268
+ // Build Map from results, filtering out nulls
269
+ const map = new Map();
270
+ decryptedCredentials.forEach(credential => {
271
+ if (credential) {
272
+ map.set(credential.id, credential);
273
+ }
274
+ });
275
+
276
+ return map;
277
+ } catch (error) {
278
+ console.error('Failed to fetch credentials bulk:', error.message);
279
+ return new Map();
280
+ }
281
+ }
282
+
283
+ /**
284
+ * Convert credential object IDs to strings for application layer
285
+ * Ensures consistent credential format across database adapters
286
+ * @private
287
+ * @param {Object|null} credential - Credential object from database
288
+ * @returns {Object|null} Credential with properly formatted IDs
289
+ */
290
+ _convertCredentialIds(credential) {
291
+ if (!credential) return credential;
292
+ return {
293
+ ...credential,
294
+ id: credential.id ? String(credential.id) : null,
295
+ userId: credential.userId ? String(credential.userId) : null,
296
+ };
297
+ }
298
+
299
+ _buildFilter(filter) {
300
+ const query = {};
301
+ if (!filter) return query;
302
+ if (filter._id || filter.id) {
303
+ const idObj = toObjectId(filter._id || filter.id);
304
+ if (idObj) query._id = idObj;
305
+ }
306
+ if (filter.user || filter.userId) {
307
+ const userObj = toObjectId(filter.user || filter.userId);
308
+ if (userObj) query.userId = userObj;
309
+ }
310
+ if (filter.credential || filter.credentialId) {
311
+ const credObj = toObjectId(filter.credential || filter.credentialId);
312
+ if (credObj) query.credentialId = credObj;
313
+ }
314
+ if (filter.name) query.name = filter.name;
315
+ if (filter.moduleName) query.moduleName = filter.moduleName;
316
+ if (filter.externalId) query.externalId = filter.externalId;
317
+ return query;
318
+ }
319
+
320
+ _mapEntity(doc, credential) {
321
+ const dynamicData = doc?.data || {};
322
+ return {
323
+ id: fromObjectId(doc?._id),
324
+ credential,
325
+ userId: fromObjectId(doc?.userId),
326
+ name: doc?.name ?? null,
327
+ externalId: doc?.externalId ?? null,
328
+ moduleName: doc?.moduleName ?? null,
329
+ ...dynamicData,
330
+ };
331
+ }
332
+ }
333
+
334
+ module.exports = { ModuleRepositoryDocumentDB };
335
+
@@ -0,0 +1,40 @@
1
+ const { ModuleRepositoryMongo } = require('./module-repository-mongo');
2
+ const { ModuleRepositoryPostgres } = require('./module-repository-postgres');
3
+ const {
4
+ ModuleRepositoryDocumentDB,
5
+ } = require('./module-repository-documentdb');
6
+ const config = require('../../database/config');
7
+
8
+ /**
9
+ * Module Repository Factory
10
+ * Creates the appropriate repository adapter based on database type
11
+ *
12
+ * @returns {ModuleRepositoryInterface} Configured repository adapter
13
+ */
14
+ function createModuleRepository() {
15
+ const dbType = config.DB_TYPE;
16
+
17
+ switch (dbType) {
18
+ case 'mongodb':
19
+ return new ModuleRepositoryMongo();
20
+
21
+ case 'postgresql':
22
+ return new ModuleRepositoryPostgres();
23
+
24
+ case 'documentdb':
25
+ return new ModuleRepositoryDocumentDB();
26
+
27
+ default:
28
+ throw new Error(
29
+ `Unsupported database type: ${dbType}. Supported values: 'mongodb', 'documentdb', 'postgresql'`
30
+ );
31
+ }
32
+ }
33
+
34
+ module.exports = {
35
+ createModuleRepository,
36
+ // Export adapters for direct testing
37
+ ModuleRepositoryMongo,
38
+ ModuleRepositoryPostgres,
39
+ ModuleRepositoryDocumentDB,
40
+ };