@friggframework/devtools 2.0.0-next.45 → 2.0.0-next.47

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 (212) hide show
  1. package/infrastructure/ARCHITECTURE.md +487 -0
  2. package/infrastructure/HEALTH.md +468 -0
  3. package/infrastructure/README.md +51 -0
  4. package/infrastructure/__tests__/postgres-config.test.js +914 -0
  5. package/infrastructure/__tests__/template-generation.test.js +687 -0
  6. package/infrastructure/create-frigg-infrastructure.js +1 -1
  7. package/infrastructure/docs/POSTGRES-CONFIGURATION.md +630 -0
  8. package/infrastructure/{DEPLOYMENT-INSTRUCTIONS.md → docs/deployment-instructions.md} +3 -3
  9. package/infrastructure/{IAM-POLICY-TEMPLATES.md → docs/iam-policy-templates.md} +9 -10
  10. package/infrastructure/domains/database/aurora-builder.js +809 -0
  11. package/infrastructure/domains/database/aurora-builder.test.js +950 -0
  12. package/infrastructure/domains/database/aurora-discovery.js +87 -0
  13. package/infrastructure/domains/database/aurora-discovery.test.js +188 -0
  14. package/infrastructure/domains/database/aurora-resolver.js +210 -0
  15. package/infrastructure/domains/database/aurora-resolver.test.js +347 -0
  16. package/infrastructure/domains/database/migration-builder.js +695 -0
  17. package/infrastructure/domains/database/migration-builder.test.js +294 -0
  18. package/infrastructure/domains/database/migration-resolver.js +163 -0
  19. package/infrastructure/domains/database/migration-resolver.test.js +337 -0
  20. package/infrastructure/domains/health/application/ports/IPropertyReconciler.js +164 -0
  21. package/infrastructure/domains/health/application/ports/IResourceDetector.js +129 -0
  22. package/infrastructure/domains/health/application/ports/IResourceImporter.js +142 -0
  23. package/infrastructure/domains/health/application/ports/IStackRepository.js +131 -0
  24. package/infrastructure/domains/health/application/ports/index.js +26 -0
  25. package/infrastructure/domains/health/application/use-cases/__tests__/execute-resource-import-use-case.test.js +679 -0
  26. package/infrastructure/domains/health/application/use-cases/__tests__/mismatch-analyzer-method-name.test.js +167 -0
  27. package/infrastructure/domains/health/application/use-cases/__tests__/repair-via-import-use-case.test.js +1130 -0
  28. package/infrastructure/domains/health/application/use-cases/execute-resource-import-use-case.js +221 -0
  29. package/infrastructure/domains/health/application/use-cases/reconcile-properties-use-case.js +152 -0
  30. package/infrastructure/domains/health/application/use-cases/reconcile-properties-use-case.test.js +343 -0
  31. package/infrastructure/domains/health/application/use-cases/repair-via-import-use-case.js +535 -0
  32. package/infrastructure/domains/health/application/use-cases/repair-via-import-use-case.test.js +376 -0
  33. package/infrastructure/domains/health/application/use-cases/run-health-check-use-case.js +213 -0
  34. package/infrastructure/domains/health/application/use-cases/run-health-check-use-case.test.js +441 -0
  35. package/infrastructure/domains/health/docs/ACME-DEV-DRIFT-ANALYSIS.md +267 -0
  36. package/infrastructure/domains/health/docs/BUILD-VS-DEPLOYED-TEMPLATE-ANALYSIS.md +324 -0
  37. package/infrastructure/domains/health/docs/ORPHAN-DETECTION-ANALYSIS.md +386 -0
  38. package/infrastructure/domains/health/docs/SPEC-CLEANUP-COMMAND.md +1419 -0
  39. package/infrastructure/domains/health/docs/TDD-IMPLEMENTATION-SUMMARY.md +391 -0
  40. package/infrastructure/domains/health/docs/TEMPLATE-COMPARISON-IMPLEMENTATION.md +551 -0
  41. package/infrastructure/domains/health/domain/entities/issue.js +299 -0
  42. package/infrastructure/domains/health/domain/entities/issue.test.js +528 -0
  43. package/infrastructure/domains/health/domain/entities/property-mismatch.js +108 -0
  44. package/infrastructure/domains/health/domain/entities/property-mismatch.test.js +275 -0
  45. package/infrastructure/domains/health/domain/entities/resource.js +159 -0
  46. package/infrastructure/domains/health/domain/entities/resource.test.js +432 -0
  47. package/infrastructure/domains/health/domain/entities/stack-health-report.js +306 -0
  48. package/infrastructure/domains/health/domain/entities/stack-health-report.test.js +601 -0
  49. package/infrastructure/domains/health/domain/services/__tests__/health-score-percentage-based.test.js +380 -0
  50. package/infrastructure/domains/health/domain/services/__tests__/import-progress-monitor.test.js +971 -0
  51. package/infrastructure/domains/health/domain/services/__tests__/import-template-generator.test.js +1150 -0
  52. package/infrastructure/domains/health/domain/services/__tests__/logical-id-mapper.test.js +672 -0
  53. package/infrastructure/domains/health/domain/services/__tests__/template-parser.test.js +496 -0
  54. package/infrastructure/domains/health/domain/services/__tests__/update-progress-monitor.test.js +419 -0
  55. package/infrastructure/domains/health/domain/services/health-score-calculator.js +248 -0
  56. package/infrastructure/domains/health/domain/services/health-score-calculator.test.js +504 -0
  57. package/infrastructure/domains/health/domain/services/import-progress-monitor.js +195 -0
  58. package/infrastructure/domains/health/domain/services/import-template-generator.js +435 -0
  59. package/infrastructure/domains/health/domain/services/logical-id-mapper.js +345 -0
  60. package/infrastructure/domains/health/domain/services/mismatch-analyzer.js +234 -0
  61. package/infrastructure/domains/health/domain/services/mismatch-analyzer.test.js +431 -0
  62. package/infrastructure/domains/health/domain/services/property-mutability-config.js +382 -0
  63. package/infrastructure/domains/health/domain/services/template-parser.js +245 -0
  64. package/infrastructure/domains/health/domain/services/update-progress-monitor.js +192 -0
  65. package/infrastructure/domains/health/domain/value-objects/health-score.js +138 -0
  66. package/infrastructure/domains/health/domain/value-objects/health-score.test.js +267 -0
  67. package/infrastructure/domains/health/domain/value-objects/property-mutability.js +161 -0
  68. package/infrastructure/domains/health/domain/value-objects/property-mutability.test.js +198 -0
  69. package/infrastructure/domains/health/domain/value-objects/resource-state.js +167 -0
  70. package/infrastructure/domains/health/domain/value-objects/resource-state.test.js +196 -0
  71. package/infrastructure/domains/health/domain/value-objects/stack-identifier.js +192 -0
  72. package/infrastructure/domains/health/domain/value-objects/stack-identifier.test.js +262 -0
  73. package/infrastructure/domains/health/infrastructure/adapters/__tests__/orphan-detection-cfn-tagged.test.js +312 -0
  74. package/infrastructure/domains/health/infrastructure/adapters/__tests__/orphan-detection-multi-stack.test.js +367 -0
  75. package/infrastructure/domains/health/infrastructure/adapters/__tests__/orphan-detection-relationship-analysis.test.js +432 -0
  76. package/infrastructure/domains/health/infrastructure/adapters/aws-property-reconciler.js +784 -0
  77. package/infrastructure/domains/health/infrastructure/adapters/aws-property-reconciler.test.js +1133 -0
  78. package/infrastructure/domains/health/infrastructure/adapters/aws-resource-detector.js +565 -0
  79. package/infrastructure/domains/health/infrastructure/adapters/aws-resource-detector.test.js +554 -0
  80. package/infrastructure/domains/health/infrastructure/adapters/aws-resource-importer.js +318 -0
  81. package/infrastructure/domains/health/infrastructure/adapters/aws-resource-importer.test.js +398 -0
  82. package/infrastructure/domains/health/infrastructure/adapters/aws-stack-repository.js +777 -0
  83. package/infrastructure/domains/health/infrastructure/adapters/aws-stack-repository.test.js +580 -0
  84. package/infrastructure/domains/integration/integration-builder.js +397 -0
  85. package/infrastructure/domains/integration/integration-builder.test.js +593 -0
  86. package/infrastructure/domains/integration/integration-resolver.js +170 -0
  87. package/infrastructure/domains/integration/integration-resolver.test.js +369 -0
  88. package/infrastructure/domains/integration/websocket-builder.js +69 -0
  89. package/infrastructure/domains/integration/websocket-builder.test.js +195 -0
  90. package/infrastructure/domains/networking/vpc-builder.js +1829 -0
  91. package/infrastructure/domains/networking/vpc-builder.test.js +1262 -0
  92. package/infrastructure/domains/networking/vpc-discovery.js +177 -0
  93. package/infrastructure/domains/networking/vpc-discovery.test.js +350 -0
  94. package/infrastructure/domains/networking/vpc-resolver.js +324 -0
  95. package/infrastructure/domains/networking/vpc-resolver.test.js +501 -0
  96. package/infrastructure/domains/parameters/ssm-builder.js +79 -0
  97. package/infrastructure/domains/parameters/ssm-builder.test.js +189 -0
  98. package/infrastructure/domains/parameters/ssm-discovery.js +84 -0
  99. package/infrastructure/domains/parameters/ssm-discovery.test.js +210 -0
  100. package/infrastructure/{iam-generator.js → domains/security/iam-generator.js} +2 -2
  101. package/infrastructure/domains/security/kms-builder.js +366 -0
  102. package/infrastructure/domains/security/kms-builder.test.js +374 -0
  103. package/infrastructure/domains/security/kms-discovery.js +80 -0
  104. package/infrastructure/domains/security/kms-discovery.test.js +177 -0
  105. package/infrastructure/domains/security/kms-resolver.js +96 -0
  106. package/infrastructure/domains/security/kms-resolver.test.js +216 -0
  107. package/infrastructure/domains/shared/base-builder.js +112 -0
  108. package/infrastructure/domains/shared/base-resolver.js +186 -0
  109. package/infrastructure/domains/shared/base-resolver.test.js +305 -0
  110. package/infrastructure/domains/shared/builder-orchestrator.js +212 -0
  111. package/infrastructure/domains/shared/builder-orchestrator.test.js +213 -0
  112. package/infrastructure/domains/shared/cloudformation-discovery-v2.js +334 -0
  113. package/infrastructure/domains/shared/cloudformation-discovery.js +375 -0
  114. package/infrastructure/domains/shared/cloudformation-discovery.test.js +590 -0
  115. package/infrastructure/domains/shared/environment-builder.js +119 -0
  116. package/infrastructure/domains/shared/environment-builder.test.js +247 -0
  117. package/infrastructure/domains/shared/providers/aws-provider-adapter.js +544 -0
  118. package/infrastructure/domains/shared/providers/aws-provider-adapter.test.js +377 -0
  119. package/infrastructure/domains/shared/providers/azure-provider-adapter.stub.js +93 -0
  120. package/infrastructure/domains/shared/providers/cloud-provider-adapter.js +136 -0
  121. package/infrastructure/domains/shared/providers/gcp-provider-adapter.stub.js +82 -0
  122. package/infrastructure/domains/shared/providers/provider-factory.js +108 -0
  123. package/infrastructure/domains/shared/providers/provider-factory.test.js +170 -0
  124. package/infrastructure/domains/shared/resource-discovery.js +192 -0
  125. package/infrastructure/domains/shared/resource-discovery.test.js +552 -0
  126. package/infrastructure/domains/shared/types/app-definition.js +205 -0
  127. package/infrastructure/domains/shared/types/discovery-result.js +106 -0
  128. package/infrastructure/domains/shared/types/discovery-result.test.js +258 -0
  129. package/infrastructure/domains/shared/types/index.js +46 -0
  130. package/infrastructure/domains/shared/types/resource-ownership.js +108 -0
  131. package/infrastructure/domains/shared/types/resource-ownership.test.js +101 -0
  132. package/infrastructure/domains/shared/utilities/base-definition-factory.js +380 -0
  133. package/infrastructure/domains/shared/utilities/base-definition-factory.js.bak +338 -0
  134. package/infrastructure/domains/shared/utilities/base-definition-factory.test.js +248 -0
  135. package/infrastructure/domains/shared/utilities/handler-path-resolver.js +134 -0
  136. package/infrastructure/domains/shared/utilities/handler-path-resolver.test.js +268 -0
  137. package/infrastructure/domains/shared/utilities/prisma-layer-manager.js +55 -0
  138. package/infrastructure/domains/shared/utilities/prisma-layer-manager.test.js +138 -0
  139. package/infrastructure/{env-validator.js → domains/shared/validation/env-validator.js} +2 -1
  140. package/infrastructure/domains/shared/validation/env-validator.test.js +173 -0
  141. package/infrastructure/esbuild.config.js +53 -0
  142. package/infrastructure/infrastructure-composer.js +87 -0
  143. package/infrastructure/{serverless-template.test.js → infrastructure-composer.test.js} +115 -24
  144. package/infrastructure/scripts/build-prisma-layer.js +553 -0
  145. package/infrastructure/scripts/build-prisma-layer.test.js +102 -0
  146. package/infrastructure/{build-time-discovery.js → scripts/build-time-discovery.js} +80 -48
  147. package/infrastructure/{build-time-discovery.test.js → scripts/build-time-discovery.test.js} +5 -4
  148. package/layers/prisma/nodejs/package.json +8 -0
  149. package/management-ui/server/utils/cliIntegration.js +1 -1
  150. package/management-ui/server/utils/environment/awsParameterStore.js +29 -18
  151. package/package.json +11 -11
  152. package/frigg-cli/.eslintrc.js +0 -141
  153. package/frigg-cli/__tests__/unit/commands/build.test.js +0 -251
  154. package/frigg-cli/__tests__/unit/commands/db-setup.test.js +0 -548
  155. package/frigg-cli/__tests__/unit/commands/install.test.js +0 -400
  156. package/frigg-cli/__tests__/unit/commands/ui.test.js +0 -346
  157. package/frigg-cli/__tests__/unit/utils/database-validator.test.js +0 -366
  158. package/frigg-cli/__tests__/unit/utils/error-messages.test.js +0 -304
  159. package/frigg-cli/__tests__/unit/utils/prisma-runner.test.js +0 -486
  160. package/frigg-cli/__tests__/utils/mock-factory.js +0 -270
  161. package/frigg-cli/__tests__/utils/prisma-mock.js +0 -194
  162. package/frigg-cli/__tests__/utils/test-fixtures.js +0 -463
  163. package/frigg-cli/__tests__/utils/test-setup.js +0 -287
  164. package/frigg-cli/build-command/index.js +0 -65
  165. package/frigg-cli/db-setup-command/index.js +0 -193
  166. package/frigg-cli/deploy-command/index.js +0 -175
  167. package/frigg-cli/generate-command/__tests__/generate-command.test.js +0 -301
  168. package/frigg-cli/generate-command/azure-generator.js +0 -43
  169. package/frigg-cli/generate-command/gcp-generator.js +0 -47
  170. package/frigg-cli/generate-command/index.js +0 -332
  171. package/frigg-cli/generate-command/terraform-generator.js +0 -555
  172. package/frigg-cli/generate-iam-command.js +0 -118
  173. package/frigg-cli/index.js +0 -75
  174. package/frigg-cli/index.test.js +0 -158
  175. package/frigg-cli/init-command/backend-first-handler.js +0 -756
  176. package/frigg-cli/init-command/index.js +0 -93
  177. package/frigg-cli/init-command/template-handler.js +0 -143
  178. package/frigg-cli/install-command/backend-js.js +0 -33
  179. package/frigg-cli/install-command/commit-changes.js +0 -16
  180. package/frigg-cli/install-command/environment-variables.js +0 -127
  181. package/frigg-cli/install-command/environment-variables.test.js +0 -136
  182. package/frigg-cli/install-command/index.js +0 -54
  183. package/frigg-cli/install-command/install-package.js +0 -13
  184. package/frigg-cli/install-command/integration-file.js +0 -30
  185. package/frigg-cli/install-command/logger.js +0 -12
  186. package/frigg-cli/install-command/template.js +0 -90
  187. package/frigg-cli/install-command/validate-package.js +0 -75
  188. package/frigg-cli/jest.config.js +0 -124
  189. package/frigg-cli/package.json +0 -54
  190. package/frigg-cli/start-command/index.js +0 -149
  191. package/frigg-cli/start-command/start-command.test.js +0 -297
  192. package/frigg-cli/test/init-command.test.js +0 -180
  193. package/frigg-cli/test/npm-registry.test.js +0 -319
  194. package/frigg-cli/ui-command/index.js +0 -154
  195. package/frigg-cli/utils/app-resolver.js +0 -319
  196. package/frigg-cli/utils/backend-path.js +0 -25
  197. package/frigg-cli/utils/database-validator.js +0 -161
  198. package/frigg-cli/utils/error-messages.js +0 -257
  199. package/frigg-cli/utils/npm-registry.js +0 -167
  200. package/frigg-cli/utils/prisma-runner.js +0 -280
  201. package/frigg-cli/utils/process-manager.js +0 -199
  202. package/frigg-cli/utils/repo-detection.js +0 -405
  203. package/infrastructure/aws-discovery.js +0 -1176
  204. package/infrastructure/aws-discovery.test.js +0 -1220
  205. package/infrastructure/serverless-template.js +0 -2094
  206. /package/infrastructure/{WEBSOCKET-CONFIGURATION.md → docs/WEBSOCKET-CONFIGURATION.md} +0 -0
  207. /package/infrastructure/{GENERATE-IAM-DOCS.md → docs/generate-iam-command.md} +0 -0
  208. /package/infrastructure/{iam-generator.test.js → domains/security/iam-generator.test.js} +0 -0
  209. /package/infrastructure/{frigg-deployment-iam-stack.yaml → domains/security/templates/frigg-deployment-iam-stack.yaml} +0 -0
  210. /package/infrastructure/{iam-policy-basic.json → domains/security/templates/iam-policy-basic.json} +0 -0
  211. /package/infrastructure/{iam-policy-full.json → domains/security/templates/iam-policy-full.json} +0 -0
  212. /package/infrastructure/{run-discovery.js → scripts/run-discovery.js} +0 -0
