npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2025 – Pkg Stats / Ryan Hefner

aargh

v1.4.0

Published

Selectively handle errors in JavaScript

Downloads

163

Readme

aargh

Yep. That's the sound of a man who just got dumped by his girlfriend, chewed by a bear and then shot 32 times. It's also the sound I make when working with JavaScript.

Lol, just kidding. It's one of the sounds I make when working with JavaScript. Aaaaaaargh.

npm version Build Status npm

Why, what, who is this?

This module gives you a simple way to selectively handle errors in JavaScript.

Supposing you have a function like this:

function fetchTweets (userId) {
    return twitterApi.fetch(userIdd).then(response => {
        if (response.error && response.error.message.includes("Too many requests")) {
            throw new RateLimitExceededError();
        }
        return response.data;
    })
}

This function is meant to throw a specific error when you exceed your Twitter API rate limits. (RateLimitExceededError is a custom Error class you created.)

So you call it like this:

fetchTweets(userId)
  .catch(e => {
      console.log('Rate limit exceeded; initiating exponential backoff');
      // do backoff
  })
  .then(tweets => {
      // do stuff with them
  })

or...

try {
    let tweets = await fetchTweets(userId);
} catch(e) {
      console.log('Rate limit exceeded; initiating exponential backoff');
      // do backoff
}

See the problem? Different errors could be thrown in that function. For instance, if you look closely at the function, you'll see I made a typo on the first line (userIdd instead of userId) which will throw a ReferenceError (internal JavaScript error) when executed. However, the calling code will expect a RateLimitError only.

There's a simple fix: use an if-statement:

try {
    let tweets = await fetchTweets(userId);
} catch(e) {
    if (e instanceof RateLimitExceededError) {
      console.log('Rate limit exceeded; initiating exponential backoff');
      // do backoff
    } else if (e instanceof SomeOtherError || e instanceof SomeOtherAnnoyingError) {
        // handle it
    } else {
        // this is important, so unexpected errors 
        // don't get swallowed by our app
        throw e;
    }
}

Hey, if that works for you, well, good, but if you're like me, you hate writing this kind of if statement. Too messy. I prefer PHP's inbuilt selective error handling:

try {
    $tweets = fetchTweets($userId);
} catch(RateLimitExceededError $e) {
    echo 'Rate limit exceeded; initiating exponential backoff';
     // do backoff
} catch (SomeOtherError | SomeOtherAnnoyingError $e) {
    // handle it
}

// Any error that doesn't match the types you specified is not caught

So I decided to do something similar for JS. Here's how you use it:

try {
    let tweets = await fetchTweets(userId);
} catch(e) {
    return aargh(e)
        .handle(RateLimitExceededError, (e) => {
            console.log('Rate limit exceeded; initiating exponential backoff');
            // do backoff
        })
        .handle([SomeOtherError, SomeOtherAnnoyingError], (e) => {
            // handle it
        })
        .throw();
}

Works with Promise#catch() too:

fetchTweets(userId)
  .catch(e => {
      return aargh(e)
          .handle(RateLimitExceededError, (e) =>  {
              console.log('Rate limit exceeded; initiating exponential backoff');
              // do backoff
          })
          .handle([SomeOtherError, SomeOtherAnnoyingError], (e) => {
              // handle it
          })
          .throw();
  })
  .then(tweets => {
      // do stuff with them
  })

Usage

const aargh = require('aargh');

aargh(e)

The entry point. You pass in the error object you want to handle. You should probably have this as the first statement in all your catch() blocks.

.handle(errorTypes, callback)

The first argument to the handle function is the type or types (as an array) of errors you want to handle. The second is a callback containing the code you want to execute for that error. Aargh will call this callback with the error as the only parameter.

You can return stuff from this callback too. Aargh will return this value to the caller.

.throw()

Calling the throw() function ends the chain and ensures any errors which weren't matched by your handle() checks are thrown back to the caller.

.others(callback)

Use the others() function to end the chain and specify a callback to be executed if the error wasn't matched by your type() checks. You should use either throw() or others(), and it should only be used after all type() calls.

try {
    return await fetchTweets(userId);
} catch(e) {
    return aargh(e)
        .handle(RateLimitExceededError, (e) => {
            // do backoff
        })
        .catch(APIUnavailableError, (e) => {
          // backoff nicely
        })
        .others((e) => {
            // This will catch any other errors
            // Maybe log the error? Idk
        });
}

Want to try it out?

npm i aargh

Hey, I'd like your opinion

I spent time thinking about the most intuitive/natural syntax/flow to use for this. If you're interested in this package, and you've got ideas, please hit me up on Twitter or open an issue.

Don't forget

A few tips for error handling in JS (whether you're using this package or if-statements):

  • Create your custom errors. Don't throw strings or just Error. Why? So you can trace exactly what went wrong in your code.
  • Make sure your errors extend from the Error class. Why? So you can leverage all the awesome debugging tools out there, plus inbuilt Error properties like .stack.
  • Handle only errors you expect. Let the rest crash your app. Why? So you know when something goes wrong.