build.js 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116
  1. let fs = require('fs');
  2. let DotJson = require('dot-json');
  3. let brotliSize = require('brotli-size');
  4. ([
  5. // Packages:
  6. 'alpinejs',
  7. 'csp',
  8. 'history',
  9. 'intersect',
  10. 'persist',
  11. 'collapse',
  12. 'morph',
  13. 'trap',
  14. ]).forEach(package => {
  15. if (! fs.existsSync(`./packages/${package}/dist`)) {
  16. fs.mkdirSync(`./packages/${package}/dist`, 0744);
  17. }
  18. // Go through each file in the package's "build" directory
  19. // and use the appropriate bundling strategy based on its name.
  20. fs.readdirSync(`./packages/${package}/builds`).forEach(file => {
  21. bundleFile(package, file)
  22. });
  23. })
  24. function bundleFile(package, file) {
  25. // Based on the filename, give esbuild a specific configuration to build.
  26. ({
  27. // This output file is meant to be loaded in a browser's <script> tag.
  28. 'cdn.js': () => {
  29. build({
  30. entryPoints: [`packages/${package}/builds/${file}`],
  31. outfile: `packages/${package}/dist/${file}`,
  32. bundle: true,
  33. platform: 'browser',
  34. define: { CDN: true },
  35. })
  36. // Build a minified version.
  37. build({
  38. entryPoints: [`packages/${package}/builds/${file}`],
  39. outfile: `packages/${package}/dist/${file.replace('.js', '.min.js')}`,
  40. bundle: true,
  41. minify: true,
  42. platform: 'browser',
  43. define: { CDN: true },
  44. }).then(() => {
  45. outputSize(package, `packages/${package}/dist/${file.replace('.js', '.min.js')}`)
  46. })
  47. },
  48. // This file outputs two files: an esm module and a cjs module.
  49. // The ESM one is meant for "import" statements (bundlers and new browsers)
  50. // and the cjs one is meant for "require" statements (node).
  51. 'module.js': () => {
  52. build({
  53. entryPoints: [`packages/${package}/builds/${file}`],
  54. outfile: `packages/${package}/dist/${file.replace('.js', '.esm.js')}`,
  55. bundle: true,
  56. platform: 'neutral',
  57. mainFields: ['main', 'module'],
  58. })
  59. build({
  60. entryPoints: [`packages/${package}/builds/${file}`],
  61. outfile: `packages/${package}/dist/${file.replace('.js', '.cjs.js')}`,
  62. bundle: true,
  63. target: ['node10.4'],
  64. platform: 'node',
  65. }).then(() => {
  66. writeToPackageDotJson(package, 'main', `dist/${file.replace('.js', '.cjs.js')}`)
  67. writeToPackageDotJson(package, 'module', `dist/${file.replace('.js', '.esm.js')}`)
  68. })
  69. },
  70. })[file]()
  71. }
  72. function build(options) {
  73. options.define || (options.define = {})
  74. options.define['ALPINE_VERSION'] = `'${getFromPackageDotJson('alpinejs', 'version')}'`
  75. options.define['process.env.NODE_ENV'] = process.argv.includes('--watch') ? `'production'` : `'development'`
  76. return require('esbuild').build({
  77. watch: process.argv.includes('--watch'),
  78. // external: ['alpinejs'],
  79. ...options,
  80. }).catch(() => process.exit(1))
  81. }
  82. function writeToPackageDotJson(package, key, value) {
  83. let dotJson = new DotJson(`./packages/${package}/package.json`)
  84. dotJson.set(key, value).save()
  85. }
  86. function getFromPackageDotJson(package, key) {
  87. let dotJson = new DotJson(`./packages/${package}/package.json`)
  88. return dotJson.get(key)
  89. }
  90. function outputSize(package, file) {
  91. let size = bytesToSize(brotliSize.sync(fs.readFileSync(file)))
  92. console.log("\x1b[32m", `${package}: ${size}`)
  93. }
  94. function bytesToSize(bytes) {
  95. const sizes = ['Bytes', 'KB', 'MB', 'GB', 'TB']
  96. if (bytes === 0) return 'n/a'
  97. const i = parseInt(Math.floor(Math.log(bytes) / Math.log(1024)), 10)
  98. if (i === 0) return `${bytes} ${sizes[i]}`
  99. return `${(bytes / (1024 ** i)).toFixed(1)} ${sizes[i]}`
  100. }