@useavalon/avalon 0.1.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 (159) hide show
  1. package/README.md +54 -0
  2. package/mod.ts +301 -0
  3. package/package.json +85 -0
  4. package/src/build/README.md +310 -0
  5. package/src/build/integration-bundler-plugin.ts +116 -0
  6. package/src/build/integration-config.ts +168 -0
  7. package/src/build/integration-detection-plugin.ts +117 -0
  8. package/src/build/integration-resolver-plugin.ts +90 -0
  9. package/src/build/island-manifest.ts +269 -0
  10. package/src/build/island-types-generator.ts +476 -0
  11. package/src/build/mdx-island-transform.ts +464 -0
  12. package/src/build/mdx-plugin.ts +98 -0
  13. package/src/build/page-island-transform.ts +598 -0
  14. package/src/build/prop-extractors/index.ts +21 -0
  15. package/src/build/prop-extractors/lit.ts +140 -0
  16. package/src/build/prop-extractors/qwik.ts +16 -0
  17. package/src/build/prop-extractors/solid.ts +125 -0
  18. package/src/build/prop-extractors/svelte.ts +194 -0
  19. package/src/build/prop-extractors/vue.ts +111 -0
  20. package/src/build/sidecar-file-manager.ts +104 -0
  21. package/src/build/sidecar-renderer.ts +30 -0
  22. package/src/client/adapters/index.ts +13 -0
  23. package/src/client/adapters/lit-adapter.ts +654 -0
  24. package/src/client/adapters/preact-adapter.ts +331 -0
  25. package/src/client/adapters/qwik-adapter.ts +345 -0
  26. package/src/client/adapters/react-adapter.ts +353 -0
  27. package/src/client/adapters/solid-adapter.ts +451 -0
  28. package/src/client/adapters/svelte-adapter.ts +524 -0
  29. package/src/client/adapters/vue-adapter.ts +467 -0
  30. package/src/client/components.ts +35 -0
  31. package/src/client/css-hmr-handler.ts +344 -0
  32. package/src/client/framework-adapter.ts +462 -0
  33. package/src/client/hmr-coordinator.ts +396 -0
  34. package/src/client/hmr-error-overlay.js +533 -0
  35. package/src/client/main.js +816 -0
  36. package/src/client/tests/css-hmr-handler.test.ts +360 -0
  37. package/src/client/tests/framework-adapter.test.ts +519 -0
  38. package/src/client/tests/hmr-coordinator.test.ts +176 -0
  39. package/src/client/tests/hydration-option-parsing.test.ts +107 -0
  40. package/src/client/tests/lit-adapter.test.ts +427 -0
  41. package/src/client/tests/preact-adapter.test.ts +353 -0
  42. package/src/client/tests/qwik-adapter.test.ts +343 -0
  43. package/src/client/tests/react-adapter.test.ts +317 -0
  44. package/src/client/tests/solid-adapter.test.ts +396 -0
  45. package/src/client/tests/svelte-adapter.test.ts +387 -0
  46. package/src/client/tests/vue-adapter.test.ts +407 -0
  47. package/src/client/types/framework-runtime.d.ts +68 -0
  48. package/src/client/types/vite-hmr.d.ts +46 -0
  49. package/src/client/types/vite-virtual-modules.d.ts +60 -0
  50. package/src/components/Image.tsx +123 -0
  51. package/src/components/IslandErrorBoundary.tsx +145 -0
  52. package/src/components/LayoutDataErrorBoundary.tsx +141 -0
  53. package/src/components/LayoutErrorBoundary.tsx +127 -0
  54. package/src/components/PersistentIsland.tsx +52 -0
  55. package/src/components/StreamingErrorBoundary.tsx +233 -0
  56. package/src/components/StreamingLayout.tsx +538 -0
  57. package/src/components/tests/component-analyzer.test.ts +96 -0
  58. package/src/components/tests/component-detection.test.ts +347 -0
  59. package/src/components/tests/persistent-islands.test.ts +398 -0
  60. package/src/core/components/component-analyzer.ts +192 -0
  61. package/src/core/components/component-detection.ts +508 -0
  62. package/src/core/components/enhanced-framework-detector.ts +500 -0
  63. package/src/core/components/framework-registry.ts +563 -0
  64. package/src/core/components/tests/enhanced-framework-detector.test.ts +577 -0
  65. package/src/core/components/tests/framework-registry.test.ts +465 -0
  66. package/src/core/content/mdx-processor.ts +46 -0
  67. package/src/core/integrations/README.md +282 -0
  68. package/src/core/integrations/index.ts +19 -0
  69. package/src/core/integrations/loader.ts +125 -0
  70. package/src/core/integrations/registry.ts +195 -0
  71. package/src/core/islands/island-persistence.ts +325 -0
  72. package/src/core/islands/island-state-serializer.ts +258 -0
  73. package/src/core/islands/persistent-island-context.tsx +80 -0
  74. package/src/core/islands/use-persistent-state.ts +68 -0
  75. package/src/core/layout/enhanced-layout-resolver.ts +322 -0
  76. package/src/core/layout/layout-cache-manager.ts +485 -0
  77. package/src/core/layout/layout-composer.ts +357 -0
  78. package/src/core/layout/layout-data-loader.ts +516 -0
  79. package/src/core/layout/layout-discovery.ts +243 -0
  80. package/src/core/layout/layout-matcher.ts +299 -0
  81. package/src/core/layout/layout-types.ts +110 -0
  82. package/src/core/layout/tests/enhanced-layout-resolver.test.ts +477 -0
  83. package/src/core/layout/tests/layout-cache-optimization.test.ts +149 -0
  84. package/src/core/layout/tests/layout-composer.test.ts +486 -0
  85. package/src/core/layout/tests/layout-data-loader.test.ts +443 -0
  86. package/src/core/layout/tests/layout-discovery.test.ts +253 -0
  87. package/src/core/layout/tests/layout-matcher.test.ts +480 -0
  88. package/src/core/modules/framework-module-resolver.ts +273 -0
  89. package/src/core/modules/tests/framework-module-resolver.test.ts +263 -0
  90. package/src/core/modules/tests/module-resolution-integration.test.ts +117 -0
  91. package/src/islands/component-analysis.ts +213 -0
  92. package/src/islands/css-utils.ts +565 -0
  93. package/src/islands/discovery/index.ts +80 -0
  94. package/src/islands/discovery/registry.ts +340 -0
  95. package/src/islands/discovery/resolver.ts +477 -0
  96. package/src/islands/discovery/scanner.ts +386 -0
  97. package/src/islands/discovery/tests/island-discovery.test.ts +881 -0
  98. package/src/islands/discovery/types.ts +117 -0
  99. package/src/islands/discovery/validator.ts +544 -0
  100. package/src/islands/discovery/watcher.ts +368 -0
  101. package/src/islands/framework-detection.ts +428 -0
  102. package/src/islands/integration-loader.ts +490 -0
  103. package/src/islands/island.tsx +565 -0
  104. package/src/islands/render-cache.ts +550 -0
  105. package/src/islands/types.ts +80 -0
  106. package/src/islands/universal-css-collector.ts +157 -0
  107. package/src/islands/universal-head-collector.ts +137 -0
  108. package/src/layout-system.d.ts +592 -0
  109. package/src/layout-system.ts +218 -0
  110. package/src/middleware/__tests__/discovery.test.ts +107 -0
  111. package/src/middleware/discovery.ts +268 -0
  112. package/src/middleware/executor.ts +315 -0
  113. package/src/middleware/index.ts +76 -0
  114. package/src/middleware/types.ts +99 -0
  115. package/src/nitro/build-config.ts +576 -0
  116. package/src/nitro/config.ts +483 -0
  117. package/src/nitro/error-handler.ts +636 -0
  118. package/src/nitro/index.ts +173 -0
  119. package/src/nitro/island-manifest.ts +584 -0
  120. package/src/nitro/middleware-adapter.ts +260 -0
  121. package/src/nitro/renderer.ts +1458 -0
  122. package/src/nitro/route-discovery.ts +439 -0
  123. package/src/nitro/types.ts +321 -0
  124. package/src/render/collect-css.ts +198 -0
  125. package/src/render/error-pages.ts +79 -0
  126. package/src/render/isolated-ssr-renderer.ts +654 -0
  127. package/src/render/ssr.ts +1030 -0
  128. package/src/schemas/api.ts +30 -0
  129. package/src/schemas/core.ts +64 -0
  130. package/src/schemas/index.ts +212 -0
  131. package/src/schemas/layout.ts +279 -0
  132. package/src/schemas/routing/index.ts +38 -0
  133. package/src/schemas/routing.ts +376 -0
  134. package/src/types/as-island.ts +20 -0
  135. package/src/types/image.d.ts +106 -0
  136. package/src/types/index.d.ts +22 -0
  137. package/src/types/island-jsx.d.ts +33 -0
  138. package/src/types/island-prop.d.ts +20 -0
  139. package/src/types/layout.ts +285 -0
  140. package/src/types/mdx.d.ts +6 -0
  141. package/src/types/routing.ts +555 -0
  142. package/src/types/tests/layout-types.test.ts +197 -0
  143. package/src/types/types.ts +5 -0
  144. package/src/types/urlpattern.d.ts +49 -0
  145. package/src/types/vite-env.d.ts +11 -0
  146. package/src/utils/dev-logger.ts +299 -0
  147. package/src/utils/fs.ts +151 -0
  148. package/src/vite-plugin/auto-discover.ts +551 -0
  149. package/src/vite-plugin/config.ts +266 -0
  150. package/src/vite-plugin/errors.ts +127 -0
  151. package/src/vite-plugin/image-optimization.ts +151 -0
  152. package/src/vite-plugin/integration-activator.ts +126 -0
  153. package/src/vite-plugin/island-sidecar-plugin.ts +176 -0
  154. package/src/vite-plugin/module-discovery.ts +189 -0
  155. package/src/vite-plugin/nitro-integration.ts +1334 -0
  156. package/src/vite-plugin/plugin.ts +329 -0
  157. package/src/vite-plugin/tests/image-optimization.test.ts +54 -0
  158. package/src/vite-plugin/types.ts +327 -0
  159. package/src/vite-plugin/validation.ts +228 -0
