sillyspec 3.18.2 → 3.18.3

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 (35) hide show
  1. package/docs/brainstorm-plan-contract.md +64 -0
  2. package/docs/plan-execute-contract.md +123 -0
  3. package/docs/revision-mode.md +115 -0
  4. package/docs/sillyspec/file-lifecycle.md +13 -4
  5. package/docs/workflow-contract-regression.md +106 -0
  6. package/package.json +1 -1
  7. package/packages/dashboard/dist/assets/{index-DpLHK4jv.js → index-Bq_Z2hne.js} +568 -568
  8. package/packages/dashboard/dist/assets/{index-BcM2J-hv.css → index-O2W5RV4z.css} +1 -1
  9. package/packages/dashboard/dist/index.html +16 -16
  10. package/packages/dashboard/src/components/PipelineStage.vue +22 -2
  11. package/packages/dashboard/src/components/PipelineView.vue +10 -2
  12. package/packages/dashboard/src/components/StageBadge.vue +17 -3
  13. package/packages/dashboard/src/components/StepCard.vue +7 -2
  14. package/src/change-risk-profile.js +167 -0
  15. package/src/db.js +6 -0
  16. package/src/index.js +17 -1
  17. package/src/knowledge-match.js +130 -0
  18. package/src/progress.js +464 -11
  19. package/src/run.js +200 -3
  20. package/src/scan-postcheck.js +34 -2
  21. package/src/stage-contract.js +86 -6
  22. package/src/stages/brainstorm.js +23 -0
  23. package/src/stages/execute.js +110 -2
  24. package/src/stages/plan.js +82 -0
  25. package/src/stages/scan.js +40 -0
  26. package/src/stages/verify.js +38 -2
  27. package/test/brainstorm-plan-contract.test.mjs +273 -0
  28. package/test/knowledge-match.test.mjs +231 -0
  29. package/test/plan-execute-contract.test.mjs +330 -0
  30. package/test/platform-failure-samples.test.mjs +4 -0
  31. package/test/revision-v1.test.mjs +1145 -0
  32. package/test/scan-knowledge.test.mjs +175 -0
  33. package/test/scan-postcheck.test.mjs +3 -0
  34. package/test/spec-dir.test.mjs +8 -3
  35. package/test/stage-definitions.test.mjs +1 -1
