@anansi/webpack-config
Version:
Production ready webpack for React
191 lines (189 loc) • 21.9 kB
JavaScript
;
var _interopRequireDefault = require("@babel/runtime-corejs3/helpers/interopRequireDefault").default;
exports.__esModule = true;
exports.default = makeProdConfig;
var _preloadWebpackPlugin = _interopRequireDefault(require("@vue/preload-webpack-plugin"));
var _cleanWebpackPlugin = require("clean-webpack-plugin");
var _cssMinimizerWebpackPlugin = _interopRequireDefault(require("css-minimizer-webpack-plugin"));
var _htmlWebpackPlugin = _interopRequireDefault(require("html-webpack-plugin"));
var _semver = _interopRequireDefault(require("semver"));
var _terserWebpackPlugin = _interopRequireDefault(require("terser-webpack-plugin"));
var _webpack = _interopRequireDefault(require("webpack"));
var _webpackRemoveEmptyScripts = _interopRequireDefault(require("webpack-remove-empty-scripts"));
var _base = require("./base");
//import CrittersPlugin from 'critters-webpack-plugin';
function makeProdConfig(baseConfig, {
rootPath,
basePath,
libraryInclude,
libraryExclude,
argv = {},
env = {},
htmlOptions = {
title: '',
scriptLoading: 'defer'
},
terserOptions,
sassOptions,
sassResources,
cssModulesOptions,
globalStyleDir,
fontPreload,
svgoOptions,
nohash,
cssExtractOptions
}) {
const config = {
...baseConfig
};
const reactVersion = getReactVersion(rootPath);
config.mode = 'production';
config.bail = true; // this helps automatic build tools not waste time
if (!argv?.target?.includes?.('node')) {
config.plugins.push(new _webpack.default.IgnorePlugin({
resourceRegExp: /DevTools/
}), new _cleanWebpackPlugin.CleanWebpackPlugin(), new _webpack.default.LoaderOptionsPlugin({
minimize: true,
debug: false
}), new _webpackRemoveEmptyScripts.default());
if (htmlOptions) {
config.plugins.unshift(new _htmlWebpackPlugin.default(htmlOptions)
//new CrittersPlugin({}),
//new InlineChunkHtmlPlugin(HtmlWebpackPlugin, [/runtime-.+[.]js/]), this is a bad idea until CSP nonce can be added
);
if (fontPreload) {
if (!['preload', 'prefetch'].includes(fontPreload)) throw new Error(`fontPreload: '${fontPreload}' is not valid.\nUse 'preload' or 'prefetch'`);
config.plugins.unshift(new _preloadWebpackPlugin.default({
rel: fontPreload,
include: 'allAssets',
fileWhitelist: [/\.(otf|woff|ttf)/],
as
}));
}
}
}
if (svgoOptions !== false) {
config.module.rules.push({
test: /\.svg(\?v=\d+\.\d+\.\d+)?$/,
issuer: {
not: [/\.(j|t)sx?$/]
},
enforce: 'pre',
use: [{
loader: require.resolve('svgo-loader'),
options: svgoOptions
}]
});
}
config.optimization = {
splitChunks: {
chunks: 'async',
maxInitialRequests: 30,
maxAsyncRequests: 30,
cacheGroups: {
react: {
test: /[\\/]node_modules[\\/](react|react-dom|scheduler|object-assign|loose-envify)[\\/]/,
name: 'react',
chunks: 'all'
},
polyfill: {
test: /[\\/]node_modules[\\/](core-js|core-js-pure|@babel\/runtime|@babel\/runtime-corejs3|regenerator-runtime|ric-shim|babel-runtime)[\\/].*/,
name: 'polyfill',
chunks: 'all'
},
styles: {
test: new RegExp(`${globalStyleDir}/.*\\.scss$`),
name: 'style',
type: 'css/mini-extract',
chunks: 'all'
}
}
},
// https://webpack.js.org/configuration/optimization/#optimizationruntimechunk
runtimeChunk: {
name: 'webpack-runtime'
}
};
if (nohash) {
config.optimization.chunkIds = 'named';
}
if (!env?.readable) {
config.optimization.minimizer = [new _terserWebpackPlugin.default({
terserOptions: {
parse: {
ecma: 9
},
compress: {
ecma: 6,
warnings: false,
// Pending further investigation:
// https://github.com/mishoo/UglifyJS2/issues/2011
comparisons: false,
// Pending futher investigation:
// https://github.com/terser-js/terser/issues/120
inline: 2
},
mangle: {
safari10: true
},
output: {
ecma: 6,
comments: false,
ascii_only: true
},
...terserOptions,
keep_classnames: !!env?.profile || (terserOptions?.keep_classnames ?? true),
keep_fnames: !!env?.profile || terserOptions?.keep_fnames
},
extractComments: true
}),
// cssnano on node_modules as well as our loaders
new _cssMinimizerWebpackPlugin.default()];
} else {
config.optimization.minimize = false;
}
config.performance = {
maxEntrypointSize: 300000,
assetFilter(assetFilename) {
return !/\.(map|LICENSE)$/.test(assetFilename);
}
};
const styleRules = (0, _base.getStyleRules)({
rootPath,
basePath,
libraryInclude,
libraryExclude,
sassOptions,
sassResources,
cssModulesOptions,
globalStyleDir,
target: argv?.target,
cssExtractOptions
});
config.module.rules = [...config.module.rules, styleRules];
if (env?.profile) {
let reactClient = 'react-dom/client$';
if (reactVersion && _semver.default.lt(reactVersion, '19.0.0')) {
reactClient = 'react-dom$';
}
config.resolve.alias = {
...config?.resolve?.alias,
[reactClient]: 'react-dom/profiling',
'scheduler/tracing': 'scheduler/tracing-profiling'
};
}
return config;
}
function as(entry) {
if (/\.css$/.test(entry)) return 'style';
if (/\.(otf|eot|woff2|woff|ttf)$/.test(entry)) return 'font';
if (/\.(svg|apng|png|jpg|gif|ico|webp|avif|cur|ani)$/.test(entry)) return 'image';
return 'script';
}
function getReactVersion(rootPath) {
const react = require(require.resolve('react', {
paths: [rootPath]
}));
return react ? react.version : null;
}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_preloadWebpackPlugin","_interopRequireDefault","require","_cleanWebpackPlugin","_cssMinimizerWebpackPlugin","_htmlWebpackPlugin","_semver","_terserWebpackPlugin","_webpack","_webpackRemoveEmptyScripts","_base","makeProdConfig","baseConfig","rootPath","basePath","libraryInclude","libraryExclude","argv","env","htmlOptions","title","scriptLoading","terserOptions","sassOptions","sassResources","cssModulesOptions","globalStyleDir","fontPreload","svgoOptions","nohash","cssExtractOptions","config","reactVersion","getReactVersion","mode","bail","target","includes","plugins","push","webpack","IgnorePlugin","resourceRegExp","CleanWebpackPlugin","LoaderOptionsPlugin","minimize","debug","RemoveEmptyScriptsPlugin","unshift","HtmlWebpackPlugin","Error","PreloadWebpackPlugin","rel","include","fileWhitelist","as","module","rules","test","issuer","not","enforce","use","loader","resolve","options","optimization","splitChunks","chunks","maxInitialRequests","maxAsyncRequests","cacheGroups","react","name","polyfill","styles","RegExp","type","runtimeChunk","chunkIds","readable","minimizer","TerserPlugin","parse","ecma","compress","warnings","comparisons","inline","mangle","safari10","output","comments","ascii_only","keep_classnames","profile","keep_fnames","extractComments","CssMinimizerPlugin","performance","maxEntrypointSize","assetFilter","assetFilename","styleRules","getStyleRules","reactClient","semver","lt","alias","entry","paths","version"],"sources":["../src/prod.js"],"sourcesContent":["import PreloadWebpackPlugin from '@vue/preload-webpack-plugin';\nimport { CleanWebpackPlugin } from 'clean-webpack-plugin';\nimport CssMinimizerPlugin from 'css-minimizer-webpack-plugin';\nimport HtmlWebpackPlugin from 'html-webpack-plugin';\nimport semver from 'semver';\nimport TerserPlugin from 'terser-webpack-plugin';\nimport webpack from 'webpack';\n//import CrittersPlugin from 'critters-webpack-plugin';\nimport RemoveEmptyScriptsPlugin from 'webpack-remove-empty-scripts';\n\nimport { getStyleRules } from './base';\n\nexport default function makeProdConfig(\n  baseConfig,\n  {\n    rootPath,\n    basePath,\n    libraryInclude,\n    libraryExclude,\n    argv = {},\n    env = {},\n    htmlOptions = { title: '', scriptLoading: 'defer' },\n    terserOptions,\n    sassOptions,\n    sassResources,\n    cssModulesOptions,\n    globalStyleDir,\n    fontPreload,\n    svgoOptions,\n    nohash,\n    cssExtractOptions,\n  },\n) {\n  const config = { ...baseConfig };\n  const reactVersion = getReactVersion(rootPath);\n\n  config.mode = 'production';\n  config.bail = true; // this helps automatic build tools not waste time\n  if (!argv?.target?.includes?.('node')) {\n    config.plugins.push(\n      new webpack.IgnorePlugin({ resourceRegExp: /DevTools/ }),\n      new CleanWebpackPlugin(),\n      new webpack.LoaderOptionsPlugin({\n        minimize: true,\n        debug: false,\n      }),\n      new RemoveEmptyScriptsPlugin(),\n    );\n    if (htmlOptions) {\n      config.plugins.unshift(\n        new HtmlWebpackPlugin(htmlOptions),\n        //new CrittersPlugin({}),\n        //new InlineChunkHtmlPlugin(HtmlWebpackPlugin, [/runtime-.+[.]js/]), this is a bad idea until CSP nonce can be added\n      );\n      if (fontPreload) {\n        if (!['preload', 'prefetch'].includes(fontPreload))\n          throw new Error(\n            `fontPreload: '${fontPreload}' is not valid.\\nUse 'preload' or 'prefetch'`,\n          );\n        config.plugins.unshift(\n          new PreloadWebpackPlugin({\n            rel: fontPreload,\n            include: 'allAssets',\n            fileWhitelist: [/\\.(otf|woff|ttf)/],\n            as,\n          }),\n        );\n      }\n    }\n  }\n  if (svgoOptions !== false) {\n    config.module.rules.push({\n      test: /\\.svg(\\?v=\\d+\\.\\d+\\.\\d+)?$/,\n      issuer: {\n        not: [/\\.(j|t)sx?$/],\n      },\n      enforce: 'pre',\n      use: [\n        {\n          loader: require.resolve('svgo-loader'),\n          options: svgoOptions,\n        },\n      ],\n    });\n  }\n  config.optimization = {\n    splitChunks: {\n      chunks: 'async',\n      maxInitialRequests: 30,\n      maxAsyncRequests: 30,\n      cacheGroups: {\n        react: {\n          test: /[\\\\/]node_modules[\\\\/](react|react-dom|scheduler|object-assign|loose-envify)[\\\\/]/,\n          name: 'react',\n          chunks: 'all',\n        },\n        polyfill: {\n          test: /[\\\\/]node_modules[\\\\/](core-js|core-js-pure|@babel\\/runtime|@babel\\/runtime-corejs3|regenerator-runtime|ric-shim|babel-runtime)[\\\\/].*/,\n          name: 'polyfill',\n          chunks: 'all',\n        },\n        styles: {\n          test: new RegExp(`${globalStyleDir}/.*\\\\.scss$`),\n          name: 'style',\n          type: 'css/mini-extract',\n          chunks: 'all',\n        },\n      },\n    },\n    // https://webpack.js.org/configuration/optimization/#optimizationruntimechunk\n    runtimeChunk: {\n      name: 'webpack-runtime',\n    },\n  };\n  if (nohash) {\n    config.optimization.chunkIds = 'named';\n  }\n  if (!env?.readable) {\n    config.optimization.minimizer = [\n      new TerserPlugin({\n        terserOptions: {\n          parse: {\n            ecma: 9,\n          },\n          compress: {\n            ecma: 6,\n            warnings: false,\n            // Pending further investigation:\n            // https://github.com/mishoo/UglifyJS2/issues/2011\n            comparisons: false,\n            // Pending futher investigation:\n            // https://github.com/terser-js/terser/issues/120\n            inline: 2,\n          },\n          mangle: {\n            safari10: true,\n          },\n          output: {\n            ecma: 6,\n            comments: false,\n            ascii_only: true,\n          },\n          ...terserOptions,\n          keep_classnames:\n            !!env?.profile || (terserOptions?.keep_classnames ?? true),\n          keep_fnames: !!env?.profile || terserOptions?.keep_fnames,\n        },\n        extractComments: true,\n      }),\n      // cssnano on node_modules as well as our loaders\n      new CssMinimizerPlugin(),\n    ];\n  } else {\n    config.optimization.minimize = false;\n  }\n  config.performance = {\n    maxEntrypointSize: 300000,\n    assetFilter(assetFilename) {\n      return !/\\.(map|LICENSE)$/.test(assetFilename);\n    },\n  };\n\n  const styleRules = getStyleRules({\n    rootPath,\n    basePath,\n    libraryInclude,\n    libraryExclude,\n    sassOptions,\n    sassResources,\n    cssModulesOptions,\n    globalStyleDir,\n    target: argv?.target,\n    cssExtractOptions,\n  });\n  config.module.rules = [...config.module.rules, styleRules];\n\n  if (env?.profile) {\n    let reactClient = 'react-dom/client$';\n    if (reactVersion && semver.lt(reactVersion, '19.0.0')) {\n      reactClient = 'react-dom$';\n    }\n    config.resolve.alias = {\n      ...config?.resolve?.alias,\n      [reactClient]: 'react-dom/profiling',\n      'scheduler/tracing': 'scheduler/tracing-profiling',\n    };\n  }\n  return config;\n}\n\nfunction as(entry) {\n  if (/\\.css$/.test(entry)) return 'style';\n  if (/\\.(otf|eot|woff2|woff|ttf)$/.test(entry)) return 'font';\n  if (/\\.(svg|apng|png|jpg|gif|ico|webp|avif|cur|ani)$/.test(entry))\n    return 'image';\n  return 'script';\n}\n\nfunction getReactVersion(rootPath) {\n  const react = require(\n    require.resolve('react', {\n      paths: [rootPath],\n    }),\n  );\n  return react ? react.version : null;\n}\n"],"mappings":"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","ignoreList":[]}