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

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (285) hide show
  1. package/CLAUDE.md +694 -0
  2. package/README.md +959 -50
  3. package/application/commands/README.md +451 -0
  4. package/application/commands/credential-commands.js +245 -0
  5. package/application/commands/entity-commands.js +336 -0
  6. package/application/commands/integration-commands.js +210 -0
  7. package/application/commands/user-commands.js +283 -0
  8. package/application/index.js +69 -0
  9. package/core/CLAUDE.md +690 -0
  10. package/core/Worker.js +8 -21
  11. package/core/create-handler.js +14 -7
  12. package/credential/repositories/credential-repository-documentdb.js +304 -0
  13. package/credential/repositories/credential-repository-factory.js +54 -0
  14. package/credential/repositories/credential-repository-interface.js +98 -0
  15. package/credential/repositories/credential-repository-mongo.js +269 -0
  16. package/credential/repositories/credential-repository-postgres.js +291 -0
  17. package/credential/repositories/credential-repository.js +302 -0
  18. package/credential/use-cases/get-credential-for-user.js +25 -0
  19. package/credential/use-cases/update-authentication-status.js +15 -0
  20. package/database/MONGODB_TRANSACTION_FIX.md +198 -0
  21. package/database/adapters/lambda-invoker.js +97 -0
  22. package/database/config.js +154 -0
  23. package/database/documentdb-encryption-service.js +330 -0
  24. package/database/documentdb-utils.js +136 -0
  25. package/database/encryption/README.md +839 -0
  26. package/database/encryption/documentdb-encryption-service.md +3575 -0
  27. package/database/encryption/encryption-schema-registry.js +268 -0
  28. package/database/encryption/field-encryption-service.js +226 -0
  29. package/database/encryption/logger.js +79 -0
  30. package/database/encryption/prisma-encryption-extension.js +222 -0
  31. package/database/index.js +61 -21
  32. package/database/models/WebsocketConnection.js +16 -10
  33. package/database/models/readme.md +1 -0
  34. package/database/prisma.js +182 -0
  35. package/database/repositories/health-check-repository-documentdb.js +134 -0
  36. package/database/repositories/health-check-repository-factory.js +48 -0
  37. package/database/repositories/health-check-repository-interface.js +82 -0
  38. package/database/repositories/health-check-repository-mongodb.js +89 -0
  39. package/database/repositories/health-check-repository-postgres.js +82 -0
  40. package/database/repositories/health-check-repository.js +108 -0
  41. package/database/repositories/migration-status-repository-s3.js +137 -0
  42. package/database/use-cases/check-database-health-use-case.js +29 -0
  43. package/database/use-cases/check-database-state-use-case.js +81 -0
  44. package/database/use-cases/check-encryption-health-use-case.js +83 -0
  45. package/database/use-cases/get-database-state-via-worker-use-case.js +61 -0
  46. package/database/use-cases/get-migration-status-use-case.js +93 -0
  47. package/database/use-cases/run-database-migration-use-case.js +139 -0
  48. package/database/use-cases/test-encryption-use-case.js +253 -0
  49. package/database/use-cases/trigger-database-migration-use-case.js +157 -0
  50. package/database/utils/mongodb-collection-utils.js +91 -0
  51. package/database/utils/mongodb-schema-init.js +106 -0
  52. package/database/utils/prisma-runner.js +477 -0
  53. package/database/utils/prisma-schema-parser.js +182 -0
  54. package/docs/PROCESS_MANAGEMENT_QUEUE_SPEC.md +517 -0
  55. package/encrypt/Cryptor.js +34 -168
  56. package/encrypt/index.js +1 -2
  57. package/encrypt/test-encrypt.js +0 -2
  58. package/errors/client-safe-error.js +26 -0
  59. package/errors/fetch-error.js +2 -1
  60. package/errors/index.js +2 -0
  61. package/generated/prisma-mongodb/client.d.ts +1 -0
  62. package/generated/prisma-mongodb/client.js +4 -0
  63. package/generated/prisma-mongodb/default.d.ts +1 -0
  64. package/generated/prisma-mongodb/default.js +4 -0
  65. package/generated/prisma-mongodb/edge.d.ts +1 -0
  66. package/generated/prisma-mongodb/edge.js +334 -0
  67. package/generated/prisma-mongodb/index-browser.js +316 -0
  68. package/generated/prisma-mongodb/index.d.ts +22903 -0
  69. package/generated/prisma-mongodb/index.js +359 -0
  70. package/generated/prisma-mongodb/package.json +183 -0
  71. package/generated/prisma-mongodb/query-engine-debian-openssl-3.0.x +0 -0
  72. package/generated/prisma-mongodb/query-engine-rhel-openssl-3.0.x +0 -0
  73. package/generated/prisma-mongodb/runtime/binary.d.ts +1 -0
  74. package/generated/prisma-mongodb/runtime/binary.js +289 -0
  75. package/generated/prisma-mongodb/runtime/edge-esm.js +34 -0
  76. package/generated/prisma-mongodb/runtime/edge.js +34 -0
  77. package/generated/prisma-mongodb/runtime/index-browser.d.ts +370 -0
  78. package/generated/prisma-mongodb/runtime/index-browser.js +16 -0
  79. package/generated/prisma-mongodb/runtime/library.d.ts +3977 -0
  80. package/generated/prisma-mongodb/runtime/react-native.js +83 -0
  81. package/generated/prisma-mongodb/runtime/wasm-compiler-edge.js +84 -0
  82. package/generated/prisma-mongodb/runtime/wasm-engine-edge.js +36 -0
  83. package/generated/prisma-mongodb/schema.prisma +360 -0
  84. package/generated/prisma-mongodb/wasm-edge-light-loader.mjs +4 -0
  85. package/generated/prisma-mongodb/wasm-worker-loader.mjs +4 -0
  86. package/generated/prisma-mongodb/wasm.d.ts +1 -0
  87. package/generated/prisma-mongodb/wasm.js +341 -0
  88. package/generated/prisma-postgresql/client.d.ts +1 -0
  89. package/generated/prisma-postgresql/client.js +4 -0
  90. package/generated/prisma-postgresql/default.d.ts +1 -0
  91. package/generated/prisma-postgresql/default.js +4 -0
  92. package/generated/prisma-postgresql/edge.d.ts +1 -0
  93. package/generated/prisma-postgresql/edge.js +356 -0
  94. package/generated/prisma-postgresql/index-browser.js +338 -0
  95. package/generated/prisma-postgresql/index.d.ts +25077 -0
  96. package/generated/prisma-postgresql/index.js +381 -0
  97. package/generated/prisma-postgresql/package.json +183 -0
  98. package/generated/prisma-postgresql/query-engine-debian-openssl-3.0.x +0 -0
  99. package/generated/prisma-postgresql/query-engine-rhel-openssl-3.0.x +0 -0
  100. package/generated/prisma-postgresql/query_engine_bg.js +2 -0
  101. package/generated/prisma-postgresql/query_engine_bg.wasm +0 -0
  102. package/generated/prisma-postgresql/runtime/binary.d.ts +1 -0
  103. package/generated/prisma-postgresql/runtime/binary.js +289 -0
  104. package/generated/prisma-postgresql/runtime/edge-esm.js +34 -0
  105. package/generated/prisma-postgresql/runtime/edge.js +34 -0
  106. package/generated/prisma-postgresql/runtime/index-browser.d.ts +370 -0
  107. package/generated/prisma-postgresql/runtime/index-browser.js +16 -0
  108. package/generated/prisma-postgresql/runtime/library.d.ts +3977 -0
  109. package/generated/prisma-postgresql/runtime/react-native.js +83 -0
  110. package/generated/prisma-postgresql/runtime/wasm-compiler-edge.js +84 -0
  111. package/generated/prisma-postgresql/runtime/wasm-engine-edge.js +36 -0
  112. package/generated/prisma-postgresql/schema.prisma +343 -0
  113. package/generated/prisma-postgresql/wasm-edge-light-loader.mjs +4 -0
  114. package/generated/prisma-postgresql/wasm-worker-loader.mjs +4 -0
  115. package/generated/prisma-postgresql/wasm.d.ts +1 -0
  116. package/generated/prisma-postgresql/wasm.js +363 -0
  117. package/handlers/WEBHOOKS.md +653 -0
  118. package/handlers/app-definition-loader.js +38 -0
  119. package/handlers/app-handler-helpers.js +56 -0
  120. package/handlers/backend-utils.js +186 -0
  121. package/handlers/database-migration-handler.js +227 -0
  122. package/handlers/integration-event-dispatcher.js +54 -0
  123. package/handlers/routers/HEALTHCHECK.md +342 -0
  124. package/handlers/routers/auth.js +15 -0
  125. package/handlers/routers/db-migration.handler.js +29 -0
  126. package/handlers/routers/db-migration.js +326 -0
  127. package/handlers/routers/health.js +516 -0
  128. package/handlers/routers/integration-defined-routers.js +45 -0
  129. package/handlers/routers/integration-webhook-routers.js +67 -0
  130. package/handlers/routers/user.js +63 -0
  131. package/handlers/routers/websocket.js +57 -0
  132. package/handlers/use-cases/check-external-apis-health-use-case.js +81 -0
  133. package/handlers/use-cases/check-integrations-health-use-case.js +44 -0
  134. package/handlers/workers/db-migration.js +352 -0
  135. package/handlers/workers/integration-defined-workers.js +27 -0
  136. package/index.js +77 -22
  137. package/integrations/WEBHOOK-QUICKSTART.md +151 -0
  138. package/integrations/index.js +12 -10
  139. package/integrations/integration-base.js +326 -55
  140. package/integrations/integration-router.js +374 -179
  141. package/integrations/options.js +1 -1
  142. package/integrations/repositories/integration-mapping-repository-documentdb.js +280 -0
  143. package/integrations/repositories/integration-mapping-repository-factory.js +57 -0
  144. package/integrations/repositories/integration-mapping-repository-interface.js +106 -0
  145. package/integrations/repositories/integration-mapping-repository-mongo.js +161 -0
  146. package/integrations/repositories/integration-mapping-repository-postgres.js +227 -0
  147. package/integrations/repositories/integration-mapping-repository.js +156 -0
  148. package/integrations/repositories/integration-repository-documentdb.js +210 -0
  149. package/integrations/repositories/integration-repository-factory.js +51 -0
  150. package/integrations/repositories/integration-repository-interface.js +127 -0
  151. package/integrations/repositories/integration-repository-mongo.js +303 -0
  152. package/integrations/repositories/integration-repository-postgres.js +352 -0
  153. package/integrations/repositories/process-repository-documentdb.js +243 -0
  154. package/integrations/repositories/process-repository-factory.js +53 -0
  155. package/integrations/repositories/process-repository-interface.js +90 -0
  156. package/integrations/repositories/process-repository-mongo.js +190 -0
  157. package/integrations/repositories/process-repository-postgres.js +217 -0
  158. package/integrations/tests/doubles/dummy-integration-class.js +83 -0
  159. package/integrations/tests/doubles/test-integration-repository.js +99 -0
  160. package/integrations/use-cases/create-integration.js +83 -0
  161. package/integrations/use-cases/create-process.js +128 -0
  162. package/integrations/use-cases/delete-integration-for-user.js +101 -0
  163. package/integrations/use-cases/find-integration-context-by-external-entity-id.js +72 -0
  164. package/integrations/use-cases/get-integration-for-user.js +78 -0
  165. package/integrations/use-cases/get-integration-instance-by-definition.js +67 -0
  166. package/integrations/use-cases/get-integration-instance.js +83 -0
  167. package/integrations/use-cases/get-integrations-for-user.js +88 -0
  168. package/integrations/use-cases/get-possible-integrations.js +27 -0
  169. package/integrations/use-cases/get-process.js +87 -0
  170. package/integrations/use-cases/index.js +19 -0
  171. package/integrations/use-cases/load-integration-context.js +71 -0
  172. package/integrations/use-cases/update-integration-messages.js +44 -0
  173. package/integrations/use-cases/update-integration-status.js +32 -0
  174. package/integrations/use-cases/update-integration.js +93 -0
  175. package/integrations/use-cases/update-process-metrics.js +201 -0
  176. package/integrations/use-cases/update-process-state.js +119 -0
  177. package/integrations/utils/map-integration-dto.js +37 -0
  178. package/jest-global-setup-noop.js +3 -0
  179. package/jest-global-teardown-noop.js +3 -0
  180. package/logs/logger.js +0 -4
  181. package/{module-plugin → modules}/entity.js +1 -1
  182. package/{module-plugin → modules}/index.js +0 -8
  183. package/modules/module-factory.js +56 -0
  184. package/modules/module.js +221 -0
  185. package/modules/repositories/module-repository-documentdb.js +307 -0
  186. package/modules/repositories/module-repository-factory.js +40 -0
  187. package/modules/repositories/module-repository-interface.js +129 -0
  188. package/modules/repositories/module-repository-mongo.js +377 -0
  189. package/modules/repositories/module-repository-postgres.js +426 -0
  190. package/modules/repositories/module-repository.js +316 -0
  191. package/modules/requester/api-key.js +52 -0
  192. package/{module-plugin → modules}/requester/requester.js +1 -0
  193. package/{module-plugin → modules}/test/mock-api/api.js +8 -3
  194. package/{module-plugin → modules}/test/mock-api/definition.js +12 -8
  195. package/modules/tests/doubles/test-module-factory.js +16 -0
  196. package/modules/tests/doubles/test-module-repository.js +39 -0
  197. package/modules/use-cases/get-entities-for-user.js +32 -0
  198. package/modules/use-cases/get-entity-options-by-id.js +71 -0
  199. package/modules/use-cases/get-entity-options-by-type.js +34 -0
  200. package/modules/use-cases/get-module-instance-from-type.js +31 -0
  201. package/modules/use-cases/get-module.js +74 -0
  202. package/modules/use-cases/process-authorization-callback.js +133 -0
  203. package/modules/use-cases/refresh-entity-options.js +72 -0
  204. package/modules/use-cases/test-module-auth.js +72 -0
  205. package/modules/utils/map-module-dto.js +18 -0
  206. package/package.json +82 -50
  207. package/prisma-mongodb/schema.prisma +360 -0
  208. package/prisma-postgresql/migrations/20250930193005_init/migration.sql +315 -0
  209. package/prisma-postgresql/migrations/20251006135218_init/migration.sql +9 -0
  210. package/prisma-postgresql/migrations/20251010000000_remove_unused_entity_reference_map/migration.sql +3 -0
  211. package/prisma-postgresql/migrations/20251112195422_update_user_unique_constraints/migration.sql +25 -0
  212. package/prisma-postgresql/migrations/migration_lock.toml +3 -0
  213. package/prisma-postgresql/schema.prisma +343 -0
  214. package/queues/queuer-util.js +27 -22
  215. package/syncs/manager.js +468 -443
  216. package/syncs/repositories/sync-repository-documentdb.js +240 -0
  217. package/syncs/repositories/sync-repository-factory.js +43 -0
  218. package/syncs/repositories/sync-repository-interface.js +109 -0
  219. package/syncs/repositories/sync-repository-mongo.js +239 -0
  220. package/syncs/repositories/sync-repository-postgres.js +319 -0
  221. package/syncs/sync.js +0 -1
  222. package/token/repositories/token-repository-documentdb.js +137 -0
  223. package/token/repositories/token-repository-factory.js +40 -0
  224. package/token/repositories/token-repository-interface.js +131 -0
  225. package/token/repositories/token-repository-mongo.js +219 -0
  226. package/token/repositories/token-repository-postgres.js +264 -0
  227. package/token/repositories/token-repository.js +219 -0
  228. package/types/core/index.d.ts +2 -2
  229. package/types/integrations/index.d.ts +2 -6
  230. package/types/module-plugin/index.d.ts +5 -59
  231. package/types/syncs/index.d.ts +0 -2
  232. package/user/repositories/user-repository-documentdb.js +441 -0
  233. package/user/repositories/user-repository-factory.js +52 -0
  234. package/user/repositories/user-repository-interface.js +201 -0
  235. package/user/repositories/user-repository-mongo.js +308 -0
  236. package/user/repositories/user-repository-postgres.js +360 -0
  237. package/user/tests/doubles/test-user-repository.js +72 -0
  238. package/user/use-cases/authenticate-user.js +127 -0
  239. package/user/use-cases/authenticate-with-shared-secret.js +48 -0
  240. package/user/use-cases/create-individual-user.js +61 -0
  241. package/user/use-cases/create-organization-user.js +47 -0
  242. package/user/use-cases/create-token-for-user-id.js +30 -0
  243. package/user/use-cases/get-user-from-adopter-jwt.js +149 -0
  244. package/user/use-cases/get-user-from-bearer-token.js +77 -0
  245. package/user/use-cases/get-user-from-x-frigg-headers.js +132 -0
  246. package/user/use-cases/login-user.js +122 -0
  247. package/user/user.js +125 -0
  248. package/utils/backend-path.js +38 -0
  249. package/utils/index.js +6 -0
  250. package/websocket/repositories/websocket-connection-repository-documentdb.js +119 -0
  251. package/websocket/repositories/websocket-connection-repository-factory.js +44 -0
  252. package/websocket/repositories/websocket-connection-repository-interface.js +106 -0
  253. package/websocket/repositories/websocket-connection-repository-mongo.js +156 -0
  254. package/websocket/repositories/websocket-connection-repository-postgres.js +196 -0
  255. package/websocket/repositories/websocket-connection-repository.js +161 -0
  256. package/database/models/State.js +0 -9
  257. package/database/models/Token.js +0 -70
  258. package/database/mongo.js +0 -45
  259. package/encrypt/Cryptor.test.js +0 -32
  260. package/encrypt/encrypt.js +0 -132
  261. package/encrypt/encrypt.test.js +0 -1069
  262. package/errors/base-error.test.js +0 -32
  263. package/errors/fetch-error.test.js +0 -79
  264. package/errors/halt-error.test.js +0 -11
  265. package/errors/validation-errors.test.js +0 -120
  266. package/integrations/create-frigg-backend.js +0 -31
  267. package/integrations/integration-factory.js +0 -251
  268. package/integrations/integration-mapping.js +0 -43
  269. package/integrations/integration-model.js +0 -46
  270. package/integrations/integration-user.js +0 -144
  271. package/integrations/test/integration-base.test.js +0 -144
  272. package/lambda/TimeoutCatcher.test.js +0 -68
  273. package/logs/logger.test.js +0 -76
  274. package/module-plugin/auther.js +0 -393
  275. package/module-plugin/credential.js +0 -22
  276. package/module-plugin/entity-manager.js +0 -70
  277. package/module-plugin/manager.js +0 -169
  278. package/module-plugin/module-factory.js +0 -61
  279. package/module-plugin/requester/api-key.js +0 -36
  280. package/module-plugin/requester/requester.test.js +0 -28
  281. package/module-plugin/test/auther.test.js +0 -97
  282. /package/{module-plugin → modules}/ModuleConstants.js +0 -0
  283. /package/{module-plugin → modules}/requester/basic.js +0 -0
  284. /package/{module-plugin → modules}/requester/oauth-2.js +0 -0
  285. /package/{module-plugin → modules}/test/mock-api/mocks/hubspot.js +0 -0
