mirror of
https://github.com/Jerryplusy/rc-plugin.git
synced 2025-10-14 08:09:19 +00:00
✨ feat: V1.4.0 新增米游社文章解析
1. 新增米游社文章解析,感谢群友mix提供的想法 2. 新增ks的图片解析,感谢群友风をした提供手机端图片的例子 3. 格式化部分代码
This commit is contained in:
parent
57c5d2b453
commit
f62bb7a695
@ -8,12 +8,17 @@ import axios from "axios";
|
||||
// 常量
|
||||
import { CAT_LIMIT } from "../constants/constant.js";
|
||||
// 书库
|
||||
import { getZHelper, getYiBook, getZBook } from "../utils/books.js";
|
||||
import { getYiBook, getZBook, getZHelper } from "../utils/books.js";
|
||||
// 工具类
|
||||
import _ from "lodash";
|
||||
import TokenBucket from '../utils/token-bucket.js'
|
||||
|
||||
export class query extends plugin {
|
||||
/**
|
||||
* 令牌桶 拿来限流
|
||||
* @type {TokenBucket}
|
||||
*/
|
||||
static #tokenBucket = new TokenBucket(1, 1, 60);
|
||||
|
||||
constructor() {
|
||||
super({
|
||||
name: "R插件查询类",
|
||||
@ -386,10 +391,4 @@ export class query extends plugin {
|
||||
const titleRex = /<[^>]+>/g;
|
||||
return title.replace(titleRex, "");
|
||||
}
|
||||
|
||||
/**
|
||||
* 令牌桶 拿来限流
|
||||
* @type {TokenBucket}
|
||||
*/
|
||||
static #tokenBucket = new TokenBucket(1, 1, 60);
|
||||
}
|
||||
|
137
apps/tools.js
137
apps/tools.js
@ -26,7 +26,7 @@ import {
|
||||
TWITTER_BEARER_TOKEN,
|
||||
XHS_NO_WATERMARK_HEADER,
|
||||
} from "../constants/constant.js";
|
||||
import { containsChinese, formatBiliInfo, getIdVideo, secondsToTime } from "../utils/common.js";
|
||||
import { containsChinese, downloadImg, formatBiliInfo, getIdVideo, secondsToTime } from "../utils/common.js";
|
||||
import config from "../model/index.js";
|
||||
import Translate from "../utils/trans-strategy.js";
|
||||
import * as xBogus from "../utils/x-bogus.cjs";
|
||||
@ -38,6 +38,7 @@ import { getWbi } from "../utils/biliWbi.js";
|
||||
import {
|
||||
BILI_SUMMARY,
|
||||
DY_INFO,
|
||||
MIYOUSHE_ARTICLE,
|
||||
TIKTOK_INFO,
|
||||
TWITTER_TWEET_INFO,
|
||||
XHS_REQ_LINK,
|
||||
@ -47,6 +48,7 @@ import {
|
||||
import child_process from 'node:child_process'
|
||||
import { getAudio, getVideo } from "../utils/y2b.js";
|
||||
import { processTikTokUrl } from "../utils/tiktok.js";
|
||||
import { getDS } from "../utils/mihoyo.js";
|
||||
|
||||
export class tools extends plugin {
|
||||
/**
|
||||
@ -134,6 +136,10 @@ export class tools extends plugin {
|
||||
{
|
||||
reg: "(ixigua.com)",
|
||||
fnc: "xigua"
|
||||
},
|
||||
{
|
||||
reg: "(miyoushe.com)",
|
||||
fnc: "miyoushe"
|
||||
}
|
||||
],
|
||||
});
|
||||
@ -566,7 +572,7 @@ export class tools extends plugin {
|
||||
for (let item of resp.includes.media) {
|
||||
if (item.type === "photo") {
|
||||
// 图片
|
||||
task.push(this.downloadImg(item.url, downloadPath, "", true));
|
||||
task.push(downloadImg(item.url, downloadPath, "", true));
|
||||
} else if (item.type === "video") {
|
||||
// 视频
|
||||
await this.downloadVideo(resp.includes.media[0].variants[0].url, true).then(
|
||||
@ -686,7 +692,7 @@ export class tools extends plugin {
|
||||
} else if (type === "normal") {
|
||||
e.reply(`识别:小红书, ${ title }\n${ desc }`);
|
||||
noteData.imageList.map(async (item, index) => {
|
||||
imgPromise.push(this.downloadImg(item.urlDefault, downloadPath, index.toString()));
|
||||
imgPromise.push(downloadImg(item.urlDefault, downloadPath, index.toString()));
|
||||
});
|
||||
}
|
||||
const paths = await Promise.all(imgPromise);
|
||||
@ -773,7 +779,7 @@ export class tools extends plugin {
|
||||
// 判断是否是海外服务器
|
||||
const isOversea = await this.isOverseasServer();
|
||||
// 简单封装图片下载
|
||||
const downloadImg = (url, destination) => {
|
||||
const downloadInsImg = (url, destination) => {
|
||||
return new Promise((resolve, reject) => {
|
||||
fetch(url, {
|
||||
timeout: 10000,
|
||||
@ -805,7 +811,7 @@ export class tools extends plugin {
|
||||
.exec(item)[0]
|
||||
.replace(/#38/g, "")
|
||||
.replace(/;/g, "");
|
||||
imgPromise.push(downloadImg(imgUrl, `${ downloadPath }/${ index }.jpg`));
|
||||
imgPromise.push(downloadInsImg(imgUrl, `${ downloadPath }/${ index }.jpg`));
|
||||
});
|
||||
}
|
||||
// TODO 视频,会出bug暂时不做
|
||||
@ -924,10 +930,25 @@ export class tools extends plugin {
|
||||
},
|
||||
timeout: 10000 // 设置超时时间
|
||||
}).then(resp => {
|
||||
// 图片:https://kph8gvfz.m.chenzhongtech.com/fw/photo/3x45s52s9wchwwm
|
||||
|
||||
if (resp.data.data?.imageUrl) {
|
||||
const imageUrl = resp.data.data.imageUrl;
|
||||
const images = imageUrl.map(item => {
|
||||
return {
|
||||
message: segment.image(item),
|
||||
nickname: this.e.sender.card || this.e.user_id,
|
||||
user_id: this.e.user_id,
|
||||
}
|
||||
})
|
||||
e.reply(Bot.makeForwardMsg(images));
|
||||
} else {
|
||||
// 视频:https://www.kuaishou.com/short-video/3xhjgcmir24m4nm
|
||||
const url = resp.data.data.url;
|
||||
this.downloadVideo(url).then(path => {
|
||||
e.reply(segment.video(path + "/temp.mp4"));
|
||||
});
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
@ -1075,6 +1096,60 @@ export class tools extends plugin {
|
||||
return true
|
||||
}
|
||||
|
||||
async miyoushe(e) {
|
||||
let msg = /(?:https?:\/\/)?(m|www)\.miyoushe\.com\/[A-Za-z\d._?%&+\-=\/#]*/.exec(e.msg)[0];
|
||||
const id = /\/(\d+)$/.exec(msg)?.[0].replace("\/", "");
|
||||
|
||||
fetch(MIYOUSHE_ARTICLE.replace("{}", id), {
|
||||
headers: {
|
||||
"Accept-Encoding": "gzip, deflate, br",
|
||||
"Accept-Language": "zh-cn",
|
||||
"Connection": "keep-alive",
|
||||
"Host": "api-takumi.mihoyo.com",
|
||||
"x-rpc-app_version": "2.11.0",
|
||||
"x-rpc-client_type": "4",
|
||||
"Referer": "https://bbs.mihoyo.com/",
|
||||
"DS": getDS(),
|
||||
}
|
||||
}).then(async resp => {
|
||||
const respJson = await resp.json();
|
||||
const data = respJson.data.post.post;
|
||||
// 分别获取:封面、主题、内容、图片
|
||||
const { cover, subject, content, images, structured_content } = data;
|
||||
let realContent = "";
|
||||
// safe JSON.parse
|
||||
try {
|
||||
realContent = JSON.parse(content);
|
||||
} catch (e) {
|
||||
realContent = content;
|
||||
}
|
||||
const normalMsg = `识别:米游社,${ subject }\n${ realContent }`;
|
||||
const replyMsg = cover ? [segment.image(cover), normalMsg] : normalMsg;
|
||||
e.reply(replyMsg);
|
||||
// 视频
|
||||
if (structured_content) {
|
||||
const sc = JSON.parse(structured_content);
|
||||
const resolutions = sc?.[1].insert.vod.resolutions;
|
||||
// 暂时选取分辨率较低的video进行解析
|
||||
const videoUrl = resolutions[0].url;
|
||||
this.downloadVideo(videoUrl).then(path => {
|
||||
e.reply(segment.video(path + "/temp.mp4"));
|
||||
});
|
||||
}
|
||||
// 这个判断防止发送重复图片
|
||||
if (images && images.length > 1) {
|
||||
const replyImages = images.map(item => {
|
||||
return {
|
||||
message: segment.image(item),
|
||||
nickname: this.e.sender.card || this.e.user_id,
|
||||
user_id: this.e.user_id,
|
||||
}
|
||||
});
|
||||
e.reply(Bot.makeForwardMsg(replyImages));
|
||||
}
|
||||
})
|
||||
}
|
||||
|
||||
/**
|
||||
* 哔哩哔哩下载
|
||||
* @param title
|
||||
@ -1111,58 +1186,6 @@ export class tools extends plugin {
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* 下载一张网络图片(自动以url的最后一个为名字)
|
||||
* @param img
|
||||
* @param dir
|
||||
* @param fileName
|
||||
* @param isProxy
|
||||
* @returns {Promise<unknown>}
|
||||
*/
|
||||
async downloadImg(img, dir, fileName = "", isProxy = false) {
|
||||
if (fileName === "") {
|
||||
fileName = img.split("/").pop();
|
||||
}
|
||||
const filepath = `${ dir }/${ fileName }`;
|
||||
await mkdirIfNotExists(dir)
|
||||
const writer = fs.createWriteStream(filepath);
|
||||
const axiosConfig = {
|
||||
headers: {
|
||||
"User-Agent":
|
||||
"Mozilla/5.0 (Linux; Android 5.0; SM-G900P Build/LRX21T) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/70.0.3538.25 Mobile Safari/537.36",
|
||||
},
|
||||
responseType: "stream",
|
||||
};
|
||||
|
||||
if (isProxy) {
|
||||
axiosConfig.httpAgent = tunnel.httpOverHttp({
|
||||
proxy: { host: this.proxyAddr, port: this.proxyPort },
|
||||
});
|
||||
axiosConfig.httpsAgent = tunnel.httpOverHttp({
|
||||
proxy: { host: this.proxyAddr, port: this.proxyPort },
|
||||
});
|
||||
}
|
||||
try {
|
||||
const res = await axios.get(img, axiosConfig);
|
||||
res.data.pipe(writer);
|
||||
|
||||
return new Promise((resolve, reject) => {
|
||||
writer.on("finish", () => {
|
||||
writer.close(() => {
|
||||
resolve(filepath);
|
||||
});
|
||||
});
|
||||
writer.on("error", err => {
|
||||
fs.unlink(filepath, () => {
|
||||
reject(err);
|
||||
});
|
||||
});
|
||||
});
|
||||
} catch (err) {
|
||||
logger.error("图片下载失败");
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* douyin 请求参数
|
||||
* @param url
|
||||
|
@ -56,6 +56,12 @@
|
||||
- icon: kuaishou
|
||||
title: "快手(测试阶段)"
|
||||
desc: 快手分享实时下载
|
||||
- icon: xigua
|
||||
title: "西瓜(测试阶段)"
|
||||
desc: 西瓜分享实时下载
|
||||
- icon: miyoushe
|
||||
title: "米游社"
|
||||
desc: 米游社文章分享实时下载
|
||||
- icon: literature
|
||||
title: "论文"
|
||||
desc: SCI论文实时解析
|
||||
|
@ -1,7 +1,8 @@
|
||||
- {
|
||||
version: 1.3.5,
|
||||
version: 1.4.0,
|
||||
data:
|
||||
[
|
||||
新增<span class="cmd">米游社解析</span>功能,
|
||||
新增<span class="cmd">🍉解析</span>功能,
|
||||
新增<span class="cmd">油管解析</span>功能,
|
||||
新增<span class="cmd">小红书无水印下载</span>功能,
|
||||
|
@ -40,6 +40,13 @@ export const BILI_VIDEO_INFO = "http://api.bilibili.com/x/web-interface/view"
|
||||
*/
|
||||
export const BILI_NAV = "https://api.bilibili.com/x/web-interface/nav"
|
||||
|
||||
/**
|
||||
* 米游社网页端获取文章
|
||||
* https://github.com/UIGF-org/mihoyo-api-collect/blob/main/hoyolab/article/article.md#%E8%8E%B7%E5%8F%96%E5%AE%8C%E6%95%B4%E6%96%87%E7%AB%A0%E4%BF%A1%E6%81%AF
|
||||
* @type {string}
|
||||
*/
|
||||
export const MIYOUSHE_ARTICLE = "https://bbs-api.miyoushe.com/post/wapi/getPostFull?post_id={}"
|
||||
|
||||
/**
|
||||
* 视频请求链接CDN
|
||||
* @type {string}
|
||||
|
BIN
resources/img/icon/miyoushe.png
Normal file
BIN
resources/img/icon/miyoushe.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 42 KiB |
BIN
resources/img/icon/xigua.png
Normal file
BIN
resources/img/icon/xigua.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 6.8 KiB |
@ -14,6 +14,7 @@ export class jFetch {
|
||||
const r = await fetch(url);
|
||||
return await r.json();
|
||||
}
|
||||
|
||||
async post(url, params) {
|
||||
const r = await fetch(url, { ...params, method: "POST" });
|
||||
return await r.json();
|
||||
@ -163,6 +164,58 @@ export async function downloadMp3(mp3Url, path, redirect = "manual") {
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* 下载一张网络图片(自动以url的最后一个为名字)
|
||||
* @param img
|
||||
* @param dir
|
||||
* @param fileName
|
||||
* @param isProxy
|
||||
* @returns {Promise<unknown>}
|
||||
*/
|
||||
export async function downloadImg(img, dir, fileName = "", isProxy = false) {
|
||||
if (fileName === "") {
|
||||
fileName = img.split("/").pop();
|
||||
}
|
||||
const filepath = `${ dir }/${ fileName }`;
|
||||
await mkdirIfNotExists(dir)
|
||||
const writer = fs.createWriteStream(filepath);
|
||||
const axiosConfig = {
|
||||
headers: {
|
||||
"User-Agent":
|
||||
"Mozilla/5.0 (Linux; Android 5.0; SM-G900P Build/LRX21T) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/70.0.3538.25 Mobile Safari/537.36",
|
||||
},
|
||||
responseType: "stream",
|
||||
};
|
||||
|
||||
if (isProxy) {
|
||||
axiosConfig.httpAgent = tunnel.httpOverHttp({
|
||||
proxy: { host: this.proxyAddr, port: this.proxyPort },
|
||||
});
|
||||
axiosConfig.httpsAgent = tunnel.httpOverHttp({
|
||||
proxy: { host: this.proxyAddr, port: this.proxyPort },
|
||||
});
|
||||
}
|
||||
try {
|
||||
const res = await axios.get(img, axiosConfig);
|
||||
res.data.pipe(writer);
|
||||
|
||||
return new Promise((resolve, reject) => {
|
||||
writer.on("finish", () => {
|
||||
writer.close(() => {
|
||||
resolve(filepath);
|
||||
});
|
||||
});
|
||||
writer.on("error", err => {
|
||||
fs.unlink(filepath, () => {
|
||||
reject(err);
|
||||
});
|
||||
});
|
||||
});
|
||||
} catch (err) {
|
||||
logger.error("图片下载失败");
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 千位数的数据处理
|
||||
* @param data
|
||||
|
14
utils/mihoyo.js
Normal file
14
utils/mihoyo.js
Normal file
@ -0,0 +1,14 @@
|
||||
import md5 from 'md5';
|
||||
|
||||
export const getDS = () => {
|
||||
const salt = "ZSHlXeQUBis52qD1kEgKt5lUYed4b7Bb";
|
||||
const lettersAndNumbers = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789'
|
||||
|
||||
const i = Math.floor(Date.now() / 1000);
|
||||
let r = ""
|
||||
for (let i; i < 6; i++) {
|
||||
r += lettersAndNumbers[Math.floor(Math.random() * lettersAndNumbers.length)]
|
||||
}
|
||||
const c = md5(`salt=${ salt }&t=${ i }&r=${ r }`);
|
||||
return `${ i },${ r },${ c }`;
|
||||
}
|
Loading…
x
Reference in New Issue
Block a user