@friggframework/devtools 2.0.0--canary.546.74db90f.0 → 2.0.0--canary.545.e7becd9.0

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 (128) hide show
  1. package/frigg-cli/README.md +1 -1
  2. package/frigg-cli/__tests__/application/use-cases/AddApiModuleToIntegrationUseCase.test.js +326 -0
  3. package/frigg-cli/__tests__/application/use-cases/CreateApiModuleUseCase.test.js +337 -0
  4. package/frigg-cli/__tests__/domain/entities/ApiModule.test.js +373 -0
  5. package/frigg-cli/__tests__/domain/entities/AppDefinition.test.js +313 -0
  6. package/frigg-cli/__tests__/domain/services/IntegrationValidator.test.js +269 -0
  7. package/frigg-cli/__tests__/domain/value-objects/IntegrationName.test.js +82 -0
  8. package/frigg-cli/__tests__/infrastructure/adapters/IntegrationJsUpdater.test.js +408 -0
  9. package/frigg-cli/__tests__/infrastructure/repositories/FileSystemApiModuleRepository.test.js +583 -0
  10. package/frigg-cli/__tests__/infrastructure/repositories/FileSystemAppDefinitionRepository.test.js +314 -0
  11. package/frigg-cli/__tests__/infrastructure/repositories/FileSystemIntegrationRepository.test.js +383 -0
  12. package/frigg-cli/__tests__/unit/commands/build.test.js +1 -1
  13. package/frigg-cli/__tests__/unit/commands/doctor.test.js +0 -2
  14. package/frigg-cli/__tests__/unit/commands/init.test.js +406 -0
  15. package/frigg-cli/__tests__/unit/commands/install.test.js +23 -19
  16. package/frigg-cli/__tests__/unit/commands/provider-dispatch.test.js +383 -0
  17. package/frigg-cli/__tests__/unit/commands/repair.test.js +275 -0
  18. package/frigg-cli/__tests__/unit/dependencies.test.js +2 -2
  19. package/frigg-cli/__tests__/unit/start-command/application/RunPreflightChecksUseCase.test.js +411 -0
  20. package/frigg-cli/__tests__/unit/start-command/infrastructure/DatabaseAdapter.test.js +405 -0
  21. package/frigg-cli/__tests__/unit/start-command/infrastructure/DockerAdapter.test.js +496 -0
  22. package/frigg-cli/__tests__/unit/start-command/presentation/InteractivePromptAdapter.test.js +474 -0
  23. package/frigg-cli/__tests__/unit/utils/output.test.js +196 -0
  24. package/frigg-cli/application/use-cases/AddApiModuleToIntegrationUseCase.js +93 -0
  25. package/frigg-cli/application/use-cases/CreateApiModuleUseCase.js +93 -0
  26. package/frigg-cli/application/use-cases/CreateIntegrationUseCase.js +103 -0
  27. package/frigg-cli/build-command/index.js +123 -11
  28. package/frigg-cli/container.js +172 -0
  29. package/frigg-cli/deploy-command/index.js +83 -1
  30. package/frigg-cli/docs/OUTPUT_MIGRATION_GUIDE.md +286 -0
  31. package/frigg-cli/doctor-command/index.js +37 -16
  32. package/frigg-cli/domain/entities/ApiModule.js +272 -0
  33. package/frigg-cli/domain/entities/AppDefinition.js +227 -0
  34. package/frigg-cli/domain/entities/Integration.js +198 -0
  35. package/frigg-cli/domain/exceptions/DomainException.js +24 -0
  36. package/frigg-cli/domain/ports/IApiModuleRepository.js +53 -0
  37. package/frigg-cli/domain/ports/IAppDefinitionRepository.js +43 -0
  38. package/frigg-cli/domain/ports/IIntegrationRepository.js +61 -0
  39. package/frigg-cli/domain/services/IntegrationValidator.js +185 -0
  40. package/frigg-cli/domain/value-objects/IntegrationId.js +42 -0
  41. package/frigg-cli/domain/value-objects/IntegrationName.js +60 -0
  42. package/frigg-cli/domain/value-objects/SemanticVersion.js +70 -0
  43. package/frigg-cli/generate-iam-command.js +21 -1
  44. package/frigg-cli/index.js +21 -6
  45. package/frigg-cli/index.test.js +7 -2
  46. package/frigg-cli/infrastructure/UnitOfWork.js +46 -0
  47. package/frigg-cli/infrastructure/adapters/BackendJsUpdater.js +197 -0
  48. package/frigg-cli/infrastructure/adapters/FileSystemAdapter.js +224 -0
  49. package/frigg-cli/infrastructure/adapters/IntegrationJsUpdater.js +249 -0
  50. package/frigg-cli/infrastructure/adapters/SchemaValidator.js +92 -0
  51. package/frigg-cli/infrastructure/repositories/FileSystemApiModuleRepository.js +373 -0
  52. package/frigg-cli/infrastructure/repositories/FileSystemAppDefinitionRepository.js +116 -0
  53. package/frigg-cli/infrastructure/repositories/FileSystemIntegrationRepository.js +277 -0
  54. package/frigg-cli/init-command/backend-first-handler.js +124 -42
  55. package/frigg-cli/init-command/index.js +2 -1
  56. package/frigg-cli/init-command/template-handler.js +13 -3
  57. package/frigg-cli/install-command/backend-js.js +3 -3
  58. package/frigg-cli/install-command/environment-variables.js +16 -19
  59. package/frigg-cli/install-command/environment-variables.test.js +12 -13
  60. package/frigg-cli/install-command/index.js +14 -9
  61. package/frigg-cli/install-command/integration-file.js +3 -3
  62. package/frigg-cli/install-command/validate-package.js +5 -9
  63. package/frigg-cli/jest.config.js +4 -1
  64. package/frigg-cli/package-lock.json +16226 -0
  65. package/frigg-cli/repair-command/index.js +121 -128
  66. package/frigg-cli/start-command/application/RunPreflightChecksUseCase.js +376 -0
  67. package/frigg-cli/start-command/index.js +324 -2
  68. package/frigg-cli/start-command/infrastructure/DatabaseAdapter.js +591 -0
  69. package/frigg-cli/start-command/infrastructure/DockerAdapter.js +306 -0
  70. package/frigg-cli/start-command/presentation/InteractivePromptAdapter.js +329 -0
  71. package/frigg-cli/templates/backend/.env.example +62 -0
  72. package/frigg-cli/templates/backend/.eslintrc.json +12 -0
  73. package/frigg-cli/templates/backend/.prettierrc +6 -0
  74. package/frigg-cli/templates/backend/docker-compose.yml +22 -0
  75. package/frigg-cli/templates/backend/index.js +96 -0
  76. package/frigg-cli/templates/backend/infrastructure.js +12 -0
  77. package/frigg-cli/templates/backend/jest.config.js +17 -0
  78. package/frigg-cli/templates/backend/package.json +50 -0
  79. package/frigg-cli/templates/backend/src/api-modules/.gitkeep +10 -0
  80. package/frigg-cli/templates/backend/src/base/.gitkeep +7 -0
  81. package/frigg-cli/templates/backend/src/integrations/.gitkeep +10 -0
  82. package/frigg-cli/templates/backend/src/integrations/ExampleIntegration.js +65 -0
  83. package/frigg-cli/templates/backend/src/utils/.gitkeep +7 -0
  84. package/frigg-cli/templates/backend/test/setup.js +30 -0
  85. package/frigg-cli/templates/backend/ui-extensions/.gitkeep +0 -0
  86. package/frigg-cli/templates/backend/ui-extensions/README.md +77 -0
  87. package/frigg-cli/ui-command/index.js +58 -36
  88. package/frigg-cli/utils/__tests__/provider-helper.test.js +55 -0
  89. package/frigg-cli/utils/__tests__/repo-detection.test.js +436 -0
  90. package/frigg-cli/utils/output.js +382 -0
  91. package/frigg-cli/utils/provider-helper.js +75 -0
  92. package/frigg-cli/utils/repo-detection.js +85 -37
  93. package/frigg-cli/validate-command/__tests__/adapters/validate-command.test.js +205 -0
  94. package/frigg-cli/validate-command/__tests__/application/validate-app-use-case.test.js +104 -0
  95. package/frigg-cli/validate-command/__tests__/domain/fix-suggestion.test.js +153 -0
  96. package/frigg-cli/validate-command/__tests__/domain/validation-error.test.js +162 -0
  97. package/frigg-cli/validate-command/__tests__/domain/validation-result.test.js +152 -0
  98. package/frigg-cli/validate-command/__tests__/infrastructure/api-module-validator.test.js +332 -0
  99. package/frigg-cli/validate-command/__tests__/infrastructure/app-definition-validator.test.js +191 -0
  100. package/frigg-cli/validate-command/__tests__/infrastructure/integration-class-validator.test.js +146 -0
  101. package/frigg-cli/validate-command/__tests__/infrastructure/template-validation.test.js +155 -0
  102. package/frigg-cli/validate-command/adapters/cli/validate-command.js +199 -0
  103. package/frigg-cli/validate-command/application/use-cases/validate-app-use-case.js +35 -0
  104. package/frigg-cli/validate-command/domain/entities/validation-result.js +74 -0
  105. package/frigg-cli/validate-command/domain/value-objects/fix-suggestion.js +74 -0
  106. package/frigg-cli/validate-command/domain/value-objects/validation-error.js +68 -0
  107. package/frigg-cli/validate-command/infrastructure/validators/api-module-validator.js +181 -0
  108. package/frigg-cli/validate-command/infrastructure/validators/app-definition-validator.js +128 -0
  109. package/frigg-cli/validate-command/infrastructure/validators/integration-class-validator.js +113 -0
  110. package/infrastructure/create-frigg-infrastructure.js +93 -0
  111. package/infrastructure/docs/iam-policy-templates.md +1 -1
  112. package/infrastructure/domains/admin-scripts/admin-script-builder.js +200 -0
  113. package/infrastructure/domains/admin-scripts/admin-script-builder.test.js +499 -0
  114. package/infrastructure/domains/admin-scripts/index.js +5 -0
  115. package/infrastructure/domains/networking/vpc-builder.test.js +2 -4
  116. package/infrastructure/domains/networking/vpc-resolver.test.js +1 -1
  117. package/infrastructure/domains/shared/cloudformation-discovery.test.js +4 -7
  118. package/infrastructure/domains/shared/resource-discovery.js +5 -5
  119. package/infrastructure/domains/shared/types/app-definition.js +21 -0
  120. package/infrastructure/domains/shared/types/discovery-result.test.js +1 -1
  121. package/infrastructure/domains/shared/utilities/base-definition-factory.js +10 -1
  122. package/infrastructure/domains/shared/utilities/base-definition-factory.test.js +2 -2
  123. package/infrastructure/infrastructure-composer.js +2 -0
  124. package/infrastructure/infrastructure-composer.test.js +2 -2
  125. package/infrastructure/jest.config.js +16 -0
  126. package/management-ui/README.md +245 -109
  127. package/package.json +8 -7
  128. package/frigg-cli/install-command/logger.js +0 -12
