vue3-yuanma/scripts/build.js

190 lines
5.3 KiB
JavaScript
Raw Normal View History

2018-09-20 00:21:00 +08:00
/*
Produces production builds and stitches together d.ts files.
2018-09-20 00:21:00 +08:00
To specify the package to build, simply pass its name and the desired build
2018-09-20 00:21:00 +08:00
formats to output (defaults to `buildOptions.formats` specified in that package,
or "esm,cjs"):
```
# name supports fuzzy match. will build all packages with name containing "dom":
yarn build dom
# specify the format to output
yarn build core --formats cjs
```
*/
2018-09-19 23:35:38 +08:00
const fs = require('fs-extra')
const path = require('path')
const chalk = require('chalk')
const execa = require('execa')
const { gzipSync } = require('zlib')
const { compress } = require('brotli')
const { targets: allTargets, fuzzyMatchTarget } = require('./utils')
2018-09-19 23:35:38 +08:00
2018-09-20 00:21:00 +08:00
const args = require('minimist')(process.argv.slice(2))
const targets = args._
2018-09-20 00:21:00 +08:00
const formats = args.formats || args.f
const devOnly = args.devOnly || args.d
const prodOnly = !devOnly && (args.prodOnly || args.p)
const sourceMap = args.sourcemap || args.s
2019-12-11 11:14:02 +08:00
const isRelease = args.release
const buildTypes = args.t || args.types || isRelease
2018-10-23 23:58:37 +08:00
const buildAllMatching = args.all || args.a
const commit = execa.sync('git', ['rev-parse', 'HEAD']).stdout.slice(0, 7)
run()
async function run() {
if (isRelease) {
// remove build cache for release builds to avoid outdated enum values
await fs.remove(path.resolve(__dirname, '../node_modules/.rts2_cache'))
}
if (!targets.length) {
await buildAll(allTargets)
checkAllSizes(allTargets)
2018-09-19 23:35:38 +08:00
} else {
await buildAll(fuzzyMatchTarget(targets, buildAllMatching))
checkAllSizes(fuzzyMatchTarget(targets, buildAllMatching))
2018-09-19 23:35:38 +08:00
}
}
2018-09-19 23:35:38 +08:00
2018-09-20 00:21:00 +08:00
async function buildAll(targets) {
await runParallel(require('os').cpus().length, targets, build)
}
async function runParallel(maxConcurrency, source, iteratorFn) {
const ret = []
const executing = []
for (const item of source) {
const p = Promise.resolve().then(() => iteratorFn(item, source))
ret.push(p)
if (maxConcurrency <= source.length) {
const e = p.then(() => executing.splice(executing.indexOf(e), 1))
executing.push(e)
if (executing.length >= maxConcurrency) {
await Promise.race(executing)
}
}
2018-09-19 23:35:38 +08:00
}
return Promise.all(ret)
2018-09-19 23:35:38 +08:00
}
2018-09-20 00:21:00 +08:00
async function build(target) {
2018-09-19 23:35:38 +08:00
const pkgDir = path.resolve(`packages/${target}`)
2018-10-17 05:41:59 +08:00
const pkg = require(`${pkgDir}/package.json`)
2018-09-19 23:35:38 +08:00
2019-12-11 11:14:02 +08:00
// only build published packages for release
if (isRelease && pkg.private) {
return
}
// if building a specific format, do not remove dist.
if (!formats) {
await fs.remove(`${pkgDir}/dist`)
}
2018-09-19 23:35:38 +08:00
2019-10-05 01:08:06 +08:00
const env =
(pkg.buildOptions && pkg.buildOptions.env) ||
(devOnly ? 'development' : 'production')
2018-09-20 00:21:00 +08:00
await execa(
'rollup',
[
'-c',
'--environment',
[
`COMMIT:${commit}`,
`NODE_ENV:${env}`,
`TARGET:${target}`,
formats ? `FORMATS:${formats}` : ``,
2019-11-01 23:32:53 +08:00
buildTypes ? `TYPES:true` : ``,
prodOnly ? `PROD_ONLY:true` : ``,
sourceMap ? `SOURCE_MAP:true` : ``
]
.filter(Boolean)
.join(',')
2018-09-20 00:21:00 +08:00
],
{ stdio: 'inherit' }
)
2018-09-19 23:35:38 +08:00
2019-11-01 23:32:53 +08:00
if (buildTypes && pkg.types) {
2018-10-17 05:41:59 +08:00
console.log()
console.log(
chalk.bold(chalk.yellow(`Rolling up type definitions for ${target}...`))
)
// build types
const { Extractor, ExtractorConfig } = require('@microsoft/api-extractor')
const extractorConfigPath = path.resolve(pkgDir, `api-extractor.json`)
const extractorConfig = ExtractorConfig.loadFileAndPrepare(
extractorConfigPath
2018-10-17 05:41:59 +08:00
)
const extractorResult = Extractor.invoke(extractorConfig, {
localBuild: true,
showVerboseMessages: true
})
if (extractorResult.succeeded) {
// concat additional d.ts to rolled-up dts
const typesDir = path.resolve(pkgDir, 'types')
if (await fs.exists(typesDir)) {
2019-11-01 23:32:53 +08:00
const dtsPath = path.resolve(pkgDir, pkg.types)
const existing = await fs.readFile(dtsPath, 'utf-8')
const typeFiles = await fs.readdir(typesDir)
2019-11-01 23:32:53 +08:00
const toAdd = await Promise.all(
typeFiles.map(file => {
return fs.readFile(path.resolve(typesDir, file), 'utf-8')
2019-11-01 23:32:53 +08:00
})
)
await fs.writeFile(dtsPath, existing + '\n' + toAdd.join('\n'))
}
console.log(
chalk.bold(chalk.green(`API Extractor completed successfully.`))
)
} else {
console.error(
`API Extractor completed with ${extractorResult.errorCount} errors` +
` and ${extractorResult.warningCount} warnings`
)
process.exitCode = 1
}
2018-09-19 23:35:38 +08:00
2018-10-17 05:41:59 +08:00
await fs.remove(`${pkgDir}/dist/packages`)
}
2018-09-19 23:35:38 +08:00
}
2018-09-20 00:21:00 +08:00
function checkAllSizes(targets) {
2019-12-03 07:18:02 +08:00
if (devOnly) {
return
}
2018-09-19 23:35:38 +08:00
console.log()
for (const target of targets) {
checkSize(target)
}
console.log()
}
2018-09-20 00:21:00 +08:00
function checkSize(target) {
2018-09-19 23:35:38 +08:00
const pkgDir = path.resolve(`packages/${target}`)
checkFileSize(`${pkgDir}/dist/${target}.global.prod.js`)
}
function checkFileSize(filePath) {
if (!fs.existsSync(filePath)) {
return
2018-09-19 23:35:38 +08:00
}
const file = fs.readFileSync(filePath)
const minSize = (file.length / 1024).toFixed(2) + 'kb'
const gzipped = gzipSync(file)
const gzippedSize = (gzipped.length / 1024).toFixed(2) + 'kb'
const compressed = compress(file)
const compressedSize = (compressed.length / 1024).toFixed(2) + 'kb'
console.log(
`${chalk.gray(
chalk.bold(path.basename(filePath))
)} min:${minSize} / gzip:${gzippedSize} / brotli:${compressedSize}`
)
2018-09-19 23:35:38 +08:00
}