import type { AstroConfig, AstroIntegration } from 'astro'; import { fileURLToPath } from 'node:url'; import { EnumChangefreq, simpleSitemapAndIndex, type LinkItem as LinkItemBase, type SitemapItemLoose, } from 'sitemap'; import { ZodError } from 'zod'; import { generateSitemap } from './generate-sitemap.js'; import { Logger } from './utils/logger.js'; import { validateOptions } from './validate-options.js'; export { EnumChangefreq as ChangeFreqEnum } from 'sitemap'; export type ChangeFreq = `${EnumChangefreq}`; export type SitemapItem = Pick< SitemapItemLoose, 'url' | 'lastmod' | 'changefreq' | 'priority' | 'links' >; export type LinkItem = LinkItemBase; export type SitemapOptions = | { filter?(page: string): boolean; customPages?: string[]; i18n?: { defaultLocale: string; locales: Record; }; // number of entries per sitemap file entryLimit?: number; // sitemap specific changefreq?: ChangeFreq; lastmod?: Date; priority?: number; // called for each sitemap item just before to save them on disk, sync or async serialize?(item: SitemapItem): SitemapItem | Promise | undefined; } | undefined; function formatConfigErrorMessage(err: ZodError) { const errorList = err.issues.map((issue) => ` ${issue.path.join('.')} ${issue.message + '.'}`); return errorList.join('\n'); } const PKG_NAME = '@astrojs/sitemap'; const OUTFILE = 'sitemap-index.xml'; const STATUS_CODE_PAGES = new Set(['404', '500']); function isStatusCodePage(pathname: string): boolean { if (pathname.endsWith('/')) { pathname = pathname.slice(0, -1); } const end = pathname.split('/').pop() ?? ''; return STATUS_CODE_PAGES.has(end); } const createPlugin = (options?: SitemapOptions): AstroIntegration => { let config: AstroConfig; const logger = new Logger(PKG_NAME); return { name: PKG_NAME, hooks: { 'astro:config:done': async ({ config: cfg }) => { config = cfg; }, 'astro:build:done': async ({ dir, routes, pages }) => { try { if (!config.site) { logger.warn( 'The Sitemap integration requires the `site` astro.config option. Skipping.' ); return; } const opts = validateOptions(config.site, options); const { filter, customPages, serialize, entryLimit } = opts; let finalSiteUrl: URL; if (config.site) { finalSiteUrl = new URL(config.base, config.site); } else { console.warn( 'The Sitemap integration requires the `site` astro.config option. Skipping.' ); return; } let pageUrls = pages .filter((p) => !isStatusCodePage(p.pathname)) .map((p) => { if (p.pathname !== '' && !finalSiteUrl.pathname.endsWith('/')) finalSiteUrl.pathname += '/'; const path = finalSiteUrl.pathname + p.pathname; return new URL(path, finalSiteUrl).href; }); let routeUrls = routes.reduce((urls, r) => { // Only expose pages, not endpoints or redirects if (r.type !== 'page') return urls; /** * Dynamic URLs have entries with `undefined` pathnames */ if (r.pathname) { if (isStatusCodePage(r.pathname ?? r.route)) return urls; /** * remove the initial slash from relative pathname * because `finalSiteUrl` always has trailing slash */ const path = finalSiteUrl.pathname + r.generate(r.pathname).substring(1); let newUrl = new URL(path, finalSiteUrl).href; if (config.trailingSlash === 'never') { urls.push(newUrl); } else if (config.build.format === 'directory' && !newUrl.endsWith('/')) { urls.push(newUrl + '/'); } else { urls.push(newUrl); } } return urls; }, []); pageUrls = Array.from(new Set([...pageUrls, ...routeUrls, ...(customPages ?? [])])); try { if (filter) { pageUrls = pageUrls.filter(filter); } } catch (err) { logger.error(`Error filtering pages\n${(err as any).toString()}`); return; } if (pageUrls.length === 0) { logger.warn(`No pages found!\n\`${OUTFILE}\` not created.`); return; } let urlData = generateSitemap(pageUrls, finalSiteUrl.href, opts); if (serialize) { try { const serializedUrls: SitemapItem[] = []; for (const item of urlData) { const serialized = await Promise.resolve(serialize(item)); if (serialized) { serializedUrls.push(serialized); } } if (serializedUrls.length === 0) { logger.warn('No pages found!'); return; } urlData = serializedUrls; } catch (err) { logger.error(`Error serializing pages\n${(err as any).toString()}`); return; } } await simpleSitemapAndIndex({ hostname: finalSiteUrl.href, destinationDir: fileURLToPath(dir), sourceData: urlData, limit: entryLimit, gzip: false, }); logger.success(`\`${OUTFILE}\` is created.`); } catch (err) { if (err instanceof ZodError) { logger.warn(formatConfigErrorMessage(err)); } else { throw err; } } }, }, }; }; export default createPlugin;