mirror of https://github.com/webpack/webpack.git
				
				
				
			
		
			
				
	
	
		
			171 lines
		
	
	
		
			6.6 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
			
		
		
	
	
			171 lines
		
	
	
		
			6.6 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
| /*
 | |
| 	MIT License http://www.opensource.org/licenses/mit-license.php
 | |
| 	Author Tobias Koppers @sokra
 | |
| */
 | |
| var path = require("path");
 | |
| var RequestShortener = require("./RequestShortener");
 | |
| var ConcatSource = require("webpack-sources").ConcatSource;
 | |
| var RawSource = require("webpack-sources").RawSource;
 | |
| var ModuleFilenameHelpers = require("./ModuleFilenameHelpers");
 | |
| var SourceMapDevToolModuleOptionsPlugin = require("./SourceMapDevToolModuleOptionsPlugin");
 | |
| 
 | |
| function SourceMapDevToolPlugin(options) {
 | |
| 	if(arguments.length > 1)
 | |
| 		throw new Error("SourceMapDevToolPlugin only takes one argument (pass an options object)");
 | |
| 	if(typeof options === "string") {
 | |
| 		options = {
 | |
| 			sourceMapFilename: options
 | |
| 		};
 | |
| 	}
 | |
| 	if(!options) options = {};
 | |
| 	this.sourceMapFilename = options.filename;
 | |
| 	this.sourceMappingURLComment = options.append === false ? false : options.append || "\n//# sourceMappingURL=[url]";
 | |
| 	this.moduleFilenameTemplate = options.moduleFilenameTemplate || "webpack:///[resourcePath]";
 | |
| 	this.fallbackModuleFilenameTemplate = options.fallbackModuleFilenameTemplate || "webpack:///[resourcePath]?[hash]";
 | |
| 	this.options = options;
 | |
| }
 | |
| module.exports = SourceMapDevToolPlugin;
 | |
