Sitemap generator for Remix applications
npm install remix-sitemap> Sitemap generator for Remix applications
- Runtime & Build time Generation
- Generate robots.txt
- v2 Route Convention Support
- Splitting Sitemaps
- ✨ Features
- 📚 Table Of Contents
- 🚀 Getting Started
- Installation
- Usage
- Runtime Generation
- Build time Generation
- 📝 Guides
- Generate Sitemap for Dynamic Routes
- Exclude Route
- Google: News, Image and Video
- Splitting Sitemaps
- Caching
- 📖 API Reference
- Config
- RobotsTxtOptions
- 👤 Author
- 🤝 Contributing
- Show your support
- Acknowledgements
``sh`
npm i remix-sitemap
#### Runtime Generation
`ts
// entry.server.tsx
import { createSitemapGenerator } from 'remix-sitemap';
// Step 1. setup the generator
const { isSitemapUrl, sitemap } = createSitemapGenerator({
siteUrl: 'https://example.com',
generateRobotsTxt: true
// configure other things here
})
export default async function handleRequest(
request: Request,
responseStatusCode: number,
responseHeaders: Headers,
remixContext: EntryContext
) {
// Step 2. add the sitemap response
if (isSitemapUrl(request))
return await sitemap(request, remixContext);
let markup = renderToString(
);
responseHeaders.set('Content-Type', 'text/html');
return new Response('' + markup, {
status: responseStatusCode,
headers: responseHeaders
});
}
`
Usage with sitemap[.]xml route (experimental)
1. Create a lib/sitemap.ts file`ts`
// lib/sitemap.ts
export const { experimental_sitemap, robots } = createSitemapGenerator({
siteUrl: 'https://example.com',
// configure other things here
})
2. Create a sitemap[.]xml route`ts
// app/routes/sitemap[.]xml.tsx
import { routes } from '@remix-run/dev/server-build';
import { experimental_sitemap } from '~/lib/sitemap';
export const loader: LoaderFunction = async ({ request }) => {
return await experimental_sitemap(request, routes);
}
`
3. Create a robots[.]txt route`ts
// app/routes/robots[.]txt.tsx
import { robots } from '~/lib/sitemap';
export const loader: LoaderFunction = ({ request }) => {
return robots();
}
`
#### Build time Generation
Create a remix-sitemap.config.js file at the project root`ts`
/* @type {import('remix-sitemap').Config} /
module.exports = {
siteUrl: 'https://example.com',
generateRobotsTxt: true
// configure other things here
}remix-sitemap
Add a script using to package.json to run after build.
For example if you are using npm-run-all`json`
{
"scripts": {
"build": "npm-run-all build:*",
"build:remix": "remix build",
"build:sitemap": "remix-sitemap"
}
}
ts
// app/routes/posts.$slug.tsx
import type { SitemapFunction } from 'remix-sitemap';export const sitemap: SitemapFunction = ({ config, request }) => {
const posts = await getPosts();
return posts.map(post => ({
loc:
/posts/${post.slug},
lastmod: post.updatedAt,
exclude: post.isDraft, // exclude this entry
// acts only in this loc
alternateRefs: [
{
href: ${config.siteUrl}/en/posts/${post.slug},
absolute: true,
hreflang: 'en'
},
{
href: ${config.siteUrl}/es,
hreflang: 'es'
}
]
}));
};
`$3
`ts
// app/routes/private.tsx
import type { SitemapFunction } from 'remix-sitemap';export const sitemap: SitemapFunction = () => ({
exclude: true
})
`$3
Url set can contain additional sitemaps defined by google. These are Google news, image or video.
You can add these sitemaps in sitemap function by adding the news, images or videos property.
`ts
export const sitemap: SitemapFunction = () => ({
images: [{ loc: 'https://example.com/example.jpg' }],
news: [{
title: 'Random News',
date: '2023-03-15',
publication: {
name: 'The Example Times',
language: 'en'
}
}]
});
`$3
If you have a lot of urls, you can split the sitemap in multiple files. You can do this by setting the size property in the config.
> This is only available in build time generation
`ts
/* @type {import('remix-sitemap').Config} /
module.exports = {
siteUrl: 'https://example.com',
size: 10000
}
`$3
you have two ways to cache the sitemap, the first one is using the Cache-Control header
> This is only available in runtime generation
`ts
createSitemapGenerator({
siteUrl: 'https://example.com',
headers: {
'Cache-Control': 'max-age=3600'
}
})
`
and the second one is using the cache property in the config
`ts
createSitemapGenerator({
siteUrl: 'https://example.com',
cache: {
get: async () => {
return await redis.get('sitemap') || null;
},
set: async (sitemap) => {
await redis.set('sitemap', sitemap, 'EX', 3600);
}
}
})
`📖 API Reference
$3
-
siteUrl: the website base url
- autoLastmod = true: (optional) Add property with the current date.
- priority = 0.7: (optional) default priority for all entries.
- changefreq = 'daily': (optional) default changefreq for all entries.
- format = false: (optional) format the sitemap for a better view.
- alternateRefs = []: (optional) default multi language support by unique url for all entries.
- generateRobotsTxt = false: (optional) generate robots.txt file.
- robotsTxtOptions: (optional) options for generating robots.txt details.
- rateLimit: (optional) limits the number of sitemap functions that can execute at once.Runtime only
-
headers = {}: (optional) headers for the sitemap and robots.txt response.
- cache: (optional) cache the sitemap details.Build Time only
-
generateIndexSitemap = true: (optional) generate index sitemap.
- sitemapBaseFileName = 'sitemap': (optional) the name of the generated sitemap file before the file extension.
- outDir = 'public': (optional) the directory to create the sitemaps files.
- size = 50000: (optional) max size of the sitemap.#### RobotsTxtOptions
-
policies = []: (optional) policies for generating robots.txt.
- additionalSitemaps = []: (optional) add additionals sitemaps to robots.txt`. Fedeya
- Website: fedeminaya.com
- Twitter: @fedeminaya
- Github: @fedeya
- LinkedIn: @federico-minaya
Contributions, issues and feature requests are welcome!
Feel free to check issues page.
Give a ⭐️ if this project helped you!
- nasa-gcn/remix-seo for finding the posibility to use the server build at runtime.