diff --git a/.editorconfig b/.editorconfig index 6e48422..93b895d 100644 --- a/.editorconfig +++ b/.editorconfig @@ -9,7 +9,7 @@ insert_final_newline = true indent_style = tab trim_trailing_whitespace = true -# YAML files require spaces -[*.{yaml,yml}] +# YAML files require spaces. JSON doesn't, but our file uses them. +[*.{yaml,yml,json}] indent_style = space indent_size = 2 diff --git a/.eslintignore b/.eslintignore index 537506f..5c7cc88 100644 --- a/.eslintignore +++ b/.eslintignore @@ -10,5 +10,4 @@ docs/ src/vendor # The test folder uses ES modules -test/src test/build diff --git a/.eslintrc.js b/.eslintrc.js index 5837717..560d2f6 100644 --- a/.eslintrc.js +++ b/.eslintrc.js @@ -105,7 +105,7 @@ module.exports = { 'new-parens': 'error', 'newline-after-var': 'off', 'newline-before-return': 'off', - 'newline-per-chained-call': 'error', + 'newline-per-chained-call': 'off', 'no-alert': 'error', 'no-array-constructor': 'error', 'no-async-promise-executor': 'error', @@ -127,7 +127,7 @@ module.exports = { 'no-extra-label': 'error', 'no-extra-parens': 'off', 'no-floating-decimal': 'error', - 'no-implicit-coercion': 'error', + 'no-implicit-coercion': 'off', 'no-implicit-globals': 'error', 'no-implied-eval': 'error', 'no-inline-comments': 'error', @@ -251,7 +251,10 @@ module.exports = { 'error', 'last' ], - 'sort-imports': 'error', + 'sort-imports': [ + 'error', + { memberSyntaxSortOrder: [ 'all', 'multiple', 'single', 'none' ] }, + ], 'sort-keys': 'off', 'sort-vars': 'error', 'space-before-blocks': 'error', diff --git a/.gitignore b/.gitignore index 6a28462..77e8e70 100644 --- a/.gitignore +++ b/.gitignore @@ -2,3 +2,5 @@ node_modules *.log package-lock.json test/build + +.DS_Store diff --git a/.travis.yml b/.travis.yml index 88bcb31..175810c 100644 --- a/.travis.yml +++ b/.travis.yml @@ -1,11 +1,12 @@ language: node_js script: "npm run lint && npm run test && npm run test-build" node_js: - - node - lts/* + - 16 + - 14 - 12 - - 10 branches: only: - main - beta + - v1 diff --git a/babel-preset.js b/babel-preset.js index ae44199..6028904 100644 --- a/babel-preset.js +++ b/babel-preset.js @@ -2,12 +2,7 @@ module.exports = ( api ) => { api.cache.forever(); return { - presets: [ '@wordpress/default' ], - plugins: [ - '@babel/plugin-proposal-class-properties', - [ 'transform-react-jsx', { - pragma: 'wp.element.createElement', - } ], - ], + presets: [ '@wordpress/babel-preset-default' ], + plugins: [ '@babel/plugin-proposal-class-properties' ], }; }; diff --git a/docs/changelog.md b/docs/changelog.md index 5b4df9b..c1687ab 100644 --- a/docs/changelog.md +++ b/docs/changelog.md @@ -6,8 +6,25 @@ nav_order: 10 # Changelog -## Next - +## v1.0 + +- **Breaking**: End support for Node v10. Node v12.13 or later is now required. +- **Breaking**: Switch to Webpack 5 and Webpack DevServer 4 +- **Breaking**: Include a contenthash string in default bundle file names. Set `filename: '[name].js'` in your output configuration to restore the old behavior. +- **Breaking**: Default DevServer manifest name is now `development-asset-manifest.json`, not `asset-manifest.json`. +- **Breaking**: Replace `filterLoaders` system with [individual hooks accesible via the new `addFilter` and `removeFilter` helpers](https://humanmade.github.io/webpack-helpers/modules/presets.html#customizing-presets). +- **Breaking**: Remove deprecated `eslint-loader` and add `eslint-webpack-plugin` to presets as `plugins.eslint()`. +- **Potentially Breaking**: Remove `loaders.url()` and `loaders.file()` in favor of Webpack 5 [`asset` modules](https://webpack.js.org/guides/asset-modules/), now usable by including `loaders.asset()` (for assets which can be inlined) and `loaders.resource()` (as a catch-all for other types) in your module rules list. Asset modules are handled automatically in both presets, so this is only breaking if `loaders.url()` or `loaders.file()` was used directly. +- A `webpack-manifest-plugin` instance is now automatically injected in development mode even if `publicPath` is not specified. +- A `webpack-bundle-analyzer` plugin is now automatically added to production builds when Webpack is invoked with the `--analyze` flag. +- Add the [`simple-build-report-webpack-plugin`](https://github.com/kadamwhite/simple-build-report-webpack-plugin) as `plugins.simpleBuildReport()` +- Include a `plugins.simpleBuildReport()` instance in production preset builds to improve legibility of Webpack console output. +- `plugins.fixStyleOnlyEntries()` now uses [`webpack-remove-empty-scripts`](https://github.com/webdiscus/webpack-remove-empty-scripts#webpack-remove-empty-scripts) instead of `webpack-fix-style-only-entries` due to Webpack 5 compatiblity issues with the original plugin. +- Allow `null` to be returned from an `addFilter` callback to skip a loader when using a configuration preset. +- Do not deep merge options passed to `plugins.terserPlugin()`: options object can now fully overwrite `terserOptions` property if needed. +- Permit filtering the Terser default configuration using `addFilter( 'plugins/terser/defaults', cb )`. +- Remove `OptimizeCssAssetsPlugin` (`plugins.optimizeCssAssets()`) in favor of Webpack 5-compatible [CssMinimizerPlugin](https://github.com/webpack-contrib/css-minimizer-webpack-plugin) (`plugins.cssMinimizer()`) +- Remove `plugins.hotModuleReplacement()`, which is now handled automatically by the DevServer in `hot` mode. - Include the `contenthash` in generated CSS filenames. [#204](https://github.com/humanmade/webpack-helpers/pull/204) ## v0.11.1 diff --git a/docs/guides/getting-started.md b/docs/guides/getting-started.md index 506d6c5..3aca816 100644 --- a/docs/guides/getting-started.md +++ b/docs/guides/getting-started.md @@ -264,7 +264,7 @@ To make use of this utility, we wrap our development config in a `choosePort` pr +] ); ``` -While these bundles are served from memory, now that we've specified a `publicPath`, a new file `asset-manifest.json` will be output into each project's `build/` folder. `themes/myproject/build/asset-manifest.json` will look like this: +While these bundles are served from memory, now that we've specified a `publicPath`, a new file `development-asset-manifest.json` will be output into each project's `build/` folder. `themes/myproject/build/development-asset-manifest.json` will look like this: ```json { "frontend.js": "http://localhost:8080/myproject-theme/frontend.js", @@ -286,8 +286,8 @@ We use the `cleanOnExit` helper to delete these files when the server shuts down + +// Clean up manifests on exit. +cleanOnExit( [ -+ filePath( 'mu-plugins/myproject-blocks/build/asset-manifest.json' ), -+ filePath( 'themes/myproject/build/asset-manifest.json' ), ++ filePath( 'mu-plugins/myproject-blocks/build/development-asset-manifest.json' ), ++ filePath( 'themes/myproject/build/development-asset-manifest.json' ), +] ); ``` diff --git a/docs/guides/upgrading-to-v1.md b/docs/guides/upgrading-to-v1.md new file mode 100644 index 0000000..60bdc3f --- /dev/null +++ b/docs/guides/upgrading-to-v1.md @@ -0,0 +1,70 @@ +--- +title: Upgrading to v1 +parent: Guides +nav_order: 1 +--- + +# Upgrading to v1 + +## Installation + +Install `@humanmade/webpack-helpers@latest` with [npm](http://npmjs.com), along with more updated versions of `webpack` and its attendant dependencies: + +```bash +npm install --save-dev @humanmade/webpack-helpers@latest webpack@5 webpack-cli@4 webpack-dev-server@4 +``` + +## Breaking Changes + +### Handling multi-config arrays in DevServer + +In a large project you may have a Webpack configuration that [exports multiple separate Webpack configuration objects](https://webpack.js.org/configuration/configuration-types/#exporting-multiple-configurations) in an array. The `presets.development()` helper adds a `devServer` property to each generated configuration, and with Webpack 4 and the pre-1.0 version of this helper library this was not a problem: if each entry in an exported array of configurations had its own `devServer` property, DevServer used the configuration in the first item in the exported array and ignored the rest. + +In Webpack 5 and the latest DevServer, however, it is an error to define `devServer` properties on any configuration but the first item in the exported array. This means that you may need to map over the exported configurations and unset the `devServer` property on subsequent configurations. + +```js +const configs = [ + presets.development( { /* ... */ } ), + presets.development( { /* ... */ } ), + presets.development( { /* ... */ } ), +]; +module.exports = configs.map( ( config, index ) => { + if ( index > 0 ) { + Reflect.deleteProperty( config, 'devServer' ); + } + return config; +} ); +``` + +### Goodbye `filterLoaders`, welcome `addFilter` + +Previously, a second argument could be passed to a preset to define a function used to filter loaders used while generating that preset configuration. In v1.0, a [hooks system](https://humanmade.github.io/webpack-helpers/reference/hooks.html) has been introduced which provides an `addFilter` method. `addFilter` can be used to register a callback function that may adjust the value of a loader globally, or make changes specific to a given preset call, in a manner similar to the WordPress PHP hooks system. Read the [hooks reference](https://humanmade.github.io/webpack-helpers/reference/hooks.html), [loaders](https://humanmade.github.io/webpack-helpers/modules/loaders.html), and [preset](https://humanmade.github.io/webpack-helpers/modules/presets.html) documentation pages for more information. + +### ESLint v6 and earlier not supported + +The ESLint integration for Webpack 5 requires ESLint v7 or later. If your project uses ESLint 6 or before, you will need to upgrade ESLint before using Webpack Helpers 1.0. + +```bash +npm install --save-dev eslint@7 +``` + +On most projects ESLint should continue to work after upgrade with no further adjustments. + +### Importing named properties from JSON + +Due to ongoing changes to how JSON files are handled by Webpack, this code + +```js +import { name } from './block.json'; +``` + +may now trigger the error + +``` +Should not import the named export 'name' (imported as 'name') from default-exporting module (only default export is available soon) +``` +This error means that your code above needs to be changed to this: +```js +import blockData from './block.json'; +const { name } = blockData; +``` diff --git a/docs/modules/helpers.md b/docs/modules/helpers.md index b0b0afc..f52aea3 100644 --- a/docs/modules/helpers.md +++ b/docs/modules/helpers.md @@ -33,15 +33,15 @@ module.exports = presets.production( { ## `cleanOnExit` -When using the `presets.development()` generator, an `asset-manifest.json` will automatically be generated so long as a `publicPath` URI can be determined. When working with an `asset-manifest.json` file, the `manifest` module provides a `cleanOnExit` method to easily remove manifests once the `webpack-dev-server` shuts down. +When using the `presets.development()` generator, a `development-asset-manifest.json` will automatically be generated so long as a `publicPath` URI can be determined. When working with a `development-asset-manifest.json` file, the `manifest` module provides a `cleanOnExit` method to easily remove manifests once the `webpack-dev-server` shuts down. ```js const { join } = require( 'path' ); const { helpers } = require( '@humanmade/webpack-helpers' ); helpers.cleanOnExit( [ - join( process.cwd(), 'content/mu-plugins/custom-blocks/build/asset-manifest.json' ), - join( process.cwd(), 'content/themes/my-theme/build/asset-manifest.json' ), + join( process.cwd(), 'content/mu-plugins/custom-blocks/build/development-asset-manifest.json' ), + join( process.cwd(), 'content/themes/my-theme/build/development-asset-manifest.json' ), ] ); ``` @@ -98,3 +98,69 @@ module.exports = helpers.choosePort( 9090 ).then( port => [ } ), ] ); ``` + +## Filters + +Webpack Helpers v1.0 introduced a new system for altering the behavior of bundled loaders, plugins, and presets, also covered under [Customizing Presets](./presets.html#customizing-presets). This system uses a filtering approach similar to [the way PHP filters work within WordPress](https://developer.wordpress.org/reference/functions/add_filter/), with `addFilter` and `removeFilter` helpers you can use to modify various types of internal data at runtime. + +```js +// webpack.config.js +const { helpers } = require( '@humanmade/webpack-helpers' ); + +const { addFilter, removeFilter } = helpers; + +// Hook onto a specific filter by name, and provide a callback function to +// alter the value returned from that filter. +addFilter( 'filter/name', function( value ) { + return 'modified value'; +} ); + +// To filter only some values, you can remove a filter by passing the same +// function reference to removeFilter. +const filterFunction = ( value ) => 'modified value'; + +addFilter( 'filter/name', filterFunction ); + +// Do something where the filter gets applied. + +removeFilter( 'filter/name', filterFunction ); + +// Now if you do that same thing, the filter will no longer apply. +``` + +### Filter List + +Each [loader](./loaders.html) exposes at minimum two filters, `loaders/{name}` and `loaders/{name}/defaults`. For example, the defaults for `loaders.ts()` can be filtered using + +```js +addFilter( 'loaders/ts/defaults', ( loaderDefaultsObject ) => { + // return a filtered value +} ); +``` +or the computed final loader object can be modified after the fact with + +```js +addFilter( 'loaders/ts', ( loaderObject ) => { + // return a filtered value +} ); +``` + +If you return `null` from the `loaders/{name}` filter, it will remove that loader from the preset entirely. + +Additional filters provided by Webpack Helpers: + +**`presets/stylesheet-loaders`** + +Filter the [stylesheet loader chain](https://webpack.js.org/concepts/loaders/#configuration) used in the `presets.production()` and `presets.development()` helpers. Functions added to this filter receive the stylesheet chain as their first argument, and the name of the compilation mode (`production` or `development`) as the second argument. + +**`plugins/terser/defaults`** + +Filter the [TerserPlugin default options object](https://webpack.js.org/plugins/terser-webpack-plugin/#options) if you wish to alter the minification settings used by the production preset. + +**`loaders/postcss/plugins`** + +Filter the list of [PostCSS plugins](https://github.com/postcss/postcss/blob/main/docs/plugins.md) used by the `loaders.postcss()` loader. + +**`loaders/postcss/preset-env`** + +Filter the [`postcss-preset-env` options](https://github.com/csstools/postcss-preset-env#options) used by the `loaders.postcss()` loader. diff --git a/docs/modules/loaders.md b/docs/modules/loaders.md index f036aa2..3dd58de 100644 --- a/docs/modules/loaders.md +++ b/docs/modules/loaders.md @@ -6,19 +6,23 @@ nav_order: 3 # Loaders Module -`const { loaders } = require( '@humanmade/webpack-helpers' );` +```js +const { loaders } = require( '@humanmade/webpack-helpers' ); +``` This module provides functions that generate configurations for commonly-needed Webpack loaders. Use them within the `.module.rules` array, or use `presets.development()`/`presets.production()` to opt-in to some opinionated defaults. -- `loaders.eslint()`: Return a configured Webpack module loader rule for `eslint-loader`. -- `loaders.js()`: Return a configured Webpack module loader rule for `js-loader`. -- `loaders.ts()`: Return a configured Webpack module loader rule for `ts-loader`. -- `loaders.url()`: Return a configured Webpack module loader rule for `url-loader`. -- `loaders.style()`: Return a configured Webpack module loader rule for `style-loader`. +- `loaders.asset()`: Return a configured Webpack module loader rule for [`asset` modules](https://webpack.js.org/guides/asset-modules/#inlining-assets) which will be inlined when small enough. - `loaders.css()`: Return a configured Webpack module loader rule for `css-loader`. +- `loaders.js()`: Return a configured Webpack module loader rule for `js-loader`. - `loaders.postcss()`: Return a configured Webpack module loader rule for `postcss-loader`. +- `loaders.resource()`: Return a configured Webpack module loader rule for [`asset/resource` modules](https://webpack.js.org/guides/asset-modules/#resource-assets). - `loaders.sass()`: Return a configured Webpack module loader rule for `sass-loader`. -- `loaders.file()`: Return a configured Webpack module loader rule for `file-loader`. +- `loaders.sourcemap()`: Return a configured Webpack module loader rule for `source-map-loader`. +- `loaders.style()`: Return a configured Webpack module loader rule for `style-loader`. +- `loaders.ts()`: Return a configured Webpack module loader rule for `ts-loader`. + +The output from these loaders can optionally be [filtered](https://humanmade.github.io/webpack-helpers/reference/hooks.html). ## Customizing Loaders @@ -41,16 +45,25 @@ module.exports = { }; ``` -To alter the configuration for a loader prior to use within a preset, you may mutate the `.defaults` property on the loader method. +To alter the configuration for a loader prior to use within a preset, you may either [filter](https://humanmade.github.io/webpack-helpers/reference/hooks.html) the default loader options, or the final merged loader configuration. ```js const { helpers, loaders, presets } = require( '@humanmade/webpack-helpers' ); +const { addFilter } = helpers; + +// Adjust the loader defaults. +addFilter( 'loaders/js/defaults', ( defaults ) => ( { + ...defaults, + include: helpers.filePath( 'themes/my-theme/src' ), +} ) ); -// Mutate the loader defaults. -loaders.js.defaults.include = helpers.filePath( 'themes/my-theme/src' ); -loaders.css.defaults.options.url = false; +addFilter( 'loaders/css/defaults', ( defaults ) => { + defaults.options.url = false; + return defaults; +} ); +// The above customizations will now apply to all calls to loader or preset factories. module.exports = presets.development( { /* ... */ } ); ``` -These loaders are also used by the [presets](https://humanmade.github.io/webpack-helpers/modules/presets.html) methods described above. To adjust the behavior of a loader for a specific configuration generated using a preset, you may pass a second argument to the preset defining a filter function which can modify loader options as they are computed. See ["Customizing Presets"](https://humanmade.github.io/webpack-helpers/modules/presets.html#customizing-presets) for more information. +These loaders are also used by the [presets](https://humanmade.github.io/webpack-helpers/modules/presets.html) methods described above. To adjust the behavior of a loader for a specific configuration generated using a preset, see ["Customizing Presets"](https://humanmade.github.io/webpack-helpers/modules/presets.html#customizing-presets). diff --git a/docs/modules/plugins.md b/docs/modules/plugins.md index ce31a83..7ef80f0 100644 --- a/docs/modules/plugins.md +++ b/docs/modules/plugins.md @@ -12,16 +12,15 @@ This module provides methods which create new instances of commonly-needed Webpa | Plugin | Description ------ | ------ | ------------ - | `plugins.bundleAnalyzer()` | Create and return a new [`webpack-bundle-analyzer`](https://github.com/webpack-contrib/webpack-bundle-analyzer) instance. When included this plugin is disabled by default unless the `--analyze` flag is passed on the command line. + | `plugins.bundleAnalyzer()` | Create and return a new [`webpack-bundle-analyzer`](https://github.com/webpack-contrib/webpack-bundle-analyzer) instance. This plugin is included in production preset builds automatically when the `--analyze` flag is passed on the command line. | `plugins.clean()` | Create and return a new [`clean-webpack-plugin`](https://github.com/johnagan/clean-webpack-plugin) instance. | `plugins.copy()` | Create and return a new [`copy-webpack-plugin`](https://github.com/webpack-contrib/copy-webpack-plugin) instance. | `plugins.errorBell()` | Create and return a new [`bell-on-bundle-error-plugin`](https://www.npmjs.com/package/bell-on-bundler-error-plugin) instance. - | `plugins.fixStyleOnlyEntries()` | Create and return a [`webpack-fix-style-only-entries`](https://github.com/fqborges/webpack-fix-style-only-entries) instance to remove empty JS bundles for style-only entrypoints. -**D** | `plugins.hotModuleReplacement()` | Create and return a new [`webpack.HotModuleReplacementPlugin`](https://webpack.js.org/plugins/hot-module-replacement-plugin/) instance. + | `plugins.eslint()` | Create and return a [`eslint-webpack-plugin`](https://webpack.js.org/plugins/eslint-webpack-plugin/) instance. + | `plugins.fixStyleOnlyEntries()` | Create and return a [`webpack-remove-empty-scripts`](https://github.com/webdiscus/webpack-remove-empty-scripts) instance (forked from the non-Webpack 5-compatible `webpack-fix-style-only-entries`) to remove empty JS bundles for style-only entrypoints. | `plugins.manifest()` | : Create and return a new [`webpack-manifest-plugin`](https://github.com/danethurber/webpack-manifest-plugin) instance, preconfigured to write the manifest file while running from a dev server. **P** | `plugins.miniCssExtract()` | Create and return a new [`mini-css-extract-plugin`](https://github.com/webpack-contrib/mini-css-extract-plugin) instance. **P** | `plugins.terser()` | Create and return a new [`terser-webpack-plugin`](https://github.com/webpack-contrib/terser-webpack-plugin) instance, preconfigured with defaults based on `create-react-app`. +**P** | `plugins.cssMinimizer()` | Create and return a new [`css-minimizer-webpack-plugin`](https://webpack.js.org/plugins/css-minimizer-webpack-plugin/) instance. **P**: Included in `presets.production()` - -**D**: Included in `presets.development()` diff --git a/docs/modules/presets.md b/docs/modules/presets.md index 205d671..c7fb68f 100644 --- a/docs/modules/presets.md +++ b/docs/modules/presets.md @@ -91,7 +91,6 @@ module.exports = { [Read up on the `...` "spread operator"](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Operators/Spread_syntax#Spread_in_object_literals) if this syntax is unfamiliar to you; it's quite useful! - Customize a generated production configuration by object mutation: ```js @@ -109,27 +108,57 @@ module.exports.optimization.minimizer = [ ]; ``` -Note that array values are _merged_, not overwritten. This allows you to easily add plugins, but it can make it hard to _remove_ values from array properties like the module loader rules. To change loader configuration options without completely removing a loader, we recommend the approach described below in "Customizing Loaders". However, if you do need to completely change or remove the loaders from a default, you may overwrite the `.module.rules` array using one of the above methods. - -### Modify loaders within a generated configuration +### Filtering values -Adjusting loaders within a generated configuration tree is difficult because loader arrays are not keyed and module rules may be nested. Instead, each `preset` generator accepts a second argument in which you can pass a callback function that will be run on the output of each computed loader definition. +Array values are _merged_ when processing a preset, not overwritten. This allows you to easily add plugins, but it can make it hard to _remove_ values from array properties like the module loader rules or plugins list. To change loader configuration options without completely removing a loader, or adjust loaders deep within a generated configuration tree, this library provides a `addFilter` helper function. `addFilter()` lets you register a callback which can transform the value of each computed loader, as well as some other useful configuration values. ```js -// Alter the publicPath value of the files-loader and url-loader. -const config = production.preset( - { /* ...configuration options described above ... */ }, - { - filterLoaders: ( loader, loaderType ) => { - if ( loaderType === 'file' || loaderType === 'url' ) { - loader.options.publicPath = '../../'; - } - return loader; - } - } +const { helpers, presets } = require( '@humanmade/webpack-helpers' ); +const { addFilter } = helpers; + +// Intercept the "sass" loader and replace it with a Stylus loader. +addFilter( 'loaders/sass', () => { + return { + loader: require.resolve( 'stylus-loader' ), + }; +} ); +// Alter presets to use a different targeting regular expression. +addFilter( 'presets/stylesheet-loaders', ( loader ) => { + loader.test = /.\styl$/; + return loader; +} ); + +// Now, use the preset as normal and it will pick up Stylus files instead! +const config = presets.production( + { /* ...normal configuration options as described above ... */ } ); ``` -The `loaderType` option will be [one of the keys in the `loaders` object](./loaders.html), or else the special key `stylesheet` which will be passed alongside the entire computed stylesheet loader chain. Filter on `stylesheet` to, for example, completely replace the Sass stylesheet support with a different preprocessor like Stylus. +If you are using a multi-configuration Webpack setup where the config file exports an array of individual configurations, you may wish to filter a loader, preset chain, or other value for only one of those configuration objects. To permit this, each filter callback will be passed the preset's configuration object as a final argument when the filter or loader is called in the context of a preset. For example, + +```js +const { helpers, externals, presets } = require( '@humanmade/webpack-helpers' ); +const { addFilter } = helpers; + +// Do not use the JS loader in the frontend build. +addFilter( 'presets/js', ( loader, config ) => { + if ( config.name === 'frontend' ) { + return null; + } + return frontend; +} ); + +module.exports = [ + presets.production( { + name: 'frontend', + // ... + } ), + presets.production( { + name: 'editor', + externals, + // ... + } ), +]; +``` -The values of the `loaderType` argument in this callback correspond to the names of the available [loader factory functions](https://humanmade.github.io/webpack-helpers/modules/loaders.html). +For more information, see [the full hooks reference.](https://humanmade.github.io/webpack-helpers/reference/hooks.html). diff --git a/docs/reference.md b/docs/reference.md new file mode 100644 index 0000000..deb8d0e --- /dev/null +++ b/docs/reference.md @@ -0,0 +1,10 @@ +--- +title: Reference +has_children: true +permalink: /reference +nav_order: 3 +--- + +# Reference Documentation + +Code-level documentation of specific components of `@humanmade/webpack-helpers`. diff --git a/docs/reference/hooks.md b/docs/reference/hooks.md new file mode 100644 index 0000000..71cbf33 --- /dev/null +++ b/docs/reference/hooks.md @@ -0,0 +1,139 @@ +--- +title: Filter Hooks +parent: Reference +nav_order: 1 +--- + +# Filter Hooks + +These hooks are provided by Webpack Helpers to customize the behavior of a specific loader or preset within your project. + +`{loader name}` in the below filters may be any slug for an available [loader](https://humanmade.github.io/webpack-helpers/modules/loaders.html): `asset`, `css`, `js`, `postcss`, `resource`, `sass`, `sourcemap`, `style`, and `ts`. + +Remember to always return a value from a filter callback. + +## `loaders/{loader name}` + +Filter the full loader configuration object after merging any user-provided options with the (filtered) defaults. + +When called in the context of a preset, the callback will receive the preset factory's configuration object as a second argument. + +**Arguments:** + + name | type | description +----- | ---- | ------ +`loader` | `Object` | Loader configuration object. +`config` | `Object` or `null` | Preset configuration object if a preset is being rendered, else `null`. + +Return `null` from the callback to remove this loader from preset-generated configurations. + +```js +addFilter( 'loaders/{loader name}', ( loader, config = null ) => { + // Filter the loader definition globally or based on the specific + // configuration options provided to a preset factory. + return loader; +} ); +``` + +## `loaders/{loader name}/defaults` + +Adjust the default values used to define a given loader configuration. + +When called in the context of a preset, the callback will receive the preset factory's configuration object as a second argument. + +**Arguments:** + + name | type | description +----- | ---- | ------ +`defaults` | `Object` | Defaults for this specific loader. +`config` | `Object` or `null` | Preset configuration object if a preset is being rendered, else `null`. + +```js +addFilter( 'loaders/{loader name}/defaults', ( defaults, config = null ) => { + // Filter the loader's defaults globally or based on the specific + // configuration options provided to a preset factory. + return defaults; +} ); +``` + +## `presets/stylesheet-loaders` + +Filter the loaders used to process stylesheet imports when building your project. + +The callback will receive the environment type of the preset being rendered as its second argument, and the preset's configuration object argument as a third argument. + +**Arguments:** + + name | type | description +----- | ---- | ------ +`loader` | `Object` | Combined stylesheet loader rule. +`environment` | `string` | `'development'` or `'production'`. +`config` | `Object` | Preset configuration object. + +```js +addFilter( 'presets/stylesheet-loaders', ( loader, environment, config ) => { + // Filter the configured stylesheet loaders based on environment or the + // specific configuration options provided to the preset factory. + return loader; +} ); +``` + +## `presets/manifest-options` + +Filter the [`webpack-manifest-plugin` options](https://github.com/shellscape/webpack-manifest-plugin#options) used when auto-injecting asset manifest plugins. + +The callback will receive the environment type of the preset being rendered as its second argument, and the preset's configuration object argument as a third argument. + +**Arguments:** + + name | type | description +----- | ---- | ------ +`options` | `Object` | Manifest plugin options object. +`environment` | `string` | `'development'` or `'production'`. +`config` | `Object` | Preset configuration object. + +```js +addFilter( 'presets/manifest-options', ( loader, environment, config ) => { + // Filter the configured stylesheet loaders based on environment or the + // specific configuration options provided to the preset factory. + return loader; +} ); +``` + +## `loaders/postcss/plugins` + +Filter the default list of PostCSS plugins used by the PostCSS loader. + +Unlike the hooks above, callbacks added to this filter do not receive the preset configuration object. + +**Arguments:** + + name | type | description +----- | ---- | ------ +`pluginsArray` | `Array` | Array of [PostCSS plugin](https://github.com/postcss/postcss#plugins) definitions. + +```js +addFilter( 'loaders/postcss/plugins', ( pluginsArray ) => { + // Filter the plugins loaded by PostCSS. + return pluginsArray; +} ); +``` + +## `loaders/postcss/preset-env` + +Filter the [`postcss-preset-env` plugin's](https://github.com/csstools/postcss-plugins/tree/main/plugin-packs/postcss-preset-env#readme) configuration object. + +Unlike the hooks above, callbacks added to this filter do not receive the preset configuration object. + +**Arguments:** + + name | type | description +----- | ---- | ------ +`pluginOptions` | `Object` | `postcss-preset-env` plugin options object. + +```js +addFilter( 'loaders/postcss/preset-env', ( pluginOptions ) => { + // Filter the PostCSS Preset Env plugin options. + return pluginOptions; +} ); +``` diff --git a/index.js b/index.js index f8bf15c..429f333 100644 --- a/index.js +++ b/index.js @@ -1,3 +1,5 @@ +const filters = require( './src/helpers/filters' ); + /** * Expose public package API. */ @@ -6,6 +8,8 @@ module.exports = { config: require( './src/config' ), externals: require( './src/externals' ), helpers: { + addFilter: filters.addFilter, + removeFilter: filters.removeFilter, choosePort: require( './src/helpers/choose-port' ), cleanOnExit: require( './src/helpers/clean-on-exit' ), filePath: require( './src/helpers/file-path' ), diff --git a/package.json b/package.json index 0ee394b..3ff1d3a 100644 --- a/package.json +++ b/package.json @@ -1,7 +1,7 @@ { "name": "@humanmade/webpack-helpers", "public": true, - "version": "0.11.1", + "version": "1.0.0-beta.17", "description": "Reusable Webpack configuration components & related helper utilities.", "main": "index.js", "author": "Human Made", @@ -18,7 +18,7 @@ "webpack" ], "scripts": { - "lint": "eslint .", + "lint": "eslint . --ignore-pattern linting-test.js", "test": "jest", "test-build": "rm -rf test/build && webpack --config=test/test-config.js", "test-dev-server": "webpack-dev-server --config=test/test-config.js" @@ -29,53 +29,53 @@ ] }, "devDependencies": { - "eslint": "^7.17.0", + "eslint": "^7.32.0", "jest": "^26.6.3", - "sass": "^1.32.4", - "typescript": "^4.0.2", - "webpack": "^4.46.0", - "webpack-cli": "^3.3.12", - "webpack-dev-server": "^3.11.2" + "sass": "^1.51.0", + "typescript": "^4.6.3", + "webpack": "^5.72.0", + "webpack-cli": "^4.9.2", + "webpack-dev-server": "^4.8.1" }, "dependencies": { - "@babel/core": "^7.10.3", - "@babel/plugin-proposal-class-properties": "^7.10.1", - "@wordpress/babel-preset-default": "^4.16.0", - "babel-loader": "^8.0.6", - "babel-plugin-transform-react-jsx": "^6.24.1", + "@babel/core": "^7.17.9", + "@babel/plugin-proposal-class-properties": "^7.16.7", + "@wordpress/babel-preset-default": "^6.9.0", + "babel-loader": "^8.2.5", "bell-on-bundler-error-plugin": "^2.0.0", - "block-editor-hmr": "^0.6.1", - "chalk": "^4.1.0", - "clean-webpack-plugin": "^3.0.0", - "copy-webpack-plugin": "^6.0.2", - "css-loader": "^5.0.1", + "block-editor-hmr": "^0.7.0", + "chalk": "^4.1.2", + "clean-webpack-plugin": "^4.0.0", + "copy-webpack-plugin": "^10.2.4", + "css-loader": "^6.7.1", + "css-minimizer-webpack-plugin": "^3.4.1", "detect-port-alt": "^1.1.6", - "eslint-loader": "^4.0.2", - "file-loader": "^6.0.0", - "inquirer": "^7.2.0", + "eslint-webpack-plugin": "^3.1.1", "is-root": "^2.1.0", - "mini-css-extract-plugin": "^1.3.4", - "optimize-css-assets-webpack-plugin": "^5.0.1", - "postcss": "^8.2.4", + "lodash.clonedeep": "^4.5.0", + "mini-css-extract-plugin": "^2.6.0", + "postcss": "^8.4.12", "postcss-flexbugs-fixes": "^5.0.2", - "postcss-loader": "^4.1.0", - "postcss-preset-env": "^6.7.0", - "run-parallel": "1.1.9", - "sass-loader": "^10.0.2", - "signal-exit": "^3.0.2", - "style-loader": "^2.0.0", - "terser-webpack-plugin": "^4.2.0", - "ts-loader": "^8.0.1", - "url-loader": "^4.1.0", - "webpack-bundle-analyzer": "^4.3.0", - "webpack-fix-style-only-entries": "^0.6.0", - "webpack-manifest-plugin": "^3.0.0" + "postcss-loader": "^6.2.1", + "postcss-preset-env": "^7.4.4", + "prompts": "^2.4.2", + "run-parallel": "^1.2.0", + "sass-loader": "^12.6.0", + "signal-exit": "^3.0.7", + "simple-build-report-webpack-plugin": "^1.0.2", + "source-map-loader": "^3.0.1", + "style-loader": "^3.3.1", + "terser-webpack-plugin": "^5.3.1", + "ts-loader": "^9.2.9", + "webpack-bundle-analyzer": "^4.5.0", + "webpack-manifest-plugin": "^5.0.0", + "webpack-remove-empty-scripts": "^0.8.0" }, "peerDependencies": { "sass": "*", "typescript": "*", - "webpack": "^4.0.0", - "webpack-cli": "^3.0.0", - "webpack-dev-server": "^3.0.0" + "webpack": "^5.0.0", + "webpack-cli": "^4.0.0", + "webpack-dev-server": "^4.0.0" } } diff --git a/src/config.js b/src/config.js index 144df73..f117d72 100644 --- a/src/config.js +++ b/src/config.js @@ -7,13 +7,21 @@ module.exports = { * @type {Object} */ devServer: { - disableHostCheck: true, + allowedHosts: 'all', headers: { 'Access-Control-Allow-Origin': '*', + 'Access-Control-Allow-Methods': '*', + 'Access-Control-Allow-Headers': '*', }, - hotOnly: true, - watchOptions: { - aggregateTimeout: 300, + // Enable gzip compression of generated files. + compress: true, + hot: 'only', + client: { + // Do not show disruptive overlay for warnings. + overlay: { + errors: true, + warnings: false, + }, }, }, @@ -22,12 +30,10 @@ module.exports = { * @type {Object} */ stats: { - all: false, + preset: 'summary', assets: true, colors: true, errors: true, - performance: true, - timings: true, warnings: true, }, }; diff --git a/src/helpers/camel-case-dash.js b/src/helpers/camel-case-dash.js index 11b3ffd..87a6b41 100644 --- a/src/helpers/camel-case-dash.js +++ b/src/helpers/camel-case-dash.js @@ -2,7 +2,7 @@ * Given a kebab-case string, returns a new camelCase string. * * @param {string} string Input kebab-case string. - * @return {string} Camel-cased string. + * @returns {string} Camel-cased string. */ module.exports = string => string.replace( /-([a-z])/g, diff --git a/src/helpers/choose-port.js b/src/helpers/choose-port.js index d4c02de..ce1b826 100644 --- a/src/helpers/choose-port.js +++ b/src/helpers/choose-port.js @@ -8,7 +8,7 @@ * */ const chalk = require( 'chalk' ); -const inquirer = require( 'inquirer' ); +const prompts = require( 'prompts' ); const detect = require( 'detect-port-alt' ); const isRoot = require( 'is-root' ); @@ -17,6 +17,8 @@ const getProcessForPort = require( '../vendor/get-process-for-port' ); const DEFAULT_PORT = process.env.PORT || 8080; const HOST = process.env.HOST || '0.0.0.0'; +const isInteractive = process.stdout.isTTY; + /** * choosePort method adapted from create-react-app's `react-dev-utils`. * @@ -33,25 +35,29 @@ const choosePort = ( host, defaultPort ) => detect( defaultPort, host ).then( const message = process.platform !== 'win32' && defaultPort < 1024 && ! isRoot() ? 'Admin permissions are required to run a server on a port below 1024.' : `Something is already running on port ${ defaultPort }.`; - // clearConsole(); - const existingProcess = getProcessForPort( defaultPort ); - const question = { - type: 'confirm', - name: 'shouldChangePort', - message: `${ - chalk.yellow( `${ message }${ - existingProcess ? ` Probably:\n ${ existingProcess }` : '' - }` ) - }\n\nWould you like to run the app on another port instead?`, - default: true, - }; - inquirer.prompt( question ).then( answer => { - if ( answer.shouldChangePort ) { - resolve( port ); - } else { - resolve( null ); - } - } ); + if ( isInteractive ) { + const existingProcess = getProcessForPort( defaultPort ); + const question = { + type: 'confirm', + name: 'shouldChangePort', + message: `${ + chalk.yellow( `${ message }${ + existingProcess ? ` Probably:\n ${ existingProcess }` : '' + }` ) + }\n\nWould you like to run the app on another port instead?`, + initial: true, + }; + prompts( question ).then( answer => { + if ( answer.shouldChangePort ) { + resolve( port ); + } else { + resolve( null ); + } + } ); + } else { + console.log( chalk.red( message ) ); + resolve( null ); + } } ), err => { throw new Error( `${ diff --git a/src/helpers/deep-merge.js b/src/helpers/deep-merge.js index cf0f3a5..586bcbd 100644 --- a/src/helpers/deep-merge.js +++ b/src/helpers/deep-merge.js @@ -12,7 +12,7 @@ const isArr = val => Array.isArray( val ); * @param {*} val A value to test for object-ness. * @returns {Boolean} Whether the provided argument is an object. */ -const isObj = val => ( typeof val === 'object' ); +const isObj = val => ( typeof val === 'object' && val !== null ); /** * Given two objects, merge array and object properties, then allow scalar diff --git a/src/helpers/deep-merge.test.js b/src/helpers/deep-merge.test.js index 1ec25dc..2867239 100644 --- a/src/helpers/deep-merge.test.js +++ b/src/helpers/deep-merge.test.js @@ -35,4 +35,19 @@ describe( 'helpers/deep-merge', () => { }, } ); } ); + + it( 'can remove a property by specifying `undefined` in the second merged object', () => { + const obj1 = { + test: '.jsx?$', + devServer: { + port: 9191, + }, + }; + const obj2 = { + devServer: undefined, + }; + expect( deepMerge( obj1, obj2 ) ).toEqual( { + test: '.jsx?$', + } ); + } ); } ); diff --git a/src/helpers/filters.js b/src/helpers/filters.js new file mode 100644 index 0000000..b43a220 --- /dev/null +++ b/src/helpers/filters.js @@ -0,0 +1,102 @@ +const registry = {}; + +/** + * Internal helper to return an ordered array of callbacks for a given priority. + * + * @private + * @param {string} hookName A hook name for which to return filters. + * @returns {Function[]} Array of filter functions. + */ +const getCallbacks = ( hookName ) => { + const priorities = Object.values( registry[ hookName ] || {} ) + // eslint-disable-next-line id-length + .sort( ( a, b ) => ( +a < +b ? -1 : 1 ) ); + return priorities.reduce( + ( callbacks, priority ) => { + return callbacks.concat( priority || [] ); + }, + [] + ); +}; + +/** + * Internal helper to set the registry to a specific state for testing. + * + * @private + * @param {Object} values New registry shape. + */ +const setupRegistry = ( values = {} ) => { + Object.keys( registry ).forEach( ( existingKey ) => { + Reflect.deleteProperty( registry, existingKey ); + } ); + + Object.keys( values ).forEach( ( newKey ) => { + registry[ newKey ] = values[ newKey ]; + } ); +}; + +/** + * Register a filter function for a given filter hook name. + * + * @param {string} hookName Name of hook to add a filter on. + * @param {Function} callback Callback to run on this hook. + * @param {Number} [priority] Optional numeric priority, default 10. + */ +const addFilter = ( hookName, callback, priority = 10 ) => { + if ( ! registry[ hookName ] ) { + registry[ hookName ] = {}; + } + if ( ! Array.isArray( registry[ hookName ][ priority ] ) ) { + registry[ hookName ][ priority ] = []; + } + registry[ hookName ][ priority ].push( callback ); +}; + +/** + * Remove a previously-added filter function for a given hook name. + * + * To remove a hook, the callback and priority arguments must match when the hook was added. + * + * @param {string} hookName Name of hook to add a filter on. + * @param {Function} callback Callback to run on this hook. + * @param {Number} [priority] Priority from which to remove the hook. + */ +const removeFilter = ( hookName, callback, priority = 10 ) => { + if ( ! registry[ hookName ] || ! registry[ hookName ][ priority ] ) { + // Not added, no action needed. + return; + } + registry[ hookName ][ priority ] = registry[ hookName ][ priority ] + .filter( ( hook ) => hook !== callback ); +}; + +/** + * Run input arguments through all registered callbacks for a specified filter, + * allowing each filter to transform the first argument. + * + * @param {string} hookName Name of hook to filter on. + * @param {...any} args Filter input arguments. + * @returns {any} Output of filter chain. + */ +const applyFilters = ( hookName, ...args ) => { + const callbacks = getCallbacks( hookName ); + + const [ firstArg, ...otherArgs ] = args; + + return callbacks.reduce( + ( val, callback ) => callback( val, ...otherArgs ), + firstArg, + ); +}; + +module.exports = { + addFilter, + removeFilter, + applyFilters, +}; + +if ( process.env.JEST_WORKER_ID ) { + // Exposed only for testing purposes. + module.exports.getCallbacks = getCallbacks; + module.exports.setupRegistry = setupRegistry; +} diff --git a/src/helpers/filters.test.js b/src/helpers/filters.test.js new file mode 100644 index 0000000..09fce2b --- /dev/null +++ b/src/helpers/filters.test.js @@ -0,0 +1,142 @@ +const { + getCallbacks, + setupRegistry, + addFilter, + removeFilter, + applyFilters, +} = require( './filters' ); + +// Functions to use when testing filtering. +const toKebabCase = ( str ) => str.split( /[_ ]+/g ).join( '-' ); +const toLowerCase = ( str ) => str.toLowerCase(); +const toUpperCase = ( str ) => str.toUpperCase(); +const reverse = ( str ) => str.split( '' ).reverse().join( '' ); +const reverseWords = ( str ) => str.split( ' ' ).reverse().join( ' ' ); + +describe( 'filters', () => { + describe( '.getCallbacks() [internal]', () => { + + it( 'is a function', () => { + expect( getCallbacks ).toBeInstanceOf( Function ); + } ); + + it( 'returns callbacks for the requested hook', () => { + setupRegistry( { + 'loaders/js': { + 10: [ + toKebabCase, + toLowerCase, + ], + }, + 'loaders/css': { + 10: [ + toUpperCase, + ], + }, + } ); + expect( getCallbacks( 'loaders/js' ) ).toEqual( [ + toKebabCase, + toLowerCase, + ] ); + expect( getCallbacks( 'loaders/css' ) ).toEqual( [ + toUpperCase, + ] ); + } ); + + it( 'returns callbacks in priority order', () => { + setupRegistry( { + 'loaders/js': { + 10: [ 'a', 'b' ], + 3: [ 'c' ], + 22: [ 'd' ], + 7: [ 'e', 'f' ], + }, + } ); + const hooks = getCallbacks( 'loaders/js' ); + expect( hooks ).toEqual( [ 'c', 'e', 'f', 'a', 'b', 'd' ] ); + } ); + + it( 'returns empty array for uninitialized hook', () => { + setupRegistry(); + const hooks = getCallbacks( 'unused/hook' ); + expect( hooks ).toEqual( [] ); + } ); + } ); + + describe( 'addFilter', () => { + beforeEach( () => { + setupRegistry(); + } ); + + it( 'adds a callback for the specified filter', () => { + const hook = () => 'Woo'; + addFilter( 'loaders/js', hook ); + expect( getCallbacks( 'loaders/js' ) ).toEqual( [ hook ] ); + } ); + + it( 'adds callbacks with the specified priority', () => { + const hook = () => 'Woo'; + const hook2 = () => 'Woo-er'; + const hook3 = () => 'Woo-est'; + addFilter( 'loaders/js', hook2, 9 ); + addFilter( 'loaders/js', hook3, 11 ); + addFilter( 'loaders/js', hook ); + expect( getCallbacks( 'loaders/js' ) ).toEqual( [ hook2, hook, hook3 ] ); + } ); + } ); + + describe( 'removeFilter', () => { + beforeEach( () => { + setupRegistry(); + } ); + + it( 'removes a previously-added callback for the specified filter', () => { + const hook = () => 'Woo'; + addFilter( 'loaders/js', hook ); + removeFilter( 'loaders/js', hook ); + expect( getCallbacks( 'loaders/js' ) ).toEqual( [] ); + } ); + + it( 'only removes the requested callback', () => { + const hook = () => 'Woo'; + const hook2 = () => 'Woo-er'; + addFilter( 'loaders/js', hook ); + addFilter( 'loaders/js', hook2 ); + removeFilter( 'loaders/js', hook ); + expect( getCallbacks( 'loaders/js' ) ).toEqual( [ hook2 ] ); + } ); + + it( 'does not remove a callback if the priority does not match', () => { + const hook = () => 'Woo'; + addFilter( 'loaders/js', hook, 9 ); + removeFilter( 'loaders/js', hook, 10 ); + expect( getCallbacks( 'loaders/js' ) ).toEqual( [ hook ] ); + } ); + } ); + + describe( 'applyFilters', () => { + const input = 'Pasta Carbonara'; + + beforeEach( () => { + setupRegistry(); + } ); + + it( 'returns input unchanged if no filters', () => { + expect( applyFilters( 'do/stuff', input ) ).toBe( 'Pasta Carbonara' ); + } ); + + it( 'processes input with the registered filters', () => { + addFilter( 'words', toKebabCase ); + addFilter( 'words', toUpperCase ); + expect( applyFilters( 'words', input ) ).toBe( 'PASTA-CARBONARA' ); + } ); + + it( 'processes input through callbacks in the specified order', () => { + addFilter( 'words', toLowerCase, 9 ); + addFilter( 'words', toKebabCase, 11 ); + addFilter( 'words', reverseWords, 10 ); + addFilter( 'words', reverse, 10 ); + expect( applyFilters( 'words', input ) ).toBe( 'atsap-aranobrac' ); + } ); + } ); +} ); diff --git a/src/helpers/infer-public-path.js b/src/helpers/infer-public-path.js index e60f024..e2a9b71 100644 --- a/src/helpers/infer-public-path.js +++ b/src/helpers/infer-public-path.js @@ -1,16 +1,35 @@ const filePath = require( './file-path' ); const findInObject = require( './find-in-object' ); +/** + * Dictionary of generated publicPath strings by file system path. + * + * @type {Object.<string, string>} + */ +const publicPaths = {}; + +/** + * Return a consistent publicPath per output directory path. + * + * Only used when automatically generating public paths. + * + * @param {String} path Output directory path + * @returns {string} Shared publicPath string. + */ +const getPublicPathForDirectory = ( path ) => { + return publicPaths[ path ]; +}; + /** * Infer the public path based on the defined output path. * * @param {webpack.Configuration} config Webpack configuration object. * @param {Number} port Port to use for webpack-dev-server. * @param {Object} [defaults] Optional config default values. - * @return {String} Public path. + * @returns {String} Public path. */ const inferPublicPath = ( config, port, defaults = {} ) => { - const protocol = findInObject( config, 'devServer.https' ) ? 'https' : 'http'; + const protocol = ( findInObject( config, 'devServer.https' ) || findInObject( config, 'devServer.server' ) === 'https' ) ? 'https' : 'http'; const outputPath = findInObject( config, 'output.path' ) || findInObject( defaults, 'output.path' ); @@ -21,7 +40,22 @@ const inferPublicPath = ( config, port, defaults = {} ) => { .replace( /^\/*/, '' ) .replace( /\/*$/, '/' ); - return `${ protocol }://localhost:${ port }/${ relPath }`; + const publicPath = `${ protocol }://localhost:${ port }/${ relPath }`; + + publicPaths[ outputPath ] = publicPath; + return publicPath; +}; + +module.exports = { + inferPublicPath, + getPublicPathForDirectory, }; -module.exports = inferPublicPath; +if ( process.env.JEST_WORKER_ID ) { + // Exposed only for testing purposes. + module.exports.resetPublicPathsCache = () => { + Object.keys( publicPaths ).forEach( ( key ) => { + Reflect.deleteProperty( publicPaths, key ); + } ); + }; +} diff --git a/src/helpers/with-dynamic-port.js b/src/helpers/with-dynamic-port.js index a22567f..b09df26 100644 --- a/src/helpers/with-dynamic-port.js +++ b/src/helpers/with-dynamic-port.js @@ -6,7 +6,7 @@ const choosePort = require( './choose-port' ); const filePath = require( './file-path' ); const findInObject = require( './find-in-object' ); -const inferPublicPath = require( './infer-public-path' ); +const { inferPublicPath } = require( './infer-public-path' ); const DEFAULT_PORT = 9090; diff --git a/src/helpers/with-dynamic-port.test.js b/src/helpers/with-dynamic-port.test.js index 7955c63..5b2b7ff 100644 --- a/src/helpers/with-dynamic-port.test.js +++ b/src/helpers/with-dynamic-port.test.js @@ -1,6 +1,7 @@ const withDynamicPort = require( './with-dynamic-port' ); const choosePort = require( './choose-port' ); +const { resetPublicPathsCache } = require( './infer-public-path' ); jest.mock( './choose-port', () => jest.fn() ); @@ -9,6 +10,7 @@ describe( 'withDynamicPort', () => { beforeEach( () => { oldArgv = process.argv; + resetPublicPathsCache(); } ); afterEach( () => { @@ -151,7 +153,7 @@ describe( 'withDynamicPort', () => { it( 'does not overwrite existing config properties', async () => { const config = { devServer: { - https: true, + server: 'https', }, entry: { 'name': './bundle.js', @@ -167,7 +169,7 @@ describe( 'withDynamicPort', () => { choosePort.mockImplementationOnce( () => Promise.resolve( 8082 ) ); expect( await withDynamicPort( 9090, config ) ).toEqual( { devServer: { - https: true, + server: 'https', port: 8082, }, entry: { diff --git a/src/loaders.js b/src/loaders.js index 0a23115..58345b9 100644 --- a/src/loaders.js +++ b/src/loaders.js @@ -1,43 +1,73 @@ /** * Export generator functions for common Webpack loader configurations. */ +const cloneDeep = require( 'lodash.clonedeep' ); const postcssFlexbugsFixes = require( 'postcss-flexbugs-fixes' ); const postcssPresetEnv = require( 'postcss-preset-env' ); const deepMerge = require( './helpers/deep-merge' ); +const { applyFilters } = require( './helpers/filters' ); /** * Export an object of named methods that generate corresponding loader config - * objects. To customize the default values of the loader, mutate the .defaults - * property exposed on each method (or pass a filterLoaders option to a preset). + * objects. To customize the default values of the loader, use addFilter() on + * either the hook `loader/loadername/defaults` (to adjust the default loader + * configuration) or `loader/loadername` (to alter the final, computed loader). */ const loaders = {}; +/** + * Create a loader factory function for a given loader slug. + * + * @private + * @param {string} loaderKey Loader slug + * @returns {Function} Factory function to generate and filter a loader with the specified slug. + */ const createLoaderFactory = loaderKey => { - const getFilteredLoader = ( options ) => { - // Handle missing options object. - if ( typeof options === 'function' ) { - return getFilteredLoader( {}, options ); - } - - // Generate the requested loader definition. - return deepMerge( loaders[ loaderKey ].defaults, options ); + return ( options = {}, config = null ) => { + /** + * Generate the requested loader definition. + * + * Allows customization of the final rendered loader via the loaders/{loaderslug} + * filter, which also receives the configuration passed to a preset factory if + * the loader is invoked in the context of a preset. + * + * @hook loaders/{$loader_slug} + * @param {Object} loader Complete loader definition object, after merging user-provided values with filtered defaults. + * @param {Object|null} [config] Configuration object for the preset being rendered, if loader is called while generating a preset. + */ + return applyFilters( + `loaders/${ loaderKey }`, + deepMerge( + /** + * Filter the loader's default configuration. + * + * @hook loaders/{$loader_slug}/defaults + * @param {Object} options Loader default options object. + * @param {Object|null} [config] Configuration object for the preset being rendered, if loader is called while generating a preset. + */ + applyFilters( `loaders/${ loaderKey }/defaults`, cloneDeep( loaders[ loaderKey ].defaults ), config ), + options + ), + config + ); }; - - return getFilteredLoader; }; // Define all supported loader factories within the loaders object. -[ 'eslint', 'js', 'ts', 'url', 'style', 'css', 'postcss', 'sass', 'file' ].forEach( loaderKey => { +[ 'asset', 'js', 'ts', 'style', 'css', 'postcss', 'sass', 'sourcemap', 'resource' ].forEach( loaderKey => { loaders[ loaderKey ] = createLoaderFactory( loaderKey ); } ); -loaders.eslint.defaults = { - test: /\.jsx?$/, - exclude: /(node_modules|bower_components)/, - enforce: 'pre', - loader: require.resolve( 'eslint-loader' ), - options: {}, +loaders.asset.defaults = { + test: /\.(png|jpg|jpeg|gif|avif|webp|svg|woff|woff2|eot|ttf)$/, + type: 'asset', + parser: { + dataUrlCondition: { + // Inline if less than 10kb. + maxSize: 10 * 1024, + }, + }, }; loaders.js.defaults = { @@ -56,14 +86,6 @@ loaders.ts.defaults = { loader: require.resolve( 'ts-loader' ), }; -loaders.url.defaults = { - test: /\.(png|jpg|jpeg|gif|svg|woff|woff2|eot|ttf)$/, - loader: require.resolve( 'url-loader' ), - options: { - limit: 10000, - }, -}; - loaders.style.defaults = { loader: require.resolve( 'style-loader' ), options: {}, @@ -80,15 +102,28 @@ loaders.postcss.defaults = { loader: require.resolve( 'postcss-loader' ), options: { postcssOptions: { - plugins: [ + ident: 'postcss', + /** + * Filter the default PostCSS Plugins array. + * + * @hook loaders/postcss/plugins + * @param {Array} plugins Array of PostCSS plugins. + */ + plugins: applyFilters( 'loaders/postcss/plugins', [ postcssFlexbugsFixes, - postcssPresetEnv( { + /** + * Filter the default PostCSS Preset Env configuration. + * + * @hook loaders/postcss/preset-env + * @param {Object} presetEnvConfig PostCSS Preset Env plugin configuration. + */ + postcssPresetEnv( applyFilters( 'loaders/postcss/preset-env', { autoprefixer: { flexbox: 'no-2009', }, stage: 3, - } ), - ], + } ) ), + ] ), }, }, }; @@ -102,11 +137,21 @@ loaders.sass.defaults = { }, }; -loaders.file.defaults = { - // Exclude `js`, `html` and `json`, but match anything else. - exclude: /\.(js|html|json)$/, - loader: require.resolve( 'file-loader' ), +loaders.sourcemap.defaults = { + test: /\.(js|mjs|jsx|ts|tsx|css)$/, + exclude: /@babel(?:\/|\\{1,2})runtime/, + enforce: 'pre', + loader: require.resolve( 'source-map-loader' ), options: {}, }; +loaders.resource.defaults = { + // Exclude `js` files to keep "css" loader working as it injects + // its runtime that would otherwise be processed through "file" loader. + // Also exclude `html` and `json` extensions so they get processed + // by webpacks internal loaders. + exclude: [ /^$/, /\.(js|mjs|jsx|ts|tsx)$/, /\.html?$/, /\.json$/ ], + type: 'asset/resource', +}; + module.exports = loaders; diff --git a/src/loaders.test.js b/src/loaders.test.js index deb445d..586731a 100644 --- a/src/loaders.test.js +++ b/src/loaders.test.js @@ -1,3 +1,4 @@ +const { addFilter, setupRegistry } = require( './helpers/filters' ); const loaders = require( './loaders' ); describe( 'loaders', () => { @@ -22,23 +23,15 @@ describe( 'loaders', () => { expect( result.value ).toBe( 42 ); } ); - describe( '.eslint()', () => { - it( 'tests for .js or .jsx files', () => { - expect( 'file.js'.match( loaders.eslint().test ) ).not.toBeNull(); - expect( 'file.jsx'.match( loaders.eslint().test ) ).not.toBeNull(); - expect( 'file.scss'.match( loaders.eslint().test ) ).toBeNull(); - } ); - } ); - describe( '.js()', () => { it( 'tests for .js or .jsx files', () => { - expect( 'file.js'.match( loaders.eslint().test ) ).not.toBeNull(); - expect( 'file.jsx'.match( loaders.eslint().test ) ).not.toBeNull(); - expect( 'file.scss'.match( loaders.eslint().test ) ).toBeNull(); + expect( 'file.js'.match( loaders.js().test ) ).not.toBeNull(); + expect( 'file.jsx'.match( loaders.js().test ) ).not.toBeNull(); + expect( 'file.scss'.match( loaders.js().test ) ).toBeNull(); } ); } ); - describe( '.url()', () => { + describe( '.asset()', () => { it( 'tests for static assets', () => { [ 'file.png', @@ -51,15 +44,33 @@ describe( 'loaders', () => { 'file.eot', 'file.ttf', ].forEach( acceptedFileType => { - expect( acceptedFileType.match( loaders.url().test ) ).not.toBeNull(); + expect( acceptedFileType.match( loaders.asset().test ) ).not.toBeNull(); } ); [ 'file.js', 'file.css', 'file.html', ].forEach( unacceptedFileType => { - expect( unacceptedFileType.match( loaders.url().test ) ).toBeNull(); + expect( unacceptedFileType.match( loaders.asset().test ) ).toBeNull(); + } ); + } ); + } ); + + describe( 'filtering', () => { + beforeEach( () => { + setupRegistry(); + } ); + + it( 'does not mutate defaults', () => { + addFilter( 'loaders/js/defaults', ( defaults ) => { + defaults.test = /\.js$/; + return defaults; } ); + + const loader = loaders.js(); + + expect( loader.test ).toEqual( /\.js$/ ); + expect( loaders.js.defaults.test ).toEqual( /\.jsx?$/ ); } ); } ); diff --git a/src/plugins.js b/src/plugins.js index 3de1ad3..094a0da 100644 --- a/src/plugins.js +++ b/src/plugins.js @@ -1,15 +1,18 @@ const { BundleAnalyzerPlugin } = require( 'webpack-bundle-analyzer' ); const { CleanWebpackPlugin } = require( 'clean-webpack-plugin' ); -const { HotModuleReplacementPlugin } = require( 'webpack' ); const BellOnBundleErrorPlugin = require( 'bell-on-bundler-error-plugin' ); const CopyPlugin = require( 'copy-webpack-plugin' ); -const FixStyleOnlyEntriesPlugin = require( 'webpack-fix-style-only-entries' ); +const ESLintPlugin = require( 'eslint-webpack-plugin' ); const { WebpackManifestPlugin: ManifestPlugin } = require( 'webpack-manifest-plugin' ); const MiniCssExtractPlugin = require( 'mini-css-extract-plugin' ); +const RemoveEmptyScriptsPlugin = require( 'webpack-remove-empty-scripts' ); +const SimpleBuildReportPlugin = require( 'simple-build-report-webpack-plugin' ); const TerserPlugin = require( 'terser-webpack-plugin' ); -const OptimizeCssAssetsPlugin = require( 'optimize-css-assets-webpack-plugin' ); +const CssMinimizerPlugin = require( 'css-minimizer-webpack-plugin' ); -const deepMerge = require( './helpers/deep-merge' ); +const { applyFilters } = require( './helpers/filters' ); + +const isProfileMode = process.argv.includes( '--profile' ); module.exports = { /** @@ -22,11 +25,12 @@ module.exports = { BundleAnalyzerPlugin, CleanWebpackPlugin, CopyPlugin, - FixStyleOnlyEntriesPlugin, - HotModuleReplacementPlugin, + ESLintPlugin, ManifestPlugin, MiniCssExtractPlugin, - OptimizeCssAssetsPlugin, + CssMinimizerPlugin, + RemoveEmptyScriptsPlugin, + SimpleBuildReportPlugin, TerserPlugin, }, @@ -47,14 +51,14 @@ module.exports = { }, /** - * Create a new BundleAnalyzerPlugin instance. The analyzer is enabled by default - * only if `--analyze` is passed on the command line. + * Create a new BundleAnalyzerPlugin instance. * * @param {Object} [options] Optional plugin options object. * @returns {BundleAnalyzerPlugin} A configured BundleAnalyzerPlugin instance. */ bundleAnalyzer: ( options = {} ) => new BundleAnalyzerPlugin( { - analyzerMode: process.argv.indexOf( '--analyze' ) >= 0 ? 'static' : 'disabled', + analyzerMode: 'static', + generateStatsFile: true, openAnalyzer: false, reportFilename: 'bundle-analyzer-report.html', ...options, @@ -99,22 +103,26 @@ module.exports = { errorBell: () => new BellOnBundleErrorPlugin(), /** - * Create a new FixStyleOnlyEntriesPlugin instance to remove unnecessary JS - * files generated for style-only bundle entries. + * Create a new ESLintPlugin instance to check your build for syntax or style issues. * - * @param {Object} [options] Optional plugin options object. - * @param {RegExp} [options.exclude] Regular expression to filter what gets cleaned. - * @returns {FixStyleOnlyEntriesPlugin} A configured FixStyleOnlyEntriesPlugin instance. + * @param {Object} options Optional plugin options object. + * @returns {ESLintPlugin} A configured ESLintPlugin instance. */ - fixStyleOnlyEntries: ( options ) => new FixStyleOnlyEntriesPlugin( options ), + eslint: ( options ) => new ESLintPlugin( { + extensions: [ 'js', 'jsx', 'ts', 'tsx' ], + exclude: [ 'node_modules', 'vendor', '*.min.js', '*.min.jsx' ], + ...options, + } ), /** - * Create a webpack.HotModuleReplacementPlugin instance. + * Create a new RemoveEmptyScriptsPlugin instance to remove unnecessary JS + * files generated for style-only bundle entries. * - * @param {Object} [options] Optional plugin options object. - * @returns {HotModuleReplacementPlugin} A configured HMR Plugin instance. + * @param {Object} [options] Optional plugin options object. + * @param {RegExp} [options.exclude] Regular expression to filter what gets cleaned. + * @returns {RemoveEmptyScriptsPlugin} A configured RemoveEmptyScriptsPlugin instance. */ - hotModuleReplacement: ( options = {} ) => new HotModuleReplacementPlugin( options ), + fixStyleOnlyEntries: ( options ) => new RemoveEmptyScriptsPlugin( options ), /** * Create a new ManifestPlugin instance to output an asset-manifest.json @@ -154,12 +162,20 @@ module.exports = { } ), /** - * Create a new OptimizeCssAssetsPlugin instance. + * Create a new CssMinimizerPlugin instance. * * @param {Object} [options] Optional plugin configuration options. - * @returns {OptimizeCssAssetsPlugin} A configured OptimizeCssAssetsPlugin instance. + * @returns {CssMinimizerPlugin} A configured CssMinimizerPlugin instance. */ - optimizeCssAssets: ( options = {} ) => new OptimizeCssAssetsPlugin( options ), + cssMinimizer: ( options = {} ) => new CssMinimizerPlugin( options ), + + /** + * Instantiate a SimpleBuildReportPlugin to render build output in a human- + * oriented fashion. + * + * @returns {SimpleBuildReportPlugin} Output formatter plugin instance. + */ + simpleBuildReport: () => new SimpleBuildReportPlugin(), /** * Create a new TerserPlugin instance, defaulting to a set of options @@ -170,41 +186,46 @@ module.exports = { * @param {Object} [options.terserOptions] Terser compressor options object. * @returns {TerserPlugin} A configured TerserPlugin instance. */ - terser: ( options = {} ) => new TerserPlugin( deepMerge( { - terserOptions: { - parse: { - // we want terser to parse ecma 8 code. However, we don't want it - // to apply any minfication steps that turns valid ecma 5 code - // into invalid ecma 5 code. This is why the 'compress' and 'output' - // sections only apply transformations that are ecma 5 safe - // https://github.com/facebook/create-react-app/pull/4234 - ecma: 8, - }, - compress: { - ecma: 5, - warnings: false, - // Disabled because of an issue with Uglify breaking seemingly valid code: - // https://github.com/facebook/create-react-app/issues/2376 - // Pending further investigation: - // https://github.com/mishoo/UglifyJS2/issues/2011 - comparisons: false, - // Disabled because of an issue with Terser breaking valid code: - // https://github.com/facebook/create-react-app/issues/5250 - // Pending futher investigation: - // https://github.com/terser-js/terser/issues/120 - inline: 2, + terser: ( options = {} ) => new TerserPlugin( { + ...applyFilters( 'plugins/terser/defaults', { + terserOptions: { + parse: { + // We want terser to parse ecma 8 code. However, we don't want it + // to apply any minification steps that turns valid ecma 5 code + // into invalid ecma 5 code. This is why the 'compress' and 'output' + // sections only apply transformations that are ecma 5 safe + // https://github.com/facebook/create-react-app/pull/4234 + ecma: 8, + }, + compress: { + ecma: 5, + warnings: false, + // Disabled because of an issue with Uglify breaking seemingly valid code: + // https://github.com/facebook/create-react-app/issues/2376 + // Pending further investigation: + // https://github.com/mishoo/UglifyJS2/issues/2011 + comparisons: false, + // Disabled because of an issue with Terser breaking valid code: + // https://github.com/facebook/create-react-app/issues/5250 + // Pending further investigation: + // https://github.com/terser-js/terser/issues/120 + inline: 2, + }, + mangle: { + safari10: true, + }, + // Added for profiling in devtools + keep_classnames: isProfileMode, + keep_fnames: isProfileMode, + output: { + ecma: 5, + comments: false, + // Turned on because emoji and regex is not minified properly using default + // https://github.com/facebook/create-react-app/issues/2488 + ascii_only: true, + }, }, - mangle: { - safari10: true, - }, - output: { - ecma: 5, - comments: false, - // Turned on because emoji and regex is not minified properly using default - // https://github.com/facebook/create-react-app/issues/2488 - ascii_only: true, - }, - }, - extractComments: false, - }, options ) ), + } ), + ...options, + } ), }; diff --git a/src/presets.js b/src/presets.js index 9f3a28a..75386f9 100644 --- a/src/presets.js +++ b/src/presets.js @@ -2,12 +2,15 @@ const { devServer, stats } = require( './config' ); const deepMerge = require( './helpers/deep-merge' ); const filePath = require( './helpers/file-path' ); const findInObject = require( './helpers/find-in-object' ); -const inferPublicPath = require( './helpers/infer-public-path' ); +const { inferPublicPath, getPublicPathForDirectory } = require( './helpers/infer-public-path' ); const isInstalled = require( './helpers/is-installed' ); +const { applyFilters } = require( './helpers/filters' ); const loaders = require( './loaders' ); const plugins = require( './plugins' ); const { ManifestPlugin, MiniCssExtractPlugin } = plugins.constructors; +const isAnalyzeMode = process.argv.includes( '--analyze' ); + /** * Dictionary of shared seed objects by path. * @@ -30,11 +33,11 @@ const getSeedByDirectory = ( path ) => { /** * Helper to detect whether a given package is installed, and return a spreadable - * array containing an appropriate loader if so. + * array containing an appropriate loader or plugin if so. * * @example - * rules: [ - * ...ifInstalled( 'eslint', loaders.eslint() ), + * plugins: [ + * ...ifInstalled( 'eslint', plugins.eslint() ), * ], * * @param {String} packageName The string name of the dependency for which to test. @@ -50,38 +53,36 @@ const ifInstalled = ( packageName, loader ) => { }; /** - * Given a reference to a (possibly-undefined) filtering method, return a pair - * of helper functions to filter a loader definition object, or to get a loader - * definition object by its loaders dictionary key and then filter it. + * Remove null entries from Webpack loaders array, in case a user returned null + * from a filter to opt out of a given loader in the preset. + * + * Detects and removes null entries from nested .oneOf or .use arrays. + * + * @param {Object[]} moduleRules Array of Webpack loader rules. * - * @param {Function} [filterLoaders] An optional filterLoaders function. Defaults - * to the identity function. - * @returns {Object} Object with `filterLoaders` and `getFilteredLoader` methods. + * @returns {Object[]} Filtered array with null items removed. */ -const createFilteringHelpers = ( filterLoaders = ( input ) => input ) => ( { - /** - * Given a loader object and its key string, pass that object through the - * filterLoaders method (if one was provided) and return the filtered output. - * - * @param {Object} loader Webpack loader configuration object. - * @param {String} loaderKey String identifying which loader is being filtered. - * @returns {Object} Filtered loader object. - */ - filterLoaders: ( loader, loaderKey ) => { - return filterLoaders( loader, loaderKey ); - }, - - /** - * Helper method to reduce duplication when accessing and invoking loader factories. - * - * @param {String} loaderKey String key of a loader factory in the loaders object. - * @param {Object} [options] Options for this loader (optional). - * @returns {Object} Configured and filtered loader definition. - */ - getFilteredLoader: ( loaderKey, options ) => { - return filterLoaders( loaders[ loaderKey ]( options ), loaderKey ); - }, -} ); +const removeNullLoaders = ( moduleRules ) => moduleRules + .map( ( rule ) => { + if ( rule && Array.isArray( rule.oneOf ) ) { + const loaders = removeNullLoaders( rule.oneOf ); + if ( ! Array.isArray( loaders ) || ! loaders.length ) { + return null; + } + return { + ...rule, + oneOf: loaders, + }; + } + if ( rule && Array.isArray( rule.use ) ) { + return { + ...rule, + use: removeNullLoaders( rule.use ), + }; + } + return rule; + } ) + .filter( Boolean ); /** * Promote a partial Webpack config into a full development-oriented configuration. @@ -95,17 +96,10 @@ const createFilteringHelpers = ( filterLoaders = ( input ) => input ) => ( { * - an `.output.publicPath` string (unless a devServer.port is specified, * in which case publicPath defaults to `http://localhost:${ port }`) * - * @param {webpack.Configuration} config Configuration options to deeply merge into the defaults. - * @param {Object} [options] Optional options to modify configuration generation. - * @param {Function} [options.filterLoaders] An optional filter function that receives each - * computed loader definition and the name of that - * loader as it is generated, to permit per-config - * customization of loader options. + * @param {webpack.Configuration} config Configuration options to deeply merge into the defaults. * @returns {webpack.Configuration} A merged Webpack configuration object. */ -const development = ( config = {}, options = {} ) => { - const { filterLoaders, getFilteredLoader } = createFilteringHelpers( options.filterLoaders ); - +const development = ( config = {} ) => { /** * Default development environment-oriented Webpack options. This object is * defined at the time of function execution so that any changes to the @@ -128,75 +122,88 @@ const development = ( config = {}, options = {} ) => { // Add /* filename */ comments to generated require()s in the output. pathinfo: true, // Provide a default output name. - filename: '[name].js', + filename: '[name].[fullhash].js', // Provide chunk filename. Requires content hash for cache busting. - chunkFilename: '[name].[contenthash].chunk.js', + chunkFilename: '[name].[fullhash].chunk.js', // `publicPath` will be inferred as a localhost URL based on output.path // when a devServer.port value is available. }, module: { strictExportPresence: true, - rules: [ - // Run all JS files through ESLint, if installed. - ...ifInstalled( 'eslint', getFilteredLoader( 'eslint', { - options: { - emitWarning: true, - }, - } ) ), + rules: removeNullLoaders( [ + // Handle node_modules packages that contain sourcemaps. + loaders.sourcemap( {}, config ), { // "oneOf" will traverse all following loaders until one will // match the requirements. If no loader matches, it will fall - // back to the "file" loader at the end of the loader list. + // back to the resource loader at the end of the loader list. oneOf: [ // Enable processing TypeScript, if installed. - ...ifInstalled( 'typescript', getFilteredLoader( 'ts' ) ), + ...ifInstalled( 'typescript', loaders.ts( {}, config ) ), // Process JS with Babel. - getFilteredLoader( 'js' ), - // Convert small files to data URIs. - getFilteredLoader( 'url' ), - // Parse styles using SASS, then PostCSS. - filterLoaders( { - test: /\.s?css$/, - use: [ - getFilteredLoader( 'style' ), - getFilteredLoader( 'css', { - options: { - sourceMap: true, - }, - } ), - getFilteredLoader( 'postcss', { - options: { - sourceMap: true, - }, - } ), - getFilteredLoader( 'sass', { - options: { - sourceMap: true, - }, - } ), - ], - }, 'stylesheet' ), - // "file" loader makes sure any non-matching assets still get served. - // When you `import` an asset you get its filename. - getFilteredLoader( 'file' ), + loaders.js( {}, config ), + // Handle static asset files. + loaders.asset( {}, config ), + /** + * Filter the full stylesheet loader definition for this preset. + * + * By default parses styles using Sass and then PostCSS. + * + * @hook presets/stylesheet-loaders + * @param {Object} loader Stylesheet loader rule. + * @param {string} environment "development" or "production". + * @param {Object} config Preset configuration object. + */ + applyFilters( + 'presets/stylesheet-loaders', + { + test: /\.s?css$/, + use: [ + loaders.style( {}, config ), + loaders.css( { + options: { + sourceMap: true, + }, + }, config ), + loaders.postcss( { + options: { + sourceMap: true, + }, + }, config ), + loaders.sass( { + options: { + sourceMap: true, + }, + }, config ), + ], + }, + 'development', + config + ), + // Resource loader makes sure any non-matching assets still get served. + // When you `import` an asset, you get its (virtual) filename. + loaders.resource( {}, config ), ], }, - ], + ] ), }, optimization: { - nodeEnv: 'development', + nodeEnv: 'development' }, - devServer: { - ...devServer, - stats, - }, + devServer, + + stats, plugins: [ - plugins.hotModuleReplacement(), - ], + // Run all JS files through ESLint, if installed. + ...ifInstalled( 'eslint', plugins.eslint( { + // But don't let errors block the build. + failOnError: false, + } ) ), + ].filter( Boolean ), }; // If no entry was provided, inject a default entry value. @@ -214,23 +221,52 @@ const development = ( config = {}, options = {} ) => { publicPath = inferPublicPath( config, port, devDefaults ); } - // If we had enough value to guess a publicPath, set that path as a default - // wherever appropriate and inject a ManifestPlugin instance to expose that - // public path to consuming applications. Any inferred values will still be - // overridden with their relevant values from `config`, when provided. + const outputPath = ( config.output && config.output.path ) || devDefaults.output.path; + // If we used a publicPath for this outputPath before, re-use it. + if ( ! publicPath && outputPath ) { + publicPath = getPublicPathForDirectory( outputPath ); + } + + // If we had enough config values to guess a publicPath, set that path in + // the default config so it can be used in generated manifests. if ( publicPath ) { + // If publicPath is a URL, default the devServer host to match. + if ( typeof publicPath === 'string' && publicPath.includes( 'http' ) ) { + devDefaults.devServer.host = new URL( publicPath ).hostname; + } devDefaults.output.publicPath = publicPath; + } - // Check for an existing ManifestPlugin instance in config.plugins. - const hasManifestPlugin = plugins.findExistingInstance( config.plugins, ManifestPlugin ); - // Add a manifest with the inferred publicPath if none was present. - if ( ! hasManifestPlugin ) { - const outputPath = ( config.output && config.output.path ) || devDefaults.output.path; - devDefaults.plugins.push( plugins.manifest( { - fileName: 'asset-manifest.json', - seed: getSeedByDirectory( outputPath ), - } ) ); - } + // Check for an existing ManifestPlugin instance in config.plugins. + // Inject a ManifestPlugin instance if none is present to ensure generated + // files can be located by consuming aplications. Any inferred values will + // still be overridden with their relevant values from `config`, if provided. + const hasManifestPlugin = plugins.findExistingInstance( config.plugins, ManifestPlugin ); + // Add a manifest if none was present. + if ( ! hasManifestPlugin ) { + /* eslint-disable function-paren-newline */ + devDefaults.plugins.push( plugins.manifest( + /** + * Filter the full stylesheet loader definition for this preset. + * + * By default parses styles using Sass and then PostCSS. + * + * @hook presets/manifest-options + * @param {Object} options Manifest plugin options object. + * @param {string} environment "development" or "production". + * @param {Object} config Preset configuration object. + */ + applyFilters( + 'presets/manifest-options', + { + fileName: 'development-asset-manifest.json', + seed: getSeedByDirectory( outputPath ), + }, + 'development', + config + ) + ) ); + /* eslint-enable */ } return deepMerge( devDefaults, config ); @@ -243,17 +279,10 @@ const development = ( config = {}, options = {} ) => { * merges specified options into an opinionated default production configuration * template. * - * @param {webpack.Configuration} config Configuration options to deeply merge into the defaults. - * @param {Object} [options] Optional options to modify configuration generation. - * @param {Function} [options.filterLoaders] An optional filter function that receives each - * computed loader definition and the name of that - * loader as it is generated, to permit per-config - * customization of loader options. + * @param {webpack.Configuration} config Configuration options to deeply merge into the defaults. * @returns {webpack.Configuration} A merged Webpack configuration object. */ -const production = ( config = {}, options = {} ) => { - const { filterLoaders, getFilteredLoader } = createFilteringHelpers( options.filterLoaders ); - +const production = ( config = {} ) => { // Determine whether source maps have been requested, and prepare an options // object to be passed to all CSS loaders to honor that request. const cssOptions = config.devtool ? @@ -281,74 +310,91 @@ const production = ( config = {}, options = {} ) => { // Inject a default entry point later on if none was specified. output: { + // Webpack 5 defaults "publicPath" to "auto," which we are not set up to handle. + publicPath: '', // Provide a default output path. path: filePath( 'build' ), pathinfo: false, // Provide a default output name. - filename: '[name].js', + filename: '[name].[contenthash].js', // Provide chunk filename. Requires content hash for cache busting. chunkFilename: '[name].[contenthash].chunk.js', }, module: { strictExportPresence: true, - rules: [ - // Run all JS files through ESLint, if installed. - ...ifInstalled( 'eslint', getFilteredLoader( 'eslint' ) ), + rules: removeNullLoaders( [ { // "oneOf" will traverse all following loaders until one will // match the requirements. If no loader matches, it will fall - // back to the "file" loader at the end of the loader list. + // back to the resource loader at the end of the loader list. oneOf: [ // Enable processing TypeScript, if installed. - ...ifInstalled( 'typescript', getFilteredLoader( 'ts' ) ), + ...ifInstalled( 'typescript', loaders.ts( {}, config ) ), // Process JS with Babel. - getFilteredLoader( 'js' ), - // Convert small files to data URIs. - getFilteredLoader( 'url' ), - // Parse styles using SASS, then PostCSS. - filterLoaders( { - test: /\.s?css$/, - use: [ - // Extract CSS to its own file. - MiniCssExtractPlugin.loader, - // Process SASS into CSS. - getFilteredLoader( 'css', cssOptions ), - getFilteredLoader( 'postcss', cssOptions ), - getFilteredLoader( 'sass', cssOptions ), - ], - }, 'stylesheet' ), - // "file" loader makes sure any non-matching assets still get served. - // When you `import` an asset you get its filename. - getFilteredLoader( 'file' ), + loaders.js( {}, config ), + // Handle static asset files. + loaders.asset( {}, config ), + /** + * Filter the full stylesheet loader definition for this preset. + * + * By default parses styles using Sass and then PostCSS. + * + * @hook presets/stylesheet-loaders + * @param {Object} loader Stylesheet loader rule. + * @param {string} environment "development" or "production". + * @param {Object} config Preset configuration object. + */ + applyFilters( + 'presets/stylesheet-loaders', + { + test: /\.s?css$/, + use: [ + // Extract CSS to its own file. + MiniCssExtractPlugin.loader, + // Process SASS into CSS. + loaders.css( cssOptions, config ), + loaders.postcss( cssOptions, config ), + loaders.sass( cssOptions, config ), + ], + }, + 'production', + config + ), + // Resource loader makes sure any non-matching assets still get served. + // When you `import` an asset, you get its (virtual) filename. + loaders.resource( {}, config ), ], }, - ], + ] ), }, optimization: { + minimize: true, minimizer: [ plugins.terser(), - plugins.optimizeCssAssets( ( - // Set option to output source maps if devtool is set. - config.devtool && ! ( /inline-/ ).test( config.devtool ) ? - { - cssProcessorOptions: { - map: { - inline: false, - }, - }, - } : - undefined - ) ), + plugins.cssMinimizer(), ], - nodeEnv: 'production', - noEmitOnErrors: true, + emitOnErrors: false, }, stats, - plugins: [], + plugins: [ + // Run all JS files through ESLint, if installed. + ...ifInstalled( 'eslint', plugins.eslint() ), + // Use the simple build report plugin to clean up Webpack's terminal output. + plugins.simpleBuildReport(), + // If webpack was invoked with the --analyze flag, include a bundleAnalyzer + // in production builds. Use the configuration name when present to separate + // output files for each webpack build in multi-configuration setups. + isAnalyzeMode ? + plugins.bundleAnalyzer( { + reportFilename: config.name ? `${ config.name }-analyzer-report.html` : 'bundle-analyzer-report.html', + statsFilename: config.name ? `${ config.name }-stats.json` : 'stats.json', + } ) : + null, + ].filter( Boolean ), }; // If no entry was provided, inject a default entry value. @@ -369,10 +415,29 @@ const production = ( config = {}, options = {} ) => { // Add a manifest with the inferred publicPath if none was present. if ( ! hasManifestPlugin ) { const outputPath = ( config.output && config.output.path ) || prodDefaults.output.path; - prodDefaults.plugins.push( plugins.manifest( { - fileName: 'production-asset-manifest.json', - seed: getSeedByDirectory( outputPath ), - } ) ); + /* eslint-disable function-paren-newline */ + prodDefaults.plugins.push( plugins.manifest( + /** + * Filter the full stylesheet loader definition for this preset. + * + * By default parses styles using Sass and then PostCSS. + * + * @hook presets/manifest-options + * @param {Object} options Manifest plugin options object. + * @param {string} environment "development" or "production". + * @param {Object} config Preset configuration object. + */ + applyFilters( + 'presets/manifest-options', + { + fileName: 'production-asset-manifest.json', + seed: getSeedByDirectory( outputPath ), + }, + 'production', + config + ) + ) ); + /* eslint-enable */ } return deepMerge( prodDefaults, config ); diff --git a/src/presets.test.js b/src/presets.test.js index b623b2a..e1821b4 100644 --- a/src/presets.test.js +++ b/src/presets.test.js @@ -3,8 +3,20 @@ const { production, } = require( './presets' ); const plugins = require( './plugins' ); +const { addFilter, setupRegistry } = require( './helpers/filters' ); +const { resetPublicPathsCache } = require( './helpers/infer-public-path' ); -jest.mock( 'process', () => ( { cwd: () => 'cwd' } ) ); +jest.mock( 'process', () => ( { + cwd: () => 'cwd', + versions: {}, +} ) ); + +/** + * Helper function to return null, used when testing filters. + * + * @returns {null} Null. + */ +const returnNull = () => null; /** * Filter an array of plugins to only contain plugins of the provided type. @@ -26,6 +38,10 @@ const getLoaderByName = ( rules, loaderType ) => { if ( rule.loader && rule.loader.indexOf( loaderType ) > -1 ) { return rule; } + // Permit selection of asset module loaders with no .loader property. + if ( rule.type && rule.type === loaderType ) { + return rule; + } if ( rule.oneOf ) { const nestedMatch = getLoaderByName( rule.oneOf, loaderType ); if ( nestedMatch ) { @@ -61,6 +77,11 @@ const getLoaderByTest = ( rules, loaderTest ) => { }; describe( 'presets', () => { + beforeEach( () => { + setupRegistry(); + resetPublicPathsCache(); + } ); + describe( 'development()', () => { it( 'is a function', () => { expect( development ).toBeInstanceOf( Function ); @@ -76,8 +97,8 @@ describe( 'presets', () => { expect( config.entry ).toEqual( 'some-file.js' ); expect( config.output ).toEqual( { pathinfo: true, - filename: '[name].js', - chunkFilename: '[name].[contenthash].chunk.js', + filename: '[name].[fullhash].js', + chunkFilename: '[name].[fullhash].chunk.js', path: 'build/', } ); } ); @@ -109,7 +130,18 @@ describe( 'presets', () => { expect( config.output.publicPath ).toBe( 'http://localhost:9090/build/' ); } ); - it( 'accounts for the value of devServer.https when inferring publicPath URI', () => { + it( 'accounts for the value of devServer.server when inferring publicPath URI', () => { + const config = development( { + devServer: { + server: 'https', + port: 9090, + }, + entry: 'some-file.js', + } ); + expect( config.output.publicPath ).toBe( 'https://localhost:9090/build/' ); + } ); + + it( 'accounts for the value of devServer.https (deprecated in favor of .server) when inferring publicPath URI', () => { const config = development( { devServer: { https: true, @@ -150,7 +182,25 @@ describe( 'presets', () => { expect( config.output.publicPath ).toBe( 'https://my-custom-domain.local/' ); } ); - it( 'injects a ManifestPlugin if publicPath can be inferred and no manifest plugin is already present', () => { + it( 're-uses previously inferred publicPath URIs in subsequent builds to the same directory', () => { + const config1 = development( { + devServer: { + port: 9090, + }, + output: { + path: 'some/folder', + } + } ); + const config2 = development( { + output: { + path: 'some/folder', + }, + } ); + expect( config1.output.publicPath ).toBe( 'http://localhost:9090/some/folder/' ); + expect( config1.output.publicPath ).toBe( config2.output.publicPath ); + } ); + + it( 'injects a ManifestPlugin if no manifest plugin is already present', () => { const { ManifestPlugin } = plugins.constructors; const config = development( { devServer: { @@ -165,19 +215,7 @@ describe( 'presets', () => { ] ) ); const manifestPlugins = config.plugins.filter( filterPlugins( ManifestPlugin ) ); expect( manifestPlugins.length ).toBe( 1 ); - expect( manifestPlugins[ 0 ].options.fileName ).toEqual( 'asset-manifest.json' ); - } ); - - it( 'does not inject a ManifestPlugin if publicPath cannot be inferred', () => { - const config = development( { - entry: { - main: 'some-file.css', - }, - } ); - expect( config.output ).not.toHaveProperty( 'publicPath' ); - expect( config.plugins ).toEqual( expect.not.arrayContaining( [ - expect.any( plugins.constructors.ManifestPlugin ), - ] ) ); + expect( manifestPlugins[ 0 ].options.fileName ).toEqual( 'development-asset-manifest.json' ); } ); it( 'does not override or duplicate existing ManifestPlugin instances', () => { @@ -252,57 +290,56 @@ describe( 'presets', () => { } ); it( 'permits filtering the computed output of individual loaders', () => { + addFilter( 'loaders/asset', ( loader ) => { + loader.test = /\.(png|jpg|jpeg|gif|svg)$/; + return loader; + } ); + addFilter( 'loaders/resource', ( loader ) => { + loader.options = { + publicPath: '../../', + }; + return loader; + } ); const config = development( { entry: { main: 'some-file.js', }, - }, { - filterLoaders: ( loader, loaderType ) => { - if ( loaderType === 'file' ) { - loader.options.publicPath = '../../'; - } - if ( loaderType === 'url' ) { - loader.test = /\.(png|jpg|jpeg|gif|svg)$/; - } - return loader; - }, } ); - const fileLoader = getLoaderByName( config.module.rules, 'file-loader' ); - const urlLoader = getLoaderByName( config.module.rules, 'url-loader' ); + const resourceLoader = getLoaderByName( config.module.rules, 'asset/resource' ); + const assetLoader = getLoaderByName( config.module.rules, 'asset' ); const jsLoader = getLoaderByName( config.module.rules, 'babel-loader' ); - expect( fileLoader ).toEqual( expect.objectContaining( { - exclude: /\.(js|html|json)$/, + expect( resourceLoader ).toEqual( expect.objectContaining( { + exclude: [ /^$/, /\.(js|mjs|jsx|ts|tsx)$/, /\.html?$/, /\.json$/ ], + type: 'asset/resource', options: { publicPath: '../../', }, } ) ); - expect( urlLoader ).toEqual( expect.objectContaining( { + expect( assetLoader ).toEqual( expect.objectContaining( { test: /\.(png|jpg|jpeg|gif|svg)$/, - options: { - limit: 10000, + type: 'asset', + parser: { + dataUrlCondition: { + maxSize: 10 * 1024, + }, }, } ) ); expect( jsLoader ).not.toBeNull(); } ); it( 'permits filtering the entire stylesheet loader chain', () => { + addFilter( 'presets/stylesheet-loaders', ( loader ) => { + loader.test = /\.styl$/; + return loader; + } ); + addFilter( 'loaders/sass', () => ( { + loader: 'stylus', + mode: 'development', + } ) ); const config = development( { entry: { main: 'some-file.js', }, - }, { - filterLoaders: ( loader, loaderType ) => { - if ( loaderType === 'stylesheet' ) { - loader.test = /\.styl$/; - } - if ( loaderType === 'sass' ) { - return { - loader: 'stylus', - mode: 'development', - }; - } - return loader; - }, } ); const styleChain = getLoaderByTest( config.module.rules, /\.styl$/ ); expect( styleChain ).toEqual( { @@ -317,6 +354,110 @@ describe( 'presets', () => { const sassLoader = getLoaderByTest( config.module.rules, /\.s?css$/ ); expect( sassLoader ).toBeNull(); } ); + + it( 'passes the preset configuration object to the stylesheet chain filter', () => { + const config = { + entry: { + main: 'some-file.js', + }, + }; + addFilter( 'presets/stylesheet-loaders', ( loader, preset, presetConfig ) => { + expect( presetConfig ).toBe( config ); + expect( preset ).toBe( 'development' ); + return loader; + } ); + development( config ); + } ); + + it( 'permits skipping a specific stylesheet loader by filtering it to null', () => { + addFilter( 'loaders/postcss', returnNull ); + addFilter( 'loaders/sass', returnNull ); + const config = development( { + entry: { + main: 'some-file.js', + }, + } ); + const styleChain = getLoaderByTest( config.module.rules, /\.s?css$/ ); + expect( styleChain ).toEqual( { + test: /\.s?css$/, + use: [ + { + loader: require.resolve( 'style-loader' ), + options: {}, + }, + { + loader: require.resolve( 'css-loader' ), + options: { + importLoaders: 1, + sourceMap: true, + }, + }, + ], + } ); + } ); + + it( 'does not include null loader entries if a loader was disabled with a filter', () => { + addFilter( 'presets/stylesheet-loaders', returnNull ); + addFilter( 'loaders/ts', returnNull ); + const config = development( { + entry: { + main: 'some-file.js', + }, + } ); + // Should have stripped out the null entries. + expect( config.module.rules[ 1 ].oneOf.length ).toBe( 3 ); + expect( config.module.rules[ 1 ].oneOf[ 0 ] ) + .toEqual( expect.objectContaining( { test: /\.jsx?$/ } ) ); + expect( config.module.rules[ 1 ].oneOf[ 1 ] ) + .toEqual( expect.objectContaining( { type: 'asset' } ) ); + expect( config.module.rules[ 1 ].oneOf[ 2 ] ) + .toEqual( expect.objectContaining( { type: 'asset/resource' } ) ); + } ); + + it( 'passes the preset config as argument 2 to loader filters', () => { + addFilter( 'loaders/js', ( options, config ) => { + expect( config ).not.toBeNull(); + expect( config.name ).toBe( 'dev-build' ); + return options; + } ); + development( { + name: 'dev-build', + } ); + } ); + + it( 'permits filtering only a specific invocation of a preset', () => { + addFilter( 'presets/stylesheet-loaders', returnNull ); + addFilter( 'loaders/ts', returnNull ); + + const filterToNullInBuild1 = ( options, config ) => { + if ( config.name === 'build1' ) { + return null; + } + // Simplified version to make test easier. + return options.test ? + { test: options.test } : + { type: options.type }; + }; + addFilter( 'loaders/js', filterToNullInBuild1 ); + addFilter( 'loaders/asset', filterToNullInBuild1 ); + addFilter( 'loaders/resource', filterToNullInBuild1 ); + addFilter( 'loaders/sourcemap', filterToNullInBuild1 ); + + const config1 = development( { name: 'build1' } ); + const config2 = development( { name: 'build2' } ); + + expect( config1.module.rules ).toEqual( [] ); + expect( config2.module.rules ).toEqual( [ + { test: /\.(js|mjs|jsx|ts|tsx|css)$/ }, + { + oneOf: [ + { test: /\.jsx?$/ }, + { test: /\.(png|jpg|jpeg|gif|avif|webp|svg|woff|woff2|eot|ttf)$/ }, + { type: 'asset/resource' }, + ], + }, + ] ); + } ); } ); describe( 'production()', () => { @@ -334,9 +475,10 @@ describe( 'presets', () => { expect( config.entry ).toEqual( 'some-file.js' ); expect( config.output ).toEqual( { pathinfo: false, - filename: '[name].js', + filename: '[name].[contenthash].js', chunkFilename: '[name].[contenthash].chunk.js', path: 'build/', + publicPath: '', } ); } ); @@ -461,56 +603,55 @@ describe( 'presets', () => { } ); it( 'permits filtering the computed output of individual loaders', () => { + addFilter( 'loaders/asset', ( loader ) => { + loader.test = /\.(png|jpg|jpeg|gif|svg)$/; + return loader; + } ); + addFilter( 'loaders/resource', ( loader ) => { + loader.options = { + publicPath: '../../', + }; + return loader; + } ); const config = production( { entry: { main: 'some-file.js', }, - }, { - filterLoaders: ( loader, loaderType ) => { - if ( loaderType === 'file' ) { - loader.options.publicPath = '../../'; - } - if ( loaderType === 'url' ) { - loader.test = /\.(png|jpg|jpeg|gif|svg)$/; - } - return loader; - }, } ); - const fileLoader = getLoaderByName( config.module.rules, 'file-loader' ); - const urlLoader = getLoaderByName( config.module.rules, 'url-loader' ); + const resourceLoader = getLoaderByName( config.module.rules, 'asset/resource' ); + const assetLoader = getLoaderByName( config.module.rules, 'asset' ); const jsLoader = getLoaderByName( config.module.rules, 'babel-loader' ); - expect( fileLoader ).toEqual( expect.objectContaining( { - exclude: /\.(js|html|json)$/, + expect( resourceLoader ).toEqual( expect.objectContaining( { + exclude: [ /^$/, /\.(js|mjs|jsx|ts|tsx)$/, /\.html?$/, /\.json$/ ], + type: 'asset/resource', options: { publicPath: '../../', }, } ) ); - expect( urlLoader ).toEqual( expect.objectContaining( { + expect( assetLoader ).toEqual( expect.objectContaining( { test: /\.(png|jpg|jpeg|gif|svg)$/, - options: { - limit: 10000, + type: 'asset', + parser: { + dataUrlCondition: { + maxSize: 10 * 1024, + }, }, } ) ); expect( jsLoader ).not.toBeNull(); } ); it( 'permits filtering the entire stylesheet loader chain', () => { + addFilter( 'presets/stylesheet-loaders', ( loader ) => { + loader.test = /\.styl$/; + return loader; + } ); + addFilter( 'loaders/sass', () => ( { + loader: 'stylus', + } ) ); const config = production( { entry: { main: 'some-file.js', }, - }, { - filterLoaders: ( loader, loaderType ) => { - if ( loaderType === 'stylesheet' ) { - loader.test = /\.styl$/; - } - if ( loaderType === 'sass' ) { - return { - loader: 'stylus', - }; - } - return loader; - }, } ); const styleChain = getLoaderByTest( config.module.rules, /\.styl$/ ); expect( styleChain ).toEqual( { @@ -524,5 +665,103 @@ describe( 'presets', () => { const sassLoader = getLoaderByTest( config.module.rules, /\.s?css$/ ); expect( sassLoader ).toBeNull(); } ); + + it( 'passes the preset configuration object to the stylesheet chain filter', () => { + const config = { + entry: { + main: 'some-file.js', + }, + }; + addFilter( 'presets/stylesheet-loaders', ( loader, preset, presetConfig ) => { + expect( presetConfig ).toBe( config ); + expect( preset ).toBe( 'production' ); + return loader; + } ); + production( config ); + } ); + + it( 'permits skipping a specific stylesheet loader by filtering it to null', () => { + addFilter( 'loaders/postcss', returnNull ); + addFilter( 'loaders/sass', returnNull ); + const config = production( { + entry: { + main: 'some-file.js', + }, + } ); + const styleChain = getLoaderByTest( config.module.rules, /\.s?css$/ ); + expect( styleChain ).toEqual( { + test: /\.s?css$/, + use: [ + plugins.constructors.MiniCssExtractPlugin.loader, + { + loader: require.resolve( 'css-loader' ), + options: { + importLoaders: 1, + }, + }, + ], + } ); + } ); + + it( 'does not include null loader entries if a loader was disabled with a filter', () => { + addFilter( 'presets/stylesheet-loaders', returnNull ); + addFilter( 'loaders/ts', returnNull ); + const config = production( { + entry: { + main: 'some-file.js', + }, + } ); + // Should have stripped out the null entries. + expect( config.module.rules[ 0 ].oneOf.length ).toBe( 3 ); + expect( config.module.rules[ 0 ].oneOf[ 0 ] ) + .toEqual( expect.objectContaining( { test: /\.jsx?$/ } ) ); + expect( config.module.rules[ 0 ].oneOf[ 1 ] ) + .toEqual( expect.objectContaining( { type: 'asset' } ) ); + expect( config.module.rules[ 0 ].oneOf[ 2 ] ) + .toEqual( expect.objectContaining( { type: 'asset/resource' } ) ); + } ); + } ); + + it( 'passes the preset config as argument 2 to loader filters', () => { + addFilter( 'loaders/js', ( options, config ) => { + expect( config ).not.toBeNull(); + expect( config.name ).toBe( 'dev-build' ); + return options; + } ); + production( { + name: 'dev-build', + } ); + } ); + + it( 'permits filtering only a specific invocation of a preset', () => { + addFilter( 'presets/stylesheet-loaders', returnNull ); + addFilter( 'loaders/ts', returnNull ); + + const filterToNullInBuild1 = ( options, config ) => { + if ( config.name === 'build1' ) { + return null; + } + // Simplified version to make test easier. + return options.test ? + { test: options.test } : + { type: options.type }; + }; + addFilter( 'loaders/js', filterToNullInBuild1 ); + addFilter( 'loaders/asset', filterToNullInBuild1 ); + addFilter( 'loaders/resource', filterToNullInBuild1 ); + + const config1 = production( { name: 'build1' } ); + const config2 = production( { name: 'build2' } ); + + expect( config1.module.rules ).toEqual( [] ); + expect( config2.module.rules ).toEqual( [ + { + oneOf: [ + { test: /\.jsx?$/ }, + { test: /\.(png|jpg|jpeg|gif|avif|webp|svg|woff|woff2|eot|ttf)$/ }, + { type: 'asset/resource' }, + ], + }, + ] ); } ); } ); diff --git a/test/src/.eslintrc.js b/test/src/.eslintrc.js new file mode 100644 index 0000000..de7ac3d --- /dev/null +++ b/test/src/.eslintrc.js @@ -0,0 +1,5 @@ +module.exports = { + parserOptions: { + sourceType: 'module', + }, +}; diff --git a/test/src/helpers.js b/test/src/helpers.js index 1c0daa0..393fd0b 100644 --- a/test/src/helpers.js +++ b/test/src/helpers.js @@ -4,3 +4,7 @@ export const getResults = async () => { await wait( 500 ); return 'Results'; }; + +export const lintingErrorsInProdButWarnsInDev = () => { + console.log( 'Very contrived example to hopefuly make test-build output clearer' ); +}; diff --git a/test/src/linting-test.js b/test/src/linting-test.js new file mode 100644 index 0000000..6495acc --- /dev/null +++ b/test/src/linting-test.js @@ -0,0 +1 @@ +import { lintingErrorsInProdButWarnsInDev } from './helpers'; diff --git a/test/test-config.js b/test/test-config.js index f196d43..7abf56d 100644 --- a/test/test-config.js +++ b/test/test-config.js @@ -7,6 +7,9 @@ module.exports = [ name: 'production-test', entry: { prod: filePath( 'test/src/index.js' ), + // Expect this not to build. Currently cannot do this because our CI script + // anticipates that all commands will return a 0 status. + // 'linting-test': filePath( 'test/src/linting-test.js' ), }, output: { path: filePath( 'test/build/prod' ), @@ -42,6 +45,8 @@ module.exports = [ name: 'dev-test', entry: { prod: filePath( 'test/src/index.js' ), + // Expect this to warn. + 'linting-test': filePath( 'test/src/linting-test.js' ), }, output: { path: filePath( 'test/build/dev' ),