vue3-core/scripts/size-report.ts

106 lines
3.0 KiB
TypeScript

import path from 'node:path'
import { markdownTable } from 'markdown-table'
import prettyBytes from 'pretty-bytes'
import { readdir } from 'node:fs/promises'
import { existsSync } from 'node:fs'
interface SizeResult {
size: number
gzip: number
brotli: number
}
interface BundleResult extends SizeResult {
file: string
}
type UsageResult = Record<string, SizeResult & { name: string }>
const currDir = path.resolve('temp/size')
const prevDir = path.resolve('temp/size-prev')
let output = '## Size Report\n\n'
const sizeHeaders = ['Size', 'Gzip', 'Brotli']
run()
async function run() {
await renderFiles()
await renderUsages()
process.stdout.write(output)
}
async function renderFiles() {
const filterFiles = (files: string[]) =>
files.filter(file => !file.startsWith('_'))
const curr = filterFiles(await readdir(currDir))
const prev = existsSync(prevDir) ? filterFiles(await readdir(prevDir)) : []
const fileList = new Set([...curr, ...prev])
const rows: string[][] = []
for (const file of fileList) {
const currPath = path.resolve(currDir, file)
const prevPath = path.resolve(prevDir, file)
const curr = await importJSON<BundleResult>(currPath)
const prev = await importJSON<BundleResult>(prevPath)
const fileName = curr?.file || prev?.file || ''
if (!curr) {
rows.push([`~~${fileName}~~`])
} else
rows.push([
fileName,
`${prettyBytes(curr.size)}${getDiff(curr.size, prev?.size)}`,
`${prettyBytes(curr.gzip)}${getDiff(curr.gzip, prev?.gzip)}`,
`${prettyBytes(curr.brotli)}${getDiff(curr.brotli, prev?.brotli)}`,
])
}
output += '### Bundles\n\n'
output += markdownTable([['File', ...sizeHeaders], ...rows])
output += '\n\n'
}
async function renderUsages() {
const curr = (await importJSON<UsageResult>(
path.resolve(currDir, '_usages.json'),
))!
const prev = await importJSON<UsageResult>(
path.resolve(prevDir, '_usages.json'),
)
output += '\n### Usages\n\n'
const data = Object.values(curr)
.map(usage => {
const prevUsage = prev?.[usage.name]
const diffSize = getDiff(usage.size, prevUsage?.size)
const diffGzipped = getDiff(usage.gzip, prevUsage?.gzip)
const diffBrotli = getDiff(usage.brotli, prevUsage?.brotli)
return [
usage.name,
`${prettyBytes(usage.size)}${diffSize}`,
`${prettyBytes(usage.gzip)}${diffGzipped}`,
`${prettyBytes(usage.brotli)}${diffBrotli}`,
]
})
.filter((usage): usage is string[] => !!usage)
output += `${markdownTable([['Name', ...sizeHeaders], ...data])}\n\n`
}
async function importJSON<T>(path: string): Promise<T | undefined> {
if (!existsSync(path)) return undefined
return (await import(path, { assert: { type: 'json' } })).default
}
function getDiff(curr: number, prev?: number) {
if (prev === undefined) return ''
const diff = curr - prev
if (diff === 0) return ''
const sign = diff > 0 ? '+' : ''
return ` (**${sign}${prettyBytes(diff)}**)`
}