@tokenbuddy/tokenbuddy 1.0.25 → 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.
Files changed (84) hide show
  1. package/bin/tb-clawtip-proof.js +2 -0
  2. package/dist/src/clawtip-bootstrap.d.ts +1 -0
  3. package/dist/src/clawtip-bootstrap.d.ts.map +1 -1
  4. package/dist/src/clawtip-bootstrap.js +1 -0
  5. package/dist/src/clawtip-bootstrap.js.map +1 -1
  6. package/dist/src/cli.d.ts +1 -0
  7. package/dist/src/cli.d.ts.map +1 -1
  8. package/dist/src/cli.js +172 -51
  9. package/dist/src/cli.js.map +1 -1
  10. package/dist/src/daemon.d.ts +6 -0
  11. package/dist/src/daemon.d.ts.map +1 -1
  12. package/dist/src/daemon.js +562 -292
  13. package/dist/src/daemon.js.map +1 -1
  14. package/dist/src/init-clawtip-activation.d.ts +12 -0
  15. package/dist/src/init-clawtip-activation.d.ts.map +1 -1
  16. package/dist/src/init-clawtip-activation.js +82 -2
  17. package/dist/src/init-clawtip-activation.js.map +1 -1
  18. package/dist/src/package-update.d.ts +60 -0
  19. package/dist/src/package-update.d.ts.map +1 -0
  20. package/dist/src/package-update.js +220 -0
  21. package/dist/src/package-update.js.map +1 -0
  22. package/dist/src/registry-trust.d.ts +7 -0
  23. package/dist/src/registry-trust.d.ts.map +1 -0
  24. package/dist/src/registry-trust.js +37 -0
  25. package/dist/src/registry-trust.js.map +1 -0
  26. package/dist/src/route-failover.d.ts +2 -2
  27. package/dist/src/route-failover.d.ts.map +1 -1
  28. package/dist/src/route-failover.js +11 -0
  29. package/dist/src/route-failover.js.map +1 -1
  30. package/dist/src/seller-catalog.d.ts +20 -0
  31. package/dist/src/seller-catalog.d.ts.map +1 -1
  32. package/dist/src/seller-catalog.js +41 -4
  33. package/dist/src/seller-catalog.js.map +1 -1
  34. package/dist/src/seller-concurrency-limiter.d.ts +36 -0
  35. package/dist/src/seller-concurrency-limiter.d.ts.map +1 -0
  36. package/dist/src/seller-concurrency-limiter.js +126 -0
  37. package/dist/src/seller-concurrency-limiter.js.map +1 -0
  38. package/dist/src/seller-pool.d.ts +7 -1
  39. package/dist/src/seller-pool.d.ts.map +1 -1
  40. package/dist/src/seller-pool.js +18 -0
  41. package/dist/src/seller-pool.js.map +1 -1
  42. package/dist/src/seller-route-planner.d.ts +21 -0
  43. package/dist/src/seller-route-planner.d.ts.map +1 -1
  44. package/dist/src/seller-route-planner.js +98 -20
  45. package/dist/src/seller-route-planner.js.map +1 -1
  46. package/dist/src/tb-clawtip-proof.d.ts +3 -0
  47. package/dist/src/tb-clawtip-proof.d.ts.map +1 -0
  48. package/dist/src/tb-clawtip-proof.js +24 -0
  49. package/dist/src/tb-clawtip-proof.js.map +1 -0
  50. package/dist/src/tb-proxyd.js +45 -3
  51. package/dist/src/tb-proxyd.js.map +1 -1
  52. package/package.json +3 -2
  53. package/src/clawtip-bootstrap.ts +1 -0
  54. package/src/cli.ts +200 -47
  55. package/src/daemon.ts +347 -50
  56. package/src/init-clawtip-activation.ts +77 -1
  57. package/src/package-update.ts +313 -0
  58. package/src/registry-trust.ts +51 -0
  59. package/src/route-failover.ts +14 -2
  60. package/src/seller-catalog.ts +67 -4
  61. package/src/seller-concurrency-limiter.ts +161 -0
  62. package/src/seller-pool.ts +20 -0
  63. package/src/seller-route-planner.ts +142 -20
  64. package/src/tb-clawtip-proof.ts +28 -0
  65. package/src/tb-proxyd.ts +48 -3
  66. package/static/ui/assets/index-Bzbrp7Qe.css +1 -0
  67. package/static/ui/assets/index-UAfOhbwC.js +236 -0
  68. package/static/ui/assets/index-UAfOhbwC.js.map +1 -0
  69. package/static/ui/index.html +2 -2
  70. package/tests/cli-routing.test.ts +37 -4
  71. package/tests/control-plane-ui-endpoints.test.ts +7 -7
  72. package/tests/daemon-trusted-registry-cache.test.ts +132 -0
  73. package/tests/e2e.test.ts +14 -1
  74. package/tests/package-update.test.ts +132 -0
  75. package/tests/registry-trust.test.ts +28 -0
  76. package/tests/route-failover.test.ts +13 -0
  77. package/tests/seller-catalog-413.test.ts +60 -1
  78. package/tests/seller-concurrency-limiter.test.ts +83 -0
  79. package/tests/seller-pool.test.ts +23 -0
  80. package/tests/seller-route-planner.test.ts +78 -0
  81. package/tests/tokenbuddy.test.ts +316 -34
  82. package/static/ui/assets/index-1uuyCCzj.css +0 -1
  83. package/static/ui/assets/index-BJSOFJIU.js +0 -236
  84. package/static/ui/assets/index-BJSOFJIU.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
+ }
@@ -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 prewarmCandidates = prewarm
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
- .filter(isSelectableCandidate);
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: indexed.ordered
232
- .filter((entry) => !metrics.blockedSellerIds.has(entry.seller.id))
233
- .map((entry) => buildCandidate({
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 blockedSellerIds = new Set((metrics ?? [])
288
- .filter((metric) => metric.circuit === "open" || isCapacityBlocked(metric, now))
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 || "https://tb-wallet-bootstrap.fly.dev/registry/sellers";
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)}}