Security News
New Python Packaging Proposal Aims to Solve Phantom Dependency Problem with SBOMs
PEP 770 proposes adding SBOM support to Python packages to improve transparency and catch hidden non-Python dependencies that security tools often miss.
@eyedea/next-roots
Advanced tools
Utility to handle internationalized routing for Next.js app folder.
NextRoots is i18n routes generator for the new Next.js APP directory. It is an alternative to officially recommended way of handling i18n routes in Next.js app.
The main idea behind is to generate all localized file-routes (slugs) in advance rather than putting everything into dynamic [lang]
segment. Read more about benefits of generated i18n routes.
If you are using old Next.js pages directory check next-roots@v2.
Working demo site built on top of next-roots can be seen https://next-roots-svobik7.vercel.app
Let's consider simple project structure that was built with English as a default locale and now needs to be localized for Czech audience.
├── app
│ ├── about
│ │ └── page.js
│ └── page.js
└── ...
The requirement is to have English localization served from /
and Czech from /cs/...
. The goal is to have following URLs:
/
/cs
/about
/cs/o-nas
NextRoots supports both prefixed
/en
and un-prefixed/
default locale.
yarn add next-roots
yarn add --dev esbuild
package.json
{
"scripts": {
"roots": "yarn next-roots",
"roots:watch": "yarn next-roots -w"
}
}
As default Next.js reads routes from the folder called app
. Using NextRoots and generating i18n routes requires you to move all your original routes into different folder called roots
(name is customizable).
Run following command from your project root:
mv ./app/ ./roots
This would be the new area where you are going to store your original routes, write your code and make changes. From now on you wont be editing the files under the app
folder. The app
folder will stands only as a keeper of localized routes and forwards everything to the original ones.
The project structure now looks like:
├── roots
│ ├── about
│ │ └── page.js
│ └── page.js
└── ...
For RSC support see this alternative configuration in FAQ section.
To tell NextRoots which locales we want to generate and where the roots files and app files can be found the roots.config.js
file must be defined in project root.
touch ./roots.config.js
Simple configuration for English and Czech localizations can looks like this:
const path = require('path')
module.exports = {
originDir: path.resolve(__dirname, 'roots'),
localizedDir: path.resolve(__dirname, 'app'),
locales: ['en', 'cs'],
defaultLocale: 'en',
prefixDefaultLocale: false, // serves "en" locale on / instead of /en
}
Generation is initiated by running following command from project root.
yarn next-roots
IMPORTANT: Please be aware that app
folder is wiped out during generation so be sure to have proper git or other backup in place.
The app
folder is now re-generated and project structure is shaped like this:
├── app
│ ├── (en)
│ │ ├── about
│ │ │ └── page.js
│ │ └── page.js
│ ├── cs
│ │ ├── about
│ │ │ └── page.js
│ │ └── page.js
├── roots
│ ├── about
│ │ └── page.js
│ └── page.js
└── ...
Without any further steps the project ends up with URLs like that:
Every URL path (slug) or even segment of the URL path can be translated or left untranslated (depends on project needs). To translate URL segment we need to add i18n.js
file into the original route directory.
Note that i18n.js, i18n.mjs and i18n.ts files are supported. Each of those file are compiled during the generation by esbuild.
├── app // app folder stays untouched now
├── roots
│ ├── about
│ │ ├── i18n.js // i18n.js file is added to the route that URL path needs to be translated
│ │ └── page.js
│ └── page.js
└── ...
Adding translated paths into i18n.js
does the trick:
module.exports.routeNames = [
{ locale: 'cs', path: 'o-nas' },
// you don't need to specify default translation as long as it match the route folder name
// { locale: 'en', path: 'about' },
]
For describing translations in promise-like way see Translation files
Running yarn roots
again will update app
folder routes with translated paths. The project structure now looks like:
├── app
│ ├── (en)
│ │ ├── about
│ │ │ └── page.js
│ │ └── page.js
│ ├── cs
│ │ ├── o-nas // translated URL path
│ │ │ └── page.js
│ │ └── page.js
├── roots
│ ├── about
│ │ └── page.js
│ └── page.js
└── ...
Finally our project is served on URLs that match perfectly the initial requirements. If you need to change your routes or translation do not forget to run yarn roots
again.
Roots comes with a strongly typed Router class for creating links between your pages. Thanks to the generated schema and types you will be notified if the desired route exists or requires additional parameters.
It is good practice to use Router only on the server side so that the list of all possible routes is not sent to the client.
Creates page href.
getHref(name: string, params?: object)
The first parameter called name
is the original route URL path.
The second parameter is an object which can define desired locale
or additional dynamic params.
Thanks to strong types you can import the RouteName
type which includes all available route name strings.
import { Router, schema, RouteName } from 'next-roots'
const router = new Router(schema)
// for getting '/cs/o-nas'
router.getHref('/about', { locale: 'cs' })
// typescript will yield at you here as /not-existing is not a valid route
router.getHref('/not-existing', { locale: 'cs' })
const routeNameValid: RouteName = '/about'
const routeNameInvalid: RouteName = '/invalid' // yields TS error
For dynamic routes like [articleId]
:
// for getting '/cs/1'
router.getHref('/[articleId]', { locale: 'cs', articleId: '1' })
// typescript will yield at you here because of the missing required parameter called articleId
router.getHref('/[articleId]', { locale: 'cs' })
const routeDynamic: RouteName = '/[articleId]'
const paramsDynamicValid: RouteParamsDynamic<typeof routeDynamic> = {
locale: 'cs',
articleId: '1',
}
// typescript will yield at you here because of the missing required parameter called articleId
const paramsDynamicInvalid: RouteParamsDynamic<typeof routeDynamic> = {
locale: 'cs',
}
Passing the locale
parameter is not required. If you do not pass any locale
param then the current page locale will be automatically used.
// on "/cs" page it will creates "/cs/o-nas" href while on "/" (en) it will create "/about" href
router.getHref('/about')
This is possible thanks to Router internal static context value of current href. Whenever user visit a page Router will sets the internal page href and determine the locale from that. If you look at generated page routes you can see that:
// in "app/cs/o-nas/page.js
export default function AboutPage(props: any) {
Router.setPageHref('/cs/about')
return <AboutPageOrigin {...props} pageHref={Router.getPageHref()} />
}
// in "app/(en)/about/page.js
export default function AboutPage(props: any) {
Router.setPageHref('/about')
return <AboutPageOrigin {...props} pageHref={Router.getPageHref()} />
}
Even you are allowed to change this static context down in the code by calling Router.setPageHref
. It is not recommended and can break the links.
When running Next.js as a standalone server for example in EC2 or docker container, the Router functionality breaks.
Since Next.js is ran as a basic Node.js server in a standalone mode, the Router class is shared for each page generation. As so, the Router.getPageHref() can return wrong values in the generation phase of the page.
in that case you always need to pass current locale
param to Router.getHref
function and do not use Router.getPageHref
. See more in https://github.com/svobik7/next-roots/issues/99.
Detects locale from given href and send it back. When no valid locale is found then the default one is retrieved.
// retrieves "en"
router.getLocaleFromHref('/about')
// retrieves "cs"
router.getLocaleFromHref('/cs/o-nas')
// retrieves "en"
router.getLocaleFromHref('/invalid-locale/o-nas')
Detects route from given href and send it back. When no valid route is found then undefined is retrieved.
// retrieves "{ name: "/about", href: "/about" }"
router.getRouteFromHref('/about')
// retrieves "{ name: "/about", href: "/cs/o-nas" }"
router.getRouteFromHref('/cs/o-nas')
// retrieves "undefined"
router.getRouteFromHref('/invalid-locale/o-nas')
NextRoots pushes some additional props to your components and functions to be able to read current page href or locale directly.
pageHref
locale
pageHref
pageLocale
Following types are available for props above and can be imported from next-roots:
Translation of URL paths is done in i18n.js
or i18n.ts
files by placing this file right next to the page.js
of page.ts
file and running yarn roots
. There are two main ways how you can define the i18n file.
Useful when you want to specify the translation in the i18n file itself:
// if you use "i18n.mjs"
export const routeNames = [
{ locale: 'en', path: 'about' },
{ locale: 'cs', path: 'o-nas' },
]
// if you use "i18n.js"
module.exports.routeNames = [
{ locale: 'en', path: 'about' },
{ locale: 'cs', path: 'o-nas' },
]
Useful when you want to store the translations in DB or other async storage:
export async function generateRouteNames() {
// "getTranslation" is custom async function that loads translated paths from DB
const { enPath, csPath } = await getTranslations('/about')
return [
{ locale: 'en', path: enPath },
{ locale: 'cs', path: csPath },
]
}
You don't need to specify translations for default locale. Routes inherit the path names from origin folders by default. If you specify the translation for default locale then it is used instead of origin folder name.
If you have intercepting route like @modal/(.)blogs/[author]/[articleId]/page.js
with corresponding normal route equals to blogs/[author]/[articleId]/page.js
and you already translated normal route by creating blogs/i18n.js
file with following contents:
module.exports.routeNames = [
{ locale: 'en', path: 'blogs' },
{ locale: 'cs', path: 'blogy' },
{ locale: 'es', path: 'blogs' },
]
then you need to translate even the intercepting route. Otherwise the intercepting route will be generated with untranslated path and interception will not work. I18n file placed in @modal/(.)blogs/i18n.js
can be use for example above with following contents:
module.exports.routeNames = [
{ locale: 'en', path: '(.)blogs' },
{ locale: 'cs', path: '(.)blogy' },
{ locale: 'es', path: '(.)blogs' },
]
name | type | default | required | description |
---|---|---|---|---|
originDir | string | ./roots | optional | absolute path to the origin un-translated routes |
localizedDir | string | ./app | optional | absolute path to the localized routes. This is where next-roots saves generated routes. |
locales | string[] | [] | required | localization prefixes that will be used in URL |
defaultLocale | string | '' | required | default locale that is specified in locales |
prefixDefaultLocale | boolean | true | optional | when default locale = en then TRUE means it will be served from "/en" and FALSE means it will be served without prefix on / |
packageDir | string | ./node_modules/next-roots | optional | absolute path to the next-root package itself. Should be changed only when package is stored in different location than project root node_modules |
afterGenerate | function | undefined | optional | custom function that will be called after files generation is done (see) |
If you need to do custom actions after the generation is done you can use afterGenerate
callback. This callback is called with following params:
type AfterGenerateCallback = (params: {
config: Config
origins: Origin[]
rewrites: Rewrite[]
routes: Route[]
routerSchema: RouterSchema
}) => Promise<void>
[lang]
approach?The [lang]
approach works well until you need to translate URL slugs. Read more about generated routes in https://dev.to/svobik7/dont-use-dynamic-lang-segment-for-your-i18n-nextjs-routes-3k05
While it is not recommended it is still possible. In that case the whole schema needs to be send to client as well which increases bundle size. Read more about server components https://beta.nextjs.org/docs/rendering/server-and-client-components
If you need to serve a content from un-translated routes like /robots.txt
or others you can easily achieve that by placing those files/routes directly into "app" folder and set localizedDir
target to nested group route.
├── app
│ ├── (routes) // translated routes will be generated within this folder
│ │ ├── en
│ │ └── cs
│ └── robots.txt
├── roots
│ ├── ...
│ └── page.js
└── ...
// roots.config.js
module.exports = {
// ...
localizedDir: path.resolve(__dirname, 'app/(routes)',
}
There are two recommended way how to achieve this:
[[...catchAll]]
route - see examples/with-preferred-language-catchall
examplemiddleware.ts
- see examples/with-preferred-language-middleware
exampleTo support RSC you need to keep your routes inside app folder. In that case your original routes can be placed in app/_roots
and translated routes into app/(routes)
. your roots.config.js
file would then look like this:
module.exports = {
originDir: path.resolve(__dirname, 'app/_roots'),
localizedDir: path.resolve(__dirname, 'app/(routes)'),
// ... other config params
}
Note that _roots
and (routes)
dir names are here just as example. you can choose your own naming.
FAQs
Utility to handle internationalized routing for Next.js app folder.
The npm package @eyedea/next-roots receives a total of 23 weekly downloads. As such, @eyedea/next-roots popularity was classified as not popular.
We found that @eyedea/next-roots demonstrated a not healthy version release cadence and project activity because the last version was released a year ago. It has 5 open source maintainers collaborating on the project.
Did you know?
Socket for GitHub automatically highlights issues in each pull request and monitors the health of all your open source dependencies. Discover the contents of your packages and block harmful activity before you install or update your dependencies.
Security News
PEP 770 proposes adding SBOM support to Python packages to improve transparency and catch hidden non-Python dependencies that security tools often miss.
Security News
Socket CEO Feross Aboukhadijeh discusses open source security challenges, including zero-day attacks and supply chain risks, on the Cyber Security Council podcast.
Security News
Research
Socket researchers uncover how threat actors weaponize Out-of-Band Application Security Testing (OAST) techniques across the npm, PyPI, and RubyGems ecosystems to exfiltrate sensitive data.