Beau's Ontology

home

Timeline

Typescript Error Handling

Started 3/28/2025

Last updated 3/28/2025

Typescript has a number of less-than-ideal features that I’ve run into consistently. These issues prompted me to research the subject and see if these problems were common, if there were common solutions for mitigating them. In this process, I discovered I had found common points of frustration for programmers of many languages, and that much of language design centers around these features.

For instance, one issue I ran into frequently (especially when using lots of packages), was effectively error handling when I don’t know when or if a given function will throw an error. In typescript, this often leads of a mess of try/catch blocks with generic error handling that usually ends up swallowing a significant amount of the context, making programs harder to debug.

Here, I learned about how Rust (a compiled general-purpose language known for employing out-of-the-box rules that enforce strong safety practices; Rust programs, once compiled, usually work as expected) handles errors with an enum type, which forces developers to handle each and every error explicitly prior to providing the ‘happy path’ of no errors. This pattern can be tedious, but it’s extremely thorough, and it forces thoughtful scenario handling where typescript makes errors easy to ignore. What’s most interesting to note is typescript has similar capabilities to Rust’s enum with discriminated unions. These are frequently used by developers for situations where you are expecting to have a set of mutually exclusive sets of data, but are unsure which you will have until runtime. For example, if you call a function that will attempt to fetch data for you, you won’t know whether the fetch was successful or not until your program is running. If it was successful, you’ll likely want to display the data that was fetched; but if it was unsuccessful, you’ll likely want to just show a message that briefly describes what went wrong. In this case, you’ll either have data of some expected format (success) or an error message (fail) - both will be in the same variable as potential options until you check.

So while Rust always uses types to manage scenarios like this, Typescript uses a throw, which cancels execution of the current path up the function call stack until it is caught in a try/catch block. If the thrown error is not caught by any try/catch block, the entire program will crash and stop executing.

This might not be so bad if try/catch blocks could look inside of the code it was running to infer the complete set of possible error types. This way, you could exhaustively manage all ways a set of code may fail. But instead, try/catch in typescript assumes ignorance, so regardless of what’s inside, the returned error type will always be unknown. In this case, the best the developer can do is throw custom errors, and check if the error is of that type later. This works well for the most straightforward code blocks, but the larger the code gets, the harder it is to know which errors are thrown, so the harder it is to handle them.

Many people have thought about this problems like these and sought to resolve them structurally with packages. Since typescript is a powerful system1, as much as there are issues with specific aspects of the language, its customizability lends itself to the ability to shape how you program with it. Packages like fp-ts take heavy inspiration from functional languages like haskell. There are also many blog2 posts about hand-built solutions and package set-ups to improve safety and the developer experience.

Footnotes

  1. Powerful type systems can express many different kinds of types, where strong type systems have heavy constraints that can’t be broken (as pointed out in this comment). Type system design is a balance between broadly constraining users for better safety while giving them the tools to make those decisions themselves. This conversation is still contentious as Rust increases adoption in the face of criticism from C & C++ developers who think expressiveness is important, and bugs resulting from poor type design is a skill issue rather than a system issue.

  2. https://blog.martijnarts.com/predictable-programming-2-making-typescript-more-like-rust/

Using Writing to Reach a Result

Started 1/24/2025

Last updated 3/28/2025

I wanted to dedicate some time to deciphering a comment left on a Terrence Tao (TT) post called ‘Maximising the results-to-effort ratio’.

The original article argued for taking low-hanging fruit in terms of illustrating natural consequences, counter-examples, and variations derived from ones main argument in math research papers, as these consequences aren’t publishable in their own right since they’re too similar to your work, and you’re leaving work, however small, to be rederived by readers at a later date (either in their head, or in writing when extending your work), rather than explicitly spelling it out for them. The comment addressing the post is as follows:

This is like the space-time tradeoff in computer science. A paper is like a program. A user uses the paper to reach a result, just like a program uses an algorithm to reach a result.

If the paper is too long, exhausting all extensions, it takes up too much space to reach a slight decrease in time, which the reader might be better off rederiving the extensions it themselves. And similarly in the opposite direction.

The space-time tradeoff in computer science describes a situation where lowering the time required to accomplish some goal requires using more memory, and vice versa. The anonymous user is likening a person to a computer and a paper to a program in that both are ‘run’ to reach some result. In the case of the paper, it is to reach an argument’s conclusion, and in the case of the program, it is to generate some output.

Papers with many arguments, extensions, and natural following conclusions use a lot of memory, which increases the time required to reach a final understanding. In this case, it may be better to have fewer arguments, explore fewer extensions, and thus reduce the amount of memory required, to decrease the time needed properly absorb.

Put simply, this is a very straightforward relationship and writing rule: increasing your complexity increases the mental burden of your reader, but delivers more results to build from in exchange.

With regard to math papers, TT’s suggestion to be thorough makes sense. Math is a rigorous domain where all stones must be overturned. Logical statements that seem like they follow have many hidden assumptions, and explicitly stating points that extend from an argument can help readers reach points they wouldn’t otherwise.

There are always hidden assumptions and conclusions that are left to the reader to fill in. It is up to the writer to decide which to state explicitly on their way to reaching a conclusion. Your level of rigor should depend on your topic and your reader.