require ( 'source-map-support' ) . install ( ) ;
require ( 'console-group' ) . install ( ) ;
const path = require ( 'path' ) ;
const sander = require ( 'sander' ) ;
const assert = require ( 'assert' ) ;
const { exec } = require ( 'child_process' ) ;
const buble = require ( 'buble' ) ;
const acorn = require ( 'acorn' ) ;
const rollup = require ( '../dist/rollup' ) ;
const FUNCTION = path . resolve ( __ dirname , 'function' ) ;
const FORM = path . resolve ( __ dirname , 'form' ) ;
const SOURCEMAPS = path . resolve ( __ dirname , 'sourcemaps' ) ;
const CLI = path . resolve ( __ dirname , 'cli' ) ;
const PROFILES = [
{ format : 'amd' } ,
{ format : 'cjs' } ,
{ format : 'es' } ,
{ format : 'iife' } ,
{ format : 'umd' }
] ;
function extend ( target ) {
[ ] . slice . call ( arguments , 1 ) . forEach ( source => {
source && Object . keys ( source ) . forEach ( key => {
target [ key ] = source [ key ] ;
} ) ;
} ) ;
return target ;
}
function normaliseOutput ( code ) {
return code . toString ( ) . trim ( ) . replace ( /\r\n/g , '\n' ) ;
}
function loadConfig ( path ) {
try {
return require ( path ) ;
} catch ( err ) {
console . error ( err . message ) ;
console . error ( err . stack ) ;
throw new Error ( ` Failed to load ${ path } . An old test perhaps? You should probably delete the directory ` ) ;
}
}
function loader ( modules ) {
return {
resolveId ( id ) {
return id in modules ? id : null ;
} ,
load ( id ) {
return modules [ id ] ;
}
} ;
}
function compareWarnings ( actual , expected ) {
assert . deepEqual (
actual . map ( warning => {
const clone = Object . assign ( { } , warning ) ;
delete clone . toString ;
if ( clone . frame ) {
clone . frame = clone . frame . replace ( /\s+$/gm , '' ) ;
}
return clone ;
} ) ,
expected . map ( warning => {
if ( warning . frame ) {
warning . frame = warning . frame . slice ( 1 ) . replace ( /^\t+/gm , '' ) . replace ( /\s+$/gm , '' ) . trim ( ) ;
}
return warning ;
} )
) ;
}
describe ( 'rollup' , function ( ) {
this . timeout ( 10000 ) ;
describe ( 'sanity checks' , ( ) => {
it ( 'exists' , ( ) => {
assert . ok ( ! ! rollup ) ;
} ) ;
it ( 'has a rollup method' , ( ) => {
assert . equal ( typeof rollup . rollup , 'function' ) ;
} ) ;
it ( 'fails without options' , ( ) => {
return rollup . rollup ( ) . then ( ( ) => {
throw new Error ( 'Missing expected error' ) ;
} , err => {
assert . equal ( 'You must supply options.entry to rollup' , err . message ) ;
} ) ;
} ) ;
it ( 'fails without options.entry' , ( ) => {
return rollup . rollup ( { } ) . then ( ( ) => {
throw new Error ( 'Missing expected error' ) ;
} , err => {
assert . equal ( 'You must supply options.entry to rollup' , err . message ) ;
} ) ;
} ) ;
it ( 'fails with invalid keys' , ( ) => {
return rollup . rollup ( { entry : 'x' , plUgins : [ ] } ) . then ( ( ) => {
throw new Error ( 'Missing expected error' ) ;
} , err => {
assert . equal ( err . message , 'Unexpected key \'plUgins\' found, expected one of: acorn, banner, cache, context, dest, entry, exports, external, footer, format, globals, indent, interop, intro, legacy, moduleContext, moduleId, moduleName, noConflict, onwarn, outro, paths, plugins, preferConst, sourceMap, sourceMapFile, targets, treeshake, useStrict' ) ;
} ) ;
} ) ;
it ( 'treats Literals as leaf nodes, even if first literal encountered is null' , ( ) => {
// this test has to be up here, otherwise the bug doesn't have
// an opportunity to present itself
return rollup . rollup ( {
entry : 'x' ,
plugins : [ loader ( { x : ` var a = null; a = 'a string'; ` } ) ]
} ) ;
} ) ;
it ( 'includes a newline at the end of the bundle' , ( ) => {
return rollup . rollup ( {
entry : 'x' ,
plugins : [ loader ( { x : ` console.log( 42 ); ` } ) ]
} ) . then ( bundle => {
const { code } = bundle . generate ( { format : 'iife' } ) ;
assert . ok ( code [ code . length - 1 ] === '\n' ) ;
} ) ;
} ) ;
it ( 'warns on missing format option' , ( ) => {
const warnings = [ ] ;
return rollup . rollup ( {
entry : 'x' ,
plugins : [ loader ( { x : ` console.log( 42 ); ` } ) ] ,
onwarn : warning => warnings . push ( warning )
} ) . then ( bundle => {
bundle . generate ( ) ;
compareWarnings ( warnings , [
{
code : 'MISSING_FORMAT' ,
message : ` No format option was supplied – defaulting to 'es' ` ,
url : ` https://github.com/rollup/rollup/wiki/JavaScript-API#format `
}
] ) ;
} ) ;
} ) ;
} ) ;
describe ( 'bundle.write()' , ( ) => {
it ( 'fails without options or options.dest' , ( ) => {
return rollup . rollup ( {
entry : 'x' ,
plugins : [ {
resolveId : ( ) => { return 'test' ; } ,
load : ( ) => {
return '// empty' ;
}
} ]
} ) . then ( bundle => {
assert . throws ( ( ) => {
bundle . write ( ) ;
} , /must supply options\.dest/ ) ;
assert . throws ( ( ) => {
bundle . write ( { } ) ;
} , /must supply options\.dest/ ) ;
} ) ;
} ) ;
it ( 'expects options.moduleName for IIFE and UMD bundles' , ( ) => {
return rollup . rollup ( {
entry : 'x' ,
plugins : [ {
resolveId : ( ) => { return 'test' ; } ,
load : ( ) => {
return 'export var foo = 42;' ;
}
} ]
} ) . then ( bundle => {
assert . throws ( ( ) => {
bundle . generate ( {
format : 'umd'
} ) ;
} , /You must supply options\.moduleName for UMD bundles/ ) ;
assert . throws ( ( ) => {
bundle . generate ( {
format : 'iife'
} ) ;
} , /You must supply options\.moduleName for IIFE bundles/ ) ;
} ) ;
} ) ;
it ( 'warns on es6 format' , ( ) => {
let warned ;
return rollup . rollup ( {
entry : 'x' ,
plugins : [ {
resolveId : ( ) => { return 'test' ; } ,
load : ( ) => {
return '// empty' ;
}
} ] ,
onwarn : msg => {
if ( /The es6 format is deprecated/ . test ( msg ) ) warned = true ;
}
} ) . then ( bundle => {
bundle . generate ( { format : 'es6' } ) ;
assert . ok ( warned ) ;
} ) ;
} ) ;
} ) ;
describe ( 'function' , ( ) => {
sander . readdirSync ( FUNCTION ) . sort ( ) . forEach ( dir => {
if ( dir [ 0 ] === '.' ) return ; // .DS_Store...
const config = loadConfig ( FUNCTION + '/' + dir + '/_config.js' ) ;
( config . skip ? it . skip : config . solo ? it . only : it ) ( dir , ( ) => {
process . chdir ( FUNCTION + '/' + dir ) ;
const warnings = [ ] ;
const captureWarning = msg => warnings . push ( msg ) ;
const options = extend ( {
entry : FUNCTION + '/' + dir + '/main.js' ,
onwarn : captureWarning
} , config . options ) ;
if ( config . solo ) console . group ( dir ) ;
return rollup . rollup ( options )
. then ( bundle => {
let unintendedError ;
if ( config . error ) {
throw new Error ( 'Expected an error while rolling up' ) ;
}
let result ;
// try to generate output
try {
result = bundle . generate ( extend ( { } , config . bundleOptions , {
format : 'cjs'
} ) ) ;
if ( config . generateError ) {
unintendedError = new Error ( 'Expected an error while generating output' ) ;
}
} catch ( err ) {
if ( config . generateError ) {
config . generateError ( err ) ;
} else {
unintendedError = err ;
}
}
if ( unintendedError ) throw unintendedError ;
if ( config . error || config . generateError ) return ;
let code = result . code ;
if ( config . buble ) {
code = buble . transform ( code , {
transforms : { modules : false }
} ) . code ;
}
if ( config . code ) config . code ( code ) ;
const module = {
exports : { }
} ;
const context = extend ( { require , module , assert , exports : module . exports } , config . context || { } ) ;
const contextKeys = Object . keys ( context ) ;
const contextValues = contextKeys . map ( key => context [ key ] ) ;
try {
const fn = new Function ( contextKeys , code ) ;
fn . apply ( { } , contextValues ) ;
if ( config . runtimeError ) {
unintendedError = new Error ( 'Expected an error while executing output' ) ;
} else {
if ( config . exports ) config . exports ( module . exports ) ;
if ( config . bundle ) config . bundle ( bundle ) ;
}
} catch ( err ) {
if ( config . runtimeError ) {
config . runtimeError ( err ) ;
} else {
unintendedError = err ;
}
}
if ( config . show || unintendedError ) {
console . log ( result . code + '\n\n\n' ) ;
}
if ( config . warnings ) {
if ( Array . isArray ( config . warnings ) ) {
compareWarnings ( warnings , config . warnings ) ;
} else {
config . warnings ( warnings ) ;
}
} else if ( warnings . length ) {
throw new Error ( ` Got unexpected warnings: \n ${ warnings . join ( '\n' ) } ` ) ;
}
if ( config . solo ) console . groupEnd ( ) ;
if ( unintendedError ) throw unintendedError ;
} , err => {
if ( config . error ) {
config . error ( err ) ;
} else {
throw err ;
}
} ) ;
} ) ;
} ) ;
} ) ;
describe ( 'form' , ( ) => {
sander . readdirSync ( FORM ) . sort ( ) . forEach ( dir => {
if ( dir [ 0 ] === '.' ) return ; // .DS_Store...
const config = loadConfig ( FORM + '/' + dir + '/_config.js' ) ;
if ( config . skipIfWindows && process . platform === 'win32' ) return ;
if ( ! config . options ) {
config . options = { } ;
}
if ( ! ( 'indent' in config . options ) ) {
config . options . indent = true ;
}
const options = extend ( { } , {
entry : FORM + '/' + dir + '/main.js' ,
onwarn : msg => {
if ( /No name was provided for/ . test ( msg ) ) return ;
if ( /as external dependency/ . test ( msg ) ) return ;
console . error ( msg ) ;
}
} , config . options ) ;
( config . skip ? describe . skip : config . solo ? describe . only : describe ) ( dir , ( ) => {
let promise ;
const createBundle = ( ) => ( promise || ( promise = rollup . rollup ( options ) ) ) ;
PROFILES . forEach ( profile => {
it ( 'generates ' + profile . format , ( ) => {
process . chdir ( FORM + '/' + dir ) ;
return createBundle ( ) . then ( bundle => {
const options = extend ( { } , config . options , {
dest : FORM + '/' + dir + '/_actual/' + profile . format + '.js' ,
format : profile . format
} ) ;
return bundle . write ( options ) . then ( ( ) => {
const actualCode = normaliseOutput ( sander . readFileSync ( FORM , dir , '_actual' , profile . format + '.js' ) ) ;
let expectedCode ;
let actualMap ;
let expectedMap ;
try {
expectedCode = normaliseOutput ( sander . readFileSync ( FORM , dir , '_expected' , profile . format + '.js' ) ) ;
} catch ( err ) {
expectedCode = 'missing file' ;
}
try {
actualMap = JSON . parse ( sander . readFileSync ( FORM , dir , '_actual' , profile . format + '.js.map' ) . toString ( ) ) ;
actualMap . sourcesContent = actualMap . sourcesContent . map ( normaliseOutput ) ;
} catch ( err ) {
assert . equal ( err . code , 'ENOENT' ) ;
}
try {
expectedMap = JSON . parse ( sander . readFileSync ( FORM , dir , '_expected' , profile . format + '.js.map' ) . toString ( ) ) ;
expectedMap . sourcesContent = expectedMap . sourcesContent . map ( normaliseOutput ) ;
} catch ( err ) {
assert . equal ( err . code , 'ENOENT' ) ;
}
if ( config . show ) {
console . log ( actualCode + '\n\n\n' ) ;
}
assert . equal ( actualCode , expectedCode ) ;
assert . deepEqual ( actualMap , expectedMap ) ;
} ) ;
} ) ;
} ) ;
} ) ;
} ) ;
} ) ;
} ) ;
describe ( 'sourcemaps' , ( ) => {
sander . readdirSync ( SOURCEMAPS ) . sort ( ) . forEach ( dir => {
if ( dir [ 0 ] === '.' ) return ; // .DS_Store...
describe ( dir , ( ) => {
process . chdir ( SOURCEMAPS + '/' + dir ) ;
const config = loadConfig ( SOURCEMAPS + '/' + dir + '/_config.js' ) ;
const entry = path . resolve ( SOURCEMAPS , dir , 'main.js' ) ;
const dest = path . resolve ( SOURCEMAPS , dir , '_actual/bundle' ) ;
let warnings ;
const options = extend ( { } , config . options , {
entry ,
onwarn : warning => warnings . push ( warning )
} ) ;
PROFILES . forEach ( profile => {
( config . skip ? it . skip : config . solo ? it . only : it ) ( 'generates ' + profile . format , ( ) => {
process . chdir ( SOURCEMAPS + '/' + dir ) ;
warnings = [ ] ;
return rollup . rollup ( options ) . then ( bundle => {
const options = extend ( { } , {
format : profile . format ,
sourceMap : true ,
dest : ` ${ dest } . ${ profile . format } .js `
} , config . options ) ;
bundle . write ( options ) ;
if ( config . test ) {
const { code , map } = bundle . generate ( options ) ;
config . test ( code , map ) ;
}
if ( config . warnings ) {
compareWarnings ( warnings , config . warnings ) ;
} else if ( warnings . length ) {
throw new Error ( ` Unexpected warnings ` ) ;
}
} ) ;
} ) ;
} ) ;
} ) ;
} ) ;
} ) ;
describe ( 'cli' , ( ) => {
sander . readdirSync ( CLI ) . sort ( ) . forEach ( dir => {
if ( dir [ 0 ] === '.' ) return ; // .DS_Store...
describe ( dir , ( ) => {
const config = loadConfig ( CLI + '/' + dir + '/_config.js' ) ;
( config . skip ? it . skip : config . solo ? it . only : it ) ( dir , done => {
process . chdir ( config . cwd || path . resolve ( CLI , dir ) ) ;
const command = 'node ' + path . resolve ( __ dirname , '../bin' ) + path . sep + config . command ;
exec ( command , { } , ( err , code , stderr ) => {
if ( err ) {
if ( config . error ) {
config . error ( err ) ;
return done ( ) ;
} else {
throw err ;
}
}
if ( stderr ) console . error ( stderr ) ;
let unintendedError ;
if ( config . execute ) {
try {
if ( config . buble ) {
code = buble . transform ( code , {
transforms : { modules : false }
} ) . code ;
}
const fn = new Function ( 'require' , 'module' , 'exports' , 'assert' , code ) ;
const module = {
exports : { }
} ;
fn ( require , module , module . exports , assert ) ;
if ( config . error ) {
unintendedError = new Error ( 'Expected an error while executing output' ) ;
}
if ( config . exports ) {
config . exports ( module . exports ) ;
}
} catch ( err ) {
if ( config . error ) {
config . error ( err ) ;
} else {
unintendedError = err ;
}
}
if ( config . show || unintendedError ) {
console . log ( code + '\n\n\n' ) ;
}
if ( config . solo ) console . groupEnd ( ) ;
unintendedError ? done ( unintendedError ) : done ( ) ;
}
else if ( config . result ) {
try {
config . result ( code ) ;
done ( ) ;
} catch ( err ) {
done ( err ) ;
}
}
else if ( sander . existsSync ( '_expected' ) && sander . statSync ( '_expected' ) . isDirectory ( ) ) {
let error = null ;
sander . readdirSync ( '_expected' ) . forEach ( child => {
const expected = sander . readFileSync ( path . join ( '_expected' , child ) ) . toString ( ) ;
const actual = sander . readFileSync ( path . join ( '_actual' , child ) ) . toString ( ) ;
try {
assert . equal ( normaliseOutput ( actual ) , normaliseOutput ( expected ) ) ;
} catch ( err ) {
error = err ;
}
} ) ;
done ( error ) ;
}
else {
const expected = sander . readFileSync ( '_expected.js' ) . toString ( ) ;
try {
assert . equal ( normaliseOutput ( code ) , normaliseOutput ( expected ) ) ;
done ( ) ;
} catch ( err ) {
done ( err ) ;
}
}
} ) ;
} ) ;
} ) ;
} ) ;
} ) ;
describe ( 'incremental' , ( ) => {
function executeBundle ( bundle ) {
const cjs = bundle . generate ( { format : 'cjs' } ) ;
const m = new Function ( 'module' , 'exports' , cjs . code ) ;
const module = { exports : { } } ;
m ( module , module . exports ) ;
return module . exports ;
}
let resolveIdCalls ;
let transformCalls ;
let modules ;
const plugin = {
resolveId : id => {
resolveIdCalls += 1 ;
return id ;
} ,
load : id => {
return modules [ id ] ;
} ,
transform : code => {
transformCalls += 1 ;
return code ;
}
} ;
beforeEach ( ( ) => {
resolveIdCalls = 0 ;
transformCalls = 0 ;
modules = {
entry : ` import foo from 'foo'; export default foo; ` ,
foo : ` export default 42 ` ,
bar : ` export default 21 `
} ;
} ) ;
it ( 'does not resolves id and transforms in the second time' , ( ) => {
return rollup . rollup ( {
entry : 'entry' ,
plugins : [ plugin ]
} ) . then ( bundle => {
assert . equal ( resolveIdCalls , 2 ) ;
assert . equal ( transformCalls , 2 ) ;
return rollup . rollup ( {
entry : 'entry' ,
plugins : [ plugin ] ,
cache : bundle
} ) ;
} ) . then ( bundle => {
assert . equal ( resolveIdCalls , 3 ) ; // +1 for entry point which is resolved every time
assert . equal ( transformCalls , 2 ) ;
assert . equal ( executeBundle ( bundle ) , 42 ) ;
} ) ;
} ) ;
it ( 'transforms modified sources' , ( ) => {
let cache ;
return rollup . rollup ( {
entry : 'entry' ,
plugins : [ plugin ]
} ) . then ( bundle => {
assert . equal ( transformCalls , 2 ) ;
assert . equal ( executeBundle ( bundle ) , 42 ) ;
modules . foo = ` export default 43 ` ;
cache = bundle ;
} ) . then ( ( ) => {
return rollup . rollup ( {
entry : 'entry' ,
plugins : [ plugin ] ,
cache
} ) ;
} ) . then ( bundle => {
assert . equal ( transformCalls , 3 ) ;
assert . equal ( executeBundle ( bundle ) , 43 ) ;
} ) ;
} ) ;
it ( 'resolves id of new imports' , ( ) => {
let cache ;
return rollup . rollup ( {
entry : 'entry' ,
plugins : [ plugin ]
} ) . then ( bundle => {
assert . equal ( resolveIdCalls , 2 ) ;
assert . equal ( executeBundle ( bundle ) , 42 ) ;
modules . entry = ` import bar from 'bar'; export default bar; ` ;
cache = bundle ;
} ) . then ( ( ) => {
return rollup . rollup ( {
entry : 'entry' ,
plugins : [ plugin ] ,
cache
} ) ;
} ) . then ( bundle => {
assert . equal ( resolveIdCalls , 4 ) ;
assert . equal ( executeBundle ( bundle ) , 21 ) ;
} ) ;
} ) ;
it ( 'keeps ASTs between runs' , ( ) => {
return rollup . rollup ( {
entry : 'entry' ,
plugins : [ plugin ]
} ) . then ( bundle => {
const asts = { } ;
bundle . modules . forEach ( module => {
asts [ module . id ] = module . ast ;
} ) ;
assert . deepEqual ( asts . entry , acorn . parse ( modules . entry , { sourceType : 'module' } ) ) ;
assert . deepEqual ( asts . foo , acorn . parse ( modules . foo , { sourceType : 'module' } ) ) ;
} ) ;
} ) ;
it ( 'recovers from errors' , ( ) => {
modules . entry = ` import foo from 'foo'; import bar from 'bar'; export default foo + bar; ` ;
return rollup . rollup ( {
entry : 'entry' ,
plugins : [ plugin ]
} ) . then ( cache => {
modules . foo = ` var 42 = nope; ` ;
return rollup . rollup ( {
entry : 'entry' ,
plugins : [ plugin ] ,
cache
} ) . catch ( err => {
return cache ;
} ) ;
} ) . then ( cache => {
modules . foo = ` export default 42; ` ;
return rollup . rollup ( {
entry : 'entry' ,
plugins : [ plugin ] ,
cache
} ) . then ( bundle => {
assert . equal ( executeBundle ( bundle ) , 63 ) ;
} ) ;
} ) ;
} ) ;
} ) ;
describe ( 'hooks' , ( ) => {
it ( 'passes bundle & output object to ongenerate & onwrite hooks' , ( ) => {
const dest = path . join ( __ dirname , 'tmp/bundle.js' ) ;
return rollup . rollup ( {
entry : 'entry' ,
plugins : [
loader ( { entry : ` alert('hello') ` } ) ,
{
ongenerate ( bundle , out ) {
out . ongenerate = true ;
} ,
onwrite ( bundle , out ) {
assert . equal ( out . ongenerate , true ) ;
}
}
]
} ) . then ( bundle => {
return bundle . write ( {
dest
} ) ;
} ) . then ( ( ) => {
return sander . unlink ( dest ) ;
} ) ;
} ) ;
it ( 'calls ongenerate hooks in sequence' , ( ) => {
const result = [ ] ;
return rollup . rollup ( {
entry : 'entry' ,
plugins : [
loader ( { entry : ` alert('hello') ` } ) ,
{
ongenerate ( info ) {
result . push ( { a : info . format } ) ;
}
} ,
{
ongenerate ( info ) {
result . push ( { b : info . format } ) ;
}
}
]
} ) . then ( bundle => {
bundle . generate ( { format : 'cjs' } ) ;
assert . deepEqual ( result , [
{ a : 'cjs' } ,
{ b : 'cjs' }
] ) ;
} ) ;
} ) ;
it ( 'calls onwrite hooks in sequence' , ( ) => {
const result = [ ] ;
const dest = path . join ( __ dirname , 'tmp/bundle.js' ) ;
return rollup . rollup ( {
entry : 'entry' ,
plugins : [
loader ( { entry : ` alert('hello') ` } ) ,
{
onwrite ( info ) {
return new Promise ( ( fulfil ) => {
result . push ( { a : info . dest , format : info . format } ) ;
fulfil ( ) ;
} ) ;
}
} ,
{
onwrite ( info ) {
result . push ( { b : info . dest , format : info . format } ) ;
}
}
]
} ) . then ( bundle => {
return bundle . write ( {
dest ,
format : 'cjs'
} ) ;
} ) . then ( ( ) => {
assert . deepEqual ( result , [
{ a : dest , format : 'cjs' } ,
{ b : dest , format : 'cjs' }
] ) ;
return sander . unlink ( dest ) ;
} ) ;
} ) ;
} ) ;
describe ( 'misc' , ( ) => {
it ( 'warns if node builtins are unresolved in a non-CJS, non-ES bundle (#1051)' , ( ) => {
const warnings = [ ] ;
return rollup . rollup ( {
entry : 'entry' ,
plugins : [
loader ( { entry : ` import { format } from 'util'; \n export default format( 'this is a %s', 'formatted string' ); ` } )
] ,
onwarn : warning => warnings . push ( warning )
} ) . then ( bundle => {
bundle . generate ( {
format : 'iife' ,
moduleName : 'myBundle'
} ) ;
const relevantWarnings = warnings . filter ( warning => warning . code === 'MISSING_NODE_BUILTINS' ) ;
assert . equal ( relevantWarnings . length , 1 ) ;
assert . equal ( relevantWarnings [ 0 ] . message , ` Creating a browser bundle that depends on Node.js built-in module ('util'). You might need to include https://www.npmjs.com/package/rollup-plugin-node-builtins ` ) ;
} ) ;
} ) ;
} ) ;
} ) ;