Files
lobehub/vite.config.ts
Innei 5f2f49a26e feat(vite): add env restart keys plugin for selective .env restart (#12561)
Add a Vite plugin that prevents server restart on every .env file change,
only restarting when whitelisted env keys actually change their values.
2026-03-01 20:58:05 +08:00

116 lines
3.6 KiB
TypeScript

import { resolve } from 'node:path';
import type { PluginOption, ViteDevServer } from 'vite';
import { defineConfig, loadEnv } from 'vite';
import { VitePWA } from 'vite-plugin-pwa';
import { viteEnvRestartKeys } from './plugins/vite/envRestartKeys';
import {
sharedOptimizeDeps,
sharedRendererDefine,
sharedRendererPlugins,
sharedRollupOutput,
} from './plugins/vite/sharedRendererConfig';
const isMobile = process.env.MOBILE === 'true';
const mode = process.env.NODE_ENV === 'production' ? 'production' : 'development';
Object.assign(process.env, loadEnv(mode, process.cwd(), ''));
const isDev = process.env.NODE_ENV !== 'production';
const platform = isMobile ? 'mobile' : 'web';
export default defineConfig({
base: isDev ? '/' : process.env.VITE_CDN_BASE || '/spa/',
build: {
outDir: isMobile ? 'dist/mobile' : 'dist/desktop',
rollupOptions: {
input: resolve(__dirname, isMobile ? 'index.mobile.html' : 'index.html'),
output: sharedRollupOutput,
},
},
define: sharedRendererDefine({ isMobile, isElectron: false }),
optimizeDeps: sharedOptimizeDeps,
plugins: [
viteEnvRestartKeys(['APP_URL']),
...sharedRendererPlugins({ platform }),
isDev && {
name: 'lobe-dev-proxy-print',
configureServer(server: ViteDevServer) {
const ONLINE_HOST = 'https://app.lobehub.com';
server.httpServer?.once('listening', () => {
const address = server.httpServer?.address();
const port = typeof address === 'object' && address ? address.port : 9876;
const localHost = `http://localhost:${port}`;
const proxyUrl = `${ONLINE_HOST}/_dangerous_local_dev_proxy?debug-host=${encodeURIComponent(localHost)}`;
setTimeout(() => {
console.info();
console.info(` \x1B[1m\x1B[35mDebug Proxy:\x1B[0m \x1B[36m${proxyUrl}\x1B[0m`);
console.info();
}, 100);
});
},
},
VitePWA({
injectRegister: null,
manifest: false,
registerType: 'prompt',
workbox: {
globPatterns: ['**/*.{js,css,html,woff2}'],
maximumFileSizeToCacheInBytes: 10 * 1024 * 1024,
runtimeCaching: [
{
handler: 'StaleWhileRevalidate',
options: { cacheName: 'google-fonts-stylesheets' },
urlPattern: /^https:\/\/fonts\.googleapis\.com\/.*/i,
},
{
handler: 'CacheFirst',
options: {
cacheName: 'google-fonts-webfonts',
expiration: { maxAgeSeconds: 60 * 60 * 24 * 365, maxEntries: 30 },
},
urlPattern: /^https:\/\/fonts\.gstatic\.com\/.*/i,
},
{
handler: 'StaleWhileRevalidate',
options: {
cacheName: 'image-assets',
expiration: { maxAgeSeconds: 60 * 60 * 24 * 30, maxEntries: 100 },
},
urlPattern: /\.(?:png|jpg|jpeg|svg|gif|webp|ico|avif)$/i,
},
{
handler: 'NetworkFirst',
options: {
cacheName: 'api-cache',
expiration: { maxAgeSeconds: 60 * 5, maxEntries: 50 },
},
urlPattern: /\/(api|trpc)\/.*/i,
},
],
},
}),
].filter(Boolean) as PluginOption[],
server: {
cors: true,
port: 9876,
host: true,
proxy: {
'/api': 'http://localhost:3010',
'/oidc': 'http://localhost:3010',
'/trpc': 'http://localhost:3010',
'/webapi': 'http://localhost:3010',
},
warmup: {
clientFiles: [
platform === 'mobile' ? './src/spa/entry.mobile.tsx' : './src/spa/entry.web.tsx',
],
},
},
});