@friggframework/core 2.0.0-next.5 → 2.0.0-next.50
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 +693 -0
- package/README.md +959 -50
- package/application/commands/README.md +421 -0
- package/application/commands/credential-commands.js +224 -0
- package/application/commands/entity-commands.js +315 -0
- package/application/commands/integration-commands.js +179 -0
- package/application/commands/user-commands.js +213 -0
- package/application/index.js +69 -0
- package/core/CLAUDE.md +690 -0
- package/core/Worker.js +8 -21
- package/core/create-handler.js +2 -7
- package/credential/repositories/credential-repository-factory.js +47 -0
- package/credential/repositories/credential-repository-interface.js +98 -0
- package/credential/repositories/credential-repository-mongo.js +307 -0
- package/credential/repositories/credential-repository-postgres.js +313 -0
- package/credential/repositories/credential-repository.js +302 -0
- package/credential/use-cases/get-credential-for-user.js +21 -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/encryption/README.md +684 -0
- package/database/encryption/encryption-schema-registry.js +141 -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 +25 -12
- package/database/models/WebsocketConnection.js +16 -10
- package/database/models/readme.md +1 -0
- package/database/prisma.js +222 -0
- package/database/repositories/health-check-repository-factory.js +43 -0
- package/database/repositories/health-check-repository-interface.js +87 -0
- package/database/repositories/health-check-repository-mongodb.js +91 -0
- package/database/repositories/health-check-repository-postgres.js +82 -0
- package/database/repositories/health-check-repository.js +108 -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 +137 -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 +91 -0
- package/database/utils/mongodb-schema-init.js +106 -0
- package/database/utils/prisma-runner.js +400 -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/encrypt/test-encrypt.js +0 -2
- 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 +334 -0
- package/generated/prisma-mongodb/index-browser.js +316 -0
- package/generated/prisma-mongodb/index.d.ts +22898 -0
- package/generated/prisma-mongodb/index.js +359 -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 +3982 -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 +341 -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 +356 -0
- package/generated/prisma-postgresql/index-browser.js +338 -0
- package/generated/prisma-postgresql/index.d.ts +25072 -0
- package/generated/prisma-postgresql/index.js +381 -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 +3982 -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 +363 -0
- package/handlers/WEBHOOKS.md +653 -0
- package/handlers/app-definition-loader.js +38 -0
- package/handlers/app-handler-helpers.js +56 -0
- package/handlers/backend-utils.js +180 -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 +256 -0
- package/handlers/routers/health.js +519 -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/integration-defined-workers.js +27 -0
- package/index.js +77 -22
- package/integrations/WEBHOOK-QUICKSTART.md +151 -0
- package/integrations/index.js +12 -10
- package/integrations/integration-base.js +296 -54
- package/integrations/integration-router.js +381 -182
- package/integrations/options.js +1 -1
- package/integrations/repositories/integration-mapping-repository-factory.js +50 -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-factory.js +44 -0
- package/integrations/repositories/integration-repository-interface.js +127 -0
- package/integrations/repositories/integration-repository-mongo.js +303 -0
- package/integrations/repositories/integration-repository-postgres.js +352 -0
- package/integrations/repositories/process-repository-factory.js +46 -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/dummy-integration-class.js +83 -0
- package/integrations/tests/doubles/test-integration-repository.js +99 -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 +87 -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 +93 -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 +36 -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}/entity.js +1 -1
- package/{module-plugin → modules}/index.js +0 -8
- package/modules/module-factory.js +56 -0
- package/modules/module.js +221 -0
- package/modules/repositories/module-repository-factory.js +33 -0
- package/modules/repositories/module-repository-interface.js +129 -0
- package/modules/repositories/module-repository-mongo.js +377 -0
- package/modules/repositories/module-repository-postgres.js +426 -0
- package/modules/repositories/module-repository.js +316 -0
- package/{module-plugin → modules}/requester/requester.js +1 -0
- package/{module-plugin → modules}/test/mock-api/api.js +8 -3
- package/{module-plugin → modules}/test/mock-api/definition.js +12 -8
- 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 +59 -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 +55 -0
- package/modules/use-cases/process-authorization-callback.js +122 -0
- package/modules/use-cases/refresh-entity-options.js +59 -0
- package/modules/use-cases/test-module-auth.js +55 -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/migration_lock.toml +3 -0
- package/prisma-postgresql/schema.prisma +345 -0
- package/queues/queuer-util.js +28 -15
- package/syncs/manager.js +468 -443
- package/syncs/repositories/sync-repository-factory.js +38 -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-factory.js +33 -0
- package/token/repositories/token-repository-interface.js +131 -0
- package/token/repositories/token-repository-mongo.js +212 -0
- package/token/repositories/token-repository-postgres.js +257 -0
- package/token/repositories/token-repository.js +219 -0
- package/types/core/index.d.ts +2 -2
- package/types/integrations/index.d.ts +2 -6
- package/types/module-plugin/index.d.ts +5 -59
- package/types/syncs/index.d.ts +0 -2
- package/user/repositories/user-repository-factory.js +46 -0
- package/user/repositories/user-repository-interface.js +198 -0
- package/user/repositories/user-repository-mongo.js +291 -0
- package/user/repositories/user-repository-postgres.js +350 -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 +106 -0
- package/user/use-cases/login-user.js +122 -0
- package/user/user.js +93 -0
- package/utils/backend-path.js +38 -0
- package/utils/index.js +6 -0
- package/websocket/repositories/websocket-connection-repository-factory.js +37 -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/database/models/State.js +0 -9
- package/database/models/Token.js +0 -70
- package/database/mongo.js +0 -45
- package/encrypt/Cryptor.test.js +0 -32
- package/encrypt/encrypt.js +0 -132
- package/encrypt/encrypt.test.js +0 -1069
- 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/manager.js +0 -169
- package/module-plugin/module-factory.js +0 -61
- package/module-plugin/requester/requester.test.js +0 -28
- package/module-plugin/test/auther.test.js +0 -97
- /package/{module-plugin → modules}/ModuleConstants.js +0 -0
- /package/{module-plugin → modules}/requester/api-key.js +0 -0
- /package/{module-plugin → modules}/requester/basic.js +0 -0
- /package/{module-plugin → modules}/requester/oauth-2.js +0 -0
- /package/{module-plugin → modules}/test/mock-api/mocks/hubspot.js +0 -0
|
@@ -0,0 +1,519 @@
|
|
|
1
|
+
const { Router } = require('express');
|
|
2
|
+
const { createAppHandler } = require('./../app-handler-helpers');
|
|
3
|
+
const { loadAppDefinition } = require('./../app-definition-loader');
|
|
4
|
+
const { ModuleFactory } = require('../../modules/module-factory');
|
|
5
|
+
const {
|
|
6
|
+
getModulesDefinitionFromIntegrationClasses,
|
|
7
|
+
} = require('../../integrations/utils/map-integration-dto');
|
|
8
|
+
const {
|
|
9
|
+
createModuleRepository,
|
|
10
|
+
} = require('../../modules/repositories/module-repository-factory');
|
|
11
|
+
const {
|
|
12
|
+
createIntegrationRepository,
|
|
13
|
+
} = require('../../integrations/repositories/integration-repository-factory');
|
|
14
|
+
const {
|
|
15
|
+
createHealthCheckRepository,
|
|
16
|
+
} = require('../../database/repositories/health-check-repository-factory');
|
|
17
|
+
const { prisma } = require('../../database/prisma');
|
|
18
|
+
const {
|
|
19
|
+
TestEncryptionUseCase,
|
|
20
|
+
} = require('../../database/use-cases/test-encryption-use-case');
|
|
21
|
+
const {
|
|
22
|
+
CheckDatabaseHealthUseCase,
|
|
23
|
+
} = require('../../database/use-cases/check-database-health-use-case');
|
|
24
|
+
const {
|
|
25
|
+
CheckEncryptionHealthUseCase,
|
|
26
|
+
} = require('../../database/use-cases/check-encryption-health-use-case');
|
|
27
|
+
const {
|
|
28
|
+
CheckExternalApisHealthUseCase,
|
|
29
|
+
} = require('../use-cases/check-external-apis-health-use-case');
|
|
30
|
+
const {
|
|
31
|
+
CheckIntegrationsHealthUseCase,
|
|
32
|
+
} = require('../use-cases/check-integrations-health-use-case');
|
|
33
|
+
|
|
34
|
+
const router = Router();
|
|
35
|
+
const healthCheckRepository = createHealthCheckRepository({ prismaClient: prisma });
|
|
36
|
+
|
|
37
|
+
// Load integrations and create factories just like auth router does
|
|
38
|
+
// This verifies the system can properly load integrations
|
|
39
|
+
let moduleFactory, integrationClasses;
|
|
40
|
+
try {
|
|
41
|
+
const appDef = loadAppDefinition();
|
|
42
|
+
integrationClasses = appDef.integrations || [];
|
|
43
|
+
|
|
44
|
+
const moduleRepository = createModuleRepository();
|
|
45
|
+
const moduleDefinitions = getModulesDefinitionFromIntegrationClasses(integrationClasses);
|
|
46
|
+
|
|
47
|
+
moduleFactory = new ModuleFactory({
|
|
48
|
+
moduleRepository,
|
|
49
|
+
moduleDefinitions,
|
|
50
|
+
});
|
|
51
|
+
} catch (error) {
|
|
52
|
+
console.error('Failed to load integrations for health check:', error.message);
|
|
53
|
+
// Factories will be undefined, health check will report unhealthy
|
|
54
|
+
moduleFactory = undefined;
|
|
55
|
+
integrationClasses = [];
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
const testEncryptionUseCase = new TestEncryptionUseCase({
|
|
59
|
+
healthCheckRepository,
|
|
60
|
+
});
|
|
61
|
+
const checkDatabaseHealthUseCase = new CheckDatabaseHealthUseCase({
|
|
62
|
+
healthCheckRepository,
|
|
63
|
+
});
|
|
64
|
+
const checkEncryptionHealthUseCase = new CheckEncryptionHealthUseCase({
|
|
65
|
+
testEncryptionUseCase,
|
|
66
|
+
});
|
|
67
|
+
const checkExternalApisHealthUseCase = new CheckExternalApisHealthUseCase();
|
|
68
|
+
const checkIntegrationsHealthUseCase = new CheckIntegrationsHealthUseCase({
|
|
69
|
+
moduleFactory,
|
|
70
|
+
integrationClasses,
|
|
71
|
+
});
|
|
72
|
+
|
|
73
|
+
const validateApiKey = (req, res, next) => {
|
|
74
|
+
const apiKey = req.headers['x-frigg-health-api-key'];
|
|
75
|
+
|
|
76
|
+
if (req.path === '/health') {
|
|
77
|
+
return next();
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
if (!apiKey || apiKey !== process.env.HEALTH_API_KEY) {
|
|
81
|
+
console.error('Unauthorized access attempt to health endpoint');
|
|
82
|
+
return res.status(401).json({
|
|
83
|
+
status: 'error',
|
|
84
|
+
message: 'Unauthorized - x-frigg-health-api-key header required',
|
|
85
|
+
});
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
next();
|
|
89
|
+
};
|
|
90
|
+
|
|
91
|
+
router.use(validateApiKey);
|
|
92
|
+
|
|
93
|
+
// Helper to detect VPC configuration
|
|
94
|
+
const detectVpcConfiguration = async () => {
|
|
95
|
+
const results = {
|
|
96
|
+
isInVpc: false,
|
|
97
|
+
hasInternetAccess: false,
|
|
98
|
+
canResolvePublicDns: false,
|
|
99
|
+
canConnectToAws: false,
|
|
100
|
+
vpcEndpoints: [],
|
|
101
|
+
};
|
|
102
|
+
|
|
103
|
+
try {
|
|
104
|
+
// Check if we're in a VPC by looking for VPC-specific environment
|
|
105
|
+
// Lambda in VPC has specific network interface configuration
|
|
106
|
+
const dns = require('dns').promises;
|
|
107
|
+
|
|
108
|
+
// Test 1: Can we resolve public DNS? (indicates DNS configuration)
|
|
109
|
+
try {
|
|
110
|
+
await Promise.race([
|
|
111
|
+
dns.resolve4('www.google.com'),
|
|
112
|
+
new Promise((_, reject) =>
|
|
113
|
+
setTimeout(() => reject(new Error('timeout')), 2000)
|
|
114
|
+
),
|
|
115
|
+
]);
|
|
116
|
+
results.canResolvePublicDns = true;
|
|
117
|
+
} catch (e) {
|
|
118
|
+
console.log('Public DNS resolution failed:', e.message);
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
// Test 2: Can we reach internet? (indicates NAT gateway)
|
|
122
|
+
try {
|
|
123
|
+
const https = require('https');
|
|
124
|
+
await new Promise((resolve, reject) => {
|
|
125
|
+
const req = https.get(
|
|
126
|
+
'https://www.google.com',
|
|
127
|
+
{ timeout: 2000 },
|
|
128
|
+
(res) => {
|
|
129
|
+
res.destroy();
|
|
130
|
+
resolve(true);
|
|
131
|
+
}
|
|
132
|
+
);
|
|
133
|
+
req.on('error', reject);
|
|
134
|
+
req.on('timeout', () => {
|
|
135
|
+
req.destroy();
|
|
136
|
+
reject(new Error('timeout'));
|
|
137
|
+
});
|
|
138
|
+
});
|
|
139
|
+
results.hasInternetAccess = true;
|
|
140
|
+
} catch (e) {
|
|
141
|
+
console.log('Internet connectivity test failed:', e.message);
|
|
142
|
+
}
|
|
143
|
+
|
|
144
|
+
// Test 3: Check for VPC endpoints by trying to resolve internal AWS endpoints
|
|
145
|
+
const region = process.env.AWS_REGION; // Lambda always provides this
|
|
146
|
+
const vpcEndpointDomains = [
|
|
147
|
+
`com.amazonaws.${region}.kms`,
|
|
148
|
+
`com.amazonaws.vpce.${region}`,
|
|
149
|
+
`kms.${region}.amazonaws.com`,
|
|
150
|
+
];
|
|
151
|
+
|
|
152
|
+
for (const domain of vpcEndpointDomains) {
|
|
153
|
+
try {
|
|
154
|
+
const addresses = await Promise.race([
|
|
155
|
+
dns.resolve4(domain).catch(() => dns.resolve6(domain)),
|
|
156
|
+
new Promise((_, reject) =>
|
|
157
|
+
setTimeout(() => reject(new Error('timeout')), 1000)
|
|
158
|
+
),
|
|
159
|
+
]);
|
|
160
|
+
if (addresses && addresses.length > 0) {
|
|
161
|
+
// Check if it's a private IP (VPC endpoint indicator)
|
|
162
|
+
const isPrivateIp = addresses.some(
|
|
163
|
+
(ip) =>
|
|
164
|
+
ip.startsWith('10.') ||
|
|
165
|
+
ip.startsWith('172.') ||
|
|
166
|
+
ip.startsWith('192.168.')
|
|
167
|
+
);
|
|
168
|
+
if (isPrivateIp) {
|
|
169
|
+
results.vpcEndpoints.push(domain);
|
|
170
|
+
}
|
|
171
|
+
}
|
|
172
|
+
} catch (e) {
|
|
173
|
+
// Expected for non-existent endpoints
|
|
174
|
+
}
|
|
175
|
+
}
|
|
176
|
+
|
|
177
|
+
// Check if Lambda is in VPC using VPC_ENABLED env var set by infrastructure
|
|
178
|
+
results.isInVpc = process.env.VPC_ENABLED === 'true' ||
|
|
179
|
+
(!results.hasInternetAccess && results.canResolvePublicDns) ||
|
|
180
|
+
results.vpcEndpoints.length > 0;
|
|
181
|
+
|
|
182
|
+
results.canConnectToAws =
|
|
183
|
+
results.hasInternetAccess || results.vpcEndpoints.length > 0;
|
|
184
|
+
} catch (error) {
|
|
185
|
+
console.error('VPC detection error:', error.message);
|
|
186
|
+
}
|
|
187
|
+
|
|
188
|
+
return results;
|
|
189
|
+
};
|
|
190
|
+
|
|
191
|
+
// KMS decrypt capability check
|
|
192
|
+
const checkKmsDecryptCapability = async () => {
|
|
193
|
+
const start = Date.now();
|
|
194
|
+
const { KMS_KEY_ARN } = process.env;
|
|
195
|
+
if (!KMS_KEY_ARN) {
|
|
196
|
+
return {
|
|
197
|
+
status: 'skipped',
|
|
198
|
+
reason: 'KMS_KEY_ARN not configured',
|
|
199
|
+
};
|
|
200
|
+
}
|
|
201
|
+
|
|
202
|
+
// Log environment for debugging
|
|
203
|
+
console.log('KMS Check Debug:', {
|
|
204
|
+
hasKmsKeyArn: !!KMS_KEY_ARN,
|
|
205
|
+
kmsKeyArnPrefix: KMS_KEY_ARN?.substring(0, 30),
|
|
206
|
+
awsRegion: process.env.AWS_REGION,
|
|
207
|
+
hasDiscoveryKey: !!process.env.AWS_DISCOVERY_KMS_KEY_ID,
|
|
208
|
+
});
|
|
209
|
+
|
|
210
|
+
// First, detect VPC configuration
|
|
211
|
+
const vpcConfig = await detectVpcConfiguration();
|
|
212
|
+
console.log('VPC Configuration:', vpcConfig);
|
|
213
|
+
|
|
214
|
+
// Test DNS resolution for KMS endpoint
|
|
215
|
+
try {
|
|
216
|
+
const dns = require('dns').promises;
|
|
217
|
+
const region = process.env.AWS_REGION; // Lambda always provides this
|
|
218
|
+
const kmsEndpoint = `kms.${region}.amazonaws.com`;
|
|
219
|
+
console.log('Testing DNS resolution for:', kmsEndpoint);
|
|
220
|
+
|
|
221
|
+
// Wrap DNS resolution in a timeout
|
|
222
|
+
const dnsPromise = dns.resolve4(kmsEndpoint);
|
|
223
|
+
const timeoutPromise = new Promise((_, reject) =>
|
|
224
|
+
setTimeout(() => reject(new Error('DNS resolution timeout')), 3000)
|
|
225
|
+
);
|
|
226
|
+
|
|
227
|
+
const addresses = await Promise.race([dnsPromise, timeoutPromise]);
|
|
228
|
+
console.log('KMS endpoint resolved to:', addresses);
|
|
229
|
+
|
|
230
|
+
// Check if resolved to private IP (VPC endpoint)
|
|
231
|
+
const isVpcEndpoint = addresses.some(
|
|
232
|
+
(ip) =>
|
|
233
|
+
ip.startsWith('10.') ||
|
|
234
|
+
ip.startsWith('172.') ||
|
|
235
|
+
ip.startsWith('192.168.')
|
|
236
|
+
);
|
|
237
|
+
|
|
238
|
+
if (isVpcEndpoint) {
|
|
239
|
+
console.log(
|
|
240
|
+
'KMS VPC Endpoint detected - using private connectivity'
|
|
241
|
+
);
|
|
242
|
+
}
|
|
243
|
+
|
|
244
|
+
// Test TCP connectivity to KMS (port 443)
|
|
245
|
+
const net = require('net');
|
|
246
|
+
const testConnection = () =>
|
|
247
|
+
new Promise((resolve) => {
|
|
248
|
+
const socket = new net.Socket();
|
|
249
|
+
const connectionTimeout = setTimeout(() => {
|
|
250
|
+
socket.destroy();
|
|
251
|
+
resolve({ connected: false, error: 'Connection timeout' });
|
|
252
|
+
}, 3000);
|
|
253
|
+
|
|
254
|
+
socket.on('connect', () => {
|
|
255
|
+
clearTimeout(connectionTimeout);
|
|
256
|
+
socket.destroy();
|
|
257
|
+
resolve({ connected: true });
|
|
258
|
+
});
|
|
259
|
+
|
|
260
|
+
socket.on('error', (err) => {
|
|
261
|
+
clearTimeout(connectionTimeout);
|
|
262
|
+
resolve({ connected: false, error: err.message });
|
|
263
|
+
});
|
|
264
|
+
|
|
265
|
+
// Try connecting to first resolved address on HTTPS port
|
|
266
|
+
socket.connect(443, addresses[0]);
|
|
267
|
+
});
|
|
268
|
+
|
|
269
|
+
const connResult = await testConnection();
|
|
270
|
+
console.log('TCP connectivity test:', connResult);
|
|
271
|
+
|
|
272
|
+
if (!connResult.connected) {
|
|
273
|
+
return {
|
|
274
|
+
status: 'unhealthy',
|
|
275
|
+
error: `Cannot connect to KMS endpoint: ${connResult.error}`,
|
|
276
|
+
dnsResolved: true,
|
|
277
|
+
tcpConnection: false,
|
|
278
|
+
vpcConfig,
|
|
279
|
+
latencyMs: Date.now() - start,
|
|
280
|
+
};
|
|
281
|
+
}
|
|
282
|
+
} catch (dnsError) {
|
|
283
|
+
console.error('DNS resolution failed:', dnsError.message);
|
|
284
|
+
return {
|
|
285
|
+
status: 'unhealthy',
|
|
286
|
+
error: `Cannot resolve KMS endpoint: ${dnsError.message}`,
|
|
287
|
+
dnsResolved: false,
|
|
288
|
+
vpcConfig,
|
|
289
|
+
latencyMs: Date.now() - start,
|
|
290
|
+
};
|
|
291
|
+
}
|
|
292
|
+
|
|
293
|
+
try {
|
|
294
|
+
// Use AWS SDK v3 for consistency with the rest of the codebase
|
|
295
|
+
// eslint-disable-next-line global-require
|
|
296
|
+
const {
|
|
297
|
+
KMSClient,
|
|
298
|
+
GenerateDataKeyCommand,
|
|
299
|
+
DecryptCommand,
|
|
300
|
+
} = require('@aws-sdk/client-kms');
|
|
301
|
+
|
|
302
|
+
// Lambda always provides AWS_REGION
|
|
303
|
+
const region = process.env.AWS_REGION;
|
|
304
|
+
|
|
305
|
+
const kms = new KMSClient({
|
|
306
|
+
region,
|
|
307
|
+
requestHandler: {
|
|
308
|
+
connectionTimeout: 10000, // 10 second connection timeout
|
|
309
|
+
requestTimeout: 25000, // 25 second timeout for slow VPC connections
|
|
310
|
+
},
|
|
311
|
+
maxAttempts: 1, // No retries on health checks
|
|
312
|
+
});
|
|
313
|
+
|
|
314
|
+
// Generate a data key (without plaintext logging) then immediately decrypt ciphertext to ensure decrypt perms.
|
|
315
|
+
const dataKeyResp = await kms.send(
|
|
316
|
+
new GenerateDataKeyCommand({
|
|
317
|
+
KeyId: KMS_KEY_ARN,
|
|
318
|
+
KeySpec: 'AES_256',
|
|
319
|
+
})
|
|
320
|
+
);
|
|
321
|
+
const decryptResp = await kms.send(
|
|
322
|
+
new DecryptCommand({ CiphertextBlob: dataKeyResp.CiphertextBlob })
|
|
323
|
+
);
|
|
324
|
+
|
|
325
|
+
const success = Boolean(
|
|
326
|
+
dataKeyResp.CiphertextBlob && decryptResp.Plaintext
|
|
327
|
+
);
|
|
328
|
+
|
|
329
|
+
return {
|
|
330
|
+
status: success ? 'healthy' : 'unhealthy',
|
|
331
|
+
kmsKeyArnSuffix: KMS_KEY_ARN.slice(-12),
|
|
332
|
+
vpcConfig,
|
|
333
|
+
latencyMs: Date.now() - start,
|
|
334
|
+
};
|
|
335
|
+
} catch (error) {
|
|
336
|
+
return {
|
|
337
|
+
status: 'unhealthy',
|
|
338
|
+
error: error.message,
|
|
339
|
+
vpcConfig,
|
|
340
|
+
latencyMs: Date.now() - start,
|
|
341
|
+
};
|
|
342
|
+
}
|
|
343
|
+
};
|
|
344
|
+
|
|
345
|
+
router.get('/health', async (_req, res) => {
|
|
346
|
+
const status = {
|
|
347
|
+
status: 'ok',
|
|
348
|
+
timestamp: new Date().toISOString(),
|
|
349
|
+
service: 'frigg-core-api',
|
|
350
|
+
};
|
|
351
|
+
|
|
352
|
+
res.status(200).json(status);
|
|
353
|
+
});
|
|
354
|
+
|
|
355
|
+
router.get('/health/detailed', async (_req, res) => {
|
|
356
|
+
console.log('Starting detailed health check');
|
|
357
|
+
const startTime = Date.now();
|
|
358
|
+
|
|
359
|
+
const response = {
|
|
360
|
+
service: 'frigg-core-api',
|
|
361
|
+
status: 'healthy',
|
|
362
|
+
timestamp: new Date().toISOString(),
|
|
363
|
+
checks: {},
|
|
364
|
+
};
|
|
365
|
+
|
|
366
|
+
console.log('Health Check Environment:', {
|
|
367
|
+
hasKmsKeyArn: !!process.env.KMS_KEY_ARN,
|
|
368
|
+
awsRegion: process.env.AWS_REGION,
|
|
369
|
+
awsDefaultRegion: process.env.AWS_DEFAULT_REGION,
|
|
370
|
+
nodeEnv: process.env.NODE_ENV,
|
|
371
|
+
stage: process.env.STAGE,
|
|
372
|
+
});
|
|
373
|
+
|
|
374
|
+
try {
|
|
375
|
+
console.log('Running network diagnostics...');
|
|
376
|
+
const networkStart = Date.now();
|
|
377
|
+
response.checks.network = await Promise.race([
|
|
378
|
+
detectVpcConfiguration(),
|
|
379
|
+
new Promise((_, reject) =>
|
|
380
|
+
setTimeout(
|
|
381
|
+
() => reject(new Error('Network diagnostics timeout')),
|
|
382
|
+
5000
|
|
383
|
+
)
|
|
384
|
+
),
|
|
385
|
+
]);
|
|
386
|
+
response.checks.network.latencyMs = Date.now() - networkStart;
|
|
387
|
+
console.log('Network diagnostics completed:', response.checks.network);
|
|
388
|
+
} catch (error) {
|
|
389
|
+
response.checks.network = {
|
|
390
|
+
status: 'error',
|
|
391
|
+
error: error.message,
|
|
392
|
+
};
|
|
393
|
+
console.log('Network diagnostics error:', error.message);
|
|
394
|
+
}
|
|
395
|
+
|
|
396
|
+
try {
|
|
397
|
+
console.log('About to check KMS capability...');
|
|
398
|
+
const kmsCheckPromise = checkKmsDecryptCapability();
|
|
399
|
+
const kmsTimeoutPromise = new Promise((_, reject) =>
|
|
400
|
+
setTimeout(
|
|
401
|
+
() => reject(new Error('KMS check timeout after 25 seconds')),
|
|
402
|
+
25000
|
|
403
|
+
)
|
|
404
|
+
);
|
|
405
|
+
|
|
406
|
+
response.checks.kms = await Promise.race([
|
|
407
|
+
kmsCheckPromise,
|
|
408
|
+
kmsTimeoutPromise,
|
|
409
|
+
]);
|
|
410
|
+
if (response.checks.kms.status === 'unhealthy') {
|
|
411
|
+
response.status = 'unhealthy';
|
|
412
|
+
}
|
|
413
|
+
console.log('KMS check completed:', response.checks.kms);
|
|
414
|
+
} catch (error) {
|
|
415
|
+
response.checks.kms = { status: 'unhealthy', error: error.message };
|
|
416
|
+
response.status = 'unhealthy';
|
|
417
|
+
console.log('KMS check error:', error.message);
|
|
418
|
+
}
|
|
419
|
+
|
|
420
|
+
try {
|
|
421
|
+
response.checks.database = await checkDatabaseHealthUseCase.execute();
|
|
422
|
+
if (response.checks.database.status === 'unhealthy') {
|
|
423
|
+
response.status = 'unhealthy';
|
|
424
|
+
}
|
|
425
|
+
console.log('Database check completed:', response.checks.database);
|
|
426
|
+
} catch (error) {
|
|
427
|
+
response.checks.database = {
|
|
428
|
+
status: 'unhealthy',
|
|
429
|
+
error: error.message,
|
|
430
|
+
};
|
|
431
|
+
response.status = 'unhealthy';
|
|
432
|
+
console.log('Database check error:', error.message);
|
|
433
|
+
}
|
|
434
|
+
|
|
435
|
+
try {
|
|
436
|
+
response.checks.encryption = await checkEncryptionHealthUseCase.execute();
|
|
437
|
+
if (response.checks.encryption.status === 'unhealthy') {
|
|
438
|
+
response.status = 'unhealthy';
|
|
439
|
+
}
|
|
440
|
+
console.log('Encryption check completed:', response.checks.encryption);
|
|
441
|
+
} catch (error) {
|
|
442
|
+
response.checks.encryption = {
|
|
443
|
+
status: 'unhealthy',
|
|
444
|
+
error: error.message,
|
|
445
|
+
};
|
|
446
|
+
response.status = 'unhealthy';
|
|
447
|
+
console.log('Encryption check error:', error.message);
|
|
448
|
+
}
|
|
449
|
+
|
|
450
|
+
try {
|
|
451
|
+
const { apiStatuses, allReachable } = await checkExternalApisHealthUseCase.execute();
|
|
452
|
+
response.checks.externalApis = apiStatuses;
|
|
453
|
+
if (!allReachable) {
|
|
454
|
+
response.status = 'unhealthy';
|
|
455
|
+
}
|
|
456
|
+
console.log('External APIs check completed:', response.checks.externalApis);
|
|
457
|
+
} catch (error) {
|
|
458
|
+
response.checks.externalApis = {
|
|
459
|
+
status: 'unhealthy',
|
|
460
|
+
error: error.message,
|
|
461
|
+
};
|
|
462
|
+
response.status = 'unhealthy';
|
|
463
|
+
console.log('External APIs check error:', error.message);
|
|
464
|
+
}
|
|
465
|
+
|
|
466
|
+
try {
|
|
467
|
+
response.checks.integrations = checkIntegrationsHealthUseCase.execute();
|
|
468
|
+
console.log('Integrations check completed:', response.checks.integrations);
|
|
469
|
+
} catch (error) {
|
|
470
|
+
response.checks.integrations = {
|
|
471
|
+
status: 'unhealthy',
|
|
472
|
+
error: error.message,
|
|
473
|
+
};
|
|
474
|
+
response.status = 'unhealthy';
|
|
475
|
+
console.log('Integrations check error:', error.message);
|
|
476
|
+
}
|
|
477
|
+
|
|
478
|
+
response.responseTime = Date.now() - startTime;
|
|
479
|
+
|
|
480
|
+
const statusCode = response.status === 'healthy' ? 200 : 503;
|
|
481
|
+
res.status(statusCode).json(response);
|
|
482
|
+
|
|
483
|
+
console.log(
|
|
484
|
+
'Final health status:',
|
|
485
|
+
response.status,
|
|
486
|
+
'Response time:',
|
|
487
|
+
response.responseTime
|
|
488
|
+
);
|
|
489
|
+
});
|
|
490
|
+
|
|
491
|
+
router.get('/health/live', (_req, res) => {
|
|
492
|
+
res.status(200).json({
|
|
493
|
+
status: 'alive',
|
|
494
|
+
timestamp: new Date().toISOString(),
|
|
495
|
+
});
|
|
496
|
+
});
|
|
497
|
+
|
|
498
|
+
router.get('/health/ready', async (_req, res) => {
|
|
499
|
+
const dbHealth = await checkDatabaseHealthUseCase.execute();
|
|
500
|
+
const isDbReady = dbHealth.status === 'healthy';
|
|
501
|
+
|
|
502
|
+
const integrationsHealth = checkIntegrationsHealthUseCase.execute();
|
|
503
|
+
const areModulesReady = integrationsHealth.modules.count > 0;
|
|
504
|
+
|
|
505
|
+
const isReady = isDbReady && areModulesReady;
|
|
506
|
+
|
|
507
|
+
res.status(isReady ? 200 : 503).json({
|
|
508
|
+
ready: isReady,
|
|
509
|
+
timestamp: new Date().toISOString(),
|
|
510
|
+
checks: {
|
|
511
|
+
database: isDbReady,
|
|
512
|
+
modules: areModulesReady,
|
|
513
|
+
},
|
|
514
|
+
});
|
|
515
|
+
});
|
|
516
|
+
|
|
517
|
+
const handler = createAppHandler('HTTP Event: Health', router);
|
|
518
|
+
|
|
519
|
+
module.exports = { handler, router };
|
|
@@ -0,0 +1,45 @@
|
|
|
1
|
+
const { createAppHandler } = require('./../app-handler-helpers');
|
|
2
|
+
const {
|
|
3
|
+
loadAppDefinition,
|
|
4
|
+
} = require('../app-definition-loader');
|
|
5
|
+
const { Router } = require('express');
|
|
6
|
+
const { loadRouterFromObject } = require('../backend-utils');
|
|
7
|
+
|
|
8
|
+
const handlers = {};
|
|
9
|
+
const { integrations: integrationClasses } = loadAppDefinition();
|
|
10
|
+
|
|
11
|
+
//todo: this should be in a use case class
|
|
12
|
+
for (const IntegrationClass of integrationClasses) {
|
|
13
|
+
const router = Router();
|
|
14
|
+
const basePath = `/api/${IntegrationClass.Definition.name}-integration`;
|
|
15
|
+
|
|
16
|
+
console.log(`\n│ Configuring routes for ${IntegrationClass.Definition.name} Integration:`);
|
|
17
|
+
|
|
18
|
+
for (const routeDef of IntegrationClass.Definition.routes) {
|
|
19
|
+
if (typeof routeDef === 'function') {
|
|
20
|
+
router.use(basePath, routeDef(IntegrationClass));
|
|
21
|
+
console.log(`│ ANY ${basePath}/* (function handler)`);
|
|
22
|
+
} else if (typeof routeDef === 'object') {
|
|
23
|
+
router.use(
|
|
24
|
+
basePath,
|
|
25
|
+
loadRouterFromObject(IntegrationClass, routeDef)
|
|
26
|
+
);
|
|
27
|
+
const method = (routeDef.method || 'ANY').toUpperCase();
|
|
28
|
+
const fullPath = `${basePath}${routeDef.path}`;
|
|
29
|
+
console.log(`│ ${method} ${fullPath}`);
|
|
30
|
+
} else if (routeDef instanceof express.Router) {
|
|
31
|
+
router.use(basePath, routeDef);
|
|
32
|
+
console.log(`│ ANY ${basePath}/* (express router)`);
|
|
33
|
+
}
|
|
34
|
+
}
|
|
35
|
+
console.log('│');
|
|
36
|
+
|
|
37
|
+
handlers[`${IntegrationClass.Definition.name}`] = {
|
|
38
|
+
handler: createAppHandler(
|
|
39
|
+
`HTTP Event: ${IntegrationClass.Definition.name}`,
|
|
40
|
+
router
|
|
41
|
+
),
|
|
42
|
+
};
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
module.exports = { handlers };
|
|
@@ -0,0 +1,67 @@
|
|
|
1
|
+
const { createAppHandler } = require('./../app-handler-helpers');
|
|
2
|
+
const { loadAppDefinition } = require('../app-definition-loader');
|
|
3
|
+
const { Router } = require('express');
|
|
4
|
+
const { IntegrationEventDispatcher } = require('../integration-event-dispatcher');
|
|
5
|
+
|
|
6
|
+
const handlers = {};
|
|
7
|
+
const { integrations: integrationClasses } = loadAppDefinition();
|
|
8
|
+
|
|
9
|
+
for (const IntegrationClass of integrationClasses) {
|
|
10
|
+
const webhookConfig = IntegrationClass.Definition.webhooks;
|
|
11
|
+
|
|
12
|
+
// Skip if webhooks not enabled
|
|
13
|
+
if (!webhookConfig || (typeof webhookConfig === 'object' && !webhookConfig.enabled)) {
|
|
14
|
+
continue;
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
const router = Router();
|
|
18
|
+
const basePath = `/api/${IntegrationClass.Definition.name}-integration/webhooks`;
|
|
19
|
+
|
|
20
|
+
console.log(`\n│ Configuring webhook routes for ${IntegrationClass.Definition.name}:`);
|
|
21
|
+
|
|
22
|
+
// General webhook route (no integration ID)
|
|
23
|
+
router.post(basePath, async (req, res, next) => {
|
|
24
|
+
try {
|
|
25
|
+
const integrationInstance = new IntegrationClass();
|
|
26
|
+
const dispatcher = new IntegrationEventDispatcher(integrationInstance);
|
|
27
|
+
await dispatcher.dispatchHttp({
|
|
28
|
+
event: 'WEBHOOK_RECEIVED',
|
|
29
|
+
req,
|
|
30
|
+
res,
|
|
31
|
+
next,
|
|
32
|
+
});
|
|
33
|
+
} catch (error) {
|
|
34
|
+
next(error);
|
|
35
|
+
}
|
|
36
|
+
});
|
|
37
|
+
console.log(`│ POST ${basePath}`);
|
|
38
|
+
|
|
39
|
+
// Integration-specific webhook route (with integration ID)
|
|
40
|
+
router.post(`${basePath}/:integrationId`, async (req, res, next) => {
|
|
41
|
+
try {
|
|
42
|
+
const integrationInstance = new IntegrationClass();
|
|
43
|
+
const dispatcher = new IntegrationEventDispatcher(integrationInstance);
|
|
44
|
+
await dispatcher.dispatchHttp({
|
|
45
|
+
event: 'WEBHOOK_RECEIVED',
|
|
46
|
+
req,
|
|
47
|
+
res,
|
|
48
|
+
next,
|
|
49
|
+
});
|
|
50
|
+
} catch (error) {
|
|
51
|
+
next(error);
|
|
52
|
+
}
|
|
53
|
+
});
|
|
54
|
+
console.log(`│ POST ${basePath}/:integrationId`);
|
|
55
|
+
console.log('│');
|
|
56
|
+
|
|
57
|
+
handlers[`${IntegrationClass.Definition.name}Webhook`] = {
|
|
58
|
+
handler: createAppHandler(
|
|
59
|
+
`HTTP Event: ${IntegrationClass.Definition.name} Webhook`,
|
|
60
|
+
router,
|
|
61
|
+
false // shouldUseDatabase = false
|
|
62
|
+
),
|
|
63
|
+
};
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
module.exports = { handlers };
|
|
67
|
+
|
|
@@ -0,0 +1,63 @@
|
|
|
1
|
+
const express = require('express');
|
|
2
|
+
const { createAppHandler } = require('../app-handler-helpers');
|
|
3
|
+
const { checkRequiredParams } = require('@friggframework/core');
|
|
4
|
+
const {
|
|
5
|
+
createUserRepository,
|
|
6
|
+
} = require('../../user/repositories/user-repository-factory');
|
|
7
|
+
const {
|
|
8
|
+
CreateIndividualUser,
|
|
9
|
+
} = require('../../user/use-cases/create-individual-user');
|
|
10
|
+
const { LoginUser } = require('../../user/use-cases/login-user');
|
|
11
|
+
const {
|
|
12
|
+
CreateTokenForUserId,
|
|
13
|
+
} = require('../../user/use-cases/create-token-for-user-id');
|
|
14
|
+
const catchAsyncError = require('express-async-handler');
|
|
15
|
+
const { loadAppDefinition } = require('../app-definition-loader');
|
|
16
|
+
|
|
17
|
+
const router = express();
|
|
18
|
+
const { userConfig } = loadAppDefinition();
|
|
19
|
+
const userRepository = createUserRepository();
|
|
20
|
+
const createIndividualUser = new CreateIndividualUser({
|
|
21
|
+
userRepository,
|
|
22
|
+
userConfig,
|
|
23
|
+
});
|
|
24
|
+
const loginUser = new LoginUser({
|
|
25
|
+
userRepository,
|
|
26
|
+
userConfig,
|
|
27
|
+
});
|
|
28
|
+
const createTokenForUserId = new CreateTokenForUserId({ userRepository });
|
|
29
|
+
|
|
30
|
+
// define the login endpoint
|
|
31
|
+
router.route('/user/login').post(
|
|
32
|
+
catchAsyncError(async (req, res) => {
|
|
33
|
+
const { username, password } = checkRequiredParams(req.body, [
|
|
34
|
+
'username',
|
|
35
|
+
'password',
|
|
36
|
+
]);
|
|
37
|
+
const user = await loginUser.execute({ username, password });
|
|
38
|
+
const token = await createTokenForUserId.execute(user.getId(), 120);
|
|
39
|
+
res.status(201);
|
|
40
|
+
res.json({ token });
|
|
41
|
+
})
|
|
42
|
+
);
|
|
43
|
+
|
|
44
|
+
router.route('/user/create').post(
|
|
45
|
+
catchAsyncError(async (req, res) => {
|
|
46
|
+
const { username, password } = checkRequiredParams(req.body, [
|
|
47
|
+
'username',
|
|
48
|
+
'password',
|
|
49
|
+
]);
|
|
50
|
+
|
|
51
|
+
const user = await createIndividualUser.execute({
|
|
52
|
+
username,
|
|
53
|
+
password,
|
|
54
|
+
});
|
|
55
|
+
const token = await createTokenForUserId.execute(user.getId(), 120);
|
|
56
|
+
res.status(201);
|
|
57
|
+
res.json({ token });
|
|
58
|
+
})
|
|
59
|
+
);
|
|
60
|
+
|
|
61
|
+
const handler = createAppHandler('HTTP Event: User', router);
|
|
62
|
+
|
|
63
|
+
module.exports = { handler, router };
|