You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
tailchat/client/web/build/webpack.config.ts

308 lines
8.3 KiB
TypeScript

/**
* Reference: https://webpack.js.org/configuration/configuration-languages/
*/
import type { Configuration } from 'webpack';
import { DefinePlugin } from 'webpack';
import type WebpackDevServer from 'webpack-dev-server';
import path from 'path';
import HtmlWebpackPlugin from 'html-webpack-plugin';
import MiniCssExtractPlugin from 'mini-css-extract-plugin';
import CopyPlugin from 'copy-webpack-plugin';
import TsconfigPathsPlugin from 'tsconfig-paths-webpack-plugin';
import { BundleAnalyzerPlugin } from 'webpack-bundle-analyzer';
import WebpackBar from 'webpackbar';
import fs from 'fs';
import WorkboxPlugin from 'workbox-webpack-plugin';
import { workboxPluginDetailPattern, workboxPluginEntryPattern } from './utils';
import dayjs from 'dayjs';
import { BundleStatsWebpackPlugin } from 'bundle-stats-webpack-plugin';
// eslint-disable-next-line @typescript-eslint/no-var-requires
require('dotenv').config();
delete process.env.TS_NODE_PROJECT; // https://github.com/dividab/tsconfig-paths-webpack-plugin/issues/32
require('../../build/script/buildPublicTranslation.js'); // 编译前先执行一下构建翻译的脚本
const ROOT_PATH = path.resolve(__dirname, '../');
const DIST_PATH = path.resolve(ROOT_PATH, './dist');
const ASSET_PATH = process.env.ASSET_PATH || '/';
const PORT = Number(process.env.PORT || 11011);
const ANALYSIS = process.env.ANALYSIS === 'true';
declare module 'webpack' {
interface Configuration {
devServer?: WebpackDevServer.Configuration;
}
}
const NODE_ENV = process.env.NODE_ENV ?? 'production';
const isDev = NODE_ENV === 'development';
const mode = isDev ? 'development' : 'production';
const plugins: Configuration['plugins'] = [
new DefinePlugin({
'process.env.NODE_ENV': JSON.stringify(NODE_ENV),
'process.env.SERVICE_URL': JSON.stringify(process.env.SERVICE_URL),
'process.env.VERSION': JSON.stringify(
process.env.VERSION || `nightly-${dayjs().format('YYYYMMDDHHmm')}`
),
}),
new HtmlWebpackPlugin({
title: 'Tailchat',
inject: true,
hash: false,
favicon: path.resolve(ROOT_PATH, './assets/images/favicon.ico'),
template: path.resolve(ROOT_PATH, './assets/template.html'),
preloadImage: `data:image/svg+xml;base64,${Buffer.from(
fs.readFileSync(path.resolve(ROOT_PATH, './assets/images/ripple.svg'), {
encoding: 'utf-8',
})
).toString('base64')}`,
}),
new CopyPlugin({
patterns: [
{
from: path.resolve(ROOT_PATH, '../locales'),
to: 'locales',
},
{
from: path.resolve(ROOT_PATH, './registry.json'),
to: 'registry.json',
},
{
from: path.resolve(ROOT_PATH, './assets/pwa.webmanifest'),
to: 'pwa.webmanifest',
},
{
from: path.resolve(ROOT_PATH, './assets/config.json'),
to: 'config.json',
},
{
from: path.resolve(ROOT_PATH, './assets/images/logo/'),
to: 'images/logo/',
},
{
from: path.resolve(ROOT_PATH, './assets/images/avatar/'),
to: 'images/avatar/',
},
{
from: path.resolve(ROOT_PATH, '../../vercel.json'),
to: 'vercel.json',
},
],
}) as any,
new MiniCssExtractPlugin({ filename: 'styles-[contenthash].css' }),
new WorkboxPlugin.GenerateSW({
// https://developers.google.com/web/tools/workbox
// these options encourage the ServiceWorkers to get in there fast
// and not allow any straggling "old" SWs to hang around
clientsClaim: true,
skipWaiting: true,
// Do not precache images
exclude: [/\.(?:png|jpg|jpeg|svg)$/, 'config.json'],
maximumFileSizeToCacheInBytes: 8 * 1024 * 1024, // 限制最大缓存 8M
// Define runtime caching rules.
runtimeCaching: [
{
// Match any request that ends with .png, .jpg, .jpeg or .svg.
urlPattern: /\.(?:png|jpg|jpeg|svg)$/,
// Apply a cache-first strategy.
// Reference: https://developers.google.com/web/tools/workbox/reference-docs/latest/module-workbox-strategies
handler: 'CacheFirst',
options: {
// Use a custom cache name.
cacheName: 'images',
// Only cache 10 images.
expiration: {
maxEntries: 10,
maxAgeSeconds: 14 * 24 * 60 * 60, // 2 week
},
},
},
//#region 插件缓存匹配
{
// 匹配内置 plugins 入口文件 以加速
urlPattern: workboxPluginEntryPattern,
handler: 'StaleWhileRevalidate',
options: {
cacheName: 'builtin-plugins-entry',
expiration: {
maxAgeSeconds: 24 * 60 * 60, // 1 day
},
cacheableResponse: {
// 只缓存js, 防止404后台直接fallback到html
headers: {
'content-type': 'application/javascript; charset=utf-8',
},
},
},
},
{
// 匹配内置 plugins 内容 以加速
urlPattern: workboxPluginDetailPattern,
handler: 'StaleWhileRevalidate',
options: {
cacheName: 'builtin-plugins-detail',
expiration: {
maxAgeSeconds: 7 * 24 * 60 * 60, // 1 week
},
cacheableResponse: {
// 只缓存js, 防止404后台直接fallback到html
headers: {
'content-type': 'application/javascript; charset=utf-8',
},
},
},
},
//#endregion
],
}),
new WebpackBar({
name: `Tailchat`,
}),
];
if (ANALYSIS) {
plugins.push(
new BundleAnalyzerPlugin({
analyzerMode: 'static',
openAnalyzer: true,
}) as any,
new BundleStatsWebpackPlugin()
);
}
const splitChunks: Required<Configuration>['optimization']['splitChunks'] = {
chunks: 'async',
minSize: 20000,
minRemainingSize: 0,
minChunks: 1,
maxAsyncRequests: 30,
maxInitialRequests: 30,
enforceSizeThreshold: 50000,
cacheGroups: {
vendors: {
chunks: 'initial',
name: 'vendors',
test: /[\\/]node_modules[\\/]/,
priority: -10,
reuseExistingChunk: true,
maxSize: 2 * 1000 * 1000,
},
default: {
minChunks: 2,
priority: -20,
reuseExistingChunk: true,
},
},
};
const config: Configuration = {
mode,
entry: {
app: path.resolve(ROOT_PATH, './src/index.tsx'),
},
output: {
path: DIST_PATH,
filename: '[name].[contenthash].js',
publicPath: ASSET_PATH,
},
devServer: {
port: PORT,
historyApiFallback: true,
static: {
directory: path.resolve(ROOT_PATH, './dist'),
},
client: {
overlay: false,
},
},
module: {
rules: [
{
test: /\.tsx?$/,
exclude: /node_modules/,
use: [
{
loader: 'esbuild-loader',
options: {
loader: 'tsx',
target: 'es2015',
tsconfigRaw: require('../tsconfig.json'),
},
},
{
loader: 'source-ref-loader',
options: {
available: isDev,
},
},
],
},
{
test: /\.(less|css)$/,
use: [
{
loader: MiniCssExtractPlugin.loader,
},
{
loader: 'css-loader',
options: {
// https://github.com/webpack-contrib/css-loader#auto
modules: {
auto: /\.module\.\w+$/i,
localIdentName: '[path][name]__[local]--[hash:base64:5]',
},
sourceMap: process.env.NODE_ENV !== 'production',
},
},
{
loader: 'postcss-loader',
options: {
postcssOptions: {
config: path.resolve(ROOT_PATH, 'postcss.config.js'),
},
},
},
{
loader: 'less-loader',
},
],
},
{
test: /\.(png|jpg|gif|woff|woff2|svg|eot|ttf)$/,
loader: 'url-loader',
options: {
limit: 8192,
name: 'assets/[name].[hash:7].[ext]',
},
},
],
},
optimization: {
splitChunks,
},
resolve: {
extensions: ['.tsx', '.ts', '.js', '.css'],
plugins: [
new TsconfigPathsPlugin({
configFile: path.resolve(ROOT_PATH, './tsconfig.json'),
}),
],
fallback: {
url: require.resolve('url/'),
},
},
plugins,
};
export default config;