@@ -0,0 +1,436 @@
1
+ /**
2
+ * Repo Detection Unit Tests
3
+ *
4
+ * Tests the Frigg repository detection algorithm to ensure
5
+ * it correctly identifies Frigg apps in various directory structures.
6
+ */
7
+
8
+ const fs = require('fs-extra');
9
+ const path = require('path');
10
+ const os = require('os');
11
+
12
+ const {
13
+ isFriggRepository,
14
+ discoverFriggRepositories,
15
+ getCurrentRepositoryInfo,
16
+ detectFramework,
17
+ } = require('../repo-detection');
18
+
19
+ // Create a temp directory for tests
20
+ let testDir;
21
+
22
+ beforeAll(async () => {
23
+ testDir = path.join(os.tmpdir(), 'frigg-repo-detection-tests');
24
+ await fs.ensureDir(testDir);
25
+ });
26
+
27
+ afterAll(async () => {
28
+ await fs.remove(testDir);
29
+ });
30
+
31
+ beforeEach(async () => {
32
+ // Clean test directory before each test
33
+ await fs.emptyDir(testDir);
34
+ });
35
+
36
+ describe('isFriggRepository', () => {
37
+ describe('positive detection', () => {
38
+ it('detects a root-level Frigg app with index.js', async () => {
39
+ const appDir = path.join(testDir, 'my-frigg-app');
40
+ await fs.ensureDir(appDir);
41
+
42
+ // Create package.json with @friggframework/core
43
+ await fs.writeJSON(path.join(appDir, 'package.json'), {
44
+ name: 'my-frigg-app',
45
+ version: '1.0.0',
46
+ dependencies: {
47
+ '@friggframework/core': '2.0.0-next.58',
48
+ },
49
+ });
50
+
51
+ // Create index.js (required for detection)
52
+ await fs.writeFile(path.join(appDir, 'index.js'), 'module.exports = {};');
53
+
54
+ const result = await isFriggRepository(appDir);
55
+
56
+ expect(result.isFriggRepo).toBe(true);
57
+ expect(result.repoInfo.name).toBe('my-frigg-app');
58
+ expect(result.repoInfo.path).toBe(appDir);
59
+ expect(result.repoInfo.friggDependencies).toContain('@friggframework/core');
60
+ });
61
+
62
+ it('detects a Frigg app with backend directory structure', async () => {
63
+ const appDir = path.join(testDir, 'workspace-frigg-app');
64
+ const backendDir = path.join(appDir, 'backend');
65
+ await fs.ensureDir(backendDir);
66
+
67
+ // Root package.json without core (workspace pattern)
68
+ await fs.writeJSON(path.join(appDir, 'package.json'), {
69
+ name: 'workspace-frigg-app',
70
+ version: '1.0.0',
71
+ workspaces: ['backend', 'frontend'],
72
+ });
73
+
74
+ // Backend package.json with core dependency
75
+ await fs.writeJSON(path.join(backendDir, 'package.json'), {
76
+ name: 'workspace-frigg-app-backend',
77
+ version: '1.0.0',
78
+ dependencies: {
79
+ '@friggframework/core': '^2.0.0',
80
+ },
81
+ });
82
+
83
+ // Backend index.js
84
+ await fs.writeFile(path.join(backendDir, 'index.js'), 'module.exports = {};');
85
+
86
+ const result = await isFriggRepository(appDir);
87
+
88
+ expect(result.isFriggRepo).toBe(true);
89
+ expect(result.repoInfo.name).toBe('workspace-frigg-app');
90
+ // Path should point to backend when using workspace structure
91
+ expect(result.repoInfo.path).toBe(backendDir);
92
+ });
93
+
94
+ it('detects a Frigg app with backend/serverless.yml', async () => {
95
+ const appDir = path.join(testDir, 'serverless-frigg-app');
96
+ const backendDir = path.join(appDir, 'backend');
97
+ await fs.ensureDir(backendDir);
98
+
99
+ await fs.writeJSON(path.join(appDir, 'package.json'), {
100
+ name: 'serverless-frigg-app',
101
+ dependencies: {
102
+ '@friggframework/core': '2.0.0-next.58',
103
+ },
104
+ });
105
+
106
+ // serverless.yml instead of index.js
107
+ await fs.writeFile(path.join(backendDir, 'serverless.yml'), 'service: my-frigg-app');
108
+
109
+ const result = await isFriggRepository(appDir);
110
+
111
+ expect(result.isFriggRepo).toBe(true);
112
+ expect(result.repoInfo.path).toBe(backendDir);
113
+ });
114
+
115
+ it('detects various version formats for @friggframework/core', async () => {
116
+ const versionFormats = [
117
+ '2.0.0-next.58',
118
+ '^2.0.0',
119
+ '~2.0.0',
120
+ '2.0.0',
121
+ 'next',
122
+ ];
123
+
124
+ for (const version of versionFormats) {
125
+ const appDir = path.join(testDir, `version-test-${version.replace(/[^a-z0-9]/gi, '-')}`);
126
+ await fs.ensureDir(appDir);
127
+
128
+ await fs.writeJSON(path.join(appDir, 'package.json'), {
129
+ name: 'version-test-app',
130
+ dependencies: {
131
+ '@friggframework/core': version,
132
+ },
133
+ });
134
+
135
+ await fs.writeFile(path.join(appDir, 'index.js'), 'module.exports = {};');
136
+
137
+ const result = await isFriggRepository(appDir);
138
+ expect(result.isFriggRepo).toBe(true);
139
+ }
140
+ });
141
+
142
+ it('includes additional frigg dependencies in repoInfo', async () => {
143
+ const appDir = path.join(testDir, 'multi-dep-app');
144
+ await fs.ensureDir(appDir);
145
+
146
+ await fs.writeJSON(path.join(appDir, 'package.json'), {
147
+ name: 'multi-dep-app',
148
+ dependencies: {
149
+ '@friggframework/core': '2.0.0-next.58',
150
+ },
151
+ devDependencies: {
152
+ '@friggframework/devtools': '2.0.0-next.58',
153
+ '@friggframework/test': '2.0.0-next.58',
154
+ },
155
+ });
156
+
157
+ await fs.writeFile(path.join(appDir, 'index.js'), 'module.exports = {};');
158
+
159
+ const result = await isFriggRepository(appDir);
160
+
161
+ expect(result.isFriggRepo).toBe(true);
162
+ expect(result.repoInfo.friggDependencies).toContain('@friggframework/core');
163
+ expect(result.repoInfo.friggDependencies).toContain('@friggframework/devtools');
164
+ expect(result.repoInfo.friggDependencies).toContain('@friggframework/test');
165
+ });
166
+ });
167
+
168
+ describe('negative detection', () => {
169
+ it('rejects directories without package.json', async () => {
170
+ const appDir = path.join(testDir, 'no-package-json');
171
+ await fs.ensureDir(appDir);
172
+ await fs.writeFile(path.join(appDir, 'index.js'), 'module.exports = {};');
173
+
174
+ const result = await isFriggRepository(appDir);
175
+
176
+ expect(result.isFriggRepo).toBe(false);
177
+ });
178
+
179
+ it('rejects directories without @friggframework/core', async () => {
180
+ const appDir = path.join(testDir, 'no-frigg-core');
181
+ await fs.ensureDir(appDir);
182
+
183
+ await fs.writeJSON(path.join(appDir, 'package.json'), {
184
+ name: 'no-frigg-core',
185
+ dependencies: {
186
+ express: '^4.0.0',
187
+ },
188
+ });
189
+
190
+ await fs.writeFile(path.join(appDir, 'index.js'), 'module.exports = {};');
191
+
192
+ const result = await isFriggRepository(appDir);
193
+
194
+ expect(result.isFriggRepo).toBe(false);
195
+ });
196
+
197
+ it('rejects @friggframework/* packages themselves', async () => {
198
+ const appDir = path.join(testDir, 'framework-package');
199
+ await fs.ensureDir(appDir);
200
+
201
+ await fs.writeJSON(path.join(appDir, 'package.json'), {
202
+ name: '@friggframework/core',
203
+ version: '2.0.0',
204
+ dependencies: {},
205
+ });
206
+
207
+ await fs.writeFile(path.join(appDir, 'index.js'), 'module.exports = {};');
208
+
209
+ const result = await isFriggRepository(appDir);
210
+
211
+ expect(result.isFriggRepo).toBe(false);
212
+ });
213
+
214
+ it('rejects directories with only devDependency on core (without app structure)', async () => {
215
+ const appDir = path.join(testDir, 'only-dev-dep');
216
+ await fs.ensureDir(appDir);
217
+
218
+ await fs.writeJSON(path.join(appDir, 'package.json'), {
219
+ name: 'test-project',
220
+ devDependencies: {
221
+ '@friggframework/core': '2.0.0-next.58',
222
+ },
223
+ });
224
+
225
+ // No index.js or backend/
226
+
227
+ const result = await isFriggRepository(appDir);
228
+
229
+ expect(result.isFriggRepo).toBe(false);
230
+ });
231
+
232
+ it('rejects v1.x versions of @friggframework/core', async () => {
233
+ const appDir = path.join(testDir, 'v1-app');
234
+ await fs.ensureDir(appDir);
235
+
236
+ await fs.writeJSON(path.join(appDir, 'package.json'), {
237
+ name: 'v1-app',
238
+ dependencies: {
239
+ '@friggframework/core': '^1.5.0',
240
+ },
241
+ });
242
+
243
+ await fs.writeFile(path.join(appDir, 'index.js'), 'module.exports = {};');
244
+
245
+ const result = await isFriggRepository(appDir);
246
+
247
+ expect(result.isFriggRepo).toBe(false);
248
+ });
249
+ });
250
+
251
+ describe('app structure detection', () => {
252
+ it('detects hasBackend correctly', async () => {
253
+ const appDir = path.join(testDir, 'has-backend');
254
+ const backendDir = path.join(appDir, 'backend');
255
+ await fs.ensureDir(backendDir);
256
+
257
+ await fs.writeJSON(path.join(appDir, 'package.json'), {
258
+ name: 'has-backend',
259
+ dependencies: { '@friggframework/core': '^2.0.0' },
260
+ });
261
+
262
+ await fs.writeFile(path.join(appDir, 'index.js'), '');
263
+
264
+ const result = await isFriggRepository(appDir);
265
+
266
+ expect(result.isFriggRepo).toBe(true);
267
+ expect(result.repoInfo.hasBackend).toBe(true);
268
+ });
269
+
270
+ it('detects frigg scripts in package.json', async () => {
271
+ const appDir = path.join(testDir, 'has-frigg-scripts');
272
+ await fs.ensureDir(appDir);
273
+
274
+ await fs.writeJSON(path.join(appDir, 'package.json'), {
275
+ name: 'has-frigg-scripts',
276
+ dependencies: { '@friggframework/core': '^2.0.0' },
277
+ scripts: {
278
+ 'frigg:start': 'frigg start',
279
+ 'frigg:deploy': 'frigg deploy',
280
+ },
281
+ });
282
+
283
+ await fs.writeFile(path.join(appDir, 'index.js'), '');
284
+
285
+ const result = await isFriggRepository(appDir);
286
+
287
+ expect(result.isFriggRepo).toBe(true);
288
+ expect(result.repoInfo.hasFriggScripts).toBe(true);
289
+ });
290
+ });
291
+ });
292
+
293
+ describe('discoverFriggRepositories', () => {
294
+ it('discovers repos within search paths', async () => {
295
+ // Create a Frigg app in test directory
296
+ const appDir = path.join(testDir, 'discoverable-app');
297
+ await fs.ensureDir(appDir);
298
+
299
+ await fs.writeJSON(path.join(appDir, 'package.json'), {
300
+ name: 'discoverable-app',
301
+ dependencies: { '@friggframework/core': '^2.0.0' },
302
+ });
303
+
304
+ await fs.writeFile(path.join(appDir, 'index.js'), '');
305
+
306
+ const repos = await discoverFriggRepositories({
307
+ searchPaths: [testDir],
308
+ maxDepth: 2,
309
+ });
310
+
311
+ expect(repos.length).toBe(1);
312
+ expect(repos[0].name).toBe('discoverable-app');
313
+ });
314
+
315
+ it('discovers repos at multiple depth levels', async () => {
316
+ // Create nested Frigg apps
317
+ const level1 = path.join(testDir, 'org');
318
+ const level2 = path.join(level1, 'apps');
319
+ const appDir = path.join(level2, 'nested-app');
320
+ await fs.ensureDir(appDir);
321
+
322
+ await fs.writeJSON(path.join(appDir, 'package.json'), {
323
+ name: 'nested-app',
324
+ dependencies: { '@friggframework/core': '^2.0.0' },
325
+ });
326
+
327
+ await fs.writeFile(path.join(appDir, 'index.js'), '');
328
+
329
+ const repos = await discoverFriggRepositories({
330
+ searchPaths: [testDir],
331
+ maxDepth: 4,
332
+ });
333
+
334
+ expect(repos.length).toBe(1);
335
+ expect(repos[0].name).toBe('nested-app');
336
+ });
337
+
338
+ it('excludes node_modules from search', async () => {
339
+ const appDir = path.join(testDir, 'main-app');
340
+ const nodeModulesApp = path.join(appDir, 'node_modules', 'some-frigg-app');
341
+ await fs.ensureDir(nodeModulesApp);
342
+
343
+ // Main app
344
+ await fs.writeJSON(path.join(appDir, 'package.json'), {
345
+ name: 'main-app',
346
+ dependencies: { '@friggframework/core': '^2.0.0' },
347
+ });
348
+ await fs.writeFile(path.join(appDir, 'index.js'), '');
349
+
350
+ // App in node_modules (should be excluded)
351
+ await fs.writeJSON(path.join(nodeModulesApp, 'package.json'), {
352
+ name: 'some-frigg-app',
353
+ dependencies: { '@friggframework/core': '^2.0.0' },
354
+ });
355
+ await fs.writeFile(path.join(nodeModulesApp, 'index.js'), '');
356
+
357
+ const repos = await discoverFriggRepositories({
358
+ searchPaths: [testDir],
359
+ maxDepth: 4,
360
+ });
361
+
362
+ expect(repos.length).toBe(1);
363
+ expect(repos[0].name).toBe('main-app');
364
+ });
365
+
366
+ it('returns unique repositories (no duplicates)', async () => {
367
+ const appDir = path.join(testDir, 'unique-app');
368
+ await fs.ensureDir(appDir);
369
+
370
+ await fs.writeJSON(path.join(appDir, 'package.json'), {
371
+ name: 'unique-app',
372
+ dependencies: { '@friggframework/core': '^2.0.0' },
373
+ });
374
+
375
+ await fs.writeFile(path.join(appDir, 'index.js'), '');
376
+
377
+ // Search same directory twice
378
+ const repos = await discoverFriggRepositories({
379
+ searchPaths: [testDir, testDir],
380
+ maxDepth: 2,
381
+ });
382
+
383
+ expect(repos.length).toBe(1);
384
+ });
385
+ });
386
+
387
+ describe('detectFramework', () => {
388
+ it('detects React from frontend/package.json', async () => {
389
+ const appDir = path.join(testDir, 'react-app');
390
+ const frontendDir = path.join(appDir, 'frontend');
391
+ await fs.ensureDir(frontendDir);
392
+
393
+ await fs.writeJSON(path.join(frontendDir, 'package.json'), {
394
+ name: 'frontend',
395
+ dependencies: { react: '^18.0.0' },
396
+ });
397
+
398
+ const framework = detectFramework(appDir, []);
399
+
400
+ expect(framework).toBe('React');
401
+ });
402
+
403
+ it('detects Vue from frontend/package.json', async () => {
404
+ const appDir = path.join(testDir, 'vue-app');
405
+ const frontendDir = path.join(appDir, 'frontend');
406
+ await fs.ensureDir(frontendDir);
407
+
408
+ await fs.writeJSON(path.join(frontendDir, 'package.json'), {
409
+ name: 'frontend',
410
+ dependencies: { vue: '^3.0.0' },
411
+ });
412
+
413
+ const framework = detectFramework(appDir, []);
414
+
415
+ expect(framework).toBe('Vue');
416
+ });
417
+
418
+ it('detects framework from named directories', async () => {
419
+ const appDir = path.join(testDir, 'named-dir-app');
420
+ const reactDir = path.join(appDir, 'react');
421
+ await fs.ensureDir(reactDir);
422
+
423
+ const framework = detectFramework(appDir, ['react']);
424
+
425
+ expect(framework).toBe('React');
426
+ });
427
+
428
+ it('returns Unknown when no framework detected', async () => {
429
+ const appDir = path.join(testDir, 'unknown-framework');
430
+ await fs.ensureDir(appDir);
431
+
432
+ const framework = detectFramework(appDir, []);
433
+
434
+ expect(framework).toBe('Unknown');
435
+ });
436
+ });