conductor-board 1.0.0 → 1.3.0
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/README.md +1 -1
- package/bin/cli.js +88 -1
- package/cli/setup.js +18 -4
- package/dist/assets/index-Dauby4vm.js +34 -0
- package/dist/assets/index-nvl4ljRP.css +1 -0
- package/dist/index.html +2 -2
- package/package.json +2 -1
- package/server/server.js +313 -61
- package/dist/assets/index--J1uxrSo.js +0 -34
- package/dist/assets/index-DMqA9hDY.css +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-border-style:solid;--tw-gradient-position:initial;--tw-gradient-from:#0000;--tw-gradient-via:#0000;--tw-gradient-to:#0000;--tw-gradient-stops:initial;--tw-gradient-via-stops:initial;--tw-gradient-from-position:0%;--tw-gradient-via-position:50%;--tw-gradient-to-position:100%;--tw-leading:initial;--tw-font-weight:initial;--tw-tracking:initial;--tw-ordinal:initial;--tw-slashed-zero:initial;--tw-numeric-figure:initial;--tw-numeric-spacing:initial;--tw-numeric-fraction: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-blur:initial;--tw-brightness:initial;--tw-contrast:initial;--tw-grayscale:initial;--tw-hue-rotate:initial;--tw-invert:initial;--tw-opacity:initial;--tw-saturate:initial;--tw-sepia:initial;--tw-drop-shadow:initial;--tw-drop-shadow-color:initial;--tw-drop-shadow-alpha:100%;--tw-drop-shadow-size:initial;--tw-backdrop-blur:initial;--tw-backdrop-brightness:initial;--tw-backdrop-contrast:initial;--tw-backdrop-grayscale:initial;--tw-backdrop-hue-rotate:initial;--tw-backdrop-invert:initial;--tw-backdrop-opacity:initial;--tw-backdrop-saturate:initial;--tw-backdrop-sepia:initial;--tw-duration:initial}}}@layer theme{:root,:host{--font-sans:"Inter", ui-sans-serif, system-ui, -apple-system, sans-serif;--font-mono:"JetBrains Mono", ui-monospace, "SF Mono", Menlo, monospace;--color-white:#fff;--spacing:.25rem;--container-md:28rem;--text-xs:.75rem;--text-xs--line-height:calc(1 / .75);--text-sm:.875rem;--text-sm--line-height:calc(1.25 / .875);--text-base:1rem;--text-base--line-height: 1.5 ;--text-xl:1.25rem;--text-xl--line-height:calc(1.75 / 1.25);--font-weight-medium:500;--font-weight-semibold:600;--tracking-wide:.025em;--leading-snug:1.375;--leading-relaxed:1.625;--radius-md:.375rem;--radius-lg:.5rem;--radius-xl:.75rem;--radius-2xl:1rem;--animate-pulse:pulse 2s cubic-bezier(.4, 0, .6, 1) infinite;--blur-xl:24px;--default-transition-duration:.15s;--default-transition-timing-function:cubic-bezier(.4, 0, .2, 1);--default-font-family:var(--font-sans);--default-mono-font-family:var(--font-mono);--color-ink:#0a0a0f;--color-ink-2:#0e0e15;--color-panel:#13131c;--color-panel-2:#181826;--color-line:#232333;--color-line-2:#2c2c40;--color-mist:#8a8aa3;--color-mist-2:#b4b4cc;--color-chalk:#e8e8f2;--color-iris:#a78bfa;--color-iris-deep:#7c5cff;--color-cyan:#22d3ee;--color-mint:#34d399;--color-amber:#fbbf24;--color-rose:#fb7185}}@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}:root{color-scheme:dark}body{background:var(--color-ink);color:var(--color-chalk);font-family:var(--font-sans);-webkit-font-smoothing:antialiased;margin:0}::selection{background:#a78bfa59}@supports (color:color-mix(in lab,red,red)){::selection{background:color-mix(in oklab,var(--color-iris) 35%,transparent)}}::selection{color:#fff}::-webkit-scrollbar{width:9px;height:9px}::-webkit-scrollbar-track{background:0 0}::-webkit-scrollbar-thumb{background:var(--color-line-2);border:2px solid var(--color-ink);border-radius:99px}}@layer components;@layer utilities{.pointer-events-none{pointer-events:none}.collapse{visibility:collapse}.visible{visibility:visible}.absolute{position:absolute}.fixed{position:fixed}.relative{position:relative}.sticky{position:sticky}.inset-x-0{inset-inline:calc(var(--spacing) * 0)}.inset-x-3{inset-inline:calc(var(--spacing) * 3)}.inset-y-0{inset-block:calc(var(--spacing) * 0)}.-top-1{top:calc(var(--spacing) * -1)}.top-0{top:calc(var(--spacing) * 0)}.top-1\.5{top:calc(var(--spacing) * 1.5)}.top-\[3px\]{top:3px}.top-\[5px\]{top:5px}.-right-1{right:calc(var(--spacing) * -1)}.right-0{right:calc(var(--spacing) * 0)}.right-4{right:calc(var(--spacing) * 4)}.bottom-0{bottom:calc(var(--spacing) * 0)}.bottom-1{bottom:calc(var(--spacing) * 1)}.bottom-3{bottom:calc(var(--spacing) * 3)}.bottom-4{bottom:calc(var(--spacing) * 4)}.bottom-20{bottom:calc(var(--spacing) * 20)}.-left-\[13px\]{left:-13px}.-left-\[15px\]{left:-15px}.left-1\/2{left:50%}.left-\[3px\]{left:3px}.z-10{z-index:10}.z-30{z-index:30}.z-40{z-index:40}.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-2\.5{margin-top:calc(var(--spacing) * 2.5)}.mt-3{margin-top:calc(var(--spacing) * 3)}.mt-4{margin-top:calc(var(--spacing) * 4)}.mt-5{margin-top:calc(var(--spacing) * 5)}.mt-6{margin-top:calc(var(--spacing) * 6)}.mt-\[3px\]{margin-top:3px}.mr-0\.5{margin-right:calc(var(--spacing) * .5)}.mr-1{margin-right:calc(var(--spacing) * 1)}.mb-1{margin-bottom:calc(var(--spacing) * 1)}.mb-2{margin-bottom:calc(var(--spacing) * 2)}.-ml-1{margin-left:calc(var(--spacing) * -1)}.ml-0\.5{margin-left:calc(var(--spacing) * .5)}.ml-1{margin-left:calc(var(--spacing) * 1)}.ml-2\.5{margin-left:calc(var(--spacing) * 2.5)}.ml-auto{margin-left:auto}.line-clamp-2{-webkit-line-clamp:2;-webkit-box-orient:vertical;display:-webkit-box;overflow:hidden}.block{display:block}.flex{display:flex}.grid{display:grid}.hidden{display:none}.inline-flex{display:inline-flex}.h-1{height:calc(var(--spacing) * 1)}.h-1\.5{height:calc(var(--spacing) * 1.5)}.h-2{height:calc(var(--spacing) * 2)}.h-2\.5{height:calc(var(--spacing) * 2.5)}.h-5{height:calc(var(--spacing) * 5)}.h-6{height:calc(var(--spacing) * 6)}.h-7{height:calc(var(--spacing) * 7)}.h-8{height:calc(var(--spacing) * 8)}.h-12{height:calc(var(--spacing) * 12)}.h-full{height:100%}.h-px{height:1px}.h-screen{height:100vh}.max-h-64{max-height:calc(var(--spacing) * 64)}.max-h-\[calc\(100vh-168px\)\]{max-height:calc(100vh - 168px)}.min-h-0{min-height:calc(var(--spacing) * 0)}.w-1{width:calc(var(--spacing) * 1)}.w-1\.5{width:calc(var(--spacing) * 1.5)}.w-2{width:calc(var(--spacing) * 2)}.w-2\.5{width:calc(var(--spacing) * 2.5)}.w-3{width:calc(var(--spacing) * 3)}.w-5{width:calc(var(--spacing) * 5)}.w-6{width:calc(var(--spacing) * 6)}.w-7{width:calc(var(--spacing) * 7)}.w-12{width:calc(var(--spacing) * 12)}.w-28{width:calc(var(--spacing) * 28)}.w-32{width:calc(var(--spacing) * 32)}.w-\[400px\]{width:400px}.w-full{width:100%}.w-px{width:1px}.max-w-\[130px\]{max-width:130px}.max-w-\[1400px\]{max-width:1400px}.max-w-full{max-width:100%}.max-w-md{max-width:var(--container-md)}.min-w-0{min-width:calc(var(--spacing) * 0)}.min-w-5{min-width:calc(var(--spacing) * 5)}.flex-1{flex:1}.shrink-0{flex-shrink:0}.-translate-x-1\/2{--tw-translate-x: -50% ;translate:var(--tw-translate-x) var(--tw-translate-y)}.animate-pulse{animation:var(--animate-pulse)}.cursor-col-resize{cursor:col-resize}.cursor-pointer{cursor:pointer}.cursor-row-resize{cursor:row-resize}.resize{resize:both}.flex-col{flex-direction:column}.flex-wrap{flex-wrap:wrap}.place-items-center{place-items:center}.items-center{align-items:center}.items-start{align-items:flex-start}.justify-between{justify-content:space-between}.justify-center{justify-content:center}.gap-1{gap:calc(var(--spacing) * 1)}.gap-1\.5{gap:calc(var(--spacing) * 1.5)}.gap-2{gap:calc(var(--spacing) * 2)}.gap-2\.5{gap:calc(var(--spacing) * 2.5)}.gap-3{gap:calc(var(--spacing) * 3)}.gap-5{gap:calc(var(--spacing) * 5)}:where(.space-y-0\.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)))}:where(.space-y-1>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing) * 1) * var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing) * 1) * calc(1 - var(--tw-space-y-reverse)))}:where(.space-y-1\.5>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing) * 1.5) * var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing) * 1.5) * calc(1 - var(--tw-space-y-reverse)))}:where(.space-y-2>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing) * 2) * var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing) * 2) * calc(1 - var(--tw-space-y-reverse)))}:where(.space-y-2\.5>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing) * 2.5) * var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing) * 2.5) * calc(1 - var(--tw-space-y-reverse)))}.gap-x-3{column-gap:calc(var(--spacing) * 3)}.gap-x-5{column-gap:calc(var(--spacing) * 5)}.gap-y-1{row-gap:calc(var(--spacing) * 1)}.gap-y-2{row-gap:calc(var(--spacing) * 2)}.truncate{text-overflow:ellipsis;white-space:nowrap;overflow:hidden}.overflow-hidden{overflow:hidden}.overflow-x-auto{overflow-x:auto}.overflow-y-auto{overflow-y:auto}.scroll-smooth{scroll-behavior:smooth}.rounded{border-radius:.25rem}.rounded-2xl{border-radius:var(--radius-2xl)}.rounded-full{border-radius:3.40282e38px}.rounded-lg{border-radius:var(--radius-lg)}.rounded-md{border-radius:var(--radius-md)}.rounded-xl{border-radius:var(--radius-xl)}.rounded-t-2xl{border-top-left-radius:var(--radius-2xl);border-top-right-radius:var(--radius-2xl)}.rounded-b-2xl{border-bottom-right-radius:var(--radius-2xl);border-bottom-left-radius:var(--radius-2xl)}.border{border-style:var(--tw-border-style);border-width:1px}.border-t{border-top-style:var(--tw-border-style);border-top-width:1px}.border-r{border-right-style:var(--tw-border-style);border-right-width:1px}.border-b{border-bottom-style:var(--tw-border-style);border-bottom-width:1px}.border-l{border-left-style:var(--tw-border-style);border-left-width:1px}.border-l-2{border-left-style:var(--tw-border-style);border-left-width:2px}.border-dashed{--tw-border-style:dashed;border-style:dashed}.border-amber\/30{border-color:#fbbf244d}@supports (color:color-mix(in lab,red,red)){.border-amber\/30{border-color:color-mix(in oklab,var(--color-amber) 30%,transparent)}}.border-amber\/40{border-color:#fbbf2466}@supports (color:color-mix(in lab,red,red)){.border-amber\/40{border-color:color-mix(in oklab,var(--color-amber) 40%,transparent)}}.border-amber\/50{border-color:#fbbf2480}@supports (color:color-mix(in lab,red,red)){.border-amber\/50{border-color:color-mix(in oklab,var(--color-amber) 50%,transparent)}}.border-cyan\/20{border-color:#22d3ee33}@supports (color:color-mix(in lab,red,red)){.border-cyan\/20{border-color:color-mix(in oklab,var(--color-cyan) 20%,transparent)}}.border-cyan\/30{border-color:#22d3ee4d}@supports (color:color-mix(in lab,red,red)){.border-cyan\/30{border-color:color-mix(in oklab,var(--color-cyan) 30%,transparent)}}.border-cyan\/40{border-color:#22d3ee66}@supports (color:color-mix(in lab,red,red)){.border-cyan\/40{border-color:color-mix(in oklab,var(--color-cyan) 40%,transparent)}}.border-iris\/25{border-color:#a78bfa40}@supports (color:color-mix(in lab,red,red)){.border-iris\/25{border-color:color-mix(in oklab,var(--color-iris) 25%,transparent)}}.border-iris\/40{border-color:#a78bfa66}@supports (color:color-mix(in lab,red,red)){.border-iris\/40{border-color:color-mix(in oklab,var(--color-iris) 40%,transparent)}}.border-iris\/50{border-color:#a78bfa80}@supports (color:color-mix(in lab,red,red)){.border-iris\/50{border-color:color-mix(in oklab,var(--color-iris) 50%,transparent)}}.border-line{border-color:var(--color-line)}.border-line-2{border-color:var(--color-line-2)}.border-line\/40{border-color:#23233366}@supports (color:color-mix(in lab,red,red)){.border-line\/40{border-color:color-mix(in oklab,var(--color-line) 40%,transparent)}}.border-line\/50{border-color:#23233380}@supports (color:color-mix(in lab,red,red)){.border-line\/50{border-color:color-mix(in oklab,var(--color-line) 50%,transparent)}}.border-line\/60{border-color:#23233399}@supports (color:color-mix(in lab,red,red)){.border-line\/60{border-color:color-mix(in oklab,var(--color-line) 60%,transparent)}}.border-line\/70{border-color:#232333b3}@supports (color:color-mix(in lab,red,red)){.border-line\/70{border-color:color-mix(in oklab,var(--color-line) 70%,transparent)}}.border-mint\/25{border-color:#34d39940}@supports (color:color-mix(in lab,red,red)){.border-mint\/25{border-color:color-mix(in oklab,var(--color-mint) 25%,transparent)}}.border-mint\/30{border-color:#34d3994d}@supports (color:color-mix(in lab,red,red)){.border-mint\/30{border-color:color-mix(in oklab,var(--color-mint) 30%,transparent)}}.border-mint\/40{border-color:#34d39966}@supports (color:color-mix(in lab,red,red)){.border-mint\/40{border-color:color-mix(in oklab,var(--color-mint) 40%,transparent)}}.border-rose\/15{border-color:#fb718526}@supports (color:color-mix(in lab,red,red)){.border-rose\/15{border-color:color-mix(in oklab,var(--color-rose) 15%,transparent)}}.border-rose\/30{border-color:#fb71854d}@supports (color:color-mix(in lab,red,red)){.border-rose\/30{border-color:color-mix(in oklab,var(--color-rose) 30%,transparent)}}.border-rose\/40{border-color:#fb718566}@supports (color:color-mix(in lab,red,red)){.border-rose\/40{border-color:color-mix(in oklab,var(--color-rose) 40%,transparent)}}.bg-\[\#08080d\]{background-color:#08080d}.bg-amber{background-color:var(--color-amber)}.bg-amber\/10{background-color:#fbbf241a}@supports (color:color-mix(in lab,red,red)){.bg-amber\/10{background-color:color-mix(in oklab,var(--color-amber) 10%,transparent)}}.bg-amber\/15{background-color:#fbbf2426}@supports (color:color-mix(in lab,red,red)){.bg-amber\/15{background-color:color-mix(in oklab,var(--color-amber) 15%,transparent)}}.bg-amber\/\[0\.05\]{background-color:#fbbf240d}@supports (color:color-mix(in lab,red,red)){.bg-amber\/\[0\.05\]{background-color:color-mix(in oklab,var(--color-amber) 5%,transparent)}}.bg-cyan{background-color:var(--color-cyan)}.bg-cyan\/10{background-color:#22d3ee1a}@supports (color:color-mix(in lab,red,red)){.bg-cyan\/10{background-color:color-mix(in oklab,var(--color-cyan) 10%,transparent)}}.bg-cyan\/\[0\.06\]{background-color:#22d3ee0f}@supports (color:color-mix(in lab,red,red)){.bg-cyan\/\[0\.06\]{background-color:color-mix(in oklab,var(--color-cyan) 6%,transparent)}}.bg-ink-2\/40{background-color:#0e0e1566}@supports (color:color-mix(in lab,red,red)){.bg-ink-2\/40{background-color:color-mix(in oklab,var(--color-ink-2) 40%,transparent)}}.bg-ink-2\/60{background-color:#0e0e1599}@supports (color:color-mix(in lab,red,red)){.bg-ink-2\/60{background-color:color-mix(in oklab,var(--color-ink-2) 60%,transparent)}}.bg-ink-2\/80{background-color:#0e0e15cc}@supports (color:color-mix(in lab,red,red)){.bg-ink-2\/80{background-color:color-mix(in oklab,var(--color-ink-2) 80%,transparent)}}.bg-ink-2\/90{background-color:#0e0e15e6}@supports (color:color-mix(in lab,red,red)){.bg-ink-2\/90{background-color:color-mix(in oklab,var(--color-ink-2) 90%,transparent)}}.bg-ink-2\/95{background-color:#0e0e15f2}@supports (color:color-mix(in lab,red,red)){.bg-ink-2\/95{background-color:color-mix(in oklab,var(--color-ink-2) 95%,transparent)}}.bg-ink\/40{background-color:#0a0a0f66}@supports (color:color-mix(in lab,red,red)){.bg-ink\/40{background-color:color-mix(in oklab,var(--color-ink) 40%,transparent)}}.bg-ink\/50{background-color:#0a0a0f80}@supports (color:color-mix(in lab,red,red)){.bg-ink\/50{background-color:color-mix(in oklab,var(--color-ink) 50%,transparent)}}.bg-ink\/60{background-color:#0a0a0f99}@supports (color:color-mix(in lab,red,red)){.bg-ink\/60{background-color:color-mix(in oklab,var(--color-ink) 60%,transparent)}}.bg-ink\/80{background-color:#0a0a0fcc}@supports (color:color-mix(in lab,red,red)){.bg-ink\/80{background-color:color-mix(in oklab,var(--color-ink) 80%,transparent)}}.bg-ink\/90{background-color:#0a0a0fe6}@supports (color:color-mix(in lab,red,red)){.bg-ink\/90{background-color:color-mix(in oklab,var(--color-ink) 90%,transparent)}}.bg-iris{background-color:var(--color-iris)}.bg-iris\/10{background-color:#a78bfa1a}@supports (color:color-mix(in lab,red,red)){.bg-iris\/10{background-color:color-mix(in oklab,var(--color-iris) 10%,transparent)}}.bg-iris\/15{background-color:#a78bfa26}@supports (color:color-mix(in lab,red,red)){.bg-iris\/15{background-color:color-mix(in oklab,var(--color-iris) 15%,transparent)}}.bg-iris\/\[0\.08\]{background-color:#a78bfa14}@supports (color:color-mix(in lab,red,red)){.bg-iris\/\[0\.08\]{background-color:color-mix(in oklab,var(--color-iris) 8%,transparent)}}.bg-line{background-color:var(--color-line)}.bg-line-2{background-color:var(--color-line-2)}.bg-mint{background-color:var(--color-mint)}.bg-mint\/10{background-color:#34d3991a}@supports (color:color-mix(in lab,red,red)){.bg-mint\/10{background-color:color-mix(in oklab,var(--color-mint) 10%,transparent)}}.bg-mint\/15{background-color:#34d39926}@supports (color:color-mix(in lab,red,red)){.bg-mint\/15{background-color:color-mix(in oklab,var(--color-mint) 15%,transparent)}}.bg-mint\/\[0\.06\]{background-color:#34d3990f}@supports (color:color-mix(in lab,red,red)){.bg-mint\/\[0\.06\]{background-color:color-mix(in oklab,var(--color-mint) 6%,transparent)}}.bg-mint\/\[0\.08\]{background-color:#34d39914}@supports (color:color-mix(in lab,red,red)){.bg-mint\/\[0\.08\]{background-color:color-mix(in oklab,var(--color-mint) 8%,transparent)}}.bg-panel{background-color:var(--color-panel)}.bg-panel-2{background-color:var(--color-panel-2)}.bg-panel\/40{background-color:#13131c66}@supports (color:color-mix(in lab,red,red)){.bg-panel\/40{background-color:color-mix(in oklab,var(--color-panel) 40%,transparent)}}.bg-panel\/50{background-color:#13131c80}@supports (color:color-mix(in lab,red,red)){.bg-panel\/50{background-color:color-mix(in oklab,var(--color-panel) 50%,transparent)}}.bg-panel\/60{background-color:#13131c99}@supports (color:color-mix(in lab,red,red)){.bg-panel\/60{background-color:color-mix(in oklab,var(--color-panel) 60%,transparent)}}.bg-rose{background-color:var(--color-rose)}.bg-rose\/10{background-color:#fb71851a}@supports (color:color-mix(in lab,red,red)){.bg-rose\/10{background-color:color-mix(in oklab,var(--color-rose) 10%,transparent)}}.bg-rose\/15{background-color:#fb718526}@supports (color:color-mix(in lab,red,red)){.bg-rose\/15{background-color:color-mix(in oklab,var(--color-rose) 15%,transparent)}}.bg-transparent{background-color:#0000}.bg-gradient-to-b{--tw-gradient-position:to bottom in oklab;background-image:linear-gradient(var(--tw-gradient-stops))}.bg-gradient-to-r{--tw-gradient-position:to right in oklab;background-image:linear-gradient(var(--tw-gradient-stops))}.bg-gradient-to-t{--tw-gradient-position:to top in oklab;background-image:linear-gradient(var(--tw-gradient-stops))}.from-ink-2{--tw-gradient-from:var(--color-ink-2);--tw-gradient-stops:var(--tw-gradient-via-stops,var(--tw-gradient-position), var(--tw-gradient-from) var(--tw-gradient-from-position), var(--tw-gradient-to) var(--tw-gradient-to-position))}.from-iris{--tw-gradient-from:var(--color-iris);--tw-gradient-stops:var(--tw-gradient-via-stops,var(--tw-gradient-position), var(--tw-gradient-from) var(--tw-gradient-from-position), var(--tw-gradient-to) var(--tw-gradient-to-position))}.from-rose{--tw-gradient-from:var(--color-rose);--tw-gradient-stops:var(--tw-gradient-via-stops,var(--tw-gradient-position), var(--tw-gradient-from) var(--tw-gradient-from-position), var(--tw-gradient-to) var(--tw-gradient-to-position))}.to-amber{--tw-gradient-to:var(--color-amber);--tw-gradient-stops:var(--tw-gradient-via-stops,var(--tw-gradient-position), var(--tw-gradient-from) var(--tw-gradient-from-position), var(--tw-gradient-to) var(--tw-gradient-to-position))}.to-cyan{--tw-gradient-to:var(--color-cyan);--tw-gradient-stops:var(--tw-gradient-via-stops,var(--tw-gradient-position), var(--tw-gradient-from) var(--tw-gradient-from-position), var(--tw-gradient-to) var(--tw-gradient-to-position))}.to-iris-deep{--tw-gradient-to:var(--color-iris-deep);--tw-gradient-stops:var(--tw-gradient-via-stops,var(--tw-gradient-position), var(--tw-gradient-from) var(--tw-gradient-from-position), var(--tw-gradient-to) var(--tw-gradient-to-position))}.to-transparent{--tw-gradient-to:transparent;--tw-gradient-stops:var(--tw-gradient-via-stops,var(--tw-gradient-position), var(--tw-gradient-from) var(--tw-gradient-from-position), var(--tw-gradient-to) var(--tw-gradient-to-position))}.p-2{padding:calc(var(--spacing) * 2)}.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)}.px-0{padding-inline:calc(var(--spacing) * 0)}.px-1{padding-inline:calc(var(--spacing) * 1)}.px-1\.5{padding-inline:calc(var(--spacing) * 1.5)}.px-2{padding-inline:calc(var(--spacing) * 2)}.px-2\.5{padding-inline:calc(var(--spacing) * 2.5)}.px-3{padding-inline:calc(var(--spacing) * 3)}.px-3\.5{padding-inline:calc(var(--spacing) * 3.5)}.px-4{padding-inline:calc(var(--spacing) * 4)}.px-5{padding-inline:calc(var(--spacing) * 5)}.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-3\.5{padding-block:calc(var(--spacing) * 3.5)}.py-6{padding-block:calc(var(--spacing) * 6)}.py-px{padding-block:1px}.pt-0\.5{padding-top:calc(var(--spacing) * .5)}.pt-1{padding-top:calc(var(--spacing) * 1)}.pt-2{padding-top:calc(var(--spacing) * 2)}.pt-3\.5{padding-top:calc(var(--spacing) * 3.5)}.pt-4{padding-top:calc(var(--spacing) * 4)}.pb-1{padding-bottom:calc(var(--spacing) * 1)}.pb-1\.5{padding-bottom:calc(var(--spacing) * 1.5)}.pb-2{padding-bottom:calc(var(--spacing) * 2)}.pl-1\.5{padding-left:calc(var(--spacing) * 1.5)}.pl-2{padding-left:calc(var(--spacing) * 2)}.pl-3\.5{padding-left:calc(var(--spacing) * 3.5)}.pl-4{padding-left:calc(var(--spacing) * 4)}.pl-7{padding-left:calc(var(--spacing) * 7)}.text-center{text-align:center}.text-left{text-align:left}.font-mono{font-family:var(--font-mono)}.text-base{font-size:var(--text-base);line-height:var(--tw-leading,var(--text-base--line-height))}.text-sm{font-size:var(--text-sm);line-height:var(--tw-leading,var(--text-sm--line-height))}.text-xl{font-size:var(--text-xl);line-height:var(--tw-leading,var(--text-xl--line-height))}.text-xs{font-size:var(--text-xs);line-height:var(--tw-leading,var(--text-xs--line-height))}.text-\[0\.85em\]{font-size:.85em}.text-\[9px\]{font-size:9px}.text-\[10\.5px\]{font-size:10.5px}.text-\[10px\]{font-size:10px}.text-\[11\.5px\]{font-size:11.5px}.text-\[11px\]{font-size:11px}.text-\[12\.5px\]{font-size:12.5px}.text-\[12px\]{font-size:12px}.text-\[13px\]{font-size:13px}.leading-none{--tw-leading:1;line-height:1}.leading-relaxed{--tw-leading:var(--leading-relaxed);line-height:var(--leading-relaxed)}.leading-snug{--tw-leading:var(--leading-snug);line-height:var(--leading-snug)}.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\.14em\]{--tw-tracking:.14em;letter-spacing:.14em}.tracking-\[0\.16em\]{--tw-tracking:.16em;letter-spacing:.16em}.tracking-wide{--tw-tracking:var(--tracking-wide);letter-spacing:var(--tracking-wide)}.text-amber{color:var(--color-amber)}.text-amber\/90{color:#fbbf24e6}@supports (color:color-mix(in lab,red,red)){.text-amber\/90{color:color-mix(in oklab,var(--color-amber) 90%,transparent)}}.text-chalk{color:var(--color-chalk)}.text-cyan{color:var(--color-cyan)}.text-cyan\/80{color:#22d3eecc}@supports (color:color-mix(in lab,red,red)){.text-cyan\/80{color:color-mix(in oklab,var(--color-cyan) 80%,transparent)}}.text-iris{color:var(--color-iris)}.text-line-2{color:var(--color-line-2)}.text-mint{color:var(--color-mint)}.text-mint\/80{color:#34d399cc}@supports (color:color-mix(in lab,red,red)){.text-mint\/80{color:color-mix(in oklab,var(--color-mint) 80%,transparent)}}.text-mist{color:var(--color-mist)}.text-mist-2{color:var(--color-mist-2)}.text-rose{color:var(--color-rose)}.text-rose\/90{color:#fb7185e6}@supports (color:color-mix(in lab,red,red)){.text-rose\/90{color:color-mix(in oklab,var(--color-rose) 90%,transparent)}}.text-white{color:var(--color-white)}.capitalize{text-transform:capitalize}.uppercase{text-transform:uppercase}.italic{font-style:italic}.not-italic{font-style:normal}.tabular-nums{--tw-numeric-spacing:tabular-nums;font-variant-numeric:var(--tw-ordinal,) var(--tw-slashed-zero,) var(--tw-numeric-figure,) var(--tw-numeric-spacing,) var(--tw-numeric-fraction,)}.underline-offset-2{text-underline-offset:2px}.opacity-60{opacity:.6}.opacity-70{opacity:.7}.opacity-80{opacity:.8}.opacity-85{opacity:.85}.opacity-90{opacity:.9}.shadow-2xl{--tw-shadow:0 25px 50px -12px var(--tw-shadow-color,#00000040);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_6px\]{--tw-shadow:0 0 6px var(--tw-shadow-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)}.shadow-lg{--tw-shadow:0 10px 15px -3px var(--tw-shadow-color,#0000001a), 0 4px 6px -4px var(--tw-shadow-color,#0000001a);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.shadow-xl{--tw-shadow:0 20px 25px -5px var(--tw-shadow-color,#0000001a), 0 8px 10px -6px var(--tw-shadow-color,#0000001a);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.shadow-mint\/60{--tw-shadow-color:#34d39999}@supports (color:color-mix(in lab,red,red)){.shadow-mint\/60{--tw-shadow-color:color-mix(in oklab, color-mix(in oklab, var(--color-mint) 60%, transparent) var(--tw-shadow-alpha), transparent)}}.filter{filter:var(--tw-blur,) var(--tw-brightness,) var(--tw-contrast,) var(--tw-grayscale,) var(--tw-hue-rotate,) var(--tw-invert,) var(--tw-saturate,) var(--tw-sepia,) var(--tw-drop-shadow,)}.backdrop-blur{--tw-backdrop-blur:blur(8px);-webkit-backdrop-filter:var(--tw-backdrop-blur,) var(--tw-backdrop-brightness,) var(--tw-backdrop-contrast,) var(--tw-backdrop-grayscale,) var(--tw-backdrop-hue-rotate,) var(--tw-backdrop-invert,) var(--tw-backdrop-opacity,) var(--tw-backdrop-saturate,) var(--tw-backdrop-sepia,);backdrop-filter:var(--tw-backdrop-blur,) var(--tw-backdrop-brightness,) var(--tw-backdrop-contrast,) var(--tw-backdrop-grayscale,) var(--tw-backdrop-hue-rotate,) var(--tw-backdrop-invert,) var(--tw-backdrop-opacity,) var(--tw-backdrop-saturate,) var(--tw-backdrop-sepia,)}.backdrop-blur-xl{--tw-backdrop-blur:blur(var(--blur-xl));-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))}.transition-opacity{transition-property:opacity;transition-timing-function:var(--tw-ease,var(--default-transition-timing-function));transition-duration:var(--tw-duration,var(--default-transition-duration))}.duration-500{--tw-duration:.5s;transition-duration:.5s}.outline-none{--tw-outline-style:none;outline-style:none}.select-none{-webkit-user-select:none;user-select:none}.placeholder\:text-mist::placeholder{color:var(--color-mist)}.last\:border-0:last-child{border-style:var(--tw-border-style);border-width:0}@media(hover:hover){.hover\:border-line-2:hover{border-color:var(--color-line-2)}.hover\:bg-amber\/15:hover{background-color:#fbbf2426}@supports (color:color-mix(in lab,red,red)){.hover\:bg-amber\/15:hover{background-color:color-mix(in oklab,var(--color-amber) 15%,transparent)}}.hover\:bg-cyan\/10:hover{background-color:#22d3ee1a}@supports (color:color-mix(in lab,red,red)){.hover\:bg-cyan\/10:hover{background-color:color-mix(in oklab,var(--color-cyan) 10%,transparent)}}.hover\:bg-mint\/25:hover{background-color:#34d39940}@supports (color:color-mix(in lab,red,red)){.hover\:bg-mint\/25:hover{background-color:color-mix(in oklab,var(--color-mint) 25%,transparent)}}.hover\:bg-panel:hover{background-color:var(--color-panel)}.hover\:bg-panel\/60:hover{background-color:#13131c99}@supports (color:color-mix(in lab,red,red)){.hover\:bg-panel\/60:hover{background-color:color-mix(in oklab,var(--color-panel) 60%,transparent)}}.hover\:text-chalk:hover{color:var(--color-chalk)}.hover\:underline:hover{text-decoration-line:underline}}.disabled\:opacity-40:disabled{opacity:.4}@media(min-width:40rem){.sm\:grid-cols-2{grid-template-columns:repeat(2,minmax(0,1fr))}}@media(min-width:64rem){.lg\:flex{display:flex}.lg\:grid-cols-4{grid-template-columns:repeat(4,minmax(0,1fr))}.lg\:grid-cols-\[repeat\(4\,minmax\(0\,1fr\)\)_0\.85fr\]{grid-template-columns:repeat(4,minmax(0,1fr)) .85fr}}}.aurora{z-index:-1;pointer-events:none;position:fixed;top:0;right:0;bottom:0;left:0;overflow:hidden}.aurora:before{content:"";filter:blur(130px);opacity:.16;background:var(--color-iris-deep);border-radius:50%;width:55vw;height:55vw;position:absolute;top:-22vw;left:30vw}@keyframes pulse-ring{0%{box-shadow:0 0 color-mix(in oklab,var(--color-cyan) 50%,transparent)}70%{box-shadow:0 0 0 8px #0000}to{box-shadow:0 0 #0000}}.pulse-ring{animation:1.8s infinite pulse-ring}@keyframes check-pop{0%{opacity:0;transform:scale(0)rotate(-12deg)}60%{transform:scale(1.18)rotate(0)}to{opacity:1;transform:scale(1)rotate(0)}}.check-pop{animation:.34s cubic-bezier(.34,1.56,.64,1) both check-pop}@keyframes spin{to{transform:rotate(360deg)}}.spin{animation:.9s linear infinite spin}@keyframes heart-rest{0%{transform:scale(1)}7%{transform:scale(1.22)}14%{transform:scale(1)}21%{transform:scale(1.14)}28%{transform:scale(1)}to{transform:scale(1)}}.heart-rest{transform-origin:50%;animation:1.15s ease-in-out infinite heart-rest}@keyframes heart-strong{0%{transform:scale(1)}28%{transform:scale(1.34)}to{transform:scale(1)}}.heart-strong{transform-origin:50%;animation:.46s cubic-bezier(.34,1.56,.64,1) heart-strong}.heart-weak{transform-origin:50%;opacity:.5;filter:drop-shadow(0 0 4px #fbbf24bf);animation:2.1s ease-in-out infinite heart-rest}@supports (color:color-mix(in lab,red,red)){.heart-weak{filter:drop-shadow(0 0 4px color-mix(in oklab,var(--color-amber) 75%,transparent))}}@keyframes tw-blink{0%,49%{opacity:1}50%,to{opacity:0}}.tw-cursor{animation:.53s step-end infinite tw-blink}.monitor-grain{background-image:repeating-linear-gradient(0deg,#e8e8f205 0,#e8e8f205 1px,#0000 1px,#0000 3px)}@supports (color:color-mix(in lab,red,red)){.monitor-grain{background-image:repeating-linear-gradient(0deg,color-mix(in oklab,var(--color-chalk) 2%,transparent) 0px,color-mix(in oklab,var(--color-chalk) 2%,transparent) 1px,transparent 1px,transparent 3px)}}.board-scroll{scrollbar-width:thin;scrollbar-color:var(--color-line-2) transparent}.board-scroll::-webkit-scrollbar{width:6px}.board-scroll::-webkit-scrollbar-track{background:0 0}.board-scroll::-webkit-scrollbar-thumb{background:var(--color-line-2);border:none;border-radius:99px}@property --tw-translate-x{syntax:"*";inherits:false;initial-value:0}@property --tw-translate-y{syntax:"*";inherits:false;initial-value:0}@property --tw-translate-z{syntax:"*";inherits:false;initial-value:0}@property --tw-space-y-reverse{syntax:"*";inherits:false;initial-value:0}@property --tw-border-style{syntax:"*";inherits:false;initial-value:solid}@property --tw-gradient-position{syntax:"*";inherits:false}@property --tw-gradient-from{syntax:"<color>";inherits:false;initial-value:#0000}@property --tw-gradient-via{syntax:"<color>";inherits:false;initial-value:#0000}@property --tw-gradient-to{syntax:"<color>";inherits:false;initial-value:#0000}@property --tw-gradient-stops{syntax:"*";inherits:false}@property --tw-gradient-via-stops{syntax:"*";inherits:false}@property --tw-gradient-from-position{syntax:"<length-percentage>";inherits:false;initial-value:0%}@property --tw-gradient-via-position{syntax:"<length-percentage>";inherits:false;initial-value:50%}@property --tw-gradient-to-position{syntax:"<length-percentage>";inherits:false;initial-value:100%}@property --tw-leading{syntax:"*";inherits:false}@property --tw-font-weight{syntax:"*";inherits:false}@property --tw-tracking{syntax:"*";inherits:false}@property --tw-ordinal{syntax:"*";inherits:false}@property --tw-slashed-zero{syntax:"*";inherits:false}@property --tw-numeric-figure{syntax:"*";inherits:false}@property --tw-numeric-spacing{syntax:"*";inherits:false}@property --tw-numeric-fraction{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-blur{syntax:"*";inherits:false}@property --tw-brightness{syntax:"*";inherits:false}@property --tw-contrast{syntax:"*";inherits:false}@property --tw-grayscale{syntax:"*";inherits:false}@property --tw-hue-rotate{syntax:"*";inherits:false}@property --tw-invert{syntax:"*";inherits:false}@property --tw-opacity{syntax:"*";inherits:false}@property --tw-saturate{syntax:"*";inherits:false}@property --tw-sepia{syntax:"*";inherits:false}@property --tw-drop-shadow{syntax:"*";inherits:false}@property --tw-drop-shadow-color{syntax:"*";inherits:false}@property --tw-drop-shadow-alpha{syntax:"<percentage>";inherits:false;initial-value:100%}@property --tw-drop-shadow-size{syntax:"*";inherits:false}@property --tw-backdrop-blur{syntax:"*";inherits:false}@property --tw-backdrop-brightness{syntax:"*";inherits:false}@property --tw-backdrop-contrast{syntax:"*";inherits:false}@property --tw-backdrop-grayscale{syntax:"*";inherits:false}@property --tw-backdrop-hue-rotate{syntax:"*";inherits:false}@property --tw-backdrop-invert{syntax:"*";inherits:false}@property --tw-backdrop-opacity{syntax:"*";inherits:false}@property --tw-backdrop-saturate{syntax:"*";inherits:false}@property --tw-backdrop-sepia{syntax:"*";inherits:false}@property --tw-duration{syntax:"*";inherits:false}@keyframes pulse{50%{opacity:.5}}
|
package/dist/index.html
CHANGED
|
@@ -6,10 +6,10 @@
|
|
|
6
6
|
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
|
7
7
|
<meta name="theme-color" content="#0a0a0f" />
|
|
8
8
|
<title>Agent Conductor — Board</title>
|
|
9
|
-
<script type="module" crossorigin src="./assets/index
|
|
9
|
+
<script type="module" crossorigin src="./assets/index-Dauby4vm.js"></script>
|
|
10
10
|
<link rel="modulepreload" crossorigin href="./assets/motion-Dmvx5jlk.js">
|
|
11
11
|
<link rel="modulepreload" crossorigin href="./assets/yaml-NA7d4LV6.js">
|
|
12
|
-
<link rel="stylesheet" crossorigin href="./assets/index-
|
|
12
|
+
<link rel="stylesheet" crossorigin href="./assets/index-nvl4ljRP.css">
|
|
13
13
|
</head>
|
|
14
14
|
<body>
|
|
15
15
|
<div id="root"></div>
|
package/package.json
CHANGED
|
@@ -1,11 +1,12 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "conductor-board",
|
|
3
|
-
"version": "1.
|
|
3
|
+
"version": "1.3.0",
|
|
4
4
|
"description": "Gated workflows for AI agents — live Kanban board included",
|
|
5
5
|
"license": "MIT",
|
|
6
6
|
"author": "mettafive",
|
|
7
7
|
"type": "module",
|
|
8
8
|
"bin": {
|
|
9
|
+
"3042": "bin/cli.js",
|
|
9
10
|
"conductor-board": "bin/cli.js"
|
|
10
11
|
},
|
|
11
12
|
"files": [
|
package/server/server.js
CHANGED
|
@@ -13,6 +13,60 @@ import http from "node:http";
|
|
|
13
13
|
import fs from "node:fs";
|
|
14
14
|
import path from "node:path";
|
|
15
15
|
import { fileURLToPath } from "node:url";
|
|
16
|
+
import yaml from "js-yaml";
|
|
17
|
+
import { validateConductor } from "../cli/validate.js";
|
|
18
|
+
|
|
19
|
+
const readBody = (req) =>
|
|
20
|
+
new Promise((resolve) => {
|
|
21
|
+
let data = "";
|
|
22
|
+
req.on("data", (c) => (data += c));
|
|
23
|
+
req.on("end", () => resolve(data));
|
|
24
|
+
});
|
|
25
|
+
|
|
26
|
+
/** Apply optimization suggestions to a parsed conductor doc (in place). */
|
|
27
|
+
function applyMutations(doc, suggestions) {
|
|
28
|
+
doc.steps = Array.isArray(doc.steps) ? doc.steps : [];
|
|
29
|
+
const byId = new Map(doc.steps.map((s) => [s.id, s]));
|
|
30
|
+
for (const sug of suggestions) {
|
|
31
|
+
const step = sug.step ? byId.get(sug.step) : null;
|
|
32
|
+
switch (sug.type) {
|
|
33
|
+
case "instruction":
|
|
34
|
+
if (step) {
|
|
35
|
+
if (sug.current && typeof step.instruction === "string" && step.instruction.includes(sug.current)) {
|
|
36
|
+
step.instruction = step.instruction.replace(sug.current, sug.proposed ?? "");
|
|
37
|
+
} else if (sug.proposed) {
|
|
38
|
+
step.instruction = sug.proposed;
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
break;
|
|
42
|
+
case "gate":
|
|
43
|
+
if (step && Array.isArray(step.gate)) {
|
|
44
|
+
const i = step.gate.findIndex((g) => g === sug.current);
|
|
45
|
+
if (i >= 0 && sug.proposed != null) step.gate[i] = sug.proposed;
|
|
46
|
+
else if (sug.proposed != null) step.gate.push(sug.proposed);
|
|
47
|
+
}
|
|
48
|
+
break;
|
|
49
|
+
case "new_gate":
|
|
50
|
+
if (step && sug.proposed != null) {
|
|
51
|
+
step.gate = Array.isArray(step.gate) ? step.gate : [];
|
|
52
|
+
step.gate.push(sug.proposed);
|
|
53
|
+
}
|
|
54
|
+
break;
|
|
55
|
+
case "new_step":
|
|
56
|
+
doc.steps.push({
|
|
57
|
+
id: sug.step || `step-${doc.steps.length + 1}`,
|
|
58
|
+
instruction: sug.proposed || "TODO",
|
|
59
|
+
gate: ["TODO: add a gate criterion"],
|
|
60
|
+
});
|
|
61
|
+
break;
|
|
62
|
+
case "remove_step":
|
|
63
|
+
if (sug.step) doc.steps = doc.steps.filter((s) => s.id !== sug.step);
|
|
64
|
+
break;
|
|
65
|
+
// `reorder` is not auto-applied in v1
|
|
66
|
+
}
|
|
67
|
+
}
|
|
68
|
+
return doc;
|
|
69
|
+
}
|
|
16
70
|
|
|
17
71
|
const __dirname = path.dirname(fileURLToPath(import.meta.url));
|
|
18
72
|
const DIST = path.resolve(__dirname, "..", "dist");
|
|
@@ -207,89 +261,279 @@ function serveStatic(req, res) {
|
|
|
207
261
|
fs.createReadStream(filePath).pipe(res);
|
|
208
262
|
}
|
|
209
263
|
|
|
264
|
+
/** Best-effort workflow name without parsing YAML (keeps the server dep-free). */
|
|
265
|
+
function workflowName(statusPath, conductorPath, dir) {
|
|
266
|
+
try {
|
|
267
|
+
if (fs.existsSync(statusPath)) {
|
|
268
|
+
const s = JSON.parse(fs.readFileSync(statusPath, "utf8"));
|
|
269
|
+
if (s && typeof s.workflow === "string") return s.workflow;
|
|
270
|
+
}
|
|
271
|
+
} catch {
|
|
272
|
+
/* ignore */
|
|
273
|
+
}
|
|
274
|
+
try {
|
|
275
|
+
if (conductorPath && fs.existsSync(conductorPath)) {
|
|
276
|
+
const m = fs.readFileSync(conductorPath, "utf8").match(/^name:\s*(.+)$/m);
|
|
277
|
+
if (m) return m[1].trim();
|
|
278
|
+
}
|
|
279
|
+
} catch {
|
|
280
|
+
/* ignore */
|
|
281
|
+
}
|
|
282
|
+
return path.basename(dir);
|
|
283
|
+
}
|
|
284
|
+
|
|
285
|
+
/**
|
|
286
|
+
* Discover workflows under the .conductor root. Supports both the flat layout
|
|
287
|
+
* (.conductor/status.json — a single workflow, for v1 backwards compatibility)
|
|
288
|
+
* and the subdirectory layout (.conductor/<name>/status.json — many workflows).
|
|
289
|
+
*/
|
|
290
|
+
function discoverWorkflows(conductorDir, explicitStatus, explicitConductor) {
|
|
291
|
+
const found = [];
|
|
292
|
+
const seen = new Set();
|
|
293
|
+
const add = (name, dir, statusPath, conductorPath) => {
|
|
294
|
+
if (seen.has(name)) return;
|
|
295
|
+
seen.add(name);
|
|
296
|
+
found.push({
|
|
297
|
+
name,
|
|
298
|
+
dir,
|
|
299
|
+
statusPath,
|
|
300
|
+
conductorPath,
|
|
301
|
+
historyDir: path.join(dir, "history"),
|
|
302
|
+
});
|
|
303
|
+
};
|
|
304
|
+
|
|
305
|
+
// flat / explicit --path
|
|
306
|
+
const flatStatus = explicitStatus || path.join(conductorDir, "status.json");
|
|
307
|
+
const flatConductor = discoverConductor(flatStatus, explicitConductor);
|
|
308
|
+
if (fs.existsSync(flatStatus) || (flatConductor && fs.existsSync(flatConductor))) {
|
|
309
|
+
add(workflowName(flatStatus, flatConductor, conductorDir), conductorDir, flatStatus, flatConductor);
|
|
310
|
+
}
|
|
311
|
+
|
|
312
|
+
// subdirectories
|
|
313
|
+
if (fs.existsSync(conductorDir)) {
|
|
314
|
+
for (const entry of fs.readdirSync(conductorDir, { withFileTypes: true })) {
|
|
315
|
+
if (!entry.isDirectory() || entry.name === "history") continue;
|
|
316
|
+
const dir = path.join(conductorDir, entry.name);
|
|
317
|
+
const sp = path.join(dir, "status.json");
|
|
318
|
+
const cp = fs.existsSync(path.join(dir, "conductor.yaml"))
|
|
319
|
+
? path.join(dir, "conductor.yaml")
|
|
320
|
+
: discoverConductor(sp, null);
|
|
321
|
+
if (fs.existsSync(sp) || (cp && fs.existsSync(cp))) {
|
|
322
|
+
add(workflowName(sp, cp, dir), dir, sp, cp);
|
|
323
|
+
}
|
|
324
|
+
}
|
|
325
|
+
}
|
|
326
|
+
return found;
|
|
327
|
+
}
|
|
328
|
+
|
|
210
329
|
export function startServer({ statusPath, conductorPath: explicitConductor, port }) {
|
|
211
330
|
const absStatus = path.resolve(process.cwd(), statusPath);
|
|
212
|
-
|
|
213
|
-
|
|
214
|
-
const watchDir = path.dirname(absStatus);
|
|
215
|
-
const historyDir = path.join(watchDir, "history");
|
|
331
|
+
const conductorDir = path.dirname(absStatus);
|
|
216
332
|
|
|
217
333
|
/** @type {Set<http.ServerResponse>} */
|
|
218
334
|
const clients = new Set();
|
|
335
|
+
/** @type {Map<string, Set<string>>} per-workflow archived run ids */
|
|
336
|
+
const archivedByWf = new Map();
|
|
337
|
+
|
|
338
|
+
const archivedSetFor = (wf) => {
|
|
339
|
+
let set = archivedByWf.get(wf.name);
|
|
340
|
+
if (!set) {
|
|
341
|
+
set = new Set();
|
|
342
|
+
for (const r of listHistory(wf.historyDir)) if (r.run_id) set.add(r.run_id);
|
|
343
|
+
archivedByWf.set(wf.name, set);
|
|
344
|
+
}
|
|
345
|
+
return set;
|
|
346
|
+
};
|
|
347
|
+
|
|
348
|
+
const snapshotFor = (wf) => {
|
|
349
|
+
const snap = readSnapshot(wf.statusPath, wf.conductorPath);
|
|
350
|
+
snap.workflow = wf.name;
|
|
351
|
+
return snap;
|
|
352
|
+
};
|
|
219
353
|
|
|
220
|
-
|
|
221
|
-
|
|
222
|
-
for (const r of listHistory(historyDir)) if (r.run_id) archivedRunIds.add(r.run_id);
|
|
354
|
+
const findWf = (name) =>
|
|
355
|
+
discoverWorkflows(conductorDir, absStatus, explicitConductor).find((w) => w.name === name);
|
|
223
356
|
|
|
224
|
-
const
|
|
225
|
-
const
|
|
226
|
-
for (const res of clients) res.write(
|
|
357
|
+
const sendAll = (event, data) => {
|
|
358
|
+
const payload = `event: ${event}\ndata: ${JSON.stringify(data)}\n\n`;
|
|
359
|
+
for (const res of clients) res.write(payload);
|
|
227
360
|
};
|
|
228
361
|
|
|
229
362
|
const broadcast = () => {
|
|
230
|
-
|
|
231
|
-
|
|
232
|
-
|
|
233
|
-
|
|
234
|
-
|
|
235
|
-
|
|
363
|
+
for (const wf of discoverWorkflows(conductorDir, absStatus, explicitConductor)) {
|
|
364
|
+
const snap = snapshotFor(wf);
|
|
365
|
+
sendAll("update", snap);
|
|
366
|
+
if (archiveIfDone(wf.historyDir, snap, archivedSetFor(wf))) {
|
|
367
|
+
sendAll("history", { workflow: wf.name, runs: listHistory(wf.historyDir) });
|
|
368
|
+
}
|
|
236
369
|
}
|
|
237
|
-
if (archiveIfDone(historyDir, snapshot, archivedRunIds)) broadcastHistory();
|
|
238
370
|
};
|
|
239
371
|
|
|
240
|
-
//
|
|
241
|
-
// watching a single file that gets atomically replaced). Debounced.
|
|
372
|
+
// Recursively watch the .conductor root so subdirectory status files are seen.
|
|
242
373
|
let timer = null;
|
|
243
374
|
const schedule = () => {
|
|
244
375
|
clearTimeout(timer);
|
|
245
376
|
timer = setTimeout(broadcast, 80);
|
|
246
377
|
};
|
|
247
378
|
try {
|
|
248
|
-
fs.mkdirSync(
|
|
249
|
-
|
|
379
|
+
fs.mkdirSync(conductorDir, { recursive: true });
|
|
380
|
+
try {
|
|
381
|
+
fs.watch(conductorDir, { recursive: true }, schedule);
|
|
382
|
+
} catch {
|
|
383
|
+
fs.watch(conductorDir, schedule); // platforms without recursive watch
|
|
384
|
+
}
|
|
250
385
|
} catch (e) {
|
|
251
|
-
console.warn(`[
|
|
386
|
+
console.warn(`[conductor-board] watch failed: ${e.message}`);
|
|
252
387
|
}
|
|
253
388
|
|
|
389
|
+
const sendSnapshotsTo = (res) => {
|
|
390
|
+
for (const wf of discoverWorkflows(conductorDir, absStatus, explicitConductor)) {
|
|
391
|
+
res.write(`event: update\ndata: ${JSON.stringify(snapshotFor(wf))}\n\n`);
|
|
392
|
+
res.write(
|
|
393
|
+
`event: history\ndata: ${JSON.stringify({
|
|
394
|
+
workflow: wf.name,
|
|
395
|
+
runs: listHistory(wf.historyDir),
|
|
396
|
+
})}\n\n`,
|
|
397
|
+
);
|
|
398
|
+
}
|
|
399
|
+
};
|
|
400
|
+
|
|
401
|
+
const json = (res, code, body) => {
|
|
402
|
+
res.writeHead(code, { "content-type": "application/json; charset=utf-8" });
|
|
403
|
+
res.end(JSON.stringify(body));
|
|
404
|
+
};
|
|
405
|
+
|
|
254
406
|
const server = http.createServer((req, res) => {
|
|
255
407
|
const url = (req.url || "/").split("?")[0];
|
|
408
|
+
const wfs = () => discoverWorkflows(conductorDir, absStatus, explicitConductor);
|
|
256
409
|
|
|
257
410
|
if (url === "/health") {
|
|
258
|
-
res
|
|
259
|
-
|
|
260
|
-
|
|
261
|
-
|
|
262
|
-
|
|
263
|
-
|
|
264
|
-
|
|
411
|
+
return json(res, 200, {
|
|
412
|
+
status: "ok",
|
|
413
|
+
version: VERSION,
|
|
414
|
+
watching: path.relative(process.cwd(), conductorDir) || conductorDir,
|
|
415
|
+
port: server.address()?.port ?? null,
|
|
416
|
+
workflows: wfs().map((w) => w.name),
|
|
417
|
+
});
|
|
418
|
+
}
|
|
419
|
+
|
|
420
|
+
// ---- multi-workflow API ----
|
|
421
|
+
if (url === "/api/workflows") {
|
|
422
|
+
return json(
|
|
423
|
+
res,
|
|
424
|
+
200,
|
|
425
|
+
wfs().map((wf) => {
|
|
426
|
+
const snap = snapshotFor(wf);
|
|
427
|
+
const st = snap.status?.status;
|
|
428
|
+
return {
|
|
429
|
+
name: wf.name,
|
|
430
|
+
status: st ?? "idle",
|
|
431
|
+
active: st === "running",
|
|
432
|
+
done: snap.status?.steps
|
|
433
|
+
? Object.values(snap.status.steps).filter((s) => s && s.status === "done").length
|
|
434
|
+
: 0,
|
|
435
|
+
total: snap.status?.steps ? Object.keys(snap.status.steps).length : 0,
|
|
436
|
+
started_at: snap.status?.started_at ?? null,
|
|
437
|
+
runs: listHistory(wf.historyDir).length,
|
|
438
|
+
hasConductor: !!snap.conductorYaml,
|
|
439
|
+
};
|
|
265
440
|
}),
|
|
266
441
|
);
|
|
267
|
-
return;
|
|
268
442
|
}
|
|
269
443
|
|
|
270
|
-
|
|
271
|
-
|
|
272
|
-
|
|
273
|
-
|
|
444
|
+
let m;
|
|
445
|
+
|
|
446
|
+
// apply optimization suggestions back to the conductor (mutate + backup + re-validate)
|
|
447
|
+
if (req.method === "POST" && (m = url.match(/^\/api\/workflow\/([^/]+)\/apply-suggestion$/))) {
|
|
448
|
+
const wf = findWf(decodeURIComponent(m[1]));
|
|
449
|
+
if (!wf) return json(res, 404, { error: "workflow not found" });
|
|
450
|
+
readBody(req).then((bodyStr) => {
|
|
451
|
+
let ids;
|
|
452
|
+
try {
|
|
453
|
+
ids = JSON.parse(bodyStr || "{}").suggestions;
|
|
454
|
+
} catch {
|
|
455
|
+
return json(res, 400, { error: "invalid request body" });
|
|
456
|
+
}
|
|
457
|
+
if (!Array.isArray(ids)) return json(res, 400, { error: "suggestions must be an array" });
|
|
458
|
+
if (!wf.conductorPath || !fs.existsSync(wf.conductorPath))
|
|
459
|
+
return json(res, 400, { error: "no conductor file for this workflow" });
|
|
460
|
+
|
|
461
|
+
let status;
|
|
462
|
+
try {
|
|
463
|
+
status = JSON.parse(fs.readFileSync(wf.statusPath, "utf8"));
|
|
464
|
+
} catch {
|
|
465
|
+
return json(res, 500, { error: "could not read status.json" });
|
|
466
|
+
}
|
|
467
|
+
const chosen = (status.suggestions || []).filter((s) => ids.includes(s.id));
|
|
468
|
+
if (chosen.length === 0) return json(res, 400, { error: "no matching suggestions" });
|
|
469
|
+
|
|
470
|
+
const original = fs.readFileSync(wf.conductorPath, "utf8");
|
|
471
|
+
const backup = `${wf.conductorPath}.bak.${new Date().toISOString().replace(/[:.]/g, "-")}`;
|
|
472
|
+
try {
|
|
473
|
+
fs.writeFileSync(backup, original);
|
|
474
|
+
} catch {
|
|
475
|
+
/* backup best-effort */
|
|
476
|
+
}
|
|
477
|
+
|
|
478
|
+
let doc;
|
|
479
|
+
try {
|
|
480
|
+
doc = yaml.load(original);
|
|
481
|
+
} catch (e) {
|
|
482
|
+
return json(res, 500, { error: `conductor parse error: ${e.message}` });
|
|
483
|
+
}
|
|
484
|
+
applyMutations(doc, chosen);
|
|
485
|
+
|
|
486
|
+
const errors = validateConductor(doc);
|
|
487
|
+
if (errors.length) {
|
|
488
|
+
// leave the original conductor untouched (we never wrote it)
|
|
489
|
+
return json(res, 422, { error: `would be invalid: ${errors[0]}`, errors });
|
|
490
|
+
}
|
|
491
|
+
try {
|
|
492
|
+
fs.writeFileSync(wf.conductorPath, yaml.dump(doc, { lineWidth: 100 }));
|
|
493
|
+
} catch (e) {
|
|
494
|
+
try {
|
|
495
|
+
fs.writeFileSync(wf.conductorPath, original); // rollback
|
|
496
|
+
} catch {
|
|
497
|
+
/* ignore */
|
|
498
|
+
}
|
|
499
|
+
return json(res, 500, { error: `write failed, rolled back: ${e.message}` });
|
|
500
|
+
}
|
|
501
|
+
return json(res, 200, {
|
|
502
|
+
ok: true,
|
|
503
|
+
applied: chosen.map((s) => s.id),
|
|
504
|
+
backup: path.basename(backup),
|
|
505
|
+
});
|
|
506
|
+
});
|
|
274
507
|
return;
|
|
275
508
|
}
|
|
276
509
|
|
|
277
|
-
if (
|
|
278
|
-
|
|
279
|
-
res
|
|
280
|
-
|
|
510
|
+
if ((m = url.match(/^\/api\/workflow\/([^/]+)\/state$/))) {
|
|
511
|
+
const wf = findWf(decodeURIComponent(m[1]));
|
|
512
|
+
return wf ? json(res, 200, snapshotFor(wf)) : json(res, 404, { error: "not found" });
|
|
513
|
+
}
|
|
514
|
+
if ((m = url.match(/^\/api\/workflow\/([^/]+)\/history$/))) {
|
|
515
|
+
const wf = findWf(decodeURIComponent(m[1]));
|
|
516
|
+
return wf ? json(res, 200, listHistory(wf.historyDir)) : json(res, 404, { error: "not found" });
|
|
517
|
+
}
|
|
518
|
+
if ((m = url.match(/^\/api\/workflow\/([^/]+)\/history\/(.+)$/))) {
|
|
519
|
+
const wf = findWf(decodeURIComponent(m[1]));
|
|
520
|
+
if (!wf) return json(res, 404, { error: "not found" });
|
|
521
|
+
const rec = getHistory(wf.historyDir, decodeURIComponent(m[2]));
|
|
522
|
+
return rec ? json(res, 200, rec) : json(res, 404, { error: "not found" });
|
|
281
523
|
}
|
|
282
524
|
|
|
525
|
+
// ---- backwards-compatible single-workflow API (primary = first found) ----
|
|
526
|
+
const primary = wfs()[0];
|
|
527
|
+
if (url === "/api/state") {
|
|
528
|
+
return json(res, 200, primary ? snapshotFor(primary) : { status: null, conductorYaml: null });
|
|
529
|
+
}
|
|
530
|
+
if (url === "/history") {
|
|
531
|
+
return json(res, 200, primary ? listHistory(primary.historyDir) : []);
|
|
532
|
+
}
|
|
283
533
|
if (url.startsWith("/history/")) {
|
|
284
534
|
const id = decodeURIComponent(url.slice("/history/".length));
|
|
285
|
-
const rec = getHistory(historyDir, id);
|
|
286
|
-
|
|
287
|
-
res.writeHead(404, { "content-type": "application/json" }).end('{"error":"not found"}');
|
|
288
|
-
return;
|
|
289
|
-
}
|
|
290
|
-
res.writeHead(200, { "content-type": "application/json; charset=utf-8" });
|
|
291
|
-
res.end(JSON.stringify(rec));
|
|
292
|
-
return;
|
|
535
|
+
const rec = primary ? getHistory(primary.historyDir, id) : null;
|
|
536
|
+
return rec ? json(res, 200, rec) : json(res, 404, { error: "not found" });
|
|
293
537
|
}
|
|
294
538
|
|
|
295
539
|
if (url === "/events") {
|
|
@@ -299,17 +543,11 @@ export function startServer({ statusPath, conductorPath: explicitConductor, port
|
|
|
299
543
|
connection: "keep-alive",
|
|
300
544
|
});
|
|
301
545
|
res.write("retry: 2000\n\n");
|
|
302
|
-
|
|
303
|
-
res.write(
|
|
304
|
-
`event: update\ndata: ${JSON.stringify(
|
|
305
|
-
readSnapshot(absStatus, conductorPath),
|
|
306
|
-
)}\n\n`,
|
|
307
|
-
);
|
|
308
|
-
res.write(`event: history\ndata: ${JSON.stringify(listHistory(historyDir))}\n\n`);
|
|
546
|
+
sendSnapshotsTo(res);
|
|
309
547
|
clients.add(res);
|
|
310
|
-
const
|
|
548
|
+
const hb = setInterval(() => res.write(": ping\n\n"), 25000);
|
|
311
549
|
req.on("close", () => {
|
|
312
|
-
clearInterval(
|
|
550
|
+
clearInterval(hb);
|
|
313
551
|
clients.delete(res);
|
|
314
552
|
});
|
|
315
553
|
return;
|
|
@@ -318,15 +556,22 @@ export function startServer({ statusPath, conductorPath: explicitConductor, port
|
|
|
318
556
|
serveStatic(req, res);
|
|
319
557
|
});
|
|
320
558
|
|
|
321
|
-
|
|
322
|
-
// so a setup conductor's health check never has to hardcode a port.
|
|
323
|
-
const serverJsonPath = path.join(watchDir, "server.json");
|
|
559
|
+
const serverJsonPath = path.join(conductorDir, "server.json");
|
|
324
560
|
|
|
325
|
-
return new Promise((resolve) => {
|
|
561
|
+
return new Promise((resolve, reject) => {
|
|
562
|
+
// Reject on listen errors (e.g. EADDRINUSE) so the CLI can walk to the next
|
|
563
|
+
// port instead of crashing on an unhandled 'error' event.
|
|
564
|
+
const onError = (e) => {
|
|
565
|
+
server.off("error", onError);
|
|
566
|
+
reject(e);
|
|
567
|
+
};
|
|
568
|
+
server.once("error", onError);
|
|
326
569
|
server.listen(port, () => {
|
|
570
|
+
server.off("error", onError);
|
|
327
571
|
const actualPort = server.address().port;
|
|
572
|
+
const discovered = discoverWorkflows(conductorDir, absStatus, explicitConductor);
|
|
328
573
|
try {
|
|
329
|
-
fs.mkdirSync(
|
|
574
|
+
fs.mkdirSync(conductorDir, { recursive: true });
|
|
330
575
|
fs.writeFileSync(
|
|
331
576
|
serverJsonPath,
|
|
332
577
|
JSON.stringify(
|
|
@@ -335,6 +580,7 @@ export function startServer({ statusPath, conductorPath: explicitConductor, port
|
|
|
335
580
|
url: `http://localhost:${actualPort}`,
|
|
336
581
|
pid: process.pid,
|
|
337
582
|
started_at: new Date().toISOString(),
|
|
583
|
+
workflows: discovered.map((w) => w.name),
|
|
338
584
|
},
|
|
339
585
|
null,
|
|
340
586
|
2,
|
|
@@ -343,7 +589,13 @@ export function startServer({ statusPath, conductorPath: explicitConductor, port
|
|
|
343
589
|
} catch (e) {
|
|
344
590
|
console.warn(`[conductor-board] could not write server.json: ${e.message}`);
|
|
345
591
|
}
|
|
346
|
-
resolve({
|
|
592
|
+
resolve({
|
|
593
|
+
server,
|
|
594
|
+
serverJsonPath,
|
|
595
|
+
absStatus,
|
|
596
|
+
conductorPath: discovered[0]?.conductorPath ?? null,
|
|
597
|
+
workflows: discovered.map((w) => w.name),
|
|
598
|
+
});
|
|
347
599
|
});
|
|
348
600
|
});
|
|
349
601
|
}
|