codekin 0.2.2 → 0.3.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (64) hide show
  1. package/dist/assets/index-CYGuwwHU.js +183 -0
  2. package/dist/assets/index-DW70VouA.css +1 -0
  3. package/dist/index.html +3 -3
  4. package/package.json +8 -5
  5. package/server/dist/approval-manager.d.ts +71 -0
  6. package/server/dist/approval-manager.js +249 -0
  7. package/server/dist/approval-manager.js.map +1 -0
  8. package/server/dist/auth-routes.d.ts +11 -0
  9. package/server/dist/auth-routes.js +11 -0
  10. package/server/dist/auth-routes.js.map +1 -1
  11. package/server/dist/claude-process.js +25 -4
  12. package/server/dist/claude-process.js.map +1 -1
  13. package/server/dist/config.d.ts +1 -2
  14. package/server/dist/config.js +16 -4
  15. package/server/dist/config.js.map +1 -1
  16. package/server/dist/crypto-utils.d.ts +16 -1
  17. package/server/dist/crypto-utils.js +44 -1
  18. package/server/dist/crypto-utils.js.map +1 -1
  19. package/server/dist/docs-routes.d.ts +16 -0
  20. package/server/dist/docs-routes.js +141 -0
  21. package/server/dist/docs-routes.js.map +1 -0
  22. package/server/dist/session-manager.d.ts +37 -84
  23. package/server/dist/session-manager.js +95 -472
  24. package/server/dist/session-manager.js.map +1 -1
  25. package/server/dist/session-naming.d.ts +35 -0
  26. package/server/dist/session-naming.js +168 -0
  27. package/server/dist/session-naming.js.map +1 -0
  28. package/server/dist/session-persistence.d.ts +30 -0
  29. package/server/dist/session-persistence.js +93 -0
  30. package/server/dist/session-persistence.js.map +1 -0
  31. package/server/dist/session-routes.d.ts +2 -1
  32. package/server/dist/session-routes.js +11 -8
  33. package/server/dist/session-routes.js.map +1 -1
  34. package/server/dist/stepflow-handler.d.ts +3 -9
  35. package/server/dist/stepflow-handler.js +19 -54
  36. package/server/dist/stepflow-handler.js.map +1 -1
  37. package/server/dist/types.d.ts +7 -0
  38. package/server/dist/upload-routes.js +6 -3
  39. package/server/dist/upload-routes.js.map +1 -1
  40. package/server/dist/webhook-github.d.ts +23 -5
  41. package/server/dist/webhook-github.js +23 -5
  42. package/server/dist/webhook-github.js.map +1 -1
  43. package/server/dist/webhook-handler-base.d.ts +45 -0
  44. package/server/dist/webhook-handler-base.js +86 -0
  45. package/server/dist/webhook-handler-base.js.map +1 -0
  46. package/server/dist/webhook-handler.d.ts +3 -10
  47. package/server/dist/webhook-handler.js +6 -46
  48. package/server/dist/webhook-handler.js.map +1 -1
  49. package/server/dist/webhook-workspace.d.ts +8 -1
  50. package/server/dist/webhook-workspace.js +71 -47
  51. package/server/dist/webhook-workspace.js.map +1 -1
  52. package/server/dist/workflow-config.d.ts +2 -0
  53. package/server/dist/workflow-config.js +1 -1
  54. package/server/dist/workflow-config.js.map +1 -1
  55. package/server/dist/workflow-loader.d.ts +2 -0
  56. package/server/dist/workflow-loader.js +13 -9
  57. package/server/dist/workflow-loader.js.map +1 -1
  58. package/server/dist/workflow-routes.js +3 -2
  59. package/server/dist/workflow-routes.js.map +1 -1
  60. package/server/dist/ws-server.js +77 -16
  61. package/server/dist/ws-server.js.map +1 -1
  62. package/server/workflows/repo-health.weekly.md +111 -0
  63. package/dist/assets/index-CQdQ4FhF.css +0 -1
  64. package/dist/assets/index-D6pRORYQ.js +0 -115
