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

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (267) hide show
  1. package/CLAUDE.md +693 -0
  2. package/README.md +959 -50
  3. package/application/commands/README.md +421 -0
  4. package/application/commands/credential-commands.js +224 -0
  5. package/application/commands/entity-commands.js +315 -0
  6. package/application/commands/integration-commands.js +179 -0
  7. package/application/commands/user-commands.js +213 -0
  8. package/application/index.js +69 -0
  9. package/core/CLAUDE.md +690 -0
  10. package/core/Worker.js +8 -21
  11. package/core/create-handler.js +2 -7
  12. package/credential/repositories/credential-repository-factory.js +47 -0
  13. package/credential/repositories/credential-repository-interface.js +98 -0
  14. package/credential/repositories/credential-repository-mongo.js +307 -0
  15. package/credential/repositories/credential-repository-postgres.js +313 -0
  16. package/credential/repositories/credential-repository.js +302 -0
  17. package/credential/use-cases/get-credential-for-user.js +21 -0
  18. package/credential/use-cases/update-authentication-status.js +15 -0
  19. package/database/MONGODB_TRANSACTION_FIX.md +198 -0
  20. package/database/adapters/lambda-invoker.js +97 -0
  21. package/database/config.js +154 -0
  22. package/database/encryption/README.md +684 -0
  23. package/database/encryption/encryption-schema-registry.js +141 -0
  24. package/database/encryption/field-encryption-service.js +226 -0
  25. package/database/encryption/logger.js +79 -0
  26. package/database/encryption/prisma-encryption-extension.js +222 -0
  27. package/database/index.js +25 -12
  28. package/database/models/WebsocketConnection.js +16 -10
  29. package/database/models/readme.md +1 -0
  30. package/database/prisma.js +222 -0
  31. package/database/repositories/health-check-repository-factory.js +43 -0
  32. package/database/repositories/health-check-repository-interface.js +87 -0
  33. package/database/repositories/health-check-repository-mongodb.js +91 -0
  34. package/database/repositories/health-check-repository-postgres.js +82 -0
  35. package/database/repositories/health-check-repository.js +108 -0
  36. package/database/repositories/migration-status-repository-s3.js +137 -0
  37. package/database/use-cases/check-database-health-use-case.js +29 -0
  38. package/database/use-cases/check-database-state-use-case.js +81 -0
  39. package/database/use-cases/check-encryption-health-use-case.js +83 -0
  40. package/database/use-cases/get-database-state-via-worker-use-case.js +61 -0
  41. package/database/use-cases/get-migration-status-use-case.js +93 -0
  42. package/database/use-cases/run-database-migration-use-case.js +137 -0
  43. package/database/use-cases/test-encryption-use-case.js +253 -0
  44. package/database/use-cases/trigger-database-migration-use-case.js +157 -0
  45. package/database/utils/mongodb-collection-utils.js +91 -0
  46. package/database/utils/mongodb-schema-init.js +106 -0
  47. package/database/utils/prisma-runner.js +400 -0
  48. package/database/utils/prisma-schema-parser.js +182 -0
  49. package/docs/PROCESS_MANAGEMENT_QUEUE_SPEC.md +517 -0
  50. package/encrypt/Cryptor.js +34 -168
  51. package/encrypt/index.js +1 -2
  52. package/encrypt/test-encrypt.js +0 -2
  53. package/generated/prisma-mongodb/client.d.ts +1 -0
  54. package/generated/prisma-mongodb/client.js +4 -0
  55. package/generated/prisma-mongodb/default.d.ts +1 -0
  56. package/generated/prisma-mongodb/default.js +4 -0
  57. package/generated/prisma-mongodb/edge.d.ts +1 -0
  58. package/generated/prisma-mongodb/edge.js +334 -0
  59. package/generated/prisma-mongodb/index-browser.js +316 -0
  60. package/generated/prisma-mongodb/index.d.ts +22898 -0
  61. package/generated/prisma-mongodb/index.js +359 -0
  62. package/generated/prisma-mongodb/package.json +183 -0
  63. package/generated/prisma-mongodb/query-engine-debian-openssl-3.0.x +0 -0
  64. package/generated/prisma-mongodb/query-engine-rhel-openssl-3.0.x +0 -0
  65. package/generated/prisma-mongodb/runtime/binary.d.ts +1 -0
  66. package/generated/prisma-mongodb/runtime/binary.js +289 -0
  67. package/generated/prisma-mongodb/runtime/edge-esm.js +34 -0
  68. package/generated/prisma-mongodb/runtime/edge.js +34 -0
  69. package/generated/prisma-mongodb/runtime/index-browser.d.ts +370 -0
  70. package/generated/prisma-mongodb/runtime/index-browser.js +16 -0
  71. package/generated/prisma-mongodb/runtime/library.d.ts +3982 -0
  72. package/generated/prisma-mongodb/runtime/react-native.js +83 -0
  73. package/generated/prisma-mongodb/runtime/wasm-compiler-edge.js +84 -0
  74. package/generated/prisma-mongodb/runtime/wasm-engine-edge.js +36 -0
  75. package/generated/prisma-mongodb/schema.prisma +362 -0
  76. package/generated/prisma-mongodb/wasm-edge-light-loader.mjs +4 -0
  77. package/generated/prisma-mongodb/wasm-worker-loader.mjs +4 -0
  78. package/generated/prisma-mongodb/wasm.d.ts +1 -0
  79. package/generated/prisma-mongodb/wasm.js +341 -0
  80. package/generated/prisma-postgresql/client.d.ts +1 -0
  81. package/generated/prisma-postgresql/client.js +4 -0
  82. package/generated/prisma-postgresql/default.d.ts +1 -0
  83. package/generated/prisma-postgresql/default.js +4 -0
  84. package/generated/prisma-postgresql/edge.d.ts +1 -0
  85. package/generated/prisma-postgresql/edge.js +356 -0
  86. package/generated/prisma-postgresql/index-browser.js +338 -0
  87. package/generated/prisma-postgresql/index.d.ts +25072 -0
  88. package/generated/prisma-postgresql/index.js +381 -0
  89. package/generated/prisma-postgresql/package.json +183 -0
  90. package/generated/prisma-postgresql/query-engine-debian-openssl-3.0.x +0 -0
  91. package/generated/prisma-postgresql/query-engine-rhel-openssl-3.0.x +0 -0
  92. package/generated/prisma-postgresql/query_engine_bg.js +2 -0
  93. package/generated/prisma-postgresql/query_engine_bg.wasm +0 -0
  94. package/generated/prisma-postgresql/runtime/binary.d.ts +1 -0
  95. package/generated/prisma-postgresql/runtime/binary.js +289 -0
  96. package/generated/prisma-postgresql/runtime/edge-esm.js +34 -0
  97. package/generated/prisma-postgresql/runtime/edge.js +34 -0
  98. package/generated/prisma-postgresql/runtime/index-browser.d.ts +370 -0
  99. package/generated/prisma-postgresql/runtime/index-browser.js +16 -0
  100. package/generated/prisma-postgresql/runtime/library.d.ts +3982 -0
  101. package/generated/prisma-postgresql/runtime/react-native.js +83 -0
  102. package/generated/prisma-postgresql/runtime/wasm-compiler-edge.js +84 -0
  103. package/generated/prisma-postgresql/runtime/wasm-engine-edge.js +36 -0
  104. package/generated/prisma-postgresql/schema.prisma +345 -0
  105. package/generated/prisma-postgresql/wasm-edge-light-loader.mjs +4 -0
  106. package/generated/prisma-postgresql/wasm-worker-loader.mjs +4 -0
  107. package/generated/prisma-postgresql/wasm.d.ts +1 -0
  108. package/generated/prisma-postgresql/wasm.js +363 -0
  109. package/handlers/WEBHOOKS.md +653 -0
  110. package/handlers/app-definition-loader.js +38 -0
  111. package/handlers/app-handler-helpers.js +56 -0
  112. package/handlers/backend-utils.js +180 -0
  113. package/handlers/database-migration-handler.js +227 -0
  114. package/handlers/integration-event-dispatcher.js +54 -0
  115. package/handlers/routers/HEALTHCHECK.md +342 -0
  116. package/handlers/routers/auth.js +15 -0
  117. package/handlers/routers/db-migration.handler.js +29 -0
  118. package/handlers/routers/db-migration.js +256 -0
  119. package/handlers/routers/health.js +519 -0
  120. package/handlers/routers/integration-defined-routers.js +45 -0
  121. package/handlers/routers/integration-webhook-routers.js +67 -0
  122. package/handlers/routers/user.js +63 -0
  123. package/handlers/routers/websocket.js +57 -0
  124. package/handlers/use-cases/check-external-apis-health-use-case.js +81 -0
  125. package/handlers/use-cases/check-integrations-health-use-case.js +44 -0
  126. package/handlers/workers/db-migration.js +352 -0
  127. package/handlers/workers/integration-defined-workers.js +27 -0
  128. package/index.js +77 -22
  129. package/integrations/WEBHOOK-QUICKSTART.md +151 -0
  130. package/integrations/index.js +12 -10
  131. package/integrations/integration-base.js +296 -54
  132. package/integrations/integration-router.js +381 -182
  133. package/integrations/options.js +1 -1
  134. package/integrations/repositories/integration-mapping-repository-factory.js +50 -0
  135. package/integrations/repositories/integration-mapping-repository-interface.js +106 -0
  136. package/integrations/repositories/integration-mapping-repository-mongo.js +161 -0
  137. package/integrations/repositories/integration-mapping-repository-postgres.js +227 -0
  138. package/integrations/repositories/integration-mapping-repository.js +156 -0
  139. package/integrations/repositories/integration-repository-factory.js +44 -0
  140. package/integrations/repositories/integration-repository-interface.js +127 -0
  141. package/integrations/repositories/integration-repository-mongo.js +303 -0
  142. package/integrations/repositories/integration-repository-postgres.js +352 -0
  143. package/integrations/repositories/process-repository-factory.js +46 -0
  144. package/integrations/repositories/process-repository-interface.js +90 -0
  145. package/integrations/repositories/process-repository-mongo.js +190 -0
  146. package/integrations/repositories/process-repository-postgres.js +217 -0
  147. package/integrations/tests/doubles/dummy-integration-class.js +83 -0
  148. package/integrations/tests/doubles/test-integration-repository.js +99 -0
  149. package/integrations/use-cases/create-integration.js +83 -0
  150. package/integrations/use-cases/create-process.js +128 -0
  151. package/integrations/use-cases/delete-integration-for-user.js +101 -0
  152. package/integrations/use-cases/find-integration-context-by-external-entity-id.js +72 -0
  153. package/integrations/use-cases/get-integration-for-user.js +78 -0
  154. package/integrations/use-cases/get-integration-instance-by-definition.js +67 -0
  155. package/integrations/use-cases/get-integration-instance.js +83 -0
  156. package/integrations/use-cases/get-integrations-for-user.js +88 -0
  157. package/integrations/use-cases/get-possible-integrations.js +27 -0
  158. package/integrations/use-cases/get-process.js +87 -0
  159. package/integrations/use-cases/index.js +19 -0
  160. package/integrations/use-cases/load-integration-context.js +71 -0
  161. package/integrations/use-cases/update-integration-messages.js +44 -0
  162. package/integrations/use-cases/update-integration-status.js +32 -0
  163. package/integrations/use-cases/update-integration.js +93 -0
  164. package/integrations/use-cases/update-process-metrics.js +201 -0
  165. package/integrations/use-cases/update-process-state.js +119 -0
  166. package/integrations/utils/map-integration-dto.js +37 -0
  167. package/jest-global-setup-noop.js +3 -0
  168. package/jest-global-teardown-noop.js +3 -0
  169. package/logs/logger.js +0 -4
  170. package/{module-plugin → modules}/entity.js +1 -1
  171. package/{module-plugin → modules}/index.js +0 -8
  172. package/modules/module-factory.js +56 -0
  173. package/modules/module.js +221 -0
  174. package/modules/repositories/module-repository-factory.js +33 -0
  175. package/modules/repositories/module-repository-interface.js +129 -0
  176. package/modules/repositories/module-repository-mongo.js +377 -0
  177. package/modules/repositories/module-repository-postgres.js +426 -0
  178. package/modules/repositories/module-repository.js +316 -0
  179. package/{module-plugin → modules}/requester/requester.js +1 -0
  180. package/{module-plugin → modules}/test/mock-api/api.js +8 -3
  181. package/{module-plugin → modules}/test/mock-api/definition.js +12 -8
  182. package/modules/tests/doubles/test-module-factory.js +16 -0
  183. package/modules/tests/doubles/test-module-repository.js +39 -0
  184. package/modules/use-cases/get-entities-for-user.js +32 -0
  185. package/modules/use-cases/get-entity-options-by-id.js +59 -0
  186. package/modules/use-cases/get-entity-options-by-type.js +34 -0
  187. package/modules/use-cases/get-module-instance-from-type.js +31 -0
  188. package/modules/use-cases/get-module.js +55 -0
  189. package/modules/use-cases/process-authorization-callback.js +122 -0
  190. package/modules/use-cases/refresh-entity-options.js +59 -0
  191. package/modules/use-cases/test-module-auth.js +55 -0
  192. package/modules/utils/map-module-dto.js +18 -0
  193. package/package.json +82 -50
  194. package/prisma-mongodb/schema.prisma +362 -0
  195. package/prisma-postgresql/migrations/20250930193005_init/migration.sql +315 -0
  196. package/prisma-postgresql/migrations/20251006135218_init/migration.sql +9 -0
  197. package/prisma-postgresql/migrations/20251010000000_remove_unused_entity_reference_map/migration.sql +3 -0
  198. package/prisma-postgresql/migrations/migration_lock.toml +3 -0
  199. package/prisma-postgresql/schema.prisma +345 -0
  200. package/queues/queuer-util.js +28 -15
  201. package/syncs/manager.js +468 -443
  202. package/syncs/repositories/sync-repository-factory.js +38 -0
  203. package/syncs/repositories/sync-repository-interface.js +109 -0
  204. package/syncs/repositories/sync-repository-mongo.js +239 -0
  205. package/syncs/repositories/sync-repository-postgres.js +319 -0
  206. package/syncs/sync.js +0 -1
  207. package/token/repositories/token-repository-factory.js +33 -0
  208. package/token/repositories/token-repository-interface.js +131 -0
  209. package/token/repositories/token-repository-mongo.js +212 -0
  210. package/token/repositories/token-repository-postgres.js +257 -0
  211. package/token/repositories/token-repository.js +219 -0
  212. package/types/core/index.d.ts +2 -2
  213. package/types/integrations/index.d.ts +2 -6
  214. package/types/module-plugin/index.d.ts +5 -59
  215. package/types/syncs/index.d.ts +0 -2
  216. package/user/repositories/user-repository-factory.js +46 -0
  217. package/user/repositories/user-repository-interface.js +198 -0
  218. package/user/repositories/user-repository-mongo.js +291 -0
  219. package/user/repositories/user-repository-postgres.js +350 -0
  220. package/user/tests/doubles/test-user-repository.js +72 -0
  221. package/user/use-cases/authenticate-user.js +127 -0
  222. package/user/use-cases/authenticate-with-shared-secret.js +48 -0
  223. package/user/use-cases/create-individual-user.js +61 -0
  224. package/user/use-cases/create-organization-user.js +47 -0
  225. package/user/use-cases/create-token-for-user-id.js +30 -0
  226. package/user/use-cases/get-user-from-adopter-jwt.js +149 -0
  227. package/user/use-cases/get-user-from-bearer-token.js +77 -0
  228. package/user/use-cases/get-user-from-x-frigg-headers.js +106 -0
  229. package/user/use-cases/login-user.js +122 -0
  230. package/user/user.js +93 -0
  231. package/utils/backend-path.js +38 -0
  232. package/utils/index.js +6 -0
  233. package/websocket/repositories/websocket-connection-repository-factory.js +37 -0
  234. package/websocket/repositories/websocket-connection-repository-interface.js +106 -0
  235. package/websocket/repositories/websocket-connection-repository-mongo.js +156 -0
  236. package/websocket/repositories/websocket-connection-repository-postgres.js +196 -0
  237. package/websocket/repositories/websocket-connection-repository.js +161 -0
  238. package/database/models/State.js +0 -9
  239. package/database/models/Token.js +0 -70
  240. package/database/mongo.js +0 -45
  241. package/encrypt/Cryptor.test.js +0 -32
  242. package/encrypt/encrypt.js +0 -132
  243. package/encrypt/encrypt.test.js +0 -1069
  244. package/errors/base-error.test.js +0 -32
  245. package/errors/fetch-error.test.js +0 -79
  246. package/errors/halt-error.test.js +0 -11
  247. package/errors/validation-errors.test.js +0 -120
  248. package/integrations/create-frigg-backend.js +0 -31
  249. package/integrations/integration-factory.js +0 -251
  250. package/integrations/integration-mapping.js +0 -43
  251. package/integrations/integration-model.js +0 -46
  252. package/integrations/integration-user.js +0 -144
  253. package/integrations/test/integration-base.test.js +0 -144
  254. package/lambda/TimeoutCatcher.test.js +0 -68
  255. package/logs/logger.test.js +0 -76
  256. package/module-plugin/auther.js +0 -393
  257. package/module-plugin/credential.js +0 -22
  258. package/module-plugin/entity-manager.js +0 -70
  259. package/module-plugin/manager.js +0 -169
  260. package/module-plugin/module-factory.js +0 -61
  261. package/module-plugin/requester/requester.test.js +0 -28
  262. package/module-plugin/test/auther.test.js +0 -97
  263. /package/{module-plugin → modules}/ModuleConstants.js +0 -0
  264. /package/{module-plugin → modules}/requester/api-key.js +0 -0
  265. /package/{module-plugin → modules}/requester/basic.js +0 -0
  266. /package/{module-plugin → modules}/requester/oauth-2.js +0 -0
  267. /package/{module-plugin → modules}/test/mock-api/mocks/hubspot.js +0 -0