| SourceMapDevToolPlugin.prototype.apply = function(compiler) {
 | |
| 	var sourceMapFilename = this.sourceMapFilename;
 | |
| 	var sourceMappingURLComment = this.sourceMappingURLComment;
 | |
| 	var moduleFilenameTemplate = this.moduleFilenameTemplate;
 | |
| 	var fallbackModuleFilenameTemplate = this.fallbackModuleFilenameTemplate;
 | |
| 	var requestShortener = new RequestShortener(compiler.context);
 | |
| 	var options = this.options;
 | |
| 	options.test = options.test || /\.(js|css)($|\?)/i;
 | |
| 	compiler.plugin("compilation", function(compilation) {
 | |
| 		new SourceMapDevToolModuleOptionsPlugin(options).apply(compilation);
 | |
| 		compilation.plugin("after-optimize-chunk-assets", function(chunks) {
 | |
| 			var allModules = [];
 | |
| 			var allModuleFilenames = [];
 | |
| 			var tasks = [];
 | |
| 			chunks.forEach(function(chunk) {
 | |
| 				chunk.files.filter(ModuleFilenameHelpers.matchObject.bind(undefined, options)).map(function(file) {
 | |
| 					var asset = this.assets[file];
 | |
| 					if(asset.__SourceMapDevToolData) {
 | |
| 						var data = asset.__SourceMapDevToolData;
 | |
| 						for(var cachedFile in data) {
 | |
| 							this.assets[cachedFile] = data[cachedFile];
 | |
| 							if(cachedFile !== file)
 | |
| 								chunk.files.push(cachedFile);
 | |
| 						}
 | |
| 						return;
 | |
| 					}
 | |
| 					var source;
 | |
| 					var sourceMap;
 | |
| 					if(asset.sourceAndMap) {
 | |
| 						var sourceAndMap = asset.sourceAndMap(options);
 | |
| 						sourceMap = sourceAndMap.map;
 | |
| 						source = sourceAndMap.source;
 | |
| 					} else {
 | |
| 						sourceMap = asset.map(options);
 | |
| 						source = asset.source();
 | |
| 					}
 | |
| 					if(sourceMap) {
 | |
| 						return {
 | |
| 							chunk: chunk,
 | |
| 							file: file,
 | |
| 							asset: asset,
 | |
| 							source: source,
 | |
| 							sourceMap: sourceMap
 | |
| 						};
 | |
| 					}
 | |
| 				}, this).filter(Boolean).map(function(task) {
 | |
| 					var modules = task.sourceMap.sources.map(function(source) {
 | |
| 						var module = compilation.findModule(source);
 | |
| 						return module || source;
 | |
| 					});
 | |
| 					var moduleFilenames = modules.map(function(module) {
 | |
| 						return ModuleFilenameHelpers.createFilename(module, moduleFilenameTemplate, requestShortener);
 | |
| 					});
 | |
| 					task.modules = modules;
 | |
| 					task.moduleFilenames = moduleFilenames;
 | |
| 					return task;
 | |
| 				}, this).forEach(function(task) {
 | |
| 					allModules = allModules.concat(task.modules);
 | |
| 					allModuleFilenames = allModuleFilenames.concat(task.moduleFilenames);
 | |
| 					tasks.push(task);
 | |
| 				}, this);
 | |
| 			}, this);
 | |
| 			allModuleFilenames = ModuleFilenameHelpers.replaceDuplicates(allModuleFilenames, function(filename, i) {
 | |
| 				return ModuleFilenameHelpers.createFilename(allModules[i], fallbackModuleFilenameTemplate, requestShortener);
 | |
| 			}, function(ai, bi) {
 | |
| 				var a = allModules[ai];
 | |
| 				var b = allModules[bi];
 | |
| 				a = !a ? "" : typeof a === "string" ? a : a.identifier();
 | |
| 				b = !b ? "" : typeof b === "string" ? b : b.identifier();
 | |
| 				return a.length - b.length;
 | |
| 			});
 | |
| 			allModuleFilenames = ModuleFilenameHelpers.replaceDuplicates(allModuleFilenames, function(filename, i, n) {
 | |
| 				for(var j = 0; j < n; j++)
 | |
| 					filename += "*";
 | |
| 				return filename;
 | |
| 			});
 | |
| 			tasks.forEach(function(task) {
 | |
| 				task.moduleFilenames = allModuleFilenames.slice(0, task.moduleFilenames.length);
 | |
| 				allModuleFilenames = allModuleFilenames.slice(task.moduleFilenames.length);
 | |
| 			}, this);
 | |
| 			tasks.forEach(function(task) {
 | |
| 				var chunk = task.chunk;
 | |
| 				var file = task.file;
 | |
| 				var asset = task.asset;
 | |
| 				var sourceMap = task.sourceMap;
 | |
| 				var source = task.source;
 | |
| 				var moduleFilenames = task.moduleFilenames;
 | |
| 				var modules = task.modules;
 | |
| 				sourceMap.sources = moduleFilenames;
 | |
| 				if(sourceMap.sourcesContent && !options.noSources) {
 | |
| 					sourceMap.sourcesContent = sourceMap.sourcesContent.map(function(content, i) {
 | |
| 						return content + "\n\n\n" + ModuleFilenameHelpers.createFooter(modules[i], requestShortener);
 | |
| 					});
 | |
| 				} else {
 | |
| 					sourceMap.sourcesContent = undefined;
 | |
| 				}
 | |
| 				sourceMap.sourceRoot = options.sourceRoot || "";
 | |
| 				sourceMap.file = file;
 | |
| 				asset.__SourceMapDevToolData = {};
 | |
| 				var currentSourceMappingURLComment = sourceMappingURLComment;
 | |
| 				if(currentSourceMappingURLComment !== false && /\.css($|\?)/i.test(file)) {
 | |
| 					currentSourceMappingURLComment = currentSourceMappingURLComment.replace(/^\n\/\/(.*)$/, "\n/*$1*/");
 | |
| 				}
 | |
| 				if(sourceMapFilename) {
 | |
| 					var filename = file,
 | |
| 						query = "";
 | |
| 					var idx = filename.indexOf("?");
 | |
| 					if(idx >= 0) {
 | |
| 						query = filename.substr(idx);
 | |
| 						filename = filename.substr(0, idx);
 | |
| 					}
 | |
| 					var sourceMapFile = this.getPath(sourceMapFilename, {
 | |
| 						chunk: chunk,
 | |
| 						filename: filename,
 | |
| 						query: query,
 | |
| 						basename: basename(filename)
 | |
| 					});
 | |
| 					var sourceMapUrl = path.relative(path.dirname(file), sourceMapFile).replace(/\\/g, "/");
 | |
| 					if(currentSourceMappingURLComment !== false) {
 | |
| 						asset.__SourceMapDevToolData[file] = this.assets[file] = new ConcatSource(new RawSource(source), currentSourceMappingURLComment.replace(/\[url\]/g, sourceMapUrl));
 | |
| 					}
 | |
| 					asset.__SourceMapDevToolData[sourceMapFile] = this.assets[sourceMapFile] = new RawSource(JSON.stringify(sourceMap));
 | |
| 					chunk.files.push(sourceMapFile);
 | |
| 				} else {
 | |
| 					asset.__SourceMapDevToolData[file] = this.assets[file] = new ConcatSource(new RawSource(source), currentSourceMappingURLComment
 | |
| 						.replace(/\[map\]/g, function() {
 | |
| 							return JSON.stringify(sourceMap);
 | |
| 						})
 | |
| 						.replace(/\[url\]/g, function() {
 | |
| 							return "data:application/json;charset=utf-8;base64," +
 | |
| 								Buffer.from(JSON.stringify(sourceMap)).toString("base64");
 | |
| 						})
 | |
| 					);
 | |
| 				}
 | |
| 			}, this);
 | |
| 		});
 | |
| 	});
 | |
| };
 | |
| 
 | |
| function basename(name) {
 | |
| 	if(name.indexOf("/") < 0) return name;
 | |
| 	return name.substr(name.lastIndexOf("/") + 1);
 | |
| }
 |