@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.
@@ -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)}}
@@ -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-YHs-Ca0f.js"></script>
16
- <link rel="stylesheet" crossorigin href="/assets/index-UMiTTeo8.css">
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>
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "id": "/",
3
- "name": "TokenBuddy Local Control",
3
+ "name": "TokenBuddy",
4
4
  "short_name": "TokenBuddy",
5
5
  "description": "Local TokenBuddy buyer control console.",
6
6
  "start_url": "/overview",
package/static/ui/sw.js CHANGED
@@ -1,4 +1,4 @@
1
- const CACHE_NAME = "tokenbuddy-ui-v1";
1
+ const CACHE_NAME = "tokenbuddy-ui-v2";
2
2
  const APP_SHELL = [
3
3
  "/",
4
4
  "/overview",
@@ -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-v1");
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 () => {
@@ -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", () => {