middleware.ts 1.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445
  1. import { match } from '@formatjs/intl-localematcher'
  2. import Negotiator from 'negotiator'
  3. import { NextResponse } from 'next/server'
  4. import type { NextRequest } from 'next/server'
  5. import type { Locale } from './i18n'
  6. import { i18n } from './i18n'
  7. export const getLocale = (request: NextRequest): Locale => {
  8. // @ts-expect-error locales are readonly
  9. const locales: Locale[] = i18n.locales
  10. let languages: string[] | undefined
  11. // get locale from cookie
  12. const localeCookie = request.cookies.get('locale')
  13. languages = localeCookie?.value ? [localeCookie.value] : []
  14. if (!languages.length) {
  15. // Negotiator expects plain object so we need to transform headers
  16. const negotiatorHeaders: Record<string, string> = {}
  17. request.headers.forEach((value, key) => (negotiatorHeaders[key] = value))
  18. // Use negotiator and intl-localematcher to get best locale
  19. languages = new Negotiator({ headers: negotiatorHeaders }).languages()
  20. }
  21. // match locale
  22. let matchedLocale: Locale = i18n.defaultLocale
  23. try {
  24. // If languages is ['*'], Error would happen in match function.
  25. matchedLocale = match(languages, locales, i18n.defaultLocale) as Locale
  26. }
  27. catch (e) {}
  28. return matchedLocale
  29. }
  30. export const middleware = async (request: NextRequest) => {
  31. const pathname = request.nextUrl.pathname
  32. if (/\.(css|js(on)?|ico|svg|png)$/.test(pathname))
  33. return
  34. const locale = getLocale(request)
  35. const response = NextResponse.next()
  36. response.cookies.set('locale', locale)
  37. return response
  38. }