@useavalon/avalon 0.1.12 → 0.1.13

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/mod.ts +302 -0
  2. package/package.json +9 -17
  3. package/src/build/integration-bundler-plugin.ts +116 -0
  4. package/src/build/integration-config.ts +168 -0
  5. package/src/build/integration-detection-plugin.ts +117 -0
  6. package/src/build/integration-resolver-plugin.ts +90 -0
  7. package/src/build/island-manifest.ts +269 -0
  8. package/src/build/island-types-generator.ts +476 -0
  9. package/src/build/mdx-island-transform.ts +464 -0
  10. package/src/build/mdx-plugin.ts +98 -0
  11. package/src/build/page-island-transform.ts +598 -0
  12. package/src/build/prop-extractors/index.ts +21 -0
  13. package/src/build/prop-extractors/lit.ts +140 -0
  14. package/src/build/prop-extractors/qwik.ts +16 -0
  15. package/src/build/prop-extractors/solid.ts +125 -0
  16. package/src/build/prop-extractors/svelte.ts +194 -0
  17. package/src/build/prop-extractors/vue.ts +111 -0
  18. package/src/build/sidecar-file-manager.ts +104 -0
  19. package/src/build/sidecar-renderer.ts +30 -0
  20. package/src/client/adapters/index.ts +21 -0
  21. package/src/client/components.ts +35 -0
  22. package/src/client/css-hmr-handler.ts +344 -0
  23. package/src/client/framework-adapter.ts +462 -0
  24. package/src/client/hmr-coordinator.ts +396 -0
  25. package/src/client/hmr-error-overlay.js +533 -0
  26. package/src/client/main.js +824 -0
  27. package/src/components/Image.tsx +123 -0
  28. package/src/components/IslandErrorBoundary.tsx +145 -0
  29. package/src/components/LayoutDataErrorBoundary.tsx +141 -0
  30. package/src/components/LayoutErrorBoundary.tsx +127 -0
  31. package/src/components/PersistentIsland.tsx +52 -0
  32. package/src/components/StreamingErrorBoundary.tsx +233 -0
  33. package/src/components/StreamingLayout.tsx +538 -0
  34. package/src/core/components/component-analyzer.ts +192 -0
  35. package/src/core/components/component-detection.ts +508 -0
  36. package/src/core/components/enhanced-framework-detector.ts +500 -0
  37. package/src/core/components/framework-registry.ts +563 -0
  38. package/src/core/content/mdx-processor.ts +46 -0
  39. package/src/core/integrations/index.ts +19 -0
  40. package/src/core/integrations/loader.ts +125 -0
  41. package/src/core/integrations/registry.ts +175 -0
  42. package/src/core/islands/island-persistence.ts +325 -0
  43. package/src/core/islands/island-state-serializer.ts +258 -0
  44. package/src/core/islands/persistent-island-context.tsx +80 -0
  45. package/src/core/islands/use-persistent-state.ts +68 -0
  46. package/src/core/layout/enhanced-layout-resolver.ts +322 -0
  47. package/src/core/layout/layout-cache-manager.ts +485 -0
  48. package/src/core/layout/layout-composer.ts +357 -0
  49. package/src/core/layout/layout-data-loader.ts +516 -0
  50. package/src/core/layout/layout-discovery.ts +243 -0
  51. package/src/core/layout/layout-matcher.ts +299 -0
  52. package/src/core/layout/layout-types.ts +110 -0
  53. package/src/core/modules/framework-module-resolver.ts +273 -0
  54. package/src/islands/component-analysis.ts +213 -0
  55. package/src/islands/css-utils.ts +565 -0
  56. package/src/islands/discovery/index.ts +80 -0
  57. package/src/islands/discovery/registry.ts +340 -0
  58. package/src/islands/discovery/resolver.ts +477 -0
  59. package/src/islands/discovery/scanner.ts +386 -0
  60. package/src/islands/discovery/types.ts +117 -0
  61. package/src/islands/discovery/validator.ts +544 -0
  62. package/src/islands/discovery/watcher.ts +368 -0
  63. package/src/islands/framework-detection.ts +428 -0
  64. package/src/islands/integration-loader.ts +490 -0
  65. package/src/islands/island.tsx +565 -0
  66. package/src/islands/render-cache.ts +550 -0
  67. package/src/islands/types.ts +80 -0
  68. package/src/islands/universal-css-collector.ts +157 -0
  69. package/src/islands/universal-head-collector.ts +137 -0
  70. package/src/layout-system.ts +218 -0
  71. package/src/middleware/discovery.ts +268 -0
  72. package/src/middleware/executor.ts +315 -0
  73. package/src/middleware/index.ts +76 -0
  74. package/src/middleware/types.ts +99 -0
  75. package/src/nitro/build-config.ts +576 -0
  76. package/src/nitro/config.ts +483 -0
  77. package/src/nitro/error-handler.ts +636 -0
  78. package/src/nitro/index.ts +173 -0
  79. package/src/nitro/island-manifest.ts +584 -0
  80. package/src/nitro/middleware-adapter.ts +260 -0
  81. package/src/nitro/renderer.ts +1471 -0
  82. package/src/nitro/route-discovery.ts +439 -0
  83. package/src/nitro/types.ts +321 -0
  84. package/src/render/collect-css.ts +198 -0
  85. package/src/render/error-pages.ts +79 -0
  86. package/src/render/isolated-ssr-renderer.ts +654 -0
  87. package/src/render/ssr.ts +1030 -0
  88. package/src/schemas/api.ts +30 -0
  89. package/src/schemas/core.ts +64 -0
  90. package/src/schemas/index.ts +212 -0
  91. package/src/schemas/layout.ts +279 -0
  92. package/src/schemas/routing/index.ts +38 -0
  93. package/src/schemas/routing.ts +376 -0
  94. package/src/types/as-island.ts +20 -0
  95. package/src/types/layout.ts +285 -0
  96. package/src/types/routing.ts +555 -0
  97. package/src/types/types.ts +5 -0
  98. package/src/utils/dev-logger.ts +299 -0
  99. package/src/utils/fs.ts +151 -0
  100. package/src/vite-plugin/auto-discover.ts +551 -0
  101. package/src/vite-plugin/config.ts +266 -0
  102. package/src/vite-plugin/errors.ts +127 -0
  103. package/src/vite-plugin/image-optimization.ts +156 -0
  104. package/src/vite-plugin/integration-activator.ts +126 -0
  105. package/src/vite-plugin/island-sidecar-plugin.ts +176 -0
  106. package/src/vite-plugin/module-discovery.ts +189 -0
  107. package/src/vite-plugin/nitro-integration.ts +1354 -0
  108. package/src/vite-plugin/plugin.ts +403 -0
  109. package/src/vite-plugin/types.ts +327 -0
  110. package/src/vite-plugin/validation.ts +228 -0
  111. package/dist/mod.js +0 -1
  112. package/dist/src/build/integration-bundler-plugin.js +0 -1
  113. package/dist/src/build/integration-config.js +0 -1
  114. package/dist/src/build/integration-detection-plugin.js +0 -1
  115. package/dist/src/build/integration-resolver-plugin.js +0 -1
  116. package/dist/src/build/island-manifest.js +0 -1
  117. package/dist/src/build/island-types-generator.js +0 -5
  118. package/dist/src/build/mdx-island-transform.js +0 -2
  119. package/dist/src/build/mdx-plugin.js +0 -1
  120. package/dist/src/build/page-island-transform.js +0 -3
  121. package/dist/src/build/prop-extractors/index.js +0 -1
  122. package/dist/src/build/prop-extractors/lit.js +0 -1
  123. package/dist/src/build/prop-extractors/qwik.js +0 -1
  124. package/dist/src/build/prop-extractors/solid.js +0 -1
  125. package/dist/src/build/prop-extractors/svelte.js +0 -1
  126. package/dist/src/build/prop-extractors/vue.js +0 -1
  127. package/dist/src/build/sidecar-file-manager.js +0 -1
  128. package/dist/src/build/sidecar-renderer.js +0 -6
  129. package/dist/src/client/adapters/index.js +0 -1
  130. package/dist/src/client/components.js +0 -1
  131. package/dist/src/client/css-hmr-handler.js +0 -1
  132. package/dist/src/client/framework-adapter.js +0 -13
  133. package/dist/src/client/hmr-coordinator.js +0 -1
  134. package/dist/src/client/hmr-error-overlay.js +0 -214
  135. package/dist/src/client/main.js +0 -39
  136. package/dist/src/components/Image.js +0 -1
  137. package/dist/src/components/IslandErrorBoundary.js +0 -1
  138. package/dist/src/components/LayoutDataErrorBoundary.js +0 -1
  139. package/dist/src/components/LayoutErrorBoundary.js +0 -1
  140. package/dist/src/components/PersistentIsland.js +0 -1
  141. package/dist/src/components/StreamingErrorBoundary.js +0 -1
  142. package/dist/src/components/StreamingLayout.js +0 -29
  143. package/dist/src/core/components/component-analyzer.js +0 -1
  144. package/dist/src/core/components/component-detection.js +0 -5
  145. package/dist/src/core/components/enhanced-framework-detector.js +0 -1
  146. package/dist/src/core/components/framework-registry.js +0 -1
  147. package/dist/src/core/content/mdx-processor.js +0 -1
  148. package/dist/src/core/integrations/index.js +0 -1
  149. package/dist/src/core/integrations/loader.js +0 -1
  150. package/dist/src/core/integrations/registry.js +0 -1
  151. package/dist/src/core/islands/island-persistence.js +0 -1
  152. package/dist/src/core/islands/island-state-serializer.js +0 -1
  153. package/dist/src/core/islands/persistent-island-context.js +0 -1
  154. package/dist/src/core/islands/use-persistent-state.js +0 -1
  155. package/dist/src/core/layout/enhanced-layout-resolver.js +0 -1
  156. package/dist/src/core/layout/layout-cache-manager.js +0 -1
  157. package/dist/src/core/layout/layout-composer.js +0 -1
  158. package/dist/src/core/layout/layout-data-loader.js +0 -1
  159. package/dist/src/core/layout/layout-discovery.js +0 -1
  160. package/dist/src/core/layout/layout-matcher.js +0 -1
  161. package/dist/src/core/layout/layout-types.js +0 -1
  162. package/dist/src/core/modules/framework-module-resolver.js +0 -1
  163. package/dist/src/islands/component-analysis.js +0 -1
  164. package/dist/src/islands/css-utils.js +0 -17
  165. package/dist/src/islands/discovery/index.js +0 -1
  166. package/dist/src/islands/discovery/registry.js +0 -1
  167. package/dist/src/islands/discovery/resolver.js +0 -2
  168. package/dist/src/islands/discovery/scanner.js +0 -1
  169. package/dist/src/islands/discovery/types.js +0 -1
  170. package/dist/src/islands/discovery/validator.js +0 -18
  171. package/dist/src/islands/discovery/watcher.js +0 -1
  172. package/dist/src/islands/framework-detection.js +0 -1
  173. package/dist/src/islands/integration-loader.js +0 -1
  174. package/dist/src/islands/island.js +0 -1
  175. package/dist/src/islands/render-cache.js +0 -1
  176. package/dist/src/islands/types.js +0 -1
  177. package/dist/src/islands/universal-css-collector.js +0 -5
  178. package/dist/src/islands/universal-head-collector.js +0 -2
  179. package/dist/src/layout-system.js +0 -1
  180. package/dist/src/middleware/discovery.js +0 -1
  181. package/dist/src/middleware/executor.js +0 -1
  182. package/dist/src/middleware/index.js +0 -1
  183. package/dist/src/middleware/types.js +0 -1
  184. package/dist/src/nitro/build-config.js +0 -1
  185. package/dist/src/nitro/config.js +0 -1
  186. package/dist/src/nitro/error-handler.js +0 -198
  187. package/dist/src/nitro/index.js +0 -1
  188. package/dist/src/nitro/island-manifest.js +0 -2
  189. package/dist/src/nitro/middleware-adapter.js +0 -1
  190. package/dist/src/nitro/renderer.js +0 -183
  191. package/dist/src/nitro/route-discovery.js +0 -1
  192. package/dist/src/nitro/types.js +0 -1
  193. package/dist/src/render/collect-css.js +0 -3
  194. package/dist/src/render/error-pages.js +0 -48
  195. package/dist/src/render/isolated-ssr-renderer.js +0 -1
  196. package/dist/src/render/ssr.js +0 -90
  197. package/dist/src/schemas/api.js +0 -1
  198. package/dist/src/schemas/core.js +0 -1
  199. package/dist/src/schemas/index.js +0 -1
  200. package/dist/src/schemas/layout.js +0 -1
  201. package/dist/src/schemas/routing/index.js +0 -1
  202. package/dist/src/schemas/routing.js +0 -1
  203. package/dist/src/types/as-island.js +0 -1
  204. package/dist/src/types/layout.js +0 -1
  205. package/dist/src/types/routing.js +0 -1
  206. package/dist/src/types/types.js +0 -1
  207. package/dist/src/utils/dev-logger.js +0 -12
  208. package/dist/src/utils/fs.js +0 -1
  209. package/dist/src/vite-plugin/auto-discover.js +0 -1
  210. package/dist/src/vite-plugin/config.js +0 -1
  211. package/dist/src/vite-plugin/errors.js +0 -1
  212. package/dist/src/vite-plugin/image-optimization.js +0 -45
  213. package/dist/src/vite-plugin/integration-activator.js +0 -1
  214. package/dist/src/vite-plugin/island-sidecar-plugin.js +0 -1
  215. package/dist/src/vite-plugin/module-discovery.js +0 -1
  216. package/dist/src/vite-plugin/nitro-integration.js +0 -42
  217. package/dist/src/vite-plugin/plugin.js +0 -1
  218. package/dist/src/vite-plugin/types.js +0 -1
  219. package/dist/src/vite-plugin/validation.js +0 -2
  220. /package/{dist/src → src}/client/types/framework-runtime.d.ts +0 -0
  221. /package/{dist/src → src}/client/types/vite-hmr.d.ts +0 -0
  222. /package/{dist/src → src}/client/types/vite-virtual-modules.d.ts +0 -0
  223. /package/{dist/src → src}/layout-system.d.ts +0 -0
  224. /package/{dist/src → src}/types/image.d.ts +0 -0
  225. /package/{dist/src → src}/types/index.d.ts +0 -0
  226. /package/{dist/src → src}/types/island-jsx.d.ts +0 -0
  227. /package/{dist/src → src}/types/island-prop.d.ts +0 -0
  228. /package/{dist/src → src}/types/mdx.d.ts +0 -0
  229. /package/{dist/src → src}/types/urlpattern.d.ts +0 -0
  230. /package/{dist/src → src}/types/vite-env.d.ts +0 -0
