* fix: filter out draft item from glob result * test: rss should ignore draft * build: add changeset * build: major version * feat: add `drafts` option * Add README docs Co-authored-by: Chris Swithinbank <swithinbank@gmail.com>
198 lines
6.2 KiB
TypeScript
198 lines
6.2 KiB
TypeScript
import { XMLBuilder, XMLParser } from 'fast-xml-parser';
|
|
import { createCanonicalURL, isValidURL } from './util.js';
|
|
|
|
type GlobResult = Record<string, () => Promise<{ [key: string]: any }>>;
|
|
|
|
type RSSOptions = {
|
|
/** (required) Title of the RSS Feed */
|
|
title: string;
|
|
/** (required) Description of the RSS Feed */
|
|
description: string;
|
|
/**
|
|
* Specify the base URL to use for RSS feed links.
|
|
* We recommend "import.meta.env.SITE" to pull in the "site"
|
|
* from your project's astro.config.
|
|
*/
|
|
site: string;
|
|
/**
|
|
* List of RSS feed items to render. Accepts either:
|
|
* a) list of RSSFeedItems
|
|
* b) import.meta.glob result. You can only glob ".md" (or alternative extensions for markdown files like ".markdown") files within src/pages/ when using this method!
|
|
*/
|
|
items: RSSFeedItem[] | GlobResult;
|
|
/** Specify arbitrary metadata on opening <xml> tag */
|
|
xmlns?: Record<string, string>;
|
|
/**
|
|
* Specifies a local custom XSL stylesheet. Ex. '/public/custom-feed.xsl'
|
|
*/
|
|
stylesheet?: string | boolean;
|
|
/** Specify custom data in opening of file */
|
|
customData?: string;
|
|
/** Whether to include drafts or not */
|
|
drafts?: boolean;
|
|
};
|
|
|
|
type RSSFeedItem = {
|
|
/** Link to item */
|
|
link: string;
|
|
/** Title of item */
|
|
title: string;
|
|
/** Publication date of item */
|
|
pubDate: Date;
|
|
/** Item description */
|
|
description?: string;
|
|
/** Full content of the item, should be valid HTML */
|
|
content?: string;
|
|
/** Append some other XML-valid data to this item */
|
|
customData?: string;
|
|
/** Whether draft or not */
|
|
draft?: boolean;
|
|
};
|
|
|
|
type GenerateRSSArgs = {
|
|
rssOptions: RSSOptions;
|
|
items: RSSFeedItem[];
|
|
};
|
|
|
|
function isGlobResult(items: RSSOptions['items']): items is GlobResult {
|
|
return typeof items === 'object' && !items.length;
|
|
}
|
|
|
|
function mapGlobResult(items: GlobResult): Promise<RSSFeedItem[]> {
|
|
return Promise.all(
|
|
Object.values(items).map(async (getInfo) => {
|
|
const { url, frontmatter } = await getInfo();
|
|
if (url === undefined || url === null) {
|
|
throw new Error(
|
|
`[RSS] When passing an import.meta.glob result directly, you can only glob ".md" (or alternative extensions for markdown files like ".markdown") files within /pages! Consider mapping the result to an array of RSSFeedItems. See the RSS docs for usage examples: https://docs.astro.build/en/guides/rss/#2-list-of-rss-feed-objects`
|
|
);
|
|
}
|
|
if (!Boolean(frontmatter.title) || !Boolean(frontmatter.pubDate)) {
|
|
throw new Error(`[RSS] "${url}" is missing a "title" and/or "pubDate" in its frontmatter.`);
|
|
}
|
|
return {
|
|
link: url,
|
|
title: frontmatter.title,
|
|
pubDate: frontmatter.pubDate,
|
|
description: frontmatter.description,
|
|
customData: frontmatter.customData,
|
|
draft: frontmatter.draft,
|
|
};
|
|
})
|
|
);
|
|
}
|
|
|
|
export default async function getRSS(rssOptions: RSSOptions) {
|
|
const { site } = rssOptions;
|
|
let { items } = rssOptions;
|
|
|
|
if (!site) {
|
|
throw new Error('[RSS] the "site" option is required, but no value was given.');
|
|
}
|
|
|
|
if (isGlobResult(items)) {
|
|
items = await mapGlobResult(items);
|
|
if (!rssOptions.drafts) {
|
|
items = items.filter((item) => !item.draft);
|
|
}
|
|
}
|
|
|
|
return {
|
|
body: await generateRSS({
|
|
rssOptions,
|
|
items,
|
|
}),
|
|
};
|
|
}
|
|
|
|
/** Generate RSS 2.0 feed */
|
|
export async function generateRSS({ rssOptions, items }: GenerateRSSArgs): Promise<string> {
|
|
const { site } = rssOptions;
|
|
const xmlOptions = { ignoreAttributes: false };
|
|
const parser = new XMLParser(xmlOptions);
|
|
const root: any = { '?xml': { '@_version': '1.0', '@_encoding': 'UTF-8' } };
|
|
if (typeof rssOptions.stylesheet === 'string') {
|
|
const isXSL = /\.xsl$/i.test(rssOptions.stylesheet);
|
|
root['?xml-stylesheet'] = {
|
|
'@_href': rssOptions.stylesheet,
|
|
...(isXSL && { '@_type': 'text/xsl' }),
|
|
};
|
|
}
|
|
root.rss = { '@_version': '2.0' };
|
|
if (items.find((result) => result.content)) {
|
|
// the namespace to be added to the xmlns:content attribute to enable the <content> RSS feature
|
|
const XMLContentNamespace = 'http://purl.org/rss/1.0/modules/content/';
|
|
root.rss['@_xmlns:content'] = XMLContentNamespace;
|
|
// Ensure that the user hasn't tried to manually include the necessary namespace themselves
|
|
if (rssOptions.xmlns?.content && rssOptions.xmlns.content === XMLContentNamespace) {
|
|
delete rssOptions.xmlns.content;
|
|
}
|
|
}
|
|
|
|
// xmlns
|
|
if (rssOptions.xmlns) {
|
|
for (const [k, v] of Object.entries(rssOptions.xmlns)) {
|
|
root.rss[`@_xmlns:${k}`] = v;
|
|
}
|
|
}
|
|
|
|
// title, description, customData
|
|
root.rss.channel = {
|
|
title: rssOptions.title,
|
|
description: rssOptions.description,
|
|
link: createCanonicalURL(site).href,
|
|
};
|
|
if (typeof rssOptions.customData === 'string')
|
|
Object.assign(
|
|
root.rss.channel,
|
|
parser.parse(`<channel>${rssOptions.customData}</channel>`).channel
|
|
);
|
|
// items
|
|
root.rss.channel.item = items.map((result) => {
|
|
validate(result);
|
|
// If the item's link is already a valid URL, don't mess with it.
|
|
const itemLink = isValidURL(result.link)
|
|
? result.link
|
|
: createCanonicalURL(result.link, site).href;
|
|
const item: any = {
|
|
title: result.title,
|
|
link: itemLink,
|
|
guid: itemLink,
|
|
};
|
|
if (result.description) {
|
|
item.description = result.description;
|
|
}
|
|
if (result.pubDate) {
|
|
// note: this should be a Date, but if user provided a string or number, we can work with that, too.
|
|
if (typeof result.pubDate === 'number' || typeof result.pubDate === 'string') {
|
|
result.pubDate = new Date(result.pubDate);
|
|
} else if (result.pubDate instanceof Date === false) {
|
|
throw new Error('[${filename}] rss.item().pubDate must be a Date');
|
|
}
|
|
item.pubDate = result.pubDate.toUTCString();
|
|
}
|
|
// include the full content of the post if the user supplies it
|
|
if (typeof result.content === 'string') {
|
|
item['content:encoded'] = result.content;
|
|
}
|
|
if (typeof result.customData === 'string') {
|
|
Object.assign(item, parser.parse(`<item>${result.customData}</item>`).item);
|
|
}
|
|
return item;
|
|
});
|
|
|
|
return new XMLBuilder(xmlOptions).build(root);
|
|
}
|
|
|
|
const requiredFields = Object.freeze(['link', 'title']);
|
|
|
|
// Perform validation to make sure all required fields are passed.
|
|
function validate(item: RSSFeedItem) {
|
|
for (const field of requiredFields) {
|
|
if (!(field in item)) {
|
|
throw new Error(
|
|
`@astrojs/rss: Required field [${field}] is missing. RSS cannot be generated without it.`
|
|
);
|
|
}
|
|
}
|
|
}
|