@friggframework/core 2.0.0-next.8 → 2.0.0-next.80
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.
- package/CLAUDE.md +694 -0
- package/README.md +959 -50
- package/application/commands/README.md +451 -0
- package/application/commands/credential-commands.js +245 -0
- package/application/commands/entity-commands.js +336 -0
- package/application/commands/integration-commands.js +210 -0
- package/application/commands/scheduler-commands.js +263 -0
- package/application/commands/user-commands.js +283 -0
- package/application/index.js +73 -0
- package/assertions/index.js +0 -3
- package/core/CLAUDE.md +690 -0
- package/core/Worker.js +60 -24
- package/core/create-handler.js +79 -8
- package/credential/repositories/credential-repository-documentdb.js +304 -0
- package/credential/repositories/credential-repository-factory.js +54 -0
- package/credential/repositories/credential-repository-interface.js +98 -0
- package/credential/repositories/credential-repository-mongo.js +269 -0
- package/credential/repositories/credential-repository-postgres.js +287 -0
- package/credential/repositories/credential-repository.js +300 -0
- package/credential/use-cases/get-credential-for-user.js +25 -0
- package/credential/use-cases/update-authentication-status.js +15 -0
- package/database/MONGODB_TRANSACTION_FIX.md +198 -0
- package/database/adapters/lambda-invoker.js +97 -0
- package/database/config.js +154 -0
- package/database/documentdb-encryption-service.js +330 -0
- package/database/documentdb-utils.js +136 -0
- package/database/encryption/README.md +839 -0
- package/database/encryption/documentdb-encryption-service.md +3575 -0
- package/database/encryption/encryption-schema-registry.js +268 -0
- package/database/encryption/field-encryption-service.js +226 -0
- package/database/encryption/logger.js +79 -0
- package/database/encryption/prisma-encryption-extension.js +222 -0
- package/database/index.js +21 -21
- package/database/prisma.js +182 -0
- package/database/repositories/health-check-repository-documentdb.js +138 -0
- package/database/repositories/health-check-repository-factory.js +48 -0
- package/database/repositories/health-check-repository-interface.js +82 -0
- package/database/repositories/health-check-repository-mongodb.js +89 -0
- package/database/repositories/health-check-repository-postgres.js +82 -0
- package/database/repositories/migration-status-repository-s3.js +137 -0
- package/database/use-cases/check-database-health-use-case.js +29 -0
- package/database/use-cases/check-database-state-use-case.js +81 -0
- package/database/use-cases/check-encryption-health-use-case.js +83 -0
- package/database/use-cases/get-database-state-via-worker-use-case.js +61 -0
- package/database/use-cases/get-migration-status-use-case.js +93 -0
- package/database/use-cases/run-database-migration-use-case.js +139 -0
- package/database/use-cases/test-encryption-use-case.js +253 -0
- package/database/use-cases/trigger-database-migration-use-case.js +157 -0
- package/database/utils/mongodb-collection-utils.js +94 -0
- package/database/utils/mongodb-schema-init.js +108 -0
- package/database/utils/prisma-runner.js +477 -0
- package/database/utils/prisma-schema-parser.js +182 -0
- package/docs/PROCESS_MANAGEMENT_QUEUE_SPEC.md +517 -0
- package/encrypt/Cryptor.js +34 -168
- package/encrypt/index.js +1 -2
- package/errors/client-safe-error.js +26 -0
- package/errors/fetch-error.js +15 -7
- package/errors/index.js +2 -0
- package/generated/prisma-mongodb/client.d.ts +1 -0
- package/generated/prisma-mongodb/client.js +4 -0
- package/generated/prisma-mongodb/default.d.ts +1 -0
- package/generated/prisma-mongodb/default.js +4 -0
- package/generated/prisma-mongodb/edge.d.ts +1 -0
- package/generated/prisma-mongodb/edge.js +335 -0
- package/generated/prisma-mongodb/index-browser.js +317 -0
- package/generated/prisma-mongodb/index.d.ts +22955 -0
- package/generated/prisma-mongodb/index.js +360 -0
- package/generated/prisma-mongodb/package.json +183 -0
- package/generated/prisma-mongodb/query-engine-debian-openssl-3.0.x +0 -0
- package/generated/prisma-mongodb/query-engine-rhel-openssl-3.0.x +0 -0
- package/generated/prisma-mongodb/runtime/binary.d.ts +1 -0
- package/generated/prisma-mongodb/runtime/binary.js +289 -0
- package/generated/prisma-mongodb/runtime/edge-esm.js +34 -0
- package/generated/prisma-mongodb/runtime/edge.js +34 -0
- package/generated/prisma-mongodb/runtime/index-browser.d.ts +370 -0
- package/generated/prisma-mongodb/runtime/index-browser.js +16 -0
- package/generated/prisma-mongodb/runtime/library.d.ts +3977 -0
- package/generated/prisma-mongodb/runtime/react-native.js +83 -0
- package/generated/prisma-mongodb/runtime/wasm-compiler-edge.js +84 -0
- package/generated/prisma-mongodb/runtime/wasm-engine-edge.js +36 -0
- package/generated/prisma-mongodb/schema.prisma +362 -0
- package/generated/prisma-mongodb/wasm-edge-light-loader.mjs +4 -0
- package/generated/prisma-mongodb/wasm-worker-loader.mjs +4 -0
- package/generated/prisma-mongodb/wasm.d.ts +1 -0
- package/generated/prisma-mongodb/wasm.js +342 -0
- package/generated/prisma-postgresql/client.d.ts +1 -0
- package/generated/prisma-postgresql/client.js +4 -0
- package/generated/prisma-postgresql/default.d.ts +1 -0
- package/generated/prisma-postgresql/default.js +4 -0
- package/generated/prisma-postgresql/edge.d.ts +1 -0
- package/generated/prisma-postgresql/edge.js +357 -0
- package/generated/prisma-postgresql/index-browser.js +339 -0
- package/generated/prisma-postgresql/index.d.ts +25131 -0
- package/generated/prisma-postgresql/index.js +382 -0
- package/generated/prisma-postgresql/package.json +183 -0
- package/generated/prisma-postgresql/query-engine-debian-openssl-3.0.x +0 -0
- package/generated/prisma-postgresql/query-engine-rhel-openssl-3.0.x +0 -0
- package/generated/prisma-postgresql/query_engine_bg.js +2 -0
- package/generated/prisma-postgresql/query_engine_bg.wasm +0 -0
- package/generated/prisma-postgresql/runtime/binary.d.ts +1 -0
- package/generated/prisma-postgresql/runtime/binary.js +289 -0
- package/generated/prisma-postgresql/runtime/edge-esm.js +34 -0
- package/generated/prisma-postgresql/runtime/edge.js +34 -0
- package/generated/prisma-postgresql/runtime/index-browser.d.ts +370 -0
- package/generated/prisma-postgresql/runtime/index-browser.js +16 -0
- package/generated/prisma-postgresql/runtime/library.d.ts +3977 -0
- package/generated/prisma-postgresql/runtime/react-native.js +83 -0
- package/generated/prisma-postgresql/runtime/wasm-compiler-edge.js +84 -0
- package/generated/prisma-postgresql/runtime/wasm-engine-edge.js +36 -0
- package/generated/prisma-postgresql/schema.prisma +345 -0
- package/generated/prisma-postgresql/wasm-edge-light-loader.mjs +4 -0
- package/generated/prisma-postgresql/wasm-worker-loader.mjs +4 -0
- package/generated/prisma-postgresql/wasm.d.ts +1 -0
- package/generated/prisma-postgresql/wasm.js +364 -0
- package/handlers/WEBHOOKS.md +653 -0
- package/handlers/app-definition-loader.js +38 -0
- package/handlers/app-handler-helpers.js +57 -0
- package/handlers/backend-utils.js +262 -0
- package/handlers/database-migration-handler.js +227 -0
- package/handlers/integration-event-dispatcher.js +54 -0
- package/handlers/routers/HEALTHCHECK.md +342 -0
- package/handlers/routers/auth.js +15 -0
- package/handlers/routers/db-migration.handler.js +29 -0
- package/handlers/routers/db-migration.js +326 -0
- package/handlers/routers/health.js +516 -0
- package/handlers/routers/integration-defined-routers.js +45 -0
- package/handlers/routers/integration-webhook-routers.js +67 -0
- package/handlers/routers/user.js +63 -0
- package/handlers/routers/websocket.js +57 -0
- package/handlers/use-cases/check-external-apis-health-use-case.js +81 -0
- package/handlers/use-cases/check-integrations-health-use-case.js +44 -0
- package/handlers/workers/db-migration.js +352 -0
- package/handlers/workers/dlq-processor.js +63 -0
- package/handlers/workers/integration-defined-workers.js +23 -0
- package/index.js +82 -46
- package/infrastructure/scheduler/eventbridge-scheduler-adapter.js +184 -0
- package/infrastructure/scheduler/index.js +33 -0
- package/infrastructure/scheduler/mock-scheduler-adapter.js +143 -0
- package/infrastructure/scheduler/scheduler-service-factory.js +73 -0
- package/infrastructure/scheduler/scheduler-service-interface.js +47 -0
- package/integrations/WEBHOOK-QUICKSTART.md +151 -0
- package/integrations/index.js +12 -10
- package/integrations/integration-base.js +364 -55
- package/integrations/integration-router.js +375 -179
- package/integrations/options.js +1 -1
- package/integrations/repositories/integration-mapping-repository-documentdb.js +280 -0
- package/integrations/repositories/integration-mapping-repository-factory.js +57 -0
- package/integrations/repositories/integration-mapping-repository-interface.js +106 -0
- package/integrations/repositories/integration-mapping-repository-mongo.js +161 -0
- package/integrations/repositories/integration-mapping-repository-postgres.js +227 -0
- package/integrations/repositories/integration-mapping-repository.js +156 -0
- package/integrations/repositories/integration-repository-documentdb.js +219 -0
- package/integrations/repositories/integration-repository-factory.js +51 -0
- package/integrations/repositories/integration-repository-interface.js +144 -0
- package/integrations/repositories/integration-repository-mongo.js +330 -0
- package/integrations/repositories/integration-repository-postgres.js +385 -0
- package/integrations/repositories/process-repository-documentdb.js +243 -0
- package/integrations/repositories/process-repository-factory.js +53 -0
- package/integrations/repositories/process-repository-interface.js +90 -0
- package/integrations/repositories/process-repository-mongo.js +190 -0
- package/integrations/repositories/process-repository-postgres.js +217 -0
- package/integrations/tests/doubles/config-capturing-integration.js +81 -0
- package/integrations/tests/doubles/dummy-integration-class.js +105 -0
- package/integrations/tests/doubles/test-integration-repository.js +112 -0
- package/integrations/use-cases/create-integration.js +83 -0
- package/integrations/use-cases/create-process.js +128 -0
- package/integrations/use-cases/delete-integration-for-user.js +101 -0
- package/integrations/use-cases/find-integration-context-by-external-entity-id.js +72 -0
- package/integrations/use-cases/get-integration-for-user.js +78 -0
- package/integrations/use-cases/get-integration-instance-by-definition.js +67 -0
- package/integrations/use-cases/get-integration-instance.js +83 -0
- package/integrations/use-cases/get-integrations-for-user.js +88 -0
- package/integrations/use-cases/get-possible-integrations.js +27 -0
- package/integrations/use-cases/get-process.js +87 -0
- package/integrations/use-cases/index.js +19 -0
- package/integrations/use-cases/load-integration-context.js +71 -0
- package/integrations/use-cases/update-integration-messages.js +44 -0
- package/integrations/use-cases/update-integration-status.js +32 -0
- package/integrations/use-cases/update-integration.js +92 -0
- package/integrations/use-cases/update-process-metrics.js +201 -0
- package/integrations/use-cases/update-process-state.js +119 -0
- package/integrations/utils/map-integration-dto.js +37 -0
- package/jest-global-setup-noop.js +3 -0
- package/jest-global-teardown-noop.js +3 -0
- package/logs/logger.js +0 -4
- package/{module-plugin → modules}/index.js +0 -10
- package/modules/module-factory.js +56 -0
- package/modules/module.js +256 -0
- package/modules/repositories/module-repository-documentdb.js +335 -0
- package/modules/repositories/module-repository-factory.js +40 -0
- package/modules/repositories/module-repository-interface.js +129 -0
- package/modules/repositories/module-repository-mongo.js +408 -0
- package/modules/repositories/module-repository-postgres.js +453 -0
- package/modules/repositories/module-repository.js +345 -0
- package/modules/requester/api-key.js +52 -0
- package/modules/requester/oauth-2.js +396 -0
- package/{module-plugin → modules}/requester/requester.js +4 -2
- package/{module-plugin → modules}/test/mock-api/api.js +8 -3
- package/{module-plugin → modules}/test/mock-api/definition.js +14 -10
- package/modules/tests/doubles/test-module-factory.js +16 -0
- package/modules/tests/doubles/test-module-repository.js +39 -0
- package/modules/use-cases/get-entities-for-user.js +32 -0
- package/modules/use-cases/get-entity-options-by-id.js +71 -0
- package/modules/use-cases/get-entity-options-by-type.js +34 -0
- package/modules/use-cases/get-module-instance-from-type.js +31 -0
- package/modules/use-cases/get-module.js +74 -0
- package/modules/use-cases/process-authorization-callback.js +177 -0
- package/modules/use-cases/refresh-entity-options.js +72 -0
- package/modules/use-cases/test-module-auth.js +72 -0
- package/modules/utils/map-module-dto.js +18 -0
- package/package.json +82 -50
- package/prisma-mongodb/schema.prisma +362 -0
- package/prisma-postgresql/migrations/20250930193005_init/migration.sql +315 -0
- package/prisma-postgresql/migrations/20251006135218_init/migration.sql +9 -0
- package/prisma-postgresql/migrations/20251010000000_remove_unused_entity_reference_map/migration.sql +3 -0
- package/prisma-postgresql/migrations/20251112195422_update_user_unique_constraints/migration.sql +25 -0
- package/prisma-postgresql/migrations/migration_lock.toml +3 -0
- package/prisma-postgresql/schema.prisma +345 -0
- package/queues/queuer-util.js +103 -21
- package/syncs/manager.js +468 -443
- package/syncs/repositories/sync-repository-documentdb.js +240 -0
- package/syncs/repositories/sync-repository-factory.js +43 -0
- package/syncs/repositories/sync-repository-interface.js +109 -0
- package/syncs/repositories/sync-repository-mongo.js +239 -0
- package/syncs/repositories/sync-repository-postgres.js +319 -0
- package/syncs/sync.js +0 -1
- package/token/repositories/token-repository-documentdb.js +137 -0
- package/token/repositories/token-repository-factory.js +40 -0
- package/token/repositories/token-repository-interface.js +131 -0
- package/token/repositories/token-repository-mongo.js +219 -0
- package/token/repositories/token-repository-postgres.js +264 -0
- package/token/repositories/token-repository.js +219 -0
- package/types/associations/index.d.ts +0 -17
- package/types/core/index.d.ts +12 -4
- package/types/database/index.d.ts +10 -2
- package/types/encrypt/index.d.ts +5 -3
- package/types/integrations/index.d.ts +3 -8
- package/types/module-plugin/index.d.ts +17 -69
- package/types/syncs/index.d.ts +0 -17
- package/user/repositories/user-repository-documentdb.js +441 -0
- package/user/repositories/user-repository-factory.js +52 -0
- package/user/repositories/user-repository-interface.js +201 -0
- package/user/repositories/user-repository-mongo.js +308 -0
- package/user/repositories/user-repository-postgres.js +360 -0
- package/user/tests/doubles/test-user-repository.js +72 -0
- package/user/use-cases/authenticate-user.js +127 -0
- package/user/use-cases/authenticate-with-shared-secret.js +48 -0
- package/user/use-cases/create-individual-user.js +61 -0
- package/user/use-cases/create-organization-user.js +47 -0
- package/user/use-cases/create-token-for-user-id.js +30 -0
- package/user/use-cases/get-user-from-adopter-jwt.js +149 -0
- package/user/use-cases/get-user-from-bearer-token.js +77 -0
- package/user/use-cases/get-user-from-x-frigg-headers.js +132 -0
- package/user/use-cases/login-user.js +122 -0
- package/user/user.js +125 -0
- package/utils/backend-path.js +38 -0
- package/utils/index.js +6 -0
- package/websocket/repositories/websocket-connection-repository-documentdb.js +119 -0
- package/websocket/repositories/websocket-connection-repository-factory.js +44 -0
- package/websocket/repositories/websocket-connection-repository-interface.js +106 -0
- package/websocket/repositories/websocket-connection-repository-mongo.js +156 -0
- package/websocket/repositories/websocket-connection-repository-postgres.js +196 -0
- package/websocket/repositories/websocket-connection-repository.js +161 -0
- package/assertions/is-equal.js +0 -17
- package/associations/model.js +0 -54
- package/database/models/IndividualUser.js +0 -76
- package/database/models/OrganizationUser.js +0 -29
- package/database/models/State.js +0 -9
- package/database/models/Token.js +0 -70
- package/database/models/UserModel.js +0 -7
- package/database/models/WebsocketConnection.js +0 -49
- package/database/mongo.js +0 -45
- package/database/mongoose.js +0 -5
- package/encrypt/Cryptor.test.js +0 -32
- package/encrypt/encrypt.js +0 -132
- package/encrypt/encrypt.test.js +0 -1069
- package/encrypt/test-encrypt.js +0 -107
- package/errors/base-error.test.js +0 -32
- package/errors/fetch-error.test.js +0 -79
- package/errors/halt-error.test.js +0 -11
- package/errors/validation-errors.test.js +0 -120
- package/integrations/create-frigg-backend.js +0 -31
- package/integrations/integration-factory.js +0 -251
- package/integrations/integration-mapping.js +0 -43
- package/integrations/integration-model.js +0 -46
- package/integrations/integration-user.js +0 -144
- package/integrations/test/integration-base.test.js +0 -144
- package/lambda/TimeoutCatcher.test.js +0 -68
- package/logs/logger.test.js +0 -76
- package/module-plugin/auther.js +0 -393
- package/module-plugin/credential.js +0 -22
- package/module-plugin/entity-manager.js +0 -70
- package/module-plugin/entity.js +0 -46
- package/module-plugin/manager.js +0 -169
- package/module-plugin/module-factory.js +0 -61
- package/module-plugin/requester/api-key.js +0 -36
- package/module-plugin/requester/oauth-2.js +0 -219
- package/module-plugin/requester/requester.test.js +0 -28
- package/module-plugin/test/auther.test.js +0 -97
- package/syncs/model.js +0 -62
- /package/{module-plugin → modules}/ModuleConstants.js +0 -0
- /package/{module-plugin → modules}/requester/basic.js +0 -0
- /package/{module-plugin → modules}/test/mock-api/mocks/hubspot.js +0 -0
|
@@ -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 };
|
|
@@ -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,119 @@
|
|
|
1
|
+
const { prisma } = require('../../database/prisma');
|
|
2
|
+
const {
|
|
3
|
+
ApiGatewayManagementApiClient,
|
|
4
|
+
PostToConnectionCommand,
|
|
5
|
+
} = require('@aws-sdk/client-apigatewaymanagementapi');
|
|
6
|
+
const {
|
|
7
|
+
toObjectId,
|
|
8
|
+
fromObjectId,
|
|
9
|
+
findMany,
|
|
10
|
+
findOne,
|
|
11
|
+
insertOne,
|
|
12
|
+
deleteOne,
|
|
13
|
+
deleteMany,
|
|
14
|
+
} = require('../../database/documentdb-utils');
|
|
15
|
+
const {
|
|
16
|
+
WebsocketConnectionRepositoryInterface,
|
|
17
|
+
} = require('./websocket-connection-repository-interface');
|
|
18
|
+
|
|
19
|
+
class WebsocketConnectionRepositoryDocumentDB extends WebsocketConnectionRepositoryInterface {
|
|
20
|
+
constructor() {
|
|
21
|
+
super();
|
|
22
|
+
this.prisma = prisma;
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
async createConnection(connectionId) {
|
|
26
|
+
const now = new Date();
|
|
27
|
+
const document = {
|
|
28
|
+
connectionId,
|
|
29
|
+
createdAt: now,
|
|
30
|
+
updatedAt: now,
|
|
31
|
+
};
|
|
32
|
+
const insertedId = await insertOne(this.prisma, 'WebsocketConnection', document);
|
|
33
|
+
const created = await findOne(this.prisma, 'WebsocketConnection', { _id: insertedId });
|
|
34
|
+
return this._mapConnection(created);
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
async deleteConnection(connectionId) {
|
|
38
|
+
const result = await deleteOne(this.prisma, 'WebsocketConnection', { connectionId });
|
|
39
|
+
const deleted = result?.n ?? 0;
|
|
40
|
+
return { acknowledged: true, deletedCount: deleted };
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
async getActiveConnections() {
|
|
44
|
+
if (!process.env.WEBSOCKET_API_ENDPOINT) {
|
|
45
|
+
return [];
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
const connections = await findMany(
|
|
49
|
+
this.prisma,
|
|
50
|
+
'WebsocketConnection',
|
|
51
|
+
{},
|
|
52
|
+
{ projection: { connectionId: 1 } }
|
|
53
|
+
);
|
|
54
|
+
|
|
55
|
+
return connections.map((conn) => ({
|
|
56
|
+
connectionId: conn.connectionId,
|
|
57
|
+
send: async (data) => {
|
|
58
|
+
const apigwManagementApi = new ApiGatewayManagementApiClient({
|
|
59
|
+
endpoint: process.env.WEBSOCKET_API_ENDPOINT,
|
|
60
|
+
});
|
|
61
|
+
|
|
62
|
+
try {
|
|
63
|
+
const command = new PostToConnectionCommand({
|
|
64
|
+
ConnectionId: conn.connectionId,
|
|
65
|
+
Data: JSON.stringify(data),
|
|
66
|
+
});
|
|
67
|
+
await apigwManagementApi.send(command);
|
|
68
|
+
} catch (error) {
|
|
69
|
+
if (error.statusCode === 410 || error.$metadata?.httpStatusCode === 410) {
|
|
70
|
+
console.log(`Stale connection ${conn.connectionId}`);
|
|
71
|
+
await deleteMany(this.prisma, 'WebsocketConnection', {
|
|
72
|
+
connectionId: conn.connectionId,
|
|
73
|
+
});
|
|
74
|
+
} else {
|
|
75
|
+
throw error;
|
|
76
|
+
}
|
|
77
|
+
}
|
|
78
|
+
},
|
|
79
|
+
}));
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
async findConnection(connectionId) {
|
|
83
|
+
const doc = await findOne(this.prisma, 'WebsocketConnection', { connectionId });
|
|
84
|
+
return doc ? this._mapConnection(doc) : null;
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
async findConnectionById(id) {
|
|
88
|
+
const objectId = toObjectId(id);
|
|
89
|
+
if (!objectId) return null;
|
|
90
|
+
const doc = await findOne(this.prisma, 'WebsocketConnection', { _id: objectId });
|
|
91
|
+
return doc ? this._mapConnection(doc) : null;
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
async getAllConnections() {
|
|
95
|
+
const docs = await findMany(this.prisma, 'WebsocketConnection');
|
|
96
|
+
return docs.map((doc) => this._mapConnection(doc));
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
async deleteAllConnections() {
|
|
100
|
+
const result = await deleteMany(this.prisma, 'WebsocketConnection', {});
|
|
101
|
+
const deleted = result?.n ?? 0;
|
|
102
|
+
return {
|
|
103
|
+
acknowledged: true,
|
|
104
|
+
deletedCount: deleted,
|
|
105
|
+
};
|
|
106
|
+
}
|
|
107
|
+
|
|
108
|
+
_mapConnection(doc) {
|
|
109
|
+
if (!doc) return null;
|
|
110
|
+
return {
|
|
111
|
+
id: fromObjectId(doc._id),
|
|
112
|
+
connectionId: doc.connectionId,
|
|
113
|
+
};
|
|
114
|
+
}
|
|
115
|
+
}
|
|
116
|
+
|
|
117
|
+
module.exports = { WebsocketConnectionRepositoryDocumentDB };
|
|
118
|
+
|
|
119
|
+
|
|
@@ -0,0 +1,44 @@
|
|
|
1
|
+
const {
|
|
2
|
+
WebsocketConnectionRepositoryMongo,
|
|
3
|
+
} = require('./websocket-connection-repository-mongo');
|
|
4
|
+
const {
|
|
5
|
+
WebsocketConnectionRepositoryPostgres,
|
|
6
|
+
} = require('./websocket-connection-repository-postgres');
|
|
7
|
+
const {
|
|
8
|
+
WebsocketConnectionRepositoryDocumentDB,
|
|
9
|
+
} = require('./websocket-connection-repository-documentdb');
|
|
10
|
+
const config = require('../../database/config');
|
|
11
|
+
|
|
12
|
+
/**
|
|
13
|
+
* Websocket Connection Repository Factory
|
|
14
|
+
* Creates the appropriate repository adapter based on database type
|
|
15
|
+
*
|
|
16
|
+
* @returns {WebsocketConnectionRepositoryInterface} Configured repository adapter
|
|
17
|
+
*/
|
|
18
|
+
function createWebsocketConnectionRepository() {
|
|
19
|
+
const dbType = config.DB_TYPE;
|
|
20
|
+
|
|
21
|
+
switch (dbType) {
|
|
22
|
+
case 'mongodb':
|
|
23
|
+
return new WebsocketConnectionRepositoryMongo();
|
|
24
|
+
|
|
25
|
+
case 'postgresql':
|
|
26
|
+
return new WebsocketConnectionRepositoryPostgres();
|
|
27
|
+
|
|
28
|
+
case 'documentdb':
|
|
29
|
+
return new WebsocketConnectionRepositoryDocumentDB();
|
|
30
|
+
|
|
31
|
+
default:
|
|
32
|
+
throw new Error(
|
|
33
|
+
`Unsupported database type: ${dbType}. Supported values: 'mongodb', 'documentdb', 'postgresql'`
|
|
34
|
+
);
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
module.exports = {
|
|
39
|
+
createWebsocketConnectionRepository,
|
|
40
|
+
// Export adapters for direct testing
|
|
41
|
+
WebsocketConnectionRepositoryMongo,
|
|
42
|
+
WebsocketConnectionRepositoryPostgres,
|
|
43
|
+
WebsocketConnectionRepositoryDocumentDB,
|
|
44
|
+
};
|
|
@@ -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 };
|