@@ -1 +1 @@
1
- .line-clamp-2[data-v-4d22d782]{display:-webkit-box;-webkit-line-clamp:2;-webkit-box-orient:vertical;overflow:hidden}.doc-tree[data-v-a51c27a2]{height:100%;min-height:0;display:flex;flex-direction:column;background:#fff}.doc-tree-search[data-v-a51c27a2]{flex-shrink:0;padding:10px 12px 8px;border-bottom:1px solid #F0F0F3}.doc-tree-list[data-v-a51c27a2]{flex:1;min-height:0;overflow:auto;padding:8px 12px 14px}.doc-tree-list[data-v-a51c27a2] .n-tree-node-content__text{white-space:normal;line-height:1.35;overflow-wrap:anywhere}.doc-preview-shell[data-v-a804f0ed]{height:100%;min-height:0;display:flex;flex-direction:column;background:#fff}.doc-empty[data-v-a804f0ed]{height:100%;display:flex;align-items:center;justify-content:center;padding:24px;text-align:center;font-size:12px;color:#636366;font-family:JetBrains Mono,monospace}.doc-toolbar[data-v-a804f0ed]{flex-shrink:0;display:flex;align-items:center;justify-content:space-between;gap:12px;padding:10px 14px;border-bottom:1px solid #F0F0F3}.doc-title[data-v-a804f0ed]{min-width:0;overflow:hidden;text-overflow:ellipsis;white-space:nowrap;font-size:12px;font-weight:600;color:#1c1c1e}.doc-preview-scroll[data-v-a804f0ed]{flex:1;min-height:0;overflow:auto;padding:14px 16px 22px}.doc-modal-body[data-v-a804f0ed]{max-height:min(72vh,760px);overflow:auto;padding-right:4px}.doc-preview[data-v-a804f0ed]{font-size:13px;line-height:1.7;color:#374151;overflow-wrap:anywhere}.doc-preview-large[data-v-a804f0ed]{font-size:14px;line-height:1.75}.doc-preview[data-v-a804f0ed] h1{color:#d97706;font-size:18px;font-weight:700;margin:0 0 16px;border-bottom:1px solid #E5E5EA;padding-bottom:8px}.doc-preview-large[data-v-a804f0ed] h1{font-size:22px}.doc-preview[data-v-a804f0ed] h2{color:#1c1c1e;font-size:15px;font-weight:600;margin:20px 0 10px}.doc-preview-large[data-v-a804f0ed] h2{font-size:17px}.doc-preview[data-v-a804f0ed] h3{color:#d97706;font-size:14px;font-weight:600;margin:16px 0 8px}.doc-preview[data-v-a804f0ed] p{margin:8px 0}.doc-preview[data-v-a804f0ed] strong{color:#1c1c1e;font-weight:600}.doc-preview[data-v-a804f0ed] a{color:#d97706;text-decoration:none}.doc-preview[data-v-a804f0ed] a:hover{text-decoration:underline}.doc-preview[data-v-a804f0ed] code{background:#e5e5ea;color:#1c1c1e;padding:1px 5px;border-radius:3px;font-size:12px;font-family:JetBrains Mono,monospace}.doc-preview[data-v-a804f0ed] pre{background:#1c1c1e;color:#e5e5e7;border:1px solid #E5E5EA;border-radius:6px;padding:12px 16px;overflow-x:auto;font-size:12px;margin:8px 0;line-height:1.5}.doc-preview[data-v-a804f0ed] pre code{background:none;color:inherit;padding:0;border-radius:0;font-size:inherit}.doc-preview[data-v-a804f0ed] ul,.doc-preview[data-v-a804f0ed] ol{padding-left:20px;margin:8px 0;color:#374151}.doc-preview[data-v-a804f0ed] li{margin:4px 0}.doc-preview[data-v-a804f0ed] blockquote{border-left:3px solid #D97706;padding-left:12px;margin:12px 0;color:#636366;font-style:italic}.doc-preview[data-v-a804f0ed] table{width:100%;border-collapse:collapse;margin:12px 0;font-size:12px}.doc-preview[data-v-a804f0ed] thead th{background:#f0f0f3;color:#1c1c1e;font-weight:600;text-align:left;padding:8px 12px;border:1px solid #E5E5EA}.doc-preview[data-v-a804f0ed] tbody td{padding:6px 12px;border:1px solid #E5E5EA;color:#374151}.doc-preview[data-v-a804f0ed] tbody tr:hover{background:#f9fafb}.doc-preview[data-v-a804f0ed] hr{border:none;border-top:1px solid #E5E5EA;margin:16px 0}.docs-panel[data-v-656ea40b]{min-height:0}.docs-tree-pane[data-v-656ea40b]{width:clamp(220px,42%,320px);min-width:180px}.project-card[data-v-814734b8]{width:280px;height:120px;background:#fff;border:2px solid #E5E7EB;border-radius:12px;padding:16px;display:flex;flex-direction:column;justify-content:space-between;cursor:pointer;transition:all .2s;flex-shrink:0}.project-card[data-v-814734b8]:hover{border-color:#d97706;box-shadow:0 4px 12px #d9770626}.project-card.selected[data-v-814734b8]{border-color:#d97706;box-shadow:0 0 0 3px #d9770633}.card-header[data-v-814734b8]{display:flex;justify-content:space-between;align-items:flex-start}.project-name[data-v-814734b8]{font-size:16px;font-weight:600;color:#1a1a1a}.last-active[data-v-814734b8]{font-size:11px;color:#9ca3af}.stage-badge[data-v-814734b8]{display:inline-block;padding:4px 10px;border-radius:12px;font-size:12px;font-weight:500}.stage-badge.in-progress[data-v-814734b8]{background:#dbeafe;color:#1d4ed8}.stage-badge.completed[data-v-814734b8]{background:#d1fae5;color:#047857}.stage-badge.pending[data-v-814734b8]{background:#f3f4f6;color:#6b7280}.progress-section[data-v-814734b8]{display:flex;align-items:center;gap:8px}.progress-bar[data-v-814734b8]{flex:1;height:6px;background:#e5e7eb;border-radius:3px;overflow:hidden}.progress-fill[data-v-814734b8]{height:100%;border-radius:3px;transition:width .3s}.progress-fill.in-progress[data-v-814734b8]{background:#3b82f6}.progress-fill.completed[data-v-814734b8]{background:#10b981}.progress-fill.pending[data-v-814734b8]{background:#9ca3af}.progress-text[data-v-814734b8]{font-size:12px;font-weight:600;color:#374151}.overview-section[data-v-863782e3]{height:100%;background:#fff;padding:16px 24px;display:flex;flex-direction:column}.section-header[data-v-863782e3]{display:flex;justify-content:space-between;align-items:center;margin-bottom:16px}.section-title[data-v-863782e3]{font-size:18px;font-weight:600;color:#1a1a1a;display:flex;align-items:center;gap:8px}.badge[data-v-863782e3]{font-size:12px;padding:2px 8px;background:#e5e7eb;border-radius:10px;color:#6b7280;font-weight:500}.section-actions[data-v-863782e3]{display:flex;gap:8px;align-items:center}.btn[data-v-863782e3]{padding:6px 12px;border-radius:6px;border:none;cursor:pointer;font-size:13px;font-weight:500;transition:all .2s}.btn[data-v-863782e3]:hover{opacity:.9}.btn-secondary[data-v-863782e3]{background:#e5e7eb;color:#374151}.btn-icon[data-v-863782e3]{padding:6px;width:32px;height:32px;display:flex;align-items:center;justify-content:center;background:#f3f4f6;border-radius:6px}.btn-icon[data-v-863782e3]:hover{background:#e5e7eb}.cards-container[data-v-863782e3]{flex:1;display:flex;gap:16px;overflow-x:auto;overflow-y:hidden;padding-bottom:8px}.cards-container[data-v-863782e3]::-webkit-scrollbar{height:8px}.cards-container[data-v-863782e3]::-webkit-scrollbar-track{background:#f3f4f6;border-radius:4px}.cards-container[data-v-863782e3]::-webkit-scrollbar-thumb{background:#d1d5db;border-radius:4px}*{margin:0;padding:0;box-sizing:border-box}body{font-family:DM Sans,-apple-system,BlinkMacSystemFont,sans-serif;-webkit-font-smoothing:antialiased;-moz-osx-smoothing:grayscale}#app{width:100vw;height:100vh;overflow:hidden;min-width:0}body.resizing{-webkit-user-select:none;user-select:none}body.resizing *{cursor:inherit!important}@media(max-width:1280px){.detail-section-title{font-size:11px}.detail-value{font-size:13px}.log-entry{font-size:11px}}.detail-column,.activity-column{padding:18px 20px}.pipeline-column{border-left:1px solid #EEF0F4;border-right:1px solid #EEF0F4}.pipeline-column>*{min-width:0;padding-left:18px;padding-right:18px}.activity-column{gap:12px}.activity-column .detail-section-title{padding:0;margin-bottom:0}.activity-column>.flex-1{padding:0}.project-info{min-width:0}.detail-section-title{font-size:12px;font-weight:600;color:#9ca3af;text-transform:uppercase;margin-bottom:12px}.detail-item{margin-bottom:16px}.detail-label{font-size:11px;color:#9ca3af;margin-bottom:4px}.detail-value{font-size:14px;color:#1a1a1a;font-weight:500;min-width:0;overflow-wrap:anywhere;word-break:break-word;line-height:1.45}.detail-path{font-size:12px;font-family:JetBrains Mono,ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,monospace;white-space:normal}.log-entry{font-family:JetBrains Mono,monospace;font-size:12px;padding:8px;background:#f9fafb;border-radius:4px;margin-bottom:8px;color:#374151}.log-entry.success{color:#047857;background:#d1fae5}.log-entry.error{color:#dc2626;background:#fee2e2}.fade-in{animation:fadeIn .2s ease-out}@keyframes fadeIn{0%{opacity:0;transform:translateY(4px)}to{opacity:1;transform:translateY(0)}}/*! tailwindcss v4.2.2 | MIT License | https://tailwindcss.com */@layer properties{@supports (((-webkit-hyphens:none)) and (not (margin-trim:inline))) or ((-moz-orient:inline) and (not (color:rgb(from red r g b)))){*,:before,:after,::backdrop{--tw-rotate-x:initial;--tw-rotate-y:initial;--tw-rotate-z:initial;--tw-skew-x:initial;--tw-skew-y:initial;--tw-space-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-outline-style:solid;--tw-blur:initial;--tw-brightness:initial;--tw-contrast:initial;--tw-grayscale:initial;--tw-hue-rotate:initial;--tw-invert:initial;--tw-opacity:initial;--tw-saturate:initial;--tw-sepia:initial;--tw-drop-shadow:initial;--tw-drop-shadow-color:initial;--tw-drop-shadow-alpha:100%;--tw-drop-shadow-size:initial;--tw-backdrop-blur:initial;--tw-backdrop-brightness:initial;--tw-backdrop-contrast:initial;--tw-backdrop-grayscale:initial;--tw-backdrop-hue-rotate:initial;--tw-backdrop-invert:initial;--tw-backdrop-opacity:initial;--tw-backdrop-saturate:initial;--tw-backdrop-sepia:initial;--tw-duration:initial}}}@layer theme{:root,:host{--font-sans:ui-sans-serif, system-ui, sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol", "Noto Color Emoji";--font-mono:ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace;--color-gray-400:oklch(70.7% .022 261.325);--color-black:#000;--color-white:#fff;--spacing:.25rem;--text-xs:.75rem;--text-xs--line-height:calc(1 / .75);--text-sm:.875rem;--text-sm--line-height:calc(1.25 / .875);--font-weight-medium:500;--font-weight-semibold:600;--font-weight-bold:700;--tracking-tight:-.025em;--tracking-wider:.05em;--tracking-widest:.1em;--leading-relaxed:1.625;--radius-sm:.25rem;--radius-md:.375rem;--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);--animate-pulse-dot:pulse-dot 1.5s ease-in-out infinite}}@layer base{*,:after,:before,::backdrop{box-sizing:border-box;border:0 solid;margin:0;padding:0}::file-selector-button{box-sizing:border-box;border:0 solid;margin:0;padding:0}html,:host{-webkit-text-size-adjust:100%;-moz-tab-size:4;tab-size:4;line-height:1.5;font-family:var(--default-font-family,ui-sans-serif, system-ui, sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol", "Noto Color Emoji");font-feature-settings:var(--default-font-feature-settings,normal);font-variation-settings:var(--default-font-variation-settings,normal);-webkit-tap-highlight-color:transparent}hr{height:0;color:inherit;border-top-width:1px}abbr:where([title]){-webkit-text-decoration:underline dotted;text-decoration:underline dotted}h1,h2,h3,h4,h5,h6{font-size:inherit;font-weight:inherit}a{color:inherit;-webkit-text-decoration:inherit;text-decoration:inherit}b,strong{font-weight:bolder}code,kbd,samp,pre{font-family:var(--default-mono-font-family,ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace);font-feature-settings:var(--default-mono-font-feature-settings,normal);font-variation-settings:var(--default-mono-font-variation-settings,normal);font-size:1em}small{font-size:80%}sub,sup{vertical-align:baseline;font-size:75%;line-height:0;position:relative}sub{bottom:-.25em}sup{top:-.5em}table{text-indent:0;border-color:inherit;border-collapse:collapse}:-moz-focusring{outline:auto}progress{vertical-align:baseline}summary{display:list-item}ol,ul,menu{list-style:none}img,svg,video,canvas,audio,iframe,embed,object{vertical-align:middle;display:block}img,video{max-width:100%;height:auto}button,input,select,optgroup,textarea{font:inherit;font-feature-settings:inherit;font-variation-settings:inherit;letter-spacing:inherit;color:inherit;opacity:1;background-color:#0000;border-radius:0}::file-selector-button{font:inherit;font-feature-settings:inherit;font-variation-settings:inherit;letter-spacing:inherit;color:inherit;opacity:1;background-color:#0000;border-radius:0}:where(select:is([multiple],[size])) optgroup{font-weight:bolder}:where(select:is([multiple],[size])) optgroup option{padding-inline-start:20px}::file-selector-button{margin-inline-end:4px}::placeholder{opacity:1}@supports (not ((-webkit-appearance:-apple-pay-button))) or (contain-intrinsic-size:1px){::placeholder{color:currentColor}@supports (color:color-mix(in lab,red,red)){::placeholder{color:color-mix(in oklab,currentcolor 50%,transparent)}}}textarea{resize:vertical}::-webkit-search-decoration{-webkit-appearance:none}::-webkit-date-and-time-value{min-height:1lh;text-align:inherit}::-webkit-datetime-edit{display:inline-flex}::-webkit-datetime-edit-fields-wrapper{padding:0}::-webkit-datetime-edit{padding-block:0}::-webkit-datetime-edit-year-field{padding-block:0}::-webkit-datetime-edit-month-field{padding-block:0}::-webkit-datetime-edit-day-field{padding-block:0}::-webkit-datetime-edit-hour-field{padding-block:0}::-webkit-datetime-edit-minute-field{padding-block:0}::-webkit-datetime-edit-second-field{padding-block:0}::-webkit-datetime-edit-millisecond-field{padding-block:0}::-webkit-datetime-edit-meridiem-field{padding-block:0}::-webkit-calendar-picker-indicator{line-height:1}:-moz-ui-invalid{box-shadow:none}button,input:where([type=button],[type=reset],[type=submit]){-webkit-appearance:button;-moz-appearance:button;appearance:button}::file-selector-button{-webkit-appearance:button;-moz-appearance:button;appearance:button}::-webkit-inner-spin-button{height:auto}::-webkit-outer-spin-button{height:auto}[hidden]:where(:not([hidden=until-found])){display:none!important}}@layer components;@layer utilities{.pointer-events-none{pointer-events:none}.collapse{visibility:collapse}.invisible{visibility:hidden}.visible{visibility:visible}.absolute{position:absolute}.fixed{position:fixed}.relative{position:relative}.static{position:static}.inset-0{inset:calc(var(--spacing) * 0)}.inset-x-0{inset-inline:calc(var(--spacing) * 0)}.start{inset-inline-start:var(--spacing)}.end{inset-inline-end:var(--spacing)}.top-0{top:calc(var(--spacing) * 0)}.bottom-0{bottom:calc(var(--spacing) * 0)}.bottom-5{bottom:calc(var(--spacing) * 5)}.left-0{left:calc(var(--spacing) * 0)}.left-5{left:calc(var(--spacing) * 5)}.z-10{z-index:10}.z-20{z-index:20}.z-50{z-index:50}.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-4{margin-inline:calc(var(--spacing) * 4)}.-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)}.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)}.ml-auto{margin-left:auto}.line-clamp-2{-webkit-line-clamp:2;-webkit-box-orient:vertical;display:-webkit-box;overflow:hidden}.block{display:block}.flex{display:flex}.grid{display:grid}.hidden{display:none}.inline{display:inline}.inline-block{display:inline-block}.inline-flex{display:inline-flex}.list-item{display:list-item}.table{display:table}.h-1{height:calc(var(--spacing) * 1)}.h-2{height:calc(var(--spacing) * 2)}.h-3{height:calc(var(--spacing) * 3)}.h-3\.5{height:calc(var(--spacing) * 3.5)}.h-4{height:calc(var(--spacing) * 4)}.h-6{height:calc(var(--spacing) * 6)}.h-8{height:calc(var(--spacing) * 8)}.h-12{height:calc(var(--spacing) * 12)}.h-\[2px\]{height:2px}.h-\[6px\]{height:6px}.h-full{height:100%}.h-px{height:1px}.h-screen{height:100vh}.max-h-32{max-height:calc(var(--spacing) * 32)}.max-h-72{max-height:calc(var(--spacing) * 72)}.w-0{width:calc(var(--spacing) * 0)}.w-1{width:calc(var(--spacing) * 1)}.w-2{width:calc(var(--spacing) * 2)}.w-3{width:calc(var(--spacing) * 3)}.w-3\.5{width:calc(var(--spacing) * 3.5)}.w-5{width:calc(var(--spacing) * 5)}.w-6{width:calc(var(--spacing) * 6)}.w-8{width:calc(var(--spacing) * 8)}.w-10{width:calc(var(--spacing) * 10)}.w-\[2px\]{width:2px}.w-\[4px\]{width:4px}.w-\[340px\]{width:340px}.w-px{width:1px}.w-screen{width:100vw}.min-w-0{min-width:calc(var(--spacing) * 0)}.flex-1{flex:1}.flex-shrink{flex-shrink:1}.flex-shrink-0{flex-shrink:0}.flex-grow{flex-grow:1}.border-collapse{border-collapse:collapse}.rotate-180{rotate:180deg}.transform{transform:var(--tw-rotate-x,) var(--tw-rotate-y,) var(--tw-rotate-z,) var(--tw-skew-x,) var(--tw-skew-y,)}.animate-pulse-dot{animation:var(--animate-pulse-dot)}.cursor-col-resize{cursor:col-resize}.cursor-pointer{cursor:pointer}.cursor-row-resize{cursor:row-resize}.resize{resize:both}.flex-col{flex-direction:column}.flex-wrap{flex-wrap:wrap}.items-center{align-items:center}.items-start{align-items:flex-start}.justify-between{justify-content:space-between}.justify-center{justify-content:center}.gap-1{gap:calc(var(--spacing) * 1)}.gap-1\.5{gap:calc(var(--spacing) * 1.5)}.gap-2{gap:calc(var(--spacing) * 2)}.gap-2\.5{gap:calc(var(--spacing) * 2.5)}.gap-3{gap:calc(var(--spacing) * 3)}.gap-4{gap:calc(var(--spacing) * 4)}.gap-6{gap:calc(var(--spacing) * 6)}:where(.space-y-0\.5>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing) * .5) * var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing) * .5) * calc(1 - var(--tw-space-y-reverse)))}:where(.space-y-1>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing) * 1) * var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing) * 1) * calc(1 - var(--tw-space-y-reverse)))}:where(.space-y-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-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-px>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(1px * var(--tw-space-y-reverse));margin-block-end:calc(1px * calc(1 - var(--tw-space-y-reverse)))}.truncate{text-overflow:ellipsis;white-space:nowrap;overflow:hidden}.overflow-hidden{overflow:hidden}.overflow-y-auto{overflow-y:auto}.rounded{border-radius:.25rem}.rounded-full{border-radius:3.40282e38px}.rounded-md{border-radius:var(--radius-md)}.rounded-sm{border-radius:var(--radius-sm)}.border{border-style:var(--tw-border-style);border-width:1px}.bg-\[\#D97706\]{background-color:#d97706}.bg-black\/70{background-color:#000000b3}@supports (color:color-mix(in lab,red,red)){.bg-black\/70{background-color:color-mix(in oklab,var(--color-black) 70%,transparent)}}.bg-white{background-color:var(--color-white)}.bg-white\/30{background-color:#ffffff4d}@supports (color:color-mix(in lab,red,red)){.bg-white\/30{background-color:color-mix(in oklab,var(--color-white) 30%,transparent)}}.p-3{padding:calc(var(--spacing) * 3)}.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-3{padding-inline:calc(var(--spacing) * 3)}.px-4{padding-inline:calc(var(--spacing) * 4)}.px-5{padding-inline:calc(var(--spacing) * 5)}.px-6{padding-inline:calc(var(--spacing) * 6)}.py-0\.5{padding-block:calc(var(--spacing) * .5)}.py-1{padding-block:calc(var(--spacing) * 1)}.py-1\.5{padding-block:calc(var(--spacing) * 1.5)}.py-2{padding-block:calc(var(--spacing) * 2)}.py-2\.5{padding-block:calc(var(--spacing) * 2.5)}.py-3{padding-block:calc(var(--spacing) * 3)}.py-px{padding-block:1px}.pt-1\.5{padding-top:calc(var(--spacing) * 1.5)}.pt-4{padding-top:calc(var(--spacing) * 4)}.pt-5{padding-top:calc(var(--spacing) * 5)}.pr-4{padding-right:calc(var(--spacing) * 4)}.pb-0{padding-bottom:calc(var(--spacing) * 0)}.pb-2{padding-bottom:calc(var(--spacing) * 2)}.pb-3{padding-bottom:calc(var(--spacing) * 3)}.pb-4{padding-bottom:calc(var(--spacing) * 4)}.pb-5{padding-bottom:calc(var(--spacing) * 5)}.pl-4{padding-left:calc(var(--spacing) * 4)}.pl-\[3px\]{padding-left:3px}.text-center{text-align:center}.font-\[DM_Sans\,sans-serif\]{font-family:DM Sans,sans-serif}.font-\[JetBrains_Mono\,monospace\]{font-family:JetBrains Mono,monospace}.font-mono{font-family:var(--font-mono)}.text-sm{font-size:var(--text-sm);line-height:var(--tw-leading,var(--text-sm--line-height))}.text-xs{font-size:var(--text-xs);line-height:var(--tw-leading,var(--text-xs--line-height))}.text-\[9px\]{font-size:9px}.text-\[10px\]{font-size:10px}.text-\[11px\]{font-size:11px}.text-\[12px\]{font-size:12px}.text-\[13px\]{font-size:13px}.text-\[18px\]{font-size:18px}.leading-relaxed{--tw-leading:var(--leading-relaxed);line-height:var(--leading-relaxed)}.font-bold{--tw-font-weight:var(--font-weight-bold);font-weight:var(--font-weight-bold)}.font-medium{--tw-font-weight:var(--font-weight-medium);font-weight:var(--font-weight-medium)}.font-semibold{--tw-font-weight:var(--font-weight-semibold);font-weight:var(--font-weight-semibold)}.tracking-\[0\.2em\]{--tw-tracking:.2em;letter-spacing:.2em}.tracking-\[0\.15em\]{--tw-tracking:.15em;letter-spacing:.15em}.tracking-\[0\.25em\]{--tw-tracking:.25em;letter-spacing:.25em}.tracking-tight{--tw-tracking:var(--tracking-tight);letter-spacing:var(--tracking-tight)}.tracking-wider{--tw-tracking:var(--tracking-wider);letter-spacing:var(--tracking-wider)}.tracking-widest{--tw-tracking:var(--tracking-widest);letter-spacing:var(--tracking-widest)}.text-black{color:var(--color-black)}.text-gray-400{color:var(--color-gray-400)}.text-white{color:var(--color-white)}.uppercase{text-transform:uppercase}.italic{font-style:italic}.ordinal{--tw-ordinal:ordinal;font-variant-numeric:var(--tw-ordinal,) var(--tw-slashed-zero,) var(--tw-numeric-figure,) var(--tw-numeric-spacing,) var(--tw-numeric-fraction,)}.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}.underline{text-decoration-line:underline}.opacity-0{opacity:0}.shadow{--tw-shadow:0 1px 3px 0 var(--tw-shadow-color,#0000001a), 0 1px 2px -1px var(--tw-shadow-color,#0000001a);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.outline{outline-style:var(--tw-outline-style);outline-width:1px}.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-filter{-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-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-100{--tw-duration:.1s;transition-duration:.1s}.duration-150{--tw-duration:.15s;transition-duration:.15s}.duration-200{--tw-duration:.2s;transition-duration:.2s}.duration-300{--tw-duration:.3s;transition-duration:.3s}.duration-500{--tw-duration:.5s;transition-duration:.5s}.select-none{-webkit-user-select:none;user-select:none}@media(hover:hover){.hover\:bg-\[\#D97706\]:hover{background-color:#d97706}.hover\:bg-\[\#FEF3C7\]:hover{background-color:#fef3c7}}.active\:bg-\[\#D97706\]:active{background-color:#d97706}}@keyframes pulse-glow{0%,to{box-shadow:0 0 #fbbf2466}50%{box-shadow:0 0 12px 2px #fbbf2426}}@keyframes pulse-dot{0%,to{opacity:1;transform:scale(1)}50%{opacity:.4;transform:scale(1.5)}}@keyframes breathe{0%,to{opacity:.5}50%{opacity:1}}@keyframes shimmer{0%{background-position:-200% 0}to{background-position:200% 0}}@keyframes slide-in{0%{opacity:0;transform:translateY(6px)}to{opacity:1;transform:translateY(0)}}@keyframes glow-border{0%,to{border-color:#fbbf2426}50%{border-color:#fbbf2480}}@keyframes float{0%,to{transform:translateY(0)}50%{transform:translateY(-3px)}}.font-mono-log{font-family:JetBrains Mono,ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,monospace}.noise-bg{position:relative}.noise-bg:before{content:"";pointer-events:none;z-index:0;background-image:url("data:image/svg+xml,%3Csvg viewBox='0 0 256 256' xmlns='http://www.w3.org/2000/svg'%3E%3Cfilter id='n'%3E%3CfeTurbulence type='fractalNoise' baseFrequency='0.9' numOctaves='4' stitchTiles='stitch'/%3E%3C/filter%3E%3Crect width='100%25' height='100%25' filter='url(%23n)' opacity='0.03'/%3E%3C/svg%3E");position:absolute;top:0;right:0;bottom:0;left:0}.accent-stripe{background:repeating-linear-gradient(-45deg,#0000,#0000 8px,#fbbf2408 8px,#fbbf2408 9px)}.progress-gradient{background:linear-gradient(90deg,#fbbf24,#f59e0b,#fb923c)}.skeleton-shimmer{background:linear-gradient(90deg,#1a1e28 25%,#2a3040,#1a1e28 75%) 0 0/200% 100%;animation:1.5s ease-in-out infinite shimmer}::-webkit-scrollbar{width:5px;height:5px}::-webkit-scrollbar-track{background:0 0}::-webkit-scrollbar-thumb{background:#2a3040;border-radius:10px}::-webkit-scrollbar-thumb:hover{background:#3a4555}*{scrollbar-width:thin;scrollbar-color:#2a2a2d transparent}::selection{color:#fbbf24;background:#fbbf2433}@property --tw-rotate-x{syntax:"*";inherits:false}@property --tw-rotate-y{syntax:"*";inherits:false}@property --tw-rotate-z{syntax:"*";inherits:false}@property --tw-skew-x{syntax:"*";inherits:false}@property --tw-skew-y{syntax:"*";inherits:false}@property --tw-space-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-outline-style{syntax:"*";inherits:false;initial-value:solid}@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}
1
+ .line-clamp-2[data-v-efc3132a]{display:-webkit-box;-webkit-line-clamp:2;-webkit-box-orient:vertical;overflow:hidden}.doc-tree[data-v-a51c27a2]{height:100%;min-height:0;display:flex;flex-direction:column;background:#fff}.doc-tree-search[data-v-a51c27a2]{flex-shrink:0;padding:10px 12px 8px;border-bottom:1px solid #F0F0F3}.doc-tree-list[data-v-a51c27a2]{flex:1;min-height:0;overflow:auto;padding:8px 12px 14px}.doc-tree-list[data-v-a51c27a2] .n-tree-node-content__text{white-space:normal;line-height:1.35;overflow-wrap:anywhere}.doc-preview-shell[data-v-a804f0ed]{height:100%;min-height:0;display:flex;flex-direction:column;background:#fff}.doc-empty[data-v-a804f0ed]{height:100%;display:flex;align-items:center;justify-content:center;padding:24px;text-align:center;font-size:12px;color:#636366;font-family:JetBrains Mono,monospace}.doc-toolbar[data-v-a804f0ed]{flex-shrink:0;display:flex;align-items:center;justify-content:space-between;gap:12px;padding:10px 14px;border-bottom:1px solid #F0F0F3}.doc-title[data-v-a804f0ed]{min-width:0;overflow:hidden;text-overflow:ellipsis;white-space:nowrap;font-size:12px;font-weight:600;color:#1c1c1e}.doc-preview-scroll[data-v-a804f0ed]{flex:1;min-height:0;overflow:auto;padding:14px 16px 22px}.doc-modal-body[data-v-a804f0ed]{max-height:min(72vh,760px);overflow:auto;padding-right:4px}.doc-preview[data-v-a804f0ed]{font-size:13px;line-height:1.7;color:#374151;overflow-wrap:anywhere}.doc-preview-large[data-v-a804f0ed]{font-size:14px;line-height:1.75}.doc-preview[data-v-a804f0ed] h1{color:#d97706;font-size:18px;font-weight:700;margin:0 0 16px;border-bottom:1px solid #E5E5EA;padding-bottom:8px}.doc-preview-large[data-v-a804f0ed] h1{font-size:22px}.doc-preview[data-v-a804f0ed] h2{color:#1c1c1e;font-size:15px;font-weight:600;margin:20px 0 10px}.doc-preview-large[data-v-a804f0ed] h2{font-size:17px}.doc-preview[data-v-a804f0ed] h3{color:#d97706;font-size:14px;font-weight:600;margin:16px 0 8px}.doc-preview[data-v-a804f0ed] p{margin:8px 0}.doc-preview[data-v-a804f0ed] strong{color:#1c1c1e;font-weight:600}.doc-preview[data-v-a804f0ed] a{color:#d97706;text-decoration:none}.doc-preview[data-v-a804f0ed] a:hover{text-decoration:underline}.doc-preview[data-v-a804f0ed] code{background:#e5e5ea;color:#1c1c1e;padding:1px 5px;border-radius:3px;font-size:12px;font-family:JetBrains Mono,monospace}.doc-preview[data-v-a804f0ed] pre{background:#1c1c1e;color:#e5e5e7;border:1px solid #E5E5EA;border-radius:6px;padding:12px 16px;overflow-x:auto;font-size:12px;margin:8px 0;line-height:1.5}.doc-preview[data-v-a804f0ed] pre code{background:none;color:inherit;padding:0;border-radius:0;font-size:inherit}.doc-preview[data-v-a804f0ed] ul,.doc-preview[data-v-a804f0ed] ol{padding-left:20px;margin:8px 0;color:#374151}.doc-preview[data-v-a804f0ed] li{margin:4px 0}.doc-preview[data-v-a804f0ed] blockquote{border-left:3px solid #D97706;padding-left:12px;margin:12px 0;color:#636366;font-style:italic}.doc-preview[data-v-a804f0ed] table{width:100%;border-collapse:collapse;margin:12px 0;font-size:12px}.doc-preview[data-v-a804f0ed] thead th{background:#f0f0f3;color:#1c1c1e;font-weight:600;text-align:left;padding:8px 12px;border:1px solid #E5E5EA}.doc-preview[data-v-a804f0ed] tbody td{padding:6px 12px;border:1px solid #E5E5EA;color:#374151}.doc-preview[data-v-a804f0ed] tbody tr:hover{background:#f9fafb}.doc-preview[data-v-a804f0ed] hr{border:none;border-top:1px solid #E5E5EA;margin:16px 0}.docs-panel[data-v-9eb3fd4e]{min-height:0}.docs-tree-pane[data-v-9eb3fd4e]{width:clamp(220px,42%,320px);min-width:180px}.project-card[data-v-814734b8]{width:280px;height:120px;background:#fff;border:2px solid #E5E7EB;border-radius:12px;padding:16px;display:flex;flex-direction:column;justify-content:space-between;cursor:pointer;transition:all .2s;flex-shrink:0}.project-card[data-v-814734b8]:hover{border-color:#d97706;box-shadow:0 4px 12px #d9770626}.project-card.selected[data-v-814734b8]{border-color:#d97706;box-shadow:0 0 0 3px #d9770633}.card-header[data-v-814734b8]{display:flex;justify-content:space-between;align-items:flex-start}.project-name[data-v-814734b8]{font-size:16px;font-weight:600;color:#1a1a1a}.last-active[data-v-814734b8]{font-size:11px;color:#9ca3af}.stage-badge[data-v-814734b8]{display:inline-block;padding:4px 10px;border-radius:12px;font-size:12px;font-weight:500}.stage-badge.in-progress[data-v-814734b8]{background:#dbeafe;color:#1d4ed8}.stage-badge.completed[data-v-814734b8]{background:#d1fae5;color:#047857}.stage-badge.pending[data-v-814734b8]{background:#f3f4f6;color:#6b7280}.progress-section[data-v-814734b8]{display:flex;align-items:center;gap:8px}.progress-bar[data-v-814734b8]{flex:1;height:6px;background:#e5e7eb;border-radius:3px;overflow:hidden}.progress-fill[data-v-814734b8]{height:100%;border-radius:3px;transition:width .3s}.progress-fill.in-progress[data-v-814734b8]{background:#3b82f6}.progress-fill.completed[data-v-814734b8]{background:#10b981}.progress-fill.pending[data-v-814734b8]{background:#9ca3af}.progress-text[data-v-814734b8]{font-size:12px;font-weight:600;color:#374151}.overview-section[data-v-863782e3]{height:100%;background:#fff;padding:16px 24px;display:flex;flex-direction:column}.section-header[data-v-863782e3]{display:flex;justify-content:space-between;align-items:center;margin-bottom:16px}.section-title[data-v-863782e3]{font-size:18px;font-weight:600;color:#1a1a1a;display:flex;align-items:center;gap:8px}.badge[data-v-863782e3]{font-size:12px;padding:2px 8px;background:#e5e7eb;border-radius:10px;color:#6b7280;font-weight:500}.section-actions[data-v-863782e3]{display:flex;gap:8px;align-items:center}.btn[data-v-863782e3]{padding:6px 12px;border-radius:6px;border:none;cursor:pointer;font-size:13px;font-weight:500;transition:all .2s}.btn[data-v-863782e3]:hover{opacity:.9}.btn-secondary[data-v-863782e3]{background:#e5e7eb;color:#374151}.btn-icon[data-v-863782e3]{padding:6px;width:32px;height:32px;display:flex;align-items:center;justify-content:center;background:#f3f4f6;border-radius:6px}.btn-icon[data-v-863782e3]:hover{background:#e5e7eb}.cards-container[data-v-863782e3]{flex:1;display:flex;gap:16px;overflow-x:auto;overflow-y:hidden;padding-bottom:8px}.cards-container[data-v-863782e3]::-webkit-scrollbar{height:8px}.cards-container[data-v-863782e3]::-webkit-scrollbar-track{background:#f3f4f6;border-radius:4px}.cards-container[data-v-863782e3]::-webkit-scrollbar-thumb{background:#d1d5db;border-radius:4px}*{margin:0;padding:0;box-sizing:border-box}body{font-family:DM Sans,-apple-system,BlinkMacSystemFont,sans-serif;-webkit-font-smoothing:antialiased;-moz-osx-smoothing:grayscale}#app{width:100vw;height:100vh;overflow:hidden;min-width:0}body.resizing{-webkit-user-select:none;user-select:none}body.resizing *{cursor:inherit!important}@media(max-width:1280px){.detail-section-title{font-size:11px}.detail-value{font-size:13px}.log-entry{font-size:11px}}.detail-column,.activity-column{padding:18px 20px}.pipeline-column{border-left:1px solid #EEF0F4;border-right:1px solid #EEF0F4}.pipeline-column>*{min-width:0;padding-left:18px;padding-right:18px}.activity-column{gap:12px}.activity-column .detail-section-title{padding:0;margin-bottom:0}.activity-column>.flex-1{padding:0}.project-info{min-width:0}.detail-section-title{font-size:12px;font-weight:600;color:#9ca3af;text-transform:uppercase;margin-bottom:12px}.detail-item{margin-bottom:16px}.detail-label{font-size:11px;color:#9ca3af;margin-bottom:4px}.detail-value{font-size:14px;color:#1a1a1a;font-weight:500;min-width:0;overflow-wrap:anywhere;word-break:break-word;line-height:1.45}.detail-path{font-size:12px;font-family:JetBrains Mono,ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,monospace;white-space:normal}.log-entry{font-family:JetBrains Mono,monospace;font-size:12px;padding:8px;background:#f9fafb;border-radius:4px;margin-bottom:8px;color:#374151}.log-entry.success{color:#047857;background:#d1fae5}.log-entry.error{color:#dc2626;background:#fee2e2}.fade-in{animation:fadeIn .2s ease-out}@keyframes fadeIn{0%{opacity:0;transform:translateY(4px)}to{opacity:1;transform:translateY(0)}}/*! tailwindcss v4.2.2 | MIT License | https://tailwindcss.com */@layer properties{@supports (((-webkit-hyphens:none)) and (not (margin-trim:inline))) or ((-moz-orient:inline) and (not (color:rgb(from red r g b)))){*,:before,:after,::backdrop{--tw-rotate-x:initial;--tw-rotate-y:initial;--tw-rotate-z:initial;--tw-skew-x:initial;--tw-skew-y:initial;--tw-space-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-outline-style:solid;--tw-blur:initial;--tw-brightness:initial;--tw-contrast:initial;--tw-grayscale:initial;--tw-hue-rotate:initial;--tw-invert:initial;--tw-opacity:initial;--tw-saturate:initial;--tw-sepia:initial;--tw-drop-shadow:initial;--tw-drop-shadow-color:initial;--tw-drop-shadow-alpha:100%;--tw-drop-shadow-size:initial;--tw-backdrop-blur:initial;--tw-backdrop-brightness:initial;--tw-backdrop-contrast:initial;--tw-backdrop-grayscale:initial;--tw-backdrop-hue-rotate:initial;--tw-backdrop-invert:initial;--tw-backdrop-opacity:initial;--tw-backdrop-saturate:initial;--tw-backdrop-sepia:initial;--tw-duration:initial}}}@layer theme{:root,:host{--font-sans:ui-sans-serif, system-ui, sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol", "Noto Color Emoji";--font-mono:ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace;--color-gray-400:oklch(70.7% .022 261.325);--color-black:#000;--color-white:#fff;--spacing:.25rem;--text-xs:.75rem;--text-xs--line-height:calc(1 / .75);--text-sm:.875rem;--text-sm--line-height:calc(1.25 / .875);--font-weight-medium:500;--font-weight-semibold:600;--font-weight-bold:700;--tracking-tight:-.025em;--tracking-wider:.05em;--tracking-widest:.1em;--leading-relaxed:1.625;--radius-sm:.25rem;--radius-md:.375rem;--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);--animate-pulse-dot:pulse-dot 1.5s ease-in-out infinite}}@layer base{*,:after,:before,::backdrop{box-sizing:border-box;border:0 solid;margin:0;padding:0}::file-selector-button{box-sizing:border-box;border:0 solid;margin:0;padding:0}html,:host{-webkit-text-size-adjust:100%;-moz-tab-size:4;tab-size:4;line-height:1.5;font-family:var(--default-font-family,ui-sans-serif, system-ui, sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol", "Noto Color Emoji");font-feature-settings:var(--default-font-feature-settings,normal);font-variation-settings:var(--default-font-variation-settings,normal);-webkit-tap-highlight-color:transparent}hr{height:0;color:inherit;border-top-width:1px}abbr:where([title]){-webkit-text-decoration:underline dotted;text-decoration:underline dotted}h1,h2,h3,h4,h5,h6{font-size:inherit;font-weight:inherit}a{color:inherit;-webkit-text-decoration:inherit;text-decoration:inherit}b,strong{font-weight:bolder}code,kbd,samp,pre{font-family:var(--default-mono-font-family,ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace);font-feature-settings:var(--default-mono-font-feature-settings,normal);font-variation-settings:var(--default-mono-font-variation-settings,normal);font-size:1em}small{font-size:80%}sub,sup{vertical-align:baseline;font-size:75%;line-height:0;position:relative}sub{bottom:-.25em}sup{top:-.5em}table{text-indent:0;border-color:inherit;border-collapse:collapse}:-moz-focusring{outline:auto}progress{vertical-align:baseline}summary{display:list-item}ol,ul,menu{list-style:none}img,svg,video,canvas,audio,iframe,embed,object{vertical-align:middle;display:block}img,video{max-width:100%;height:auto}button,input,select,optgroup,textarea{font:inherit;font-feature-settings:inherit;font-variation-settings:inherit;letter-spacing:inherit;color:inherit;opacity:1;background-color:#0000;border-radius:0}::file-selector-button{font:inherit;font-feature-settings:inherit;font-variation-settings:inherit;letter-spacing:inherit;color:inherit;opacity:1;background-color:#0000;border-radius:0}:where(select:is([multiple],[size])) optgroup{font-weight:bolder}:where(select:is([multiple],[size])) optgroup option{padding-inline-start:20px}::file-selector-button{margin-inline-end:4px}::placeholder{opacity:1}@supports (not ((-webkit-appearance:-apple-pay-button))) or (contain-intrinsic-size:1px){::placeholder{color:currentColor}@supports (color:color-mix(in lab,red,red)){::placeholder{color:color-mix(in oklab,currentcolor 50%,transparent)}}}textarea{resize:vertical}::-webkit-search-decoration{-webkit-appearance:none}::-webkit-date-and-time-value{min-height:1lh;text-align:inherit}::-webkit-datetime-edit{display:inline-flex}::-webkit-datetime-edit-fields-wrapper{padding:0}::-webkit-datetime-edit{padding-block:0}::-webkit-datetime-edit-year-field{padding-block:0}::-webkit-datetime-edit-month-field{padding-block:0}::-webkit-datetime-edit-day-field{padding-block:0}::-webkit-datetime-edit-hour-field{padding-block:0}::-webkit-datetime-edit-minute-field{padding-block:0}::-webkit-datetime-edit-second-field{padding-block:0}::-webkit-datetime-edit-millisecond-field{padding-block:0}::-webkit-datetime-edit-meridiem-field{padding-block:0}::-webkit-calendar-picker-indicator{line-height:1}:-moz-ui-invalid{box-shadow:none}button,input:where([type=button],[type=reset],[type=submit]){-webkit-appearance:button;-moz-appearance:button;appearance:button}::file-selector-button{-webkit-appearance:button;-moz-appearance:button;appearance:button}::-webkit-inner-spin-button{height:auto}::-webkit-outer-spin-button{height:auto}[hidden]:where(:not([hidden=until-found])){display:none!important}}@layer components;@layer utilities{.pointer-events-none{pointer-events:none}.collapse{visibility:collapse}.invisible{visibility:hidden}.visible{visibility:visible}.absolute{position:absolute}.fixed{position:fixed}.relative{position:relative}.static{position:static}.inset-0{inset:calc(var(--spacing) * 0)}.inset-x-0{inset-inline:calc(var(--spacing) * 0)}.start{inset-inline-start:var(--spacing)}.end{inset-inline-end:var(--spacing)}.top-0{top:calc(var(--spacing) * 0)}.bottom-0{bottom:calc(var(--spacing) * 0)}.bottom-5{bottom:calc(var(--spacing) * 5)}.left-0{left:calc(var(--spacing) * 0)}.left-5{left:calc(var(--spacing) * 5)}.z-10{z-index:10}.z-20{z-index:20}.z-50{z-index:50}.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-4{margin-inline:calc(var(--spacing) * 4)}.-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)}.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)}.ml-auto{margin-left:auto}.line-clamp-2{-webkit-line-clamp:2;-webkit-box-orient:vertical;display:-webkit-box;overflow:hidden}.block{display:block}.flex{display:flex}.grid{display:grid}.hidden{display:none}.inline{display:inline}.inline-block{display:inline-block}.inline-flex{display:inline-flex}.list-item{display:list-item}.table{display:table}.h-1{height:calc(var(--spacing) * 1)}.h-2{height:calc(var(--spacing) * 2)}.h-3{height:calc(var(--spacing) * 3)}.h-3\.5{height:calc(var(--spacing) * 3.5)}.h-4{height:calc(var(--spacing) * 4)}.h-6{height:calc(var(--spacing) * 6)}.h-8{height:calc(var(--spacing) * 8)}.h-12{height:calc(var(--spacing) * 12)}.h-\[2px\]{height:2px}.h-\[6px\]{height:6px}.h-full{height:100%}.h-px{height:1px}.h-screen{height:100vh}.max-h-32{max-height:calc(var(--spacing) * 32)}.max-h-72{max-height:calc(var(--spacing) * 72)}.w-0{width:calc(var(--spacing) * 0)}.w-1{width:calc(var(--spacing) * 1)}.w-2{width:calc(var(--spacing) * 2)}.w-3{width:calc(var(--spacing) * 3)}.w-3\.5{width:calc(var(--spacing) * 3.5)}.w-5{width:calc(var(--spacing) * 5)}.w-6{width:calc(var(--spacing) * 6)}.w-8{width:calc(var(--spacing) * 8)}.w-10{width:calc(var(--spacing) * 10)}.w-\[2px\]{width:2px}.w-\[4px\]{width:4px}.w-\[340px\]{width:340px}.w-px{width:1px}.w-screen{width:100vw}.min-w-0{min-width:calc(var(--spacing) * 0)}.flex-1{flex:1}.flex-shrink{flex-shrink:1}.flex-shrink-0{flex-shrink:0}.flex-grow{flex-grow:1}.border-collapse{border-collapse:collapse}.rotate-180{rotate:180deg}.transform{transform:var(--tw-rotate-x,) var(--tw-rotate-y,) var(--tw-rotate-z,) var(--tw-skew-x,) var(--tw-skew-y,)}.animate-pulse-dot{animation:var(--animate-pulse-dot)}.cursor-col-resize{cursor:col-resize}.cursor-pointer{cursor:pointer}.cursor-row-resize{cursor:row-resize}.resize{resize:both}.flex-col{flex-direction:column}.flex-wrap{flex-wrap:wrap}.items-center{align-items:center}.items-start{align-items:flex-start}.justify-between{justify-content:space-between}.justify-center{justify-content:center}.gap-1{gap:calc(var(--spacing) * 1)}.gap-1\.5{gap:calc(var(--spacing) * 1.5)}.gap-2{gap:calc(var(--spacing) * 2)}.gap-2\.5{gap:calc(var(--spacing) * 2.5)}.gap-3{gap:calc(var(--spacing) * 3)}.gap-4{gap:calc(var(--spacing) * 4)}.gap-6{gap:calc(var(--spacing) * 6)}:where(.space-y-0\.5>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing) * .5) * var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing) * .5) * calc(1 - var(--tw-space-y-reverse)))}:where(.space-y-1>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing) * 1) * var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing) * 1) * calc(1 - var(--tw-space-y-reverse)))}:where(.space-y-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-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-px>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(1px * var(--tw-space-y-reverse));margin-block-end:calc(1px * calc(1 - var(--tw-space-y-reverse)))}.truncate{text-overflow:ellipsis;white-space:nowrap;overflow:hidden}.overflow-hidden{overflow:hidden}.overflow-y-auto{overflow-y:auto}.rounded{border-radius:.25rem}.rounded-full{border-radius:3.40282e38px}.rounded-md{border-radius:var(--radius-md)}.rounded-sm{border-radius:var(--radius-sm)}.border{border-style:var(--tw-border-style);border-width:1px}.bg-\[\#D97706\]{background-color:#d97706}.bg-black\/70{background-color:#000000b3}@supports (color:color-mix(in lab,red,red)){.bg-black\/70{background-color:color-mix(in oklab,var(--color-black) 70%,transparent)}}.bg-white{background-color:var(--color-white)}.bg-white\/30{background-color:#ffffff4d}@supports (color:color-mix(in lab,red,red)){.bg-white\/30{background-color:color-mix(in oklab,var(--color-white) 30%,transparent)}}.p-3{padding:calc(var(--spacing) * 3)}.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-3{padding-inline:calc(var(--spacing) * 3)}.px-4{padding-inline:calc(var(--spacing) * 4)}.px-5{padding-inline:calc(var(--spacing) * 5)}.px-6{padding-inline:calc(var(--spacing) * 6)}.py-0\.5{padding-block:calc(var(--spacing) * .5)}.py-1{padding-block:calc(var(--spacing) * 1)}.py-1\.5{padding-block:calc(var(--spacing) * 1.5)}.py-2{padding-block:calc(var(--spacing) * 2)}.py-2\.5{padding-block:calc(var(--spacing) * 2.5)}.py-3{padding-block:calc(var(--spacing) * 3)}.py-px{padding-block:1px}.pt-1\.5{padding-top:calc(var(--spacing) * 1.5)}.pt-4{padding-top:calc(var(--spacing) * 4)}.pt-5{padding-top:calc(var(--spacing) * 5)}.pr-4{padding-right:calc(var(--spacing) * 4)}.pb-0{padding-bottom:calc(var(--spacing) * 0)}.pb-2{padding-bottom:calc(var(--spacing) * 2)}.pb-3{padding-bottom:calc(var(--spacing) * 3)}.pb-4{padding-bottom:calc(var(--spacing) * 4)}.pb-5{padding-bottom:calc(var(--spacing) * 5)}.pl-4{padding-left:calc(var(--spacing) * 4)}.pl-\[3px\]{padding-left:3px}.text-center{text-align:center}.font-\[DM_Sans\,sans-serif\]{font-family:DM Sans,sans-serif}.font-\[JetBrains_Mono\,monospace\]{font-family:JetBrains Mono,monospace}.font-mono{font-family:var(--font-mono)}.text-sm{font-size:var(--text-sm);line-height:var(--tw-leading,var(--text-sm--line-height))}.text-xs{font-size:var(--text-xs);line-height:var(--tw-leading,var(--text-xs--line-height))}.text-\[9px\]{font-size:9px}.text-\[10px\]{font-size:10px}.text-\[11px\]{font-size:11px}.text-\[12px\]{font-size:12px}.text-\[13px\]{font-size:13px}.text-\[18px\]{font-size:18px}.leading-relaxed{--tw-leading:var(--leading-relaxed);line-height:var(--leading-relaxed)}.font-bold{--tw-font-weight:var(--font-weight-bold);font-weight:var(--font-weight-bold)}.font-medium{--tw-font-weight:var(--font-weight-medium);font-weight:var(--font-weight-medium)}.font-semibold{--tw-font-weight:var(--font-weight-semibold);font-weight:var(--font-weight-semibold)}.tracking-\[0\.2em\]{--tw-tracking:.2em;letter-spacing:.2em}.tracking-\[0\.15em\]{--tw-tracking:.15em;letter-spacing:.15em}.tracking-\[0\.25em\]{--tw-tracking:.25em;letter-spacing:.25em}.tracking-tight{--tw-tracking:var(--tracking-tight);letter-spacing:var(--tracking-tight)}.tracking-wider{--tw-tracking:var(--tracking-wider);letter-spacing:var(--tracking-wider)}.tracking-widest{--tw-tracking:var(--tracking-widest);letter-spacing:var(--tracking-widest)}.text-black{color:var(--color-black)}.text-gray-400{color:var(--color-gray-400)}.text-white{color:var(--color-white)}.uppercase{text-transform:uppercase}.italic{font-style:italic}.ordinal{--tw-ordinal:ordinal;font-variant-numeric:var(--tw-ordinal,) var(--tw-slashed-zero,) var(--tw-numeric-figure,) var(--tw-numeric-spacing,) var(--tw-numeric-fraction,)}.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}.underline{text-decoration-line:underline}.opacity-0{opacity:0}.shadow{--tw-shadow:0 1px 3px 0 var(--tw-shadow-color,#0000001a), 0 1px 2px -1px var(--tw-shadow-color,#0000001a);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.outline{outline-style:var(--tw-outline-style);outline-width:1px}.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-filter{-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-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-100{--tw-duration:.1s;transition-duration:.1s}.duration-150{--tw-duration:.15s;transition-duration:.15s}.duration-200{--tw-duration:.2s;transition-duration:.2s}.duration-300{--tw-duration:.3s;transition-duration:.3s}.duration-500{--tw-duration:.5s;transition-duration:.5s}.select-none{-webkit-user-select:none;user-select:none}@media(hover:hover){.hover\:bg-\[\#D97706\]:hover{background-color:#d97706}.hover\:bg-\[\#FEF3C7\]:hover{background-color:#fef3c7}}.active\:bg-\[\#D97706\]:active{background-color:#d97706}}@keyframes pulse-glow{0%,to{box-shadow:0 0 #fbbf2466}50%{box-shadow:0 0 12px 2px #fbbf2426}}@keyframes pulse-dot{0%,to{opacity:1;transform:scale(1)}50%{opacity:.4;transform:scale(1.5)}}@keyframes breathe{0%,to{opacity:.5}50%{opacity:1}}@keyframes shimmer{0%{background-position:-200% 0}to{background-position:200% 0}}@keyframes slide-in{0%{opacity:0;transform:translateY(6px)}to{opacity:1;transform:translateY(0)}}@keyframes glow-border{0%,to{border-color:#fbbf2426}50%{border-color:#fbbf2480}}@keyframes float{0%,to{transform:translateY(0)}50%{transform:translateY(-3px)}}.font-mono-log{font-family:JetBrains Mono,ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,monospace}.noise-bg{position:relative}.noise-bg:before{content:"";pointer-events:none;z-index:0;background-image:url("data:image/svg+xml,%3Csvg viewBox='0 0 256 256' xmlns='http://www.w3.org/2000/svg'%3E%3Cfilter id='n'%3E%3CfeTurbulence type='fractalNoise' baseFrequency='0.9' numOctaves='4' stitchTiles='stitch'/%3E%3C/filter%3E%3Crect width='100%25' height='100%25' filter='url(%23n)' opacity='0.03'/%3E%3C/svg%3E");position:absolute;top:0;right:0;bottom:0;left:0}.accent-stripe{background:repeating-linear-gradient(-45deg,#0000,#0000 8px,#fbbf2408 8px,#fbbf2408 9px)}.progress-gradient{background:linear-gradient(90deg,#fbbf24,#f59e0b,#fb923c)}.skeleton-shimmer{background:linear-gradient(90deg,#1a1e28 25%,#2a3040,#1a1e28 75%) 0 0/200% 100%;animation:1.5s ease-in-out infinite shimmer}::-webkit-scrollbar{width:5px;height:5px}::-webkit-scrollbar-track{background:0 0}::-webkit-scrollbar-thumb{background:#2a3040;border-radius:10px}::-webkit-scrollbar-thumb:hover{background:#3a4555}*{scrollbar-width:thin;scrollbar-color:#2a2a2d transparent}::selection{color:#fbbf24;background:#fbbf2433}@property --tw-rotate-x{syntax:"*";inherits:false}@property --tw-rotate-y{syntax:"*";inherits:false}@property --tw-rotate-z{syntax:"*";inherits:false}@property --tw-skew-x{syntax:"*";inherits:false}@property --tw-skew-y{syntax:"*";inherits:false}@property --tw-space-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-outline-style{syntax:"*";inherits:false;initial-value:solid}@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}
@@ -1,16 +1,16 @@
1
- <!DOCTYPE html>
2
- <html lang="en">
3
- <head>
4
- <meta charset="UTF-8">
5
- <meta name="viewport" content="width=device-width, initial-scale=1.0">
6
- <title>SillySpec Dashboard</title>
7
- <link rel="preconnect" href="https://fonts.googleapis.com">
8
- <link rel="preconnect" href="https://fonts.gstatic.com" crossorigin>
9
- <link href="https://fonts.googleapis.com/css2?family=DM+Sans:ital,opsz,wght@0,9..40,300;0,9..40,400;0,9..40,500;0,9..40,600;0,9..40,700&family=JetBrains+Mono:wght@400;500;600;700&display=swap" rel="stylesheet">
10
- <script type="module" crossorigin src="/assets/index-DpLHK4jv.js"></script>
11
- <link rel="stylesheet" crossorigin href="/assets/index-BcM2J-hv.css">
12
- </head>
13
- <body style="background-color: #0A0A0B; margin: 0;">
14
- <div id="app"></div>
15
- </body>
16
- </html>
1
+ <!DOCTYPE html>
2
+ <html lang="en">
3
+ <head>
4
+ <meta charset="UTF-8">
5
+ <meta name="viewport" content="width=device-width, initial-scale=1.0">
6
+ <title>SillySpec Dashboard</title>
7
+ <link rel="preconnect" href="https://fonts.googleapis.com">
8
+ <link rel="preconnect" href="https://fonts.gstatic.com" crossorigin>
9
+ <link href="https://fonts.googleapis.com/css2?family=DM+Sans:ital,opsz,wght@0,9..40,300;0,9..40,400;0,9..40,500;0,9..40,600;0,9..40,700&family=JetBrains+Mono:wght@400;500;600;700&display=swap" rel="stylesheet">
10
+ <script type="module" crossorigin src="/assets/index-Bq_Z2hne.js"></script>
11
+ <link rel="stylesheet" crossorigin href="/assets/index-O2W5RV4z.css">
12
+ </head>
13
+ <body style="background-color: #0A0A0B; margin: 0;">
14
+ <div id="app"></div>
15
+ </body>
16
+ </html>
@@ -24,6 +24,13 @@
24
24
  </div>
25
25
 
26
26
  <!-- Steps -->
27
+ <div v-if="status === 'stale' && staleReason" class="mb-2 text-[10px] italic" style="color: #CA8A04;">
28
+ ⚠️ {{ staleReason }}
29
+ <span v-if="name === 'archive'">(已有归档文件保留但不再可信)</span>
30
+ </div>
31
+ <div v-if="status === 'revising' && revisionInfo" class="mb-2 text-[10px]" style="color: #7C3AED;">
32
+ 🔧 {{ revisionInfo }}
33
+ </div>
27
34
  <div class="space-y-1">
28
35
  <div v-if="steps.length === 0" class="text-[11px] italic py-1" style="color: #6B7280;">
29
36
  No steps yet
@@ -52,11 +59,21 @@ const props = defineProps({
52
59
  steps: { type: Array, default: () => [] },
53
60
  status: { type: String, default: 'pending' },
54
61
  isActive: { type: Boolean, default: false },
55
- activeStep: { type: Object, default: null }
62
+ activeStep: { type: Object, default: null },
63
+ staleReason: { type: String, default: '' },
64
+ revision: { type: Number, default: 0 },
65
+ reopenedFromStep: { type: String, default: '' },
56
66
  })
57
67
 
58
68
  const emit = defineEmits(['select-step'])
59
69
 
70
+ const revisionInfo = computed(() => {
71
+ if (!props.revision) return ''
72
+ const parts = [`revision ${props.revision}`]
73
+ if (props.reopenedFromStep) parts.push(`from: ${props.reopenedFromStep}`)
74
+ return parts.join(', ')
75
+ })
76
+
60
77
  const nodeStyle = computed(() => {
61
78
  if (props.isActive) return { background: '#D97706', boxShadow: '0 0 8px rgba(251,191,36,0.4)' }
62
79
  const colors = {
@@ -64,7 +81,10 @@ const nodeStyle = computed(() => {
64
81
  'in-progress': '#D97706',
65
82
  'blocked': '#EA580C',
66
83
  'failed': '#DC2626',
67
- 'pending': '#E5E5EA'
84
+ 'pending': '#E5E5EA',
85
+ 'revising': '#7C3AED',
86
+ 'stale': '#CA8A04',
87
+ 'waiting': '#2563EB',
68
88
  }
69
89
  return { background: colors[props.status] || colors.pending }
70
90
  })
@@ -36,7 +36,7 @@
36
36
  <div v-if="hasStage('verify')" class="flex items-center pl-[3px]"><div class="w-px h-4" style="background: #F0F0F3;" /></div>
37
37
  <PipelineStage name="verify" title="验证" :steps="getStageSteps('verify')" :status="getStageStatus('verify')" :is-active="currentStage === 'verify'" :active-step="activeStep" @select-step="handleSelectStep" />
38
38
  <div v-if="hasStage('archive')" class="flex items-center pl-[3px]"><div class="w-px h-4" style="background: #F0F0F3;" /></div>
39
- <PipelineStage name="archive" title="归档" :steps="getStageSteps('archive')" :status="getStageStatus('archive')" :is-active="currentStage === 'archive'" :active-step="activeStep" @select-step="handleSelectStep" />
39
+ <PipelineStage name="archive" title="归档" :steps="getStageSteps('archive')" :status="getStageStatus('archive')" :is-active="currentStage === 'archive'" :active-step="activeStep" :stale-reason="getStageStaleReason('archive')" :revision="getStageRevision('archive')" :reopened-from-step="getStageReopenedFromStep('archive')" @select-step="handleSelectStep" />
40
40
  </div>
41
41
 
42
42
  <!-- Activity Log -->
@@ -50,7 +50,7 @@
50
50
  <n-timeline-item
51
51
  v-for="(log, i) in activityLogs"
52
52
  :key="i"
53
- :type="log.status === 'completed' ? 'success' : 'warning'"
53
+ :type="log.status === 'completed' ? 'success' : (log.status === 'stale' ? 'warning' : (log.status === 'revising' ? 'info' : 'default'))"
54
54
  >
55
55
  <template #default>
56
56
  <span class="text-[11px]" style="color: #636366;">{{ log.description }}</span>
@@ -117,6 +117,8 @@ const activityLogs = computed(() => {
117
117
  const label = stageNameMap[name] || name
118
118
  if (data.status === 'completed') logs.push({ time: data.completedAt ? formatTime(data.completedAt) : '--:--', status: 'completed', description: `${label} 已完成` })
119
119
  else if (data.status === 'in-progress') logs.push({ time: '--:--', status: 'in-progress', description: `${label} 运行中` })
120
+ else if (data.status === 'revising') logs.push({ time: '--:--', status: 'revising', description: `${label} 修订中 (revision ${data.revision || 1})` })
121
+ else if (data.status === 'stale') logs.push({ time: '--:--', status: 'stale', description: `${label} 已失效:${data.staleReason || '上游修订'}` })
120
122
  }
121
123
  return logs.reverse()
122
124
  })
@@ -126,6 +128,9 @@ function hasStage(n) { return !!stages.value[n] }
126
128
  function getStageSteps(n) { return stages.value[n]?.steps || [] }
127
129
  function getStageStatus(n) {
128
130
  const s = stages.value[n]; if (!s) return 'pending'
131
+ // 优先用 stage 自身的 status(支持 revising/stale)
132
+ if (s.status === 'revising' || s.status === 'stale') return s.status
133
+ // 没有明确 status 时从 step 推导
129
134
  const steps = s.steps || []
130
135
  if (steps.some(x => x.status === 'failed')) return 'failed'
131
136
  if (steps.some(x => x.status === 'blocked')) return 'blocked'
@@ -133,6 +138,9 @@ function getStageStatus(n) {
133
138
  if (steps.every(x => x.status === 'completed')) return 'completed'
134
139
  return 'pending'
135
140
  }
141
+ function getStageStaleReason(n) { return stages.value[n]?.staleReason || '' }
142
+ function getStageRevision(n) { return stages.value[n]?.revision || 0 }
143
+ function getStageReopenedFromStep(n) { return stages.value[n]?.reopenedFromStep || '' }
136
144
  function handleSelectStep(step) { emit('select-step', step) }
137
145
  </script>
138
146
 
@@ -19,7 +19,15 @@ const props = defineProps({
19
19
 
20
20
  const displayLabel = computed(() => {
21
21
  if (props.label) return props.label
22
- const labels = { 'completed': 'done', 'in-progress': 'running', 'blocked': 'blocked', 'failed': 'error' }
22
+ const labels = {
23
+ 'completed': 'done',
24
+ 'in-progress': 'running',
25
+ 'blocked': 'blocked',
26
+ 'failed': 'error',
27
+ 'revising': 'revising',
28
+ 'stale': 'stale',
29
+ 'waiting': 'waiting',
30
+ }
23
31
  return labels[props.status] || ''
24
32
  })
25
33
 
@@ -31,7 +39,10 @@ const badgeStyle = computed(() => {
31
39
  'in-progress': { background: 'rgba(251,191,36,0.1)', color: '#D97706' },
32
40
  'blocked': { background: 'rgba(251,146,60,0.1)', color: '#EA580C' },
33
41
  'failed': { background: 'rgba(239,68,68,0.1)', color: '#DC2626' },
34
- 'pending': { background: 'rgba(82,82,82,0.15)', color: '#6B7280' }
42
+ 'pending': { background: 'rgba(82,82,82,0.15)', color: '#6B7280' },
43
+ 'revising': { background: 'rgba(139,92,246,0.1)', color: '#7C3AED' },
44
+ 'stale': { background: 'rgba(234,179,8,0.1)', color: '#CA8A04' },
45
+ 'waiting': { background: 'rgba(59,130,246,0.1)', color: '#2563EB' },
35
46
  }
36
47
  return styles[props.status] || styles.pending
37
48
  })
@@ -46,7 +57,10 @@ const dotStyle = computed(() => {
46
57
  'in-progress': '#D97706',
47
58
  'blocked': '#EA580C',
48
59
  'failed': '#DC2626',
49
- 'pending': '#6B7280'
60
+ 'pending': '#6B7280',
61
+ 'revising': '#7C3AED',
62
+ 'stale': '#CA8A04',
63
+ 'waiting': '#2563EB',
50
64
  }
51
65
  return { background: colors[props.status] || '#6B7280' }
52
66
  })
@@ -62,7 +62,10 @@ const emit = defineEmits(['select'])
62
62
  const hovered = ref(false)
63
63
 
64
64
  const barColor = computed(() => {
65
- const colors = { 'completed': '#16A34A', 'in-progress': '#D97706', 'blocked': '#EA580C', 'failed': '#DC2626', 'pending': '#E5E5EA' }
65
+ const colors = {
66
+ 'completed': '#16A34A', 'in-progress': '#D97706', 'blocked': '#EA580C', 'failed': '#DC2626', 'pending': '#E5E5EA',
67
+ 'revising': '#7C3AED', 'stale': '#CA8A04', 'waiting': '#2563EB',
68
+ }
66
69
  return colors[props.step.status] || colors.pending
67
70
  })
68
71
 
@@ -72,7 +75,9 @@ const cardStyle = computed(() => ({
72
75
  }))
73
76
 
74
77
  const statusLabel = computed(() => {
75
- const labels = { 'completed': '完成', 'in-progress': '进行中', 'blocked': '阻塞', 'failed': '失败', 'pending': '待办' }
78
+ const labels = { 'completed': '完成', 'in-progress': '进行中', 'blocked': '阻塞', 'failed': '失败', 'pending': '待办',
79
+ 'revising': '修订中', 'stale': '已失效', 'waiting': '等待中',
80
+ }
76
81
  return labels[props.step.status] || '待办'
77
82
  })
78
83
 
@@ -0,0 +1,167 @@
1
+ /**
2
+ * change-risk-profile.js — 变更风险分级检测
3
+ *
4
+ * 根据变更涉及的文件类型、关键词,自动判定 verify 所需的验收强度。
5
+ */
6
+
7
+ /**
8
+ * 触发 integration-critical 风险等级的关键词
9
+ */
10
+ const INTEGRATION_CRITICAL_PATTERNS = [
11
+ // 跨进程通信
12
+ /\bdaemon\b/i,
13
+ /\bbackend\b/i,
14
+ /\bclient.*api\b/i,
15
+ /\bgrpc\b/i,
16
+ /\bwebsocket\b/i,
17
+ /\bhttp.*client\b/i,
18
+ // 状态机 / 生命周期
19
+ /\bsession\b/i,
20
+ /\blease\b/i,
21
+ /\bagent.?run\b/i,
22
+ /\blifecycle\b/i,
23
+ /\bstate.?transition\b/i,
24
+ /\bclaim\b/i,
25
+ /\bheartbeat\b/i,
26
+ // 跨进程协议
27
+ /\bcross.?process\b/i,
28
+ /\bipc\b/i,
29
+ /\bmessage.?queue\b/i,
30
+ /\bpub.?sub\b/i,
31
+ // 部署/启动路径
32
+ /\bcli\.ts\b/i,
33
+ /\bmain\.ts\b/i,
34
+ /\bentrypoint\b/i,
35
+ /\bserver\.(js|ts)\b/i,
36
+ /\bbootstrap\b/i,
37
+ /\bdockerfile\b/i,
38
+ /\bdocker.?compose\b/i,
39
+ ]
40
+
41
+ /**
42
+ * 需要集成验证的文件路径关键词
43
+ */
44
+ const INTEGRATION_FILE_PATTERNS = [
45
+ /daemon/i,
46
+ /session.?manager/i,
47
+ /agent.?run/i,
48
+ /lifecycle/i,
49
+ /state.?machine/i,
50
+ /lease/i,
51
+ /cli\.(js|ts)$/,
52
+ /main\.(js|ts)$/,
53
+ /server\.(js|ts)$/,
54
+ /bootstrap/i,
55
+ /startup/i,
56
+ ]
57
+
58
+ /**
59
+ * 检测变更风险等级
60
+ * @param {object} opts
61
+ * @param {string} [opts.designContent] - design.md 内容
62
+ * @param {string} [opts.planContent] - plan.md 内容
63
+ * @param {string[]} [opts.changedFiles] - 变更文件列表
64
+ * @returns {{ level: string, triggers: string[], requiredVerification: string[] }}
65
+ */
66
+ export function detectChangeRisk({ designContent = '', planContent = '', changedFiles = [] } = {}) {
67
+ const triggers = []
68
+ const combined = [designContent, planContent].join('\n')
69
+
70
+ for (const pattern of INTEGRATION_CRITICAL_PATTERNS) {
71
+ if (pattern.test(combined)) {
72
+ pattern.lastIndex = 0
73
+ const match = combined.match(pattern)
74
+ if (match && !triggers.includes(match[0])) triggers.push(match[0])
75
+ }
76
+ }
77
+
78
+ for (const file of changedFiles) {
79
+ for (const pattern of INTEGRATION_FILE_PATTERNS) {
80
+ if (pattern.test(file)) {
81
+ pattern.lastIndex = 0
82
+ const match = file.match(pattern)
83
+ if (match && !triggers.includes(match[0])) triggers.push(match[0])
84
+ }
85
+ }
86
+ }
87
+
88
+ if (triggers.length === 0) {
89
+ return { level: 'doc-only', triggers: [], requiredVerification: ['static_check'] }
90
+ }
91
+
92
+ const deploymentTrigger = triggers.some(t => /cli\.ts|main\.ts|server\.(js|ts)|bootstrap|entrypoint/i.test(t))
93
+ const lifecycleTrigger = triggers.some(t => /session|lease|agent.?run|lifecycle|state.?transition|claim|heartbeat/i.test(t))
94
+ const crossProcessTrigger = triggers.some(t => /daemon|backend|client.*api|grpc|websocket|cross.?process|ipc|message.?queue/i.test(t))
95
+
96
+ let level
97
+ const requiredVerification = ['unit_tests']
98
+
99
+ if (deploymentTrigger) {
100
+ level = 'deployment-critical'
101
+ requiredVerification.push('contract_tests', 'real_daemon_backend_integration', 'runtime_log_evidence', 'real_startup_once')
102
+ } else if (lifecycleTrigger || crossProcessTrigger) {
103
+ level = 'integration-critical'
104
+ requiredVerification.push('contract_tests', 'real_daemon_backend_integration', 'runtime_log_evidence', 'terminal_state_assertion')
105
+ } else if (triggers.some(t => /api|client|contract|dto/i.test(t))) {
106
+ level = 'contract-required'
107
+ requiredVerification.push('contract_tests')
108
+ } else {
109
+ level = 'unit-sufficient'
110
+ }
111
+
112
+ return { level, triggers, requiredVerification }
113
+ }
114
+
115
+ /**
116
+ * 检查 verify-result.md 是否包含集成验证证据
117
+ * @param {string} verifyContent
118
+ * @param {string[]} requiredVerification
119
+ * @returns {{ ok: boolean, errors: string[], warnings: string[] }}
120
+ */
121
+ export function checkIntegrationEvidence(verifyContent, requiredVerification) {
122
+ const errors = []
123
+ const warnings = []
124
+ const lower = verifyContent.toLowerCase()
125
+
126
+ const needsIntegration = requiredVerification.includes('real_daemon_backend_integration')
127
+ const needsLogEvidence = requiredVerification.includes('runtime_log_evidence')
128
+ const needsTerminalState = requiredVerification.includes('terminal_state_assertion')
129
+ const needsRealStartup = requiredVerification.includes('real_startup_once')
130
+
131
+ if (needsIntegration) {
132
+ const hasMockOnly = /mock.*test.*passed|unit.*test.*passed/i.test(lower)
133
+ const hasIntegrationEvidence =
134
+ /集成测试|integration.*test|e2e.*test|端到端/i.test(lower) ||
135
+ /daemon.*backend|backend.*daemon|真实.*集成|real.*integration/i.test(lower) ||
136
+ /runtime.*evidence|运行时.*证据/i.test(lower)
137
+
138
+ if (!hasIntegrationEvidence && hasMockOnly) {
139
+ errors.push('integration-critical 变更只提供了 mock 单测证据,缺少真实 daemon↔backend 集成验证')
140
+ } else if (!hasIntegrationEvidence) {
141
+ errors.push('integration-critical 变更缺少集成验证证据 — 需要真实 daemon↔backend 测试结果或运行时日志')
142
+ }
143
+ }
144
+
145
+ if (needsLogEvidence) {
146
+ const hasRuntimeSection = /runtime.*evidence|运行时.*证据|daemon.*log|日志.*片段/i.test(lower)
147
+ if (!hasRuntimeSection) {
148
+ errors.push('integration-critical 变更的 verify-result.md 缺少 Runtime Evidence section')
149
+ }
150
+ }
151
+
152
+ if (needsTerminalState) {
153
+ const hasTerminalState = /terminal.*state|终态|running.*completed|completed.*failed|session.*end|lease.*end/i.test(lower)
154
+ if (!hasTerminalState) {
155
+ warnings.push('建议检查终态断言:AgentRun running→completed/failed、session end 状态同步')
156
+ }
157
+ }
158
+
159
+ if (needsRealStartup) {
160
+ const hasStartupEvidence = /启动.*一次|real.*startup|实际.*启动|docker.*up|npm.*start|node.*server/i.test(lower)
161
+ if (!hasStartupEvidence) {
162
+ errors.push('deployment-critical 变更需要真实启动验证证据')
163
+ }
164
+ }
165
+
166
+ return { ok: errors.length === 0, errors, warnings }
167
+ }
package/src/db.js CHANGED
@@ -179,6 +179,12 @@ export class DB {
179
179
  this._migrateAddColumn('steps', 'wait_answers', 'TEXT'); // JSON array
180
180
  this._migrateAddColumn('steps', 'wait_round', 'INTEGER');
181
181
  this._migrateAddColumn('steps', 'max_wait_rounds', 'INTEGER');
182
+
183
+ // Revision v1 support
184
+ this._migrateAddColumn('stages', 'revision', 'INTEGER DEFAULT 0');
185
+ this._migrateAddColumn('stages', 'reopened_from_step', 'TEXT');
186
+ this._migrateAddColumn('stages', 'reopened_at', 'TEXT');
187
+ this._migrateAddColumn('stages', 'stale_reason', 'TEXT');
182
188
  }
183
189
 
184
190
  /**
package/src/index.js CHANGED
@@ -31,7 +31,9 @@ SillySpec CLI — 规范驱动开发工具包
31
31
  --done --output "..." 完成当前步骤
32
32
  --skip 跳过可选步骤
33
33
  --status 查看阶段进度
34
- --reset 重置阶段
34
+ --reset 重置阶段(从头开始)
35
+ --reopen 重新打开已完成阶段进入修订模式
36
+ --from-step <index|name> 配合 --reopen:从指定步骤开始修订
35
37
  --change <name> 设置当前变更名
36
38
  --spec-dir <path> 指定规范目录(默认 <项目>/.sillyspec)
37
39
  --runtime-root <path> 平台模式:运行时产物根路径
@@ -43,6 +45,10 @@ SillySpec CLI — 规范驱动开发工具包
43
45
  scan, brainstorm, plan, execute, verify, archive
44
46
  quick, explore, status, doctor
45
47
 
48
+ Revision mode:
49
+ 已完成阶段不能直接重跑。使用 --reopen --from-step 进入受控修订。
50
+ 重开会使下游阶段自动标记为 stale,但不修改已有产物文件。
51
+
46
52
  sillyspec progress <cmd> 进度记录(轻量,不强制顺序)
47
53
  init 初始化项目数据库
48
54
  show 查看当前进度
@@ -50,6 +56,8 @@ SillySpec CLI — 规范驱动开发工具包
50
56
  add-step <stage> <name> 添加步骤
51
57
  update-step <s> <n> --status <st> [--output <t>]
52
58
  complete-stage <stage> 标记阶段完成
59
+ check 状态一致性检查(只报告,不修复)
60
+ repair [--apply] 修复状态元数据(默认 dry-run,--apply 才修改)
53
61
  validate 校验并修复
54
62
  reset [--stage X] 重置进度
55
63
 
@@ -174,6 +182,14 @@ async function main() {
174
182
  case 'show':
175
183
  pm.show(dir, progChangeName);
176
184
  break;
185
+ case 'check':
186
+ await pm.checkConsistency(dir, progChangeName);
187
+ break;
188
+ case 'repair': {
189
+ const repairApply = filteredArgs.includes('--apply');
190
+ await pm.repairConsistency(dir, { apply: repairApply, changeName: progChangeName });
191
+ break;
192
+ }
177
193
  case 'validate':
178
194
  await pm.validate(dir, progChangeName);
179
195
  break;