You cannot select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
	
	
		
			175 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			JavaScript
		
	
		
		
			
		
	
	
			175 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			JavaScript
		
	
| 
											9 months ago
										 | /* | ||
|  | 	MIT License http://www.opensource.org/licenses/mit-license.php
 | ||
|  | 	Author Tobias Koppers @sokra | ||
|  | */ | ||
|  | 
 | ||
|  | "use strict"; | ||
|  | 
 | ||
|  | const parseJson = require("json-parse-even-better-errors"); | ||
|  | const DelegatedModuleFactoryPlugin = require("./DelegatedModuleFactoryPlugin"); | ||
|  | const ExternalModuleFactoryPlugin = require("./ExternalModuleFactoryPlugin"); | ||
|  | const WebpackError = require("./WebpackError"); | ||
|  | const DelegatedSourceDependency = require("./dependencies/DelegatedSourceDependency"); | ||
|  | const createSchemaValidation = require("./util/create-schema-validation"); | ||
|  | const makePathsRelative = require("./util/identifier").makePathsRelative; | ||
|  | 
 | ||
|  | /** @typedef {import("../declarations/WebpackOptions").Externals} Externals */ | ||
|  | /** @typedef {import("../declarations/plugins/DllReferencePlugin").DllReferencePluginOptions} DllReferencePluginOptions */ | ||
|  | /** @typedef {import("../declarations/plugins/DllReferencePlugin").DllReferencePluginOptionsManifest} DllReferencePluginOptionsManifest */ | ||
|  | /** @typedef {import("./Compiler")} Compiler */ | ||
|  | 
 | ||
|  | const validate = createSchemaValidation( | ||
|  | 	require("../schemas/plugins/DllReferencePlugin.check.js"), | ||
|  | 	() => require("../schemas/plugins/DllReferencePlugin.json"), | ||
|  | 	{ | ||
|  | 		name: "Dll Reference Plugin", | ||
|  | 		baseDataPath: "options" | ||
|  | 	} | ||
|  | ); | ||
|  | 
 | ||
