@useavalon/avalon 0.1.11 → 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.
- package/README.md +54 -54
- package/mod.ts +302 -302
- package/package.json +49 -26
- package/src/build/integration-bundler-plugin.ts +116 -116
- package/src/build/integration-config.ts +168 -168
- package/src/build/integration-detection-plugin.ts +117 -117
- package/src/build/integration-resolver-plugin.ts +90 -90
- package/src/build/island-manifest.ts +269 -269
- package/src/build/island-types-generator.ts +476 -476
- package/src/build/mdx-island-transform.ts +464 -464
- package/src/build/mdx-plugin.ts +98 -98
- package/src/build/page-island-transform.ts +598 -598
- package/src/build/prop-extractors/index.ts +21 -21
- package/src/build/prop-extractors/lit.ts +140 -140
- package/src/build/prop-extractors/qwik.ts +16 -16
- package/src/build/prop-extractors/solid.ts +125 -125
- package/src/build/prop-extractors/svelte.ts +194 -194
- package/src/build/prop-extractors/vue.ts +111 -111
- package/src/build/sidecar-file-manager.ts +104 -104
- package/src/build/sidecar-renderer.ts +30 -30
- package/src/client/adapters/index.ts +21 -13
- package/src/client/components.ts +35 -35
- package/src/client/css-hmr-handler.ts +344 -344
- package/src/client/framework-adapter.ts +462 -462
- package/src/client/hmr-coordinator.ts +396 -396
- package/src/client/hmr-error-overlay.js +533 -533
- package/src/client/main.js +824 -816
- package/src/client/types/framework-runtime.d.ts +68 -68
- package/src/client/types/vite-hmr.d.ts +46 -46
- package/src/client/types/vite-virtual-modules.d.ts +70 -60
- package/src/components/Image.tsx +123 -123
- package/src/components/IslandErrorBoundary.tsx +145 -145
- package/src/components/LayoutDataErrorBoundary.tsx +141 -141
- package/src/components/LayoutErrorBoundary.tsx +127 -127
- package/src/components/PersistentIsland.tsx +52 -52
- package/src/components/StreamingErrorBoundary.tsx +233 -233
- package/src/components/StreamingLayout.tsx +538 -538
- package/src/core/components/component-analyzer.ts +192 -192
- package/src/core/components/component-detection.ts +508 -508
- package/src/core/components/enhanced-framework-detector.ts +500 -500
- package/src/core/components/framework-registry.ts +563 -563
- package/src/core/content/mdx-processor.ts +46 -46
- package/src/core/integrations/index.ts +19 -19
- package/src/core/integrations/loader.ts +125 -125
- package/src/core/integrations/registry.ts +175 -175
- package/src/core/islands/island-persistence.ts +325 -325
- package/src/core/islands/island-state-serializer.ts +258 -258
- package/src/core/islands/persistent-island-context.tsx +80 -80
- package/src/core/islands/use-persistent-state.ts +68 -68
- package/src/core/layout/enhanced-layout-resolver.ts +322 -322
- package/src/core/layout/layout-cache-manager.ts +485 -485
- package/src/core/layout/layout-composer.ts +357 -357
- package/src/core/layout/layout-data-loader.ts +516 -516
- package/src/core/layout/layout-discovery.ts +243 -243
- package/src/core/layout/layout-matcher.ts +299 -299
- package/src/core/layout/layout-types.ts +110 -110
- package/src/core/modules/framework-module-resolver.ts +273 -273
- package/src/islands/component-analysis.ts +213 -213
- package/src/islands/css-utils.ts +565 -565
- package/src/islands/discovery/index.ts +80 -80
- package/src/islands/discovery/registry.ts +340 -340
- package/src/islands/discovery/resolver.ts +477 -477
- package/src/islands/discovery/scanner.ts +386 -386
- package/src/islands/discovery/types.ts +117 -117
- package/src/islands/discovery/validator.ts +544 -544
- package/src/islands/discovery/watcher.ts +368 -368
- package/src/islands/framework-detection.ts +428 -428
- package/src/islands/integration-loader.ts +490 -490
- package/src/islands/island.tsx +565 -565
- package/src/islands/render-cache.ts +550 -550
- package/src/islands/types.ts +80 -80
- package/src/islands/universal-css-collector.ts +157 -157
- package/src/islands/universal-head-collector.ts +137 -137
- package/src/layout-system.d.ts +592 -592
- package/src/layout-system.ts +218 -218
- package/src/middleware/discovery.ts +268 -268
- package/src/middleware/executor.ts +315 -315
- package/src/middleware/index.ts +76 -76
- package/src/middleware/types.ts +99 -99
- package/src/nitro/build-config.ts +575 -575
- package/src/nitro/config.ts +483 -483
- package/src/nitro/error-handler.ts +636 -636
- package/src/nitro/index.ts +173 -173
- package/src/nitro/island-manifest.ts +584 -584
- package/src/nitro/middleware-adapter.ts +260 -260
- package/src/nitro/renderer.ts +1471 -1471
- package/src/nitro/route-discovery.ts +439 -439
- package/src/nitro/types.ts +321 -321
- package/src/render/collect-css.ts +198 -198
- package/src/render/error-pages.ts +79 -79
- package/src/render/isolated-ssr-renderer.ts +654 -654
- package/src/render/ssr.ts +1030 -1030
- package/src/schemas/api.ts +30 -30
- package/src/schemas/core.ts +64 -64
- package/src/schemas/index.ts +212 -212
- package/src/schemas/layout.ts +279 -279
- package/src/schemas/routing/index.ts +38 -38
- package/src/schemas/routing.ts +376 -376
- package/src/types/as-island.ts +20 -20
- package/src/types/image.d.ts +106 -106
- package/src/types/index.d.ts +22 -22
- package/src/types/island-jsx.d.ts +33 -33
- package/src/types/island-prop.d.ts +20 -20
- package/src/types/layout.ts +285 -285
- package/src/types/mdx.d.ts +6 -6
- package/src/types/routing.ts +555 -555
- package/src/types/types.ts +5 -5
- package/src/types/urlpattern.d.ts +49 -49
- package/src/types/vite-env.d.ts +11 -11
- package/src/utils/dev-logger.ts +299 -299
- package/src/utils/fs.ts +151 -151
- package/src/vite-plugin/auto-discover.ts +551 -551
- package/src/vite-plugin/config.ts +266 -266
- package/src/vite-plugin/errors.ts +127 -127
- package/src/vite-plugin/image-optimization.ts +156 -156
- package/src/vite-plugin/integration-activator.ts +126 -126
- package/src/vite-plugin/island-sidecar-plugin.ts +176 -176
- package/src/vite-plugin/module-discovery.ts +189 -189
- package/src/vite-plugin/nitro-integration.ts +1354 -1354
- package/src/vite-plugin/plugin.ts +403 -409
- package/src/vite-plugin/types.ts +327 -327
- package/src/vite-plugin/validation.ts +228 -228
- package/src/client/adapters/index.js +0 -12
- package/src/client/adapters/lit-adapter.js +0 -467
- package/src/client/adapters/lit-adapter.ts +0 -654
- package/src/client/adapters/preact-adapter.js +0 -223
- package/src/client/adapters/preact-adapter.ts +0 -331
- package/src/client/adapters/qwik-adapter.js +0 -259
- package/src/client/adapters/qwik-adapter.ts +0 -345
- package/src/client/adapters/react-adapter.js +0 -220
- package/src/client/adapters/react-adapter.ts +0 -353
- package/src/client/adapters/solid-adapter.js +0 -295
- package/src/client/adapters/solid-adapter.ts +0 -451
- package/src/client/adapters/svelte-adapter.js +0 -368
- package/src/client/adapters/svelte-adapter.ts +0 -524
- package/src/client/adapters/vue-adapter.js +0 -278
- package/src/client/adapters/vue-adapter.ts +0 -467
- package/src/client/components.js +0 -23
- package/src/client/css-hmr-handler.js +0 -263
- package/src/client/framework-adapter.js +0 -283
- package/src/client/hmr-coordinator.js +0 -274
|
@@ -1,576 +1,576 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Nitro Build Configuration Module for Avalon
|
|
3
|
-
*
|
|
4
|
-
* This module provides build configuration utilities for integrating Nitro
|
|
5
|
-
* with Avalon's Vite-based build pipeline. It handles:
|
|
6
|
-
* - Vite client build configuration
|
|
7
|
-
* - Nitro server build configuration
|
|
8
|
-
* - Preset-specific output configuration
|
|
9
|
-
* - Source map generation
|
|
10
|
-
*
|
|
11
|
-
* @module nitro/build-config
|
|
12
|
-
*/
|
|
13
|
-
|
|
14
|
-
import type { Plugin, UserConfig, BuildOptions } from "vite";
|
|
15
|
-
import type { ResolvedAvalonConfig } from "../vite-plugin/types.ts";
|
|
16
|
-
import type { AvalonNitroConfig } from "./config.ts";
|
|
17
|
-
import { DEFAULT_NITRO_CONFIG, VALID_V3_PRESETS } from "./config.ts";
|
|
18
|
-
|
|
19
|
-
/**
|
|
20
|
-
* Build mode for the Avalon application
|
|
21
|
-
*/
|
|
22
|
-
export type BuildMode = "client" | "server" | "both";
|
|
23
|
-
|
|
24
|
-
/**
|
|
25
|
-
* Sourcemap option type
|
|
26
|
-
*/
|
|
27
|
-
export type SourcemapOption = boolean | "inline" | "hidden";
|
|
28
|
-
|
|
29
|
-
/**
|
|
30
|
-
* Minify option type
|
|
31
|
-
*/
|
|
32
|
-
export type MinifyOption = boolean | "oxc" | "esbuild" | "terser";
|
|
33
|
-
|
|
34
|
-
/**
|
|
35
|
-
* Preset-specific build output configuration
|
|
36
|
-
*/
|
|
37
|
-
export interface PresetOutputConfig {
|
|
38
|
-
/** Output directory for the preset */
|
|
39
|
-
outputDir: string;
|
|
40
|
-
/** Server entry file name */
|
|
41
|
-
serverEntry: string;
|
|
42
|
-
/** Whether the preset supports streaming */
|
|
43
|
-
supportsStreaming: boolean;
|
|
44
|
-
/** Whether the preset requires bundled dependencies */
|
|
45
|
-
bundleDependencies: boolean;
|
|
46
|
-
/** Additional files to include in output */
|
|
47
|
-
additionalFiles?: string[];
|
|
48
|
-
/** Environment-specific configuration */
|
|
49
|
-
env?: Record<string, string>;
|
|
50
|
-
}
|
|
51
|
-
|
|
52
|
-
/**
|
|
53
|
-
* Avalon build configuration options
|
|
54
|
-
*/
|
|
55
|
-
export interface AvalonBuildConfig {
|
|
56
|
-
/** Build mode: client, server, or both */
|
|
57
|
-
mode: BuildMode;
|
|
58
|
-
/** Output directory for client assets */
|
|
59
|
-
clientOutDir: string;
|
|
60
|
-
/** Output directory for server bundle */
|
|
61
|
-
serverOutDir: string;
|
|
62
|
-
/** Enable source maps */
|
|
63
|
-
sourcemap: SourcemapOption;
|
|
64
|
-
/** Minify output */
|
|
65
|
-
minify: MinifyOption;
|
|
66
|
-
/** Target environment */
|
|
67
|
-
target: string | string[];
|
|
68
|
-
/** Enable SSR build */
|
|
69
|
-
ssr: boolean;
|
|
70
|
-
/** Nitro preset for deployment */
|
|
71
|
-
preset: string;
|
|
72
|
-
/** Enable verbose logging */
|
|
73
|
-
verbose: boolean;
|
|
74
|
-
/** Nitro v3: Compatibility date for preset features (YYYY-MM-DD) */
|
|
75
|
-
compatibilityDate?: string;
|
|
76
|
-
/** Nitro v3: Dependencies to trace instead of bundle */
|
|
77
|
-
traceDeps?: string[];
|
|
78
|
-
/** Nitro v3: Rolldown-specific configuration */
|
|
79
|
-
rolldownConfig?: Record<string, unknown>;
|
|
80
|
-
/** Nitro v3: Custom server entry point */
|
|
81
|
-
serverEntry?: string;
|
|
82
|
-
}
|
|
83
|
-
|
|
84
|
-
/**
|
|
85
|
-
* Default build configuration values
|
|
86
|
-
*/
|
|
87
|
-
export const DEFAULT_BUILD_CONFIG: AvalonBuildConfig = {
|
|
88
|
-
mode: "both",
|
|
89
|
-
clientOutDir: "dist/client",
|
|
90
|
-
serverOutDir: "dist/server",
|
|
91
|
-
sourcemap: true,
|
|
92
|
-
minify: "oxc",
|
|
93
|
-
target: "es2020",
|
|
94
|
-
ssr: true,
|
|
95
|
-
preset: "node_server",
|
|
96
|
-
verbose: false,
|
|
97
|
-
};
|
|
98
|
-
|
|
99
|
-
/**
|
|
100
|
-
* Preset-specific output configurations
|
|
101
|
-
*/
|
|
102
|
-
export const PRESET_OUTPUT_CONFIGS: Record<string, PresetOutputConfig> = {
|
|
103
|
-
"node_server": {
|
|
104
|
-
outputDir: ".output",
|
|
105
|
-
serverEntry: "server/index.mjs",
|
|
106
|
-
supportsStreaming: true,
|
|
107
|
-
bundleDependencies: false,
|
|
108
|
-
},
|
|
109
|
-
"node_middleware": {
|
|
110
|
-
outputDir: ".output",
|
|
111
|
-
serverEntry: "server/index.mjs",
|
|
112
|
-
supportsStreaming: true,
|
|
113
|
-
bundleDependencies: false,
|
|
114
|
-
},
|
|
115
|
-
"vercel": {
|
|
116
|
-
outputDir: ".vercel/output",
|
|
117
|
-
serverEntry: "functions/render.func/index.mjs",
|
|
118
|
-
supportsStreaming: true,
|
|
119
|
-
bundleDependencies: true,
|
|
120
|
-
additionalFiles: ["config.json"],
|
|
121
|
-
},
|
|
122
|
-
"cloudflare_module": {
|
|
123
|
-
outputDir: "dist",
|
|
124
|
-
serverEntry: "server/index.mjs",
|
|
125
|
-
supportsStreaming: false,
|
|
126
|
-
bundleDependencies: true,
|
|
127
|
-
},
|
|
128
|
-
"cloudflare_pages": {
|
|
129
|
-
outputDir: "dist",
|
|
130
|
-
serverEntry: "_worker.js",
|
|
131
|
-
supportsStreaming: false,
|
|
132
|
-
bundleDependencies: true,
|
|
133
|
-
additionalFiles: ["_routes.json"],
|
|
134
|
-
},
|
|
135
|
-
"deno_deploy": {
|
|
136
|
-
outputDir: ".output",
|
|
137
|
-
serverEntry: "server/index.ts",
|
|
138
|
-
supportsStreaming: true,
|
|
139
|
-
bundleDependencies: false,
|
|
140
|
-
},
|
|
141
|
-
"deno_server": {
|
|
142
|
-
outputDir: ".output",
|
|
143
|
-
serverEntry: "server/index.ts",
|
|
144
|
-
supportsStreaming: true,
|
|
145
|
-
bundleDependencies: false,
|
|
146
|
-
},
|
|
147
|
-
"netlify_functions": {
|
|
148
|
-
outputDir: ".netlify",
|
|
149
|
-
serverEntry: "functions-internal/render.mjs",
|
|
150
|
-
supportsStreaming: true,
|
|
151
|
-
bundleDependencies: true,
|
|
152
|
-
},
|
|
153
|
-
"netlify_edge": {
|
|
154
|
-
outputDir: ".netlify/edge-functions",
|
|
155
|
-
serverEntry: "render.js",
|
|
156
|
-
supportsStreaming: true,
|
|
157
|
-
bundleDependencies: true,
|
|
158
|
-
},
|
|
159
|
-
"aws_lambda": {
|
|
160
|
-
outputDir: ".output",
|
|
161
|
-
serverEntry: "server/index.mjs",
|
|
162
|
-
supportsStreaming: false,
|
|
163
|
-
bundleDependencies: true,
|
|
164
|
-
},
|
|
165
|
-
"azure_swa": {
|
|
166
|
-
outputDir: ".output",
|
|
167
|
-
serverEntry: "server/index.mjs",
|
|
168
|
-
supportsStreaming: false,
|
|
169
|
-
bundleDependencies: true,
|
|
170
|
-
},
|
|
171
|
-
"firebase_functions": {
|
|
172
|
-
outputDir: ".output",
|
|
173
|
-
serverEntry: "server/index.mjs",
|
|
174
|
-
supportsStreaming: true,
|
|
175
|
-
bundleDependencies: true,
|
|
176
|
-
},
|
|
177
|
-
"render_com": {
|
|
178
|
-
outputDir: ".output",
|
|
179
|
-
serverEntry: "server/index.mjs",
|
|
180
|
-
supportsStreaming: true,
|
|
181
|
-
bundleDependencies: false,
|
|
182
|
-
},
|
|
183
|
-
"static": {
|
|
184
|
-
outputDir: "dist",
|
|
185
|
-
serverEntry: "",
|
|
186
|
-
supportsStreaming: false,
|
|
187
|
-
bundleDependencies: false,
|
|
188
|
-
},
|
|
189
|
-
"browser": {
|
|
190
|
-
outputDir: "dist",
|
|
191
|
-
serverEntry: "",
|
|
192
|
-
supportsStreaming: false,
|
|
193
|
-
bundleDependencies: true,
|
|
194
|
-
},
|
|
195
|
-
};
|
|
196
|
-
|
|
197
|
-
/**
|
|
198
|
-
* Creates the Vite client build configuration
|
|
199
|
-
*
|
|
200
|
-
* @param avalonConfig - Resolved Avalon configuration
|
|
201
|
-
* @param buildConfig - Build configuration options
|
|
202
|
-
* @returns Vite build options for client build
|
|
203
|
-
*/
|
|
204
|
-
export function createClientBuildConfig(
|
|
205
|
-
avalonConfig: ResolvedAvalonConfig,
|
|
206
|
-
buildConfig: Partial<AvalonBuildConfig> = {}
|
|
207
|
-
): BuildOptions {
|
|
208
|
-
const config = { ...DEFAULT_BUILD_CONFIG, ...buildConfig };
|
|
209
|
-
|
|
210
|
-
return {
|
|
211
|
-
outDir: config.clientOutDir,
|
|
212
|
-
emptyOutDir: true,
|
|
213
|
-
sourcemap: config.sourcemap,
|
|
214
|
-
minify: config.minify,
|
|
215
|
-
target: config.target,
|
|
216
|
-
rolldownOptions: {
|
|
217
|
-
output: {
|
|
218
|
-
entryFileNames: "[name].[hash].js",
|
|
219
|
-
chunkFileNames: "chunks/[name].[hash].js",
|
|
220
|
-
assetFileNames: "assets/[name].[hash].[ext]",
|
|
221
|
-
codeSplitting: {
|
|
222
|
-
groups: [
|
|
223
|
-
{ name: "vendor-react", test: /node_modules\/(react|react-dom)/ },
|
|
224
|
-
{ name: "vendor-vue", test: /node_modules\/(vue|@vue)/ },
|
|
225
|
-
{ name: "vendor-svelte", test: /node_modules\/svelte/ },
|
|
226
|
-
{ name: "vendor-preact", test: /node_modules\/preact/ },
|
|
227
|
-
{ name: "vendor-solid", test: /node_modules\/solid-js/ },
|
|
228
|
-
{ name: "vendor-lit", test: /node_modules\/(lit|@lit)/ },
|
|
229
|
-
{ name: "vendor", test: /node_modules/ },
|
|
230
|
-
{ name: "islands", test: /\/islands\// },
|
|
231
|
-
],
|
|
232
|
-
},
|
|
233
|
-
},
|
|
234
|
-
},
|
|
235
|
-
reportCompressedSize: !avalonConfig.isDev,
|
|
236
|
-
cssCodeSplit: true,
|
|
237
|
-
};
|
|
238
|
-
}
|
|
239
|
-
|
|
240
|
-
/**
|
|
241
|
-
* Creates the Vite server build configuration for Nitro
|
|
242
|
-
*
|
|
243
|
-
* @param avalonConfig - Resolved Avalon configuration
|
|
244
|
-
* @param nitroConfig - Nitro configuration
|
|
245
|
-
* @param buildConfig - Build configuration options
|
|
246
|
-
* @returns Vite build options for server build
|
|
247
|
-
*/
|
|
248
|
-
export function createServerBuildConfig(
|
|
249
|
-
avalonConfig: ResolvedAvalonConfig,
|
|
250
|
-
nitroConfig: AvalonNitroConfig,
|
|
251
|
-
buildConfig: Partial<AvalonBuildConfig> = {}
|
|
252
|
-
): BuildOptions {
|
|
253
|
-
const config = { ...DEFAULT_BUILD_CONFIG, ...buildConfig };
|
|
254
|
-
const preset = nitroConfig.preset ?? DEFAULT_NITRO_CONFIG.preset;
|
|
255
|
-
const presetConfig = PRESET_OUTPUT_CONFIGS[preset] ?? PRESET_OUTPUT_CONFIGS["node_server"];
|
|
256
|
-
|
|
257
|
-
return {
|
|
258
|
-
outDir: config.serverOutDir,
|
|
259
|
-
emptyOutDir: true,
|
|
260
|
-
sourcemap: config.sourcemap,
|
|
261
|
-
minify: config.minify,
|
|
262
|
-
target: config.target,
|
|
263
|
-
ssr: true,
|
|
264
|
-
rolldownOptions: {
|
|
265
|
-
input: {
|
|
266
|
-
index: "./server/index.ts",
|
|
267
|
-
},
|
|
268
|
-
output: {
|
|
269
|
-
format: "esm",
|
|
270
|
-
entryFileNames: "[name].mjs",
|
|
271
|
-
chunkFileNames: "chunks/[name].[hash].mjs",
|
|
272
|
-
preserveModules: !presetConfig.bundleDependencies,
|
|
273
|
-
},
|
|
274
|
-
external: presetConfig.bundleDependencies
|
|
275
|
-
? []
|
|
276
|
-
: getServerExternals(preset),
|
|
277
|
-
},
|
|
278
|
-
};
|
|
279
|
-
}
|
|
280
|
-
|
|
281
|
-
/**
|
|
282
|
-
* Gets external dependencies for server build based on preset
|
|
283
|
-
*
|
|
284
|
-
* @param preset - Nitro preset name
|
|
285
|
-
* @returns Array of external dependency patterns
|
|
286
|
-
*/
|
|
287
|
-
function getServerExternals(preset: string): (string | RegExp)[] {
|
|
288
|
-
const baseExternals: (string | RegExp)[] = [
|
|
289
|
-
/^node:/,
|
|
290
|
-
/^deno:/,
|
|
291
|
-
];
|
|
292
|
-
|
|
293
|
-
switch (preset) {
|
|
294
|
-
case "deno_deploy":
|
|
295
|
-
case "deno_server":
|
|
296
|
-
return [...baseExternals];
|
|
297
|
-
|
|
298
|
-
case "node_server":
|
|
299
|
-
return [
|
|
300
|
-
...baseExternals,
|
|
301
|
-
/^[a-z@]/i,
|
|
302
|
-
];
|
|
303
|
-
|
|
304
|
-
default:
|
|
305
|
-
return baseExternals;
|
|
306
|
-
}
|
|
307
|
-
}
|
|
308
|
-
|
|
309
|
-
/**
|
|
310
|
-
* Gets the preset output configuration
|
|
311
|
-
*
|
|
312
|
-
* @param preset - Nitro preset name
|
|
313
|
-
* @returns Preset output configuration
|
|
314
|
-
*/
|
|
315
|
-
export function getPresetOutputConfig(preset: string): PresetOutputConfig {
|
|
316
|
-
return PRESET_OUTPUT_CONFIGS[preset] ?? PRESET_OUTPUT_CONFIGS["node_server"];
|
|
317
|
-
}
|
|
318
|
-
|
|
319
|
-
/**
|
|
320
|
-
* Checks if a preset supports streaming SSR
|
|
321
|
-
*
|
|
322
|
-
* @param preset - Nitro preset name
|
|
323
|
-
* @returns True if the preset supports streaming
|
|
324
|
-
*/
|
|
325
|
-
export function presetSupportsStreaming(preset: string): boolean {
|
|
326
|
-
const config = getPresetOutputConfig(preset);
|
|
327
|
-
return config.supportsStreaming;
|
|
328
|
-
}
|
|
329
|
-
|
|
330
|
-
/**
|
|
331
|
-
* Creates the combined build configuration for both client and server
|
|
332
|
-
*
|
|
333
|
-
* @param avalonConfig - Resolved Avalon configuration
|
|
334
|
-
* @param nitroConfig - Nitro configuration
|
|
335
|
-
* @param buildConfig - Build configuration options
|
|
336
|
-
* @returns Combined Vite user config
|
|
337
|
-
*/
|
|
338
|
-
export function createCombinedBuildConfig(
|
|
339
|
-
avalonConfig: ResolvedAvalonConfig,
|
|
340
|
-
nitroConfig: AvalonNitroConfig,
|
|
341
|
-
buildConfig: Partial<AvalonBuildConfig> = {}
|
|
342
|
-
): Partial<UserConfig> {
|
|
343
|
-
const preset = nitroConfig.preset ?? DEFAULT_NITRO_CONFIG.preset;
|
|
344
|
-
const presetConfig = getPresetOutputConfig(preset);
|
|
345
|
-
|
|
346
|
-
return {
|
|
347
|
-
build: {
|
|
348
|
-
...createClientBuildConfig(avalonConfig, buildConfig),
|
|
349
|
-
outDir: presetConfig.outputDir,
|
|
350
|
-
},
|
|
351
|
-
define: {
|
|
352
|
-
__DEV__: false,
|
|
353
|
-
__PROD__: true,
|
|
354
|
-
"process.env.NODE_ENV": JSON.stringify("production"),
|
|
355
|
-
...createPresetDefines(preset, presetConfig),
|
|
356
|
-
},
|
|
357
|
-
};
|
|
358
|
-
}
|
|
359
|
-
|
|
360
|
-
/**
|
|
361
|
-
* Creates preset-specific define constants
|
|
362
|
-
*/
|
|
363
|
-
function createPresetDefines(
|
|
364
|
-
preset: string,
|
|
365
|
-
presetConfig: PresetOutputConfig
|
|
366
|
-
): Record<string, string> {
|
|
367
|
-
const defines: Record<string, string> = {
|
|
368
|
-
__NITRO_PRESET__: JSON.stringify(preset),
|
|
369
|
-
__SUPPORTS_STREAMING__: JSON.stringify(presetConfig.supportsStreaming),
|
|
370
|
-
};
|
|
371
|
-
|
|
372
|
-
if (presetConfig.env) {
|
|
373
|
-
for (const [key, value] of Object.entries(presetConfig.env)) {
|
|
374
|
-
defines[`process.env.${key}`] = JSON.stringify(value);
|
|
375
|
-
}
|
|
376
|
-
}
|
|
377
|
-
|
|
378
|
-
return defines;
|
|
379
|
-
}
|
|
380
|
-
|
|
381
|
-
/**
|
|
382
|
-
* Creates a Vite plugin for Nitro build integration
|
|
383
|
-
*/
|
|
384
|
-
export function createNitroBuildPlugin(
|
|
385
|
-
avalonConfig: ResolvedAvalonConfig,
|
|
386
|
-
nitroConfig: AvalonNitroConfig
|
|
387
|
-
): Plugin {
|
|
388
|
-
const preset = nitroConfig.preset ?? DEFAULT_NITRO_CONFIG.preset;
|
|
389
|
-
const presetConfig = getPresetOutputConfig(preset);
|
|
390
|
-
|
|
391
|
-
return {
|
|
392
|
-
name: "avalon:nitro-build",
|
|
393
|
-
enforce: "post",
|
|
394
|
-
|
|
395
|
-
config(_config: UserConfig, { command }: { command: string }) {
|
|
396
|
-
if (command !== "build") {
|
|
397
|
-
return;
|
|
398
|
-
}
|
|
399
|
-
|
|
400
|
-
return createCombinedBuildConfig(avalonConfig, nitroConfig, {
|
|
401
|
-
sourcemap: true,
|
|
402
|
-
verbose: avalonConfig.verbose,
|
|
403
|
-
});
|
|
404
|
-
},
|
|
405
|
-
|
|
406
|
-
buildStart() {
|
|
407
|
-
if (avalonConfig.verbose) {
|
|
408
|
-
console.log("🚀 Avalon Nitro build starting...");
|
|
409
|
-
console.log(` Preset: ${preset}`);
|
|
410
|
-
console.log(` Output: ${presetConfig.outputDir}`);
|
|
411
|
-
console.log(` Streaming: ${presetConfig.supportsStreaming}`);
|
|
412
|
-
}
|
|
413
|
-
},
|
|
414
|
-
|
|
415
|
-
writeBundle() {
|
|
416
|
-
if (avalonConfig.verbose) {
|
|
417
|
-
console.log("✅ Avalon client build complete");
|
|
418
|
-
}
|
|
419
|
-
},
|
|
420
|
-
|
|
421
|
-
closeBundle() {
|
|
422
|
-
if (avalonConfig.verbose) {
|
|
423
|
-
console.log("📦 Avalon build finished");
|
|
424
|
-
console.log(` Output directory: ${presetConfig.outputDir}`);
|
|
425
|
-
}
|
|
426
|
-
},
|
|
427
|
-
};
|
|
428
|
-
}
|
|
429
|
-
|
|
430
|
-
/**
|
|
431
|
-
* Validates build configuration
|
|
432
|
-
*/
|
|
433
|
-
export function validateBuildConfig(
|
|
434
|
-
config: Partial<AvalonBuildConfig>
|
|
435
|
-
): { valid: boolean; errors: string[] } {
|
|
436
|
-
const errors: string[] = [];
|
|
437
|
-
|
|
438
|
-
if (config.preset) {
|
|
439
|
-
if (!VALID_V3_PRESETS.includes(config.preset)) {
|
|
440
|
-
errors.push(
|
|
441
|
-
`Unknown preset: ${config.preset}. Valid presets: ${VALID_V3_PRESETS.join(", ")}`
|
|
442
|
-
);
|
|
443
|
-
}
|
|
444
|
-
}
|
|
445
|
-
|
|
446
|
-
if (config.mode && !["client", "server", "both"].includes(config.mode)) {
|
|
447
|
-
errors.push(`Invalid build mode: ${config.mode}. Must be 'client', 'server', or 'both'`);
|
|
448
|
-
}
|
|
449
|
-
|
|
450
|
-
if (
|
|
451
|
-
config.sourcemap !== undefined &&
|
|
452
|
-
typeof config.sourcemap !== "boolean" &&
|
|
453
|
-
!["inline", "hidden"].includes(config.sourcemap as string)
|
|
454
|
-
) {
|
|
455
|
-
errors.push(`Invalid sourcemap option: ${config.sourcemap}`);
|
|
456
|
-
}
|
|
457
|
-
|
|
458
|
-
return {
|
|
459
|
-
valid: errors.length === 0,
|
|
460
|
-
errors,
|
|
461
|
-
};
|
|
462
|
-
}
|
|
463
|
-
|
|
464
|
-
/**
|
|
465
|
-
* Source map configuration options
|
|
466
|
-
*/
|
|
467
|
-
export interface SourceMapConfig {
|
|
468
|
-
/** Enable source maps */
|
|
469
|
-
enabled: boolean;
|
|
470
|
-
/** Source map type: true for external, 'inline' for inline, 'hidden' for hidden */
|
|
471
|
-
type: SourcemapOption;
|
|
472
|
-
/** Include source content in source maps */
|
|
473
|
-
includeContent: boolean;
|
|
474
|
-
/** Source map URL prefix */
|
|
475
|
-
sourceRoot?: string;
|
|
476
|
-
}
|
|
477
|
-
|
|
478
|
-
/**
|
|
479
|
-
* Default source map configuration
|
|
480
|
-
*/
|
|
481
|
-
export const DEFAULT_SOURCEMAP_CONFIG: SourceMapConfig = {
|
|
482
|
-
enabled: true,
|
|
483
|
-
type: true,
|
|
484
|
-
includeContent: true,
|
|
485
|
-
};
|
|
486
|
-
|
|
487
|
-
/**
|
|
488
|
-
* Creates source map configuration based on environment and preset
|
|
489
|
-
*/
|
|
490
|
-
export function createSourceMapConfig(
|
|
491
|
-
preset: string,
|
|
492
|
-
isDev: boolean
|
|
493
|
-
): SourceMapConfig {
|
|
494
|
-
if (isDev) {
|
|
495
|
-
return {
|
|
496
|
-
enabled: true,
|
|
497
|
-
type: true,
|
|
498
|
-
includeContent: true,
|
|
499
|
-
};
|
|
500
|
-
}
|
|
501
|
-
|
|
502
|
-
if (
|
|
503
|
-
preset.includes("edge") ||
|
|
504
|
-
preset.includes("cloudflare") ||
|
|
505
|
-
preset.includes("lambda")
|
|
506
|
-
) {
|
|
507
|
-
return {
|
|
508
|
-
enabled: true,
|
|
509
|
-
type: "hidden",
|
|
510
|
-
includeContent: false,
|
|
511
|
-
};
|
|
512
|
-
}
|
|
513
|
-
|
|
514
|
-
return {
|
|
515
|
-
enabled: true,
|
|
516
|
-
type: true,
|
|
517
|
-
includeContent: true,
|
|
518
|
-
};
|
|
519
|
-
}
|
|
520
|
-
|
|
521
|
-
/**
|
|
522
|
-
* Gets the Vite sourcemap option from SourceMapConfig
|
|
523
|
-
*/
|
|
524
|
-
export function getViteSourceMapOption(
|
|
525
|
-
config: SourceMapConfig
|
|
526
|
-
): SourcemapOption {
|
|
527
|
-
if (!config.enabled) {
|
|
528
|
-
return false;
|
|
529
|
-
}
|
|
530
|
-
return config.type;
|
|
531
|
-
}
|
|
532
|
-
|
|
533
|
-
/**
|
|
534
|
-
* Creates a Vite plugin for source map handling
|
|
535
|
-
*/
|
|
536
|
-
export function createSourceMapPlugin(config: SourceMapConfig): Plugin {
|
|
537
|
-
return {
|
|
538
|
-
name: "avalon:sourcemap",
|
|
539
|
-
enforce: "post",
|
|
540
|
-
|
|
541
|
-
config(_viteConfig: UserConfig, { command }: { command: string }) {
|
|
542
|
-
if (command !== "build") {
|
|
543
|
-
return;
|
|
544
|
-
}
|
|
545
|
-
|
|
546
|
-
return {
|
|
547
|
-
build: {
|
|
548
|
-
sourcemap: getViteSourceMapOption(config),
|
|
549
|
-
},
|
|
550
|
-
...(config.sourceRoot && {
|
|
551
|
-
rolldownOptions: {
|
|
552
|
-
output: {
|
|
553
|
-
sourcemapPathTransform: (relativeSourcePath: string) => {
|
|
554
|
-
return `${config.sourceRoot}/${relativeSourcePath}`;
|
|
555
|
-
},
|
|
556
|
-
},
|
|
557
|
-
},
|
|
558
|
-
}),
|
|
559
|
-
};
|
|
560
|
-
},
|
|
561
|
-
|
|
562
|
-
generateBundle(_options: unknown, bundle: Record<string, unknown>) {
|
|
563
|
-
if (!config.enabled) {
|
|
564
|
-
return;
|
|
565
|
-
}
|
|
566
|
-
|
|
567
|
-
const sourceMapCount = Object.keys(bundle).filter(
|
|
568
|
-
(key) => key.endsWith(".map")
|
|
569
|
-
).length;
|
|
570
|
-
|
|
571
|
-
if (sourceMapCount > 0 && globalThis.__avalonConfig?.verbose) {
|
|
572
|
-
console.log(`📍 Generated ${sourceMapCount} source map(s)`);
|
|
573
|
-
}
|
|
574
|
-
},
|
|
575
|
-
};
|
|
1
|
+
/**
|
|
2
|
+
* Nitro Build Configuration Module for Avalon
|
|
3
|
+
*
|
|
4
|
+
* This module provides build configuration utilities for integrating Nitro
|
|
5
|
+
* with Avalon's Vite-based build pipeline. It handles:
|
|
6
|
+
* - Vite client build configuration
|
|
7
|
+
* - Nitro server build configuration
|
|
8
|
+
* - Preset-specific output configuration
|
|
9
|
+
* - Source map generation
|
|
10
|
+
*
|
|
11
|
+
* @module nitro/build-config
|
|
12
|
+
*/
|
|
13
|
+
|
|
14
|
+
import type { Plugin, UserConfig, BuildOptions } from "vite";
|
|
15
|
+
import type { ResolvedAvalonConfig } from "../vite-plugin/types.ts";
|
|
16
|
+
import type { AvalonNitroConfig } from "./config.ts";
|
|
17
|
+
import { DEFAULT_NITRO_CONFIG, VALID_V3_PRESETS } from "./config.ts";
|
|
18
|
+
|
|
19
|
+
/**
|
|
20
|
+
* Build mode for the Avalon application
|
|
21
|
+
*/
|
|
22
|
+
export type BuildMode = "client" | "server" | "both";
|
|
23
|
+
|
|
24
|
+
/**
|
|
25
|
+
* Sourcemap option type
|
|
26
|
+
*/
|
|
27
|
+
export type SourcemapOption = boolean | "inline" | "hidden";
|
|
28
|
+
|
|
29
|
+
/**
|
|
30
|
+
* Minify option type
|
|
31
|
+
*/
|
|
32
|
+
export type MinifyOption = boolean | "oxc" | "esbuild" | "terser";
|
|
33
|
+
|
|
34
|
+
/**
|
|
35
|
+
* Preset-specific build output configuration
|
|
36
|
+
*/
|
|
37
|
+
export interface PresetOutputConfig {
|
|
38
|
+
/** Output directory for the preset */
|
|
39
|
+
outputDir: string;
|
|
40
|
+
/** Server entry file name */
|
|
41
|
+
serverEntry: string;
|
|
42
|
+
/** Whether the preset supports streaming */
|
|
43
|
+
supportsStreaming: boolean;
|
|
44
|
+
/** Whether the preset requires bundled dependencies */
|
|
45
|
+
bundleDependencies: boolean;
|
|
46
|
+
/** Additional files to include in output */
|
|
47
|
+
additionalFiles?: string[];
|
|
48
|
+
/** Environment-specific configuration */
|
|
49
|
+
env?: Record<string, string>;
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
/**
|
|
53
|
+
* Avalon build configuration options
|
|
54
|
+
*/
|
|
55
|
+
export interface AvalonBuildConfig {
|
|
56
|
+
/** Build mode: client, server, or both */
|
|
57
|
+
mode: BuildMode;
|
|
58
|
+
/** Output directory for client assets */
|
|
59
|
+
clientOutDir: string;
|
|
60
|
+
/** Output directory for server bundle */
|
|
61
|
+
serverOutDir: string;
|
|
62
|
+
/** Enable source maps */
|
|
63
|
+
sourcemap: SourcemapOption;
|
|
64
|
+
/** Minify output */
|
|
65
|
+
minify: MinifyOption;
|
|
66
|
+
/** Target environment */
|
|
67
|
+
target: string | string[];
|
|
68
|
+
/** Enable SSR build */
|
|
69
|
+
ssr: boolean;
|
|
70
|
+
/** Nitro preset for deployment */
|
|
71
|
+
preset: string;
|
|
72
|
+
/** Enable verbose logging */
|
|
73
|
+
verbose: boolean;
|
|
74
|
+
/** Nitro v3: Compatibility date for preset features (YYYY-MM-DD) */
|
|
75
|
+
compatibilityDate?: string;
|
|
76
|
+
/** Nitro v3: Dependencies to trace instead of bundle */
|
|
77
|
+
traceDeps?: string[];
|
|
78
|
+
/** Nitro v3: Rolldown-specific configuration */
|
|
79
|
+
rolldownConfig?: Record<string, unknown>;
|
|
80
|
+
/** Nitro v3: Custom server entry point */
|
|
81
|
+
serverEntry?: string;
|
|
82
|
+
}
|
|
83
|
+
|
|
84
|
+
/**
|
|
85
|
+
* Default build configuration values
|
|
86
|
+
*/
|
|
87
|
+
export const DEFAULT_BUILD_CONFIG: AvalonBuildConfig = {
|
|
88
|
+
mode: "both",
|
|
89
|
+
clientOutDir: "dist/client",
|
|
90
|
+
serverOutDir: "dist/server",
|
|
91
|
+
sourcemap: true,
|
|
92
|
+
minify: "oxc",
|
|
93
|
+
target: "es2020",
|
|
94
|
+
ssr: true,
|
|
95
|
+
preset: "node_server",
|
|
96
|
+
verbose: false,
|
|
97
|
+
};
|
|
98
|
+
|
|
99
|
+
/**
|
|
100
|
+
* Preset-specific output configurations
|
|
101
|
+
*/
|
|
102
|
+
export const PRESET_OUTPUT_CONFIGS: Record<string, PresetOutputConfig> = {
|
|
103
|
+
"node_server": {
|
|
104
|
+
outputDir: ".output",
|
|
105
|
+
serverEntry: "server/index.mjs",
|
|
106
|
+
supportsStreaming: true,
|
|
107
|
+
bundleDependencies: false,
|
|
108
|
+
},
|
|
109
|
+
"node_middleware": {
|
|
110
|
+
outputDir: ".output",
|
|
111
|
+
serverEntry: "server/index.mjs",
|
|
112
|
+
supportsStreaming: true,
|
|
113
|
+
bundleDependencies: false,
|
|
114
|
+
},
|
|
115
|
+
"vercel": {
|
|
116
|
+
outputDir: ".vercel/output",
|
|
117
|
+
serverEntry: "functions/render.func/index.mjs",
|
|
118
|
+
supportsStreaming: true,
|
|
119
|
+
bundleDependencies: true,
|
|
120
|
+
additionalFiles: ["config.json"],
|
|
121
|
+
},
|
|
122
|
+
"cloudflare_module": {
|
|
123
|
+
outputDir: "dist",
|
|
124
|
+
serverEntry: "server/index.mjs",
|
|
125
|
+
supportsStreaming: false,
|
|
126
|
+
bundleDependencies: true,
|
|
127
|
+
},
|
|
128
|
+
"cloudflare_pages": {
|
|
129
|
+
outputDir: "dist",
|
|
130
|
+
serverEntry: "_worker.js",
|
|
131
|
+
supportsStreaming: false,
|
|
132
|
+
bundleDependencies: true,
|
|
133
|
+
additionalFiles: ["_routes.json"],
|
|
134
|
+
},
|
|
135
|
+
"deno_deploy": {
|
|
136
|
+
outputDir: ".output",
|
|
137
|
+
serverEntry: "server/index.ts",
|
|
138
|
+
supportsStreaming: true,
|
|
139
|
+
bundleDependencies: false,
|
|
140
|
+
},
|
|
141
|
+
"deno_server": {
|
|
142
|
+
outputDir: ".output",
|
|
143
|
+
serverEntry: "server/index.ts",
|
|
144
|
+
supportsStreaming: true,
|
|
145
|
+
bundleDependencies: false,
|
|
146
|
+
},
|
|
147
|
+
"netlify_functions": {
|
|
148
|
+
outputDir: ".netlify",
|
|
149
|
+
serverEntry: "functions-internal/render.mjs",
|
|
150
|
+
supportsStreaming: true,
|
|
151
|
+
bundleDependencies: true,
|
|
152
|
+
},
|
|
153
|
+
"netlify_edge": {
|
|
154
|
+
outputDir: ".netlify/edge-functions",
|
|
155
|
+
serverEntry: "render.js",
|
|
156
|
+
supportsStreaming: true,
|
|
157
|
+
bundleDependencies: true,
|
|
158
|
+
},
|
|
159
|
+
"aws_lambda": {
|
|
160
|
+
outputDir: ".output",
|
|
161
|
+
serverEntry: "server/index.mjs",
|
|
162
|
+
supportsStreaming: false,
|
|
163
|
+
bundleDependencies: true,
|
|
164
|
+
},
|
|
165
|
+
"azure_swa": {
|
|
166
|
+
outputDir: ".output",
|
|
167
|
+
serverEntry: "server/index.mjs",
|
|
168
|
+
supportsStreaming: false,
|
|
169
|
+
bundleDependencies: true,
|
|
170
|
+
},
|
|
171
|
+
"firebase_functions": {
|
|
172
|
+
outputDir: ".output",
|
|
173
|
+
serverEntry: "server/index.mjs",
|
|
174
|
+
supportsStreaming: true,
|
|
175
|
+
bundleDependencies: true,
|
|
176
|
+
},
|
|
177
|
+
"render_com": {
|
|
178
|
+
outputDir: ".output",
|
|
179
|
+
serverEntry: "server/index.mjs",
|
|
180
|
+
supportsStreaming: true,
|
|
181
|
+
bundleDependencies: false,
|
|
182
|
+
},
|
|
183
|
+
"static": {
|
|
184
|
+
outputDir: "dist",
|
|
185
|
+
serverEntry: "",
|
|
186
|
+
supportsStreaming: false,
|
|
187
|
+
bundleDependencies: false,
|
|
188
|
+
},
|
|
189
|
+
"browser": {
|
|
190
|
+
outputDir: "dist",
|
|
191
|
+
serverEntry: "",
|
|
192
|
+
supportsStreaming: false,
|
|
193
|
+
bundleDependencies: true,
|
|
194
|
+
},
|
|
195
|
+
};
|
|
196
|
+
|
|
197
|
+
/**
|
|
198
|
+
* Creates the Vite client build configuration
|
|
199
|
+
*
|
|
200
|
+
* @param avalonConfig - Resolved Avalon configuration
|
|
201
|
+
* @param buildConfig - Build configuration options
|
|
202
|
+
* @returns Vite build options for client build
|
|
203
|
+
*/
|
|
204
|
+
export function createClientBuildConfig(
|
|
205
|
+
avalonConfig: ResolvedAvalonConfig,
|
|
206
|
+
buildConfig: Partial<AvalonBuildConfig> = {}
|
|
207
|
+
): BuildOptions {
|
|
208
|
+
const config = { ...DEFAULT_BUILD_CONFIG, ...buildConfig };
|
|
209
|
+
|
|
210
|
+
return {
|
|
211
|
+
outDir: config.clientOutDir,
|
|
212
|
+
emptyOutDir: true,
|
|
213
|
+
sourcemap: config.sourcemap,
|
|
214
|
+
minify: config.minify,
|
|
215
|
+
target: config.target,
|
|
216
|
+
rolldownOptions: {
|
|
217
|
+
output: {
|
|
218
|
+
entryFileNames: "[name].[hash].js",
|
|
219
|
+
chunkFileNames: "chunks/[name].[hash].js",
|
|
220
|
+
assetFileNames: "assets/[name].[hash].[ext]",
|
|
221
|
+
codeSplitting: {
|
|
222
|
+
groups: [
|
|
223
|
+
{ name: "vendor-react", test: /node_modules\/(react|react-dom)/ },
|
|
224
|
+
{ name: "vendor-vue", test: /node_modules\/(vue|@vue)/ },
|
|
225
|
+
{ name: "vendor-svelte", test: /node_modules\/svelte/ },
|
|
226
|
+
{ name: "vendor-preact", test: /node_modules\/preact/ },
|
|
227
|
+
{ name: "vendor-solid", test: /node_modules\/solid-js/ },
|
|
228
|
+
{ name: "vendor-lit", test: /node_modules\/(lit|@lit)/ },
|
|
229
|
+
{ name: "vendor", test: /node_modules/ },
|
|
230
|
+
{ name: "islands", test: /\/islands\// },
|
|
231
|
+
],
|
|
232
|
+
},
|
|
233
|
+
},
|
|
234
|
+
},
|
|
235
|
+
reportCompressedSize: !avalonConfig.isDev,
|
|
236
|
+
cssCodeSplit: true,
|
|
237
|
+
};
|
|
238
|
+
}
|
|
239
|
+
|
|
240
|
+
/**
|
|
241
|
+
* Creates the Vite server build configuration for Nitro
|
|
242
|
+
*
|
|
243
|
+
* @param avalonConfig - Resolved Avalon configuration
|
|
244
|
+
* @param nitroConfig - Nitro configuration
|
|
245
|
+
* @param buildConfig - Build configuration options
|
|
246
|
+
* @returns Vite build options for server build
|
|
247
|
+
*/
|
|
248
|
+
export function createServerBuildConfig(
|
|
249
|
+
avalonConfig: ResolvedAvalonConfig,
|
|
250
|
+
nitroConfig: AvalonNitroConfig,
|
|
251
|
+
buildConfig: Partial<AvalonBuildConfig> = {}
|
|
252
|
+
): BuildOptions {
|
|
253
|
+
const config = { ...DEFAULT_BUILD_CONFIG, ...buildConfig };
|
|
254
|
+
const preset = nitroConfig.preset ?? DEFAULT_NITRO_CONFIG.preset;
|
|
255
|
+
const presetConfig = PRESET_OUTPUT_CONFIGS[preset] ?? PRESET_OUTPUT_CONFIGS["node_server"];
|
|
256
|
+
|
|
257
|
+
return {
|
|
258
|
+
outDir: config.serverOutDir,
|
|
259
|
+
emptyOutDir: true,
|
|
260
|
+
sourcemap: config.sourcemap,
|
|
261
|
+
minify: config.minify,
|
|
262
|
+
target: config.target,
|
|
263
|
+
ssr: true,
|
|
264
|
+
rolldownOptions: {
|
|
265
|
+
input: {
|
|
266
|
+
index: "./server/index.ts",
|
|
267
|
+
},
|
|
268
|
+
output: {
|
|
269
|
+
format: "esm",
|
|
270
|
+
entryFileNames: "[name].mjs",
|
|
271
|
+
chunkFileNames: "chunks/[name].[hash].mjs",
|
|
272
|
+
preserveModules: !presetConfig.bundleDependencies,
|
|
273
|
+
},
|
|
274
|
+
external: presetConfig.bundleDependencies
|
|
275
|
+
? []
|
|
276
|
+
: getServerExternals(preset),
|
|
277
|
+
},
|
|
278
|
+
};
|
|
279
|
+
}
|
|
280
|
+
|
|
281
|
+
/**
|
|
282
|
+
* Gets external dependencies for server build based on preset
|
|
283
|
+
*
|
|
284
|
+
* @param preset - Nitro preset name
|
|
285
|
+
* @returns Array of external dependency patterns
|
|
286
|
+
*/
|
|
287
|
+
function getServerExternals(preset: string): (string | RegExp)[] {
|
|
288
|
+
const baseExternals: (string | RegExp)[] = [
|
|
289
|
+
/^node:/,
|
|
290
|
+
/^deno:/,
|
|
291
|
+
];
|
|
292
|
+
|
|
293
|
+
switch (preset) {
|
|
294
|
+
case "deno_deploy":
|
|
295
|
+
case "deno_server":
|
|
296
|
+
return [...baseExternals];
|
|
297
|
+
|
|
298
|
+
case "node_server":
|
|
299
|
+
return [
|
|
300
|
+
...baseExternals,
|
|
301
|
+
/^[a-z@]/i,
|
|
302
|
+
];
|
|
303
|
+
|
|
304
|
+
default:
|
|
305
|
+
return baseExternals;
|
|
306
|
+
}
|
|
307
|
+
}
|
|
308
|
+
|
|
309
|
+
/**
|
|
310
|
+
* Gets the preset output configuration
|
|
311
|
+
*
|
|
312
|
+
* @param preset - Nitro preset name
|
|
313
|
+
* @returns Preset output configuration
|
|
314
|
+
*/
|
|
315
|
+
export function getPresetOutputConfig(preset: string): PresetOutputConfig {
|
|
316
|
+
return PRESET_OUTPUT_CONFIGS[preset] ?? PRESET_OUTPUT_CONFIGS["node_server"];
|
|
317
|
+
}
|
|
318
|
+
|
|
319
|
+
/**
|
|
320
|
+
* Checks if a preset supports streaming SSR
|
|
321
|
+
*
|
|
322
|
+
* @param preset - Nitro preset name
|
|
323
|
+
* @returns True if the preset supports streaming
|
|
324
|
+
*/
|
|
325
|
+
export function presetSupportsStreaming(preset: string): boolean {
|
|
326
|
+
const config = getPresetOutputConfig(preset);
|
|
327
|
+
return config.supportsStreaming;
|
|
328
|
+
}
|
|
329
|
+
|
|
330
|
+
/**
|
|
331
|
+
* Creates the combined build configuration for both client and server
|
|
332
|
+
*
|
|
333
|
+
* @param avalonConfig - Resolved Avalon configuration
|
|
334
|
+
* @param nitroConfig - Nitro configuration
|
|
335
|
+
* @param buildConfig - Build configuration options
|
|
336
|
+
* @returns Combined Vite user config
|
|
337
|
+
*/
|
|
338
|
+
export function createCombinedBuildConfig(
|
|
339
|
+
avalonConfig: ResolvedAvalonConfig,
|
|
340
|
+
nitroConfig: AvalonNitroConfig,
|
|
341
|
+
buildConfig: Partial<AvalonBuildConfig> = {}
|
|
342
|
+
): Partial<UserConfig> {
|
|
343
|
+
const preset = nitroConfig.preset ?? DEFAULT_NITRO_CONFIG.preset;
|
|
344
|
+
const presetConfig = getPresetOutputConfig(preset);
|
|
345
|
+
|
|
346
|
+
return {
|
|
347
|
+
build: {
|
|
348
|
+
...createClientBuildConfig(avalonConfig, buildConfig),
|
|
349
|
+
outDir: presetConfig.outputDir,
|
|
350
|
+
},
|
|
351
|
+
define: {
|
|
352
|
+
__DEV__: false,
|
|
353
|
+
__PROD__: true,
|
|
354
|
+
"process.env.NODE_ENV": JSON.stringify("production"),
|
|
355
|
+
...createPresetDefines(preset, presetConfig),
|
|
356
|
+
},
|
|
357
|
+
};
|
|
358
|
+
}
|
|
359
|
+
|
|
360
|
+
/**
|
|
361
|
+
* Creates preset-specific define constants
|
|
362
|
+
*/
|
|
363
|
+
function createPresetDefines(
|
|
364
|
+
preset: string,
|
|
365
|
+
presetConfig: PresetOutputConfig
|
|
366
|
+
): Record<string, string> {
|
|
367
|
+
const defines: Record<string, string> = {
|
|
368
|
+
__NITRO_PRESET__: JSON.stringify(preset),
|
|
369
|
+
__SUPPORTS_STREAMING__: JSON.stringify(presetConfig.supportsStreaming),
|
|
370
|
+
};
|
|
371
|
+
|
|
372
|
+
if (presetConfig.env) {
|
|
373
|
+
for (const [key, value] of Object.entries(presetConfig.env)) {
|
|
374
|
+
defines[`process.env.${key}`] = JSON.stringify(value);
|
|
375
|
+
}
|
|
376
|
+
}
|
|
377
|
+
|
|
378
|
+
return defines;
|
|
379
|
+
}
|
|
380
|
+
|
|
381
|
+
/**
|
|
382
|
+
* Creates a Vite plugin for Nitro build integration
|
|
383
|
+
*/
|
|
384
|
+
export function createNitroBuildPlugin(
|
|
385
|
+
avalonConfig: ResolvedAvalonConfig,
|
|
386
|
+
nitroConfig: AvalonNitroConfig
|
|
387
|
+
): Plugin {
|
|
388
|
+
const preset = nitroConfig.preset ?? DEFAULT_NITRO_CONFIG.preset;
|
|
389
|
+
const presetConfig = getPresetOutputConfig(preset);
|
|
390
|
+
|
|
391
|
+
return {
|
|
392
|
+
name: "avalon:nitro-build",
|
|
393
|
+
enforce: "post",
|
|
394
|
+
|
|
395
|
+
config(_config: UserConfig, { command }: { command: string }) {
|
|
396
|
+
if (command !== "build") {
|
|
397
|
+
return;
|
|
398
|
+
}
|
|
399
|
+
|
|
400
|
+
return createCombinedBuildConfig(avalonConfig, nitroConfig, {
|
|
401
|
+
sourcemap: true,
|
|
402
|
+
verbose: avalonConfig.verbose,
|
|
403
|
+
});
|
|
404
|
+
},
|
|
405
|
+
|
|
406
|
+
buildStart() {
|
|
407
|
+
if (avalonConfig.verbose) {
|
|
408
|
+
console.log("🚀 Avalon Nitro build starting...");
|
|
409
|
+
console.log(` Preset: ${preset}`);
|
|
410
|
+
console.log(` Output: ${presetConfig.outputDir}`);
|
|
411
|
+
console.log(` Streaming: ${presetConfig.supportsStreaming}`);
|
|
412
|
+
}
|
|
413
|
+
},
|
|
414
|
+
|
|
415
|
+
writeBundle() {
|
|
416
|
+
if (avalonConfig.verbose) {
|
|
417
|
+
console.log("✅ Avalon client build complete");
|
|
418
|
+
}
|
|
419
|
+
},
|
|
420
|
+
|
|
421
|
+
closeBundle() {
|
|
422
|
+
if (avalonConfig.verbose) {
|
|
423
|
+
console.log("📦 Avalon build finished");
|
|
424
|
+
console.log(` Output directory: ${presetConfig.outputDir}`);
|
|
425
|
+
}
|
|
426
|
+
},
|
|
427
|
+
};
|
|
428
|
+
}
|
|
429
|
+
|
|
430
|
+
/**
|
|
431
|
+
* Validates build configuration
|
|
432
|
+
*/
|
|
433
|
+
export function validateBuildConfig(
|
|
434
|
+
config: Partial<AvalonBuildConfig>
|
|
435
|
+
): { valid: boolean; errors: string[] } {
|
|
436
|
+
const errors: string[] = [];
|
|
437
|
+
|
|
438
|
+
if (config.preset) {
|
|
439
|
+
if (!VALID_V3_PRESETS.includes(config.preset)) {
|
|
440
|
+
errors.push(
|
|
441
|
+
`Unknown preset: ${config.preset}. Valid presets: ${VALID_V3_PRESETS.join(", ")}`
|
|
442
|
+
);
|
|
443
|
+
}
|
|
444
|
+
}
|
|
445
|
+
|
|
446
|
+
if (config.mode && !["client", "server", "both"].includes(config.mode)) {
|
|
447
|
+
errors.push(`Invalid build mode: ${config.mode}. Must be 'client', 'server', or 'both'`);
|
|
448
|
+
}
|
|
449
|
+
|
|
450
|
+
if (
|
|
451
|
+
config.sourcemap !== undefined &&
|
|
452
|
+
typeof config.sourcemap !== "boolean" &&
|
|
453
|
+
!["inline", "hidden"].includes(config.sourcemap as string)
|
|
454
|
+
) {
|
|
455
|
+
errors.push(`Invalid sourcemap option: ${config.sourcemap}`);
|
|
456
|
+
}
|
|
457
|
+
|
|
458
|
+
return {
|
|
459
|
+
valid: errors.length === 0,
|
|
460
|
+
errors,
|
|
461
|
+
};
|
|
462
|
+
}
|
|
463
|
+
|
|
464
|
+
/**
|
|
465
|
+
* Source map configuration options
|
|
466
|
+
*/
|
|
467
|
+
export interface SourceMapConfig {
|
|
468
|
+
/** Enable source maps */
|
|
469
|
+
enabled: boolean;
|
|
470
|
+
/** Source map type: true for external, 'inline' for inline, 'hidden' for hidden */
|
|
471
|
+
type: SourcemapOption;
|
|
472
|
+
/** Include source content in source maps */
|
|
473
|
+
includeContent: boolean;
|
|
474
|
+
/** Source map URL prefix */
|
|
475
|
+
sourceRoot?: string;
|
|
476
|
+
}
|
|
477
|
+
|
|
478
|
+
/**
|
|
479
|
+
* Default source map configuration
|
|
480
|
+
*/
|
|
481
|
+
export const DEFAULT_SOURCEMAP_CONFIG: SourceMapConfig = {
|
|
482
|
+
enabled: true,
|
|
483
|
+
type: true,
|
|
484
|
+
includeContent: true,
|
|
485
|
+
};
|
|
486
|
+
|
|
487
|
+
/**
|
|
488
|
+
* Creates source map configuration based on environment and preset
|
|
489
|
+
*/
|
|
490
|
+
export function createSourceMapConfig(
|
|
491
|
+
preset: string,
|
|
492
|
+
isDev: boolean
|
|
493
|
+
): SourceMapConfig {
|
|
494
|
+
if (isDev) {
|
|
495
|
+
return {
|
|
496
|
+
enabled: true,
|
|
497
|
+
type: true,
|
|
498
|
+
includeContent: true,
|
|
499
|
+
};
|
|
500
|
+
}
|
|
501
|
+
|
|
502
|
+
if (
|
|
503
|
+
preset.includes("edge") ||
|
|
504
|
+
preset.includes("cloudflare") ||
|
|
505
|
+
preset.includes("lambda")
|
|
506
|
+
) {
|
|
507
|
+
return {
|
|
508
|
+
enabled: true,
|
|
509
|
+
type: "hidden",
|
|
510
|
+
includeContent: false,
|
|
511
|
+
};
|
|
512
|
+
}
|
|
513
|
+
|
|
514
|
+
return {
|
|
515
|
+
enabled: true,
|
|
516
|
+
type: true,
|
|
517
|
+
includeContent: true,
|
|
518
|
+
};
|
|
519
|
+
}
|
|
520
|
+
|
|
521
|
+
/**
|
|
522
|
+
* Gets the Vite sourcemap option from SourceMapConfig
|
|
523
|
+
*/
|
|
524
|
+
export function getViteSourceMapOption(
|
|
525
|
+
config: SourceMapConfig
|
|
526
|
+
): SourcemapOption {
|
|
527
|
+
if (!config.enabled) {
|
|
528
|
+
return false;
|
|
529
|
+
}
|
|
530
|
+
return config.type;
|
|
531
|
+
}
|
|
532
|
+
|
|
533
|
+
/**
|
|
534
|
+
* Creates a Vite plugin for source map handling
|
|
535
|
+
*/
|
|
536
|
+
export function createSourceMapPlugin(config: SourceMapConfig): Plugin {
|
|
537
|
+
return {
|
|
538
|
+
name: "avalon:sourcemap",
|
|
539
|
+
enforce: "post",
|
|
540
|
+
|
|
541
|
+
config(_viteConfig: UserConfig, { command }: { command: string }) {
|
|
542
|
+
if (command !== "build") {
|
|
543
|
+
return;
|
|
544
|
+
}
|
|
545
|
+
|
|
546
|
+
return {
|
|
547
|
+
build: {
|
|
548
|
+
sourcemap: getViteSourceMapOption(config),
|
|
549
|
+
},
|
|
550
|
+
...(config.sourceRoot && {
|
|
551
|
+
rolldownOptions: {
|
|
552
|
+
output: {
|
|
553
|
+
sourcemapPathTransform: (relativeSourcePath: string) => {
|
|
554
|
+
return `${config.sourceRoot}/${relativeSourcePath}`;
|
|
555
|
+
},
|
|
556
|
+
},
|
|
557
|
+
},
|
|
558
|
+
}),
|
|
559
|
+
};
|
|
560
|
+
},
|
|
561
|
+
|
|
562
|
+
generateBundle(_options: unknown, bundle: Record<string, unknown>) {
|
|
563
|
+
if (!config.enabled) {
|
|
564
|
+
return;
|
|
565
|
+
}
|
|
566
|
+
|
|
567
|
+
const sourceMapCount = Object.keys(bundle).filter(
|
|
568
|
+
(key) => key.endsWith(".map")
|
|
569
|
+
).length;
|
|
570
|
+
|
|
571
|
+
if (sourceMapCount > 0 && globalThis.__avalonConfig?.verbose) {
|
|
572
|
+
console.log(`📍 Generated ${sourceMapCount} source map(s)`);
|
|
573
|
+
}
|
|
574
|
+
},
|
|
575
|
+
};
|
|
576
576
|
}
|