mirror of
https://github.com/DIYgod/RSSHub.git
synced 2025-12-08 14:07:54 +08:00
fix(route): github comments use api (#9358)
fix(route): github comments use api
This commit is contained in:
@@ -534,7 +534,7 @@ See docs of the specified route and `lib/config.js` for detailed information.
|
||||
- `EH_SK`: The value of `sk` in the cookie header after logging in E-Hentai
|
||||
- `EH_IGNEOUS`: The value of `igneous` in the cookie header after logging in ExHentai. If this value is set, RSS will be generated from ExHentai, `EH_SK` will be ignored.
|
||||
|
||||
- github: [Access Token application](https://github.com/settings/tokens)
|
||||
- GitHub: [Access Token application](https://github.com/settings/tokens)
|
||||
|
||||
- `GITHUB_ACCESS_TOKEN`: GitHub Access Token
|
||||
|
||||
|
||||
@@ -191,9 +191,9 @@ For instance, the `/github/topics/framework/l=php&o=desc&s=stars` route will gen
|
||||
|
||||
<RouteEn author="zoenglinghou" example="/github/contributors/DIYgod/RSSHub" path="/github/contributors/:user/:repo/:order?/:anon?" :paramsDesc="['User name','Repo name','Sort order by commit numbers, desc and asc (descending by default)','Show anonymous users. Defaults to no, use any values for yes.']" radar="1" rssbud="1"/>
|
||||
|
||||
### Issues / Pull Requests comments
|
||||
### Issue / Pull Request comments
|
||||
|
||||
<RouteEn author="TonyRL" example="/github/comments/DIYgod/RSSHub/issues/8116" path="/github/comments/:user/:repo/:type/:number" :paramsDesc="['User / Org name', 'Repo name', 'Type, `issues` or `pull`', 'Number']"/>
|
||||
<RouteEn author="TonyRL" example="/github/comments/DIYgod/RSSHub/8116" path="/github/comments/:user/:repo/:number" :paramsDesc="['User / Org name', 'Repo name', 'Issue or pull number']"/>
|
||||
|
||||
## GitLab
|
||||
|
||||
|
||||
@@ -565,7 +565,7 @@ RSSHub 支持使用访问密钥 / 码,白名单和黑名单三种方式进行
|
||||
- `EH_SK`: E-Hentai 账户登录后 cookie 中的`sk`值
|
||||
- `EH_IGNEOUS`: ExHentai 账户登录后 cookie 中的`igneous`值。若设置此值,RSS 数据将全部从里站获取,`EH_SK`将被忽略
|
||||
|
||||
- github 全部路由:[申请地址](https://github.com/settings/tokens)
|
||||
- GitHub 全部路由:[申请地址](https://github.com/settings/tokens)
|
||||
|
||||
- `GITHUB_ACCESS_TOKEN`: GitHub Access Token
|
||||
|
||||
|
||||
@@ -269,9 +269,9 @@ GitHub 官方也提供了一些 RSS:
|
||||
|
||||
<Route author="zoenglinghou" example="/github/contributors/DIYgod/RSSHub" path="/github/contributors/:user/:repo/:order?/:anon?" :paramsDesc="['用户名', '仓库名', 'Commit 数量排序顺序,desc和asc(默认desc降序)', '是否包括匿名用户,默认不包含,任意值包含匿名用户']" radar="1" rssbud="1"/>
|
||||
|
||||
### Issues / Pull Requests 评论
|
||||
### Issue / Pull Request 评论
|
||||
|
||||
<Route author="TonyRL" example="/github/comments/DIYgod/RSSHub/issues/8116" path="/github/comments/:user/:repo/:type/:number" :paramsDesc="['用户名', '仓库', '类型,`issues`或`pull`', '编号']"/>
|
||||
<Route author="TonyRL" example="/github/comments/DIYgod/RSSHub/8116" path="/github/comments/:user/:repo/:number" :paramsDesc="['用户名', '仓库', 'Issue 或 pull 编号']"/>
|
||||
|
||||
## GitLab
|
||||
|
||||
|
||||
@@ -1,64 +1,122 @@
|
||||
const got = require('@/utils/got');
|
||||
const cheerio = require('cheerio');
|
||||
const { parseDate } = require('@/utils/parse-date');
|
||||
const { art } = require('@/utils/render');
|
||||
const path = require('path');
|
||||
|
||||
const md = require('markdown-it')({
|
||||
html: true,
|
||||
});
|
||||
const apiUrl = 'https://api.github.com';
|
||||
const config = require('@/config').value;
|
||||
const typeDict = {
|
||||
issues: {
|
||||
typeRootUrl: '/issues',
|
||||
issue: {
|
||||
title: 'Issue',
|
||||
},
|
||||
pull: {
|
||||
typeRootUrl: '/pulls',
|
||||
title: 'Pull request',
|
||||
},
|
||||
};
|
||||
|
||||
module.exports = async (ctx) => {
|
||||
const user = ctx.params.user;
|
||||
const repo = ctx.params.repo;
|
||||
const type = ctx.params.type ?? 'issues';
|
||||
const number = ctx.params.number ?? '1';
|
||||
|
||||
const rootUrl = 'https://github.com';
|
||||
const number = isNaN(parseInt(ctx.params.number)) ? 1 : parseInt(ctx.params.number);
|
||||
const limit = ctx.query.limit ? parseInt(ctx.params.limit) : 100;
|
||||
const headers =
|
||||
config.github && config.github.access_token
|
||||
? {
|
||||
Accept: 'application/vnd.github.v3+json',
|
||||
Authorization: `token ${config.github.access_token}`,
|
||||
}
|
||||
: {
|
||||
Accept: 'application/vnd.github.v3+json',
|
||||
};
|
||||
|
||||
const response = await got({
|
||||
method: 'get',
|
||||
url: `${rootUrl}/${user}/${repo}/${type}/${number}`,
|
||||
header: {
|
||||
Referer: `${rootUrl}/${user}/${repo}${typeDict[type].typeRootUrl}`,
|
||||
url: `${apiUrl}/repos/${user}/${repo}/issues/${number}`,
|
||||
headers,
|
||||
});
|
||||
const issue = response.data;
|
||||
const type = issue.pull_request ? 'pull' : 'issue';
|
||||
|
||||
const timelineResponse = await got({
|
||||
url: issue.timeline_url,
|
||||
headers,
|
||||
searchParams: {
|
||||
per_page: limit,
|
||||
},
|
||||
});
|
||||
const timeline = timelineResponse.data;
|
||||
|
||||
const $ = cheerio.load(response.data);
|
||||
let items = [
|
||||
{
|
||||
title: `${issue.user.login} created ${user}/${repo}: ${typeDict[type].title} #${issue.number}`,
|
||||
description: md.render(issue.body),
|
||||
author: issue.user.login,
|
||||
pubDate: parseDate(issue.created_at),
|
||||
link: `${issue.html_url}#issue-${issue.id}`,
|
||||
},
|
||||
];
|
||||
|
||||
const list = $('div.timeline-comment-group.js-minimizable-comment-group.js-targetable-element.TimelineItem-body.my-0')
|
||||
.map((_, item) => {
|
||||
item = $(item);
|
||||
return {
|
||||
title: `${item.find('strong.css-truncate').text().trim()} commented`,
|
||||
description: art(path.join(__dirname, 'templates/comments-description.art'), {
|
||||
desc: item.find('div.edit-comment-hide').html(),
|
||||
}),
|
||||
author: item.find('strong.css-truncate').text().trim(),
|
||||
pubDate: parseDate(item.find('relative-time.no-wrap').attr('datetime')),
|
||||
link: `${rootUrl}/${user}/${repo}/${type}/${number}` + item.find('a.Link--secondary.js-timestamp').attr('href'),
|
||||
};
|
||||
})
|
||||
.get();
|
||||
timeline.forEach((item) => {
|
||||
switch (item.event) {
|
||||
case 'closed':
|
||||
items.push({
|
||||
title: `${item.actor.login} ${item.event} ${user}/${repo}: ${typeDict[type].title} #${issue.number}`,
|
||||
author: item.actor.login,
|
||||
pubDate: parseDate(item.created_at),
|
||||
link: item.url,
|
||||
});
|
||||
break;
|
||||
case 'commented':
|
||||
items.push({
|
||||
title: `${item.actor.login} ${item.event} on ${user}/${repo}: ${typeDict[type].title} #${issue.number}`,
|
||||
description: md.render(item.body),
|
||||
author: item.actor.login,
|
||||
pubDate: parseDate(item.created_at),
|
||||
link: item.html_url,
|
||||
});
|
||||
break;
|
||||
case 'cross-referenced':
|
||||
items.push({
|
||||
title: `${item.actor.login} ${item.event} on ${user}/${repo}: ${typeDict[type].title} #${issue.number}`,
|
||||
description: `${item.actor.login} mentioned this issue in <a href='${item.source.issue.html_url}'><b>${item.source.issue.title}</b> #${item.source.issue.number}</a>`,
|
||||
author: item.actor.login,
|
||||
pubDate: parseDate(item.created_at),
|
||||
guid: `${item.actor.login} ${item.event} on ${user}/${repo}: ${typeDict[type].title} #${issue.number} on ${item.created_at}`,
|
||||
});
|
||||
break;
|
||||
case 'renamed':
|
||||
items.push({
|
||||
title: `${item.actor.login} ${item.event} on ${user}/${repo}: ${typeDict[type].title} #${issue.number}`,
|
||||
description: `${item.actor.login} changed the title <del>${item.rename.from}</del> ${item.rename.to}`,
|
||||
author: item.actor.login,
|
||||
pubDate: parseDate(item.created_at),
|
||||
link: item.url,
|
||||
});
|
||||
break;
|
||||
case 'reviewed':
|
||||
items.push({
|
||||
title: `${item.user.login} ${item.event} on ${user}/${repo}: ${typeDict[type].title} #${issue.number}`,
|
||||
description: item.body ? md.render(item.body) : item.state.replace('_', ' '),
|
||||
author: item.user.login,
|
||||
pubDate: parseDate(item.submitted_at),
|
||||
link: item.html_url,
|
||||
});
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
});
|
||||
|
||||
const items = await Promise.all(list && list.map((item) => ctx.cache.tryGet(item.link, () => item)));
|
||||
items = await Promise.all(items.map((item) => ctx.cache.tryGet(item.link, () => item)));
|
||||
|
||||
ctx.state.data = {
|
||||
title: $('span.author.flex-self-stretch').text().trim() + '/' + $('strong.mr-2.flex-self-stretch').text().trim() + `: ${type} #${number}`,
|
||||
description: $('span.author.flex-self-stretch').text().trim() + '/' + $('strong.mr-2.flex-self-stretch').text().trim() + ': ' + $('h1.gh-header-title.mb-2.lh-condensed.f1.mr-0.flex-auto.break-word').text().trim(),
|
||||
link: `${rootUrl}/${user}/${repo}/${type}/${number}`,
|
||||
title: `${user}/${repo}: ${typeDict[type].title} #${number} - ${issue.title}`,
|
||||
link: issue.html_url,
|
||||
item: items,
|
||||
};
|
||||
|
||||
ctx.state.json = {
|
||||
title: $('span.author.flex-self-stretch').text().trim() + '/' + $('strong.mr-2.flex-self-stretch').text().trim() + `: ${type} #${number}`,
|
||||
description: $('span.author.flex-self-stretch').text().trim() + '/' + $('strong.mr-2.flex-self-stretch').text().trim() + ': ' + $('h1.gh-header-title.mb-2.lh-condensed.f1.mr-0.flex-auto.break-word').text().trim(),
|
||||
link: `${rootUrl}/${user}/${repo}/${type}/${number}`,
|
||||
title: `${user}/${repo}: ${typeDict[type].title} #${number} - ${issue.title}`,
|
||||
link: issue.html_url,
|
||||
item: items,
|
||||
};
|
||||
};
|
||||
|
||||
@@ -3,7 +3,7 @@ const config = require('@/config').value;
|
||||
|
||||
module.exports = async (ctx) => {
|
||||
if (!config.github || !config.github.access_token) {
|
||||
throw 'GitHub follower RSS is disabled due to the lack of <a href="https://docs.rsshub.app/install/#%E9%83%A8%E5%88%86-rss-%E6%A8%A1%E5%9D%97%E9%85%8D%E7%BD%AE">relevant config</a>';
|
||||
throw 'GitHub follower RSS is disabled due to the lack of <a href="https://docs.rsshub.app/install/#pei-zhi-bu-fen-rss-mo-kuai-pei-zhi">relevant config</a>';
|
||||
}
|
||||
const user = ctx.params.user;
|
||||
|
||||
|
||||
@@ -6,7 +6,7 @@ module.exports = {
|
||||
title: 'Issues / Pull Requests 评论',
|
||||
docs: 'https://docs.rsshub.app/programming.html#github',
|
||||
source: ['/:user/:repo/:type/:number'],
|
||||
target: '/github/comments/:user/:repo/:type/:number',
|
||||
target: '/github/comments/:user/:repo/:number',
|
||||
},
|
||||
{
|
||||
title: '用户仓库',
|
||||
|
||||
@@ -1,6 +1,7 @@
|
||||
module.exports = function (router) {
|
||||
router.get('/branches/:user/:repo', require('./branches'));
|
||||
router.get('/comments/:user/:repo/:type/:number', require('./comments'));
|
||||
router.get('/comments/:user/:repo/:type/:number', require('./comments')); // deprecated
|
||||
router.get('/comments/:user/:repo/:number', require('./comments'));
|
||||
router.get('/contributors/:user/:repo/:order?/:anon?', require('./contributors'));
|
||||
router.get('/file/:user/:repo/:branch/:filepath+', require('./file'));
|
||||
router.get('/issue/:user/:repo/:state?/:labels?', require('./issue'));
|
||||
|
||||
@@ -3,7 +3,7 @@ const config = require('@/config').value;
|
||||
|
||||
module.exports = async (ctx) => {
|
||||
if (!config.github || !config.github.access_token) {
|
||||
throw 'GitHub star RSS is disabled due to the lack of <a href="https://docs.rsshub.app/install/#%E9%83%A8%E5%88%86-rss-%E6%A8%A1%E5%9D%97%E9%85%8D%E7%BD%AE">relevant config</a>';
|
||||
throw 'GitHub star RSS is disabled due to the lack of <a href="https://docs.rsshub.app/install/#pei-zhi-bu-fen-rss-mo-kuai-pei-zhi">relevant config</a>';
|
||||
}
|
||||
const user = ctx.params.user;
|
||||
const repo = ctx.params.repo;
|
||||
|
||||
@@ -3,7 +3,7 @@ const config = require('@/config').value;
|
||||
|
||||
module.exports = async (ctx) => {
|
||||
if (!config.github || !config.github.access_token) {
|
||||
throw 'GitHub star RSS is disabled due to the lack of <a href="https://docs.rsshub.app/install/#%E9%83%A8%E5%88%86-rss-%E6%A8%A1%E5%9D%97%E9%85%8D%E7%BD%AE">relevant config</a>';
|
||||
throw 'GitHub star RSS is disabled due to the lack of <a href="https://docs.rsshub.app/install/#pei-zhi-bu-fen-rss-mo-kuai-pei-zhi">relevant config</a>';
|
||||
}
|
||||
const user = ctx.params.user;
|
||||
|
||||
|
||||
Reference in New Issue
Block a user