react-native/packager
Jean Lauliac a79043035e packager: ResolutionRequest: refactor _loadAsDir and dependents
Summary:
Working on refactoring error handling in `_loadAsDir` I figured out it was oftentimes problematic to pass on the candidates out of the functions as an array, as in practice there's always a single "candidates" object passed out. Also, using an array prevented nice Flow typing and forced additional invariants to be added. So I replaced that by a return value that explicitely can be either a module, or resolution candidates. That way the semantic is more clear: we don't get any candidates if we did resolve properly, and at the same time we enforce returning candidates if we could not resolve any module.

At first I wanted to just have type `{module: TModule} | {candidate: TCandidate}`, but Flow would hit edge cases, so instead I added a field `type` that make it explicit what is the result of the resolution, and allows Flow to refine the type fully after we test that field. This allows us to remove the extraneous invariants. Also, a nice thing is that at a few places, even if the type of the candididate is different, Flow allows us to return the "resolved" object just as it is, that prevents using more memory and causing more garbage collection than necessary.

Since we're creating more objects with that solution, this will be slightly less performant than returning `Module` objects directly, but I don't think it is worth micro optimizing this at that point. If really we see this to be causing trouble later, I'd try to find solutions such as reusing a pool of objects. Ex. we could pass the result `Resolution` object as argument instead of returning a fresh one, but that would make the code less legible, more complex.

Reviewed By: davidaurelio

Differential Revision: D5111501

fbshipit-source-id: f41cdab00640124081cfdf07668169bb2d5c00be
2017-05-23 12:46:56 -07:00
..
src packager: ResolutionRequest: refactor _loadAsDir and dependents 2017-05-23 12:46:56 -07:00
.eslintrc Add `"node"` env to packager `.eslintrc` 2017-05-19 07:32:36 -07:00
Glossary.md Add glossary, some renames 2017-05-11 16:47:43 -07:00
README.md packager: create the reporter higher in the stack 2017-01-09 10:13:33 -08:00
babelRegisterOnly.js packager: add transform-class-properties transform 2017-05-12 07:39:10 -07:00
blacklist.js RN: Cleanup Flow / Jest / Metro Configs 2017-03-26 16:05:10 -07:00
defaults.js packager: add support for relative files with custom extensions 2017-05-04 05:21:44 -07:00
index.js Remove `react-packager` indirection. 2017-02-02 05:44:15 -08:00
launchPackager.bat Launch the packager with `react-native run-android` on Windows 2016-04-29 04:14:22 -07:00
launchPackager.command Run packager.sh using '.' instead of 'source'. 2017-05-19 14:50:53 -07:00
package.json packager: add missing dep 2017-05-23 09:16:16 -07:00
packager.sh Fix cli entry points 2015-10-22 10:13:13 -07:00
react-native-xcode.sh Don't build js bundles for debug simulator build configurations 2017-04-15 10:17:51 -07:00
react-packager.js packager: runServer: @flow 2017-05-17 05:03:41 -07:00
rn-babelrc.json Remove `react-packager` indirection. 2017-02-02 05:44:15 -08:00
rn-cli.config.js packager: add support for relative files with custom extensions 2017-05-04 05:21:44 -07:00
setupNodePolyfills.js Separate node polyfills from runtime transforms 2017-05-02 05:35:22 -07:00
transformer.js Minor lint fixes 2017-05-23 06:46:37 -07:00

README.md

React Native Packager

React Native Packager is a project similar in scope to browserify or webpack, it provides a CommonJS-like module system, JavaScript compilation (ES6, Flow, JSX), bundling, and asset loading.

The main difference is the Packager's focus on compilation and bundling speed. We aim for a sub-second edit-reload cycles. Additionally, we don't want users -- with large code bases -- to wait more than a few seconds after starting the packager.

The main deviation from the node module system is the support for our proprietary module format known as @providesModule. However, we discourage people from using this module format because going forward we want to completely separate our infrastructure from React Native and provide an experience most JavaScript developers are familiar with, namely the node module format. We want to even go further, and let you choose your own packager and asset pipeline or even integrate into your existing infrastructure.

