Implementing Multiple Middleware in Next.js: Combining NextAuth V(5) and Internationalization

tanzimhossain2

Tanzim Hossain

Posted on May 21, 2024

Implementing Multiple Middleware in Next.js: Combining NextAuth V(5) and Internationalization



Implementing Multiple Middleware Functions in Next.js

Welcome to my first blog post on Dev Community


Today, I'll walk you through implementing multiple middleware functions in a Next.js application. Specifically, we'll combine authentication using NextAuth (v5) and internationalization. By the end of this tutorial, you'll know how to chain middleware functions to create a modular and maintainable setup.



We know next js middleware just export one funtion. But what if we want to use multiple middleware functions in our Next.js application? In this tutorial, we'll show you how to combine multiple middleware functions using a chain utility. We'll create middleware for internationalization and authentication, and then chain them together to create a single middleware function. This approach allows us to easily add more middleware functions in the future without cluttering our codebase. You can use this setup to add additional features like logging, caching, or error handling to your Next.js application.


Let's get started! 🚀

File Structure


Before diving into the code, let's set up our project structure to keep everything organized. Here's a simplified version of our file structure:

/my-next-app
|-- /src
| |-- /middlewares
| | |-- chain.ts
| | |-- withI18nMiddleware.ts
| | -- withAuthMiddleware.ts
| |-- auth.config.ts
| |-- auth.ts
| |-- middleware.ts
| |-- i18n.config.ts
|
-- next.config.js

Step 1: Setting Up the Chain Utility


The first step is to create a utility to chain multiple middleware functions. This utility will ensure that each middleware function is executed in sequence. We'll use this utility to combine the internationalization and authentication middlewares. This will allow us to easily add more middleware functions in the future.

The chain utility takes an array of middleware functions and returns a single middleware function that executes each middleware in order. If a middleware function returns a response, the chain stops and returns that response. Otherwise, it calls the next middleware function in the chain. It continues this process until all middleware functions have been executed. So let's create the chain utility.

Create a chain.ts file in the /middlewares directory

Here's the content of the chain.ts file:

 <code>
import { NextMiddlewareResult } from 'next/dist/server/web/types';
import { NextResponse } from 'next/server';
import type { NextFetchEvent, NextRequest } from 'next/server';

export type CustomMiddleware = (
request: NextRequest,
event: NextFetchEvent,
response: NextResponse
) => NextMiddlewareResult | Promise<NextMiddlewareResult>;

type MiddlewareFactory = (middleware: CustomMiddleware) => CustomMiddleware;

export function chain(
functions: MiddlewareFactory[],
index = 0
): CustomMiddleware {
const current = functions[index];

if (current) {
const next = chain(functions, index + 1);
return current(next);
}

return (
request: NextRequest,
event: NextFetchEvent,
response: NextResponse
) => {
return response;
};
}
</code>

Step 2: Creating the Internationalization Middleware


Next, we'll create middleware for handling internationalization. This middleware will detect the user's locale and redirect them if necessary.



We'll use the @formatjs/intl-localematcher library to match the user's preferred languages with the available locales in our application. If the user's preferred language is not supported, we'll redirect them to the default locale. This ensures that the user always sees content in a supported language.

 Create a withI18nMiddleware.ts file in the /middlewares directory 

Here's the content of the withI18nMiddleware.ts file:

<code>
// middlewares/withI18nMiddleware.ts
import { NextResponse } from 'next/server';
import type { NextFetchEvent, NextRequest } from 'next/server';

import { i18n } from '@/i18n.config';
import { match as matchLocale } from '@formatjs/intl-localematcher';
import Negotiator from 'negotiator';
import { CustomMiddleware } from './chain';