@@ -0,0 +1,47 @@
1
+ const { get } = require('../../assertions');
2
+ const { User } = require('../user');
3
+
4
+ /**
5
+ * Use case for creating an organization user.
6
+ * @class CreateOrganizationUser
7
+ */
8
+ class CreateOrganizationUser {
9
+ /**
10
+ * Creates a new CreateOrganizationUser instance.
11
+ * @param {Object} params - Configuration parameters.
12
+ * @param {import('../user-repository-interface').UserRepositoryInterface} params.userRepository - Repository for user data operations.
13
+ * @param {Object} params.userConfig - The user properties inside of the app definition.
14
+ */
15
+ constructor({ userRepository, userConfig }) {
16
+ this.userRepository = userRepository;
17
+ this.userConfig = userConfig;
18
+ }
19
+
20
+ /**
21
+ * Executes the use case.
22
+ * @async
23
+ * @param {Object} params - The parameters for creating the user.
24
+ * @returns {Promise<import('../user').User>} The newly created user object.
25
+ */
26
+ async execute(params) {
27
+ const name = get(params, 'name');
28
+ const appOrgId = get(params, 'appOrgId');
29
+
30
+ const organizationUserData =
31
+ await this.userRepository.createOrganizationUser({
32
+ name,
33
+ appOrgId,
34
+ });
35
+
36
+ return new User(
37
+ null,
38
+ organizationUserData,
39
+ this.userConfig.usePassword,
40
+ this.userConfig.primary,
41
+ this.userConfig.individualUserRequired,
42
+ this.userConfig.organizationUserRequired
43
+ );
44
+ }
45
+ }
46
+
47
+ module.exports = { CreateOrganizationUser };
@@ -0,0 +1,30 @@
1
+ const crypto = require('crypto');
2
+
3
+ /**
4
+ * Use case for creating a token for a user ID.
5
+ * @class CreateTokenForUserId
6
+ */
7
+ class CreateTokenForUserId {
8
+ /**
9
+ * Creates a new CreateTokenForUserId instance.
10
+ * @param {Object} params - Configuration parameters.
11
+ * @param {import('../user-repository-interface').UserRepositoryInterface} params.userRepository - Repository for user data operations.
12
+ */
13
+ constructor({ userRepository }) {
14
+ this.userRepository = userRepository;
15
+ }
16
+
17
+ /**
18
+ * Executes the use case.
19
+ * @async
20
+ * @param {string} userId - The ID of the user to create a token for.
21
+ * @param {number} minutes - The number of minutes until the token expires.
22
+ * @returns {Promise<string>} The user token.
23
+ */
24
+ async execute(userId, minutes) {
25
+ const rawToken = crypto.randomBytes(20).toString('hex');
26
+ return this.userRepository.createToken(userId, rawToken, minutes);
27
+ }
28
+ }
29
+
30
+ module.exports = { CreateTokenForUserId };
@@ -0,0 +1,149 @@
1
+ const Boom = require('@hapi/boom');
2
+
3
+ /**
4
+ * STUB: Use case for retrieving a user from adopter-provided JWT token.
5
+ *
6
+ * This is a stub implementation for future JWT authentication support.
7
+ * When implemented, this will allow adopters to use their own JWT tokens
8
+ * for authentication instead of Frigg's native token system.
9
+ *
10
+ * FUTURE IMPLEMENTATION REQUIREMENTS:
11
+ * - Validate JWT signature using jwtConfig.secret from app definition
12
+ * - Support configurable signing algorithms (HS256, HS384, HS512, RS256, RS384, RS512)
13
+ * - Extract user identifiers from JWT claims based on jwtConfig.userIdClaim and jwtConfig.orgIdClaim
14
+ * - Find or create user based on extracted claim values
15
+ * - Handle token expiration and validation errors
16
+ * - Support refresh tokens (optional)
17
+ * - Validate user ID conflicts if both individual and org IDs present in JWT
18
+ *
19
+ * RECOMMENDED IMPLEMENTATION:
20
+ * - Use 'jsonwebtoken' package for JWT parsing and validation
21
+ * - Cache JWT public keys for RS* algorithms
22
+ * - Add comprehensive error handling for invalid tokens
23
+ * - Log authentication attempts for security auditing
24
+ *
25
+ * @todo Implement JWT validation with jsonwebtoken package
26
+ * @todo Add unit tests for JWT parsing and claim extraction
27
+ * @todo Document adopter JWT integration guide in Frigg docs
28
+ * @todo Add support for JWT refresh tokens
29
+ * @todo Implement JWT public key caching for RS* algorithms
30
+ *
31
+ * @class GetUserFromAdopterJwt
32
+ */
33
+ class GetUserFromAdopterJwt {
34
+ /**
35
+ * Creates a new GetUserFromAdopterJwt instance.
36
+ * @param {Object} params - Configuration parameters.
37
+ * @param {import('../repositories/user-repository-interface').UserRepositoryInterface} params.userRepository - Repository for user data operations.
38
+ * @param {Object} params.userConfig - The user config in the app definition.
39
+ */
40
+ constructor({ userRepository, userConfig }) {
41
+ this.userRepository = userRepository;
42
+ this.userConfig = userConfig;
43
+ }
44
+
45
+ /**
46
+ * Executes the use case.
47
+ * @async
48
+ * @param {string} jwtToken - The JWT token from the Authorization header.
49
+ * @returns {Promise<import('../user').User>} The authenticated user object.
50
+ * @throws {Boom} 501 Not Implemented - This feature is not yet available.
51
+ */
52
+ async execute(jwtToken) {
53
+ throw Boom.notImplemented(
54
+ 'Adopter JWT authentication is not yet implemented. ' +
55
+ 'This feature is planned for a future Frigg release. ' +
56
+ 'Please use one of the supported authentication modes instead: ' +
57
+ 'friggToken (native bearer token) or xFriggHeaders (backend-to-backend with x-frigg-appUserId/appOrgId headers).'
58
+ );
59
+
60
+ /* FUTURE IMPLEMENTATION PSEUDOCODE:
61
+
62
+ const jwt = require('jsonwebtoken');
63
+
64
+ // Validate JWT configuration exists
65
+ if (!this.userConfig.jwtConfig || !this.userConfig.jwtConfig.secret) {
66
+ throw Boom.badImplementation('JWT configuration is required when adopterJwt auth mode is enabled');
67
+ }
68
+
69
+ try {
70
+ // Verify and decode JWT
71
+ const decoded = jwt.verify(jwtToken, this.userConfig.jwtConfig.secret, {
72
+ algorithms: [this.userConfig.jwtConfig.algorithm || 'HS256']
73
+ });
74
+
75
+ // Extract user identifiers from claims
76
+ const appUserId = decoded[this.userConfig.jwtConfig.userIdClaim || 'sub'];
77
+ const appOrgId = decoded[this.userConfig.jwtConfig.orgIdClaim || 'org_id'];
78
+
79
+ // At least one identifier required
80
+ if (!appUserId && !appOrgId) {
81
+ throw Boom.badRequest('JWT must contain user or organization identifier claims');
82
+ }
83
+
84
+ // Find existing users
85
+ let individualUserData = null;
86
+ let organizationUserData = null;
87
+
88
+ if (appUserId) {
89
+ individualUserData = await this.userRepository.findIndividualUserByAppUserId(appUserId);
90
+ }
91
+
92
+ if (appOrgId) {
93
+ organizationUserData = await this.userRepository.findOrganizationUserByAppOrgId(appOrgId);
94
+ }
95
+
96
+ // Validate no conflicts if both IDs present
97
+ if (appUserId && appOrgId && individualUserData && organizationUserData) {
98
+ const individualOrgId = individualUserData.organizationUser?.toString();
99
+ const expectedOrgId = organizationUserData.id?.toString();
100
+
101
+ if (individualOrgId !== expectedOrgId) {
102
+ throw Boom.badRequest(
103
+ 'User ID mismatch: JWT claims refer to different users. ' +
104
+ 'Individual and organization IDs must belong to the same user.'
105
+ );
106
+ }
107
+ }
108
+
109
+ // Auto-create if not found
110
+ if (!individualUserData && !organizationUserData) {
111
+ if (appUserId) {
112
+ individualUserData = await this.userRepository.createIndividualUser({
113
+ appUserId,
114
+ username: `jwt-user-${appUserId}`,
115
+ email: decoded.email || `${appUserId}@jwt.local`,
116
+ });
117
+ } else {
118
+ organizationUserData = await this.userRepository.createOrganizationUser({
119
+ appOrgId,
120
+ });
121
+ }
122
+ }
123
+
124
+ return new User(
125
+ individualUserData,
126
+ organizationUserData,
127
+ this.userConfig.usePassword,
128
+ this.userConfig.primary,
129
+ this.userConfig.individualUserRequired,
130
+ this.userConfig.organizationUserRequired
131
+ );
132
+
133
+ } catch (error) {
134
+ if (error.name === 'TokenExpiredError') {
135
+ throw Boom.unauthorized('JWT token has expired');
136
+ } else if (error.name === 'JsonWebTokenError') {
137
+ throw Boom.unauthorized('Invalid JWT token');
138
+ } else if (error.isBoom) {
139
+ throw error;
140
+ }
141
+ throw Boom.unauthorized('JWT authentication failed');
142
+ }
143
+ */
144
+ }
145
+ }
146
+
147
+ module.exports = { GetUserFromAdopterJwt };
148
+
149
+
@@ -0,0 +1,77 @@
1
+ const Boom = require('@hapi/boom');
2
+ const { User } = require('../user');
3
+
4
+ /**
5
+ * Use case for retrieving a user from a bearer token.
6
+ * @class GetUserFromBearerToken
7
+ */
8
+ class GetUserFromBearerToken {
9
+ /**
10
+ * Creates a new GetUserFromBearerToken instance.
11
+ * @param {Object} params - Configuration parameters.
12
+ * @param {import('../user-repository-interface').UserRepositoryInterface} params.userRepository - Repository for user data operations.
13
+ * @param {Object} params.userConfig - The user config in the app definition.
14
+ */
15
+ constructor({ userRepository, userConfig }) {
16
+ this.userRepository = userRepository;
17
+ this.userConfig = userConfig;
18
+ }
19
+
20
+ /**
21
+ * Executes the use case.
22
+ * @async
23
+ * @param {string} bearerToken - The bearer token from the authorization header.
24
+ * @returns {Promise<import('../user').User>} The authenticated user object.
25
+ * @throws {Boom} 401 Unauthorized if the token is missing, malformed, or invalid.
26
+ */
27
+ async execute(bearerToken) {
28
+ if (!bearerToken) {
29
+ throw Boom.unauthorized('Missing Authorization Header');
30
+ }
31
+
32
+ const token = bearerToken.split(' ')[1]?.trim();
33
+ if (!token) {
34
+ throw Boom.unauthorized('Invalid Token Format');
35
+ }
36
+
37
+ const sessionToken = await this.userRepository.getSessionToken(token);
38
+
39
+ if (!sessionToken) {
40
+ throw Boom.unauthorized('Session Token Not Found');
41
+ }
42
+
43
+ if (this.userConfig.primary === 'organization') {
44
+ const organizationUserData = await this.userRepository.findOrganizationUserById(sessionToken.user);
45
+
46
+ if (!organizationUserData) {
47
+ throw Boom.unauthorized('Organization User Not Found');
48
+ }
49
+
50
+ return new User(
51
+ null,
52
+ organizationUserData,
53
+ this.userConfig.usePassword,
54
+ this.userConfig.primary,
55
+ this.userConfig.individualUserRequired,
56
+ this.userConfig.organizationUserRequired
57
+ );
58
+ }
59
+
60
+ const individualUserData = await this.userRepository.findIndividualUserById(sessionToken.user);
61
+
62
+ if (!individualUserData) {
63
+ throw Boom.unauthorized('Individual User Not Found');
64
+ }
65
+
66
+ return new User(
67
+ individualUserData,
68
+ null,
69
+ this.userConfig.usePassword,
70
+ this.userConfig.primary,
71
+ this.userConfig.individualUserRequired,
72
+ this.userConfig.organizationUserRequired
73
+ );
74
+ }
75
+ }
76
+
77
+ module.exports = { GetUserFromBearerToken };
@@ -0,0 +1,132 @@
1
+ const Boom = require('@hapi/boom');
2
+ const { User } = require('../user');
3
+
4
+ /**
5
+ * Use case for retrieving or creating a user from x-frigg header identifiers.
6
+ * Supports backend-to-backend API communication using application user IDs.
7
+ *
8
+ * @class GetUserFromXFriggHeaders
9
+ */
10
+ class GetUserFromXFriggHeaders {
11
+ /**
12
+ * Creates a new GetUserFromXFriggHeaders instance.
13
+ * @param {Object} params - Configuration parameters.
14
+ * @param {import('../repositories/user-repository-interface').UserRepositoryInterface} params.userRepository - Repository for user data operations.
15
+ * @param {Object} params.userConfig - The user config in the app definition.
16
+ */
17
+ constructor({ userRepository, userConfig }) {
18
+ this.userRepository = userRepository;
19
+ this.userConfig = userConfig;
20
+ }
21
+
22
+ /**
23
+ * Executes the use case.
24
+ * @async
25
+ * @param {string} [appUserId] - The app user ID from x-frigg-appUserId header.
26
+ * @param {string} [appOrgId] - The app organization ID from x-frigg-appOrgId header.
27
+ * @returns {Promise<import('../user').User>} The authenticated user object.
28
+ * @throws {Boom} 400 Bad Request if neither ID is provided or if both IDs are provided but belong to different users.
29
+ */
30
+ async execute(appUserId, appOrgId) {
31
+ // At least one header must be provided
32
+ if (!appUserId && !appOrgId) {
33
+ throw Boom.badRequest(
34
+ 'At least one of x-frigg-appUserId or x-frigg-appOrgId headers is required for backend-to-backend authentication'
35
+ );
36
+ }
37
+
38
+ // Find users by both IDs if both are provided
39
+ let individualUserData = null;
40
+ let organizationUserData = null;
41
+
42
+ if (appUserId && this.userConfig.individualUserRequired !== false) {
43
+ individualUserData =
44
+ await this.userRepository.findIndividualUserByAppUserId(
45
+ appUserId
46
+ );
47
+ }
48
+
49
+ if (appOrgId && this.userConfig.organizationUserRequired) {
50
+ organizationUserData =
51
+ await this.userRepository.findOrganizationUserByAppOrgId(
52
+ appOrgId
53
+ );
54
+ }
55
+
56
+ // VALIDATION/AUTO-LINKING: If both IDs provided and both users exist, handle mismatch
57
+ if (
58
+ appUserId &&
59
+ appOrgId &&
60
+ individualUserData &&
61
+ organizationUserData
62
+ ) {
63
+ // Check if individual user is linked to the org user
64
+ const individualOrgId =
65
+ individualUserData.organizationUser?.toString();
66
+ const expectedOrgId = organizationUserData.id?.toString();
67
+
68
+ if (individualOrgId !== expectedOrgId) {
69
+ // Default behavior: Auto-link disconnected users
70
+ // Opt-in strict mode: Throw error on mismatch
71
+ if (this.userConfig.strictUserValidation) {
72
+ throw Boom.badRequest(
73
+ 'User ID mismatch: x-frigg-appUserId and x-frigg-appOrgId refer to different users. ' +
74
+ 'Provide only one identifier or ensure they belong to the same user.'
75
+ );
76
+ }
77
+
78
+ // Auto-link the users
79
+ individualUserData = await this.userRepository.linkIndividualToOrganization(
80
+ individualUserData.id,
81
+ organizationUserData.id
82
+ );
83
+ }
84
+ }
85
+
86
+ // Auto-create users independently if they don't exist and are required
87
+ if (
88
+ !individualUserData &&
89
+ appUserId &&
90
+ this.userConfig.individualUserRequired !== false
91
+ ) {
92
+ individualUserData =
93
+ await this.userRepository.createIndividualUser({
94
+ appUserId,
95
+ username: `app-user-${appUserId}`,
96
+ email: `${appUserId}@app.local`,
97
+ });
98
+ }
99
+
100
+ if (
101
+ !organizationUserData &&
102
+ appOrgId &&
103
+ this.userConfig.organizationUserRequired
104
+ ) {
105
+ organizationUserData =
106
+ await this.userRepository.createOrganizationUser({
107
+ appOrgId,
108
+ });
109
+
110
+ // Link individual user to newly created org user if individual exists
111
+ if (individualUserData && organizationUserData) {
112
+ individualUserData = await this.userRepository.linkIndividualToOrganization(
113
+ individualUserData.id,
114
+ organizationUserData.id
115
+ );
116
+ }
117
+ }
118
+
119
+ const user = new User(
120
+ individualUserData,
121
+ organizationUserData,
122
+ this.userConfig.usePassword,
123
+ this.userConfig.primary,
124
+ this.userConfig.individualUserRequired,
125
+ this.userConfig.organizationUserRequired
126
+ );
127
+
128
+ return user;
129
+ }
130
+ }
131
+
132
+ module.exports = { GetUserFromXFriggHeaders };
@@ -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
+ throw new Error('User configuration must require either individualUserRequired or organizationUserRequired');
119
+ }
120
+ }
121
+
122
+ module.exports = { LoginUser };
package/user/user.js ADDED
@@ -0,0 +1,125 @@
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
+ * Checks if a given userId belongs to this user (either primary or linked).
94
+ * When primary is 'organization', entities owned by the linked individual user
95
+ * should still be accessible to the organization.
96
+ *
97
+ * @param {string|number} userId - The userId to check
98
+ * @returns {boolean} True if the userId belongs to this user or their linked user
99
+ */
100
+ ownsUserId(userId) {
101
+ const userIdStr = userId?.toString();
102
+ const primaryId = this.getPrimaryUser()?.id?.toString();
103
+ const individualId = this.individualUser?.id?.toString();
104
+ const organizationId = this.organizationUser?.id?.toString();
105
+
106
+ // Check if userId matches primary user
107
+ if (userIdStr === primaryId) {
108
+ return true;
109
+ }
110
+
111
+ // When primary is 'organization', also check linked individual user
112
+ if (this.config.primary === 'organization' && userIdStr === individualId) {
113
+ return true;
114
+ }
115
+
116
+ // When primary is 'individual', also check linked organization user if required
117
+ if (this.config.primary === 'individual' && this.config.organizationUserRequired && userIdStr === organizationId) {
118
+ return true;
119
+ }
120
+
121
+ return false;
122
+ }
123
+ }
124
+
125
+ module.exports = { User };