@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.
- package/README.md +54 -0
- package/mod.ts +301 -0
- package/package.json +85 -0
- package/src/build/README.md +310 -0
- package/src/build/integration-bundler-plugin.ts +116 -0
- package/src/build/integration-config.ts +168 -0
- package/src/build/integration-detection-plugin.ts +117 -0
- package/src/build/integration-resolver-plugin.ts +90 -0
- package/src/build/island-manifest.ts +269 -0
- package/src/build/island-types-generator.ts +476 -0
- package/src/build/mdx-island-transform.ts +464 -0
- package/src/build/mdx-plugin.ts +98 -0
- package/src/build/page-island-transform.ts +598 -0
- package/src/build/prop-extractors/index.ts +21 -0
- package/src/build/prop-extractors/lit.ts +140 -0
- package/src/build/prop-extractors/qwik.ts +16 -0
- package/src/build/prop-extractors/solid.ts +125 -0
- package/src/build/prop-extractors/svelte.ts +194 -0
- package/src/build/prop-extractors/vue.ts +111 -0
- package/src/build/sidecar-file-manager.ts +104 -0
- package/src/build/sidecar-renderer.ts +30 -0
- package/src/client/adapters/index.ts +13 -0
- package/src/client/adapters/lit-adapter.ts +654 -0
- package/src/client/adapters/preact-adapter.ts +331 -0
- package/src/client/adapters/qwik-adapter.ts +345 -0
- package/src/client/adapters/react-adapter.ts +353 -0
- package/src/client/adapters/solid-adapter.ts +451 -0
- package/src/client/adapters/svelte-adapter.ts +524 -0
- package/src/client/adapters/vue-adapter.ts +467 -0
- package/src/client/components.ts +35 -0
- package/src/client/css-hmr-handler.ts +344 -0
- package/src/client/framework-adapter.ts +462 -0
- package/src/client/hmr-coordinator.ts +396 -0
- package/src/client/hmr-error-overlay.js +533 -0
- package/src/client/main.js +816 -0
- package/src/client/tests/css-hmr-handler.test.ts +360 -0
- package/src/client/tests/framework-adapter.test.ts +519 -0
- package/src/client/tests/hmr-coordinator.test.ts +176 -0
- package/src/client/tests/hydration-option-parsing.test.ts +107 -0
- package/src/client/tests/lit-adapter.test.ts +427 -0
- package/src/client/tests/preact-adapter.test.ts +353 -0
- package/src/client/tests/qwik-adapter.test.ts +343 -0
- package/src/client/tests/react-adapter.test.ts +317 -0
- package/src/client/tests/solid-adapter.test.ts +396 -0
- package/src/client/tests/svelte-adapter.test.ts +387 -0
- package/src/client/tests/vue-adapter.test.ts +407 -0
- package/src/client/types/framework-runtime.d.ts +68 -0
- package/src/client/types/vite-hmr.d.ts +46 -0
- package/src/client/types/vite-virtual-modules.d.ts +60 -0
- package/src/components/Image.tsx +123 -0
- package/src/components/IslandErrorBoundary.tsx +145 -0
- package/src/components/LayoutDataErrorBoundary.tsx +141 -0
- package/src/components/LayoutErrorBoundary.tsx +127 -0
- package/src/components/PersistentIsland.tsx +52 -0
- package/src/components/StreamingErrorBoundary.tsx +233 -0
- package/src/components/StreamingLayout.tsx +538 -0
- package/src/components/tests/component-analyzer.test.ts +96 -0
- package/src/components/tests/component-detection.test.ts +347 -0
- package/src/components/tests/persistent-islands.test.ts +398 -0
- package/src/core/components/component-analyzer.ts +192 -0
- package/src/core/components/component-detection.ts +508 -0
- package/src/core/components/enhanced-framework-detector.ts +500 -0
- package/src/core/components/framework-registry.ts +563 -0
- package/src/core/components/tests/enhanced-framework-detector.test.ts +577 -0
- package/src/core/components/tests/framework-registry.test.ts +465 -0
- package/src/core/content/mdx-processor.ts +46 -0
- package/src/core/integrations/README.md +282 -0
- package/src/core/integrations/index.ts +19 -0
- package/src/core/integrations/loader.ts +125 -0
- package/src/core/integrations/registry.ts +195 -0
- package/src/core/islands/island-persistence.ts +325 -0
- package/src/core/islands/island-state-serializer.ts +258 -0
- package/src/core/islands/persistent-island-context.tsx +80 -0
- package/src/core/islands/use-persistent-state.ts +68 -0
- package/src/core/layout/enhanced-layout-resolver.ts +322 -0
- package/src/core/layout/layout-cache-manager.ts +485 -0
- package/src/core/layout/layout-composer.ts +357 -0
- package/src/core/layout/layout-data-loader.ts +516 -0
- package/src/core/layout/layout-discovery.ts +243 -0
- package/src/core/layout/layout-matcher.ts +299 -0
- package/src/core/layout/layout-types.ts +110 -0
- package/src/core/layout/tests/enhanced-layout-resolver.test.ts +477 -0
- package/src/core/layout/tests/layout-cache-optimization.test.ts +149 -0
- package/src/core/layout/tests/layout-composer.test.ts +486 -0
- package/src/core/layout/tests/layout-data-loader.test.ts +443 -0
- package/src/core/layout/tests/layout-discovery.test.ts +253 -0
- package/src/core/layout/tests/layout-matcher.test.ts +480 -0
- package/src/core/modules/framework-module-resolver.ts +273 -0
- package/src/core/modules/tests/framework-module-resolver.test.ts +263 -0
- package/src/core/modules/tests/module-resolution-integration.test.ts +117 -0
- package/src/islands/component-analysis.ts +213 -0
- package/src/islands/css-utils.ts +565 -0
- package/src/islands/discovery/index.ts +80 -0
- package/src/islands/discovery/registry.ts +340 -0
- package/src/islands/discovery/resolver.ts +477 -0
- package/src/islands/discovery/scanner.ts +386 -0
- package/src/islands/discovery/tests/island-discovery.test.ts +881 -0
- package/src/islands/discovery/types.ts +117 -0
- package/src/islands/discovery/validator.ts +544 -0
- package/src/islands/discovery/watcher.ts +368 -0
- package/src/islands/framework-detection.ts +428 -0
- package/src/islands/integration-loader.ts +490 -0
- package/src/islands/island.tsx +565 -0
- package/src/islands/render-cache.ts +550 -0
- package/src/islands/types.ts +80 -0
- package/src/islands/universal-css-collector.ts +157 -0
- package/src/islands/universal-head-collector.ts +137 -0
- package/src/layout-system.d.ts +592 -0
- package/src/layout-system.ts +218 -0
- package/src/middleware/__tests__/discovery.test.ts +107 -0
- package/src/middleware/discovery.ts +268 -0
- package/src/middleware/executor.ts +315 -0
- package/src/middleware/index.ts +76 -0
- package/src/middleware/types.ts +99 -0
- package/src/nitro/build-config.ts +576 -0
- package/src/nitro/config.ts +483 -0
- package/src/nitro/error-handler.ts +636 -0
- package/src/nitro/index.ts +173 -0
- package/src/nitro/island-manifest.ts +584 -0
- package/src/nitro/middleware-adapter.ts +260 -0
- package/src/nitro/renderer.ts +1458 -0
- package/src/nitro/route-discovery.ts +439 -0
- package/src/nitro/types.ts +321 -0
- package/src/render/collect-css.ts +198 -0
- package/src/render/error-pages.ts +79 -0
- package/src/render/isolated-ssr-renderer.ts +654 -0
- package/src/render/ssr.ts +1030 -0
- package/src/schemas/api.ts +30 -0
- package/src/schemas/core.ts +64 -0
- package/src/schemas/index.ts +212 -0
- package/src/schemas/layout.ts +279 -0
- package/src/schemas/routing/index.ts +38 -0
- package/src/schemas/routing.ts +376 -0
- package/src/types/as-island.ts +20 -0
- package/src/types/image.d.ts +106 -0
- package/src/types/index.d.ts +22 -0
- package/src/types/island-jsx.d.ts +33 -0
- package/src/types/island-prop.d.ts +20 -0
- package/src/types/layout.ts +285 -0
- package/src/types/mdx.d.ts +6 -0
- package/src/types/routing.ts +555 -0
- package/src/types/tests/layout-types.test.ts +197 -0
- package/src/types/types.ts +5 -0
- package/src/types/urlpattern.d.ts +49 -0
- package/src/types/vite-env.d.ts +11 -0
- package/src/utils/dev-logger.ts +299 -0
- package/src/utils/fs.ts +151 -0
- package/src/vite-plugin/auto-discover.ts +551 -0
- package/src/vite-plugin/config.ts +266 -0
- package/src/vite-plugin/errors.ts +127 -0
- package/src/vite-plugin/image-optimization.ts +151 -0
- package/src/vite-plugin/integration-activator.ts +126 -0
- package/src/vite-plugin/island-sidecar-plugin.ts +176 -0
- package/src/vite-plugin/module-discovery.ts +189 -0
- package/src/vite-plugin/nitro-integration.ts +1334 -0
- package/src/vite-plugin/plugin.ts +329 -0
- package/src/vite-plugin/tests/image-optimization.test.ts +54 -0
- package/src/vite-plugin/types.ts +327 -0
- package/src/vite-plugin/validation.ts +228 -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
|
+
}
|