UNPKG

convex

Version:

Client for the Convex Cloud

186 lines (168 loc) 5.91 kB
import * as Sentry from "@sentry/node"; import chalk from "chalk"; import ora, { Ora } from "ora"; import { Filesystem, nodeFs } from "./fs.js"; import { format } from "util"; // How the error should be handled when running `npx convex dev`. export type ErrorType = // The error was likely caused by the state of the developer's local // file system (e.g. `tsc` fails due to a syntax error). The `convex dev` // command will then print out the error and wait for the file to change before // retrying. | "invalid filesystem data" // The error was caused by either the local state (ie schema.ts content) // or the state of the db (ie documents not matching the new schema). // The `convex dev` command will wait for either file OR table data change // to retry (if a table name is specified as the value in this Object). | { "invalid filesystem or db data": string | null; } // The error was caused by either the local state (ie schema.ts content) // or the state of the deployment environment variables. // The `convex dev` command will wait for either file OR env var change // before retrying. | "invalid filesystem or env vars" // The error was some transient issue (e.g. a network // error). This will then cause a retry after an exponential backoff. | "transient" // This error is truly permanent. Exit `npx convex dev` because the // developer will need to take a manual commandline action. | "fatal"; export interface Context { fs: Filesystem; deprecationMessagePrinted: boolean; spinner: Ora | undefined; // Reports to Sentry and either throws FatalError or exits the process. // Prints the `printedMessage` if provided crash(args: { exitCode: number; errorType: ErrorType; errForSentry?: any; printedMessage: string | null; }): Promise<never>; } async function flushAndExit(exitCode: number, err?: any) { if (err) { Sentry.captureException(err); } await Sentry.close(); // eslint-disable-next-line no-restricted-syntax return process.exit(exitCode); } export type OneoffCtx = Context & { // Generally `ctx.crash` is better to use since it handles printing a message // for the user, and then calls this. // // This function reports to Sentry + exits the process, but does not handle // printing a message for the user. flushAndExit: (exitCode: number, err?: any) => Promise<never>; }; export const oneoffContext: OneoffCtx = { fs: nodeFs, deprecationMessagePrinted: false, spinner: undefined, async crash(args: { exitCode: number; errorType?: ErrorType; errForSentry?: any; printedMessage: string | null; }) { if (args.printedMessage !== null) { logFailure(oneoffContext, args.printedMessage); } return await flushAndExit(args.exitCode, args.errForSentry); }, flushAndExit, }; // console.error before it started being red by default in Node v20 function logToStderr(...args: unknown[]) { process.stderr.write(`${format(...args)}\n`); } // Handles clearing spinner so that it doesn't get messed up export function logError(ctx: Context, message: string) { ctx.spinner?.clear(); logToStderr(message); } // Handles clearing spinner so that it doesn't get messed up export function logWarning(ctx: Context, message: string) { ctx.spinner?.clear(); logToStderr(message); } // Handles clearing spinner so that it doesn't get messed up export function logMessage(ctx: Context, ...logged: any) { ctx.spinner?.clear(); logToStderr(...logged); } // For the rare case writing output to stdout. Status and error messages // (logMesage, logWarning, etc.) should be written to stderr. export function logOutput(ctx: Context, ...logged: any) { ctx.spinner?.clear(); console.log(...logged); } export function logVerbose(ctx: Context, ...logged: any) { if (process.env.CONVEX_VERBOSE) { logMessage(ctx, `[verbose] ${new Date().toISOString()}`, ...logged); } } // Start a spinner. // To change its message use changeSpinner. // To print warnings/erros while it's running use logError or logWarning. // To stop it due to an error use logFailure. // To stop it due to success use logFinishedStep. export function showSpinner(ctx: Context, message: string) { ctx.spinner?.stop(); ctx.spinner = ora({ // Add newline to prevent clobbering when a message // we can't pipe through `logMessage` et al gets printed text: message + "\n", stream: process.stderr, // hideCursor: true doesn't work with `tsx`. // see https://github.com/tapjs/signal-exit/issues/49#issuecomment-1459408082 // See CX-6822 for an issue to bring back cursor hiding, probably by upgrading libraries. hideCursor: process.env.CONVEX_RUNNING_LIVE_IN_MONOREPO ? false : true, }).start(); } export function changeSpinner(ctx: Context, message: string) { if (ctx.spinner) { // Add newline to prevent clobbering ctx.spinner.text = message + "\n"; } else { logToStderr(message); } } export function logFailure(ctx: Context, message: string) { if (ctx.spinner) { ctx.spinner.fail(message); ctx.spinner = undefined; } else { logToStderr(`${chalk.red(`✖`)} ${message}`); } } // Stops and removes spinner if one is active export function logFinishedStep(ctx: Context, message: string) { if (ctx.spinner) { ctx.spinner.succeed(message); ctx.spinner = undefined; } else { logToStderr(`${chalk.green(`✔`)} ${message}`); } } export function stopSpinner(ctx: Context) { if (ctx.spinner) { ctx.spinner.stop(); ctx.spinner = undefined; } } // Only shows the spinner if the async `fn` takes longer than `delayMs` export async function showSpinnerIfSlow( ctx: Context, message: string, delayMs: number, fn: () => Promise<any>, ) { const timeout = setTimeout(() => { showSpinner(ctx, message); }, delayMs); await fn(); clearTimeout(timeout); }