Skip to content

Commit

Permalink
[FEAT] Adds an option to emit external babel helpers
Browse files Browse the repository at this point in the history
This PR adds the `includeExternalHelpers` config option which allows
users to explicitly opt into using externalized helpers for transpiled
code. In large apps this can help with app size, especially in apps
which are making heavy usage of new ES features like native classes and
decorators.

This option is a _global_ option, meaning it _must_ be configured in the
root application. The reason for this is because there must be one
canonical source of truth for which version of helpers are being used,
and whether or not helpers are being used at all. If one addon decides
to use helpers, the app must include them, and they must be the right
version.

By default, ec-babel will check the dependency tree to see if there are
any addons which exist which will incur heavy expenses, such as
decorators. If so, it'll externalize helpers by default. Eventually this
behavior will be unnecessary due to treeshaking.

The current setup will _only_ support babel 7 helpers. This can be
expanded upon in the future, but the complexity of doing so makes it
beyond the scope of the initial work here.
  • Loading branch information
pzuraq committed Dec 18, 2018
1 parent 2dc8ecf commit 9291be1
Show file tree
Hide file tree
Showing 10 changed files with 2,392 additions and 1,042 deletions.
1 change: 1 addition & 0 deletions .eslintrc.js
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,7 @@ module.exports = {
'blueprints/*/index.js',
'config/**/*.js',
'tests/dummy/config/**/*.js',
'lib/**/*.js'
],
excludedFiles: [
'addon/**',
Expand Down
28 changes: 28 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -94,6 +94,7 @@ interface EmberCLIBabelConfig {
*/
'ember-cli-babel'?: {
includePolyfill?: boolean;
includeExternalHelpers?: boolean;
compileModules?: boolean;
disableDebugTooling?: boolean;
disablePresetEnv?: boolean;
Expand Down Expand Up @@ -159,6 +160,33 @@ let app = new EmberApp(defaults, {
});
```

#### External Helpers

Babel often includes helper functions to handle some of the more complex logic
in codemods. These functions are inlined by default, so they are duplicated in
every file that they are used in, which adds some extra weight to final builds.

Enabling `includeExternalHelpers` will cause Babel to import these helpers from
a shared module, reducing app size overall. This option is available _only_ to
the root application, because it is a global configuration value. There must be
one agreed upon Babel version to generate helpers, and if any addon opts into
generating helpers then they must be included and synced with any other addons
and the root app.

```js
// ember-cli-build.js

let app = new EmberApp(defaults, {
'ember-cli-babel': {
includeExternalHelpers: true
}
});
```

Note that there is currently no way to whitelist or blacklist helpers, so all
helpers will be included, even ones which are not used. If your app is small,
this could add to overall build size, so be sure to check.

#### Enabling Source Maps

Babel generated source maps will enable you to debug your original ES6 source
Expand Down
3 changes: 2 additions & 1 deletion ember-cli-build.js
Original file line number Diff line number Diff line change
Expand Up @@ -19,10 +19,11 @@ module.exports = function(defaults) {
let app = new EmberAddon(defaults, {
'ember-cli-babel': {
includePolyfill: true,
includeExternalHelpers: true,
// ember-cli-babels defaults, should be parallelizable. If they are not,
// it should fail to build. This flag being enabled ensure that to be the
// case.
throwUnlessParallelizable: true
throwUnlessParallelizable: true,
}
});

Expand Down
82 changes: 79 additions & 3 deletions index.js
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,8 @@ const clone = require('clone');
const path = require('path');
const semver = require('semver');

const defaultShouldIncludeHelpers = require('./lib/default-should-include-helpers');

let count = 0;

module.exports = {
Expand Down Expand Up @@ -84,19 +86,85 @@ module.exports = {
}
},

_shouldIncludeHelpers() {
let customAddonOptions = this.parent && this.parent.options && this.parent.options['ember-cli-babel'];

if (customAddonOptions && 'includeExternalHelpers' in customAddonOptions) {
throw new Error('includeExternalHelpers is not supported in addon configurations, it is an app-wide configuration option');
}

let hostOptions = this._getHostOptions();
let customOptions = hostOptions['ember-cli-babel'];

if (customOptions && 'includeExternalHelpers' in customOptions) {
return customOptions.includeExternalHelpers === true;
} else {
// Check the project to see if we should include helpers based on heuristics.
return defaultShouldIncludeHelpers(this.project);
}
},

_getHelperVersion() {
if (!this._helperVersion) {
let checker = new VersionChecker(this.project);
this._helperVersion = checker.for('@babel/core', 'npm').version;
}

return this._helperVersion;
},

_getHelpersPlugin() {
return [
[
require.resolve('@babel/plugin-transform-runtime'),
{
version: this._getHelperVersion(),
regenerator: false,
useESModules: true
}
]
]
},

treeForAddon() {
// Helpers are a global config, so only the root application should bother
// generating and including the file.
if (!(this.parent === this.project && this._shouldIncludeHelpers())) return;

const path = require('path');
const Funnel = require('broccoli-funnel');
const UnwatchedDir = require('broccoli-source').UnwatchedDir;

const babelHelpersPath = path.dirname(require.resolve('@babel/runtime/package.json'));

let babelHelpersTree = new Funnel(new UnwatchedDir(babelHelpersPath), {
srcDir: 'helpers/esm',
destDir: '@babel/runtime/helpers/esm'
});

return this.transpileTree(babelHelpersTree, {
'ember-cli-babel': {
// prevents the helpers from being double transpiled, and including themselves
disablePresetEnv: true
}
});
},

treeForVendor() {
if (!this._shouldIncludePolyfill()) { return; }
if (!this._shouldIncludePolyfill()) return;

const Funnel = require('broccoli-funnel');
const UnwatchedDir = require('broccoli-source').UnwatchedDir;

// Find babel-core's browser polyfill and use its directory as our vendor tree
let polyfillDir = path.dirname(require.resolve('@babel/polyfill/dist/polyfill'));

return new Funnel(new UnwatchedDir(polyfillDir), {
let polyfillTree = new Funnel(new UnwatchedDir(polyfillDir), {
destDir: 'babel-polyfill'
});
},

return polyfillTree;
},

included: function(app) {
this._super.included.apply(this, arguments);
Expand All @@ -123,6 +191,12 @@ module.exports = {
return (this.parent && this.parent.options) || (this.app && this.app.options) || {};
},

_getHostOptions() {
let host = this._findHost();

return (host && host.options) || {};
},

_parentName() {
let parentName;

Expand Down Expand Up @@ -158,6 +232,7 @@ module.exports = {
_getBabelOptions(config) {
let addonProvidedConfig = this._getAddonProvidedConfig(config);
let shouldCompileModules = this._shouldCompileModules(config);
let shouldIncludeHelpers = this._shouldIncludeHelpers();

let emberCLIBabelConfig = config['ember-cli-babel'];
let shouldRunPresetEnv = true;
Expand Down Expand Up @@ -188,6 +263,7 @@ module.exports = {
let userPostTransformPlugins = addonProvidedConfig.postTransformPlugins;

options.plugins = [].concat(
shouldIncludeHelpers && this._getHelpersPlugin(),
userPlugins,
this._getDebugMacroPlugins(config),
this._getEmberModulesAPIPolyfill(config),
Expand Down
38 changes: 38 additions & 0 deletions lib/default-should-include-helpers.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,38 @@
const VersionChecker = require('ember-cli-version-checker');

const SHOULD_INCLUDE_HELPERS = new WeakMap();

function shouldIncludeHelpers(addonOrProject) {
// Currently we check for @ember-decorators transforms specifically, but we
// could check for any number of heuristics in this function. What we want is
// to default to on if we reasonably believe that users will incur massive
// cost for inlining helpers. Stage 2+ decorators are a very clear indicator
// that helpers should be included, at 12kb for the helpers, it pays for
// itself after usage in 5 files. With stage 1 decorators, it pays for itself
// after 25 files.
let checker = new VersionChecker(addonOrProject).for('@ember-decorators/babel-transforms', 'npm');

if (checker.exists()) {
return true;
}

if (addonOrProject.addons) {
return addonOrProject.addons.some(hasDecoratorsTransform);
}

return false;
}


module.exports = function defaultShouldIncludeHelpers(project) {
if (SHOULD_INCLUDE_HELPERS.has(project)) {
return SHOULD_INCLUDE_HELPERS.get(project);
}

let hasDecorators = shouldIncludeHelpers(project);

SHOULD_INCLUDE_HELPERS.set(project, hasDecorators);

return hasDecorators;
}

1 change: 0 additions & 1 deletion lib/relative-module-paths.js
Original file line number Diff line number Diff line change
@@ -1,4 +1,3 @@
/* eslint-env node */
'use strict';

const path = require('path');
Expand Down
69 changes: 68 additions & 1 deletion node-tests/addon-test.js
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,11 @@ const createBuilder = BroccoliTestHelper.createBuilder;
const createTempDir = BroccoliTestHelper.createTempDir;
const terminateWorkerPool = require('./utils/terminate-workers');

let Addon = CoreObject.extend(AddonMixin);
let Addon = CoreObject.extend(AddonMixin).extend({
_findHost() {
return this.__host;
}
});

describe('ember-cli-babel', function() {

Expand All @@ -23,6 +27,7 @@ describe('ember-cli-babel', function() {
this.ui = new MockUI();
let project = { root: __dirname, emberCLIVersion: () => '2.16.2' };
this.addon = new Addon({
__host: { options: {} },
project,
parent: project,
ui: this.ui,
Expand Down Expand Up @@ -456,6 +461,68 @@ describe('ember-cli-babel', function() {
});
});

describe('_shouldIncludeHelpers()', function() {
it('should return false without any includeExternalHelpers option set', function() {
expect(this.addon._shouldIncludeHelpers()).to.be.false;
});

it('should throw an error with ember-cli-babel.includeExternalHelpers = true in parent', function() {
this.addon.parent.options = { 'ember-cli-babel': { includeExternalHelpers: true } };

expect(this.addon._shouldIncludeHelpers).to.throw;
});

it('should return true with ember-cli-babel.includeExternalHelpers = true in host', function() {
this.addon.__host.options = { 'ember-cli-babel': { includeExternalHelpers: true } };

expect(this.addon._shouldIncludeHelpers()).to.be.true;
});

it('should return false with ember-cli-babel.includeExternalHelpers = false in host', function() {
this.addon.__host.options = { 'ember-cli-babel': { includeExternalHelpers: false } };

expect(this.addon._shouldIncludeHelpers()).to.be.false;
});

describe('autodetection', function() {
let input;

beforeEach(co.wrap(function* () {
input = yield createTempDir();
}));

afterEach(co.wrap(function* () {
yield input.dispose();

// shut down workers after the tests are run so that mocha doesn't hang
yield terminateWorkerPool();
}));

it('should return true if @ember-decorators/babel-transforms exists', function() {
input.write({
node_modules: {
'some-addon': {
node_modules: {
'@ember-decorators': {
'babel-transforms': {
'package.json': JSON.stringify({ name: '@ember-decorators/babel-transforms', version: '1.2.3' }),
'index.js': 'module.exports = {};',
},
},
},
},
},
});

this.addon.project.addons = [{
root: input.path('node_modules/some-addon')
}];

expect(this.addon._shouldIncludeHelpers()).to.be.true;
});
})
});

describe('_shouldCompileModules()', function() {
beforeEach(function() {
this.addon.parent = {
Expand Down
3 changes: 3 additions & 0 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -39,9 +39,12 @@
},
"dependencies": {
"@babel/core": "^7.0.0",
"@babel/helper-module-imports": "^7.0.0",
"@babel/plugin-transform-modules-amd": "^7.0.0",
"@babel/plugin-transform-runtime": "^7.2.0",
"@babel/polyfill": "^7.0.0",
"@babel/preset-env": "^7.0.0",
"@babel/runtime": "^7.2.0",
"amd-name-resolver": "^1.2.1",
"babel-plugin-debug-macros": "^0.2.0-beta.6",
"babel-plugin-ember-modules-api-polyfill": "^2.6.0",
Expand Down
14 changes: 14 additions & 0 deletions tests/unit/external-helpers-test.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,14 @@
import { module, test } from 'qunit';

module('Unit | external helpers test', function() {
test('external helpers work ', function(assert) {
assert.expect(0);

// This test will transpile to use external helpers depending on targets. If
// those helpers are not present, it will break. If IE11 is removed from
// targets we should find another way to test this.
class Foo {}

new Foo();
});
});
Loading

0 comments on commit 9291be1

Please sign in to comment.