import { DEFAULT_CONFIG } from "~/env"; export interface ErrorResponse { title: string; message: string; } export class HttpClient { private static instance: HttpClient; private timeout: number; private constructor(timeout = 10000) { this.timeout = timeout; } public static getInstance(timeout?: number): HttpClient { if (!this.instance) { this.instance = new HttpClient(timeout); } return this.instance; } private async setHeaders(options: RequestInit = {}): Promise { const headers = new Headers(options.headers); if (!headers.has("Content-Type")) { headers.set("Content-Type", "application/json"); } const token = localStorage.getItem("token"); if (token) { headers.set("Authorization", `Bearer ${token}`); } return { ...options, headers }; } private async handleResponse(response: Response): Promise { if (!response.ok) { const contentType = response.headers.get("content-type"); let message; try { if (contentType?.includes("application/json")) { const error = await response.json(); message = error.message || ""; } else { const textError = await response.text(); message = textError || ""; } } catch (e) { console.error("解析响应错误:", e); } switch (response.status) { case 404: message = "请求的资源不存在"; break; } const errorResponse: ErrorResponse = { title: `${response.status} ${response.statusText}`, message: message, }; throw errorResponse; } const contentType = response.headers.get("content-type"); return contentType?.includes("application/json") ? response.json() : response.text(); } private async request( endpoint: string, options: RequestInit = {}, url: string, ): Promise { const controller = new AbortController(); const timeoutId = setTimeout(() => controller.abort(), this.timeout); try { const config = await this.setHeaders(options); const baseUrl = url.startsWith("http") ? url : `http://${url}`; const newUrl = new URL(endpoint, baseUrl); const response = await fetch(newUrl, { ...config, signal: controller.signal, credentials: "include", mode: "cors", }); return await this.handleResponse(response); } catch (error: any) { if (error.name === "AbortError") { const errorResponse: ErrorResponse = { title: "请求超时", message: "服务器响应时间过长,请稍后重试", }; throw errorResponse; } if ((error as ErrorResponse).title && (error as ErrorResponse).message) { throw error; } console.log(error); const errorResponse: ErrorResponse = { title: "未知错误", message: error.message || "发生未知错误", }; throw errorResponse; } finally { clearTimeout(timeoutId); } } public async api(endpoint: string, options: RequestInit = {}): Promise { const url = import.meta.env.VITE_API_BASE_URL ?? DEFAULT_CONFIG.VITE_API_BASE_URL; return this.request(endpoint, options, url); } public async dev(endpoint: string, options: RequestInit = {}): Promise { const address = import.meta.env.VITE_ADDRESS ?? DEFAULT_CONFIG.VITE_ADDRESS; const port = Number(import.meta.env.VITE_PORT ?? DEFAULT_CONFIG.VITE_PORT) + 1; return this.request(endpoint, options, `${address}:${port}`); } public async get(endpoint: string, options: RequestInit = {}): Promise { return this.api(endpoint, { ...options, method: "GET" }); } public async post( endpoint: string, data?: any, options: RequestInit = {}, ): Promise { return this.api(endpoint, { ...options, method: "POST", body: JSON.stringify(data), }); } public async put( endpoint: string, data?: any, options: RequestInit = {}, ): Promise { return this.api(endpoint, { ...options, method: "PUT", body: JSON.stringify(data), }); } public async delete( endpoint: string, options: RequestInit = {}, ): Promise { return this.api(endpoint, { ...options, method: "DELETE" }); } public async systemToken(): Promise { const formData = { username: import.meta.env.VITE_API_USERNAME, password: import.meta.env.VITE_API_PASSWORD, }; return this.api("/auth/token/system", { method: "POST", body: JSON.stringify(formData), }); } }