2023-07-18 02:17:59 +02:00
|
|
|
import { existsSync } from 'node:fs';
|
2024-11-18 13:35:34 +00:00
|
|
|
import { build } from 'esbuild';
|
2022-05-26 11:49:29 -05:00
|
|
|
|
|
|
|
const CLIENT_RUNTIME_PATH = 'packages/astro/src/runtime/client/';
|
|
|
|
|
|
|
|
function formatBytes(bytes, decimals = 2) {
|
2024-11-18 13:34:43 +00:00
|
|
|
if (bytes === 0) return '0 B';
|
2022-05-26 11:49:29 -05:00
|
|
|
|
2024-11-18 13:34:43 +00:00
|
|
|
const k = 1024;
|
|
|
|
const dm = decimals < 0 ? 0 : decimals;
|
|
|
|
const sizes = ['B', 'KB', 'MB', 'GB', 'TB', 'PB', 'EB', 'ZB', 'YB'];
|
2022-05-26 11:49:29 -05:00
|
|
|
|
2024-11-18 13:34:43 +00:00
|
|
|
const i = Math.floor(Math.log(bytes) / Math.log(k));
|
2022-05-26 11:49:29 -05:00
|
|
|
|
2024-11-18 13:34:43 +00:00
|
|
|
return parseFloat((bytes / Math.pow(k, i)).toFixed(dm)) + ' ' + sizes[i];
|
2022-05-26 11:49:29 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
export default async function checkBundleSize({ github, context }) {
|
|
|
|
const PR_NUM = context.payload.pull_request.number;
|
|
|
|
const SHA = context.payload.pull_request.head.sha;
|
|
|
|
|
|
|
|
const { data: files } = await github.rest.pulls.listFiles({
|
|
|
|
...context.repo,
|
|
|
|
pull_number: PR_NUM,
|
|
|
|
});
|
2023-06-27 21:16:10 +08:00
|
|
|
const clientRuntimeFiles = files.filter((file) => {
|
2024-11-18 13:34:43 +00:00
|
|
|
return file.filename.startsWith(CLIENT_RUNTIME_PATH) && file.status !== 'removed';
|
2023-06-27 21:16:10 +08:00
|
|
|
});
|
2022-05-26 11:49:29 -05:00
|
|
|
if (clientRuntimeFiles.length === 0) return;
|
2023-07-18 02:17:59 +02:00
|
|
|
|
2022-05-26 11:49:29 -05:00
|
|
|
const table = [
|
|
|
|
'| File | Old Size | New Size | Change |',
|
|
|
|
'| ---- | -------- | -------- | ------ |',
|
|
|
|
];
|
|
|
|
const output = await bundle(clientRuntimeFiles);
|
2023-07-18 02:17:59 +02:00
|
|
|
|
2022-05-26 11:49:29 -05:00
|
|
|
for (let [filename, { oldSize, newSize, sourceFile }] of Object.entries(output)) {
|
2024-11-18 13:34:43 +00:00
|
|
|
filename = ['idle', 'load', 'media', 'only', 'visible'].includes(filename)
|
|
|
|
? `client:${filename}`
|
|
|
|
: filename;
|
|
|
|
const prefix = newSize - oldSize === 0 ? '' : newSize - oldSize > 0 ? '+ ' : '- ';
|
2022-05-26 11:49:29 -05:00
|
|
|
const change = `${prefix}${formatBytes(newSize - oldSize)}`;
|
2024-11-18 13:34:43 +00:00
|
|
|
table.push(
|
|
|
|
`| [\`${filename}\`](https://github.com/${context.repo.owner}/${context.repo.repo}/tree/${context.payload.pull_request.head.ref}/${sourceFile}) | ${formatBytes(oldSize)} | ${formatBytes(newSize)} | ${change} |`,
|
|
|
|
);
|
2022-05-26 11:49:29 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
const { data: comments } = await github.rest.issues.listComments({
|
|
|
|
...context.repo,
|
2024-11-18 13:34:43 +00:00
|
|
|
issue_number: PR_NUM,
|
|
|
|
});
|
|
|
|
const comment = comments.find(
|
|
|
|
(comment) =>
|
|
|
|
comment.user.login === 'github-actions[bot]' && comment.body.includes('Bundle Size Check'),
|
|
|
|
);
|
2022-05-26 11:49:29 -05:00
|
|
|
const method = comment ? 'updateComment' : 'createComment';
|
|
|
|
const payload = comment ? { comment_id: comment.id } : { issue_number: PR_NUM };
|
|
|
|
await github.rest.issues[method]({
|
|
|
|
...context.repo,
|
|
|
|
...payload,
|
|
|
|
body: `### ⚖️ Bundle Size Check
|
|
|
|
|
|
|
|
Latest commit: ${SHA}
|
|
|
|
|
|
|
|
${table.join('\n')}`,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
async function bundle(files) {
|
|
|
|
const { metafile } = await build({
|
2024-11-18 13:34:43 +00:00
|
|
|
entryPoints: [
|
|
|
|
...files.map(({ filename }) => filename),
|
|
|
|
...files.map(({ filename }) => `main/${filename}`).filter((f) => existsSync(f)),
|
|
|
|
],
|
2022-05-26 11:49:29 -05:00
|
|
|
bundle: true,
|
|
|
|
minify: true,
|
|
|
|
sourcemap: false,
|
|
|
|
target: ['es2018'],
|
|
|
|
outdir: 'out',
|
2024-02-07 17:52:14 +08:00
|
|
|
external: ['astro:*', 'aria-query', 'axobject-query'],
|
2022-05-26 11:49:29 -05:00
|
|
|
metafile: true,
|
2024-11-18 13:34:43 +00:00
|
|
|
});
|
2022-05-26 11:49:29 -05:00
|
|
|
|
|
|
|
return Object.entries(metafile.outputs).reduce((acc, [filename, info]) => {
|
|
|
|
filename = filename.slice('out/'.length);
|
|
|
|
if (filename.startsWith('main/')) {
|
|
|
|
filename = filename.slice('main/'.length).replace(CLIENT_RUNTIME_PATH, '').replace('.js', '');
|
|
|
|
const oldSize = info.bytes;
|
2024-11-18 13:34:43 +00:00
|
|
|
return Object.assign(acc, {
|
|
|
|
[filename]: Object.assign(acc[filename] ?? { oldSize: 0, newSize: 0 }, { oldSize }),
|
|
|
|
});
|
2022-05-26 11:49:29 -05:00
|
|
|
}
|
|
|
|
filename = filename.replace(CLIENT_RUNTIME_PATH, '').replace('.js', '');
|
|
|
|
const newSize = info.bytes;
|
2024-11-18 13:34:43 +00:00
|
|
|
return Object.assign(acc, {
|
|
|
|
[filename]: Object.assign(acc[filename] ?? { oldSize: 0, newSize: 0 }, {
|
|
|
|
newSize,
|
|
|
|
sourceFile: Object.keys(info.inputs).find((src) => src.endsWith('.ts')),
|
|
|
|
}),
|
|
|
|
});
|
2022-05-26 11:49:29 -05:00
|
|
|
}, {});
|
|
|
|
}
|