David Aurelio b62ed2f291 Add support for module groups to iOS Random Access Bundle format
Summary:
Adds the possibility to specify an array of files (group roots) that are used to bundle modules outside of the “startup section” into bigger groups by colocating their code.
A require call for any grouped module will load all modules in that group.
Files contained by multiple groups are deoptimized (i.e. bundled as individual script)

Reviewed By: martinbigio

Differential Revision: D3841780

fbshipit-source-id: 8d37782792efd66b5f557c7567489f68c9b229d8
2016-09-16 10:13:49 -07:00

96 lines
2.5 KiB
JavaScript

/**
* Copyright (c) 2016-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';
const newline = /\r\n?|\n|\u2028|\u2029/g;
const countLines =
string => (string.match(newline) || []).length + 1; // fastest implementation
function lineToLineSourceMap(source, filename) {
// The first line mapping in our package is the base64vlq code for zeros (A).
const firstLine = 'AAAA;';
// Most other lines in our mappings are all zeros (for module, column etc)
// except for the lineno mapping: curLineno - prevLineno = 1; Which is C.
const line = 'AACA;';
return {
version: 3,
sources: [filename],
mappings: firstLine + Array(countLines(source)).join(line),
};
}
const wrapperEnd = wrappedCode => wrappedCode.indexOf('{') + 1;
const Section = (line, column, map) => ({map, offset: {line, column}});
function combineSourceMaps({modules, withCustomOffsets, moduleGroups}) {
let offsets;
const sections = [];
const sourceMap = {
version: 3,
sections,
};
if (withCustomOffsets) {
offsets = sourceMap.x_facebook_offsets = [];
}
let line = 0;
modules.forEach(({code, id, map, name}) => {
let column = 0;
let hasOffset = false;
let group;
let groupLines = 0;
if (withCustomOffsets) {
if (moduleGroups && moduleGroups.modulesInGroups.has(id)) {
// this is a module appended to another module
return;
}
if (moduleGroups && moduleGroups.groups.has(id)) {
group = moduleGroups.groups.get(id);
const otherModules = Array.from(group).map(
moduleId => moduleGroups.modulesById.get(moduleId));
otherModules.forEach(m => {
groupLines += countLines(m.code);
});
map = combineSourceMaps({
modules: [{code, id, map, name}].concat(otherModules),
});
}
hasOffset = id != null;
column = wrapperEnd(code);
}
sections.push(Section(line, column, map || lineToLineSourceMap(code, name)));
if (hasOffset) {
offsets[id] = line;
for (const moduleId of group || []) {
offsets[moduleId] = line;
}
}
line += countLines(code) + groupLines;
});
return sourceMap;
}
const joinModules = modules => modules.map(m => m.code).join('\n');
module.exports = {
countLines,
lineToLineSourceMap,
combineSourceMaps,
joinModules,
};