JSPM

@byteslice/result

0.2.2
  • ESM via JSPM
  • ES Module Entrypoint
  • Export Map
  • Keywords
  • License
  • Repository URL
  • TypeScript Types
  • README
  • Created
  • Published
  • Downloads 4267
  • Score
    100M100P100Q128088F

Package Exports

  • @byteslice/result

Readme

@byteslice/result

A lightweight TypeScript utility for wrapping operations in a structured Result type, mitigating the need for exception-handling boilerplate.

This package enables developers to clearly represent both success and failure states, ensuring a predictable and type-safe approach to managing operation results.

🍕 Built by the team at ByteSlice.

Table of Contents

Installation

npm install @byteslice/result
# or
yarn add @byteslice/result
# or
pnpm add @byteslice/result
# or
bun add @byteslice/result

Motivation

To fully understand the purpose and application of this package, it's essential to provide some context.

Errors vs. Exceptions

Exceptions are particularly useful in scenarios where a program must terminate quickly in response to serious problems or unforeseen circumstances. As the term suggests, they signify exceptions that arise when standard operations are disrupted by the unexpected.

In TypeScript, any value can be "thrown" as an exception: errors, strings, numbers, you name it. This is why "caught" exceptions are type unknown.

Errors, on the other hand, are values that represent anticipated—albeit undesired—behavior. They denote an error state and typically contain a descriptive message that explains the nature of the problem.

Function Signatures

TypeScript—while providing excellent type safety—lacks a built-in mechanism to indicate when a function might throw an exception.

Consider the following function. While the implementation indicates that an exception could be thrown, the type signature fails to convey this information.

function fetchUser(id: string): User {
  throw new Error("Oh, no! Mr. Bill!")
}

This becomes especially problematic if the developer is not familiar with the underlying implementation. They may need to resort to defensive try/catch blocks or risk having exceptions propagate unexpectedly.

Success and Failure States

Every operation can lead to one of two possible outcomes: success or failure. Establishing a standard pattern to represent both of these potential states is crucial.

The @byteslice/result package provides this pattern through a Result type that effectively captures these two distinct states.

Instead of an operation simply returning a value (indicating success) or throwing an exception (indicating failure), it can return a type-safe Result that represents either outcome.

Overview

@byteslice/result provides two key exports:

  1. Result<S, F = Error> – A discriminated union type representing either:

    • Success: { data: S }
    • Failure: { failure: F }
  2. withResult – An asynchronous function wrapper that:

    • Executes a provided operation.
    • Catches any thrown exception.
    • Returns a success or failure object rather than throwing.

This pattern is particularly helpful when you want to avoid using try/catch directly in your code, or if you need a standardized way to capture failure details.

Usage

Basic Example

import { withResult } from '@byteslice/result';

async function fetchData(): Promise<string> {
  // Imagine this might fail
  return "Data fetched successfully!";
}

async function main() {
  const result = await withResult(
    () => fetchData(),
    (error) => error // Pass the error through as-is (default)
  );

  if (!result.failure) {
    console.log('Success:', result.data);
  } else {
    console.error('Failure:', result.failure);
  }
}

main();

In this example:

  • The fetchData function may throw.
  • withResult catches any thrown exceptions and calls onError, returning a failure object if something goes wrong.
  • The caller only needs to check if result contains a data or a failure property.

Custom Failure Types

By default, the failure type (F) is Error, but you can define your own custom failure structure:

import { withResult, Result } from '@byteslice/result';

interface MyCustomFailure {
  type: 'NETWORK_ERROR' | 'VALIDATION_ERROR';
  message: string;
}

async function fetchUser(): Promise<Result<{ name: string }, MyCustomFailure>> {
  return withResult(
    async () => {
      // Potentially throwing code
      return { name: 'Alice' };
    },
    (error) => ({
      type: 'NETWORK_ERROR',
      message: error.message
    })
  );
}

async function main() {
  const result = await fetchUser();

  if (!result.failure) {
    console.log('User:', result.data.name);
  } else {
    console.log('Failed with:', result.failure);
  }
}

Using onException Hook

You can optionally provide an onException hook to transform or log the original exception before onError is called:

import { withResult } from '@byteslice/result';

async function riskyOperation() {
  throw new Error("Something unexpected occurred!");
}

async function main() {
  const result = await withResult(
    () => riskyOperation(),
    (err) => ({ type: 'CUSTOM_FAILURE', message: err.message }),
    {
      onException: (ex) => {
        // Log, transform, or capture `ex` before it's passed to onError
        console.error('Caught exception:', ex);
        return new Error('Wrapped exception details');
      }
    }
  );

  if (!result.failure) {
    console.log('Operation succeeded:', result.data);
  } else {
    console.warn('Operation failed:', result.failure);
  }
}

main();

In this scenario:

  • onException is called first, receiving the thrown value (ex), and returns an Error.
  • That Error is then passed to your onError function.

Contributing

Please see CONTRIBUTING.md for details.

License

MIT © ByteSlice See the LICENSE file for more details.