制作自己的 RSSHub 路由
如前所述,我们以 GitHub 仓库 Issues 为例制作 RSS 源。我们将展示前面提到的四种数据获取方法:
通过 API
查看 API 文档
不同的站点有不同的 API。您可以查看要为其制作 RSS 源的站点的 API 文档。在本例中,我们将使用 GitHub Issues API。
创建主文件
打开您的代码编辑器并创建一个新文件。由于我们要为 GitHub 仓库 Issues 制作 RSS 源,因此建议将文件命名为 issue.js
。
以下是让您开始的基本代码:
- issue.js
// 导入所需模组
const got = require('@/utils/got'); // 自订的 got
const { parseDate } = require('@/utils/parse-date');
module.exports = async (ctx) => {
// 在此处编写您的逻辑
ctx.state.data = {
// 在此处输出您的 RSS
};
};
获取用户输入
如前所述,我们需要从用户输入中获取 GitHub 用户名和仓库名称。如果请求 URL 中未提供仓库名称,则应默认为 RSSHub
。您可以使用以下代码实现:
- 解构赋值
- 传统赋值
module.exports = async (ctx) => {
const { user, repo = 'RSSHub' } = ctx.params;
ctx.state.data = {
// 在此处输出您的 RSS
};
};
module.exports = async (ctx) => {
const user = ctx.params.user;
const repo = ctx.params.repo ?? 'RSSHub';
ctx.state.data = {
// 在此处输出您的 RSS
};
};
这两个代码片段都执行相同的操作。第一个使用对象解构将 user
和 repo
变量赋值,而第二个使用传统赋值和空值合并运算符在请求 URL 中未提供它的情况下将 repo
变量分配默认值 RSSHub
。
从 API 获取数据
在获取用户输入后,我们可以使用它向 API 发送请求。大多数情况下,您需要使用 @/utils/got
中的 got
(一个自订的 got 包装函数)发送 HTTP 请求。有关更多信息,请参阅 got 文档。
- 解构赋值
- 传统赋值
module.exports = async (ctx) => {
const { user, repo = 'RSSHub' } = ctx.params;
// 发送 HTTP GET 请求到 API 并解构返回的数据对 象
const { data } = await got(`https://api.github.com/repos/${user}/${repo}/issues`, {
headers: {
// 为简单起见,此示例使用 HTML 而不是推荐的 'application/vnd.github+json',
// 因后者返回 Markdown 并需要进一步处理
accept: 'application/vnd.github.html+json',
},
searchParams: {
// 这允许用户设置条数限制
per_page: ctx.query.limit ? parseInt(ctx.query.limit, 10) : 30,
},
});
ctx.state.data = {
// 在此处输出您的 RSS
};
};
module.exports = async (ctx) => {
const user = ctx.params.user;
const repo = ctx.params.repo ?? 'RSSHub';
// 发送 HTTP GET 请求到 API
const response = await got(`https://api.github.com/repos/${user}/${repo}/issues`, {
headers: {
accept: 'application/vnd.github.html+json',
},
searchParams: {
per_page: ctx.query.limit ? parseInt(ctx.query.limit, 10) : 30,
},
});
// response.data 是上述请求返回的数据对象
const data = response.data;
ctx.state.data = {
// 在此处输出您的 RSS
};
};
生成 RSS 源
一旦我们从 API 获取到数据,我们需要进一步处理它以生成符合 RSS 规范的 RSS 源。具体来说,我们需要提取源标题、源链接、 文章标题、文章链接、文章正文和文章发布日期。
为此,我们可以将相关数据赋值给 ctx.state.data
对象,RSSHub 的中间件将处理其余部分。
以下是应有的最终代码:
- 最终代码
- 替代代码
const got = require('@/utils/got');
const { parseDate } = require('@/utils/parse-date');
module.exports = async (ctx) => {
const { user, repo = 'RSSHub' } = ctx.params;
const { data } = await got(`https://api.github.com/repos/${user}/${repo}/issues`, {
headers: {
accept: 'application/vnd.github.html+json',
},
searchParams: {
per_page: ctx.query.limit ? parseInt(ctx.query.limit, 10) : 30,
},
});
// 从 API 响应中提取相关数据
const items = data.map((item) => ({
// 文章标题
title: item.title,
// 文章链接
link: item.html_url,
// 文章正文
description: item.body_html,
// 文章发布日期
pubDate: parseDate(item.created_at),
// 如果有的话,文章作者
author: item.user.login,
// 如果有的话,文章分类
category: item.labels.map((label) => label.name),
}));
ctx.state.data = {
// 源标题
title: `${user}/${repo} issues`,
// 源链接
link: `https://github.com/${user}/${repo}/issues`,
// 源文章
item: items,
};
};
const got = require('@/utils/got');
const { parseDate } = require('@/utils/parse-date');
module.exports = async (ctx) => {
const { user, repo = 'RSSHub' } = ctx.params;
const { data } = await got(`https://api.github.com/repos/${user}/${repo}/issues`, {
headers: {
accept: 'application/vnd.github.html+json',
},
searchParams: {
per_page: ctx.query.limit ? parseInt(ctx.query.limit, 10) : 30,
},
});
ctx.state.data = {
// 源标题
title: `${user}/${repo} issues`,
// 源链接
link: `https://github.com/${user}/${repo}/issues`,
// 遍历所有此前获取的数据
item: data.map((item) => ({
// 文章标题
title: item.title,
// 文章链接
link: item.html_url,
// 文章正文
description: item.body_html,
// 文章发布日期
pubDate: parseDate(item.created_at),
// 如果有的话,文章作者
author: item.user.login,
// 如果有的话,文章分类
category: item.labels.map((label) => label.name),
}));
};
};