Astro.build: Generate a simple sitemap.xml

This post is outdated and needs to be rewritten to reflect the current state of Astro.build and how I’ve implemented the sitemap.xml on my website. It has been 2 years since I wrote this, and a lot of things have changed since. I’ll update this post as soon as possible.

It might not be perfect code, I’m well aware of that. And it might not work for your setup, but with some modifications it might :)

In this post I’ll explain the solution I’ve implemented on my photography portfolio site 31f-fotografie.nl

First I’ve created a file lib/util.ts with 2 functions.

import { getCollection } from 'astro:content';

export async function getSortedCollectionPosts({ collection }) {
  return await getCollection(collection).then((entries) =>
    entries
      .filter((entry) => !entry.data.draft)
      .sort(
        (a, b) =>
          new Date(b.data.date).valueOf() - new Date(a.data.date).valueOf()
      )
  );
}

export async function generateSitemapXml({ posts, site }) {
  return `
    <?xml version="1.0" encoding="UTF-8"?>
    <urlset xsi:schemaLocation="http://www.sitemaps.org/schemas/sitemap/0.9 http://www.sitemaps.org/schemas/sitemap/0.9/sitemap.xsd" xmlns="http://www.sitemaps.org/schemas/sitemap/0.9" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
    <url>
        <loc>${site.url}</loc>
        <lastmod>${posts[0].data.date.toISOString()}</lastmod>
        <priority>1.00</priority>
    </url>
    ${posts
      .map((post) => {
        const loc = new URL(`/${post.collection}/${post.slug}`, site.url).href;
        return `
            <url>
                <loc>${loc}</loc>
                <lastmod>${post.data.date.toISOString()}</lastmod>
                <priority>0.80</priority>
            </url>
        `.trim();
      })
      .join('')}
    </urlset>
`.trim();
}

The getSortedCollectionPosts() function is also used on the homepage to generate the lost of posts.

Then the pages/sitemap.xml.ts file that looks something like this:

import { getSortedCollectionPosts, generateSitemapXml } from '../lib/util';

export async function get() {
  const posts = await getSortedCollectionPosts({ collection: 'post' });

  return {
    body: await generateSitemapXml({
      posts,
      site: { url: import.meta.env.SITE },
    }),
  };
}

Make sure you have defined your site in astro.config.mjs

export default defineConfig({
  site: 'https://31f-fotografie.nl',
  ...
});

Now you should be able to load your websites’ sitemap.xml

last updated: 2025-03-11

I hope this post was helpful to you.

If you have any questions or feedback, feel free to contact me on Twitter/X, Bluesky, or Mastodon. I’d love to hear from you! 🚀