import axios, { AxiosInstance } from "axios"; import { getClashInfo } from "./cmds"; import { invoke } from "@tauri-apps/api/core"; import { useLockFn } from "ahooks"; let instancePromise: Promise = null!; async function getInstancePromise() { let server = ""; let secret = ""; try { const info = await getClashInfo(); if (info?.server) { server = info.server; // compatible width `external-controller` if (server.startsWith(":")) server = `127.0.0.1${server}`; else if (/^\d+$/.test(server)) server = `127.0.0.1:${server}`; } if (info?.secret) secret = info?.secret; } catch {} const axiosIns = axios.create({ baseURL: `http://${server}`, headers: secret ? { Authorization: `Bearer ${secret}` } : {}, timeout: 15000, }); axiosIns.interceptors.response.use((r) => r.data); return axiosIns; } /// initialize some information /// enable force update axiosIns export const getAxios = async (force: boolean = false) => { if (!instancePromise || force) { instancePromise = getInstancePromise(); } return instancePromise; }; /// Get Version export const getVersion = async () => { const instance = await getAxios(); return instance.get("/version") as Promise<{ premium: boolean; meta?: boolean; version: string; }>; }; /// Get current base configs export const getClashConfig = async () => { const instance = await getAxios(); return instance.get("/configs") as Promise; }; /// Update geo data export const updateGeoData = async () => { const instance = await getAxios(); return instance.post("/configs/geo"); }; /// Upgrade clash core export const upgradeCore = async () => { const instance = await getAxios(); return instance.post("/upgrade"); }; /// Get current rules export const getRules = async () => { const instance = await getAxios(); const response = await instance.get("/rules"); return response?.rules as IRuleItem[]; }; /// Get Proxy delay export const getProxyDelay = async ( name: string, url?: string, timeout?: number, ) => { const params = { timeout: timeout || 10000, url: url || "http://cp.cloudflare.com/generate_204", }; const instance = await getAxios(); const result = await instance.get( `/proxies/${encodeURIComponent(name)}/delay`, { params }, ); return result as any as { delay: number }; }; /// Update the Proxy Choose export const updateProxy = async (group: string, proxy: string) => { const instance = await getAxios(); return instance.put(`/proxies/${encodeURIComponent(group)}`, { name: proxy }); }; // get proxy export const getProxiesInner = async () => { const response = await invoke<{ proxies: Record }>( "get_proxies", ); return response.proxies as Record; }; /// Get the Proxy information export const getProxies = async (): Promise<{ global: IProxyGroupItem; direct: IProxyItem; groups: IProxyGroupItem[]; records: Record; proxies: IProxyItem[]; }> => { const [proxyRecord, providerRecord] = await Promise.all([ getProxiesInner(), getProxyProviders(), ]); // provider name map const providerMap = Object.fromEntries( Object.entries(providerRecord).flatMap(([provider, item]) => item.proxies.map((p) => [p.name, { ...p, provider }]), ), ); // compatible with proxy-providers const generateItem = (name: string) => { if (proxyRecord[name]) return proxyRecord[name]; if (providerMap[name]) return providerMap[name]; return { name, type: "unknown", udp: false, xudp: false, tfo: false, mptcp: false, smux: false, history: [], }; }; const { GLOBAL: global, DIRECT: direct, REJECT: reject } = proxyRecord; let groups: IProxyGroupItem[] = Object.values(proxyRecord).reduce< IProxyGroupItem[] >((acc, each) => { if (each.name !== "GLOBAL" && each.all) { acc.push({ ...each, all: each.all!.map((item) => generateItem(item)), }); } return acc; }, []); if (global?.all) { let globalGroups: IProxyGroupItem[] = global.all.reduce( (acc, name) => { if (proxyRecord[name]?.all) { acc.push({ ...proxyRecord[name], all: proxyRecord[name].all!.map((item) => generateItem(item)), }); } return acc; }, [], ); let globalNames = new Set(globalGroups.map((each) => each.name)); groups = groups .filter((group) => { return !globalNames.has(group.name); }) .concat(globalGroups); } const proxies = [direct, reject].concat( Object.values(proxyRecord).filter( (p) => !p.all?.length && p.name !== "DIRECT" && p.name !== "REJECT", ), ); const _global: IProxyGroupItem = { ...global, all: global?.all?.map((item) => generateItem(item)) || [], }; return { global: _global, direct, groups, records: proxyRecord, proxies }; }; // get proxy providers export const getProxyProviders = async () => { const response = await invoke<{ providers: Record; }>("get_providers_proxies"); const providers = response.providers as Record; return Object.fromEntries( Object.entries(providers).filter(([key, item]) => { const type = item.vehicleType.toLowerCase(); return type === "http" || type === "file"; }), ); }; export const getRuleProviders = async () => { const instance = await getAxios(); const response = await instance.get("/providers/rules"); const providers = (response.providers || {}) as Record< string, IRuleProviderItem >; return Object.fromEntries( Object.entries(providers).filter(([key, item]) => { const type = item.vehicleType.toLowerCase(); return type === "http" || type === "file"; }), ); }; // proxy providers health check export const providerHealthCheck = async (name: string) => { const instance = await getAxios(); return instance.get( `/providers/proxies/${encodeURIComponent(name)}/healthcheck`, ); }; export const proxyProviderUpdate = async (name: string) => { const instance = await getAxios(); return instance.put(`/providers/proxies/${encodeURIComponent(name)}`); }; export const ruleProviderUpdate = async (name: string) => { const instance = await getAxios(); return instance.put(`/providers/rules/${encodeURIComponent(name)}`); }; export const getConnections = async () => { const instance = await getAxios(); const result = await instance.get("/connections"); return result as any as IConnections; }; // Close specific connection export const deleteConnection = async (id: string) => { const instance = await getAxios(); await instance.delete(`/connections/${encodeURIComponent(id)}`); }; // Close all connections export const closeAllConnections = async () => { const instance = await getAxios(); await instance.delete("/connections"); }; // Get Group Proxy Delays export const getGroupProxyDelays = async ( groupName: string, url?: string, timeout?: number, ) => { const params = { timeout: timeout || 10000, url: url || "http://cp.cloudflare.com/generate_204", }; console.log( `[API] 获取代理组延迟,组: ${groupName}, URL: ${params.url}, 超时: ${params.timeout}ms`, ); try { const instance = await getAxios(); console.log( `[API] 发送HTTP请求: GET /group/${encodeURIComponent(groupName)}/delay`, ); const result = await instance.get( `/group/${encodeURIComponent(groupName)}/delay`, { params }, ); console.log( `[API] 获取代理组延迟成功,组: ${groupName}, 结果数量:`, Object.keys(result || {}).length, ); return result as any as Record; } catch (error) { console.error(`[API] 获取代理组延迟失败,组: ${groupName}`, error); throw error; } }; // Is debug enabled export const isDebugEnabled = async () => { try { const instance = await getAxios(); await instance.get("/debug/pprof"); return true; } catch { return false; } }; // GC export const gc = async () => { try { const instance = await getAxios(); await instance.put("/debug/gc"); } catch (error) { console.error(`Error gcing: ${error}`); } }; // Get current IP and geolocation information export const getIpInfo = async () => { // 添加重试机制 const maxRetries = 3; const retryDelay = 1500; const timeout = 5000; let lastError; for (let attempt = 0; attempt < maxRetries; attempt++) { try { // 使用axios直接请求IP.sb的API,不通过clash代理 const response = await axios.get("https://api.ip.sb/geoip", { timeout }); return response.data as { ip: string; country_code: string; country: string; region: string; city: string; organization: string; asn: number; asn_organization: string; longitude: number; latitude: number; timezone: string; }; } catch (error) { console.log(`获取IP信息失败,尝试 ${attempt + 1}/${maxRetries}`, error); lastError = error; // 如果不是最后一次尝试,则等待后重试 if (attempt < maxRetries - 1) { await new Promise(resolve => setTimeout(resolve, retryDelay)); } } } throw lastError; };