fix(route): github comments use api (#9358)

fix(route): github comments use api
This commit is contained in:
Tony
2022-03-21 17:17:59 +00:00
committed by GitHub
parent 6b3d21438e
commit fe775f5198
10 changed files with 107 additions and 48 deletions

View File

@@ -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,
};
};