2012-03-12 04:50:55 +08:00
/ *
MIT License http : //www.opensource.org/licenses/mit-license.php
Author Tobias Koppers @ sokra
* /
2018-07-30 23:08:51 +08:00
2017-01-02 03:19:00 +08:00
"use strict" ;
2020-09-30 02:20:19 +08:00
const util = require ( "util" ) ;
2025-07-02 20:10:54 +08:00
const webpackOptionsSchemaCheck = require ( "../schemas/WebpackOptions.check" ) ;
2018-07-30 23:08:51 +08:00
const webpackOptionsSchema = require ( "../schemas/WebpackOptions.json" ) ;
2017-01-02 03:19:00 +08:00
const Compiler = require ( "./Compiler" ) ;
const MultiCompiler = require ( "./MultiCompiler" ) ;
const WebpackOptionsApply = require ( "./WebpackOptionsApply" ) ;
2020-02-17 17:27:46 +08:00
const {
2025-07-03 17:06:45 +08:00
applyWebpackOptionsBaseDefaults ,
applyWebpackOptionsDefaults
2020-02-17 17:27:46 +08:00
} = require ( "./config/defaults" ) ;
const { getNormalizedWebpackOptions } = require ( "./config/normalization" ) ;
2018-07-30 23:08:51 +08:00
const NodeEnvironmentPlugin = require ( "./node/NodeEnvironmentPlugin" ) ;
2021-04-16 21:35:18 +08:00
const memoize = require ( "./util/memoize" ) ;
2018-02-27 01:22:25 +08:00
2018-09-19 19:05:22 +08:00
/** @typedef {import("../declarations/WebpackOptions").WebpackOptions} WebpackOptions */
2024-02-17 01:39:12 +08:00
/** @typedef {import("../declarations/WebpackOptions").WebpackPluginFunction} WebpackPluginFunction */
2019-11-08 18:13:16 +08:00
/** @typedef {import("./Compiler").WatchOptions} WatchOptions */
2021-02-14 02:53:50 +08:00
/** @typedef {import("./MultiCompiler").MultiCompilerOptions} MultiCompilerOptions */
2025-07-18 20:28:16 +08:00
/** @typedef {import("./MultiCompiler").MultiWebpackOptions} MultiWebpackOptions */
2018-12-10 18:34:59 +08:00
/** @typedef {import("./MultiStats")} MultiStats */
2018-09-27 13:22:19 +08:00
/** @typedef {import("./Stats")} Stats */
2018-09-19 19:05:22 +08:00
2021-04-16 21:35:18 +08:00
const getValidateSchema = memoize ( ( ) => require ( "./validateSchema" ) ) ;
2018-12-09 21:26:35 +08:00
/ * *
2018-12-10 18:34:59 +08:00
* @ template T
* @ callback Callback
2024-02-21 22:05:29 +08:00
* @ param { Error | null } err
2018-12-10 18:34:59 +08:00
* @ param { T = } stats
2018-12-09 21:26:35 +08:00
* @ returns { void }
* /
/ * *
2021-03-08 16:06:04 +08:00
* @ param { ReadonlyArray < WebpackOptions > } childOptions options array
2021-02-14 02:53:50 +08:00
* @ param { MultiCompilerOptions } options options
2018-12-09 21:26:35 +08:00
* @ returns { MultiCompiler } a multi - compiler
* /
2021-02-14 02:53:50 +08:00
const createMultiCompiler = ( childOptions , options ) => {
2024-05-08 00:01:41 +08:00
const compilers = childOptions . map ( ( options , index ) =>
createCompiler ( options , index )
) ;
2021-02-14 02:53:50 +08:00
const compiler = new MultiCompiler ( compilers , options ) ;
2018-12-09 21:26:35 +08:00
for ( const childCompiler of compilers ) {
if ( childCompiler . options . dependencies ) {
compiler . setDependencies (
childCompiler ,
childCompiler . options . dependencies
) ;
}
}
return compiler ;
} ;
2018-09-19 19:05:22 +08:00
/ * *
2020-02-17 17:27:46 +08:00
* @ param { WebpackOptions } rawOptions options object
2025-04-16 22:04:11 +08:00
* @ param { number = } compilerIndex index of compiler
2018-12-09 21:26:35 +08:00
* @ returns { Compiler } a compiler
* /
2024-05-08 00:03:46 +08:00
const createCompiler = ( rawOptions , compilerIndex ) => {
2020-02-17 17:27:46 +08:00
const options = getNormalizedWebpackOptions ( rawOptions ) ;
applyWebpackOptionsBaseDefaults ( options ) ;
2023-06-12 22:21:21 +08:00
const compiler = new Compiler (
/** @type {string} */ ( options . context ) ,
options
) ;
2019-07-24 16:51:04 +08:00
new NodeEnvironmentPlugin ( {
infrastructureLogging : options . infrastructureLogging
} ) . apply ( compiler ) ;
2018-12-09 21:26:35 +08:00
if ( Array . isArray ( options . plugins ) ) {
for ( const plugin of options . plugins ) {
if ( typeof plugin === "function" ) {
2024-02-17 01:39:12 +08:00
/** @type {WebpackPluginFunction} */
( plugin ) . call ( compiler , compiler ) ;
2023-06-07 09:07:31 +08:00
} else if ( plugin ) {
2018-12-09 21:26:35 +08:00
plugin . apply ( compiler ) ;
}
}
}
2024-06-07 22:29:10 +08:00
const resolvedDefaultOptions = applyWebpackOptionsDefaults (
options ,
compilerIndex
) ;
if ( resolvedDefaultOptions . platform ) {
compiler . platform = resolvedDefaultOptions . platform ;
2024-05-01 22:22:45 +08:00
}
2018-12-09 21:26:35 +08:00
compiler . hooks . environment . call ( ) ;
compiler . hooks . afterEnvironment . call ( ) ;
2020-02-17 17:27:46 +08:00
new WebpackOptionsApply ( ) . process ( options , compiler ) ;
compiler . hooks . initialize . call ( ) ;
2018-12-09 21:26:35 +08:00
return compiler ;
} ;
/ * *
2020-04-21 14:54:59 +08:00
* @ callback WebpackFunctionSingle
* @ param { WebpackOptions } options options object
* @ param { Callback < Stats >= } callback callback
2025-07-18 20:28:16 +08:00
* @ returns { Compiler | null } the compiler object
2018-09-19 19:05:22 +08:00
* /
2020-04-21 14:54:59 +08:00
/ * *
* @ callback WebpackFunctionMulti
2025-07-18 20:28:16 +08:00
* @ param { MultiWebpackOptions } options options objects
2020-04-21 14:54:59 +08:00
* @ param { Callback < MultiStats >= } callback callback
2025-07-18 20:28:16 +08:00
* @ returns { MultiCompiler | null } the multi compiler object
2020-04-21 14:54:59 +08:00
* /
2023-06-12 22:21:21 +08:00
/ * *
* @ template T
* @ param { Array < T > | T } options options
* @ returns { Array < T > } array of options
* /
2025-07-17 00:13:14 +08:00
const asArray = ( options ) =>
Array . isArray ( options ) ? [ ... options ] : [ options ] ;
2021-11-24 15:32:07 +08:00
2025-07-18 20:28:16 +08:00
/ * *
* @ callback WebpackCallback
* @ param { WebpackOptions | MultiWebpackOptions } options options
* @ param { Callback < Stats > & Callback < MultiStats >= } callback callback
* @ returns { Compiler | MultiCompiler | null } Compiler or MultiCompiler
* /
2021-02-14 02:53:50 +08:00
const webpack = /** @type {WebpackFunctionSingle & WebpackFunctionMulti} */ (
2025-07-18 20:28:16 +08:00
/** @type {WebpackCallback} */
2021-02-14 02:53:50 +08:00
( options , callback ) => {
const create = ( ) => {
2021-11-24 15:32:07 +08:00
if ( ! asArray ( options ) . every ( webpackOptionsSchemaCheck ) ) {
2021-04-16 21:35:18 +08:00
getValidateSchema ( ) ( webpackOptionsSchema , options ) ;
2021-11-24 15:32:07 +08:00
util . deprecate (
( ) => { } ,
"webpack bug: Pre-compiled schema reports error while real schema is happy. This has performance drawbacks." ,
"DEP_WEBPACK_PRE_COMPILED_SCHEMA_INVALID"
) ( ) ;
2021-04-16 21:35:18 +08:00
}
2021-02-14 02:53:50 +08:00
/** @type {MultiCompiler|Compiler} */
let compiler ;
2023-06-12 22:21:21 +08:00
/** @type {boolean | undefined} */
2021-02-14 02:53:50 +08:00
let watch = false ;
/** @type {WatchOptions|WatchOptions[]} */
let watchOptions ;
if ( Array . isArray ( options ) ) {
/** @type {MultiCompiler} */
2021-05-18 23:10:36 +08:00
compiler = createMultiCompiler (
options ,
/** @type {MultiCompilerOptions} */ ( options )
) ;
2025-07-17 00:13:14 +08:00
watch = options . some ( ( options ) => options . watch ) ;
watchOptions = options . map ( ( options ) => options . watchOptions || { } ) ;
2020-09-28 16:51:12 +08:00
} else {
2021-04-02 05:52:21 +08:00
const webpackOptions = /** @type {WebpackOptions} */ ( options ) ;
2021-02-14 02:53:50 +08:00
/** @type {Compiler} */
2021-03-08 16:06:04 +08:00
compiler = createCompiler ( webpackOptions ) ;
watch = webpackOptions . watch ;
watchOptions = webpackOptions . watchOptions || { } ;
2021-02-14 02:53:50 +08:00
}
return { compiler , watch , watchOptions } ;
} ;
if ( callback ) {
try {
const { compiler , watch , watchOptions } = create ( ) ;
if ( watch ) {
compiler . watch ( watchOptions , callback ) ;
} else {
compiler . run ( ( err , stats ) => {
2025-07-17 00:13:14 +08:00
compiler . close ( ( err2 ) => {
2024-02-21 22:55:02 +08:00
callback (
err || err2 ,
/** @type {options extends WebpackOptions ? Stats : MultiStats} */
( stats )
) ;
2021-02-14 02:53:50 +08:00
} ) ;
2020-09-28 16:51:12 +08:00
} ) ;
2021-02-14 02:53:50 +08:00
}
return compiler ;
} catch ( err ) {
2024-02-17 01:39:12 +08:00
process . nextTick ( ( ) => callback ( /** @type {Error} */ ( err ) ) ) ;
2021-02-14 02:53:50 +08:00
return null ;
}
} else {
const { compiler , watch } = create ( ) ;
if ( watch ) {
util . deprecate (
( ) => { } ,
2021-07-12 18:15:59 +08:00
"A 'callback' argument needs to be provided to the 'webpack(options, callback)' function when the 'watch' option is set. There is no way to handle the 'watch' option without a callback." ,
2021-02-14 02:53:50 +08:00
"DEP_WEBPACK_WATCH_WITHOUT_CALLBACK"
) ( ) ;
2020-09-28 16:51:12 +08:00
}
return compiler ;
2018-12-09 21:26:35 +08:00
}
2012-10-20 21:08:12 +08:00
}
2021-02-14 02:53:50 +08:00
) ;
2017-11-08 18:32:05 +08:00
2018-12-09 21:26:35 +08:00
module . exports = webpack ;