| #!/usr/bin/env node | 
 |  | 
 | /* | 
 | * Licensed to the Apache Software Foundation (ASF) under one | 
 | * or more contributor license agreements.  See the NOTICE file | 
 | * distributed with this work for additional information | 
 | * regarding copyright ownership.  The ASF licenses this file | 
 | * to you under the Apache License, Version 2.0 (the | 
 | * "License"); you may not use this file except in compliance | 
 | * with the License.  You may obtain a copy of the License at | 
 | * | 
 | *   http://www.apache.org/licenses/LICENSE-2.0 | 
 | * | 
 | * Unless required by applicable law or agreed to in writing, | 
 | * software distributed under the License is distributed on an | 
 | * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY | 
 | * KIND, either express or implied.  See the License for the | 
 | * specific language governing permissions and limitations | 
 | * under the License. | 
 | */ | 
 |  | 
 | const fs = require('fs'); | 
 | const config = require('./config.js'); | 
 | const commander = require('commander'); | 
 | const chalk = require('chalk'); | 
 | const rollup = require('rollup'); | 
 | const prePublish = require('./pre-publish'); | 
 | const transformDEV = require('./transform-dev'); | 
 |  | 
 | async function run() { | 
 |  | 
 |     /** | 
 |      * Tips for `commander`: | 
 |      * (1) If arg xxx not specified, `commander.xxx` is undefined. | 
 |      *     Otherwise: | 
 |      *      If '-x, --xxx', `commander.xxx` can only be true/false, even if '--xxx yyy' input. | 
 |      *      If '-x, --xxx <some>', the 'some' string is required, or otherwise error will be thrown. | 
 |      *      If '-x, --xxx [some]', the 'some' string is optional, that is, `commander.xxx` can be boolean or string. | 
 |      * (2) `node ./build/build.js --help` will print helper info and exit. | 
 |      */ | 
 |  | 
 |     let descIndent = '                                 '; | 
 |     let egIndent = '    '; | 
 |  | 
 |     commander | 
 |         .usage('[options]') | 
 |         .description([ | 
 |             'Build echarts and generate result files in directory `echarts/dist`.', | 
 |             '', | 
 |             '  For example:', | 
 |             '', | 
 |             egIndent + 'node build/build.js --prepublish' | 
 |                 + '\n' + descIndent + '# Only prepublish.', | 
 |             egIndent + 'node build/build.js --type ""' | 
 |                 + '\n' + descIndent + '# Only generate `dist/echarts.js`.', | 
 |             egIndent + 'node build/build.js --type common --min' | 
 |                 + '\n' + descIndent + '# Only generate `dist/echarts.common.min.js`.', | 
 |             egIndent + 'node build/build.js --type simple --min' | 
 |                 + '\n' + descIndent + '# Only generate `dist/echarts-en.simple.min.js`.', | 
 |         ].join('\n')) | 
 |         .option( | 
 |             '--prepublish', | 
 |             'Build all for release' | 
 |         ) | 
 |         .option( | 
 |             '--min', | 
 |             'Whether to compress the output file, and remove error-log-print code.' | 
 |         ) | 
 |         .option( | 
 |             '--type <type name>', [ | 
 |             'Can be "simple" or "common" or "all" (default). Or can be simple,common,all to build multiple. For example,', | 
 |             descIndent + '`--type ""` or `--type "common"`.' | 
 |         ].join('\n')) | 
 |         .option( | 
 |             '--format <format>', | 
 |             'The format of output bundle. Can be "umd", "amd", "iife", "cjs", "esm".' | 
 |         ) | 
 |         .parse(process.argv); | 
 |  | 
 |     let isPrePublish = !!commander.prepublish; | 
 |     let buildType = commander.type || 'all'; | 
 |  | 
 |     let opt = { | 
 |         min: commander.min, | 
 |         format: commander.format || 'umd' | 
 |     }; | 
 |  | 
 |     validateIO(opt.input, opt.output); | 
 |  | 
 |     if (isPrePublish) { | 
 |         await prePublish(); | 
 |     } | 
 |     else if (buildType === 'extension') { | 
 |         const cfgs = [ | 
 |             config.createBMap(opt), | 
 |             config.createDataTool(opt) | 
 |         ]; | 
 |         await build(cfgs); | 
 |     } | 
 |     else if (buildType === 'myTransform') { | 
 |         const cfgs = [ | 
 |             config.createMyTransform(opt) | 
 |         ]; | 
 |         await build(cfgs); | 
 |     } | 
 |     else { | 
 |         const types = buildType.split(',').map(a => a.trim()); | 
 |         const cfgs = types.map(type => | 
 |             config.createECharts({ | 
 |                 ...opt, | 
 |                 type | 
 |             }) | 
 |         ); | 
 |         await build(cfgs); | 
 |     } | 
 | } | 
 |  | 
 | function checkBundleCode(cfg) { | 
 |     // Make sure process.env.NODE_ENV is eliminated. | 
 |     for (let output of cfg.output) { | 
 |         let code = fs.readFileSync(output.file, {encoding: 'utf-8'}); | 
 |         if (!code) { | 
 |             throw new Error(`${output.file} is empty`); | 
 |         } | 
 |         transformDEV.recheckDEV(code); | 
 |         console.log(chalk.green.dim('Check code: correct.')); | 
 |     } | 
 | } | 
 |  | 
 | function validateIO(input, output) { | 
 |     if ((input != null && output == null) | 
 |         || (input == null && output != null) | 
 |     ) { | 
 |         throw new Error('`input` and `output` must be both set.'); | 
 |     } | 
 | } | 
 |  | 
 | /** | 
 |  * @param {Array.<Object>} configs A list of rollup configs: | 
 |  *  See: <https://rollupjs.org/#big-list-of-options> | 
 |  *  For example: | 
 |  *  [ | 
 |  *      { | 
 |  *          ...inputOptions, | 
 |  *          output: [outputOptions], | 
 |  *      }, | 
 |  *      ... | 
 |  *  ] | 
 |  */ | 
 | async function build(configs) { | 
 |     console.log(chalk.yellow(` | 
 |     NOTICE: If you are using 'npm run build'. Run 'npm run prepublish' before build !!! | 
 | `)); | 
 |  | 
 |     console.log(chalk.yellow(` | 
 |     NOTICE: If you are using syslink on zrender. Run 'npm run prepublish' in zrender first !! | 
 | `)); | 
 |  | 
 |     for (let singleConfig of configs) { | 
 |         console.log( | 
 |             chalk.cyan.dim('\Bundling '), | 
 |             chalk.cyan(singleConfig.input) | 
 |         ); | 
 |  | 
 |         console.time('rollup build'); | 
 |         const bundle = await rollup.rollup(singleConfig); | 
 |  | 
 |         for (let output of singleConfig.output) { | 
 |             console.log( | 
 |                 chalk.green.dim('Created '), | 
 |                 chalk.green(output.file), | 
 |                 chalk.green.dim(' successfully.') | 
 |             ); | 
 |  | 
 |             await bundle.write(output); | 
 |  | 
 |         }; | 
 |         console.timeEnd('rollup build'); | 
 |  | 
 |         checkBundleCode(singleConfig); | 
 |     } | 
 | } | 
 |  | 
 | async function main() { | 
 |     try { | 
 |         await run(); | 
 |     } | 
 |     catch (err) { | 
 |         console.log(chalk.red('BUILD ERROR!')); | 
 |         // rollup parse error. | 
 |         if (err) { | 
 |             if (err.loc) { | 
 |                 console.warn(chalk.red(`${err.loc.file} (${err.loc.line}:${err.loc.column})`)); | 
 |                 console.warn(chalk.red(err.message)); | 
 |             } | 
 |             if (err.frame) { | 
 |                 console.warn(chalk.red(err.frame)); | 
 |             } | 
 |             console.log(chalk.red(err ? err.stack : err)); | 
 |  | 
 |             err.id != null && console.warn(chalk.red(`id: ${err.id}`)); | 
 |             err.hook != null && console.warn(chalk.red(`hook: ${err.hook}`)); | 
 |             err.code != null && console.warn(chalk.red(`code: ${err.code}`)); | 
 |             err.plugin != null && console.warn(chalk.red(`plugin: ${err.plugin}`)); | 
 |         } | 
 |         // console.log(err); | 
 |     } | 
 | } | 
 |  | 
 | main(); |