@@ -0,0 +1,428 @@
1
+ import type { Framework } from './types.ts';
2
+ import { registry } from '../core/integrations/registry.ts';
3
+ import { IslandRegistry, createIslandRegistry } from './discovery/index.ts';
4
+ import { getCachedPath, setCachedPath } from './render-cache.ts';
5
+ import { stat as fsStat, readFile } from 'node:fs/promises';
6
+
7
+ /** Known synchronous framework types (excludes 'unknown') */
8
+ type SyncFramework = 'solid' | 'vue' | 'svelte' | 'preact' | 'react' | 'lit' | 'qwik';
9
+
10
+ /**
11
+ * Resolve an island path using the island registry.
12
+ * This enables resolution of nested island paths by component name.
13
+ *
14
+ * @param src - The source path or component name
15
+ * @param registry - The island registry to use for resolution
16
+ * @returns The resolved path or null if not found
17
+ */
18
+ function resolveIslandPathFromRegistry(src: string, registry: IslandRegistry): string | null {
19
+ // Extract component name from path
20
+ const componentName = extractComponentName(src);
21
+ if (!componentName) {
22
+ return null;
23
+ }
24
+
25
+ // Try to resolve by name
26
+ const island = registry.resolve(componentName);
27
+ if (island) {
28
+ // Return the relative path with leading slash
29
+ return '/' + island.relativePath;
30
+ }
31
+
32
+ // Try to resolve by qualified name (namespace/name)
33
+ if (src.includes('/')) {
34
+ // Extract potential namespace from path
35
+ const namespace = extractNamespaceFromPath(src);
36
+ if (namespace) {
37
+ const islandByNamespace = registry.resolve(componentName, namespace);
38
+ if (islandByNamespace) {
39
+ return '/' + islandByNamespace.relativePath;
40
+ }
41
+ }
42
+ }
43
+
44
+ return null;
45
+ }
46
+
47
+ /**
48
+ * Extract component name from a path.
49
+ * Handles various path formats and removes extensions.
50
+ *
51
+ * @param path - The path to extract from
52
+ * @returns The component name or null
53
+ */
54
+ function extractComponentName(path: string): string | null {
55
+ // Get the filename from the path
56
+ const parts = path.split('/').filter(Boolean);
57
+ if (parts.length === 0) {
58
+ return null;
59
+ }
60
+
61
+ let filename = parts.at(-1)!;
62
+
63
+ // Remove framework-specific extensions first (e.g., .solid.tsx -> .tsx)
64
+ const frameworkPatterns = [/\.solid\.(tsx|jsx)$/, /\.react\.(tsx|jsx)$/, /\.lit\.(ts|js)$/, /\.preact\.(tsx|jsx)$/];
65
+
66
+ for (const pattern of frameworkPatterns) {
67
+ if (pattern.test(filename)) {
68
+ filename = filename.replace(pattern, '');
69
+ break;
70
+ }
71
+ }
72
+
73
+ // Remove standard extensions
74
+ filename = filename.replace(/\.(tsx|ts|jsx|js|vue|svelte)$/, '');
75
+
76
+ return filename || null;
77
+ }
78
+
79
+ /**
80
+ * Extract namespace from a nested island path.
81
+ *
82
+ * @param path - The path to extract namespace from
83
+ * @returns The namespace or empty string for default
84
+ */
85
+ function extractNamespaceFromPath(path: string): string {
86
+ // Match patterns like /src/modules/auth/islands/ or /modules/auth/islands/
87
+ const match = new RegExp(/(?:\/src)?\/(.+?)\/islands\//).exec(path);
88
+ if (match) {
89
+ return match[1];
90
+ }
91
+ return '';
92
+ }
93
+
94
+ /**
95
+ * Check if a file exists asynchronously
96
+ */
97
+ async function fileExists(path: string): Promise<boolean> {
98
+ try {
99
+ await fsStat(path);
100
+ return true;
101
+ } catch {
102
+ return false;
103
+ }
104
+ }
105
+
106
+ /**
107
+ * Normalize nested island paths to include /src/ prefix.
108
+ */
109
+ function normalizeIslandPath(resolvedPath: string): string {
110
+ if (resolvedPath.includes('/islands/') && !resolvedPath.startsWith('/src/')) {
111
+ if (/^\/(?:modules\/)?[^/]+\/islands\//.test(resolvedPath)) {
112
+ return '/src' + resolvedPath;
113
+ }
114
+ if (resolvedPath.startsWith('/islands/')) {
115
+ return resolvedPath.replace('/islands/', '/src/islands/');
116
+ }
117
+ }
118
+ return resolvedPath;
119
+ }
120
+
121
+ /**
122
+ * Try to resolve a .tsx path to a framework-specific file that exists on disk.
123
+ */
124
+ async function resolveFrameworkSpecificPath(resolvedPath: string): Promise<string | null> {
125
+ const integrations = registry.getAll();
126
+ const possiblePaths: string[] = [];
127
+ const basePath = resolvedPath.replace('.tsx', '');
128
+
129
+ for (const integration of integrations) {
130
+ const config = integration.config();
131
+ for (const ext of config.fileExtensions) {
132
+ if (ext === '.tsx' || ext === '.jsx') {
133
+ possiblePaths.push(`${basePath}.${config.name}${ext}`);
134
+ } else {
135
+ possiblePaths.push(`${basePath}${ext}`);
136
+ }
137
+ }
138
+ }
139
+
140
+ possiblePaths.push(resolvedPath);
141
+
142
+ for (const possiblePath of possiblePaths) {
143
+ const pathVariation = possiblePath.startsWith('/') ? possiblePath.substring(1) : possiblePath;
144
+ if (await fileExists(pathVariation)) {
145
+ return possiblePath;
146
+ }
147
+ }
148
+
149
+ return null;
150
+ }
151
+
152
+ /**
153
+ * Resolve Island component path for Vite SSR loading
154
+ * Converts /islands/* paths to /src/islands/* for proper resolution
155
+ * Also handles framework-specific naming conventions and nested islands.
156
+ * Uses async file operations and caching for better performance.
157
+ *
158
+ * @param src - The source path or component name
159
+ * @returns The resolved path
160
+ */
161
+ export async function resolveIslandPath(src: string): Promise<string> {
162
+ // Check cache first
163
+ const cachedPath = getCachedPath(src);
164
+ if (cachedPath !== null) {
165
+ return cachedPath;
166
+ }
167
+
168
+ // Normalize path separators and nested island paths
169
+ let resolvedPath = normalizeIslandPath(src.replaceAll('\\', '/'));
170
+
171
+ // Try to resolve using the island registry if available
172
+ const islandRegistry = _global.__islandRegistry;
173
+ if (islandRegistry) {
174
+ const resolvedFromRegistry = resolveIslandPathFromRegistry(resolvedPath, islandRegistry);
175
+ if (resolvedFromRegistry) {
176
+ setCachedPath(src, resolvedFromRegistry);
177
+ return resolvedFromRegistry;
178
+ }
179
+ }
180
+
181
+ // Handle framework-specific naming conventions
182
+ if (resolvedPath.endsWith('.tsx') && !resolvedPath.includes('.solid.') && !resolvedPath.includes('.preact.')) {
183
+ const frameworkPath = await resolveFrameworkSpecificPath(resolvedPath);
184
+ if (frameworkPath) {
185
+ setCachedPath(src, frameworkPath);
186
+ return frameworkPath;
187
+ }
188
+ }
189
+
190
+ // Cache the resolved path (even if it's the same as input)
191
+ setCachedPath(src, resolvedPath);
192
+ return resolvedPath;
193
+ }
194
+
195
+ /**
196
+ * Check if a path is a nested island path (not in default /src/islands/).
197
+ *
198
+ * @param path - The path to check
199
+ * @returns True if the path is a nested island path
200
+ */
201
+ export function isNestedIslandPath(path: string): boolean {
202
+ const normalized = path.replaceAll('\\', '/');
203
+
204
+ // Check if it contains /islands/ but not at the root level
205
+ if (!normalized.includes('/islands/')) {
206
+ return false;
207
+ }
208
+
209
+ // Default path patterns
210
+ const defaultPatterns = [/^\/islands\//, /^\/src\/islands\//, /^src\/islands\//, /^islands\//];
211
+
212
+ for (const pattern of defaultPatterns) {
213
+ if (pattern.test(normalized)) {
214
+ return false;
215
+ }
216
+ }
217
+
218
+ // If it contains /islands/ but doesn't match default patterns, it's nested
219
+ return true;
220
+ }
221
+
222
+ /** Typed accessor for the global island registry */
223
+ const _global = globalThis as unknown as {
224
+ __islandRegistry?: IslandRegistry;
225
+ __viteDevServer?: { ssrLoadModule: (path: string) => Promise<Record<string, unknown>> };
226
+ };
227
+
228
+ /**
229
+ * Get the island registry, initializing it if necessary.
230
+ *
231
+ * @param projectRoot - The project root directory
232
+ * @returns The island registry
233
+ */
234
+ export async function getOrCreateIslandRegistry(projectRoot: string = process.cwd()): Promise<IslandRegistry> {
235
+ _global.__islandRegistry ??= await createIslandRegistry(projectRoot);
236
+ return _global.__islandRegistry;
237
+ }
238
+
239
+ /**
240
+ * Set the global island registry.
241
+ * Useful for testing or when the registry is created elsewhere.
242
+ *
243
+ * @param registry - The registry to set
244
+ */
245
+ export function setIslandRegistry(registry: IslandRegistry): void {
246
+ _global.__islandRegistry = registry;
247
+ }
248
+
249
+ /**
250
+ * Clear the global island registry.
251
+ * Useful for testing or hot module replacement.
252
+ */
253
+ export function clearIslandRegistry(): void {
254
+ _global.__islandRegistry = undefined;
255
+ }
256
+
257
+ /**
258
+ * Quick framework detection based on file extension and naming conventions
259
+ * Used for setting framework attributes without async file reading
260
+ *
261
+ * Updated to query integration configs for detection patterns
262
+ */
263
+ export function detectFrameworkFromSrc(src: string): SyncFramework {
264
+ // Normalize path separators
265
+ const normalizedSrc = src.replaceAll('\\', '/');
266
+
267
+ // Get all registered integrations
268
+ const integrations = registry.getAll();
269
+
270
+ // First pass: Check for framework-specific naming conventions (e.g., .solid.tsx)
271
+ // This takes priority over generic extensions
272
+ for (const integration of integrations) {
273
+ const config = integration.config();
274
+
275
+ if (normalizedSrc.includes(`.${config.name}.`)) {
276
+ return config.name as SyncFramework;
277
+ }
278
+ }
279
+
280
+ // Second pass: Check file extensions for unique extensions (e.g., .vue, .svelte)
281
+ for (const integration of integrations) {
282
+ const config = integration.config();
283
+
284
+ // Check if file extension matches
285
+ for (const ext of config.fileExtensions) {
286
+ if (normalizedSrc.endsWith(ext)) {
287
+ return config.name as SyncFramework;
288
+ }
289
+ }
290
+ }
291
+
292
+ return detectFrameworkFromFallback(normalizedSrc);
293
+ }
294
+
295
+ /**
296
+ * Fallback detection for when no integrations are loaded yet.
297
+ */
298
+ function detectFrameworkFromFallback(normalizedSrc: string): SyncFramework {
299
+ if (normalizedSrc.endsWith('.vue')) {
300
+ return 'vue';
301
+ }
302
+ if (normalizedSrc.endsWith('.svelte')) {
303
+ return 'svelte';
304
+ }
305
+ if (normalizedSrc.includes('.solid.') || normalizedSrc.toLowerCase().includes('solid')) {
306
+ return 'solid';
307
+ }
308
+ if (normalizedSrc.includes('.qwik.') || normalizedSrc.toLowerCase().includes('qwik')) {
309
+ return 'qwik';
310
+ }
311
+ if (normalizedSrc.includes('react') || normalizedSrc.toLowerCase().includes('react')) {
312
+ return 'react';
313
+ }
314
+
315
+ // Default to preact for .tsx/.jsx files
316
+ return 'preact';
317
+ }
318
+
319
+ /**
320
+ * Detect framework from file content by checking integration detection patterns.
321
+ */
322
+ function detectFrameworkFromContent(
323
+ fileContent: string,
324
+ integrations: ReturnType<typeof registry.getAll>,
325
+ ): Framework | null {
326
+ for (const integration of integrations) {
327
+ const config = integration.config();
328
+
329
+ for (const pattern of config.detectionPatterns.imports) {
330
+ if (pattern.test(fileContent)) {
331
+ return config.name as Framework;
332
+ }
333
+ }
334
+
335
+ for (const pattern of config.detectionPatterns.content) {
336
+ if (pattern.test(fileContent)) {
337
+ return config.name as Framework;
338
+ }
339
+ }
340
+ }
341
+
342
+ return null;
343
+ }
344
+
345
+ /**
346
+ * Fallback content-based detection when no integrations are loaded.
347
+ */
348
+ function detectFrameworkFromContentFallback(fileContent: string): Framework {
349
+ const checks = [
350
+ { pattern: /solid-js|@jsxImportSource solid-js/, framework: 'solid' as const },
351
+ { pattern: /@builder\.io\/qwik|@jsxImportSource @builder\.io\/qwik/, framework: 'qwik' as const },
352
+ { pattern: /vue|Vue/, framework: 'vue' as const },
353
+ { pattern: /svelte/, framework: 'svelte' as const },
354
+ { pattern: /react/, framework: 'react' as const },
355
+ { pattern: /preact/, framework: 'preact' as const },
356
+ ];
357
+
358
+ for (const check of checks) {
359
+ if (check.pattern.test(fileContent)) {
360
+ return check.framework;
361
+ }
362
+ }
363
+
364
+ return 'preact';
365
+ }
366
+
367
+ /**
368
+ * Read file content for framework detection, trying direct read then Vite SSR.
369
+ */
370
+ async function readFileContentForDetection(src: string): Promise<string | null> {
371
+ try {
372
+ const resolvedPath = await resolveIslandPath(src);
373
+ const filePath = resolvedPath.replace(/^\//, '');
374
+ return await readFile(filePath, 'utf-8');
375
+ } catch {
376
+ const viteServer = _global.__viteDevServer;
377
+ if (viteServer) {
378
+ const resolvedPath = await resolveIslandPath(src);
379
+ const mod = await viteServer.ssrLoadModule(resolvedPath);
380
+ return JSON.stringify(mod);
381
+ }
382
+ return null;
383
+ }
384
+ }
385
+
386
+ /**
387
+ * Detect the framework used by a component file
388
+ * Updated to query integration configs for detection patterns
389
+ */
390
+ export async function detectFramework(src: string): Promise<Framework> {
391
+ const integrations = registry.getAll();
392
+
393
+ // Quick filename-based detection using integration configs
394
+ for (const integration of integrations) {
395
+ const config = integration.config();
396
+
397
+ for (const ext of config.fileExtensions) {
398
+ if (src.endsWith(ext)) {
399
+ return config.name as Framework;
400
+ }
401
+ }
402
+
403
+ if (src.includes(`.${config.name}.`)) {
404
+ return config.name as Framework;
405
+ }
406
+ }
407
+
408
+ // Try to read file content for more accurate detection
409
+ try {
410
+ const fileContent = await readFileContentForDetection(src);
411
+ if (!fileContent) {
412
+ return 'unknown';
413
+ }
414
+
415
+ const detected = detectFrameworkFromContent(fileContent, integrations);
416
+ if (detected) {
417
+ return detected;
418
+ }
419
+
420
+ if (integrations.length === 0) {
421
+ return detectFrameworkFromContentFallback(fileContent);
422
+ }
423
+
424
+ return 'preact';
425
+ } catch {
426
+ return 'unknown';
427
+ }
428
+ }