mirror of https://github.com/status-im/metro.git
Move wrapModule logic for scripts, modules and JSON files to the transformer
Reviewed By: davidaurelio Differential Revision: D6389116 fbshipit-source-id: fa43bd54669849c22b6b9d155ab07676b2455ef7
This commit is contained in:
parent
b193fc3436
commit
b08ce130fd
|
@ -213,6 +213,7 @@ class Bundler {
|
||||||
module.path,
|
module.path,
|
||||||
module.localPath,
|
module.localPath,
|
||||||
code,
|
code,
|
||||||
|
module.isPolyfill(),
|
||||||
transformCodeOptions,
|
transformCodeOptions,
|
||||||
),
|
),
|
||||||
transformCache: opts.transformCache,
|
transformCache: opts.transformCache,
|
||||||
|
@ -240,14 +241,11 @@ class Bundler {
|
||||||
}
|
}
|
||||||
|
|
||||||
async generateAssetObjAndCode(
|
async generateAssetObjAndCode(
|
||||||
module: Module,
|
path: string,
|
||||||
assetPlugins: Array<string>,
|
assetPlugins: Array<string>,
|
||||||
platform: ?string = null,
|
platform: ?string = null,
|
||||||
) {
|
) {
|
||||||
const assetData = await this._assetServer.getAssetData(
|
const assetData = await this._assetServer.getAssetData(path, platform);
|
||||||
module.path,
|
|
||||||
platform,
|
|
||||||
);
|
|
||||||
const asset = await this._applyAssetPlugins(assetPlugins, assetData);
|
const asset = await this._applyAssetPlugins(assetPlugins, assetData);
|
||||||
|
|
||||||
const {
|
const {
|
||||||
|
|
|
@ -14,6 +14,7 @@
|
||||||
|
|
||||||
const DeltaCalculator = require('./DeltaCalculator');
|
const DeltaCalculator = require('./DeltaCalculator');
|
||||||
|
|
||||||
|
const addParamsToDefineCall = require('../lib/addParamsToDefineCall');
|
||||||
const createModuleIdFactory = require('../lib/createModuleIdFactory');
|
const createModuleIdFactory = require('../lib/createModuleIdFactory');
|
||||||
|
|
||||||
const {EventEmitter} = require('events');
|
const {EventEmitter} = require('events');
|
||||||
|
@ -425,27 +426,37 @@ class DeltaTransformer extends EventEmitter {
|
||||||
): Promise<[number, ?DeltaEntry]> {
|
): Promise<[number, ?DeltaEntry]> {
|
||||||
const name = module.getName();
|
const name = module.getName();
|
||||||
const metadata = await this._getMetadata(module, transformOptions);
|
const metadata = await this._getMetadata(module, transformOptions);
|
||||||
|
|
||||||
const edge = dependencyEdges.get(module.path);
|
const edge = dependencyEdges.get(module.path);
|
||||||
const dependencyPairs = edge ? edge.dependencies : new Map();
|
const dependencyPairs = edge ? edge.dependencies : new Map();
|
||||||
|
|
||||||
const wrapped = this._resolver.wrapModule({
|
let wrappedCode;
|
||||||
module,
|
|
||||||
getModuleId: this._getModuleId,
|
if (module.isAsset()) {
|
||||||
dependencyPairs,
|
wrappedCode = await this._wrapAsset({
|
||||||
dependencyOffsets: metadata.dependencyOffsets || [],
|
|
||||||
name,
|
|
||||||
code: metadata.code,
|
code: metadata.code,
|
||||||
map: metadata.map,
|
dependencyPairs,
|
||||||
dev: this._bundleOptions.dev,
|
name,
|
||||||
|
path: module.path,
|
||||||
});
|
});
|
||||||
|
} else if (!module.isPolyfill()) {
|
||||||
|
wrappedCode = this._addDependencyMap({
|
||||||
|
code: metadata.code,
|
||||||
|
dependencyPairs,
|
||||||
|
name,
|
||||||
|
path: module.path,
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
wrappedCode = metadata.code;
|
||||||
|
}
|
||||||
|
|
||||||
const {code, map} = transformOptions.minify
|
const {code, map} = transformOptions.minify
|
||||||
? await this._resolver.minifyModule(
|
? await this._resolver.minifyModule(
|
||||||
module.path,
|
module.path,
|
||||||
wrapped.code,
|
wrappedCode,
|
||||||
wrapped.map,
|
metadata.map,
|
||||||
)
|
)
|
||||||
: wrapped;
|
: {code: wrappedCode, map: metadata.map};
|
||||||
|
|
||||||
const id = this._getModuleId(module.path);
|
const id = this._getModuleId(module.path);
|
||||||
|
|
||||||
|
@ -463,6 +474,70 @@ class DeltaTransformer extends EventEmitter {
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Function to add the mapping object between local module ids and
|
||||||
|
* actual bundle module ids for dependencies. This way, we can do the path
|
||||||
|
* replacements on require() calls on transformers (since local ids do not
|
||||||
|
* change between bundles).
|
||||||
|
*/
|
||||||
|
_addDependencyMap({
|
||||||
|
code,
|
||||||
|
dependencyPairs,
|
||||||
|
name,
|
||||||
|
path,
|
||||||
|
}: {
|
||||||
|
code: string,
|
||||||
|
dependencyPairs: Map<string, string>,
|
||||||
|
name: string,
|
||||||
|
path: string,
|
||||||
|
}): string {
|
||||||
|
const moduleId = this._getModuleId(path);
|
||||||
|
const params = [
|
||||||
|
moduleId,
|
||||||
|
Array.from(dependencyPairs.values()).map(this._getModuleId),
|
||||||
|
];
|
||||||
|
|
||||||
|
// Add the module name as the last parameter (to make it easier to do
|
||||||
|
// requires by name when debugging).
|
||||||
|
if (this._bundleOptions.dev) {
|
||||||
|
params.push(name);
|
||||||
|
}
|
||||||
|
|
||||||
|
return addParamsToDefineCall(code, ...params);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Temporary function to wrap an asset. This logic will go away once we
|
||||||
|
* generate the needed JS code for assets in the transformer.
|
||||||
|
*/
|
||||||
|
async _wrapAsset({
|
||||||
|
code,
|
||||||
|
dependencyPairs,
|
||||||
|
name,
|
||||||
|
path,
|
||||||
|
}: {
|
||||||
|
code: string,
|
||||||
|
dependencyPairs: Map<string, string>,
|
||||||
|
name: string,
|
||||||
|
path: string,
|
||||||
|
}): Promise<string> {
|
||||||
|
const asset = await this._bundler.generateAssetObjAndCode(
|
||||||
|
path,
|
||||||
|
this._bundleOptions.assetPlugins,
|
||||||
|
this._bundleOptions.platform,
|
||||||
|
);
|
||||||
|
|
||||||
|
return await this._resolver.wrapModule({
|
||||||
|
path,
|
||||||
|
getModuleId: this._getModuleId,
|
||||||
|
dependencyPairs,
|
||||||
|
dependencyOffsets: asset.meta.dependencyOffsets,
|
||||||
|
name,
|
||||||
|
code: asset.code,
|
||||||
|
dev: this._bundleOptions.dev,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
_getModuleType(module: Module): DeltaEntryType {
|
_getModuleType(module: Module): DeltaEntryType {
|
||||||
if (module.isAsset()) {
|
if (module.isAsset()) {
|
||||||
return 'asset';
|
return 'asset';
|
||||||
|
@ -480,25 +555,10 @@ class DeltaTransformer extends EventEmitter {
|
||||||
transformOptions: JSTransformerOptions,
|
transformOptions: JSTransformerOptions,
|
||||||
): Promise<{
|
): Promise<{
|
||||||
+code: string,
|
+code: string,
|
||||||
+dependencyOffsets: ?Array<number>,
|
+dependencies: Array<string>,
|
||||||
+map: CompactRawMappings,
|
+map: CompactRawMappings,
|
||||||
+source: string,
|
+source: string,
|
||||||
}> {
|
}> {
|
||||||
if (module.isAsset()) {
|
|
||||||
const asset = await this._bundler.generateAssetObjAndCode(
|
|
||||||
module,
|
|
||||||
this._bundleOptions.assetPlugins,
|
|
||||||
this._bundleOptions.platform,
|
|
||||||
);
|
|
||||||
|
|
||||||
return {
|
|
||||||
code: asset.code,
|
|
||||||
dependencyOffsets: asset.meta.dependencyOffsets,
|
|
||||||
map: [],
|
|
||||||
source: '',
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
return await module.read(transformOptions);
|
return await module.read(transformOptions);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -59,6 +59,7 @@ describe('Transformer', function() {
|
||||||
fileName,
|
fileName,
|
||||||
localPath,
|
localPath,
|
||||||
code,
|
code,
|
||||||
|
false,
|
||||||
transformOptions,
|
transformOptions,
|
||||||
);
|
);
|
||||||
|
|
||||||
|
@ -67,6 +68,7 @@ describe('Transformer', function() {
|
||||||
fileName,
|
fileName,
|
||||||
localPath,
|
localPath,
|
||||||
code,
|
code,
|
||||||
|
false,
|
||||||
transformOptions,
|
transformOptions,
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
@ -92,7 +94,7 @@ describe('Transformer', function() {
|
||||||
expect.assertions(6);
|
expect.assertions(6);
|
||||||
|
|
||||||
return transformer
|
return transformer
|
||||||
.transformFile(fileName, localPath, '', {})
|
.transformFile(fileName, localPath, '', true, {})
|
||||||
.catch(function(error) {
|
.catch(function(error) {
|
||||||
expect(error.type).toEqual('TransformError');
|
expect(error.type).toEqual('TransformError');
|
||||||
expect(error.message).toBe(
|
expect(error.message).toBe(
|
||||||
|
|
|
@ -87,6 +87,7 @@ module.exports = class Transformer {
|
||||||
filename: string,
|
filename: string,
|
||||||
localPath: LocalPath,
|
localPath: LocalPath,
|
||||||
code: string,
|
code: string,
|
||||||
|
isScript: boolean,
|
||||||
options: Options,
|
options: Options,
|
||||||
): Promise<TransformedCode> {
|
): Promise<TransformedCode> {
|
||||||
try {
|
try {
|
||||||
|
@ -97,6 +98,7 @@ module.exports = class Transformer {
|
||||||
filename,
|
filename,
|
||||||
localPath,
|
localPath,
|
||||||
code,
|
code,
|
||||||
|
isScript,
|
||||||
options,
|
options,
|
||||||
);
|
);
|
||||||
|
|
||||||
|
|
|
@ -13,243 +13,95 @@
|
||||||
|
|
||||||
jest
|
jest
|
||||||
.mock('../constant-folding')
|
.mock('../constant-folding')
|
||||||
.mock('../extract-dependencies')
|
|
||||||
.mock('../inline')
|
.mock('../inline')
|
||||||
.mock('../minify')
|
.mock('../minify');
|
||||||
.mock('babel-generator');
|
|
||||||
|
|
||||||
const {objectContaining} = jasmine;
|
const path = require('path');
|
||||||
|
const transformCode = require('..').transformAndExtractDependencies;
|
||||||
|
|
||||||
describe('code transformation worker:', () => {
|
describe('code transformation worker:', () => {
|
||||||
let transformCode;
|
it('transforms a simple script', async () => {
|
||||||
let babelGenerator;
|
const {result} = await transformCode(
|
||||||
|
path.join(__dirname, '../../../transformer.js'),
|
||||||
let extractDependencies, transformer;
|
'arbitrary/file.js',
|
||||||
beforeEach(() => {
|
`local/file.js`,
|
||||||
jest.resetModules();
|
'someReallyArbitrary(code)',
|
||||||
({transformCode} = require('..'));
|
true,
|
||||||
extractDependencies = require('../extract-dependencies').mockReturnValue(
|
{
|
||||||
{},
|
|
||||||
);
|
|
||||||
transformer = {
|
|
||||||
transform: jest.fn(({filename, options, src}) => ({
|
|
||||||
code: src,
|
|
||||||
map: [],
|
|
||||||
})),
|
|
||||||
};
|
|
||||||
|
|
||||||
babelGenerator = require('babel-generator');
|
|
||||||
|
|
||||||
babelGenerator.default.mockReturnValue({
|
|
||||||
code: '',
|
|
||||||
map: [],
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
it('calls the transform with file name, source code, and transform options', function() {
|
|
||||||
const filename = 'arbitrary/file.js';
|
|
||||||
const localPath = `local/${filename}`;
|
|
||||||
const sourceCode = 'arbitrary(code)';
|
|
||||||
const transformOptions = {arbitrary: 'options'};
|
|
||||||
transformCode(transformer, filename, localPath, sourceCode, {
|
|
||||||
dev: true,
|
dev: true,
|
||||||
transform: transformOptions,
|
transform: {},
|
||||||
});
|
},
|
||||||
expect(transformer.transform).toBeCalledWith({
|
|
||||||
filename,
|
|
||||||
localPath,
|
|
||||||
options: transformOptions,
|
|
||||||
plugins: [],
|
|
||||||
src: sourceCode,
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
it('calls the transform with two plugins when not in dev mode', () => {
|
|
||||||
const filename = 'arbitrary/file.js';
|
|
||||||
const localPath = `local/${filename}`;
|
|
||||||
const sourceCode = 'arbitrary(code)';
|
|
||||||
const options = {dev: false, transform: {arbitrary: 'options'}};
|
|
||||||
|
|
||||||
transformCode(transformer, filename, localPath, sourceCode, options);
|
|
||||||
|
|
||||||
const plugins = transformer.transform.mock.calls[0][0].plugins;
|
|
||||||
|
|
||||||
expect(plugins[0]).toEqual([expect.any(Object), options]);
|
|
||||||
expect(plugins[1]).toEqual([expect.any(Object), options]);
|
|
||||||
});
|
|
||||||
|
|
||||||
it('prefixes JSON files with an assignment to module.exports to make the code valid', function() {
|
|
||||||
const filename = 'arbitrary/file.json';
|
|
||||||
const localPath = `local/${filename}`;
|
|
||||||
const sourceCode = '{"arbitrary":"property"}';
|
|
||||||
transformCode(transformer, filename, localPath, sourceCode, {dev: true});
|
|
||||||
|
|
||||||
expect(transformer.transform).toBeCalledWith({
|
|
||||||
filename,
|
|
||||||
localPath,
|
|
||||||
options: undefined,
|
|
||||||
plugins: [],
|
|
||||||
src: `module.exports=${sourceCode}`,
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
it('calls back with the result of the transform in the cache', async () => {
|
|
||||||
const result = {
|
|
||||||
code: 'some.other(code)',
|
|
||||||
map: [],
|
|
||||||
};
|
|
||||||
|
|
||||||
babelGenerator.default.mockReturnValue({
|
|
||||||
code: 'some.other(code)',
|
|
||||||
map: [],
|
|
||||||
});
|
|
||||||
|
|
||||||
const data = await transformCode(
|
|
||||||
transformer,
|
|
||||||
'filename',
|
|
||||||
'local/filename',
|
|
||||||
result.code,
|
|
||||||
{},
|
|
||||||
);
|
);
|
||||||
|
|
||||||
expect(data.result).toEqual(objectContaining(result));
|
expect(result.code).toBe(
|
||||||
|
[
|
||||||
|
'(function (global) {',
|
||||||
|
' someReallyArbitrary(code);',
|
||||||
|
'})(this);',
|
||||||
|
].join('\n'),
|
||||||
|
);
|
||||||
|
expect(result.map).toHaveLength(3);
|
||||||
|
expect(result.dependencies).toEqual([]);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('removes the leading `module.exports` before returning if the file is a JSON file, even if minified', async () => {
|
it('transforms a simple module', async () => {
|
||||||
const code = '{a:1,b:2}';
|
const {result} = await transformCode(
|
||||||
const filePath = 'arbitrary/file.json';
|
path.join(__dirname, '../../../transformer.js'),
|
||||||
|
'arbitrary/file.js',
|
||||||
babelGenerator.default.mockReturnValue({
|
`local/file.js`,
|
||||||
code: '{a:1,b:2}',
|
'arbitrary(code)',
|
||||||
map: [],
|
false,
|
||||||
});
|
{
|
||||||
|
dev: true,
|
||||||
const data = await transformCode(transformer, filePath, filePath, code, {});
|
transform: {},
|
||||||
|
},
|
||||||
expect(data.result.code).toEqual(code);
|
|
||||||
});
|
|
||||||
|
|
||||||
it('removes shebang when present', async () => {
|
|
||||||
const shebang = '#!/usr/bin/env node';
|
|
||||||
const result = {
|
|
||||||
code: `${shebang} \n arbitrary(code)`,
|
|
||||||
};
|
|
||||||
const filePath = 'arbitrary/file.js';
|
|
||||||
|
|
||||||
babelGenerator.default.mockReturnValue({
|
|
||||||
code: `${shebang} \n arbitrary(code)`,
|
|
||||||
map: [],
|
|
||||||
});
|
|
||||||
|
|
||||||
const data = await transformCode(
|
|
||||||
transformer,
|
|
||||||
filePath,
|
|
||||||
filePath,
|
|
||||||
result.code,
|
|
||||||
{},
|
|
||||||
);
|
);
|
||||||
|
|
||||||
const {code} = data.result;
|
expect(result.code).toBe(
|
||||||
expect(code).not.toContain(shebang);
|
[
|
||||||
expect(code.split('\n').length).toEqual(result.code.split('\n').length);
|
'__d(function (global, require, module, exports, _dependencyMap) {',
|
||||||
});
|
' arbitrary(code);',
|
||||||
|
'});',
|
||||||
it('calls back with any error yielded by the transform', async () => {
|
].join('\n'),
|
||||||
const message = 'SyntaxError: this code is broken.';
|
|
||||||
|
|
||||||
transformer.transform.mockImplementation(() => {
|
|
||||||
throw new Error(message);
|
|
||||||
});
|
|
||||||
|
|
||||||
expect.assertions(1);
|
|
||||||
|
|
||||||
try {
|
|
||||||
await transformCode(
|
|
||||||
transformer,
|
|
||||||
'filename',
|
|
||||||
'local/filename',
|
|
||||||
'code',
|
|
||||||
{},
|
|
||||||
);
|
);
|
||||||
} catch (error) {
|
expect(result.map).toHaveLength(3);
|
||||||
expect(error.message).toBe(message);
|
expect(result.dependencies).toEqual([]);
|
||||||
}
|
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('dependency extraction', () => {
|
it('transforms a module with dependencies', async () => {
|
||||||
it('passes the transformed code the `extractDependencies`', async () => {
|
const {result} = await transformCode(
|
||||||
const code = 'arbitrary(code)';
|
path.join(__dirname, '../../../transformer.js'),
|
||||||
|
'arbitrary/file.js',
|
||||||
babelGenerator.default.mockReturnValue({
|
`local/file.js`,
|
||||||
code: 'arbitrary(code)',
|
[
|
||||||
map: [],
|
'require("./a");',
|
||||||
});
|
'arbitrary(code);',
|
||||||
|
'const b = require("b");',
|
||||||
await transformCode(transformer, 'filename', 'local/filename', code, {});
|
'import c from "./c";',
|
||||||
|
].join('\n'),
|
||||||
expect(extractDependencies).toBeCalledWith(code, 'filename');
|
false,
|
||||||
});
|
{
|
||||||
|
dev: true,
|
||||||
it('uses `dependencies` and `dependencyOffsets` provided by `extractDependencies` for the result', async () => {
|
transform: {},
|
||||||
const dependencyData = {
|
},
|
||||||
dependencies: ['arbitrary', 'list', 'of', 'dependencies'],
|
|
||||||
dependencyOffsets: [12, 119, 185, 328, 471],
|
|
||||||
};
|
|
||||||
|
|
||||||
extractDependencies.mockReturnValue(dependencyData);
|
|
||||||
|
|
||||||
const data = await transformCode(
|
|
||||||
transformer,
|
|
||||||
'filename',
|
|
||||||
'local/filename',
|
|
||||||
'code',
|
|
||||||
{},
|
|
||||||
);
|
);
|
||||||
|
|
||||||
expect(data.result).toEqual(objectContaining(dependencyData));
|
expect(result.code).toBe(
|
||||||
});
|
[
|
||||||
|
'__d(function (global, require, module, exports, _dependencyMap) {',
|
||||||
it('does not extract requires of JSON files', async () => {
|
' var _c = require(_dependencyMap[0], "./c");',
|
||||||
const jsonStr = '{"arbitrary":"json"}';
|
'',
|
||||||
|
' var _c2 = babelHelpers.interopRequireDefault(_c);',
|
||||||
babelGenerator.default.mockReturnValue({
|
'',
|
||||||
code: '{"arbitrary":"json"}',
|
' require(_dependencyMap[1], "./a");',
|
||||||
map: [],
|
'',
|
||||||
});
|
' arbitrary(code);',
|
||||||
|
'',
|
||||||
const data = await transformCode(
|
' var b = require(_dependencyMap[2], "b");',
|
||||||
transformer,
|
'});',
|
||||||
'arbitrary.json',
|
].join('\n'),
|
||||||
'local/arbitrary.json',
|
|
||||||
jsonStr,
|
|
||||||
{},
|
|
||||||
);
|
);
|
||||||
|
expect(result.map).toHaveLength(13);
|
||||||
const {dependencies, dependencyOffsets} = data.result;
|
expect(result.dependencies).toEqual(['./c', './a', 'b']);
|
||||||
|
|
||||||
expect(extractDependencies).not.toBeCalled();
|
|
||||||
expect(dependencies).toEqual([]);
|
|
||||||
expect(dependencyOffsets).toEqual([]);
|
|
||||||
});
|
|
||||||
|
|
||||||
it('calls back with every error thrown by `extractDependencies`', async () => {
|
|
||||||
const error = new Error('arbitrary');
|
|
||||||
|
|
||||||
extractDependencies.mockImplementation(() => {
|
|
||||||
throw error;
|
|
||||||
});
|
|
||||||
|
|
||||||
try {
|
|
||||||
await transformCode(
|
|
||||||
transformer,
|
|
||||||
'arbitrary.js',
|
|
||||||
'local/arbitrary.js',
|
|
||||||
'code',
|
|
||||||
{},
|
|
||||||
);
|
|
||||||
} catch (err) {
|
|
||||||
expect(err).toBe(error);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
|
@ -12,28 +12,25 @@
|
||||||
|
|
||||||
'use strict';
|
'use strict';
|
||||||
|
|
||||||
|
const JsFileWrapping = require('../../ModuleGraph/worker/JsFileWrapping');
|
||||||
|
|
||||||
|
const collectDependencies = require('../../ModuleGraph/worker/collect-dependencies');
|
||||||
const constantFolding = require('./constant-folding');
|
const constantFolding = require('./constant-folding');
|
||||||
const extractDependencies = require('./extract-dependencies');
|
|
||||||
const generate = require('babel-generator').default;
|
const generate = require('babel-generator').default;
|
||||||
const inline = require('./inline');
|
const inline = require('./inline');
|
||||||
const minify = require('./minify');
|
const minify = require('./minify');
|
||||||
|
|
||||||
const {compactMapping, toRawMappings} = require('../../Bundler/source-map');
|
const {compactMapping} = require('../../Bundler/source-map');
|
||||||
|
|
||||||
import type {LogEntry} from '../../Logger/Types';
|
import type {LogEntry} from '../../Logger/Types';
|
||||||
import type {
|
import type {CompactRawMappings, MappingsMap} from '../../lib/SourceMap';
|
||||||
CompactRawMappings,
|
|
||||||
MappingsMap,
|
|
||||||
RawMappings,
|
|
||||||
} from '../../lib/SourceMap';
|
|
||||||
import type {LocalPath} from '../../node-haste/lib/toLocalPath';
|
import type {LocalPath} from '../../node-haste/lib/toLocalPath';
|
||||||
import type {ResultWithMap} from './minify';
|
import type {ResultWithMap} from './minify';
|
||||||
import type {Ast, Plugins as BabelPlugins} from 'babel-core';
|
import type {Ast, Plugins as BabelPlugins} from 'babel-core';
|
||||||
|
|
||||||
export type TransformedCode = {
|
export type TransformedCode = {
|
||||||
code: string,
|
code: string,
|
||||||
dependencies: Array<string>,
|
dependencies: $ReadOnlyArray<string>,
|
||||||
dependencyOffsets: Array<number>,
|
|
||||||
map: CompactRawMappings,
|
map: CompactRawMappings,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -96,6 +93,7 @@ async function transformCode(
|
||||||
filename: string,
|
filename: string,
|
||||||
localPath: LocalPath,
|
localPath: LocalPath,
|
||||||
sourceCode: string,
|
sourceCode: string,
|
||||||
|
isScript: boolean,
|
||||||
options: Options,
|
options: Options,
|
||||||
): Promise<Data> {
|
): Promise<Data> {
|
||||||
const isJson = filename.endsWith('.json');
|
const isJson = filename.endsWith('.json');
|
||||||
|
@ -116,7 +114,7 @@ async function transformCode(
|
||||||
? []
|
? []
|
||||||
: [[inline.plugin, options], [constantFolding.plugin, options]];
|
: [[inline.plugin, options], [constantFolding.plugin, options]];
|
||||||
|
|
||||||
const result = await transformer.transform({
|
const {ast} = await transformer.transform({
|
||||||
filename,
|
filename,
|
||||||
localPath,
|
localPath,
|
||||||
options: options.transform,
|
options: options.transform,
|
||||||
|
@ -124,9 +122,45 @@ async function transformCode(
|
||||||
src: sourceCode,
|
src: sourceCode,
|
||||||
});
|
});
|
||||||
|
|
||||||
// Serialize the AST received from the transformer.
|
const timeDelta = process.hrtime(transformFileStartLogEntry.start_timestamp);
|
||||||
const transformed = generate(
|
const duration_ms = Math.round((timeDelta[0] * 1e9 + timeDelta[1]) / 1e6);
|
||||||
result.ast,
|
const transformFileEndLogEntry = {
|
||||||
|
action_name: 'Transforming file',
|
||||||
|
action_phase: 'end',
|
||||||
|
file_name: filename,
|
||||||
|
duration_ms,
|
||||||
|
log_entry_label: 'Transforming file',
|
||||||
|
};
|
||||||
|
|
||||||
|
let dependencies, wrappedAst;
|
||||||
|
|
||||||
|
// If the module to transform is a script (meaning that is not part of the
|
||||||
|
// dependency graph and it code will just be prepended to the bundle modules),
|
||||||
|
// we need to wrap it differently than a commonJS module (also, scripts do
|
||||||
|
// not have dependencies).
|
||||||
|
if (isScript) {
|
||||||
|
dependencies = [];
|
||||||
|
wrappedAst = JsFileWrapping.wrapPolyfill(ast);
|
||||||
|
} else {
|
||||||
|
let dependencyData = collectDependencies(ast);
|
||||||
|
|
||||||
|
if (!options.dev) {
|
||||||
|
dependencyData = collectDependencies.forOptimization(
|
||||||
|
ast,
|
||||||
|
dependencyData.dependencies,
|
||||||
|
dependencyData.dependencyMapName,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
dependencies = dependencyData.dependencies.map(dep => dep.name);
|
||||||
|
wrappedAst = JsFileWrapping.wrapModule(
|
||||||
|
ast,
|
||||||
|
dependencyData.dependencyMapName,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
const result = generate(
|
||||||
|
wrappedAst,
|
||||||
{
|
{
|
||||||
code: false,
|
code: false,
|
||||||
comments: false,
|
comments: false,
|
||||||
|
@ -139,40 +173,10 @@ async function transformCode(
|
||||||
sourceCode,
|
sourceCode,
|
||||||
);
|
);
|
||||||
|
|
||||||
// If the transformer returns standard sourcemaps, we need to transform them
|
const map = result.rawMappings ? result.rawMappings.map(compactMapping) : [];
|
||||||
// to rawMappings so we can process them correctly.
|
|
||||||
const rawMappings =
|
|
||||||
transformed.map && !Array.isArray(transformed.map)
|
|
||||||
? toRawMappings(transformed.map)
|
|
||||||
: transformed.map;
|
|
||||||
|
|
||||||
// Convert the sourcemaps to Compact Raw source maps.
|
|
||||||
const map = rawMappings ? rawMappings.map(compactMapping) : [];
|
|
||||||
|
|
||||||
let code = transformed.code;
|
|
||||||
if (isJson) {
|
|
||||||
code = code.replace(/^\w+\.exports=/, '');
|
|
||||||
} else {
|
|
||||||
// Remove shebang
|
|
||||||
code = code.replace(/^#!.*/, '');
|
|
||||||
}
|
|
||||||
|
|
||||||
const depsResult = isJson
|
|
||||||
? {dependencies: [], dependencyOffsets: []}
|
|
||||||
: extractDependencies(code, filename);
|
|
||||||
|
|
||||||
const timeDelta = process.hrtime(transformFileStartLogEntry.start_timestamp);
|
|
||||||
const duration_ms = Math.round((timeDelta[0] * 1e9 + timeDelta[1]) / 1e6);
|
|
||||||
const transformFileEndLogEntry = {
|
|
||||||
action_name: 'Transforming file',
|
|
||||||
action_phase: 'end',
|
|
||||||
file_name: filename,
|
|
||||||
duration_ms,
|
|
||||||
log_entry_label: 'Transforming file',
|
|
||||||
};
|
|
||||||
|
|
||||||
return {
|
return {
|
||||||
result: {...depsResult, code, map},
|
result: {dependencies, code: result.code, map},
|
||||||
transformFileStartLogEntry,
|
transformFileStartLogEntry,
|
||||||
transformFileEndLogEntry,
|
transformFileEndLogEntry,
|
||||||
};
|
};
|
||||||
|
@ -201,6 +205,7 @@ exports.transformAndExtractDependencies = async function(
|
||||||
filename: string,
|
filename: string,
|
||||||
localPath: LocalPath,
|
localPath: LocalPath,
|
||||||
sourceCode: string,
|
sourceCode: string,
|
||||||
|
isScript: boolean,
|
||||||
options: Options,
|
options: Options,
|
||||||
): Promise<Data> {
|
): Promise<Data> {
|
||||||
// $FlowFixMe: impossible to type a dynamic require.
|
// $FlowFixMe: impossible to type a dynamic require.
|
||||||
|
@ -211,6 +216,7 @@ exports.transformAndExtractDependencies = async function(
|
||||||
filename,
|
filename,
|
||||||
localPath,
|
localPath,
|
||||||
sourceCode,
|
sourceCode,
|
||||||
|
isScript,
|
||||||
options,
|
options,
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
|
@ -133,7 +133,14 @@ function createMapLookup(dependencyMapIdentifier, propertyIdentifier) {
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
function collectDependencies(ast, replacement, dependencyMapIdentifier) {
|
function collectDependencies(
|
||||||
|
ast,
|
||||||
|
replacement,
|
||||||
|
dependencyMapIdentifier,
|
||||||
|
): {
|
||||||
|
dependencies: $ReadOnlyArray<TransformResultDependency>,
|
||||||
|
dependencyMapName: string,
|
||||||
|
} {
|
||||||
const visited = new WeakSet();
|
const visited = new WeakSet();
|
||||||
const traversalState = {dependencyMapIdentifier};
|
const traversalState = {dependencyMapIdentifier};
|
||||||
traverse(
|
traverse(
|
||||||
|
|
|
@ -161,8 +161,8 @@ describe('Resolver', function() {
|
||||||
dependencyPairs.set(relativePath, dependencyModule.path);
|
dependencyPairs.set(relativePath, dependencyModule.path);
|
||||||
}
|
}
|
||||||
|
|
||||||
const {code: processedCode} = depResolver.wrapModule({
|
const processedCode = depResolver.wrapModule({
|
||||||
module: module,
|
path: module.path,
|
||||||
getModuleId: resolutionResponse.getModuleId,
|
getModuleId: resolutionResponse.getModuleId,
|
||||||
dependencyPairs,
|
dependencyPairs,
|
||||||
name: 'test module',
|
name: 'test module',
|
||||||
|
@ -201,11 +201,11 @@ describe('Resolver', function() {
|
||||||
mainModuleId: 'test module',
|
mainModuleId: 'test module',
|
||||||
});
|
});
|
||||||
|
|
||||||
const {code: processedCode} = depResolver.wrapModule({
|
const processedCode = depResolver.wrapModule({
|
||||||
getModuleId: resolutionResponse.getModuleId,
|
getModuleId: resolutionResponse.getModuleId,
|
||||||
dependencyPairs: resolutionResponse.getResolvedDependencyPairs(module),
|
dependencyPairs: resolutionResponse.getResolvedDependencyPairs(module),
|
||||||
code,
|
code,
|
||||||
module,
|
path: module.path,
|
||||||
name: 'test module',
|
name: 'test module',
|
||||||
dev: true,
|
dev: true,
|
||||||
});
|
});
|
||||||
|
@ -220,90 +220,6 @@ describe('Resolver', function() {
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should pass through passed-in source maps', () => {
|
|
||||||
expect.assertions(1);
|
|
||||||
const module = createModule('test module');
|
|
||||||
const resolutionResponse = new ResolutionResponseMock({
|
|
||||||
dependencies: [module],
|
|
||||||
mainModuleId: 'test module',
|
|
||||||
});
|
|
||||||
const inputMap = {version: 3, mappings: 'ARBITRARY'};
|
|
||||||
|
|
||||||
const {map} = depResolver.wrapModule({
|
|
||||||
getModuleId: resolutionResponse.getModuleId,
|
|
||||||
dependencyPairs: resolutionResponse.getResolvedDependencyPairs(module),
|
|
||||||
module,
|
|
||||||
name: 'test module',
|
|
||||||
code: 'arbitrary(code)',
|
|
||||||
map: inputMap,
|
|
||||||
});
|
|
||||||
expect(map).toBe(inputMap);
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should resolve polyfills', async function() {
|
|
||||||
expect.assertions(1);
|
|
||||||
return Resolver.load({
|
|
||||||
projectRoot: '/root',
|
|
||||||
}).then(depResolver => {
|
|
||||||
const polyfill = createPolyfill('test polyfill', []);
|
|
||||||
const code = ['global.fetch = () => 1;'].join('');
|
|
||||||
|
|
||||||
const {code: processedCode} = depResolver.wrapModule({
|
|
||||||
module: polyfill,
|
|
||||||
code,
|
|
||||||
});
|
|
||||||
|
|
||||||
expect(processedCode).toEqual(
|
|
||||||
[
|
|
||||||
'(function(global) {',
|
|
||||||
'global.fetch = () => 1;',
|
|
||||||
'\n})' +
|
|
||||||
"(typeof global !== 'undefined' ? global : typeof self !== 'undefined' ? self : this);",
|
|
||||||
].join(''),
|
|
||||||
);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
describe('JSON files:', () => {
|
|
||||||
const code = JSON.stringify({arbitrary: 'data'});
|
|
||||||
const id = 'arbitrary.json';
|
|
||||||
let depResolver, module, resolutionResponse;
|
|
||||||
|
|
||||||
beforeEach(() => {
|
|
||||||
return Resolver.load({projectRoot: '/root'}).then(r => {
|
|
||||||
depResolver = r;
|
|
||||||
module = createJsonModule(id);
|
|
||||||
resolutionResponse = new ResolutionResponseMock({
|
|
||||||
dependencies: [module],
|
|
||||||
mainModuleId: id,
|
|
||||||
});
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should prefix JSON files with `module.exports=`', () => {
|
|
||||||
expect.assertions(1);
|
|
||||||
const {code: processedCode} = depResolver.wrapModule({
|
|
||||||
getModuleId: resolutionResponse.getModuleId,
|
|
||||||
dependencyPairs: resolutionResponse.getResolvedDependencyPairs(
|
|
||||||
module,
|
|
||||||
),
|
|
||||||
module,
|
|
||||||
name: id,
|
|
||||||
code,
|
|
||||||
dev: false,
|
|
||||||
});
|
|
||||||
|
|
||||||
expect(processedCode).toEqual(
|
|
||||||
[
|
|
||||||
`__d(/* ${id} */function(global, require, module, exports) {`,
|
|
||||||
`module.exports = ${code}\n}, ${resolutionResponse.getModuleId(
|
|
||||||
module.path,
|
|
||||||
)});`,
|
|
||||||
].join(''),
|
|
||||||
);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
describe('minification:', () => {
|
describe('minification:', () => {
|
||||||
const code = 'arbitrary(code)';
|
const code = 'arbitrary(code)';
|
||||||
const id = 'arbitrary.js';
|
const id = 'arbitrary.js';
|
||||||
|
|
|
@ -120,7 +120,6 @@ class Resolver {
|
||||||
}
|
}
|
||||||
|
|
||||||
resolveRequires(
|
resolveRequires(
|
||||||
module: Module,
|
|
||||||
getModuleId: (path: string) => number,
|
getModuleId: (path: string) => number,
|
||||||
code: string,
|
code: string,
|
||||||
dependencyPairs: Map<string, string>,
|
dependencyPairs: Map<string, string>,
|
||||||
|
@ -154,44 +153,30 @@ class Resolver {
|
||||||
}
|
}
|
||||||
|
|
||||||
wrapModule({
|
wrapModule({
|
||||||
module,
|
path,
|
||||||
getModuleId,
|
getModuleId,
|
||||||
dependencyPairs,
|
dependencyPairs,
|
||||||
dependencyOffsets,
|
dependencyOffsets,
|
||||||
name,
|
name,
|
||||||
map,
|
|
||||||
code,
|
code,
|
||||||
dev = true,
|
dev = true,
|
||||||
}: {
|
}: {
|
||||||
module: Module,
|
path: string,
|
||||||
getModuleId: (path: string) => number,
|
getModuleId: (path: string) => number,
|
||||||
dependencyPairs: Map<string, string>,
|
dependencyPairs: Map<string, string>,
|
||||||
dependencyOffsets: Array<number>,
|
dependencyOffsets: Array<number>,
|
||||||
name: string,
|
name: string,
|
||||||
map: CompactRawMappings,
|
|
||||||
code: string,
|
code: string,
|
||||||
dev?: boolean,
|
dev?: boolean,
|
||||||
}): {code: string, map: CompactRawMappings} {
|
}): string {
|
||||||
if (module.isJSON()) {
|
|
||||||
code = `module.exports = ${code}`;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (module.isPolyfill()) {
|
|
||||||
code = definePolyfillCode(code);
|
|
||||||
} else {
|
|
||||||
const moduleId = getModuleId(module.path);
|
|
||||||
|
|
||||||
code = this.resolveRequires(
|
code = this.resolveRequires(
|
||||||
module,
|
|
||||||
getModuleId,
|
getModuleId,
|
||||||
code,
|
code,
|
||||||
dependencyPairs,
|
dependencyPairs,
|
||||||
dependencyOffsets,
|
dependencyOffsets,
|
||||||
);
|
);
|
||||||
code = defineModuleCode(moduleId, code, name, dev);
|
|
||||||
}
|
|
||||||
|
|
||||||
return {code, map};
|
return defineModuleCode(getModuleId(path), code, name, dev);
|
||||||
}
|
}
|
||||||
|
|
||||||
async minifyModule(
|
async minifyModule(
|
||||||
|
|
|
@ -1,26 +1,22 @@
|
||||||
// Jest Snapshot v1, https://goo.gl/fbAQLP
|
// Jest Snapshot v1, https://goo.gl/fbAQLP
|
||||||
|
|
||||||
exports[`basic_bundle bundles package with polyfills 1`] = `
|
exports[`basic_bundle bundles package with polyfills 1`] = `
|
||||||
"(function(global) {
|
"(function (global) {
|
||||||
|
global.__DEV__ = false;
|
||||||
|
global.__BUNDLE_START_TIME__ = global.nativePerformanceNow ? global.nativePerformanceNow() : Date.now();
|
||||||
|
})(this);
|
||||||
|
(function (global) {
|
||||||
|
'use strict';
|
||||||
|
|
||||||
global.__DEV__ = false;
|
global.require = _require;
|
||||||
|
global.__d = define;
|
||||||
|
var modules = Object.create(null);
|
||||||
|
|
||||||
global.__BUNDLE_START_TIME__ = global.nativePerformanceNow ? global.nativePerformanceNow() : Date.now();
|
function define(factory, moduleId, dependencyMap) {
|
||||||
})(typeof global !== 'undefined' ? global : typeof self !== 'undefined' ? self : this);
|
|
||||||
(function(global) {
|
|
||||||
|
|
||||||
'use strict';
|
|
||||||
|
|
||||||
global.require = _require;
|
|
||||||
global.__d = define;
|
|
||||||
|
|
||||||
var modules = Object.create(null);
|
|
||||||
|
|
||||||
|
|
||||||
function define(factory, moduleId, dependencyMap) {
|
|
||||||
if (moduleId in modules) {
|
if (moduleId in modules) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
modules[moduleId] = {
|
modules[moduleId] = {
|
||||||
dependencyMap: dependencyMap,
|
dependencyMap: dependencyMap,
|
||||||
exports: undefined,
|
exports: undefined,
|
||||||
|
@ -28,43 +24,51 @@ function define(factory, moduleId, dependencyMap) {
|
||||||
hasError: false,
|
hasError: false,
|
||||||
isInitialized: false
|
isInitialized: false
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
function _require(moduleId) {
|
function _require(moduleId) {
|
||||||
var moduleIdReallyIsNumber = moduleId;
|
var moduleIdReallyIsNumber = moduleId;
|
||||||
var module = modules[moduleIdReallyIsNumber];
|
var module = modules[moduleIdReallyIsNumber];
|
||||||
return module && module.isInitialized ? module.exports : guardedLoadModule(moduleIdReallyIsNumber, module);
|
return module && module.isInitialized ? module.exports : guardedLoadModule(moduleIdReallyIsNumber, module);
|
||||||
}
|
}
|
||||||
|
|
||||||
var inGuard = false;
|
var inGuard = false;
|
||||||
function guardedLoadModule(moduleId, module) {
|
|
||||||
|
function guardedLoadModule(moduleId, module) {
|
||||||
if (!inGuard && global.ErrorUtils) {
|
if (!inGuard && global.ErrorUtils) {
|
||||||
inGuard = true;
|
inGuard = true;
|
||||||
var returnValue = void 0;
|
var returnValue = void 0;
|
||||||
|
|
||||||
try {
|
try {
|
||||||
returnValue = loadModuleImplementation(moduleId, module);
|
returnValue = loadModuleImplementation(moduleId, module);
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
global.ErrorUtils.reportFatalError(e);
|
global.ErrorUtils.reportFatalError(e);
|
||||||
}
|
}
|
||||||
|
|
||||||
inGuard = false;
|
inGuard = false;
|
||||||
return returnValue;
|
return returnValue;
|
||||||
} else {
|
} else {
|
||||||
return loadModuleImplementation(moduleId, module);
|
return loadModuleImplementation(moduleId, module);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
var ID_MASK_SHIFT = 16;
|
var ID_MASK_SHIFT = 16;
|
||||||
var LOCAL_ID_MASK = ~0 >>> ID_MASK_SHIFT;
|
var LOCAL_ID_MASK = ~0 >>> ID_MASK_SHIFT;
|
||||||
|
|
||||||
function unpackModuleId(moduleId) {
|
function unpackModuleId(moduleId) {
|
||||||
var segmentId = moduleId >>> ID_MASK_SHIFT;
|
var segmentId = moduleId >>> ID_MASK_SHIFT;
|
||||||
var localId = moduleId & LOCAL_ID_MASK;
|
var localId = moduleId & LOCAL_ID_MASK;
|
||||||
return { segmentId: segmentId, localId: localId };
|
return {
|
||||||
}
|
segmentId: segmentId,
|
||||||
_require.unpackModuleId = unpackModuleId;
|
localId: localId
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
function loadModuleImplementation(moduleId, module) {
|
_require.unpackModuleId = unpackModuleId;
|
||||||
|
|
||||||
|
function loadModuleImplementation(moduleId, module) {
|
||||||
var nativeRequire = global.nativeRequire;
|
var nativeRequire = global.nativeRequire;
|
||||||
|
|
||||||
if (!module && nativeRequire) {
|
if (!module && nativeRequire) {
|
||||||
var _unpackModuleId = unpackModuleId(moduleId),
|
var _unpackModuleId = unpackModuleId(moduleId),
|
||||||
segmentId = _unpackModuleId.segmentId,
|
segmentId = _unpackModuleId.segmentId,
|
||||||
|
@ -89,16 +93,14 @@ function loadModuleImplementation(moduleId, module) {
|
||||||
dependencyMap = _module.dependencyMap;
|
dependencyMap = _module.dependencyMap;
|
||||||
|
|
||||||
try {
|
try {
|
||||||
|
var _moduleObject = {
|
||||||
var _moduleObject = { exports: exports };
|
exports: exports
|
||||||
|
};
|
||||||
factory(global, _require, _moduleObject, exports, dependencyMap);
|
factory(global, _require, _moduleObject, exports, dependencyMap);
|
||||||
|
|
||||||
{
|
{
|
||||||
module.factory = undefined;
|
module.factory = undefined;
|
||||||
module.dependencyMap = undefined;
|
module.dependencyMap = undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
return module.exports = _moduleObject.exports;
|
return module.exports = _moduleObject.exports;
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
module.hasError = true;
|
module.hasError = true;
|
||||||
|
@ -107,90 +109,89 @@ function loadModuleImplementation(moduleId, module) {
|
||||||
module.exports = undefined;
|
module.exports = undefined;
|
||||||
throw e;
|
throw e;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function unknownModuleError(id) {
|
function unknownModuleError(id) {
|
||||||
var message = 'Requiring unknown module \\"' + id + '\\".';
|
var message = 'Requiring unknown module \\"' + id + '\\".';
|
||||||
|
|
||||||
return Error(message);
|
return Error(message);
|
||||||
}
|
}
|
||||||
|
|
||||||
function moduleThrewError(id, error) {
|
function moduleThrewError(id, error) {
|
||||||
var displayName = id;
|
var displayName = id;
|
||||||
return Error('Requiring module \\"' + displayName + '\\", which threw an exception: ' + error);
|
return Error('Requiring module \\"' + displayName + '\\", which threw an exception: ' + error);
|
||||||
}
|
}
|
||||||
})(typeof global !== 'undefined' ? global : typeof self !== 'undefined' ? self : this);
|
})(this);
|
||||||
(function(global) {
|
(function (global) {
|
||||||
|
'use strict';
|
||||||
|
|
||||||
'use strict';
|
if (!Object.keys) {
|
||||||
|
|
||||||
if (!Object.keys) {
|
|
||||||
Object.keys = function () {};
|
Object.keys = function () {};
|
||||||
}
|
}
|
||||||
})(typeof global !== 'undefined' ? global : typeof self !== 'undefined' ? self : this);
|
})(this);
|
||||||
(function(global) {
|
(function (global) {
|
||||||
|
'use strict';
|
||||||
|
|
||||||
'use strict';
|
if (!String.prototype.repeat) {
|
||||||
|
|
||||||
if (!String.prototype.repeat) {
|
|
||||||
String.prototype.repeat = function () {};
|
String.prototype.repeat = function () {};
|
||||||
}
|
}
|
||||||
})(typeof global !== 'undefined' ? global : typeof self !== 'undefined' ? self : this);
|
})(this);
|
||||||
__d(/* /TestBundle.js */function(global, require, module, exports) {
|
__d(function (global, require, module, exports, _dependencyMap) {
|
||||||
|
'use strict';
|
||||||
|
|
||||||
'use strict';
|
var Bar = require(_dependencyMap[0]);
|
||||||
|
|
||||||
var Bar = require(5); // 5 = ./Bar
|
var Foo = require(_dependencyMap[1]);
|
||||||
var Foo = require(6); // 6 = ./Foo
|
|
||||||
|
|
||||||
module.exports = { Foo: Foo, Bar: Bar };
|
module.exports = {
|
||||||
}, 4);
|
Foo: Foo,
|
||||||
__d(/* /Bar.js */function(global, require, module, exports) {
|
Bar: Bar
|
||||||
|
};
|
||||||
|
},4,[5,6]);
|
||||||
|
__d(function (global, require, module, exports, _dependencyMap) {
|
||||||
|
'use strict';
|
||||||
|
|
||||||
'use strict';
|
var Foo = require(_dependencyMap[0]);
|
||||||
|
|
||||||
var Foo = require(6); // 6 = ./Foo
|
module.exports = {
|
||||||
|
type: 'bar',
|
||||||
|
foo: Foo.type
|
||||||
|
};
|
||||||
|
},5,[6]);
|
||||||
|
__d(function (global, require, module, exports, _dependencyMap) {
|
||||||
|
'use strict';
|
||||||
|
|
||||||
module.exports = { type: 'bar', foo: Foo.type };
|
var asset = require(_dependencyMap[0]);
|
||||||
}, 5);
|
|
||||||
__d(/* /Foo.js */function(global, require, module, exports) {
|
|
||||||
|
|
||||||
'use strict';
|
module.exports = {
|
||||||
|
type: 'foo',
|
||||||
var asset = require(7); // 7 = ./test.png
|
asset: asset
|
||||||
|
};
|
||||||
module.exports = { type: 'foo', asset: asset };
|
},6,[7]);
|
||||||
}, 6);
|
|
||||||
__d(/* /test.png */function(global, require, module, exports) {module.exports=require(8).registerAsset({\\"__packager_asset\\":true,\\"httpServerLocation\\":\\"/assets\\",\\"width\\":8,\\"height\\":8,\\"scales\\":[1],\\"hash\\":\\"77d45c1f7fa73c0f6c444a830dc42f67\\",\\"name\\":\\"test\\",\\"type\\":\\"png\\"}); // 8 = /AssetRegistry
|
__d(/* /test.png */function(global, require, module, exports) {module.exports=require(8).registerAsset({\\"__packager_asset\\":true,\\"httpServerLocation\\":\\"/assets\\",\\"width\\":8,\\"height\\":8,\\"scales\\":[1],\\"hash\\":\\"77d45c1f7fa73c0f6c444a830dc42f67\\",\\"name\\":\\"test\\",\\"type\\":\\"png\\"}); // 8 = /AssetRegistry
|
||||||
}, 7);
|
}, 7);
|
||||||
__d(/* /AssetRegistry.js */function(global, require, module, exports) {
|
__d(function (global, require, module, exports, _dependencyMap) {
|
||||||
|
'use strict';
|
||||||
'use strict';
|
},8,[]);
|
||||||
}, 8);
|
|
||||||
require(4);"
|
require(4);"
|
||||||
`;
|
`;
|
||||||
|
|
||||||
exports[`basic_bundle bundles package without polyfills 1`] = `
|
exports[`basic_bundle bundles package without polyfills 1`] = `
|
||||||
"(function(global) {
|
"(function (global) {
|
||||||
|
global.__DEV__ = false;
|
||||||
|
global.__BUNDLE_START_TIME__ = global.nativePerformanceNow ? global.nativePerformanceNow() : Date.now();
|
||||||
|
})(this);
|
||||||
|
(function (global) {
|
||||||
|
'use strict';
|
||||||
|
|
||||||
global.__DEV__ = false;
|
global.require = _require;
|
||||||
|
global.__d = define;
|
||||||
|
var modules = Object.create(null);
|
||||||
|
|
||||||
global.__BUNDLE_START_TIME__ = global.nativePerformanceNow ? global.nativePerformanceNow() : Date.now();
|
function define(factory, moduleId, dependencyMap) {
|
||||||
})(typeof global !== 'undefined' ? global : typeof self !== 'undefined' ? self : this);
|
|
||||||
(function(global) {
|
|
||||||
|
|
||||||
'use strict';
|
|
||||||
|
|
||||||
global.require = _require;
|
|
||||||
global.__d = define;
|
|
||||||
|
|
||||||
var modules = Object.create(null);
|
|
||||||
|
|
||||||
|
|
||||||
function define(factory, moduleId, dependencyMap) {
|
|
||||||
if (moduleId in modules) {
|
if (moduleId in modules) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
modules[moduleId] = {
|
modules[moduleId] = {
|
||||||
dependencyMap: dependencyMap,
|
dependencyMap: dependencyMap,
|
||||||
exports: undefined,
|
exports: undefined,
|
||||||
|
@ -198,43 +199,51 @@ function define(factory, moduleId, dependencyMap) {
|
||||||
hasError: false,
|
hasError: false,
|
||||||
isInitialized: false
|
isInitialized: false
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
function _require(moduleId) {
|
function _require(moduleId) {
|
||||||
var moduleIdReallyIsNumber = moduleId;
|
var moduleIdReallyIsNumber = moduleId;
|
||||||
var module = modules[moduleIdReallyIsNumber];
|
var module = modules[moduleIdReallyIsNumber];
|
||||||
return module && module.isInitialized ? module.exports : guardedLoadModule(moduleIdReallyIsNumber, module);
|
return module && module.isInitialized ? module.exports : guardedLoadModule(moduleIdReallyIsNumber, module);
|
||||||
}
|
}
|
||||||
|
|
||||||
var inGuard = false;
|
var inGuard = false;
|
||||||
function guardedLoadModule(moduleId, module) {
|
|
||||||
|
function guardedLoadModule(moduleId, module) {
|
||||||
if (!inGuard && global.ErrorUtils) {
|
if (!inGuard && global.ErrorUtils) {
|
||||||
inGuard = true;
|
inGuard = true;
|
||||||
var returnValue = void 0;
|
var returnValue = void 0;
|
||||||
|
|
||||||
try {
|
try {
|
||||||
returnValue = loadModuleImplementation(moduleId, module);
|
returnValue = loadModuleImplementation(moduleId, module);
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
global.ErrorUtils.reportFatalError(e);
|
global.ErrorUtils.reportFatalError(e);
|
||||||
}
|
}
|
||||||
|
|
||||||
inGuard = false;
|
inGuard = false;
|
||||||
return returnValue;
|
return returnValue;
|
||||||
} else {
|
} else {
|
||||||
return loadModuleImplementation(moduleId, module);
|
return loadModuleImplementation(moduleId, module);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
var ID_MASK_SHIFT = 16;
|
var ID_MASK_SHIFT = 16;
|
||||||
var LOCAL_ID_MASK = ~0 >>> ID_MASK_SHIFT;
|
var LOCAL_ID_MASK = ~0 >>> ID_MASK_SHIFT;
|
||||||
|
|
||||||
function unpackModuleId(moduleId) {
|
function unpackModuleId(moduleId) {
|
||||||
var segmentId = moduleId >>> ID_MASK_SHIFT;
|
var segmentId = moduleId >>> ID_MASK_SHIFT;
|
||||||
var localId = moduleId & LOCAL_ID_MASK;
|
var localId = moduleId & LOCAL_ID_MASK;
|
||||||
return { segmentId: segmentId, localId: localId };
|
return {
|
||||||
}
|
segmentId: segmentId,
|
||||||
_require.unpackModuleId = unpackModuleId;
|
localId: localId
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
function loadModuleImplementation(moduleId, module) {
|
_require.unpackModuleId = unpackModuleId;
|
||||||
|
|
||||||
|
function loadModuleImplementation(moduleId, module) {
|
||||||
var nativeRequire = global.nativeRequire;
|
var nativeRequire = global.nativeRequire;
|
||||||
|
|
||||||
if (!module && nativeRequire) {
|
if (!module && nativeRequire) {
|
||||||
var _unpackModuleId = unpackModuleId(moduleId),
|
var _unpackModuleId = unpackModuleId(moduleId),
|
||||||
segmentId = _unpackModuleId.segmentId,
|
segmentId = _unpackModuleId.segmentId,
|
||||||
|
@ -259,16 +268,14 @@ function loadModuleImplementation(moduleId, module) {
|
||||||
dependencyMap = _module.dependencyMap;
|
dependencyMap = _module.dependencyMap;
|
||||||
|
|
||||||
try {
|
try {
|
||||||
|
var _moduleObject = {
|
||||||
var _moduleObject = { exports: exports };
|
exports: exports
|
||||||
|
};
|
||||||
factory(global, _require, _moduleObject, exports, dependencyMap);
|
factory(global, _require, _moduleObject, exports, dependencyMap);
|
||||||
|
|
||||||
{
|
{
|
||||||
module.factory = undefined;
|
module.factory = undefined;
|
||||||
module.dependencyMap = undefined;
|
module.dependencyMap = undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
return module.exports = _moduleObject.exports;
|
return module.exports = _moduleObject.exports;
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
module.hasError = true;
|
module.hasError = true;
|
||||||
|
@ -277,49 +284,54 @@ function loadModuleImplementation(moduleId, module) {
|
||||||
module.exports = undefined;
|
module.exports = undefined;
|
||||||
throw e;
|
throw e;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function unknownModuleError(id) {
|
function unknownModuleError(id) {
|
||||||
var message = 'Requiring unknown module \\"' + id + '\\".';
|
var message = 'Requiring unknown module \\"' + id + '\\".';
|
||||||
|
|
||||||
return Error(message);
|
return Error(message);
|
||||||
}
|
}
|
||||||
|
|
||||||
function moduleThrewError(id, error) {
|
function moduleThrewError(id, error) {
|
||||||
var displayName = id;
|
var displayName = id;
|
||||||
return Error('Requiring module \\"' + displayName + '\\", which threw an exception: ' + error);
|
return Error('Requiring module \\"' + displayName + '\\", which threw an exception: ' + error);
|
||||||
}
|
}
|
||||||
})(typeof global !== 'undefined' ? global : typeof self !== 'undefined' ? self : this);
|
})(this);
|
||||||
__d(/* /TestBundle.js */function(global, require, module, exports) {
|
__d(function (global, require, module, exports, _dependencyMap) {
|
||||||
|
'use strict';
|
||||||
|
|
||||||
'use strict';
|
var Bar = require(_dependencyMap[0]);
|
||||||
|
|
||||||
var Bar = require(3); // 3 = ./Bar
|
var Foo = require(_dependencyMap[1]);
|
||||||
var Foo = require(4); // 4 = ./Foo
|
|
||||||
|
|
||||||
module.exports = { Foo: Foo, Bar: Bar };
|
module.exports = {
|
||||||
}, 2);
|
Foo: Foo,
|
||||||
__d(/* /Bar.js */function(global, require, module, exports) {
|
Bar: Bar
|
||||||
|
};
|
||||||
|
},2,[3,4]);
|
||||||
|
__d(function (global, require, module, exports, _dependencyMap) {
|
||||||
|
'use strict';
|
||||||
|
|
||||||
'use strict';
|
var Foo = require(_dependencyMap[0]);
|
||||||
|
|
||||||
var Foo = require(4); // 4 = ./Foo
|
module.exports = {
|
||||||
|
type: 'bar',
|
||||||
|
foo: Foo.type
|
||||||
|
};
|
||||||
|
},3,[4]);
|
||||||
|
__d(function (global, require, module, exports, _dependencyMap) {
|
||||||
|
'use strict';
|
||||||
|
|
||||||
module.exports = { type: 'bar', foo: Foo.type };
|
var asset = require(_dependencyMap[0]);
|
||||||
}, 3);
|
|
||||||
__d(/* /Foo.js */function(global, require, module, exports) {
|
|
||||||
|
|
||||||
'use strict';
|
module.exports = {
|
||||||
|
type: 'foo',
|
||||||
var asset = require(5); // 5 = ./test.png
|
asset: asset
|
||||||
|
};
|
||||||
module.exports = { type: 'foo', asset: asset };
|
},4,[5]);
|
||||||
}, 4);
|
|
||||||
__d(/* /test.png */function(global, require, module, exports) {module.exports=require(6).registerAsset({\\"__packager_asset\\":true,\\"httpServerLocation\\":\\"/assets\\",\\"width\\":8,\\"height\\":8,\\"scales\\":[1],\\"hash\\":\\"77d45c1f7fa73c0f6c444a830dc42f67\\",\\"name\\":\\"test\\",\\"type\\":\\"png\\"}); // 6 = /AssetRegistry
|
__d(/* /test.png */function(global, require, module, exports) {module.exports=require(6).registerAsset({\\"__packager_asset\\":true,\\"httpServerLocation\\":\\"/assets\\",\\"width\\":8,\\"height\\":8,\\"scales\\":[1],\\"hash\\":\\"77d45c1f7fa73c0f6c444a830dc42f67\\",\\"name\\":\\"test\\",\\"type\\":\\"png\\"}); // 6 = /AssetRegistry
|
||||||
}, 5);
|
}, 5);
|
||||||
__d(/* /AssetRegistry.js */function(global, require, module, exports) {
|
__d(function (global, require, module, exports, _dependencyMap) {
|
||||||
|
'use strict';
|
||||||
'use strict';
|
},6,[]);
|
||||||
}, 6);
|
|
||||||
require(2);"
|
require(2);"
|
||||||
`;
|
`;
|
||||||
|
|
|
@ -212,9 +212,7 @@ function validateCachedResult(cachedResult: mixed): ?CachedResult {
|
||||||
typeof cachedResult === 'object' &&
|
typeof cachedResult === 'object' &&
|
||||||
typeof cachedResult.code === 'string' &&
|
typeof cachedResult.code === 'string' &&
|
||||||
Array.isArray(cachedResult.dependencies) &&
|
Array.isArray(cachedResult.dependencies) &&
|
||||||
cachedResult.dependencies.every(dep => typeof dep === 'string') &&
|
cachedResult.dependencies.every(dep => typeof dep === 'string')
|
||||||
Array.isArray(cachedResult.dependencyOffsets) &&
|
|
||||||
cachedResult.dependencyOffsets.every(offset => typeof offset === 'number')
|
|
||||||
) {
|
) {
|
||||||
return (cachedResult: any);
|
return (cachedResult: any);
|
||||||
}
|
}
|
||||||
|
|
|
@ -33,8 +33,7 @@ const CACHE_SUB_DIR = 'cache';
|
||||||
|
|
||||||
export type CachedResult = {
|
export type CachedResult = {
|
||||||
code: string,
|
code: string,
|
||||||
dependencies: Array<string>,
|
dependencies: $ReadOnlyArray<string>,
|
||||||
dependencyOffsets: Array<number>,
|
|
||||||
map: CompactRawMappings,
|
map: CompactRawMappings,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -143,7 +142,6 @@ class FileBasedCache {
|
||||||
.digest('hex'),
|
.digest('hex'),
|
||||||
hashSourceCode(props),
|
hashSourceCode(props),
|
||||||
result.dependencies,
|
result.dependencies,
|
||||||
result.dependencyOffsets,
|
|
||||||
result.map,
|
result.map,
|
||||||
]),
|
]),
|
||||||
);
|
);
|
||||||
|
@ -214,7 +212,6 @@ class FileBasedCache {
|
||||||
result: {
|
result: {
|
||||||
code: transformedCode,
|
code: transformedCode,
|
||||||
dependencies: metadata.dependencies,
|
dependencies: metadata.dependencies,
|
||||||
dependencyOffsets: metadata.dependencyOffsets,
|
|
||||||
map: metadata.sourceMap,
|
map: metadata.sourceMap,
|
||||||
},
|
},
|
||||||
outdatedDependencies: EMPTY_ARRAY,
|
outdatedDependencies: EMPTY_ARRAY,
|
||||||
|
@ -335,7 +332,6 @@ function readMetadataFileSync(
|
||||||
cachedResultHash: string,
|
cachedResultHash: string,
|
||||||
cachedSourceHash: string,
|
cachedSourceHash: string,
|
||||||
dependencies: Array<string>,
|
dependencies: Array<string>,
|
||||||
dependencyOffsets: Array<number>,
|
|
||||||
sourceMap: CompactRawMappings,
|
sourceMap: CompactRawMappings,
|
||||||
} {
|
} {
|
||||||
const metadataStr = fs.readFileSync(metadataFilePath, 'utf8');
|
const metadataStr = fs.readFileSync(metadataFilePath, 'utf8');
|
||||||
|
@ -347,7 +343,6 @@ function readMetadataFileSync(
|
||||||
cachedResultHash,
|
cachedResultHash,
|
||||||
cachedSourceHash,
|
cachedSourceHash,
|
||||||
dependencies,
|
dependencies,
|
||||||
dependencyOffsets,
|
|
||||||
sourceMap,
|
sourceMap,
|
||||||
] = metadata;
|
] = metadata;
|
||||||
if (
|
if (
|
||||||
|
@ -357,10 +352,6 @@ function readMetadataFileSync(
|
||||||
Array.isArray(dependencies) &&
|
Array.isArray(dependencies) &&
|
||||||
dependencies.every(dep => typeof dep === 'string')
|
dependencies.every(dep => typeof dep === 'string')
|
||||||
) ||
|
) ||
|
||||||
!(
|
|
||||||
Array.isArray(dependencyOffsets) &&
|
|
||||||
dependencyOffsets.every(offset => typeof offset === 'number')
|
|
||||||
) ||
|
|
||||||
!(sourceMap == null || typeof sourceMap === 'object')
|
!(sourceMap == null || typeof sourceMap === 'object')
|
||||||
) {
|
) {
|
||||||
return null;
|
return null;
|
||||||
|
@ -369,7 +360,6 @@ function readMetadataFileSync(
|
||||||
cachedResultHash,
|
cachedResultHash,
|
||||||
cachedSourceHash,
|
cachedSourceHash,
|
||||||
dependencies,
|
dependencies,
|
||||||
dependencyOffsets,
|
|
||||||
sourceMap,
|
sourceMap,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
|
@ -71,7 +71,6 @@ describe('TransformCaching.FileBasedCache', () => {
|
||||||
result: {
|
result: {
|
||||||
code: `/* result for ${key} */`,
|
code: `/* result for ${key} */`,
|
||||||
dependencies: ['foo', `dep of ${key}`],
|
dependencies: ['foo', `dep of ${key}`],
|
||||||
dependencyOffsets: [12, 34],
|
|
||||||
map: {desc: `source map for ${key}`},
|
map: {desc: `source map for ${key}`},
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
@ -104,7 +103,6 @@ describe('TransformCaching.FileBasedCache', () => {
|
||||||
result: {
|
result: {
|
||||||
code: `/* result for ${key} */`,
|
code: `/* result for ${key} */`,
|
||||||
dependencies: ['foo', 'bar'],
|
dependencies: ['foo', 'bar'],
|
||||||
dependencyOffsets: [12, 34],
|
|
||||||
map: {desc: `source map for ${key}`},
|
map: {desc: `source map for ${key}`},
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
|
@ -39,8 +39,7 @@ import type {LocalPath} from './lib/toLocalPath';
|
||||||
|
|
||||||
export type ReadResult = {
|
export type ReadResult = {
|
||||||
+code: string,
|
+code: string,
|
||||||
+dependencies: Array<string>,
|
+dependencies: $ReadOnlyArray<string>,
|
||||||
+dependencyOffsets?: ?Array<number>,
|
|
||||||
+map: CompactRawMappings,
|
+map: CompactRawMappings,
|
||||||
+source: string,
|
+source: string,
|
||||||
};
|
};
|
||||||
|
|
Loading…
Reference in New Issue