Browse Source

omit globals for empty imports (#1217)

gh-1187
Rich-Harris 8 years ago
parent
commit
502e58d0b0
  1. 20
      src/finalisers/iife.js
  2. 16
      src/finalisers/shared/getGlobalNameMaker.js
  3. 18
      src/finalisers/shared/propertyStringFor.js
  4. 11
      src/finalisers/shared/sanitize.js
  5. 12
      src/finalisers/shared/trimEmptyImports.js
  6. 23
      src/finalisers/umd.js
  7. 11
      test/form/external-empty-import-no-global-b/_config.js
  8. 9
      test/form/external-empty-import-no-global-b/_expected/amd.js
  9. 10
      test/form/external-empty-import-no-global-b/_expected/cjs.js
  10. 8
      test/form/external-empty-import-no-global-b/_expected/es.js
  11. 10
      test/form/external-empty-import-no-global-b/_expected/iife.js
  12. 13
      test/form/external-empty-import-no-global-b/_expected/umd.js
  13. 6
      test/form/external-empty-import-no-global-b/main.js
  14. 10
      test/form/external-empty-import-no-global/_config.js
  15. 7
      test/form/external-empty-import-no-global/_expected/amd.js
  16. 7
      test/form/external-empty-import-no-global/_expected/cjs.js
  17. 5
      test/form/external-empty-import-no-global/_expected/es.js
  18. 8
      test/form/external-empty-import-no-global/_expected/iife.js
  19. 11
      test/form/external-empty-import-no-global/_expected/umd.js
  20. 3
      test/form/external-empty-import-no-global/main.js

20
src/finalisers/iife.js

@ -4,14 +4,9 @@ import error from '../utils/error.js';
import getInteropBlock from './shared/getInteropBlock.js'; import getInteropBlock from './shared/getInteropBlock.js';
import getExportBlock from './shared/getExportBlock.js'; import getExportBlock from './shared/getExportBlock.js';
import getGlobalNameMaker from './shared/getGlobalNameMaker.js'; import getGlobalNameMaker from './shared/getGlobalNameMaker.js';
import propertyStringFor from './shared/propertyStringFor'; import { property, keypath } from './shared/sanitize.js';
import warnOnBuiltins from './shared/warnOnBuiltins.js'; import warnOnBuiltins from './shared/warnOnBuiltins.js';
import trimEmptyImports from './shared/trimEmptyImports.js';
// thisProp('foo.bar-baz.qux') === "this.foo['bar-baz'].qux"
const thisProp = propertyStringFor('this');
// propString('foo.bar-baz.qux') === ".foo['bar-baz'].qux"
const propString = propertyStringFor('');
function setupNamespace ( keypath ) { function setupNamespace ( keypath ) {
const parts = keypath.split( '.' ); const parts = keypath.split( '.' );
@ -21,20 +16,23 @@ function setupNamespace ( keypath ) {
let acc = 'this'; let acc = 'this';
return parts return parts
.map( part => ( acc += propString(part), `${acc} = ${acc} || {};` ) ) .map( part => ( acc += property( part ), `${acc} = ${acc} || {};` ) )
.join( '\n' ) + '\n'; .join( '\n' ) + '\n';
} }
const thisProp = name => `this${keypath( name )}`;
export default function iife ( bundle, magicString, { exportMode, indentString, intro, outro }, options ) { export default function iife ( bundle, magicString, { exportMode, indentString, intro, outro }, options ) {
const globalNameMaker = getGlobalNameMaker( options.globals || blank(), bundle ); const globalNameMaker = getGlobalNameMaker( options.globals || blank(), bundle, 'null' );
const name = options.moduleName; const name = options.moduleName;
const isNamespaced = name && ~name.indexOf( '.' ); const isNamespaced = name && ~name.indexOf( '.' );
warnOnBuiltins( bundle ); warnOnBuiltins( bundle );
const dependencies = bundle.externalModules.map( globalNameMaker ); const external = trimEmptyImports( bundle.externalModules );
const args = bundle.externalModules.map( getName ); const dependencies = external.map( globalNameMaker );
const args = external.map( getName );
if ( exportMode !== 'none' && !name ) { if ( exportMode !== 'none' && !name ) {
error({ error({

16
src/finalisers/shared/getGlobalNameMaker.js

@ -1,15 +1,19 @@
export default function getGlobalNameMaker ( globals, bundle ) { export default function getGlobalNameMaker ( globals, bundle, fallback = null ) {
const fn = typeof globals === 'function' ? globals : id => globals[ id ]; const fn = typeof globals === 'function' ? globals : id => globals[ id ];
return function ( module ) { return function ( module ) {
const name = fn( module.id ); const name = fn( module.id );
if ( name ) return name; if ( name ) return name;
bundle.warn({ if ( Object.keys( module.declarations ).length > 0 ) {
code: 'MISSING_GLOBAL_NAME', bundle.warn({
message: `No name was provided for external module '${module.id}' in options.globals – guessing '${module.name}'` code: 'MISSING_GLOBAL_NAME',
}); message: `No name was provided for external module '${module.id}' in options.globals – guessing '${module.name}'`
});
return module.name; return module.name;
}
return fallback;
}; };
} }

18
src/finalisers/shared/propertyStringFor.js

@ -1,18 +0,0 @@
// Generate strings which dereference dotted properties, but use array notation `['prop-deref']`
// if the property name isn't trivial
const shouldUseDot = /^[a-zA-Z$_][a-zA-Z0-9$_]*$/;
const dereferenceString = prop =>
prop.match(shouldUseDot) ? `.${prop}` : `['${prop}']`;
/**
* returns a function which generates property dereference strings for the given name
*
* const getGlobalProp = propertyStringFor('global');
* getGlobalProp('foo.bar-baz.qux') => `global.bar['bar-baz'].qux`
*/
const propertyStringFor = objName => propName =>
objName + propName.split('.').map(dereferenceString).join('');
export default propertyStringFor;

11
src/finalisers/shared/sanitize.js

@ -0,0 +1,11 @@
// Generate strings which dereference dotted properties, but use array notation `['prop-deref']`
// if the property name isn't trivial
const shouldUseDot = /^[a-zA-Z$_][a-zA-Z0-9$_]*$/;
export function property ( prop ) {
return shouldUseDot.test( prop ) ? `.${prop}` : `['${prop}']`;
}
export function keypath ( keypath ) {
return keypath.split( '.' ).map( property ).join( '' );
}

12
src/finalisers/shared/trimEmptyImports.js

@ -0,0 +1,12 @@
export default function trimEmptyImports ( modules ) {
let i = modules.length;
while ( i-- ) {
const module = modules[i];
if ( Object.keys( module.declarations ).length > 0 ) {
return modules.slice( 0, i + 1 );
}
}
return [];
}

23
src/finalisers/umd.js

@ -5,23 +5,23 @@ import getInteropBlock from './shared/getInteropBlock.js';
import getExportBlock from './shared/getExportBlock.js'; import getExportBlock from './shared/getExportBlock.js';
import getGlobalNameMaker from './shared/getGlobalNameMaker.js'; import getGlobalNameMaker from './shared/getGlobalNameMaker.js';
import esModuleExport from './shared/esModuleExport.js'; import esModuleExport from './shared/esModuleExport.js';
import propertyStringFor from './shared/propertyStringFor.js'; import { property, keypath } from './shared/sanitize.js';
import warnOnBuiltins from './shared/warnOnBuiltins.js'; import warnOnBuiltins from './shared/warnOnBuiltins.js';
import trimEmptyImports from './shared/trimEmptyImports.js';
// globalProp('foo.bar-baz') === "global.foo['bar-baz']" function globalProp ( name ) {
const globalProp = propertyStringFor('global'); if ( !name ) return 'null';
return `global${ keypath( name ) }`;
// propString('foo.bar-baz') === ".foo['bar']" }
const propString = propertyStringFor('');
function setupNamespace ( name ) { function setupNamespace ( name ) {
const parts = name.split( '.' ); const parts = name.split( '.' );
parts.pop(); const last = property( parts.pop() );
let acc = 'global'; let acc = 'global';
return parts return parts
.map( part => ( acc += propString(part), `${acc} = ${acc} || {}` ) ) .map( part => ( acc += property( part ), `${acc} = ${acc} || {}` ) )
.concat( globalProp(name) ) .concat( `${acc}${last}` )
.join( ', ' ); .join( ', ' );
} }
@ -41,9 +41,10 @@ export default function umd ( bundle, magicString, { exportMode, indentString, i
const amdDeps = bundle.externalModules.map( quotePath ); const amdDeps = bundle.externalModules.map( quotePath );
const cjsDeps = bundle.externalModules.map( req ); const cjsDeps = bundle.externalModules.map( req );
const globalDeps = bundle.externalModules.map( module => globalProp(globalNameMaker( module )) );
const args = bundle.externalModules.map( getName ); const trimmed = trimEmptyImports( bundle.externalModules );
const globalDeps = trimmed.map( module => globalProp( globalNameMaker( module ) ) );
const args = trimmed.map( getName );
if ( exportMode === 'named' ) { if ( exportMode === 'named' ) {
amdDeps.unshift( `'exports'` ); amdDeps.unshift( `'exports'` );

11
test/form/external-empty-import-no-global-b/_config.js

@ -0,0 +1,11 @@
module.exports = {
description: 'does not expect a global to be provided for empty imports (#1217)',
options: {
external: [ 'babel-polyfill', 'other' ],
moduleName: 'myBundle',
globals: { other: 'other' },
onwarn ( warning ) {
throw new Error( warning.message );
}
}
};

9
test/form/external-empty-import-no-global-b/_expected/amd.js

@ -0,0 +1,9 @@
define(['babel-polyfill', 'other'], function (babelPolyfill, other) { 'use strict';
other.x();
var main = new WeakMap();
return main;
});

10
test/form/external-empty-import-no-global-b/_expected/cjs.js

@ -0,0 +1,10 @@
'use strict';
require('babel-polyfill');
var other = require('other');
other.x();
var main = new WeakMap();
module.exports = main;

8
test/form/external-empty-import-no-global-b/_expected/es.js

@ -0,0 +1,8 @@
import 'babel-polyfill';
import { x } from 'other';
x();
var main = new WeakMap();
export default main;

10
test/form/external-empty-import-no-global-b/_expected/iife.js

@ -0,0 +1,10 @@
var myBundle = (function (babelPolyfill,other) {
'use strict';
other.x();
var main = new WeakMap();
return main;
}(null,other));

13
test/form/external-empty-import-no-global-b/_expected/umd.js

@ -0,0 +1,13 @@
(function (global, factory) {
typeof exports === 'object' && typeof module !== 'undefined' ? module.exports = factory(require('babel-polyfill'), require('other')) :
typeof define === 'function' && define.amd ? define(['babel-polyfill', 'other'], factory) :
(global.myBundle = factory(null,global.other));
}(this, (function (babelPolyfill,other) { 'use strict';
other.x();
var main = new WeakMap();
return main;
})));

6
test/form/external-empty-import-no-global-b/main.js

@ -0,0 +1,6 @@
import 'babel-polyfill';
import { x } from 'other';
x();
export default new WeakMap();

10
test/form/external-empty-import-no-global/_config.js

@ -0,0 +1,10 @@
module.exports = {
description: 'does not expect a global to be provided for empty imports (#1217)',
options: {
external: [ 'babel-polyfill' ],
moduleName: 'myBundle',
onwarn ( warning ) {
throw new Error( warning.message );
}
}
};

7
test/form/external-empty-import-no-global/_expected/amd.js

@ -0,0 +1,7 @@
define(['babel-polyfill'], function (babelPolyfill) { 'use strict';
var main = new WeakMap();
return main;
});

7
test/form/external-empty-import-no-global/_expected/cjs.js

@ -0,0 +1,7 @@
'use strict';
require('babel-polyfill');
var main = new WeakMap();
module.exports = main;

5
test/form/external-empty-import-no-global/_expected/es.js

@ -0,0 +1,5 @@
import 'babel-polyfill';
var main = new WeakMap();
export default main;

8
test/form/external-empty-import-no-global/_expected/iife.js

@ -0,0 +1,8 @@
var myBundle = (function () {
'use strict';
var main = new WeakMap();
return main;
}());

11
test/form/external-empty-import-no-global/_expected/umd.js

@ -0,0 +1,11 @@
(function (global, factory) {
typeof exports === 'object' && typeof module !== 'undefined' ? module.exports = factory(require('babel-polyfill')) :
typeof define === 'function' && define.amd ? define(['babel-polyfill'], factory) :
(global.myBundle = factory());
}(this, (function () { 'use strict';
var main = new WeakMap();
return main;
})));

3
test/form/external-empty-import-no-global/main.js

@ -0,0 +1,3 @@
import 'babel-polyfill';
export default new WeakMap();
Loading…
Cancel
Save