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 }; | |
} | |
} |
- This is personal preference. Throwing errors are actually much slower (about 300x) and not as type safe as you don't know the error type thrown.
- You can include stack trace in your error type with a single line:
new Error().stack
. Some errors don't need a stack trace like errors thrown by a JWT verifier. - You can return the error directly:
if (st.isErr(result)) return result;
Or if you don't care much about perf and want a nice API:
import * as flow from 'safe-throw/flow';
import * as st from 'safe-throw';
import * as native from 'safe-throw/native';
// Thrown errors will be wrapped with native.err
const asyncFn = native.asyncTry(() => fetch('http://example.com'));
const syncFn = () => Math.random() < 0.5 ? st.err('Random') : 200;
const fn = function*() {
// Unwrap async and result
const fetchResult = yield* flow.unwrap(asyncFn());
const expectedStatus = yield* flow.unwrap(syncFn());
return fetchResult.status === expectedStatus;
};
const res = await flow.run(fn());
if (st.isErr(res)) {
// Handle error
} else {
res; // boolean
}
This project is trying to be neverthrow
but faster and uses less memory.
-
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
@aquapi i have a few problems with your solution:
having a function to just create an error is rather clunky, especially when
import * as st from "safe-throw"
now has to be included every time you wanted to throw something.in a large codebase, dev-ex will become severely diminished, since you dont have stack traces using error symbols, and if you use your solution as a very fundamental and core function of your code, and something went wrong with that function when you use it later on, you are likely to suffer since the error cant be natively traced back to the original source, making debugging very difficult.
continuing number 2, in more complex functions, the only way to properly use your solution is by having a temporary variable and if it fails, assign your error to it and then
return
it to prevent further execution. again this is bad dev-ex and you shouldn't need to do this.