WebpackOptionsApply.js 26 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698
  1. /*
  2. MIT License http://www.opensource.org/licenses/mit-license.php
  3. Author Tobias Koppers @sokra
  4. */
  5. "use strict";
  6. const OptionsApply = require("./OptionsApply");
  7. const AssetModulesPlugin = require("./asset/AssetModulesPlugin");
  8. const JavascriptModulesPlugin = require("./javascript/JavascriptModulesPlugin");
  9. const JsonModulesPlugin = require("./json/JsonModulesPlugin");
  10. const ChunkPrefetchPreloadPlugin = require("./prefetch/ChunkPrefetchPreloadPlugin");
  11. const EntryOptionPlugin = require("./EntryOptionPlugin");
  12. const RecordIdsPlugin = require("./RecordIdsPlugin");
  13. const RuntimePlugin = require("./RuntimePlugin");
  14. const APIPlugin = require("./APIPlugin");
  15. const CompatibilityPlugin = require("./CompatibilityPlugin");
  16. const ConstPlugin = require("./ConstPlugin");
  17. const ExportsInfoApiPlugin = require("./ExportsInfoApiPlugin");
  18. const WebpackIsIncludedPlugin = require("./WebpackIsIncludedPlugin");
  19. const TemplatedPathPlugin = require("./TemplatedPathPlugin");
  20. const UseStrictPlugin = require("./UseStrictPlugin");
  21. const WarnCaseSensitiveModulesPlugin = require("./WarnCaseSensitiveModulesPlugin");
  22. const DataUriPlugin = require("./schemes/DataUriPlugin");
  23. const FileUriPlugin = require("./schemes/FileUriPlugin");
  24. const ResolverCachePlugin = require("./cache/ResolverCachePlugin");
  25. const CommonJsPlugin = require("./dependencies/CommonJsPlugin");
  26. const HarmonyModulesPlugin = require("./dependencies/HarmonyModulesPlugin");
  27. const ImportMetaPlugin = require("./dependencies/ImportMetaPlugin");
  28. const ImportPlugin = require("./dependencies/ImportPlugin");
  29. const LoaderPlugin = require("./dependencies/LoaderPlugin");
  30. const RequireContextPlugin = require("./dependencies/RequireContextPlugin");
  31. const RequireEnsurePlugin = require("./dependencies/RequireEnsurePlugin");
  32. const RequireIncludePlugin = require("./dependencies/RequireIncludePlugin");
  33. const SystemPlugin = require("./dependencies/SystemPlugin");
  34. const URLPlugin = require("./dependencies/URLPlugin");
  35. const WorkerPlugin = require("./dependencies/WorkerPlugin");
  36. const InferAsyncModulesPlugin = require("./async-modules/InferAsyncModulesPlugin");
  37. const JavascriptMetaInfoPlugin = require("./JavascriptMetaInfoPlugin");
  38. const DefaultStatsFactoryPlugin = require("./stats/DefaultStatsFactoryPlugin");
  39. const DefaultStatsPresetPlugin = require("./stats/DefaultStatsPresetPlugin");
  40. const DefaultStatsPrinterPlugin = require("./stats/DefaultStatsPrinterPlugin");
  41. const { cleverMerge } = require("./util/cleverMerge");
  42. /** @typedef {import("../declarations/WebpackOptions").WebpackOptionsNormalized} WebpackOptions */
  43. /** @typedef {import("./Compiler")} Compiler */
  44. class WebpackOptionsApply extends OptionsApply {
  45. constructor() {
  46. super();
  47. }
  48. /**
  49. * @param {WebpackOptions} options options object
  50. * @param {Compiler} compiler compiler object
  51. * @returns {WebpackOptions} options object
  52. */
  53. process(options, compiler) {
  54. compiler.outputPath = options.output.path;
  55. compiler.recordsInputPath = options.recordsInputPath || null;
  56. compiler.recordsOutputPath = options.recordsOutputPath || null;
  57. compiler.name = options.name;
  58. if (options.externals) {
  59. //@ts-expect-error https://github.com/microsoft/TypeScript/issues/41697
  60. const ExternalsPlugin = require("./ExternalsPlugin");
  61. new ExternalsPlugin(options.externalsType, options.externals).apply(
  62. compiler
  63. );
  64. }
  65. if (options.externalsPresets.node) {
  66. const NodeTargetPlugin = require("./node/NodeTargetPlugin");
  67. new NodeTargetPlugin().apply(compiler);
  68. }
  69. if (options.externalsPresets.electronMain) {
  70. //@ts-expect-error https://github.com/microsoft/TypeScript/issues/41697
  71. const ElectronTargetPlugin = require("./electron/ElectronTargetPlugin");
  72. new ElectronTargetPlugin("main").apply(compiler);
  73. }
  74. if (options.externalsPresets.electronPreload) {
  75. //@ts-expect-error https://github.com/microsoft/TypeScript/issues/41697
  76. const ElectronTargetPlugin = require("./electron/ElectronTargetPlugin");
  77. new ElectronTargetPlugin("preload").apply(compiler);
  78. }
  79. if (options.externalsPresets.electronRenderer) {
  80. //@ts-expect-error https://github.com/microsoft/TypeScript/issues/41697
  81. const ElectronTargetPlugin = require("./electron/ElectronTargetPlugin");
  82. new ElectronTargetPlugin("renderer").apply(compiler);
  83. }
  84. if (
  85. options.externalsPresets.electron &&
  86. !options.externalsPresets.electronMain &&
  87. !options.externalsPresets.electronPreload &&
  88. !options.externalsPresets.electronRenderer
  89. ) {
  90. //@ts-expect-error https://github.com/microsoft/TypeScript/issues/41697
  91. const ElectronTargetPlugin = require("./electron/ElectronTargetPlugin");
  92. new ElectronTargetPlugin().apply(compiler);
  93. }
  94. if (options.externalsPresets.nwjs) {
  95. //@ts-expect-error https://github.com/microsoft/TypeScript/issues/41697
  96. const ExternalsPlugin = require("./ExternalsPlugin");
  97. new ExternalsPlugin("node-commonjs", "nw.gui").apply(compiler);
  98. }
  99. if (options.externalsPresets.webAsync) {
  100. //@ts-expect-error https://github.com/microsoft/TypeScript/issues/41697
  101. const ExternalsPlugin = require("./ExternalsPlugin");
  102. new ExternalsPlugin(
  103. "import",
  104. options.experiments.css
  105. ? ({ request, dependencyType }, callback) => {
  106. if (dependencyType === "url") {
  107. if (/^(\/\/|https?:\/\/)/.test(request))
  108. return callback(null, `asset ${request}`);
  109. } else if (dependencyType === "css-import") {
  110. if (/^(\/\/|https?:\/\/)/.test(request))
  111. return callback(null, `css-import ${request}`);
  112. } else if (/^(\/\/|https?:\/\/|std:)/.test(request)) {
  113. if (/^\.css(\?|$)/.test(request))
  114. return callback(null, `css-import ${request}`);
  115. return callback(null, `import ${request}`);
  116. }
  117. callback();
  118. }
  119. : /^(\/\/|https?:\/\/|std:)/
  120. ).apply(compiler);
  121. } else if (options.externalsPresets.web) {
  122. //@ts-expect-error https://github.com/microsoft/TypeScript/issues/41697
  123. const ExternalsPlugin = require("./ExternalsPlugin");
  124. new ExternalsPlugin(
  125. "module",
  126. options.experiments.css
  127. ? ({ request, dependencyType }, callback) => {
  128. if (dependencyType === "url") {
  129. if (/^(\/\/|https?:\/\/)/.test(request))
  130. return callback(null, `asset ${request}`);
  131. } else if (dependencyType === "css-import") {
  132. if (/^(\/\/|https?:\/\/)/.test(request))
  133. return callback(null, `css-import ${request}`);
  134. } else if (/^(\/\/|https?:\/\/|std:)/.test(request)) {
  135. if (/^\.css(\?|$)/.test(request))
  136. return callback(null, `css-import ${request}`);
  137. return callback(null, `module ${request}`);
  138. }
  139. callback();
  140. }
  141. : /^(\/\/|https?:\/\/|std:)/
  142. ).apply(compiler);
  143. }
  144. new ChunkPrefetchPreloadPlugin().apply(compiler);
  145. if (typeof options.output.chunkFormat === "string") {
  146. switch (options.output.chunkFormat) {
  147. case "array-push": {
  148. const ArrayPushCallbackChunkFormatPlugin = require("./javascript/ArrayPushCallbackChunkFormatPlugin");
  149. new ArrayPushCallbackChunkFormatPlugin().apply(compiler);
  150. break;
  151. }
  152. case "commonjs": {
  153. const CommonJsChunkFormatPlugin = require("./javascript/CommonJsChunkFormatPlugin");
  154. new CommonJsChunkFormatPlugin().apply(compiler);
  155. break;
  156. }
  157. case "module": {
  158. const ModuleChunkFormatPlugin = require("./esm/ModuleChunkFormatPlugin");
  159. new ModuleChunkFormatPlugin().apply(compiler);
  160. break;
  161. }
  162. default:
  163. throw new Error(
  164. "Unsupported chunk format '" + options.output.chunkFormat + "'."
  165. );
  166. }
  167. }
  168. if (options.output.enabledChunkLoadingTypes.length > 0) {
  169. for (const type of options.output.enabledChunkLoadingTypes) {
  170. const EnableChunkLoadingPlugin = require("./javascript/EnableChunkLoadingPlugin");
  171. new EnableChunkLoadingPlugin(type).apply(compiler);
  172. }
  173. }
  174. if (options.output.enabledWasmLoadingTypes.length > 0) {
  175. for (const type of options.output.enabledWasmLoadingTypes) {
  176. const EnableWasmLoadingPlugin = require("./wasm/EnableWasmLoadingPlugin");
  177. new EnableWasmLoadingPlugin(type).apply(compiler);
  178. }
  179. }
  180. if (options.output.enabledLibraryTypes.length > 0) {
  181. for (const type of options.output.enabledLibraryTypes) {
  182. const EnableLibraryPlugin = require("./library/EnableLibraryPlugin");
  183. new EnableLibraryPlugin(type).apply(compiler);
  184. }
  185. }
  186. if (options.output.pathinfo) {
  187. const ModuleInfoHeaderPlugin = require("./ModuleInfoHeaderPlugin");
  188. new ModuleInfoHeaderPlugin(options.output.pathinfo !== true).apply(
  189. compiler
  190. );
  191. }
  192. if (options.output.clean) {
  193. const CleanPlugin = require("./CleanPlugin");
  194. new CleanPlugin(
  195. options.output.clean === true ? {} : options.output.clean
  196. ).apply(compiler);
  197. }
  198. if (options.devtool) {
  199. if (options.devtool.includes("source-map")) {
  200. const hidden = options.devtool.includes("hidden");
  201. const inline = options.devtool.includes("inline");
  202. const evalWrapped = options.devtool.includes("eval");
  203. const cheap = options.devtool.includes("cheap");
  204. const moduleMaps = options.devtool.includes("module");
  205. const noSources = options.devtool.includes("nosources");
  206. const Plugin = evalWrapped
  207. ? require("./EvalSourceMapDevToolPlugin")
  208. : require("./SourceMapDevToolPlugin");
  209. new Plugin({
  210. filename: inline ? null : options.output.sourceMapFilename,
  211. moduleFilenameTemplate: options.output.devtoolModuleFilenameTemplate,
  212. fallbackModuleFilenameTemplate:
  213. options.output.devtoolFallbackModuleFilenameTemplate,
  214. append: hidden ? false : undefined,
  215. module: moduleMaps ? true : cheap ? false : true,
  216. columns: cheap ? false : true,
  217. noSources: noSources,
  218. namespace: options.output.devtoolNamespace
  219. }).apply(compiler);
  220. } else if (options.devtool.includes("eval")) {
  221. const EvalDevToolModulePlugin = require("./EvalDevToolModulePlugin");
  222. new EvalDevToolModulePlugin({
  223. moduleFilenameTemplate: options.output.devtoolModuleFilenameTemplate,
  224. namespace: options.output.devtoolNamespace
  225. }).apply(compiler);
  226. }
  227. }
  228. new JavascriptModulesPlugin().apply(compiler);
  229. new JsonModulesPlugin().apply(compiler);
  230. new AssetModulesPlugin().apply(compiler);
  231. if (!options.experiments.outputModule) {
  232. if (options.output.module) {
  233. throw new Error(
  234. "'output.module: true' is only allowed when 'experiments.outputModule' is enabled"
  235. );
  236. }
  237. if (options.output.enabledLibraryTypes.includes("module")) {
  238. throw new Error(
  239. "library type \"module\" is only allowed when 'experiments.outputModule' is enabled"
  240. );
  241. }
  242. if (options.externalsType === "module") {
  243. throw new Error(
  244. "'externalsType: \"module\"' is only allowed when 'experiments.outputModule' is enabled"
  245. );
  246. }
  247. }
  248. if (options.experiments.syncWebAssembly) {
  249. const WebAssemblyModulesPlugin = require("./wasm-sync/WebAssemblyModulesPlugin");
  250. new WebAssemblyModulesPlugin({
  251. mangleImports: options.optimization.mangleWasmImports
  252. }).apply(compiler);
  253. }
  254. if (options.experiments.asyncWebAssembly) {
  255. const AsyncWebAssemblyModulesPlugin = require("./wasm-async/AsyncWebAssemblyModulesPlugin");
  256. new AsyncWebAssemblyModulesPlugin({
  257. mangleImports: options.optimization.mangleWasmImports
  258. }).apply(compiler);
  259. }
  260. if (options.experiments.css) {
  261. const CssModulesPlugin = require("./css/CssModulesPlugin");
  262. new CssModulesPlugin().apply(compiler);
  263. }
  264. if (options.experiments.lazyCompilation) {
  265. const LazyCompilationPlugin = require("./hmr/LazyCompilationPlugin");
  266. const lazyOptions =
  267. typeof options.experiments.lazyCompilation === "object"
  268. ? options.experiments.lazyCompilation
  269. : null;
  270. new LazyCompilationPlugin({
  271. backend:
  272. typeof lazyOptions.backend === "function"
  273. ? lazyOptions.backend
  274. : require("./hmr/lazyCompilationBackend")({
  275. ...lazyOptions.backend,
  276. client:
  277. (lazyOptions.backend && lazyOptions.backend.client) ||
  278. require.resolve(
  279. `../hot/lazy-compilation-${
  280. options.externalsPresets.node ? "node" : "web"
  281. }.js`
  282. )
  283. }),
  284. entries: !lazyOptions || lazyOptions.entries !== false,
  285. imports: !lazyOptions || lazyOptions.imports !== false,
  286. test: (lazyOptions && lazyOptions.test) || undefined
  287. }).apply(compiler);
  288. }
  289. if (options.experiments.buildHttp) {
  290. const HttpUriPlugin = require("./schemes/HttpUriPlugin");
  291. const httpOptions = options.experiments.buildHttp;
  292. new HttpUriPlugin(httpOptions).apply(compiler);
  293. }
  294. new EntryOptionPlugin().apply(compiler);
  295. compiler.hooks.entryOption.call(options.context, options.entry);
  296. new RuntimePlugin().apply(compiler);
  297. new InferAsyncModulesPlugin().apply(compiler);
  298. new DataUriPlugin().apply(compiler);
  299. new FileUriPlugin().apply(compiler);
  300. new CompatibilityPlugin().apply(compiler);
  301. new HarmonyModulesPlugin({
  302. topLevelAwait: options.experiments.topLevelAwait
  303. }).apply(compiler);
  304. if (options.amd !== false) {
  305. const AMDPlugin = require("./dependencies/AMDPlugin");
  306. const RequireJsStuffPlugin = require("./RequireJsStuffPlugin");
  307. new AMDPlugin(options.amd || {}).apply(compiler);
  308. new RequireJsStuffPlugin().apply(compiler);
  309. }
  310. new CommonJsPlugin().apply(compiler);
  311. new LoaderPlugin({}).apply(compiler);
  312. if (options.node !== false) {
  313. const NodeStuffPlugin = require("./NodeStuffPlugin");
  314. new NodeStuffPlugin(options.node).apply(compiler);
  315. }
  316. new APIPlugin().apply(compiler);
  317. new ExportsInfoApiPlugin().apply(compiler);
  318. new WebpackIsIncludedPlugin().apply(compiler);
  319. new ConstPlugin().apply(compiler);
  320. new UseStrictPlugin().apply(compiler);
  321. new RequireIncludePlugin().apply(compiler);
  322. new RequireEnsurePlugin().apply(compiler);
  323. new RequireContextPlugin().apply(compiler);
  324. new ImportPlugin().apply(compiler);
  325. new SystemPlugin().apply(compiler);
  326. new ImportMetaPlugin().apply(compiler);
  327. new URLPlugin().apply(compiler);
  328. new WorkerPlugin(
  329. options.output.workerChunkLoading,
  330. options.output.workerWasmLoading,
  331. options.output.module
  332. ).apply(compiler);
  333. new DefaultStatsFactoryPlugin().apply(compiler);
  334. new DefaultStatsPresetPlugin().apply(compiler);
  335. new DefaultStatsPrinterPlugin().apply(compiler);
  336. new JavascriptMetaInfoPlugin().apply(compiler);
  337. if (typeof options.mode !== "string") {
  338. const WarnNoModeSetPlugin = require("./WarnNoModeSetPlugin");
  339. new WarnNoModeSetPlugin().apply(compiler);
  340. }
  341. const EnsureChunkConditionsPlugin = require("./optimize/EnsureChunkConditionsPlugin");
  342. new EnsureChunkConditionsPlugin().apply(compiler);
  343. if (options.optimization.removeAvailableModules) {
  344. const RemoveParentModulesPlugin = require("./optimize/RemoveParentModulesPlugin");
  345. new RemoveParentModulesPlugin().apply(compiler);
  346. }
  347. if (options.optimization.removeEmptyChunks) {
  348. const RemoveEmptyChunksPlugin = require("./optimize/RemoveEmptyChunksPlugin");
  349. new RemoveEmptyChunksPlugin().apply(compiler);
  350. }
  351. if (options.optimization.mergeDuplicateChunks) {
  352. const MergeDuplicateChunksPlugin = require("./optimize/MergeDuplicateChunksPlugin");
  353. new MergeDuplicateChunksPlugin().apply(compiler);
  354. }
  355. if (options.optimization.flagIncludedChunks) {
  356. const FlagIncludedChunksPlugin = require("./optimize/FlagIncludedChunksPlugin");
  357. new FlagIncludedChunksPlugin().apply(compiler);
  358. }
  359. if (options.optimization.sideEffects) {
  360. const SideEffectsFlagPlugin = require("./optimize/SideEffectsFlagPlugin");
  361. new SideEffectsFlagPlugin(
  362. options.optimization.sideEffects === true
  363. ).apply(compiler);
  364. }
  365. if (options.optimization.providedExports) {
  366. const FlagDependencyExportsPlugin = require("./FlagDependencyExportsPlugin");
  367. new FlagDependencyExportsPlugin().apply(compiler);
  368. }
  369. if (options.optimization.usedExports) {
  370. const FlagDependencyUsagePlugin = require("./FlagDependencyUsagePlugin");
  371. new FlagDependencyUsagePlugin(
  372. options.optimization.usedExports === "global"
  373. ).apply(compiler);
  374. }
  375. if (options.optimization.innerGraph) {
  376. const InnerGraphPlugin = require("./optimize/InnerGraphPlugin");
  377. new InnerGraphPlugin().apply(compiler);
  378. }
  379. if (options.optimization.mangleExports) {
  380. const MangleExportsPlugin = require("./optimize/MangleExportsPlugin");
  381. new MangleExportsPlugin(
  382. options.optimization.mangleExports !== "size"
  383. ).apply(compiler);
  384. }
  385. if (options.optimization.concatenateModules) {
  386. const ModuleConcatenationPlugin = require("./optimize/ModuleConcatenationPlugin");
  387. new ModuleConcatenationPlugin().apply(compiler);
  388. }
  389. if (options.optimization.splitChunks) {
  390. const SplitChunksPlugin = require("./optimize/SplitChunksPlugin");
  391. new SplitChunksPlugin(options.optimization.splitChunks).apply(compiler);
  392. }
  393. if (options.optimization.runtimeChunk) {
  394. const RuntimeChunkPlugin = require("./optimize/RuntimeChunkPlugin");
  395. new RuntimeChunkPlugin(options.optimization.runtimeChunk).apply(compiler);
  396. }
  397. if (!options.optimization.emitOnErrors) {
  398. const NoEmitOnErrorsPlugin = require("./NoEmitOnErrorsPlugin");
  399. new NoEmitOnErrorsPlugin().apply(compiler);
  400. }
  401. if (options.optimization.realContentHash) {
  402. const RealContentHashPlugin = require("./optimize/RealContentHashPlugin");
  403. new RealContentHashPlugin({
  404. hashFunction: options.output.hashFunction,
  405. hashDigest: options.output.hashDigest
  406. }).apply(compiler);
  407. }
  408. if (options.optimization.checkWasmTypes) {
  409. const WasmFinalizeExportsPlugin = require("./wasm-sync/WasmFinalizeExportsPlugin");
  410. new WasmFinalizeExportsPlugin().apply(compiler);
  411. }
  412. const moduleIds = options.optimization.moduleIds;
  413. if (moduleIds) {
  414. switch (moduleIds) {
  415. case "natural": {
  416. const NaturalModuleIdsPlugin = require("./ids/NaturalModuleIdsPlugin");
  417. new NaturalModuleIdsPlugin().apply(compiler);
  418. break;
  419. }
  420. case "named": {
  421. const NamedModuleIdsPlugin = require("./ids/NamedModuleIdsPlugin");
  422. new NamedModuleIdsPlugin().apply(compiler);
  423. break;
  424. }
  425. case "hashed": {
  426. const WarnDeprecatedOptionPlugin = require("./WarnDeprecatedOptionPlugin");
  427. const HashedModuleIdsPlugin = require("./ids/HashedModuleIdsPlugin");
  428. new WarnDeprecatedOptionPlugin(
  429. "optimization.moduleIds",
  430. "hashed",
  431. "deterministic"
  432. ).apply(compiler);
  433. new HashedModuleIdsPlugin({
  434. hashFunction: options.output.hashFunction
  435. }).apply(compiler);
  436. break;
  437. }
  438. case "deterministic": {
  439. const DeterministicModuleIdsPlugin = require("./ids/DeterministicModuleIdsPlugin");
  440. new DeterministicModuleIdsPlugin().apply(compiler);
  441. break;
  442. }
  443. case "size": {
  444. const OccurrenceModuleIdsPlugin = require("./ids/OccurrenceModuleIdsPlugin");
  445. new OccurrenceModuleIdsPlugin({
  446. prioritiseInitial: true
  447. }).apply(compiler);
  448. break;
  449. }
  450. default:
  451. throw new Error(
  452. `webpack bug: moduleIds: ${moduleIds} is not implemented`
  453. );
  454. }
  455. }
  456. const chunkIds = options.optimization.chunkIds;
  457. if (chunkIds) {
  458. switch (chunkIds) {
  459. case "natural": {
  460. const NaturalChunkIdsPlugin = require("./ids/NaturalChunkIdsPlugin");
  461. new NaturalChunkIdsPlugin().apply(compiler);
  462. break;
  463. }
  464. case "named": {
  465. const NamedChunkIdsPlugin = require("./ids/NamedChunkIdsPlugin");
  466. new NamedChunkIdsPlugin().apply(compiler);
  467. break;
  468. }
  469. case "deterministic": {
  470. const DeterministicChunkIdsPlugin = require("./ids/DeterministicChunkIdsPlugin");
  471. new DeterministicChunkIdsPlugin().apply(compiler);
  472. break;
  473. }
  474. case "size": {
  475. //@ts-expect-error https://github.com/microsoft/TypeScript/issues/41697
  476. const OccurrenceChunkIdsPlugin = require("./ids/OccurrenceChunkIdsPlugin");
  477. new OccurrenceChunkIdsPlugin({
  478. prioritiseInitial: true
  479. }).apply(compiler);
  480. break;
  481. }
  482. case "total-size": {
  483. //@ts-expect-error https://github.com/microsoft/TypeScript/issues/41697
  484. const OccurrenceChunkIdsPlugin = require("./ids/OccurrenceChunkIdsPlugin");
  485. new OccurrenceChunkIdsPlugin({
  486. prioritiseInitial: false
  487. }).apply(compiler);
  488. break;
  489. }
  490. default:
  491. throw new Error(
  492. `webpack bug: chunkIds: ${chunkIds} is not implemented`
  493. );
  494. }
  495. }
  496. if (options.optimization.nodeEnv) {
  497. const DefinePlugin = require("./DefinePlugin");
  498. new DefinePlugin({
  499. "process.env.NODE_ENV": JSON.stringify(options.optimization.nodeEnv)
  500. }).apply(compiler);
  501. }
  502. if (options.optimization.minimize) {
  503. for (const minimizer of options.optimization.minimizer) {
  504. if (typeof minimizer === "function") {
  505. minimizer.call(compiler, compiler);
  506. } else if (minimizer !== "...") {
  507. minimizer.apply(compiler);
  508. }
  509. }
  510. }
  511. if (options.performance) {
  512. const SizeLimitsPlugin = require("./performance/SizeLimitsPlugin");
  513. new SizeLimitsPlugin(options.performance).apply(compiler);
  514. }
  515. new TemplatedPathPlugin().apply(compiler);
  516. new RecordIdsPlugin({
  517. portableIds: options.optimization.portableRecords
  518. }).apply(compiler);
  519. new WarnCaseSensitiveModulesPlugin().apply(compiler);
  520. const AddManagedPathsPlugin = require("./cache/AddManagedPathsPlugin");
  521. new AddManagedPathsPlugin(
  522. options.snapshot.managedPaths,
  523. options.snapshot.immutablePaths
  524. ).apply(compiler);
  525. if (options.cache && typeof options.cache === "object") {
  526. const cacheOptions = options.cache;
  527. switch (cacheOptions.type) {
  528. case "memory": {
  529. if (isFinite(cacheOptions.maxGenerations)) {
  530. //@ts-expect-error https://github.com/microsoft/TypeScript/issues/41697
  531. const MemoryWithGcCachePlugin = require("./cache/MemoryWithGcCachePlugin");
  532. new MemoryWithGcCachePlugin({
  533. maxGenerations: cacheOptions.maxGenerations
  534. }).apply(compiler);
  535. } else {
  536. //@ts-expect-error https://github.com/microsoft/TypeScript/issues/41697
  537. const MemoryCachePlugin = require("./cache/MemoryCachePlugin");
  538. new MemoryCachePlugin().apply(compiler);
  539. }
  540. if (cacheOptions.cacheUnaffected) {
  541. if (!options.experiments.cacheUnaffected) {
  542. throw new Error(
  543. "'cache.cacheUnaffected: true' is only allowed when 'experiments.cacheUnaffected' is enabled"
  544. );
  545. }
  546. compiler.moduleMemCaches = new Map();
  547. }
  548. break;
  549. }
  550. case "filesystem": {
  551. const AddBuildDependenciesPlugin = require("./cache/AddBuildDependenciesPlugin");
  552. for (const key in cacheOptions.buildDependencies) {
  553. const list = cacheOptions.buildDependencies[key];
  554. new AddBuildDependenciesPlugin(list).apply(compiler);
  555. }
  556. if (!isFinite(cacheOptions.maxMemoryGenerations)) {
  557. //@ts-expect-error https://github.com/microsoft/TypeScript/issues/41697
  558. const MemoryCachePlugin = require("./cache/MemoryCachePlugin");
  559. new MemoryCachePlugin().apply(compiler);
  560. } else if (cacheOptions.maxMemoryGenerations !== 0) {
  561. //@ts-expect-error https://github.com/microsoft/TypeScript/issues/41697
  562. const MemoryWithGcCachePlugin = require("./cache/MemoryWithGcCachePlugin");
  563. new MemoryWithGcCachePlugin({
  564. maxGenerations: cacheOptions.maxMemoryGenerations
  565. }).apply(compiler);
  566. }
  567. if (cacheOptions.memoryCacheUnaffected) {
  568. if (!options.experiments.cacheUnaffected) {
  569. throw new Error(
  570. "'cache.memoryCacheUnaffected: true' is only allowed when 'experiments.cacheUnaffected' is enabled"
  571. );
  572. }
  573. compiler.moduleMemCaches = new Map();
  574. }
  575. switch (cacheOptions.store) {
  576. case "pack": {
  577. const IdleFileCachePlugin = require("./cache/IdleFileCachePlugin");
  578. const PackFileCacheStrategy = require("./cache/PackFileCacheStrategy");
  579. new IdleFileCachePlugin(
  580. new PackFileCacheStrategy({
  581. compiler,
  582. fs: compiler.intermediateFileSystem,
  583. context: options.context,
  584. cacheLocation: cacheOptions.cacheLocation,
  585. version: cacheOptions.version,
  586. logger: compiler.getInfrastructureLogger(
  587. "webpack.cache.PackFileCacheStrategy"
  588. ),
  589. snapshot: options.snapshot,
  590. maxAge: cacheOptions.maxAge,
  591. profile: cacheOptions.profile,
  592. allowCollectingMemory: cacheOptions.allowCollectingMemory,
  593. compression: cacheOptions.compression
  594. }),
  595. cacheOptions.idleTimeout,
  596. cacheOptions.idleTimeoutForInitialStore,
  597. cacheOptions.idleTimeoutAfterLargeChanges
  598. ).apply(compiler);
  599. break;
  600. }
  601. default:
  602. throw new Error("Unhandled value for cache.store");
  603. }
  604. break;
  605. }
  606. default:
  607. // @ts-expect-error Property 'type' does not exist on type 'never'. ts(2339)
  608. throw new Error(`Unknown cache type ${cacheOptions.type}`);
  609. }
  610. }
  611. new ResolverCachePlugin().apply(compiler);
  612. if (options.ignoreWarnings && options.ignoreWarnings.length > 0) {
  613. const IgnoreWarningsPlugin = require("./IgnoreWarningsPlugin");
  614. new IgnoreWarningsPlugin(options.ignoreWarnings).apply(compiler);
  615. }
  616. compiler.hooks.afterPlugins.call(compiler);
  617. if (!compiler.inputFileSystem) {
  618. throw new Error("No input filesystem provided");
  619. }
  620. compiler.resolverFactory.hooks.resolveOptions
  621. .for("normal")
  622. .tap("WebpackOptionsApply", resolveOptions => {
  623. resolveOptions = cleverMerge(options.resolve, resolveOptions);
  624. resolveOptions.fileSystem = compiler.inputFileSystem;
  625. return resolveOptions;
  626. });
  627. compiler.resolverFactory.hooks.resolveOptions
  628. .for("context")
  629. .tap("WebpackOptionsApply", resolveOptions => {
  630. resolveOptions = cleverMerge(options.resolve, resolveOptions);
  631. resolveOptions.fileSystem = compiler.inputFileSystem;
  632. resolveOptions.resolveToContext = true;
  633. return resolveOptions;
  634. });
  635. compiler.resolverFactory.hooks.resolveOptions
  636. .for("loader")
  637. .tap("WebpackOptionsApply", resolveOptions => {
  638. resolveOptions = cleverMerge(options.resolveLoader, resolveOptions);
  639. resolveOptions.fileSystem = compiler.inputFileSystem;
  640. return resolveOptions;
  641. });
  642. compiler.hooks.afterResolvers.call(compiler);
  643. return options;
  644. }
  645. }
  646. module.exports = WebpackOptionsApply;