import fs from 'fs' import path from 'path' import ts from 'rollup-plugin-typescript2' import replace from 'rollup-plugin-replace' import alias from 'rollup-plugin-alias' import json from 'rollup-plugin-json' import lernaJson from './lerna.json' if (!process.env.TARGET) { throw new Error('TARGET package must be specified via --environment flag.') } const packagesDir = path.resolve(__dirname, 'packages') const packageDir = path.resolve(packagesDir, process.env.TARGET) const name = path.basename(packageDir) const resolve = p => path.resolve(packageDir, p) const pkg = require(resolve(`package.json`)) const packageOptions = pkg.buildOptions || {} // build aliases dynamically const aliasOptions = { resolve: ['.ts'] } fs.readdirSync(packagesDir).forEach(dir => { if (dir === 'vue') { return } if (fs.statSync(path.resolve(packagesDir, dir)).isDirectory()) { aliasOptions[`@vue/${dir}`] = path.resolve(packagesDir, `${dir}/src/index`) } }) const aliasPlugin = alias(aliasOptions) // ensure TS checks only once for each build let hasTSChecked = false const configs = { esm: { file: resolve(`dist/${name}.esm-bundler.js`), format: `es` }, cjs: { file: resolve(`dist/${name}.cjs.js`), format: `cjs` }, global: { file: resolve(`dist/${name}.global.js`), format: `iife` }, 'esm-browser': { file: resolve(`dist/${name}.esm-browser.js`), format: `es` } } const defaultFormats = ['esm', '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(configs[format])) if (process.env.NODE_ENV === 'production') { packageFormats.forEach(format => { if (format === 'cjs') { packageConfigs.push(createProductionConfig(format)) } if (format === 'global' || format === 'esm-browser') { packageConfigs.push(createMinifiedConfig(format)) } }) } export default packageConfigs function createConfig(output, plugins = []) { const isProductionBuild = process.env.__DEV__ === 'false' || /\.prod\.js$/.test(output.file) const isGlobalBuild = /\.global(\.prod)?\.js$/.test(output.file) const isBundlerESMBuild = /\.esm-bundler\.js$/.test(output.file) const isBrowserESMBuild = /esm-browser(\.prod)?\.js$/.test(output.file) const isRuntimeCompileBuild = /vue\./.test(output.file) if (isGlobalBuild) { output.name = packageOptions.name } const shouldEmitDeclarations = process.env.TYPES != null && process.env.NODE_ENV === 'production' && !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: { 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 const externals = Object.keys(aliasOptions).filter(p => p !== '@vue/shared') return { input: resolve(`src/index.ts`), // Global and Browser ESM builds inlines everything so that they can be // used alone. external: isGlobalBuild || isBrowserESMBuild ? [] : externals, plugins: [ json({ namedExports: false }), tsPlugin, aliasPlugin, createReplacePlugin( isProductionBuild, isBundlerESMBuild, (isGlobalBuild || isBrowserESMBuild) && !packageOptions.enableNonBrowserBranches, isRuntimeCompileBuild ), ...plugins ], output, onwarn: (msg, warn) => { if (!/Circular/.test(msg)) { warn(msg) } } } } function createReplacePlugin( isProduction, isBundlerESMBuild, isBrowserBuild, isRuntimeCompileBuild ) { return replace({ __COMMIT__: `"${process.env.COMMIT}"`, __VERSION__: `"${lernaJson.version}"`, __DEV__: isBundlerESMBuild ? // preserve to be handled by bundlers `(process.env.NODE_ENV !== 'production')` : // hard coded dev/prod builds !isProduction, // this is only used during tests __TEST__: isBundlerESMBuild ? `(process.env.NODE_ENV === 'test')` : false, // If the build is expected to run directly in the browser (global / esm-browser builds) __BROWSER__: isBrowserBuild, // support compile in browser? __RUNTIME_COMPILE__: isRuntimeCompileBuild, // support options? // the lean build drops options related code with buildOptions.lean: true __FEATURE_OPTIONS__: !packageOptions.lean && !process.env.LEAN, __FEATURE_SUSPENSE__: true }) } function createProductionConfig(format) { return createConfig({ file: resolve(`dist/${name}.${format}.prod.js`), format: configs[format].format }) } function createMinifiedConfig(format) { const { terser } = require('rollup-plugin-terser') return createConfig( { file: resolve(`dist/${name}.${format}.prod.js`), format: configs[format].format }, [ terser({ module: /^esm/.test(format) }) ] ) }