import type { AstroConfig, AstroIntegration } from 'astro'; import { EnumChangefreq, LinkItem as LinkItemBase, simpleSitemapAndIndex, SitemapItemLoose, } from 'sitemap'; import { fileURLToPath } from 'url'; import { ZodError } from 'zod'; import { generateSitemap } from './generate-sitemap'; import { Logger } from './utils/logger'; import { validateOptions } from './validate-options'; 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[]; canonicalURL?: 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; 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; return { name: PKG_NAME, hooks: { 'astro:config:done': async ({ config: cfg }) => { config = cfg; }, 'astro:build:done': async ({ dir, pages }) => { const logger = new Logger(PKG_NAME); try { const opts = validateOptions(config.site, options); const { filter, customPages, canonicalURL, serialize, entryLimit } = opts; let finalSiteUrl: URL; if (canonicalURL) { finalSiteUrl = new URL(canonicalURL); if (!finalSiteUrl.pathname.endsWith('/')) { finalSiteUrl.pathname += '/'; // normalizes the final url since it's provided by user } } else { // `validateOptions` forces to provide `canonicalURL` or `config.site` at least. // So step to check on empty values of `canonicalURL` and `config.site` is dropped. finalSiteUrl = new URL(config.base, config.site); } let pageUrls = pages.map((p) => { const path = finalSiteUrl.pathname + p.pathname; return new URL(path, finalSiteUrl).href; }); try { if (filter) { pageUrls = pageUrls.filter(filter); } } catch (err) { logger.error(`Error filtering pages\n${(err as any).toString()}`); return; } if (customPages) { pageUrls = [...pageUrls, ...customPages]; } if (pageUrls.length === 0) { logger.warn(`No data for sitemap.\n\`${OUTFILE}\` is 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)); serializedUrls.push(serialized); } 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;