codex-proxy / shared /hooks /use-usage-stats.ts
icebear
feat: usage stats page with time-series token tracking (#147)
df56b50 unverified
raw
history blame
1.88 kB
/**
* Hooks for fetching usage stats data.
*/
import { useState, useEffect, useCallback } from "preact/hooks";
export interface UsageSummary {
total_input_tokens: number;
total_output_tokens: number;
total_request_count: number;
total_accounts: number;
active_accounts: number;
}
export interface UsageDataPoint {
timestamp: string;
input_tokens: number;
output_tokens: number;
request_count: number;
}
export type Granularity = "raw" | "hourly" | "daily";
export function useUsageSummary(refreshIntervalMs = 30_000) {
const [summary, setSummary] = useState<UsageSummary | null>(null);
const [loading, setLoading] = useState(true);
const load = useCallback(async () => {
try {
const resp = await fetch("/admin/usage-stats/summary");
if (resp.ok) setSummary(await resp.json());
} catch { /* ignore */ }
setLoading(false);
}, []);
useEffect(() => {
load();
const id = setInterval(load, refreshIntervalMs);
return () => clearInterval(id);
}, [load, refreshIntervalMs]);
return { summary, loading };
}
export function useUsageHistory(granularity: Granularity, hours: number, refreshIntervalMs = 60_000) {
const [dataPoints, setDataPoints] = useState<UsageDataPoint[]>([]);
const [loading, setLoading] = useState(true);
const load = useCallback(async () => {
try {
const resp = await fetch(
`/admin/usage-stats/history?granularity=${granularity}&hours=${hours}`,
);
if (resp.ok) {
const body = await resp.json();
setDataPoints(body.data_points);
}
} catch { /* ignore */ }
setLoading(false);
}, [granularity, hours]);
useEffect(() => {
setLoading(true);
load();
const id = setInterval(load, refreshIntervalMs);
return () => clearInterval(id);
}, [load, refreshIntervalMs]);
return { dataPoints, loading };
}