309 lines
9.1 KiB
TypeScript
309 lines
9.1 KiB
TypeScript
/* eslint-disable no-console, @typescript-eslint/no-explicit-any, @typescript-eslint/no-non-null-assertion */
|
|
|
|
import { Redis } from '@upstash/redis';
|
|
|
|
import { AdminConfig } from './admin.types';
|
|
import { Favorite, IStorage, PlayRecord } from './types';
|
|
|
|
// 搜索历史最大条数
|
|
const SEARCH_HISTORY_LIMIT = 20;
|
|
|
|
// 数据类型转换辅助函数
|
|
function ensureString(value: any): string {
|
|
return String(value);
|
|
}
|
|
|
|
function ensureStringArray(value: any[]): string[] {
|
|
return value.map((item) => String(item));
|
|
}
|
|
|
|
// 添加Upstash Redis操作重试包装器
|
|
async function withRetry<T>(
|
|
operation: () => Promise<T>,
|
|
maxRetries = 3
|
|
): Promise<T> {
|
|
for (let i = 0; i < maxRetries; i++) {
|
|
try {
|
|
return await operation();
|
|
} catch (err: any) {
|
|
const isLastAttempt = i === maxRetries - 1;
|
|
const isConnectionError =
|
|
err.message?.includes('Connection') ||
|
|
err.message?.includes('ECONNREFUSED') ||
|
|
err.message?.includes('ENOTFOUND') ||
|
|
err.code === 'ECONNRESET' ||
|
|
err.code === 'EPIPE' ||
|
|
err.name === 'UpstashError';
|
|
|
|
if (isConnectionError && !isLastAttempt) {
|
|
console.log(
|
|
`Upstash Redis operation failed, retrying... (${i + 1}/${maxRetries})`
|
|
);
|
|
console.error('Error:', err.message);
|
|
|
|
// 等待一段时间后重试
|
|
await new Promise((resolve) => setTimeout(resolve, 1000 * (i + 1)));
|
|
continue;
|
|
}
|
|
|
|
throw err;
|
|
}
|
|
}
|
|
|
|
throw new Error('Max retries exceeded');
|
|
}
|
|
|
|
export class UpstashRedisStorage implements IStorage {
|
|
private client: Redis;
|
|
|
|
constructor() {
|
|
this.client = getUpstashRedisClient();
|
|
}
|
|
|
|
// ---------- 播放记录 ----------
|
|
private prKey(user: string, key: string) {
|
|
return `u:${user}:pr:${key}`; // u:username:pr:source+id
|
|
}
|
|
|
|
async getPlayRecord(
|
|
userName: string,
|
|
key: string
|
|
): Promise<PlayRecord | null> {
|
|
const val = await withRetry(() =>
|
|
this.client.get(this.prKey(userName, key))
|
|
);
|
|
return val ? (val as PlayRecord) : null;
|
|
}
|
|
|
|
async setPlayRecord(
|
|
userName: string,
|
|
key: string,
|
|
record: PlayRecord
|
|
): Promise<void> {
|
|
await withRetry(() => this.client.set(this.prKey(userName, key), record));
|
|
}
|
|
|
|
async getAllPlayRecords(
|
|
userName: string
|
|
): Promise<Record<string, PlayRecord>> {
|
|
const pattern = `u:${userName}:pr:*`;
|
|
const keys: string[] = await withRetry(() => this.client.keys(pattern));
|
|
if (keys.length === 0) return {};
|
|
|
|
const result: Record<string, PlayRecord> = {};
|
|
for (const fullKey of keys) {
|
|
const value = await withRetry(() => this.client.get(fullKey));
|
|
if (value) {
|
|
// 截取 source+id 部分
|
|
const keyPart = ensureString(fullKey.replace(`u:${userName}:pr:`, ''));
|
|
result[keyPart] = value as PlayRecord;
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
async deletePlayRecord(userName: string, key: string): Promise<void> {
|
|
await withRetry(() => this.client.del(this.prKey(userName, key)));
|
|
}
|
|
|
|
// ---------- 收藏 ----------
|
|
private favKey(user: string, key: string) {
|
|
return `u:${user}:fav:${key}`;
|
|
}
|
|
|
|
async getFavorite(userName: string, key: string): Promise<Favorite | null> {
|
|
const val = await withRetry(() =>
|
|
this.client.get(this.favKey(userName, key))
|
|
);
|
|
return val ? (val as Favorite) : null;
|
|
}
|
|
|
|
async setFavorite(
|
|
userName: string,
|
|
key: string,
|
|
favorite: Favorite
|
|
): Promise<void> {
|
|
await withRetry(() =>
|
|
this.client.set(this.favKey(userName, key), favorite)
|
|
);
|
|
}
|
|
|
|
async getAllFavorites(userName: string): Promise<Record<string, Favorite>> {
|
|
const pattern = `u:${userName}:fav:*`;
|
|
const keys: string[] = await withRetry(() => this.client.keys(pattern));
|
|
if (keys.length === 0) return {};
|
|
|
|
const result: Record<string, Favorite> = {};
|
|
for (const fullKey of keys) {
|
|
const value = await withRetry(() => this.client.get(fullKey));
|
|
if (value) {
|
|
const keyPart = ensureString(fullKey.replace(`u:${userName}:fav:`, ''));
|
|
result[keyPart] = value as Favorite;
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
async deleteFavorite(userName: string, key: string): Promise<void> {
|
|
await withRetry(() => this.client.del(this.favKey(userName, key)));
|
|
}
|
|
|
|
// ---------- 用户注册 / 登录 ----------
|
|
private userPwdKey(user: string) {
|
|
return `u:${user}:pwd`;
|
|
}
|
|
|
|
async registerUser(userName: string, password: string): Promise<void> {
|
|
// 简单存储明文密码,生产环境应加密
|
|
await withRetry(() => this.client.set(this.userPwdKey(userName), password));
|
|
}
|
|
|
|
async verifyUser(userName: string, password: string): Promise<boolean> {
|
|
const stored = await withRetry(() =>
|
|
this.client.get(this.userPwdKey(userName))
|
|
);
|
|
if (stored === null) return false;
|
|
// 确保比较时都是字符串类型
|
|
return ensureString(stored) === password;
|
|
}
|
|
|
|
// 检查用户是否存在
|
|
async checkUserExist(userName: string): Promise<boolean> {
|
|
// 使用 EXISTS 判断 key 是否存在
|
|
const exists = await withRetry(() =>
|
|
this.client.exists(this.userPwdKey(userName))
|
|
);
|
|
return exists === 1;
|
|
}
|
|
|
|
// 修改用户密码
|
|
async changePassword(userName: string, newPassword: string): Promise<void> {
|
|
// 简单存储明文密码,生产环境应加密
|
|
await withRetry(() =>
|
|
this.client.set(this.userPwdKey(userName), newPassword)
|
|
);
|
|
}
|
|
|
|
// 删除用户及其所有数据
|
|
async deleteUser(userName: string): Promise<void> {
|
|
// 删除用户密码
|
|
await withRetry(() => this.client.del(this.userPwdKey(userName)));
|
|
|
|
// 删除搜索历史
|
|
await withRetry(() => this.client.del(this.shKey(userName)));
|
|
|
|
// 删除播放记录
|
|
const playRecordPattern = `u:${userName}:pr:*`;
|
|
const playRecordKeys = await withRetry(() =>
|
|
this.client.keys(playRecordPattern)
|
|
);
|
|
if (playRecordKeys.length > 0) {
|
|
await withRetry(() => this.client.del(...playRecordKeys));
|
|
}
|
|
|
|
// 删除收藏夹
|
|
const favoritePattern = `u:${userName}:fav:*`;
|
|
const favoriteKeys = await withRetry(() =>
|
|
this.client.keys(favoritePattern)
|
|
);
|
|
if (favoriteKeys.length > 0) {
|
|
await withRetry(() => this.client.del(...favoriteKeys));
|
|
}
|
|
}
|
|
|
|
// ---------- 搜索历史 ----------
|
|
private shKey(user: string) {
|
|
return `u:${user}:sh`; // u:username:sh
|
|
}
|
|
|
|
async getSearchHistory(userName: string): Promise<string[]> {
|
|
const result = await withRetry(() =>
|
|
this.client.lrange(this.shKey(userName), 0, -1)
|
|
);
|
|
// 确保返回的都是字符串类型
|
|
return ensureStringArray(result as any[]);
|
|
}
|
|
|
|
async addSearchHistory(userName: string, keyword: string): Promise<void> {
|
|
const key = this.shKey(userName);
|
|
// 先去重
|
|
await withRetry(() => this.client.lrem(key, 0, ensureString(keyword)));
|
|
// 插入到最前
|
|
await withRetry(() => this.client.lpush(key, ensureString(keyword)));
|
|
// 限制最大长度
|
|
await withRetry(() => this.client.ltrim(key, 0, SEARCH_HISTORY_LIMIT - 1));
|
|
}
|
|
|
|
async deleteSearchHistory(userName: string, keyword?: string): Promise<void> {
|
|
const key = this.shKey(userName);
|
|
if (keyword) {
|
|
await withRetry(() => this.client.lrem(key, 0, ensureString(keyword)));
|
|
} else {
|
|
await withRetry(() => this.client.del(key));
|
|
}
|
|
}
|
|
|
|
// ---------- 获取全部用户 ----------
|
|
async getAllUsers(): Promise<string[]> {
|
|
const keys = await withRetry(() => this.client.keys('u:*:pwd'));
|
|
return keys
|
|
.map((k) => {
|
|
const match = k.match(/^u:(.+?):pwd$/);
|
|
return match ? ensureString(match[1]) : undefined;
|
|
})
|
|
.filter((u): u is string => typeof u === 'string');
|
|
}
|
|
|
|
// ---------- 管理员配置 ----------
|
|
private adminConfigKey() {
|
|
return 'admin:config';
|
|
}
|
|
|
|
async getAdminConfig(): Promise<AdminConfig | null> {
|
|
const val = await withRetry(() => this.client.get(this.adminConfigKey()));
|
|
return val ? (val as AdminConfig) : null;
|
|
}
|
|
|
|
async setAdminConfig(config: AdminConfig): Promise<void> {
|
|
await withRetry(() => this.client.set(this.adminConfigKey(), config));
|
|
}
|
|
}
|
|
|
|
// 单例 Upstash Redis 客户端
|
|
function getUpstashRedisClient(): Redis {
|
|
const legacyKey = Symbol.for('__MOONTV_UPSTASH_REDIS_CLIENT__');
|
|
const globalKey = Symbol.for('__KATELYATV_UPSTASH_REDIS_CLIENT__');
|
|
let client: Redis | undefined = (global as any)[globalKey] || (global as any)[legacyKey];
|
|
|
|
if (!client) {
|
|
const upstashUrl = process.env.UPSTASH_URL;
|
|
const upstashToken = process.env.UPSTASH_TOKEN;
|
|
|
|
if (!upstashUrl || !upstashToken) {
|
|
throw new Error(
|
|
'UPSTASH_URL and UPSTASH_TOKEN env variables must be set'
|
|
);
|
|
}
|
|
|
|
// 创建 Upstash Redis 客户端
|
|
client = new Redis({
|
|
url: upstashUrl,
|
|
token: upstashToken,
|
|
// 可选配置
|
|
retry: {
|
|
retries: 3,
|
|
backoff: (retryCount: number) =>
|
|
Math.min(1000 * Math.pow(2, retryCount), 30000),
|
|
},
|
|
});
|
|
|
|
console.log('Upstash Redis client created successfully');
|
|
|
|
(global as any)[globalKey] = client;
|
|
// 同步设置旧的全局键,保持向后兼容
|
|
(global as any)[legacyKey] = client;
|
|
}
|
|
|
|
return client;
|
|
}
|