Last active
April 4, 2025 01:43
-
Star
(630)
You must be signed in to star a gist -
Fork
(101)
You must be signed in to fork a gist
-
-
Save t3dotgg/a486c4ae66d32bf17c09c73609dacc5b to your computer and use it in GitHub Desktop.
Theo's preferred way of handling try/catch in TypeScript
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
// Types for the result object with discriminated union | |
type Success<T> = { | |
data: T; | |
error: null; | |
}; | |
type Failure<E> = { | |
data: null; | |
error: E; | |
}; | |
type Result<T, E = Error> = Success<T> | Failure<E>; | |
// Main wrapper function | |
export async function tryCatch<T, E = Error>( | |
promise: Promise<T>, | |
): Promise<Result<T, E>> { | |
try { | |
const data = await promise; | |
return { data, error: null }; | |
} catch (error) { | |
return { data: null, error: error as E }; | |
} | |
} |
-
i dont think you understand when your code is type-safe. throwing errors will always throw the same error and everything regardless of its type, and the result will be correctly-typed if you checked for errors beforehand.
-
you still have to explicit state that you want a stack trace in your error instead of the solution doing it for you, that's bad dev-ex. when i am writing my app, i dont want to deal with something as trivial as error creation and handling.
- How do you know if a function throws? With this you always know the result may throw by returning an error (plus it's way faster to not capture stack trace and allocate an Error instance).
- With this you don't need stack traces. You can always throw if you want to for unrecoverable errors (you should check out
neverthrow
).
I think just like promises , it might be useful to try catch normal callback functions
export function tryCatchSync<T, E = Error>(callback: () => T): Result<T, E> {
try {
const data = callback();
return { data, error: null };
} catch (error) {
return { data: null, error: error as E };
}
}
So now you can use it like
const {data,error} = tryCatchSync(()=>JSON.parse(someString))
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
@osoclos
new Error().stack
. Some errors don't need a stack trace like errors thrown by a JWT verifier.if (st.isErr(result)) return result;
Or if you don't care much about perf and want a nice API:
This project is trying to be
neverthrow
but faster and uses less memory.