@@ -1,319 +0,0 @@
1
- const fs = require('fs').promises;
2
- const path = require('path');
3
- const os = require('os');
4
-
5
- class AppResolver {
6
- constructor() {
7
- this.cache = new Map();
8
- }
9
-
10
- async resolveAppPath(options = {}) {
11
- const cacheKey = JSON.stringify(options);
12
- if (this.cache.has(cacheKey)) {
13
- return this.cache.get(cacheKey);
14
- }
15
-
16
- let resolvedPath;
17
-
18
- // Priority 1: Explicit flags (--app-path, --config, --app)
19
- if (options.appPath || options.config || options.app) {
20
- const explicitPath = options.appPath || options.config || options.app;
21
- resolvedPath = await this.validateAndResolvePath(explicitPath);
22
- if (resolvedPath) {
23
- this.cache.set(cacheKey, resolvedPath);
24
- return resolvedPath;
25
- }
26
- throw new Error(`Invalid app path specified: ${explicitPath}`);
27
- }
28
-
29
- // Priority 2: Environment variable
30
- if (process.env.FRIGG_APP_PATH) {
31
- resolvedPath = await this.validateAndResolvePath(process.env.FRIGG_APP_PATH);
32
- if (resolvedPath) {
33
- this.cache.set(cacheKey, resolvedPath);
34
- return resolvedPath;
35
- }
36
- console.warn(`Warning: FRIGG_APP_PATH environment variable points to invalid path: ${process.env.FRIGG_APP_PATH}`);
37
- }
38
-
39
- // Priority 3: Current directory auto-detection (backward compatibility)
40
- resolvedPath = await this.autoDetectFriggApp();
41
- if (resolvedPath) {
42
- this.cache.set(cacheKey, resolvedPath);
43
- return resolvedPath;
44
- }
45
-
46
- // Priority 4: Search common development directories
47
- resolvedPath = await this.searchCommonDirectories();
48
- if (resolvedPath) {
49
- this.cache.set(cacheKey, resolvedPath);
50
- return resolvedPath;
51
- }
52
-
53
- throw new Error('No Frigg application found. Use --app-path to specify the application directory.');
54
- }
55
-
56
- async validateAndResolvePath(inputPath) {
57
- if (!inputPath) return null;
58
-
59
- // Handle different path formats
60
- let resolvedPath;
61
- if (inputPath.startsWith('~/')) {
62
- resolvedPath = path.join(os.homedir(), inputPath.slice(2));
63
- } else if (path.isAbsolute(inputPath)) {
64
- resolvedPath = inputPath;
65
- } else {
66
- resolvedPath = path.resolve(process.cwd(), inputPath);
67
- }
68
-
69
- try {
70
- const stats = await fs.stat(resolvedPath);
71
- if (!stats.isDirectory()) {
72
- // If it's a file, check if it's a config file and use its directory
73
- if (await this.isConfigFile(resolvedPath)) {
74
- resolvedPath = path.dirname(resolvedPath);
75
- } else {
76
- return null;
77
- }
78
- }
79
-
80
- // Validate that this is a Frigg application
81
- if (await this.isFriggApplication(resolvedPath)) {
82
- return resolvedPath;
83
- }
84
- } catch (error) {
85
- // Path doesn't exist or is not accessible
86
- return null;
87
- }
88
-
89
- return null;
90
- }
91
-
92
- async isConfigFile(filePath) {
93
- const basename = path.basename(filePath);
94
- const configFiles = [
95
- 'frigg.config.js',
96
- 'frigg.config.json',
97
- '.friggrc',
98
- '.friggrc.js',
99
- '.friggrc.json',
100
- 'package.json'
101
- ];
102
-
103
- return configFiles.includes(basename);
104
- }
105
-
106
- async isFriggApplication(dirPath) {
107
- try {
108
- // Check for package.json with Frigg dependencies
109
- const packageJsonPath = path.join(dirPath, 'package.json');
110
- try {
111
- const packageJsonContent = await fs.readFile(packageJsonPath, 'utf8');
112
- const packageJson = JSON.parse(packageJsonContent);
113
-
114
- // Check for @friggframework dependencies
115
- const deps = {
116
- ...packageJson.dependencies,
117
- ...packageJson.devDependencies,
118
- ...packageJson.peerDependencies
119
- };
120
-
121
- if (Object.keys(deps).some(dep => dep.startsWith('@friggframework/'))) {
122
- return true;
123
- }
124
-
125
- // Check for frigg scripts
126
- if (packageJson.scripts) {
127
- const scriptNames = Object.keys(packageJson.scripts);
128
- if (scriptNames.some(script => script.includes('frigg'))) {
129
- return true;
130
- }
131
- }
132
- } catch (error) {
133
- // package.json doesn't exist or is invalid, continue checking other indicators
134
- }
135
-
136
- // Check for Frigg configuration files
137
- const configFiles = [
138
- 'frigg.config.js',
139
- 'frigg.config.json',
140
- '.friggrc',
141
- '.friggrc.js',
142
- '.friggrc.json'
143
- ];
144
-
145
- for (const configFile of configFiles) {
146
- try {
147
- await fs.access(path.join(dirPath, configFile));
148
- return true;
149
- } catch (error) {
150
- // File doesn't exist, continue
151
- }
152
- }
153
-
154
- // Check for Frigg-specific directories
155
- const friggDirectories = [
156
- '.frigg',
157
- 'frigg-modules',
158
- 'api-modules'
159
- ];
160
-
161
- for (const friggDir of friggDirectories) {
162
- try {
163
- const dirStat = await fs.stat(path.join(dirPath, friggDir));
164
- if (dirStat.isDirectory()) {
165
- return true;
166
- }
167
- } catch (error) {
168
- // Directory doesn't exist, continue
169
- }
170
- }
171
-
172
- // Check for serverless.yml with Frigg references
173
- try {
174
- const serverlessPath = path.join(dirPath, 'serverless.yml');
175
- const serverlessContent = await fs.readFile(serverlessPath, 'utf8');
176
- if (serverlessContent.includes('frigg') || serverlessContent.includes('Frigg')) {
177
- return true;
178
- }
179
- } catch (error) {
180
- // serverless.yml doesn't exist or can't be read
181
- }
182
-
183
- // Check for infrastructure.js (common in Frigg apps)
184
- try {
185
- await fs.access(path.join(dirPath, 'infrastructure.js'));
186
- return true;
187
- } catch (error) {
188
- // infrastructure.js doesn't exist
189
- }
190
-
191
- return false;
192
- } catch (error) {
193
- return false;
194
- }
195
- }
196
-
197
- async autoDetectFriggApp() {
198
- // Start from current directory and search up to 3 levels
199
- let currentDir = process.cwd();
200
-
201
- for (let i = 0; i < 3; i++) {
202
- if (await this.isFriggApplication(currentDir)) {
203
- return currentDir;
204
- }
205
-
206
- const parentDir = path.dirname(currentDir);
207
- if (parentDir === currentDir) {
208
- // Reached filesystem root
209
- break;
210
- }
211
- currentDir = parentDir;
212
- }
213
-
214
- return null;
215
- }
216
-
217
- async searchCommonDirectories() {
218
- const commonDirs = [
219
- path.join(os.homedir(), 'Documents'),
220
- path.join(os.homedir(), 'Projects'),
221
- path.join(os.homedir(), 'Development'),
222
- path.join(os.homedir(), 'dev'),
223
- path.join(os.homedir(), 'workspace')
224
- ];
225
-
226
- for (const baseDir of commonDirs) {
227
- try {
228
- const friggApp = await this.searchDirectoryRecursively(baseDir, 3);
229
- if (friggApp) {
230
- return friggApp;
231
- }
232
- } catch (error) {
233
- // Directory doesn't exist or can't be accessed, continue
234
- }
235
- }
236
-
237
- return null;
238
- }
239
-
240
- async searchDirectoryRecursively(dirPath, maxDepth) {
241
- if (maxDepth <= 0) return null;
242
-
243
- try {
244
- const entries = await fs.readdir(dirPath, { withFileTypes: true });
245
-
246
- // First check if current directory is a Frigg app
247
- if (await this.isFriggApplication(dirPath)) {
248
- return dirPath;
249
- }
250
-
251
- // Then search subdirectories
252
- for (const entry of entries) {
253
- if (!entry.isDirectory()) continue;
254
-
255
- // Skip common directories that shouldn't contain Frigg apps
256
- const skipDirs = ['node_modules', '.git', 'dist', 'build', '.next', 'coverage'];
257
- if (skipDirs.includes(entry.name)) continue;
258
-
259
- const subDirPath = path.join(dirPath, entry.name);
260
- const result = await this.searchDirectoryRecursively(subDirPath, maxDepth - 1);
261
- if (result) {
262
- return result;
263
- }
264
- }
265
- } catch (error) {
266
- // Directory can't be read, skip
267
- }
268
-
269
- return null;
270
- }
271
-
272
- async loadAppConfig(appPath) {
273
- const configPaths = [
274
- path.join(appPath, 'frigg.config.js'),
275
- path.join(appPath, 'frigg.config.json'),
276
- path.join(appPath, '.friggrc.js'),
277
- path.join(appPath, '.friggrc.json'),
278
- path.join(appPath, '.friggrc')
279
- ];
280
-
281
- for (const configPath of configPaths) {
282
- try {
283
- const stats = await fs.stat(configPath);
284
- if (stats.isFile()) {
285
- if (configPath.endsWith('.js')) {
286
- delete require.cache[require.resolve(configPath)];
287
- return require(configPath);
288
- } else {
289
- const content = await fs.readFile(configPath, 'utf8');
290
- return JSON.parse(content);
291
- }
292
- }
293
- } catch (error) {
294
- // Config file doesn't exist or can't be read, continue
295
- }
296
- }
297
-
298
- // Fallback to package.json frigg configuration
299
- try {
300
- const packageJsonPath = path.join(appPath, 'package.json');
301
- const packageJsonContent = await fs.readFile(packageJsonPath, 'utf8');
302
- const packageJson = JSON.parse(packageJsonContent);
303
-
304
- if (packageJson.frigg) {
305
- return packageJson.frigg;
306
- }
307
- } catch (error) {
308
- // package.json doesn't exist or doesn't have frigg config
309
- }
310
-
311
- return {};
312
- }
313
-
314
- clearCache() {
315
- this.cache.clear();
316
- }
317
- }
318
-
319
- module.exports = { AppResolver };
@@ -1,25 +0,0 @@
1
- const path = require('path');
2
- const fs = require('fs');
3
-
4
- /**
5
- * Finds the nearest backend package.json by traversing up the directory tree
6
- * @param {string} startDir - Directory to start searching from
7
- * @returns {string|null} - Path to the backend directory or null if not found
8
- */
9
- function findNearestBackendPackageJson(startDir = process.cwd()) {
10
- let currentDir = startDir;
11
-
12
- while (currentDir !== path.dirname(currentDir)) {
13
- const backendPath = path.join(currentDir, 'backend', 'package.json');
14
- if (fs.existsSync(backendPath)) {
15
- return path.dirname(backendPath);
16
- }
17
- currentDir = path.dirname(currentDir);
18
- }
19
-
20
- return null;
21
- }
22
-
23
- module.exports = {
24
- findNearestBackendPackageJson
25
- };
@@ -1,161 +0,0 @@
1
- const path = require('path');
2
- const fs = require('fs');
3
- const { getDatabaseType: getDatabaseTypeFromCore } = require('@friggframework/core/database/config');
4
- const { connectPrisma, disconnectPrisma } = require('@friggframework/core/database/prisma');
5
-
6
- /**
7
- * Database Validation Utility
8
- * Validates database configuration and connectivity for Frigg applications
9
- */
10
-
11
- /**
12
- * Validates that DATABASE_URL environment variable exists and has a value
13
- * @returns {Object} { valid: boolean, url?: string, error?: string }
14
- */
15
- function validateDatabaseUrl() {
16
- const databaseUrl = process.env.DATABASE_URL;
17
-
18
- if (!databaseUrl) {
19
- return {
20
- valid: false,
21
- error: 'DATABASE_URL environment variable not found'
22
- };
23
- }
24
-
25
- if (databaseUrl.trim() === '') {
26
- return {
27
- valid: false,
28
- error: 'DATABASE_URL environment variable is empty'
29
- };
30
- }
31
-
32
- return {
33
- valid: true,
34
- url: databaseUrl
35
- };
36
- }
37
-
38
- /**
39
- * Determines database type from backend app definition
40
- * Reuses core logic from @friggframework/core/database/config
41
- *
42
- * @returns {Object} { dbType?: 'mongodb'|'postgresql', error?: string }
43
- */
44
- function getDatabaseType() {
45
- try {
46
- // Use the core database config logic (same logic used at runtime)
47
- const dbType = getDatabaseTypeFromCore();
48
- return { dbType };
49
- } catch (error) {
50
- // Convert thrown errors to error object format for CLI
51
- // Strip [Frigg] prefix from error messages for cleaner CLI output
52
- const errorMessage = error.message.replace(/^\[Frigg\]\s*/, '');
53
- return {
54
- error: errorMessage,
55
- stack: error.stack // Include stack trace for debugging
56
- };
57
- }
58
- }
59
-
60
- /**
61
- * Tests database connectivity by attempting to connect
62
- * Uses the same Prisma client configuration as runtime
63
- *
64
- * @param {string} databaseUrl - Database connection URL (for validation purposes)
65
- * @param {'mongodb'|'postgresql'} dbType - Database type to determine appropriate health check
66
- * @param {number} timeout - Connection timeout in milliseconds (default: 5000)
67
- * @returns {Promise<Object>} { connected: boolean, error?: string }
68
- */
69
- async function testDatabaseConnection(databaseUrl, dbType, timeout = 5000) {
70
- try {
71
- // Use the core Prisma client (same client used at runtime)
72
- // This automatically uses the correct client based on DB_TYPE
73
- const connectPromise = connectPrisma();
74
- const timeoutPromise = new Promise((_, reject) =>
75
- setTimeout(() => reject(new Error('Connection timeout')), timeout)
76
- );
77
-
78
- const client = await Promise.race([connectPromise, timeoutPromise]);
79
-
80
- // Test with database-appropriate health check
81
- // MongoDB doesn't support SQL, so we use the native ping command
82
- if (dbType === 'mongodb') {
83
- // Use MongoDB's native ping command via $runCommandRaw
84
- await client.$runCommandRaw({ ping: 1 });
85
- } else {
86
- // PostgreSQL: use a simple SQL query
87
- await client.$queryRaw`SELECT 1`;
88
- }
89
-
90
- await disconnectPrisma();
91
-
92
- return { connected: true };
93
-
94
- } catch (error) {
95
- try {
96
- await disconnectPrisma();
97
- } catch (disconnectError) {
98
- // Ignore disconnect errors
99
- }
100
-
101
- return {
102
- connected: false,
103
- error: error.message
104
- };
105
- }
106
- }
107
-
108
- /**
109
- * Checks if Prisma client is generated for the database type
110
- * Uses require.resolve to find the client in node_modules
111
- *
112
- * @param {'mongodb'|'postgresql'} dbType - Database type
113
- * @param {string} projectRoot - Project root directory (used for require.resolve context)
114
- * @returns {Object} { generated: boolean, path?: string, error?: string }
115
- */
116
- function checkPrismaClientGenerated(dbType, projectRoot = process.cwd()) {
117
- const clientPackageName = `@prisma-${dbType}/client`;
118
-
119
- try {
120
- // First, resolve where @friggframework/core actually is
121
- // This handles file: dependencies and symlinks correctly
122
- const corePackagePath = require.resolve('@friggframework/core', {
123
- paths: [projectRoot]
124
- });
125
- const corePackageDir = path.dirname(corePackagePath);
126
-
127
- // Now look for the Prisma client within the resolved core package
128
- const clientPath = require.resolve(clientPackageName, {
129
- paths: [
130
- corePackageDir, // Look in the actual core package location
131
- projectRoot // Fallback to project root
132
- ]
133
- });
134
-
135
- return {
136
- generated: true,
137
- path: path.dirname(clientPath)
138
- };
139
-
140
- } catch (error) {
141
- // require.resolve throws MODULE_NOT_FOUND if the client doesn't exist
142
- if (error.code === 'MODULE_NOT_FOUND') {
143
- return {
144
- generated: false,
145
- error: `Prisma client for ${dbType} (${clientPackageName}) not found. Run 'frigg db:setup' to generate it.`
146
- };
147
- }
148
-
149
- return {
150
- generated: false,
151
- error: `Failed to check Prisma client: ${error.message}`
152
- };
153
- }
154
- }
155
-
156
- module.exports = {
157
- validateDatabaseUrl,
158
- getDatabaseType,
159
- testDatabaseConnection,
160
- checkPrismaClientGenerated
161
- };