function getLocale(request: NextRequest): string | undefined {
const negotiatorHeaders: Record<string, string> = {};
request.headers.forEach((value, key) => (negotiatorHeaders[key] = value));

// @ts-ignore locales are readonly
const locales: string[] = i18n.locales;
const languages = new Negotiator({ headers: negotiatorHeaders }).languages();

const locale = matchLocale(languages, locales, i18n.defaultLocale);
return locale;
}

export function withI18nMiddleware(middleware: CustomMiddleware) {
return async (
request: NextRequest,
event: NextFetchEvent,
response: NextResponse
) => {
const pathname = request.nextUrl.pathname;
const pathnameIsMissingLocale = i18n.locales.every(
locale => !pathname.startsWith(/${locale}/) && pathname !== /${locale}
);

if (pathnameIsMissingLocale) {
  const locale = getLocale(request)
  const redirectURL = new URL(request.url)

  if (locale) {
    redirectURL.pathname = `/${locale}${pathname.startsWith('/') ? '' : '/'}${pathname}`
  }
    // Preserve query parameters
    redirectURL.search = request.nextUrl.search

    return NextResponse.redirect(redirectURL.toString())
}

return middleware(request, event, response);

};
}
</code>

Step 3: Creating the Authentication Middleware


Now, let's create middleware for handling authentication using NextAuth.



We'll use the getToken function from next-auth/jwt to retrieve the user's token. If the token is missing, we'll redirect the user to the sign-in page. This ensures that only authenticated users can access protected routes in our application.



Note: Make sure to set the NEXTAUTH_SECRET environment variable in your Next.js application. You can generate a secure secret using a tool like openssl or a password manager.

 Create a withAuthMiddleware.ts file in the /middlewares directory 

Here's the content of the withAuthMiddleware.ts file:

<code>
// middlewares/withAuthMiddleware.ts
import { getToken } from 'next-auth/jwt';
import { NextResponse } from 'next/server';
import type { NextFetchEvent, NextRequest } from 'next/server';
import { CustomMiddleware } from './chain';

export function withAuthMiddleware(middleware: CustomMiddleware): CustomMiddleware {
return async (request: NextRequest, event: NextFetchEvent, response: NextResponse) => {
const token = await getToken({ req: request, secret: process.env.NEXTAUTH_SECRET });
if (!token) {
return NextResponse.redirect(new URL('/api/auth/signin', request.url));
}

return middleware(request, event, response);

};
}
</code>

Step 4: Combining the Middlewares


Finally, we'll chain the internationalization and authentication middlewares together in a single middleware file. We combine them using the chain utility we created earlier. Now we can use this middleware in our Next.js application.



In main middleware file, we can define the order of the middleware functions , remember order is important. If you want to change the order of the middleware functions, you can simply rearrange them in the chain array. You need to make sure what middleware function should be executed first and what should be executed later.



In this example, we're using the withI18nMiddleware first to handle internationalization and then withAuthMiddleware to handle authentication. You can add more middleware functions as needed.



 Create a middleware.ts file at the root of your project 

Here's the content of the middleware.ts file:

<code>
// middleware.ts
import { chain } from '@/middlewares/chain';
import { withI18nMiddleware } from '@/middlewares/withI18nMiddleware';
import { withAuthMiddleware } from '@/middlewares/withAuthMiddleware';

export default chain([withI18nMiddleware, withAuthMiddleware]);

export const config = {
matcher: ['/((?!api|_next/static|_next/image|favicon.ico|images).*)'],
};
</code>

Conclusion

By following these steps, you've successfully combined NextAuth and internationalization middleware in your Next.js application. This modular approach allows you to add more middleware functions as needed without cluttering your codebase. You can now easily extend your application with additional features while keeping your code clean and maintainable.

Thanks for reading! I hope you found this tutorial helpful. If you have any questions or feedback, feel free to leave a comment below.


💖 💪 🙅 🚩
tanzimhossain2
Tanzim Hossain

Posted on May 21, 2024

Join Our Newsletter. No Spam, Only the good stuff.

Sign up to receive the latest update from our blog.

Related