packager: buck assets: wrap in __d

Reviewed By: davidaurelio

Differential Revision: D5003464

fbshipit-source-id: 59800dded389d133394b60530eb97642d67a5c89
This commit is contained in:
Jean Lauliac 2017-05-08 04:28:26 -07:00 committed by Facebook Github Bot
parent 43900f558c
commit 021ff010d8
3 changed files with 88 additions and 43 deletions

View File

@ -11,6 +11,10 @@
'use strict'; 'use strict';
const babel = require('babel-core');
const babelGenerate = require('babel-generator').default;
const babylon = require('babylon');
import type {AssetDescriptor} from '.'; import type {AssetDescriptor} from '.';
const assetPropertyBlacklist = new Set([ const assetPropertyBlacklist = new Set([
@ -19,18 +23,33 @@ const assetPropertyBlacklist = new Set([
'path', 'path',
]); ]);
const ASSET_REGISTRY_PATH = 'react-native/Libraries/Image/AssetRegistry';
function generateAssetCodeFileAst(assetDescriptor: AssetDescriptor): Object {
const properDescriptor = filterObject(assetDescriptor, assetPropertyBlacklist);
const descriptorAst = babylon.parseExpression(JSON.stringify(properDescriptor));
const t = babel.types;
const moduleExports = t.memberExpression(t.identifier('module'), t.identifier('exports'));
const requireCall =
t.callExpression(t.identifier('require'), [t.stringLiteral(ASSET_REGISTRY_PATH)]);
const registerAssetFunction = t.memberExpression(requireCall, t.identifier('registerAsset'));
const registerAssetCall = t.callExpression(registerAssetFunction, [descriptorAst]);
return t.file(t.program([
t.expressionStatement(t.assignmentExpression('=', moduleExports, registerAssetCall)),
]));
}
function generateAssetTransformResult(assetDescriptor: AssetDescriptor): {| function generateAssetTransformResult(assetDescriptor: AssetDescriptor): {|
code: string, code: string,
dependencies: Array<string>, dependencies: Array<string>,
dependencyOffsets: Array<number>, dependencyOffsets: Array<number>,
|} { |} {
const properDescriptor = filterObject(assetDescriptor, assetPropertyBlacklist); const {code} = babelGenerate(
const json = JSON.stringify(properDescriptor); generateAssetCodeFileAst(assetDescriptor),
const assetRegistryPath = 'react-native/Libraries/Image/AssetRegistry'; {comments: false, compact: true},
const code = );
`module.exports = require(${JSON.stringify(assetRegistryPath)}).registerAsset(${json});`; const dependencies = [ASSET_REGISTRY_PATH];
const dependencies = [assetRegistryPath]; const dependencyOffsets = [code.indexOf(ASSET_REGISTRY_PATH) - 1];
const dependencyOffsets = [code.indexOf(assetRegistryPath) - 1];
return {code, dependencies, dependencyOffsets}; return {code, dependencies, dependencyOffsets};
} }
@ -50,5 +69,8 @@ function filterObject(object, blacklist) {
return copied; return copied;
} }
exports.generateAssetTransformResult = generateAssetTransformResult; module.exports = {
exports.isAssetTypeAnImage = isAssetTypeAnImage; generateAssetCodeFileAst,
generateAssetTransformResult,
isAssetTypeAnImage
};

View File

@ -0,0 +1,52 @@
/**
* 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.
*
* @flow
*/
'use strict';
const babel = require('babel-core');
const MODULE_FACTORY_PARAMETERS = ['global', 'require', 'module', 'exports'];
const POLYFILL_FACTORY_PARAMETERS = ['global'];
function wrapModule(fileAst: Object, dependencyMapName: string): Object {
const t = babel.types;
const params = MODULE_FACTORY_PARAMETERS.concat(dependencyMapName);
const factory = functionFromProgram(fileAst.program, params);
const def = t.callExpression(t.identifier('__d'), [factory]);
return t.file(t.program([t.expressionStatement(def)]));
}
function wrapPolyfill(fileAst: Object): Object {
const t = babel.types;
const factory = functionFromProgram(fileAst.program, POLYFILL_FACTORY_PARAMETERS);
const iife = t.callExpression(factory, [t.identifier('this')]);
return t.file(t.program([t.expressionStatement(iife)]));
}
function functionFromProgram(program: Object, parameters: Array<string>): Object {
const t = babel.types;
return t.functionExpression(
t.identifier(''),
parameters.map(makeIdentifier),
t.blockStatement(program.body, program.directives),
);
}
function makeIdentifier(name: string): Object {
return babel.types.identifier(name);
}
module.exports = {
MODULE_FACTORY_PARAMETERS,
POLYFILL_FACTORY_PARAMETERS,
wrapModule,
wrapPolyfill,
};

View File

@ -10,7 +10,8 @@
*/ */
'use strict'; 'use strict';
const babel = require('babel-core'); const JsFileWrapping = require('./JsFileWrapping');
const collectDependencies = require('./collect-dependencies'); const collectDependencies = require('./collect-dependencies');
const defaults = require('../../../defaults'); const defaults = require('../../../defaults');
const docblock = require('../../node-haste/DependencyGraph/docblock'); const docblock = require('../../node-haste/DependencyGraph/docblock');
@ -38,8 +39,6 @@ export type TransformOptions = {|
|}; |};
const defaultVariants = {default: {}}; const defaultVariants = {default: {}};
const moduleFactoryParameters = ['global', 'require', 'module', 'exports'];
const polyfillFactoryParameters = ['global'];
const ASSET_EXTENSIONS = new Set(defaults.assetExts); const ASSET_EXTENSIONS = new Set(defaults.assetExts);
@ -109,7 +108,7 @@ function transformJSON(json, options, callback) {
const value = JSON.parse(json); const value = JSON.parse(json);
const {filename} = options; const {filename} = options;
const code = const code =
`__d(function(${moduleFactoryParameters.join(', ')}) { module.exports = \n${ `__d(function(${JsFileWrapping.MODULE_FACTORY_PARAMETERS.join(', ')}) { module.exports = \n${
json json
}\n});`; }\n});`;
@ -162,42 +161,14 @@ function makeResult(ast, filename, sourceCode, isPolyfill = false) {
let dependencies, dependencyMapName, file; let dependencies, dependencyMapName, file;
if (isPolyfill) { if (isPolyfill) {
dependencies = []; dependencies = [];
file = wrapPolyfill(ast); file = JsFileWrapping.wrapPolyfill(ast);
} else { } else {
({dependencies, dependencyMapName} = collectDependencies(ast)); ({dependencies, dependencyMapName} = collectDependencies(ast));
file = wrapModule(ast, dependencyMapName); file = JsFileWrapping.wrapModule(ast, dependencyMapName);
} }
const gen = generate(file, filename, sourceCode); const gen = generate(file, filename, sourceCode);
return {code: gen.code, map: gen.map, dependencies, dependencyMapName}; return {code: gen.code, map: gen.map, dependencies, dependencyMapName};
} }
function wrapModule(file, dependencyMapName) {
const t = babel.types;
const params = moduleFactoryParameters.concat(dependencyMapName);
const factory = functionFromProgram(file.program, params);
const def = t.callExpression(t.identifier('__d'), [factory]);
return t.file(t.program([t.expressionStatement(def)]));
}
function wrapPolyfill(file) {
const t = babel.types;
const factory = functionFromProgram(file.program, polyfillFactoryParameters);
const iife = t.callExpression(factory, [t.identifier('this')]);
return t.file(t.program([t.expressionStatement(iife)]));
}
function functionFromProgram(program, parameters) {
const t = babel.types;
return t.functionExpression(
t.identifier(''),
parameters.map(makeIdentifier),
t.blockStatement(program.body, program.directives),
);
}
function makeIdentifier(name) {
return babel.types.identifier(name);
}
module.exports = transformModule; module.exports = transformModule;