build.js 3.9 KB

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