Skip to content

Commit

Permalink
Merge pull request #64 from DIYgod/master
Browse files Browse the repository at this point in the history
[pull] master from diygod:master
  • Loading branch information
pull[bot] authored Apr 14, 2024
2 parents 79d232a + 517fa8d commit 6063786
Show file tree
Hide file tree
Showing 5 changed files with 167 additions and 76 deletions.
41 changes: 41 additions & 0 deletions lib/routes/njuferret/blog.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,41 @@
import { Route } from '@/types';
import ofetch from '@/utils/ofetch';
import { load } from 'cheerio';
import { parseDate } from '@/utils/parse-date';

export const route: Route = {
path: '/blog',
categories: ['blog'],
example: '/njuferret/blog',
radar: [
{
source: ['njuferret.github.io'],
},
],
name: 'Blogs',
maintainers: ['tyl0622'],
handler,
};

async function handler() {
const baseUrl = 'https://njuferret.github.io';
const response = await ofetch(baseUrl);
const $ = load(response);

const items = $('div.post-block')
.toArray()
.map((item) => {
item = $(item);
const a = item.find('a').first();
return {
title: a.text(),
link: `${baseUrl}${a.attr('href')}`,
pubDate: parseDate(item.find('time').attr('datetime')),
};
});

return {
title: 'njuferret - blog',
item: items,
};
}
10 changes: 10 additions & 0 deletions lib/routes/njuferret/namespace.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
import type { Namespace } from '@/types';

export const namespace: Namespace = {
name: 'njuferret',
url: 'njuferret.github.io',
description: '',
zh: {
name: '有点博客',
},
};
159 changes: 100 additions & 59 deletions lib/routes/papers/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,86 +2,127 @@ import { Route } from '@/types';
import { getCurrentPath } from '@/utils/helpers';
const __dirname = getCurrentPath(import.meta.url);

import got from '@/utils/got';
import { load } from 'cheerio';
import { parseDate } from '@/utils/parse-date';
import { art } from '@/utils/render';
import path from 'node:path';
import parser from '@/utils/rss-parser';

