Stop throwing errors, and instead return Results!
npm install neverthrow
Encode failure into your program.
This package contains a Result type that represents either success (Ok) or failure (Err).
For asynchronous tasks, neverthrow offers a ResultAsync class which wraps a Promise and gives you the same level of expressivity and control as a regular Result.
ResultAsync is thenable meaning it behaves exactly like a native Promise ... except you have access to the same methods that Result provides without having to await or .then the promise! Check out the wiki for examples and best practices.
> Need to see real-life examples of how to leverage this package for error handling? See this repo: https://github.com/parlez-vous/server
* Installation
* Recommended: Use eslint-plugin-neverthrow
* Top-Level API
* API Documentation
+ Synchronous API (Result)
- ok
- err
- Result.isOk (method)
- Result.isErr (method)
- Result.map (method)
- Result.mapErr (method)
- Result.unwrapOr (method)
- Result.andThen (method)
- Result.asyncAndThen (method)
- Result.orElse (method)
- Result.match (method)
- Result.asyncMap (method)
- Result.andTee (method)
- Result.orTee (method)
- Result.andThrough (method)
- Result.asyncAndThrough (method)
- Result.fromThrowable (static class method)
- Result.combine (static class method)
- Result.combineWithAllErrors (static class method)
- Result.safeUnwrap()
+ Asynchronous API (ResultAsync)
- okAsync
- errAsync
- ResultAsync.fromThrowable (static class method)
- ResultAsync.fromPromise (static class method)
- ResultAsync.fromSafePromise (static class method)
- ResultAsync.map (method)
- ResultAsync.mapErr (method)
- ResultAsync.unwrapOr (method)
- ResultAsync.andThen (method)
- ResultAsync.orElse (method)
- ResultAsync.match (method)
- ResultAsync.andTee (method)
- ResultAsync.orTee (method)
- ResultAsync.andThrough (method)
- ResultAsync.combine (static class method)
- ResultAsync.combineWithAllErrors (static class method)
- ResultAsync.safeUnwrap()
+ Utilities
- fromThrowable
- fromAsyncThrowable
- fromPromise
- fromSafePromise
- safeTry
+ Testing
* A note on the Package Name
``sh`
> npm install neverthrow
As part of neverthrows bounty program, user mdbetancourt created eslint-plugin-neverthrow to ensure that errors are not gone unhandled.
Install by running:
`sh`
> npm install eslint-plugin-neverthrow
With eslint-plugin-neverthrow, you are forced to consume the result in one of the following three ways:
- Calling .match.unwrapOr
- Calling ._unsafeUnwrap
- Calling
This ensures that you're explicitly handling the error of your Result.
This plugin is essentially a porting of Rust's must-use attribute.
neverthrow exposes the following:
- ok convenience function to create an Ok variant of Resulterr
- convenience function to create an Err variant of ResultOk
- class and typeErr
- class and typeResult
- Type as well as namespace / object from which to call Result.fromThrowable, Result.combine.ResultAsync
- classokAsync
- convenience function to create a ResultAsync containing an Ok type ResulterrAsync
- convenience function to create a ResultAsync containing an Err type Result
`typescript`
import {
ok,
Ok,
err,
Err,
Result,
okAsync,
errAsync,
ResultAsync,
fromAsyncThrowable,
fromThrowable,
fromPromise,
fromSafePromise,
safeTry,
} from 'neverthrow'
---
Check out the wiki for help on how to make the most of neverthrow.
If you find this package useful, please consider sponsoring me or simply buying me a coffee!
---
#### ok
Constructs an Ok variant of Result
Signature:
`typescript`
ok
Example:
`typescript
import { ok } from 'neverthrow'
const myResult = ok({ myData: 'test' }) // instance of Ok
myResult.isOk() // true
myResult.isErr() // false
`
---
#### err
Constructs an Err variant of Result
Signature:
`typescript`
err
Example:
`typescript
import { err } from 'neverthrow'
const myResult = err('Oh noooo') // instance of Err
myResult.isOk() // false
myResult.isErr() // true
`
---
#### Result.isOk (method)
Returns true if the result is an Ok variant
Signature:
`typescript`
isOk(): boolean { ... }
---
#### Result.isErr (method)
Returns true if the result is an Err variant
Signature:
`typescript`
isErr(): boolean { ... }
---
#### Result.map (method)
Maps a Result to Result by applying a function to a contained Ok value, leaving an Err value untouched.
This function can be used to compose the results of two functions.
Signature:
`typescript`
class Result
map(callback: (value: T) => U): Result { ... }
}
Example:
`typescript
import { getLines } from 'imaginary-parser'
// ^ assume getLines has the following signature:
// getLines(str: string): Result
// since the formatting is deemed correct by getLineslinesResult
// then it means that is an Ok
// containing an Array of strings for each line of code
const linesResult = getLines('1\n2\n3\n4\n')
// this Result now has a Array
const newResult = linesResult.map(
(arr: Array
)
newResult.isOk() // true
`
---
#### Result.mapErr (method)
Maps a Result to Result by applying a function to a contained Err value, leaving an Ok value untouched.
This function can be used to pass through a successful result while handling an error.
Signature:
`typescript`
class Result
mapErr
}
Example:
`typescript
import { parseHeaders } from 'imaginary-http-parser'
// imagine that parseHeaders has the following signature:
// parseHeaders(raw: string): Result
const rawHeaders = 'nonsensical gibberish and badly formatted stuff'
const parseResult = parseHeaders(rawHeaders)
parseResult.mapErr(parseError => {
res.status(400).json({
error: parseError
})
})
parseResult.isErr() // true
`
---
#### Result.unwrapOr (method)
Unwrap the Ok value, or return the default if there is an Err
Signature:
`typescript`
class Result
unwrapOr
}
Example:
`typescript
const myResult = err('Oh noooo')
const multiply = (value: number): number => value * 2
const unwrapped: number = myResult.map(multiply).unwrapOr(10)
`
---
#### Result.andThen (method)
Same idea as map above. Except you must return a new Result.
The returned value will be a Result. As of v4.1.0-beta, you are able to return distinct error types (see signature below). Prior to v4.1.0-beta, the error type could not be distinct.
This is useful for when you need to do a subsequent computation using the inner T value, but that computation might fail.
Additionally, andThen is really useful as a tool to flatten a Result into a Result (see example below).
Signature:
`typescriptstring | string
class Result
// Note that the latest version lets you return distinct errors as well.
// If the error types (E and F) are the same (like )string
// then they will be merged into one type ()`
andThen(
callback: (value: T) => Result
): Result { ... }
}
Example 1: Chaining Results
`typescript
import { err, ok } from 'neverthrow'
const sq = (n: number): Result
ok(2)
.andThen(sq)
.andThen(sq) // Ok(16)
ok(2)
.andThen(sq)
.andThen(err) // Err(4)
ok(2)
.andThen(err)
.andThen(sq) // Err(2)
err(3)
.andThen(sq)
.andThen(sq) // Err(3)
`
Example 2: Flattening Nested Results
`typescript
// It's common to have nested Results
const nested = ok(ok(1234))
// notNested is a Ok(1234)
const notNested = nested.andThen((innerResult) => innerResult)
`
---
#### Result.asyncAndThen (method)
Same idea as andThen above, except you must return a new ResultAsync.
The returned value will be a ResultAsync.
Signature:
`typescript`
class Result
asyncAndThen(
callback: (value: T) => ResultAsync
): ResultAsync { ... }
}
---
#### Result.orElse (method)
Takes an Err value and maps it to a Result. This is useful for error recovery.
Signature:
`typescript`
class Result
orElse(
callback: (error: E) => Result
): Result { ... }
}
Example:
`typescript
enum DatabaseError {
PoolExhausted = 'PoolExhausted',
NotFound = 'NotFound',
}
const dbQueryResult: Result
const updatedQueryResult = dbQueryResult.orElse((dbError) =>
dbError === DatabaseError.NotFound
? ok('User does not exist') // error recovery branch: ok() must be called with a value of type string
//
//
// err() can be called with a value of any new type that you want
// it could also be called with the same error value
//
// err(dbError)
: err(500)
)
`
---
#### Result.match (method)
Given 2 functions (one for the Ok variant and one for the Err variant) execute the function that matches the Result variant.
Match callbacks do not necessitate to return a Result, however you can return a Result if you want to.
Signature:
`typescript`
class Result
match(
okCallback: (value: T) => A,
errorCallback: (error: E) => B
): A | B => { ... }
}
match is like chaining map and mapErr, with the distinction that with match both functions must have the same return type.match
The differences between and chaining map and mapErr are that:match
- with both functions must have the same return type Amatch
- unwraps the Result into an A (the match functions' return type)
- This makes no difference if you are performing side effects only
Example:
`typescript
// map/mapErr api
// note that you DON'T have to append mapErr
// after map which means that you are not required to do
// error handling
computationThatMightFail().map(console.log).mapErr(console.error)
// match api
// works exactly the same as above since both callbacks
// only perform side effects,
// except, now you HAVE to do error handling :)
computationThatMightFail().match(console.log, console.error)
// Returning values
const attempt = computationThatMightFail()
.map((str) => str.toUpperCase())
.mapErr((err) => Error: ${err})attempt
// is of type Result
const answer = computationThatMightFail().match(
(str) => str.toUpperCase(),
(err) => Error: ${err}answer
)
// is of type string`
If you don't use the error parameter in your match callback then match is equivalent to chaining map with unwrapOr:`tsanswer
const answer = computationThatMightFail().match(
(str) => str.toUpperCase(),
() => 'ComputationError'
)
// is of type string
const answer = computationThatMightFail()
.map((str) => str.toUpperCase())
.unwrapOr('ComputationError')
`
---
#### Result.asyncMap (method)
Similar to map except for two things:
- the mapping function must return a PromiseResultAsync
- asyncMap returns a
You can then chain the result of asyncMap using the ResultAsync apis (like map, mapErr, andThen, etc.)
Signature:
`typescript`
class Result
asyncMap(
callback: (value: T) => Promise
): ResultAsync { ... }
}
Example:
`typescript
import { parseHeaders } from 'imaginary-http-parser'
// imagine that parseHeaders has the following signature:
// parseHeaders(raw: string): Result
const asyncRes = parseHeaders(rawHeader)
.map(headerKvMap => headerKvMap.Authorization)
.asyncMap(findUserInDatabase)
`
Note that in the above example if parseHeaders returns an Err then .map and .asyncMap will not be invoked, and asyncRes variable will resolve to an Err when turned into a Result using await or .then().
⬆️ Back to top
---
#### Result.andTee (method)
Takes a Result and lets the original Result pass through regardless the result of the passed-in function.
This is a handy way to handle side effects whose failure or success should not affect your main logics such as logging.
Signature:
`typescript`
class Result
andTee(
callback: (value: T) => unknown
): Result
}
Example:
`typescript
import { parseUserInput } from 'imaginary-parser'
import { logUser } from 'imaginary-logger'
import { insertUser } from 'imaginary-database'
// ^ assume parseUserInput, logUser and insertUser have the following signatures:
// parseUserInput(input: RequestData): Result
// logUser(user: User): Result
// insertUser(user: User): ResultAsync
// Note logUser returns void upon success but insertUser takes User type.
const resAsync = parseUserInput(userInput)
.andTee(logUser)
.asyncAndThen(insertUser)
// Note no LogError shows up in the Result type
resAsync.then((res: Result
if(res.isErr()){
console.log("Oops, at least one step failed", res.error)
}
else{
console.log("User input has been parsed and inserted successfully.")
}
}))
`
---
#### Result.orTee (method)
Like andTee for the error track. Takes a Result and lets the Err value pass through regardless the result of the passed-in function.
This is a handy way to handle side effects whose failure or success should not affect your main logics such as logging.
Signature:
`typescript`
class Result
orTee(
callback: (value: E) => unknown
): Result
}
Example:
`typescript
import { parseUserInput } from 'imaginary-parser'
import { logParseError } from 'imaginary-logger'
import { insertUser } from 'imaginary-database'
// ^ assume parseUserInput, logParseError and insertUser have the following signatures:
// parseUserInput(input: RequestData): Result
// logParseError(parseError: ParseError): Result
// insertUser(user: User): ResultAsync
// Note logParseError returns void upon success but insertUser takes User type.
const resAsync = parseUserInput(userInput)
.orTee(logParseError)
.asyncAndThen(insertUser)
// Note no LogError shows up in the Result type
resAsync.then((res: Result
if(res.isErr()){
console.log("Oops, at least one step failed", res.error)
}
else{
console.log("User input has been parsed and inserted successfully.")
}
}))
`
---
#### Result.andThrough (method)
Similar to andTee except for:
- when there is an error from the passed-in function, that error will be passed along.
Signature:
`typescript`
class Result
andThrough
callback: (value: T) => Result
): Result
}
Example:
`typescript
import { parseUserInput } from 'imaginary-parser'
import { validateUser } from 'imaginary-validator'
import { insertUser } from 'imaginary-database'
// ^ assume parseUseInput, validateUser and insertUser have the following signatures:
// parseUserInput(input: RequestData): Result
// validateUser(user: User): Result
// insertUser(user: User): ResultAsync
// Note validateUser returns void upon success but insertUser takes User type.
const resAsync = parseUserInput(userInput)
.andThrough(validateUser)
.asyncAndThen(insertUser)
resAsync.then((res: Result
if(res.isErr()){
console.log("Oops, at least one step failed", res.error)
}
else{
console.log("User input has been parsed, validated, inserted successfully.")
}
}))
`
⬆️ Back to top
---
#### Result.asyncAndThrough (method)
Similar to andThrough except you must return a ResultAsync.
You can then chain the result of asyncAndThrough using the ResultAsync apis (like map, mapErr, andThen, etc.)
Signature:
`typescript
import { parseUserInput } from 'imaginary-parser'
import { insertUser } from 'imaginary-database'
import { sendNotification } from 'imaginary-service'
// ^ assume parseUserInput, insertUser and sendNotification have the following signatures:
// parseUserInput(input: RequestData): Result
// insertUser(user: User): ResultAsync
// sendNotification(user: User): ResultAsync
// Note insertUser returns void upon success but sendNotification takes User type.
const resAsync = parseUserInput(userInput)
.asyncAndThrough(insertUser)
.andThen(sendNotification)
resAsync.then((res: Result
if(res.isErr()){
console.log("Oops, at least one step failed", res.error)
}
else{
console.log("User has been parsed, inserted and notified successfully.")
}
}))
`
⬆️ Back to top
---
#### Result.fromThrowable (static class method)
> Although Result is not an actual JS class, the way that fromThrowable has been implemented requires that you call fromThrowable as though it were a static method on Result. See examples below.
The JavaScript community has agreed on the convention of throwing exceptions.
As such, when interfacing with third party libraries it's imperative that you
wrap third-party code in try / catch blocks.
This function will create a new function that returns an Err when the original
function throws.
It is not possible to know the types of the errors thrown in the original
function, therefore it is recommended to use the second argument errorFn to
map what is thrown to a known type.
Example:
`typescript
import { Result } from 'neverthrow'
type ParseError = { message: string }
const toParseError = (): ParseError => ({ message: "Parse Error" })
const safeJsonParse = Result.fromThrowable(JSON.parse, toParseError)
// the function can now be used safely, if the function throws, the result will be an Err
const res = safeJsonParse("{");
`
---
#### Result.combine (static class method)
> Although Result is not an actual JS class, the way that combine has been implemented requires that you call combine as though it were a static method on Result. See examples below.
Combine lists of Results.
If you're familiar with Promise.all, the combine function works conceptually the same.
combine works on both heterogeneous and homogeneous lists. This means that you can have lists that contain different kinds of Results and still be able to combine them. Note that you cannot combine lists that contain both Results and ResultAsyncs.
The combine function takes a list of results and returns a single result. If all the results in the list are Ok, then the return value will be a Ok containing a list of all the individual Ok values.
If just one of the results in the list is an Err then the combine function returns that Err value (it short circuits and returns the first Err that it finds).
Formally speaking:
`typescript
// homogeneous lists
function combine
// heterogeneous lists
function combine
function combine
function combine
// ... etc etc ad infinitum
`
Example:
`typescript
const resultList: Result
[ok(1), ok(2)]
const combinedList: Result
Result.combine(resultList)
`
Example with tuples:
`typescript
/* @example tuple(1, 2, 3) === [1, 2, 3] // with type [number, number, number] /
const tuple =
const resultTuple: [Result
tuple(ok('a'), ok('b'))
const combinedTuple: Result<[string, string], unknown> =
Result.combine(resultTuple)
`
---
#### Result.combineWithAllErrors (static class method)
> Although Result is not an actual JS class, the way that combineWithAllErrors has been implemented requires that you call combineWithAllErrors as though it were a static method on Result. See examples below.
Like combine but without short-circuiting. Instead of just the first error value, you get a list of all error values of the input result list.
If only some results fail, the new combined error list will only contain the error value of the failed results, meaning that there is no guarantee of the length of the new error list.
Function signature:
`typescript
// homogeneous lists
function combineWithAllErrors
// heterogeneous lists
function combineWithAllErrors
function combineWithAllErrors
function combineWithAllErrors
// ... etc etc ad infinitum
`
Example usage:
`typescript
const resultList: Result
ok(123),
err('boooom!'),
ok(456),
err('ahhhhh!'),
]
const result = Result.combineWithAllErrors(resultList)
// result is Err(['boooom!', 'ahhhhh!'])
`
#### Result.safeUnwrap()
Deprecated. You don't need to use this method anymore.
Allows for unwrapping a Result or returning an Err implicitly, thereby reducing boilerplate.
---
#### okAsync
Constructs an Ok variant of ResultAsync
Signature:
`typescript`
okAsync
Example:
`typescript
import { okAsync } from 'neverthrow'
const myResultAsync = okAsync({ myData: 'test' }) // instance of ResultAsync
const myResult = await myResultAsync // instance of Ok
myResult.isOk() // true
myResult.isErr() // false
`
---
#### errAsync
Constructs an Err variant of ResultAsync
Signature:
`typescript`
errAsync
Example:
`typescript
import { errAsync } from 'neverthrow'
const myResultAsync = errAsync('Oh nooo') // instance of ResultAsync
const myResult = await myResultAsync // instance of Err
myResult.isOk() // false
myResult.isErr() // true
`
---
#### ResultAsync.fromThrowable (static class method)
Similar to Result.fromThrowable, but for functions that return a Promise.
Example:
`typescript
import { ResultAsync } from 'neverthrow'
import { insertIntoDb } from 'imaginary-database'
// insertIntoDb(user: User): Promise
const insertUser = ResultAsync.fromThrowable(insertIntoDb, () => new Error('Database error'))
// res has a type of (user: User) => ResultAsync`
Note that this can be safer than using ResultAsync.fromPromise with
the result of a function call, because not all functions that return a Promise are async, and thus they can throwPromise
errors synchronously rather than returning a rejected . For example:
`typescript
// NOT SAFE !!
import { ResultAsync } from 'neverthrow'
import { db } from 'imaginary-database'
// db.insert
const insertUser = (user: User): Promise
if (!user.id) {
// this throws synchronously!
throw new TypeError('missing user id')
}
return db.insert('users', user)
}
// this will throw, NOT return a ResultAsync`
const res = ResultAsync.fromPromise(insertIntoDb(myUser), () => new Error('Database error'))
---
#### ResultAsync.fromPromise (static class method)
Transforms a PromiseLike (that may throw) into a ResultAsync.
The second argument handles the rejection case of the promise and maps the error from unknown into some type E.
Signature:
`typescript`
// fromPromise is a static class method
// also available as a standalone function
// import { fromPromise } from 'neverthrow'
ResultAsync.fromPromise
promise: PromiseLike
errorHandler: (unknownError: unknown) => E)
): ResultAsync
If you are working with PromiseLike objects that you know for a fact will not throw, then use fromSafePromise in order to avoid having to pass a redundant errorHandler argument.
Example:
`typescript
import { ResultAsync } from 'neverthrow'
import { insertIntoDb } from 'imaginary-database'
// insertIntoDb(user: User): Promise
const res = ResultAsync.fromPromise(insertIntoDb(myUser), () => new Error('Database error'))
// res has a type of ResultAsync`
---
#### ResultAsync.fromSafePromise (static class method)
Same as ResultAsync.fromPromise except that it does not handle the rejection of the promise. Ensure you know what you're doing, otherwise a thrown exception within this promise will cause ResultAsync to reject, instead of resolve to a Result.
Signature:
`typescript`
// fromPromise is a static class method
// also available as a standalone function
// import { fromPromise } from 'neverthrow'
ResultAsync.fromSafePromise
promise: PromiseLike
): ResultAsync
Example:
`typescript
import { RouteError } from 'routes/error'
// simulate slow routes in an http server that works in a Result / ResultAsync context
// Adopted from https://github.com/parlez-vous/server/blob/2496bacf55a2acbebc30631b5562f34272794d76/src/routes/common/signup.ts
export const slowDown =
ResultAsync.fromSafePromise
new Promise((resolve) => {
setTimeout(() => {
resolve(value)
}, ms)
})
)
export const signupHandler = route
decode(userSignupDecoder, req.body, 'Invalid request body').map((parsed) => {
return createUser(parsed)
.andThen(slowDown(3000)) // slowdown by 3 seconds
.andThen(sessionManager.createSession)
.map(({ sessionToken, admin }) => AppData.init(admin, sessionToken))
})
)
`
---
#### ResultAsync.map (method)
Maps a ResultAsync to ResultAsync by applying a function to a contained Ok value, leaving an Err value untouched.
The applied function can be synchronous or asynchronous (returning a Promise) with no impact to the return type.
This function can be used to compose the results of two functions.
Signature:
`typescript`
class ResultAsync
map(
callback: (value: T) => U | Promise
): ResultAsync { ... }
}
Example:
`typescript
import { findUsersIn } from 'imaginary-database'
// ^ assume findUsersIn has the following signature:
// findUsersIn(country: string): ResultAsync
const usersInCanada = findUsersIn("Canada")
// Let's assume we only need their names
const namesInCanada = usersInCanada.map((users: Array
// namesInCanada is of type ResultAsync
// We can extract the Result using .then() or await
namesInCanada.then((namesResult: Result
if(namesResult.isErr()){
console.log("Couldn't get the users from the database", namesResult.error)
}
else{
console.log("Users in Canada are named: " + namesResult.value.join(','))
}
})
`
---
#### ResultAsync.mapErr (method)
Maps a ResultAsync to ResultAsync by applying a function to a contained Err value, leaving an Ok value untouched.
The applied function can be synchronous or asynchronous (returning a Promise) with no impact to the return type.
This function can be used to pass through a successful result while handling an error.
Signature:
`typescript`
class ResultAsync
mapErr
callback: (error: E) => F | Promise
): ResultAsync
}
Example:
`typescript
import { findUsersIn } from 'imaginary-database'
// ^ assume findUsersIn has the following signature:
// findUsersIn(country: string): ResultAsync
// Let's say we need to low-level errors from findUsersIn to be more readable
const usersInCanada = findUsersIn("Canada").mapErr((error: Error) => {
// The only error we want to pass to the user is "Unknown country"
if(error.message === "Unknown country"){
return error.message
}
// All other errors will be labelled as a system error
return "System error, please contact an administrator."
})
// usersInCanada is of type ResultAsync
usersInCanada.then((usersResult: Result
if(usersResult.isErr()){
res.status(400).json({
error: usersResult.error
})
}
else{
res.status(200).json({
users: usersResult.value
})
}
})
`
---
#### ResultAsync.unwrapOr (method)
Unwrap the Ok value, or return the default if there is an Err. Result.unwrapOr
Works just like but returns a Promise instead of T.
Signature:
`typescript`
class ResultAsync
unwrapOr
}
Example:
`typescript`
const unwrapped: number = await errAsync(0).unwrapOr(10)
// unwrapped = 10
---
#### ResultAsync.andThen (method)
Same idea as map above. Except the applied function must return a Result or ResultAsync.
ResultAsync.andThen always returns a ResultAsync no matter the return type of the applied function.
This is useful for when you need to do a subsequent computation using the inner T value, but that computation might fail.
andThen is really useful as a tool to flatten a ResultAsync into a ResultAsync (see example below).
Signature:
`typescriptstring | string
// Note that the latest version (v4.1.0-beta) lets you return distinct errors as well.
// If the error types (E and F) are the same (like )string
// then they will be merged into one type ()
class ResultAsync
andThen(
callback: (value: T) => Result | ResultAsync
): ResultAsync { ... }
}
`
Example
`typescript
import { validateUser } from 'imaginary-validator'
import { insertUser } from 'imaginary-database'
import { sendNotification } from 'imaginary-service'
// ^ assume validateUser, insertUser and sendNotification have the following signatures:
// validateUser(user: User): Result
// insertUser(user): ResultAsync
// sendNotification(user): ResultAsync
const resAsync = validateUser(user)
.andThen(insertUser)
.andThen(sendNotification)
// resAsync is a ResultAsync
resAsync.then((res: Result
if(res.isErr()){
console.log("Oops, at least one step failed", res.error)
}
else{
console.log("User has been validated, inserted and notified successfully.")
}
})
`
---
#### ResultAsync.orElse (method)
Takes an Err value and maps it to a ResultAsync. This is useful for error recovery.
Signature:
`typescript`
class ResultAsync
orElse(
callback: (error: E) => Result | ResultAsync
): ResultAsync { ... }
}
---
#### ResultAsync.match (method)
Given 2 functions (one for the Ok variant and one for the Err variant) execute the function that matches the ResultAsync variant.
The difference with Result.match is that it always returns a Promise because of the asynchronous nature of the ResultAsync.
Signature:
`typescript`
class ResultAsync
match(
okCallback: (value: T) => A,
errorCallback: (error: E) => B
): Promise => { ... }
}
Example:
`typescript
import { validateUser } from 'imaginary-validator'
import { insertUser } from 'imaginary-database'
// ^ assume validateUser and insertUser have the following signatures:
// validateUser(user: User): Result
// insertUser(user): ResultAsync
// Handle both cases at the end of the chain using match
const resultMessage = await validateUser(user)
.andThen(insertUser)
.match(
(user: User) => User ${user.name} has been successfully created,User could not be created because ${error.message}
(error: Error) =>
)
// resultMessage is a string
`
---
#### ResultAsync.andTee (method)
Takes a ResultAsync and lets the original ResultAsync pass through regardless
the result of the passed-in function.
This is a handy way to handle side effects whose failure or success should not affect your main logics such as logging.
Signature:
`typescript`
class ResultAsync
andTee(
callback: (value: T) => unknown
): ResultAsync
}
Example:
`typescript
import { insertUser } from 'imaginary-database'
import { logUser } from 'imaginary-logger'
import { sendNotification } from 'imaginary-service'
// ^ assume insertUser, logUser and sendNotification have the following signatures:
// insertUser(user: User): ResultAsync
// logUser(user: User): Result
// sendNotification(user: User): ResultAsync
// Note logUser returns void on success but sendNotification takes User type.
const resAsync = insertUser(user)
.andTee(logUser)
.andThen(sendNotification)
// Note there is no LogError in the types below
resAsync.then((res: Result
if(res.isErr()){
console.log("Oops, at least one step failed", res.error)
}
else{
console.log("User has been inserted and notified successfully.")
}
}))
`
---
#### ResultAsync.orTee (method)
Like andTee for the error track. Takes a ResultAsync and lets the original Err value pass through regardless
the result of the passed-in function.
This is a handy way to handle side effects whose failure or success should not affect your main logics such as logging.
Signature:
`typescript`
class ResultAsync
orTee(
callback: (value: E) => unknown
): ResultAsync
}
Example:
`typescript
import { insertUser } from 'imaginary-database'
import { logInsertError } from 'imaginary-logger'
import { sendNotification } from 'imaginary-service'
// ^ assume insertUser, logInsertError and sendNotification have the following signatures:
// insertUser(user: User): ResultAsync
// logInsertError(insertError: InsertError): Result
// sendNotification(user: User): ResultAsync
// Note logInsertError returns void on success but sendNotification takes User type.
const resAsync = insertUser(user)
.orTee(logUser)
.andThen(sendNotification)
// Note there is no LogError in the types below
resAsync.then((res: Result
if(res.isErr()){
console.log("Oops, at least one step failed", res.error)
}
else{
console.log("User has been inserted and notified successfully.")
}
}))
`
---
#### ResultAsync.andThrough (method)
Similar to andTee except for:
- when there is an error from the passed-in function, that error will be passed along.
Signature:
`typescript`
class ResultAsync
andThrough
callback: (value: T) => Result
): ResultAsync
}
Example:
`typescript
import { buildUser } from 'imaginary-builder'
import { insertUser } from 'imaginary-database'
import { sendNotification } from 'imaginary-service'
// ^ assume buildUser, insertUser and sendNotification have the following signatures:
// buildUser(userRaw: UserRaw): ResultAsync
// insertUser(user: User): ResultAsync
// sendNotification(user: User): ResultAsync
// Note insertUser returns void upon success but sendNotification takes User type.
const resAsync = buildUser(userRaw)
.andThrough(insertUser)
.andThen(sendNotification)
resAsync.then((res: Result
if(res.isErr()){
console.log("Oops, at least one step failed", res.error)
}
else{
console.log("User data has been built, inserted and notified successfully.")
}
}))
`
---
#### ResultAsync.combine (static class method)
Combine lists of ResultAsyncs.
If you're familiar with Promise.all, the combine function works conceptually the same.
combine works on both heterogeneous and homogeneous lists. This means that you can have lists that contain different kinds of ResultAsyncs and still be able to combine them. Note that you cannot combine lists that contain both Results and ResultAsyncs.
The combine function takes a list of results and returns a single result. If all the results in the list are Ok, then the return value will be a Ok containing a list of all the individual Ok values.
If just one of the results in the list is an Err then the combine function returns that Err value (it short circuits and returns the first Err that it finds).
Formally speaking:
`typescript
// homogeneous lists
function combine
// heterogeneous lists
function combine
function combine
function combine
// ... etc etc ad infinitum
`
Example:
`typescript
const resultList: ResultAsync
[okAsync(1), okAsync(2)]
const combinedList: ResultAsync
ResultAsync.combine(resultList)
`
Example with tuples:
`typescript
/* @example tuple(1, 2, 3) === [1, 2, 3] // with type [number, number, number] /
const tuple =
const resultTuple: [ResultAsync
tuple(okAsync('a'), okAsync('b'))
const combinedTuple: ResultAsync<[string, string], unknown> =
ResultAsync.combine(resultTuple)
`
⬆️ Back to top
---
#### ResultAsync.combineWithAllErrors (static class method)
Like combine but without short-circuiting. Instead of just the first error value, you get a list of all error values of the input result list.
If only some results fail, the new combined error list will only contain the error value of the failed results, meaning that there is no guarantee of the length of the new error list.
Function signature:
`typescript
// homogeneous lists
function combineWithAllErrors
// heterogeneous lists
function combineWithAllErrors
function combineWithAllErrors
function combineWithAllErrors
// ... etc etc ad infinitum
`
Example usage:
`typescript
const resultList: ResultAsync
okAsync(123),
errAsync('boooom!'),
okAsync(456),
errAsync('ahhhhh!'),
]
const result = ResultAsync.combineWithAllErrors(resultList)
// result is Err(['boooom!', 'ahhhhh!'])
`
#### ResultAsync.safeUnwrap()
Deprecated. You don't need to use this method anymore.
Allows for unwrapping a Result or returning an Err implicitly, thereby reducing boilerplate.
---
#### fromThrowable
Top level export of Result.fromThrowable.
Please find documentation at Result.fromThrowable
#### fromAsyncThrowable
Top level export of ResultAsync.fromThrowable.
Please find documentation at ResultAsync.fromThrowable
#### fromPromise
Top level export of ResultAsync.fromPromise.
Please find documentation at ResultAsync.fromPromise
#### fromSafePromise
Top level export of ResultAsync.fromSafePromise.
Please find documentation at ResultAsync.fromSafePromise
#### safeTry
Used to implicitly return errors and reduce boilerplate.
Let's say we are writing a function that returns a Result, and in that function we call some functions which also return Results and we check those results to see whether we should keep going or abort. Usually, we will write like the following.`typescript
declare function mayFail1(): Result
declare function mayFail2(): Result
function myFunc(): Result
// We have to define a constant to hold the result to check and unwrap its value.
const result1 = mayFail1();
if (result1.isErr()) {
return err(aborted by an error from 1st function, ${result1.error});
}
const value1 = result1.value
// Again, we need to define a constant and then check and unwrap.
const result2 = mayFail2();
if (result2.isErr()) {
return err(aborted by an error from 2nd function, ${result2.error});
}
const value2 = result2.value
// And finally we return what we want to calculate
return ok(value1 + value2);
}
`Ok
Basically, we need to define a constant for each result to check whether it's a and read its .value or .error.
With safeTry, we can state 'Return here if its an Err, otherwise unwrap it here and keep going.' in just one expression.`typescript
declare function mayFail1(): Result
declare function mayFail2(): Result
function myFunc(): Result
return safeTry
return ok(
// If the result of mayFail1().mapErr() is an Err, the evaluation issafeTry
// aborted here and the enclosing block is evaluated to that Err.(yield* ...)
// Otherwise, this is evaluated to its .value.aborted by an error from 1st function, ${e}
(yield* mayFail1()
.mapErr(e => ))aborted by an error from 2nd function, ${e}
+
// The same as above.
(yield* mayFail2()
.mapErr(e => ))`
)
})
}
To use safeTry, the points are as follows.yield
Wrap the entire block in a generator function
In that block, you can use to state 'Return if it's an Err, otherwise evaluate to its .value'safeTry
* Pass the generator function to
You can also use async generator function to pass an async block to safeTry.`typescript
// You can use either Promise
declare function mayFail1(): Promise
declare function mayFail2(): ResultAsync
function myFunc(): Promise
return safeTry
return ok(
// You have to await if the expression is Promise
(yield* (await mayFail1())
.mapErr(e => aborted by an error from 1st function, ${e}))safeUnwrap
+
// You can call directly if its ResultAsyncaborted by an error from 2nd function, ${e}
(yield* mayFail2()
.mapErr(e => ))`
)
})
}
For more information, see https://github.com/supermacro/neverthrow/pull/448 and https://github.com/supermacro/neverthrow/issues/444
---
Result instances have two unsafe methods, aptly called _unsafeUnwrap and _unsafeUnwrapErr which should only be used in a test environment.
_unsafeUnwrap takes a Result and returns a T when the result is an Ok, otherwise it throws a custom object.
_unsafeUnwrapErr takes a Result and returns a E when the result is an Err, otherwise it throws a custom object.
That way you can do something like:
`typescript`
expect(myResult._unsafeUnwrap()).toBe(someExpectation)
However, do note that Result instances are comparable. So you don't necessarily need to unwrap them in order to assert expectations in your tests. So you could also do something like this:
`typescript
import { ok } from 'neverthrow'
// ...
expect(callSomeFunctionThatReturnsAResult("with", "some", "args")).toEqual(ok(someExpectation));
`
By default, the thrown value does not contain a stack trace. This is because stack trace generation makes error messages in Jest harder to understand. If you want stack traces to be generated, call _unsafeUnwrap and / or _unsafeUnwrapErr with a config object:
`typescript
_unsafeUnwrapErr({
withStackTrace: true,
})
// ^ Now the error object will have a .stack property containing the current stack`
---
If you find this package useful, please consider sponsoring me or simply buying me a coffee!
---
Although the package is called neverthrow, please don't take this literally. I am simply encouraging the developer to think a bit more about the ergonomics and usage of whatever software they are writing.
Throwing and catching is very similar to using goto statements - in other words; it makes reasoning about your programs harder. Secondly, by using throw you make the assumption that the caller of your function is implementing catch. This is a known source of errors. Example: One dev throw`s and another dev uses the function without prior knowledge that the function will throw. Thus, and edge case has been left unhandled and now you have unhappy users, bosses, cats, etc.
With all that said, there are definitely good use cases for throwing in your program. But much less than you might think.
The neverthrow project is available as open source under the terms of the MIT license.