What are the types of the four parameters for the error handling function when coding in Typescript?
app.use((err: ??, req: ??, res: ??, next: ??) => { }
I don't know why Typescript does not get that we are passing an error handler.
These are the solutions I found possible :
app.use((err, req, res, next) => {}) // KO
const inVariable : ErrorRequestHandler = (err, req, res, next) => {};
app.use(inVariable); // ok
app.use(((err, req, res, next) => {}) as ErrorRequestHandler); // ok
Using any
completely gives up the type checking advantage of the Typescript. For err
parameter, it's recommended to create your own type or class like following:
// HttpException.ts in exceptions directory of your project.
export class HttpException extends Error {
public status: number
public message: string
constructor(status: number, message: string) {
super(message)
this.status = status
this.message = message
}
}
And then import it for your error-handling middleware:
import { HttpException } from './exceptions/HttpException'
app.use((err: HttpException, req: Request, res: Response, next: NextFunction) => {
err.status = 404
err.message = 'Not Found'
next(err)
})
We can add any properties and methods to the HttpException
class as and when required, like we have added status
and message
.
In case you haven't done already, you need to install type definitions for Node.js and Express.js for your Typescript project. This will make sure the types Request
, Response
and NextFunction
will be recognised and auto-imported.
To install type definitions for Node.js:
npm install --save-dev @types/node
To install type definitions for Express.js:
npm install --save-dev @types/express
That's it! Hope that helps.
The function itself has the following signature (taken from DefinitelyTyped):
export type ErrorRequestHandler = (err: any, req: Request, res: Response, next: NextFunction) => any;
So you can either declare the function as a variable of type ErrorRequestHandler
or type the parameters according to that definition.
Note: the typings for "express-serve-static-core" are imported by the typings for "express", which was where I looked for the above definition.
Regarding your second question related to implicit any
, it is the "implicit" part which is causing the problem, If you explicitly type as any
then there won't be any error (but there won't be any typings either).
You can also disable noImplicitAny
in your compiler config but I wouldn't recommend it personally, as it protects you from several classes of bugs.