Fix bin/publish: use correct .env path for rspade_system Fix bin/publish script: prevent grep exit code 1 from terminating script 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude <noreply@anthropic.com>
237 lines
5.7 KiB
JavaScript
Executable File
237 lines
5.7 KiB
JavaScript
Executable File
const _ = require('lodash');
|
|
const chalk = require('chalk');
|
|
const Table = require('cli-table3');
|
|
const readline = require('readline');
|
|
const stripAnsi = require('strip-ansi');
|
|
const { formatSize } = require('webpack/lib/SizeFormatHelpers');
|
|
const { version } = require('../../package.json');
|
|
|
|
/**
|
|
* @typedef {object} BuildOutputOptions
|
|
* @property {boolean} clearConsole
|
|
* @property {boolean} showRelated
|
|
**/
|
|
|
|
/**
|
|
* @typedef {object} StatsAsset
|
|
* @property {string} name
|
|
* @property {number} size
|
|
* @property {StatsAsset[]|{}} related
|
|
**/
|
|
|
|
/**
|
|
* @typedef {object} StatsData
|
|
* @property {StatsAsset[]} assets
|
|
**/
|
|
|
|
class BuildOutputPlugin {
|
|
/**
|
|
*
|
|
* @param {BuildOutputOptions} options
|
|
*/
|
|
constructor(options) {
|
|
this.options = options;
|
|
this.patched = false;
|
|
}
|
|
|
|
/**
|
|
* Apply the plugin.
|
|
*
|
|
* @param {import("webpack").Compiler} compiler
|
|
*/
|
|
apply(compiler) {
|
|
// TODO: Refactor setup to allow removing this check
|
|
if (process.env.NODE_ENV === 'test') {
|
|
return;
|
|
}
|
|
|
|
compiler.hooks.done.tap('BuildOutputPlugin', stats => {
|
|
if (stats.hasErrors()) {
|
|
return false;
|
|
}
|
|
|
|
if (this.options.clearConsole) {
|
|
this.clearConsole();
|
|
}
|
|
|
|
let data = stats.toJson({
|
|
assets: true,
|
|
builtAt: true,
|
|
hash: true,
|
|
performance: true,
|
|
relatedAssets: this.options.showRelated
|
|
});
|
|
|
|
this.heading(`Laravel Mix v${version}`);
|
|
|
|
console.log(chalk.green.bold(`✔ Compiled Successfully in ${data.time}ms`));
|
|
|
|
if (data.assets.length) {
|
|
console.log(this.statsTable(data));
|
|
}
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Print a block section heading.
|
|
*
|
|
* @param {string} text
|
|
*/
|
|
heading(text) {
|
|
console.log();
|
|
|
|
console.log(chalk.bgBlue.white.bold(this.section(text)));
|
|
|
|
console.log();
|
|
}
|
|
|
|
/**
|
|
* Create a block section.
|
|
*
|
|
* @param {string} text
|
|
*/
|
|
section(text) {
|
|
const padLength = 3;
|
|
const padding = ' '.repeat(padLength);
|
|
|
|
text = `${padding}${text}${padding}`;
|
|
|
|
const line = ' '.repeat(text.length);
|
|
|
|
return `${line}\n${text}\n${line}`;
|
|
}
|
|
|
|
/**
|
|
* Generate the stats table.
|
|
*
|
|
* @param {StatsData} data
|
|
* @returns {string}
|
|
*/
|
|
statsTable(data) {
|
|
const assets = this.sortAssets(data);
|
|
|
|
const table = new Table({
|
|
head: [chalk.bold('File'), chalk.bold('Size')],
|
|
colWidths: [35],
|
|
colAligns: ['right'],
|
|
style: {
|
|
head: [],
|
|
compact: true
|
|
}
|
|
});
|
|
|
|
for (const asset of assets) {
|
|
table.push([chalk.green(asset.name), formatSize(asset.size)]);
|
|
}
|
|
|
|
this.extendTableWidth(table);
|
|
this.monkeyPatchTruncate();
|
|
|
|
return table.toString();
|
|
}
|
|
|
|
/**
|
|
*
|
|
* @param {StatsData} data
|
|
*/
|
|
sortAssets(data) {
|
|
let assets = data.assets;
|
|
|
|
assets = _.flatMap(assets, asset => [
|
|
asset,
|
|
...(Array.isArray(asset.related) ? asset.related : [])
|
|
]);
|
|
|
|
assets = _.orderBy(assets, ['name', 'size'], ['asc', 'asc']);
|
|
|
|
return assets;
|
|
}
|
|
|
|
/**
|
|
* Clear the entire screen.
|
|
*/
|
|
clearConsole() {
|
|
const blank = '\n'.repeat(process.stdout.rows);
|
|
console.log(blank);
|
|
|
|
readline.cursorTo(process.stdout, 0, 0);
|
|
readline.clearScreenDown(process.stdout);
|
|
}
|
|
|
|
/**
|
|
* Extend the width of the table
|
|
*
|
|
* Currently only increases the file column size
|
|
*
|
|
* @param {import("cli-table3").Table} table
|
|
* @param {number|null} targetWidth
|
|
* @param {number} maxWidth
|
|
*/
|
|
extendTableWidth(table, targetWidth = null, maxWidth = Infinity) {
|
|
targetWidth = targetWidth === null ? process.stdout.columns : targetWidth;
|
|
|
|
if (!targetWidth) {
|
|
return;
|
|
}
|
|
|
|
const tableWidth = this.calculateTableWidth(table);
|
|
const fileColIncrease = Math.min(targetWidth - tableWidth, maxWidth - tableWidth);
|
|
|
|
if (fileColIncrease <= 0) {
|
|
return;
|
|
}
|
|
|
|
// @ts-ignore
|
|
table.options.colWidths[0] += fileColIncrease;
|
|
}
|
|
|
|
// Yeah, I know.
|
|
monkeyPatchTruncate() {
|
|
if (this.patched) {
|
|
return;
|
|
}
|
|
|
|
this.patched = true;
|
|
|
|
// @ts-ignore
|
|
const utils = require('cli-table3/src/utils');
|
|
const oldTruncate = utils.truncate;
|
|
|
|
// cli-table3 can only do truncation at the end
|
|
// We want the asset name to be truncated at the beginning if it's too long
|
|
// FIXME: We really should set truncation location via a paramter or something
|
|
// (or PR support for alignment-based truncation)
|
|
|
|
/**
|
|
*
|
|
* @param {string} str
|
|
* @param {number} desiredLength
|
|
* @param {string} truncateChar
|
|
*/
|
|
utils.truncate = (str, desiredLength, truncateChar) => {
|
|
if (stripAnsi(str).length > desiredLength) {
|
|
str = `…${str.substr(-desiredLength + 2)}`;
|
|
}
|
|
|
|
return oldTruncate(str, desiredLength, truncateChar);
|
|
};
|
|
}
|
|
|
|
/**
|
|
* Calculate the width of the CLI Table
|
|
*
|
|
* `table.width` does not report the correct width
|
|
* because it includes ANSI control characters
|
|
*
|
|
* @internal
|
|
* @param {import("cli-table3").Table} table
|
|
*/
|
|
calculateTableWidth(table) {
|
|
const firstRow = table.toString().split('\n')[0];
|
|
|
|
return stripAnsi(firstRow).length;
|
|
}
|
|
}
|
|
|
|
module.exports = BuildOutputPlugin;
|