From b94207aa05f80c1d56bae29cabdbe0b26b929be8 Mon Sep 17 00:00:00 2001 From: Ethan Shen <42264778+nczitzk@users.noreply.github.com> Date: Thu, 12 Sep 2024 01:25:02 +0800 Subject: [PATCH] =?UTF-8?q?feat(route):=20add=20=E6=9E=81=E5=AE=A2?= =?UTF-8?q?=E5=85=AC=E5=9B=AD=20(#16696)?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit * feat(route): add 极客公园 * fix typo * fix typo --------- --- lib/router.js | 2 +- .../geekpark/breakingnews.js | 25 -- lib/routes/geekpark/index.ts | 219 ++++++++++++++++++ lib/routes/geekpark/namespace.ts | 8 + lib/routes/geekpark/templates/description.art | 21 ++ 5 files changed, 249 insertions(+), 26 deletions(-) delete mode 100644 lib/routes-deprecated/geekpark/breakingnews.js create mode 100644 lib/routes/geekpark/index.ts create mode 100644 lib/routes/geekpark/namespace.ts create mode 100644 lib/routes/geekpark/templates/description.art diff --git a/lib/router.js b/lib/router.js index 738753d3235f0c..4033bb6bad12fe 100644 --- a/lib/router.js +++ b/lib/router.js @@ -361,7 +361,7 @@ router.get('/ltaaa/:category?', lazyloadRouteHandler('./routes/ltaaa/index')); router.get('/autotrader/:query', lazyloadRouteHandler('./routes/autotrader')); // 极客公园 -router.get('/geekpark/breakingnews', lazyloadRouteHandler('./routes/geekpark/breakingnews')); +// router.get('/geekpark/breakingnews', lazyloadRouteHandler('./routes/geekpark/breakingnews')); // 搜狗 // router.get('/sogou/doodles', lazyloadRouteHandler('./routes/sogou/doodles')); diff --git a/lib/routes-deprecated/geekpark/breakingnews.js b/lib/routes-deprecated/geekpark/breakingnews.js deleted file mode 100644 index 67f1e4255f7b7e..00000000000000 --- a/lib/routes-deprecated/geekpark/breakingnews.js +++ /dev/null @@ -1,25 +0,0 @@ -const got = require('@/utils/got'); - -module.exports = async (ctx) => { - const url = 'https://mainssl.geekpark.net/api/v1/posts'; - const link = 'https://www.geekpark.net'; - - const response = await got({ - method: 'get', - url, - }); - const data = response.data.posts; - - ctx.state.data = { - title: '极客公园 - 资讯', - description: - '极客公园聚焦互联网领域,跟踪最新的科技新闻动态,关注极具创新精神的科技产品。目前涵盖前沿科技、游戏、手机评测、硬件测评、出行方式、共享经济、人工智能等全方位的科技生活内容。现有前沿社、挖App、深度报道、极客养成指南等多个内容栏目。', - link, - item: data.map(({ title, content, published_at, id }) => ({ - title, - link: `https://www.geekpark.net/news/${id}`, - description: content, - pubDate: new Date(published_at).toUTCString(), - })), - }; -}; diff --git a/lib/routes/geekpark/index.ts b/lib/routes/geekpark/index.ts new file mode 100644 index 00000000000000..7fd077b92e3af2 --- /dev/null +++ b/lib/routes/geekpark/index.ts @@ -0,0 +1,219 @@ +import { Route } from '@/types'; +import { getCurrentPath } from '@/utils/helpers'; +const __dirname = getCurrentPath(import.meta.url); + +import cache from '@/utils/cache'; +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'; + +export const handler = async (ctx) => { + const { column } = ctx.req.param(); + const limit = ctx.req.query('limit') ? Number.parseInt(ctx.req.query('limit'), 10) : 20; + + const rootUrl = 'https://geekpark.net'; + const apiRootUrl = 'https://mainssl.geekpark.net'; + const currentUrl = new URL(column ? `column/${column}` : '', rootUrl).href; + const apiUrl = new URL(column ? `api/v1/columns/${column}` : 'api/v2', apiRootUrl).href; + + const { data: response } = await got(apiUrl); + + let items = (response.homepage_posts ?? response.column.posts).slice(0, limit).map((item) => { + item = item.post ?? item; + + const title = item.title; + const image = item.cover_url; + const description = art(path.join(__dirname, 'templates/description.art'), { + image: image + ? [ + { + src: image, + alt: title, + }, + ] + : undefined, + intro: item.abstract, + }); + const guid = `geekpark-${item.id}`; + + return { + title, + description, + pubDate: parseDate(item.published_timestamp, 'X'), + link: new URL(`api/v1/posts/${item.id}`, apiRootUrl).href, + category: [...new Set([...item.tags, item.column?.title])].filter(Boolean), + author: item.authors.map((a) => a.realname ?? a.nickname).join('/'), + guid, + id: guid, + content: { + html: description, + text: item.abstract, + }, + image, + banner: image, + }; + }); + + items = await Promise.all( + items.map((item) => + cache.tryGet(item.link, async () => { + const { data: detailResponse } = await got(item.link); + + const data = detailResponse.post; + + const title = data.title; + const image = data.cover_url; + const description = art(path.join(__dirname, 'templates/description.art'), { + image: image + ? [ + { + src: image, + alt: title, + }, + ] + : undefined, + intro: data.abstract, + description: data.content, + }); + const guid = `geekpark-${data.id}`; + + item.title = title; + item.description = description; + item.pubDate = parseDate(data.published_timestamp, 'X'); + item.link = new URL(`news/${data.id}`, rootUrl).href; + item.category = [...new Set([...data.tags, data.column?.title])].filter(Boolean); + item.author = data.authors.map((a) => a.realname ?? a.nickname).join('/'); + item.guid = guid; + item.id = guid; + item.content = { + html: description, + text: data.content, + }; + item.image = image; + item.banner = image; + item.updated = parseDate(data.updated_at); + + return item; + }) + ) + ); + + const data = { + title: '', + description: '', + link: currentUrl, + item: items, + allowEmpty: true, + image: '', + author: '', + }; + + if (column) { + data.title = `${response.column.title} | 极客公园`; + data.description = response.column.description; + data.image = response.column.banner_url; + } else { + const { data: currentResponse } = await got(currentUrl); + + const $ = load(currentResponse); + + data.title = $('title').text(); + data.description = $('meta[property="og:description"]').prop('content'); + data.image = `https:${$('meta[name="og:image"]').prop('content')}`; + data.author = $('meta[property="og:site_name"]').prop('content'); + } + + return data; +}; + +export const route: Route = { + path: '/:column?', + name: '栏目', + url: 'geekpark.net', + maintainers: ['nczitzk'], + handler, + example: '/geekpark', + parameters: { column: '栏目 id,默认为空,即首页资讯,可在对应栏目页 URL 中找到' }, + description: `:::tip + 若订阅 [综合报道](https://www.geekpark.net/column/179),网址为 \`https://www.geekpark.net/column/179\`。截取 \`https://www.geekpark.net/column/\` 到末尾的部分 \`179\` 作为参数填入,此时路由为 [\`/geekpark/179\`](https://rsshub.app/geekpark/179)。 + ::: + + | 栏目 | ID | + | ------------------------------------------------------------ | -------------------------------------- | + | [综合报道](https://www.geekpark.net/column/179) | [179](https://rsshub.app/geekpark/179) | + | [AI新浪潮观察](https://www.geekpark.net/column/304) | [304](https://rsshub.app/geekpark/304) | + | [新造车观察](https://www.geekpark.net/column/305) | [305](https://rsshub.app/geekpark/305) | + | [财报解读](https://www.geekpark.net/column/271) | [271](https://rsshub.app/geekpark/271) | + | [底稿对话CEO系列](https://www.geekpark.net/column/308) | [308](https://rsshub.app/geekpark/308) | + | [Geek Insight 特稿系列](https://www.geekpark.net/column/306) | [306](https://rsshub.app/geekpark/306) | + | [心科技](https://www.geekpark.net/column/307) | [307](https://rsshub.app/geekpark/307) | + | [行业资讯](https://www.geekpark.net/column/2) | [2](https://rsshub.app/geekpark/2) | + `, + categories: ['new-media'], + + features: { + requireConfig: false, + requirePuppeteer: false, + antiCrawler: false, + supportRadar: true, + supportBT: false, + supportPodcast: false, + supportScihub: false, + }, + radar: [ + { + source: ['geekpark.net'], + target: '/', + }, + { + source: ['geekpark.net/column/:column?'], + target: (params) => { + const column = params.column; + + return column ? `/${column}` : ''; + }, + }, + { + title: '综合报道', + source: ['www.geekpark.net/column/179'], + target: '/179', + }, + { + title: 'AI新浪潮观察', + source: ['www.geekpark.net/column/304'], + target: '/304', + }, + { + title: '新造车观察', + source: ['www.geekpark.net/column/305'], + target: '/305', + }, + { + title: '财报解读', + source: ['www.geekpark.net/column/271'], + target: '/271', + }, + { + title: '底稿对话CEO系列', + source: ['www.geekpark.net/column/308'], + target: '/308', + }, + { + title: 'Geek Insight 特稿系列', + source: ['www.geekpark.net/column/306'], + target: '/306', + }, + { + title: '心科技', + source: ['www.geekpark.net/column/307'], + target: '/307', + }, + { + title: '行业资讯', + source: ['www.geekpark.net/column/2'], + target: '/2', + }, + ], +}; diff --git a/lib/routes/geekpark/namespace.ts b/lib/routes/geekpark/namespace.ts new file mode 100644 index 00000000000000..8b5c57b22afc51 --- /dev/null +++ b/lib/routes/geekpark/namespace.ts @@ -0,0 +1,8 @@ +import type { Namespace } from '@/types'; + +export const namespace: Namespace = { + name: '极客公园', + url: 'geekpark.net', + categories: ['new-media'], + description: '', +}; diff --git a/lib/routes/geekpark/templates/description.art b/lib/routes/geekpark/templates/description.art new file mode 100644 index 00000000000000..249654e7e618a4 --- /dev/null +++ b/lib/routes/geekpark/templates/description.art @@ -0,0 +1,21 @@ +{{ if images }} + {{ each images image }} + {{ if image?.src }} +
+ {{ image.alt }} +
+ {{ /if }} + {{ /each }} +{{ /if }} + +{{ if intro }} +
{{ intro }}
+{{ /if }} + +{{ if description }} + {{@ description }} +{{ /if }} \ No newline at end of file