Browse Source

Merge pull request #27 from rollup/gh-26

fix #26
contingency-plan
Rich Harris 10 years ago
parent
commit
f2324de2fe
  1. 45
      src/Module.js
  2. 13
      test/function/statement-order/_config.js
  3. 17
      test/function/statement-order/answer.js
  4. 3
      test/function/statement-order/main.js

45
src/Module.js

@ -28,12 +28,19 @@ export default class Module {
// Try to extract a list of top-level statements/declarations. If // Try to extract a list of top-level statements/declarations. If
// the parse fails, attach file info and abort // the parse fails, attach file info and abort
let ast;
try { try {
const ast = parse( source, { ast = parse( source, {
ecmaVersion: 6, ecmaVersion: 6,
sourceType: 'module', sourceType: 'module',
onComment: ( block, text, start, end ) => this.comments.push({ block, text, start, end }) onComment: ( block, text, start, end ) => this.comments.push({ block, text, start, end })
}); });
} catch ( err ) {
err.code = 'PARSE_ERROR';
err.file = path;
throw err;
}
walk( ast, { walk( ast, {
enter: node => { enter: node => {
@ -42,16 +49,38 @@ export default class Module {
} }
}); });
this.statements = ast.body.map( ( node, i ) => { this.statements = [];
const magicString = this.magicString.snip( node.start, node.end ).trim();
return new Statement( node, magicString, this, i ); ast.body.map( node => {
// special case - top-level var declarations with multiple declarators
// should be split up. Otherwise, we may end up including code we
// don't need, just because an unwanted declarator is included
if ( node.type === 'VariableDeclaration' && node.declarations.length > 1 ) {
node.declarations.forEach( declarator => {
const magicString = this.magicString.snip( declarator.start, declarator.end ).trim();
magicString.prepend( `${node.kind} ` ).append( ';' );
const syntheticNode = {
type: 'VariableDeclaration',
kind: node.kind,
start: node.start,
end: node.end,
declarations: [ declarator ]
};
const statement = new Statement( syntheticNode, magicString, this, this.statements.length );
this.statements.push( statement );
}); });
} catch ( err ) {
err.code = 'PARSE_ERROR';
err.file = path;
throw err;
} }
else {
const magicString = this.magicString.snip( node.start, node.end ).trim();
const statement = new Statement( node, magicString, this, this.statements.length );
this.statements.push( statement );
}
});
this.importDeclarations = this.statements.filter( isImportDeclaration ); this.importDeclarations = this.statements.filter( isImportDeclaration );
this.exportDeclarations = this.statements.filter( isExportDeclaration ); this.exportDeclarations = this.statements.filter( isExportDeclaration );

13
test/function/statement-order/_config.js

@ -0,0 +1,13 @@
var assert = require( 'assert' );
module.exports = {
description: 'correct statement order is preserved even in weird edge cases',
context: {
getAnswer: function ( obj ) {
return obj.answer;
}
},
exports: function ( exports ) {
assert.equal( exports, 'right' );
}
};

17
test/function/statement-order/answer.js

@ -0,0 +1,17 @@
var prop,
answer;
var foo = { answer: 'wrong' };
var bar = { answer: 'right' };
if ( typeof bar === "object" ) {
for ( prop in bar ) {
if ( bar.hasOwnProperty(prop) ) {
foo[prop] = bar[prop];
}
}
}
answer = getAnswer( foo );
export default answer;

3
test/function/statement-order/main.js

@ -0,0 +1,3 @@
import answer from './answer';
var answer2 = answer;
export default answer2;
Loading…
Cancel
Save