Files
Nitro-V3/vite.config.mjs
T
2026-06-15 02:20:54 +02:00

223 lines
9.9 KiB
JavaScript

import react from '@vitejs/plugin-react';
import { existsSync, readFileSync } from 'fs';
import { resolve } from 'path';
import { defineConfig } from 'vite';
import sirv from 'sirv';
const legacyRendererRoot = resolve(__dirname, '..', 'renderer');
const currentRendererRoot = resolve(__dirname, '..', 'Nitro_Render_V3');
const rendererRoot = existsSync(currentRendererRoot) ? currentRendererRoot : legacyRendererRoot;
// Game assets live outside the repo, in a sibling directory next to Nitro-V3.
// They are NOT placed under public/ on purpose: with ~177k files a symlink
// under public/ makes chokidar try to install a watcher on each one and the
// dev server takes minutes to start on Windows. Serving them with a
// dedicated sirv middleware (below) bypasses chokidar entirely.
const nitroFilesRoot = resolve(__dirname, '..', 'Nitro-Files');
const nitroAssetsRoot = resolve(nitroFilesRoot, 'nitro-assets');
const swfRoot = resolve(nitroFilesRoot, 'swf');
const nitroAssetsServer = () => ({
name: 'nitro-assets-serve',
configureServer(server)
{
if(existsSync(nitroAssetsRoot))
{
server.middlewares.use('/nitro-assets', sirv(nitroAssetsRoot, { dev: true, etag: true, maxAge: 0 }));
}
else
{
server.config.logger.warn(`[nitro-assets-serve] ${ nitroAssetsRoot } not found — /nitro-assets/* requests will 404.`);
}
if(existsSync(swfRoot))
{
server.middlewares.use('/swf', sirv(swfRoot, { dev: true, etag: true, maxAge: 0 }));
}
else
{
server.config.logger.warn(`[nitro-assets-serve] ${ swfRoot } not found — /swf/* requests will 404.`);
}
},
configurePreviewServer(server)
{
if(existsSync(nitroAssetsRoot))
{
server.middlewares.use('/nitro-assets', sirv(nitroAssetsRoot, { dev: false, etag: true }));
}
if(existsSync(swfRoot))
{
server.middlewares.use('/swf', sirv(swfRoot, { dev: false, etag: true }));
}
}
});
if(!existsSync(rendererRoot))
{
// Fail fast with a useful message instead of waiting for Rolldown to
// report "Failed to resolve import @nitrots/nitro-renderer" deep
// inside the bundle pass.
throw new Error(
'\n Nitro renderer SDK not found.\n\n' +
' vite.config.mjs expects one of these directories to exist as a sibling of this repo:\n' +
` - ${ currentRendererRoot } (preferred)\n` +
` - ${ legacyRendererRoot } (legacy)\n\n` +
' Clone the Nitro_Render_V3 repo next to Nitro-V3 and rerun:\n' +
' git clone <renderer-repo> ../Nitro_Render_V3\n' +
' cd ../Nitro_Render_V3 && yarn install\n\n' +
' (See CLAUDE.md "Commands" section for the full setup walkthrough.)\n'
);
}
const ReactCompilerConfig = {
target: '19'
};
const resolveJsonMode = () =>
{
const envOverride = process.env.NITRO_JSON_MODE;
if(envOverride === 'legacy' || envOverride === 'json5' || envOverride === 'auto') return envOverride;
const configFile = resolve(__dirname, '.nitro-build.json');
if(existsSync(configFile))
{
try
{
const parsed = JSON.parse(readFileSync(configFile, 'utf8'));
if(parsed?.jsonMode === 'legacy' || parsed?.jsonMode === 'json5' || parsed?.jsonMode === 'auto') return parsed.jsonMode;
}
catch {}
}
return 'auto';
};
const nitroJsonMode = resolveJsonMode();
const nitroSingleBundle = process.env.NITRO_SINGLE_BUNDLE === '1';
process.stdout.write(`[vite] __NITRO_JSON_MODE__ = ${ nitroJsonMode }\n`);
process.stdout.write(`[vite] NITRO_SINGLE_BUNDLE = ${ nitroSingleBundle ? '1' : '0' }\n`);
export default defineConfig({
base: process.env.VITE_BASE || './',
plugins: [
react({
babel: {
plugins: [
[ 'babel-plugin-react-compiler', ReactCompilerConfig ]
]
}
}),
nitroAssetsServer()
],
define: {
__NITRO_JSON_MODE__: JSON.stringify(nitroJsonMode)
},
server: {
fs: {
allow: [
resolve(__dirname),
rendererRoot,
]
},
proxy: {
'/api': {
target: process.env.AUTH_PROXY_TARGET || 'http://localhost:2096',
changeOrigin: true,
}
}
},
resolve: {
tsconfigPaths: true,
alias: {
'@': resolve(__dirname, 'src'),
'~': resolve(__dirname, 'node_modules'),
// Force the umbrella to the source index.ts. Without this,
// node-module resolution (via the symlink at
// node_modules/@nitrots/nitro-renderer -> ../Nitro_Render_V3)
// can land on the stale `dist/index.js` when one exists in
// the renderer working tree — leaving the bundle with
// pre-snapshot-pattern stubs and producing runtime errors
// like "TypeError: (intermediate value)() is undefined"
// when newer code calls getUserDataSnapshot() / .subscribe()
// / NitroEventType.SESSION_DATA_UPDATED etc.
'@nitrots/nitro-renderer': resolve(rendererRoot, 'index.ts'),
'@nitrots/api': resolve(rendererRoot, 'packages/api/src/index.ts'),
'@nitrots/assets': resolve(rendererRoot, 'packages/assets/src/index.ts'),
'@nitrots/avatar': resolve(rendererRoot, 'packages/avatar/src/index.ts'),
'@nitrots/camera': resolve(rendererRoot, 'packages/camera/src/index.ts'),
'@nitrots/communication': resolve(rendererRoot, 'packages/communication/src/index.ts'),
'@nitrots/configuration': resolve(rendererRoot, 'packages/configuration/src/index.ts'),
'@nitrots/events': resolve(rendererRoot, 'packages/events/src/index.ts'),
'@nitrots/localization': resolve(rendererRoot, 'packages/localization/src/index.ts'),
'@nitrots/room': resolve(rendererRoot, 'packages/room/src/index.ts'),
'@nitrots/session': resolve(rendererRoot, 'packages/session/src/index.ts'),
'@nitrots/sound': resolve(rendererRoot, 'packages/sound/src/index.ts'),
'@nitrots/utils/src': resolve(rendererRoot, 'packages/utils/src'),
'@nitrots/utils': resolve(rendererRoot, 'packages/utils/src/index.ts'),
'pixi.js': resolve(rendererRoot, 'node_modules/pixi.js'),
'pixi-filters': resolve(rendererRoot, 'node_modules/pixi-filters'),
'howler': resolve(rendererRoot, 'node_modules/howler'),
}
},
build: {
assetsInlineLimit: 102400,
chunkSizeWarningLimit: 200000,
manifest: true,
rollupOptions: {
checks: {
pluginTimings: false
},
output: nitroSingleBundle ? {
assetFileNames: 'src/assets/[name]-[hash].[ext]',
entryFileNames: 'assets/app.js',
inlineDynamicImports: true
} : {
assetFileNames: 'src/assets/[name]-[hash].[ext]',
// Granular chunking: split the monolithic vendor / nitro-renderer
// bundles into smaller chunks so the browser can fetch them in
// parallel and CF can cache each independently. Splits chosen
// by size impact (pixi ~600KB, react ~150KB, framer-motion ~100KB,
// jodit ~250KB lazy-loaded only by admin news, etc.).
manualChunks: id =>
{
// Vendor checks first — pixi.js/howler are aliased to
// ../Nitro_Render_V3/node_modules so they match
// `Nitro_Render_V3` too. Without this priority, they end
// up bundled into nitro-renderer instead of getting their
// own chunks (pixi alone is ~600KB). Use `/pixi.js/` to
// avoid matching path fragments like `assets/pixi.js/`.
const norm = id.replace(/\\/g, '/');
if(norm.includes('pixi.js') || norm.includes('pixi-filters')) return 'vendor-pixi';
if(norm.includes('howler')) return 'vendor-audio';
if(norm.includes('@emoji-mart')) return 'vendor-emoji';
if(norm.includes('jodit') || norm.includes('@react-page')) return 'vendor-editor';
if(id.includes('Nitro_Render_V3') || id.includes(`${ rendererRoot }`))
{
// Heaviest renderer packages get their own chunks so
// pages that don't touch them (login flow, very early
// boot) don't have to pay for them upfront.
if(id.includes('/packages/avatar/')) return 'nitro-renderer-avatar';
if(id.includes('/packages/communication/')) return 'nitro-renderer-comm';
if(id.includes('/packages/room/')) return 'nitro-renderer-room';
if(id.includes('/packages/assets/')) return 'nitro-renderer-assets';
return 'nitro-renderer';
}
if(id.includes('node_modules'))
{
if(id.includes('@nitrots/nitro-renderer') || id.includes('renderer3')) return 'nitro-renderer';
if(id.match(/\/react(-dom)?\/|\/scheduler\//) || id.includes('react-error-boundary')) return 'vendor-react';
if(id.includes('framer-motion')) return 'vendor-motion';
if(id.includes('@tanstack')) return 'vendor-query';
if(id.includes('zustand') || id.includes('use-between')) return 'vendor-state';
if(id.includes('react-icons')) return 'vendor-icons';
if(id.includes('json5')) return 'vendor-json5';
return 'vendor';
}
}
}
}
}
});