Files
012-kaopeilian/frontend/vite.config.ts.timestamp-1758830290264-a627566ebc111.mjs
111 998211c483 feat: 初始化考培练系统项目
- 从服务器拉取完整代码
- 按框架规范整理项目结构
- 配置 Drone CI 测试环境部署
- 包含后端(FastAPI)、前端(Vue3)、管理端

技术栈: Vue3 + TypeScript + FastAPI + MySQL
2026-01-24 19:33:28 +08:00

199 lines
18 KiB
JavaScript

var __require = /* @__PURE__ */ ((x) => typeof require !== "undefined" ? require : typeof Proxy !== "undefined" ? new Proxy(x, {
get: (a, b) => (typeof require !== "undefined" ? require : a)[b]
}) : x)(function(x) {
if (typeof require !== "undefined") return require.apply(this, arguments);
throw Error('Dynamic require of "' + x + '" is not supported');
});
// vite.config.ts
import { defineConfig, loadEnv } from "file:///app/node_modules/vite/dist/node/index.js";
import vue from "file:///app/node_modules/@vitejs/plugin-vue/dist/index.mjs";
import path from "path";
import { visualizer } from "file:///app/node_modules/rollup-plugin-visualizer/dist/plugin/index.js";
import { createHtmlPlugin } from "file:///app/node_modules/vite-plugin-html/dist/index.mjs";
var __vite_injected_original_dirname = "/app";
var vite_config_default = defineConfig(({ command, mode }) => {
const env = loadEnv(mode, process.cwd(), "");
const isProduction = mode === "production";
return {
plugins: [
vue(),
// HTML 模板处理
createHtmlPlugin({
minify: isProduction,
inject: {
data: {
title: env.VITE_APP_TITLE || "\u8003\u57F9\u7EC3\u7CFB\u7EDF",
description: "\u8003\u57F9\u7EC3\u7CFB\u7EDF - \u667A\u80FD\u5B66\u4E60\u5E73\u53F0"
}
}
}),
// 打包分析(仅在分析模式下)
...process.env.ANALYZE === "true" ? [visualizer({
filename: "dist/stats.html",
open: true,
gzipSize: true,
brotliSize: true
})] : []
].filter(Boolean),
resolve: {
alias: {
"@": path.resolve(__vite_injected_original_dirname, "./src")
}
},
// CSS 配置
css: {
preprocessorOptions: {
scss: {
additionalData: `@import "@/style/variables.scss";`
}
},
// 生产环境压缩CSS
...isProduction && {
postcss: {
plugins: [
__require("file:///app/node_modules/autoprefixer/lib/autoprefixer.js"),
__require("file:///app/node_modules/cssnano/src/index.js")({
preset: "default"
})
]
}
}
},
// 构建配置
build: {
target: "es2015",
outDir: "dist",
assetsDir: "assets",
sourcemap: !isProduction,
minify: isProduction ? "terser" : false,
// Terser 压缩配置
terserOptions: {
compress: {
drop_console: isProduction,
drop_debugger: isProduction,
pure_funcs: isProduction ? ["console.log", "console.info"] : []
}
},
// 代码分割配置
rollupOptions: {
output: {
// 手动分割代码块
manualChunks: {
// Vue 相关
vue: ["vue", "vue-router", "pinia"],
// UI 库
"element-plus": ["element-plus", "@element-plus/icons-vue"],
// 图表库
echarts: ["echarts"],
// 工具库
utils: ["axios"]
},
// 文件命名
chunkFileNames: "assets/js/[name]-[hash].js",
entryFileNames: "assets/js/[name]-[hash].js",
assetFileNames: (assetInfo) => {
const info = assetInfo.name.split(".");
let extType = info[info.length - 1];
if (/\.(mp4|webm|ogg|mp3|wav|flac|aac)(\?.*)?$/i.test(assetInfo.name)) {
extType = "media";
} else if (/\.(png|jpe?g|gif|svg)(\?.*)?$/i.test(assetInfo.name)) {
extType = "img";
} else if (/\.(woff2?|eot|ttf|otf)(\?.*)?$/i.test(assetInfo.name)) {
extType = "fonts";
}
return `assets/${extType}/[name]-[hash].[ext]`;
}
}
},
// 警告大小限制
chunkSizeWarningLimit: 1e3,
// 压缩资源
assetsInlineLimit: 4096
},
// 开发服务器配置
server: {
port: 3001,
host: true,
open: false,
cors: true,
proxy: {
"/api": {
target: env.VITE_API_BASE_URL || "http://localhost:8000",
changeOrigin: true,
rewrite: (path2) => path2,
configure: (proxy, options) => {
proxy.on("proxyRes", (proxyRes, req, res) => {
proxyRes.headers["content-type"] = "application/json; charset=utf-8";
});
}
},
"/ws": {
target: env.VITE_WS_BASE_URL || "ws://localhost:8000",
ws: true,
changeOrigin: true
}
}
},
// 预览服务器配置
preview: {
port: 4173,
host: true,
cors: true
},
// 优化配置
optimizeDeps: {
include: [
"vue",
"vue-router",
"pinia",
"element-plus",
"@element-plus/icons-vue",
"echarts"
],
exclude: ["@vitejs/plugin-vue"]
},
// 测试配置
test: {
environment: "jsdom",
globals: true,
setupFiles: ["./src/test/setup.ts"],
include: ["src/**/*.{test,spec}.{js,mjs,cjs,ts,mts,cts,jsx,tsx}"],
exclude: ["node_modules", "dist", ".idea", ".git", ".cache"],
coverage: {
provider: "v8",
reporter: ["text", "json", "html"],
exclude: [
"node_modules/",
"src/test/",
"**/*.d.ts",
"**/*.config.*",
"**/coverage/**",
"**/dist/**",
"**/.{idea,git,cache,output,temp}/**"
],
thresholds: {
global: {
branches: 60,
functions: 60,
lines: 60,
statements: 60
}
}
}
},
// 环境变量配置
define: {
__VUE_OPTIONS_API__: true,
__VUE_PROD_DEVTOOLS__: !isProduction,
__APP_VERSION__: JSON.stringify(process.env.npm_package_version || "1.0.0")
},
// 日志级别
logLevel: isProduction ? "info" : "warn"
};
});
export {
vite_config_default as default
};
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["vite.config.ts"],
  "sourcesContent": ["const __vite_injected_original_dirname = \"/app\";const __vite_injected_original_filename = \"/app/vite.config.ts\";const __vite_injected_original_import_meta_url = \"file:///app/vite.config.ts\";/// <reference types=\"vitest\" />\nimport { defineConfig, loadEnv } from 'vite'\nimport vue from '@vitejs/plugin-vue'\nimport path from 'path'\nimport { visualizer } from 'rollup-plugin-visualizer'\nimport { createHtmlPlugin } from 'vite-plugin-html'\n\n// https://vitejs.dev/config/\nexport default defineConfig(({ command, mode }) => {\n  const env = loadEnv(mode, process.cwd(), '')\n  const isProduction = mode === 'production'\n\n  return {\n    plugins: [\n      vue(),\n      \n      // HTML \u6A21\u677F\u5904\u7406\n      createHtmlPlugin({\n        minify: isProduction,\n        inject: {\n          data: {\n            title: env.VITE_APP_TITLE || '\u8003\u57F9\u7EC3\u7CFB\u7EDF',\n            description: '\u8003\u57F9\u7EC3\u7CFB\u7EDF - \u667A\u80FD\u5B66\u4E60\u5E73\u53F0'\n          }\n        }\n      }),\n\n      // \u6253\u5305\u5206\u6790\uFF08\u4EC5\u5728\u5206\u6790\u6A21\u5F0F\u4E0B\uFF09\n      ...(process.env.ANALYZE === 'true' ? [visualizer({\n        filename: 'dist/stats.html',\n        open: true,\n        gzipSize: true,\n        brotliSize: true\n      })] : [])\n    ].filter(Boolean),\n\n    resolve: {\n      alias: {\n        '@': path.resolve(__dirname, './src')\n      }\n    },\n\n    // CSS \u914D\u7F6E\n    css: {\n      preprocessorOptions: {\n        scss: {\n          additionalData: `@import \"@/style/variables.scss\";`\n        }\n      },\n      // \u751F\u4EA7\u73AF\u5883\u538B\u7F29CSS\n      ...(isProduction && {\n        postcss: {\n          plugins: [\n            require('autoprefixer'),\n            require('cssnano')({\n              preset: 'default'\n            })\n          ]\n        }\n      })\n    },\n\n    // \u6784\u5EFA\u914D\u7F6E\n    build: {\n      target: 'es2015',\n      outDir: 'dist',\n      assetsDir: 'assets',\n      sourcemap: !isProduction,\n      minify: isProduction ? 'terser' : false,\n      \n      // Terser \u538B\u7F29\u914D\u7F6E\n      terserOptions: {\n        compress: {\n          drop_console: isProduction,\n          drop_debugger: isProduction,\n          pure_funcs: isProduction ? ['console.log', 'console.info'] : []\n        }\n      },\n\n      // \u4EE3\u7801\u5206\u5272\u914D\u7F6E\n      rollupOptions: {\n        output: {\n          // \u624B\u52A8\u5206\u5272\u4EE3\u7801\u5757\n          manualChunks: {\n            // Vue \u76F8\u5173\n            vue: ['vue', 'vue-router', 'pinia'],\n            \n            // UI \u5E93\n            'element-plus': ['element-plus', '@element-plus/icons-vue'],\n            \n            // \u56FE\u8868\u5E93\n            echarts: ['echarts'],\n            \n            // \u5DE5\u5177\u5E93\n            utils: ['axios']\n          },\n          \n          // \u6587\u4EF6\u547D\u540D\n          chunkFileNames: 'assets/js/[name]-[hash].js',\n          entryFileNames: 'assets/js/[name]-[hash].js',\n          assetFileNames: (assetInfo) => {\n            const info = assetInfo.name.split('.')\n            let extType = info[info.length - 1]\n            \n            if (/\\.(mp4|webm|ogg|mp3|wav|flac|aac)(\\?.*)?$/i.test(assetInfo.name)) {\n              extType = 'media'\n            } else if (/\\.(png|jpe?g|gif|svg)(\\?.*)?$/i.test(assetInfo.name)) {\n              extType = 'img'\n            } else if (/\\.(woff2?|eot|ttf|otf)(\\?.*)?$/i.test(assetInfo.name)) {\n              extType = 'fonts'\n            }\n            \n            return `assets/${extType}/[name]-[hash].[ext]`\n          }\n        }\n      },\n\n      // \u8B66\u544A\u5927\u5C0F\u9650\u5236\n      chunkSizeWarningLimit: 1000,\n      \n      // \u538B\u7F29\u8D44\u6E90\n      assetsInlineLimit: 4096\n    },\n\n    // \u5F00\u53D1\u670D\u52A1\u5668\u914D\u7F6E\n    server: {\n      port: 3001,\n      host: true,\n      open: false,\n      cors: true,\n      proxy: {\n        '/api': {\n          target: env.VITE_API_BASE_URL || 'http://localhost:8000',\n          changeOrigin: true,\n          rewrite: (path) => path,\n          configure: (proxy, options) => {\n            proxy.on('proxyRes', (proxyRes, req, res) => {\n              // \u786E\u4FDD\u4EE3\u7406\u54CD\u5E94\u6B63\u786E\u8BBE\u7F6E UTF-8 \u7F16\u7801\n              proxyRes.headers['content-type'] = 'application/json; charset=utf-8'\n            })\n          }\n        },\n        '/ws': {\n          target: env.VITE_WS_BASE_URL || 'ws://localhost:8000',\n          ws: true,\n          changeOrigin: true\n        }\n      }\n    },\n\n    // \u9884\u89C8\u670D\u52A1\u5668\u914D\u7F6E\n    preview: {\n      port: 4173,\n      host: true,\n      cors: true\n    },\n\n    // \u4F18\u5316\u914D\u7F6E\n    optimizeDeps: {\n      include: [\n        'vue',\n        'vue-router',\n        'pinia',\n        'element-plus',\n        '@element-plus/icons-vue',\n        'echarts'\n      ],\n      exclude: ['@vitejs/plugin-vue']\n    },\n\n    // \u6D4B\u8BD5\u914D\u7F6E\n    test: {\n      environment: 'jsdom',\n      globals: true,\n      setupFiles: ['./src/test/setup.ts'],\n      include: ['src/**/*.{test,spec}.{js,mjs,cjs,ts,mts,cts,jsx,tsx}'],\n      exclude: ['node_modules', 'dist', '.idea', '.git', '.cache'],\n      coverage: {\n        provider: 'v8',\n        reporter: ['text', 'json', 'html'],\n        exclude: [\n          'node_modules/',\n          'src/test/',\n          '**/*.d.ts',\n          '**/*.config.*',\n          '**/coverage/**',\n          '**/dist/**',\n          '**/.{idea,git,cache,output,temp}/**'\n        ],\n        thresholds: {\n          global: {\n            branches: 60,\n            functions: 60,\n            lines: 60,\n            statements: 60\n          }\n        }\n      }\n    },\n\n    // \u73AF\u5883\u53D8\u91CF\u914D\u7F6E\n    define: {\n      __VUE_OPTIONS_API__: true,\n      __VUE_PROD_DEVTOOLS__: !isProduction,\n      __APP_VERSION__: JSON.stringify(process.env.npm_package_version || '1.0.0')\n    },\n\n    // \u65E5\u5FD7\u7EA7\u522B\n    logLevel: isProduction ? 'info' : 'warn'\n  }\n})"],
  "mappings": "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",
  "names": ["path"]
}
