aigetwey 1.4.2 → 1.4.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +20 -0
- package/README.md +8 -1
- package/dashboard/.next/BUILD_ID +1 -1
- package/dashboard/.next/build-manifest.json +3 -3
- package/dashboard/.next/diagnostics/route-bundle-stats.json +26 -26
- package/dashboard/.next/fallback-build-manifest.json +3 -3
- package/dashboard/.next/server/app/(console)/combos/page_client-reference-manifest.js +1 -1
- package/dashboard/.next/server/app/(console)/config/page_client-reference-manifest.js +1 -1
- package/dashboard/.next/server/app/(console)/console/page_client-reference-manifest.js +1 -1
- package/dashboard/.next/server/app/(console)/endpoint/page_client-reference-manifest.js +1 -1
- package/dashboard/.next/server/app/(console)/keys/page_client-reference-manifest.js +1 -1
- package/dashboard/.next/server/app/(console)/page_client-reference-manifest.js +1 -1
- package/dashboard/.next/server/app/(console)/providers/[id]/page_client-reference-manifest.js +1 -1
- package/dashboard/.next/server/app/(console)/providers/page_client-reference-manifest.js +1 -1
- package/dashboard/.next/server/app/(console)/quota/page_client-reference-manifest.js +1 -1
- package/dashboard/.next/server/app/(console)/tools/[id]/page_client-reference-manifest.js +1 -1
- package/dashboard/.next/server/app/(console)/tools/page_client-reference-manifest.js +1 -1
- package/dashboard/.next/server/app/(console)/usage/page_client-reference-manifest.js +1 -1
- package/dashboard/.next/server/app/_global-error.html +1 -1
- package/dashboard/.next/server/app/_global-error.rsc +1 -1
- package/dashboard/.next/server/app/_global-error.segments/__PAGE__.segment.rsc +1 -1
- package/dashboard/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
- package/dashboard/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
- package/dashboard/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
- package/dashboard/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
- package/dashboard/.next/server/app/_not-found/page_client-reference-manifest.js +1 -1
- package/dashboard/.next/server/app/_not-found.html +1 -1
- package/dashboard/.next/server/app/_not-found.rsc +2 -2
- package/dashboard/.next/server/app/_not-found.segments/_full.segment.rsc +2 -2
- package/dashboard/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
- package/dashboard/.next/server/app/_not-found.segments/_index.segment.rsc +2 -2
- package/dashboard/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
- package/dashboard/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
- package/dashboard/.next/server/app/_not-found.segments/_tree.segment.rsc +2 -2
- package/dashboard/.next/server/app/combos.html +1 -1
- package/dashboard/.next/server/app/combos.rsc +5 -5
- package/dashboard/.next/server/app/combos.segments/!KGNvbnNvbGUp/combos/__PAGE__.segment.rsc +2 -2
- package/dashboard/.next/server/app/combos.segments/!KGNvbnNvbGUp/combos.segment.rsc +1 -1
- package/dashboard/.next/server/app/combos.segments/!KGNvbnNvbGUp.segment.rsc +3 -3
- package/dashboard/.next/server/app/combos.segments/_full.segment.rsc +5 -5
- package/dashboard/.next/server/app/combos.segments/_head.segment.rsc +1 -1
- package/dashboard/.next/server/app/combos.segments/_index.segment.rsc +2 -2
- package/dashboard/.next/server/app/combos.segments/_tree.segment.rsc +2 -2
- package/dashboard/.next/server/app/config.html +1 -1
- package/dashboard/.next/server/app/config.rsc +5 -5
- package/dashboard/.next/server/app/config.segments/!KGNvbnNvbGUp/config/__PAGE__.segment.rsc +2 -2
- package/dashboard/.next/server/app/config.segments/!KGNvbnNvbGUp/config.segment.rsc +1 -1
- package/dashboard/.next/server/app/config.segments/!KGNvbnNvbGUp.segment.rsc +3 -3
- package/dashboard/.next/server/app/config.segments/_full.segment.rsc +5 -5
- package/dashboard/.next/server/app/config.segments/_head.segment.rsc +1 -1
- package/dashboard/.next/server/app/config.segments/_index.segment.rsc +2 -2
- package/dashboard/.next/server/app/config.segments/_tree.segment.rsc +2 -2
- package/dashboard/.next/server/app/console.html +1 -1
- package/dashboard/.next/server/app/console.rsc +5 -5
- package/dashboard/.next/server/app/console.segments/!KGNvbnNvbGUp/console/__PAGE__.segment.rsc +2 -2
- package/dashboard/.next/server/app/console.segments/!KGNvbnNvbGUp/console.segment.rsc +1 -1
- package/dashboard/.next/server/app/console.segments/!KGNvbnNvbGUp.segment.rsc +3 -3
- package/dashboard/.next/server/app/console.segments/_full.segment.rsc +5 -5
- package/dashboard/.next/server/app/console.segments/_head.segment.rsc +1 -1
- package/dashboard/.next/server/app/console.segments/_index.segment.rsc +2 -2
- package/dashboard/.next/server/app/console.segments/_tree.segment.rsc +2 -2
- package/dashboard/.next/server/app/endpoint.html +1 -1
- package/dashboard/.next/server/app/endpoint.rsc +5 -5
- package/dashboard/.next/server/app/endpoint.segments/!KGNvbnNvbGUp/endpoint/__PAGE__.segment.rsc +2 -2
- package/dashboard/.next/server/app/endpoint.segments/!KGNvbnNvbGUp/endpoint.segment.rsc +1 -1
- package/dashboard/.next/server/app/endpoint.segments/!KGNvbnNvbGUp.segment.rsc +3 -3
- package/dashboard/.next/server/app/endpoint.segments/_full.segment.rsc +5 -5
- package/dashboard/.next/server/app/endpoint.segments/_head.segment.rsc +1 -1
- package/dashboard/.next/server/app/endpoint.segments/_index.segment.rsc +2 -2
- package/dashboard/.next/server/app/endpoint.segments/_tree.segment.rsc +2 -2
- package/dashboard/.next/server/app/index.html +1 -1
- package/dashboard/.next/server/app/index.rsc +5 -5
- package/dashboard/.next/server/app/index.segments/!KGNvbnNvbGUp/__PAGE__.segment.rsc +2 -2
- package/dashboard/.next/server/app/index.segments/!KGNvbnNvbGUp.segment.rsc +3 -3
- package/dashboard/.next/server/app/index.segments/_full.segment.rsc +5 -5
- package/dashboard/.next/server/app/index.segments/_head.segment.rsc +1 -1
- package/dashboard/.next/server/app/index.segments/_index.segment.rsc +2 -2
- package/dashboard/.next/server/app/index.segments/_tree.segment.rsc +2 -2
- package/dashboard/.next/server/app/keys.html +1 -1
- package/dashboard/.next/server/app/keys.rsc +5 -5
- package/dashboard/.next/server/app/keys.segments/!KGNvbnNvbGUp/keys/__PAGE__.segment.rsc +2 -2
- package/dashboard/.next/server/app/keys.segments/!KGNvbnNvbGUp/keys.segment.rsc +1 -1
- package/dashboard/.next/server/app/keys.segments/!KGNvbnNvbGUp.segment.rsc +3 -3
- package/dashboard/.next/server/app/keys.segments/_full.segment.rsc +5 -5
- package/dashboard/.next/server/app/keys.segments/_head.segment.rsc +1 -1
- package/dashboard/.next/server/app/keys.segments/_index.segment.rsc +2 -2
- package/dashboard/.next/server/app/keys.segments/_tree.segment.rsc +2 -2
- package/dashboard/.next/server/app/login/page_client-reference-manifest.js +1 -1
- package/dashboard/.next/server/app/login.html +1 -1
- package/dashboard/.next/server/app/login.rsc +2 -2
- package/dashboard/.next/server/app/login.segments/_full.segment.rsc +2 -2
- package/dashboard/.next/server/app/login.segments/_head.segment.rsc +1 -1
- package/dashboard/.next/server/app/login.segments/_index.segment.rsc +2 -2
- package/dashboard/.next/server/app/login.segments/_tree.segment.rsc +2 -2
- package/dashboard/.next/server/app/login.segments/login/__PAGE__.segment.rsc +1 -1
- package/dashboard/.next/server/app/login.segments/login.segment.rsc +1 -1
- package/dashboard/.next/server/app/providers.html +1 -1
- package/dashboard/.next/server/app/providers.rsc +5 -5
- package/dashboard/.next/server/app/providers.segments/!KGNvbnNvbGUp/providers/__PAGE__.segment.rsc +2 -2
- package/dashboard/.next/server/app/providers.segments/!KGNvbnNvbGUp/providers.segment.rsc +1 -1
- package/dashboard/.next/server/app/providers.segments/!KGNvbnNvbGUp.segment.rsc +3 -3
- package/dashboard/.next/server/app/providers.segments/_full.segment.rsc +5 -5
- package/dashboard/.next/server/app/providers.segments/_head.segment.rsc +1 -1
- package/dashboard/.next/server/app/providers.segments/_index.segment.rsc +2 -2
- package/dashboard/.next/server/app/providers.segments/_tree.segment.rsc +2 -2
- package/dashboard/.next/server/app/quota.html +1 -1
- package/dashboard/.next/server/app/quota.rsc +5 -5
- package/dashboard/.next/server/app/quota.segments/!KGNvbnNvbGUp/quota/__PAGE__.segment.rsc +2 -2
- package/dashboard/.next/server/app/quota.segments/!KGNvbnNvbGUp/quota.segment.rsc +1 -1
- package/dashboard/.next/server/app/quota.segments/!KGNvbnNvbGUp.segment.rsc +3 -3
- package/dashboard/.next/server/app/quota.segments/_full.segment.rsc +5 -5
- package/dashboard/.next/server/app/quota.segments/_head.segment.rsc +1 -1
- package/dashboard/.next/server/app/quota.segments/_index.segment.rsc +2 -2
- package/dashboard/.next/server/app/quota.segments/_tree.segment.rsc +2 -2
- package/dashboard/.next/server/app/tools.html +1 -1
- package/dashboard/.next/server/app/tools.rsc +5 -5
- package/dashboard/.next/server/app/tools.segments/!KGNvbnNvbGUp/tools/__PAGE__.segment.rsc +2 -2
- package/dashboard/.next/server/app/tools.segments/!KGNvbnNvbGUp/tools.segment.rsc +1 -1
- package/dashboard/.next/server/app/tools.segments/!KGNvbnNvbGUp.segment.rsc +3 -3
- package/dashboard/.next/server/app/tools.segments/_full.segment.rsc +5 -5
- package/dashboard/.next/server/app/tools.segments/_head.segment.rsc +1 -1
- package/dashboard/.next/server/app/tools.segments/_index.segment.rsc +2 -2
- package/dashboard/.next/server/app/tools.segments/_tree.segment.rsc +2 -2
- package/dashboard/.next/server/chunks/ssr/[root-of-the-server]__0zf4yft._.js +1 -1
- package/dashboard/.next/server/chunks/ssr/[root-of-the-server]__0zf4yft._.js.map +1 -1
- package/dashboard/.next/server/chunks/ssr/_0p90gyy._.js +1 -1
- package/dashboard/.next/server/chunks/ssr/_0p90gyy._.js.map +1 -1
- package/dashboard/.next/server/chunks/ssr/src_components_10h9ide._.js +1 -1
- package/dashboard/.next/server/chunks/ssr/src_components_10h9ide._.js.map +1 -1
- package/dashboard/.next/server/middleware-build-manifest.js +3 -3
- package/dashboard/.next/server/pages/404.html +1 -1
- package/dashboard/.next/server/pages/500.html +1 -1
- package/dashboard/.next/static/chunks/{40pm83c1_d0jz.js → 0lcp6ibeqkbpk.js} +19 -19
- package/dashboard/.next/static/chunks/{0-qqj0_fg6ef0.js → 0wb0jg5y8b49o.js} +1 -1
- package/dashboard/.next/static/chunks/2ltht8xwlssac.js +1 -0
- package/dashboard/.next/static/chunks/2mpe3tpzx0wtj.css +3 -0
- package/dashboard/.next/trace +1 -1
- package/dashboard/.next/trace-build +1 -1
- package/dashboard/src/components/AreaChart.tsx +88 -19
- package/dashboard/src/components/AutostartToggle.tsx +63 -0
- package/dashboard/src/components/ConfigEditor.tsx +3 -0
- package/dashboard/src/components/TopBar.tsx +1 -1
- package/dist/cli/tray/autostart.js +3 -3
- package/dist/cli/tray/autostart.js.map +1 -1
- package/dist/cli.js +5 -3
- package/dist/cli.js.map +1 -1
- package/dist/core/handler.js +23 -3
- package/dist/core/handler.js.map +1 -1
- package/dist/routes/admin.js +28 -0
- package/dist/routes/admin.js.map +1 -1
- package/dist/routes/v1.js +4 -2
- package/dist/routes/v1.js.map +1 -1
- package/dist/server.js +32 -5
- package/dist/server.js.map +1 -1
- package/package.json +1 -1
- package/src/cli/tray/autostart.ts +3 -3
- package/src/cli.ts +5 -3
- package/src/core/handler.ts +22 -3
- package/src/routes/admin.ts +27 -0
- package/src/routes/v1.ts +4 -2
- package/src/server.ts +33 -5
- package/dashboard/.next/static/chunks/0347y27liw05t.js +0 -1
- package/dashboard/.next/static/chunks/0iv69e64f1ywa.css +0 -3
- /package/dashboard/.next/static/{jKcczqYA93-Kuqa9A9H9h → 3ctBU8K8tClTVQCiGy65u}/_buildManifest.js +0 -0
- /package/dashboard/.next/static/{jKcczqYA93-Kuqa9A9H9h → 3ctBU8K8tClTVQCiGy65u}/_clientMiddlewareManifest.js +0 -0
- /package/dashboard/.next/static/{jKcczqYA93-Kuqa9A9H9h → 3ctBU8K8tClTVQCiGy65u}/_ssgManifest.js +0 -0
|
@@ -78,7 +78,7 @@ function enableMac(node: string, script: string): boolean {
|
|
|
78
78
|
<plist version="1.0"><dict>
|
|
79
79
|
<key>Label</key><string>${APP_LABEL}</string>
|
|
80
80
|
<key>ProgramArguments</key><array>
|
|
81
|
-
<string>${node}</string><string>${script}</string><string>--tray</string>
|
|
81
|
+
<string>${node}</string><string>${script}</string><string>--tray</string><string>--skip-update</string>
|
|
82
82
|
</array>
|
|
83
83
|
<key>EnvironmentVariables</key><dict><key>PATH</key><string>${path}</string></dict>
|
|
84
84
|
<key>RunAtLoad</key><true/>
|
|
@@ -100,7 +100,7 @@ function disableMac(): boolean {
|
|
|
100
100
|
function enableWin(node: string, script: string): boolean {
|
|
101
101
|
const dir = join(process.env.APPDATA || "", "Microsoft", "Windows", "Start Menu", "Programs", "Startup");
|
|
102
102
|
if (!existsSync(dir)) return false;
|
|
103
|
-
const vbs = `Set WshShell = CreateObject("WScript.Shell")\nWshShell.Run """${node}"" ""${script}"" --tray", 0, False\n`;
|
|
103
|
+
const vbs = `Set WshShell = CreateObject("WScript.Shell")\nWshShell.Run """${node}"" ""${script}"" --tray --skip-update", 0, False\n`;
|
|
104
104
|
writeFileSync(join(dir, `${APP_NAME}.vbs`), vbs);
|
|
105
105
|
return true;
|
|
106
106
|
}
|
|
@@ -118,7 +118,7 @@ function enableLinux(node: string, script: string): boolean {
|
|
|
118
118
|
Type=Application
|
|
119
119
|
Name=aigetwey
|
|
120
120
|
Comment=Personal AI gateway
|
|
121
|
-
Exec=${node} ${script} --tray
|
|
121
|
+
Exec=${node} ${script} --tray --skip-update
|
|
122
122
|
Hidden=false
|
|
123
123
|
NoDisplay=false
|
|
124
124
|
X-GNOME-Autostart-enabled=true
|
package/src/cli.ts
CHANGED
|
@@ -34,15 +34,17 @@ interface CliOpts {
|
|
|
34
34
|
help: boolean;
|
|
35
35
|
version: boolean;
|
|
36
36
|
tray: boolean;
|
|
37
|
+
skipUpdate: boolean;
|
|
37
38
|
}
|
|
38
39
|
function parseArgs(argv: string[]): CliOpts {
|
|
39
|
-
const o: CliOpts = { noBrowser: false, yes: false, help: false, version: false, tray: false };
|
|
40
|
+
const o: CliOpts = { noBrowser: false, yes: false, help: false, version: false, tray: false, skipUpdate: false };
|
|
40
41
|
for (let i = 0; i < argv.length; i++) {
|
|
41
42
|
const a = argv[i];
|
|
42
43
|
if (a === "-p" || a === "--port") o.port = Number(argv[++i]);
|
|
43
44
|
else if (a === "-n" || a === "--no-browser") o.noBrowser = true;
|
|
44
45
|
else if (a === "-y" || a === "--yes") o.yes = true;
|
|
45
46
|
else if (a === "-t" || a === "--tray") o.tray = true;
|
|
47
|
+
else if (a === "--skip-update") o.skipUpdate = true;
|
|
46
48
|
else if (a === "-v" || a === "--version") o.version = true;
|
|
47
49
|
else if (a === "-h" || a === "--help") o.help = true;
|
|
48
50
|
}
|
|
@@ -67,7 +69,7 @@ const HELP = `
|
|
|
67
69
|
`;
|
|
68
70
|
|
|
69
71
|
const GATEWAY_PORT = opts.port ?? Number(process.env.AIGETWEY_PORT ?? 18080);
|
|
70
|
-
const DASHBOARD_PORT = Number(process.env.DASHBOARD_PORT ??
|
|
72
|
+
const DASHBOARD_PORT = Number(process.env.DASHBOARD_PORT ?? 18081);
|
|
71
73
|
|
|
72
74
|
const adminPassword = process.env.AIGETWEY_ADMIN_PASSWORD ?? "123456";
|
|
73
75
|
const generatedPw = !process.env.AIGETWEY_ADMIN_PASSWORD;
|
|
@@ -188,7 +190,7 @@ async function ensurePortFree(port: number, envVar: string): Promise<void> {
|
|
|
188
190
|
// ps failed — fall through to the unknown-owner branch
|
|
189
191
|
}
|
|
190
192
|
|
|
191
|
-
if (!/
|
|
193
|
+
if (!/aigetwey/.test(cmd)) {
|
|
192
194
|
console.error(
|
|
193
195
|
` port ${port} is in use by another process (pid ${pid}). free it or set ${envVar}.`,
|
|
194
196
|
);
|
package/src/core/handler.ts
CHANGED
|
@@ -18,6 +18,25 @@ import { adapterFor } from "../adapters/index.js";
|
|
|
18
18
|
import type { UpstreamError } from "../upstream/client.js";
|
|
19
19
|
import { parseSSE, encodeSSE } from "../stream/sse.js";
|
|
20
20
|
import { streamAdapterFor } from "../stream/index.js";
|
|
21
|
+
|
|
22
|
+
function humanDuration(ms: number): string {
|
|
23
|
+
if (ms <= 0) return "now";
|
|
24
|
+
const s = Math.ceil(ms / 1000);
|
|
25
|
+
if (s < 60) return `${s}s`;
|
|
26
|
+
const m = Math.ceil(s / 60);
|
|
27
|
+
if (m < 60) return `${m}m`;
|
|
28
|
+
const h = Math.floor(m / 60);
|
|
29
|
+
const rm = m % 60;
|
|
30
|
+
return rm ? `${h}h ${rm}m` : `${h}h`;
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
function budgetErrorBody(resetMs: number): { error: string; reset_in_ms: number; reset_at: string } {
|
|
34
|
+
const resetAt = new Date(Date.now() + resetMs).toISOString();
|
|
35
|
+
const msg = resetMs > 0
|
|
36
|
+
? `budget exceeded — resets in ${humanDuration(resetMs)}`
|
|
37
|
+
: "budget exceeded";
|
|
38
|
+
return { error: msg, reset_in_ms: resetMs, reset_at: resetAt };
|
|
39
|
+
}
|
|
21
40
|
import type { CanonicalChunk } from "../stream/chunk.js";
|
|
22
41
|
import type { KeyPool } from "./keypool.js";
|
|
23
42
|
import { executeWithFallback } from "./fallback.js";
|
|
@@ -139,15 +158,15 @@ export async function handle(
|
|
|
139
158
|
// matching routes; if every candidate is barred, there's nothing to serve → 402.
|
|
140
159
|
if (deps.budget) {
|
|
141
160
|
const g = deps.budget.globalStatus();
|
|
142
|
-
if (g?.exhausted) throw new GatewayError(402,
|
|
161
|
+
if (g?.exhausted) throw new GatewayError(402, budgetErrorBody(g.reset_in_ms));
|
|
143
162
|
if (deps.clientKeyFp) {
|
|
144
163
|
const kb = deps.budget.blocksKey(deps.clientKeyFp);
|
|
145
|
-
if (kb?.exhausted) throw new GatewayError(402,
|
|
164
|
+
if (kb?.exhausted) throw new GatewayError(402, budgetErrorBody(kb.reset_in_ms));
|
|
146
165
|
}
|
|
147
166
|
const eligible = routes.filter((r) => !deps.budget!.blocks(r.provider.id, r.model));
|
|
148
167
|
if (eligible.length === 0) {
|
|
149
168
|
const b = deps.budget.blocks(routes[0]!.provider.id, routes[0]!.model);
|
|
150
|
-
throw new GatewayError(402,
|
|
169
|
+
throw new GatewayError(402, budgetErrorBody(b?.reset_in_ms ?? 0));
|
|
151
170
|
}
|
|
152
171
|
routes = eligible;
|
|
153
172
|
}
|
package/src/routes/admin.ts
CHANGED
|
@@ -757,6 +757,33 @@ export function registerAdminRoutes(app: FastifyInstance, deps: AdminDeps): void
|
|
|
757
757
|
reply.send({ current, latest, updateAvailable: !!(latest && isNewerVersion(latest, current)) });
|
|
758
758
|
});
|
|
759
759
|
|
|
760
|
+
// ---- autostart: query/toggle run-on-OS-startup ----
|
|
761
|
+
app.get("/admin/autostart", requireAdmin, (_req, reply) => {
|
|
762
|
+
try {
|
|
763
|
+
const { isAutoStartEnabled } = require("../cli/tray/autostart.js") as typeof import("../cli/tray/autostart.js");
|
|
764
|
+
reply.send({ enabled: isAutoStartEnabled() });
|
|
765
|
+
} catch {
|
|
766
|
+
reply.send({ enabled: false });
|
|
767
|
+
}
|
|
768
|
+
});
|
|
769
|
+
|
|
770
|
+
app.post("/admin/autostart", requireAdmin, (req, reply) => {
|
|
771
|
+
const { enabled } = (req.body ?? {}) as { enabled?: boolean };
|
|
772
|
+
try {
|
|
773
|
+
if (enabled) {
|
|
774
|
+
const { enableAutoStart } = require("../cli/tray/autostart.js") as typeof import("../cli/tray/autostart.js");
|
|
775
|
+
const ok = enableAutoStart();
|
|
776
|
+
reply.send({ enabled: ok });
|
|
777
|
+
} else {
|
|
778
|
+
const { disableAutoStart } = require("../cli/tray/autostart.js") as typeof import("../cli/tray/autostart.js");
|
|
779
|
+
disableAutoStart();
|
|
780
|
+
reply.send({ enabled: false });
|
|
781
|
+
}
|
|
782
|
+
} catch (e) {
|
|
783
|
+
reply.code(500).send({ error: String(e) });
|
|
784
|
+
}
|
|
785
|
+
});
|
|
786
|
+
|
|
760
787
|
// ---- shutdown: stop the gateway process (dashboard power button) ----
|
|
761
788
|
// Matches aigetwey's POST /api/shutdown: reply first, then exit after a short
|
|
762
789
|
// delay so the response reaches the browser. Admin-gated like everything else;
|
package/src/routes/v1.ts
CHANGED
|
@@ -25,13 +25,15 @@ export function registerV1Routes(app: FastifyInstance, state: GatewayState, db?:
|
|
|
25
25
|
|
|
26
26
|
const presented = extractKey(req);
|
|
27
27
|
if (presented && isKeyExpired(state.config.server, presented, Date.now())) {
|
|
28
|
-
|
|
28
|
+
const expAt = state.config.server.key_expires?.[presented];
|
|
29
|
+
const expDate = expAt ? new Date(expAt).toISOString() : undefined;
|
|
30
|
+
reply.code(403).send({ error: "access key expired", expired_at: expDate });
|
|
29
31
|
return; // short-circuit
|
|
30
32
|
}
|
|
31
33
|
|
|
32
34
|
const rpm = presented ? state.config.server.key_rpm?.[presented] : undefined;
|
|
33
35
|
if (presented && rpm && limiter.over(clientKeyFingerprint(presented), rpm)) {
|
|
34
|
-
reply.code(429).send({ error:
|
|
36
|
+
reply.code(429).send({ error: `rate limit exceeded — max ${rpm} req/min`, retry_after_ms: 60000 });
|
|
35
37
|
return; // short-circuit
|
|
36
38
|
}
|
|
37
39
|
|
package/src/server.ts
CHANGED
|
@@ -44,10 +44,33 @@ async function main(): Promise<void> {
|
|
|
44
44
|
|
|
45
45
|
const app = Fastify({
|
|
46
46
|
logger: { level: process.env.LOG_LEVEL ?? "info", stream: logStream },
|
|
47
|
-
// gateway proxies large prompts; raise the JSON body cap.
|
|
48
47
|
bodyLimit: 32 * 1024 * 1024,
|
|
49
48
|
});
|
|
50
49
|
|
|
50
|
+
// Security headers — safe defaults for all responses.
|
|
51
|
+
app.addHook("onSend", (_req, reply, payload, done) => {
|
|
52
|
+
reply.header("X-Content-Type-Options", "nosniff");
|
|
53
|
+
reply.header("X-Frame-Options", "DENY");
|
|
54
|
+
reply.header("X-XSS-Protection", "0");
|
|
55
|
+
reply.header("Referrer-Policy", "no-referrer");
|
|
56
|
+
if (!reply.hasHeader("Cache-Control")) {
|
|
57
|
+
reply.header("Cache-Control", "no-store");
|
|
58
|
+
}
|
|
59
|
+
done(null, payload);
|
|
60
|
+
});
|
|
61
|
+
|
|
62
|
+
// CORS — allow any origin for /v1 (LLM clients), block dashboard from foreign origins.
|
|
63
|
+
app.addHook("onRequest", (req, reply, done) => {
|
|
64
|
+
const origin = req.headers.origin;
|
|
65
|
+
if (req.url?.startsWith("/v1")) {
|
|
66
|
+
reply.header("Access-Control-Allow-Origin", origin ?? "*");
|
|
67
|
+
reply.header("Access-Control-Allow-Headers", "Authorization, Content-Type, x-api-key, anthropic-version");
|
|
68
|
+
reply.header("Access-Control-Allow-Methods", "GET, POST, OPTIONS");
|
|
69
|
+
if (req.method === "OPTIONS") { reply.code(204).send(); return; }
|
|
70
|
+
}
|
|
71
|
+
done();
|
|
72
|
+
});
|
|
73
|
+
|
|
51
74
|
const dataDir = getDataDir();
|
|
52
75
|
const db = new UsageDB(join(dataDir, "usage.sqlite"));
|
|
53
76
|
|
|
@@ -91,11 +114,16 @@ async function main(): Promise<void> {
|
|
|
91
114
|
process.on("SIGTERM", close);
|
|
92
115
|
|
|
93
116
|
try {
|
|
94
|
-
// AIGETWEY_PORT (set by the CLI launcher) overrides the config port so the
|
|
95
|
-
// launcher can pin the gateway port without editing config.yaml.
|
|
96
117
|
const port = process.env.AIGETWEY_PORT ? Number(process.env.AIGETWEY_PORT) : config.server.port;
|
|
97
|
-
|
|
98
|
-
|
|
118
|
+
const host = config.server.host;
|
|
119
|
+
|
|
120
|
+
if (host !== "127.0.0.1" && host !== "localhost" && config.server.api_keys.length === 0) {
|
|
121
|
+
app.log.warn("⚠ SECURITY: binding on %s with NO api_keys — all requests unauthenticated!", host);
|
|
122
|
+
app.log.warn("⚠ Set server.api_keys in config.yaml or AIGETWEY_ADMIN_PASSWORD to secure the gateway.");
|
|
123
|
+
}
|
|
124
|
+
|
|
125
|
+
await app.listen({ host, port });
|
|
126
|
+
app.log.info(`aigetwey listening on http://${host}:${port}`);
|
|
99
127
|
} catch (e) {
|
|
100
128
|
app.log.error(e);
|
|
101
129
|
process.exit(1);
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
(globalThis.TURBOPACK||(globalThis.TURBOPACK=[])).push(["object"==typeof document?document.currentScript:void 0,38286,e=>{"use strict";var t=e.i(43476);e.s(["Empty",0,function({children:e}){return(0,t.jsx)("div",{className:"px-1 py-8 text-center text-[13px] text-text-muted",children:e})},"Stat",0,function({label:e,value:s,sub:l}){return(0,t.jsxs)("div",{className:"rounded-brand-lg border border-border bg-surface px-4 py-3.5 shadow-soft",children:[(0,t.jsx)("div",{className:"text-[11px] font-medium uppercase tracking-wider text-text-subtle",children:e}),(0,t.jsx)("div",{className:"mt-1 tnum text-[22px] font-semibold tracking-tight text-text",children:s}),l&&(0,t.jsx)("div",{className:"mt-0.5 text-[12px] text-text-muted",children:l})]})},"fmt",0,{int:e=>e.toLocaleString("en-US"),compact:e=>Math.abs(e)>=1e6?`${(e/1e6).toFixed(1)}M`:Math.abs(e)>=1e3?`${(e/1e3).toFixed(1)}K`:String(Math.round(e)),cost:e=>0===e?"$0":e<.01?`$${e.toFixed(4)}`:`$${e.toFixed(2)}`,time:e=>new Date(e).toLocaleString("en-US",{hour12:!1}),date(e){let t=new Date(e);return`${String(t.getDate()).padStart(2,"0")}/${String(t.getMonth()+1).padStart(2,"0")}/${t.getFullYear()}`},ago(e){let t=Math.max(0,Math.round((Date.now()-e)/1e3));return t<60?`${t}s`:t<3600?`${Math.floor(t/60)}m`:t<86400?`${Math.floor(t/3600)}h`:`${Math.floor(t/86400)}d`},duration(e){let t=Math.max(0,Math.round(e/1e3));if(t<60)return`${t}s`;if(t<3600)return`${Math.floor(t/60)}m ${t%60}s`;let s=Math.floor(t/3600),l=Math.floor(t%3600/60);if(s<24)return`${s}h ${l}m`;let r=Math.floor(s/24);return`${r}d ${s%24}h`}}])},81189,e=>{"use strict";var t=e.i(43476);let s={live:"bg-success/12 text-success",down:"bg-danger/12 text-danger",danger:"bg-danger/12 text-danger",warn:"bg-warning/12 text-warning",info:"bg-info/12 text-info",neutral:"bg-surface-2 text-text-muted"};function l({tone:e="neutral",children:r,className:a}){return(0,t.jsx)("span",{className:`inline-flex items-center gap-1 rounded-full px-2 py-0.5 text-[11px] font-medium ${s[e]}${a?` ${a}`:""}`,children:r})}e.s(["Badge",0,l,"FormatBadge",0,function({format:e}){return(0,t.jsx)(l,{tone:"info",children:e})}])},69163,e=>{"use strict";var t=e.i(43476);e.s(["CardTitle",0,function({title:e,sub:s,icon:l}){return(0,t.jsxs)("div",{className:"flex items-center gap-2.5 min-w-0",children:[l,(0,t.jsxs)("div",{className:"min-w-0",children:[(0,t.jsx)("div",{className:"truncate text-[14px] font-semibold tracking-tight text-text",children:e}),s&&(0,t.jsx)("div",{className:"truncate text-[12px] text-text-muted",children:s})]})]})},"RichCard",0,function({header:e,footer:s,children:l,className:r}){return(0,t.jsxs)("section",{className:`overflow-hidden rounded-brand-lg border border-border bg-surface shadow-soft${r?` ${r}`:""}`,children:[e&&(0,t.jsx)("header",{className:"flex items-center justify-between gap-3 border-b border-border-subtle px-4 py-3",children:e}),(0,t.jsx)("div",{className:"p-4",children:l}),s&&(0,t.jsx)("footer",{className:"flex items-center justify-between gap-3 border-t border-border-subtle bg-bg-alt px-4 py-2.5",children:s})]})}])},49080,e=>{"use strict";var t=e.i(43476);e.s(["Lamp",0,function({state:e,title:s}){return(0,t.jsx)("span",{className:`lamp lamp-${e}`,title:s,"aria-label":e})}])},24966,e=>{"use strict";var t=e.i(43476),s=e.i(71645);let l=[{key:"tokens",label:"Tokens"},{key:"cost",label:"Cost"},{key:"requests",label:"Requests"}];function r(e,t){return"cost"===t?e<.01?`$${e.toFixed(4)}`:`$${e.toFixed(2)}`:e>=1e6?`${(e/1e6).toFixed(1)}M`:e>=1e3?`${(e/1e3).toFixed(1)}K`:String(Math.round(e))}function a(e){return new Date(e).toLocaleTimeString("en-US",{hour:"2-digit",minute:"2-digit",hour12:!1})}function n({series:e}){let[i,o]=(0,s.useState)("tokens"),d=e.map(e=>"tokens"===i?e.tokens_in+e.tokens_out:"cost"===i?e.cost:e.requests),c=Math.max(1,...d),x=e.length,u=e=>x<=1?8:8+e/(x-1)*984,m=e=>224-e/c*208,h=d.map((e,t)=>`${0===t?"M":"L"}${u(t).toFixed(1)},${m(e).toFixed(1)}`).join(" "),p=`${h} L${u(x-1).toFixed(1)},224 L${u(0).toFixed(1)},224 Z`,b=d.length?d.indexOf(Math.max(...d)):-1,f=d.reduce((e,t)=>e+t,0);return(0,t.jsxs)("div",{className:"overflow-hidden rounded-brand-lg border border-border bg-surface shadow-soft",children:[(0,t.jsxs)("header",{className:"flex items-center justify-between gap-3 border-b border-border-subtle px-4 py-3",children:[(0,t.jsxs)("div",{className:"flex items-center gap-2.5",children:[(0,t.jsx)("span",{className:"text-[13px] font-semibold text-text",children:"Over time"}),(0,t.jsxs)("span",{className:"tnum text-[12px] text-text-muted",children:[r(f,i)," total"]})]}),(0,t.jsx)("div",{className:"flex items-center gap-1",children:l.map(e=>(0,t.jsx)("button",{onClick:()=>o(e.key),className:`rounded-full px-3 py-1 text-[12px] font-medium transition-colors ${i===e.key?"bg-surface-2 text-text":"text-text-muted hover:text-text"}`,children:e.label},e.key))})]}),(0,t.jsx)("div",{className:"px-2 py-3",children:0===x||0===f?(0,t.jsx)("div",{className:"flex h-[200px] items-center justify-center text-[13px] text-text-muted",children:"No activity in this range."}):(0,t.jsxs)("svg",{viewBox:"0 0 1000 240",className:"h-[200px] w-full",preserveAspectRatio:"none",children:[(0,t.jsx)("defs",{children:(0,t.jsxs)("linearGradient",{id:"areaFill",x1:"0",y1:"0",x2:"0",y2:"1",children:[(0,t.jsx)("stop",{offset:"0%",stopColor:"var(--color-accent)",stopOpacity:"0.28"}),(0,t.jsx)("stop",{offset:"100%",stopColor:"var(--color-accent)",stopOpacity:"0"})]})}),(0,t.jsx)("path",{d:p,fill:"url(#areaFill)"}),(0,t.jsx)("path",{d:h,fill:"none",stroke:"var(--color-accent)",strokeWidth:"2",vectorEffect:"non-scaling-stroke"}),b>=0&&(0,t.jsx)("circle",{cx:u(b),cy:m(d[b]),r:"3",fill:"var(--color-accent)",vectorEffect:"non-scaling-stroke"})]})}),x>0&&f>0&&b>=0&&(0,t.jsxs)("div",{className:"flex justify-between border-t border-border-subtle px-4 py-2 tnum text-[10px] text-text-subtle",children:[(0,t.jsx)("span",{children:a(e[0].ts)}),(0,t.jsxs)("span",{children:["peak ",r(d[b],i)," @ ",a(e[b].ts)]}),(0,t.jsx)("span",{children:a(e[x-1].ts)})]})]})}var i=e.i(38286),o=e.i(69163);let d=[{label:"Today",key:"today",bucketMs:36e5},{label:"24h",key:"24h",bucketMs:36e5},{label:"7D",key:"7d",bucketMs:216e5},{label:"30D",key:"30d",bucketMs:864e5},{label:"60D",key:"60d",bucketMs:1728e5}];function c({rows:e,loading:s}){return(0,t.jsx)("div",{className:`table-wrap ${s?"opacity-50":""}`,children:(0,t.jsxs)("table",{className:"w-full border-collapse",children:[(0,t.jsx)("thead",{children:(0,t.jsx)("tr",{className:"text-text-subtle",children:["","Reqs","In","Out","Cost"].map((e,s)=>(0,t.jsx)("th",{className:`pb-2 text-[10px] font-medium uppercase tracking-wider ${0===s?"text-left":"text-right"}`,children:e},e+s))})}),(0,t.jsx)("tbody",{children:e.map(e=>(0,t.jsxs)("tr",{className:"border-t border-border-subtle",children:[(0,t.jsx)("td",{className:"py-2 text-[12.5px] text-text",children:e.label}),(0,t.jsx)("td",{className:"py-2 text-right tnum text-[12.5px] text-text-muted",children:i.fmt.int(e.requests)}),(0,t.jsx)("td",{className:"py-2 text-right tnum text-[12.5px] text-text-muted",children:i.fmt.compact(e.tokens_in)}),(0,t.jsx)("td",{className:"py-2 text-right tnum text-[12.5px] text-text-muted",children:i.fmt.compact(e.tokens_out)}),(0,t.jsx)("td",{className:"py-2 text-right tnum text-[12.5px] text-text",children:i.fmt.cost(e.cost)})]},e.label))})]})})}e.s(["UsageView",0,function(){let[e,l]=(0,s.useState)(d[0]),[r,a]=(0,s.useState)(null),[x,u]=(0,s.useState)([]),[m,h]=(0,s.useState)(""),[p,b]=(0,s.useState)(!0),f=(0,s.useCallback)(async e=>{b(!0),h("");let t=function(e){let t=Date.now();switch(e){case"today":{let e=new Date;return e.setHours(0,0,0,0),e.getTime()}case"24h":return t-864e5;case"7d":return t-6048e5;case"30d":return t-2592e6;case"60d":return t-5184e6}}(e.key),[s,l]=await Promise.all([fetch(`/api/gw/admin/usage?since=${t}`),fetch(`/api/gw/admin/usage/series?since=${t}&bucket=${e.bucketMs}`)]);if(!s.ok){h("could not load usage"),b(!1);return}a(await s.json()),u((l.ok?await l.json():{series:[]}).series),b(!1)},[]);(0,s.useEffect)(()=>{f(e)},[e,f]);let j=r?.total;return(0,t.jsxs)("div",{children:[(0,t.jsxs)("div",{className:"mb-6 flex items-center justify-between gap-3",children:[(0,t.jsxs)("div",{children:[(0,t.jsx)("h1",{className:"text-[22px] font-semibold tracking-tight text-text",children:"Usage"}),(0,t.jsx)("p",{className:"mt-1 text-[13px] text-text-muted",children:"Tokens and cost across providers and models."})]}),(0,t.jsx)("div",{className:"flex items-center gap-1 rounded-full border border-border bg-surface p-1",children:d.map(s=>(0,t.jsx)("button",{onClick:()=>l(s),className:`rounded-full px-3 py-1 text-[12px] font-medium transition-colors ${e.label===s.label?"bg-surface-2 text-text":"text-text-muted hover:text-text"}`,children:s.label},s.label))})]}),m?(0,t.jsx)(i.Empty,{children:m}):(0,t.jsxs)(t.Fragment,{children:[(0,t.jsxs)("div",{className:"mb-5 grid grid-cols-2 gap-3 sm:grid-cols-4",children:[(0,t.jsx)(i.Stat,{label:"Requests",value:i.fmt.int(j?.requests??0)}),(0,t.jsx)(i.Stat,{label:"Tokens in",value:i.fmt.compact(j?.tokens_in??0)}),(0,t.jsx)(i.Stat,{label:"Tokens out",value:i.fmt.compact(j?.tokens_out??0)}),(0,t.jsx)(i.Stat,{label:"Cost",value:i.fmt.cost(j?.cost??0)})]}),(0,t.jsx)("div",{className:"mb-5",children:(0,t.jsx)(n,{series:x})}),(0,t.jsxs)("div",{className:"grid gap-4 lg:grid-cols-2",children:[(0,t.jsx)(o.RichCard,{header:(0,t.jsx)("span",{className:"text-[13px] font-semibold text-text",children:"By provider"}),children:r?.by_provider.length?(0,t.jsx)(c,{rows:r.by_provider.map(e=>({label:e.provider,...e})),loading:p}):(0,t.jsx)(i.Empty,{children:"No usage in this window."})}),(0,t.jsx)(o.RichCard,{header:(0,t.jsx)("span",{className:"text-[13px] font-semibold text-text",children:"By model"}),children:r?.by_model.length?(0,t.jsx)(c,{rows:r.by_model.map(e=>({label:`${e.alias} → ${e.model}`,...e})),loading:p}):(0,t.jsx)(i.Empty,{children:"No usage in this window."})})]})]})]})}],24966)},95509,e=>{"use strict";var t=e.i(43476),s=e.i(71645),l=e.i(49080),r=e.i(81189),a=e.i(51514),n=e.i(38286);let i=[{key:"all",label:"All"},{key:"ok",label:"Success"},{key:"error",label:"Errors"}],o=[{label:"1h",ms:36e5},{label:"6h",ms:216e5},{label:"24h",ms:864e5},{label:"7d",ms:6048e5},{label:"All",ms:null}];function d({label:e,children:s}){return(0,t.jsxs)("label",{className:"flex flex-col gap-1",children:[(0,t.jsx)("span",{className:"text-[10px] font-medium uppercase tracking-wider text-text-subtle",children:e}),s]})}function c({children:e,muted:s,right:l,title:r,className:a,suppressHydrationWarning:n}){return(0,t.jsx)("td",{title:r,suppressHydrationWarning:n,className:`whitespace-nowrap px-4 py-2.5 tnum text-[12.5px] ${l?"text-right":"text-left"} ${s?"text-text-muted":"text-text"}${a?` ${a}`:""}`,children:e})}function x({log:e,onClose:s}){let l=e.status>=200&&e.status<300,r=e.tokens_in+e.tokens_out;return(0,t.jsxs)("div",{className:"border-t border-border-subtle bg-surface-2/40 px-5 py-4",children:[(0,t.jsxs)("div",{className:"mb-3 flex items-center justify-between",children:[(0,t.jsx)("span",{className:"text-[12px] font-semibold text-text",children:"Request Details"}),(0,t.jsx)("button",{onClick:s,className:"text-text-subtle hover:text-text",children:(0,t.jsx)(a.Icon,{name:"close",size:16})})]}),(0,t.jsxs)("div",{className:"grid grid-cols-2 gap-x-8 gap-y-2 text-[12px] sm:grid-cols-4",children:[(0,t.jsx)(u,{label:"Status",value:`${e.status} ${l?"OK":"Error"}`}),(0,t.jsx)(u,{label:"Timestamp",value:new Date(e.ts).toLocaleString()}),(0,t.jsx)(u,{label:"Alias",value:e.alias}),(0,t.jsx)(u,{label:"Provider",value:e.provider}),(0,t.jsx)(u,{label:"Model",value:e.model}),(0,t.jsx)(u,{label:"Mode",value:e.stream?"Streaming":"Unary"}),(0,t.jsx)(u,{label:"Latency",value:`${e.latency_ms}ms`}),(0,t.jsx)(u,{label:"Cost",value:n.fmt.cost(e.cost)}),(0,t.jsx)(u,{label:"Input tokens",value:n.fmt.int(e.tokens_in)}),(0,t.jsx)(u,{label:"Output tokens",value:n.fmt.int(e.tokens_out)}),(0,t.jsx)(u,{label:"Cached tokens",value:n.fmt.int(e.cached_tokens)}),(0,t.jsx)(u,{label:"Total tokens",value:n.fmt.int(r)})]})]})}function u({label:e,value:s}){return(0,t.jsxs)("div",{children:[(0,t.jsx)("span",{className:"text-[10px] uppercase tracking-wider text-text-subtle",children:e}),(0,t.jsx)("div",{className:"tnum text-text",children:s})]})}e.s(["LogTable",0,function({logs:e}){let[u,m]=(0,s.useState)(e),[h,p]=(0,s.useState)("all"),[b,f]=(0,s.useState)("all"),[j,g]=(0,s.useState)(null),[v,y]=(0,s.useState)(!0),[N,k]=(0,s.useState)(!1),[w,$]=(0,s.useState)(null),S=(0,s.useRef)(null),M=(0,s.useCallback)(async()=>{try{let e=await fetch("/api/gw/admin/logs?limit=200");if(e.ok){let t=await e.json();m(t.logs)}}catch{}},[]);(0,s.useEffect)(()=>{if(v)return S.current=setInterval(M,5e3),()=>{S.current&&clearInterval(S.current)};S.current&&clearInterval(S.current)},[v,M]);let C=[...new Set(u.map(e=>e.provider))].sort(),_=u.filter(e=>e.status>=200&&e.status<300).length,F=u.length-_,T=null!==j?Date.now()-j:null,D=u.filter(e=>("ok"!==h||!!(e.status>=200&&e.status<300))&&("error"!==h||!(e.status>=200)||!(e.status<300))&&("all"===b||e.provider===b)&&(null===T||!(e.ts<T))),q="all"!==h||"all"!==b||null!==j;return(0,t.jsxs)("div",{className:"overflow-hidden rounded-brand-lg border border-border bg-surface shadow-soft",children:[(0,t.jsxs)("header",{className:"flex flex-col gap-3 border-b border-border-subtle px-4 py-3",children:[(0,t.jsxs)("div",{className:"flex flex-wrap items-center justify-between gap-3",children:[(0,t.jsxs)("div",{className:"flex items-center gap-1.5",children:[(0,t.jsx)("button",{onClick:()=>k(e=>!e),className:"flex-none rounded p-0.5 text-text-subtle transition-colors hover:text-text","aria-label":N?"Expand requests":"Collapse requests",title:N?"Expand":"Collapse",children:(0,t.jsx)(a.Icon,{name:N?"chevron_right":"expand_more",size:18})}),(0,t.jsx)("div",{className:"flex items-center gap-1",children:i.map(e=>{let s;return(0,t.jsxs)("button",{onClick:()=>p(e.key),className:`flex items-center gap-1.5 rounded-full px-3 py-1.5 text-[12px] font-medium transition-colors ${h===e.key?"bg-surface-2 text-text":"text-text-muted hover:text-text"}`,children:[e.label,(0,t.jsx)("span",{className:"tnum text-text-subtle",children:"all"===(s=e.key)?u.length:"ok"===s?_:F})]},e.key)})})]}),(0,t.jsxs)("button",{onClick:()=>y(e=>!e),className:`flex items-center gap-1 rounded-full px-2.5 py-1 text-[11px] font-medium transition-colors ${v?"bg-success/15 text-success":"text-text-muted hover:text-text"}`,title:v?"Auto-refresh ON (5s)":"Auto-refresh OFF",children:[(0,t.jsx)(a.Icon,{name:v?"radio_button_checked":"radio_button_unchecked",size:12}),"Live"]})]}),!N&&(0,t.jsxs)("div",{className:"flex flex-wrap items-end gap-2.5",children:[(0,t.jsx)(d,{label:"Provider",children:(0,t.jsxs)("select",{value:b,onChange:e=>f(e.target.value),className:"h-9 rounded-brand border border-border bg-surface-2 px-2.5 text-[12.5px] text-text focus:border-accent focus:outline-none w-40",children:[(0,t.jsx)("option",{value:"all",children:"All providers"}),C.map(e=>(0,t.jsx)("option",{value:e,children:e},e))]})}),(0,t.jsx)(d,{label:"Since",children:(0,t.jsx)("div",{className:"flex h-9 items-center gap-0.5 rounded-brand border border-border bg-surface-2 px-1",children:o.map(e=>{let s;return(0,t.jsx)("button",{type:"button",onClick:()=>g(e.ms),className:(s=j===e.ms,`rounded-full px-2.5 py-1 text-[12px] font-medium transition-colors ${s?"bg-accent/15 text-accent":"text-text-muted hover:text-text"}`),children:e.label},e.label)})})}),(0,t.jsxs)("button",{onClick:()=>{p("all"),f("all"),g(null)},disabled:!q,className:"flex h-9 items-center gap-1.5 rounded-brand border border-border bg-surface-2 px-3 text-[12.5px] font-medium text-text-muted transition-colors hover:border-text-subtle hover:text-text disabled:opacity-40 disabled:hover:border-border disabled:hover:text-text-muted",title:q?"Reset all filters":"No filters applied",children:[(0,t.jsx)(a.Icon,{name:"filter_alt_off",size:15}),"Clear"]})]})]}),N?null:0===D.length?(0,t.jsx)("div",{className:"px-4 py-8 text-center text-[13px] text-text-muted",children:0===u.length?"No requests recorded yet.":"No requests match this filter."}):(0,t.jsxs)(t.Fragment,{children:[(0,t.jsx)("div",{className:"table-wrap",children:(0,t.jsxs)("table",{className:"w-full min-w-[720px] border-collapse",children:[(0,t.jsx)("thead",{children:(0,t.jsx)("tr",{className:"text-text-subtle",children:["Status","Time","Alias","Provider","Model","In","Out","ms","Mode"].map((e,s)=>(0,t.jsx)("th",{className:`whitespace-nowrap px-4 pb-2.5 pt-3 text-[10px] font-medium uppercase tracking-wider ${s>=5&&s<=7?"text-right":"text-left"}`,children:e},e+s))})}),(0,t.jsx)("tbody",{children:D.map((e,s)=>{let a=e.status>=200&&e.status<300,i=w===s;return(0,t.jsxs)("tr",{onClick:()=>$(i?null:s),className:`border-t border-border-subtle cursor-pointer transition-colors ${i?"bg-surface-2/60":"hover:bg-surface-2/70"}`,children:[(0,t.jsx)(c,{children:(0,t.jsxs)(r.Badge,{tone:a?"live":"down",children:[(0,t.jsx)(l.Lamp,{state:a?"live":"down"}),e.status]})}),(0,t.jsxs)(c,{muted:!0,title:n.fmt.time(e.ts),suppressHydrationWarning:!0,children:[n.fmt.ago(e.ts)," ago"]}),(0,t.jsx)(c,{className:"text-text",children:e.alias}),(0,t.jsx)(c,{muted:!0,children:e.provider}),(0,t.jsx)(c,{muted:!0,children:e.model}),(0,t.jsx)(c,{right:!0,children:n.fmt.int(e.tokens_in)}),(0,t.jsx)(c,{right:!0,children:n.fmt.int(e.tokens_out)}),(0,t.jsx)(c,{right:!0,children:n.fmt.int(e.latency_ms)}),(0,t.jsx)(c,{muted:!0,children:e.stream?"stream":"unary"})]},s)})})]})}),null!==w&&D[w]&&(0,t.jsx)(x,{log:D[w],onClose:()=>$(null)})]})]})}])}]);
|
|
@@ -1,3 +0,0 @@
|
|
|
1
|
-
@font-face{font-family:Inter;font-style:normal;font-weight:100 900;font-display:swap;src:url(../media/2c55a0e60120577a-s.0-dom-5bn10r2.woff2)format("woff2");unicode-range:U+460-52F,U+1C80-1C8A,U+20B4,U+2DE0-2DFF,U+A640-A69F,U+FE2E-FE2F}@font-face{font-family:Inter;font-style:normal;font-weight:100 900;font-display:swap;src:url(../media/9c72aa0f40e4eef8-s.1y4-pdgsjb-pw.woff2)format("woff2");unicode-range:U+301,U+400-45F,U+490-491,U+4B0-4B1,U+2116}@font-face{font-family:Inter;font-style:normal;font-weight:100 900;font-display:swap;src:url(../media/ad66f9afd8947f86-s.3lvt2whj97whp.woff2)format("woff2");unicode-range:U+1F??}@font-face{font-family:Inter;font-style:normal;font-weight:100 900;font-display:swap;src:url(../media/5476f68d60460930-s.2uwcyprjm3xu3.woff2)format("woff2");unicode-range:U+370-377,U+37A-37F,U+384-38A,U+38C,U+38E-3A1,U+3A3-3FF}@font-face{font-family:Inter;font-style:normal;font-weight:100 900;font-display:swap;src:url(../media/2bbe8d2671613f1f-s.0k62hbripvv8p.woff2)format("woff2");unicode-range:U+102-103,U+110-111,U+128-129,U+168-169,U+1A0-1A1,U+1AF-1B0,U+300-301,U+303-304,U+308-309,U+323,U+329,U+1EA0-1EF9,U+20AB}@font-face{font-family:Inter;font-style:normal;font-weight:100 900;font-display:swap;src:url(../media/1bffadaabf893a1e-s.3-6t-g6q0vh0a.woff2)format("woff2");unicode-range:U+100-2BA,U+2BD-2C5,U+2C7-2CC,U+2CE-2D7,U+2DD-2FF,U+304,U+308,U+329,U+1D00-1DBF,U+1E00-1E9F,U+1EF2-1EFF,U+2020,U+20A0-20AB,U+20AD-20C0,U+2113,U+2C60-2C7F,U+A720-A7FF}@font-face{font-family:Inter;font-style:normal;font-weight:100 900;font-display:swap;src:url(../media/83afe278b6a6bb3c-s.p.2bn3s6zvc0dyp.woff2)format("woff2");unicode-range:U+??,U+131,U+152-153,U+2BB-2BC,U+2C6,U+2DA,U+2DC,U+304,U+308,U+329,U+2000-206F,U+20AC,U+2122,U+2191,U+2193,U+2212,U+2215,U+FEFF,U+FFFD}@font-face{font-family:Inter Fallback;src:local(Arial);ascent-override:90.44%;descent-override:22.52%;line-gap-override:0.0%;size-adjust:107.12%}.inter_c15e96cb-module__0bjUvq__className{font-family:Inter,Inter Fallback;font-style:normal}.inter_c15e96cb-module__0bjUvq__variable{--font-inter:"Inter", "Inter Fallback"}
|
|
2
|
-
@font-face{font-family:JetBrains Mono;font-style:normal;font-weight:100 800;font-display:swap;src:url(../media/13bf9871fe164e7f-s.2f7nqdagzwx2-.woff2)format("woff2");unicode-range:U+460-52F,U+1C80-1C8A,U+20B4,U+2DE0-2DFF,U+A640-A69F,U+FE2E-FE2F}@font-face{font-family:JetBrains Mono;font-style:normal;font-weight:100 800;font-display:swap;src:url(../media/cc545e633e20c56d-s.176arc174-8zp.woff2)format("woff2");unicode-range:U+301,U+400-45F,U+490-491,U+4B0-4B1,U+2116}@font-face{font-family:JetBrains Mono;font-style:normal;font-weight:100 800;font-display:swap;src:url(../media/71b036adf157cdcf-s.0bp8oijd_gu96.woff2)format("woff2");unicode-range:U+370-377,U+37A-37F,U+384-38A,U+38C,U+38E-3A1,U+3A3-3FF}@font-face{font-family:JetBrains Mono;font-style:normal;font-weight:100 800;font-display:swap;src:url(../media/89b21bb081cb7469-s.1fby2rem9ngyr.woff2)format("woff2");unicode-range:U+102-103,U+110-111,U+128-129,U+168-169,U+1A0-1A1,U+1AF-1B0,U+300-301,U+303-304,U+308-309,U+323,U+329,U+1EA0-1EF9,U+20AB}@font-face{font-family:JetBrains Mono;font-style:normal;font-weight:100 800;font-display:swap;src:url(../media/3fe682a82f50d426-s.0vfdmo25voy_0.woff2)format("woff2");unicode-range:U+100-2BA,U+2BD-2C5,U+2C7-2CC,U+2CE-2D7,U+2DD-2FF,U+304,U+308,U+329,U+1D00-1DBF,U+1E00-1E9F,U+1EF2-1EFF,U+2020,U+20A0-20AB,U+20AD-20C0,U+2113,U+2C60-2C7F,U+A720-A7FF}@font-face{font-family:JetBrains Mono;font-style:normal;font-weight:100 800;font-display:swap;src:url(../media/70bc3e132a0a741e-s.p.3t6q91iet4nsy.woff2)format("woff2");unicode-range:U+??,U+131,U+152-153,U+2BB-2BC,U+2C6,U+2DA,U+2DC,U+304,U+308,U+329,U+2000-206F,U+20AC,U+2122,U+2191,U+2193,U+2212,U+2215,U+FEFF,U+FFFD}@font-face{font-family:JetBrains Mono Fallback;src:local(Arial);ascent-override:75.79%;descent-override:22.29%;line-gap-override:0.0%;size-adjust:134.59%}.jetbrains_mono_7ea1d0f9-module__6GV5LG__className{font-family:JetBrains Mono,JetBrains Mono Fallback;font-style:normal}.jetbrains_mono_7ea1d0f9-module__6GV5LG__variable{--font-jetbrains-mono:"JetBrains Mono", "JetBrains Mono Fallback"}
|
|
3
|
-
@layer properties{@supports (((-webkit-hyphens:none)) and (not (margin-trim:inline))) or ((-moz-orient:inline) and (not (color:rgb(from red r g b)))){*,:before,:after,::backdrop{--tw-translate-x:0;--tw-translate-y:0;--tw-translate-z:0;--tw-rotate-x:initial;--tw-rotate-y:initial;--tw-rotate-z:initial;--tw-skew-x:initial;--tw-skew-y:initial;--tw-space-y-reverse:0;--tw-divide-y-reverse:0;--tw-border-style:solid;--tw-leading:initial;--tw-font-weight:initial;--tw-tracking:initial;--tw-shadow:0 0 #0000;--tw-shadow-color:initial;--tw-shadow-alpha:100%;--tw-inset-shadow:0 0 #0000;--tw-inset-shadow-color:initial;--tw-inset-shadow-alpha:100%;--tw-ring-color:initial;--tw-ring-shadow:0 0 #0000;--tw-inset-ring-color:initial;--tw-inset-ring-shadow:0 0 #0000;--tw-ring-inset:initial;--tw-ring-offset-width:0px;--tw-ring-offset-color:#fff;--tw-ring-offset-shadow:0 0 #0000;--tw-blur:initial;--tw-brightness:initial;--tw-contrast:initial;--tw-grayscale:initial;--tw-hue-rotate:initial;--tw-invert:initial;--tw-opacity:initial;--tw-saturate:initial;--tw-sepia:initial;--tw-drop-shadow:initial;--tw-drop-shadow-color:initial;--tw-drop-shadow-alpha:100%;--tw-drop-shadow-size:initial;--tw-duration:initial}}}@layer theme{:root,:host{--font-sans:var(--font-inter), -apple-system, BlinkMacSystemFont, "SF Pro Text", system-ui, sans-serif;--font-mono:var(--font-jetbrains-mono), ui-monospace, "SF Mono", Menlo, monospace;--color-black:#000;--color-white:#fff;--spacing:.25rem;--container-sm:24rem;--container-md:28rem;--container-lg:32rem;--font-weight-medium:500;--font-weight-semibold:600;--font-weight-bold:700;--tracking-tight:-.025em;--tracking-normal:0em;--tracking-wider:.05em;--leading-relaxed:1.625;--animate-spin:spin 1s linear infinite;--default-transition-duration:.15s;--default-transition-timing-function:cubic-bezier(.4, 0, .2, 1);--default-font-family:var(--font-sans);--default-mono-font-family:var(--font-mono);--color-bg:#0f0f0e;--color-bg-alt:#141413;--color-surface:#1a1a18;--color-surface-2:#222220;--color-surface-3:#2c2c29;--color-sidebar:#131312;--color-border:#2a2a27;--color-border-subtle:#1f1f1d;--color-text:#ededec;--color-text-muted:#8d8d86;--color-text-subtle:#5c5c56;--color-accent:#cbe85a;--color-accent-hover:#d6f06a;--color-accent-ink:#14140f;--color-accent-soft:#3a3f1c;--color-success:#7dd87f;--color-warning:#e8c55a;--color-danger:#e8806a;--color-info:#7fb0e8;--radius-brand:12px;--radius-brand-lg:16px;--shadow-soft:0 1px 2px 0 #0006;--shadow-warm:0 4px 16px -4px #cbe85a2e}}@layer base{*,:after,:before,::backdrop{box-sizing:border-box;border:0 solid;margin:0;padding:0}::file-selector-button{box-sizing:border-box;border:0 solid;margin:0;padding:0}html,:host{-webkit-text-size-adjust:100%;tab-size:4;line-height:1.5;font-family:var(--default-font-family,ui-sans-serif, system-ui, sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol", "Noto Color Emoji");font-feature-settings:var(--default-font-feature-settings,normal);font-variation-settings:var(--default-font-variation-settings,normal);-webkit-tap-highlight-color:transparent}hr{height:0;color:inherit;border-top-width:1px}abbr:where([title]){-webkit-text-decoration:underline dotted;text-decoration:underline dotted}h1,h2,h3,h4,h5,h6{font-size:inherit;font-weight:inherit}a{color:inherit;-webkit-text-decoration:inherit;-webkit-text-decoration:inherit;-webkit-text-decoration:inherit;-webkit-text-decoration:inherit;text-decoration:inherit}b,strong{font-weight:bolder}code,kbd,samp,pre{font-family:var(--default-mono-font-family,ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace);font-feature-settings:var(--default-mono-font-feature-settings,normal);font-variation-settings:var(--default-mono-font-variation-settings,normal);font-size:1em}small{font-size:80%}sub,sup{vertical-align:baseline;font-size:75%;line-height:0;position:relative}sub{bottom:-.25em}sup{top:-.5em}table{text-indent:0;border-color:inherit;border-collapse:collapse}:-moz-focusring{outline:auto}progress{vertical-align:baseline}summary{display:list-item}ol,ul,menu{list-style:none}img,svg,video,canvas,audio,iframe,embed,object{vertical-align:middle;display:block}img,video{max-width:100%;height:auto}button,input,select,optgroup,textarea{font:inherit;font-feature-settings:inherit;font-variation-settings:inherit;letter-spacing:inherit;color:inherit;opacity:1;background-color:#0000;border-radius:0}::file-selector-button{font:inherit;font-feature-settings:inherit;font-variation-settings:inherit;letter-spacing:inherit;color:inherit;opacity:1;background-color:#0000;border-radius:0}:where(select:is([multiple],[size])) optgroup{font-weight:bolder}:where(select:is([multiple],[size])) optgroup option{padding-inline-start:20px}::file-selector-button{margin-inline-end:4px}::placeholder{opacity:1}@supports (not ((-webkit-appearance:-apple-pay-button))) or (contain-intrinsic-size:1px){::placeholder{color:currentColor}@supports (color:color-mix(in lab, red, red)){::placeholder{color:color-mix(in oklab, currentcolor 50%, transparent)}}}textarea{resize:vertical}::-webkit-search-decoration{-webkit-appearance:none}::-webkit-date-and-time-value{min-height:1lh;text-align:inherit}::-webkit-datetime-edit{display:inline-flex}::-webkit-datetime-edit-fields-wrapper{padding:0}::-webkit-datetime-edit{padding-block:0}::-webkit-datetime-edit-year-field{padding-block:0}::-webkit-datetime-edit-month-field{padding-block:0}::-webkit-datetime-edit-day-field{padding-block:0}::-webkit-datetime-edit-hour-field{padding-block:0}::-webkit-datetime-edit-minute-field{padding-block:0}::-webkit-datetime-edit-second-field{padding-block:0}::-webkit-datetime-edit-millisecond-field{padding-block:0}::-webkit-datetime-edit-meridiem-field{padding-block:0}::-webkit-calendar-picker-indicator{line-height:1}:-moz-ui-invalid{box-shadow:none}button,input:where([type=button],[type=reset],[type=submit]){appearance:button}::file-selector-button{appearance:button}::-webkit-inner-spin-button{height:auto}::-webkit-outer-spin-button{height:auto}[hidden]:where(:not([hidden=until-found])){display:none!important}}@layer components;@layer utilities{.pointer-events-none{pointer-events:none}.invisible{visibility:hidden}.visible{visibility:visible}.absolute{position:absolute}.fixed{position:fixed}.relative{position:relative}.static{position:static}.sticky{position:sticky}.inset-0{inset:0}.inset-x-0{inset-inline:0}.top-0{top:0}.top-0\.5{top:calc(var(--spacing) * .5)}.top-1\/2{top:50%}.top-2{top:calc(var(--spacing) * 2)}.right-2{right:calc(var(--spacing) * 2)}.right-5{right:calc(var(--spacing) * 5)}.bottom-5{bottom:calc(var(--spacing) * 5)}.left-0\.5{left:calc(var(--spacing) * .5)}.left-2\.5{left:calc(var(--spacing) * 2.5)}.z-10{z-index:10}.z-50{z-index:50}.z-\[100\]{z-index:100}.mx-auto{margin-inline:auto}.mt-0\.5{margin-top:calc(var(--spacing) * .5)}.mt-1{margin-top:var(--spacing)}.mt-1\.5{margin-top:calc(var(--spacing) * 1.5)}.mt-2{margin-top:calc(var(--spacing) * 2)}.mt-2\.5{margin-top:calc(var(--spacing) * 2.5)}.mt-3{margin-top:calc(var(--spacing) * 3)}.mt-4{margin-top:calc(var(--spacing) * 4)}.mt-5{margin-top:calc(var(--spacing) * 5)}.mt-6{margin-top:calc(var(--spacing) * 6)}.mr-1{margin-right:var(--spacing)}.mb-1{margin-bottom:var(--spacing)}.mb-1\.5{margin-bottom:calc(var(--spacing) * 1.5)}.mb-2{margin-bottom:calc(var(--spacing) * 2)}.mb-2\.5{margin-bottom:calc(var(--spacing) * 2.5)}.mb-3{margin-bottom:calc(var(--spacing) * 3)}.mb-4{margin-bottom:calc(var(--spacing) * 4)}.mb-5{margin-bottom:calc(var(--spacing) * 5)}.mb-6{margin-bottom:calc(var(--spacing) * 6)}.ml-1\.5{margin-left:calc(var(--spacing) * 1.5)}.ml-2{margin-left:calc(var(--spacing) * 2)}.ml-auto{margin-left:auto}.block{display:block}.flex{display:flex}.grid{display:grid}.hidden{display:none}.inline{display:inline}.inline-flex{display:inline-flex}.h-1\.5{height:calc(var(--spacing) * 1.5)}.h-3{height:calc(var(--spacing) * 3)}.h-4{height:calc(var(--spacing) * 4)}.h-5{height:calc(var(--spacing) * 5)}.h-6{height:calc(var(--spacing) * 6)}.h-7{height:calc(var(--spacing) * 7)}.h-8{height:calc(var(--spacing) * 8)}.h-9{height:calc(var(--spacing) * 9)}.h-10{height:calc(var(--spacing) * 10)}.h-\[3px\]{height:3px}.h-\[18px\]{height:18px}.h-\[55vh\]{height:55vh}.h-\[200px\]{height:200px}.h-\[calc\(100vh-200px\)\]{height:calc(100vh - 200px)}.h-full{height:100%}.max-h-\[80vh\]{max-height:80vh}.max-h-\[360px\]{max-height:360px}.min-h-\[34px\]{min-height:34px}.min-h-screen{min-height:100vh}.w-3{width:calc(var(--spacing) * 3)}.w-4{width:calc(var(--spacing) * 4)}.w-5{width:calc(var(--spacing) * 5)}.w-6{width:calc(var(--spacing) * 6)}.w-7{width:calc(var(--spacing) * 7)}.w-8{width:calc(var(--spacing) * 8)}.w-9{width:calc(var(--spacing) * 9)}.w-10{width:calc(var(--spacing) * 10)}.w-11{width:calc(var(--spacing) * 11)}.w-16{width:calc(var(--spacing) * 16)}.w-24{width:calc(var(--spacing) * 24)}.w-32{width:calc(var(--spacing) * 32)}.w-40{width:calc(var(--spacing) * 40)}.w-48{width:calc(var(--spacing) * 48)}.w-\[18px\]{width:18px}.w-full{width:100%}.max-w-\[180px\]{max-width:180px}.max-w-\[280px\]{max-width:280px}.max-w-\[360px\]{max-width:360px}.max-w-\[480px\]{max-width:480px}.max-w-lg{max-width:var(--container-lg)}.max-w-md{max-width:var(--container-md)}.max-w-sm{max-width:var(--container-sm)}.min-w-0{min-width:0}.min-w-\[720px\]{min-width:720px}.flex-1{flex:1}.flex-none{flex:none}.grow{flex-grow:1}.border-collapse{border-collapse:collapse}.translate-x-0{--tw-translate-x:0;translate:var(--tw-translate-x) var(--tw-translate-y)}.translate-x-5{--tw-translate-x:calc(var(--spacing) * 5);translate:var(--tw-translate-x) var(--tw-translate-y)}.translate-x-\[14px\]{--tw-translate-x:14px;translate:var(--tw-translate-x) var(--tw-translate-y)}.translate-x-\[18px\]{--tw-translate-x:18px;translate:var(--tw-translate-x) var(--tw-translate-y)}.-translate-y-1\/2{--tw-translate-y:calc(calc(1 / 2 * 100%) * -1);translate:var(--tw-translate-x) var(--tw-translate-y)}.transform{transform:var(--tw-rotate-x,) var(--tw-rotate-y,) var(--tw-rotate-z,) var(--tw-skew-x,) var(--tw-skew-y,)}.animate-\[slideIn_0\.2s_ease\]{animation:.2s slideIn}.animate-spin{animation:var(--animate-spin)}.cursor-grab{cursor:grab}.cursor-pointer{cursor:pointer}.resize-none{resize:none}.list-none{list-style-type:none}.grid-cols-2{grid-template-columns:repeat(2,minmax(0,1fr))}.grid-cols-\[7rem_1fr\]{grid-template-columns:7rem 1fr}.flex-col{flex-direction:column}.flex-wrap{flex-wrap:wrap}.place-items-center{place-items:center}.items-center{align-items:center}.items-end{align-items:flex-end}.items-start{align-items:flex-start}.justify-between{justify-content:space-between}.justify-center{justify-content:center}.justify-end{justify-content:flex-end}.gap-0\.5{gap:calc(var(--spacing) * .5)}.gap-1{gap:var(--spacing)}.gap-1\.5{gap:calc(var(--spacing) * 1.5)}.gap-2{gap:calc(var(--spacing) * 2)}.gap-2\.5{gap:calc(var(--spacing) * 2.5)}.gap-3{gap:calc(var(--spacing) * 3)}.gap-4{gap:calc(var(--spacing) * 4)}:where(.space-y-1\.5>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing) * 1.5) * var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing) * 1.5) * calc(1 - var(--tw-space-y-reverse)))}:where(.space-y-2>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing) * 2) * var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing) * 2) * calc(1 - var(--tw-space-y-reverse)))}:where(.space-y-2\.5>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing) * 2.5) * var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing) * 2.5) * calc(1 - var(--tw-space-y-reverse)))}:where(.space-y-3>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing) * 3) * var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing) * 3) * calc(1 - var(--tw-space-y-reverse)))}:where(.space-y-4>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing) * 4) * var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing) * 4) * calc(1 - var(--tw-space-y-reverse)))}:where(.space-y-7>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing) * 7) * var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing) * 7) * calc(1 - var(--tw-space-y-reverse)))}.gap-x-8{column-gap:calc(var(--spacing) * 8)}.gap-y-2{row-gap:calc(var(--spacing) * 2)}:where(.divide-y>:not(:last-child)){--tw-divide-y-reverse:0;border-bottom-style:var(--tw-border-style);border-top-style:var(--tw-border-style);border-top-width:calc(1px * var(--tw-divide-y-reverse));border-bottom-width:calc(1px * calc(1 - var(--tw-divide-y-reverse)))}:where(.divide-border-subtle>:not(:last-child)){border-color:var(--color-border-subtle)}.truncate{text-overflow:ellipsis;white-space:nowrap;overflow:hidden}.overflow-hidden{overflow:hidden}.overflow-x-auto{overflow-x:auto}.overflow-y-auto{overflow-y:auto}.rounded{border-radius:.25rem}.rounded-\[5px\]{border-radius:5px}.rounded-brand{border-radius:var(--radius-brand)}.rounded-brand-lg{border-radius:var(--radius-brand-lg)}.rounded-full{border-radius:3.40282e38px}.border{border-style:var(--tw-border-style);border-width:1px}.border-t{border-top-style:var(--tw-border-style);border-top-width:1px}.border-b{border-bottom-style:var(--tw-border-style);border-bottom-width:1px}.border-dashed{--tw-border-style:dashed;border-style:dashed}.border-accent{border-color:var(--color-accent)}.border-accent\/40{border-color:#cbe85a66}@supports (color:color-mix(in lab, red, red)){.border-accent\/40{border-color:color-mix(in oklab, var(--color-accent) 40%, transparent)}}.border-border{border-color:var(--color-border)}.border-border-subtle{border-color:var(--color-border-subtle)}.border-danger\/30{border-color:#e8806a4d}@supports (color:color-mix(in lab, red, red)){.border-danger\/30{border-color:color-mix(in oklab, var(--color-danger) 30%, transparent)}}.border-danger\/35{border-color:#e8806a59}@supports (color:color-mix(in lab, red, red)){.border-danger\/35{border-color:color-mix(in oklab, var(--color-danger) 35%, transparent)}}.border-danger\/40{border-color:#e8806a66}@supports (color:color-mix(in lab, red, red)){.border-danger\/40{border-color:color-mix(in oklab, var(--color-danger) 40%, transparent)}}.border-info\/30{border-color:#7fb0e84d}@supports (color:color-mix(in lab, red, red)){.border-info\/30{border-color:color-mix(in oklab, var(--color-info) 30%, transparent)}}.border-success\/30{border-color:#7dd87f4d}@supports (color:color-mix(in lab, red, red)){.border-success\/30{border-color:color-mix(in oklab, var(--color-success) 30%, transparent)}}.border-transparent{border-color:#0000}.border-warning\/30{border-color:#e8c55a4d}@supports (color:color-mix(in lab, red, red)){.border-warning\/30{border-color:color-mix(in oklab, var(--color-warning) 30%, transparent)}}.border-warning\/40{border-color:#e8c55a66}@supports (color:color-mix(in lab, red, red)){.border-warning\/40{border-color:color-mix(in oklab, var(--color-warning) 40%, transparent)}}.bg-\[\#0a0a09\]{background-color:#0a0a09}.bg-accent{background-color:var(--color-accent)}.bg-accent-soft{background-color:var(--color-accent-soft)}.bg-accent-soft\/40{background-color:#3a3f1c66}@supports (color:color-mix(in lab, red, red)){.bg-accent-soft\/40{background-color:color-mix(in oklab, var(--color-accent-soft) 40%, transparent)}}.bg-accent\/12{background-color:#cbe85a1f}@supports (color:color-mix(in lab, red, red)){.bg-accent\/12{background-color:color-mix(in oklab, var(--color-accent) 12%, transparent)}}.bg-accent\/15{background-color:#cbe85a26}@supports (color:color-mix(in lab, red, red)){.bg-accent\/15{background-color:color-mix(in oklab, var(--color-accent) 15%, transparent)}}.bg-bg{background-color:var(--color-bg)}.bg-bg-alt{background-color:var(--color-bg-alt)}.bg-black\/50{background-color:#00000080}@supports (color:color-mix(in lab, red, red)){.bg-black\/50{background-color:color-mix(in oklab, var(--color-black) 50%, transparent)}}.bg-black\/60{background-color:#0009}@supports (color:color-mix(in lab, red, red)){.bg-black\/60{background-color:color-mix(in oklab, var(--color-black) 60%, transparent)}}.bg-black\/70{background-color:#000000b3}@supports (color:color-mix(in lab, red, red)){.bg-black\/70{background-color:color-mix(in oklab, var(--color-black) 70%, transparent)}}.bg-border-subtle{background-color:var(--color-border-subtle)}.bg-danger{background-color:var(--color-danger)}.bg-danger\/8{background-color:#e8806a14}@supports (color:color-mix(in lab, red, red)){.bg-danger\/8{background-color:color-mix(in oklab, var(--color-danger) 8%, transparent)}}.bg-danger\/10{background-color:#e8806a1a}@supports (color:color-mix(in lab, red, red)){.bg-danger\/10{background-color:color-mix(in oklab, var(--color-danger) 10%, transparent)}}.bg-danger\/12{background-color:#e8806a1f}@supports (color:color-mix(in lab, red, red)){.bg-danger\/12{background-color:color-mix(in oklab, var(--color-danger) 12%, transparent)}}.bg-info\/10{background-color:#7fb0e81a}@supports (color:color-mix(in lab, red, red)){.bg-info\/10{background-color:color-mix(in oklab, var(--color-info) 10%, transparent)}}.bg-info\/12{background-color:#7fb0e81f}@supports (color:color-mix(in lab, red, red)){.bg-info\/12{background-color:color-mix(in oklab, var(--color-info) 12%, transparent)}}.bg-success\/10{background-color:#7dd87f1a}@supports (color:color-mix(in lab, red, red)){.bg-success\/10{background-color:color-mix(in oklab, var(--color-success) 10%, transparent)}}.bg-success\/12{background-color:#7dd87f1f}@supports (color:color-mix(in lab, red, red)){.bg-success\/12{background-color:color-mix(in oklab, var(--color-success) 12%, transparent)}}.bg-success\/15{background-color:#7dd87f26}@supports (color:color-mix(in lab, red, red)){.bg-success\/15{background-color:color-mix(in oklab, var(--color-success) 15%, transparent)}}.bg-surface{background-color:var(--color-surface)}.bg-surface-2{background-color:var(--color-surface-2)}.bg-surface-2\/40{background-color:#22222066}@supports (color:color-mix(in lab, red, red)){.bg-surface-2\/40{background-color:color-mix(in oklab, var(--color-surface-2) 40%, transparent)}}.bg-surface-2\/60{background-color:#22222099}@supports (color:color-mix(in lab, red, red)){.bg-surface-2\/60{background-color:color-mix(in oklab, var(--color-surface-2) 60%, transparent)}}.bg-surface-3{background-color:var(--color-surface-3)}.bg-transparent{background-color:#0000}.bg-warning{background-color:var(--color-warning)}.bg-warning\/8{background-color:#e8c55a14}@supports (color:color-mix(in lab, red, red)){.bg-warning\/8{background-color:color-mix(in oklab, var(--color-warning) 8%, transparent)}}.bg-warning\/10{background-color:#e8c55a1a}@supports (color:color-mix(in lab, red, red)){.bg-warning\/10{background-color:color-mix(in oklab, var(--color-warning) 10%, transparent)}}.bg-warning\/12{background-color:#e8c55a1f}@supports (color:color-mix(in lab, red, red)){.bg-warning\/12{background-color:color-mix(in oklab, var(--color-warning) 12%, transparent)}}.bg-white{background-color:var(--color-white)}.p-0\.5{padding:calc(var(--spacing) * .5)}.p-1{padding:var(--spacing)}.p-4{padding:calc(var(--spacing) * 4)}.p-5{padding:calc(var(--spacing) * 5)}.p-6{padding:calc(var(--spacing) * 6)}.p-7{padding:calc(var(--spacing) * 7)}.px-1{padding-inline:var(--spacing)}.px-1\.5{padding-inline:calc(var(--spacing) * 1.5)}.px-2{padding-inline:calc(var(--spacing) * 2)}.px-2\.5{padding-inline:calc(var(--spacing) * 2.5)}.px-3{padding-inline:calc(var(--spacing) * 3)}.px-3\.5{padding-inline:calc(var(--spacing) * 3.5)}.px-4{padding-inline:calc(var(--spacing) * 4)}.px-5{padding-inline:calc(var(--spacing) * 5)}.py-0\.5{padding-block:calc(var(--spacing) * .5)}.py-1{padding-block:var(--spacing)}.py-1\.5{padding-block:calc(var(--spacing) * 1.5)}.py-2{padding-block:calc(var(--spacing) * 2)}.py-2\.5{padding-block:calc(var(--spacing) * 2.5)}.py-3{padding-block:calc(var(--spacing) * 3)}.py-3\.5{padding-block:calc(var(--spacing) * 3.5)}.py-4{padding-block:calc(var(--spacing) * 4)}.py-6{padding-block:calc(var(--spacing) * 6)}.py-8{padding-block:calc(var(--spacing) * 8)}.pt-1{padding-top:var(--spacing)}.pt-1\.5{padding-top:calc(var(--spacing) * 1.5)}.pt-2\.5{padding-top:calc(var(--spacing) * 2.5)}.pt-3{padding-top:calc(var(--spacing) * 3)}.pr-3{padding-right:calc(var(--spacing) * 3)}.pr-5{padding-right:calc(var(--spacing) * 5)}.pr-9{padding-right:calc(var(--spacing) * 9)}.pb-2{padding-bottom:calc(var(--spacing) * 2)}.pb-2\.5{padding-bottom:calc(var(--spacing) * 2.5)}.pb-4{padding-bottom:calc(var(--spacing) * 4)}.pl-1{padding-left:var(--spacing)}.pl-8{padding-left:calc(var(--spacing) * 8)}.text-center{text-align:center}.text-left{text-align:left}.text-right{text-align:right}.align-text-bottom{vertical-align:text-bottom}.font-mono{font-family:var(--font-mono)}.text-\[10px\]{font-size:10px}.text-\[11\.5px\]{font-size:11.5px}.text-\[11px\]{font-size:11px}.text-\[12\.5px\]{font-size:12.5px}.text-\[12px\]{font-size:12px}.text-\[13\.5px\]{font-size:13.5px}.text-\[13px\]{font-size:13px}.text-\[14px\]{font-size:14px}.text-\[15px\]{font-size:15px}.text-\[16px\]{font-size:16px}.text-\[19px\]{font-size:19px}.text-\[22px\]{font-size:22px}.leading-none{--tw-leading:1;line-height:1}.leading-relaxed{--tw-leading:var(--leading-relaxed);line-height:var(--leading-relaxed)}.font-bold{--tw-font-weight:var(--font-weight-bold);font-weight:var(--font-weight-bold)}.font-medium{--tw-font-weight:var(--font-weight-medium);font-weight:var(--font-weight-medium)}.font-semibold{--tw-font-weight:var(--font-weight-semibold);font-weight:var(--font-weight-semibold)}.tracking-normal{--tw-tracking:var(--tracking-normal);letter-spacing:var(--tracking-normal)}.tracking-tight{--tw-tracking:var(--tracking-tight);letter-spacing:var(--tracking-tight)}.tracking-wider{--tw-tracking:var(--tracking-wider);letter-spacing:var(--tracking-wider)}.break-all{word-break:break-all}.whitespace-nowrap{white-space:nowrap}.whitespace-pre-wrap{white-space:pre-wrap}.text-accent{color:var(--color-accent)}.text-accent-ink{color:var(--color-accent-ink)}.text-danger{color:var(--color-danger)}.text-info{color:var(--color-info)}.text-success{color:var(--color-success)}.text-text{color:var(--color-text)}.text-text-muted{color:var(--color-text-muted)}.text-text-subtle{color:var(--color-text-subtle)}.text-text-subtle\/70{color:#5c5c56b3}@supports (color:color-mix(in lab, red, red)){.text-text-subtle\/70{color:color-mix(in oklab, var(--color-text-subtle) 70%, transparent)}}.text-transparent{color:#0000}.text-warning{color:var(--color-warning)}.capitalize{text-transform:capitalize}.lowercase{text-transform:lowercase}.uppercase{text-transform:uppercase}.opacity-0{opacity:0}.opacity-40{opacity:.4}.opacity-50{opacity:.5}.opacity-60{opacity:.6}.opacity-70{opacity:.7}.shadow-elevated{--tw-shadow:0 16px 40px -8px var(--tw-shadow-color,#0000008c);box-shadow:var(--tw-inset-shadow), var(--tw-inset-ring-shadow), var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow)}.shadow-soft{--tw-shadow:0 1px 2px 0 var(--tw-shadow-color,#0006);box-shadow:var(--tw-inset-shadow), var(--tw-inset-ring-shadow), var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow)}.shadow-warm{--tw-shadow:0 4px 16px -4px var(--tw-shadow-color,#cbe85a2e);box-shadow:var(--tw-inset-shadow), var(--tw-inset-ring-shadow), var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow)}.filter{filter:var(--tw-blur,) var(--tw-brightness,) var(--tw-contrast,) var(--tw-grayscale,) var(--tw-hue-rotate,) var(--tw-invert,) var(--tw-saturate,) var(--tw-sepia,) var(--tw-drop-shadow,)}.transition-all{transition-property:all;transition-timing-function:var(--tw-ease,var(--default-transition-timing-function));transition-duration:var(--tw-duration,var(--default-transition-duration))}.transition-colors{transition-property:color,background-color,border-color,outline-color,text-decoration-color,fill,stroke,--tw-gradient-from,--tw-gradient-via,--tw-gradient-to;transition-timing-function:var(--tw-ease,var(--default-transition-timing-function));transition-duration:var(--tw-duration,var(--default-transition-duration))}.transition-opacity{transition-property:opacity;transition-timing-function:var(--tw-ease,var(--default-transition-timing-function));transition-duration:var(--tw-duration,var(--default-transition-duration))}.transition-transform{transition-property:transform,translate,scale,rotate;transition-timing-function:var(--tw-ease,var(--default-transition-timing-function));transition-duration:var(--tw-duration,var(--default-transition-duration))}.duration-150{--tw-duration:.15s;transition-duration:.15s}.outline-none{--tw-outline-style:none;outline-style:none}.select-all{-webkit-user-select:all;user-select:all}.group-open\:rotate-180:is(:where(.group):is([open],:popover-open,:open) *),.group-open\/p\:rotate-180:is(:where(.group\/p):is([open],:popover-open,:open) *){rotate:180deg}@media (hover:hover){.group-hover\:bg-text-subtle:is(:where(.group):hover *){background-color:var(--color-text-subtle)}.group-hover\:text-accent:is(:where(.group):hover *){color:var(--color-accent)}.group-hover\:text-text:is(:where(.group):hover *){color:var(--color-text)}.group-hover\:opacity-100:is(:where(.group):hover *){opacity:1}}.placeholder\:text-text-subtle::placeholder{color:var(--color-text-subtle)}.placeholder\:text-text-subtle\/60::placeholder{color:#5c5c5699}@supports (color:color-mix(in lab, red, red)){.placeholder\:text-text-subtle\/60::placeholder{color:color-mix(in oklab, var(--color-text-subtle) 60%, transparent)}}.last\:border-b-0:last-child{border-bottom-style:var(--tw-border-style);border-bottom-width:0}@media (hover:hover){.hover\:border-accent:hover{border-color:var(--color-accent)}.hover\:border-border:hover{border-color:var(--color-border)}.hover\:border-danger\/40:hover{border-color:#e8806a66}@supports (color:color-mix(in lab, red, red)){.hover\:border-danger\/40:hover{border-color:color-mix(in oklab, var(--color-danger) 40%, transparent)}}.hover\:border-danger\/50:hover{border-color:#e8806a80}@supports (color:color-mix(in lab, red, red)){.hover\:border-danger\/50:hover{border-color:color-mix(in oklab, var(--color-danger) 50%, transparent)}}.hover\:border-danger\/60:hover{border-color:#e8806a99}@supports (color:color-mix(in lab, red, red)){.hover\:border-danger\/60:hover{border-color:color-mix(in oklab, var(--color-danger) 60%, transparent)}}.hover\:border-text-subtle:hover{border-color:var(--color-text-subtle)}.hover\:bg-accent-hover:hover{background-color:var(--color-accent-hover)}.hover\:bg-accent-soft:hover{background-color:var(--color-accent-soft)}.hover\:bg-bg:hover{background-color:var(--color-bg)}.hover\:bg-surface:hover{background-color:var(--color-surface)}.hover\:bg-surface-2:hover{background-color:var(--color-surface-2)}.hover\:bg-surface-2\/70:hover{background-color:#222220b3}@supports (color:color-mix(in lab, red, red)){.hover\:bg-surface-2\/70:hover{background-color:color-mix(in oklab, var(--color-surface-2) 70%, transparent)}}.hover\:bg-warning\/20:hover{background-color:#e8c55a33}@supports (color:color-mix(in lab, red, red)){.hover\:bg-warning\/20:hover{background-color:color-mix(in oklab, var(--color-warning) 20%, transparent)}}.hover\:text-accent:hover{color:var(--color-accent)}.hover\:text-danger:hover{color:var(--color-danger)}.hover\:text-text:hover{color:var(--color-text)}.hover\:underline:hover{text-decoration-line:underline}.hover\:opacity-100:hover{opacity:1}}.focus\:border-accent:focus{border-color:var(--color-accent)}.focus\:outline-none:focus{--tw-outline-style:none;outline-style:none}.active\:cursor-grabbing:active{cursor:grabbing}.disabled\:cursor-not-allowed:disabled{cursor:not-allowed}.disabled\:opacity-30:disabled{opacity:.3}.disabled\:opacity-40:disabled{opacity:.4}.disabled\:opacity-45:disabled{opacity:.45}.disabled\:opacity-50:disabled{opacity:.5}.disabled\:opacity-60:disabled{opacity:.6}@media (hover:hover){.disabled\:hover\:border-border:disabled:hover{border-color:var(--color-border)}.disabled\:hover\:text-text-muted:disabled:hover{color:var(--color-text-muted)}}@media (min-width:40rem){.sm\:grid-cols-2{grid-template-columns:repeat(2,minmax(0,1fr))}.sm\:grid-cols-3{grid-template-columns:repeat(3,minmax(0,1fr))}.sm\:grid-cols-4{grid-template-columns:repeat(4,minmax(0,1fr))}.sm\:p-10{padding:calc(var(--spacing) * 10)}}@media (min-width:64rem){.lg\:col-span-2{grid-column:span 2/span 2}.lg\:grid-cols-2{grid-template-columns:repeat(2,minmax(0,1fr))}.lg\:grid-cols-3{grid-template-columns:repeat(3,minmax(0,1fr))}}.\[\&\:\:-webkit-details-marker\]\:hidden::-webkit-details-marker{display:none}}*{box-sizing:border-box}html.light{--color-bg:#fafaf9;--color-bg-alt:#f5f5f3;--color-surface:#fff;--color-surface-2:#f0f0ee;--color-surface-3:#e8e8e5;--color-sidebar:#f7f7f6;--color-border:#e0e0dc;--color-border-subtle:#eaeae7;--color-text:#1a1a18;--color-text-muted:#6b6b65;--color-text-subtle:#9b9b94;--color-accent:#6b8f20;--color-accent-hover:#7ba025;--color-accent-ink:#fff;--color-accent-soft:#eef5d8;--shadow-soft:0 1px 3px 0 #00000014;--shadow-warm:0 4px 16px -4px #6b8f201f;--shadow-elevated:0 16px 40px -8px #0000001f}html,body{background:var(--color-bg);color:var(--color-text);font-family:var(--font-sans);-webkit-font-smoothing:antialiased;text-rendering:optimizelegibility;margin:0;padding:0}.material-symbols-outlined{font-variation-settings:"FILL" 0, "wght" 400, "GRAD" 0, "opsz" 24;-webkit-user-select:none;user-select:none;font-size:20px;line-height:1}.tnum{font-family:var(--font-mono);font-variant-numeric:tabular-nums;font-feature-settings:"tnum"}a{color:inherit;text-decoration:none}a:focus-visible,button:focus-visible,input:focus-visible,textarea:focus-visible,select:focus-visible{outline:2px solid var(--color-accent);outline-offset:2px;border-radius:6px}.dark select option{background-color:var(--color-surface);color:var(--color-text)}html.light select option{color:#1a1a18;background-color:#fff}html.light .rail-icon-active{color:#fff;background:#1a1a18}html.light .rail-icon-active:hover{color:#fff;background:#333330}.lamp{border-radius:50%;flex:none;width:8px;height:8px;display:inline-block}.lamp-live{background:var(--color-success);box-shadow:0 0 6px 1px #7dd87f99}@supports (color:color-mix(in lab, red, red)){.lamp-live{box-shadow:0 0 6px 1px color-mix(in srgb, var(--color-success) 60%, transparent)}}.lamp-idle{background:var(--color-text-subtle)}.lamp-down{background:var(--color-danger);box-shadow:0 0 6px 1px #e8806a99}@supports (color:color-mix(in lab, red, red)){.lamp-down{box-shadow:0 0 6px 1px color-mix(in srgb, var(--color-danger) 60%, transparent)}}.lamp-down{animation:1.4s ease-in-out infinite pulse}@keyframes pulse{50%{opacity:.5}}@media (prefers-reduced-motion:reduce){.lamp-down{animation:none}}.console-grid{min-height:100vh;padding-left:108px}.console-rail{z-index:30;background:var(--color-sidebar);border:1px solid var(--color-border-subtle);width:76px;box-shadow:var(--shadow-soft);border-radius:24px;flex-direction:column;align-items:center;gap:1rem;padding:1.25rem 0;display:flex;position:fixed;top:50%;left:32px;transform:translateY(-50%)}.rail-divider{background:var(--color-border);flex:none;width:26px;height:1px;margin:.2rem 0 .4rem}.console-col{flex-direction:column;min-width:0;display:flex}.console-topbar{z-index:20;background:var(--color-bg);align-items:center;gap:1rem;height:56px;padding:0 1.5rem;display:flex;position:sticky;top:0}.console-main{zoom:1.06;width:100%;min-width:0;max-width:1600px;margin:0 auto;padding:2rem 3rem 4rem}.rail-icon{background:var(--color-surface);width:42px;height:42px;color:var(--color-text-subtle);border-radius:50%;justify-content:center;align-items:center;transition:color .15s,background .15s;display:flex;position:relative}.rail-icon:hover{color:var(--color-text);background:var(--color-surface-3)}.rail-icon-active{color:var(--color-bg);background:var(--color-text)}.rail-icon-active:hover{color:var(--color-bg);background:#fff}.rail-brand{background:var(--color-accent);width:42px;height:42px;color:var(--color-accent-ink);box-shadow:var(--shadow-warm);border-radius:13px;flex:none;justify-content:center;align-items:center;font-size:17px;font-weight:700;display:flex}.rail-icon[data-label]:after{content:attr(data-label);white-space:nowrap;background:var(--color-surface-3);color:var(--color-text);opacity:0;pointer-events:none;z-index:40;box-shadow:var(--shadow-soft);border-radius:7px;padding:4px 9px;font-size:12px;transition:opacity .12s;position:absolute;top:50%;left:calc(100% + 12px);transform:translateY(-50%)}.rail-icon:hover[data-label]:after{opacity:1}.kpi{background:var(--color-surface);border:1px solid var(--color-border);white-space:nowrap;border-radius:999px;align-items:center;gap:7px;padding:5px 11px;font-size:12px;display:flex}.kpi-k{color:var(--color-text-subtle)}.kpi-v{color:var(--color-text);font-family:var(--font-mono);font-variant-numeric:tabular-nums}@media (max-width:760px){.console-grid{grid-template-columns:72px 1fr}.console-main{padding:1.25rem 1.1rem 3rem}.console-topbar{padding:0 1rem}}.table-wrap{overflow-x:auto}button:not(:disabled),[role=button]:not([aria-disabled=true]){cursor:pointer}label:has(>input[type=checkbox]:not(:disabled)){cursor:pointer}button:disabled{cursor:not-allowed}::selection{background:#cbe85a4d}@supports (color:color-mix(in lab, red, red)){::selection{background:color-mix(in srgb, var(--color-accent) 30%, transparent)}}@keyframes slideIn{0%{opacity:0;transform:translate(20px)}to{opacity:1;transform:translate(0)}}input[type=range].range-accent{appearance:none;background:var(--color-surface-2);border-radius:9999px;outline:none;height:6px}input[type=range].range-accent::-webkit-slider-thumb{appearance:none;background:var(--color-accent);border:2px solid var(--color-surface);width:16px;height:16px;box-shadow:0 0 0 1px var(--color-accent), 0 0 8px #cbe85a8c;border-radius:9999px}@supports (color:color-mix(in lab, red, red)){input[type=range].range-accent::-webkit-slider-thumb{box-shadow:0 0 0 1px var(--color-accent), 0 0 8px color-mix(in srgb, var(--color-accent) 55%, transparent)}}input[type=range].range-accent::-webkit-slider-thumb{cursor:pointer;transition:box-shadow .15s}input[type=range].range-accent:hover::-webkit-slider-thumb{box-shadow:0 0 0 1px var(--color-accent), 0 0 12px #cbe85ad9}@supports (color:color-mix(in lab, red, red)){input[type=range].range-accent:hover::-webkit-slider-thumb{box-shadow:0 0 0 1px var(--color-accent), 0 0 12px color-mix(in srgb, var(--color-accent) 85%, transparent)}}input[type=range].range-accent:focus-visible::-webkit-slider-thumb{box-shadow:0 0 0 1px var(--color-accent), 0 0 12px #cbe85ad9}@supports (color:color-mix(in lab, red, red)){input[type=range].range-accent:focus-visible::-webkit-slider-thumb{box-shadow:0 0 0 1px var(--color-accent), 0 0 12px color-mix(in srgb, var(--color-accent) 85%, transparent)}}input[type=range].range-accent::-moz-range-track{background:var(--color-surface-2);border-radius:9999px;height:6px}input[type=range].range-accent::-moz-range-progress{background:var(--color-accent);border-radius:9999px;height:6px}input[type=range].range-accent::-moz-range-thumb{border:2px solid var(--color-surface);background:var(--color-accent);width:16px;height:16px;box-shadow:0 0 0 1px var(--color-accent), 0 0 8px #cbe85a8c;border-radius:9999px}@supports (color:color-mix(in lab, red, red)){input[type=range].range-accent::-moz-range-thumb{box-shadow:0 0 0 1px var(--color-accent), 0 0 8px color-mix(in srgb, var(--color-accent) 55%, transparent)}}input[type=range].range-accent::-moz-range-thumb{cursor:pointer}@property --tw-translate-x{syntax:"*";inherits:false;initial-value:0}@property --tw-translate-y{syntax:"*";inherits:false;initial-value:0}@property --tw-translate-z{syntax:"*";inherits:false;initial-value:0}@property --tw-rotate-x{syntax:"*";inherits:false}@property --tw-rotate-y{syntax:"*";inherits:false}@property --tw-rotate-z{syntax:"*";inherits:false}@property --tw-skew-x{syntax:"*";inherits:false}@property --tw-skew-y{syntax:"*";inherits:false}@property --tw-space-y-reverse{syntax:"*";inherits:false;initial-value:0}@property --tw-divide-y-reverse{syntax:"*";inherits:false;initial-value:0}@property --tw-border-style{syntax:"*";inherits:false;initial-value:solid}@property --tw-leading{syntax:"*";inherits:false}@property --tw-font-weight{syntax:"*";inherits:false}@property --tw-tracking{syntax:"*";inherits:false}@property --tw-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-shadow-color{syntax:"*";inherits:false}@property --tw-shadow-alpha{syntax:"<percentage>";inherits:false;initial-value:100%}@property --tw-inset-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-inset-shadow-color{syntax:"*";inherits:false}@property --tw-inset-shadow-alpha{syntax:"<percentage>";inherits:false;initial-value:100%}@property --tw-ring-color{syntax:"*";inherits:false}@property --tw-ring-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-inset-ring-color{syntax:"*";inherits:false}@property --tw-inset-ring-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-ring-inset{syntax:"*";inherits:false}@property --tw-ring-offset-width{syntax:"<length>";inherits:false;initial-value:0}@property --tw-ring-offset-color{syntax:"*";inherits:false;initial-value:#fff}@property --tw-ring-offset-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-blur{syntax:"*";inherits:false}@property --tw-brightness{syntax:"*";inherits:false}@property --tw-contrast{syntax:"*";inherits:false}@property --tw-grayscale{syntax:"*";inherits:false}@property --tw-hue-rotate{syntax:"*";inherits:false}@property --tw-invert{syntax:"*";inherits:false}@property --tw-opacity{syntax:"*";inherits:false}@property --tw-saturate{syntax:"*";inherits:false}@property --tw-sepia{syntax:"*";inherits:false}@property --tw-drop-shadow{syntax:"*";inherits:false}@property --tw-drop-shadow-color{syntax:"*";inherits:false}@property --tw-drop-shadow-alpha{syntax:"<percentage>";inherits:false;initial-value:100%}@property --tw-drop-shadow-size{syntax:"*";inherits:false}@property --tw-duration{syntax:"*";inherits:false}@keyframes spin{to{transform:rotate(360deg)}}
|
/package/dashboard/.next/static/{jKcczqYA93-Kuqa9A9H9h → 3ctBU8K8tClTVQCiGy65u}/_buildManifest.js
RENAMED
|
File without changes
|
|
File without changes
|
/package/dashboard/.next/static/{jKcczqYA93-Kuqa9A9H9h → 3ctBU8K8tClTVQCiGy65u}/_ssgManifest.js
RENAMED
|
File without changes
|