mirror of
https://github.com/MoonTechLab/LunaTV.git
synced 2026-02-15 04:14:40 +08:00
258 lines
8.2 KiB
TypeScript
258 lines
8.2 KiB
TypeScript
/* eslint-disable no-console,@typescript-eslint/no-explicit-any */
|
||
|
||
import * as crypto from 'crypto';
|
||
import { NextRequest, NextResponse } from 'next/server';
|
||
|
||
import { getConfig, refineConfig } from '@/lib/config';
|
||
import { db } from '@/lib/db';
|
||
import { fetchVideoDetail } from '@/lib/fetchVideoDetail';
|
||
import { refreshLiveChannels } from '@/lib/live';
|
||
import { SearchResult } from '@/lib/types';
|
||
|
||
export const runtime = 'nodejs';
|
||
|
||
export async function GET(request: NextRequest) {
|
||
console.log(request.url);
|
||
try {
|
||
console.log('Cron job triggered:', new Date().toISOString());
|
||
|
||
cronJob();
|
||
|
||
return NextResponse.json({
|
||
success: true,
|
||
message: 'Cron job executed successfully',
|
||
timestamp: new Date().toISOString(),
|
||
});
|
||
} catch (error) {
|
||
console.error('Cron job failed:', error);
|
||
|
||
return NextResponse.json(
|
||
{
|
||
success: false,
|
||
message: 'Cron job failed',
|
||
error: error instanceof Error ? error.message : 'Unknown error',
|
||
timestamp: new Date().toISOString(),
|
||
},
|
||
{ status: 500 }
|
||
);
|
||
}
|
||
}
|
||
|
||
async function cronJob() {
|
||
await refreshConfig();
|
||
await refreshAllLiveChannels();
|
||
await refreshRecordAndFavorites();
|
||
}
|
||
|
||
async function refreshAllLiveChannels() {
|
||
const config = await getConfig();
|
||
|
||
// 并发刷新所有启用的直播源
|
||
const refreshPromises = (config.LiveConfig || [])
|
||
.filter(liveInfo => !liveInfo.disabled)
|
||
.map(async (liveInfo) => {
|
||
try {
|
||
const nums = await refreshLiveChannels(liveInfo);
|
||
liveInfo.channelNumber = nums;
|
||
} catch (error) {
|
||
console.error(`刷新直播源失败 [${liveInfo.name || liveInfo.key}]:`, error);
|
||
liveInfo.channelNumber = 0;
|
||
}
|
||
});
|
||
|
||
// 等待所有刷新任务完成
|
||
await Promise.all(refreshPromises);
|
||
|
||
// 保存配置
|
||
await db.saveAdminConfig(config);
|
||
}
|
||
|
||
async function refreshConfig() {
|
||
let config = await getConfig();
|
||
if (config && config.ConfigSubscribtion && config.ConfigSubscribtion.URL && config.ConfigSubscribtion.AutoUpdate) {
|
||
try {
|
||
const response = await fetch(config.ConfigSubscribtion.URL);
|
||
|
||
if (!response.ok) {
|
||
throw new Error(`请求失败: ${response.status} ${response.statusText}`);
|
||
}
|
||
|
||
const configContent = await response.text();
|
||
|
||
// 对 configContent 进行 base58 解码
|
||
let decodedContent;
|
||
try {
|
||
const bs58 = (await import('bs58')).default;
|
||
const decodedBytes = bs58.decode(configContent);
|
||
decodedContent = new TextDecoder().decode(decodedBytes);
|
||
} catch (decodeError) {
|
||
console.warn('Base58 解码失败:', decodeError);
|
||
throw decodeError;
|
||
}
|
||
|
||
try {
|
||
JSON.parse(decodedContent);
|
||
} catch (e) {
|
||
throw new Error('配置文件格式错误,请检查 JSON 语法');
|
||
}
|
||
config.ConfigFile = decodedContent;
|
||
config.ConfigSubscribtion.LastCheck = new Date().toISOString();
|
||
config = refineConfig(config);
|
||
await db.saveAdminConfig(config);
|
||
} catch (e) {
|
||
console.error('刷新配置失败:', e);
|
||
}
|
||
} else {
|
||
console.log('跳过刷新:未配置订阅地址或自动更新');
|
||
}
|
||
}
|
||
|
||
async function refreshRecordAndFavorites() {
|
||
try {
|
||
const users = await db.getAllUsers();
|
||
if (process.env.USERNAME && !users.includes(process.env.USERNAME)) {
|
||
users.push(process.env.USERNAME);
|
||
}
|
||
// 函数级缓存:key 为 `${source}+${id}`,值为 Promise<VideoDetail | null>
|
||
const detailCache = new Map<string, Promise<SearchResult | null>>();
|
||
|
||
// 获取详情 Promise(带缓存和错误处理)
|
||
const getDetail = async (
|
||
source: string,
|
||
id: string,
|
||
fallbackTitle: string
|
||
): Promise<SearchResult | null> => {
|
||
const key = `${source}+${id}`;
|
||
let promise = detailCache.get(key);
|
||
if (!promise) {
|
||
promise = fetchVideoDetail({
|
||
source,
|
||
id,
|
||
fallbackTitle: fallbackTitle.trim(),
|
||
})
|
||
.then((detail) => {
|
||
// 成功时才缓存结果
|
||
const successPromise = Promise.resolve(detail);
|
||
detailCache.set(key, successPromise);
|
||
return detail;
|
||
})
|
||
.catch((err) => {
|
||
console.error(`获取视频详情失败 (${source}+${id}):`, err);
|
||
return null;
|
||
});
|
||
}
|
||
return promise;
|
||
};
|
||
|
||
for (const user of users) {
|
||
console.log(`开始处理用户: ${user}`);
|
||
|
||
// 播放记录
|
||
try {
|
||
const playRecords = await db.getAllPlayRecords(user);
|
||
const totalRecords = Object.keys(playRecords).length;
|
||
let processedRecords = 0;
|
||
|
||
for (const [key, record] of Object.entries(playRecords)) {
|
||
try {
|
||
const [source, id] = key.split('+');
|
||
if (!source || !id) {
|
||
console.warn(`跳过无效的播放记录键: ${key}`);
|
||
continue;
|
||
}
|
||
|
||
const detail = await getDetail(source, id, record.title);
|
||
if (!detail) {
|
||
console.warn(`跳过无法获取详情的播放记录: ${key}`);
|
||
continue;
|
||
}
|
||
|
||
const episodeCount = detail.episodes?.length || 0;
|
||
if (episodeCount > 0 && episodeCount !== record.total_episodes) {
|
||
await db.savePlayRecord(user, source, id, {
|
||
title: detail.title || record.title,
|
||
source_name: record.source_name,
|
||
cover: detail.poster || record.cover,
|
||
index: record.index,
|
||
total_episodes: episodeCount,
|
||
play_time: record.play_time,
|
||
year: detail.year || record.year,
|
||
total_time: record.total_time,
|
||
save_time: record.save_time,
|
||
search_title: record.search_title,
|
||
});
|
||
console.log(
|
||
`更新播放记录: ${record.title} (${record.total_episodes} -> ${episodeCount})`
|
||
);
|
||
}
|
||
|
||
processedRecords++;
|
||
} catch (err) {
|
||
console.error(`处理播放记录失败 (${key}):`, err);
|
||
// 继续处理下一个记录
|
||
}
|
||
}
|
||
|
||
console.log(`播放记录处理完成: ${processedRecords}/${totalRecords}`);
|
||
} catch (err) {
|
||
console.error(`获取用户播放记录失败 (${user}):`, err);
|
||
}
|
||
|
||
// 收藏
|
||
try {
|
||
let favorites = await db.getAllFavorites(user);
|
||
favorites = Object.fromEntries(
|
||
Object.entries(favorites).filter(([_, fav]) => fav.origin !== 'live')
|
||
);
|
||
const totalFavorites = Object.keys(favorites).length;
|
||
let processedFavorites = 0;
|
||
|
||
for (const [key, fav] of Object.entries(favorites)) {
|
||
try {
|
||
const [source, id] = key.split('+');
|
||
if (!source || !id) {
|
||
console.warn(`跳过无效的收藏键: ${key}`);
|
||
continue;
|
||
}
|
||
|
||
const favDetail = await getDetail(source, id, fav.title);
|
||
if (!favDetail) {
|
||
console.warn(`跳过无法获取详情的收藏: ${key}`);
|
||
continue;
|
||
}
|
||
|
||
const favEpisodeCount = favDetail.episodes?.length || 0;
|
||
if (favEpisodeCount > 0 && favEpisodeCount !== fav.total_episodes) {
|
||
await db.saveFavorite(user, source, id, {
|
||
title: favDetail.title || fav.title,
|
||
source_name: fav.source_name,
|
||
cover: favDetail.poster || fav.cover,
|
||
year: favDetail.year || fav.year,
|
||
total_episodes: favEpisodeCount,
|
||
save_time: fav.save_time,
|
||
search_title: fav.search_title,
|
||
});
|
||
console.log(
|
||
`更新收藏: ${fav.title} (${fav.total_episodes} -> ${favEpisodeCount})`
|
||
);
|
||
}
|
||
|
||
processedFavorites++;
|
||
} catch (err) {
|
||
console.error(`处理收藏失败 (${key}):`, err);
|
||
// 继续处理下一个收藏
|
||
}
|
||
}
|
||
|
||
console.log(`收藏处理完成: ${processedFavorites}/${totalFavorites}`);
|
||
} catch (err) {
|
||
console.error(`获取用户收藏失败 (${user}):`, err);
|
||
}
|
||
}
|
||
|
||
console.log('刷新播放记录/收藏任务完成');
|
||
} catch (err) {
|
||
console.error('刷新播放记录/收藏任务启动失败', err);
|
||
}
|
||
}
|