React Native users need not to understand how the packager work, however, this documentation might be useful for advanced users and people who want to fix bugs or add features to the packager (patches welcome!).

HTTP interface

The main way you'd interact with the packager is via the HTTP interface. The following is the list of endpoints and their respective functions.

/path/to/moduleName.bundle

Does the following in order:

  • parse out path/to/moduleName
  • add a .js suffix to the path
  • looks in your project root(s) for the file
  • recursively collects all the dependencies from an in memory graph
  • runs the modules through the transformer (might just be cached)
  • concatenate the modules' content into a bundle
  • responds to the client with the bundle (and a SourceMap URL)

/path/to/moduleName.map

  • if the package has been previously generated via the .bundle endpoint then the source map will be generated from that package
  • if the package has not been previously asked for, this will go through the same steps outlined in the .bundle endpoint then generate the source map.

Note that source map generation currently assumes that the code has been compiled with jstransform, which preserves line and column numbers which allows us to generate source maps super fast.

/path/to/moduleName.(map|bundle) query params

You can pass options for the bundle creation through the query params, if the option is boolean 1/0 or true/false is accepted.

Here are the current options the packager accepts:

  • dev boolean, defaults to true: sets a global __DEV__ variable which will effect how the React Native core libraries behave.
  • minify boolean, defaults to false: whether to minify the bundle.
  • runModule boolean, defaults to true: whether to require your entry point module. So if you requested moduleName, this option will add a require('moduleName') the end of your bundle.
  • inlineSourceMap boolean, defaults to false: whether to inline source maps.

/debug

This is a page used for debugging, it offers a link to a single page :

  • Cached Packages: which shows you the packages that's been already generated and cached

Programmatic API

The packager is made of two things:

  • The core packager (which we're calling ReactPackager)
  • The scripts, devtools launcher, server run etc.

ReactPackager is how you mainly interact with the API.

var ReactPackager = require('./react-packager');

ReactPackager.buildBundle(serverOptions, bundleOptions)

Builds a bundle according to the provided options.

serverOptions

  • projectRoots array (required): Is the roots where your JavaScript file will exist
  • blacklistRE regexp: Is a pattern to ignore certain paths from the packager
  • polyfillModuleName array: Paths to polyfills you want to be included at the start of the bundle
  • cacheVersion string: used in creating the cache file
  • resetCache boolean, defaults to false: whether to use the cache on disk
  • transformModulePath string: Path to the module used as a JavaScript transformer
  • nonPersistent boolean, defaults to false: Whether the server should be used as a persistent deamon to watch files and update itself
  • getTransformOptions function: A function that acts as a middleware for generating options to pass to the transformer based on the bundle being built.
  • reporter object (required): An object with a single function update that is called when events are happening: build updates, warnings, errors.

bundleOptions

  • entryFile string (required): the entry file of the bundle, relative to one of the roots.
  • dev boolean (defaults to true): sets a global __DEV__ variable which will effect how the React Native core libraries behave.
  • minify boolean: Whether to minify code and apply production optimizations.
  • runModule boolean (defaults to true): whether to require your entry point module.
  • inlineSourceMap boolean, defaults to false: whether to inline source maps.
  • platform string: The target platform for the build
  • generateSourceMaps boolean: Whether to generate source maps.
  • sourceMapUrl string: The url of the source map (will be appended to the bundle).

Debugging

To get verbose output when running the packager, define an environment variable:

export DEBUG=RNP:*

You can combine this with other values, e.g. DEBUG=babel,RNP:*. Under the hood this uses the debug package, see its documentation for all the available options.

The /debug endpoint discussed above is also useful.

FAQ

Can I use this in my own non-React Native project?

Yes. It's not really tied to React Native, however feature development is informed by React Native needs.

Why didn't you use webpack?

We love webpack, however, when we tried on our codebase it was slower than our developers would like it to be. You can find more discussion about the subject here.