webpack/lib/SourceMapDevToolPlugin.js

141 lines
5.7 KiB
JavaScript
Raw Normal View History

2013-03-26 23:54:41 +08:00
/*
MIT License http://www.opensource.org/licenses/mit-license.php
Author Tobias Koppers @sokra
*/
var path = require("path");
2013-03-26 23:54:41 +08:00
var RequestShortener = require("./RequestShortener");
var Template = require("./Template");
var ConcatSource = require("webpack-core/lib/ConcatSource");
var RawSource = require("webpack-core/lib/RawSource");
2014-07-18 19:31:50 +08:00
var ModuleFilenameHelpers = require("./ModuleFilenameHelpers");
2013-03-26 23:54:41 +08:00
2014-07-18 19:31:50 +08:00
function SourceMapDevToolPlugin(sourceMapFilename, sourceMappingURLComment, moduleFilenameTemplate, fallbackModuleFilenameTemplate) {
2013-03-26 23:54:41 +08:00
this.sourceMapFilename = sourceMapFilename;
this.sourceMappingURLComment = sourceMappingURLComment === false ? false : sourceMappingURLComment || "\n//# sourceMappingURL=[url]";
2014-07-18 19:31:50 +08:00
this.moduleFilenameTemplate = moduleFilenameTemplate || "webpack:///[resourcePath]";
this.fallbackModuleFilenameTemplate = fallbackModuleFilenameTemplate || "webpack:///[resourcePath]?[hash]";
2013-03-26 23:54:41 +08:00
}
module.exports = SourceMapDevToolPlugin;
SourceMapDevToolPlugin.prototype.apply = function(compiler) {
var sourceMapFilename = this.sourceMapFilename;
var sourceMappingURLComment = this.sourceMappingURLComment;
2014-07-18 19:31:50 +08:00
var moduleFilenameTemplate = this.moduleFilenameTemplate;
var fallbackModuleFilenameTemplate = this.fallbackModuleFilenameTemplate;
var requestShortener = new RequestShortener(compiler.context);
2013-03-26 23:54:41 +08:00
compiler.plugin("compilation", function(compilation) {
2013-10-14 19:59:44 +08:00
compilation.plugin("build-module", function(module) {
module.useSourceMap = true;
});
2013-03-26 23:54:41 +08:00
compilation.plugin("after-optimize-chunk-assets", function(chunks) {
var allModules = [];
var allModuleFilenames = [];
var tasks = [];
2013-03-26 23:54:41 +08:00
chunks.forEach(function(chunk) {
chunk.files.slice().map(function(file) {
2013-03-26 23:54:41 +08:00
var asset = this.assets[file];
2013-07-04 17:55:37 +08:00
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);
2013-07-04 17:55:37 +08:00
}
return;
}
2013-03-26 23:54:41 +08:00
var sourceMap = asset.map();
if(sourceMap) {
return {
chunk: chunk,
file: file,
asset: asset,
sourceMap: sourceMap
2013-03-26 23:54:41 +08:00
}
}
}, 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);
2013-03-26 23:54:41 +08:00
}, 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 = typeof a === "string" ? a : a.identifier();
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 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);
});
}
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);
}
2014-08-22 19:51:24 +08:00
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(asset, 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(asset, currentSourceMappingURLComment.replace(/\[url\]/g, "data:application/json;base64," + new Buffer(JSON.stringify(sourceMap)).toString("base64")));
}
}, this);
2013-03-26 23:54:41 +08:00
});
});
};
function basename(name) {
if(name.indexOf("/") < 0) return name;
return name.substr(name.lastIndexOf("/")+1);
}