cli

Errors

Errors can and will happen, and handling them gracefully is essential to provide a first-class developer experience.

When designing code, you can leverage tools to minimize the likelihood of errors happening. For example, you can leverage Typescript to prevent invalid states from entering the system. It requires additional time to figure out how to model the state, but the investment pays off significantly. If an external state enters the system, for example, a project in the file system, validate it and return early if the shape is invalid. It is similar to what web servers do with incoming HTTP requests. We also recommend modeling business logic as a combination of pure functions that are predictable. Functions with the freedom to mutate input state might get mutations introduced in the future that breaks the contract with callers of the function.

Moreover, you should think about scenarios other than the happy path and build for them. There are infinite of those, especially considering that the CLI runs in environments we don’t control, so it’s essential that you focus on the most obvious ones and don’t get too obsessed with trying to anticipate every possible error. Excellent error management is an iterative process using unhandled errors as input to inform the improvements.

Aborting the execution using errors

If your logic needs to abort the execution, throw instantiating any of the errors exported by @shopify/cli-kit: You can pass an array of TokenItems as the second param, the tryMessage, to format it properly.

import {
  AbortError,
  AbortSilentError,
  BugError,
} from "@shopify/cli-kit/node/error"

throw new AbortError(
  "The project doesn't exist",
  ["Make sure the command", {command: "npm install"}, "is executed from a project's directory"]
)

You can also pass some next steps to the error constructor to help the user recover from the error. The next steps are displayed after the error message.

For example

throw new AbortError(
  "The project doesn't exist",
  undefined,
  [
    "Make sure the command is executed from a project's directory",
    "Run the command again",
  ]
)

Please, don’t use the global process.exit and process.abort APIs. Also, don’t try {} catch {} abort errors. If you need to communicate the failure of an operation to the caller (e.g., a 5xx HTTP response), use the result type from the following section.

Report a result from a function

There are scenarios where a function needs to inform the caller about the success or failure of the operation. For that, @shopify/cli-kit provides a result utility:

import { FatalError } from "@shopify/cli-kit/node/error"
import {err, ok, Result} from '@shopify/cli-kit/node/result'

class ActionError extends FatalError {}

function action({success}: {success: boolean}): Result<string, ActionError> {
  if (success) {
    return ok("ok")
  } else {
    return err(new ActionError("err"))
  }
}

// OK result
let result = action({success: true})
result.isErr() // false
result.valueOrBug() // ok
result.mapError((error) => new FatalError("other error"))

// Error result
let result = action({success: false})
result.isErr() // true
result.valueOrBug() // throws!
result.mapError((error) => new FatalError("other error"))