2ff11df438
* Add .js to imports in the sitemap package * Adds a smoke test * Adds a changeset
142 lines
3.7 KiB
TypeScript
142 lines
3.7 KiB
TypeScript
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.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[];
|
|
canonicalURL?: string;
|
|
|
|
i18n?: {
|
|
defaultLocale: string;
|
|
locales: Record<string, string>;
|
|
};
|
|
// 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<SitemapItem>;
|
|
}
|
|
| 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;
|