mirror of
https://github.com/zimplexing/OrionTV.git
synced 2026-02-04 03:36:29 +08:00
247 lines
6.7 KiB
TypeScript
247 lines
6.7 KiB
TypeScript
|
|
// region: --- Interface Definitions ---
|
|
export interface DoubanItem {
|
|
title: string;
|
|
poster: string;
|
|
rate?: string;
|
|
}
|
|
|
|
export interface DoubanResponse {
|
|
code: number;
|
|
message: string;
|
|
list: DoubanItem[];
|
|
}
|
|
|
|
export interface VideoDetail {
|
|
id: string;
|
|
title: string;
|
|
poster: string;
|
|
source: string;
|
|
source_name: string;
|
|
desc?: string;
|
|
type?: string;
|
|
year?: string;
|
|
area?: string;
|
|
director?: string;
|
|
actor?: string;
|
|
remarks?: string;
|
|
}
|
|
|
|
export interface SearchResult {
|
|
id: number;
|
|
title: string;
|
|
poster: string;
|
|
episodes: string[];
|
|
source: string;
|
|
source_name: string;
|
|
class?: string;
|
|
year: string;
|
|
desc?: string;
|
|
type_name?: string;
|
|
}
|
|
|
|
export interface Favorite {
|
|
cover: string;
|
|
title: string;
|
|
source_name: string;
|
|
total_episodes: number;
|
|
search_title: string;
|
|
year: string;
|
|
save_time?: number;
|
|
}
|
|
|
|
export interface PlayRecord {
|
|
title: string;
|
|
source_name: string;
|
|
cover: string;
|
|
index: number;
|
|
total_episodes: number;
|
|
play_time: number;
|
|
total_time: number;
|
|
save_time: number;
|
|
year: string;
|
|
}
|
|
|
|
export interface ApiSite {
|
|
key: string;
|
|
api: string;
|
|
name: string;
|
|
detail?: string;
|
|
}
|
|
|
|
export interface ServerConfig {
|
|
SiteName: string;
|
|
StorageType: "localstorage" | "redis" | string;
|
|
}
|
|
|
|
export class API {
|
|
public baseURL: string = "";
|
|
|
|
constructor(baseURL?: string) {
|
|
if (baseURL) {
|
|
this.baseURL = baseURL;
|
|
}
|
|
}
|
|
|
|
public setBaseUrl(url: string) {
|
|
this.baseURL = url;
|
|
}
|
|
|
|
private async _fetch(url: string, options: RequestInit = {}): Promise<Response> {
|
|
if (!this.baseURL) {
|
|
throw new Error("API_URL_NOT_SET");
|
|
}
|
|
|
|
const response = await fetch(`${this.baseURL}${url}`, options);
|
|
|
|
if (response.status === 401) {
|
|
throw new Error("UNAUTHORIZED");
|
|
}
|
|
|
|
if (!response.ok) {
|
|
throw new Error(`HTTP error! status: ${response.status}`);
|
|
}
|
|
|
|
return response;
|
|
}
|
|
|
|
async getServerConfig(): Promise<ServerConfig> {
|
|
const response = await this._fetch("/api/server-config");
|
|
return response.json();
|
|
}
|
|
|
|
async login(username?: string | undefined, password?: string): Promise<{ ok: boolean }> {
|
|
const response = await this._fetch("/api/login", {
|
|
method: "POST",
|
|
headers: { "Content-Type": "application/json" },
|
|
body: JSON.stringify({ username, password }),
|
|
});
|
|
return response.json();
|
|
}
|
|
|
|
async getFavorites(key?: string): Promise<Record<string, Favorite> | Favorite | null> {
|
|
const url = key ? `/api/favorites?key=${encodeURIComponent(key)}` : "/api/favorites";
|
|
const response = await this._fetch(url);
|
|
return response.json();
|
|
}
|
|
|
|
async addFavorite(key: string, favorite: Omit<Favorite, "save_time">): Promise<{ success: boolean }> {
|
|
const response = await this._fetch("/api/favorites", {
|
|
method: "POST",
|
|
headers: { "Content-Type": "application/json" },
|
|
body: JSON.stringify({ key, favorite }),
|
|
});
|
|
return response.json();
|
|
}
|
|
|
|
async deleteFavorite(key?: string): Promise<{ success: boolean }> {
|
|
const url = key ? `/api/favorites?key=${encodeURIComponent(key)}` : "/api/favorites";
|
|
const response = await this._fetch(url, { method: "DELETE" });
|
|
return response.json();
|
|
}
|
|
|
|
async getPlayRecords(): Promise<Record<string, PlayRecord>> {
|
|
const response = await this._fetch("/api/playrecords");
|
|
return response.json();
|
|
}
|
|
|
|
async savePlayRecord(key: string, record: Omit<PlayRecord, "save_time">): Promise<{ success: boolean }> {
|
|
const response = await this._fetch("/api/playrecords", {
|
|
method: "POST",
|
|
headers: { "Content-Type": "application/json" },
|
|
body: JSON.stringify({ key, record }),
|
|
});
|
|
return response.json();
|
|
}
|
|
|
|
async deletePlayRecord(key?: string): Promise<{ success: boolean }> {
|
|
const url = key ? `/api/playrecords?key=${encodeURIComponent(key)}` : "/api/playrecords";
|
|
const response = await this._fetch(url, { method: "DELETE" });
|
|
return response.json();
|
|
}
|
|
|
|
async getSearchHistory(): Promise<string[]> {
|
|
const response = await this._fetch("/api/searchhistory");
|
|
return response.json();
|
|
}
|
|
|
|
async addSearchHistory(keyword: string): Promise<string[]> {
|
|
const response = await this._fetch("/api/searchhistory", {
|
|
method: "POST",
|
|
headers: { "Content-Type": "application/json" },
|
|
body: JSON.stringify({ keyword }),
|
|
});
|
|
return response.json();
|
|
}
|
|
|
|
async deleteSearchHistory(keyword?: string): Promise<{ success: boolean }> {
|
|
const url = keyword ? `/api/searchhistory?keyword=${keyword}` : "/api/searchhistory";
|
|
const response = await this._fetch(url, { method: "DELETE" });
|
|
return response.json();
|
|
}
|
|
|
|
getImageProxyUrl(imageUrl: string): string {
|
|
return `${this.baseURL}/api/image-proxy?url=${encodeURIComponent(imageUrl)}`;
|
|
}
|
|
|
|
async getDoubanData(
|
|
type: "movie" | "tv",
|
|
tag: string,
|
|
pageSize: number = 16,
|
|
pageStart: number = 0
|
|
): Promise<DoubanResponse> {
|
|
const url = `/api/douban?type=${type}&tag=${encodeURIComponent(tag)}&pageSize=${pageSize}&pageStart=${pageStart}`;
|
|
const response = await this._fetch(url);
|
|
return response.json();
|
|
}
|
|
|
|
async searchVideos(query: string): Promise<{ results: SearchResult[] }> {
|
|
const url = `/api/search?q=${encodeURIComponent(query)}`;
|
|
const response = await this._fetch(url);
|
|
return response.json();
|
|
}
|
|
|
|
async searchVideo(query: string, resourceId: string, signal?: AbortSignal): Promise<{ results: SearchResult[] }> {
|
|
const url = `/api/search/one?q=${encodeURIComponent(query)}&resourceId=${encodeURIComponent(resourceId)}`;
|
|
const response = await this._fetch(url, { signal });
|
|
return response.json();
|
|
}
|
|
|
|
async getResources(signal?: AbortSignal): Promise<ApiSite[]> {
|
|
const url = `/api/admin/config`;
|
|
const response = await this._fetch(url, { signal });
|
|
const config = await response.json();
|
|
|
|
// 添加安全检查
|
|
if (!config || !config.Config.SourceConfig) {
|
|
console.warn('API response missing SourceConfig:', config);
|
|
return [];
|
|
}
|
|
|
|
// 确保 SourceConfig 是数组
|
|
if (!Array.isArray(config.Config.SourceConfig)) {
|
|
console.warn('SourceConfig is not an array:', config.Config.SourceConfig);
|
|
return [];
|
|
}
|
|
|
|
// 过滤并验证每个站点配置
|
|
return config.Config.SourceConfig
|
|
.filter((site: any) => site && !site.disabled)
|
|
.map((site: any) => ({
|
|
key: site.key || '',
|
|
api: site.api || '',
|
|
name: site.name || '',
|
|
detail: site.detail
|
|
}));
|
|
}
|
|
|
|
async getVideoDetail(source: string, id: string): Promise<VideoDetail> {
|
|
const url = `/api/detail?source=${source}&id=${id}`;
|
|
const response = await this._fetch(url);
|
|
return response.json();
|
|
}
|
|
}
|
|
|
|
// 默认实例
|
|
export let api = new API();
|