@curatedmcp/tokenshield 0.2.0

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.
Files changed (55) hide show
  1. package/dist/cli.d.ts +2 -0
  2. package/dist/cli.js +238 -0
  3. package/dist/cli.js.map +1 -0
  4. package/dist/commands/bench.d.ts +5 -0
  5. package/dist/commands/bench.js +191 -0
  6. package/dist/commands/bench.js.map +1 -0
  7. package/dist/commands/demo.d.ts +3 -0
  8. package/dist/commands/demo.js +99 -0
  9. package/dist/commands/demo.js.map +1 -0
  10. package/dist/commands/doctor.d.ts +1 -0
  11. package/dist/commands/doctor.js +119 -0
  12. package/dist/commands/doctor.js.map +1 -0
  13. package/dist/commands/estimate.d.ts +3 -0
  14. package/dist/commands/estimate.js +90 -0
  15. package/dist/commands/estimate.js.map +1 -0
  16. package/dist/commands/integrations.d.ts +14 -0
  17. package/dist/commands/integrations.js +138 -0
  18. package/dist/commands/integrations.js.map +1 -0
  19. package/dist/commands/logs.d.ts +6 -0
  20. package/dist/commands/logs.js +66 -0
  21. package/dist/commands/logs.js.map +1 -0
  22. package/dist/commands/setup.d.ts +9 -0
  23. package/dist/commands/setup.js +159 -0
  24. package/dist/commands/setup.js.map +1 -0
  25. package/dist/commands/status.d.ts +1 -0
  26. package/dist/commands/status.js +102 -0
  27. package/dist/commands/status.js.map +1 -0
  28. package/dist/commands/stop.d.ts +1 -0
  29. package/dist/commands/stop.js +27 -0
  30. package/dist/commands/stop.js.map +1 -0
  31. package/dist/commands/up.d.ts +15 -0
  32. package/dist/commands/up.js +145 -0
  33. package/dist/commands/up.js.map +1 -0
  34. package/dist/dashboard.d.ts +4 -0
  35. package/dist/dashboard.js +186 -0
  36. package/dist/dashboard.js.map +1 -0
  37. package/dist/lib/daemon.d.ts +23 -0
  38. package/dist/lib/daemon.js +164 -0
  39. package/dist/lib/daemon.js.map +1 -0
  40. package/dist/lib/errors.d.ts +23 -0
  41. package/dist/lib/errors.js +131 -0
  42. package/dist/lib/errors.js.map +1 -0
  43. package/dist/lib/integrations.d.ts +47 -0
  44. package/dist/lib/integrations.js +210 -0
  45. package/dist/lib/integrations.js.map +1 -0
  46. package/dist/lib/paths.d.ts +5 -0
  47. package/dist/lib/paths.js +18 -0
  48. package/dist/lib/paths.js.map +1 -0
  49. package/dist/lib/preflight.d.ts +20 -0
  50. package/dist/lib/preflight.js +97 -0
  51. package/dist/lib/preflight.js.map +1 -0
  52. package/dist/lib/ui.d.ts +68 -0
  53. package/dist/lib/ui.js +221 -0
  54. package/dist/lib/ui.js.map +1 -0
  55. package/package.json +44 -0
