sillyspec 3.18.1 → 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.
- package/.claude/skills/sillyspec-brainstorm/SKILL.md +24 -23
- package/.claude/skills/sillyspec-execute/SKILL.md +8 -1
- package/docs/brainstorm-plan-contract.md +64 -0
- package/docs/plan-execute-contract.md +123 -0
- package/docs/revision-mode.md +115 -0
- package/docs/sillyspec/file-lifecycle.md +13 -4
- package/docs/workflow-contract-regression.md +106 -0
- package/package.json +1 -1
- package/packages/dashboard/dist/assets/{index-DpLHK4jv.js → index-Bq_Z2hne.js} +568 -568
- package/packages/dashboard/dist/assets/{index-BcM2J-hv.css → index-O2W5RV4z.css} +1 -1
- package/packages/dashboard/dist/index.html +16 -16
- package/packages/dashboard/src/components/PipelineStage.vue +22 -2
- package/packages/dashboard/src/components/PipelineView.vue +10 -2
- package/packages/dashboard/src/components/StageBadge.vue +17 -3
- package/packages/dashboard/src/components/StepCard.vue +7 -2
- package/src/change-risk-profile.js +167 -0
- package/src/db.js +6 -0
- package/src/index.js +17 -1
- package/src/knowledge-match.js +130 -0
- package/src/progress.js +464 -11
- package/src/run.js +269 -29
- package/src/scan-postcheck.js +34 -2
- package/src/stage-contract.js +90 -5
- package/src/stages/brainstorm.js +23 -0
- package/src/stages/execute.js +122 -16
- package/src/stages/plan.js +82 -0
- package/src/stages/scan.js +40 -0
- package/src/stages/verify.js +38 -2
- package/test/brainstorm-plan-contract.test.mjs +273 -0
- package/test/knowledge-match.test.mjs +231 -0
- package/test/plan-execute-contract.test.mjs +330 -0
- package/test/platform-failure-samples.test.mjs +4 -0
- package/test/revision-v1.test.mjs +1145 -0
- package/test/scan-knowledge.test.mjs +175 -0
- package/test/scan-postcheck.test.mjs +3 -0
- package/test/spec-dir.test.mjs +8 -3
- 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-
|
|
11
|
-
<link rel="stylesheet" crossorigin href="/assets/index-
|
|
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 = {
|
|
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 = {
|
|
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;
|