|  | class DllReferencePlugin { | ||
|  | 	/** | ||
|  | 	 * @param {DllReferencePluginOptions} options options object | ||
|  | 	 */ | ||
|  | 	constructor(options) { | ||
|  | 		validate(options); | ||
|  | 		this.options = options; | ||
|  | 		/** @type {WeakMap<Object, {path: string, data: DllReferencePluginOptionsManifest?, error: Error?}>} */ | ||
|  | 		this._compilationData = new WeakMap(); | ||
|  | 	} | ||
|  | 
 | ||
|  | 	/** | ||
|  | 	 * Apply the plugin | ||
|  | 	 * @param {Compiler} compiler the compiler instance | ||
|  | 	 * @returns {void} | ||
|  | 	 */ | ||
|  | 	apply(compiler) { | ||
|  | 		compiler.hooks.compilation.tap( | ||
|  | 			"DllReferencePlugin", | ||
|  | 			(compilation, { normalModuleFactory }) => { | ||
|  | 				compilation.dependencyFactories.set( | ||
|  | 					DelegatedSourceDependency, | ||
|  | 					normalModuleFactory | ||
|  | 				); | ||
|  | 			} | ||
|  | 		); | ||
|  | 
 | ||
|  | 		compiler.hooks.beforeCompile.tapAsync( | ||
|  | 			"DllReferencePlugin", | ||
|  | 			(params, callback) => { | ||
|  | 				if ("manifest" in this.options) { | ||
|  | 					const manifest = this.options.manifest; | ||
|  | 					if (typeof manifest === "string") { | ||
|  | 						compiler.inputFileSystem.readFile(manifest, (err, result) => { | ||
|  | 							if (err) return callback(err); | ||
|  | 							const data = { | ||
|  | 								path: manifest, | ||
|  | 								data: undefined, | ||
|  | 								error: undefined | ||
|  | 							}; | ||
|  | 							// Catch errors parsing the manifest so that blank
 | ||
|  | 							// or malformed manifest files don't kill the process.
 | ||
|  | 							try { | ||
|  | 								data.data = parseJson(result.toString("utf-8")); | ||
|  | 							} catch (e) { | ||
|  | 								// Store the error in the params so that it can
 | ||
|  | 								// be added as a compilation error later on.
 | ||
|  | 								const manifestPath = makePathsRelative( | ||
|  | 									compiler.options.context, | ||
|  | 									manifest, | ||
|  | 									compiler.root | ||
|  | 								); | ||
|  | 								data.error = new DllManifestError(manifestPath, e.message); | ||
|  | 							} | ||
|  | 							this._compilationData.set(params, data); | ||
|  | 							return callback(); | ||
|  | 						}); | ||
|  | 						return; | ||
|  | 					} | ||
|  | 				} | ||
|  | 				return callback(); | ||
|  | 			} | ||
|  | 		); | ||
|  | 
 | ||
|  | 		compiler.hooks.compile.tap("DllReferencePlugin", params => { | ||
|  | 			let name = this.options.name; | ||
|  | 			let sourceType = this.options.sourceType; | ||
|  | 			let content = | ||
|  | 				"content" in this.options ? this.options.content : undefined; | ||
|  | 			if ("manifest" in this.options) { | ||
|  | 				let manifestParameter = this.options.manifest; | ||
|  | 				let manifest; | ||
|  | 				if (typeof manifestParameter === "string") { | ||
|  | 					const data = this._compilationData.get(params); | ||
|  | 					// If there was an error parsing the manifest
 | ||
|  | 					// file, exit now because the error will be added
 | ||
|  | 					// as a compilation error in the "compilation" hook.
 | ||
|  | 					if (data.error) { | ||
|  | 						return; | ||
|  | 					} | ||
|  | 					manifest = data.data; | ||
|  | 				} else { | ||
|  | 					manifest = manifestParameter; | ||
|  | 				} | ||
|  | 				if (manifest) { | ||
|  | 					if (!name) name = manifest.name; | ||
|  | 					if (!sourceType) sourceType = manifest.type; | ||
|  | 					if (!content) content = manifest.content; | ||
|  | 				} | ||
|  | 			} | ||
|  | 			/** @type {Externals} */ | ||
|  | 			const externals = {}; | ||
|  | 			const source = "dll-reference " + name; | ||
|  | 			externals[source] = name; | ||
|  | 			const normalModuleFactory = params.normalModuleFactory; | ||
|  | 			new ExternalModuleFactoryPlugin(sourceType || "var", externals).apply( | ||
|  | 				normalModuleFactory | ||
|  | 			); | ||
|  | 			new DelegatedModuleFactoryPlugin({ | ||
|  | 				source: source, | ||
|  | 				type: this.options.type, | ||
|  | 				scope: this.options.scope, | ||
|  | 				context: this.options.context || compiler.options.context, | ||
|  | 				content, | ||
|  | 				extensions: this.options.extensions, | ||
|  | 				associatedObjectForCache: compiler.root | ||
|  | 			}).apply(normalModuleFactory); | ||
|  | 		}); | ||
|  | 
 | ||
|  | 		compiler.hooks.compilation.tap( | ||
|  | 			"DllReferencePlugin", | ||
|  | 			(compilation, params) => { | ||
|  | 				if ("manifest" in this.options) { | ||
|  | 					let manifest = this.options.manifest; | ||
|  | 					if (typeof manifest === "string") { | ||
|  | 						const data = this._compilationData.get(params); | ||
|  | 						// If there was an error parsing the manifest file, add the
 | ||
|  | 						// error as a compilation error to make the compilation fail.
 | ||
|  | 						if (data.error) { | ||
|  | 							compilation.errors.push( | ||
|  | 								/** @type {DllManifestError} */ (data.error) | ||
|  | 							); | ||
|  | 						} | ||
|  | 						compilation.fileDependencies.add(manifest); | ||
|  | 					} | ||
|  | 				} | ||
|  | 			} | ||
|  | 		); | ||
|  | 	} | ||
|  | } | ||
|  | 
 | ||
|  | class DllManifestError extends WebpackError { | ||
|  | 	/** | ||
|  | 	 * @param {string} filename filename of the manifest | ||
|  | 	 * @param {string} message error message | ||
|  | 	 */ | ||
|  | 	constructor(filename, message) { | ||
|  | 		super(); | ||
|  | 
 | ||
|  | 		this.name = "DllManifestError"; | ||
|  | 		this.message = `Dll manifest ${filename}\n${message}`; | ||
|  | 	} | ||
|  | } | ||
|  | 
 | ||
|  | module.exports = DllReferencePlugin; |