// @ts-check import path from 'path' import ts from 'rollup-plugin-typescript2' import replace from '@rollup/plugin-replace' import json from '@rollup/plugin-json' if (!process.env.TARGET) { throw new Error('TARGET package must be specified via --environment flag.') } const masterVersion = require('./package.json').version const packagesDir = path.resolve(__dirname, 'packages') const packageDir = path.resolve(packagesDir, process.env.TARGET) const resolve = p => path.resolve(packageDir, p) const pkg = require(resolve(`package.json`)) const packageOptions = pkg.buildOptions || {} const name = packageOptions.filename || path.basename(packageDir) // ensure TS checks only once for each build let hasTSChecked = false const outputConfigs = { 'esm-bundler': { file: resolve(`dist/${name}.esm-bundler.js`), format: `es` }, 'esm-browser': { file: resolve(`dist/${name}.esm-browser.js`), format: `es` }, cjs: { file: resolve(`dist/${name}.cjs.js`), format: `cjs` }, global: { file: resolve(`dist/${name}.global.js`), format: `iife` }, // runtime-only builds, for main "vue" package only 'esm-bundler-runtime': { file: resolve(`dist/${name}.runtime.esm-bundler.js`), format: `es` }, 'esm-browser-runtime': { file: resolve(`dist/${name}.runtime.esm-browser.js`), format: 'es' }, 'global-runtime': { file: resolve(`dist/${name}.runtime.global.js`), format: 'iife' } } const defaultFormats = ['esm-bundler', 'cjs'] const inlineFormats = process.env.FORMATS && process.env.FORMATS.split(',') const packageFormats = inlineFormats || packageOptions.formats || defaultFormats const packageConfigs = process.env.PROD_ONLY ? [] : packageFormats.map(format => createConfig(format, outputConfigs[format])) if (process.env.NODE_ENV === 'production') { packageFormats.forEach(format => { if (packageOptions.prod === false) { return } if (format === 'cjs') { packageConfigs.push(createProductionConfig(format)) } if (/^(global|esm-browser)(-runtime)?/.test(format)) { packageConfigs.push(createMinifiedConfig(format)) } }) } export default packageConfigs function createConfig(format, output, plugins = []) { if (!output) { console.log(require('chalk').yellow(`invalid format: "${format}"`)) process.exit(1) } const isProductionBuild = process.env.__DEV__ === 'false' || /\.prod\.js$/.test(output.file) const isBundlerESMBuild = /esm-bundler/.test(format) const isBrowserESMBuild = /esm-browser/.test(format) const isNodeBuild = format === 'cjs' const isGlobalBuild = /global/.test(format) const isCompatPackage = pkg.name === '@vue/compat' const isCompatBuild = !!packageOptions.compat output.exports = isCompatPackage ? 'auto' : 'named' output.sourcemap = !!process.env.SOURCE_MAP output.externalLiveBindings = false if (isGlobalBuild) { output.name = packageOptions.name } const shouldEmitDeclarations = pkg.types && process.env.TYPES != null && !hasTSChecked const tsPlugin = ts({ check: process.env.NODE_ENV === 'production' && !hasTSChecked, tsconfig: path.resolve(__dirname, 'tsconfig.json'), cacheRoot: path.resolve(__dirname, 'node_modules/.rts2_cache'), tsconfigOverride: { compilerOptions: { sourceMap: output.sourcemap, declaration: shouldEmitDeclarations, declarationMap: shouldEmitDeclarations }, exclude: ['**/__tests__', 'test-dts'] } }) // we only need to check TS and generate declarations once for each build. // it also seems to run into weird issues when checking multiple times // during a single build. hasTSChecked = true let entryFile = /runtime$/.test(format) ? `src/runtime.ts` : `src/index.ts` // the compat build needs both default AND named exports. This will cause // Rollup to complain for non-ESM targets, so we use separate entries for // esm vs. non-esm builds. if (isCompatPackage && (isBrowserESMBuild || isBundlerESMBuild)) { entryFile = /runtime$/.test(format) ? `src/esm-runtime.ts` : `src/esm-index.ts` } let external = [] if (isGlobalBuild || isBrowserESMBuild || isCompatPackage) { if (!packageOptions.enableNonBrowserBranches) { // normal browser builds - non-browser only imports are tree-shaken, // they are only listed here to suppress warnings. external = ['source-map', '@babel/parser', 'estree-walker'] } } else { // Node / esm-bundler builds. // externalize all direct deps unless it's the compat build. external = [ ...Object.keys(pkg.dependencies || {}), ...Object.keys(pkg.peerDependencies || {}), ...['path', 'url', 'stream'] // for @vue/compiler-sfc / server-renderer ] } // we are bundling forked consolidate.js in compiler-sfc which dynamically // requires a ton of template engines which should be ignored. let cjsIgnores = [] if (pkg.name === '@vue/compiler-sfc') { cjsIgnores = [ ...Object.keys(require('@vue/consolidate/package.json').devDependencies), 'vm', 'crypto', 'react-dom/server', 'teacup/lib/express', 'arc-templates/dist/es5', 'then-pug', 'then-jade' ] } const nodePlugins = (format === 'cjs' && Object.keys(pkg.devDependencies || {}).length) || packageOptions.enableNonBrowserBranches ? [ // @ts-ignore require('@rollup/plugin-commonjs')({ sourceMap: false, ignore: cjsIgnores }), ...(format === 'cjs' ? [] : // @ts-ignore [require('rollup-plugin-polyfill-node')()]), require('@rollup/plugin-node-resolve').nodeResolve() ] : [] return { input: resolve(entryFile), // Global and Browser ESM builds inlines everything so that they can be // used alone. external, plugins: [ json({ namedExports: false }), tsPlugin, createReplacePlugin( isProductionBuild, isBundlerESMBuild, isBrowserESMBuild, // isBrowserBuild? (isGlobalBuild || isBrowserESMBuild || isBundlerESMBuild) && !packageOptions.enableNonBrowserBranches, isGlobalBuild, isNodeBuild, isCompatBuild ), ...nodePlugins, ...plugins ], output, onwarn: (msg, warn) => { if (!/Circular/.test(msg)) { warn(msg) } }, treeshake: { moduleSideEffects: false } } } function createReplacePlugin( isProduction, isBundlerESMBuild, isBrowserESMBuild, isBrowserBuild, isGlobalBuild, isNodeBuild, isCompatBuild ) { const replacements = { __COMMIT__: `"${process.env.COMMIT}"`, __VERSION__: `"${masterVersion}"`, __DEV__: isBundlerESMBuild ? // preserve to be handled by bundlers `(process.env.NODE_ENV !== 'production')` : // hard coded dev/prod builds !isProduction, // this is only used during Vue's internal tests __TEST__: false, // If the build is expected to run directly in the browser (global / esm builds) __BROWSER__: isBrowserBuild, __GLOBAL__: isGlobalBuild, __ESM_BUNDLER__: isBundlerESMBuild, __ESM_BROWSER__: isBrowserESMBuild, // is targeting Node (SSR)? __NODE_JS__: isNodeBuild, // for compiler-sfc browser build inlined deps ...(isBrowserESMBuild ? { 'process.env': '({})', 'process.platform': '""', 'process.stdout': 'null' } : {}), // 2.x compat build __COMPAT__: isCompatBuild, // feature flags __FEATURE_SUSPENSE__: true, __FEATURE_OPTIONS_API__: isBundlerESMBuild ? `__VUE_OPTIONS_API__` : true, __FEATURE_PROD_DEVTOOLS__: isBundlerESMBuild ? `__VUE_PROD_DEVTOOLS__` : false, ...(isProduction && isBrowserBuild ? { 'context.onError(': `/*#__PURE__*/ context.onError(`, 'emitError(': `/*#__PURE__*/ emitError(`, 'createCompilerError(': `/*#__PURE__*/ createCompilerError(`, 'createDOMCompilerError(': `/*#__PURE__*/ createDOMCompilerError(` } : {}) } // allow inline overrides like //__RUNTIME_COMPILE__=true yarn build runtime-core Object.keys(replacements).forEach(key => { if (key in process.env) { replacements[key] = process.env[key] } }) return replace({ // @ts-ignore values: replacements, preventAssignment: true }) } function createProductionConfig(format) { return createConfig(format, { file: resolve(`dist/${name}.${format}.prod.js`), format: outputConfigs[format].format }) } function createMinifiedConfig(format) { const { terser } = require('rollup-plugin-terser') return createConfig( format, { file: outputConfigs[format].file.replace(/\.js$/, '.prod.js'), format: outputConfigs[format].format }, [ terser({ module: /^esm/.test(format), compress: { ecma: 2015, pure_getters: true }, safari10: true }) ] ) }