feat: 新增快手热榜及网易新闻
This commit is contained in:
parent
8cffc6c701
commit
c42573c37c
10
README.md
10
README.md
@ -14,7 +14,7 @@
|
|||||||
|
|
||||||
> 🟢 状态正常
|
> 🟢 状态正常
|
||||||
> 🟠 可能失效
|
> 🟠 可能失效
|
||||||
> 🔴 无法使用
|
> ❌ 无法使用
|
||||||
|
|
||||||
| **站点** | **类别** | **调用名称** | **状态** |
|
| **站点** | **类别** | **调用名称** | **状态** |
|
||||||
| ------------ | -------- | ------------------- | -------- |
|
| ------------ | -------- | ------------------- | -------- |
|
||||||
@ -26,15 +26,17 @@
|
|||||||
| 抖音 | 热歌榜 | douyin_music | 🟢 |
|
| 抖音 | 热歌榜 | douyin_music | 🟢 |
|
||||||
| 百度贴吧 | 热议榜 | tieba | 🟢 |
|
| 百度贴吧 | 热议榜 | tieba | 🟢 |
|
||||||
| 少数派 | 热榜 | sspai | 🟢 |
|
| 少数派 | 热榜 | sspai | 🟢 |
|
||||||
| IT 之家 | 热榜 | ithome | 🟠 |
|
| IT 之家 | 热榜 | ithome | 🟢 |
|
||||||
| 澎湃新闻 | 热榜 | thepaper | 🟢 |
|
| 澎湃新闻 | 热榜 | thepaper | 🟢 |
|
||||||
| 今日头条 | 热榜 | toutiao | 🟢 |
|
| 今日头条 | 热榜 | toutiao | 🟢 |
|
||||||
| 36 氪 | 热榜 | 36kr | 🟢 |
|
| 36 氪 | 热榜 | 36kr | 🟢 |
|
||||||
| 稀土掘金 | 热榜 | juejin | 🟢 |
|
| 稀土掘金 | 热榜 | juejin | 🟢 |
|
||||||
| 腾讯新闻 | 热点榜 | newsqq | 🟢 |
|
| 腾讯新闻 | 热点榜 | newsqq | 🟢 |
|
||||||
|
| 网易新闻 | 热点榜 | netease | 🟢 |
|
||||||
| 英雄联盟 | 更新公告 | lol | 🟢 |
|
| 英雄联盟 | 更新公告 | lol | 🟢 |
|
||||||
| 原神 | 最新消息 | genshin | 🟢 |
|
| 原神 | 最新消息 | genshin | 🟢 |
|
||||||
| 微信读书 | 飙升榜 | weread | 🟢 |
|
| 微信读书 | 飙升榜 | weread | 🟢 |
|
||||||
|
| 快手 | 热榜 | kuaishou | 🟢 |
|
||||||
| 历史上的今天 | 指定日期 | calendar | 🟢 |
|
| 历史上的今天 | 指定日期 | calendar | 🟢 |
|
||||||
|
|
||||||
### 特殊接口说明
|
### 特殊接口说明
|
||||||
@ -44,7 +46,7 @@
|
|||||||
获取除了下方特殊接口外的全部接口列表
|
获取除了下方特殊接口外的全部接口列表
|
||||||
|
|
||||||
```http
|
```http
|
||||||
GET https://{example.com}/all
|
GET https://example.com/all
|
||||||
```
|
```
|
||||||
|
|
||||||
#### 历史上的今天(指定日期)
|
#### 历史上的今天(指定日期)
|
||||||
@ -52,7 +54,7 @@ GET https://{example.com}/all
|
|||||||
将指定的月份和日期传入即可得到当天数据,请注意格式
|
将指定的月份和日期传入即可得到当天数据,请注意格式
|
||||||
|
|
||||||
```http
|
```http
|
||||||
GET https://{example.com}/calendar/date?month=06&day=01
|
GET https://example.com/calendar/date?month=06&day=01
|
||||||
```
|
```
|
||||||
|
|
||||||
## 部署
|
## 部署
|
||||||
|
@ -1,11 +1,11 @@
|
|||||||
{
|
{
|
||||||
"name": "dailyhot_api",
|
"name": "dailyhot_api",
|
||||||
"version": "1.0.2",
|
"version": "1.0.3",
|
||||||
"description": "一个今日热榜",
|
"description": "一个今日热榜",
|
||||||
"main": "index.js",
|
"main": "index.js",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"start": "node index.js",
|
"start": "node index.js",
|
||||||
"dev": "./node_modules/.bin/nodemon index.js",
|
"dev": "npx nodemon index.js",
|
||||||
"prd": "pm2 start index.js",
|
"prd": "pm2 start index.js",
|
||||||
"build": "node index.js"
|
"build": "node index.js"
|
||||||
},
|
},
|
||||||
|
@ -5,6 +5,7 @@ const { get, set, del } = require("../utils/cacheData");
|
|||||||
|
|
||||||
// 接口信息
|
// 接口信息
|
||||||
const routerInfo = {
|
const routerInfo = {
|
||||||
|
name: "36kr",
|
||||||
title: "36氪",
|
title: "36氪",
|
||||||
subtitle: "热榜",
|
subtitle: "热榜",
|
||||||
};
|
};
|
||||||
|
@ -4,7 +4,7 @@ const axios = require("axios");
|
|||||||
const { get, set, del } = require("../utils/cacheData");
|
const { get, set, del } = require("../utils/cacheData");
|
||||||
|
|
||||||
// 接口信息
|
// 接口信息
|
||||||
const routerInfo = { title: "百度", subtitle: "热搜榜" };
|
const routerInfo = { name: "baidu", title: "百度", subtitle: "热搜榜" };
|
||||||
|
|
||||||
// 缓存键名
|
// 缓存键名
|
||||||
const cacheKey = "baiduData";
|
const cacheKey = "baiduData";
|
||||||
|
@ -5,6 +5,7 @@ const { get, set, del } = require("../utils/cacheData");
|
|||||||
|
|
||||||
// 接口信息
|
// 接口信息
|
||||||
const routerInfo = {
|
const routerInfo = {
|
||||||
|
name: "bilibili",
|
||||||
title: "哔哩哔哩",
|
title: "哔哩哔哩",
|
||||||
subtitle: "热门榜",
|
subtitle: "热门榜",
|
||||||
};
|
};
|
||||||
|
@ -5,6 +5,7 @@ const { get, set, del } = require("../utils/cacheData");
|
|||||||
|
|
||||||
// 接口信息
|
// 接口信息
|
||||||
const routerInfo = {
|
const routerInfo = {
|
||||||
|
name: "douyin",
|
||||||
title: "抖音",
|
title: "抖音",
|
||||||
subtitle: "热点榜",
|
subtitle: "热点榜",
|
||||||
};
|
};
|
||||||
|
@ -1,5 +1,4 @@
|
|||||||
/*
|
/*
|
||||||
* @version: 1.0.0
|
|
||||||
* @author: WangPeng
|
* @author: WangPeng
|
||||||
* @date: 2023-07-11 16:41:48
|
* @date: 2023-07-11 16:41:48
|
||||||
* @customEditors: imsyy
|
* @customEditors: imsyy
|
||||||
@ -13,6 +12,7 @@ const { get, set, del } = require("../utils/cacheData");
|
|||||||
|
|
||||||
// 接口信息
|
// 接口信息
|
||||||
const routerInfo = {
|
const routerInfo = {
|
||||||
|
name: "douyin",
|
||||||
title: "抖音",
|
title: "抖音",
|
||||||
subtitle: "热歌榜",
|
subtitle: "热歌榜",
|
||||||
};
|
};
|
||||||
|
@ -1,5 +1,4 @@
|
|||||||
/*
|
/*
|
||||||
* @version: 1.0.0
|
|
||||||
* @author: WangPeng
|
* @author: WangPeng
|
||||||
* @date: 2023-07-10 16:56:01
|
* @date: 2023-07-10 16:56:01
|
||||||
* @customEditors: imsyy
|
* @customEditors: imsyy
|
||||||
@ -13,6 +12,7 @@ const { get, set, del } = require("../utils/cacheData");
|
|||||||
|
|
||||||
// 接口信息
|
// 接口信息
|
||||||
const routerInfo = {
|
const routerInfo = {
|
||||||
|
name: "douyin",
|
||||||
title: "抖音",
|
title: "抖音",
|
||||||
subtitle: "热点榜",
|
subtitle: "热点榜",
|
||||||
};
|
};
|
||||||
|
@ -5,6 +5,7 @@ const { get, set, del } = require("../utils/cacheData");
|
|||||||
|
|
||||||
// 接口信息
|
// 接口信息
|
||||||
const routerInfo = {
|
const routerInfo = {
|
||||||
|
name: "genshin",
|
||||||
title: "原神",
|
title: "原神",
|
||||||
subtitle: "最新信息",
|
subtitle: "最新信息",
|
||||||
};
|
};
|
||||||
|
@ -6,6 +6,7 @@ const { get, set, del } = require("../utils/cacheData");
|
|||||||
|
|
||||||
// 接口信息
|
// 接口信息
|
||||||
const routerInfo = {
|
const routerInfo = {
|
||||||
|
name: "ithome",
|
||||||
title: "IT之家",
|
title: "IT之家",
|
||||||
subtitle: "热榜",
|
subtitle: "热榜",
|
||||||
};
|
};
|
||||||
|
@ -5,6 +5,7 @@ const { get, set, del } = require("../utils/cacheData");
|
|||||||
|
|
||||||
// 接口信息
|
// 接口信息
|
||||||
const routerInfo = {
|
const routerInfo = {
|
||||||
|
name: "juejin",
|
||||||
title: "稀土掘金",
|
title: "稀土掘金",
|
||||||
subtitle: "热榜",
|
subtitle: "热榜",
|
||||||
};
|
};
|
||||||
|
163
routes/kuaishou.js
Normal file
163
routes/kuaishou.js
Normal file
@ -0,0 +1,163 @@
|
|||||||
|
/*
|
||||||
|
* @author: MCBBC
|
||||||
|
* @date: 2023-07-17
|
||||||
|
* @customEditors: imsyy
|
||||||
|
* @lastEditTime: 2023-07-17
|
||||||
|
*/
|
||||||
|
|
||||||
|
const Router = require("koa-router");
|
||||||
|
const kuaishouRouter = new Router();
|
||||||
|
const axios = require("axios");
|
||||||
|
const { get, set, del } = require("../utils/cacheData");
|
||||||
|
|
||||||
|
// 接口信息
|
||||||
|
const routerInfo = {
|
||||||
|
name: "kuaishou",
|
||||||
|
title: "快手",
|
||||||
|
subtitle: "热榜",
|
||||||
|
};
|
||||||
|
|
||||||
|
// 缓存键名
|
||||||
|
const cacheKey = "kuaishouData";
|
||||||
|
|
||||||
|
// 调用时间
|
||||||
|
let updateTime = new Date().toISOString();
|
||||||
|
|
||||||
|
// 调用路径
|
||||||
|
const url = "https://www.kuaishou.com/?isHome=1";
|
||||||
|
|
||||||
|
// Unicode 解码
|
||||||
|
const decodedString = (encodedString) => {
|
||||||
|
return encodedString.replace(/\\u([\d\w]{4})/gi, (match, grp) =>
|
||||||
|
String.fromCharCode(parseInt(grp, 16))
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
// 数据处理
|
||||||
|
const getData = (data) => {
|
||||||
|
if (!data) return [];
|
||||||
|
const dataList = [];
|
||||||
|
try {
|
||||||
|
const pattern = /window.__APOLLO_STATE__=(.*);\(function\(\)/s;
|
||||||
|
const idPattern = /clientCacheKey=([A-Za-z0-9]+)/s;
|
||||||
|
const matchResult = data.match(pattern);
|
||||||
|
const jsonObject = JSON.parse(matchResult[1])["defaultClient"];
|
||||||
|
|
||||||
|
// 获取所有分类
|
||||||
|
const allItems =
|
||||||
|
jsonObject['$ROOT_QUERY.visionHotRank({"page":"home"})']["items"];
|
||||||
|
// 遍历所有分类
|
||||||
|
allItems.forEach((v) => {
|
||||||
|
// 基础数据
|
||||||
|
const image = jsonObject[v.id]["poster"];
|
||||||
|
const id = image.match(idPattern)[1];
|
||||||
|
// 数据处理
|
||||||
|
dataList.push({
|
||||||
|
title: jsonObject[v.id]["name"],
|
||||||
|
pic: decodedString(image),
|
||||||
|
hot: jsonObject[v.id]["hotValue"],
|
||||||
|
url: `https://www.kuaishou.com/short-video/${id}`,
|
||||||
|
mobileUrl: `https://www.kuaishou.com/short-video/${id}`,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
return dataList;
|
||||||
|
} catch (error) {
|
||||||
|
console.error("数据处理出错" + error);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// 快手热榜
|
||||||
|
kuaishouRouter.get("/kuaishou", async (ctx) => {
|
||||||
|
console.log("获取快手热榜");
|
||||||
|
try {
|
||||||
|
// 从缓存中获取数据
|
||||||
|
let data = await get(cacheKey);
|
||||||
|
const from = data ? "cache" : "server";
|
||||||
|
if (!data) {
|
||||||
|
// 如果缓存中不存在数据
|
||||||
|
console.log("从服务端重新获取快手热榜");
|
||||||
|
// 从服务器拉取数据
|
||||||
|
const response = await axios.get(url);
|
||||||
|
data = getData(response.data);
|
||||||
|
updateTime = new Date().toISOString();
|
||||||
|
if (!data) {
|
||||||
|
ctx.body = {
|
||||||
|
code: 500,
|
||||||
|
...routerInfo,
|
||||||
|
message: "获取失败",
|
||||||
|
};
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
// 将数据写入缓存
|
||||||
|
await set(cacheKey, data);
|
||||||
|
}
|
||||||
|
ctx.body = {
|
||||||
|
code: 200,
|
||||||
|
message: "获取成功",
|
||||||
|
...routerInfo,
|
||||||
|
from,
|
||||||
|
total: data.length,
|
||||||
|
updateTime,
|
||||||
|
data,
|
||||||
|
};
|
||||||
|
} catch (error) {
|
||||||
|
console.error(error);
|
||||||
|
ctx.body = {
|
||||||
|
code: 500,
|
||||||
|
...routerInfo,
|
||||||
|
message: "获取失败",
|
||||||
|
};
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 快手热榜 - 获取最新数据
|
||||||
|
kuaishouRouter.get("/kuaishou/new", async (ctx) => {
|
||||||
|
console.log("获取快手热榜 - 最新数据");
|
||||||
|
try {
|
||||||
|
// 从服务器拉取最新数据
|
||||||
|
const response = await axios.get(url);
|
||||||
|
const newData = getData(response.data);
|
||||||
|
updateTime = new Date().toISOString();
|
||||||
|
console.log("从服务端重新获取快手热榜");
|
||||||
|
|
||||||
|
// 返回最新数据
|
||||||
|
ctx.body = {
|
||||||
|
code: 200,
|
||||||
|
message: "获取成功",
|
||||||
|
...routerInfo,
|
||||||
|
total: newData.length,
|
||||||
|
updateTime,
|
||||||
|
data: newData,
|
||||||
|
};
|
||||||
|
|
||||||
|
// 删除旧数据
|
||||||
|
await del(cacheKey);
|
||||||
|
// 将最新数据写入缓存
|
||||||
|
await set(cacheKey, newData);
|
||||||
|
} catch (error) {
|
||||||
|
// 如果拉取最新数据失败,尝试从缓存中获取数据
|
||||||
|
console.error(error);
|
||||||
|
const cachedData = await get(cacheKey);
|
||||||
|
if (cachedData) {
|
||||||
|
ctx.body = {
|
||||||
|
code: 200,
|
||||||
|
message: "获取成功",
|
||||||
|
...routerInfo,
|
||||||
|
total: cachedData.length,
|
||||||
|
updateTime,
|
||||||
|
data: cachedData,
|
||||||
|
};
|
||||||
|
} else {
|
||||||
|
// 如果缓存中也没有数据,则返回错误信息
|
||||||
|
ctx.body = {
|
||||||
|
code: 500,
|
||||||
|
...routerInfo,
|
||||||
|
message: "获取失败",
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
kuaishouRouter.info = routerInfo;
|
||||||
|
module.exports = kuaishouRouter;
|
@ -5,6 +5,7 @@ const { get, set, del } = require("../utils/cacheData");
|
|||||||
|
|
||||||
// 接口信息
|
// 接口信息
|
||||||
const routerInfo = {
|
const routerInfo = {
|
||||||
|
name: "lol",
|
||||||
title: "英雄联盟",
|
title: "英雄联盟",
|
||||||
subtitle: "更新公告",
|
subtitle: "更新公告",
|
||||||
};
|
};
|
||||||
|
130
routes/netease.js
Normal file
130
routes/netease.js
Normal file
@ -0,0 +1,130 @@
|
|||||||
|
/*
|
||||||
|
* @author: MCBBC
|
||||||
|
* @date: 2023-07-17
|
||||||
|
* @customEditors: imsyy
|
||||||
|
* @lastEditTime: 2023-07-17
|
||||||
|
*/
|
||||||
|
|
||||||
|
const Router = require("koa-router");
|
||||||
|
const neteaseRouter = new Router();
|
||||||
|
const axios = require("axios");
|
||||||
|
const { get, set, del } = require("../utils/cacheData");
|
||||||
|
|
||||||
|
// 接口信息
|
||||||
|
const routerInfo = {
|
||||||
|
name: "netease",
|
||||||
|
title: "网易新闻",
|
||||||
|
subtitle: "热点榜",
|
||||||
|
};
|
||||||
|
|
||||||
|
// 缓存键名
|
||||||
|
const cacheKey = "neteaseData";
|
||||||
|
|
||||||
|
// 调用时间
|
||||||
|
let updateTime = new Date().toISOString();
|
||||||
|
|
||||||
|
// 调用路径
|
||||||
|
const url = "https://m.163.com/fe/api/hot/news/flow";
|
||||||
|
|
||||||
|
// 数据处理
|
||||||
|
const getData = (data) => {
|
||||||
|
if (!data) return [];
|
||||||
|
return data.map((v) => {
|
||||||
|
return {
|
||||||
|
id: v.skipID,
|
||||||
|
title: v.title,
|
||||||
|
desc: v._keyword,
|
||||||
|
pic: v.imgsrc,
|
||||||
|
owner: v.source,
|
||||||
|
url: `https://www.163.com/dy/article/${v.skipID}.html`,
|
||||||
|
mobileUrl: v.url,
|
||||||
|
};
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
// 网易新闻热榜
|
||||||
|
neteaseRouter.get("/netease", async (ctx) => {
|
||||||
|
console.log("获取网易新闻热榜");
|
||||||
|
try {
|
||||||
|
// 从缓存中获取数据
|
||||||
|
let data = await get(cacheKey);
|
||||||
|
const from = data ? "cache" : "server";
|
||||||
|
if (!data) {
|
||||||
|
// 如果缓存中不存在数据
|
||||||
|
console.log("从服务端重新获取网易新闻热榜");
|
||||||
|
// 从服务器拉取数据
|
||||||
|
const response = await axios.get(url);
|
||||||
|
data = getData(response.data.data.list);
|
||||||
|
updateTime = new Date().toISOString();
|
||||||
|
// 将数据写入缓存
|
||||||
|
await set(cacheKey, data);
|
||||||
|
}
|
||||||
|
ctx.body = {
|
||||||
|
code: 200,
|
||||||
|
message: "获取成功",
|
||||||
|
...routerInfo,
|
||||||
|
from,
|
||||||
|
total: data.length,
|
||||||
|
updateTime,
|
||||||
|
data,
|
||||||
|
};
|
||||||
|
} catch (error) {
|
||||||
|
console.error(error);
|
||||||
|
ctx.body = {
|
||||||
|
code: 500,
|
||||||
|
...routerInfo,
|
||||||
|
message: "获取失败",
|
||||||
|
};
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 网易新闻热榜 - 获取最新数据
|
||||||
|
neteaseRouter.get("/netease/new", async (ctx) => {
|
||||||
|
console.log("获取网易新闻热榜 - 最新数据");
|
||||||
|
try {
|
||||||
|
// 从服务器拉取最新数据
|
||||||
|
const response = await axios.get(url);
|
||||||
|
const newData = getData(response.data.data.list);
|
||||||
|
updateTime = new Date().toISOString();
|
||||||
|
console.log("从服务端重新获取网易新闻热榜");
|
||||||
|
|
||||||
|
// 返回最新数据
|
||||||
|
ctx.body = {
|
||||||
|
code: 200,
|
||||||
|
message: "获取成功",
|
||||||
|
...routerInfo,
|
||||||
|
total: newData.length,
|
||||||
|
updateTime,
|
||||||
|
data: newData,
|
||||||
|
};
|
||||||
|
|
||||||
|
// 删除旧数据
|
||||||
|
await del(cacheKey);
|
||||||
|
// 将最新数据写入缓存
|
||||||
|
await set(cacheKey, newData);
|
||||||
|
} catch (error) {
|
||||||
|
// 如果拉取最新数据失败,尝试从缓存中获取数据
|
||||||
|
console.error(error);
|
||||||
|
const cachedData = await get(cacheKey);
|
||||||
|
if (cachedData) {
|
||||||
|
ctx.body = {
|
||||||
|
code: 200,
|
||||||
|
message: "获取成功",
|
||||||
|
...routerInfo,
|
||||||
|
total: cachedData.length,
|
||||||
|
updateTime,
|
||||||
|
data: cachedData,
|
||||||
|
};
|
||||||
|
} else {
|
||||||
|
// 如果缓存中也没有数据,则返回错误信息
|
||||||
|
ctx.body = {
|
||||||
|
code: 500,
|
||||||
|
...routerInfo,
|
||||||
|
message: "获取失败",
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
neteaseRouter.info = routerInfo;
|
||||||
|
module.exports = neteaseRouter;
|
@ -5,6 +5,7 @@ const { get, set, del } = require("../utils/cacheData");
|
|||||||
|
|
||||||
// 接口信息
|
// 接口信息
|
||||||
const routerInfo = {
|
const routerInfo = {
|
||||||
|
name: "newsqq",
|
||||||
title: "腾讯新闻",
|
title: "腾讯新闻",
|
||||||
subtitle: "热点榜",
|
subtitle: "热点榜",
|
||||||
};
|
};
|
||||||
|
@ -5,6 +5,7 @@ const { get, set, del } = require("../utils/cacheData");
|
|||||||
|
|
||||||
// 接口信息
|
// 接口信息
|
||||||
const routerInfo = {
|
const routerInfo = {
|
||||||
|
name: "sspai",
|
||||||
title: "少数派",
|
title: "少数派",
|
||||||
subtitle: "热榜",
|
subtitle: "热榜",
|
||||||
};
|
};
|
||||||
|
@ -5,6 +5,7 @@ const { get, set, del } = require("../utils/cacheData");
|
|||||||
|
|
||||||
// 接口信息
|
// 接口信息
|
||||||
const routerInfo = {
|
const routerInfo = {
|
||||||
|
name: "thepaper",
|
||||||
title: "澎湃新闻",
|
title: "澎湃新闻",
|
||||||
subtitle: "热榜",
|
subtitle: "热榜",
|
||||||
};
|
};
|
||||||
|
@ -5,6 +5,7 @@ const { get, set, del } = require("../utils/cacheData");
|
|||||||
|
|
||||||
// 接口信息
|
// 接口信息
|
||||||
const routerInfo = {
|
const routerInfo = {
|
||||||
|
name: "tieba",
|
||||||
title: "百度贴吧",
|
title: "百度贴吧",
|
||||||
subtitle: "热议榜",
|
subtitle: "热议榜",
|
||||||
};
|
};
|
||||||
|
@ -5,6 +5,7 @@ const { get, set, del } = require("../utils/cacheData");
|
|||||||
|
|
||||||
// 接口信息
|
// 接口信息
|
||||||
const routerInfo = {
|
const routerInfo = {
|
||||||
|
name: "toutiao",
|
||||||
title: "今日头条",
|
title: "今日头条",
|
||||||
subtitle: "热榜",
|
subtitle: "热榜",
|
||||||
};
|
};
|
||||||
|
@ -6,6 +6,7 @@ const { get, set, del } = require("../utils/cacheData");
|
|||||||
|
|
||||||
// 接口信息
|
// 接口信息
|
||||||
const routerInfo = {
|
const routerInfo = {
|
||||||
|
name: "weibo",
|
||||||
title: "微博",
|
title: "微博",
|
||||||
subtitle: "热搜榜",
|
subtitle: "热搜榜",
|
||||||
};
|
};
|
||||||
|
@ -1,10 +1,12 @@
|
|||||||
const Router = require("koa-router");
|
const Router = require("koa-router");
|
||||||
const wereadRouter = new Router();
|
const wereadRouter = new Router();
|
||||||
const axios = require("axios");
|
const axios = require("axios");
|
||||||
|
const getWereadID = require("../utils/getWereadID");
|
||||||
const { get, set, del } = require("../utils/cacheData");
|
const { get, set, del } = require("../utils/cacheData");
|
||||||
|
|
||||||
// 接口信息
|
// 接口信息
|
||||||
const routerInfo = {
|
const routerInfo = {
|
||||||
|
name: "weread",
|
||||||
title: "微信读书",
|
title: "微信读书",
|
||||||
subtitle: "飙升榜",
|
subtitle: "飙升榜",
|
||||||
};
|
};
|
||||||
@ -23,7 +25,6 @@ const getData = (data) => {
|
|||||||
if (!data) return [];
|
if (!data) return [];
|
||||||
return data.map((v) => {
|
return data.map((v) => {
|
||||||
const book = v.bookInfo;
|
const book = v.bookInfo;
|
||||||
console.log(book);
|
|
||||||
return {
|
return {
|
||||||
id: book.bookId,
|
id: book.bookId,
|
||||||
title: book.title,
|
title: book.title,
|
||||||
@ -31,8 +32,10 @@ const getData = (data) => {
|
|||||||
pic: book.cover.replace("s_", "t9_"),
|
pic: book.cover.replace("s_", "t9_"),
|
||||||
hot: v.readingCount,
|
hot: v.readingCount,
|
||||||
author: book.author,
|
author: book.author,
|
||||||
url: "https://weread.qq.com/web/category/rising",
|
url: `https://weread.qq.com/web/bookDetail/${getWereadID(book.bookId)}`,
|
||||||
mobileUrl: "https://weread.qq.com/web/category/rising",
|
mobileUrl: `https://weread.qq.com/web/bookDetail/${getWereadID(
|
||||||
|
book.bookId
|
||||||
|
)}`,
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
82
utils/getWereadID.js
Normal file
82
utils/getWereadID.js
Normal file
@ -0,0 +1,82 @@
|
|||||||
|
const crypto = require("crypto");
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取微信读书的书籍 ID
|
||||||
|
* 感谢 @MCBBC 及 ChatGPT
|
||||||
|
* @param {string} bookId - 书籍 ID
|
||||||
|
* @returns {string} - 唯一的书籍 ID
|
||||||
|
*/
|
||||||
|
const getWereadID = (bookId) => {
|
||||||
|
try {
|
||||||
|
// 使用 MD5 哈希算法创建哈希对象
|
||||||
|
const hash = crypto.createHash("md5");
|
||||||
|
hash.update(bookId);
|
||||||
|
const str = hash.digest("hex");
|
||||||
|
|
||||||
|
// 取哈希结果的前三个字符作为初始值
|
||||||
|
let strSub = str.substring(0, 3);
|
||||||
|
|
||||||
|
// 判断书籍 ID 的类型并进行转换
|
||||||
|
let fa;
|
||||||
|
if (/^\d*$/.test(bookId)) {
|
||||||
|
// 如果书籍 ID 只包含数字,则将其拆分成长度为 9 的子字符串,并转换为十六进制表示
|
||||||
|
const chunks = [];
|
||||||
|
for (let i = 0; i < bookId.length; i += 9) {
|
||||||
|
const chunk = bookId.substring(i, i + 9);
|
||||||
|
chunks.push(parseInt(chunk).toString(16));
|
||||||
|
}
|
||||||
|
fa = ["3", chunks];
|
||||||
|
} else {
|
||||||
|
// 如果书籍 ID 包含其他字符,则将每个字符的 Unicode 编码转换为十六进制表示
|
||||||
|
let hexStr = "";
|
||||||
|
for (let i = 0; i < bookId.length; i++) {
|
||||||
|
hexStr += bookId.charCodeAt(i).toString(16);
|
||||||
|
}
|
||||||
|
fa = ["4", [hexStr]];
|
||||||
|
}
|
||||||
|
|
||||||
|
// 将类型添加到初始值中
|
||||||
|
strSub += fa[0];
|
||||||
|
|
||||||
|
// 将数字2和哈希结果的后两个字符添加到初始值中
|
||||||
|
strSub += "2" + str.substring(str.length - 2);
|
||||||
|
|
||||||
|
// 处理转换后的子字符串数组
|
||||||
|
for (let i = 0; i < fa[1].length; i++) {
|
||||||
|
const sub = fa[1][i];
|
||||||
|
const subLength = sub.length.toString(16);
|
||||||
|
|
||||||
|
// 如果长度只有一位数,则在前面添加0
|
||||||
|
const subLengthPadded =
|
||||||
|
subLength.length === 1 ? "0" + subLength : subLength;
|
||||||
|
|
||||||
|
// 将长度和子字符串添加到初始值中
|
||||||
|
strSub += subLengthPadded + sub;
|
||||||
|
|
||||||
|
// 如果不是最后一个子字符串,则添加分隔符 'g'
|
||||||
|
if (i < fa[1].length - 1) {
|
||||||
|
strSub += "g";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 如果初始值长度不足 20,从哈希结果中取足够的字符补齐
|
||||||
|
if (strSub.length < 20) {
|
||||||
|
strSub += str.substring(0, 20 - strSub.length);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 使用 MD5 哈希算法创建新的哈希对象
|
||||||
|
const finalHash = crypto.createHash("md5");
|
||||||
|
finalHash.update(strSub);
|
||||||
|
const finalStr = finalHash.digest("hex");
|
||||||
|
|
||||||
|
// 取最终哈希结果的前三个字符并添加到初始值的末尾
|
||||||
|
strSub += finalStr.substring(0, 3);
|
||||||
|
|
||||||
|
return strSub;
|
||||||
|
} catch (error) {
|
||||||
|
console.error("处理微信读书 ID 时出现错误:" + error);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
module.exports = getWereadID;
|
Loading…
Reference in New Issue
Block a user