/** * 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. * * @flow */ 'use strict'; const blacklist = require('metro-bundler/src/blacklist'); const findSymlinksPaths = require('./findSymlinksPaths'); const fs = require('fs'); const invariant = require('fbjs/lib/invariant'); const path = require('path'); const {providesModuleNodeModules} = require('metro-bundler/src/defaults'); const RN_CLI_CONFIG = 'rn-cli.config.js'; import type { GetTransformOptions, PostMinifyProcess, PostProcessModules, // $FlowFixMe: Exported by metro bundler PostProcessBundleSourcemap } from 'metro-bundler/src/Bundler'; import type {HasteImpl} from 'metro-bundler/src/node-haste/Module'; import type {TransformVariants} from 'metro-bundler/src/ModuleGraph/types.flow'; import type {PostProcessModules as PostProcessModulesForBuck} from 'metro-bundler/src/ModuleGraph/types.flow.js'; /** * Configuration file of the CLI. */ export type ConfigT = { extraNodeModules: {[id: string]: string}, /** * Specify any additional asset file extensions to be used by the packager. * For example, if you want to include a .ttf file, you would return ['ttf'] * from here and use `require('./fonts/example.ttf')` inside your app. */ getAssetExts: () => Array, /** * Returns a regular expression for modules that should be ignored by the * packager on a given platform. */ getBlacklistRE(): RegExp, /** * Specify any additional polyfill modules that should be processed * before regular module loading. */ getPolyfillModuleNames: () => Array, /** * Specify any additional platforms to be used by the packager. * For example, if you want to add a "custom" platform, and use modules * ending in .custom.js, you would return ['custom'] here. */ getPlatforms: () => Array, getProjectRoots(): Array, /** * Specify any additional node modules that should be processed for * providesModule declarations. */ getProvidesModuleNodeModules?: () => Array, /** * Specify any additional source file extensions to be used by the packager. * For example, if you want to include a .ts file, you would return ['ts'] * from here and use `require('./module/example')` to require the file with * path 'module/example.ts' inside your app. */ getSourceExts: () => Array, /** * Returns the path to a custom transformer. This can also be overridden * with the --transformer commandline argument. */ getTransformModulePath: () => string, getTransformOptions: GetTransformOptions, /** * Returns the path to the worker that is used for transformation. */ getWorkerPath: () => ?string, /** * An optional function that can modify the code and source map of bundle * after the minifaction took place. (Function applied per module). */ postMinifyProcess: PostMinifyProcess, /** * An optional function that can modify the module array before the bundle is * finalized. */ postProcessModules: PostProcessModules, /** * An optional function that can modify the code and source map of the bundle * before it is written. Applied once for the entire bundle, only works if * output is a plainBundle. */ postProcessBundleSourcemap: PostProcessBundleSourcemap, /** * Same as `postProcessModules` but for the Buck worker. Eventually we do want * to unify both variants. */ postProcessModulesForBuck: PostProcessModulesForBuck, /** * A module that exports: * - a `getHasteName(filePath)` method that returns `hasteName` for module at * `filePath`, or undefined if `filePath` is not a haste module. */ hasteImpl?: HasteImpl, transformVariants: () => TransformVariants, }; function getProjectPath() { if (__dirname.match(/node_modules[\/\\]react-native[\/\\]local-cli[\/\\]util$/)) { // Packager is running from node_modules. // This is the default case for all projects created using 'react-native init'. return path.resolve(__dirname, '../../../..'); } else if (__dirname.match(/Pods[\/\\]React[\/\\]packager$/)) { // React Native was installed using CocoaPods. return path.resolve(__dirname, '../../../..'); } return path.resolve(__dirname, '../..'); } const resolveSymlink = (roots) => roots.concat( findSymlinksPaths( path.join(getProjectPath(), 'node_modules'), roots ) ); /** * Module capable of getting the configuration out of a given file. * * The function will return all the default configuration, as specified by the * `DEFAULTS` param overriden by those found on `rn-cli.config.js` files, if any. If no * default config is provided and no configuration can be found in the directory * hierarchy, an error will be thrown. */ const Config = { DEFAULTS: ({ extraNodeModules: Object.create(null), getAssetExts: () => [], getBlacklistRE: () => blacklist(), getPlatforms: () => [], getPolyfillModuleNames: () => [], getProjectRoots: () => { const root = process.env.REACT_NATIVE_APP_ROOT; if (root) { return resolveSymlink([path.resolve(root)]); } return resolveSymlink([getProjectPath()]); }, getProvidesModuleNodeModules: () => providesModuleNodeModules.slice(), getSourceExts: () => [], getTransformModulePath: () => require.resolve('metro-bundler/src/transformer.js'), getTransformOptions: async () => ({}), postMinifyProcess: x => x, postProcessModules: modules => modules, postProcessModulesForBuck: modules => modules, postProcessBundleSourcemap: ({code, map, outFileName}) => ({code, map}), transformVariants: () => ({default: {}}), getWorkerPath: () => null, }: ConfigT), find(startDir: string): ConfigT { return Config.findCustom(startDir, Config.DEFAULTS); }, /** * This allows a callsite to grab a config that may have custom fields or * a different version of the config. In that case the defaults have to be * specified explicitely. */ findCustom(startDir: string, defaults: TConfig): TConfig { return Config.findWithPathCustom(startDir, defaults).config; }, findWithPath(startDir: string): {config: ConfigT, projectPath: string} { return Config.findWithPathCustom(startDir, Config.DEFAULTS); }, findWithPathCustom(startDir: string, defaults: TConfig): {config: TConfig, projectPath: string} { const configPath = findConfigPath(startDir); invariant( configPath, `Can't find "${RN_CLI_CONFIG}" file in any parent folder of "${startDir}"`, ); const projectPath = path.dirname(configPath); return {config: Config.loadFileCustom(configPath, defaults), projectPath}; }, findOptional(startDir: string): ConfigT { return Config.findOptionalCustom(startDir, Config.DEFAULTS); }, findOptionalCustom(startDir: string, defaults: TConfig): TConfig { const configPath = findConfigPath(startDir); return configPath ? Config.loadFileCustom(configPath, defaults) : {...defaults}; }, loadFile(pathToConfig: string): ConfigT { return Config.loadFileCustom(pathToConfig, Config.DEFAULTS); }, loadFileCustom(pathToConfig: string, defaults: TConfig): TConfig { //$FlowFixMe: necessary dynamic require const config: {} = require(pathToConfig); return {...defaults, ...config}; }, }; function findConfigPath(cwd: string): ?string { const parentDir = findParentDirectory(cwd, RN_CLI_CONFIG); return parentDir ? path.join(parentDir, RN_CLI_CONFIG) : null; } // Finds the most near ancestor starting at `currentFullPath` that has // a file named `filename` function findParentDirectory(currentFullPath, filename) { const root = path.parse(currentFullPath).root; const testDir = (parts) => { if (parts.length === 0) { return null; } const fullPath = path.join(root, parts.join(path.sep)); var exists = fs.existsSync(path.join(fullPath, filename)); return exists ? fullPath : testDir(parts.slice(0, -1)); }; return testDir(currentFullPath.substring(root.length).split(path.sep)); } module.exports = Config;