2025-10-18 16:52:02 +08:00
|
|
|
|
import {
|
|
|
|
|
|
useQuery,
|
|
|
|
|
|
type UseQueryResult,
|
|
|
|
|
|
keepPreviousData,
|
|
|
|
|
|
} from "@tanstack/react-query";
|
2025-10-30 14:59:15 +08:00
|
|
|
|
import { providersApi, settingsApi, usageApi, type AppId } from "@/lib/api";
|
2025-10-17 19:18:10 +08:00
|
|
|
|
import type { Provider, Settings, UsageResult } from "@/types";
|
2025-10-16 10:00:22 +08:00
|
|
|
|
|
|
|
|
|
|
const sortProviders = (
|
2025-10-16 12:13:51 +08:00
|
|
|
|
providers: Record<string, Provider>,
|
2025-10-16 10:00:22 +08:00
|
|
|
|
): Record<string, Provider> => {
|
|
|
|
|
|
const sortedEntries = Object.values(providers)
|
|
|
|
|
|
.sort((a, b) => {
|
|
|
|
|
|
const indexA = a.sortIndex ?? Number.MAX_SAFE_INTEGER;
|
|
|
|
|
|
const indexB = b.sortIndex ?? Number.MAX_SAFE_INTEGER;
|
|
|
|
|
|
if (indexA !== indexB) {
|
|
|
|
|
|
return indexA - indexB;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const timeA = a.createdAt ?? 0;
|
|
|
|
|
|
const timeB = b.createdAt ?? 0;
|
|
|
|
|
|
if (timeA === timeB) {
|
|
|
|
|
|
return a.name.localeCompare(b.name, "zh-CN");
|
|
|
|
|
|
}
|
|
|
|
|
|
return timeA - timeB;
|
|
|
|
|
|
})
|
|
|
|
|
|
.map((provider) => [provider.id, provider] as const);
|
|
|
|
|
|
|
|
|
|
|
|
return Object.fromEntries(sortedEntries);
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
export interface ProvidersQueryData {
|
|
|
|
|
|
providers: Record<string, Provider>;
|
|
|
|
|
|
currentProviderId: string;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
export const useProvidersQuery = (
|
2025-10-30 14:59:15 +08:00
|
|
|
|
appId: AppId,
|
2025-10-16 10:00:22 +08:00
|
|
|
|
): UseQueryResult<ProvidersQueryData> => {
|
|
|
|
|
|
return useQuery({
|
2025-10-30 14:59:15 +08:00
|
|
|
|
queryKey: ["providers", appId],
|
2025-10-17 21:51:37 +08:00
|
|
|
|
placeholderData: keepPreviousData,
|
2025-10-16 10:00:22 +08:00
|
|
|
|
queryFn: async () => {
|
|
|
|
|
|
let providers: Record<string, Provider> = {};
|
|
|
|
|
|
let currentProviderId = "";
|
|
|
|
|
|
|
|
|
|
|
|
try {
|
2025-10-30 14:59:15 +08:00
|
|
|
|
providers = await providersApi.getAll(appId);
|
2025-10-16 10:00:22 +08:00
|
|
|
|
} catch (error) {
|
|
|
|
|
|
console.error("获取供应商列表失败:", error);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
try {
|
2025-10-30 14:59:15 +08:00
|
|
|
|
currentProviderId = await providersApi.getCurrent(appId);
|
2025-10-16 10:00:22 +08:00
|
|
|
|
} catch (error) {
|
|
|
|
|
|
console.error("获取当前供应商失败:", error);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (Object.keys(providers).length === 0) {
|
|
|
|
|
|
try {
|
2025-10-30 14:59:15 +08:00
|
|
|
|
const success = await providersApi.importDefault(appId);
|
2025-10-16 10:00:22 +08:00
|
|
|
|
if (success) {
|
2025-10-30 14:59:15 +08:00
|
|
|
|
providers = await providersApi.getAll(appId);
|
|
|
|
|
|
currentProviderId = await providersApi.getCurrent(appId);
|
2025-10-16 10:00:22 +08:00
|
|
|
|
}
|
|
|
|
|
|
} catch (error) {
|
|
|
|
|
|
console.error("导入默认配置失败:", error);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return {
|
|
|
|
|
|
providers: sortProviders(providers),
|
|
|
|
|
|
currentProviderId,
|
|
|
|
|
|
};
|
|
|
|
|
|
},
|
|
|
|
|
|
});
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
export const useSettingsQuery = (): UseQueryResult<Settings> => {
|
|
|
|
|
|
return useQuery({
|
|
|
|
|
|
queryKey: ["settings"],
|
|
|
|
|
|
queryFn: async () => settingsApi.get(),
|
|
|
|
|
|
});
|
|
|
|
|
|
};
|
2025-10-17 19:18:10 +08:00
|
|
|
|
|
2025-11-05 21:40:06 +08:00
|
|
|
|
export interface UseUsageQueryOptions {
|
|
|
|
|
|
enabled?: boolean;
|
|
|
|
|
|
autoQueryInterval?: number; // 自动查询间隔(分钟),0 表示禁用
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-10-17 19:18:10 +08:00
|
|
|
|
export const useUsageQuery = (
|
|
|
|
|
|
providerId: string,
|
2025-10-30 14:59:15 +08:00
|
|
|
|
appId: AppId,
|
2025-11-05 21:40:06 +08:00
|
|
|
|
options?: UseUsageQueryOptions,
|
|
|
|
|
|
) => {
|
|
|
|
|
|
const { enabled = true, autoQueryInterval = 0 } = options || {};
|
|
|
|
|
|
|
|
|
|
|
|
const query = useQuery<UsageResult>({
|
2025-10-30 14:59:15 +08:00
|
|
|
|
queryKey: ["usage", providerId, appId],
|
|
|
|
|
|
queryFn: async () => usageApi.query(providerId, appId),
|
2025-10-17 19:18:10 +08:00
|
|
|
|
enabled: enabled && !!providerId,
|
2025-11-05 21:40:06 +08:00
|
|
|
|
refetchInterval:
|
|
|
|
|
|
autoQueryInterval > 0
|
|
|
|
|
|
? Math.max(autoQueryInterval, 1) * 60 * 1000 // 最小1分钟
|
|
|
|
|
|
: false,
|
2025-10-17 19:18:10 +08:00
|
|
|
|
refetchOnWindowFocus: false,
|
2025-11-03 10:24:59 +08:00
|
|
|
|
retry: false,
|
2025-10-17 19:18:10 +08:00
|
|
|
|
staleTime: 5 * 60 * 1000, // 5分钟
|
|
|
|
|
|
});
|
2025-11-05 21:40:06 +08:00
|
|
|
|
|
|
|
|
|
|
return {
|
|
|
|
|
|
...query,
|
|
|
|
|
|
lastQueriedAt: query.dataUpdatedAt || null,
|
|
|
|
|
|
};
|
2025-10-17 19:18:10 +08:00
|
|
|
|
};
|