webpack/lib/SourceMapDevToolPlugin.js

170 lines
6.9 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");
var SourceMapDevToolModuleOptionsPlugin = require("./SourceMapDevToolModuleOptionsPlugin");
2013-03-26 23:54:41 +08:00
2015-02-19 08:11:29 +08:00
function SourceMapDevToolPlugin(options, sourceMappingURLComment, moduleFilenameTemplate, fallbackModuleFilenameTemplate) {
2015-03-04 14:45:26 +08:00
if(!options || typeof options !== "object") {
2015-02-19 08:11:29 +08:00
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 = {};
2015-02-19 08:11:29 +08:00
} 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;
2015-02-19 08:11:29 +08:00
}
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);
var options = this.options;
2015-03-21 02:00:39 +08:00
options.test = options.test || /\.js($|\?)/i;
2013-03-26 23:54:41 +08:00
compiler.plugin("compilation", function(compilation) {
new SourceMapDevToolModuleOptionsPlugin(options).apply(compilation);
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) {
2015-03-21 02:00:39 +08:00
chunk.files.filter(ModuleFilenameHelpers.matchObject.bind(undefined, options)).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;
}
2015-04-03 18:38:56 +08:00
if(asset.sourceAndMap) {
var sourceAndMap = asset.sourceAndMap(options);
2015-04-03 18:38:56 +08:00
var sourceMap = sourceAndMap.map;
var source = sourceAndMap.source;
} else {
var sourceMap = asset.map(options);
2015-04-03 18:38:56 +08:00
var source = asset.source();
}
2013-03-26 23:54:41 +08:00
if(sourceMap) {
return {
chunk: chunk,
file: file,
asset: asset,
2015-04-03 18:38:56 +08:00
source: source,
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];
2015-04-08 21:10:09 +08:00
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;
2015-04-03 18:38:56 +08:00
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);
});
2015-02-19 08:11:29 +08:00
} 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);
}
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) {
2015-04-03 18:38:56 +08:00
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 {
2015-04-03 18:38:56 +08:00
asset.__SourceMapDevTool_Data[file] = this.assets[file] = new ConcatSource(new RawSource(source), currentSourceMappingURLComment
2015-02-19 08:11:29 +08:00
.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);
2013-03-26 23:54:41 +08:00
});
});
};
function basename(name) {
if(name.indexOf("/") < 0) return name;
return name.substr(name.lastIndexOf("/")+1);
}