mirror of
https://github.com/zhigang1992/react-native.git
synced 2026-02-14 22:36:21 +08:00
301 lines
8.0 KiB
JavaScript
301 lines
8.0 KiB
JavaScript
/**
|
|
* Copyright (c) 2015-present, Facebook, Inc.
|
|
* All rights reserved.
|
|
*
|
|
* This source code is licensed under the BSD-style license found in the
|
|
* LICENSE file in the root directory of this source tree. An additional grant
|
|
* of patent rights can be found in the PATENTS file in the same directory.
|
|
*/
|
|
'use strict';
|
|
|
|
var _ = require('underscore');
|
|
var base64VLQ = require('./base64-vlq');
|
|
var UglifyJS = require('uglify-js');
|
|
var ModuleTransport = require('../lib/ModuleTransport');
|
|
|
|
module.exports = Package;
|
|
|
|
var SOURCEMAPPING_URL = '\n\/\/@ sourceMappingURL=';
|
|
|
|
function Package(sourceMapUrl) {
|
|
this._finalized = false;
|
|
this._modules = [];
|
|
this._assets = [];
|
|
this._sourceMapUrl = sourceMapUrl;
|
|
this._shouldCombineSourceMaps = false;
|
|
}
|
|
|
|
Package.prototype.setMainModuleId = function(moduleId) {
|
|
this._mainModuleId = moduleId;
|
|
};
|
|
|
|
Package.prototype.addModule = function(module) {
|
|
if (!(module instanceof ModuleTransport)) {
|
|
throw new Error('Expeceted a ModuleTransport object');
|
|
}
|
|
|
|
// If we get a map from the transformer we'll switch to a mode
|
|
// were we're combining the source maps as opposed to
|
|
if (!this._shouldCombineSourceMaps && module.map != null) {
|
|
this._shouldCombineSourceMaps = true;
|
|
}
|
|
|
|
this._modules.push(module);
|
|
};
|
|
|
|
Package.prototype.addAsset = function(asset) {
|
|
this._assets.push(asset);
|
|
};
|
|
|
|
Package.prototype.finalize = function(options) {
|
|
options = options || {};
|
|
if (options.runMainModule) {
|
|
var runCode = ';require("' + this._mainModuleId + '");';
|
|
this.addModule(new ModuleTransport({
|
|
code: runCode,
|
|
virtual: true,
|
|
sourceCode: runCode,
|
|
sourcePath: 'RunMainModule.js'
|
|
}));
|
|
}
|
|
|
|
Object.freeze(this._modules);
|
|
Object.seal(this._modules);
|
|
Object.freeze(this._assets);
|
|
Object.seal(this._assets);
|
|
this._finalized = true;
|
|
};
|
|
|
|
Package.prototype._assertFinalized = function() {
|
|
if (!this._finalized) {
|
|
throw new Error('Package need to be finalized before getting any source');
|
|
}
|
|
};
|
|
|
|
Package.prototype._getSource = function() {
|
|
if (this._source == null) {
|
|
this._source = _.pluck(this._modules, 'code').join('\n');
|
|
}
|
|
return this._source;
|
|
};
|
|
|
|
Package.prototype._getInlineSourceMap = function() {
|
|
if (this._inlineSourceMap == null) {
|
|
var sourceMap = this.getSourceMap({excludeSource: true});
|
|
var encoded = new Buffer(JSON.stringify(sourceMap)).toString('base64');
|
|
this._inlineSourceMap = 'data:application/json;base64,' + encoded;
|
|
}
|
|
return this._inlineSourceMap;
|
|
};
|
|
|
|
Package.prototype.getSource = function(options) {
|
|
this._assertFinalized();
|
|
|
|
options = options || {};
|
|
|
|
if (options.minify) {
|
|
return this.getMinifiedSourceAndMap().code;
|
|
}
|
|
|
|
var source = this._getSource();
|
|
|
|
if (options.inlineSourceMap) {
|
|
source += SOURCEMAPPING_URL + this._getInlineSourceMap();
|
|
} else if (this._sourceMapUrl) {
|
|
source += SOURCEMAPPING_URL + this._sourceMapUrl;
|
|
}
|
|
|
|
return source;
|
|
};
|
|
|
|
Package.prototype.getMinifiedSourceAndMap = function() {
|
|
this._assertFinalized();
|
|
|
|
var source = this._getSource();
|
|
try {
|
|
return UglifyJS.minify(source, {
|
|
fromString: true,
|
|
outSourceMap: 'bundle.js',
|
|
inSourceMap: this.getSourceMap(),
|
|
});
|
|
} catch(e) {
|
|
// Sometimes, when somebody is using a new syntax feature that we
|
|
// don't yet have transform for, the untransformed line is sent to
|
|
// uglify, and it chokes on it. This code tries to print the line
|
|
// and the module for easier debugging
|
|
var errorMessage = 'Error while minifying JS\n';
|
|
if (e.line) {
|
|
errorMessage += 'Transformed code line: "' +
|
|
source.split('\n')[e.line - 1] + '"\n';
|
|
}
|
|
if (e.pos) {
|
|
var fromIndex = source.lastIndexOf('__d(\'', e.pos);
|
|
if (fromIndex > -1) {
|
|
fromIndex += '__d(\''.length;
|
|
var toIndex = source.indexOf('\'', fromIndex);
|
|
errorMessage += 'Module name (best guess): ' +
|
|
source.substring(fromIndex, toIndex) + '\n';
|
|
}
|
|
}
|
|
errorMessage += e.toString();
|
|
throw new Error(errorMessage);
|
|
}
|
|
};
|
|
|
|
/**
|
|
* I found a neat trick in the sourcemap spec that makes it easy
|
|
* to concat sourcemaps. The `sections` field allows us to combine
|
|
* the sourcemap easily by adding an offset. Tested on chrome.
|
|
* Seems like it's not yet in Firefox but that should be fine for
|
|
* now.
|
|
*/
|
|
Package.prototype._getCombinedSourceMaps = function(options) {
|
|
var result = {
|
|
version: 3,
|
|
file: 'bundle.js',
|
|
sections: [],
|
|
};
|
|
|
|
var line = 0;
|
|
this._modules.forEach(function(module) {
|
|
var map = module.map;
|
|
if (module.virtual) {
|
|
map = generateSourceMapForVirtualModule(module);
|
|
}
|
|
|
|
if (options.excludeSource) {
|
|
map = _.extend({}, map, {sourcesContent: []});
|
|
}
|
|
|
|
result.sections.push({
|
|
offset: { line: line, column: 0 },
|
|
map: map,
|
|
});
|
|
line += module.code.split('\n').length;
|
|
});
|
|
|
|
return result;
|
|
};
|
|
|
|
Package.prototype.getSourceMap = function(options) {
|
|
this._assertFinalized();
|
|
|
|
options = options || {};
|
|
|
|
if (this._shouldCombineSourceMaps) {
|
|
return this._getCombinedSourceMaps(options);
|
|
}
|
|
|
|
var mappings = this._getMappings();
|
|
var map = {
|
|
file: 'bundle.js',
|
|
sources: _.pluck(this._modules, 'sourcePath'),
|
|
version: 3,
|
|
names: [],
|
|
mappings: mappings,
|
|
sourcesContent: options.excludeSource
|
|
? [] : _.pluck(this._modules, 'sourceCode')
|
|
};
|
|
return map;
|
|
};
|
|
|
|
Package.prototype.getAssets = function() {
|
|
return this._assets;
|
|
};
|
|
|
|
Package.prototype._getMappings = function() {
|
|
var modules = this._modules;
|
|
|
|
// The first line mapping in our package is basically the base64vlq code for
|
|
// zeros (A).
|
|
var firstLine = 'AAAA';
|
|
|
|
// Most other lines in our mappings are all zeros (for module, column etc)
|
|
// except for the lineno mappinp: curLineno - prevLineno = 1; Which is C.
|
|
var line = 'AACA';
|
|
|
|
var moduleLines = Object.create(null);
|
|
var mappings = '';
|
|
for (var i = 0; i < modules.length; i++) {
|
|
var module = modules[i];
|
|
var code = module.code;
|
|
var lastCharNewLine = false;
|
|
moduleLines[module.sourcePath] = 0;
|
|
for (var t = 0; t < code.length; t++) {
|
|
if (t === 0 && i === 0) {
|
|
mappings += firstLine;
|
|
} else if (t === 0) {
|
|
mappings += 'AC';
|
|
|
|
// This is the only place were we actually don't know the mapping ahead
|
|
// of time. When it's a new module (and not the first) the lineno
|
|
// mapping is 0 (current) - number of lines in prev module.
|
|
mappings += base64VLQ.encode(
|
|
0 - moduleLines[modules[i - 1].sourcePath]
|
|
);
|
|
mappings += 'A';
|
|
} else if (lastCharNewLine) {
|
|
moduleLines[module.sourcePath]++;
|
|
mappings += line;
|
|
}
|
|
lastCharNewLine = code[t] === '\n';
|
|
if (lastCharNewLine) {
|
|
mappings += ';';
|
|
}
|
|
}
|
|
if (i !== modules.length - 1) {
|
|
mappings += ';';
|
|
}
|
|
}
|
|
return mappings;
|
|
};
|
|
|
|
Package.prototype.getJSModulePaths = function() {
|
|
return this._modules.filter(function(module) {
|
|
// Filter out non-js files. Like images etc.
|
|
return !module.virtual;
|
|
}).map(function(module) {
|
|
return module.sourcePath;
|
|
});
|
|
};
|
|
|
|
Package.prototype.getDebugInfo = function() {
|
|
return [
|
|
'<div><h3>Main Module:</h3> ' + this._mainModuleId + '</div>',
|
|
'<style>',
|
|
'pre.collapsed {',
|
|
' height: 10px;',
|
|
' width: 100px;',
|
|
' display: block;',
|
|
' text-overflow: ellipsis;',
|
|
' overflow: hidden;',
|
|
' cursor: pointer;',
|
|
'}',
|
|
'</style>',
|
|
'<h3> Module paths and transformed code: </h3>',
|
|
this._modules.map(function(m) {
|
|
return '<div> <h4> Path: </h4>' + m.sourcePath + '<br/> <h4> Source: </h4>' +
|
|
'<code><pre class="collapsed" onclick="this.classList.remove(\'collapsed\')">' +
|
|
_.escape(m.code) + '</pre></code></div>';
|
|
}).join('\n'),
|
|
].join('\n');
|
|
};
|
|
|
|
function generateSourceMapForVirtualModule(module) {
|
|
// All lines map 1-to-1
|
|
var mappings = 'AAAA;';
|
|
|
|
for (var i = 1; i < module.code.split('\n').length; i++) {
|
|
mappings += 'AACA;';
|
|
}
|
|
|
|
return {
|
|
version: 3,
|
|
sources: [ module.sourcePath ],
|
|
names: [],
|
|
mappings: mappings,
|
|
file: module.sourcePath,
|
|
sourcesContent: [ module.sourceCode ],
|
|
};
|
|
}
|