mirror of
https://github.com/Jerryplusy/crystelf-plugin.git
synced 2025-07-04 06:09:19 +00:00
226 lines
5.9 KiB
JavaScript
226 lines
5.9 KiB
JavaScript
import fs from 'node:fs';
|
|
import path from 'path';
|
|
import chokidar from 'chokidar';
|
|
import ConfigControl from '../lib/config/configControl.js';
|
|
import Fanqie from '../models/apps/fanqie/fanqie.js';
|
|
|
|
/**
|
|
* 本功能由 y68(github@yeqiu6080) 提供技术支持
|
|
*/
|
|
export default class FanqiePlugin extends plugin {
|
|
constructor() {
|
|
super({
|
|
name: 'crystelf-fanqie',
|
|
dsc: '番茄小说下载器',
|
|
event: 'message',
|
|
priority: -114,
|
|
rule: [
|
|
{
|
|
reg: '(changdunovel.com/wap/share-v2.html|fanqienovel.com/page)',
|
|
fnc: 'handleFanqieLink',
|
|
},
|
|
{
|
|
reg: '#?fq下载(.*)',
|
|
fnc: 'downloadByBookId',
|
|
},
|
|
{
|
|
reg: '^fq清(理|除|空)缓存$',
|
|
fnc: 'clearFanqieCache',
|
|
},
|
|
],
|
|
});
|
|
|
|
this.initPromise = this.initFanqieConfig();
|
|
this.fanqieClient = null;
|
|
|
|
// 注册计划任务
|
|
this.task = {
|
|
cron: '0 0 16 * * ?',
|
|
name: '定时清理番茄缓存',
|
|
fnc: () => this.clearFanqieCache(false, true),
|
|
};
|
|
}
|
|
|
|
async initFanqieConfig() {
|
|
this.outDir = await ConfigControl.get('fanqieConfig')?.outDir;
|
|
this.apiUrl = await ConfigControl.get('fanqieConfig')?.url;
|
|
this.fanqieClient = new Fanqie(this.apiUrl);
|
|
}
|
|
|
|
/**
|
|
* 监听下载输出目录
|
|
*/
|
|
async waitForOutputFile(dir, timeout = 30000) {
|
|
if (!dir) return false;
|
|
|
|
return new Promise((resolve) => {
|
|
const watcher = chokidar.watch(dir, {
|
|
persistent: true,
|
|
ignoreInitial: true,
|
|
});
|
|
|
|
const timer = setTimeout(() => {
|
|
watcher.close();
|
|
resolve(false);
|
|
}, timeout);
|
|
|
|
watcher.on('add', (filePath) => {
|
|
clearTimeout(timer);
|
|
watcher.close();
|
|
resolve(filePath);
|
|
});
|
|
});
|
|
}
|
|
|
|
/**
|
|
* 清理缓存
|
|
*/
|
|
async clearFanqieCache(e, isScheduled = false, specificId = false) {
|
|
if (!isScheduled && e && !e.isMaster) {
|
|
e.reply('你没有权限使用此功能', true);
|
|
return false;
|
|
}
|
|
|
|
if (!this.outDir) {
|
|
await this.initPromise;
|
|
if (!this.outDir) {
|
|
if (e) e.reply('缓存目录未初始化,无法清理', true);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
if (specificId) {
|
|
const specificPath = path.join(this.outDir, 'files', specificId);
|
|
if (fs.existsSync(specificPath)) {
|
|
fs.rmSync(specificPath, { recursive: true, force: true });
|
|
}
|
|
}
|
|
|
|
const mainCachePath = path.join(this.outDir, 'fanqie');
|
|
if (fs.existsSync(mainCachePath)) {
|
|
fs.readdirSync(mainCachePath).forEach((file) => {
|
|
const fullPath = path.join(mainCachePath, file);
|
|
const stat = fs.statSync(fullPath);
|
|
if (stat.isDirectory()) {
|
|
fs.rmSync(fullPath, { recursive: true, force: true });
|
|
} else {
|
|
fs.unlinkSync(fullPath);
|
|
}
|
|
});
|
|
}
|
|
|
|
if (!isScheduled && e) e.reply('缓存清理完成', true);
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* 解析网页链接中的 book_id
|
|
*/
|
|
async handleFanqieLink(e) {
|
|
const message = e.msg.trim();
|
|
let bookId = null;
|
|
|
|
try {
|
|
if (message.includes('changdunovel.com')) {
|
|
bookId = message.match(/book_id=(\d+)/)[1];
|
|
} else {
|
|
bookId = message.match(/page\/(\d+)/)[1];
|
|
}
|
|
} catch {
|
|
return e.reply('解析失败,请检查链接是否正确', true);
|
|
}
|
|
|
|
return this.downloadFanqieBook(e, bookId);
|
|
}
|
|
|
|
/**
|
|
* 使用 #fq下载 命令下载
|
|
*/
|
|
async downloadByBookId(e) {
|
|
const bookId = e.msg.replace(/^#?fq下载/, '').trim();
|
|
return this.downloadFanqieBook(e, bookId);
|
|
}
|
|
|
|
/**
|
|
* 执行下载并上传文件
|
|
*/
|
|
async downloadFanqieBook(e, bookId) {
|
|
await this.initPromise;
|
|
|
|
let bookInfo;
|
|
try {
|
|
bookInfo = await this.fanqieClient.get_info(bookId);
|
|
} catch (err) {
|
|
logger.error(err);
|
|
return e.reply('获取小说信息失败', true);
|
|
}
|
|
|
|
if (!bookInfo) return e.reply('获取失败,请稍后再试', true);
|
|
|
|
e.reply(
|
|
`识别小说:[番茄小说]《${bookInfo.book_name}》\n作者:${bookInfo.author}\n原名:${bookInfo.original_book_name}`,
|
|
true
|
|
);
|
|
|
|
e.reply('开始下载,请稍等片刻...', true);
|
|
const startTime = Date.now();
|
|
|
|
try {
|
|
await this.fanqieClient.down(bookId, e.message_id);
|
|
} catch (err) {
|
|
logger.error(err);
|
|
return e.reply('下载失败,请稍后重试', true);
|
|
}
|
|
|
|
const outPath = path.join(this.outDir, 'files', String(e.message_id));
|
|
let finalFilePath = await this.waitForOutputFile(outPath);
|
|
if (!finalFilePath) return e.reply('下载超时', true);
|
|
|
|
// 文件重命名防止空格
|
|
const safePath = finalFilePath.replace(/ /g, '_');
|
|
if (finalFilePath !== safePath) {
|
|
try {
|
|
fs.renameSync(finalFilePath, safePath);
|
|
finalFilePath = safePath;
|
|
} catch (err) {
|
|
logger.error(`重命名失败:${err.stack}`);
|
|
return e.reply('重命名失败', true);
|
|
}
|
|
}
|
|
|
|
const uploaded = await this.sendFileToUser(e, finalFilePath);
|
|
await this.clearFanqieCache(false, true, String(e.message_id));
|
|
|
|
if (!uploaded) return e.reply('上传失败', true);
|
|
|
|
e.reply(`《${bookInfo.book_name}》上传成功,耗时 ${(Date.now() - startTime) / 1000}s`);
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* 上传文件至群或私聊
|
|
*/
|
|
async sendFileToUser(e, filePath) {
|
|
try {
|
|
const fileName = path.basename(filePath);
|
|
if (e.isGroup) {
|
|
return await e.bot.sendApi('upload_group_file', {
|
|
group_id: e.group_id,
|
|
file: filePath,
|
|
name: fileName,
|
|
});
|
|
} else if (e.friend) {
|
|
return await e.bot.sendApi('upload_private_file', {
|
|
user_id: e.user_id,
|
|
file: filePath,
|
|
name: fileName,
|
|
});
|
|
}
|
|
} catch (err) {
|
|
logger.error(`文件上传失败:${logger.red(err.stack)}`);
|
|
e.reply(`上传失败:${err.message}`, true);
|
|
return null;
|
|
}
|
|
}
|
|
}
|