Browse Source

Merge pull request #472 from rollup/export-global

Support export { global }
gh-669
Rich Harris 9 years ago
parent
commit
51a87a85c9
  1. 9
      src/Bundle.js
  2. 56
      src/Declaration.js
  3. 14
      src/Module.js
  4. 6
      src/utils/object.js
  5. 9
      test/function/export-global/_config.js
  6. 1
      test/function/export-global/main.js

9
src/Bundle.js

@ -1,7 +1,7 @@
import Promise from 'es6-promise/lib/es6-promise/promise.js'; import Promise from 'es6-promise/lib/es6-promise/promise.js';
import MagicString from 'magic-string'; import MagicString from 'magic-string';
import first from './utils/first.js'; import first from './utils/first.js';
import { blank, keys } from './utils/object.js'; import { blank, forOwn, keys } from './utils/object.js';
import Module from './Module.js'; import Module from './Module.js';
import ExternalModule from './ExternalModule.js'; import ExternalModule from './ExternalModule.js';
import finalisers from './finalisers/index.js'; import finalisers from './finalisers/index.js';
@ -130,15 +130,14 @@ export default class Bundle {
// ensure we don't shadow named external imports, if // ensure we don't shadow named external imports, if
// we're creating an ES6 bundle // we're creating an ES6 bundle
keys( module.declarations ).forEach( name => { forOwn( module.declarations, ( declaration, name ) => {
const declaration = module.declarations[ name ];
declaration.setSafeName( getSafeName( name ) ); declaration.setSafeName( getSafeName( name ) );
}); });
}); });
this.modules.forEach( module => { this.modules.forEach( module => {
keys( module.declarations ).forEach( originalName => { forOwn( module.declarations, ( declaration, originalName ) => {
const declaration = module.declarations[ originalName ]; if ( declaration.isGlobal ) return;
if ( originalName === 'default' ) { if ( originalName === 'default' ) {
if ( declaration.original && !declaration.original.isReassigned ) return; if ( declaration.original && !declaration.original.isReassigned ) return;

56
src/Declaration.js

@ -1,7 +1,9 @@
import { blank, keys } from './utils/object.js'; import { blank, forOwn, keys } from './utils/object.js';
import run from './utils/run.js'; import run from './utils/run.js';
import { SyntheticReference } from './Reference.js'; import { SyntheticReference } from './Reference.js';
const use = alias => alias.use();
export default class Declaration { export default class Declaration {
constructor ( node, isParam, statement ) { constructor ( node, isParam, statement ) {
if ( node ) { if ( node ) {
@ -68,7 +70,7 @@ export default class Declaration {
this.isUsed = true; this.isUsed = true;
if ( this.statement ) this.statement.mark(); if ( this.statement ) this.statement.mark();
this.aliases.forEach( alias => alias.use() ); this.aliases.forEach( use );
} }
} }
@ -129,7 +131,44 @@ export class SyntheticDefaultDeclaration {
if ( this.original ) this.original.use(); if ( this.original ) this.original.use();
this.aliases.forEach( alias => alias.use() ); this.aliases.forEach( use );
}
}
export class SyntheticGlobalDeclaration {
constructor ( name ) {
this.name = name;
this.isExternal = true;
this.isGlobal = true;
this.isReassigned = false;
this.aliases = [];
this.isUsed = false;
}
addAlias ( declaration ) {
this.aliases.push( declaration );
}
addReference ( reference ) {
reference.declaration = this;
if ( reference.isReassignment ) this.isReassigned = true;
}
render () {
return this.name;
}
run () {
return true;
}
use () {
if ( this.isUsed ) return;
this.isUsed = true;
this.aliases.forEach( use );
} }
} }
@ -182,8 +221,8 @@ export class SyntheticNamespaceDeclaration {
// add synthetic references, in case of chained // add synthetic references, in case of chained
// namespace imports // namespace imports
keys( this.originals ).forEach( name => { forOwn( this.originals, ( original, name ) => {
this.originals[ name ].addReference( new SyntheticReference( name ) ); original.addReference( new SyntheticReference( name ) );
}); });
} }
@ -210,11 +249,8 @@ export class SyntheticNamespaceDeclaration {
} }
use () { use () {
keys( this.originals ).forEach( name => { forOwn( this.originals, use );
this.originals[ name ].use(); this.aliases.forEach( use );
});
this.aliases.forEach( alias => alias.use() );
} }
} }

14
src/Module.js

@ -7,7 +7,11 @@ import { basename, extname } from './utils/path.js';
import getLocation from './utils/getLocation.js'; import getLocation from './utils/getLocation.js';
import makeLegalIdentifier from './utils/makeLegalIdentifier.js'; import makeLegalIdentifier from './utils/makeLegalIdentifier.js';
import SOURCEMAPPING_URL from './utils/sourceMappingURL.js'; import SOURCEMAPPING_URL from './utils/sourceMappingURL.js';
import { SyntheticDefaultDeclaration, SyntheticNamespaceDeclaration } from './Declaration.js'; import {
SyntheticDefaultDeclaration,
SyntheticGlobalDeclaration,
SyntheticNamespaceDeclaration
} from './Declaration.js';
import { isFalsy, isTruthy } from './ast/conditions.js'; import { isFalsy, isTruthy } from './ast/conditions.js';
import { emptyBlockStatement } from './ast/create.js'; import { emptyBlockStatement } from './ast/create.js';
import extractNames from './ast/extractNames.js'; import extractNames from './ast/extractNames.js';
@ -633,7 +637,13 @@ export default class Module {
const exportDeclaration = this.exports[ name ]; const exportDeclaration = this.exports[ name ];
if ( exportDeclaration ) { if ( exportDeclaration ) {
return this.trace( exportDeclaration.localName ); const name = exportDeclaration.localName;
const declaration = this.trace( name );
if ( declaration ) return declaration;
this.bundle.assumedGlobals[ name ] = true;
return ( this.declarations[ name ] = new SyntheticGlobalDeclaration( name ) );
} }
for ( let i = 0; i < this.exportAllModules.length; i += 1 ) { for ( let i = 0; i < this.exportAllModules.length; i += 1 ) {

6
src/utils/object.js

@ -1,5 +1,9 @@
export const keys = Object.keys; export const { keys } = Object;
export function blank () { export function blank () {
return Object.create( null ); return Object.create( null );
} }
export function forOwn ( object, func ) {
Object.keys( object ).forEach( key => func( object[ key ], key ) );
}

9
test/function/export-global/_config.js

@ -0,0 +1,9 @@
var assert = require( 'assert' );
module.exports = {
description: 'any global variables in scope can be re-exported',
exports: function ( exports ) {
assert.equal( exports.Buffer, Buffer );
}
};

1
test/function/export-global/main.js

@ -0,0 +1 @@
export { Buffer };
Loading…
Cancel
Save