@@ -0,0 +1,500 @@
1
+ /**
2
+ * Enhanced Framework Detection System
3
+ *
4
+ * This module provides advanced framework detection with multi-evidence analysis,
5
+ * confidence scoring, and JSX import source parsing to accurately identify
6
+ * component frameworks and prevent cross-framework conflicts.
7
+ */
8
+
9
+ export interface FrameworkDetectionResult {
10
+ framework: 'preact' | 'solid' | 'vue' | 'svelte' | 'react' | 'lit' | 'qwik' | 'unknown';
11
+ confidence: 'high' | 'medium' | 'low';
12
+ evidence: string[];
13
+ warnings: string[];
14
+ }
15
+
16
+ export interface DetectionCriteria {
17
+ fileExtension: string;
18
+ jsxImportSource?: string;
19
+ imports: string[];
20
+ content: string;
21
+ }
22
+
23
+ export interface FrameworkConfig {
24
+ name: string;
25
+ fileExtensions: string[];
26
+ jsxImportSources: string[];
27
+ ssrModules: string[];
28
+ hydrationModules: string[];
29
+ detectionPatterns: {
30
+ imports: RegExp[];
31
+ content: RegExp[];
32
+ jsxPragmas: string[];
33
+ };
34
+ }
35
+
36
+ /**
37
+ * Enhanced Framework Detector with multi-evidence analysis
38
+ */
39
+ export class EnhancedFrameworkDetector {
40
+ private readonly frameworkRegistry: Map<string, FrameworkConfig>;
41
+
42
+ constructor(frameworkConfigs?: Record<string, FrameworkConfig>) {
43
+ this.frameworkRegistry = new Map();
44
+
45
+ // Initialize with provided configs or defaults
46
+ const configs = frameworkConfigs || this.getDefaultFrameworkConfigs();
47
+ Object.entries(configs).forEach(([name, config]) => {
48
+ this.frameworkRegistry.set(name, config);
49
+ });
50
+ }
51
+
52
+ /**
53
+ * Detects framework with multi-evidence analysis
54
+ */
55
+ detectFramework(filePath: string, content: string): FrameworkDetectionResult {
56
+ const criteria = this.extractDetectionCriteria(filePath, content);
57
+ const evidence: string[] = [];
58
+ const warnings: string[] = [];
59
+
60
+ // Check for explicit naming conventions first (highest priority)
61
+ if (filePath.includes('.solid.')) {
62
+ evidence.push('Explicit Solid naming convention (.solid.tsx/.solid.jsx)');
63
+ return {
64
+ framework: 'solid',
65
+ confidence: 'high',
66
+ evidence,
67
+ warnings,
68
+ };
69
+ }
70
+
71
+ if (filePath.includes('.preact.')) {
72
+ evidence.push('Explicit Preact naming convention (.preact.tsx/.preact.jsx)');
73
+ return {
74
+ framework: 'preact',
75
+ confidence: 'high',
76
+ evidence,
77
+ warnings,
78
+ };
79
+ }
80
+
81
+ if (filePath.includes('.react.')) {
82
+ evidence.push('Explicit React naming convention (.react.tsx/.react.jsx)');
83
+ return {
84
+ framework: 'react',
85
+ confidence: 'high',
86
+ evidence,
87
+ warnings,
88
+ };
89
+ }
90
+
91
+ if (filePath.includes('.lit.')) {
92
+ evidence.push('Explicit Lit naming convention (.lit.ts/.lit.js)');
93
+ return {
94
+ framework: 'lit',
95
+ confidence: 'high',
96
+ evidence,
97
+ warnings,
98
+ };
99
+ }
100
+
101
+ if (filePath.includes('.qwik.')) {
102
+ evidence.push('Explicit Qwik naming convention (.qwik.tsx/.qwik.jsx)');
103
+ return {
104
+ framework: 'qwik',
105
+ confidence: 'high',
106
+ evidence,
107
+ warnings,
108
+ };
109
+ }
110
+
111
+ // Score each framework based on evidence
112
+ const frameworkScores = new Map<string, number>();
113
+
114
+ for (const [frameworkName, config] of this.frameworkRegistry) {
115
+ const score = this.calculateFrameworkScore(criteria, config, evidence);
116
+ frameworkScores.set(frameworkName, score);
117
+ }
118
+
119
+ // Find the highest scoring framework
120
+ const sortedFrameworks = Array.from(frameworkScores.entries()).sort(([, a], [, b]) => b - a);
121
+
122
+ const [topFramework, topScore] = sortedFrameworks[0] || ['unknown', 0];
123
+ const [secondFramework, secondScore] = sortedFrameworks[1] || ['unknown', 0];
124
+
125
+ // Determine confidence based on score and evidence
126
+ let confidence: FrameworkDetectionResult['confidence'];
127
+ if (topScore >= 3 && topScore - secondScore >= 2) {
128
+ confidence = 'high';
129
+ } else if (topScore >= 2) {
130
+ confidence = 'medium';
131
+ } else {
132
+ confidence = 'low';
133
+ warnings.push(
134
+ 'Framework detection has low confidence - consider adding explicit JSX import source or use naming convention (.solid.tsx, .preact.tsx)'
135
+ );
136
+ }
137
+
138
+ // Handle ambiguous cases
139
+ if (topScore === secondScore && topScore > 0) {
140
+ warnings.push(
141
+ `Ambiguous detection between ${topFramework} and ${secondFramework} - consider using naming convention`
142
+ );
143
+ confidence = 'low';
144
+ }
145
+
146
+ return {
147
+ framework: topScore > 0 ? (topFramework as FrameworkDetectionResult['framework']) : 'unknown',
148
+ confidence,
149
+ evidence,
150
+ warnings,
151
+ };
152
+ }
153
+
154
+ /**
155
+ * Extracts detection criteria from file path and content
156
+ */
157
+ private extractDetectionCriteria(filePath: string, content: string): DetectionCriteria {
158
+ const fileExtension = this.getFileExtension(filePath);
159
+ const jsxImportSource = this.parseJSXImportSource(content);
160
+ const imports = this.extractImportStatements(content);
161
+
162
+ return {
163
+ fileExtension,
164
+ jsxImportSource,
165
+ imports,
166
+ content,
167
+ };
168
+ }
169
+
170
+ /**
171
+ * Calculates framework score based on multiple evidence points
172
+ */
173
+ private calculateFrameworkScore(criteria: DetectionCriteria, config: FrameworkConfig, evidence: string[]): number {
174
+ let score = 0;
175
+
176
+ // JSX Import Source (highest priority - 3 points)
177
+ if (criteria.jsxImportSource && config.jsxImportSources.includes(criteria.jsxImportSource)) {
178
+ score += 3;
179
+ evidence.push(`JSX import source: @jsxImportSource ${criteria.jsxImportSource}`);
180
+ }
181
+
182
+ // Import statements (2 points each)
183
+ for (const importPattern of config.detectionPatterns.imports) {
184
+ if (criteria.imports.some(imp => importPattern.test(imp))) {
185
+ score += 2;
186
+ evidence.push(`Framework import detected: ${importPattern.source}`);
187
+ }
188
+ }
189
+
190
+ // Content patterns (1 point each)
191
+ for (const contentPattern of config.detectionPatterns.content) {
192
+ if (contentPattern.test(criteria.content)) {
193
+ score += 1;
194
+ evidence.push(`Framework-specific content pattern: ${contentPattern.source}`);
195
+ }
196
+ }
197
+
198
+ // File extension (0.5 points - lowest priority)
199
+ if (config.fileExtensions.includes(criteria.fileExtension)) {
200
+ score += 0.5;
201
+ evidence.push(`File extension: ${criteria.fileExtension}`);
202
+ }
203
+
204
+ return score;
205
+ }
206
+
207
+ /**
208
+ * Parses JSX import source from content
209
+ */
210
+ private parseJSXImportSource(content: string): string | undefined {
211
+ // Look for @jsxImportSource pragma (handles both // and /** */ comment styles)
212
+ const jsxImportSourceRegex = /@jsxImportSource\s+([^\s*]+)/;
213
+ const match = jsxImportSourceRegex.exec(content);
214
+
215
+ if (match) {
216
+ return match[1];
217
+ }
218
+
219
+ return undefined;
220
+ }
221
+
222
+ /**
223
+ * Extracts import statements from content
224
+ */
225
+ private extractImportStatements(content: string): string[] {
226
+ const imports: string[] = [];
227
+
228
+ // Match ES6 import specifiers: import ... from 'module'
229
+ const importFromRegex = /from\s+['"]([^'"]+)['"]/g;
230
+ // Match side-effect imports: import 'module'
231
+ const sideEffectRegex = /import\s+['"]([^'"]+)['"]/g;
232
+ // Match require statements: require('module')
233
+ const requireRegex = /require\s*\(\s*['"]([^'"]+)['"]\s*\)/g;
234
+
235
+ let match;
236
+ while ((match = importFromRegex.exec(content)) !== null) {
237
+ imports.push(match[1]);
238
+ }
239
+ while ((match = sideEffectRegex.exec(content)) !== null) {
240
+ imports.push(match[1]);
241
+ }
242
+ while ((match = requireRegex.exec(content)) !== null) {
243
+ imports.push(match[1]);
244
+ }
245
+
246
+ return imports;
247
+ }
248
+
249
+ /**
250
+ * Gets file extension from path, handling framework-specific naming conventions
251
+ */
252
+ private getFileExtension(filePath: string): string {
253
+ // Handle framework-specific naming conventions first
254
+ if (filePath.includes('.solid.')) {
255
+ return '.solid.tsx'; // Treat as special Solid extension
256
+ }
257
+ if (filePath.includes('.preact.')) {
258
+ return '.preact.tsx'; // Treat as special Preact extension
259
+ }
260
+ if (filePath.includes('.react.')) {
261
+ return '.react.tsx'; // Treat as special React extension
262
+ }
263
+ if (filePath.includes('.lit.')) {
264
+ return '.lit.ts'; // Treat as special Lit extension
265
+ }
266
+
267
+ const lastDot = filePath.lastIndexOf('.');
268
+ return lastDot === -1 ? '' : filePath.substring(lastDot);
269
+ }
270
+
271
+ /**
272
+ * Gets default framework configurations
273
+ */
274
+ private getDefaultFrameworkConfigs(): Record<string, FrameworkConfig> {
275
+ return {
276
+ preact: {
277
+ name: 'preact',
278
+ fileExtensions: ['.tsx', '.jsx', '.preact.tsx', '.preact.jsx'],
279
+ jsxImportSources: ['preact'],
280
+ ssrModules: ['preact-render-to-string'],
281
+ hydrationModules: ['preact'],
282
+ detectionPatterns: {
283
+ imports: [/^preact$/, /^preact\//, /preact-render-to-string/],
284
+ content: [
285
+ /\buseState\b/,
286
+ /\buseEffect\b/,
287
+ /\buseCallback\b/,
288
+ /\buseMemo\b/,
289
+ /\buseRef\b/,
290
+ /from\s+['"]preact['"]/,
291
+ ],
292
+ jsxPragmas: ['@jsxImportSource preact'],
293
+ },
294
+ },
295
+ solid: {
296
+ name: 'solid',
297
+ fileExtensions: ['.tsx', '.jsx', '.solid.tsx', '.solid.jsx'],
298
+ jsxImportSources: ['solid-js'],
299
+ ssrModules: ['solid-js/web'],
300
+ hydrationModules: ['solid-js/web'],
301
+ detectionPatterns: {
302
+ imports: [/^solid-js$/, /^solid-js\//, /solid-js\/web/],
303
+ content: [
304
+ /\bcreateSignal\b/,
305
+ /\bcreateEffect\b/,
306
+ /\bcreateMemo\b/,
307
+ /\bcreateResource\b/,
308
+ /\bonMount\b/,
309
+ /\bonCleanup\b/,
310
+ /from\s+['"]solid-js['"]/,
311
+ ],
312
+ jsxPragmas: ['@jsxImportSource solid-js'],
313
+ },
314
+ },
315
+ vue: {
316
+ name: 'vue',
317
+ fileExtensions: ['.vue'],
318
+ jsxImportSources: ['vue'],
319
+ ssrModules: ['vue/server-renderer'],
320
+ hydrationModules: ['vue'],
321
+ detectionPatterns: {
322
+ imports: [/^vue$/, /^@vue\//, /vue\/server-renderer/],
323
+ content: [
324
+ /<template>/,
325
+ /<script>/,
326
+ /<style>/,
327
+ /\bref\b/,
328
+ /\breactive\b/,
329
+ /\bcomputed\b/,
330
+ /\bwatchEffect\b/,
331
+ /from\s+['"]vue['"]/,
332
+ ],
333
+ jsxPragmas: ['@jsxImportSource vue'],
334
+ },
335
+ },
336
+ svelte: {
337
+ name: 'svelte',
338
+ fileExtensions: ['.svelte'],
339
+ jsxImportSources: ['svelte'],
340
+ ssrModules: ['svelte/server'],
341
+ hydrationModules: ['svelte'],
342
+ detectionPatterns: {
343
+ imports: [/^svelte$/, /^svelte\//, /svelte\/store/],
344
+ content: [
345
+ /<script>/,
346
+ /<style>/,
347
+ /\$:/,
348
+ /\bonMount\b/,
349
+ /\bafterUpdate\b/,
350
+ /\bbeforeUpdate\b/,
351
+ /from\s+['"]svelte['"]/,
352
+ ],
353
+ jsxPragmas: ['@jsxImportSource svelte'],
354
+ },
355
+ },
356
+ react: {
357
+ name: 'react',
358
+ fileExtensions: ['.jsx', '.tsx', '.react.jsx', '.react.tsx'],
359
+ jsxImportSources: ['react'],
360
+ ssrModules: ['react-dom/server'],
361
+ hydrationModules: ['react-dom/client'],
362
+ detectionPatterns: {
363
+ imports: [
364
+ /^react$/,
365
+ /^react\//,
366
+ /^react-dom$/,
367
+ /^react-dom\//,
368
+ /from\s+['"]react['"]/,
369
+ /from\s+['"]react\/[^'"]+['"]/,
370
+ /from\s+['"]react-dom['"]/,
371
+ ],
372
+ content: [
373
+ /\buseState\b/,
374
+ /\buseEffect\b/,
375
+ /\buseContext\b/,
376
+ /\buseReducer\b/,
377
+ /\buseCallback\b/,
378
+ /\buseMemo\b/,
379
+ /\buseRef\b/,
380
+ /\buseTransition\b/,
381
+ /\buseDeferredValue\b/,
382
+ /\buseId\b/,
383
+ /\buseImperativeHandle\b/,
384
+ /\buseLayoutEffect\b/,
385
+ /["']use client["']/,
386
+ /["']use server["']/,
387
+ /from\s+['"]react['"]/,
388
+ /import\s+.*\s+from\s+['"]react['"]/,
389
+ ],
390
+ jsxPragmas: ['@jsxImportSource react'],
391
+ },
392
+ },
393
+ lit: {
394
+ name: 'lit',
395
+ fileExtensions: ['.ts', '.js', '.lit.ts', '.lit.js'],
396
+ jsxImportSources: ['lit'],
397
+ ssrModules: ['@lit-labs/ssr'],
398
+ hydrationModules: ['lit'],
399
+ detectionPatterns: {
400
+ imports: [
401
+ /^lit$/,
402
+ /^lit\//,
403
+ /^@lit\//,
404
+ /^@lit-labs\/ssr/,
405
+ /from\s+['"]lit['"]/,
406
+ /from\s+['"]lit\/[^'"]+['"]/,
407
+ /from\s+['"]@lit\/[^'"]+['"]/,
408
+ ],
409
+ content: [
410
+ /\bLitElement\b/,
411
+ /\bcustomElement\b/,
412
+ /@customElement/,
413
+ /@property/,
414
+ /@state/,
415
+ /@query/,
416
+ /@queryAll/,
417
+ /\bhtml`/,
418
+ /\bcss`/,
419
+ /extends\s+LitElement/,
420
+ /from\s+['"]lit['"]/,
421
+ /import\s+.*\s+from\s+['"]lit['"]/,
422
+ ],
423
+ jsxPragmas: ['@jsxImportSource lit'],
424
+ },
425
+ },
426
+ qwik: {
427
+ name: 'qwik',
428
+ fileExtensions: ['.tsx', '.jsx', '.qwik.tsx', '.qwik.jsx'],
429
+ jsxImportSources: ['@builder.io/qwik'],
430
+ ssrModules: ['@builder.io/qwik/server'],
431
+ hydrationModules: ['@builder.io/qwik'],
432
+ detectionPatterns: {
433
+ imports: [
434
+ /^@builder\.io\/qwik$/,
435
+ /^@builder\.io\/qwik\//,
436
+ /from\s+['"]@builder\.io\/qwik['"]/,
437
+ /from\s+['"]@builder\.io\/qwik\/[^'"]+['"]/,
438
+ ],
439
+ content: [
440
+ /\bcomponent\$/,
441
+ /\buseSignal\b/,
442
+ /\buseStore\b/,
443
+ /\buseTask\$/,
444
+ /\buseVisibleTask\$/,
445
+ /\buseResource\$/,
446
+ /\buseContext\b/,
447
+ /\buseContextProvider\b/,
448
+ /\$\(\s*\(/,
449
+ /from\s+['"]@builder\.io\/qwik['"]/,
450
+ /import\s+.*\s+from\s+['"]@builder\.io\/qwik['"]/,
451
+ ],
452
+ jsxPragmas: ['@jsxImportSource @builder.io/qwik'],
453
+ },
454
+ },
455
+ };
456
+ }
457
+
458
+ /**
459
+ * Adds or updates a framework configuration
460
+ */
461
+ addFrameworkConfig(name: string, config: FrameworkConfig): void {
462
+ this.frameworkRegistry.set(name, config);
463
+ }
464
+
465
+ /**
466
+ * Gets all registered framework configurations
467
+ */
468
+ getFrameworkConfigs(): Map<string, FrameworkConfig> {
469
+ return new Map(this.frameworkRegistry);
470
+ }
471
+
472
+ /**
473
+ * Validates framework configuration completeness
474
+ */
475
+ validateFrameworkConfig(config: FrameworkConfig): string[] {
476
+ const errors: string[] = [];
477
+
478
+ if (!config.name || config.name.trim() === '') {
479
+ errors.push('Framework name is required');
480
+ }
481
+
482
+ if (!config.fileExtensions || config.fileExtensions.length === 0) {
483
+ errors.push('At least one file extension is required');
484
+ }
485
+
486
+ if (!config.detectionPatterns.imports || config.detectionPatterns.imports.length === 0) {
487
+ errors.push('At least one import pattern is required for detection');
488
+ }
489
+
490
+ if (!config.ssrModules || config.ssrModules.length === 0) {
491
+ errors.push('At least one SSR module is required');
492
+ }
493
+
494
+ if (!config.hydrationModules || config.hydrationModules.length === 0) {
495
+ errors.push('At least one hydration module is required');
496
+ }
497
+
498
+ return errors;
499
+ }
500
+ }