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

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (267) hide show
  1. package/CLAUDE.md +693 -0
  2. package/README.md +959 -50
  3. package/application/commands/README.md +421 -0
  4. package/application/commands/credential-commands.js +224 -0
  5. package/application/commands/entity-commands.js +315 -0
  6. package/application/commands/integration-commands.js +179 -0
  7. package/application/commands/user-commands.js +213 -0
  8. package/application/index.js +69 -0
  9. package/core/CLAUDE.md +690 -0
  10. package/core/Worker.js +8 -21
  11. package/core/create-handler.js +2 -7
  12. package/credential/repositories/credential-repository-factory.js +47 -0
  13. package/credential/repositories/credential-repository-interface.js +98 -0
  14. package/credential/repositories/credential-repository-mongo.js +307 -0
  15. package/credential/repositories/credential-repository-postgres.js +313 -0
  16. package/credential/repositories/credential-repository.js +302 -0
  17. package/credential/use-cases/get-credential-for-user.js +21 -0
  18. package/credential/use-cases/update-authentication-status.js +15 -0
  19. package/database/MONGODB_TRANSACTION_FIX.md +198 -0
  20. package/database/adapters/lambda-invoker.js +97 -0
  21. package/database/config.js +154 -0
  22. package/database/encryption/README.md +684 -0
  23. package/database/encryption/encryption-schema-registry.js +141 -0
  24. package/database/encryption/field-encryption-service.js +226 -0
  25. package/database/encryption/logger.js +79 -0
  26. package/database/encryption/prisma-encryption-extension.js +222 -0
  27. package/database/index.js +25 -12
  28. package/database/models/WebsocketConnection.js +16 -10
  29. package/database/models/readme.md +1 -0
  30. package/database/prisma.js +222 -0
  31. package/database/repositories/health-check-repository-factory.js +43 -0
  32. package/database/repositories/health-check-repository-interface.js +87 -0
  33. package/database/repositories/health-check-repository-mongodb.js +91 -0
  34. package/database/repositories/health-check-repository-postgres.js +82 -0
  35. package/database/repositories/health-check-repository.js +108 -0
  36. package/database/repositories/migration-status-repository-s3.js +137 -0
  37. package/database/use-cases/check-database-health-use-case.js +29 -0
  38. package/database/use-cases/check-database-state-use-case.js +81 -0
  39. package/database/use-cases/check-encryption-health-use-case.js +83 -0
  40. package/database/use-cases/get-database-state-via-worker-use-case.js +61 -0
  41. package/database/use-cases/get-migration-status-use-case.js +93 -0
  42. package/database/use-cases/run-database-migration-use-case.js +137 -0
  43. package/database/use-cases/test-encryption-use-case.js +253 -0
  44. package/database/use-cases/trigger-database-migration-use-case.js +157 -0
  45. package/database/utils/mongodb-collection-utils.js +91 -0
  46. package/database/utils/mongodb-schema-init.js +106 -0
  47. package/database/utils/prisma-runner.js +400 -0
  48. package/database/utils/prisma-schema-parser.js +182 -0
  49. package/docs/PROCESS_MANAGEMENT_QUEUE_SPEC.md +517 -0
  50. package/encrypt/Cryptor.js +34 -168
  51. package/encrypt/index.js +1 -2
  52. package/encrypt/test-encrypt.js +0 -2
  53. package/generated/prisma-mongodb/client.d.ts +1 -0
  54. package/generated/prisma-mongodb/client.js +4 -0
  55. package/generated/prisma-mongodb/default.d.ts +1 -0
  56. package/generated/prisma-mongodb/default.js +4 -0
  57. package/generated/prisma-mongodb/edge.d.ts +1 -0
  58. package/generated/prisma-mongodb/edge.js +334 -0
  59. package/generated/prisma-mongodb/index-browser.js +316 -0
  60. package/generated/prisma-mongodb/index.d.ts +22898 -0
  61. package/generated/prisma-mongodb/index.js +359 -0
  62. package/generated/prisma-mongodb/package.json +183 -0
  63. package/generated/prisma-mongodb/query-engine-debian-openssl-3.0.x +0 -0
  64. package/generated/prisma-mongodb/query-engine-rhel-openssl-3.0.x +0 -0
  65. package/generated/prisma-mongodb/runtime/binary.d.ts +1 -0
  66. package/generated/prisma-mongodb/runtime/binary.js +289 -0
  67. package/generated/prisma-mongodb/runtime/edge-esm.js +34 -0
  68. package/generated/prisma-mongodb/runtime/edge.js +34 -0
  69. package/generated/prisma-mongodb/runtime/index-browser.d.ts +370 -0
  70. package/generated/prisma-mongodb/runtime/index-browser.js +16 -0
  71. package/generated/prisma-mongodb/runtime/library.d.ts +3982 -0
  72. package/generated/prisma-mongodb/runtime/react-native.js +83 -0
  73. package/generated/prisma-mongodb/runtime/wasm-compiler-edge.js +84 -0
  74. package/generated/prisma-mongodb/runtime/wasm-engine-edge.js +36 -0
  75. package/generated/prisma-mongodb/schema.prisma +362 -0
  76. package/generated/prisma-mongodb/wasm-edge-light-loader.mjs +4 -0
  77. package/generated/prisma-mongodb/wasm-worker-loader.mjs +4 -0
  78. package/generated/prisma-mongodb/wasm.d.ts +1 -0
  79. package/generated/prisma-mongodb/wasm.js +341 -0
  80. package/generated/prisma-postgresql/client.d.ts +1 -0
  81. package/generated/prisma-postgresql/client.js +4 -0
  82. package/generated/prisma-postgresql/default.d.ts +1 -0
  83. package/generated/prisma-postgresql/default.js +4 -0
  84. package/generated/prisma-postgresql/edge.d.ts +1 -0
  85. package/generated/prisma-postgresql/edge.js +356 -0
  86. package/generated/prisma-postgresql/index-browser.js +338 -0
  87. package/generated/prisma-postgresql/index.d.ts +25072 -0
  88. package/generated/prisma-postgresql/index.js +381 -0
  89. package/generated/prisma-postgresql/package.json +183 -0
  90. package/generated/prisma-postgresql/query-engine-debian-openssl-3.0.x +0 -0
  91. package/generated/prisma-postgresql/query-engine-rhel-openssl-3.0.x +0 -0
  92. package/generated/prisma-postgresql/query_engine_bg.js +2 -0
  93. package/generated/prisma-postgresql/query_engine_bg.wasm +0 -0
  94. package/generated/prisma-postgresql/runtime/binary.d.ts +1 -0
  95. package/generated/prisma-postgresql/runtime/binary.js +289 -0
  96. package/generated/prisma-postgresql/runtime/edge-esm.js +34 -0
  97. package/generated/prisma-postgresql/runtime/edge.js +34 -0
  98. package/generated/prisma-postgresql/runtime/index-browser.d.ts +370 -0
  99. package/generated/prisma-postgresql/runtime/index-browser.js +16 -0
  100. package/generated/prisma-postgresql/runtime/library.d.ts +3982 -0
  101. package/generated/prisma-postgresql/runtime/react-native.js +83 -0
  102. package/generated/prisma-postgresql/runtime/wasm-compiler-edge.js +84 -0
  103. package/generated/prisma-postgresql/runtime/wasm-engine-edge.js +36 -0
  104. package/generated/prisma-postgresql/schema.prisma +345 -0
  105. package/generated/prisma-postgresql/wasm-edge-light-loader.mjs +4 -0
  106. package/generated/prisma-postgresql/wasm-worker-loader.mjs +4 -0
  107. package/generated/prisma-postgresql/wasm.d.ts +1 -0
  108. package/generated/prisma-postgresql/wasm.js +363 -0
  109. package/handlers/WEBHOOKS.md +653 -0
  110. package/handlers/app-definition-loader.js +38 -0
  111. package/handlers/app-handler-helpers.js +56 -0
  112. package/handlers/backend-utils.js +180 -0
  113. package/handlers/database-migration-handler.js +227 -0
  114. package/handlers/integration-event-dispatcher.js +54 -0
  115. package/handlers/routers/HEALTHCHECK.md +342 -0
  116. package/handlers/routers/auth.js +15 -0
  117. package/handlers/routers/db-migration.handler.js +29 -0
  118. package/handlers/routers/db-migration.js +256 -0
  119. package/handlers/routers/health.js +519 -0
  120. package/handlers/routers/integration-defined-routers.js +45 -0
  121. package/handlers/routers/integration-webhook-routers.js +67 -0
  122. package/handlers/routers/user.js +63 -0
  123. package/handlers/routers/websocket.js +57 -0
  124. package/handlers/use-cases/check-external-apis-health-use-case.js +81 -0
  125. package/handlers/use-cases/check-integrations-health-use-case.js +44 -0
  126. package/handlers/workers/db-migration.js +352 -0
  127. package/handlers/workers/integration-defined-workers.js +27 -0
  128. package/index.js +77 -22
  129. package/integrations/WEBHOOK-QUICKSTART.md +151 -0
  130. package/integrations/index.js +12 -10
  131. package/integrations/integration-base.js +296 -54
  132. package/integrations/integration-router.js +381 -182
  133. package/integrations/options.js +1 -1
  134. package/integrations/repositories/integration-mapping-repository-factory.js +50 -0
  135. package/integrations/repositories/integration-mapping-repository-interface.js +106 -0
  136. package/integrations/repositories/integration-mapping-repository-mongo.js +161 -0
  137. package/integrations/repositories/integration-mapping-repository-postgres.js +227 -0
  138. package/integrations/repositories/integration-mapping-repository.js +156 -0
  139. package/integrations/repositories/integration-repository-factory.js +44 -0
  140. package/integrations/repositories/integration-repository-interface.js +127 -0
  141. package/integrations/repositories/integration-repository-mongo.js +303 -0
  142. package/integrations/repositories/integration-repository-postgres.js +352 -0
  143. package/integrations/repositories/process-repository-factory.js +46 -0
  144. package/integrations/repositories/process-repository-interface.js +90 -0
  145. package/integrations/repositories/process-repository-mongo.js +190 -0
  146. package/integrations/repositories/process-repository-postgres.js +217 -0
  147. package/integrations/tests/doubles/dummy-integration-class.js +83 -0
  148. package/integrations/tests/doubles/test-integration-repository.js +99 -0
  149. package/integrations/use-cases/create-integration.js +83 -0
  150. package/integrations/use-cases/create-process.js +128 -0
  151. package/integrations/use-cases/delete-integration-for-user.js +101 -0
  152. package/integrations/use-cases/find-integration-context-by-external-entity-id.js +72 -0
  153. package/integrations/use-cases/get-integration-for-user.js +78 -0
  154. package/integrations/use-cases/get-integration-instance-by-definition.js +67 -0
  155. package/integrations/use-cases/get-integration-instance.js +83 -0
  156. package/integrations/use-cases/get-integrations-for-user.js +88 -0
  157. package/integrations/use-cases/get-possible-integrations.js +27 -0
  158. package/integrations/use-cases/get-process.js +87 -0
  159. package/integrations/use-cases/index.js +19 -0
  160. package/integrations/use-cases/load-integration-context.js +71 -0
  161. package/integrations/use-cases/update-integration-messages.js +44 -0
  162. package/integrations/use-cases/update-integration-status.js +32 -0
  163. package/integrations/use-cases/update-integration.js +93 -0
  164. package/integrations/use-cases/update-process-metrics.js +201 -0
  165. package/integrations/use-cases/update-process-state.js +119 -0
  166. package/integrations/utils/map-integration-dto.js +37 -0
  167. package/jest-global-setup-noop.js +3 -0
  168. package/jest-global-teardown-noop.js +3 -0
  169. package/logs/logger.js +0 -4
  170. package/{module-plugin → modules}/entity.js +1 -1
  171. package/{module-plugin → modules}/index.js +0 -8
  172. package/modules/module-factory.js +56 -0
  173. package/modules/module.js +221 -0
  174. package/modules/repositories/module-repository-factory.js +33 -0
  175. package/modules/repositories/module-repository-interface.js +129 -0
  176. package/modules/repositories/module-repository-mongo.js +377 -0
  177. package/modules/repositories/module-repository-postgres.js +426 -0
  178. package/modules/repositories/module-repository.js +316 -0
  179. package/{module-plugin → modules}/requester/requester.js +1 -0
  180. package/{module-plugin → modules}/test/mock-api/api.js +8 -3
  181. package/{module-plugin → modules}/test/mock-api/definition.js +12 -8
  182. package/modules/tests/doubles/test-module-factory.js +16 -0
  183. package/modules/tests/doubles/test-module-repository.js +39 -0
  184. package/modules/use-cases/get-entities-for-user.js +32 -0
  185. package/modules/use-cases/get-entity-options-by-id.js +59 -0
  186. package/modules/use-cases/get-entity-options-by-type.js +34 -0
  187. package/modules/use-cases/get-module-instance-from-type.js +31 -0
  188. package/modules/use-cases/get-module.js +55 -0
  189. package/modules/use-cases/process-authorization-callback.js +122 -0
  190. package/modules/use-cases/refresh-entity-options.js +59 -0
  191. package/modules/use-cases/test-module-auth.js +55 -0
  192. package/modules/utils/map-module-dto.js +18 -0
  193. package/package.json +82 -50
  194. package/prisma-mongodb/schema.prisma +362 -0
  195. package/prisma-postgresql/migrations/20250930193005_init/migration.sql +315 -0
  196. package/prisma-postgresql/migrations/20251006135218_init/migration.sql +9 -0
  197. package/prisma-postgresql/migrations/20251010000000_remove_unused_entity_reference_map/migration.sql +3 -0
  198. package/prisma-postgresql/migrations/migration_lock.toml +3 -0
  199. package/prisma-postgresql/schema.prisma +345 -0
  200. package/queues/queuer-util.js +28 -15
  201. package/syncs/manager.js +468 -443
  202. package/syncs/repositories/sync-repository-factory.js +38 -0
  203. package/syncs/repositories/sync-repository-interface.js +109 -0
  204. package/syncs/repositories/sync-repository-mongo.js +239 -0
  205. package/syncs/repositories/sync-repository-postgres.js +319 -0
  206. package/syncs/sync.js +0 -1
  207. package/token/repositories/token-repository-factory.js +33 -0
  208. package/token/repositories/token-repository-interface.js +131 -0
  209. package/token/repositories/token-repository-mongo.js +212 -0
  210. package/token/repositories/token-repository-postgres.js +257 -0
  211. package/token/repositories/token-repository.js +219 -0
  212. package/types/core/index.d.ts +2 -2
  213. package/types/integrations/index.d.ts +2 -6
  214. package/types/module-plugin/index.d.ts +5 -59
  215. package/types/syncs/index.d.ts +0 -2
  216. package/user/repositories/user-repository-factory.js +46 -0
  217. package/user/repositories/user-repository-interface.js +198 -0
  218. package/user/repositories/user-repository-mongo.js +291 -0
  219. package/user/repositories/user-repository-postgres.js +350 -0
  220. package/user/tests/doubles/test-user-repository.js +72 -0
  221. package/user/use-cases/authenticate-user.js +127 -0
  222. package/user/use-cases/authenticate-with-shared-secret.js +48 -0
  223. package/user/use-cases/create-individual-user.js +61 -0
  224. package/user/use-cases/create-organization-user.js +47 -0
  225. package/user/use-cases/create-token-for-user-id.js +30 -0
  226. package/user/use-cases/get-user-from-adopter-jwt.js +149 -0
  227. package/user/use-cases/get-user-from-bearer-token.js +77 -0
  228. package/user/use-cases/get-user-from-x-frigg-headers.js +106 -0
  229. package/user/use-cases/login-user.js +122 -0
  230. package/user/user.js +93 -0
  231. package/utils/backend-path.js +38 -0
  232. package/utils/index.js +6 -0
  233. package/websocket/repositories/websocket-connection-repository-factory.js +37 -0
  234. package/websocket/repositories/websocket-connection-repository-interface.js +106 -0
  235. package/websocket/repositories/websocket-connection-repository-mongo.js +156 -0
  236. package/websocket/repositories/websocket-connection-repository-postgres.js +196 -0
  237. package/websocket/repositories/websocket-connection-repository.js +161 -0
  238. package/database/models/State.js +0 -9
  239. package/database/models/Token.js +0 -70
  240. package/database/mongo.js +0 -45
  241. package/encrypt/Cryptor.test.js +0 -32
  242. package/encrypt/encrypt.js +0 -132
  243. package/encrypt/encrypt.test.js +0 -1069
  244. package/errors/base-error.test.js +0 -32
  245. package/errors/fetch-error.test.js +0 -79
  246. package/errors/halt-error.test.js +0 -11
  247. package/errors/validation-errors.test.js +0 -120
  248. package/integrations/create-frigg-backend.js +0 -31
  249. package/integrations/integration-factory.js +0 -251
  250. package/integrations/integration-mapping.js +0 -43
  251. package/integrations/integration-model.js +0 -46
  252. package/integrations/integration-user.js +0 -144
  253. package/integrations/test/integration-base.test.js +0 -144
  254. package/lambda/TimeoutCatcher.test.js +0 -68
  255. package/logs/logger.test.js +0 -76
  256. package/module-plugin/auther.js +0 -393
  257. package/module-plugin/credential.js +0 -22
  258. package/module-plugin/entity-manager.js +0 -70
  259. package/module-plugin/manager.js +0 -169
  260. package/module-plugin/module-factory.js +0 -61
  261. package/module-plugin/requester/requester.test.js +0 -28
  262. package/module-plugin/test/auther.test.js +0 -97
  263. /package/{module-plugin → modules}/ModuleConstants.js +0 -0
  264. /package/{module-plugin → modules}/requester/api-key.js +0 -0
  265. /package/{module-plugin → modules}/requester/basic.js +0 -0
  266. /package/{module-plugin → modules}/requester/oauth-2.js +0 -0
  267. /package/{module-plugin → modules}/test/mock-api/mocks/hubspot.js +0 -0
@@ -0,0 +1,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 };