export const route: Route = {
path: '/:category{.+}?',
name: 'Unknown',
maintainers: [],
handler,
const pdfUrlGenerators = {
arxiv: (id: string) => `https://arxiv.org/pdf/${id}.pdf`,
};

async function handler(ctx) {
const { category = 'arxiv/cs.CL' } = ctx.req.param();
export const handler = async (ctx) => {
const { category = 'arxiv/cs.AI' } = ctx.req.param();
const limit = ctx.req.query('limit') ? Number.parseInt(ctx.req.query('limit'), 10) : 150;

const rootUrl = 'https://papers.cool';
const currentUrl = new URL(category, rootUrl).href;
const feedUrl = new URL(`${category}/feed`, rootUrl).href;

const site = category.split(/\//)[0];
const apiKimiUrl = new URL(`${site}/kimi/`, rootUrl).href;

const { data: response } = await got(currentUrl);

const $ = load(response);
const feed = await parser.parseURL(feedUrl);

const pubDate = parseDate(
$('p.info')
.first()
.text()
.match(/(\d+\s\w+\s\d{4})/)[1],
['DD MMM YYYY', 'D MMM YYYY']
);
const language = 'en';

const items = $('div.panel')
.slice(0, limit)
.toArray()
.map((item) => {
item = $(item);
const items = feed.items.slice(0, limit).map((item) => {
const title = item.title;
const guid = item.guid;

const id = item.prop('id');
const kimiUrl = new URL(id, apiKimiUrl).href;
const enclosureUrl =
item
.find('a.pdf-preview')
.prop('onclick')
.match(/'(http.*?)'/)?.[1] ?? undefined;
const id = item.link?.split(/\//).pop() ?? '';
const kimiUrl = new URL(id, apiKimiUrl).href;
const pdfUrl = Object.hasOwn(pdfUrlGenerators, site) ? pdfUrlGenerators[site](id) : undefined;

return {
title: item.find('span[id]').first().text(),
link: kimiUrl,
description: art(path.join(__dirname, 'templates/description.art'), {
kimiUrl,
siteUrl: item.find('a').first().prop('href'),
pdfUrl: enclosureUrl,
summary: item.find('p.summary').text(),
}),
author: item
.find('p.authors a')
.toArray()
.map((a) => $(a).text())
.join('; '),
guid: `${currentUrl}#${id}`,
pubDate,
enclosure_url: enclosureUrl,
enclosure_type: enclosureUrl ? 'application/pdf' : undefined,
};
const description = art(path.join(__dirname, 'templates/description.art'), {
pdfUrl,
siteUrl: item.link,
kimiUrl,
summary: item.summary,
});

const title = $('title').text();
const icon = new URL('favicon.ico', rootUrl).href;
return {
title,
description,
pubDate: parseDate(item.pubDate ?? ''),
link: item.link,
category: item.categories,
author: item.creator,
doi: `${site}${id}`,
guid,
id: guid,
content: {
html: description,
text: item.content,
},
language,
enclosure_url: pdfUrl,
enclosure_type: 'application/pdf',
enclosure_title: title,
};
});

return {
item: items,
title: title.split(/-/)[0].trim(),
title: feed.title,
description: feed.description,
link: currentUrl,
description: title,
icon,
logo: icon,
subtitle: $('h1').first().text(),
item: items,
allowEmpty: true,
image: feed.image?.url,
language: feed.language,
};
}
};

export const route: Route = {
path: '/:category{.+}?',
name: 'Topic',
url: 'papers.cool',
maintainers: ['nczitzk'],
handler,
example: '/papers/arxiv/cs.AI',
parameters: { category: 'Category, arXiv Artificial Intelligence (cs.AI) by default' },
description: `:::tip
If you subscribe to [arXiv Artificial Intelligence (cs.AI)](https://papers.cool/arxiv/cs.AI),where the URL is \`https://papers.cool/arxiv/cs.AI\`, extract the part \`https://papers.cool/\` to the end, and use it as the parameter to fill in. Therefore, the route will be [\`/papers/arxiv/cs.AI\`](https://rsshub.app/papers/arxiv/cs.AI).
:::
| Category | id |
| ----------------------------------------------------- | ----------- |
| arXiv Artificial Intelligence (cs.AI) | arxiv/cs.AI |
| arXiv Computation and Language (cs.CL) | arxiv/cs.CL |
| arXiv Computer Vision and Pattern Recognition (cs.CV) | arxiv/cs.CV |
| arXiv Machine Learning (cs.LG) | arxiv/cs.LG |
`,
categories: ['journal'],

features: {
requireConfig: false,
requirePuppeteer: false,
antiCrawler: false,
supportRadar: true,
supportBT: false,
supportPodcast: false,
supportScihub: true,
},
radar: [
{
title: 'arXiv Artificial Intelligence (cs.AI)',
source: ['papers.cool/arxiv/cs.AI'],
target: '/arxiv/cs.AI',
},
{
title: 'arXiv Computation and Language (cs.CL)',
source: ['papers.cool/arxiv/cs.CL'],
target: '/arxiv/cs.CL',
},
{
title: 'arXiv Computer Vision and Pattern Recognition (cs.CV)',
source: ['papers.cool/arxiv/cs.CV'],
target: '/arxiv/cs.CV',
},
{
title: 'arXiv Machine Learning (cs.LG)',
source: ['papers.cool/arxiv/cs.LG'],
target: '/arxiv/cs.LG',
},
],
};
2 changes: 2 additions & 0 deletions lib/routes/papers/namespace.ts
Original file line number Diff line number Diff line change
Expand Up @@ -3,4 +3,6 @@ import type { Namespace } from '@/types';
export const namespace: Namespace = {
name: 'Cool Papers',
url: 'papers.cool',
categories: ['journal'],
description: '',
};
31 changes: 14 additions & 17 deletions lib/routes/xueqiu/user-stock.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,5 @@
import { Route } from '@/types';
import got from '@/utils/got';
import queryString from 'query-string';
import ofetch from '@/utils/ofetch';
import { parseDate } from '@/utils/parse-date';

export const route: Route = {
Expand Down Expand Up @@ -29,34 +28,32 @@ export const route: Route = {
async function handler(ctx) {
const id = ctx.req.param('id');

const res1 = await got({
method: 'get',
url: 'https://xueqiu.com/',
});
const token = res1.headers['set-cookie'].find((s) => s.startsWith('xq_a_token=')).split(';')[0];
const { headers } = await ofetch.raw('https://xueqiu.com/');
const token = headers
?.getSetCookie()
.find((s) => s.startsWith('xq_a_token='))
?.split(';')[0] as string;

const res2 = await got({
method: 'get',
url: `https://stock.xueqiu.com/v5/stock/portfolio/stock/list.json?category=1&size=1000&uid=${id}`,
const {
data: { stocks: data },
} = await ofetch(`https://stock.xueqiu.com/v5/stock/portfolio/stock/list.json?category=1&size=1000&uid=${id}`, {
headers: {
Cookie: token,
Referer: `https://xueqiu.com/u/${id}`,
},
});
const data = res2.data.data.stocks;

const res3 = await got({
method: 'get',
url: 'https://xueqiu.com/statuses/original/show.json',
searchParams: queryString.stringify({
const {
user: { screen_name },
} = await ofetch('https://xueqiu.com/statuses/original/show.json', {
query: {
user_id: id,
}),
},
headers: {
Cookie: token,
Referer: `https://xueqiu.com/u/${id}`,
},
});
const screen_name = res3.data.user.screen_name;

return {
title: `${screen_name} 的雪球自选动态`,
Expand Down

0 comments on commit 6063786

Please sign in to comment.