@tokenbuddy/tokenbuddy 1.0.16 → 1.0.18
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-proxyd.js +2 -0
- package/bin/tb.js +4 -0
- package/dist/src/cli.d.ts +52 -1
- package/dist/src/cli.d.ts.map +1 -1
- package/dist/src/cli.js +178 -8
- package/dist/src/cli.js.map +1 -1
- package/dist/src/daemon.d.ts +8 -0
- package/dist/src/daemon.d.ts.map +1 -1
- package/dist/src/daemon.js +335 -21
- package/dist/src/daemon.js.map +1 -1
- package/dist/src/init-setup.d.ts +25 -0
- package/dist/src/init-setup.d.ts.map +1 -0
- package/dist/src/init-setup.js +125 -0
- package/dist/src/init-setup.js.map +1 -0
- package/package.json +1 -1
- package/src/cli.ts +233 -10
- package/src/daemon.ts +385 -20
- package/src/init-setup.ts +165 -0
- package/static/ui/assets/index-BILQcD8g.js +226 -0
- package/static/ui/assets/index-BILQcD8g.js.map +1 -0
- package/static/ui/assets/index-C5KsebCA.css +1 -0
- package/static/ui/index.html +2 -2
- package/static/ui/manifest.webmanifest +1 -1
- package/static/ui/sw.js +1 -1
- package/tests/control-plane-ui-endpoints.test.ts +193 -1
- package/tests/tokenbuddy.test.ts +209 -0
- package/static/ui/assets/index-UMiTTeo8.css +0 -1
- package/static/ui/assets/index-YHs-Ca0f.js +0 -206
- package/static/ui/assets/index-YHs-Ca0f.js.map +0 -1
|
@@ -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:"";--tw-scale-x:1;--tw-scale-y:1;--tw-scale-z:1}}}@layer theme{:root,:host{--font-sans:ui-sans-serif, system-ui, -apple-system, BlinkMacSystemFont, "Segoe UI", sans-serif;--color-amber-50:oklch(98.7% .022 95.277);--color-amber-100:oklch(96.2% .059 95.617);--color-amber-200:oklch(92.4% .12 95.746);--color-amber-500:oklch(76.9% .188 70.08);--color-amber-700:oklch(55.5% .163 48.998);--color-amber-800:oklch(47.3% .137 46.201);--color-amber-950:oklch(27.9% .077 45.635);--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-emerald-950:oklch(26.2% .051 172.552);--color-teal-50:oklch(98.4% .014 180.72);--color-teal-100:oklch(95.3% .051 180.801);--color-teal-200:oklch(91% .096 180.426);--color-teal-600:oklch(60% .118 184.704);--color-teal-700:oklch(51.1% .096 186.391);--color-teal-900:oklch(38.6% .063 188.416);--color-blue-50:oklch(97% .014 254.604);--color-blue-700:oklch(48.8% .243 264.376);--color-indigo-50:oklch(96.2% .018 272.314);--color-indigo-100:oklch(93% .034 272.788);--color-indigo-600:oklch(51.1% .262 276.966);--color-indigo-700:oklch(45.7% .24 277.023);--color-indigo-900:oklch(35.9% .144 278.697);--color-violet-100:oklch(94.3% .029 294.588);--color-violet-200:oklch(89.4% .057 293.283);--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-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-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-lg:.5rem;--radius-xl:.75rem;--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:ui-sans-serif, system-ui, -apple-system, BlinkMacSystemFont, "Segoe UI", sans-serif;--default-mono-font-family:ui-monospace, SFMono-Regular, Menlo, monospace;--color-page:#f8f7ff;--color-ink:#201a38;--color-text:#312a4f;--color-muted:#665f80;--color-faint:#837d9b;--color-line:#ddd6f1;--color-line-2:#eee9fb;--color-lavender:#efe8ff;--color-lavender-2:#e3d7ff;--color-purple:#7c3df0;--color-purple-2:#9b6dff;--color-green:#34d399;--color-amber:#f6b73c;--color-red:#ef5b78}}@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-0\.5{margin-left:calc(var(--spacing) * .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-\[120px\]{min-height:120px}.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-\[180px\]{max-width:180px}.max-w-\[190px\]{max-width:190px}.max-w-\[210px\]{max-width:210px}.max-w-\[220px\]{max-width:220px}.max-w-\[240px\]{max-width:240px}.max-w-\[260px\]{max-width:260px}.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-\[260px\]{min-width:260px}.min-w-\[760px\]{min-width:760px}.min-w-\[980px\]{min-width:980px}.min-w-\[1080px\]{min-width:1080px}.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}.grid-cols-\[minmax\(0\,1fr\)_auto\]{grid-template-columns:minmax(0,1fr) auto}.flex-col{flex-direction:column}.flex-wrap{flex-wrap:wrap}.place-items-center{place-items:center}.content-start{align-content:flex-start}.items-baseline{align-items:baseline}.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\.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:16px}.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-full{border-radius:3.40282e38px}.rounded-lg{border-radius:var(--radius-lg)}.rounded-xl{border-radius:var(--radius-xl)}.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-ink\)\]{border-color:var(--color-ink)}.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-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-indigo-100{border-color:var(--color-indigo-100)}.border-rose-100{border-color:var(--color-rose-100)}.border-rose-200{border-color:var(--color-rose-200)}.border-teal-100{border-color:var(--color-teal-100)}.border-violet-100{border-color:var(--color-violet-100)}.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-\[\#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-amber\)\]{background-color:var(--color-amber)}.bg-\[var\(--color-card\)\]{background-color:var(--color-card)}.bg-\[var\(--color-green\)\]{background-color:var(--color-green)}.bg-\[var\(--color-ink\)\]{background-color:var(--color-ink)}.bg-\[var\(--color-lavender\)\]{background-color:var(--color-lavender)}.bg-\[var\(--color-lavender\)\]\/35{background-color:#efe8ff59}@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:#efe8ff8c}@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:#efe8ffb3}@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\)\]\/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-purple\)\]{background-color:var(--color-purple)}.bg-\[var\(--color-purple\)\]\/75{background-color:#7c3df0bf}@supports (color:color-mix(in lab,red,red)){.bg-\[var\(--color-purple\)\]\/75{background-color:color-mix(in oklab,var(--color-purple) 75%,transparent)}}.bg-\[var\(--color-red\)\]{background-color:var(--color-red)}.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-indigo-50{background-color:var(--color-indigo-50)}.bg-indigo-600{background-color:var(--color-indigo-600)}.bg-rose-50{background-color:var(--color-rose-50)}.bg-slate-100{background-color:var(--color-slate-100)}.bg-slate-300{background-color:var(--color-slate-300)}.bg-teal-50{background-color:var(--color-teal-50)}.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-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-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}.font-mono{font-family:ui-monospace,SFMono-Regular,Menlo,monospace}.text-\[0\.7rem\]{font-size:.7rem}.text-\[0\.8rem\]{font-size:.8rem}.text-\[0\.9rem\]{font-size:.9rem}.text-\[0\.62rem\]{font-size:.62rem}.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\.06em\]{--tw-tracking:.06em;letter-spacing:.06em}.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-ink\)\]{color:var(--color-ink)}.text-\[var\(--color-muted\)\]{color:var(--color-muted)}.text-\[var\(--color-purple\)\]{color:var(--color-purple)}.text-\[var\(--color-purple\)\]\/70{color:#7c3df0b3}@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-purple\)\]\/72{color:#7c3df0b8}@supports (color:color-mix(in lab,red,red)){.text-\[var\(--color-purple\)\]\/72{color:color-mix(in oklab,var(--color-purple) 72%,transparent)}}.text-\[var\(--color-purple-2\)\]{color:var(--color-purple-2)}.text-\[var\(--color-red\)\]{color:var(--color-red)}.text-\[var\(--color-text\)\]{color:var(--color-text)}.text-amber-700{color:var(--color-amber-700)}.text-amber-800{color:var(--color-amber-800)}.text-amber-800\/70{color:#953d00b3}@supports (color:color-mix(in lab,red,red)){.text-amber-800\/70{color:color-mix(in oklab,var(--color-amber-800) 70%,transparent)}}.text-amber-950\/58{color:#46190194}@supports (color:color-mix(in lab,red,red)){.text-amber-950\/58{color:color-mix(in oklab,var(--color-amber-950) 58%,transparent)}}.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-700\/72{color:#007956b8}@supports (color:color-mix(in lab,red,red)){.text-emerald-700\/72{color:color-mix(in oklab,var(--color-emerald-700) 72%,transparent)}}.text-emerald-800{color:var(--color-emerald-800)}.text-emerald-900{color:var(--color-emerald-900)}.text-emerald-950\/58{color:#002c2294}@supports (color:color-mix(in lab,red,red)){.text-emerald-950\/58{color:color-mix(in oklab,var(--color-emerald-950) 58%,transparent)}}.text-indigo-700{color:var(--color-indigo-700)}.text-indigo-900{color:var(--color-indigo-900)}.text-indigo-900\/70{color:#312c85b3}@supports (color:color-mix(in lab,red,red)){.text-indigo-900\/70{color:color-mix(in oklab,var(--color-indigo-900) 70%,transparent)}}.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-teal-700{color:var(--color-teal-700)}.text-teal-700\/72{color:#00776eb8}@supports (color:color-mix(in lab,red,red)){.text-teal-700\/72{color:color-mix(in oklab,var(--color-teal-700) 72%,transparent)}}.text-teal-900\/62{color:#0b4f4a9e}@supports (color:color-mix(in lab,red,red)){.text-teal-900\/62{color:color-mix(in oklab,var(--color-teal-900) 62%,transparent)}}.text-transparent{color:#0000}.text-white{color:var(--color-white)}.uppercase{text-transform:uppercase}.underline{text-decoration-line:underline}.decoration-\[var\(--color-purple\)\]{-webkit-text-decoration-color:var(--color-purple);text-decoration-color:var(--color-purple)}.decoration-2{text-decoration-thickness:2px}.underline-offset-\[10px\]{text-underline-offset:10px}.opacity-45{opacity:.45}.opacity-60{opacity:.6}.shadow-\[0_0_0_3px_rgba\(52\,211\,153\,0\.18\)\]{--tw-shadow:0 0 0 3px var(--tw-shadow-color,#34d3992e);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\(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\(246\,183\,60\,0\.18\)\]{--tw-shadow:0 0 0 3px var(--tw-shadow-color,#f6b73c2e);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\(79\,70\,229\,0\.16\)\]{--tw-shadow:0 8px 18px var(--tw-shadow-color,#4f46e529);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_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\(15\,118\,110\,0\.08\)\]{--tw-shadow:0 10px 24px var(--tw-shadow-color,#0f766e14);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\(16\,185\,129\,0\.08\)\]{--tw-shadow:0 10px 24px var(--tw-shadow-color,#10b98114);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_10px_24px_rgba\(124\,61\,240\,0\.08\)\]{--tw-shadow:0 10px 24px 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\(245\,158\,11\,0\.08\)\]{--tw-shadow:0 10px 24px var(--tw-shadow-color,#f59e0b14);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_12px_28px_rgba\(41\,31\,84\,0\.06\)\]{--tw-shadow:0 12px 28px var(--tw-shadow-color,#291f540f);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_36px_rgba\(60\,41\,112\,0\.08\)\]{--tw-shadow:0 12px 36px 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_14px_30px_rgba\(124\,61\,240\,0\.14\)\]{--tw-shadow:0 14px 30px var(--tw-shadow-color,#7c3df024);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-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-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-purple\)\]:before{content:var(--tw-content);background-color:var(--color-purple)}.before\:bg-amber-500:before{content:var(--tw-content);background-color:var(--color-amber-500)}.before\:bg-emerald-500:before{content:var(--tw-content);background-color:var(--color-emerald-500)}.before\:bg-teal-600:before{content:var(--tw-content);background-color:var(--color-teal-600)}.after\:absolute:after{content:var(--tw-content);position:absolute}.after\:-bottom-1:after{content:var(--tw-content);bottom:calc(var(--spacing) * -1)}.after\:left-0:after{content:var(--tw-content);left:calc(var(--spacing) * 0)}.after\:h-0\.5:after{content:var(--tw-content);height:calc(var(--spacing) * .5)}.after\:w-full:after{content:var(--tw-content);width:100%}.after\:origin-left:after{content:var(--tw-content);transform-origin:0}.after\:scale-x-75:after{content:var(--tw-content);--tw-scale-x:75%;scale:var(--tw-scale-x) var(--tw-scale-y)}.after\:rounded-full:after{content:var(--tw-content);border-radius:3.40282e38px}.after\:bg-\[var\(--color-purple\)\]:after{content:var(--tw-content);background-color:var(--color-purple)}.after\:opacity-0:after{content:var(--tw-content);opacity:0}.after\:transition:after{content:var(--tw-content);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))}@media(hover:hover){.group-hover\:after\:scale-x-100:is(:where(.group):hover *):after{content:var(--tw-content);--tw-scale-x:100%;scale:var(--tw-scale-x) var(--tw-scale-y)}.group-hover\:after\:opacity-100:is(:where(.group):hover *):after{content:var(--tw-content);opacity:1}}.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-lavender-2\)\]:hover{border-color:var(--color-lavender-2)}.hover\:border-\[var\(--color-line\)\]:hover{border-color:var(--color-line)}.hover\:border-\[var\(--color-purple\)\]:hover{border-color:var(--color-purple)}.hover\:border-amber-200:hover{border-color:var(--color-amber-200)}.hover\:border-emerald-200:hover{border-color:var(--color-emerald-200)}.hover\:border-teal-200:hover{border-color:var(--color-teal-200)}.hover\:border-violet-200:hover{border-color:var(--color-violet-200)}.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:#efe8ff40}@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:#efe8ff59}@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:#efe8ff66}@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:#efe8ff73}@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-indigo-100:hover{background-color:var(--color-indigo-100)}.hover\:bg-indigo-700:hover{background-color:var(--color-indigo-700)}.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-purple\)\]:focus{--tw-ring-color:var(--color-purple)}.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\:opacity-40:disabled{opacity:.4}.disabled\:opacity-45:disabled{opacity:.45}@media(min-width:40rem){.sm\:mt-0{margin-top:calc(var(--spacing) * 0)}.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-\[150px_minmax\(0\,1fr\)\]{grid-template-columns:150px minmax(0,1fr)}.sm\:grid-cols-\[minmax\(0\,1\.45fr\)_minmax\(260px\,0\.8fr\)\]{grid-template-columns:minmax(0,1.45fr) minmax(260px,.8fr)}.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\.86rem\]{font-size:.86rem}.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\: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-\[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\:col-span-2{grid-column:span 2/span 2}.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)}.xl\:grid-cols-\[minmax\(340px\,380px\)_minmax\(0\,1fr\)\]{grid-template-columns:minmax(340px,380px) minmax(0,1fr)}}}html,body,#root{height:100%}body{color:var(--color-text);background:var(--color-page);font-family:var(--font-sans);margin:0}@property --tw-translate-x{syntax:"*";inherits:false;initial-value:0}@property --tw-translate-y{syntax:"*";inherits:false;initial-value:0}@property --tw-translate-z{syntax:"*";inherits:false;initial-value:0}@property --tw-space-y-reverse{syntax:"*";inherits:false;initial-value:0}@property --tw-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:""}@property --tw-scale-x{syntax:"*";inherits:false;initial-value:1}@property --tw-scale-y{syntax:"*";inherits:false;initial-value:1}@property --tw-scale-z{syntax:"*";inherits:false;initial-value:1}@keyframes spin{to{transform:rotate(360deg)}}
|
package/static/ui/index.html
CHANGED
|
@@ -12,8 +12,8 @@
|
|
|
12
12
|
<link rel="icon" type="image/png" sizes="192x192" href="/icons/tokenbuddy-192.png" />
|
|
13
13
|
<link rel="apple-touch-icon" href="/icons/apple-touch-icon.png" />
|
|
14
14
|
<title>TokenBuddy · Local Control</title>
|
|
15
|
-
<script type="module" crossorigin src="/assets/index-
|
|
16
|
-
<link rel="stylesheet" crossorigin href="/assets/index-
|
|
15
|
+
<script type="module" crossorigin src="/assets/index-BILQcD8g.js"></script>
|
|
16
|
+
<link rel="stylesheet" crossorigin href="/assets/index-C5KsebCA.css">
|
|
17
17
|
</head>
|
|
18
18
|
<body>
|
|
19
19
|
<div id="root"></div>
|
package/static/ui/sw.js
CHANGED
|
@@ -3,6 +3,7 @@ import * as fs from "fs";
|
|
|
3
3
|
import * as path from "path";
|
|
4
4
|
import { AddressInfo } from "net";
|
|
5
5
|
import { TokenbuddyDaemon, type DaemonConfig } from "../src/daemon.js";
|
|
6
|
+
import { DEFAULT_INIT_RECOMMENDED_MODELS } from "../src/init-setup.js";
|
|
6
7
|
import type { SellerRegistryDocument } from "../src/seller-catalog.js";
|
|
7
8
|
|
|
8
9
|
/**
|
|
@@ -70,6 +71,7 @@ describe("TokenbuddyDaemon control-plane UI endpoints (PR-0)", () => {
|
|
|
70
71
|
dbPath: TEMP_DB,
|
|
71
72
|
sellerRegistryUrl: "http://127.0.0.1:1/registry/sellers", // 不会真正被触发
|
|
72
73
|
clawtipHomeDir: TEMP_HOME,
|
|
74
|
+
providerHomeDir: TEMP_HOME,
|
|
73
75
|
...overrides
|
|
74
76
|
});
|
|
75
77
|
daemon.start();
|
|
@@ -314,6 +316,7 @@ describe("TokenbuddyDaemon control-plane UI endpoints (PR-0)", () => {
|
|
|
314
316
|
expect(manifestRes.status).toBe(200);
|
|
315
317
|
expect(manifestRes.headers.get("content-type")).toContain("application/manifest+json");
|
|
316
318
|
const manifest = await manifestRes.json() as { start_url: string; icons: Array<{ src: string; sizes: string }> };
|
|
319
|
+
expect(manifest).toMatchObject({ name: "TokenBuddy", short_name: "TokenBuddy" });
|
|
317
320
|
expect(manifest.start_url).toBe("/overview");
|
|
318
321
|
expect(manifest.icons).toEqual(expect.arrayContaining([
|
|
319
322
|
expect.objectContaining({ src: "/icons/tokenbuddy-192.png", sizes: "192x192" }),
|
|
@@ -323,7 +326,7 @@ describe("TokenbuddyDaemon control-plane UI endpoints (PR-0)", () => {
|
|
|
323
326
|
const serviceWorkerRes = await fetch(controlUrl("/sw.js"));
|
|
324
327
|
expect(serviceWorkerRes.status).toBe(200);
|
|
325
328
|
expect(serviceWorkerRes.headers.get("content-type")).toContain("application/javascript");
|
|
326
|
-
expect(await serviceWorkerRes.text()).toContain("tokenbuddy-ui-
|
|
329
|
+
expect(await serviceWorkerRes.text()).toContain("tokenbuddy-ui-v2");
|
|
327
330
|
|
|
328
331
|
const iconRes = await fetch(controlUrl("/icons/tokenbuddy-192.png"));
|
|
329
332
|
expect(iconRes.status).toBe(200);
|
|
@@ -332,6 +335,195 @@ describe("TokenbuddyDaemon control-plane UI endpoints (PR-0)", () => {
|
|
|
332
335
|
});
|
|
333
336
|
});
|
|
334
337
|
|
|
338
|
+
// ─── Init Wizard State ────────────────────────────────────────
|
|
339
|
+
describe("web init wizard state", () => {
|
|
340
|
+
it("returns a fresh-machine setup snapshot for the web wizard", async () => {
|
|
341
|
+
daemon.stop();
|
|
342
|
+
await startDaemon({ providerHomeDir: TEMP_HOME });
|
|
343
|
+
|
|
344
|
+
const res = await fetch(controlUrl("/init/state"));
|
|
345
|
+
expect(res.status).toBe(200);
|
|
346
|
+
const body = await res.json() as {
|
|
347
|
+
setup: { status: string; completedSteps: string[] };
|
|
348
|
+
freshMachine: boolean;
|
|
349
|
+
runtime: { status: string; controlPort: number; proxyPort: number };
|
|
350
|
+
payments: unknown[];
|
|
351
|
+
clients: Array<{ id: string; name: string }>;
|
|
352
|
+
clientsSummary: { installCommand: string; totalCount: number };
|
|
353
|
+
routing: { strategy: { mode: string; scorer: string }; source: string };
|
|
354
|
+
focusSet: string[];
|
|
355
|
+
recommendedModels: string[];
|
|
356
|
+
};
|
|
357
|
+
|
|
358
|
+
expect(body.setup).toMatchObject({
|
|
359
|
+
status: "not_started",
|
|
360
|
+
completedSteps: []
|
|
361
|
+
});
|
|
362
|
+
expect(body.freshMachine).toBe(true);
|
|
363
|
+
expect(body.runtime).toMatchObject({
|
|
364
|
+
status: "running",
|
|
365
|
+
controlPort,
|
|
366
|
+
proxyPort
|
|
367
|
+
});
|
|
368
|
+
expect(body.payments).toEqual([]);
|
|
369
|
+
expect(body.clients).toEqual(expect.arrayContaining([
|
|
370
|
+
expect.objectContaining({ id: "custom", name: "Custom client" })
|
|
371
|
+
]));
|
|
372
|
+
expect(body.clientsSummary.installCommand).toBe("tb init");
|
|
373
|
+
expect(body.clientsSummary.totalCount).toBe(body.clients.length);
|
|
374
|
+
expect(body.routing.strategy).toMatchObject({ mode: "fullAuto", scorer: "balanced" });
|
|
375
|
+
expect(body.routing.source).toBe("default");
|
|
376
|
+
expect(body.focusSet).toEqual([]);
|
|
377
|
+
expect(body.recommendedModels).toEqual([...DEFAULT_INIT_RECOMMENDED_MODELS]);
|
|
378
|
+
});
|
|
379
|
+
|
|
380
|
+
it("returns configured init recommended models in stable order", async () => {
|
|
381
|
+
daemon.stop();
|
|
382
|
+
await startDaemon({
|
|
383
|
+
providerHomeDir: TEMP_HOME,
|
|
384
|
+
initRecommendedModels: [" custom-a ", "", "custom-b", "custom-a"]
|
|
385
|
+
});
|
|
386
|
+
|
|
387
|
+
const res = await fetch(controlUrl("/init/state"));
|
|
388
|
+
expect(res.status).toBe(200);
|
|
389
|
+
const body = await res.json() as { recommendedModels: string[] };
|
|
390
|
+
|
|
391
|
+
expect(body.recommendedModels).toEqual(["custom-a", "custom-b"]);
|
|
392
|
+
});
|
|
393
|
+
|
|
394
|
+
it("marks the web init wizard as completed and rejects unknown steps", async () => {
|
|
395
|
+
const badRes = await fetch(controlUrl("/init/complete"), {
|
|
396
|
+
method: "POST",
|
|
397
|
+
headers: { "content-type": "application/json" },
|
|
398
|
+
body: JSON.stringify({ completedSteps: ["gateway_intro", "unknown_step"] })
|
|
399
|
+
});
|
|
400
|
+
expect(badRes.status).toBe(400);
|
|
401
|
+
const badBody = await badRes.json() as { error: { code: string; message: string } };
|
|
402
|
+
expect(badBody.error.code).toBe("init_setup_complete_failed");
|
|
403
|
+
expect(badBody.error.message).toContain("unknown init setup step");
|
|
404
|
+
|
|
405
|
+
const completeRes = await fetch(controlUrl("/init/complete"), {
|
|
406
|
+
method: "POST",
|
|
407
|
+
headers: { "content-type": "application/json" },
|
|
408
|
+
body: JSON.stringify({ completedSteps: ["gateway_intro", "model_access"] })
|
|
409
|
+
});
|
|
410
|
+
expect(completeRes.status).toBe(200);
|
|
411
|
+
const completed = await completeRes.json() as {
|
|
412
|
+
setup: { status: string; completedSteps: string[]; completedAt?: string };
|
|
413
|
+
freshMachine: boolean;
|
|
414
|
+
repairMode: boolean;
|
|
415
|
+
repairReasons: string[];
|
|
416
|
+
};
|
|
417
|
+
expect(completed.setup.status).toBe("completed");
|
|
418
|
+
expect(completed.setup.completedSteps).toEqual(["gateway_intro", "model_access"]);
|
|
419
|
+
expect(completed.setup.completedAt).toEqual(expect.any(String));
|
|
420
|
+
expect(completed.freshMachine).toBe(false);
|
|
421
|
+
expect(completed.repairMode).toBe(true);
|
|
422
|
+
expect(completed.repairReasons).toEqual(expect.arrayContaining([
|
|
423
|
+
expect.stringContaining("missing_steps:"),
|
|
424
|
+
"missing_focus_models",
|
|
425
|
+
"missing_payment_method",
|
|
426
|
+
"missing_connected_tools"
|
|
427
|
+
]));
|
|
428
|
+
|
|
429
|
+
const stateRes = await fetch(controlUrl("/init/state"));
|
|
430
|
+
expect(stateRes.status).toBe(200);
|
|
431
|
+
const state = await stateRes.json() as {
|
|
432
|
+
setup: { status: string; completedSteps: string[] };
|
|
433
|
+
freshMachine: boolean;
|
|
434
|
+
repairMode: boolean;
|
|
435
|
+
repairReasons: string[];
|
|
436
|
+
};
|
|
437
|
+
expect(state.setup.status).toBe("completed");
|
|
438
|
+
expect(state.setup.completedSteps).toEqual(["gateway_intro", "model_access"]);
|
|
439
|
+
expect(state.freshMachine).toBe(false);
|
|
440
|
+
expect(state.repairMode).toBe(true);
|
|
441
|
+
expect(state.repairReasons).toEqual(completed.repairReasons);
|
|
442
|
+
});
|
|
443
|
+
|
|
444
|
+
it("keeps completed setup out of repair mode when key wizard outputs exist", async () => {
|
|
445
|
+
daemon.stop();
|
|
446
|
+
await startDaemon({ providerHomeDir: TEMP_HOME });
|
|
447
|
+
const codexConfig = path.join(TEMP_HOME, ".codex", "config.toml");
|
|
448
|
+
fs.mkdirSync(path.dirname(codexConfig), { recursive: true });
|
|
449
|
+
fs.writeFileSync(codexConfig, "[tokenbuddy]\nproxy_url = \"http://127.0.0.1:17821\"\n", "utf8");
|
|
450
|
+
|
|
451
|
+
const store = (daemon as unknown as { tokenStore: { savePayment(input: unknown): void } }).tokenStore;
|
|
452
|
+
store.savePayment({
|
|
453
|
+
method: "mock",
|
|
454
|
+
enabled: true,
|
|
455
|
+
isDefault: true,
|
|
456
|
+
config: { source: "test" }
|
|
457
|
+
});
|
|
458
|
+
|
|
459
|
+
const focusRes = await fetch(controlUrl("/prewarm/focus-set"), {
|
|
460
|
+
method: "PUT",
|
|
461
|
+
headers: { "content-type": "application/json" },
|
|
462
|
+
body: JSON.stringify({ models: ["gpt-4o"] })
|
|
463
|
+
});
|
|
464
|
+
expect(focusRes.status).toBe(200);
|
|
465
|
+
|
|
466
|
+
const completeRes = await fetch(controlUrl("/init/complete"), {
|
|
467
|
+
method: "POST",
|
|
468
|
+
headers: { "content-type": "application/json" },
|
|
469
|
+
body: JSON.stringify({})
|
|
470
|
+
});
|
|
471
|
+
expect(completeRes.status).toBe(200);
|
|
472
|
+
|
|
473
|
+
const stateRes = await fetch(controlUrl("/init/state"));
|
|
474
|
+
expect(stateRes.status).toBe(200);
|
|
475
|
+
const state = await stateRes.json() as {
|
|
476
|
+
setup: { status: string; completedSteps: string[] };
|
|
477
|
+
freshMachine: boolean;
|
|
478
|
+
repairMode: boolean;
|
|
479
|
+
repairReasons: string[];
|
|
480
|
+
focusSet: string[];
|
|
481
|
+
payments: Array<{ method: string; enabled: boolean }>;
|
|
482
|
+
clientsSummary: { configuredCount: number };
|
|
483
|
+
};
|
|
484
|
+
expect(state.setup.status).toBe("completed");
|
|
485
|
+
expect(state.setup.completedSteps).toEqual([
|
|
486
|
+
"gateway_intro",
|
|
487
|
+
"model_access",
|
|
488
|
+
"supplier_routing",
|
|
489
|
+
"auto_purchase",
|
|
490
|
+
"connect_tools",
|
|
491
|
+
"verify_gateway",
|
|
492
|
+
"install_app"
|
|
493
|
+
]);
|
|
494
|
+
expect(state.freshMachine).toBe(false);
|
|
495
|
+
expect(state.repairMode).toBe(false);
|
|
496
|
+
expect(state.repairReasons).toEqual([]);
|
|
497
|
+
expect(state.focusSet).toEqual(["gpt-4o"]);
|
|
498
|
+
expect(state.payments).toEqual(expect.arrayContaining([
|
|
499
|
+
expect.objectContaining({ method: "mock", enabled: true })
|
|
500
|
+
]));
|
|
501
|
+
expect(state.clientsSummary.configuredCount).toBeGreaterThanOrEqual(1);
|
|
502
|
+
});
|
|
503
|
+
|
|
504
|
+
it("runs structured init doctor checks for the web wizard", async () => {
|
|
505
|
+
const res = await fetch(controlUrl("/init/doctor/run"), { method: "POST" });
|
|
506
|
+
expect(res.status).toBe(200);
|
|
507
|
+
const body = await res.json() as {
|
|
508
|
+
status: "passed" | "warning" | "failed";
|
|
509
|
+
generatedAt: string;
|
|
510
|
+
checks: Array<{ id: string; label: string; status: string; message: string; details: string[] }>;
|
|
511
|
+
};
|
|
512
|
+
|
|
513
|
+
expect(body.generatedAt).toEqual(expect.any(String));
|
|
514
|
+
expect(body.status).toBe("warning");
|
|
515
|
+
expect(body.checks).toEqual(expect.arrayContaining([
|
|
516
|
+
expect.objectContaining({ id: "local_service", label: "本地服务", status: "passed" }),
|
|
517
|
+
expect.objectContaining({ id: "model_catalog", label: "模型目录", status: "passed" }),
|
|
518
|
+
expect.objectContaining({ id: "routing_strategy", label: "路由策略", status: "passed" }),
|
|
519
|
+
expect.objectContaining({ id: "payment_method", label: "支付方式", status: "warning" }),
|
|
520
|
+
expect.objectContaining({ id: "connected_tools", label: "已连接工具", status: "warning" })
|
|
521
|
+
]));
|
|
522
|
+
const routing = body.checks.find((check) => check.id === "routing_strategy");
|
|
523
|
+
expect(routing?.details.join(" ")).toContain("fullAuto");
|
|
524
|
+
});
|
|
525
|
+
});
|
|
526
|
+
|
|
335
527
|
// ─── GET /providers/status ───────────────────────────────────
|
|
336
528
|
describe("GET /providers/status", () => {
|
|
337
529
|
it("returns read-only client tool status with a CLI install command", async () => {
|
package/tests/tokenbuddy.test.ts
CHANGED
|
@@ -7,6 +7,7 @@ import {
|
|
|
7
7
|
fetchClawtipBootstrap,
|
|
8
8
|
normalizeClawtipBootstrapResourceUrl,
|
|
9
9
|
restartLaunchAgent,
|
|
10
|
+
runWebInitLauncher,
|
|
10
11
|
} from "../src/cli.js";
|
|
11
12
|
import {
|
|
12
13
|
checkOpenClawRuntime,
|
|
@@ -218,6 +219,214 @@ describe("TokenBuddy CLI command surface", () => {
|
|
|
218
219
|
});
|
|
219
220
|
expect(launchctlCalls).toEqual([]);
|
|
220
221
|
});
|
|
222
|
+
|
|
223
|
+
test("web init launcher installs the macOS LaunchAgent and opens /init", async () => {
|
|
224
|
+
const writtenFiles: Array<{ filePath: string; content: string }> = [];
|
|
225
|
+
const installed: Array<{ plistPath: string; label: string }> = [];
|
|
226
|
+
const createdDirs: string[] = [];
|
|
227
|
+
const opened: Array<{ controlPort: number; pathname?: string }> = [];
|
|
228
|
+
const result = await runWebInitLauncher({
|
|
229
|
+
platform: "darwin",
|
|
230
|
+
controlPort: 3210,
|
|
231
|
+
proxyPort: 3211,
|
|
232
|
+
sellerRegistryUrl: "https://registry.example.test/sellers",
|
|
233
|
+
homeDir: "/Users/example",
|
|
234
|
+
nodePath: "/opt/node",
|
|
235
|
+
scriptPath: "/opt/tb-proxyd.js",
|
|
236
|
+
mkdirSync: (dirPath) => {
|
|
237
|
+
createdDirs.push(dirPath);
|
|
238
|
+
},
|
|
239
|
+
writeFileSync: (filePath, content) => {
|
|
240
|
+
writtenFiles.push({ filePath, content });
|
|
241
|
+
},
|
|
242
|
+
installLaunchAgent: (plistPath, label) => {
|
|
243
|
+
installed.push({ plistPath, label });
|
|
244
|
+
},
|
|
245
|
+
waitForDaemonStatus: async () => ({
|
|
246
|
+
running: true,
|
|
247
|
+
status: { pid: 42, controlPort: 3210, proxyPort: 3211 }
|
|
248
|
+
}),
|
|
249
|
+
fetchInitState: async () => ({ freshMachine: true, setup: { status: "not_started" } }),
|
|
250
|
+
launchControlUi: (controlPort, pathname) => {
|
|
251
|
+
opened.push({ controlPort, pathname });
|
|
252
|
+
return `http://127.0.0.1:${controlPort}${pathname}`;
|
|
253
|
+
}
|
|
254
|
+
});
|
|
255
|
+
|
|
256
|
+
expect(result).toMatchObject({
|
|
257
|
+
method: "launchd",
|
|
258
|
+
controlPort: 3210,
|
|
259
|
+
proxyPort: 3211,
|
|
260
|
+
serviceInstalled: true,
|
|
261
|
+
url: "http://127.0.0.1:3210/init",
|
|
262
|
+
probe: { running: true }
|
|
263
|
+
});
|
|
264
|
+
expect(result.plistPath).toBe("/Users/example/Library/LaunchAgents/com.tokenbuddy.proxyd.plist");
|
|
265
|
+
expect(installed).toEqual([{
|
|
266
|
+
plistPath: "/Users/example/Library/LaunchAgents/com.tokenbuddy.proxyd.plist",
|
|
267
|
+
label: "com.tokenbuddy.proxyd"
|
|
268
|
+
}]);
|
|
269
|
+
expect(createdDirs).toEqual(expect.arrayContaining([
|
|
270
|
+
"/Users/example/Library/LaunchAgents",
|
|
271
|
+
"/Users/example/.tokenbuddy-store"
|
|
272
|
+
]));
|
|
273
|
+
expect(writtenFiles[0].content).toContain("<string>3210</string>");
|
|
274
|
+
expect(writtenFiles[0].content).toContain("<string>3211</string>");
|
|
275
|
+
expect(writtenFiles[0].content).toContain("https://registry.example.test/sellers");
|
|
276
|
+
expect(opened).toEqual([{ controlPort: 3210, pathname: "/init" }]);
|
|
277
|
+
});
|
|
278
|
+
|
|
279
|
+
test("web init launcher recovers when launchctl fails after the service becomes ready", async () => {
|
|
280
|
+
const opened: Array<{ controlPort: number; pathname?: string }> = [];
|
|
281
|
+
const result = await runWebInitLauncher({
|
|
282
|
+
platform: "darwin",
|
|
283
|
+
controlPort: 3220,
|
|
284
|
+
proxyPort: 3221,
|
|
285
|
+
homeDir: "/Users/example",
|
|
286
|
+
nodePath: "/opt/node",
|
|
287
|
+
scriptPath: "/opt/tb-proxyd.js",
|
|
288
|
+
mkdirSync: () => undefined,
|
|
289
|
+
writeFileSync: () => undefined,
|
|
290
|
+
installLaunchAgent: () => {
|
|
291
|
+
throw new Error("launchctl bootstrap failed");
|
|
292
|
+
},
|
|
293
|
+
waitForDaemonStatus: async () => ({
|
|
294
|
+
running: true,
|
|
295
|
+
status: { pid: 43, controlPort: 3220, proxyPort: 3221 }
|
|
296
|
+
}),
|
|
297
|
+
fetchInitState: async () => ({ freshMachine: false, repairMode: false, setup: { status: "completed" } }),
|
|
298
|
+
launchControlUi: (controlPort, pathname) => {
|
|
299
|
+
opened.push({ controlPort, pathname });
|
|
300
|
+
return `http://127.0.0.1:${controlPort}${pathname}`;
|
|
301
|
+
}
|
|
302
|
+
});
|
|
303
|
+
|
|
304
|
+
expect(result).toMatchObject({
|
|
305
|
+
method: "launchd",
|
|
306
|
+
controlPort: 3220,
|
|
307
|
+
proxyPort: 3221,
|
|
308
|
+
serviceInstalled: true,
|
|
309
|
+
url: "http://127.0.0.1:3220/overview",
|
|
310
|
+
probe: { running: true }
|
|
311
|
+
});
|
|
312
|
+
expect(result.error).toBeUndefined();
|
|
313
|
+
expect(opened).toEqual([{ controlPort: 3220, pathname: "/overview" }]);
|
|
314
|
+
});
|
|
315
|
+
|
|
316
|
+
test("web init launcher opens overview after setup is completed", async () => {
|
|
317
|
+
const opened: Array<{ controlPort: number; pathname?: string }> = [];
|
|
318
|
+
const result = await runWebInitLauncher({
|
|
319
|
+
platform: "linux",
|
|
320
|
+
controlPort: 4340,
|
|
321
|
+
proxyPort: 4341,
|
|
322
|
+
repairDaemon: async () => ({
|
|
323
|
+
repair: { attempted: false, fixed: false },
|
|
324
|
+
probe: { running: true, status: { pid: 101 } }
|
|
325
|
+
}),
|
|
326
|
+
fetchInitState: async () => ({ freshMachine: false, repairMode: false, setup: { status: "completed" } }),
|
|
327
|
+
launchControlUi: (controlPort, pathname) => {
|
|
328
|
+
opened.push({ controlPort, pathname });
|
|
329
|
+
return `http://127.0.0.1:${controlPort}${pathname}`;
|
|
330
|
+
}
|
|
331
|
+
});
|
|
332
|
+
|
|
333
|
+
expect(result).toMatchObject({
|
|
334
|
+
method: "detached",
|
|
335
|
+
controlPort: 4340,
|
|
336
|
+
proxyPort: 4341,
|
|
337
|
+
serviceInstalled: true,
|
|
338
|
+
url: "http://127.0.0.1:4340/overview",
|
|
339
|
+
probe: { running: true }
|
|
340
|
+
});
|
|
341
|
+
expect(opened).toEqual([{ controlPort: 4340, pathname: "/overview" }]);
|
|
342
|
+
});
|
|
343
|
+
|
|
344
|
+
test("web init launcher reopens /init when completed setup needs repair", async () => {
|
|
345
|
+
const opened: Array<{ controlPort: number; pathname?: string }> = [];
|
|
346
|
+
const result = await runWebInitLauncher({
|
|
347
|
+
platform: "linux",
|
|
348
|
+
controlPort: 4350,
|
|
349
|
+
proxyPort: 4351,
|
|
350
|
+
repairDaemon: async () => ({
|
|
351
|
+
repair: { attempted: false, fixed: false },
|
|
352
|
+
probe: { running: true, status: { pid: 102 } }
|
|
353
|
+
}),
|
|
354
|
+
fetchInitState: async () => ({
|
|
355
|
+
freshMachine: false,
|
|
356
|
+
repairMode: true,
|
|
357
|
+
setup: { status: "completed" }
|
|
358
|
+
}),
|
|
359
|
+
launchControlUi: (controlPort, pathname) => {
|
|
360
|
+
opened.push({ controlPort, pathname });
|
|
361
|
+
return `http://127.0.0.1:${controlPort}${pathname}`;
|
|
362
|
+
}
|
|
363
|
+
});
|
|
364
|
+
|
|
365
|
+
expect(result).toMatchObject({
|
|
366
|
+
method: "detached",
|
|
367
|
+
controlPort: 4350,
|
|
368
|
+
proxyPort: 4351,
|
|
369
|
+
serviceInstalled: true,
|
|
370
|
+
url: "http://127.0.0.1:4350/init",
|
|
371
|
+
probe: { running: true }
|
|
372
|
+
});
|
|
373
|
+
expect(opened).toEqual([{ controlPort: 4350, pathname: "/init" }]);
|
|
374
|
+
});
|
|
375
|
+
|
|
376
|
+
test("web init launcher uses detached startup on non-macOS and reports failures without opening a browser", async () => {
|
|
377
|
+
const opened: Array<{ controlPort: number; pathname?: string }> = [];
|
|
378
|
+
const failed = await runWebInitLauncher({
|
|
379
|
+
platform: "linux",
|
|
380
|
+
controlPort: 4320,
|
|
381
|
+
proxyPort: 4321,
|
|
382
|
+
repairDaemon: async () => ({
|
|
383
|
+
repair: { attempted: true, fixed: false, error: "spawn failed" },
|
|
384
|
+
probe: { running: false, error: "offline" }
|
|
385
|
+
}),
|
|
386
|
+
launchControlUi: (controlPort, pathname) => {
|
|
387
|
+
opened.push({ controlPort, pathname });
|
|
388
|
+
return `http://127.0.0.1:${controlPort}${pathname}`;
|
|
389
|
+
}
|
|
390
|
+
});
|
|
391
|
+
|
|
392
|
+
expect(failed).toMatchObject({
|
|
393
|
+
method: "detached",
|
|
394
|
+
controlPort: 4320,
|
|
395
|
+
proxyPort: 4321,
|
|
396
|
+
serviceInstalled: false,
|
|
397
|
+
url: "http://127.0.0.1:4320/init",
|
|
398
|
+
probe: { running: false },
|
|
399
|
+
error: "spawn failed"
|
|
400
|
+
});
|
|
401
|
+
expect(opened).toEqual([]);
|
|
402
|
+
|
|
403
|
+
const openedAfterSuccess: Array<{ controlPort: number; pathname?: string }> = [];
|
|
404
|
+
const succeeded = await runWebInitLauncher({
|
|
405
|
+
platform: "linux",
|
|
406
|
+
controlPort: 4330,
|
|
407
|
+
proxyPort: 4331,
|
|
408
|
+
repairDaemon: async () => ({
|
|
409
|
+
repair: { attempted: true, fixed: true, pid: 99 },
|
|
410
|
+
probe: { running: true, status: { pid: 99 } }
|
|
411
|
+
}),
|
|
412
|
+
fetchInitState: async () => ({ freshMachine: true, setup: { status: "not_started" } }),
|
|
413
|
+
launchControlUi: (controlPort, pathname) => {
|
|
414
|
+
openedAfterSuccess.push({ controlPort, pathname });
|
|
415
|
+
return `http://127.0.0.1:${controlPort}${pathname}`;
|
|
416
|
+
}
|
|
417
|
+
});
|
|
418
|
+
|
|
419
|
+
expect(succeeded).toMatchObject({
|
|
420
|
+
method: "detached",
|
|
421
|
+
controlPort: 4330,
|
|
422
|
+
proxyPort: 4331,
|
|
423
|
+
serviceInstalled: true,
|
|
424
|
+
url: "http://127.0.0.1:4330/init",
|
|
425
|
+
probe: { running: true },
|
|
426
|
+
repair: { attempted: true, fixed: true, pid: 99 }
|
|
427
|
+
});
|
|
428
|
+
expect(openedAfterSuccess).toEqual([{ controlPort: 4330, pathname: "/init" }]);
|
|
429
|
+
});
|
|
221
430
|
});
|
|
222
431
|
|
|
223
432
|
describe("BuyerStore safe SQLite persistence", () => {
|