@tokenbuddy/tokenbuddy 1.0.26 → 1.0.27
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/bin/tb-clawtip-proof.js +2 -0
- package/dist/src/clawtip-bootstrap.d.ts +1 -0
- package/dist/src/clawtip-bootstrap.d.ts.map +1 -1
- package/dist/src/clawtip-bootstrap.js +1 -0
- package/dist/src/clawtip-bootstrap.js.map +1 -1
- package/dist/src/cli.d.ts +1 -0
- package/dist/src/cli.d.ts.map +1 -1
- package/dist/src/cli.js +172 -51
- package/dist/src/cli.js.map +1 -1
- package/dist/src/daemon.d.ts +6 -0
- package/dist/src/daemon.d.ts.map +1 -1
- package/dist/src/daemon.js +562 -292
- package/dist/src/daemon.js.map +1 -1
- package/dist/src/init-clawtip-activation.d.ts +5 -0
- package/dist/src/init-clawtip-activation.d.ts.map +1 -1
- package/dist/src/init-clawtip-activation.js +61 -1
- package/dist/src/init-clawtip-activation.js.map +1 -1
- package/dist/src/package-update.d.ts +60 -0
- package/dist/src/package-update.d.ts.map +1 -0
- package/dist/src/package-update.js +220 -0
- package/dist/src/package-update.js.map +1 -0
- package/dist/src/registry-trust.d.ts +7 -0
- package/dist/src/registry-trust.d.ts.map +1 -0
- package/dist/src/registry-trust.js +37 -0
- package/dist/src/registry-trust.js.map +1 -0
- package/dist/src/route-failover.d.ts +2 -2
- package/dist/src/route-failover.d.ts.map +1 -1
- package/dist/src/route-failover.js +11 -0
- package/dist/src/route-failover.js.map +1 -1
- package/dist/src/seller-catalog.d.ts +20 -0
- package/dist/src/seller-catalog.d.ts.map +1 -1
- package/dist/src/seller-catalog.js +41 -4
- package/dist/src/seller-catalog.js.map +1 -1
- package/dist/src/seller-concurrency-limiter.d.ts +36 -0
- package/dist/src/seller-concurrency-limiter.d.ts.map +1 -0
- package/dist/src/seller-concurrency-limiter.js +126 -0
- package/dist/src/seller-concurrency-limiter.js.map +1 -0
- package/dist/src/seller-pool.d.ts +7 -1
- package/dist/src/seller-pool.d.ts.map +1 -1
- package/dist/src/seller-pool.js +18 -0
- package/dist/src/seller-pool.js.map +1 -1
- package/dist/src/seller-route-planner.d.ts +21 -0
- package/dist/src/seller-route-planner.d.ts.map +1 -1
- package/dist/src/seller-route-planner.js +98 -20
- package/dist/src/seller-route-planner.js.map +1 -1
- package/dist/src/tb-clawtip-proof.d.ts +3 -0
- package/dist/src/tb-clawtip-proof.d.ts.map +1 -0
- package/dist/src/tb-clawtip-proof.js +24 -0
- package/dist/src/tb-clawtip-proof.js.map +1 -0
- package/dist/src/tb-proxyd.js +45 -3
- package/dist/src/tb-proxyd.js.map +1 -1
- package/package.json +3 -2
- package/src/clawtip-bootstrap.ts +1 -0
- package/src/cli.ts +200 -47
- package/src/daemon.ts +347 -50
- package/src/init-clawtip-activation.ts +77 -1
- package/src/package-update.ts +313 -0
- package/src/registry-trust.ts +51 -0
- package/src/route-failover.ts +14 -2
- package/src/seller-catalog.ts +67 -4
- package/src/seller-concurrency-limiter.ts +161 -0
- package/src/seller-pool.ts +20 -0
- package/src/seller-route-planner.ts +142 -20
- package/src/tb-clawtip-proof.ts +28 -0
- package/src/tb-proxyd.ts +48 -3
- package/static/ui/assets/index-Bzbrp7Qe.css +1 -0
- package/static/ui/assets/index-UAfOhbwC.js +236 -0
- package/static/ui/assets/index-UAfOhbwC.js.map +1 -0
- package/static/ui/index.html +2 -2
- package/tests/cli-routing.test.ts +37 -4
- package/tests/control-plane-ui-endpoints.test.ts +7 -7
- package/tests/daemon-trusted-registry-cache.test.ts +132 -0
- package/tests/e2e.test.ts +14 -1
- package/tests/package-update.test.ts +132 -0
- package/tests/registry-trust.test.ts +28 -0
- package/tests/route-failover.test.ts +13 -0
- package/tests/seller-catalog-413.test.ts +60 -1
- package/tests/seller-concurrency-limiter.test.ts +83 -0
- package/tests/seller-pool.test.ts +23 -0
- package/tests/seller-route-planner.test.ts +78 -0
- package/tests/tokenbuddy.test.ts +316 -34
- package/static/ui/assets/index-1uuyCCzj.css +0 -1
- package/static/ui/assets/index-cm_EgQZ-.js +0 -236
- package/static/ui/assets/index-cm_EgQZ-.js.map +0 -1
|
@@ -0,0 +1,161 @@
|
|
|
1
|
+
import { createModuleLogger } from "@tokenbuddy/logging";
|
|
2
|
+
|
|
3
|
+
const logger = createModuleLogger("tb-proxyd:seller-concurrency");
|
|
4
|
+
|
|
5
|
+
const DEFAULT_MAX_IN_FLIGHT_PER_SELLER = 2;
|
|
6
|
+
const DEFAULT_LEASE_TTL_MS = 185_000;
|
|
7
|
+
|
|
8
|
+
export interface SellerConcurrencyLimiterOptions {
|
|
9
|
+
enabled?: boolean;
|
|
10
|
+
maxInFlightPerSeller?: number;
|
|
11
|
+
leaseTtlMs?: number;
|
|
12
|
+
}
|
|
13
|
+
|
|
14
|
+
export interface SellerConcurrencyLease {
|
|
15
|
+
sellerId: string;
|
|
16
|
+
activeCount: number;
|
|
17
|
+
maxInFlight: number;
|
|
18
|
+
refresh(): void;
|
|
19
|
+
release(): void;
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
export interface SellerConcurrencySnapshot {
|
|
23
|
+
enabled: boolean;
|
|
24
|
+
maxInFlightPerSeller: number;
|
|
25
|
+
leaseTtlMs: number;
|
|
26
|
+
active: Array<{ sellerId: string; activeCount: number }>;
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
export class SellerConcurrencyLimiter {
|
|
30
|
+
private readonly enabled: boolean;
|
|
31
|
+
private readonly maxInFlightPerSeller: number;
|
|
32
|
+
private readonly leaseTtlMs: number;
|
|
33
|
+
private readonly active = new Map<string, number>();
|
|
34
|
+
|
|
35
|
+
constructor(options: SellerConcurrencyLimiterOptions = {}) {
|
|
36
|
+
this.enabled = options.enabled === true;
|
|
37
|
+
this.maxInFlightPerSeller = positiveIntegerOrDefault(
|
|
38
|
+
options.maxInFlightPerSeller,
|
|
39
|
+
DEFAULT_MAX_IN_FLIGHT_PER_SELLER
|
|
40
|
+
);
|
|
41
|
+
this.leaseTtlMs = positiveIntegerOrDefault(options.leaseTtlMs, DEFAULT_LEASE_TTL_MS);
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
isEnabled(): boolean {
|
|
45
|
+
return this.enabled;
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
tryAcquire(sellerId: string, context: { requestId?: string; modelId?: string; endpoint?: string } = {}): SellerConcurrencyLease | undefined {
|
|
49
|
+
if (!this.enabled) {
|
|
50
|
+
return {
|
|
51
|
+
sellerId,
|
|
52
|
+
activeCount: 0,
|
|
53
|
+
maxInFlight: this.maxInFlightPerSeller,
|
|
54
|
+
refresh: () => undefined,
|
|
55
|
+
release: () => undefined
|
|
56
|
+
};
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
const activeCount = this.active.get(sellerId) ?? 0;
|
|
60
|
+
if (activeCount >= this.maxInFlightPerSeller) {
|
|
61
|
+
logger.info("seller_concurrency.local_full", "seller local concurrency limit reached", {
|
|
62
|
+
sellerId,
|
|
63
|
+
requestId: context.requestId,
|
|
64
|
+
model: context.modelId,
|
|
65
|
+
endpoint: context.endpoint,
|
|
66
|
+
activeCount,
|
|
67
|
+
maxInFlight: this.maxInFlightPerSeller
|
|
68
|
+
});
|
|
69
|
+
return undefined;
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
const nextCount = activeCount + 1;
|
|
73
|
+
this.active.set(sellerId, nextCount);
|
|
74
|
+
logger.info("seller_concurrency.lease_acquired", "seller local concurrency lease acquired", {
|
|
75
|
+
sellerId,
|
|
76
|
+
requestId: context.requestId,
|
|
77
|
+
model: context.modelId,
|
|
78
|
+
endpoint: context.endpoint,
|
|
79
|
+
activeCount: nextCount,
|
|
80
|
+
maxInFlight: this.maxInFlightPerSeller
|
|
81
|
+
});
|
|
82
|
+
|
|
83
|
+
let released = false;
|
|
84
|
+
let timer: NodeJS.Timeout | undefined;
|
|
85
|
+
const scheduleExpiry = (): void => {
|
|
86
|
+
timer = setTimeout(() => {
|
|
87
|
+
if (released) {
|
|
88
|
+
return;
|
|
89
|
+
}
|
|
90
|
+
logger.warn("seller_concurrency.lease_expired", "seller local concurrency lease expired", {
|
|
91
|
+
sellerId,
|
|
92
|
+
requestId: context.requestId,
|
|
93
|
+
model: context.modelId,
|
|
94
|
+
endpoint: context.endpoint,
|
|
95
|
+
leaseTtlMs: this.leaseTtlMs
|
|
96
|
+
});
|
|
97
|
+
release();
|
|
98
|
+
}, this.leaseTtlMs);
|
|
99
|
+
timer.unref?.();
|
|
100
|
+
};
|
|
101
|
+
const refresh = (): void => {
|
|
102
|
+
if (released) {
|
|
103
|
+
return;
|
|
104
|
+
}
|
|
105
|
+
if (timer) {
|
|
106
|
+
clearTimeout(timer);
|
|
107
|
+
}
|
|
108
|
+
scheduleExpiry();
|
|
109
|
+
};
|
|
110
|
+
scheduleExpiry();
|
|
111
|
+
|
|
112
|
+
const release = (): void => {
|
|
113
|
+
if (released) {
|
|
114
|
+
return;
|
|
115
|
+
}
|
|
116
|
+
released = true;
|
|
117
|
+
if (timer) {
|
|
118
|
+
clearTimeout(timer);
|
|
119
|
+
timer = undefined;
|
|
120
|
+
}
|
|
121
|
+
const current = this.active.get(sellerId) ?? 0;
|
|
122
|
+
const remaining = Math.max(0, current - 1);
|
|
123
|
+
if (remaining === 0) {
|
|
124
|
+
this.active.delete(sellerId);
|
|
125
|
+
} else {
|
|
126
|
+
this.active.set(sellerId, remaining);
|
|
127
|
+
}
|
|
128
|
+
logger.info("seller_concurrency.lease_released", "seller local concurrency lease released", {
|
|
129
|
+
sellerId,
|
|
130
|
+
requestId: context.requestId,
|
|
131
|
+
model: context.modelId,
|
|
132
|
+
endpoint: context.endpoint,
|
|
133
|
+
activeCount: remaining,
|
|
134
|
+
maxInFlight: this.maxInFlightPerSeller
|
|
135
|
+
});
|
|
136
|
+
};
|
|
137
|
+
|
|
138
|
+
return {
|
|
139
|
+
sellerId,
|
|
140
|
+
activeCount: nextCount,
|
|
141
|
+
maxInFlight: this.maxInFlightPerSeller,
|
|
142
|
+
refresh,
|
|
143
|
+
release
|
|
144
|
+
};
|
|
145
|
+
}
|
|
146
|
+
|
|
147
|
+
snapshot(): SellerConcurrencySnapshot {
|
|
148
|
+
return {
|
|
149
|
+
enabled: this.enabled,
|
|
150
|
+
maxInFlightPerSeller: this.maxInFlightPerSeller,
|
|
151
|
+
leaseTtlMs: this.leaseTtlMs,
|
|
152
|
+
active: Array.from(this.active.entries())
|
|
153
|
+
.sort(([a], [b]) => a.localeCompare(b))
|
|
154
|
+
.map(([sellerId, activeCount]) => ({ sellerId, activeCount }))
|
|
155
|
+
};
|
|
156
|
+
}
|
|
157
|
+
}
|
|
158
|
+
|
|
159
|
+
function positiveIntegerOrDefault(value: number | undefined, fallback: number): number {
|
|
160
|
+
return Number.isInteger(value) && (value as number) > 0 ? value as number : fallback;
|
|
161
|
+
}
|
package/src/seller-pool.ts
CHANGED
|
@@ -23,6 +23,7 @@ export type FailureKind =
|
|
|
23
23
|
| "auth_invalid" // 401/403 token invalid
|
|
24
24
|
| "insufficient_funds" // 402
|
|
25
25
|
| "busy_capacity" // 429 busy_capacity — seller is temporarily full
|
|
26
|
+
| "purchase_failed" // purchase/create or purchase/complete failed
|
|
26
27
|
| "soft_5xx" // 429/5xx/timeout/network
|
|
27
28
|
| "deadline" // buyer deadline exceeded
|
|
28
29
|
| "stream_aborted" // upstream stream broken after first chunk
|
|
@@ -409,6 +410,25 @@ export class SellerPool {
|
|
|
409
410
|
return { entry, freshPurchase, autoPurchaseAvailable };
|
|
410
411
|
}
|
|
411
412
|
|
|
413
|
+
/**
|
|
414
|
+
* Recycle expired open circuits before route planning paths that consume
|
|
415
|
+
* `snapshot()` directly. This keeps the `open -> half_open` recovery path
|
|
416
|
+
* active even when the newer route planner is used instead of `pick()`.
|
|
417
|
+
*/
|
|
418
|
+
recycleOpenCircuits(now: number = this.now()): number {
|
|
419
|
+
let recycled = 0;
|
|
420
|
+
for (const entry of this.entries.values()) {
|
|
421
|
+
if (entry.circuit !== "open") {
|
|
422
|
+
continue;
|
|
423
|
+
}
|
|
424
|
+
const next = this.maybeRecycleFromOpen(entry, now);
|
|
425
|
+
if (next.circuit === "half_open") {
|
|
426
|
+
recycled += 1;
|
|
427
|
+
}
|
|
428
|
+
}
|
|
429
|
+
return recycled;
|
|
430
|
+
}
|
|
431
|
+
|
|
412
432
|
/**
|
|
413
433
|
* Manually mark an entry as `open`. Used by the registry loop when a
|
|
414
434
|
* seller is removed from the registry: the entry lingers for a grace
|
|
@@ -34,6 +34,10 @@ export interface SellerRouteMetric {
|
|
|
34
34
|
circuit?: SellerCircuitState;
|
|
35
35
|
/** 临时容量避让截止时间;大于当前时间时直接剔除候选 */
|
|
36
36
|
capacityBlockedUntil?: number;
|
|
37
|
+
/** 当前 `tb-proxyd` 进程内该 seller 的活跃 lease 数。 */
|
|
38
|
+
localConcurrencyActive?: number;
|
|
39
|
+
/** 当前 `tb-proxyd` 进程内该 seller 的最大活跃 lease 数。 */
|
|
40
|
+
localConcurrencyLimit?: number;
|
|
37
41
|
}
|
|
38
42
|
|
|
39
43
|
/**
|
|
@@ -123,12 +127,39 @@ export interface SellerRoutePlan {
|
|
|
123
127
|
scorer: SellerRoutingPlan["scorer"];
|
|
124
128
|
/** 候选总数(prewarm 阶段剔除了不支持模型 / 协议 / 支付的 seller 之前) */
|
|
125
129
|
candidateCount: number;
|
|
130
|
+
/** 候选构造与排除摘要,用于日志解释本轮决策为什么只剩这些 seller。 */
|
|
131
|
+
diagnostics: SellerRouteDiagnostics;
|
|
132
|
+
}
|
|
133
|
+
|
|
134
|
+
export interface SellerRouteDiagnostics {
|
|
135
|
+
registryVisibleCount: number;
|
|
136
|
+
prewarmCandidateCount: number;
|
|
137
|
+
prewarmUsableCount: number;
|
|
138
|
+
prewarmMissingSellerIds: string[];
|
|
139
|
+
prewarmBlockedSellerIds: string[];
|
|
140
|
+
prewarmIncompatibleSellerIds: string[];
|
|
141
|
+
sourceCandidateCount: number;
|
|
142
|
+
blockedOpenCircuitCount: number;
|
|
143
|
+
blockedCapacityCount: number;
|
|
144
|
+
blockedLocalConcurrencyCount: number;
|
|
145
|
+
blockedSellerIds: string[];
|
|
146
|
+
incompatibleCount: number;
|
|
147
|
+
incompatibleSellerIds: string[];
|
|
126
148
|
}
|
|
127
149
|
|
|
128
150
|
interface CandidateSourceResult {
|
|
129
151
|
source: SellerRouteSource;
|
|
130
152
|
sourceReason: string;
|
|
131
153
|
candidates: RoutingCandidate[];
|
|
154
|
+
incompatibleSellerIds: string[];
|
|
155
|
+
prewarmDiagnostics: PrewarmSourceDiagnostics;
|
|
156
|
+
}
|
|
157
|
+
|
|
158
|
+
interface PrewarmSourceDiagnostics {
|
|
159
|
+
usableCount: number;
|
|
160
|
+
missingSellerIds: string[];
|
|
161
|
+
blockedSellerIds: string[];
|
|
162
|
+
incompatibleSellerIds: string[];
|
|
132
163
|
}
|
|
133
164
|
|
|
134
165
|
interface IndexedSeller {
|
|
@@ -139,6 +170,9 @@ interface IndexedSeller {
|
|
|
139
170
|
interface MetricIndex {
|
|
140
171
|
bySellerId: Map<string, SellerRouteMetric>;
|
|
141
172
|
blockedSellerIds: Set<string>;
|
|
173
|
+
blockedOpenCircuitSellerIds: Set<string>;
|
|
174
|
+
blockedCapacitySellerIds: Set<string>;
|
|
175
|
+
blockedLocalConcurrencySellerIds: Set<string>;
|
|
142
176
|
}
|
|
143
177
|
|
|
144
178
|
/**
|
|
@@ -184,7 +218,8 @@ export function planSellerRouteSet(input: SellerRoutePlannerInput): SellerRouteP
|
|
|
184
218
|
reason: strategyPlan.reason,
|
|
185
219
|
mode: strategyPlan.mode,
|
|
186
220
|
scorer: strategyPlan.scorer,
|
|
187
|
-
candidateCount: source.candidates.length
|
|
221
|
+
candidateCount: source.candidates.length,
|
|
222
|
+
diagnostics: buildDiagnostics(input, indexed, metrics, source)
|
|
188
223
|
};
|
|
189
224
|
}
|
|
190
225
|
|
|
@@ -194,14 +229,19 @@ function chooseCandidateSource(
|
|
|
194
229
|
metrics: MetricIndex
|
|
195
230
|
): CandidateSourceResult {
|
|
196
231
|
const prewarm = input.prewarmCandidates ?? [];
|
|
232
|
+
let prewarmDiagnostics: PrewarmSourceDiagnostics = emptyPrewarmDiagnostics();
|
|
197
233
|
if (prewarm.length > 0) {
|
|
198
|
-
const
|
|
234
|
+
const missingSellerIds: string[] = [];
|
|
235
|
+
const blockedSellerIds: string[] = [];
|
|
236
|
+
const prewarmCandidatesBeforeCompatibility = prewarm
|
|
199
237
|
.map((candidate) => {
|
|
200
238
|
const indexedSeller = indexed.bySellerId.get(candidate.sellerId);
|
|
201
239
|
if (!indexedSeller) {
|
|
240
|
+
missingSellerIds.push(candidate.sellerId);
|
|
202
241
|
return undefined;
|
|
203
242
|
}
|
|
204
243
|
if (metrics.blockedSellerIds.has(indexedSeller.seller.id)) {
|
|
244
|
+
blockedSellerIds.push(candidate.sellerId);
|
|
205
245
|
return undefined;
|
|
206
246
|
}
|
|
207
247
|
return buildCandidate({
|
|
@@ -213,32 +253,43 @@ function chooseCandidateSource(
|
|
|
213
253
|
metric: mergeMetric(metrics.bySellerId.get(candidate.sellerId), candidate)
|
|
214
254
|
});
|
|
215
255
|
})
|
|
216
|
-
.filter((candidate): candidate is RoutingCandidate => Boolean(candidate))
|
|
217
|
-
|
|
256
|
+
.filter((candidate): candidate is RoutingCandidate => Boolean(candidate));
|
|
257
|
+
const prewarmCandidates = prewarmCandidatesBeforeCompatibility.filter(isSelectableCandidate);
|
|
258
|
+
prewarmDiagnostics = {
|
|
259
|
+
usableCount: prewarmCandidates.length,
|
|
260
|
+
missingSellerIds,
|
|
261
|
+
blockedSellerIds,
|
|
262
|
+
incompatibleSellerIds: incompatibleSellerIds(prewarmCandidatesBeforeCompatibility)
|
|
263
|
+
};
|
|
218
264
|
|
|
219
265
|
if (prewarmCandidates.length > 0) {
|
|
220
266
|
return {
|
|
221
267
|
source: "prewarm_cache",
|
|
222
268
|
sourceReason: "prewarm_candidates_compatible",
|
|
223
|
-
candidates: prewarmCandidates
|
|
269
|
+
candidates: prewarmCandidates,
|
|
270
|
+
incompatibleSellerIds: prewarmDiagnostics.incompatibleSellerIds,
|
|
271
|
+
prewarmDiagnostics
|
|
224
272
|
};
|
|
225
273
|
}
|
|
226
274
|
}
|
|
227
275
|
|
|
276
|
+
const registryCandidatesBeforeCompatibility = indexed.ordered
|
|
277
|
+
.filter((entry) => !metrics.blockedSellerIds.has(entry.seller.id))
|
|
278
|
+
.map((entry) => buildCandidate({
|
|
279
|
+
seller: entry.seller,
|
|
280
|
+
registryOrder: entry.registryOrder,
|
|
281
|
+
modelId: input.modelId,
|
|
282
|
+
protocol: input.protocol,
|
|
283
|
+
paymentMethod: input.paymentMethod,
|
|
284
|
+
metric: metrics.bySellerId.get(entry.seller.id)
|
|
285
|
+
}));
|
|
286
|
+
|
|
228
287
|
return {
|
|
229
288
|
source: "registry_fallback",
|
|
230
289
|
sourceReason: prewarm.length > 0 ? "prewarm_no_compatible_candidates" : "prewarm_missing",
|
|
231
|
-
candidates:
|
|
232
|
-
|
|
233
|
-
|
|
234
|
-
seller: entry.seller,
|
|
235
|
-
registryOrder: entry.registryOrder,
|
|
236
|
-
modelId: input.modelId,
|
|
237
|
-
protocol: input.protocol,
|
|
238
|
-
paymentMethod: input.paymentMethod,
|
|
239
|
-
metric: metrics.bySellerId.get(entry.seller.id)
|
|
240
|
-
}))
|
|
241
|
-
.filter(isSelectableCandidate)
|
|
290
|
+
candidates: registryCandidatesBeforeCompatibility.filter(isSelectableCandidate),
|
|
291
|
+
incompatibleSellerIds: incompatibleSellerIds(registryCandidatesBeforeCompatibility),
|
|
292
|
+
prewarmDiagnostics
|
|
242
293
|
};
|
|
243
294
|
}
|
|
244
295
|
|
|
@@ -284,14 +335,77 @@ function indexRegistrySellers(sellers: RegistrySeller[]): {
|
|
|
284
335
|
}
|
|
285
336
|
|
|
286
337
|
function indexMetrics(metrics: SellerRouteMetric[] | undefined, now: number): MetricIndex {
|
|
287
|
-
const
|
|
288
|
-
.filter((metric) => metric.circuit === "open"
|
|
338
|
+
const blockedOpenCircuitSellerIds = new Set((metrics ?? [])
|
|
339
|
+
.filter((metric) => metric.circuit === "open")
|
|
289
340
|
.map((metric) => metric.sellerId));
|
|
341
|
+
const blockedCapacitySellerIds = new Set((metrics ?? [])
|
|
342
|
+
.filter((metric) => isCapacityBlocked(metric, now))
|
|
343
|
+
.map((metric) => metric.sellerId));
|
|
344
|
+
const blockedLocalConcurrencySellerIds = new Set((metrics ?? [])
|
|
345
|
+
.filter(isLocalConcurrencyBlocked)
|
|
346
|
+
.map((metric) => metric.sellerId));
|
|
347
|
+
const blockedSellerIds = new Set([
|
|
348
|
+
...blockedOpenCircuitSellerIds,
|
|
349
|
+
...blockedCapacitySellerIds,
|
|
350
|
+
...blockedLocalConcurrencySellerIds
|
|
351
|
+
]);
|
|
290
352
|
return {
|
|
291
353
|
bySellerId: new Map((metrics ?? [])
|
|
292
354
|
.filter((metric) => !blockedSellerIds.has(metric.sellerId))
|
|
293
355
|
.map((metric) => [metric.sellerId, metric])),
|
|
294
|
-
blockedSellerIds
|
|
356
|
+
blockedSellerIds,
|
|
357
|
+
blockedOpenCircuitSellerIds,
|
|
358
|
+
blockedCapacitySellerIds,
|
|
359
|
+
blockedLocalConcurrencySellerIds
|
|
360
|
+
};
|
|
361
|
+
}
|
|
362
|
+
|
|
363
|
+
function buildDiagnostics(
|
|
364
|
+
input: SellerRoutePlannerInput,
|
|
365
|
+
indexed: ReturnType<typeof indexRegistrySellers>,
|
|
366
|
+
metrics: MetricIndex,
|
|
367
|
+
source: CandidateSourceResult
|
|
368
|
+
): SellerRouteDiagnostics {
|
|
369
|
+
const visibleOpenBlocked = Array.from(metrics.blockedOpenCircuitSellerIds)
|
|
370
|
+
.filter((sellerId) => indexed.bySellerId.has(sellerId));
|
|
371
|
+
const visibleCapacityBlocked = Array.from(metrics.blockedCapacitySellerIds)
|
|
372
|
+
.filter((sellerId) => indexed.bySellerId.has(sellerId));
|
|
373
|
+
const visibleLocalConcurrencyBlocked = Array.from(metrics.blockedLocalConcurrencySellerIds)
|
|
374
|
+
.filter((sellerId) => indexed.bySellerId.has(sellerId));
|
|
375
|
+
const visibleBlockedSellerIds = Array.from(new Set([
|
|
376
|
+
...visibleOpenBlocked,
|
|
377
|
+
...visibleCapacityBlocked,
|
|
378
|
+
...visibleLocalConcurrencyBlocked
|
|
379
|
+
])).sort();
|
|
380
|
+
return {
|
|
381
|
+
registryVisibleCount: indexed.ordered.length,
|
|
382
|
+
prewarmCandidateCount: input.prewarmCandidates?.length ?? 0,
|
|
383
|
+
prewarmUsableCount: source.prewarmDiagnostics.usableCount,
|
|
384
|
+
prewarmMissingSellerIds: [...source.prewarmDiagnostics.missingSellerIds].sort(),
|
|
385
|
+
prewarmBlockedSellerIds: [...source.prewarmDiagnostics.blockedSellerIds].sort(),
|
|
386
|
+
prewarmIncompatibleSellerIds: [...source.prewarmDiagnostics.incompatibleSellerIds].sort(),
|
|
387
|
+
sourceCandidateCount: source.candidates.length,
|
|
388
|
+
blockedOpenCircuitCount: visibleOpenBlocked.length,
|
|
389
|
+
blockedCapacityCount: visibleCapacityBlocked.length,
|
|
390
|
+
blockedLocalConcurrencyCount: visibleLocalConcurrencyBlocked.length,
|
|
391
|
+
blockedSellerIds: visibleBlockedSellerIds,
|
|
392
|
+
incompatibleCount: source.incompatibleSellerIds.length,
|
|
393
|
+
incompatibleSellerIds: [...source.incompatibleSellerIds].sort()
|
|
394
|
+
};
|
|
395
|
+
}
|
|
396
|
+
|
|
397
|
+
function incompatibleSellerIds(candidates: RoutingCandidate[]): string[] {
|
|
398
|
+
return candidates
|
|
399
|
+
.filter((candidate) => !isSelectableCandidate(candidate))
|
|
400
|
+
.map((candidate) => candidate.sellerId);
|
|
401
|
+
}
|
|
402
|
+
|
|
403
|
+
function emptyPrewarmDiagnostics(): PrewarmSourceDiagnostics {
|
|
404
|
+
return {
|
|
405
|
+
usableCount: 0,
|
|
406
|
+
missingSellerIds: [],
|
|
407
|
+
blockedSellerIds: [],
|
|
408
|
+
incompatibleSellerIds: []
|
|
295
409
|
};
|
|
296
410
|
}
|
|
297
411
|
|
|
@@ -307,7 +421,9 @@ function mergeMetric(
|
|
|
307
421
|
avgInferenceMs: metric?.avgInferenceMs,
|
|
308
422
|
discountRatio: metric?.discountRatio,
|
|
309
423
|
circuit: metric?.circuit,
|
|
310
|
-
capacityBlockedUntil: metric?.capacityBlockedUntil
|
|
424
|
+
capacityBlockedUntil: metric?.capacityBlockedUntil,
|
|
425
|
+
localConcurrencyActive: metric?.localConcurrencyActive,
|
|
426
|
+
localConcurrencyLimit: metric?.localConcurrencyLimit
|
|
311
427
|
};
|
|
312
428
|
}
|
|
313
429
|
|
|
@@ -315,6 +431,12 @@ function isCapacityBlocked(metric: SellerRouteMetric, now: number): boolean {
|
|
|
315
431
|
return Number.isFinite(metric.capacityBlockedUntil) && (metric.capacityBlockedUntil as number) > now;
|
|
316
432
|
}
|
|
317
433
|
|
|
434
|
+
function isLocalConcurrencyBlocked(metric: SellerRouteMetric): boolean {
|
|
435
|
+
return Number.isFinite(metric.localConcurrencyActive) &&
|
|
436
|
+
Number.isFinite(metric.localConcurrencyLimit) &&
|
|
437
|
+
(metric.localConcurrencyActive as number) >= (metric.localConcurrencyLimit as number);
|
|
438
|
+
}
|
|
439
|
+
|
|
318
440
|
function sellerSupportsModel(seller: RegistrySeller, modelId: string): boolean {
|
|
319
441
|
const normalized = normalizeLookupValue(modelId);
|
|
320
442
|
return (seller.models ?? []).some((model) => normalizeLookupValue(model) === normalized);
|
|
@@ -0,0 +1,28 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
import { createClawtipPaymentProof } from "./init-clawtip-activation.js";
|
|
3
|
+
|
|
4
|
+
async function readStdin(): Promise<string> {
|
|
5
|
+
const chunks: Buffer[] = [];
|
|
6
|
+
for await (const chunk of process.stdin) {
|
|
7
|
+
chunks.push(Buffer.isBuffer(chunk) ? chunk : Buffer.from(String(chunk)));
|
|
8
|
+
}
|
|
9
|
+
return Buffer.concat(chunks).toString("utf8");
|
|
10
|
+
}
|
|
11
|
+
|
|
12
|
+
async function main(): Promise<void> {
|
|
13
|
+
const input = (await readStdin()).trim();
|
|
14
|
+
if (!input) {
|
|
15
|
+
throw new Error("ClawTip proof provider requires a JSON payload on stdin");
|
|
16
|
+
}
|
|
17
|
+
const payload = JSON.parse(input) as unknown;
|
|
18
|
+
const proof = await createClawtipPaymentProof(
|
|
19
|
+
payload && typeof payload === "object" ? payload as { paymentInstructions?: unknown; quote?: unknown } : {},
|
|
20
|
+
);
|
|
21
|
+
process.stdout.write(`${proof}\n`);
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
main().catch((error: unknown) => {
|
|
25
|
+
const message = error instanceof Error ? error.message : String(error);
|
|
26
|
+
process.stderr.write(`${message}\n`);
|
|
27
|
+
process.exitCode = 1;
|
|
28
|
+
});
|
package/src/tb-proxyd.ts
CHANGED
|
@@ -2,22 +2,25 @@ import { TokenbuddyDaemon } from "./daemon.js";
|
|
|
2
2
|
import { createModuleLogger } from "@tokenbuddy/logging";
|
|
3
3
|
import { resolveBuyerStorePath } from "./buyer-store.js";
|
|
4
4
|
import { normalizeSellerRoutingConfig, parseSellerRoutingEnv } from "./seller-routing-config.js";
|
|
5
|
+
import { DEFAULT_SELLER_REGISTRY_URL } from "./registry-trust.js";
|
|
5
6
|
|
|
6
7
|
const logger = createModuleLogger("tb-proxyd");
|
|
7
8
|
|
|
8
9
|
const dbPath = resolveBuyerStorePath();
|
|
9
10
|
const controlPort = parsePortEnv("TB_PROXYD_CONTROL_PORT", 17820);
|
|
10
11
|
const proxyPort = parsePortEnv("TB_PROXYD_PROXY_PORT", 17821);
|
|
11
|
-
const sellerRegistryUrl = process.env.TB_PROXYD_SELLER_REGISTRY_URL ||
|
|
12
|
+
const sellerRegistryUrl = process.env.TB_PROXYD_SELLER_REGISTRY_URL || DEFAULT_SELLER_REGISTRY_URL;
|
|
12
13
|
const sellerRoutingEnv = parseSellerRoutingEnv();
|
|
13
14
|
const sellerRouting = sellerRoutingEnv ? normalizeSellerRoutingConfig(sellerRoutingEnv) : undefined;
|
|
15
|
+
const sellerConcurrency = parseSellerConcurrencyEnv();
|
|
14
16
|
|
|
15
17
|
const daemon = new TokenbuddyDaemon({
|
|
16
18
|
controlPort,
|
|
17
19
|
proxyPort,
|
|
18
20
|
dbPath,
|
|
19
21
|
sellerRegistryUrl,
|
|
20
|
-
...(sellerRouting ? { sellerRouting } : {})
|
|
22
|
+
...(sellerRouting ? { sellerRouting } : {}),
|
|
23
|
+
...(sellerConcurrency ? { sellerConcurrency } : {})
|
|
21
24
|
});
|
|
22
25
|
|
|
23
26
|
logger.info("proxy.process.initializing", "tb-proxyd process initializing", {
|
|
@@ -27,7 +30,9 @@ logger.info("proxy.process.initializing", "tb-proxyd process initializing", {
|
|
|
27
30
|
sellerRegistryUrl,
|
|
28
31
|
sellerRoutingEnvOverride: Boolean(sellerRouting),
|
|
29
32
|
sellerRoutingMode: sellerRouting?.mode,
|
|
30
|
-
sellerRoutingScorer: sellerRouting?.scorer
|
|
33
|
+
sellerRoutingScorer: sellerRouting?.scorer,
|
|
34
|
+
sellerConcurrencyEnabled: sellerConcurrency?.enabled ?? false,
|
|
35
|
+
sellerMaxInFlight: sellerConcurrency?.maxInFlightPerSeller
|
|
31
36
|
});
|
|
32
37
|
daemon.start();
|
|
33
38
|
|
|
@@ -54,3 +59,43 @@ function parsePortEnv(name: string, fallback: number): number {
|
|
|
54
59
|
}
|
|
55
60
|
return port;
|
|
56
61
|
}
|
|
62
|
+
|
|
63
|
+
function parseSellerConcurrencyEnv(): { enabled: boolean; maxInFlightPerSeller?: number; leaseTtlMs?: number } | undefined {
|
|
64
|
+
const enabled = parseBooleanEnv("TB_PROXYD_SELLER_CONCURRENCY_ENABLED");
|
|
65
|
+
const maxInFlightPerSeller = parsePositiveIntegerEnv("TB_PROXYD_SELLER_MAX_IN_FLIGHT");
|
|
66
|
+
const leaseTtlMs = parsePositiveIntegerEnv("TB_PROXYD_SELLER_LEASE_TTL_MS");
|
|
67
|
+
if (enabled === undefined && maxInFlightPerSeller === undefined && leaseTtlMs === undefined) {
|
|
68
|
+
return undefined;
|
|
69
|
+
}
|
|
70
|
+
return {
|
|
71
|
+
enabled: enabled ?? false,
|
|
72
|
+
...(maxInFlightPerSeller !== undefined ? { maxInFlightPerSeller } : {}),
|
|
73
|
+
...(leaseTtlMs !== undefined ? { leaseTtlMs } : {})
|
|
74
|
+
};
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
function parseBooleanEnv(name: string): boolean | undefined {
|
|
78
|
+
const rawValue = process.env[name]?.trim().toLowerCase();
|
|
79
|
+
if (!rawValue) {
|
|
80
|
+
return undefined;
|
|
81
|
+
}
|
|
82
|
+
if (["1", "true", "yes", "on"].includes(rawValue)) {
|
|
83
|
+
return true;
|
|
84
|
+
}
|
|
85
|
+
if (["0", "false", "no", "off"].includes(rawValue)) {
|
|
86
|
+
return false;
|
|
87
|
+
}
|
|
88
|
+
throw new Error(`${name} must be one of 1, true, yes, on, 0, false, no, off`);
|
|
89
|
+
}
|
|
90
|
+
|
|
91
|
+
function parsePositiveIntegerEnv(name: string): number | undefined {
|
|
92
|
+
const rawValue = process.env[name];
|
|
93
|
+
if (!rawValue) {
|
|
94
|
+
return undefined;
|
|
95
|
+
}
|
|
96
|
+
const parsed = Number(rawValue);
|
|
97
|
+
if (!Number.isInteger(parsed) || parsed <= 0) {
|
|
98
|
+
throw new Error(`${name} must be a positive integer`);
|
|
99
|
+
}
|
|
100
|
+
return parsed;
|
|
101
|
+
}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
/*! tailwindcss v4.3.0 | 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-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-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-content:""}}}@layer theme{:root,:host{--font-sans:Inter, Geist, ui-sans-serif, system-ui, -apple-system, BlinkMacSystemFont, "Segoe UI", sans-serif;--color-red-100:oklch(93.6% .032 17.717);--color-amber-50:oklch(98.7% .022 95.277);--color-amber-100:oklch(96.2% .059 95.617);--color-amber-700:oklch(55.5% .163 48.998);--color-amber-800:oklch(47.3% .137 46.201);--color-emerald-50:oklch(97.9% .021 166.113);--color-emerald-100:oklch(95% .052 163.051);--color-emerald-200:oklch(90.5% .093 164.15);--color-emerald-500:oklch(69.6% .17 162.48);--color-emerald-600:oklch(59.6% .145 163.225);--color-emerald-700:oklch(50.8% .118 165.612);--color-emerald-800:oklch(43.2% .095 166.913);--color-emerald-900:oklch(37.8% .077 168.94);--color-blue-50:oklch(97% .014 254.604);--color-blue-700:oklch(48.8% .243 264.376);--color-rose-50:oklch(96.9% .015 12.422);--color-rose-100:oklch(94.1% .03 12.58);--color-rose-200:oklch(89.2% .058 10.001);--color-rose-700:oklch(51.4% .222 16.935);--color-slate-50:oklch(98.4% .003 247.858);--color-slate-100:oklch(96.8% .007 247.896);--color-slate-200:oklch(92.9% .013 255.508);--color-slate-300:oklch(86.9% .022 252.894);--color-slate-500:oklch(55.4% .046 257.417);--color-slate-600:oklch(44.6% .043 257.281);--color-slate-700:oklch(37.2% .044 257.287);--color-white:#fff;--spacing:.25rem;--font-weight-medium:500;--font-weight-semibold:600;--font-weight-bold:700;--font-weight-black:900;--tracking-tight:-.025em;--tracking-normal:0em;--tracking-wider:.05em;--leading-snug:1.375;--radius-sm:6px;--radius-md:8px;--radius-lg:10px;--radius-xl:12px;--animate-spin:spin 1s linear infinite;--blur-sm:8px;--default-transition-duration:.15s;--default-transition-timing-function:cubic-bezier(.4, 0, .2, 1);--default-font-family:Inter, Geist, ui-sans-serif, system-ui, -apple-system, BlinkMacSystemFont, "Segoe UI", sans-serif;--default-mono-font-family:"Geist Mono", ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, monospace;--color-page:#f8f7ff;--color-ink:#201a38;--color-text:#312a4f;--color-muted:#6b6384;--color-faint:#8d86a3;--color-line:#ded6f2;--color-line-2:#eee9fb;--color-hairline:#eee9fb;--color-hairline-strong:#ded6f2;--color-lavender:#f1ebff;--color-lavender-2:#ded6f2;--color-primary:#7c3df0;--color-router:#6d42e8;--color-spend:#0f766e;--color-spend-soft:#e8fbf7;--color-spend-border:#cceee6;--color-tokens:#2563eb;--color-tokens-soft:#eaf2ff;--color-tokens-border:#d9e7ff;--color-inventory:#9333ea;--color-inventory-soft:#f7ecff;--color-inventory-border:#ead5ff;--color-success:#10b981;--color-success-soft:#e8fff6;--color-warning:#f59e0b;--color-warning-soft:#fff7df;--color-danger:#ef5b78;--color-danger-soft:#fff0f3;--color-chart-input:#7c3df0;--color-chart-cost:#2563eb;--color-chart-grid:#ece6f8;--radius-pill:9999px;--shadow-panel:0 12px 36px #3c297014;--shadow-panel-strong:0 14px 30px #7c3df024;--shadow-chip:0 12px 28px #291f540f}}@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{.pointer-events-none{pointer-events:none}.sr-only{clip-path:inset(50%);white-space:nowrap;border-width:0;width:1px;height:1px;margin:-1px;padding:0;position:absolute;overflow:hidden}.absolute{position:absolute}.fixed{position:fixed}.relative{position:relative}.static{position:static}.sticky{position:sticky}.inset-0{inset:calc(var(--spacing) * 0)}.inset-y-0{inset-block:calc(var(--spacing) * 0)}.top-0{top:calc(var(--spacing) * 0)}.top-1\/2{top:50%}.left-0{left:calc(var(--spacing) * 0)}.left-3{left:calc(var(--spacing) * 3)}.z-10{z-index:10}.z-30{z-index:30}.z-50{z-index:50}.mx-auto{margin-inline:auto}.mt-0\.5{margin-top:calc(var(--spacing) * .5)}.mt-1{margin-top:calc(var(--spacing) * 1)}.mt-1\.5{margin-top:calc(var(--spacing) * 1.5)}.mt-2{margin-top:calc(var(--spacing) * 2)}.mt-3{margin-top:calc(var(--spacing) * 3)}.mt-4{margin-top:calc(var(--spacing) * 4)}.mt-5{margin-top:calc(var(--spacing) * 5)}.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)}.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}.contents{display:contents}.flex{display:flex}.grid{display:grid}.hidden{display:none}.inline-block{display:inline-block}.inline-flex{display:inline-flex}.size-1\.5{width:calc(var(--spacing) * 1.5);height:calc(var(--spacing) * 1.5)}.size-2{width:calc(var(--spacing) * 2);height:calc(var(--spacing) * 2)}.size-2\.5{width:calc(var(--spacing) * 2.5);height:calc(var(--spacing) * 2.5)}.size-3{width:calc(var(--spacing) * 3);height:calc(var(--spacing) * 3)}.size-3\.5{width:calc(var(--spacing) * 3.5);height:calc(var(--spacing) * 3.5)}.size-4{width:calc(var(--spacing) * 4);height:calc(var(--spacing) * 4)}.size-5{width:calc(var(--spacing) * 5);height:calc(var(--spacing) * 5)}.size-6{width:calc(var(--spacing) * 6);height:calc(var(--spacing) * 6)}.size-7{width:calc(var(--spacing) * 7);height:calc(var(--spacing) * 7)}.size-8{width:calc(var(--spacing) * 8);height:calc(var(--spacing) * 8)}.size-9{width:calc(var(--spacing) * 9);height:calc(var(--spacing) * 9)}.size-11{width:calc(var(--spacing) * 11);height:calc(var(--spacing) * 11)}.size-12{width:calc(var(--spacing) * 12);height:calc(var(--spacing) * 12)}.size-full{width:100%;height:100%}.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-12{height:calc(var(--spacing) * 12)}.h-\[60px\]{height:60px}.h-\[66px\]{height:66px}.h-\[160px\]{height:160px}.h-px{height:1px}.max-h-\[72vh\]{max-height:72vh}.max-h-\[86dvh\]{max-height:86dvh}.max-h-\[92px\]{max-height:92px}.max-h-\[180px\]{max-height:180px}.max-h-\[300px\]{max-height:300px}.max-h-\[320px\]{max-height:320px}.max-h-\[340px\]{max-height:340px}.max-h-\[380px\]{max-height:380px}.max-h-\[calc\(86dvh-65px\)\]{max-height:calc(86dvh - 65px)}.max-h-full{max-height:100%}.min-h-0{min-height:calc(var(--spacing) * 0)}.min-h-5{min-height:calc(var(--spacing) * 5)}.min-h-9{min-height:calc(var(--spacing) * 9)}.min-h-14{min-height:calc(var(--spacing) * 14)}.min-h-\[7rem\]{min-height:7rem}.min-h-\[64px\]{min-height:64px}.min-h-\[72px\]{min-height:72px}.min-h-\[86px\]{min-height:86px}.min-h-\[94px\]{min-height:94px}.min-h-\[124px\]{min-height:124px}.min-h-\[220px\]{min-height:220px}.min-h-\[320px\]{min-height:320px}.min-h-\[360px\]{min-height:360px}.min-h-full{min-height:100%}.w-1{width:calc(var(--spacing) * 1)}.w-4{width:calc(var(--spacing) * 4)}.w-10{width:calc(var(--spacing) * 10)}.w-12{width:calc(var(--spacing) * 12)}.w-14{width:calc(var(--spacing) * 14)}.w-fit{width:fit-content}.w-full{width:100%}.max-w-\[9rem\]{max-width:9rem}.max-w-\[150px\]{max-width:150px}.max-w-\[160px\]{max-width:160px}.max-w-\[180px\]{max-width:180px}.max-w-\[190px\]{max-width:190px}.max-w-\[210px\]{max-width:210px}.max-w-\[220px\]{max-width:220px}.max-w-\[230px\]{max-width:230px}.max-w-\[240px\]{max-width:240px}.max-w-\[340px\]{max-width:340px}.max-w-\[360px\]{max-width:360px}.max-w-\[720px\]{max-width:720px}.max-w-\[820px\]{max-width:820px}.max-w-\[1180px\]{max-width:1180px}.max-w-full{max-width:100%}.min-w-0{min-width:calc(var(--spacing) * 0)}.min-w-8{min-width:calc(var(--spacing) * 8)}.min-w-\[190px\]{min-width:190px}.min-w-\[760px\]{min-width:760px}.min-w-\[980px\]{min-width:980px}.min-w-\[1080px\]{min-width:1080px}.min-w-\[1180px\]{min-width:1180px}.flex-1{flex:1}.shrink-0{flex-shrink:0}.border-collapse{border-collapse:collapse}.-translate-y-1\/2{--tw-translate-y: -50% ;translate:var(--tw-translate-x) var(--tw-translate-y)}.animate-spin{animation:var(--animate-spin)}.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-\[auto_minmax\(0\,1fr\)_auto\]{grid-template-columns:auto minmax(0,1fr) auto}.grid-cols-\[minmax\(0\,1fr\)_74px_82px\]{grid-template-columns:minmax(0,1fr) 74px 82px}.flex-col{flex-direction:column}.flex-wrap{flex-wrap:wrap}.place-items-center{place-items:center}.content-start{align-content:flex-start}.items-center{align-items:center}.items-end{align-items:flex-end}.items-start{align-items:flex-start}.items-stretch{align-items:stretch}.justify-between{justify-content:space-between}.justify-center{justify-content:center}.justify-end{justify-content:flex-end}.justify-start{justify-content:flex-start}.gap-0\.5{gap:calc(var(--spacing) * .5)}.gap-1{gap:calc(var(--spacing) * 1)}.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-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)))}: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-5>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing) * 5) * var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing) * 5) * calc(1 - var(--tw-space-y-reverse)))}.gap-x-2{column-gap:calc(var(--spacing) * 2)}.gap-x-2\.5{column-gap:calc(var(--spacing) * 2.5)}.gap-x-4{column-gap:calc(var(--spacing) * 4)}.gap-y-1{row-gap:calc(var(--spacing) * 1)}.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-\[var\(--color-line-2\)\]>:not(:last-child)){border-color:var(--color-line-2)}.truncate{text-overflow:ellipsis;white-space:nowrap;overflow:hidden}.overflow-auto{overflow:auto}.overflow-hidden{overflow:hidden}.overflow-x-auto{overflow-x:auto}.overflow-x-hidden{overflow-x:hidden}.overflow-y-auto{overflow-y:auto}.rounded{border-radius:12px}.rounded-\[5px\]{border-radius:5px}.rounded-\[6px\]{border-radius:6px}.rounded-\[7px\]{border-radius:7px}.rounded-\[8px\]{border-radius:8px}.rounded-\[9px\]{border-radius:9px}.rounded-\[10px\]{border-radius:10px}.rounded-\[12px\]{border-radius:12px}.rounded-\[14px\]{border-radius:14px}.rounded-\[var\(--radius-lg\)\]{border-radius:var(--radius-lg)}.rounded-\[var\(--radius-md\)\]{border-radius:var(--radius-md)}.rounded-\[var\(--radius-pill\)\]{border-radius:var(--radius-pill)}.rounded-\[var\(--radius-sm\)\]{border-radius:var(--radius-sm)}.rounded-\[var\(--radius-xl\)\]{border-radius:var(--radius-xl)}.rounded-full{border-radius:3.40282e38px}.rounded-lg{border-radius:10px}.rounded-xl{border-radius:12px}.rounded-t-\[3px\]{border-top-left-radius:3px;border-top-right-radius:3px}.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-\[\#cceee6\]{border-color:#cceee6}.border-\[\#d9e7ff\]{border-color:#d9e7ff}.border-\[\#dff8ec\]{border-color:#dff8ec}.border-\[\#f8d5de\]{border-color:#f8d5de}.border-\[var\(--color-danger\)\]\/30{border-color:#ef5b784d}@supports (color:color-mix(in lab,red,red)){.border-\[var\(--color-danger\)\]\/30{border-color:color-mix(in oklab,var(--color-danger) 30%,transparent)}}.border-\[var\(--color-hairline\)\]{border-color:var(--color-hairline)}.border-\[var\(--color-hairline-strong\)\]{border-color:var(--color-hairline-strong)}.border-\[var\(--color-ink\)\]{border-color:var(--color-ink)}.border-\[var\(--color-inventory-border\)\]{border-color:var(--color-inventory-border)}.border-\[var\(--color-lavender-2\)\]{border-color:var(--color-lavender-2)}.border-\[var\(--color-line\)\]{border-color:var(--color-line)}.border-\[var\(--color-line-2\)\]{border-color:var(--color-line-2)}.border-\[var\(--color-purple\)\]{border-color:var(--color-purple)}.border-\[var\(--color-spend-border\)\]{border-color:var(--color-spend-border)}.border-\[var\(--color-tokens-border\)\]{border-color:var(--color-tokens-border)}.border-amber-100{border-color:var(--color-amber-100)}.border-emerald-100{border-color:var(--color-emerald-100)}.border-emerald-200{border-color:var(--color-emerald-200)}.border-red-100{border-color:var(--color-red-100)}.border-rose-100{border-color:var(--color-rose-100)}.border-rose-200{border-color:var(--color-rose-200)}.bg-\[\#0f766e\]{background-color:#0f766e}.bg-\[\#201a38\]\/35{background-color:#201a3859}.bg-\[\#2563eb\]{background-color:#2563eb}.bg-\[\#e8fbf7\]{background-color:#e8fbf7}.bg-\[\#e8fff6\]{background-color:#e8fff6}.bg-\[\#e1251b\]{background-color:#e1251b}.bg-\[\#eaf2ff\]{background-color:#eaf2ff}.bg-\[\#ef5b78\]{background-color:#ef5b78}.bg-\[\#fff0f3\]{background-color:#fff0f3}.bg-\[rgba\(32\,26\,56\,0\.22\)\]{background-color:#201a3838}.bg-\[rgba\(32\,26\,56\,0\.28\)\]{background-color:#201a3847}.bg-\[var\(--color-card\)\]{background-color:var(--color-card)}.bg-\[var\(--color-danger\)\]{background-color:var(--color-danger)}.bg-\[var\(--color-danger-soft\)\]{background-color:var(--color-danger-soft)}.bg-\[var\(--color-ink\)\]{background-color:var(--color-ink)}.bg-\[var\(--color-inventory-soft\)\]{background-color:var(--color-inventory-soft)}.bg-\[var\(--color-lavender\)\]{background-color:var(--color-lavender)}.bg-\[var\(--color-lavender\)\]\/35{background-color:#f1ebff59}@supports (color:color-mix(in lab,red,red)){.bg-\[var\(--color-lavender\)\]\/35{background-color:color-mix(in oklab,var(--color-lavender) 35%,transparent)}}.bg-\[var\(--color-lavender\)\]\/55{background-color:#f1ebff8c}@supports (color:color-mix(in lab,red,red)){.bg-\[var\(--color-lavender\)\]\/55{background-color:color-mix(in oklab,var(--color-lavender) 55%,transparent)}}.bg-\[var\(--color-lavender\)\]\/70{background-color:#f1ebffb3}@supports (color:color-mix(in lab,red,red)){.bg-\[var\(--color-lavender\)\]\/70{background-color:color-mix(in oklab,var(--color-lavender) 70%,transparent)}}.bg-\[var\(--color-page\)\]{background-color:var(--color-page)}.bg-\[var\(--color-page\)\]\/55{background-color:#f8f7ff8c}@supports (color:color-mix(in lab,red,red)){.bg-\[var\(--color-page\)\]\/55{background-color:color-mix(in oklab,var(--color-page) 55%,transparent)}}.bg-\[var\(--color-page\)\]\/60{background-color:#f8f7ff99}@supports (color:color-mix(in lab,red,red)){.bg-\[var\(--color-page\)\]\/60{background-color:color-mix(in oklab,var(--color-page) 60%,transparent)}}.bg-\[var\(--color-page\)\]\/70{background-color:#f8f7ffb3}@supports (color:color-mix(in lab,red,red)){.bg-\[var\(--color-page\)\]\/70{background-color:color-mix(in oklab,var(--color-page) 70%,transparent)}}.bg-\[var\(--color-page\)\]\/95{background-color:#f8f7fff2}@supports (color:color-mix(in lab,red,red)){.bg-\[var\(--color-page\)\]\/95{background-color:color-mix(in oklab,var(--color-page) 95%,transparent)}}.bg-\[var\(--color-primary\)\]{background-color:var(--color-primary)}.bg-\[var\(--color-purple\)\]{background-color:var(--color-purple)}.bg-\[var\(--color-spend-soft\)\]{background-color:var(--color-spend-soft)}.bg-\[var\(--color-success\)\]{background-color:var(--color-success)}.bg-\[var\(--color-success-soft\)\]{background-color:var(--color-success-soft)}.bg-\[var\(--color-tokens\)\]\/75{background-color:#2563ebbf}@supports (color:color-mix(in lab,red,red)){.bg-\[var\(--color-tokens\)\]\/75{background-color:color-mix(in oklab,var(--color-tokens) 75%,transparent)}}.bg-\[var\(--color-tokens-soft\)\]{background-color:var(--color-tokens-soft)}.bg-\[var\(--color-warning\)\]{background-color:var(--color-warning)}.bg-\[var\(--color-warning-soft\)\]{background-color:var(--color-warning-soft)}.bg-amber-50{background-color:var(--color-amber-50)}.bg-blue-50{background-color:var(--color-blue-50)}.bg-current{background-color:currentColor}.bg-emerald-50{background-color:var(--color-emerald-50)}.bg-emerald-500{background-color:var(--color-emerald-500)}.bg-rose-50{background-color:var(--color-rose-50)}.bg-slate-50{background-color:var(--color-slate-50)}.bg-slate-100{background-color:var(--color-slate-100)}.bg-slate-300{background-color:var(--color-slate-300)}.bg-transparent{background-color:#0000}.bg-white{background-color:var(--color-white)}.bg-white\/70{background-color:#ffffffb3}@supports (color:color-mix(in lab,red,red)){.bg-white\/70{background-color:color-mix(in oklab,var(--color-white) 70%,transparent)}}.fill-\[var\(--color-muted\)\]{fill:var(--color-muted)}.object-contain{object-fit:contain}.p-0{padding:calc(var(--spacing) * 0)}.p-1{padding:calc(var(--spacing) * 1)}.p-1\.5{padding:calc(var(--spacing) * 1.5)}.p-2\.5{padding:calc(var(--spacing) * 2.5)}.p-3{padding:calc(var(--spacing) * 3)}.p-3\.5{padding:calc(var(--spacing) * 3.5)}.p-4{padding:calc(var(--spacing) * 4)}.p-5{padding:calc(var(--spacing) * 5)}.px-1{padding-inline:calc(var(--spacing) * 1)}.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)}.px-6{padding-inline:calc(var(--spacing) * 6)}.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{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-4{padding-block:calc(var(--spacing) * 4)}.py-5{padding-block:calc(var(--spacing) * 5)}.py-6{padding-block:calc(var(--spacing) * 6)}.py-7{padding-block:calc(var(--spacing) * 7)}.py-8{padding-block:calc(var(--spacing) * 8)}.py-10{padding-block:calc(var(--spacing) * 10)}.pt-3{padding-top:calc(var(--spacing) * 3)}.pr-1{padding-right:calc(var(--spacing) * 1)}.pr-3{padding-right:calc(var(--spacing) * 3)}.pb-3{padding-bottom:calc(var(--spacing) * 3)}.pl-9{padding-left:calc(var(--spacing) * 9)}.text-center{text-align:center}.text-left{text-align:left}.text-right{text-align:right}.align-top{vertical-align:top}.font-mono{font-family:Geist Mono,ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,monospace}.text-\[0\.7rem\]{font-size:.7rem}.text-\[0\.8rem\]{font-size:.8rem}.text-\[0\.9rem\]{font-size:.9rem}.text-\[0\.58rem\]{font-size:.58rem}.text-\[0\.62rem\]{font-size:.62rem}.text-\[0\.65rem\]{font-size:.65rem}.text-\[0\.66rem\]{font-size:.66rem}.text-\[0\.68rem\]{font-size:.68rem}.text-\[0\.72rem\]{font-size:.72rem}.text-\[0\.73rem\]{font-size:.73rem}.text-\[0\.74rem\]{font-size:.74rem}.text-\[0\.75rem\]{font-size:.75rem}.text-\[0\.76rem\]{font-size:.76rem}.text-\[0\.78rem\]{font-size:.78rem}.text-\[0\.82rem\]{font-size:.82rem}.text-\[0\.84rem\]{font-size:.84rem}.text-\[0\.85rem\]{font-size:.85rem}.text-\[0\.86rem\]{font-size:.86rem}.text-\[0\.88rem\]{font-size:.88rem}.text-\[0\.92rem\]{font-size:.92rem}.text-\[0\.94rem\]{font-size:.94rem}.text-\[1\.05rem\]{font-size:1.05rem}.text-\[1\.25rem\]{font-size:1.25rem}.text-\[1\.55rem\]{font-size:1.55rem}.text-\[1\.95rem\]{font-size:1.95rem}.text-\[1rem\]{font-size:1rem}.text-\[10px\]{font-size:10px}.text-\[12px\]{font-size:12px}.leading-3{--tw-leading:calc(var(--spacing) * 3);line-height:calc(var(--spacing) * 3)}.leading-5{--tw-leading:calc(var(--spacing) * 5);line-height:calc(var(--spacing) * 5)}.leading-6{--tw-leading:calc(var(--spacing) * 6);line-height:calc(var(--spacing) * 6)}.leading-none{--tw-leading:1;line-height:1}.leading-snug{--tw-leading:var(--leading-snug);line-height:var(--leading-snug)}.font-black{--tw-font-weight:var(--font-weight-black);font-weight:var(--font-weight-black)}.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-\[0\.08em\]{--tw-tracking:.08em;letter-spacing:.08em}.tracking-\[0\.14em\]{--tw-tracking:.14em;letter-spacing:.14em}.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}.text-\[\#0f766e\]{color:#0f766e}.text-\[\#201a38\]{color:#201a38}.text-\[\#2563eb\]{color:#2563eb}.text-\[\#ef5b78\]{color:#ef5b78}.text-\[var\(--color-danger\)\]{color:var(--color-danger)}.text-\[var\(--color-faint\)\]{color:var(--color-faint)}.text-\[var\(--color-ink\)\]{color:var(--color-ink)}.text-\[var\(--color-inventory\)\]{color:var(--color-inventory)}.text-\[var\(--color-muted\)\]{color:var(--color-muted)}.text-\[var\(--color-primary\)\]{color:var(--color-primary)}.text-\[var\(--color-purple\)\],.text-\[var\(--color-purple\)\]\/70{color:var(--color-purple)}@supports (color:color-mix(in lab,red,red)){.text-\[var\(--color-purple\)\]\/70{color:color-mix(in oklab,var(--color-purple) 70%,transparent)}}.text-\[var\(--color-router\)\]{color:var(--color-router)}.text-\[var\(--color-spend\)\]{color:var(--color-spend)}.text-\[var\(--color-success\)\]{color:var(--color-success)}.text-\[var\(--color-text\)\]{color:var(--color-text)}.text-\[var\(--color-tokens\)\]{color:var(--color-tokens)}.text-\[var\(--color-warning\)\]{color:var(--color-warning)}.text-amber-700{color:var(--color-amber-700)}.text-amber-800{color:var(--color-amber-800)}.text-blue-700{color:var(--color-blue-700)}.text-emerald-600{color:var(--color-emerald-600)}.text-emerald-700{color:var(--color-emerald-700)}.text-emerald-800{color:var(--color-emerald-800)}.text-emerald-900{color:var(--color-emerald-900)}.text-rose-700{color:var(--color-rose-700)}.text-rose-700\/70{color:#c20039b3}@supports (color:color-mix(in lab,red,red)){.text-rose-700\/70{color:color-mix(in oklab,var(--color-rose-700) 70%,transparent)}}.text-slate-500{color:var(--color-slate-500)}.text-slate-600{color:var(--color-slate-600)}.text-slate-700{color:var(--color-slate-700)}.text-transparent{color:#0000}.text-white{color:var(--color-white)}.uppercase{text-transform:uppercase}.underline{text-decoration-line:underline}.decoration-\[var\(--color-primary\)\]{-webkit-text-decoration-color:var(--color-primary);text-decoration-color:var(--color-primary)}.decoration-2{text-decoration-thickness:2px}.underline-offset-\[10px\]{text-underline-offset:10px}.opacity-45{opacity:.45}.opacity-60{opacity:.6}.opacity-70{opacity:.7}.shadow-\[0_0_0_3px_rgba\(16\,185\,129\,0\.18\)\]{--tw-shadow:0 0 0 3px var(--tw-shadow-color,#10b9812e);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.shadow-\[0_0_0_3px_rgba\(124\,61\,240\,0\.1\)\]{--tw-shadow:0 0 0 3px var(--tw-shadow-color,#7c3df01a);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.shadow-\[0_0_0_3px_rgba\(124\,61\,240\,0\.18\)\]{--tw-shadow:0 0 0 3px var(--tw-shadow-color,#7c3df02e);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.shadow-\[0_0_0_3px_rgba\(203\,213\,225\,0\.25\)\]{--tw-shadow:0 0 0 3px var(--tw-shadow-color,#cbd5e140);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.shadow-\[0_0_0_3px_rgba\(239\,91\,120\,0\.18\)\]{--tw-shadow:0 0 0 3px var(--tw-shadow-color,#ef5b782e);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.shadow-\[0_0_0_3px_rgba\(245\,158\,11\,0\.18\)\]{--tw-shadow:0 0 0 3px var(--tw-shadow-color,#f59e0b2e);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.shadow-\[0_8px_18px_rgba\(124\,61\,240\,0\.16\)\]{--tw-shadow:0 8px 18px var(--tw-shadow-color,#7c3df029);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.shadow-\[0_8px_18px_rgba\(225\,37\,27\,0\.16\)\]{--tw-shadow:0 8px 18px var(--tw-shadow-color,#e1251b29);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.shadow-\[0_8px_22px_rgba\(60\,41\,112\,0\.045\)\]{--tw-shadow:0 8px 22px var(--tw-shadow-color,#3c29700b);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.shadow-\[0_8px_22px_rgba\(124\,61\,240\,0\.08\)\]{--tw-shadow:0 8px 22px var(--tw-shadow-color,#7c3df014);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.shadow-\[0_10px_24px_rgba\(60\,41\,112\,0\.08\)\]{--tw-shadow:0 10px 24px var(--tw-shadow-color,#3c297014);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.shadow-\[0_12px_28px_rgba\(15\,23\,42\,0\.1\)\]{--tw-shadow:0 12px 28px var(--tw-shadow-color,#0f172a1a);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.shadow-\[0_20px_56px_rgba\(32\,26\,56\,0\.20\)\]{--tw-shadow:0 20px 56px var(--tw-shadow-color,#201a3833);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.shadow-\[0_24px_70px_rgba\(32\,26\,56\,0\.22\)\]{--tw-shadow:0 24px 70px var(--tw-shadow-color,#201a3838);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.shadow-\[0_24px_80px_rgba\(32\,26\,56\,0\.22\)\]{--tw-shadow:0 24px 80px var(--tw-shadow-color,#201a3838);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.shadow-\[var\(--shadow-chip\)\]{--tw-shadow:var(--shadow-chip);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.shadow-\[var\(--shadow-panel\)\]{--tw-shadow:var(--shadow-panel);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.shadow-\[var\(--shadow-panel-strong\)\]{--tw-shadow:var(--shadow-panel-strong);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.shadow-sm{--tw-shadow:0 1px 3px 0 var(--tw-shadow-color,#0000001a), 0 1px 2px -1px 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)}.ring-1{--tw-ring-shadow:var(--tw-ring-inset,) 0 0 0 calc(1px + var(--tw-ring-offset-width)) var(--tw-ring-color,currentcolor);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.ring-2{--tw-ring-shadow:var(--tw-ring-inset,) 0 0 0 calc(2px + var(--tw-ring-offset-width)) var(--tw-ring-color,currentcolor);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.ring-\[var\(--color-lavender-2\)\]{--tw-ring-color:var(--color-lavender-2)}.ring-emerald-100{--tw-ring-color:var(--color-emerald-100)}.ring-slate-200{--tw-ring-color:var(--color-slate-200)}.ring-offset-1{--tw-ring-offset-width:1px;--tw-ring-offset-shadow:var(--tw-ring-inset,) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color)}.ring-offset-white{--tw-ring-offset-color:var(--color-white)}.backdrop-blur{--tw-backdrop-blur:blur(8px);-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,)}.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{transition-property:color,background-color,border-color,outline-color,text-decoration-color,fill,stroke,--tw-gradient-from,--tw-gradient-via,--tw-gradient-to,opacity,box-shadow,transform,translate,scale,rotate,filter,-webkit-backdrop-filter,backdrop-filter,display,content-visibility,overlay,pointer-events;transition-timing-function:var(--tw-ease,var(--default-transition-timing-function));transition-duration:var(--tw-duration,var(--default-transition-duration))}.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))}.outline-none{--tw-outline-style:none;outline-style:none}.placeholder\:text-\[var\(--color-faint\)\]::placeholder{color:var(--color-faint)}.placeholder\:text-\[var\(--color-muted\)\]::placeholder{color:var(--color-muted)}.before\:absolute:before{content:var(--tw-content);position:absolute}.before\:inset-y-0:before{content:var(--tw-content);inset-block:calc(var(--spacing) * 0)}.before\:left-0:before{content:var(--tw-content);left:calc(var(--spacing) * 0)}.before\:w-1:before{content:var(--tw-content);width:calc(var(--spacing) * 1)}.before\:bg-\[var\(--color-inventory\)\]:before{content:var(--tw-content);background-color:var(--color-inventory)}.before\:bg-\[var\(--color-router\)\]:before{content:var(--tw-content);background-color:var(--color-router)}.before\:bg-\[var\(--color-spend\)\]:before{content:var(--tw-content);background-color:var(--color-spend)}.before\:bg-\[var\(--color-success\)\]:before{content:var(--tw-content);background-color:var(--color-success)}.before\:bg-\[var\(--color-tokens\)\]:before{content:var(--tw-content);background-color:var(--color-tokens)}.last\:border-b-0:last-child{border-bottom-style:var(--tw-border-style);border-bottom-width:0}@media(hover:hover){.hover\:-translate-y-0\.5:hover{--tw-translate-y:calc(var(--spacing) * -.5);translate:var(--tw-translate-x) var(--tw-translate-y)}.hover\:border-\[var\(--color-inventory\)\]:hover{border-color:var(--color-inventory)}.hover\:border-\[var\(--color-lavender-2\)\]:hover{border-color:var(--color-lavender-2)}.hover\:border-\[var\(--color-line\)\]:hover{border-color:var(--color-line)}.hover\:border-\[var\(--color-primary\)\]:hover{border-color:var(--color-primary)}.hover\:border-\[var\(--color-purple\)\]:hover{border-color:var(--color-purple)}.hover\:border-\[var\(--color-spend\)\]:hover{border-color:var(--color-spend)}.hover\:border-\[var\(--color-success\)\]:hover{border-color:var(--color-success)}.hover\:border-\[var\(--color-tokens\)\]:hover{border-color:var(--color-tokens)}.hover\:bg-\[\#6e32dc\]:hover{background-color:#6e32dc}.hover\:bg-\[var\(--color-lavender\)\]:hover{background-color:var(--color-lavender)}.hover\:bg-\[var\(--color-lavender\)\]\/25:hover{background-color:#f1ebff40}@supports (color:color-mix(in lab,red,red)){.hover\:bg-\[var\(--color-lavender\)\]\/25:hover{background-color:color-mix(in oklab,var(--color-lavender) 25%,transparent)}}.hover\:bg-\[var\(--color-lavender\)\]\/35:hover{background-color:#f1ebff59}@supports (color:color-mix(in lab,red,red)){.hover\:bg-\[var\(--color-lavender\)\]\/35:hover{background-color:color-mix(in oklab,var(--color-lavender) 35%,transparent)}}.hover\:bg-\[var\(--color-lavender\)\]\/40:hover{background-color:#f1ebff66}@supports (color:color-mix(in lab,red,red)){.hover\:bg-\[var\(--color-lavender\)\]\/40:hover{background-color:color-mix(in oklab,var(--color-lavender) 40%,transparent)}}.hover\:bg-\[var\(--color-lavender\)\]\/45:hover{background-color:#f1ebff73}@supports (color:color-mix(in lab,red,red)){.hover\:bg-\[var\(--color-lavender\)\]\/45:hover{background-color:color-mix(in oklab,var(--color-lavender) 45%,transparent)}}.hover\:bg-\[var\(--color-page\)\]:hover{background-color:var(--color-page)}.hover\:bg-\[var\(--color-purple\)\]:hover{background-color:var(--color-purple)}.hover\:bg-\[var\(--color-purple-2\)\]:hover{background-color:var(--color-purple-2)}.hover\:bg-rose-50:hover{background-color:var(--color-rose-50)}.hover\:bg-white:hover{background-color:var(--color-white)}.hover\:text-\[var\(--color-ink\)\]:hover{color:var(--color-ink)}.hover\:text-\[var\(--color-purple\)\]:hover{color:var(--color-purple)}.hover\:text-rose-700:hover{color:var(--color-rose-700)}}.focus\:border-\[var\(--color-purple\)\]:focus{border-color:var(--color-purple)}.focus\:ring-2:focus{--tw-ring-shadow:var(--tw-ring-inset,) 0 0 0 calc(2px + var(--tw-ring-offset-width)) var(--tw-ring-color,currentcolor);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.focus\:ring-\[var\(--color-primary\)\]:focus{--tw-ring-color:var(--color-primary)}.focus\:outline-none:focus{--tw-outline-style:none;outline-style:none}.active\:translate-y-px:active{--tw-translate-y:1px;translate:var(--tw-translate-x) var(--tw-translate-y)}.disabled\:cursor-not-allowed:disabled{cursor:not-allowed}.disabled\:cursor-wait:disabled{cursor:wait}.disabled\:opacity-40:disabled{opacity:.4}.disabled\:opacity-45:disabled{opacity:.45}.disabled\:opacity-70:disabled{opacity:.7}@media(min-width:40rem){.sm\:flex{display:flex}.sm\:min-h-\[560px\]{min-height:560px}.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\:grid-cols-\[150px_minmax\(0\,1fr\)\]{grid-template-columns:150px minmax(0,1fr)}.sm\:grid-cols-\[minmax\(0\,1fr\)_96px_92px\]{grid-template-columns:minmax(0,1fr) 96px 92px}.sm\:grid-cols-\[minmax\(0\,1fr\)_190px\]{grid-template-columns:minmax(0,1fr) 190px}.sm\:flex-row{flex-direction:row}.sm\:items-center{align-items:center}.sm\:items-start{align-items:flex-start}.sm\:justify-between{justify-content:space-between}.sm\:gap-1{gap:calc(var(--spacing) * 1)}.sm\:gap-3{gap:calc(var(--spacing) * 3)}.sm\:gap-5{gap:calc(var(--spacing) * 5)}.sm\:p-4{padding:calc(var(--spacing) * 4)}.sm\:p-6{padding:calc(var(--spacing) * 6)}.sm\:px-3{padding-inline:calc(var(--spacing) * 3)}.sm\:px-3\.5{padding-inline:calc(var(--spacing) * 3.5)}.sm\:px-6{padding-inline:calc(var(--spacing) * 6)}.sm\:py-7{padding-block:calc(var(--spacing) * 7)}.sm\:text-\[0\.92rem\]{font-size:.92rem}}@media(min-width:48rem){.md\:grid-cols-3{grid-template-columns:repeat(3,minmax(0,1fr))}}@media(min-width:64rem){.lg\:max-w-\[360px\]{max-width:360px}.lg\:grid-cols-2{grid-template-columns:repeat(2,minmax(0,1fr))}.lg\:grid-cols-3{grid-template-columns:repeat(3,minmax(0,1fr))}.lg\:grid-cols-6{grid-template-columns:repeat(6,minmax(0,1fr))}.lg\:grid-cols-\[minmax\(0\,1\.05fr\)_minmax\(0\,2fr\)\]{grid-template-columns:minmax(0,1.05fr) minmax(0,2fr)}.lg\:grid-cols-\[minmax\(0\,300px\)_minmax\(0\,1fr\)\]{grid-template-columns:minmax(0,300px) minmax(0,1fr)}.lg\:grid-cols-\[minmax\(220px\,1fr\)_160px_200px\]{grid-template-columns:minmax(220px,1fr) 160px 200px}.lg\:flex-row{flex-direction:row}.lg\:items-center{align-items:center}.lg\:justify-between{justify-content:space-between}}@media(min-width:80rem){.xl\:block{display:block}.xl\:hidden{display:none}.xl\:grid-cols-2{grid-template-columns:repeat(2,minmax(0,1fr))}.xl\:grid-cols-4{grid-template-columns:repeat(4,minmax(0,1fr))}.xl\:grid-cols-6{grid-template-columns:repeat(6,minmax(0,1fr))}.xl\:grid-cols-\[minmax\(0\,290px\)_minmax\(0\,1fr\)\]{grid-template-columns:minmax(0,290px) minmax(0,1fr)}.xl\:grid-cols-\[minmax\(0\,320px\)_minmax\(0\,1fr\)\]{grid-template-columns:minmax(0,320px) minmax(0,1fr)}}}html,body,#root{height:100%}body{color:var(--color-text);background:var(--color-page);font-family:var(--font-sans);font-variant-numeric:tabular-nums;margin:0}button,a,input,select{transition-duration:.18s}.font-mono,code,pre,td,th{font-variant-numeric:tabular-nums}@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-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-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-content{syntax:"*";inherits:false;initial-value:""}@keyframes spin{to{transform:rotate(360deg)}}
|