import type { AstroConfig, AstroIntegration } from 'astro'; import { EnumChangefreq, simpleSitemapAndIndex, type LinkItem as LinkItemBase, type SitemapItemLoose, } from 'sitemap'; import { fileURLToPath } from 'url'; import { ZodError } from 'zod'; import { generateSitemap } from './generate-sitemap.js'; import { Logger } from './utils/logger.js'; import { validateOptions } from './validate-options.js'; 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 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 }) => { 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 { // eslint-disable-next-line no-console console.warn( 'The Sitemap integration requires the `site` astro.config option. Skipping.' ); return; } let pageUrls = routes.reduce((urls, r) => { /** * Dynamic URLs have entries with `undefined` pathnames */ if (r.pathname) { /** * 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; }, []); try { if (filter) { pageUrls = pageUrls.filter(filter); } } catch (err) { logger.error(`Error filtering pages\n${(err as any).toString()}`); return; } if (customPages) { pageUrls = Array.from(new Set([...pageUrls, ...customPages])); } 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;