@@ -0,0 +1,122 @@
1
+ const Boom = require('@hapi/boom');
2
+ const {
3
+ RequiredPropertyError,
4
+ } = require('../../errors');
5
+ const { User } = require('../user');
6
+
7
+ /**
8
+ * Use case for logging in a user.
9
+ * @class LoginUser
10
+ */
11
+ class LoginUser {
12
+ /**
13
+ * Creates a new LoginUser instance.
14
+ * @param {Object} params - Configuration parameters.
15
+ * @param {import('../user-repository-interface').UserRepositoryInterface} params.userRepository - Repository for user data operations.
16
+ * @param {Object} params.userConfig - The user properties inside of the app definition.
17
+ */
18
+ constructor({ userRepository, userConfig }) {
19
+ this.userRepository = userRepository;
20
+ this.userConfig = userConfig;
21
+ }
22
+
23
+ /**
24
+ * Executes the use case.
25
+ * @async
26
+ * @param {Object} userCredentials - The user's credentials for authentication.
27
+ * @param {string} [userCredentials.username] - The username for authentication.
28
+ * @param {string} [userCredentials.password] - The password for authentication.
29
+ * @param {string} [userCredentials.appUserId] - The app user id for authentication if no username and password are provided.
30
+ * @param {string} [userCredentials.appOrgId] - The app organization id for authentication if no username and password are provided.
31
+ * @returns {Promise<import('../user').User>} The authenticated user object.
32
+ */
33
+ async execute(userCredentials) {
34
+ const { username, password, appUserId, appOrgId } = userCredentials;
35
+ if (this.userConfig.individualUserRequired) {
36
+ if (this.userConfig.usePassword) {
37
+ if (!username) {
38
+ throw new RequiredPropertyError({
39
+ parent: this,
40
+ key: 'username',
41
+ });
42
+ }
43
+ if (!password) {
44
+ throw new RequiredPropertyError({
45
+ parent: this,
46
+ key: 'password',
47
+ });
48
+ }
49
+
50
+ const individualUserData =
51
+ await this.userRepository.findIndividualUserByUsername(
52
+ username
53
+ );
54
+
55
+ if (!individualUserData) {
56
+ throw Boom.unauthorized('user not found');
57
+ }
58
+
59
+ const individualUser = new User(
60
+ individualUserData,
61
+ null,
62
+ this.userConfig.usePassword,
63
+ this.userConfig.primary,
64
+ this.userConfig.individualUserRequired,
65
+ this.userConfig.organizationUserRequired
66
+ );
67
+
68
+ if (!(await individualUser.isPasswordValid(password))) {
69
+ throw Boom.unauthorized('Incorrect username or password');
70
+ }
71
+
72
+ return individualUser;
73
+ } else {
74
+ const individualUserData =
75
+ await this.userRepository.findIndividualUserByAppUserId(
76
+ appUserId
77
+ );
78
+
79
+ if (!individualUserData) {
80
+ throw Boom.unauthorized('user not found');
81
+ }
82
+
83
+ const individualUser = new User(
84
+ individualUserData,
85
+ null,
86
+ this.userConfig.usePassword,
87
+ this.userConfig.primary,
88
+ this.userConfig.individualUserRequired,
89
+ this.userConfig.organizationUserRequired
90
+ );
91
+
92
+ return individualUser;
93
+ }
94
+ }
95
+
96
+
97
+ if (this.userConfig.organizationUserRequired) {
98
+
99
+ const organizationUserData =
100
+ await this.userRepository.findOrganizationUserByAppOrgId(appOrgId);
101
+
102
+ if (!organizationUserData) {
103
+ throw Boom.unauthorized(`org user ${appOrgId} not found`);
104
+ }
105
+
106
+ const organizationUser = new User(
107
+ null,
108
+ organizationUserData,
109
+ this.userConfig.usePassword,
110
+ this.userConfig.primary,
111
+ this.userConfig.individualUserRequired,
112
+ this.userConfig.organizationUserRequired
113
+ );
114
+
115
+ return organizationUser;
116
+ }
117
+
118
+ return null;
119
+ }
120
+ }
121
+
122
+ module.exports = { LoginUser };
package/user/user.js ADDED
@@ -0,0 +1,93 @@
1
+ const bcrypt = require('bcryptjs');
2
+
3
+ /**
4
+ * Represents a user in the system. The User class is a domain entity,
5
+ * @class User
6
+ */
7
+ class User {
8
+ /**
9
+ * Creates a new User instance.
10
+ * @param {import('../database/models/IndividualUser').IndividualUser} [individualUser=null] - The individual user for the user.
11
+ * @param {import('../database/models/OrganizationUser').OrganizationUser} [organizationUser=null] - The organization user for the user.
12
+ * @param {boolean} [usePassword=false] - Whether the user has a password.
13
+ * @param {string} [primary='individual'] - The primary user type.
14
+ * @param {boolean} [individualUserRequired=true] - Whether the user is required to have an individual user.
15
+ * @param {boolean} [organizationUserRequired=false] - Whether the user is required to have an organization user.
16
+ */
17
+ constructor(individualUser = null, organizationUser = null, usePassword = false, primary = 'individual', individualUserRequired = true, organizationUserRequired = false) {
18
+ this.individualUser = individualUser;
19
+ this.organizationUser = organizationUser;
20
+ this.usePassword = usePassword;
21
+
22
+ this.config = {
23
+ primary,
24
+ individualUserRequired,
25
+ organizationUserRequired,
26
+ };
27
+ }
28
+
29
+ getPrimaryUser() {
30
+ if (this.config.primary === 'organization') {
31
+ return this.organizationUser;
32
+ }
33
+ return this.individualUser;
34
+ }
35
+
36
+ getId() {
37
+ return this.getPrimaryUser()?.id;
38
+ }
39
+
40
+ isPasswordRequired() {
41
+ return this.usePassword;
42
+ }
43
+
44
+ async isPasswordValid(password) {
45
+ if (!this.isPasswordRequired()) {
46
+ return true;
47
+ }
48
+
49
+ return await bcrypt.compare(password, this.getPrimaryUser().hashword);
50
+ }
51
+
52
+ setIndividualUser(individualUser) {
53
+ this.individualUser = individualUser;
54
+ }
55
+
56
+ setOrganizationUser(organizationUser) {
57
+ this.organizationUser = organizationUser;
58
+ }
59
+
60
+ isOrganizationUserRequired() {
61
+ return this.config.organizationUserRequired;
62
+ }
63
+
64
+ isIndividualUserRequired() {
65
+ return this.config.individualUserRequired;
66
+ }
67
+
68
+ getIndividualUser() {
69
+ return this.individualUser;
70
+ }
71
+
72
+ getOrganizationUser() {
73
+ return this.organizationUser;
74
+ }
75
+
76
+ /**
77
+ * Gets the appUserId from the individual user if present.
78
+ * @returns {string|null} The app user ID or null
79
+ */
80
+ getAppUserId() {
81
+ return this.individualUser?.appUserId || null;
82
+ }
83
+
84
+ /**
85
+ * Gets the appOrgId from the organization user if present.
86
+ * @returns {string|null} The app organization ID or null
87
+ */
88
+ getAppOrgId() {
89
+ return this.organizationUser?.appOrgId || null;
90
+ }
91
+ }
92
+
93
+ module.exports = { User };
@@ -0,0 +1,38 @@
1
+ const fs = require('fs-extra');
2
+ const path = require('node:path');
3
+ const PACKAGE_JSON = 'package.json';
4
+
5
+ function findNearestBackendPackageJson() {
6
+ let currentDir = process.cwd();
7
+
8
+ // First check if we're in production by looking for package.json in the current directory
9
+ const rootPackageJson = path.join(currentDir, PACKAGE_JSON);
10
+ if (fs.existsSync(rootPackageJson)) {
11
+ // In production environment, check for index.js in the same directory
12
+ const indexJs = path.join(currentDir, 'index.js');
13
+ if (fs.existsSync(indexJs)) {
14
+ return rootPackageJson;
15
+ }
16
+ }
17
+
18
+ // If not found at root or not in production, look for it in the backend directory
19
+ while (currentDir !== path.parse(currentDir).root) {
20
+ const packageJsonPath = path.join(currentDir, 'backend', PACKAGE_JSON);
21
+ if (fs.existsSync(packageJsonPath)) {
22
+ return packageJsonPath;
23
+ }
24
+ currentDir = path.dirname(currentDir);
25
+ }
26
+ return null;
27
+ }
28
+
29
+ function validateBackendPath(backendPath) {
30
+ if (!backendPath) {
31
+ throw new Error('Could not find a backend package.json file.');
32
+ }
33
+ }
34
+
35
+ module.exports = {
36
+ findNearestBackendPackageJson,
37
+ validateBackendPath,
38
+ };
package/utils/index.js ADDED
@@ -0,0 +1,6 @@
1
+ const { findNearestBackendPackageJson, validateBackendPath } = require('./backend-path');
2
+
3
+ module.exports = {
4
+ findNearestBackendPackageJson,
5
+ validateBackendPath,
6
+ };
@@ -0,0 +1,37 @@
1
+ const {
2
+ WebsocketConnectionRepositoryMongo,
3
+ } = require('./websocket-connection-repository-mongo');
4
+ const {
5
+ WebsocketConnectionRepositoryPostgres,
6
+ } = require('./websocket-connection-repository-postgres');
7
+ const config = require('../../database/config');
8
+
9
+ /**
10
+ * Websocket Connection Repository Factory
11
+ * Creates the appropriate repository adapter based on database type
12
+ *
13
+ * @returns {WebsocketConnectionRepositoryInterface} Configured repository adapter
14
+ */
15
+ function createWebsocketConnectionRepository() {
16
+ const dbType = config.DB_TYPE;
17
+
18
+ switch (dbType) {
19
+ case 'mongodb':
20
+ return new WebsocketConnectionRepositoryMongo();
21
+
22
+ case 'postgresql':
23
+ return new WebsocketConnectionRepositoryPostgres();
24
+
25
+ default:
26
+ throw new Error(
27
+ `Unsupported database type: ${dbType}. Supported values: 'mongodb', 'postgresql'`
28
+ );
29
+ }
30
+ }
31
+
32
+ module.exports = {
33
+ createWebsocketConnectionRepository,
34
+ // Export adapters for direct testing
35
+ WebsocketConnectionRepositoryMongo,
36
+ WebsocketConnectionRepositoryPostgres,
37
+ };
@@ -0,0 +1,106 @@
1
+ /**
2
+ * Websocket Connection Repository Interface
3
+ * Abstract base class defining the contract for websocket connection 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, WebsocketConnection model has identical structure across MongoDB and PostgreSQL,
11
+ * so WebsocketConnectionRepository serves both. This interface exists for consistency and
12
+ * future-proofing if database-specific implementations become needed.
13
+ *
14
+ * @abstract
15
+ */
16
+ class WebsocketConnectionRepositoryInterface {
17
+ /**
18
+ * Create a new websocket connection
19
+ *
20
+ * @param {string} connectionId - Connection ID
21
+ * @returns {Promise<Object>} Created connection object
22
+ * @abstract
23
+ */
24
+ async createConnection(connectionId) {
25
+ throw new Error(
26
+ 'Method createConnection must be implemented by subclass'
27
+ );
28
+ }
29
+
30
+ /**
31
+ * Delete a websocket connection
32
+ *
33
+ * @param {string} connectionId - Connection ID
34
+ * @returns {Promise<Object>} Deletion result
35
+ * @abstract
36
+ */
37
+ async deleteConnection(connectionId) {
38
+ throw new Error(
39
+ 'Method deleteConnection must be implemented by subclass'
40
+ );
41
+ }
42
+
43
+ /**
44
+ * Get active connections
45
+ *
46
+ * @returns {Promise<Array>} Array of active connection objects
47
+ * @abstract
48
+ */
49
+ async getActiveConnections() {
50
+ throw new Error(
51
+ 'Method getActiveConnections must be implemented by subclass'
52
+ );
53
+ }
54
+
55
+ /**
56
+ * Find connection by connection ID
57
+ *
58
+ * @param {string} connectionId - Connection ID
59
+ * @returns {Promise<Object|null>} Connection object or null
60
+ * @abstract
61
+ */
62
+ async findConnection(connectionId) {
63
+ throw new Error(
64
+ 'Method findConnection must be implemented by subclass'
65
+ );
66
+ }
67
+
68
+ /**
69
+ * Find connection by database ID
70
+ *
71
+ * @param {string|number} id - Database ID
72
+ * @returns {Promise<Object|null>} Connection object or null
73
+ * @abstract
74
+ */
75
+ async findConnectionById(id) {
76
+ throw new Error(
77
+ 'Method findConnectionById must be implemented by subclass'
78
+ );
79
+ }
80
+
81
+ /**
82
+ * Get all connections
83
+ *
84
+ * @returns {Promise<Array>} Array of all connection objects
85
+ * @abstract
86
+ */
87
+ async getAllConnections() {
88
+ throw new Error(
89
+ 'Method getAllConnections must be implemented by subclass'
90
+ );
91
+ }
92
+
93
+ /**
94
+ * Delete all connections
95
+ *
96
+ * @returns {Promise<Object>} Deletion result
97
+ * @abstract
98
+ */
99
+ async deleteAllConnections() {
100
+ throw new Error(
101
+ 'Method deleteAllConnections must be implemented by subclass'
102
+ );
103
+ }
104
+ }
105
+
106
+ module.exports = { WebsocketConnectionRepositoryInterface };
@@ -0,0 +1,156 @@
1
+ const { prisma } = require('../../database/prisma');
2
+ const {
3
+ ApiGatewayManagementApiClient,
4
+ PostToConnectionCommand,
5
+ } = require('@aws-sdk/client-apigatewaymanagementapi');
6
+ const {
7
+ WebsocketConnectionRepositoryInterface,
8
+ } = require('./websocket-connection-repository-interface');
9
+
10
+ /**
11
+ * MongoDB WebSocket Connection Repository Adapter
12
+ * Handles persistence of active WebSocket connections
13
+ *
14
+ * MongoDB-specific characteristics:
15
+ * - Uses String IDs (ObjectId)
16
+ * - No ID conversion needed (IDs are already strings)
17
+ * - AWS API Gateway Management API integration preserved
18
+ */
19
+ class WebsocketConnectionRepositoryMongo extends WebsocketConnectionRepositoryInterface {
20
+ constructor() {
21
+ super();
22
+ this.prisma = prisma;
23
+ }
24
+
25
+ /**
26
+ * Create a new WebSocket connection record
27
+ * Replaces: WebsocketConnection.create({ connectionId })
28
+ *
29
+ * @param {string} connectionId - The WebSocket connection ID
30
+ * @returns {Promise<Object>} The created connection record with string IDs
31
+ */
32
+ async createConnection(connectionId) {
33
+ return await this.prisma.websocketConnection.create({
34
+ data: { connectionId },
35
+ });
36
+ }
37
+
38
+ /**
39
+ * Delete a WebSocket connection record
40
+ * Replaces: WebsocketConnection.deleteOne({ connectionId })
41
+ *
42
+ * @param {string} connectionId - The WebSocket connection ID to delete
43
+ * @returns {Promise<Object>} The deletion result
44
+ */
45
+ async deleteConnection(connectionId) {
46
+ try {
47
+ await this.prisma.websocketConnection.delete({
48
+ where: { connectionId },
49
+ });
50
+ return { acknowledged: true, deletedCount: 1 };
51
+ } catch (error) {
52
+ if (error.code === 'P2025') {
53
+ // Record not found
54
+ return { acknowledged: true, deletedCount: 0 };
55
+ }
56
+ throw error;
57
+ }
58
+ }
59
+
60
+ /**
61
+ * Get all active WebSocket connections with send capability
62
+ * Replaces: WebsocketConnection.getActiveConnections()
63
+ *
64
+ * @returns {Promise<Array>} Array of active connection objects with send capability
65
+ */
66
+ async getActiveConnections() {
67
+ try {
68
+ // Return empty array if websockets are not configured
69
+ if (!process.env.WEBSOCKET_API_ENDPOINT) {
70
+ return [];
71
+ }
72
+
73
+ const connections = await this.prisma.websocketConnection.findMany({
74
+ select: { connectionId: true },
75
+ });
76
+
77
+ return connections.map((conn) => ({
78
+ connectionId: conn.connectionId,
79
+ send: async (data) => {
80
+ const apigwManagementApi = new ApiGatewayManagementApiClient({
81
+ endpoint: process.env.WEBSOCKET_API_ENDPOINT,
82
+ });
83
+
84
+ try {
85
+ const command = new PostToConnectionCommand({
86
+ ConnectionId: conn.connectionId,
87
+ Data: JSON.stringify(data),
88
+ });
89
+ await apigwManagementApi.send(command);
90
+ } catch (error) {
91
+ if (error.statusCode === 410 || error.$metadata?.httpStatusCode === 410) {
92
+ console.log(
93
+ `Stale connection ${conn.connectionId}`
94
+ );
95
+ // Delete stale connection
96
+ await this.prisma.websocketConnection.deleteMany({
97
+ where: { connectionId: conn.connectionId },
98
+ });
99
+ } else {
100
+ throw error;
101
+ }
102
+ }
103
+ },
104
+ }));
105
+ } catch (error) {
106
+ console.error('Error getting active connections:', error);
107
+ throw error;
108
+ }
109
+ }
110
+
111
+ /**
112
+ * Find a connection by connection ID
113
+ * Replaces: WebsocketConnection.findOne({ connectionId })
114
+ *
115
+ * @param {string} connectionId - The WebSocket connection ID
116
+ * @returns {Promise<Object|null>} The connection record with string IDs or null
117
+ */
118
+ async findConnection(connectionId) {
119
+ return await this.prisma.websocketConnection.findFirst({
120
+ where: { connectionId },
121
+ });
122
+ }
123
+
124
+ /**
125
+ * Find connection by internal ID
126
+ * @param {string} id - The internal connection ID
127
+ * @returns {Promise<Object|null>} The connection record with string IDs or null
128
+ */
129
+ async findConnectionById(id) {
130
+ return await this.prisma.websocketConnection.findUnique({
131
+ where: { id },
132
+ });
133
+ }
134
+
135
+ /**
136
+ * Get all connections
137
+ * @returns {Promise<Array>} Array of all connection records with string IDs
138
+ */
139
+ async getAllConnections() {
140
+ return await this.prisma.websocketConnection.findMany();
141
+ }
142
+
143
+ /**
144
+ * Delete all connections
145
+ * @returns {Promise<Object>} The deletion result
146
+ */
147
+ async deleteAllConnections() {
148
+ const result = await this.prisma.websocketConnection.deleteMany();
149
+ return {
150
+ acknowledged: true,
151
+ deletedCount: result.count,
152
+ };
153
+ }
154
+ }
155
+
156
+ module.exports = { WebsocketConnectionRepositoryMongo };