mirror of
https://github.com/vercel/commerce.git
synced 2025-05-15 05:56:59 +00:00
60 lines
1.9 KiB
TypeScript
60 lines
1.9 KiB
TypeScript
import type { NextRequest } from 'next/server';
|
|
import { NextResponse } from 'next/server';
|
|
|
|
import { match as matchLocale } from '@formatjs/intl-localematcher';
|
|
import { i18n } from 'i18n-config';
|
|
import Negotiator from 'negotiator';
|
|
|
|
function getLocale(request: NextRequest): string | undefined {
|
|
// Negotiator expects plain object so we need to transform headers
|
|
const negotiatorHeaders: Record<string, string> = {};
|
|
request.headers.forEach((value, key) => (negotiatorHeaders[key] = value));
|
|
|
|
// @ts-ignore locales are readonly
|
|
const locales: string[] = i18n.locales;
|
|
|
|
// Use negotiator and intl-localematcher to get best locale
|
|
let languages = new Negotiator({ headers: negotiatorHeaders }).languages(locales);
|
|
|
|
const locale = matchLocale(languages, locales, i18n.defaultLocale);
|
|
|
|
return locale;
|
|
}
|
|
|
|
export function middleware(request: NextRequest) {
|
|
const pathname = request.nextUrl.pathname;
|
|
|
|
// `/_next/` and `/api/` are ignored by the watcher, but we need to ignore files in `public` manually.
|
|
// If you have one
|
|
if (
|
|
[
|
|
'/public/assets/images/logo.png',
|
|
'/public/assets/images/logo+namemark.png',
|
|
'/public/assets/images/namemark.png'
|
|
// Your other files in `public`
|
|
].includes(pathname)
|
|
)
|
|
return;
|
|
|
|
// Check if there is any supported locale in the pathname
|
|
const pathnameIsMissingLocale = i18n.locales.every(
|
|
(locale: any) => !pathname.startsWith(`/${locale}/`) && pathname !== `/${locale}`
|
|
);
|
|
|
|
// Redirect if there is no locale
|
|
if (pathnameIsMissingLocale) {
|
|
const locale = getLocale(request);
|
|
|
|
// e.g. incoming request is /products
|
|
// The new URL is now /en-US/products
|
|
return NextResponse.redirect(
|
|
new URL(`/${locale}${pathname.startsWith('/') ? '' : '/'}${pathname}`, request.url)
|
|
);
|
|
}
|
|
}
|
|
|
|
export const config = {
|
|
// Matcher ignoring `/_next/` and `/api/`
|
|
matcher: ['/((?!api|_next/static|_next/image|favicon.ico).*)']
|
|
};
|