@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,508 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Component Detection System
|
|
3
|
+
*
|
|
4
|
+
* This module provides utilities to analyze components and determine their
|
|
5
|
+
* hydration requirements based on script content and framework patterns.
|
|
6
|
+
*/
|
|
7
|
+
|
|
8
|
+
export interface ComponentAnalysis {
|
|
9
|
+
hasScript: boolean;
|
|
10
|
+
hasHydrateFunction: boolean;
|
|
11
|
+
framework: 'vue' | 'svelte' | 'solid' | 'preact' | 'react' | 'lit' | 'qwik' | 'unknown';
|
|
12
|
+
recommendedStrategy: 'hydrate' | 'ssr-only';
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
export interface DetectionResult {
|
|
16
|
+
shouldHydrate: boolean;
|
|
17
|
+
reason: string;
|
|
18
|
+
warnings?: string[];
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
export interface ComponentMetadata {
|
|
22
|
+
path: string;
|
|
23
|
+
framework: 'vue' | 'svelte' | 'solid' | 'preact' | 'react' | 'lit' | 'qwik';
|
|
24
|
+
hasScript: boolean;
|
|
25
|
+
hasHydrateFunction: boolean;
|
|
26
|
+
renderStrategy: 'hydrate' | 'ssr-only';
|
|
27
|
+
detectionConfidence: 'high' | 'medium' | 'low';
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
// Framework detection patterns
|
|
31
|
+
const FRAMEWORK_PATTERNS = {
|
|
32
|
+
vue: {
|
|
33
|
+
fileExtensions: ['.vue'],
|
|
34
|
+
scriptTags: ['<script>', '<script setup>', '<script lang="ts">', '<script setup lang="ts">'],
|
|
35
|
+
hydratePatterns: ['hydrate', 'mount', 'createApp', 'Vue.createApp'],
|
|
36
|
+
imports: ['vue', '@vue/', 'vue/'],
|
|
37
|
+
},
|
|
38
|
+
svelte: {
|
|
39
|
+
fileExtensions: ['.svelte'],
|
|
40
|
+
scriptTags: ['<script>', '<script lang="ts">', '<script context="module">'],
|
|
41
|
+
hydratePatterns: ['hydrate', 'mount', '$:', 'onMount'],
|
|
42
|
+
imports: ['svelte', 'svelte/'],
|
|
43
|
+
},
|
|
44
|
+
solid: {
|
|
45
|
+
fileExtensions: ['.tsx', '.jsx', '.solid.tsx', '.solid.jsx'],
|
|
46
|
+
scriptTags: [], // Solid uses JSX, no separate script tags
|
|
47
|
+
hydratePatterns: ['hydrate', 'render', 'createSignal', 'createEffect'],
|
|
48
|
+
imports: ['solid-js', 'solid-js/web'],
|
|
49
|
+
},
|
|
50
|
+
preact: {
|
|
51
|
+
fileExtensions: ['.tsx', '.jsx', '.preact.tsx', '.preact.jsx'],
|
|
52
|
+
scriptTags: [], // Preact uses JSX, no separate script tags
|
|
53
|
+
hydratePatterns: ['hydrate', 'render'],
|
|
54
|
+
imports: ['preact', 'preact/hooks'],
|
|
55
|
+
},
|
|
56
|
+
react: {
|
|
57
|
+
fileExtensions: ['.tsx', '.jsx', '.react.tsx', '.react.jsx'],
|
|
58
|
+
scriptTags: [], // React uses JSX, no separate script tags
|
|
59
|
+
hydratePatterns: ['hydrate', 'render'],
|
|
60
|
+
imports: ['react', 'react-dom'],
|
|
61
|
+
},
|
|
62
|
+
lit: {
|
|
63
|
+
fileExtensions: ['.ts', '.js'],
|
|
64
|
+
scriptTags: [], // Lit uses TypeScript/JavaScript classes
|
|
65
|
+
hydratePatterns: ['LitElement', 'customElement', '@customElement'],
|
|
66
|
+
imports: ['lit', 'lit-element', 'lit/'],
|
|
67
|
+
},
|
|
68
|
+
qwik: {
|
|
69
|
+
fileExtensions: ['.tsx', '.jsx', '.qwik.tsx', '.qwik.jsx'],
|
|
70
|
+
scriptTags: [], // Qwik uses JSX with component$
|
|
71
|
+
hydratePatterns: ['component$', 'useSignal', 'useStore', 'useTask$', 'useVisibleTask$'],
|
|
72
|
+
imports: ['@builder.io/qwik', '@builder.io/qwik/'],
|
|
73
|
+
},
|
|
74
|
+
} as const;
|
|
75
|
+
|
|
76
|
+
/** Detect framework from explicit naming conventions in the file path */
|
|
77
|
+
function detectByNamingConvention(filePath: string): ComponentAnalysis['framework'] | null {
|
|
78
|
+
if (filePath.includes('.solid.')) return 'solid';
|
|
79
|
+
if (filePath.includes('.preact.')) return 'preact';
|
|
80
|
+
if (filePath.includes('.react.')) return 'react';
|
|
81
|
+
if (filePath.includes('.qwik.')) return 'qwik';
|
|
82
|
+
return null;
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
/** Detect framework from file extension alone */
|
|
86
|
+
function detectByExtension(filePath: string, content: string): ComponentAnalysis['framework'] | null {
|
|
87
|
+
if (filePath.endsWith('.vue')) return 'vue';
|
|
88
|
+
if (filePath.endsWith('.svelte')) return 'svelte';
|
|
89
|
+
|
|
90
|
+
if (filePath.endsWith('.ts') || filePath.endsWith('.js')) {
|
|
91
|
+
if (content.includes('lit') || content.includes('LitElement') || content.includes('@customElement')) {
|
|
92
|
+
return 'lit';
|
|
93
|
+
}
|
|
94
|
+
}
|
|
95
|
+
|
|
96
|
+
if (filePath.endsWith('.tsx') || filePath.endsWith('.jsx')) {
|
|
97
|
+
return detectJSXFramework(content);
|
|
98
|
+
}
|
|
99
|
+
|
|
100
|
+
return null;
|
|
101
|
+
}
|
|
102
|
+
|
|
103
|
+
/** Distinguish JSX frameworks by content patterns */
|
|
104
|
+
function detectJSXFramework(content: string): ComponentAnalysis['framework'] {
|
|
105
|
+
if (content.includes('solid-js') || content.includes('from "solid-js"') || content.includes("from 'solid-js'")) {
|
|
106
|
+
return 'solid';
|
|
107
|
+
}
|
|
108
|
+
if (content.includes('@builder.io/qwik') || content.includes('from "@builder.io/qwik"') || content.includes("from '@builder.io/qwik'")) {
|
|
109
|
+
return 'qwik';
|
|
110
|
+
}
|
|
111
|
+
if (content.includes('preact') || content.includes('from "preact"') || content.includes("from 'preact'")) {
|
|
112
|
+
return 'preact';
|
|
113
|
+
}
|
|
114
|
+
if (content.includes('react') && !content.includes('preact')) {
|
|
115
|
+
return 'react';
|
|
116
|
+
}
|
|
117
|
+
return 'preact';
|
|
118
|
+
}
|
|
119
|
+
|
|
120
|
+
/**
|
|
121
|
+
* Detects the framework type based on file extension and content
|
|
122
|
+
*/
|
|
123
|
+
export function detectFramework(filePath: string, content: string): ComponentAnalysis['framework'] {
|
|
124
|
+
return (
|
|
125
|
+
detectByNamingConvention(filePath) ??
|
|
126
|
+
detectByExtension(filePath, content) ??
|
|
127
|
+
detectByImports(content)
|
|
128
|
+
);
|
|
129
|
+
}
|
|
130
|
+
|
|
131
|
+
/** Fallback: detect framework by scanning imports against known patterns */
|
|
132
|
+
function detectByImports(content: string): ComponentAnalysis['framework'] {
|
|
133
|
+
for (const [framework, patterns] of Object.entries(FRAMEWORK_PATTERNS)) {
|
|
134
|
+
if (patterns.imports.some(importPattern => content.includes(importPattern))) {
|
|
135
|
+
return framework as ComponentAnalysis['framework'];
|
|
136
|
+
}
|
|
137
|
+
}
|
|
138
|
+
return 'unknown';
|
|
139
|
+
}
|
|
140
|
+
|
|
141
|
+
/**
|
|
142
|
+
* Detects if a component has script sections
|
|
143
|
+
*/
|
|
144
|
+
export function hasScriptSection(content: string, framework: ComponentAnalysis['framework']): boolean {
|
|
145
|
+
switch (framework) {
|
|
146
|
+
case 'vue':
|
|
147
|
+
// Check for any <script> tag (more flexible than exact string matching)
|
|
148
|
+
return /<script[^>]*>/i.test(content);
|
|
149
|
+
|
|
150
|
+
case 'svelte':
|
|
151
|
+
// Svelte components have script sections if they contain <script> tags
|
|
152
|
+
// Also check for module context scripts
|
|
153
|
+
return (
|
|
154
|
+
content.includes('<script>') ||
|
|
155
|
+
content.includes('<script ') ||
|
|
156
|
+
content.includes('<script\n') ||
|
|
157
|
+
content.includes('<script\t')
|
|
158
|
+
);
|
|
159
|
+
|
|
160
|
+
case 'solid':
|
|
161
|
+
case 'preact':
|
|
162
|
+
case 'react':
|
|
163
|
+
case 'qwik':
|
|
164
|
+
// JSX frameworks inherently have script content
|
|
165
|
+
return (
|
|
166
|
+
content.includes('function') || content.includes('=>') || content.includes('const') || content.includes('let')
|
|
167
|
+
);
|
|
168
|
+
|
|
169
|
+
case 'lit':
|
|
170
|
+
// Lit components are TypeScript/JavaScript classes
|
|
171
|
+
// They always have script content (class definitions)
|
|
172
|
+
return (
|
|
173
|
+
content.includes('class') || content.includes('LitElement') || content.includes('@customElement')
|
|
174
|
+
);
|
|
175
|
+
|
|
176
|
+
default:
|
|
177
|
+
// For unknown frameworks, look for any script-like patterns
|
|
178
|
+
return content.includes('<script>') || content.includes('function') || content.includes('=>');
|
|
179
|
+
}
|
|
180
|
+
}
|
|
181
|
+
|
|
182
|
+
/**
|
|
183
|
+
* Detects if a component has hydration functions
|
|
184
|
+
*/
|
|
185
|
+
export function hasHydrateFunction(content: string, framework: ComponentAnalysis['framework']): boolean {
|
|
186
|
+
const patterns = FRAMEWORK_PATTERNS[framework as keyof typeof FRAMEWORK_PATTERNS];
|
|
187
|
+
if (!patterns) return false;
|
|
188
|
+
|
|
189
|
+
// Look for explicit hydrate function exports or declarations
|
|
190
|
+
const explicitHydratePatterns = [
|
|
191
|
+
'export function hydrate',
|
|
192
|
+
'export const hydrate',
|
|
193
|
+
'function hydrate(',
|
|
194
|
+
'const hydrate =',
|
|
195
|
+
'let hydrate =',
|
|
196
|
+
'var hydrate =',
|
|
197
|
+
];
|
|
198
|
+
|
|
199
|
+
// Check for explicit hydrate functions first
|
|
200
|
+
const hasExplicitHydrate = explicitHydratePatterns.some(pattern =>
|
|
201
|
+
content.toLowerCase().includes(pattern.toLowerCase())
|
|
202
|
+
);
|
|
203
|
+
|
|
204
|
+
if (hasExplicitHydrate) return true;
|
|
205
|
+
|
|
206
|
+
// For framework-specific patterns, be more selective
|
|
207
|
+
switch (framework) {
|
|
208
|
+
case 'vue':
|
|
209
|
+
// Look for Vue-specific hydration patterns
|
|
210
|
+
return content.includes('createApp') && content.includes('mount');
|
|
211
|
+
|
|
212
|
+
case 'svelte':
|
|
213
|
+
// Look for Svelte-specific hydration patterns - be more restrictive for explicit hydrate detection
|
|
214
|
+
// Only consider it hydration-ready if it has explicit hydrate function
|
|
215
|
+
// or uses Svelte's hydrate import
|
|
216
|
+
return (
|
|
217
|
+
content.includes('import') &&
|
|
218
|
+
content.includes('hydrate') &&
|
|
219
|
+
(content.includes('svelte') || content.includes('hydrate('))
|
|
220
|
+
);
|
|
221
|
+
|
|
222
|
+
case 'solid':
|
|
223
|
+
// Look for Solid-specific hydration patterns
|
|
224
|
+
return (
|
|
225
|
+
(content.includes('hydrate') && content.includes('solid-js')) ||
|
|
226
|
+
(content.includes('render') && content.includes('solid-js/web'))
|
|
227
|
+
);
|
|
228
|
+
|
|
229
|
+
case 'preact':
|
|
230
|
+
// Look for Preact-specific hydration patterns
|
|
231
|
+
return (
|
|
232
|
+
(content.includes('hydrate') && content.includes('preact')) ||
|
|
233
|
+
(content.includes('render') && content.includes('preact'))
|
|
234
|
+
);
|
|
235
|
+
|
|
236
|
+
case 'react':
|
|
237
|
+
// Look for React-specific hydration patterns
|
|
238
|
+
return (
|
|
239
|
+
(content.includes('hydrate') && content.includes('react')) ||
|
|
240
|
+
(content.includes('render') && content.includes('react-dom'))
|
|
241
|
+
);
|
|
242
|
+
|
|
243
|
+
case 'qwik':
|
|
244
|
+
// Qwik uses resumability instead of hydration — component$ is the marker
|
|
245
|
+
return (
|
|
246
|
+
content.includes('component$') ||
|
|
247
|
+
content.includes('@builder.io/qwik')
|
|
248
|
+
);
|
|
249
|
+
|
|
250
|
+
default:
|
|
251
|
+
return false;
|
|
252
|
+
}
|
|
253
|
+
}
|
|
254
|
+
|
|
255
|
+
/**
|
|
256
|
+
* Extracts script content from Vue components
|
|
257
|
+
*/
|
|
258
|
+
export function extractVueScript(content: string): string {
|
|
259
|
+
return extractScriptTags(content);
|
|
260
|
+
}
|
|
261
|
+
|
|
262
|
+
/**
|
|
263
|
+
* Extracts script content from Svelte components
|
|
264
|
+
*/
|
|
265
|
+
export function extractSvelteScript(content: string): string {
|
|
266
|
+
return extractScriptTags(content);
|
|
267
|
+
}
|
|
268
|
+
|
|
269
|
+
/** Shared: extract `<script>` tag contents from SFC-style components */
|
|
270
|
+
function extractScriptTags(content: string): string {
|
|
271
|
+
const scriptRegex = /<script[^>]*>([\s\S]*?)<\/script>/gi;
|
|
272
|
+
const results: string[] = [];
|
|
273
|
+
let match;
|
|
274
|
+
while ((match = scriptRegex.exec(content)) !== null) {
|
|
275
|
+
results.push(match[0]);
|
|
276
|
+
}
|
|
277
|
+
return results.join('\n');
|
|
278
|
+
}
|
|
279
|
+
|
|
280
|
+
const JSX_RETURN_RE = /^\s*return\s*\(/;
|
|
281
|
+
|
|
282
|
+
/** Shared: extract script-like lines from JSX component files */
|
|
283
|
+
function extractJSXScript(content: string): string {
|
|
284
|
+
return content
|
|
285
|
+
.split('\n')
|
|
286
|
+
.filter(line => {
|
|
287
|
+
const trimmed = line.trim();
|
|
288
|
+
return !trimmed.startsWith('<') && !trimmed.startsWith('</') && !JSX_RETURN_RE.exec(trimmed);
|
|
289
|
+
})
|
|
290
|
+
.join('\n');
|
|
291
|
+
}
|
|
292
|
+
|
|
293
|
+
/**
|
|
294
|
+
* Extracts script content from Solid components (JSX)
|
|
295
|
+
*/
|
|
296
|
+
export function extractSolidScript(content: string): string {
|
|
297
|
+
return extractJSXScript(content);
|
|
298
|
+
}
|
|
299
|
+
|
|
300
|
+
/**
|
|
301
|
+
* Extracts script content from Preact components (JSX)
|
|
302
|
+
*/
|
|
303
|
+
export function extractPreactScript(content: string): string {
|
|
304
|
+
return extractJSXScript(content);
|
|
305
|
+
}
|
|
306
|
+
|
|
307
|
+
function extractQwikScript(content: string): string {
|
|
308
|
+
return extractJSXScript(content);
|
|
309
|
+
}
|
|
310
|
+
|
|
311
|
+
/**
|
|
312
|
+
* Extracts script content from React components (JSX)
|
|
313
|
+
*/
|
|
314
|
+
export function extractReactScript(content: string): string {
|
|
315
|
+
return extractJSXScript(content);
|
|
316
|
+
}
|
|
317
|
+
|
|
318
|
+
/**
|
|
319
|
+
* Analyzes a component and returns detailed analysis
|
|
320
|
+
*/
|
|
321
|
+
export function analyzeComponent(filePath: string, content: string): ComponentAnalysis {
|
|
322
|
+
const framework = detectFramework(filePath, content);
|
|
323
|
+
const hasScript = hasScriptSection(content, framework);
|
|
324
|
+
const hasHydrate = hasScript ? hasHydrateFunction(content, framework) : false;
|
|
325
|
+
|
|
326
|
+
// Determine recommended strategy with framework-specific logic
|
|
327
|
+
let recommendedStrategy: ComponentAnalysis['recommendedStrategy'];
|
|
328
|
+
|
|
329
|
+
if (hasScript) {
|
|
330
|
+
// Framework-specific hydration strategy decisions
|
|
331
|
+
switch (framework) {
|
|
332
|
+
case 'svelte':
|
|
333
|
+
// Svelte components with script sections are typically interactive
|
|
334
|
+
// Use more intelligent detection based on content patterns
|
|
335
|
+
recommendedStrategy = analyzeSvelteHydrationStrategy(content, hasHydrate);
|
|
336
|
+
break;
|
|
337
|
+
|
|
338
|
+
case 'solid':
|
|
339
|
+
case 'preact':
|
|
340
|
+
case 'react':
|
|
341
|
+
// JSX frameworks typically need hydration if they have script content
|
|
342
|
+
recommendedStrategy = 'hydrate';
|
|
343
|
+
break;
|
|
344
|
+
|
|
345
|
+
case 'qwik':
|
|
346
|
+
// Qwik uses resumability — components with component$ always resume on client
|
|
347
|
+
recommendedStrategy = 'hydrate';
|
|
348
|
+
break;
|
|
349
|
+
|
|
350
|
+
case 'vue':
|
|
351
|
+
// Vue components need explicit hydration indicators
|
|
352
|
+
recommendedStrategy = hasHydrate ? 'hydrate' : 'ssr-only';
|
|
353
|
+
break;
|
|
354
|
+
|
|
355
|
+
default:
|
|
356
|
+
// Conservative approach for unknown frameworks
|
|
357
|
+
recommendedStrategy = hasHydrate ? 'hydrate' : 'ssr-only';
|
|
358
|
+
break;
|
|
359
|
+
}
|
|
360
|
+
} else {
|
|
361
|
+
// No script = pure template component = SSR-only
|
|
362
|
+
recommendedStrategy = 'ssr-only';
|
|
363
|
+
}
|
|
364
|
+
|
|
365
|
+
return {
|
|
366
|
+
hasScript,
|
|
367
|
+
hasHydrateFunction: hasHydrate,
|
|
368
|
+
framework,
|
|
369
|
+
recommendedStrategy,
|
|
370
|
+
};
|
|
371
|
+
}
|
|
372
|
+
|
|
373
|
+
/**
|
|
374
|
+
* Analyzes Svelte component to determine if it needs hydration
|
|
375
|
+
*/
|
|
376
|
+
function analyzeSvelteHydrationStrategy(content: string, hasExplicitHydrate: boolean): ComponentAnalysis['recommendedStrategy'] {
|
|
377
|
+
// If there's an explicit hydrate function, definitely hydrate
|
|
378
|
+
if (hasExplicitHydrate) {
|
|
379
|
+
return 'hydrate';
|
|
380
|
+
}
|
|
381
|
+
|
|
382
|
+
// Check for interactive patterns that indicate need for hydration
|
|
383
|
+
const interactivePatterns = [
|
|
384
|
+
'on:', // Event handlers (Svelte 4 style)
|
|
385
|
+
'onclick', // Event handlers (HTML style)
|
|
386
|
+
'onchange',
|
|
387
|
+
'oninput',
|
|
388
|
+
'onsubmit',
|
|
389
|
+
'onkeydown',
|
|
390
|
+
'onkeyup',
|
|
391
|
+
'onmousedown',
|
|
392
|
+
'onmouseup',
|
|
393
|
+
'bind:', // Two-way bindings
|
|
394
|
+
'$:', // Reactive statements
|
|
395
|
+
'$state', // Svelte 5 runes
|
|
396
|
+
'$derived',
|
|
397
|
+
'$effect',
|
|
398
|
+
'$props',
|
|
399
|
+
'onMount', // Lifecycle functions
|
|
400
|
+
'onDestroy',
|
|
401
|
+
'beforeUpdate',
|
|
402
|
+
'afterUpdate',
|
|
403
|
+
'tick',
|
|
404
|
+
'writable', // Stores
|
|
405
|
+
'readable',
|
|
406
|
+
'derived',
|
|
407
|
+
'get(',
|
|
408
|
+
'set(',
|
|
409
|
+
'update(',
|
|
410
|
+
'subscribe(',
|
|
411
|
+
];
|
|
412
|
+
|
|
413
|
+
const hasInteractiveFeatures = interactivePatterns.some(pattern => content.includes(pattern));
|
|
414
|
+
|
|
415
|
+
// Check for static-only patterns
|
|
416
|
+
const staticPatterns = [
|
|
417
|
+
'export let', // Only props, no interactivity
|
|
418
|
+
];
|
|
419
|
+
|
|
420
|
+
const isLikelyStatic = staticPatterns.some(pattern => content.includes(pattern)) && !hasInteractiveFeatures;
|
|
421
|
+
|
|
422
|
+
if (isLikelyStatic) {
|
|
423
|
+
return 'ssr-only';
|
|
424
|
+
}
|
|
425
|
+
|
|
426
|
+
// Default to hydration for Svelte components with script sections
|
|
427
|
+
// This is safer and aligns with Svelte's typical usage patterns
|
|
428
|
+
return 'hydrate';
|
|
429
|
+
}
|
|
430
|
+
|
|
431
|
+
/** Framework-specific hydration reasons for known frameworks with script sections */
|
|
432
|
+
const FRAMEWORK_HYDRATION_REASONS: Record<string, string> = {
|
|
433
|
+
solid: 'SolidJS component detected - uses integration system',
|
|
434
|
+
preact: 'preact component with script content - likely needs hydration',
|
|
435
|
+
react: 'react component with script content - likely needs hydration',
|
|
436
|
+
svelte: 'Svelte component with script section - uses Svelte hydration system',
|
|
437
|
+
vue: 'Vue component with script section - uses Vue integration system',
|
|
438
|
+
lit: 'Lit component detected - Web Components require client-side registration',
|
|
439
|
+
qwik: 'Qwik component detected - uses resumability instead of hydration',
|
|
440
|
+
};
|
|
441
|
+
|
|
442
|
+
/**
|
|
443
|
+
* Determines if a component should be hydrated based on analysis
|
|
444
|
+
*/
|
|
445
|
+
export function shouldHydrateComponent(
|
|
446
|
+
analysis: ComponentAnalysis,
|
|
447
|
+
options: { forceSSROnly?: boolean; detectScripts?: boolean } = {}
|
|
448
|
+
): DetectionResult {
|
|
449
|
+
if (options.forceSSROnly) {
|
|
450
|
+
return { shouldHydrate: false, reason: 'Explicitly configured for SSR-only rendering' };
|
|
451
|
+
}
|
|
452
|
+
|
|
453
|
+
if (options.detectScripts === false) {
|
|
454
|
+
return { shouldHydrate: true, reason: 'Script detection disabled, defaulting to hydration' };
|
|
455
|
+
}
|
|
456
|
+
|
|
457
|
+
if (!analysis.hasScript) {
|
|
458
|
+
return { shouldHydrate: false, reason: 'No script section detected, using SSR-only rendering' };
|
|
459
|
+
}
|
|
460
|
+
|
|
461
|
+
// Known frameworks with script sections always hydrate
|
|
462
|
+
const frameworkReason = FRAMEWORK_HYDRATION_REASONS[analysis.framework];
|
|
463
|
+
if (frameworkReason) {
|
|
464
|
+
return { shouldHydrate: true, reason: frameworkReason };
|
|
465
|
+
}
|
|
466
|
+
|
|
467
|
+
// Unknown frameworks need explicit hydrate functions
|
|
468
|
+
if (analysis.hasHydrateFunction) {
|
|
469
|
+
return { shouldHydrate: true, reason: 'Component has script section with explicit hydration functions' };
|
|
470
|
+
}
|
|
471
|
+
|
|
472
|
+
return {
|
|
473
|
+
shouldHydrate: false,
|
|
474
|
+
reason: 'Component has script section but no explicit hydrate function, using SSR-only',
|
|
475
|
+
warnings: ['Component has script section but no clear hydrate function detected'],
|
|
476
|
+
};
|
|
477
|
+
}
|
|
478
|
+
|
|
479
|
+
/**
|
|
480
|
+
* Creates component metadata with confidence scoring
|
|
481
|
+
*/
|
|
482
|
+
export function createComponentMetadata(
|
|
483
|
+
filePath: string,
|
|
484
|
+
content: string,
|
|
485
|
+
analysis: ComponentAnalysis
|
|
486
|
+
): ComponentMetadata {
|
|
487
|
+
let detectionConfidence: ComponentMetadata['detectionConfidence'] = 'medium';
|
|
488
|
+
|
|
489
|
+
// High confidence cases
|
|
490
|
+
if (analysis.framework !== 'unknown' && analysis.hasScript && analysis.hasHydrateFunction) {
|
|
491
|
+
detectionConfidence = 'high';
|
|
492
|
+
} else if (analysis.framework !== 'unknown' && !analysis.hasScript) {
|
|
493
|
+
detectionConfidence = 'high';
|
|
494
|
+
}
|
|
495
|
+
// Low confidence cases
|
|
496
|
+
else if (analysis.framework === 'unknown') {
|
|
497
|
+
detectionConfidence = 'low';
|
|
498
|
+
}
|
|
499
|
+
|
|
500
|
+
return {
|
|
501
|
+
path: filePath,
|
|
502
|
+
framework: analysis.framework === 'unknown' ? 'vue' : analysis.framework, // Default fallback to vue for backward compatibility
|
|
503
|
+
hasScript: analysis.hasScript,
|
|
504
|
+
hasHydrateFunction: analysis.hasHydrateFunction,
|
|
505
|
+
renderStrategy: analysis.recommendedStrategy,
|
|
506
|
+
detectionConfidence,
|
|
507
|
+
};
|
|
508
|
+
}
|