/* eslint no-console: 'off' */ import { color, label, say as houston, spinner as load } from '@astrojs/cli-kit'; import { align, sleep } from '@astrojs/cli-kit/utils'; import { exec } from 'node:child_process'; import { get } from 'node:https'; import stripAnsi from 'strip-ansi'; let stdout = process.stdout; /** @internal Used to mock `process.stdout.write` for testing purposes */ export function setStdout(writable: typeof process.stdout) { stdout = writable; } export async function say(messages: string | string[], { clear = false, hat = '' } = {}) { return houston(messages, { clear, hat, stdout }); } export async function spinner(args: { start: string; end: string; while: (...args: any) => Promise; }) { await load(args, { stdout }); } export const title = (text: string) => align(label(text), 'end', 7) + ' '; export const welcome = [ `Let's claim your corner of the internet.`, `I'll be your assistant today.`, `Let's build something awesome!`, `Let's build something great!`, `Let's build something fast!`, `Let's build the web we want.`, `Let's make the web weird!`, `Let's make the web a better place!`, `Let's create a new project!`, `Let's create something unique!`, `Time to build a new website.`, `Time to build a faster website.`, `Time to build a sweet new website.`, `We're glad to have you on board.`, `Keeping the internet weird since 2021.`, `Initiating launch sequence...`, `Initiating launch sequence... right... now!`, `Awaiting further instructions.`, ]; export const getName = () => new Promise((resolve) => { exec('git config user.name', { encoding: 'utf-8' }, (_1, gitName, _2) => { if (gitName.trim()) { return resolve(gitName.split(' ')[0].trim()); } exec('whoami', { encoding: 'utf-8' }, (_3, whoami, _4) => { if (whoami.trim()) { return resolve(whoami.split(' ')[0].trim()); } return resolve('astronaut'); }); }); }); let v: string; export const getVersion = () => new Promise((resolve) => { if (v) return resolve(v); get('https://registry.npmjs.org/astro/latest', (res) => { let body = ''; res.on('data', (chunk) => (body += chunk)); res.on('end', () => { const { version } = JSON.parse(body); v = version; resolve(version); }); }); }); export const log = (message: string) => stdout.write(message + '\n'); export const banner = async (version: string) => log( `\n${label('astro', color.bgGreen, color.black)} ${color.green( color.bold(`v${version}`) )} ${color.bold('Launch sequence initiated.')}` ); export const info = async (prefix: string, text: string) => { await sleep(100); if (stdout.columns < 80) { log(`${' '.repeat(5)} ${color.cyan('◼')} ${color.cyan(prefix)}`); log(`${' '.repeat(9)}${color.dim(text)}`); } else { log(`${' '.repeat(5)} ${color.cyan('◼')} ${color.cyan(prefix)} ${color.dim(text)}`); } }; export const error = async (prefix: string, text: string) => { if (stdout.columns < 80) { log(`${' '.repeat(5)} ${color.red('▲')} ${color.red(prefix)}`); log(`${' '.repeat(9)}${color.dim(text)}`); } else { log(`${' '.repeat(5)} ${color.red('▲')} ${color.red(prefix)} ${color.dim(text)}`); } }; export const typescriptByDefault = async () => { await info(`No worries!`, 'TypeScript is supported in Astro by default,'); log(`${' '.repeat(9)}${color.dim('but you are free to continue writing JavaScript instead.')}`); await sleep(1000); }; export const nextSteps = async ({ projectDir, devCmd }: { projectDir: string; devCmd: string }) => { const max = stdout.columns; const prefix = max < 80 ? ' ' : ' '.repeat(9); await sleep(200); log( `\n ${color.bgCyan(` ${color.black('next')} `)} ${color.bold( 'Liftoff confirmed. Explore your project!' )}` ); await sleep(100); if (projectDir !== '') { const enter = [ `\n${prefix}Enter your project directory using`, color.cyan(`cd ./${projectDir}`, ''), ]; const len = enter[0].length + stripAnsi(enter[1]).length; log(enter.join(len > max ? '\n' + prefix : ' ')); } log( `${prefix}Run ${color.cyan(devCmd)} to start the dev server. ${color.cyan('CTRL+C')} to stop.` ); await sleep(100); log( `${prefix}Add frameworks like ${color.cyan(`react`)} or ${color.cyan( 'tailwind' )} using ${color.cyan('astro add')}.` ); await sleep(100); log(`\n${prefix}Stuck? Join us at ${color.cyan(`https://astro.build/chat`)}`); await sleep(200); }; export function printHelp({ commandName, headline, usage, tables, description, }: { commandName: string; headline?: string; usage?: string; tables?: Record; description?: string; }) { const linebreak = () => ''; const table = (rows: [string, string][], { padding }: { padding: number }) => { const split = stdout.columns < 60; let raw = ''; for (const row of rows) { if (split) { raw += ` ${row[0]}\n `; } else { raw += `${`${row[0]}`.padStart(padding)}`; } raw += ' ' + color.dim(row[1]) + '\n'; } return raw.slice(0, -1); // remove latest \n }; let message = []; if (headline) { message.push( linebreak(), `${title(commandName)} ${color.green(`v${process.env.PACKAGE_VERSION ?? ''}`)} ${headline}` ); } if (usage) { message.push(linebreak(), `${color.green(commandName)} ${color.bold(usage)}`); } if (tables) { function calculateTablePadding(rows: [string, string][]) { return rows.reduce((val, [first]) => Math.max(val, first.length), 0); } const tableEntries = Object.entries(tables); const padding = Math.max(...tableEntries.map(([, rows]) => calculateTablePadding(rows))); for (const [, tableRows] of tableEntries) { message.push(linebreak(), table(tableRows, { padding })); } } if (description) { message.push(linebreak(), `${description}`); } log(message.join('\n') + '\n'); }