@@ -0,0 +1 @@
1
+ @import"https://fonts.googleapis.com/css2?family=Inconsolata:wght@300..700&family=Lato:wght@300;400;700&display=swap";@layer components;@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-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;--tw-ease:initial}}}@layer theme{:root,:host{--font-sans:"Lato", sans-serif;--font-mono:ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace;--color-yellow-500:oklch(79.5% .184 86.047);--color-yellow-900:oklch(42.1% .095 57.708);--color-black:#000;--color-white:#fff;--spacing:.25rem;--container-xs:20rem;--container-md:28rem;--container-lg:32rem;--container-2xl:42rem;--font-weight-normal:400;--font-weight-medium:500;--font-weight-semibold:600;--tracking-wide:.025em;--tracking-wider:.05em;--tracking-widest:.1em;--leading-snug:1.375;--leading-relaxed:1.625;--radius-md:.375rem;--radius-lg:.5rem;--radius-xl:.75rem;--ease-out:cubic-bezier(0, 0, .2, 1);--animate-spin:spin 1s linear infinite;--animate-pulse:pulse 2s cubic-bezier(.4, 0, .6, 1) infinite;--blur-sm:8px;--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-primary-2:#fcedcc;--color-primary-3:#f8dca0;--color-primary-4:#f0c56e;--color-primary-5:#e4ae42;--color-primary-6:#cf9a2d;--color-primary-7:#9d7440;--color-primary-8:#9e7335;--color-primary-9:#7d5b2a;--color-primary-10:#5c4420;--color-primary-11:#3a2c15;--color-accent-2:#d0f0f5;--color-accent-3:#a5e0ea;--color-accent-4:#74cbda;--color-accent-5:#45b5c9;--color-accent-6:#2d9bb0;--color-accent-7:#248192;--color-accent-8:#1d6876;--color-accent-9:#17515c;--color-accent-11:#0c282e;--color-neutral-1:#dce4e5;--color-neutral-2:#c6d4d5;--color-neutral-3:#b8ccce;--color-neutral-4:#94aeb1;--color-neutral-5:#748d91;--color-neutral-6:#5c7377;--color-neutral-7:#465c60;--color-neutral-8:#354a4e;--color-neutral-9:#263a3e;--color-neutral-10:#1a2c30;--color-neutral-11:#101e21;--color-neutral-12:#090e0f;--color-error-2:#fddcdc;--color-error-4:#f28a8a;--color-error-5:#e86060;--color-error-7:#c03636;--color-error-8:#a12c2c;--color-error-9:#822323;--color-error-10:#631b1b;--color-warning-2:#fdf1c8;--color-warning-3:#fae295;--color-warning-4:#f4ce5e;--color-warning-5:#e8b830;--color-warning-6:#d0a020;--color-warning-7:#b08618;--color-warning-8:#906d12;--color-warning-9:#73560e;--color-warning-10:#56400a;--color-warning-11:#382a07;--color-success-2:#d0f2e2;--color-success-3:#a0e4c4;--color-success-4:#6dd1a2;--color-success-5:#40bc80;--color-success-6:#2da368;--color-success-7:#248854;--color-success-8:#1d6e43;--color-success-9:#175634;--color-success-10:#114027}}@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%;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]){appearance:button}::file-selector-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 utilities{.pointer-events-none{pointer-events:none}.collapse{visibility:collapse}.visible{visibility:visible}.absolute{position:absolute}.fixed{position:fixed}.relative{position:relative}.static{position:static}.sticky{position:sticky}.inset-0{inset:calc(var(--spacing) * 0)}.start{inset-inline-start:var(--spacing)}.end{inset-inline-end:var(--spacing)}.top-0{top:calc(var(--spacing) * 0)}.top-2{top:calc(var(--spacing) * 2)}.top-3{top:calc(var(--spacing) * 3)}.top-full{top:100%}.right-0{right:calc(var(--spacing) * 0)}.right-2{right:calc(var(--spacing) * 2)}.right-3{right:calc(var(--spacing) * 3)}.bottom-0{bottom:calc(var(--spacing) * 0)}.bottom-3{bottom:calc(var(--spacing) * 3)}.bottom-full{bottom:100%}.left-0{left:calc(var(--spacing) * 0)}.left-1\/2{left:50%}.left-full{left:100%}.isolate{isolation:isolate}.z-10{z-index:10}.z-20{z-index:20}.z-40{z-index:40}.z-50{z-index:50}.col-span-2{grid-column:span 2/span 2}.container{width:100%}@media(min-width:40rem){.container{max-width:40rem}}@media(min-width:48rem){.container{max-width:48rem}}@media(min-width:64rem){.container{max-width:64rem}}@media(min-width:80rem){.container{max-width:80rem}}@media(min-width:96rem){.container{max-width:96rem}}.mx-1{margin-inline:calc(var(--spacing) * 1)}.mx-3{margin-inline:calc(var(--spacing) * 3)}.mx-4{margin-inline:calc(var(--spacing) * 4)}.mx-10{margin-inline:calc(var(--spacing) * 10)}.mx-auto{margin-inline:auto}.my-0\.5{margin-block:calc(var(--spacing) * .5)}.my-1{margin-block:calc(var(--spacing) * 1)}.my-2{margin-block:calc(var(--spacing) * 2)}.-mt-0\.5{margin-top:calc(var(--spacing) * -.5)}.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)}.mt-6{margin-top:calc(var(--spacing) * 6)}.mt-\[3px\]{margin-top:3px}.mt-\[7px\]{margin-top:7px}.mt-auto{margin-top:auto}.mr-1{margin-right:calc(var(--spacing) * 1)}.mb-1{margin-bottom:calc(var(--spacing) * 1)}.mb-1\.5{margin-bottom:calc(var(--spacing) * 1.5)}.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)}.mb-6{margin-bottom:calc(var(--spacing) * 6)}.ml-0\.5{margin-left:calc(var(--spacing) * .5)}.ml-1{margin-left:calc(var(--spacing) * 1)}.ml-1\.5{margin-left:calc(var(--spacing) * 1.5)}.ml-2{margin-left:calc(var(--spacing) * 2)}.ml-3{margin-left:calc(var(--spacing) * 3)}.ml-6{margin-left:calc(var(--spacing) * 6)}.ml-auto{margin-left:auto}.block{display:block}.contents{display:contents}.flex{display:flex}.grid{display:grid}.hidden{display:none}.inline{display:inline}.inline-block{display:inline-block}.inline-flex{display:inline-flex}.table{display:table}.size-10{width:calc(var(--spacing) * 10);height:calc(var(--spacing) * 10)}.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-4{height:calc(var(--spacing) * 4)}.h-6{height:calc(var(--spacing) * 6)}.h-10{height:calc(var(--spacing) * 10)}.h-12{height:calc(var(--spacing) * 12)}.h-52{height:calc(var(--spacing) * 52)}.h-full{height:100%}.h-px{height:1px}.max-h-48{max-height:calc(var(--spacing) * 48)}.max-h-64{max-height:calc(var(--spacing) * 64)}.max-h-72{max-height:calc(var(--spacing) * 72)}.max-h-96{max-height:calc(var(--spacing) * 96)}.max-h-\[50vh\]{max-height:50vh}.max-h-\[85vh\]{max-height:85vh}.max-h-\[90vh\]{max-height:90vh}.max-h-\[95vh\]{max-height:95vh}.max-h-\[200px\]{max-height:200px}.max-h-\[300px\]{max-height:300px}.min-h-0{min-height:calc(var(--spacing) * 0)}.min-h-\[200px\]{min-height:200px}.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-4{width:calc(var(--spacing) * 4)}.w-6{width:calc(var(--spacing) * 6)}.w-10{width:calc(var(--spacing) * 10)}.w-12{width:calc(var(--spacing) * 12)}.w-20{width:calc(var(--spacing) * 20)}.w-40{width:calc(var(--spacing) * 40)}.w-56{width:calc(var(--spacing) * 56)}.w-64{width:calc(var(--spacing) * 64)}.w-72{width:calc(var(--spacing) * 72)}.w-\[12px\]{width:12px}.w-\[14px\]{width:14px}.w-\[280px\]{width:280px}.w-\[286px\]{width:286px}.w-\[480px\]{width:480px}.w-\[520px\]{width:520px}.w-full{width:100%}.w-px{width:1px}.max-w-2xl{max-width:var(--container-2xl)}.max-w-24{max-width:calc(var(--spacing) * 24)}.max-w-\[80\%\]{max-width:80%}.max-w-\[85vw\]{max-width:85vw}.max-w-\[95\%\]{max-width:95%}.max-w-\[140px\]{max-width:140px}.max-w-\[150px\]{max-width:150px}.max-w-\[180px\]{max-width:180px}.max-w-\[400px\]{max-width:400px}.max-w-\[720px\]{max-width:720px}.max-w-full{max-width:100%}.max-w-lg{max-width:var(--container-lg)}.max-w-md{max-width:var(--container-md)}.max-w-none{max-width:none}.max-w-xs{max-width:var(--container-xs)}.min-w-0{min-width:calc(var(--spacing) * 0)}.min-w-\[160px\]{min-width:160px}.flex-1{flex:1}.flex-shrink-0,.shrink-0{flex-shrink:0}.border-collapse{border-collapse:collapse}.-translate-x-1\/2{--tw-translate-x: -50% ;translate:var(--tw-translate-x) var(--tw-translate-y)}.-translate-x-full{--tw-translate-x:-100%;translate:var(--tw-translate-x) var(--tw-translate-y)}.translate-x-0{--tw-translate-x:calc(var(--spacing) * 0);translate:var(--tw-translate-x) var(--tw-translate-y)}.rotate-90{rotate:90deg}.animate-\[spin_3s_linear_infinite\]{animation:3s linear infinite spin}.animate-pulse{animation:var(--animate-pulse)}.animate-spin{animation:var(--animate-spin)}.cursor-col-resize{cursor:col-resize}.cursor-pointer{cursor:pointer}.cursor-row-resize{cursor:row-resize}.cursor-wait{cursor:wait}.resize{resize:both}.resize-none{resize:none}.list-inside{list-style-position:inside}.list-disc{list-style-type:disc}.list-none{list-style-type:none}.grid-cols-2{grid-template-columns:repeat(2,minmax(0,1fr))}.flex-col{flex-direction:column}.flex-row{flex-direction:row}.flex-wrap{flex-wrap:wrap}.items-baseline{align-items:baseline}.items-center{align-items:center}.items-end{align-items:flex-end}.items-start{align-items:flex-start}.justify-between{justify-content:space-between}.justify-center{justify-content:center}.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-3{gap:calc(var(--spacing) * 3)}: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\.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)))}: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-4{column-gap:calc(var(--spacing) * 4)}.gap-x-6{column-gap:calc(var(--spacing) * 6)}.gap-y-0\.5{row-gap:calc(var(--spacing) * .5)}.gap-y-4{row-gap:calc(var(--spacing) * 4)}: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-neutral-9\/30>:not(:last-child)){border-color:#263a3e4d}@supports (color:color-mix(in lab,red,red)){:where(.divide-neutral-9\/30>:not(:last-child)){border-color:color-mix(in oklab,var(--color-neutral-9) 30%,transparent)}}:where(.divide-neutral-9\/50>:not(:last-child)){border-color:#263a3e80}@supports (color:color-mix(in lab,red,red)){:where(.divide-neutral-9\/50>:not(:last-child)){border-color:color-mix(in oklab,var(--color-neutral-9) 50%,transparent)}}.self-center{align-self:center}.truncate{text-overflow:ellipsis;white-space:nowrap;overflow:hidden}.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:.25rem}.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-xl{border-top-left-radius:var(--radius-xl);border-top-right-radius:var(--radius-xl)}.border{border-style:var(--tw-border-style);border-width:1px}.border-2{border-style:var(--tw-border-style);border-width:2px}.border-t{border-top-style:var(--tw-border-style);border-top-width:1px}.border-t-2{border-top-style:var(--tw-border-style);border-top-width:2px}.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-accent-6{border-color:var(--color-accent-6)}.border-accent-6\/60{border-color:#2d9bb099}@supports (color:color-mix(in lab,red,red)){.border-accent-6\/60{border-color:color-mix(in oklab,var(--color-accent-6) 60%,transparent)}}.border-accent-7\/60{border-color:#24819299}@supports (color:color-mix(in lab,red,red)){.border-accent-7\/60{border-color:color-mix(in oklab,var(--color-accent-7) 60%,transparent)}}.border-accent-8\/40{border-color:#1d687666}@supports (color:color-mix(in lab,red,red)){.border-accent-8\/40{border-color:color-mix(in oklab,var(--color-accent-8) 40%,transparent)}}.border-error-7{border-color:var(--color-error-7)}.border-error-7\/60{border-color:#c0363699}@supports (color:color-mix(in lab,red,red)){.border-error-7\/60{border-color:color-mix(in oklab,var(--color-error-7) 60%,transparent)}}.border-error-8\/50{border-color:#a12c2c80}@supports (color:color-mix(in lab,red,red)){.border-error-8\/50{border-color:color-mix(in oklab,var(--color-error-8) 50%,transparent)}}.border-error-9\/50{border-color:#82232380}@supports (color:color-mix(in lab,red,red)){.border-error-9\/50{border-color:color-mix(in oklab,var(--color-error-9) 50%,transparent)}}.border-neutral-5{border-color:var(--color-neutral-5)}.border-neutral-6{border-color:var(--color-neutral-6)}.border-neutral-7{border-color:var(--color-neutral-7)}.border-neutral-7\/30{border-color:#465c604d}@supports (color:color-mix(in lab,red,red)){.border-neutral-7\/30{border-color:color-mix(in oklab,var(--color-neutral-7) 30%,transparent)}}.border-neutral-7\/50{border-color:#465c6080}@supports (color:color-mix(in lab,red,red)){.border-neutral-7\/50{border-color:color-mix(in oklab,var(--color-neutral-7) 50%,transparent)}}.border-neutral-7\/60{border-color:#465c6099}@supports (color:color-mix(in lab,red,red)){.border-neutral-7\/60{border-color:color-mix(in oklab,var(--color-neutral-7) 60%,transparent)}}.border-neutral-8{border-color:var(--color-neutral-8)}.border-neutral-8\/30{border-color:#354a4e4d}@supports (color:color-mix(in lab,red,red)){.border-neutral-8\/30{border-color:color-mix(in oklab,var(--color-neutral-8) 30%,transparent)}}.border-neutral-8\/40{border-color:#354a4e66}@supports (color:color-mix(in lab,red,red)){.border-neutral-8\/40{border-color:color-mix(in oklab,var(--color-neutral-8) 40%,transparent)}}.border-neutral-8\/50{border-color:#354a4e80}@supports (color:color-mix(in lab,red,red)){.border-neutral-8\/50{border-color:color-mix(in oklab,var(--color-neutral-8) 50%,transparent)}}.border-neutral-8\/60{border-color:#354a4e99}@supports (color:color-mix(in lab,red,red)){.border-neutral-8\/60{border-color:color-mix(in oklab,var(--color-neutral-8) 60%,transparent)}}.border-neutral-9{border-color:var(--color-neutral-9)}.border-neutral-9\/40{border-color:#263a3e66}@supports (color:color-mix(in lab,red,red)){.border-neutral-9\/40{border-color:color-mix(in oklab,var(--color-neutral-9) 40%,transparent)}}.border-neutral-9\/50{border-color:#263a3e80}@supports (color:color-mix(in lab,red,red)){.border-neutral-9\/50{border-color:color-mix(in oklab,var(--color-neutral-9) 50%,transparent)}}.border-neutral-9\/60{border-color:#263a3e99}@supports (color:color-mix(in lab,red,red)){.border-neutral-9\/60{border-color:color-mix(in oklab,var(--color-neutral-9) 60%,transparent)}}.border-neutral-10{border-color:var(--color-neutral-10)}.border-primary-6{border-color:var(--color-primary-6)}.border-primary-7{border-color:var(--color-primary-7)}.border-primary-9\/50{border-color:#7d5b2a80}@supports (color:color-mix(in lab,red,red)){.border-primary-9\/50{border-color:color-mix(in oklab,var(--color-primary-9) 50%,transparent)}}.border-success-7{border-color:var(--color-success-7)}.border-success-7\/60{border-color:#24885499}@supports (color:color-mix(in lab,red,red)){.border-success-7\/60{border-color:color-mix(in oklab,var(--color-success-7) 60%,transparent)}}.border-transparent{border-color:#0000}.border-warning-6{border-color:var(--color-warning-6)}.border-warning-7{border-color:var(--color-warning-7)}.border-warning-7\/60{border-color:#b0861899}@supports (color:color-mix(in lab,red,red)){.border-warning-7\/60{border-color:color-mix(in oklab,var(--color-warning-7) 60%,transparent)}}.border-warning-8\/50{border-color:#906d1280}@supports (color:color-mix(in lab,red,red)){.border-warning-8\/50{border-color:color-mix(in oklab,var(--color-warning-8) 50%,transparent)}}.border-warning-9\/50{border-color:#73560e80}@supports (color:color-mix(in lab,red,red)){.border-warning-9\/50{border-color:color-mix(in oklab,var(--color-warning-9) 50%,transparent)}}.border-l-accent-6{border-left-color:var(--color-accent-6)}.bg-accent-2{background-color:var(--color-accent-2)}.bg-accent-3{background-color:var(--color-accent-3)}.bg-accent-4{background-color:var(--color-accent-4)}.bg-accent-5{background-color:var(--color-accent-5)}.bg-accent-6{background-color:var(--color-accent-6)}.bg-accent-7{background-color:var(--color-accent-7)}.bg-accent-8{background-color:var(--color-accent-8)}.bg-accent-8\/50{background-color:#1d687680}@supports (color:color-mix(in lab,red,red)){.bg-accent-8\/50{background-color:color-mix(in oklab,var(--color-accent-8) 50%,transparent)}}.bg-accent-9\/20{background-color:#17515c33}@supports (color:color-mix(in lab,red,red)){.bg-accent-9\/20{background-color:color-mix(in oklab,var(--color-accent-9) 20%,transparent)}}.bg-accent-9\/30{background-color:#17515c4d}@supports (color:color-mix(in lab,red,red)){.bg-accent-9\/30{background-color:color-mix(in oklab,var(--color-accent-9) 30%,transparent)}}.bg-accent-9\/40{background-color:#17515c66}@supports (color:color-mix(in lab,red,red)){.bg-accent-9\/40{background-color:color-mix(in oklab,var(--color-accent-9) 40%,transparent)}}.bg-accent-11\/20{background-color:#0c282e33}@supports (color:color-mix(in lab,red,red)){.bg-accent-11\/20{background-color:color-mix(in oklab,var(--color-accent-11) 20%,transparent)}}.bg-black\/50{background-color:#00000080}@supports (color:color-mix(in lab,red,red)){.bg-black\/50{background-color:color-mix(in oklab,var(--color-black) 50%,transparent)}}.bg-black\/60{background-color:#0009}@supports (color:color-mix(in lab,red,red)){.bg-black\/60{background-color:color-mix(in oklab,var(--color-black) 60%,transparent)}}.bg-error-5{background-color:var(--color-error-5)}.bg-error-7{background-color:var(--color-error-7)}.bg-error-8{background-color:var(--color-error-8)}.bg-error-9\/30{background-color:#8223234d}@supports (color:color-mix(in lab,red,red)){.bg-error-9\/30{background-color:color-mix(in oklab,var(--color-error-9) 30%,transparent)}}.bg-error-9\/40{background-color:#82232366}@supports (color:color-mix(in lab,red,red)){.bg-error-9\/40{background-color:color-mix(in oklab,var(--color-error-9) 40%,transparent)}}.bg-error-10\/30{background-color:#631b1b4d}@supports (color:color-mix(in lab,red,red)){.bg-error-10\/30{background-color:color-mix(in oklab,var(--color-error-10) 30%,transparent)}}.bg-error-10\/50{background-color:#631b1b80}@supports (color:color-mix(in lab,red,red)){.bg-error-10\/50{background-color:color-mix(in oklab,var(--color-error-10) 50%,transparent)}}.bg-neutral-5{background-color:var(--color-neutral-5)}.bg-neutral-6{background-color:var(--color-neutral-6)}.bg-neutral-7{background-color:var(--color-neutral-7)}.bg-neutral-7\/50{background-color:#465c6080}@supports (color:color-mix(in lab,red,red)){.bg-neutral-7\/50{background-color:color-mix(in oklab,var(--color-neutral-7) 50%,transparent)}}.bg-neutral-8{background-color:var(--color-neutral-8)}.bg-neutral-8\/40{background-color:#354a4e66}@supports (color:color-mix(in lab,red,red)){.bg-neutral-8\/40{background-color:color-mix(in oklab,var(--color-neutral-8) 40%,transparent)}}.bg-neutral-8\/50{background-color:#354a4e80}@supports (color:color-mix(in lab,red,red)){.bg-neutral-8\/50{background-color:color-mix(in oklab,var(--color-neutral-8) 50%,transparent)}}.bg-neutral-8\/80{background-color:#354a4ecc}@supports (color:color-mix(in lab,red,red)){.bg-neutral-8\/80{background-color:color-mix(in oklab,var(--color-neutral-8) 80%,transparent)}}.bg-neutral-8\/90{background-color:#354a4ee6}@supports (color:color-mix(in lab,red,red)){.bg-neutral-8\/90{background-color:color-mix(in oklab,var(--color-neutral-8) 90%,transparent)}}.bg-neutral-9{background-color:var(--color-neutral-9)}.bg-neutral-9\/40{background-color:#263a3e66}@supports (color:color-mix(in lab,red,red)){.bg-neutral-9\/40{background-color:color-mix(in oklab,var(--color-neutral-9) 40%,transparent)}}.bg-neutral-9\/50{background-color:#263a3e80}@supports (color:color-mix(in lab,red,red)){.bg-neutral-9\/50{background-color:color-mix(in oklab,var(--color-neutral-9) 50%,transparent)}}.bg-neutral-9\/70{background-color:#263a3eb3}@supports (color:color-mix(in lab,red,red)){.bg-neutral-9\/70{background-color:color-mix(in oklab,var(--color-neutral-9) 70%,transparent)}}.bg-neutral-9\/80{background-color:#263a3ecc}@supports (color:color-mix(in lab,red,red)){.bg-neutral-9\/80{background-color:color-mix(in oklab,var(--color-neutral-9) 80%,transparent)}}.bg-neutral-9\/90{background-color:#263a3ee6}@supports (color:color-mix(in lab,red,red)){.bg-neutral-9\/90{background-color:color-mix(in oklab,var(--color-neutral-9) 90%,transparent)}}.bg-neutral-10{background-color:var(--color-neutral-10)}.bg-neutral-10\/20{background-color:#1a2c3033}@supports (color:color-mix(in lab,red,red)){.bg-neutral-10\/20{background-color:color-mix(in oklab,var(--color-neutral-10) 20%,transparent)}}.bg-neutral-10\/30{background-color:#1a2c304d}@supports (color:color-mix(in lab,red,red)){.bg-neutral-10\/30{background-color:color-mix(in oklab,var(--color-neutral-10) 30%,transparent)}}.bg-neutral-10\/50{background-color:#1a2c3080}@supports (color:color-mix(in lab,red,red)){.bg-neutral-10\/50{background-color:color-mix(in oklab,var(--color-neutral-10) 50%,transparent)}}.bg-neutral-10\/60{background-color:#1a2c3099}@supports (color:color-mix(in lab,red,red)){.bg-neutral-10\/60{background-color:color-mix(in oklab,var(--color-neutral-10) 60%,transparent)}}.bg-neutral-10\/90{background-color:#1a2c30e6}@supports (color:color-mix(in lab,red,red)){.bg-neutral-10\/90{background-color:color-mix(in oklab,var(--color-neutral-10) 90%,transparent)}}.bg-neutral-11{background-color:var(--color-neutral-11)}.bg-neutral-11\/50{background-color:#101e2180}@supports (color:color-mix(in lab,red,red)){.bg-neutral-11\/50{background-color:color-mix(in oklab,var(--color-neutral-11) 50%,transparent)}}.bg-neutral-11\/60{background-color:#101e2199}@supports (color:color-mix(in lab,red,red)){.bg-neutral-11\/60{background-color:color-mix(in oklab,var(--color-neutral-11) 60%,transparent)}}.bg-neutral-12{background-color:var(--color-neutral-12)}.bg-neutral-12\/80{background-color:#090e0fcc}@supports (color:color-mix(in lab,red,red)){.bg-neutral-12\/80{background-color:color-mix(in oklab,var(--color-neutral-12) 80%,transparent)}}.bg-primary-5{background-color:var(--color-primary-5)}.bg-primary-8{background-color:var(--color-primary-8)}.bg-primary-9{background-color:var(--color-primary-9)}.bg-primary-9\/30{background-color:#7d5b2a4d}@supports (color:color-mix(in lab,red,red)){.bg-primary-9\/30{background-color:color-mix(in oklab,var(--color-primary-9) 30%,transparent)}}.bg-primary-10\/30{background-color:#5c44204d}@supports (color:color-mix(in lab,red,red)){.bg-primary-10\/30{background-color:color-mix(in oklab,var(--color-primary-10) 30%,transparent)}}.bg-primary-10\/50{background-color:#5c442080}@supports (color:color-mix(in lab,red,red)){.bg-primary-10\/50{background-color:color-mix(in oklab,var(--color-primary-10) 50%,transparent)}}.bg-primary-11\/60{background-color:#3a2c1599}@supports (color:color-mix(in lab,red,red)){.bg-primary-11\/60{background-color:color-mix(in oklab,var(--color-primary-11) 60%,transparent)}}.bg-primary-11\/80{background-color:#3a2c15cc}@supports (color:color-mix(in lab,red,red)){.bg-primary-11\/80{background-color:color-mix(in oklab,var(--color-primary-11) 80%,transparent)}}.bg-success-3{background-color:var(--color-success-3)}.bg-success-5{background-color:var(--color-success-5)}.bg-success-6{background-color:var(--color-success-6)}.bg-success-7{background-color:var(--color-success-7)}.bg-success-8\/40{background-color:#1d6e4366}@supports (color:color-mix(in lab,red,red)){.bg-success-8\/40{background-color:color-mix(in oklab,var(--color-success-8) 40%,transparent)}}.bg-success-9\/30{background-color:#1756344d}@supports (color:color-mix(in lab,red,red)){.bg-success-9\/30{background-color:color-mix(in oklab,var(--color-success-9) 30%,transparent)}}.bg-success-10\/50{background-color:#11402780}@supports (color:color-mix(in lab,red,red)){.bg-success-10\/50{background-color:color-mix(in oklab,var(--color-success-10) 50%,transparent)}}.bg-transparent{background-color:#0000}.bg-warning-3{background-color:var(--color-warning-3)}.bg-warning-5{background-color:var(--color-warning-5)}.bg-warning-6{background-color:var(--color-warning-6)}.bg-warning-8{background-color:var(--color-warning-8)}.bg-warning-9\/30{background-color:#73560e4d}@supports (color:color-mix(in lab,red,red)){.bg-warning-9\/30{background-color:color-mix(in oklab,var(--color-warning-9) 30%,transparent)}}.bg-warning-9\/40{background-color:#73560e66}@supports (color:color-mix(in lab,red,red)){.bg-warning-9\/40{background-color:color-mix(in oklab,var(--color-warning-9) 40%,transparent)}}.bg-warning-10\/50{background-color:#56400a80}@supports (color:color-mix(in lab,red,red)){.bg-warning-10\/50{background-color:color-mix(in oklab,var(--color-warning-10) 50%,transparent)}}.bg-warning-11\/20{background-color:#382a0733}@supports (color:color-mix(in lab,red,red)){.bg-warning-11\/20{background-color:color-mix(in oklab,var(--color-warning-11) 20%,transparent)}}.bg-warning-11\/30{background-color:#382a074d}@supports (color:color-mix(in lab,red,red)){.bg-warning-11\/30{background-color:color-mix(in oklab,var(--color-warning-11) 30%,transparent)}}.bg-yellow-900\/30{background-color:#733e0a4d}@supports (color:color-mix(in lab,red,red)){.bg-yellow-900\/30{background-color:color-mix(in oklab,var(--color-yellow-900) 30%,transparent)}}.object-contain{object-fit:contain}.p-0\.5{padding:calc(var(--spacing) * .5)}.p-1{padding:calc(var(--spacing) * 1)}.p-1\.5{padding:calc(var(--spacing) * 1.5)}.p-2{padding:calc(var(--spacing) * 2)}.p-2\.5{padding:calc(var(--spacing) * 2.5)}.p-5{padding:calc(var(--spacing) * 5)}.p-6{padding:calc(var(--spacing) * 6)}.p-8{padding:calc(var(--spacing) * 8)}.px-0\.5{padding-inline:calc(var(--spacing) * .5)}.px-1{padding-inline:calc(var(--spacing) * 1)}.px-1\.5{padding-inline:calc(var(--spacing) * 1.5)}.px-2{padding-inline:calc(var(--spacing) * 2)}.px-2\.5{padding-inline:calc(var(--spacing) * 2.5)}.px-3{padding-inline:calc(var(--spacing) * 3)}.px-3\.5{padding-inline:calc(var(--spacing) * 3.5)}.px-4{padding-inline:calc(var(--spacing) * 4)}.px-5{padding-inline:calc(var(--spacing) * 5)}.px-6{padding-inline:calc(var(--spacing) * 6)}.py-0{padding-block:calc(var(--spacing) * 0)}.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)}.py-12{padding-block:calc(var(--spacing) * 12)}.pt-0\.5{padding-top:calc(var(--spacing) * .5)}.pt-1{padding-top:calc(var(--spacing) * 1)}.pt-1\.5{padding-top:calc(var(--spacing) * 1.5)}.pt-2{padding-top:calc(var(--spacing) * 2)}.pt-3{padding-top:calc(var(--spacing) * 3)}.pt-4{padding-top:calc(var(--spacing) * 4)}.pt-5{padding-top:calc(var(--spacing) * 5)}.pt-\[20vh\]{padding-top:20vh}.pr-2{padding-right:calc(var(--spacing) * 2)}.pb-0\.5{padding-bottom:calc(var(--spacing) * .5)}.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)}.pb-3{padding-bottom:calc(var(--spacing) * 3)}.pb-4{padding-bottom:calc(var(--spacing) * 4)}.pl-2{padding-left:calc(var(--spacing) * 2)}.pl-3{padding-left:calc(var(--spacing) * 3)}.pl-4{padding-left:calc(var(--spacing) * 4)}.pl-10{padding-left:calc(var(--spacing) * 10)}.pl-12{padding-left:calc(var(--spacing) * 12)}.text-center{text-align:center}.text-left{text-align:left}.font-mono{font-family:var(--font-mono)}.text-\[11px\]{font-size:11px}.text-\[12px\]{font-size:12px}.text-\[13px\]{font-size:13px}.text-\[14px\]{font-size:14px}.text-\[15px\]{font-size:15px}.text-\[16px\]{font-size:16px}.text-\[17px\]{font-size:17px}.text-\[18px\]{font-size:18px}.text-\[19px\]{font-size:19px}.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-normal{--tw-font-weight:var(--font-weight-normal);font-weight:var(--font-weight-normal)}.font-semibold{--tw-font-weight:var(--font-weight-semibold);font-weight:var(--font-weight-semibold)}.tracking-wide{--tw-tracking:var(--tracking-wide);letter-spacing:var(--tracking-wide)}.tracking-wider{--tw-tracking:var(--tracking-wider);letter-spacing:var(--tracking-wider)}.tracking-widest{--tw-tracking:var(--tracking-widest);letter-spacing:var(--tracking-widest)}.break-all{word-break:break-all}.whitespace-nowrap{white-space:nowrap}.whitespace-pre-wrap{white-space:pre-wrap}.text-accent-2{color:var(--color-accent-2)}.text-accent-3{color:var(--color-accent-3)}.text-accent-4{color:var(--color-accent-4)}.text-accent-6{color:var(--color-accent-6)}.text-error-2{color:var(--color-error-2)}.text-error-4{color:var(--color-error-4)}.text-error-5{color:var(--color-error-5)}.text-neutral-1{color:var(--color-neutral-1)}.text-neutral-2{color:var(--color-neutral-2)}.text-neutral-3{color:var(--color-neutral-3)}.text-neutral-4{color:var(--color-neutral-4)}.text-neutral-5{color:var(--color-neutral-5)}.text-neutral-6{color:var(--color-neutral-6)}.text-neutral-7{color:var(--color-neutral-7)}.text-primary-3{color:var(--color-primary-3)}.text-primary-4{color:var(--color-primary-4)}.text-primary-5{color:var(--color-primary-5)}.text-primary-6{color:var(--color-primary-6)}.text-primary-7{color:var(--color-primary-7)}.text-success-2{color:var(--color-success-2)}.text-success-4{color:var(--color-success-4)}.text-success-5{color:var(--color-success-5)}.text-success-6{color:var(--color-success-6)}.text-transparent{color:#0000}.text-warning-2{color:var(--color-warning-2)}.text-warning-3{color:var(--color-warning-3)}.text-warning-4{color:var(--color-warning-4)}.text-warning-5{color:var(--color-warning-5)}.text-white{color:var(--color-white)}.text-yellow-500{color:var(--color-yellow-500)}.capitalize{text-transform:capitalize}.lowercase{text-transform:lowercase}.uppercase{text-transform:uppercase}.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,)}.line-through{text-decoration-line:line-through}.placeholder-neutral-5::placeholder{color:var(--color-neutral-5)}.opacity-0{opacity:0}.opacity-40{opacity:.4}.opacity-50{opacity:.5}.opacity-60{opacity:.6}.opacity-100\!{opacity:1!important}.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-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)}.ring,.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-accent-6\/30{--tw-ring-color:#2d9bb04d}@supports (color:color-mix(in lab,red,red)){.ring-accent-6\/30{--tw-ring-color:color-mix(in oklab, var(--color-accent-6) 30%, transparent)}}.blur{--tw-blur:blur(8px);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,)}.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-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))}.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))}.transition-transform{transition-property:transform,translate,scale,rotate;transition-timing-function:var(--tw-ease,var(--default-transition-timing-function));transition-duration:var(--tw-duration,var(--default-transition-duration))}.duration-200{--tw-duration:.2s;transition-duration:.2s}.ease-out{--tw-ease:var(--ease-out);transition-timing-function:var(--ease-out)}.outline-none{--tw-outline-style:none;outline-style:none}.select-all{-webkit-user-select:all;user-select:all}.select-none{-webkit-user-select:none;user-select:none}@media(hover:hover){.group-hover\:text-accent-5:is(:where(.group):hover *){color:var(--color-accent-5)}.group-hover\:text-neutral-1:is(:where(.group):hover *){color:var(--color-neutral-1)}.group-hover\:text-neutral-3:is(:where(.group):hover *){color:var(--color-neutral-3)}.group-hover\:text-neutral-4:is(:where(.group):hover *){color:var(--color-neutral-4)}.group-hover\:text-neutral-5:is(:where(.group):hover *){color:var(--color-neutral-5)}.group-hover\:text-neutral-6:is(:where(.group):hover *){color:var(--color-neutral-6)}.group-hover\:text-primary-5:is(:where(.group):hover *){color:var(--color-primary-5)}.group-hover\:opacity-100:is(:where(.group):hover *),.group-hover\/codeblock\:opacity-100:is(:where(.group\/codeblock):hover *),.group-hover\/header\:opacity-100:is(:where(.group\/header):hover *),.group-hover\/repo\:opacity-100:is(:where(.group\/repo):hover *){opacity:1}}.placeholder\:text-neutral-5::placeholder{color:var(--color-neutral-5)}.last\:border-b-0:last-child{border-bottom-style:var(--tw-border-style);border-bottom-width:0}@media(hover:hover){.hover\:border-neutral-5:hover{border-color:var(--color-neutral-5)}.hover\:border-neutral-6:hover{border-color:var(--color-neutral-6)}.hover\:border-primary-7:hover{border-color:var(--color-primary-7)}.hover\:bg-error-9\/50:hover{background-color:#82232380}@supports (color:color-mix(in lab,red,red)){.hover\:bg-error-9\/50:hover{background-color:color-mix(in oklab,var(--color-error-9) 50%,transparent)}}.hover\:bg-neutral-6:hover{background-color:var(--color-neutral-6)}.hover\:bg-neutral-6\/25:hover{background-color:#5c737740}@supports (color:color-mix(in lab,red,red)){.hover\:bg-neutral-6\/25:hover{background-color:color-mix(in oklab,var(--color-neutral-6) 25%,transparent)}}.hover\:bg-neutral-6\/50:hover{background-color:#5c737780}@supports (color:color-mix(in lab,red,red)){.hover\:bg-neutral-6\/50:hover{background-color:color-mix(in oklab,var(--color-neutral-6) 50%,transparent)}}.hover\:bg-neutral-7:hover{background-color:var(--color-neutral-7)}.hover\:bg-neutral-7\/20:hover{background-color:#465c6033}@supports (color:color-mix(in lab,red,red)){.hover\:bg-neutral-7\/20:hover{background-color:color-mix(in oklab,var(--color-neutral-7) 20%,transparent)}}.hover\:bg-neutral-8:hover{background-color:var(--color-neutral-8)}.hover\:bg-neutral-8\/50:hover{background-color:#354a4e80}@supports (color:color-mix(in lab,red,red)){.hover\:bg-neutral-8\/50:hover{background-color:color-mix(in oklab,var(--color-neutral-8) 50%,transparent)}}.hover\:bg-neutral-8\/60:hover{background-color:#354a4e99}@supports (color:color-mix(in lab,red,red)){.hover\:bg-neutral-8\/60:hover{background-color:color-mix(in oklab,var(--color-neutral-8) 60%,transparent)}}.hover\:bg-neutral-8\/90:hover{background-color:#354a4ee6}@supports (color:color-mix(in lab,red,red)){.hover\:bg-neutral-8\/90:hover{background-color:color-mix(in oklab,var(--color-neutral-8) 90%,transparent)}}.hover\:bg-neutral-9:hover{background-color:var(--color-neutral-9)}.hover\:bg-neutral-10\/30:hover{background-color:#1a2c304d}@supports (color:color-mix(in lab,red,red)){.hover\:bg-neutral-10\/30:hover{background-color:color-mix(in oklab,var(--color-neutral-10) 30%,transparent)}}.hover\:bg-neutral-10\/50:hover{background-color:#1a2c3080}@supports (color:color-mix(in lab,red,red)){.hover\:bg-neutral-10\/50:hover{background-color:color-mix(in oklab,var(--color-neutral-10) 50%,transparent)}}.hover\:bg-primary-7:hover{background-color:var(--color-primary-7)}.hover\:bg-primary-7\/40:hover{background-color:#9d744066}@supports (color:color-mix(in lab,red,red)){.hover\:bg-primary-7\/40:hover{background-color:color-mix(in oklab,var(--color-primary-7) 40%,transparent)}}.hover\:bg-primary-7\/50:hover{background-color:#9d744080}@supports (color:color-mix(in lab,red,red)){.hover\:bg-primary-7\/50:hover{background-color:color-mix(in oklab,var(--color-primary-7) 50%,transparent)}}.hover\:bg-primary-8:hover{background-color:var(--color-primary-8)}.hover\:bg-primary-9\/40:hover{background-color:#7d5b2a66}@supports (color:color-mix(in lab,red,red)){.hover\:bg-primary-9\/40:hover{background-color:color-mix(in oklab,var(--color-primary-9) 40%,transparent)}}.hover\:bg-primary-9\/50:hover{background-color:#7d5b2a80}@supports (color:color-mix(in lab,red,red)){.hover\:bg-primary-9\/50:hover{background-color:color-mix(in oklab,var(--color-primary-9) 50%,transparent)}}.hover\:bg-success-9\/50:hover{background-color:#17563480}@supports (color:color-mix(in lab,red,red)){.hover\:bg-success-9\/50:hover{background-color:color-mix(in oklab,var(--color-success-9) 50%,transparent)}}.hover\:bg-warning-8\/50:hover{background-color:#906d1280}@supports (color:color-mix(in lab,red,red)){.hover\:bg-warning-8\/50:hover{background-color:color-mix(in oklab,var(--color-warning-8) 50%,transparent)}}.hover\:bg-warning-9\/50:hover{background-color:#73560e80}@supports (color:color-mix(in lab,red,red)){.hover\:bg-warning-9\/50:hover{background-color:color-mix(in oklab,var(--color-warning-9) 50%,transparent)}}.hover\:text-accent-3:hover{color:var(--color-accent-3)}.hover\:text-error-4:hover{color:var(--color-error-4)}.hover\:text-error-5:hover{color:var(--color-error-5)}.hover\:text-neutral-1:hover{color:var(--color-neutral-1)}.hover\:text-neutral-2:hover{color:var(--color-neutral-2)}.hover\:text-neutral-2\!:hover{color:var(--color-neutral-2)!important}.hover\:text-neutral-3:hover{color:var(--color-neutral-3)}.hover\:text-primary-3:hover{color:var(--color-primary-3)}.hover\:text-primary-5:hover{color:var(--color-primary-5)}.hover\:text-success-3:hover{color:var(--color-success-3)}.hover\:text-warning-4:hover{color:var(--color-warning-4)}}.focus\:border-accent-6:focus{border-color:var(--color-accent-6)}.focus\:border-primary-6:focus{border-color:var(--color-primary-6)}.focus\:border-primary-7:focus{border-color:var(--color-primary-7)}.focus\:outline-none:focus{--tw-outline-style:none;outline-style:none}.active\:bg-primary-6\/60:active{background-color:#cf9a2d99}@supports (color:color-mix(in lab,red,red)){.active\:bg-primary-6\/60:active{background-color:color-mix(in oklab,var(--color-primary-6) 60%,transparent)}}.active\:bg-primary-7\/60:active{background-color:#9d744099}@supports (color:color-mix(in lab,red,red)){.active\:bg-primary-7\/60:active{background-color:color-mix(in oklab,var(--color-primary-7) 60%,transparent)}}.disabled\:pointer-events-none:disabled{pointer-events:none}.disabled\:opacity-30:disabled{opacity:.3}.disabled\:opacity-40:disabled{opacity:.4}.disabled\:opacity-50:disabled{opacity:.5}.aria-selected\:bg-primary-8\/20[aria-selected=true]{background-color:#9e733533}@supports (color:color-mix(in lab,red,red)){.aria-selected\:bg-primary-8\/20[aria-selected=true]{background-color:color-mix(in oklab,var(--color-primary-8) 20%,transparent)}}.aria-selected\:text-primary-4[aria-selected=true]{color:var(--color-primary-4)}.\[\&_\[cmdk-group-heading\]\]\:px-2 [cmdk-group-heading]{padding-inline:calc(var(--spacing) * 2)}.\[\&_\[cmdk-group-heading\]\]\:pt-2 [cmdk-group-heading]{padding-top:calc(var(--spacing) * 2)}.\[\&_\[cmdk-group-heading\]\]\:pb-1 [cmdk-group-heading]{padding-bottom:calc(var(--spacing) * 1)}.\[\&_\[cmdk-group-heading\]\]\:text-\[12px\] [cmdk-group-heading]{font-size:12px}.\[\&_\[cmdk-group-heading\]\]\:font-medium [cmdk-group-heading]{--tw-font-weight:var(--font-weight-medium);font-weight:var(--font-weight-medium)}.\[\&_\[cmdk-group-heading\]\]\:tracking-wider [cmdk-group-heading]{--tw-tracking:var(--tracking-wider);letter-spacing:var(--tracking-wider)}.\[\&_\[cmdk-group-heading\]\]\:text-neutral-5 [cmdk-group-heading]{color:var(--color-neutral-5)}.\[\&_\[cmdk-group-heading\]\]\:uppercase [cmdk-group-heading]{text-transform:uppercase}}.terminal-area{--color-primary-1:#fef9f1;--color-primary-2:#faefd8;--color-primary-3:#f5e1b6;--color-primary-4:#eccc8e;--color-primary-5:#deb769;--color-primary-6:#cba557;--color-primary-7:#96754b;--color-primary-8:#977443;--color-primary-9:#775c35;--color-primary-10:#584528;--color-primary-11:#382c1a;--color-primary-12:#1d170d;--color-secondary-1:#fef6f3;--color-secondary-2:#fbe9e2;--color-secondary-3:#f2d3c3;--color-secondary-4:#e4b8a2;--color-secondary-5:#d09984;--color-secondary-6:#b5826b;--color-secondary-7:#93604b;--color-secondary-8:#774e3c;--color-secondary-9:#5d3c2e;--color-secondary-10:#452d22;--color-secondary-11:#2f1e17;--color-secondary-12:#1a110d;--color-accent-1:#f4f9f8;--color-accent-2:#ddf0ee;--color-accent-3:#bae1dc;--color-accent-4:#93ccc6;--color-accent-5:#6cb7af;--color-accent-6:#559f97;--color-accent-7:#42847d;--color-accent-8:#356b65;--color-accent-9:#29524d;--color-accent-10:#1e3c39;--color-accent-11:#152928;--color-accent-12:#0c1918;--color-neutral-1:#e2e1de;--color-neutral-2:#d1cfcb;--color-neutral-3:#cccac5;--color-neutral-4:#b0ada7;--color-neutral-5:#93918c;--color-neutral-6:#777571;--color-neutral-7:#555350;--color-neutral-8:#424040;--color-neutral-9:#333130;--color-neutral-10:#272524;--color-neutral-11:#1b1a18;--color-neutral-12:#0f0e0d;--color-error-1:#fdf5f5;--color-error-2:#f9e1e1;--color-error-3:#f1c0c0;--color-error-4:#e69b9b;--color-error-5:#d97777;--color-error-6:#ca5f5f;--color-error-7:#ad3e3e;--color-error-8:#913333;--color-error-9:#752929;--color-error-10:#591f1f;--color-error-11:#3a1515;--color-error-12:#1f0b0b;--color-warning-1:#fefbf1;--color-warning-2:#fbf2d6;--color-warning-3:#f7e5ae;--color-warning-4:#eed382;--color-warning-5:#e2bf5d;--color-warning-6:#ccaa4e;--color-warning-7:#a8862e;--color-warning-8:#896d24;--color-warning-9:#6d561c;--color-warning-10:#524015;--color-warning-11:#352a0e;--color-warning-12:#1c1608;--color-success-1:#f4faf5;--color-success-2:#daf2e5;--color-success-3:#b5e2ca;--color-success-4:#8ccfab;--color-success-5:#66bb8e;--color-success-6:#53a578;--color-success-7:#3f8559;--color-success-8:#326c48;--color-success-9:#275438;--color-success-10:#1d3f2a;--color-success-11:#142b1d;--color-success-12:#0b1811;--color-info-1:#f3f9f9;--color-info-2:#daecf0;--color-info-3:#b2d9e0;--color-info-4:#87c1cd;--color-info-5:#5ea8b8;--color-info-6:#478fa0;--color-info-7:#387486;--color-info-8:#2d5d6c;--color-info-9:#234854;--color-info-10:#1a353e;--color-info-11:#12242b;--color-info-12:#0a1519}[data-theme=light]{--color-neutral-1:#0f0e0d;--color-neutral-2:#1b1a18;--color-neutral-3:#272524;--color-neutral-4:#424040;--color-neutral-5:#555350;--color-neutral-6:#777571;--color-neutral-7:#c8c7c4;--color-neutral-8:#f0ede8;--color-neutral-9:#f5f3ee;--color-neutral-10:#f8f6f2;--color-neutral-11:#faf9f6;--color-neutral-12:#fdfcfa;--color-primary-1:#3a2c15;--color-primary-2:#5c4420;--color-primary-3:#7d5b2a;--color-primary-4:#9e7335;--color-primary-5:#b7884a;--color-primary-6:#cf9a2d;--color-primary-7:#c39538;--color-primary-8:#f0c56e;--color-primary-9:#f8dca0;--color-primary-10:#fcedcc;--color-primary-11:#fef3dc;--color-primary-12:#fef8ec;--color-secondary-1:#321d14;--color-secondary-2:#4a2b1e;--color-secondary-3:#633a29;--color-secondary-4:#7f4b35;--color-secondary-5:#9c5d42;--color-secondary-6:#b8704f;--color-secondary-7:#d4896a;--color-secondary-8:#e8ab8d;--color-secondary-9:#f5cbb5;--color-secondary-10:#fce6da;--color-secondary-11:#fef0e8;--color-secondary-12:#fef5f0;--color-accent-1:#07171a;--color-accent-2:#113c44;--color-accent-3:#17515c;--color-accent-4:#1d6876;--color-accent-5:#248192;--color-accent-6:#2d9bb0;--color-accent-7:#45b5c9;--color-accent-8:#74cbda;--color-accent-9:#a5e0ea;--color-accent-10:#d0f0f5;--color-accent-11:#e5f7fa;--color-accent-12:#eef9fb;--color-error-1:#230a0a;--color-error-2:#631b1b;--color-error-3:#822323;--color-error-4:#a12c2c;--color-error-5:#c03636;--color-error-6:#d94444;--color-error-7:#e86060;--color-error-8:#f28a8a;--color-error-9:#f9b5b5;--color-error-10:#fddcdc;--color-error-11:#feeded;--color-error-12:#fef2f2;--color-warning-1:#1e1604;--color-warning-2:#56400a;--color-warning-3:#73560e;--color-warning-4:#906d12;--color-warning-5:#b08618;--color-warning-6:#d0a020;--color-warning-7:#e8b830;--color-warning-8:#f4ce5e;--color-warning-9:#fae295;--color-warning-10:#fdf1c8;--color-warning-11:#fef8e5;--color-warning-12:#fefaec;--color-success-1:#07180e;--color-success-2:#114027;--color-success-3:#175634;--color-success-4:#1d6e43;--color-success-5:#248854;--color-success-6:#2da368;--color-success-7:#40bc80;--color-success-8:#6dd1a2;--color-success-9:#a0e4c4;--color-success-10:#d0f2e2;--color-success-11:#e5f8ef;--color-success-12:#effaf5;--color-info-1:#07151a;--color-info-2:#113644;--color-info-3:#17495c;--color-info-4:#1d5e76;--color-info-5:#247592;--color-info-6:#2d8fb0;--color-info-7:#40a9c8;--color-info-8:#6dc2da;--color-info-9:#a0d9ea;--color-info-10:#d0edf5;--color-info-11:#e5f4fa;--color-info-12:#eff8fb;font-weight:465}[data-theme=light] .app-logo-circle{background-color:#ffffff8c}[data-theme=dark] .app-logo-circle{background-color:var(--color-primary-2)}[data-theme=light] .app-right-sidebar{background-color:var(--color-neutral-7)}[data-theme=light] .app-left-sidebar{background-color:var(--color-neutral-9)}[data-theme=light] .app-left-sidebar button:hover{background-color:#748d912e}@supports (color:color-mix(in lab,red,red)){[data-theme=light] .app-left-sidebar button:hover{background-color:color-mix(in srgb,var(--color-neutral-5) 18%,transparent)}}[data-theme=light] .app-new-session-btn:hover{background-color:#748d912e}@supports (color:color-mix(in lab,red,red)){[data-theme=light] .app-new-session-btn:hover{background-color:color-mix(in srgb,var(--color-neutral-5) 18%,transparent)}}[data-theme=light] .app-session-tab:hover{background-color:#748d912e}@supports (color:color-mix(in lab,red,red)){[data-theme=light] .app-session-tab:hover{background-color:color-mix(in srgb,var(--color-neutral-5) 18%,transparent)}}[data-theme=light] .app-input-bar{background-color:var(--color-neutral-10)}[data-theme=light] .app-thinking-badge{background-color:#465c60e6}@supports (color:color-mix(in lab,red,red)){[data-theme=light] .app-thinking-badge{background-color:color-mix(in srgb,var(--color-neutral-7) 90%,transparent)}}[data-theme=light] .settings-section-card{background-color:var(--color-neutral-12);border-color:var(--color-neutral-7)}[data-theme=light] .settings-section-card>div:first-child{border-color:var(--color-neutral-7)}[data-theme=light] .workflow-card{background-color:var(--color-neutral-12);border-color:var(--color-neutral-7)}[data-theme=light] .workflow-card>div:first-child,[data-theme=light] .workflow-card .divide-neutral-9\/30>:not(:first-child){border-color:var(--color-neutral-7)}[data-theme=light] .terminal-area .user-bubble{background-color:#354a4ecc}@supports (color:color-mix(in lab,red,red)){[data-theme=light] .terminal-area .user-bubble{background-color:color-mix(in srgb,var(--color-neutral-8) 80%,transparent)}}[data-theme=light] .terminal-area{--color-neutral-1:#0f0e0d;--color-neutral-2:#1b1a18;--color-neutral-3:#333131;--color-neutral-4:#424040;--color-neutral-5:#777571;--color-neutral-6:#93918c;--color-neutral-7:#e4e2de;--color-neutral-8:#eeebe6;--color-neutral-9:#f3f1ec;--color-neutral-10:#f7f5f0;--color-neutral-11:#faf9f6;--color-neutral-12:#fdf9f4;--color-primary-1:#382c1a;--color-primary-2:#584528;--color-primary-3:#775c35;--color-primary-4:#977443;--color-primary-5:#af8958;--color-primary-6:#cba557;--color-primary-7:#be9d5a;--color-primary-8:#eccc8e;--color-primary-9:#f5e1b6;--color-primary-10:#faefd8;--color-primary-11:#fdf5e8;--color-primary-12:#fef9f1;--color-accent-1:#0c1918;--color-accent-2:#1e3c39;--color-accent-3:#29524d;--color-accent-4:#356b65;--color-accent-5:#42847d;--color-accent-6:#559f97;--color-accent-7:#6cb7af;--color-accent-8:#93ccc6;--color-accent-9:#bae1dc;--color-accent-10:#ddf0ee;--color-accent-11:#edf8f7;--color-accent-12:#f4f9f8;--color-error-1:#1f0b0b;--color-error-2:#591f1f;--color-error-3:#752929;--color-error-4:#913333;--color-error-5:#ad3e3e;--color-error-6:#ca5f5f;--color-error-7:#d97777;--color-error-8:#e69b9b;--color-error-9:#f1c0c0;--color-error-10:#f9e1e1;--color-error-11:#fceeed;--color-error-12:#fdf5f5;--color-warning-1:#1c1608;--color-warning-2:#524015;--color-warning-3:#6d561c;--color-warning-4:#896d24;--color-warning-5:#a8862e;--color-warning-6:#ccaa4e;--color-warning-7:#e2bf5d;--color-warning-8:#eed382;--color-warning-9:#f7e5ae;--color-warning-10:#fbf2d6;--color-warning-11:#fdf8e8;--color-warning-12:#fefbf1;--color-success-1:#0b1811;--color-success-2:#1d3f2a;--color-success-3:#275438;--color-success-4:#326c48;--color-success-5:#3f8559;--color-success-6:#53a578;--color-success-7:#66bb8e;--color-success-8:#8ccfab;--color-success-9:#b5e2ca;--color-success-10:#daf2e5;--color-success-11:#ecf9f2;--color-success-12:#f4faf5}.app-left-sidebar,.app-right-sidebar{--color-neutral-1:#e0e0e0;--color-neutral-2:#ccc;--color-neutral-3:silver;--color-neutral-4:#a3a3a3;--color-neutral-5:#858585;--color-neutral-6:#6b6b6b}[data-theme=light] .app-left-sidebar,[data-theme=light] .app-right-sidebar{--color-neutral-1:#1a1a1a;--color-neutral-2:#2e2e2e;--color-neutral-3:#3d3d3d;--color-neutral-4:#555;--color-neutral-5:#707070;--color-neutral-6:#8a8a8a}html,body,#root{height:100%;margin:0;font-family:Lato,sans-serif;overflow:hidden}.app-input-bar{padding-bottom:env(safe-area-inset-bottom,0px)}@supports (padding-top:env(safe-area-inset-top)){.mobile-top-bar-safe{padding-top:env(safe-area-inset-top,0px)}}.chat-scroll::-webkit-scrollbar{width:6px}.chat-scroll::-webkit-scrollbar-track{background:0 0}.chat-scroll::-webkit-scrollbar-thumb{background:#304346}@supports (color:color-mix(in lab,red,red)){.chat-scroll::-webkit-scrollbar-thumb{background:color-mix(in srgb,var(--color-neutral-8),black 10%)}}.chat-scroll::-webkit-scrollbar-thumb{border-radius:3px}.chat-scroll::-webkit-scrollbar-thumb:hover{background:var(--color-neutral-6)}.approvals-scroll::-webkit-scrollbar{width:5px}.approvals-scroll::-webkit-scrollbar-track{background:0 0}.approvals-scroll::-webkit-scrollbar-thumb{background:#304346}@supports (color:color-mix(in lab,red,red)){.approvals-scroll::-webkit-scrollbar-thumb{background:color-mix(in srgb,var(--color-neutral-8),black 10%)}}.approvals-scroll::-webkit-scrollbar-thumb{border-radius:3px}.approvals-scroll::-webkit-scrollbar-thumb:hover{background:var(--color-neutral-6)}.prose.prose-themed code{font-family:Inconsolata,monospace}.prose.prose-themed pre{background:var(--color-neutral-11);border:1px solid var(--color-neutral-9)}.prose.prose-themed a{text-underline-offset:2px;text-decoration:underline}.prose.prose-themed p{margin-top:.5em;margin-bottom:.5em}.prose.prose-themed h1,.prose.prose-themed h2,.prose.prose-themed h3,.prose.prose-themed h4{margin-top:1em;margin-bottom:.5em}.prose.prose-themed ul,.prose.prose-themed ol{margin-top:.25em;margin-bottom:.25em}.prose.prose-themed table{border-collapse:collapse}.prose.prose-themed th,.prose.prose-themed td{border:1px solid var(--color-neutral-9);padding:.25em .75em}[data-theme=dark] .prose.prose-themed{--tw-prose-body:var(--color-neutral-2);--tw-prose-headings:var(--color-neutral-2);--tw-prose-links:var(--color-accent-5);--tw-prose-bold:var(--color-neutral-2);--tw-prose-code:var(--color-accent-4);--tw-prose-pre-bg:var(--color-neutral-11);--tw-prose-pre-code:var(--color-neutral-2);--tw-prose-quotes:var(--color-neutral-3);--tw-prose-quote-borders:var(--color-neutral-8);--tw-prose-hr:var(--color-neutral-9);--tw-prose-th-borders:var(--color-neutral-8);--tw-prose-td-borders:var(--color-neutral-9);color:var(--color-neutral-2)}[data-theme=light] .prose.prose-themed{--tw-prose-body:var(--color-neutral-2);--tw-prose-headings:var(--color-neutral-1);--tw-prose-links:var(--color-accent-5);--tw-prose-bold:var(--color-neutral-1);--tw-prose-code:var(--color-accent-4);--tw-prose-pre-bg:var(--color-neutral-9);--tw-prose-pre-code:var(--color-neutral-2);--tw-prose-quotes:var(--color-neutral-3);--tw-prose-quote-borders:var(--color-neutral-7);--tw-prose-hr:var(--color-neutral-8);--tw-prose-th-borders:var(--color-neutral-7);--tw-prose-td-borders:var(--color-neutral-8);color:var(--color-neutral-2)}.prose.prose-invert{--tw-prose-body:var(--color-neutral-2);--tw-prose-headings:var(--color-neutral-2);--tw-prose-links:var(--color-accent-5);--tw-prose-bold:var(--color-neutral-2);--tw-prose-code:var(--color-accent-4);--tw-prose-pre-bg:var(--color-neutral-11);--tw-prose-pre-code:var(--color-neutral-2);--tw-prose-quotes:var(--color-neutral-3);--tw-prose-quote-borders:var(--color-neutral-8);--tw-prose-hr:var(--color-neutral-9);--tw-prose-th-borders:var(--color-neutral-8);--tw-prose-td-borders:var(--color-neutral-9)}.prose.prose-invert code{font-family:Inconsolata,monospace}.prose.prose-invert pre{background:var(--color-neutral-11);border:1px solid var(--color-neutral-9)}.prose.prose-invert a{text-underline-offset:2px;text-decoration:underline}.prose.prose-invert p{margin-top:.5em;margin-bottom:.5em}.prose.prose-invert h1,.prose.prose-invert h2,.prose.prose-invert h3,.prose.prose-invert h4{margin-top:1em;margin-bottom:.5em}.prose.prose-invert ul,.prose.prose-invert ol{margin-top:.25em;margin-bottom:.25em}.prose.prose-invert table{border-collapse:collapse}.prose.prose-invert th,.prose.prose-invert td{border:1px solid var(--color-neutral-9);padding:.25em .75em}.docs-header{background:var(--color-neutral-2);border-color:var(--color-neutral-3)}.docs-header-back{color:var(--color-neutral-8)}.docs-header-back:hover{color:var(--color-neutral-11)}.docs-header-path{color:var(--color-neutral-6)}.docs-header-slash{color:var(--color-neutral-5)}.docs-header-toggle{color:var(--color-neutral-7)}.docs-header-toggle:hover{color:var(--color-neutral-10);background:var(--color-neutral-4)}.docs-header-star{color:var(--color-neutral-6)}.docs-header-star:hover,.docs-header-star.is-starred{color:var(--color-primary-5)}.docs-header-toggle.is-active{background:var(--color-neutral-5);color:var(--color-neutral-11)}.docs-prose{color:var(--color-neutral-2);font-family:Lato,sans-serif;font-size:15px;line-height:1.65}.docs-prose h1{color:var(--color-neutral-2);margin:1.5em 0 .5em;font-size:1.5em;font-weight:700}.docs-prose h2{color:var(--color-neutral-2);margin:1.25em 0 .5em;font-size:1.25em;font-weight:600}.docs-prose h3{color:var(--color-neutral-2);margin:1em 0 .5em;font-size:1.1em;font-weight:600}.docs-prose h4{color:var(--color-neutral-2);margin:1em 0 .5em;font-size:1em;font-weight:600}.docs-prose h1:first-child,.docs-prose h2:first-child,.docs-prose h3:first-child{margin-top:0}.docs-prose p{margin:.5em 0}.docs-prose pre{background:var(--color-neutral-11);border:1px solid var(--color-neutral-9);border-radius:6px;margin:.5em 0;padding:.75em 1em;font-size:14px;overflow-x:auto}.docs-prose code{font-family:Inconsolata,monospace}.docs-prose :not(pre)>code{background:var(--color-neutral-10);color:var(--color-accent-4);border-radius:3px;padding:.1em .4em}.docs-prose table{border-collapse:collapse;width:100%;margin:.5em 0;font-size:14px}.docs-prose th,.docs-prose td{border:1px solid var(--color-neutral-9);text-align:left;padding:.25em .75em}.docs-prose th{background:var(--color-neutral-11);color:var(--color-neutral-2);font-weight:600}.docs-prose a{color:var(--color-accent-5);text-underline-offset:2px;text-decoration:underline}.docs-prose a:hover{color:var(--color-accent-4)}.docs-prose blockquote{border-left:3px solid var(--color-neutral-8);color:var(--color-neutral-4);margin:.5em 0;padding-left:1em}.docs-prose ul{margin:.5em 0;padding-left:1.5em;list-style:outside}.docs-prose ol{margin:.5em 0;padding-left:1.5em;list-style:decimal}.docs-prose li{margin:.25em 0}.docs-prose hr{border:none;border-top:1px solid var(--color-neutral-9);margin:1.5em 0}.docs-prose img{max-width:100%;height:auto}@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-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}@property --tw-ease{syntax:"*";inherits:false}@keyframes spin{to{transform:rotate(360deg)}}@keyframes pulse{50%{opacity:.5}}
package/dist/index.html CHANGED
@@ -2,11 +2,11 @@
2
2
  <html lang="en" class="dark">
3
3
  <head>
4
4
  <meta charset="UTF-8" />
5
- <meta name="viewport" content="width=device-width, initial-scale=1.0" />
5
+ <meta name="viewport" content="width=device-width, initial-scale=1.0, viewport-fit=cover" />
6
6
  <link rel="icon" type="image/svg+xml" href="/favicon.svg" />
7
7
  <title>Codekin</title>
8
- <script type="module" crossorigin src="/assets/index-D6pRORYQ.js"></script>
9
- <link rel="stylesheet" crossorigin href="/assets/index-CQdQ4FhF.css">
8
+ <script type="module" crossorigin src="/assets/index-CYGuwwHU.js"></script>
9
+ <link rel="stylesheet" crossorigin href="/assets/index-DW70VouA.css">
10
10
  </head>
11
11
  <body class="bg-neutral-12 text-neutral-2">
12
12
  <div id="root"></div>
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "codekin",
3
- "version": "0.2.2",
3
+ "version": "0.3.1",
4
4
  "license": "MIT",
5
5
  "author": "multiplier-labs",
6
6
  "description": "Web UI for Claude Code sessions with multi-session support and WebSocket streaming",
@@ -36,28 +36,31 @@
36
36
  "test:watch": "vitest"
37
37
  },
38
38
  "dependencies": {
39
- "@ai-sdk/groq": "^3.0.29",
40
39
  "@tabler/icons-react": "^3.37.1",
41
40
  "@tailwindcss/vite": "^4.2.0",
42
- "ai": "^6.0.116",
41
+ "@types/dompurify": "^3.0.5",
43
42
  "cmdk": "^1.1.1",
43
+ "dompurify": "^3.3.2",
44
+ "highlight.js": "^11.11.1",
45
+ "marked": "^17.0.4",
46
+ "marked-highlight": "^2.2.3",
44
47
  "react": "^19.2.0",
45
48
  "react-dom": "^19.2.0",
46
49
  "react-markdown": "^10.1.0",
47
50
  "react-syntax-highlighter": "^16.1.0",
48
51
  "remark-gfm": "^4.0.1",
49
- "better-sqlite3": "^12.6.2",
50
52
  "tailwindcss": "^4.2.0"
51
53
  },
52
54
  "devDependencies": {
53
55
  "@eslint/js": "^9.39.1",
56
+ "@types/better-sqlite3": "^7.6.13",
54
57
  "@types/node": "^24.10.1",
55
58
  "@types/react": "^19.2.7",
56
59
  "@types/react-dom": "^19.2.3",
57
60
  "@types/react-syntax-highlighter": "^15.5.13",
58
- "@types/better-sqlite3": "^7.6.13",
59
61
  "@vitejs/plugin-react": "^5.1.1",
60
62
  "@vitest/coverage-v8": "^4.0.18",
63
+ "better-sqlite3": "^12.6.2",
61
64
  "eslint": "^9.39.1",
62
65
  "eslint-plugin-react-hooks": "^7.0.1",
63
66
  "eslint-plugin-react-refresh": "^0.4.24",
@@ -0,0 +1,71 @@
1
+ /**
2
+ * Approval manager for Codekin.
3
+ *
4
+ * Manages repo-level auto-approval rules for tools and Bash commands.
5
+ * Approvals are stored per-repo (keyed by workingDir) in
6
+ * ~/.codekin/repo-approvals.json, so they persist across sessions
7
+ * sharing the same repo.
8
+ */
9
+ export declare class ApprovalManager {
10
+ /** Repo-level auto-approval store, keyed by workingDir (repo path). */
11
+ private repoApprovals;
12
+ private _approvalPersistTimer;
13
+ constructor();
14
+ /** Get or create the approval entry for a repo (workingDir). */
15
+ private getRepoApprovalEntry;
16
+ /** Add an auto-approval rule for a repo and persist. */
17
+ addRepoApproval(workingDir: string, opts: {
18
+ tool?: string;
19
+ command?: string;
20
+ pattern?: string;
21
+ }): void;
22
+ /**
23
+ * Command prefixes where prefix-based auto-approval is safe.
24
+ * Only commands whose behavior is determined by later arguments (not by target)
25
+ * should be listed here. Dangerous commands like rm, sudo, curl, etc. require
26
+ * exact match to prevent escalation (e.g. approving `rm -rf /tmp/x` should NOT
27
+ * also approve `rm -rf /`).
28
+ */
29
+ private static readonly SAFE_PREFIX_COMMANDS;
30
+ /**
31
+ * Check if a tool/command is auto-approved for a repo.
32
+ * For Bash commands, uses prefix matching only for safe commands;
33
+ * dangerous commands require exact match to prevent escalation.
34
+ */
35
+ checkAutoApproval(workingDir: string, toolName: string, toolInput: Record<string, unknown>): boolean;
36
+ /** Extract the command prefix (first two tokens) for prefix-based matching. */
37
+ private commandPrefix;
38
+ /**
39
+ * Derive a glob pattern from a tool invocation for "Approve Pattern".
40
+ * Returns a string like "cat *" or "git diff *", or null if no safe pattern applies.
41
+ * Patterns use the format "<prefix> *" meaning "this prefix followed by anything".
42
+ */
43
+ derivePattern(toolName: string, toolInput: Record<string, unknown>): string | null;
44
+ /**
45
+ * Check if a bash command matches a stored pattern.
46
+ * Patterns of the form "<prefix> *" match any command starting with <prefix>.
47
+ */
48
+ private matchesPattern;
49
+ /** Save an "Always Allow" approval for a tool/command. */
50
+ saveAlwaysAllow(workingDir: string, toolName: string, toolInput: Record<string, unknown>): void;
51
+ /** Save a pattern-based approval (e.g. "cat *") for a tool/command. */
52
+ savePatternApproval(workingDir: string, toolName: string, toolInput: Record<string, unknown>): void;
53
+ /** Return the auto-approved tools, commands, and patterns for a repo (workingDir). */
54
+ getApprovals(workingDir: string): {
55
+ tools: string[];
56
+ commands: string[];
57
+ patterns: string[];
58
+ };
59
+ /** Remove an auto-approval rule for a repo (workingDir) and persist to disk.
60
+ * Returns 'invalid' if no tool/command provided, or boolean indicating if something was deleted.
61
+ * Pass skipPersist=true for bulk operations (caller must call persistRepoApprovals after). */
62
+ removeApproval(workingDir: string, opts: {
63
+ tool?: string;
64
+ command?: string;
65
+ pattern?: string;
66
+ }, skipPersist?: boolean): 'invalid' | boolean;
67
+ /** Write repo-level approvals to disk (atomic rename). */
68
+ persistRepoApprovals(): void;
69
+ private persistRepoApprovalsDebounced;
70
+ private restoreRepoApprovalsFromDisk;
71
+ }
@@ -0,0 +1,249 @@
1
+ /**
2
+ * Approval manager for Codekin.
3
+ *
4
+ * Manages repo-level auto-approval rules for tools and Bash commands.
5
+ * Approvals are stored per-repo (keyed by workingDir) in
6
+ * ~/.codekin/repo-approvals.json, so they persist across sessions
7
+ * sharing the same repo.
8
+ */
9
+ import { existsSync, mkdirSync, readFileSync, renameSync, writeFileSync } from 'fs';
10
+ import { join } from 'path';
11
+ import { DATA_DIR } from './config.js';
12
+ const REPO_APPROVALS_FILE = join(DATA_DIR, 'repo-approvals.json');
13
+ const PERSIST_DEBOUNCE_MS = 2000;
14
+ export class ApprovalManager {
15
+ /** Repo-level auto-approval store, keyed by workingDir (repo path). */
16
+ repoApprovals = new Map();
17
+ _approvalPersistTimer = null;
18
+ constructor() {
19
+ this.restoreRepoApprovalsFromDisk();
20
+ }
21
+ /** Get or create the approval entry for a repo (workingDir). */
22
+ getRepoApprovalEntry(workingDir) {
23
+ let entry = this.repoApprovals.get(workingDir);
24
+ if (!entry) {
25
+ entry = { tools: new Set(), commands: new Set(), patterns: new Set() };
26
+ this.repoApprovals.set(workingDir, entry);
27
+ }
28
+ return entry;
29
+ }
30
+ /** Add an auto-approval rule for a repo and persist. */
31
+ addRepoApproval(workingDir, opts) {
32
+ const entry = this.getRepoApprovalEntry(workingDir);
33
+ if (opts.tool)
34
+ entry.tools.add(opts.tool);
35
+ if (opts.command)
36
+ entry.commands.add(opts.command);
37
+ if (opts.pattern)
38
+ entry.patterns.add(opts.pattern);
39
+ this.persistRepoApprovalsDebounced();
40
+ }
41
+ /**
42
+ * Command prefixes where prefix-based auto-approval is safe.
43
+ * Only commands whose behavior is determined by later arguments (not by target)
44
+ * should be listed here. Dangerous commands like rm, sudo, curl, etc. require
45
+ * exact match to prevent escalation (e.g. approving `rm -rf /tmp/x` should NOT
46
+ * also approve `rm -rf /`).
47
+ */
48
+ static SAFE_PREFIX_COMMANDS = new Set([
49
+ 'git add', 'git commit', 'git diff', 'git log', 'git show', 'git stash',
50
+ 'git status', 'git branch', 'git checkout', 'git switch', 'git rebase',
51
+ 'git fetch', 'git pull', 'git merge', 'git tag', 'git rev-parse',
52
+ 'npm run', 'npm test', 'npm install', 'npm ci', 'npm exec',
53
+ 'npx', 'node', 'bun', 'deno',
54
+ 'cargo build', 'cargo test', 'cargo run', 'cargo check', 'cargo clippy',
55
+ 'make', 'cmake',
56
+ 'python', 'python3', 'pip install',
57
+ 'go build', 'go test', 'go run', 'go vet',
58
+ 'tsc', 'eslint', 'prettier',
59
+ 'cat', 'head', 'tail', 'wc', 'sort', 'uniq', 'diff', 'less',
60
+ 'ls', 'pwd', 'echo', 'date', 'which', 'whoami', 'env', 'printenv',
61
+ 'find', 'grep', 'rg', 'ag', 'fd',
62
+ 'mkdir', 'touch',
63
+ ]);
64
+ /**
65
+ * Check if a tool/command is auto-approved for a repo.
66
+ * For Bash commands, uses prefix matching only for safe commands;
67
+ * dangerous commands require exact match to prevent escalation.
68
+ */
69
+ checkAutoApproval(workingDir, toolName, toolInput) {
70
+ const approvals = this.getRepoApprovalEntry(workingDir);
71
+ if (approvals.tools.has(toolName))
72
+ return true;
73
+ if (toolName === 'Bash') {
74
+ const cmd = (typeof toolInput.command === 'string' ? toolInput.command : '').trim();
75
+ // Exact match always works
76
+ if (approvals.commands.has(cmd))
77
+ return true;
78
+ // Pattern match (e.g. "cat *" matches any cat command)
79
+ for (const pattern of approvals.patterns) {
80
+ if (this.matchesPattern(pattern, cmd))
81
+ return true;
82
+ }
83
+ // Prefix match only for safe commands
84
+ const cmdPrefix = this.commandPrefix(cmd);
85
+ if (cmdPrefix && ApprovalManager.SAFE_PREFIX_COMMANDS.has(cmdPrefix)) {
86
+ for (const approved of approvals.commands) {
87
+ if (this.commandPrefix(approved) === cmdPrefix)
88
+ return true;
89
+ }
90
+ }
91
+ }
92
+ return false;
93
+ }
94
+ /** Extract the command prefix (first two tokens) for prefix-based matching. */
95
+ commandPrefix(cmd) {
96
+ const tokens = cmd.split(/\s+/).filter(Boolean);
97
+ if (tokens.length === 0)
98
+ return '';
99
+ if (tokens.length === 1)
100
+ return tokens[0];
101
+ return `${tokens[0]} ${tokens[1]}`;
102
+ }
103
+ /**
104
+ * Derive a glob pattern from a tool invocation for "Approve Pattern".
105
+ * Returns a string like "cat *" or "git diff *", or null if no safe pattern applies.
106
+ * Patterns use the format "<prefix> *" meaning "this prefix followed by anything".
107
+ */
108
+ derivePattern(toolName, toolInput) {
109
+ if (toolName !== 'Bash')
110
+ return null;
111
+ const cmd = (typeof toolInput.command === 'string' ? toolInput.command : '').trim();
112
+ const tokens = cmd.split(/\s+/).filter(Boolean);
113
+ if (tokens.length === 0)
114
+ return null;
115
+ // Check single-token safe commands (cat, grep, ls, etc.)
116
+ const first = tokens[0];
117
+ if (ApprovalManager.SAFE_PREFIX_COMMANDS.has(first)) {
118
+ return `${first} *`;
119
+ }
120
+ // Check two-token safe commands (git diff, npm run, etc.)
121
+ if (tokens.length >= 2) {
122
+ const twoToken = `${tokens[0]} ${tokens[1]}`;
123
+ if (ApprovalManager.SAFE_PREFIX_COMMANDS.has(twoToken)) {
124
+ return `${twoToken} *`;
125
+ }
126
+ }
127
+ return null;
128
+ }
129
+ /**
130
+ * Check if a bash command matches a stored pattern.
131
+ * Patterns of the form "<prefix> *" match any command starting with <prefix>.
132
+ */
133
+ matchesPattern(pattern, cmd) {
134
+ if (pattern.endsWith(' *')) {
135
+ const prefix = pattern.slice(0, -2);
136
+ return cmd === prefix || cmd.startsWith(prefix + ' ');
137
+ }
138
+ return cmd === pattern;
139
+ }
140
+ /** Save an "Always Allow" approval for a tool/command. */
141
+ saveAlwaysAllow(workingDir, toolName, toolInput) {
142
+ if (toolName === 'Bash') {
143
+ const cmd = (typeof toolInput.command === 'string' ? toolInput.command : '').trim();
144
+ this.addRepoApproval(workingDir, { command: cmd });
145
+ console.log(`[auto-approve] saved command for repo ${workingDir}: ${cmd.slice(0, 80)}`);
146
+ }
147
+ else {
148
+ this.addRepoApproval(workingDir, { tool: toolName });
149
+ console.log(`[auto-approve] saved tool for repo ${workingDir}: ${toolName}`);
150
+ }
151
+ }
152
+ /** Save a pattern-based approval (e.g. "cat *") for a tool/command. */
153
+ savePatternApproval(workingDir, toolName, toolInput) {
154
+ const pattern = this.derivePattern(toolName, toolInput);
155
+ if (pattern) {
156
+ this.addRepoApproval(workingDir, { pattern });
157
+ console.log(`[auto-approve] saved pattern for repo ${workingDir}: ${pattern}`);
158
+ }
159
+ else {
160
+ // No pattern derivable — skip saving rather than silently escalating to always-allow
161
+ console.log(`[auto-approve] no pattern derivable for ${toolName}, skipping pattern save`);
162
+ }
163
+ }
164
+ /** Return the auto-approved tools, commands, and patterns for a repo (workingDir). */
165
+ getApprovals(workingDir) {
166
+ const entry = this.repoApprovals.get(workingDir);
167
+ if (!entry)
168
+ return { tools: [], commands: [], patterns: [] };
169
+ return {
170
+ tools: Array.from(entry.tools).sort(),
171
+ commands: Array.from(entry.commands).sort(),
172
+ patterns: Array.from(entry.patterns).sort(),
173
+ };
174
+ }
175
+ /** Remove an auto-approval rule for a repo (workingDir) and persist to disk.
176
+ * Returns 'invalid' if no tool/command provided, or boolean indicating if something was deleted.
177
+ * Pass skipPersist=true for bulk operations (caller must call persistRepoApprovals after). */
178
+ removeApproval(workingDir, opts, skipPersist = false) {
179
+ const tool = typeof opts.tool === 'string' ? opts.tool.trim() : '';
180
+ const command = typeof opts.command === 'string' ? opts.command.trim() : '';
181
+ const pattern = typeof opts.pattern === 'string' ? opts.pattern.trim() : '';
182
+ if (!tool && !command && !pattern)
183
+ return 'invalid';
184
+ const entry = this.repoApprovals.get(workingDir);
185
+ if (!entry)
186
+ return false;
187
+ let removed = false;
188
+ if (tool)
189
+ removed = entry.tools.delete(tool) || removed;
190
+ if (command)
191
+ removed = entry.commands.delete(command) || removed;
192
+ if (pattern)
193
+ removed = entry.patterns.delete(pattern) || removed;
194
+ if (removed && !skipPersist)
195
+ this.persistRepoApprovals();
196
+ return removed;
197
+ }
198
+ /** Write repo-level approvals to disk (atomic rename). */
199
+ persistRepoApprovals() {
200
+ const data = {};
201
+ for (const [dir, entry] of this.repoApprovals) {
202
+ // Only persist non-empty entries
203
+ if (entry.tools.size > 0 || entry.commands.size > 0 || entry.patterns.size > 0) {
204
+ data[dir] = {
205
+ tools: Array.from(entry.tools).sort(),
206
+ commands: Array.from(entry.commands).sort(),
207
+ patterns: Array.from(entry.patterns).sort(),
208
+ };
209
+ }
210
+ }
211
+ try {
212
+ mkdirSync(DATA_DIR, { recursive: true });
213
+ const tmp = REPO_APPROVALS_FILE + '.tmp';
214
+ writeFileSync(tmp, JSON.stringify(data, null, 2), { mode: 0o600 });
215
+ renameSync(tmp, REPO_APPROVALS_FILE);
216
+ }
217
+ catch (err) {
218
+ console.error('Failed to persist repo approvals:', err);
219
+ }
220
+ }
221
+ persistRepoApprovalsDebounced() {
222
+ if (this._approvalPersistTimer)
223
+ return;
224
+ this._approvalPersistTimer = setTimeout(() => {
225
+ this._approvalPersistTimer = null;
226
+ this.persistRepoApprovals();
227
+ }, PERSIST_DEBOUNCE_MS);
228
+ }
229
+ restoreRepoApprovalsFromDisk() {
230
+ if (!existsSync(REPO_APPROVALS_FILE))
231
+ return;
232
+ try {
233
+ const raw = readFileSync(REPO_APPROVALS_FILE, 'utf-8');
234
+ const data = JSON.parse(raw);
235
+ for (const [dir, entry] of Object.entries(data)) {
236
+ this.repoApprovals.set(dir, {
237
+ tools: new Set(entry.tools || []),
238
+ commands: new Set(entry.commands || []),
239
+ patterns: new Set(entry.patterns || []),
240
+ });
241
+ }
242
+ console.log(`Restored repo approvals for ${Object.keys(data).length} repo(s) from disk`);
243
+ }
244
+ catch (err) {
245
+ console.error('Failed to restore repo approvals from disk:', err);
246
+ }
247
+ }
248
+ }
249
+ //# sourceMappingURL=approval-manager.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"approval-manager.js","sourceRoot":"","sources":["../approval-manager.ts"],"names":[],"mappings":"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"}
@@ -8,5 +8,16 @@ import type { Request } from 'express';
8
8
  import type { SessionManager } from './session-manager.js';
9
9
  type VerifyFn = (token: string | undefined) => boolean;
10
10
  type ExtractFn = (req: Request) => string | undefined;
11
+ /**
12
+ * Create an Express router with auth verification and health-check endpoints.
13
+ *
14
+ * @param verifyToken - Returns true if the given bearer token is valid.
15
+ * @param extractToken - Extracts the bearer token from an incoming request (header, cookie, etc.).
16
+ * @param sessions - Session manager instance, used to report active/total session counts.
17
+ * @param claudeAvailable - Whether the Claude CLI binary was found at startup.
18
+ * @param claudeVersion - Version string of the detected Claude CLI.
19
+ * @param apiKeySet - Whether an Anthropic API key is configured.
20
+ * @returns An Express Router with POST `/auth-verify` and GET `/api/health` routes.
21
+ */
11
22
  export declare function createAuthRouter(verifyToken: VerifyFn, extractToken: ExtractFn, sessions: SessionManager, claudeAvailable: boolean, claudeVersion: string, apiKeySet: boolean): Router;
12
23
  export {};
@@ -4,6 +4,17 @@
4
4
  * Mounted directly on the Express app (no path prefix).
5
5
  */
6
6
  import { Router } from 'express';
7
+ /**
8
+ * Create an Express router with auth verification and health-check endpoints.
9
+ *
10
+ * @param verifyToken - Returns true if the given bearer token is valid.
11
+ * @param extractToken - Extracts the bearer token from an incoming request (header, cookie, etc.).
12
+ * @param sessions - Session manager instance, used to report active/total session counts.
13
+ * @param claudeAvailable - Whether the Claude CLI binary was found at startup.
14
+ * @param claudeVersion - Version string of the detected Claude CLI.
15
+ * @param apiKeySet - Whether an Anthropic API key is configured.
16
+ * @returns An Express Router with POST `/auth-verify` and GET `/api/health` routes.
17
+ */
7
18
  export function createAuthRouter(verifyToken, extractToken, sessions, claudeAvailable, claudeVersion, apiKeySet) {
8
19
  const router = Router();
9
20
  router.post('/auth-verify', (req, res) => {
@@ -1 +1 @@
1
- {"version":3,"file":"auth-routes.js","sourceRoot":"","sources":["../auth-routes.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AAEH,OAAO,EAAE,MAAM,EAAE,MAAM,SAAS,CAAA;AAOhC,MAAM,UAAU,gBAAgB,CAC9B,WAAqB,EACrB,YAAuB,EACvB,QAAwB,EACxB,eAAwB,EACxB,aAAqB,EACrB,SAAkB;IAElB,MAAM,MAAM,GAAG,MAAM,EAAE,CAAA;IAEvB,MAAM,CAAC,IAAI,CAAC,cAAc,EAAE,CAAC,GAAG,EAAE,GAAG,EAAE,EAAE;QACvC,MAAM,KAAK,GAAG,YAAY,CAAC,GAAG,CAAC,CAAA;QAC/B,GAAG,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,WAAW,CAAC,KAAK,CAAC,EAAE,CAAC,CAAA;IACzC,CAAC,CAAC,CAAA;IAEF,MAAM,CAAC,GAAG,CAAC,aAAa,EAAE,CAAC,GAAG,EAAE,GAAG,EAAE,EAAE;QACrC,MAAM,KAAK,GAAG,YAAY,CAAC,GAAG,CAAC,CAAA;QAC/B,IAAI,CAAC,WAAW,CAAC,KAAK,CAAC;YAAE,OAAO,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,cAAc,EAAE,CAAC,CAAA;QAE/E,MAAM,WAAW,GAAG,QAAQ,CAAC,IAAI,EAAE,CAAA;QACnC,GAAG,CAAC,IAAI,CAAC;YACP,MAAM,EAAE,IAAI;YACZ,eAAe;YACf,aAAa;YACb,SAAS;YACT,cAAc,EAAE,WAAW,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,MAAM;YACxD,aAAa,EAAE,WAAW,CAAC,MAAM;SAClC,CAAC,CAAA;IACJ,CAAC,CAAC,CAAA;IAEF,OAAO,MAAM,CAAA;AACf,CAAC"}
1
+ {"version":3,"file":"auth-routes.js","sourceRoot":"","sources":["../auth-routes.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AAEH,OAAO,EAAE,MAAM,EAAE,MAAM,SAAS,CAAA;AAOhC;;;;;;;;;;GAUG;AACH,MAAM,UAAU,gBAAgB,CAC9B,WAAqB,EACrB,YAAuB,EACvB,QAAwB,EACxB,eAAwB,EACxB,aAAqB,EACrB,SAAkB;IAElB,MAAM,MAAM,GAAG,MAAM,EAAE,CAAA;IAEvB,MAAM,CAAC,IAAI,CAAC,cAAc,EAAE,CAAC,GAAG,EAAE,GAAG,EAAE,EAAE;QACvC,MAAM,KAAK,GAAG,YAAY,CAAC,GAAG,CAAC,CAAA;QAC/B,GAAG,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,WAAW,CAAC,KAAK,CAAC,EAAE,CAAC,CAAA;IACzC,CAAC,CAAC,CAAA;IAEF,MAAM,CAAC,GAAG,CAAC,aAAa,EAAE,CAAC,GAAG,EAAE,GAAG,EAAE,EAAE;QACrC,MAAM,KAAK,GAAG,YAAY,CAAC,GAAG,CAAC,CAAA;QAC/B,IAAI,CAAC,WAAW,CAAC,KAAK,CAAC;YAAE,OAAO,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,cAAc,EAAE,CAAC,CAAA;QAE/E,MAAM,WAAW,GAAG,QAAQ,CAAC,IAAI,EAAE,CAAA;QACnC,GAAG,CAAC,IAAI,CAAC;YACP,MAAM,EAAE,IAAI;YACZ,eAAe;YACf,aAAa;YACb,SAAS;YACT,cAAc,EAAE,WAAW,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,MAAM;YACxD,aAAa,EAAE,WAAW,CAAC,MAAM;SAClC,CAAC,CAAA;IACJ,CAAC,CAAC,CAAA;IAEF,OAAO,MAAM,CAAA;AACf,CAAC"}