@@ -0,0 +1 @@
1
+ {"version":3,"file":"setup.js","sourceRoot":"","sources":["../../src/commands/setup.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1 @@
1
+ export declare function runStatus(): Promise<void>;
@@ -0,0 +1,102 @@
1
+ import { request as httpRequest } from "node:http";
2
+ import { readDaemon } from "../lib/daemon.js";
3
+ import { Ledger, defaultConfig } from "@curatedmcp/tokenshield-core";
4
+ import { c, sym, dim, emit, emitJson, isJson, table, heading, say } from "../lib/ui.js";
5
+ function fmtDollars(n) {
6
+ if (n === 0)
7
+ return "$0.00";
8
+ return "$" + n.toFixed(n < 1 ? 4 : 2);
9
+ }
10
+ function fmtN(n) { return n.toLocaleString(); }
11
+ function fmtUptime(ms) {
12
+ const s = Math.floor(ms / 1000);
13
+ if (s < 60)
14
+ return `${s}s`;
15
+ if (s < 3600)
16
+ return `${Math.floor(s / 60)}m ${s % 60}s`;
17
+ if (s < 86400)
18
+ return `${Math.floor(s / 3600)}h ${Math.floor((s % 3600) / 60)}m`;
19
+ return `${Math.floor(s / 86400)}d ${Math.floor((s % 86400) / 3600)}h`;
20
+ }
21
+ async function checkProxyHealth(host, port) {
22
+ return new Promise((resolve) => {
23
+ const req = httpRequest({ host, port, path: "/__tokenshield/health", method: "GET", timeout: 1500 }, (res) => {
24
+ const chunks = [];
25
+ res.on("data", (c) => chunks.push(c));
26
+ res.on("end", () => {
27
+ try {
28
+ const body = JSON.parse(Buffer.concat(chunks).toString("utf8"));
29
+ resolve({ ok: body.ok === true, status: res.statusCode ?? 0 });
30
+ }
31
+ catch {
32
+ resolve({ ok: false, status: res.statusCode ?? 0, detail: "non-JSON response" });
33
+ }
34
+ });
35
+ });
36
+ req.on("error", (err) => resolve({ ok: false, detail: err.message }));
37
+ req.on("timeout", () => { req.destroy(); resolve({ ok: false, detail: "timed out" }); });
38
+ req.end();
39
+ });
40
+ }
41
+ export async function runStatus() {
42
+ const cfg = defaultConfig();
43
+ const info = readDaemon();
44
+ let summary = null;
45
+ try {
46
+ const ledger = new Ledger(cfg.ledgerPath);
47
+ summary = ledger.summary(Date.now() - 24 * 60 * 60 * 1000);
48
+ ledger.close();
49
+ }
50
+ catch {
51
+ summary = null;
52
+ }
53
+ let health = null;
54
+ if (info !== null) {
55
+ health = await checkProxyHealth(info.bind, info.port);
56
+ }
57
+ if (isJson()) {
58
+ emitJson({
59
+ ok: true,
60
+ daemon: info,
61
+ health,
62
+ ledger: summary,
63
+ });
64
+ return;
65
+ }
66
+ heading("TokenShield status");
67
+ if (info === null) {
68
+ emit(` ${sym.dot} Proxy ${c.gray("not running")}`);
69
+ emit(dim(" Start with: tokenshield up (or `tokenshield up --daemon` for background)"));
70
+ }
71
+ else {
72
+ const aliveSym = health?.ok ? sym.check : sym.warn;
73
+ const aliveLabel = health?.ok ? "running" : `running but not responding ${health?.detail ?? ""}`;
74
+ emit(` ${aliveSym} Proxy ${aliveLabel} ${dim(`pid ${info.pid}`)}`);
75
+ emit(` ${sym.bullet} URL http://${info.bind}:${info.port}`);
76
+ emit(` ${sym.bullet} Dashboard http://${info.bind}:${info.dashboardPort}`);
77
+ emit(` ${sym.bullet} Uptime ${fmtUptime(Date.now() - info.startedAt)}`);
78
+ }
79
+ emit("");
80
+ if (summary === null) {
81
+ emit(dim(" (no ledger yet)"));
82
+ return;
83
+ }
84
+ emit(c.bold(" Last 24 hours"));
85
+ emit("");
86
+ emit(` ${sym.bullet} Requests ${fmtN(summary.requestCount)}`);
87
+ emit(` ${sym.bullet} Spent ${fmtDollars(summary.dollarsRaw)}`);
88
+ emit(` ${sym.bullet} Input tokens ${fmtN(summary.totalInputTokensRaw)}`);
89
+ emit(` ${sym.bullet} Output tokens ${fmtN(summary.totalOutputTokensRaw)}`);
90
+ if (summary.byModel.length > 0) {
91
+ say("");
92
+ say(table(summary.byModel.map((m) => [
93
+ " " + m.model,
94
+ fmtN(m.requests),
95
+ fmtN(m.inputTokens),
96
+ fmtN(m.outputTokens),
97
+ fmtDollars(m.dollars),
98
+ ]), { header: [" Model", "Reqs", "Input", "Output", "$"] }));
99
+ }
100
+ emit("");
101
+ }
102
+ //# sourceMappingURL=status.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"status.js","sourceRoot":"","sources":["../../src/commands/status.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1 @@
1
+ export declare function runStop(): Promise<void>;
@@ -0,0 +1,27 @@
1
+ import { stopDaemon, readDaemon } from "../lib/daemon.js";
2
+ import { c, sym, emit, emitJson, isJson, dim } from "../lib/ui.js";
3
+ import { TokenShieldError } from "../lib/errors.js";
4
+ export async function runStop() {
5
+ const info = readDaemon();
6
+ if (info === null) {
7
+ if (isJson()) {
8
+ emitJson({ ok: true, stopped: null });
9
+ return;
10
+ }
11
+ throw new TokenShieldError({
12
+ code: "DAEMON_NOT_RUNNING",
13
+ message: "No TokenShield daemon is running.",
14
+ hint: "Nothing to stop. Start one with `tokenshield up --daemon`.",
15
+ });
16
+ }
17
+ const stopped = await stopDaemon();
18
+ if (isJson()) {
19
+ emitJson({ ok: true, stopped });
20
+ return;
21
+ }
22
+ emit(`${sym.check} Stopped TokenShield ${dim(`(pid ${stopped?.pid ?? info.pid})`)}`);
23
+ emit(dim(` proxy was on port ${stopped?.port ?? info.port}, dashboard ${stopped?.dashboardPort ?? info.dashboardPort}`));
24
+ // Silence unused-symbol lint
25
+ void c;
26
+ }
27
+ //# sourceMappingURL=stop.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"stop.js","sourceRoot":"","sources":["../../src/commands/stop.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,UAAU,EAAE,MAAM,kBAAkB,CAAC;AAC1D,OAAO,EAAE,CAAC,EAAE,GAAG,EAAE,IAAI,EAAE,QAAQ,EAAE,MAAM,EAAE,GAAG,EAAE,MAAM,cAAc,CAAC;AACnE,OAAO,EAAE,gBAAgB,EAAE,MAAM,kBAAkB,CAAC;AAEpD,MAAM,CAAC,KAAK,UAAU,OAAO;IAC3B,MAAM,IAAI,GAAG,UAAU,EAAE,CAAC;IAC1B,IAAI,IAAI,KAAK,IAAI,EAAE,CAAC;QAClB,IAAI,MAAM,EAAE,EAAE,CAAC;YACb,QAAQ,CAAC,EAAE,EAAE,EAAE,IAAI,EAAE,OAAO,EAAE,IAAI,EAAE,CAAC,CAAC;YACtC,OAAO;QACT,CAAC;QACD,MAAM,IAAI,gBAAgB,CAAC;YACzB,IAAI,EAAE,oBAAoB;YAC1B,OAAO,EAAE,mCAAmC;YAC5C,IAAI,EAAE,4DAA4D;SACnE,CAAC,CAAC;IACL,CAAC;IACD,MAAM,OAAO,GAAG,MAAM,UAAU,EAAE,CAAC;IACnC,IAAI,MAAM,EAAE,EAAE,CAAC;QACb,QAAQ,CAAC,EAAE,EAAE,EAAE,IAAI,EAAE,OAAO,EAAE,CAAC,CAAC;QAChC,OAAO;IACT,CAAC;IACD,IAAI,CAAC,GAAG,GAAG,CAAC,KAAK,wBAAwB,GAAG,CAAC,QAAQ,OAAO,EAAE,GAAG,IAAI,IAAI,CAAC,GAAG,GAAG,CAAC,EAAE,CAAC,CAAC;IACrF,IAAI,CAAC,GAAG,CAAC,uBAAuB,OAAO,EAAE,IAAI,IAAI,IAAI,CAAC,IAAI,eAAe,OAAO,EAAE,aAAa,IAAI,IAAI,CAAC,aAAa,EAAE,CAAC,CAAC,CAAC;IAC1H,6BAA6B;IAC7B,KAAK,CAAC,CAAC;AACT,CAAC"}
@@ -0,0 +1,15 @@
1
+ export interface UpOptions {
2
+ port: number;
3
+ dashboardPort: number;
4
+ bind: string;
5
+ upstream: string;
6
+ ledger?: string;
7
+ retentionDays: number;
8
+ daemon: boolean;
9
+ }
10
+ export declare function runUp(options: UpOptions): Promise<void>;
11
+ /**
12
+ * Internal: the entrypoint a detached daemon process invokes.
13
+ * Does NOT print the banner — output goes to ~/.tokenshield/proxy.log.
14
+ */
15
+ export declare function runSupervised(options: UpOptions): Promise<void>;
@@ -0,0 +1,145 @@
1
+ import { start, defaultConfig } from "@curatedmcp/tokenshield-core";
2
+ import { dashboardHtml } from "../dashboard.js";
3
+ import { c, sym, box, link, say, emit, dim } from "../lib/ui.js";
4
+ import { TokenShieldError } from "../lib/errors.js";
5
+ import { requirePortFree, classifyApiKey } from "../lib/preflight.js";
6
+ import { readDaemon, spawnDaemon } from "../lib/daemon.js";
7
+ import { logFile } from "../lib/paths.js";
8
+ function banner(config, opts) {
9
+ const proxyUrl = `http://${config.bind}:${config.port}`;
10
+ const dashUrl = `http://${config.bind}:${config.dashboardPort}`;
11
+ const lines = [];
12
+ lines.push(c.brightGreen("TokenShield is live") + dim(opts.daemon ? " (daemon mode)" : ""));
13
+ lines.push("");
14
+ lines.push(`${sym.dot} Proxy ${link(proxyUrl, proxyUrl)}`);
15
+ lines.push(`${sym.dot} Dashboard ${link(dashUrl, dashUrl)}`);
16
+ lines.push(`${sym.dot} Upstream ${dim(config.upstreamBaseUrl)}`);
17
+ lines.push(`${sym.dot} Ledger ${dim(config.ledgerPath)}`);
18
+ if (opts.logPath) {
19
+ lines.push(`${sym.dot} Log ${dim(opts.logPath)}`);
20
+ }
21
+ lines.push("");
22
+ lines.push(c.bold("Point Claude Code at the proxy:"));
23
+ lines.push(` ${c.cyan("export ANTHROPIC_BASE_URL=" + proxyUrl)}`);
24
+ lines.push("");
25
+ lines.push(dim("Your ANTHROPIC_API_KEY never leaves this machine."));
26
+ if (!opts.daemon) {
27
+ lines.push(dim("Press Ctrl-C to stop."));
28
+ }
29
+ else {
30
+ lines.push(dim("Run `tokenshield stop` when you're done."));
31
+ }
32
+ return box(lines.join("\n"), { color: c.gray, padding: 2 });
33
+ }
34
+ function maybeWarnAboutLocalKey() {
35
+ const key = process.env["ANTHROPIC_API_KEY"];
36
+ const cls = classifyApiKey(key);
37
+ if (cls.state === "missing") {
38
+ // Not fatal — the user runs Claude Code in a DIFFERENT shell. Just inform.
39
+ say("");
40
+ say(`${sym.info} ANTHROPIC_API_KEY is not set in this shell.`);
41
+ say(dim(" That's fine — set it in the shell that runs Claude Code, not here."));
42
+ }
43
+ else if (cls.state === "wrong_prefix") {
44
+ say("");
45
+ say(`${sym.warn} ANTHROPIC_API_KEY in this shell doesn't look like an Anthropic key.`);
46
+ say(dim(` ${cls.hint}`));
47
+ }
48
+ }
49
+ export async function runUp(options) {
50
+ const config = defaultConfig({
51
+ port: options.port,
52
+ dashboardPort: options.dashboardPort,
53
+ bind: options.bind,
54
+ upstreamBaseUrl: options.upstream,
55
+ ...(options.ledger ? { ledgerPath: options.ledger } : {}),
56
+ retentionDays: options.retentionDays,
57
+ });
58
+ // Preflight: don't fail mysteriously inside server.listen()
59
+ if (config.bind === "127.0.0.1" || config.bind === "localhost") {
60
+ await requirePortFree(config.port, "proxy");
61
+ await requirePortFree(config.dashboardPort, "dashboard");
62
+ }
63
+ // Bind-other warning
64
+ if (config.bind !== "127.0.0.1" && config.bind !== "localhost") {
65
+ say("");
66
+ say(`${sym.warn} Binding to ${c.bold(config.bind)} — proxy is reachable beyond localhost.`);
67
+ say(dim(" Only do this on a trusted network. Starting in 3 seconds (Ctrl-C to abort)."));
68
+ await new Promise((r) => setTimeout(r, 3000));
69
+ }
70
+ if (options.daemon) {
71
+ const existing = readDaemon();
72
+ if (existing !== null) {
73
+ throw new TokenShieldError({
74
+ code: "DAEMON_ALREADY_RUNNING",
75
+ message: `TokenShield is already running (pid ${existing.pid}, port ${existing.port}).`,
76
+ nextSteps: ["tokenshield status", "tokenshield stop"],
77
+ });
78
+ }
79
+ const info = await spawnDaemon({
80
+ port: config.port,
81
+ dashboardPort: config.dashboardPort,
82
+ bind: config.bind,
83
+ upstream: config.upstreamBaseUrl,
84
+ ledger: config.ledgerPath,
85
+ retentionDays: config.retentionDays,
86
+ });
87
+ emit(banner(config, { daemon: true, logPath: logFile() }));
88
+ say("");
89
+ say(`${sym.check} Started as daemon ${dim(`(pid ${info.pid})`)}.`);
90
+ return;
91
+ }
92
+ // Foreground mode
93
+ const handle = await start({
94
+ config,
95
+ renderDashboard: () => dashboardHtml({ proxyPort: config.port, bind: config.bind }),
96
+ });
97
+ emit(banner(config, { daemon: false }));
98
+ maybeWarnAboutLocalKey();
99
+ let shuttingDown = false;
100
+ const shutdown = (signal) => {
101
+ if (shuttingDown)
102
+ return;
103
+ shuttingDown = true;
104
+ say("");
105
+ say(`${sym.arrow} Caught ${signal}, shutting down…`);
106
+ handle
107
+ .close()
108
+ .then(() => {
109
+ say(`${sym.check} Stopped cleanly.`);
110
+ process.exit(0);
111
+ })
112
+ .catch((err) => {
113
+ say(`${sym.cross} shutdown error: ${err.message}`);
114
+ process.exit(1);
115
+ });
116
+ };
117
+ process.on("SIGINT", () => shutdown("SIGINT"));
118
+ process.on("SIGTERM", () => shutdown("SIGTERM"));
119
+ }
120
+ /**
121
+ * Internal: the entrypoint a detached daemon process invokes.
122
+ * Does NOT print the banner — output goes to ~/.tokenshield/proxy.log.
123
+ */
124
+ export async function runSupervised(options) {
125
+ const config = defaultConfig({
126
+ port: options.port,
127
+ dashboardPort: options.dashboardPort,
128
+ bind: options.bind,
129
+ upstreamBaseUrl: options.upstream,
130
+ ...(options.ledger ? { ledgerPath: options.ledger } : {}),
131
+ retentionDays: options.retentionDays,
132
+ });
133
+ const handle = await start({
134
+ config,
135
+ renderDashboard: () => dashboardHtml({ proxyPort: config.port, bind: config.bind }),
136
+ });
137
+ const shutdown = async (signal) => {
138
+ process.stderr.write(`[tokenshield] caught ${signal}, shutting down\n`);
139
+ await handle.close();
140
+ process.exit(0);
141
+ };
142
+ process.on("SIGTERM", () => { void shutdown("SIGTERM"); });
143
+ process.on("SIGINT", () => { void shutdown("SIGINT"); });
144
+ }
145
+ //# sourceMappingURL=up.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"up.js","sourceRoot":"","sources":["../../src/commands/up.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,4 @@
1
+ export declare function dashboardHtml(opts: {
2
+ proxyPort: number;
3
+ bind: string;
4
+ }): string;
@@ -0,0 +1,186 @@
1
+ export function dashboardHtml(opts) {
2
+ return `<!doctype html>
3
+ <html lang="en">
4
+ <head>
5
+ <meta charset="utf-8" />
6
+ <meta name="viewport" content="width=device-width,initial-scale=1" />
7
+ <title>TokenShield — live savings</title>
8
+ <style>
9
+ :root {
10
+ color-scheme: dark light;
11
+ --bg: #0b0d10;
12
+ --panel: #14171c;
13
+ --panel-2: #1a1e25;
14
+ --border: #2a2f38;
15
+ --text: #e7e9ee;
16
+ --muted: #8b94a3;
17
+ --accent: #6dd3a8;
18
+ --warn: #f0b35e;
19
+ --bad: #ef6868;
20
+ --link: #7eb8ff;
21
+ }
22
+ * { box-sizing: border-box }
23
+ html, body { margin: 0; background: var(--bg); color: var(--text); font: 14px/1.5 system-ui, -apple-system, "Segoe UI", Roboto, sans-serif; }
24
+ .wrap { max-width: 1100px; margin: 0 auto; padding: 24px; }
25
+ header { display: flex; align-items: baseline; justify-content: space-between; margin-bottom: 18px; }
26
+ h1 { margin: 0; font-size: 18px; letter-spacing: 0.02em; }
27
+ h1 small { color: var(--muted); font-weight: 400; margin-left: 8px; }
28
+ .status { font-size: 12px; color: var(--muted); }
29
+ .status .dot { display: inline-block; width: 8px; height: 8px; border-radius: 999px; background: var(--accent); margin-right: 6px; vertical-align: middle; }
30
+ .grid { display: grid; grid-template-columns: repeat(4, minmax(0, 1fr)); gap: 12px; }
31
+ .card { background: var(--panel); border: 1px solid var(--border); border-radius: 10px; padding: 14px; }
32
+ .card .label { font-size: 11px; text-transform: uppercase; letter-spacing: 0.08em; color: var(--muted); }
33
+ .card .value { font-size: 22px; font-weight: 600; margin-top: 6px; font-variant-numeric: tabular-nums; }
34
+ .card .sub { font-size: 11px; color: var(--muted); margin-top: 4px; }
35
+ .section { margin-top: 24px; }
36
+ .section h2 { font-size: 13px; text-transform: uppercase; letter-spacing: 0.08em; color: var(--muted); margin: 0 0 10px; }
37
+ table { width: 100%; border-collapse: collapse; font-variant-numeric: tabular-nums; }
38
+ th, td { text-align: left; padding: 8px 10px; border-bottom: 1px solid var(--border); font-size: 13px; }
39
+ th { color: var(--muted); font-weight: 500; font-size: 11px; text-transform: uppercase; letter-spacing: 0.06em; }
40
+ tr:last-child td { border-bottom: 0; }
41
+ .muted { color: var(--muted); }
42
+ .num { text-align: right; }
43
+ .pill { display: inline-block; padding: 1px 8px; border-radius: 999px; font-size: 11px; border: 1px solid var(--border); }
44
+ .pill.ok { color: var(--accent); border-color: rgba(109,211,168,0.35); }
45
+ .pill.err { color: var(--bad); border-color: rgba(239,104,104,0.35); }
46
+ .cmd { background: var(--panel-2); border: 1px solid var(--border); border-radius: 6px; padding: 10px 12px; font-family: ui-monospace, "SF Mono", Menlo, monospace; font-size: 12px; }
47
+ .footer { color: var(--muted); font-size: 12px; margin-top: 28px; padding-top: 16px; border-top: 1px solid var(--border); }
48
+ .kbd { font-family: ui-monospace, monospace; background: var(--panel-2); border: 1px solid var(--border); border-radius: 4px; padding: 1px 5px; font-size: 11px; }
49
+ a { color: var(--link); text-decoration: none; }
50
+ a:hover { text-decoration: underline; }
51
+ </style>
52
+ </head>
53
+ <body>
54
+ <div class="wrap">
55
+ <header>
56
+ <h1>TokenShield <small>v0.1 — Estimate mode</small></h1>
57
+ <div class="status"><span class="dot"></span><span id="status-text">proxy live on ${opts.bind}:${opts.proxyPort}</span></div>
58
+ </header>
59
+
60
+ <div class="grid">
61
+ <div class="card">
62
+ <div class="label">Spent (24h)</div>
63
+ <div class="value" id="dollars-spent">$0.00</div>
64
+ <div class="sub" id="request-count">0 requests</div>
65
+ </div>
66
+ <div class="card">
67
+ <div class="label">Input tokens (24h)</div>
68
+ <div class="value" id="input-tokens">0</div>
69
+ <div class="sub" id="input-cache">cache: 0 read, 0 write</div>
70
+ </div>
71
+ <div class="card">
72
+ <div class="label">Output tokens (24h)</div>
73
+ <div class="value" id="output-tokens">0</div>
74
+ <div class="sub">most expensive line item on Opus</div>
75
+ </div>
76
+ <div class="card">
77
+ <div class="label">Projected weekly</div>
78
+ <div class="value" id="weekly-projected">$0.00</div>
79
+ <div class="sub">extrapolated from last 24h</div>
80
+ </div>
81
+ </div>
82
+
83
+ <div class="section">
84
+ <h2>Spend by model (24h)</h2>
85
+ <table id="by-model">
86
+ <thead>
87
+ <tr>
88
+ <th>Model</th>
89
+ <th class="num">Requests</th>
90
+ <th class="num">Input</th>
91
+ <th class="num">Output</th>
92
+ <th class="num">$</th>
93
+ </tr>
94
+ </thead>
95
+ <tbody><tr><td colspan="5" class="muted">No traffic yet. Run Claude Code with <span class="kbd">ANTHROPIC_BASE_URL=http://${opts.bind}:${opts.proxyPort}</span>.</td></tr></tbody>
96
+ </table>
97
+ </div>
98
+
99
+ <div class="section">
100
+ <h2>Recent requests</h2>
101
+ <table id="recent">
102
+ <thead>
103
+ <tr>
104
+ <th>Time</th>
105
+ <th>Model</th>
106
+ <th>Endpoint</th>
107
+ <th class="num">Input</th>
108
+ <th class="num">Output</th>
109
+ <th class="num">Duration</th>
110
+ <th class="num">$</th>
111
+ <th>Status</th>
112
+ </tr>
113
+ </thead>
114
+ <tbody><tr><td colspan="8" class="muted">No requests recorded yet.</td></tr></tbody>
115
+ </table>
116
+ </div>
117
+
118
+ <div class="footer">
119
+ Privacy: TokenShield never stores prompt content. Your Anthropic API key stays in process memory.
120
+ Set <span class="cmd">export ANTHROPIC_BASE_URL=http://${opts.bind}:${opts.proxyPort}</span> in the shell you run Claude Code from.
121
+ </div>
122
+ </div>
123
+
124
+ <script>
125
+ const fmtNum = (n) => Number(n || 0).toLocaleString();
126
+ const fmtDollars = (d) => '$' + Number(d || 0).toFixed(Math.abs(d) < 1 ? 4 : 2);
127
+ const fmtTime = (ms) => new Date(ms).toLocaleTimeString();
128
+ const fmtMs = (ms) => ms < 1000 ? ms + 'ms' : (ms / 1000).toFixed(1) + 's';
129
+
130
+ async function refresh() {
131
+ try {
132
+ const [sumRes, recRes] = await Promise.all([
133
+ fetch('/api/summary'),
134
+ fetch('/api/recent'),
135
+ ]);
136
+ const sum = await sumRes.json();
137
+ const rec = await recRes.json();
138
+
139
+ document.getElementById('dollars-spent').textContent = fmtDollars(sum.dollarsRaw);
140
+ document.getElementById('request-count').textContent = fmtNum(sum.requestCount) + ' requests';
141
+ document.getElementById('input-tokens').textContent = fmtNum(sum.totalInputTokensRaw);
142
+ document.getElementById('output-tokens').textContent = fmtNum(sum.totalOutputTokensRaw);
143
+
144
+ const windowMs = Math.max(1, sum.windowEnd - sum.windowStart);
145
+ const weeklyProjection = sum.dollarsRaw / (windowMs / (7 * 24 * 60 * 60 * 1000));
146
+ document.getElementById('weekly-projected').textContent = fmtDollars(weeklyProjection);
147
+
148
+ const modelBody = document.querySelector('#by-model tbody');
149
+ if (sum.byModel && sum.byModel.length > 0) {
150
+ modelBody.innerHTML = sum.byModel.map((m) =>
151
+ '<tr><td>' + m.model + '</td>' +
152
+ '<td class="num">' + fmtNum(m.requests) + '</td>' +
153
+ '<td class="num">' + fmtNum(m.inputTokens) + '</td>' +
154
+ '<td class="num">' + fmtNum(m.outputTokens) + '</td>' +
155
+ '<td class="num">' + fmtDollars(m.dollars) + '</td></tr>'
156
+ ).join('');
157
+ }
158
+
159
+ const recBody = document.querySelector('#recent tbody');
160
+ if (rec && rec.length > 0) {
161
+ recBody.innerHTML = rec.map((r) => {
162
+ const ok = r.upstreamStatus >= 200 && r.upstreamStatus < 300;
163
+ return '<tr>' +
164
+ '<td class="muted">' + fmtTime(r.timestamp) + '</td>' +
165
+ '<td>' + r.model + '</td>' +
166
+ '<td class="muted">' + r.endpoint + '</td>' +
167
+ '<td class="num">' + fmtNum(r.usageRaw.inputTokens) + '</td>' +
168
+ '<td class="num">' + fmtNum(r.usageRaw.outputTokens) + '</td>' +
169
+ '<td class="num muted">' + fmtMs(r.durationMs) + '</td>' +
170
+ '<td class="num">' + fmtDollars(r.dollarsRaw) + '</td>' +
171
+ '<td><span class="pill ' + (ok ? 'ok' : 'err') + '">' + r.upstreamStatus + '</span></td>' +
172
+ '</tr>';
173
+ }).join('');
174
+ }
175
+ } catch (e) {
176
+ document.getElementById('status-text').textContent = 'dashboard error: ' + e.message;
177
+ }
178
+ }
179
+
180
+ refresh();
181
+ setInterval(refresh, 2000);
182
+ </script>
183
+ </body>
184
+ </html>`;
185
+ }
186
+ //# sourceMappingURL=dashboard.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"dashboard.js","sourceRoot":"","sources":["../src/dashboard.ts"],"names":[],"mappings":"AAAA,MAAM,UAAU,aAAa,CAAC,IAAyC;IACrE,OAAO;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;wFAuD+E,IAAI,CAAC,IAAI,IAAI,IAAI,CAAC,SAAS;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;kIAsCe,IAAI,CAAC,IAAI,IAAI,IAAI,CAAC,SAAS;;;;;;;;;;;;;;;;;;;;;;;;;6DAyBhG,IAAI,CAAC,IAAI,IAAI,IAAI,CAAC,SAAS;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;QAgEhF,CAAC;AACT,CAAC"}
@@ -0,0 +1,23 @@
1
+ export interface DaemonInfo {
2
+ pid: number;
3
+ startedAt: number;
4
+ port: number;
5
+ dashboardPort: number;
6
+ bind: string;
7
+ }
8
+ declare function isAliveSync(pid: number): boolean;
9
+ export declare function readDaemon(): DaemonInfo | null;
10
+ export declare function writeDaemon(info: DaemonInfo): void;
11
+ export declare function clearDaemon(): void;
12
+ export interface SpawnOptions {
13
+ port: number;
14
+ dashboardPort: number;
15
+ bind: string;
16
+ upstream: string;
17
+ ledger: string;
18
+ retentionDays: number;
19
+ }
20
+ /** Spawn the proxy as a detached background process and return its DaemonInfo. */
21
+ export declare function spawnDaemon(opts: SpawnOptions): Promise<DaemonInfo>;
22
+ export declare function stopDaemon(timeoutMs?: number): Promise<DaemonInfo | null>;
23
+ export { isAliveSync };