@useavalon/avalon 0.1.13 → 0.1.15

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 (230) hide show
  1. package/dist/mod.js +1 -0
  2. package/dist/src/build/integration-bundler-plugin.js +1 -0
  3. package/dist/src/build/integration-config.js +1 -0
  4. package/dist/src/build/integration-detection-plugin.js +1 -0
  5. package/dist/src/build/integration-resolver-plugin.js +1 -0
  6. package/dist/src/build/island-manifest.js +1 -0
  7. package/dist/src/build/island-types-generator.js +5 -0
  8. package/dist/src/build/mdx-island-transform.js +2 -0
  9. package/dist/src/build/mdx-plugin.js +1 -0
  10. package/dist/src/build/page-island-transform.js +3 -0
  11. package/dist/src/build/prop-extractors/index.js +1 -0
  12. package/dist/src/build/prop-extractors/lit.js +1 -0
  13. package/dist/src/build/prop-extractors/qwik.js +1 -0
  14. package/dist/src/build/prop-extractors/solid.js +1 -0
  15. package/dist/src/build/prop-extractors/svelte.js +1 -0
  16. package/dist/src/build/prop-extractors/vue.js +1 -0
  17. package/dist/src/build/sidecar-file-manager.js +1 -0
  18. package/dist/src/build/sidecar-renderer.js +6 -0
  19. package/dist/src/client/adapters/index.js +1 -0
  20. package/dist/src/client/components.js +1 -0
  21. package/dist/src/client/css-hmr-handler.js +1 -0
  22. package/dist/src/client/framework-adapter.js +13 -0
  23. package/dist/src/client/hmr-coordinator.js +1 -0
  24. package/dist/src/client/hmr-error-overlay.js +214 -0
  25. package/dist/src/client/main.js +39 -0
  26. package/dist/src/components/Image.js +1 -0
  27. package/dist/src/components/IslandErrorBoundary.js +1 -0
  28. package/dist/src/components/LayoutDataErrorBoundary.js +1 -0
  29. package/dist/src/components/LayoutErrorBoundary.js +1 -0
  30. package/dist/src/components/PersistentIsland.js +1 -0
  31. package/dist/src/components/StreamingErrorBoundary.js +1 -0
  32. package/dist/src/components/StreamingLayout.js +29 -0
  33. package/dist/src/core/components/component-analyzer.js +1 -0
  34. package/dist/src/core/components/component-detection.js +5 -0
  35. package/dist/src/core/components/enhanced-framework-detector.js +1 -0
  36. package/dist/src/core/components/framework-registry.js +1 -0
  37. package/dist/src/core/content/mdx-processor.js +1 -0
  38. package/dist/src/core/integrations/index.js +1 -0
  39. package/dist/src/core/integrations/loader.js +1 -0
  40. package/dist/src/core/integrations/registry.js +1 -0
  41. package/dist/src/core/islands/island-persistence.js +1 -0
  42. package/dist/src/core/islands/island-state-serializer.js +1 -0
  43. package/dist/src/core/islands/persistent-island-context.js +1 -0
  44. package/dist/src/core/islands/use-persistent-state.js +1 -0
  45. package/dist/src/core/layout/enhanced-layout-resolver.js +1 -0
  46. package/dist/src/core/layout/layout-cache-manager.js +1 -0
  47. package/dist/src/core/layout/layout-composer.js +1 -0
  48. package/dist/src/core/layout/layout-data-loader.js +1 -0
  49. package/dist/src/core/layout/layout-discovery.js +1 -0
  50. package/dist/src/core/layout/layout-matcher.js +1 -0
  51. package/dist/src/core/layout/layout-types.js +1 -0
  52. package/dist/src/core/modules/framework-module-resolver.js +1 -0
  53. package/dist/src/islands/component-analysis.js +1 -0
  54. package/dist/src/islands/css-utils.js +17 -0
  55. package/dist/src/islands/discovery/index.js +1 -0
  56. package/dist/src/islands/discovery/registry.js +1 -0
  57. package/dist/src/islands/discovery/resolver.js +2 -0
  58. package/dist/src/islands/discovery/scanner.js +1 -0
  59. package/dist/src/islands/discovery/types.js +1 -0
  60. package/dist/src/islands/discovery/validator.js +18 -0
  61. package/dist/src/islands/discovery/watcher.js +1 -0
  62. package/dist/src/islands/framework-detection.js +1 -0
  63. package/dist/src/islands/integration-loader.js +1 -0
  64. package/dist/src/islands/island.js +1 -0
  65. package/dist/src/islands/render-cache.js +1 -0
  66. package/dist/src/islands/types.js +1 -0
  67. package/dist/src/islands/universal-css-collector.js +5 -0
  68. package/dist/src/islands/universal-head-collector.js +2 -0
  69. package/dist/src/layout-system.js +1 -0
  70. package/dist/src/middleware/discovery.js +1 -0
  71. package/dist/src/middleware/executor.js +1 -0
  72. package/dist/src/middleware/index.js +1 -0
  73. package/dist/src/middleware/types.js +1 -0
  74. package/dist/src/nitro/build-config.js +1 -0
  75. package/dist/src/nitro/config.js +1 -0
  76. package/dist/src/nitro/error-handler.js +198 -0
  77. package/dist/src/nitro/index.js +1 -0
  78. package/dist/src/nitro/island-manifest.js +2 -0
  79. package/dist/src/nitro/middleware-adapter.js +1 -0
  80. package/dist/src/nitro/renderer.js +183 -0
  81. package/dist/src/nitro/route-discovery.js +1 -0
  82. package/dist/src/nitro/types.js +1 -0
  83. package/dist/src/render/collect-css.js +3 -0
  84. package/dist/src/render/error-pages.js +48 -0
  85. package/dist/src/render/isolated-ssr-renderer.js +1 -0
  86. package/dist/src/render/ssr.js +90 -0
  87. package/dist/src/schemas/api.js +1 -0
  88. package/dist/src/schemas/core.js +1 -0
  89. package/dist/src/schemas/index.js +1 -0
  90. package/dist/src/schemas/layout.js +1 -0
  91. package/dist/src/schemas/routing/index.js +1 -0
  92. package/dist/src/schemas/routing.js +1 -0
  93. package/dist/src/types/as-island.js +1 -0
  94. package/dist/src/types/layout.js +1 -0
  95. package/dist/src/types/routing.js +1 -0
  96. package/dist/src/types/types.js +1 -0
  97. package/dist/src/utils/dev-logger.js +12 -0
  98. package/dist/src/utils/fs.js +1 -0
  99. package/dist/src/vite-plugin/auto-discover.js +1 -0
  100. package/dist/src/vite-plugin/config.js +1 -0
  101. package/dist/src/vite-plugin/errors.js +1 -0
  102. package/dist/src/vite-plugin/image-optimization.js +45 -0
  103. package/dist/src/vite-plugin/integration-activator.js +1 -0
  104. package/dist/src/vite-plugin/island-sidecar-plugin.js +1 -0
  105. package/dist/src/vite-plugin/module-discovery.js +1 -0
  106. package/dist/src/vite-plugin/nitro-integration.js +42 -0
  107. package/dist/src/vite-plugin/plugin.js +1 -0
  108. package/dist/src/vite-plugin/types.js +1 -0
  109. package/dist/src/vite-plugin/validation.js +2 -0
  110. package/package.json +14 -20
  111. package/mod.ts +0 -302
  112. package/src/build/integration-bundler-plugin.ts +0 -116
  113. package/src/build/integration-config.ts +0 -168
  114. package/src/build/integration-detection-plugin.ts +0 -117
  115. package/src/build/integration-resolver-plugin.ts +0 -90
  116. package/src/build/island-manifest.ts +0 -269
  117. package/src/build/island-types-generator.ts +0 -476
  118. package/src/build/mdx-island-transform.ts +0 -464
  119. package/src/build/mdx-plugin.ts +0 -98
  120. package/src/build/page-island-transform.ts +0 -598
  121. package/src/build/prop-extractors/index.ts +0 -21
  122. package/src/build/prop-extractors/lit.ts +0 -140
  123. package/src/build/prop-extractors/qwik.ts +0 -16
  124. package/src/build/prop-extractors/solid.ts +0 -125
  125. package/src/build/prop-extractors/svelte.ts +0 -194
  126. package/src/build/prop-extractors/vue.ts +0 -111
  127. package/src/build/sidecar-file-manager.ts +0 -104
  128. package/src/build/sidecar-renderer.ts +0 -30
  129. package/src/client/adapters/index.ts +0 -21
  130. package/src/client/components.ts +0 -35
  131. package/src/client/css-hmr-handler.ts +0 -344
  132. package/src/client/framework-adapter.ts +0 -462
  133. package/src/client/hmr-coordinator.ts +0 -396
  134. package/src/client/hmr-error-overlay.js +0 -533
  135. package/src/client/main.js +0 -824
  136. package/src/components/Image.tsx +0 -123
  137. package/src/components/IslandErrorBoundary.tsx +0 -145
  138. package/src/components/LayoutDataErrorBoundary.tsx +0 -141
  139. package/src/components/LayoutErrorBoundary.tsx +0 -127
  140. package/src/components/PersistentIsland.tsx +0 -52
  141. package/src/components/StreamingErrorBoundary.tsx +0 -233
  142. package/src/components/StreamingLayout.tsx +0 -538
  143. package/src/core/components/component-analyzer.ts +0 -192
  144. package/src/core/components/component-detection.ts +0 -508
  145. package/src/core/components/enhanced-framework-detector.ts +0 -500
  146. package/src/core/components/framework-registry.ts +0 -563
  147. package/src/core/content/mdx-processor.ts +0 -46
  148. package/src/core/integrations/index.ts +0 -19
  149. package/src/core/integrations/loader.ts +0 -125
  150. package/src/core/integrations/registry.ts +0 -175
  151. package/src/core/islands/island-persistence.ts +0 -325
  152. package/src/core/islands/island-state-serializer.ts +0 -258
  153. package/src/core/islands/persistent-island-context.tsx +0 -80
  154. package/src/core/islands/use-persistent-state.ts +0 -68
  155. package/src/core/layout/enhanced-layout-resolver.ts +0 -322
  156. package/src/core/layout/layout-cache-manager.ts +0 -485
  157. package/src/core/layout/layout-composer.ts +0 -357
  158. package/src/core/layout/layout-data-loader.ts +0 -516
  159. package/src/core/layout/layout-discovery.ts +0 -243
  160. package/src/core/layout/layout-matcher.ts +0 -299
  161. package/src/core/layout/layout-types.ts +0 -110
  162. package/src/core/modules/framework-module-resolver.ts +0 -273
  163. package/src/islands/component-analysis.ts +0 -213
  164. package/src/islands/css-utils.ts +0 -565
  165. package/src/islands/discovery/index.ts +0 -80
  166. package/src/islands/discovery/registry.ts +0 -340
  167. package/src/islands/discovery/resolver.ts +0 -477
  168. package/src/islands/discovery/scanner.ts +0 -386
  169. package/src/islands/discovery/types.ts +0 -117
  170. package/src/islands/discovery/validator.ts +0 -544
  171. package/src/islands/discovery/watcher.ts +0 -368
  172. package/src/islands/framework-detection.ts +0 -428
  173. package/src/islands/integration-loader.ts +0 -490
  174. package/src/islands/island.tsx +0 -565
  175. package/src/islands/render-cache.ts +0 -550
  176. package/src/islands/types.ts +0 -80
  177. package/src/islands/universal-css-collector.ts +0 -157
  178. package/src/islands/universal-head-collector.ts +0 -137
  179. package/src/layout-system.ts +0 -218
  180. package/src/middleware/discovery.ts +0 -268
  181. package/src/middleware/executor.ts +0 -315
  182. package/src/middleware/index.ts +0 -76
  183. package/src/middleware/types.ts +0 -99
  184. package/src/nitro/build-config.ts +0 -576
  185. package/src/nitro/config.ts +0 -483
  186. package/src/nitro/error-handler.ts +0 -636
  187. package/src/nitro/index.ts +0 -173
  188. package/src/nitro/island-manifest.ts +0 -584
  189. package/src/nitro/middleware-adapter.ts +0 -260
  190. package/src/nitro/renderer.ts +0 -1471
  191. package/src/nitro/route-discovery.ts +0 -439
  192. package/src/nitro/types.ts +0 -321
  193. package/src/render/collect-css.ts +0 -198
  194. package/src/render/error-pages.ts +0 -79
  195. package/src/render/isolated-ssr-renderer.ts +0 -654
  196. package/src/render/ssr.ts +0 -1030
  197. package/src/schemas/api.ts +0 -30
  198. package/src/schemas/core.ts +0 -64
  199. package/src/schemas/index.ts +0 -212
  200. package/src/schemas/layout.ts +0 -279
  201. package/src/schemas/routing/index.ts +0 -38
  202. package/src/schemas/routing.ts +0 -376
  203. package/src/types/as-island.ts +0 -20
  204. package/src/types/layout.ts +0 -285
  205. package/src/types/routing.ts +0 -555
  206. package/src/types/types.ts +0 -5
  207. package/src/utils/dev-logger.ts +0 -299
  208. package/src/utils/fs.ts +0 -151
  209. package/src/vite-plugin/auto-discover.ts +0 -551
  210. package/src/vite-plugin/config.ts +0 -266
  211. package/src/vite-plugin/errors.ts +0 -127
  212. package/src/vite-plugin/image-optimization.ts +0 -156
  213. package/src/vite-plugin/integration-activator.ts +0 -126
  214. package/src/vite-plugin/island-sidecar-plugin.ts +0 -176
  215. package/src/vite-plugin/module-discovery.ts +0 -189
  216. package/src/vite-plugin/nitro-integration.ts +0 -1354
  217. package/src/vite-plugin/plugin.ts +0 -403
  218. package/src/vite-plugin/types.ts +0 -327
  219. package/src/vite-plugin/validation.ts +0 -228
  220. /package/{src → dist/src}/client/types/framework-runtime.d.ts +0 -0
  221. /package/{src → dist/src}/client/types/vite-hmr.d.ts +0 -0
  222. /package/{src → dist/src}/client/types/vite-virtual-modules.d.ts +0 -0
  223. /package/{src → dist/src}/layout-system.d.ts +0 -0
  224. /package/{src → dist/src}/types/image.d.ts +0 -0
  225. /package/{src → dist/src}/types/index.d.ts +0 -0
  226. /package/{src → dist/src}/types/island-jsx.d.ts +0 -0
  227. /package/{src → dist/src}/types/island-prop.d.ts +0 -0
  228. /package/{src → dist/src}/types/mdx.d.ts +0 -0
  229. /package/{src → dist/src}/types/urlpattern.d.ts +0 -0
  230. /package/{src → dist/src}/types/vite-env.d.ts +0 -0
@@ -1,428 +0,0 @@
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
- }