mirror of https://github.com/webpack/webpack.git
170 lines
6.9 KiB
JavaScript
170 lines
6.9 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 Template = require("./Template");
|
|
var ConcatSource = require("webpack-core/lib/ConcatSource");
|
|
var RawSource = require("webpack-core/lib/RawSource");
|
|
var ModuleFilenameHelpers = require("./ModuleFilenameHelpers");
|
|
var SourceMapDevToolModuleOptionsPlugin = require("./SourceMapDevToolModuleOptionsPlugin");
|
|
|
|
function SourceMapDevToolPlugin(options, sourceMappingURLComment, moduleFilenameTemplate, fallbackModuleFilenameTemplate) {
|
|
if(!options || typeof options !== "object") {
|
|
this.sourceMapFilename = options;
|
|
this.sourceMappingURLComment = sourceMappingURLComment === false ? false : sourceMappingURLComment || "\n//# sourceMappingURL=[url]";
|
|
this.moduleFilenameTemplate = moduleFilenameTemplate || "webpack:///[resourcePath]";
|
|
this.fallbackModuleFilenameTemplate = fallbackModuleFilenameTemplate || "webpack:///[resourcePath]?[hash]";
|
|
this.options = {};
|
|
} else {
|
|
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($|\?)/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.__SourceMapDevTool_Data) {
|
|
var data = asset.__SourceMapDevTool_Data;
|
|
for(var cachedFile in data) {
|
|
this.assets[cachedFile] = data[cachedFile];
|
|
if(cachedFile !== file)
|
|
chunk.files.push(cachedFile);
|
|
}
|
|
return;
|
|
}
|
|
if(asset.sourceAndMap) {
|
|
var sourceAndMap = asset.sourceAndMap(options);
|
|
var sourceMap = sourceAndMap.map;
|
|
var source = sourceAndMap.source;
|
|
} else {
|
|
var sourceMap = asset.map(options);
|
|
var 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) {
|
|
sourceMap.sourcesContent = sourceMap.sourcesContent.map(function(content, i) {
|
|
return content + "\n\n\n" + ModuleFilenameHelpers.createFooter(modules[i], requestShortener);
|
|
});
|
|
} else {
|
|
sourceMap.sourcesContent = undefined;
|
|
}
|
|
sourceMap.sourceRoot = "";
|
|
sourceMap.file = file;
|
|
asset.__SourceMapDevTool_Data = {};
|
|
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.__SourceMapDevTool_Data[file] = this.assets[file] = new ConcatSource(new RawSource(source), currentSourceMappingURLComment.replace(/\[url\]/g, sourceMapUrl));
|
|
}
|
|
asset.__SourceMapDevTool_Data[sourceMapFile] = this.assets[sourceMapFile] = new RawSource(JSON.stringify(sourceMap));
|
|
chunk.files.push(sourceMapFile);
|
|
} else {
|
|
asset.__SourceMapDevTool_Data[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;base64," +
|
|
new Buffer(JSON.stringify(sourceMap)).toString("base64");
|
|
})
|
|
);
|
|
}
|
|
}, this);
|
|
});
|
|
});
|
|
};
|
|
|
|
function basename(name) {
|
|
if(name.indexOf("/") < 0) return name;
|
|
return name.substr(name.lastIndexOf("/")+1);
|
|
}
|
|
|