Links

handle-errors-and-exceptions-in-javascript.s

Owner: Maxime Sraïki

Why

  • On scaling projects you can easily be overwhelmed by a pretty large code source. When this occurs, if you don't have a proper way to handle errors/exceptions, you could easily start raising a lot of bug or regression and spend a lot of time trying to fix it.

Definitions

  • First let's make the distinction between Erros and Exceptions:
    • Errors are javascript objects that are thrown when something goes wrong in the code during runtime. It will usually have:
      • a message key containing a human readable error message
      • informations about where the error was raised like fileName, lineNumber, or even the whole stack
    • Exceptions are custom Errors objects created by developers to implement business logic.

Checks

  • Every Exceptions should be caught in a .catch statement and treated.
  • Every caught Error should be rethrown (Even if it makes your application crash, you have to see it QUICKLY)
  • Every .catch statement should take the error as argument
  • [OPTIONAL] If you have a logging system, in every .catch statement, you should log a message with the context and the whole error object

Examples

Bad Examples:

Example 1: No catch statement

myFunction().then(res => {
//Do something with res
})
Here if myFunction throws an error, it will remain unhandled and can make my server crash without prior notice !

Example 2: Error no rethrown

myFunction().then(res => {
//Do something with res
}).catch(console.log)
Here if myFunction throws an error, it will be caught in the .catch statement and logged to the console logger. The problem are:
  • The statement
    .catch(console.log)
    is equivalent to
    .catch(err => console.log(err))
    which is
    .catch(err => {return console.log(err)})
    therefore it will go in your next .then and act as if everything was normal with an undefined response...
    • You make the error disappear and it will not be caught in your next .catch

Example 3: Error not even considered

myFunction().then(res => {
//Do something with res
}).catch(() => {
//Whatever
})
If you don't even use the error that has been caught, you are sure you don't react correctly to it.

Good Examples:

Example 1: Error

myFunction(argument).then(res => {
//Do something with res
}).catch(error => {
logger.log('The following Error occured when executing myFunction with argument', { error, argument })
return Promise.reject(err)
})

Example 2: Exception

myFunction(argument).then(res => {
//Do something with res
}).catch(error => {
if (error.msg === EXPECTED_MESSAGE) {
// Do something with error
}
logger.log('The following Error occured when executing myFunction with argument', { error, argument })
return Promise.reject(err)
})