Decorator is a neat design pattern which lets you extend the functionality without modifying it. It’s usually also a higher order function = function that takes another function as an argument.
I need a way connect to Mongodb
In my NextJS project I use mongodb as a persistence layer, and to be able to do queries I need to :
await connect(config.mongoose.uri, config.mongoose.options)
This turns out to be a repetitive task for any api route or a page.
I could use a middleware
NextJS has a middleware to do that, but I’m not sure that I like using middleware for that purpose.
Middleware gets called for everything, even the _next
routes. And If you don’t want to call it for everything, you’d use a matcher
, something like:
export const config = { matcher: [ /* * Match all request paths except for the ones starting with: * - api (API routes) * - _next/static (static files) * - favicon.ico (favicon file) */ '/((?!api|_next/static|favicon.ico).*)', ], }
I prefer decorators instead
Now, a middleware may be perfectly fine for logging, but it looked off to me to use it to connect to the mongo database.
Instead, I created a decorator that connects to the database under the hood. It still is a repetitive task, but I prefer to explicitly call it whenever I need mongodb in the scope.
export function withMongooseSessionApiRoute<T>( handler: NextApiHandler<T>, uri: string, options: ConnectOptions ): NextApiHandler<T> { return async function nextApiHandler(req: NextApiRequest, res: NextApiResponse) { await connect(uri, options); return handler(req, res); }; }
withMongooseSessionApiRoute
is a function that takes another function (nextjs handler) and mongodb connect options. Inside the withMongooseSessionApiRoute
, a connection to mongo gets done first and nextjs handler call is second.
And for an api route, you’d call like:
export default withMongooseSessionApiRoute( async function handler(req, res) { } );
Small caveat on the current implementation is that cache is not supported for multiple different uris. If you need that implementation, please open a new issue request at https://github.com/eisberg-labs/next-mongoose-session/issues.
You can check out the code and instructions at https://github.com/eisberg-labs/next-mongoose-session.
If you’re not sure how to configure mongoose schemas, take a look at my other post!