@ecadlabs/tezosx-mcp 1.0.3 → 1.0.5
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/dist/api.js +1 -0
- package/dist/live-config.d.ts +1 -0
- package/dist/live-config.js +3 -0
- package/dist/tools/index.d.ts +21 -0
- package/dist/tools/index.js +2 -0
- package/dist/tools/recover_spender_funds.d.ts +24 -0
- package/dist/tools/recover_spender_funds.js +113 -0
- package/dist/webserver.js +17 -7
- package/frontend/dist/assets/index-BjYU9URm.css +1 -0
- package/frontend/dist/assets/{index-BM2KDhgo.js → index-CfgL_Mz0.js} +50 -50
- package/frontend/dist/index.html +2 -2
- package/package.json +1 -1
- package/frontend/dist/assets/index-DxuaKI2K.css +0 -1
package/dist/api.js
CHANGED
|
@@ -50,6 +50,7 @@ export function createApiRouter(liveConfig) {
|
|
|
50
50
|
configured: liveConfig.configured,
|
|
51
51
|
spenderAddress: liveConfig.spendingAddress || undefined,
|
|
52
52
|
contractAddress: liveConfig.spendingContract || undefined,
|
|
53
|
+
network: liveConfig.networkName,
|
|
53
54
|
});
|
|
54
55
|
});
|
|
55
56
|
// Generate keypair server-side, persist as *pending* key, return only public info.
|
package/dist/live-config.d.ts
CHANGED
package/dist/live-config.js
CHANGED
|
@@ -34,6 +34,7 @@ export function createLiveConfig(networkName) {
|
|
|
34
34
|
spendingContract: '',
|
|
35
35
|
spendingAddress: '',
|
|
36
36
|
tzktApi: network.tzktApi,
|
|
37
|
+
networkName,
|
|
37
38
|
configured: false,
|
|
38
39
|
};
|
|
39
40
|
}
|
|
@@ -46,6 +47,7 @@ export async function configureLiveConfig(config, privateKey, spendingContract,
|
|
|
46
47
|
const network = NETWORKS[networkName];
|
|
47
48
|
config.Tezos = new TezosToolkit(network.rpcUrl);
|
|
48
49
|
config.tzktApi = network.tzktApi;
|
|
50
|
+
config.networkName = networkName;
|
|
49
51
|
}
|
|
50
52
|
const signer = await InMemorySigner.fromSecretKey(privateKey);
|
|
51
53
|
config.Tezos.setSignerProvider(signer);
|
|
@@ -62,6 +64,7 @@ export function resetLiveConfig(config, networkName) {
|
|
|
62
64
|
config.Tezos = new TezosToolkit(rpcUrl);
|
|
63
65
|
if (networkName) {
|
|
64
66
|
config.tzktApi = NETWORKS[networkName].tzktApi;
|
|
67
|
+
config.networkName = networkName;
|
|
65
68
|
}
|
|
66
69
|
config.spendingContract = '';
|
|
67
70
|
config.spendingAddress = '';
|
package/dist/tools/index.d.ts
CHANGED
|
@@ -88,6 +88,27 @@ export declare const createTools: (liveConfig: LiveConfig, http: boolean) => ({
|
|
|
88
88
|
text: string;
|
|
89
89
|
}[];
|
|
90
90
|
}>;
|
|
91
|
+
} | {
|
|
92
|
+
name: string;
|
|
93
|
+
config: {
|
|
94
|
+
title: string;
|
|
95
|
+
description: string;
|
|
96
|
+
inputSchema: import("zod").ZodObject<{
|
|
97
|
+
destination: import("zod").ZodOptional<import("zod").ZodString>;
|
|
98
|
+
}, import("zod/v4/core").$strip>;
|
|
99
|
+
annotations: {
|
|
100
|
+
readOnlyHint: boolean;
|
|
101
|
+
destructiveHint: boolean;
|
|
102
|
+
idempotentHint: boolean;
|
|
103
|
+
openWorldHint: boolean;
|
|
104
|
+
};
|
|
105
|
+
};
|
|
106
|
+
handler: (params: any) => Promise<{
|
|
107
|
+
content: {
|
|
108
|
+
type: "text";
|
|
109
|
+
text: string;
|
|
110
|
+
}[];
|
|
111
|
+
}>;
|
|
91
112
|
} | {
|
|
92
113
|
name: string;
|
|
93
114
|
config: {
|
package/dist/tools/index.js
CHANGED
|
@@ -6,6 +6,7 @@ import { createGetLimitsTool } from "./get_limits.js";
|
|
|
6
6
|
import { createGetOperationHistoryTool } from "./get_operation_history.js";
|
|
7
7
|
import { createParseX402RequirementsTool } from "./parse_x402_requirements.js";
|
|
8
8
|
import { createRevealAccountTool } from "./reveal_account.js";
|
|
9
|
+
import { createRecoverSpenderFundsTool } from "./recover_spender_funds.js";
|
|
9
10
|
import { createSendXtzTool } from "./send_xtz.js";
|
|
10
11
|
import { createGetDashboardTool } from "./get_dashboard.js";
|
|
11
12
|
const getNotConfiguredMessage = () => `Wallet not configured.
|
|
@@ -39,6 +40,7 @@ export const createTools = (liveConfig, http) => {
|
|
|
39
40
|
createGetLimitsTool(liveConfig),
|
|
40
41
|
createGetOperationHistoryTool(liveConfig),
|
|
41
42
|
createParseX402RequirementsTool(),
|
|
43
|
+
createRecoverSpenderFundsTool(liveConfig),
|
|
42
44
|
createRevealAccountTool(liveConfig),
|
|
43
45
|
createSendXtzTool(liveConfig),
|
|
44
46
|
];
|
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
import z from "zod";
|
|
2
|
+
import type { LiveConfig } from "../live-config.js";
|
|
3
|
+
export declare const createRecoverSpenderFundsTool: (config: LiveConfig) => {
|
|
4
|
+
name: string;
|
|
5
|
+
config: {
|
|
6
|
+
title: string;
|
|
7
|
+
description: string;
|
|
8
|
+
inputSchema: z.ZodObject<{
|
|
9
|
+
destination: z.ZodOptional<z.ZodString>;
|
|
10
|
+
}, z.z.core.$strip>;
|
|
11
|
+
annotations: {
|
|
12
|
+
readOnlyHint: boolean;
|
|
13
|
+
destructiveHint: boolean;
|
|
14
|
+
idempotentHint: boolean;
|
|
15
|
+
openWorldHint: boolean;
|
|
16
|
+
};
|
|
17
|
+
};
|
|
18
|
+
handler: (params: any) => Promise<{
|
|
19
|
+
content: {
|
|
20
|
+
type: "text";
|
|
21
|
+
text: string;
|
|
22
|
+
}[];
|
|
23
|
+
}>;
|
|
24
|
+
};
|
|
@@ -0,0 +1,113 @@
|
|
|
1
|
+
import z from "zod";
|
|
2
|
+
import { ensureRevealed } from "./reveal_account.js";
|
|
3
|
+
const CONFIRMATIONS_TO_WAIT = 3;
|
|
4
|
+
const inputSchema = z.object({
|
|
5
|
+
destination: z.string().optional().describe("Address to send recovered funds to. Defaults to the contract owner's address."),
|
|
6
|
+
});
|
|
7
|
+
export const createRecoverSpenderFundsTool = (config) => ({
|
|
8
|
+
name: "tezos_recover_spender_funds",
|
|
9
|
+
config: {
|
|
10
|
+
title: "Recover Spender Gas Funds",
|
|
11
|
+
description: "Transfers the spender address's XTZ balance (used for gas fees) to the contract owner or a specified address. " +
|
|
12
|
+
"Useful for recovering funds when decommissioning a spender or rebalancing. " +
|
|
13
|
+
"Always fetches the live on-chain balance — call this tool every time the user asks, even if a previous call returned zero.",
|
|
14
|
+
inputSchema,
|
|
15
|
+
annotations: {
|
|
16
|
+
readOnlyHint: false,
|
|
17
|
+
destructiveHint: true,
|
|
18
|
+
idempotentHint: false,
|
|
19
|
+
openWorldHint: true,
|
|
20
|
+
},
|
|
21
|
+
},
|
|
22
|
+
handler: async (params) => {
|
|
23
|
+
params = params;
|
|
24
|
+
const { Tezos, spendingContract, spendingAddress, tzktApi } = config;
|
|
25
|
+
// Resolve destination: explicit param > contract owner > spending contract
|
|
26
|
+
let destination = params.destination;
|
|
27
|
+
if (!destination) {
|
|
28
|
+
if (spendingContract) {
|
|
29
|
+
const contract = await Tezos.contract.at(spendingContract);
|
|
30
|
+
const storage = await contract.storage();
|
|
31
|
+
destination = storage.owner;
|
|
32
|
+
}
|
|
33
|
+
}
|
|
34
|
+
if (!destination) {
|
|
35
|
+
throw new Error("No destination provided and no spending contract configured.");
|
|
36
|
+
}
|
|
37
|
+
const balance = await Tezos.tz.getBalance(spendingAddress);
|
|
38
|
+
const balanceMutez = balance.toNumber();
|
|
39
|
+
if (balanceMutez === 0) {
|
|
40
|
+
return {
|
|
41
|
+
content: [{
|
|
42
|
+
type: "text",
|
|
43
|
+
text: `Spender address (${spendingAddress}) has zero balance. Nothing to recover.`,
|
|
44
|
+
}],
|
|
45
|
+
};
|
|
46
|
+
}
|
|
47
|
+
await ensureRevealed(Tezos);
|
|
48
|
+
// Drain account pattern (Taquito docs):
|
|
49
|
+
// For KT1 destinations, call the default entrypoint explicitly.
|
|
50
|
+
// For implicit accounts, use a simple transfer.
|
|
51
|
+
const isContract = destination.startsWith("KT1");
|
|
52
|
+
if (isContract) {
|
|
53
|
+
const contract = await Tezos.contract.at(destination);
|
|
54
|
+
const depositCall = contract.methodsObject.default_(null);
|
|
55
|
+
const estimate = await Tezos.estimate.contractCall(depositCall);
|
|
56
|
+
const maxAmount = balanceMutez - estimate.suggestedFeeMutez;
|
|
57
|
+
if (maxAmount <= 0) {
|
|
58
|
+
return {
|
|
59
|
+
content: [{
|
|
60
|
+
type: "text",
|
|
61
|
+
text: `Spender balance (${balanceMutez} mutez) is too low to cover transfer fees (${estimate.suggestedFeeMutez} mutez). Nothing to recover.`,
|
|
62
|
+
}],
|
|
63
|
+
};
|
|
64
|
+
}
|
|
65
|
+
const operation = await contract.methodsObject.default_(null).send({
|
|
66
|
+
amount: maxAmount,
|
|
67
|
+
mutez: true,
|
|
68
|
+
fee: estimate.suggestedFeeMutez,
|
|
69
|
+
gasLimit: estimate.gasLimit,
|
|
70
|
+
storageLimit: 0,
|
|
71
|
+
});
|
|
72
|
+
await operation.confirmation(CONFIRMATIONS_TO_WAIT);
|
|
73
|
+
const tzktBase = tzktApi.replace("api.", "");
|
|
74
|
+
return {
|
|
75
|
+
content: [{
|
|
76
|
+
type: "text",
|
|
77
|
+
text: `Recovered ${(maxAmount / 1_000_000).toFixed(6)} XTZ from spender (${spendingAddress}) to ${destination}.\n${tzktBase}/${operation.hash}`,
|
|
78
|
+
}],
|
|
79
|
+
};
|
|
80
|
+
}
|
|
81
|
+
// Implicit account (tz1/tz2/tz3)
|
|
82
|
+
const estimate = await Tezos.estimate.transfer({
|
|
83
|
+
to: destination,
|
|
84
|
+
amount: balanceMutez,
|
|
85
|
+
mutez: true,
|
|
86
|
+
});
|
|
87
|
+
const maxAmount = balanceMutez - estimate.suggestedFeeMutez;
|
|
88
|
+
if (maxAmount <= 0) {
|
|
89
|
+
return {
|
|
90
|
+
content: [{
|
|
91
|
+
type: "text",
|
|
92
|
+
text: `Spender balance (${balanceMutez} mutez) is too low to cover transfer fees (${estimate.suggestedFeeMutez} mutez). Nothing to recover.`,
|
|
93
|
+
}],
|
|
94
|
+
};
|
|
95
|
+
}
|
|
96
|
+
const operation = await Tezos.contract.transfer({
|
|
97
|
+
to: destination,
|
|
98
|
+
amount: maxAmount,
|
|
99
|
+
mutez: true,
|
|
100
|
+
fee: estimate.suggestedFeeMutez,
|
|
101
|
+
gasLimit: estimate.gasLimit,
|
|
102
|
+
storageLimit: 0,
|
|
103
|
+
});
|
|
104
|
+
await operation.confirmation(CONFIRMATIONS_TO_WAIT);
|
|
105
|
+
const tzktBase = tzktApi.replace("api.", "");
|
|
106
|
+
return {
|
|
107
|
+
content: [{
|
|
108
|
+
type: "text",
|
|
109
|
+
text: `Recovered ${(maxAmount / 1_000_000).toFixed(6)} XTZ from spender (${spendingAddress}) to ${destination}.\n${tzktBase}/${operation.hash}`,
|
|
110
|
+
}],
|
|
111
|
+
};
|
|
112
|
+
},
|
|
113
|
+
});
|
package/dist/webserver.js
CHANGED
|
@@ -14,28 +14,38 @@ export const startWebServer = (port, apiRouter) => {
|
|
|
14
14
|
const distPath = join(__dirname, "../frontend/dist");
|
|
15
15
|
app.use(sirv(distPath, { single: true }));
|
|
16
16
|
let retries = 0;
|
|
17
|
-
const maxRetries =
|
|
17
|
+
const maxRetries = 5;
|
|
18
|
+
const retryDelay = 1000;
|
|
19
|
+
let activeServer = null;
|
|
18
20
|
const tryListen = () => {
|
|
19
21
|
const server = app.listen(port);
|
|
20
22
|
server.on("error", (err) => {
|
|
21
23
|
if (err.code === "EADDRINUSE" && retries < maxRetries) {
|
|
22
24
|
retries++;
|
|
23
25
|
console.error(`[tezosx-mcp] Port ${port} in use, retrying (${retries}/${maxRetries})...`);
|
|
24
|
-
setTimeout(tryListen,
|
|
26
|
+
setTimeout(tryListen, retryDelay);
|
|
25
27
|
}
|
|
26
28
|
else if (err.code === "EADDRINUSE") {
|
|
27
|
-
console.error(`[tezosx-mcp] Port ${port} in use, frontend dashboard unavailable`);
|
|
29
|
+
console.error(`[tezosx-mcp] Port ${port} still in use after ${maxRetries} retries, frontend dashboard unavailable`);
|
|
28
30
|
}
|
|
29
31
|
else {
|
|
30
32
|
console.error(`[tezosx-mcp] Web server error:`, err.message);
|
|
31
33
|
}
|
|
32
34
|
});
|
|
33
35
|
server.on("listening", () => {
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
process.on("SIGINT", shutdown);
|
|
37
|
-
process.on("exit", shutdown);
|
|
36
|
+
activeServer = server;
|
|
37
|
+
console.error(`[tezosx-mcp] Dashboard running on http://localhost:${port}`);
|
|
38
38
|
});
|
|
39
39
|
};
|
|
40
|
+
// Ensure the server is fully closed before the process exits
|
|
41
|
+
const shutdown = () => {
|
|
42
|
+
if (activeServer) {
|
|
43
|
+
activeServer.closeAllConnections();
|
|
44
|
+
activeServer.close();
|
|
45
|
+
activeServer = null;
|
|
46
|
+
}
|
|
47
|
+
};
|
|
48
|
+
process.on("SIGTERM", () => { shutdown(); process.exit(0); });
|
|
49
|
+
process.on("SIGINT", () => { shutdown(); process.exit(0); });
|
|
40
50
|
tryListen();
|
|
41
51
|
};
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
/*! tailwindcss v4.1.17 | MIT License | https://tailwindcss.com */@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-space-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-backdrop-blur:initial;--tw-backdrop-brightness:initial;--tw-backdrop-contrast:initial;--tw-backdrop-grayscale:initial;--tw-backdrop-hue-rotate:initial;--tw-backdrop-invert:initial;--tw-backdrop-opacity:initial;--tw-backdrop-saturate:initial;--tw-backdrop-sepia:initial;--tw-duration:initial;--tw-ease:initial}}}@layer theme{:root,:host{--font-sans:"Inter",system-ui,-apple-system,sans-serif;--font-mono:"JetBrains Mono","SF Mono","Fira Code",monospace;--color-red-50:oklch(97.1% .013 17.38);--color-red-100:oklch(93.6% .032 17.717);--color-red-200:oklch(88.5% .062 18.334);--color-red-600:oklch(57.7% .245 27.325);--color-red-700:oklch(50.5% .213 27.518);--color-red-800:oklch(44.4% .177 26.899);--color-amber-50:oklch(98.7% .022 95.277);--color-amber-100:oklch(96.2% .059 95.617);--color-amber-200:oklch(92.4% .12 95.746);--color-amber-600:oklch(66.6% .179 58.318);--color-amber-700:oklch(55.5% .163 48.998);--color-amber-800:oklch(47.3% .137 46.201);--color-green-50:oklch(98.2% .018 155.826);--color-green-100:oklch(96.2% .044 156.743);--color-green-200:oklch(92.5% .084 155.995);--color-green-600:oklch(62.7% .194 149.214);--color-green-700:oklch(52.7% .154 150.069);--color-green-800:oklch(44.8% .119 151.328);--color-gray-50:oklch(98.5% .002 247.839);--color-gray-200:oklch(92.8% .006 264.531);--color-gray-300:oklch(87.2% .01 258.338);--color-gray-900:oklch(21% .034 264.665);--color-black:#000;--color-white:#fff;--spacing:.25rem;--container-md:28rem;--container-2xl:42rem;--text-xs:.75rem;--text-xs--line-height:calc(1/.75);--text-sm:.875rem;--text-sm--line-height:calc(1.25/.875);--text-lg:1.125rem;--text-lg--line-height:calc(1.75/1.125);--text-xl:1.25rem;--text-xl--line-height:calc(1.75/1.25);--text-3xl:1.875rem;--text-3xl--line-height: 1.2 ;--font-weight-medium:500;--font-weight-semibold:600;--tracking-tight:-.025em;--tracking-wider:.05em;--leading-relaxed:1.625;--radius-lg:.5rem;--blur-sm:8px;--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-primary-50:#fafafa;--color-primary-100:#f5f5f5;--color-primary-200:#e5e5e5;--color-primary-600:#525252;--color-accent-400:#fb923c;--color-accent-500:#f97316;--color-accent-600:#ea580c;--color-text-primary:#171717;--color-text-secondary:#525252;--color-text-muted:#a3a3a3;--color-warning:#f59e0b;--color-error:#ef4444;--radius-card:12px;--radius-button:6px;--radius-badge:4px}}@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%;-moz-tab-size:4;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;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]){-webkit-appearance:button;-moz-appearance:button;appearance:button}::file-selector-button{-webkit-appearance:button;-moz-appearance: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{.absolute{position:absolute}.fixed{position:fixed}.relative{position:relative}.inset-0{inset:calc(var(--spacing)*0)}.top-full{top:100%}.bottom-full{bottom:100%}.left-1\/2{left:50%}.z-10{z-index:10}.z-50{z-index:50}.mx-4{margin-inline:calc(var(--spacing)*4)}.mx-auto{margin-inline:auto}.-mt-1{margin-top:calc(var(--spacing)*-1)}.mt-0\.5{margin-top:calc(var(--spacing)*.5)}.mt-1\.5{margin-top:calc(var(--spacing)*1.5)}.mt-2{margin-top:calc(var(--spacing)*2)}.mt-4{margin-top:calc(var(--spacing)*4)}.mt-5{margin-top:calc(var(--spacing)*5)}.mt-12{margin-top:calc(var(--spacing)*12)}.\!mb-0{margin-bottom:calc(var(--spacing)*0)!important}.mb-1{margin-bottom:calc(var(--spacing)*1)}.mb-2{margin-bottom:calc(var(--spacing)*2)}.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)}.block{display:block}.flex{display:flex}.grid{display:grid}.hidden{display:none}.inline-flex{display:inline-flex}.\!h-3{height:calc(var(--spacing)*3)!important}.\!h-4{height:calc(var(--spacing)*4)!important}.h-3\.5{height:calc(var(--spacing)*3.5)}.h-4{height:calc(var(--spacing)*4)}.h-5{height:calc(var(--spacing)*5)}.h-6{height:calc(var(--spacing)*6)}.h-8{height:calc(var(--spacing)*8)}.h-12{height:calc(var(--spacing)*12)}.h-\[38px\]{height:38px}.h-px{height:1px}.min-h-screen{min-height:100vh}.\!w-3{width:calc(var(--spacing)*3)!important}.\!w-4{width:calc(var(--spacing)*4)!important}.w-3\.5{width:calc(var(--spacing)*3.5)}.w-4{width:calc(var(--spacing)*4)}.w-5{width:calc(var(--spacing)*5)}.w-6{width:calc(var(--spacing)*6)}.w-8{width:calc(var(--spacing)*8)}.w-12{width:calc(var(--spacing)*12)}.w-20{width:calc(var(--spacing)*20)}.w-24{width:calc(var(--spacing)*24)}.w-64{width:calc(var(--spacing)*64)}.w-fit{width:fit-content}.w-full{width:100%}.max-w-2xl{max-width:var(--container-2xl)}.max-w-md{max-width:var(--container-md)}.flex-1{flex:1}.flex-shrink-0,.shrink-0{flex-shrink:0}.-translate-x-1\/2{--tw-translate-x: -50% ;translate:var(--tw-translate-x)var(--tw-translate-y)}.rotate-180{rotate:180deg}.cursor-pointer{cursor:pointer}.grid-cols-1{grid-template-columns:repeat(1,minmax(0,1fr))}.grid-cols-2{grid-template-columns:repeat(2,minmax(0,1fr))}.grid-cols-\[160px_1fr\]{grid-template-columns:160px 1fr}.grid-rows-\[0fr\]{grid-template-rows:0fr}.grid-rows-\[1fr\]{grid-template-rows:1fr}.items-baseline{align-items:baseline}.items-center{align-items:center}.items-start{align-items:flex-start}.items-stretch{align-items:stretch}.justify-between{justify-content:space-between}.justify-center{justify-content:center}.gap-1{gap:calc(var(--spacing)*1)}.gap-1\.5{gap:calc(var(--spacing)*1.5)}.gap-2{gap:calc(var(--spacing)*2)}.gap-3{gap:calc(var(--spacing)*3)}: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-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)))}.truncate{text-overflow:ellipsis;white-space:nowrap;overflow:hidden}.overflow-hidden{overflow:hidden}.overflow-x-auto{overflow-x:auto}.rounded{border-radius:.25rem}.rounded-full{border-radius:3.40282e38px}.rounded-lg{border-radius:var(--radius-lg)}.border{border-style:var(--tw-border-style);border-width:1px}.border-2{border-style:var(--tw-border-style);border-width:2px}.border-4{border-style:var(--tw-border-style);border-width:4px}.border-t{border-top-style:var(--tw-border-style);border-top-width:1px}.border-none{--tw-border-style:none;border-style:none}.border-amber-200{border-color:var(--color-amber-200)}.border-error\/20{border-color:#ef444433}@supports (color:color-mix(in lab,red,red)){.border-error\/20{border-color:color-mix(in oklab,var(--color-error)20%,transparent)}}.border-gray-200{border-color:var(--color-gray-200)}.border-green-200{border-color:var(--color-green-200)}.border-primary-200{border-color:var(--color-primary-200)}.border-red-200{border-color:var(--color-red-200)}.border-transparent{border-color:#0000}.border-warning\/20{border-color:#f59e0b33}@supports (color:color-mix(in lab,red,red)){.border-warning\/20{border-color:color-mix(in oklab,var(--color-warning)20%,transparent)}}.border-t-gray-900{border-top-color:var(--color-gray-900)}.bg-amber-50\/50{background-color:#fffbeb80}@supports (color:color-mix(in lab,red,red)){.bg-amber-50\/50{background-color:color-mix(in oklab,var(--color-amber-50)50%,transparent)}}.bg-amber-100{background-color:var(--color-amber-100)}.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-error\/5{background-color:#ef44440d}@supports (color:color-mix(in lab,red,red)){.bg-error\/5{background-color:color-mix(in oklab,var(--color-error)5%,transparent)}}.bg-gray-50{background-color:var(--color-gray-50)}.bg-gray-900{background-color:var(--color-gray-900)}.bg-green-50\/50{background-color:#f0fdf480}@supports (color:color-mix(in lab,red,red)){.bg-green-50\/50{background-color:color-mix(in oklab,var(--color-green-50)50%,transparent)}}.bg-green-100{background-color:var(--color-green-100)}.bg-primary-100{background-color:var(--color-primary-100)}.bg-primary-200{background-color:var(--color-primary-200)}.bg-red-50\/50{background-color:#fef2f280}@supports (color:color-mix(in lab,red,red)){.bg-red-50\/50{background-color:color-mix(in oklab,var(--color-red-50)50%,transparent)}}.bg-red-100{background-color:var(--color-red-100)}.bg-transparent{background-color:#0000}.bg-warning\/10{background-color:#f59e0b1a}@supports (color:color-mix(in lab,red,red)){.bg-warning\/10{background-color:color-mix(in oklab,var(--color-warning)10%,transparent)}}.bg-white{background-color:var(--color-white)}.p-3{padding:calc(var(--spacing)*3)}.p-4{padding:calc(var(--spacing)*4)}.p-5{padding:calc(var(--spacing)*5)}.p-6{padding:calc(var(--spacing)*6)}.\!px-2{padding-inline:calc(var(--spacing)*2)!important}.\!px-3{padding-inline:calc(var(--spacing)*3)!important}.px-1{padding-inline:calc(var(--spacing)*1)}.px-2{padding-inline:calc(var(--spacing)*2)}.px-3{padding-inline:calc(var(--spacing)*3)}.px-4{padding-inline:calc(var(--spacing)*4)}.px-5{padding-inline:calc(var(--spacing)*5)}.\!py-1{padding-block:calc(var(--spacing)*1)!important}.\!py-1\.5{padding-block:calc(var(--spacing)*1.5)!important}.\!py-2{padding-block:calc(var(--spacing)*2)!important}.py-0\.5{padding-block:calc(var(--spacing)*.5)}.py-1{padding-block:calc(var(--spacing)*1)}.py-1\.5{padding-block:calc(var(--spacing)*1.5)}.py-2\.5{padding-block:calc(var(--spacing)*2.5)}.py-4{padding-block:calc(var(--spacing)*4)}.py-12{padding-block:calc(var(--spacing)*12)}.pt-2{padding-top:calc(var(--spacing)*2)}.pb-5{padding-bottom:calc(var(--spacing)*5)}.text-center{text-align:center}.text-left{text-align:left}.text-3xl{font-size:var(--text-3xl);line-height:var(--tw-leading,var(--text-3xl--line-height))}.text-lg{font-size:var(--text-lg);line-height:var(--tw-leading,var(--text-lg--line-height))}.text-sm{font-size:var(--text-sm);line-height:var(--tw-leading,var(--text-sm--line-height))}.text-xl{font-size:var(--text-xl);line-height:var(--tw-leading,var(--text-xl--line-height))}.text-xs{font-size:var(--text-xs);line-height:var(--tw-leading,var(--text-xs--line-height))}.\!text-\[11px\]{font-size:11px!important}.leading-relaxed{--tw-leading:var(--leading-relaxed);line-height:var(--leading-relaxed)}.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-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-pre-wrap{white-space:pre-wrap}.text-accent-600{color:var(--color-accent-600)}.text-amber-600{color:var(--color-amber-600)}.text-amber-700{color:var(--color-amber-700)}.text-amber-800{color:var(--color-amber-800)}.text-error{color:var(--color-error)}.text-gray-300{color:var(--color-gray-300)}.text-green-600{color:var(--color-green-600)}.text-green-700{color:var(--color-green-700)}.text-green-800{color:var(--color-green-800)}.text-primary-600{color:var(--color-primary-600)}.text-red-600{color:var(--color-red-600)}.text-red-700{color:var(--color-red-700)}.text-red-800{color:var(--color-red-800)}.text-text-muted{color:var(--color-text-muted)}.text-text-primary{color:var(--color-text-primary)}.text-text-secondary{color:var(--color-text-secondary)}.text-warning{color:var(--color-warning)}.text-warning\/70{color:#f59e0bb3}@supports (color:color-mix(in lab,red,red)){.text-warning\/70{color:color-mix(in oklab,var(--color-warning)70%,transparent)}}.text-white{color:var(--color-white)}.uppercase{text-transform:uppercase}.italic{font-style:italic}.shadow-lg{--tw-shadow:0 10px 15px -3px var(--tw-shadow-color,#0000001a),0 4px 6px -4px var(--tw-shadow-color,#0000001a);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.shadow-xl{--tw-shadow:0 20px 25px -5px var(--tw-shadow-color,#0000001a),0 8px 10px -6px var(--tw-shadow-color,#0000001a);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.backdrop-blur-sm{--tw-backdrop-blur:blur(var(--blur-sm));-webkit-backdrop-filter:var(--tw-backdrop-blur,)var(--tw-backdrop-brightness,)var(--tw-backdrop-contrast,)var(--tw-backdrop-grayscale,)var(--tw-backdrop-hue-rotate,)var(--tw-backdrop-invert,)var(--tw-backdrop-opacity,)var(--tw-backdrop-saturate,)var(--tw-backdrop-sepia,);backdrop-filter:var(--tw-backdrop-blur,)var(--tw-backdrop-brightness,)var(--tw-backdrop-contrast,)var(--tw-backdrop-grayscale,)var(--tw-backdrop-hue-rotate,)var(--tw-backdrop-invert,)var(--tw-backdrop-opacity,)var(--tw-backdrop-saturate,)var(--tw-backdrop-sepia,)}.transition-\[grid-template-rows\]{transition-property:grid-template-rows;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-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-200{--tw-duration:.2s;transition-duration:.2s}.duration-300{--tw-duration:.3s;transition-duration:.3s}.ease-\[cubic-bezier\(0\.16\,1\,0\.3\,1\)\]{--tw-ease:cubic-bezier(.16,1,.3,1);transition-timing-function:cubic-bezier(.16,1,.3,1)}@media (hover:hover){.hover\:border-text-muted:hover{border-color:var(--color-text-muted)}.hover\:bg-primary-50\/50:hover{background-color:#fafafa80}@supports (color:color-mix(in lab,red,red)){.hover\:bg-primary-50\/50:hover{background-color:color-mix(in oklab,var(--color-primary-50)50%,transparent)}}.hover\:text-text-secondary:hover{color:var(--color-text-secondary)}}@media (min-width:40rem){.sm\:block{display:block}.sm\:grid-cols-2{grid-template-columns:repeat(2,minmax(0,1fr))}.sm\:grid-cols-\[170px_1fr_1fr\]{grid-template-columns:170px 1fr 1fr}.sm\:px-6{padding-inline:calc(var(--spacing)*6)}}@media (min-width:64rem){.lg\:px-8{padding-inline:calc(var(--spacing)*8)}}}html{font-family:var(--font-sans);color:var(--color-text-primary);background:#f5f5f5;min-height:100vh}body{min-height:100vh;position:relative}.film-grain{pointer-events:none;z-index:0;width:100%;height:100%;position:fixed;top:0;left:0}.film-grain:before{content:"";opacity:.08;background-image:url("data:image/svg+xml,%3Csvg viewBox='0 0 200 200' xmlns='http://www.w3.org/2000/svg'%3E%3Cfilter id='noise'%3E%3CfeTurbulence type='fractalNoise' baseFrequency='0.8' numOctaves='4' stitchTiles='stitch'/%3E%3C/filter%3E%3Crect width='100%25' height='100%25' filter='url(%23noise)'/%3E%3C/svg%3E");width:100%;height:100%;position:absolute;top:0;left:0}.film-grain:after{content:"";background:linear-gradient(135deg,#fb923c1f,#f9731614,#fed7aa0d,#0000 75%);width:100%;height:100%;position:absolute;top:0;left:0}.card{border-radius:var(--radius-card);background:#fff;box-shadow:0 1px 3px #0000000a,0 1px 2px #0000000f}.card-subtle{border-radius:var(--radius-button);background:#fafafa;border:1px solid #e5e5e5}.section-label{font-family:var(--font-mono);color:var(--color-text-muted);letter-spacing:-.01em;font-size:.8125rem}.section-label:before{content:"// "}.btn-primary{background:var(--color-accent-500);color:#fff;border-radius:var(--radius-button);cursor:pointer;border:none;padding:.5rem 1rem;font-size:.875rem;font-weight:500;transition:background .15s}.btn-primary:hover{background:var(--color-accent-600)}.btn-primary:disabled{opacity:.5;cursor:not-allowed}.btn-secondary{color:var(--color-text-primary);border-radius:var(--radius-button);cursor:pointer;background:#fff;border:1px solid #e5e5e5;padding:.5rem 1rem;font-size:.875rem;font-weight:500;transition:all .15s}.btn-secondary:hover{background:#fafafa;border-color:#d4d4d4}.btn-secondary:disabled{opacity:.5;cursor:not-allowed}.btn-danger{background:var(--color-error);color:#fff;border-radius:var(--radius-button);cursor:pointer;border:none;padding:.5rem 1rem;font-size:.875rem;font-weight:500;transition:background .15s}.btn-danger:hover{background:#dc2626}.btn-danger:disabled{opacity:.5;cursor:not-allowed}.input-field{border-radius:var(--radius-button);width:100%;color:var(--color-text-primary);background:#fff;border:1px solid #e5e5e5;padding:.5rem .75rem;font-family:inherit;font-size:.875rem;transition:border-color .15s}.input-field:focus{border-color:var(--color-accent-500);outline:none}.input-field::placeholder{color:var(--color-text-muted)}.mono{font-family:var(--font-mono);letter-spacing:-.02em;font-size:.8125rem}.label{font-family:var(--font-mono);color:var(--color-text-muted);text-transform:lowercase;margin-bottom:.375rem;font-size:.75rem;display:block}.tech-badge{font-family:var(--font-mono);color:var(--color-text-secondary);border-radius:var(--radius-badge);background:#f5f5f5;border:1px solid #e5e5e5;align-items:center;padding:.25rem .5rem;font-size:.6875rem;display:inline-flex}.badge{border-radius:var(--radius-badge);align-items:center;padding:.25rem .5rem;font-size:.75rem;font-weight:500;display:inline-flex}.badge-success{color:#16a34a;background:#22c55e1a}.badge-warning{color:#d97706;background:#f59e0b1a}.badge-error{color:#dc2626;background:#ef44441a}.badge-muted{color:#6b7280;background:#6b72801a}.accent-text{color:var(--color-accent-600);font-family:var(--font-mono);font-size:.875rem}.divider{background:#e5e5e5;height:1px;margin:1.5rem 0}@keyframes spin{to{transform:rotate(360deg)}}.spinner{border:2px solid #ffffff4d;border-top-color:#fff;border-radius:50%;width:1rem;height:1rem;animation:.8s linear infinite spin}.spinner-dark{border-color:#e5e5e5;border-top-color:var(--color-text-secondary)}.progress-bar{background:#e5e5e5;border-radius:2px;height:4px;overflow:hidden}.progress-bar-fill{background:linear-gradient(90deg,var(--color-accent-400),var(--color-accent-500));height:100%;transition:width .3s}.value-display{color:var(--color-text-primary);font-size:1.5rem;font-weight:600;line-height:1}.value-unit{color:var(--color-text-muted);margin-left:.25rem;font-size:.875rem;font-weight:400}a{color:var(--color-accent-500);text-decoration:none;transition:color .15s}a:hover{color:var(--color-accent-600)}.fade-enter-active,.fade-leave-active{transition:opacity .2s}.fade-enter-from,.fade-leave-to{opacity:0}@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-space-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-backdrop-blur{syntax:"*";inherits:false}@property --tw-backdrop-brightness{syntax:"*";inherits:false}@property --tw-backdrop-contrast{syntax:"*";inherits:false}@property --tw-backdrop-grayscale{syntax:"*";inherits:false}@property --tw-backdrop-hue-rotate{syntax:"*";inherits:false}@property --tw-backdrop-invert{syntax:"*";inherits:false}@property --tw-backdrop-opacity{syntax:"*";inherits:false}@property --tw-backdrop-saturate{syntax:"*";inherits:false}@property --tw-backdrop-sepia{syntax:"*";inherits:false}@property --tw-duration{syntax:"*";inherits:false}@property --tw-ease{syntax:"*";inherits:false}
|