cli-jaw 2.0.1 → 2.0.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.ko.md +3 -2
- package/README.md +41 -12
- package/dist/bin/_http-client.js +34 -0
- package/dist/bin/_http-client.js.map +1 -0
- package/dist/bin/cli-jaw.js +10 -5
- package/dist/bin/cli-jaw.js.map +1 -1
- package/dist/bin/commands/browser-web-ai.js +155 -38
- package/dist/bin/commands/browser-web-ai.js.map +1 -1
- package/dist/bin/commands/browser.js +265 -37
- package/dist/bin/commands/browser.js.map +1 -1
- package/dist/bin/commands/chat.js +18 -14
- package/dist/bin/commands/chat.js.map +1 -1
- package/dist/bin/commands/clone.js +3 -2
- package/dist/bin/commands/clone.js.map +1 -1
- package/dist/bin/commands/dashboard.js +22 -18
- package/dist/bin/commands/dashboard.js.map +1 -1
- package/dist/bin/commands/dispatch.js +9 -11
- package/dist/bin/commands/dispatch.js.map +1 -1
- package/dist/bin/commands/doctor.js +26 -14
- package/dist/bin/commands/doctor.js.map +1 -1
- package/dist/bin/commands/employee.js +12 -7
- package/dist/bin/commands/employee.js.map +1 -1
- package/dist/bin/commands/init.js +9 -9
- package/dist/bin/commands/init.js.map +1 -1
- package/dist/bin/commands/launchd.js +17 -7
- package/dist/bin/commands/launchd.js.map +1 -1
- package/dist/bin/commands/mcp.js +7 -4
- package/dist/bin/commands/mcp.js.map +1 -1
- package/dist/bin/commands/memory.js +7 -5
- package/dist/bin/commands/memory.js.map +1 -1
- package/dist/bin/commands/orchestrate.js +6 -3
- package/dist/bin/commands/orchestrate.js.map +1 -1
- package/dist/bin/commands/reset.js +4 -3
- package/dist/bin/commands/reset.js.map +1 -1
- package/dist/bin/commands/serve.js +3 -2
- package/dist/bin/commands/serve.js.map +1 -1
- package/dist/bin/commands/service.js +2 -2
- package/dist/bin/commands/service.js.map +1 -1
- package/dist/bin/commands/status.js +10 -9
- package/dist/bin/commands/status.js.map +1 -1
- package/dist/bin/commands/tui/api.js +26 -18
- package/dist/bin/commands/tui/api.js.map +1 -1
- package/dist/bin/commands/tui/overlays.js +3 -1
- package/dist/bin/commands/tui/overlays.js.map +1 -1
- package/dist/bin/commands/tui/simple-mode.js +1 -1
- package/dist/bin/commands/tui/simple-mode.js.map +1 -1
- package/dist/bin/commands/tui/types.js.map +1 -1
- package/dist/bin/postinstall.js +153 -65
- package/dist/bin/postinstall.js.map +1 -1
- package/dist/bin/star-prompt.js +4 -3
- package/dist/bin/star-prompt.js.map +1 -1
- package/dist/lib/mcp/format-converters.js +15 -10
- package/dist/lib/mcp/format-converters.js.map +1 -1
- package/dist/lib/mcp/mcp-install.js +2 -2
- package/dist/lib/mcp/mcp-install.js.map +1 -1
- package/dist/lib/mcp/skills-distribution.js +2 -2
- package/dist/lib/mcp/skills-distribution.js.map +1 -1
- package/dist/lib/mcp/skills-reset.js +5 -4
- package/dist/lib/mcp/skills-reset.js.map +1 -1
- package/dist/lib/mcp/skills-symlinks.js +9 -9
- package/dist/lib/mcp/skills-symlinks.js.map +1 -1
- package/dist/lib/mcp/skills-utils.js +5 -4
- package/dist/lib/mcp/skills-utils.js.map +1 -1
- package/dist/lib/mcp/unified-config.js +3 -2
- package/dist/lib/mcp/unified-config.js.map +1 -1
- package/dist/lib/mime-detect.js +65 -0
- package/dist/lib/mime-detect.js.map +1 -0
- package/dist/lib/quota-copilot.js +12 -10
- package/dist/lib/quota-copilot.js.map +1 -1
- package/dist/lib/stt.js +3 -3
- package/dist/lib/stt.js.map +1 -1
- package/dist/lib/upload.js +14 -5
- package/dist/lib/upload.js.map +1 -1
- package/dist/scripts/i18n-registry.js +4 -4
- package/dist/scripts/i18n-registry.js.map +1 -1
- package/dist/server.js +61 -57
- package/dist/server.js.map +1 -1
- package/dist/src/agent/alert-escalation.js +61 -0
- package/dist/src/agent/alert-escalation.js.map +1 -0
- package/dist/src/agent/args.js +71 -2
- package/dist/src/agent/args.js.map +1 -1
- package/dist/src/agent/error-classifier.js +9 -4
- package/dist/src/agent/error-classifier.js.map +1 -1
- package/dist/src/agent/events.js +130 -101
- package/dist/src/agent/events.js.map +1 -1
- package/dist/src/agent/lifecycle-handler.js +44 -19
- package/dist/src/agent/lifecycle-handler.js.map +1 -1
- package/dist/src/agent/live-run-state.js +7 -3
- package/dist/src/agent/live-run-state.js.map +1 -1
- package/dist/src/agent/memory-flush-controller.js +6 -4
- package/dist/src/agent/memory-flush-controller.js.map +1 -1
- package/dist/src/agent/opencode-diagnostics.js +7 -6
- package/dist/src/agent/opencode-diagnostics.js.map +1 -1
- package/dist/src/agent/session-persistence.js +1 -1
- package/dist/src/agent/session-persistence.js.map +1 -1
- package/dist/src/agent/spawn-env.js +33 -16
- package/dist/src/agent/spawn-env.js.map +1 -1
- package/dist/src/agent/spawn.js +176 -110
- package/dist/src/agent/spawn.js.map +1 -1
- package/dist/src/agent/watchdog.js +56 -0
- package/dist/src/agent/watchdog.js.map +1 -0
- package/dist/src/browser/actions.js +179 -70
- package/dist/src/browser/actions.js.map +1 -1
- package/dist/src/browser/connection.js +296 -19
- package/dist/src/browser/connection.js.map +1 -1
- package/dist/src/browser/index.js +4 -1
- package/dist/src/browser/index.js.map +1 -1
- package/dist/src/browser/launch-policy.js +1 -1
- package/dist/src/browser/launch-policy.js.map +1 -1
- package/dist/src/browser/primitives.js +11 -7
- package/dist/src/browser/primitives.js.map +1 -1
- package/dist/src/browser/runtime-diagnostics.js +45 -0
- package/dist/src/browser/runtime-diagnostics.js.map +1 -0
- package/dist/src/browser/runtime-orphans.js +127 -0
- package/dist/src/browser/runtime-orphans.js.map +1 -0
- package/dist/src/browser/runtime-owner-store.js +51 -0
- package/dist/src/browser/runtime-owner-store.js.map +1 -0
- package/dist/src/browser/runtime-owner.js +2 -2
- package/dist/src/browser/runtime-owner.js.map +1 -1
- package/dist/src/browser/tab-lifecycle.js +157 -0
- package/dist/src/browser/tab-lifecycle.js.map +1 -0
- package/dist/src/browser/vision.js +35 -11
- package/dist/src/browser/vision.js.map +1 -1
- package/dist/src/browser/web-ai/action-breadth.js +56 -0
- package/dist/src/browser/web-ai/action-breadth.js.map +1 -0
- package/dist/src/browser/web-ai/action-cache.js +9 -8
- package/dist/src/browser/web-ai/action-cache.js.map +1 -1
- package/dist/src/browser/web-ai/action-intent.js +64 -0
- package/dist/src/browser/web-ai/action-intent.js.map +1 -0
- package/dist/src/browser/web-ai/action-memory.js +82 -0
- package/dist/src/browser/web-ai/action-memory.js.map +1 -0
- package/dist/src/browser/web-ai/action-trace.js +11 -6
- package/dist/src/browser/web-ai/action-trace.js.map +1 -1
- package/dist/src/browser/web-ai/annotated-screenshot.js +2 -1
- package/dist/src/browser/web-ai/annotated-screenshot.js.map +1 -1
- package/dist/src/browser/web-ai/answer-artifact.js +97 -0
- package/dist/src/browser/web-ai/answer-artifact.js.map +1 -0
- package/dist/src/browser/web-ai/ax-snapshot.js +3 -2
- package/dist/src/browser/web-ai/ax-snapshot.js.map +1 -1
- package/dist/src/browser/web-ai/browser-primitives.js +10 -7
- package/dist/src/browser/web-ai/browser-primitives.js.map +1 -1
- package/dist/src/browser/web-ai/capability-registry.js +1 -1
- package/dist/src/browser/web-ai/capability-registry.js.map +1 -1
- package/dist/src/browser/web-ai/chatgpt-composer.js +12 -12
- package/dist/src/browser/web-ai/chatgpt-composer.js.map +1 -1
- package/dist/src/browser/web-ai/chatgpt-model.js +469 -50
- package/dist/src/browser/web-ai/chatgpt-model.js.map +1 -1
- package/dist/src/browser/web-ai/chatgpt-response.js +82 -10
- package/dist/src/browser/web-ai/chatgpt-response.js.map +1 -1
- package/dist/src/browser/web-ai/chatgpt.js +325 -49
- package/dist/src/browser/web-ai/chatgpt.js.map +1 -1
- package/dist/src/browser/web-ai/churn-log.js +1 -1
- package/dist/src/browser/web-ai/churn-log.js.map +1 -1
- package/dist/src/browser/web-ai/cli-sessions.js +42 -26
- package/dist/src/browser/web-ai/cli-sessions.js.map +1 -1
- package/dist/src/browser/web-ai/context-pack/builder.js +1 -1
- package/dist/src/browser/web-ai/context-pack/builder.js.map +1 -1
- package/dist/src/browser/web-ai/context-pack/file-selector.js +2 -2
- package/dist/src/browser/web-ai/context-pack/file-selector.js.map +1 -1
- package/dist/src/browser/web-ai/context-pack/report.js +1 -1
- package/dist/src/browser/web-ai/context-pack/report.js.map +1 -1
- package/dist/src/browser/web-ai/copy-markdown.js +25 -5
- package/dist/src/browser/web-ai/copy-markdown.js.map +1 -1
- package/dist/src/browser/web-ai/diagnostics.js +17 -14
- package/dist/src/browser/web-ai/diagnostics.js.map +1 -1
- package/dist/src/browser/web-ai/doctor.js +10 -6
- package/dist/src/browser/web-ai/doctor.js.map +1 -1
- package/dist/src/browser/web-ai/dom-hash.js.map +1 -1
- package/dist/src/browser/web-ai/errors.js +5 -22
- package/dist/src/browser/web-ai/errors.js.map +1 -1
- package/dist/src/browser/web-ai/gemini-live.js +7 -6
- package/dist/src/browser/web-ai/gemini-live.js.map +1 -1
- package/dist/src/browser/web-ai/grok-live.js +9 -5
- package/dist/src/browser/web-ai/grok-live.js.map +1 -1
- package/dist/src/browser/web-ai/index.js +5 -0
- package/dist/src/browser/web-ai/index.js.map +1 -1
- package/dist/src/browser/web-ai/observation-bundle.js +90 -0
- package/dist/src/browser/web-ai/observation-bundle.js.map +1 -0
- package/dist/src/browser/web-ai/observe-actions.js +186 -0
- package/dist/src/browser/web-ai/observe-actions.js.map +1 -0
- package/dist/src/browser/web-ai/observe-targets.js +6 -1
- package/dist/src/browser/web-ai/observe-targets.js.map +1 -1
- package/dist/src/browser/web-ai/planner-contract.js +18 -0
- package/dist/src/browser/web-ai/planner-contract.js.map +1 -0
- package/dist/src/browser/web-ai/post-action-assert.js +8 -7
- package/dist/src/browser/web-ai/post-action-assert.js.map +1 -1
- package/dist/src/browser/web-ai/product-surfaces.js +2 -1
- package/dist/src/browser/web-ai/product-surfaces.js.map +1 -1
- package/dist/src/browser/web-ai/provider-adapter.js.map +1 -1
- package/dist/src/browser/web-ai/question.js +14 -7
- package/dist/src/browser/web-ai/question.js.map +1 -1
- package/dist/src/browser/web-ai/ref-registry.js.map +1 -1
- package/dist/src/browser/web-ai/self-heal.js +20 -14
- package/dist/src/browser/web-ai/self-heal.js.map +1 -1
- package/dist/src/browser/web-ai/session-store.js +61 -1
- package/dist/src/browser/web-ai/session-store.js.map +1 -1
- package/dist/src/browser/web-ai/session.js +43 -2
- package/dist/src/browser/web-ai/session.js.map +1 -1
- package/dist/src/browser/web-ai/source-audit.js +116 -0
- package/dist/src/browser/web-ai/source-audit.js.map +1 -0
- package/dist/src/browser/web-ai/tab-finalizer.js +18 -0
- package/dist/src/browser/web-ai/tab-finalizer.js.map +1 -0
- package/dist/src/browser/web-ai/tab-lease-store.js +390 -0
- package/dist/src/browser/web-ai/tab-lease-store.js.map +1 -0
- package/dist/src/browser/web-ai/tab-pool.js +44 -0
- package/dist/src/browser/web-ai/tab-pool.js.map +1 -0
- package/dist/src/browser/web-ai/target-resolver.js +31 -0
- package/dist/src/browser/web-ai/target-resolver.js.map +1 -0
- package/dist/src/browser/web-ai/trace-persistence.js +4 -2
- package/dist/src/browser/web-ai/trace-persistence.js.map +1 -1
- package/dist/src/browser/web-ai/vendor-editor-contract.js.map +1 -1
- package/dist/src/browser/web-ai/watcher.js +8 -7
- package/dist/src/browser/web-ai/watcher.js.map +1 -1
- package/dist/src/cli/acp-client.js +40 -26
- package/dist/src/cli/acp-client.js.map +1 -1
- package/dist/src/cli/command-context.js +3 -3
- package/dist/src/cli/command-context.js.map +1 -1
- package/dist/src/cli/commands.js +25 -16
- package/dist/src/cli/commands.js.map +1 -1
- package/dist/src/cli/compact.js.map +1 -1
- package/dist/src/cli/handlers-completions.js +6 -4
- package/dist/src/cli/handlers-completions.js.map +1 -1
- package/dist/src/cli/handlers-runtime.js +45 -38
- package/dist/src/cli/handlers-runtime.js.map +1 -1
- package/dist/src/cli/handlers.js +44 -36
- package/dist/src/cli/handlers.js.map +1 -1
- package/dist/src/cli/readiness.js +3 -1
- package/dist/src/cli/readiness.js.map +1 -1
- package/dist/src/cli/registry.js +1 -1
- package/dist/src/cli/registry.js.map +1 -1
- package/dist/src/cli/tui/overlay.js +4 -1
- package/dist/src/cli/tui/overlay.js.map +1 -1
- package/dist/src/cli/types.js +4 -0
- package/dist/src/cli/types.js.map +1 -0
- package/dist/src/command-contract/catalog.js +1 -1
- package/dist/src/command-contract/catalog.js.map +1 -1
- package/dist/src/command-contract/help-renderer.js +1 -1
- package/dist/src/command-contract/help-renderer.js.map +1 -1
- package/dist/src/core/browser-open-default.js +13 -0
- package/dist/src/core/browser-open-default.js.map +1 -0
- package/dist/src/core/browser-open.js +41 -0
- package/dist/src/core/browser-open.js.map +1 -0
- package/dist/src/core/bus.js +13 -2
- package/dist/src/core/bus.js.map +1 -1
- package/dist/src/core/compact.js +5 -4
- package/dist/src/core/compact.js.map +1 -1
- package/dist/src/core/config.js +51 -52
- package/dist/src/core/config.js.map +1 -1
- package/dist/src/core/db.js +48 -6
- package/dist/src/core/db.js.map +1 -1
- package/dist/src/core/employees.js +9 -8
- package/dist/src/core/employees.js.map +1 -1
- package/dist/src/core/instance.js +7 -6
- package/dist/src/core/instance.js.map +1 -1
- package/dist/src/core/logger.js +1 -1
- package/dist/src/core/logger.js.map +1 -1
- package/dist/src/core/main-session.js +7 -7
- package/dist/src/core/main-session.js.map +1 -1
- package/dist/src/core/path-expand.js +10 -0
- package/dist/src/core/path-expand.js.map +1 -0
- package/dist/src/core/runtime-path.js +1 -1
- package/dist/src/core/runtime-path.js.map +1 -1
- package/dist/src/core/runtime-settings.js +13 -13
- package/dist/src/core/runtime-settings.js.map +1 -1
- package/dist/src/core/settings-merge.js +14 -14
- package/dist/src/core/settings-merge.js.map +1 -1
- package/dist/src/core/strip-undefined.js +13 -0
- package/dist/src/core/strip-undefined.js.map +1 -0
- package/dist/src/core/tcc.js +1 -1
- package/dist/src/core/tcc.js.map +1 -1
- package/dist/src/discord/bot.js +19 -18
- package/dist/src/discord/bot.js.map +1 -1
- package/dist/src/discord/channel-types.js +14 -0
- package/dist/src/discord/channel-types.js.map +1 -0
- package/dist/src/discord/commands.js +7 -6
- package/dist/src/discord/commands.js.map +1 -1
- package/dist/src/discord/discord-file.js.map +1 -1
- package/dist/src/discord/forwarder.js +3 -3
- package/dist/src/discord/forwarder.js.map +1 -1
- package/dist/src/http/error-middleware.js +3 -3
- package/dist/src/http/error-middleware.js.map +1 -1
- package/dist/src/http/response.js +2 -2
- package/dist/src/http/response.js.map +1 -1
- package/dist/src/ide/diff.js +7 -7
- package/dist/src/ide/diff.js.map +1 -1
- package/dist/src/manager/board/routes.js +28 -27
- package/dist/src/manager/board/routes.js.map +1 -1
- package/dist/src/manager/dashboard-home.js +3 -5
- package/dist/src/manager/dashboard-home.js.map +1 -1
- package/dist/src/manager/dashboard-service.js +2 -2
- package/dist/src/manager/dashboard-service.js.map +1 -1
- package/dist/src/manager/health-history.js +10 -9
- package/dist/src/manager/health-history.js.map +1 -1
- package/dist/src/manager/launchd-service.js +2 -2
- package/dist/src/manager/launchd-service.js.map +1 -1
- package/dist/src/manager/lifecycle-helpers.js +7 -6
- package/dist/src/manager/lifecycle-helpers.js.map +1 -1
- package/dist/src/manager/lifecycle.js +7 -6
- package/dist/src/manager/lifecycle.js.map +1 -1
- package/dist/src/manager/logs.js +18 -18
- package/dist/src/manager/logs.js.map +1 -1
- package/dist/src/manager/metadata.js +10 -2
- package/dist/src/manager/metadata.js.map +1 -1
- package/dist/src/manager/notes/assets.js +216 -0
- package/dist/src/manager/notes/assets.js.map +1 -0
- package/dist/src/manager/notes/capabilities.js +56 -0
- package/dist/src/manager/notes/capabilities.js.map +1 -0
- package/dist/src/manager/notes/constants.js +11 -0
- package/dist/src/manager/notes/constants.js.map +1 -0
- package/dist/src/manager/notes/frontmatter.js +120 -0
- package/dist/src/manager/notes/frontmatter.js.map +1 -0
- package/dist/src/manager/notes/link-resolver.js +96 -0
- package/dist/src/manager/notes/link-resolver.js.map +1 -0
- package/dist/src/manager/notes/path-guards.js +4 -0
- package/dist/src/manager/notes/path-guards.js.map +1 -1
- package/dist/src/manager/notes/remote-assets.js +128 -0
- package/dist/src/manager/notes/remote-assets.js.map +1 -0
- package/dist/src/manager/notes/routes.js +63 -13
- package/dist/src/manager/notes/routes.js.map +1 -1
- package/dist/src/manager/notes/store.js +3 -2
- package/dist/src/manager/notes/store.js.map +1 -1
- package/dist/src/manager/notes/system-trash.js +3 -3
- package/dist/src/manager/notes/system-trash.js.map +1 -1
- package/dist/src/manager/notes/vault-index.js +220 -0
- package/dist/src/manager/notes/vault-index.js.map +1 -0
- package/dist/src/manager/notes/watcher.js +27 -0
- package/dist/src/manager/notes/watcher.js.map +1 -0
- package/dist/src/manager/notes/wiki-links.js +109 -0
- package/dist/src/manager/notes/wiki-links.js.map +1 -0
- package/dist/src/manager/process-verify.js +29 -0
- package/dist/src/manager/process-verify.js.map +1 -1
- package/dist/src/manager/profiles.js +3 -2
- package/dist/src/manager/profiles.js.map +1 -1
- package/dist/src/manager/registry.js +60 -58
- package/dist/src/manager/registry.js.map +1 -1
- package/dist/src/manager/routes/electron-metrics.js +22 -22
- package/dist/src/manager/routes/electron-metrics.js.map +1 -1
- package/dist/src/manager/scan.js +2 -2
- package/dist/src/manager/scan.js.map +1 -1
- package/dist/src/manager/schedule/routes.js +24 -23
- package/dist/src/manager/schedule/routes.js.map +1 -1
- package/dist/src/manager/server.js +28 -28
- package/dist/src/manager/server.js.map +1 -1
- package/dist/src/manager/systemd-service.js +1 -1
- package/dist/src/manager/systemd-service.js.map +1 -1
- package/dist/src/memory/bootstrap.js +5 -4
- package/dist/src/memory/bootstrap.js.map +1 -1
- package/dist/src/memory/heartbeat-schedule.js +10 -10
- package/dist/src/memory/heartbeat-schedule.js.map +1 -1
- package/dist/src/memory/heartbeat.js +23 -22
- package/dist/src/memory/heartbeat.js.map +1 -1
- package/dist/src/memory/indexing.js.map +1 -1
- package/dist/src/memory/keyword-expand.js +1 -1
- package/dist/src/memory/keyword-expand.js.map +1 -1
- package/dist/src/memory/memory.js +2 -2
- package/dist/src/memory/memory.js.map +1 -1
- package/dist/src/memory/runtime.js +6 -5
- package/dist/src/memory/runtime.js.map +1 -1
- package/dist/src/memory/shared.js +2 -1
- package/dist/src/memory/shared.js.map +1 -1
- package/dist/src/memory/worklog.js +1 -1
- package/dist/src/memory/worklog.js.map +1 -1
- package/dist/src/messaging/runtime.js +13 -10
- package/dist/src/messaging/runtime.js.map +1 -1
- package/dist/src/messaging/send.js +16 -15
- package/dist/src/messaging/send.js.map +1 -1
- package/dist/src/orchestrator/collect.js +6 -6
- package/dist/src/orchestrator/collect.js.map +1 -1
- package/dist/src/orchestrator/distribute.js +112 -77
- package/dist/src/orchestrator/distribute.js.map +1 -1
- package/dist/src/orchestrator/gateway.js +6 -5
- package/dist/src/orchestrator/gateway.js.map +1 -1
- package/dist/src/orchestrator/pipeline.js +42 -42
- package/dist/src/orchestrator/pipeline.js.map +1 -1
- package/dist/src/orchestrator/state-machine.js +3 -3
- package/dist/src/orchestrator/state-machine.js.map +1 -1
- package/dist/src/orchestrator/worker-registry.js +4 -3
- package/dist/src/orchestrator/worker-registry.js.map +1 -1
- package/dist/src/prompt/builder.js +24 -19
- package/dist/src/prompt/builder.js.map +1 -1
- package/dist/src/prompt/templates/a1-system.md +15 -10
- package/dist/src/prompt/templates/control-system.md +9 -7
- package/dist/src/prompt/templates/employee.md +5 -4
- package/dist/src/prompt/templates/vision-click.md +1 -1
- package/dist/src/routes/_http-error.js +15 -0
- package/dist/src/routes/_http-error.js.map +1 -0
- package/dist/src/routes/avatar.js +6 -5
- package/dist/src/routes/avatar.js.map +1 -1
- package/dist/src/routes/browser.js +132 -45
- package/dist/src/routes/browser.js.map +1 -1
- package/dist/src/routes/employees.js +28 -21
- package/dist/src/routes/employees.js.map +1 -1
- package/dist/src/routes/heartbeat.js +12 -9
- package/dist/src/routes/heartbeat.js.map +1 -1
- package/dist/src/routes/i18n.js +13 -7
- package/dist/src/routes/i18n.js.map +1 -1
- package/dist/src/routes/jaw-memory.js +15 -11
- package/dist/src/routes/jaw-memory.js.map +1 -1
- package/dist/src/routes/memory.js +25 -20
- package/dist/src/routes/memory.js.map +1 -1
- package/dist/src/routes/messaging.js +50 -29
- package/dist/src/routes/messaging.js.map +1 -1
- package/dist/src/routes/orchestrate.js +38 -24
- package/dist/src/routes/orchestrate.js.map +1 -1
- package/dist/src/routes/quota.js +81 -17
- package/dist/src/routes/quota.js.map +1 -1
- package/dist/src/routes/settings.js +28 -18
- package/dist/src/routes/settings.js.map +1 -1
- package/dist/src/routes/skills.js +22 -13
- package/dist/src/routes/skills.js.map +1 -1
- package/dist/src/routes/traces.js +77 -0
- package/dist/src/routes/traces.js.map +1 -0
- package/dist/src/security/path-guards.js +2 -1
- package/dist/src/security/path-guards.js.map +1 -1
- package/dist/src/shared/tool-log-sanitize.js +191 -0
- package/dist/src/shared/tool-log-sanitize.js.map +1 -0
- package/dist/src/telegram/bot.js +65 -55
- package/dist/src/telegram/bot.js.map +1 -1
- package/dist/src/telegram/forwarder.js +5 -9
- package/dist/src/telegram/forwarder.js.map +1 -1
- package/dist/src/telegram/telegram-file.js +28 -24
- package/dist/src/telegram/telegram-file.js.map +1 -1
- package/dist/src/telegram/voice.js +4 -3
- package/dist/src/telegram/voice.js.map +1 -1
- package/dist/src/trace/redact.js +50 -0
- package/dist/src/trace/redact.js.map +1 -0
- package/dist/src/trace/store.js +162 -0
- package/dist/src/trace/store.js.map +1 -0
- package/dist/src/trace/types.js +2 -0
- package/dist/src/trace/types.js.map +1 -0
- package/dist/src/types/cli-engine.js +34 -0
- package/dist/src/types/cli-engine.js.map +1 -0
- package/dist/src/types/cli-events.js +31 -0
- package/dist/src/types/cli-events.js.map +1 -0
- package/package.json +22 -2
- package/public/css/tool-ui.css +3 -1
- package/public/css/trace-drawer.css +48 -0
- package/public/dist/assets/{AdvancedExport-3WAYIabE.js → AdvancedExport-DJZ2VmBR.js} +1 -1
- package/public/dist/assets/Agent-CgpLT8IY.js +1 -0
- package/public/dist/assets/Browser-CrkiQoB8.js +1 -0
- package/public/dist/assets/{ChannelsDiscord-UzFPlWT4.js → ChannelsDiscord-CVUC22D4.js} +1 -1
- package/public/dist/assets/{ChannelsTelegram-DNWtPX0w.js → ChannelsTelegram-DEatIQNM.js} +1 -1
- package/public/dist/assets/{DashboardMeta-Y_6nVeJO.js → DashboardMeta-BKoxRc7i.js} +1 -1
- package/public/dist/assets/{Display-D8vGOl4s.js → Display-DnNGV9Km.js} +1 -1
- package/public/dist/assets/{Employees-YR_sIRK4.js → Employees-DZ2iJYKy.js} +1 -1
- package/public/dist/assets/{HealthBadge-CPePajyU.js → HealthBadge-Cq2c7G9s.js} +1 -1
- package/public/dist/assets/{Heartbeat-DTpAULQR.js → Heartbeat-BML6eTXZ.js} +1 -1
- package/public/dist/assets/{Mcp-DM-PgG6z.js → Mcp-BlEviQ3h.js} +1 -1
- package/public/dist/assets/{Memory-C_LvJnkn.js → Memory-BRyH80He.js} +1 -1
- package/public/dist/assets/MilkdownWysiwygEditor-Cm3uXfWf.js +52 -0
- package/public/dist/assets/ModelProvider-DxyR7EL9.js +1 -0
- package/public/dist/assets/Network-DDOOESh1.js +1 -0
- package/public/dist/assets/{Permissions-B1naJjjw.js → Permissions-Br0eSbKb.js} +1 -1
- package/public/dist/assets/{Permissions-BKffrMJD.js → Permissions-QHkzStqQ.js} +1 -1
- package/public/dist/assets/{Profile-DIqjSe2C.js → Profile-C79NKumk.js} +1 -1
- package/public/dist/assets/{Prompts-BMfbV6Y4.js → Prompts-BmiIDiXW.js} +1 -1
- package/public/dist/assets/{SpeechKeys-DjiQTzSL.js → SpeechKeys-B8304XJK.js} +1 -1
- package/public/dist/assets/app-Be58Cs3Y.js +32 -0
- package/public/dist/assets/{app-BzvwJJiv.css → app-CphocJzo.css} +1 -1
- package/public/dist/assets/{dist-DTvxN3ux.js → dist-BD0UXfgF2.js} +1 -1
- package/public/dist/assets/{dist-CASeq-Tl.js → dist-BNfXO3Yr.js} +1 -1
- package/public/dist/assets/{dist-BMPiaUzF.js → dist-BUnPYbK3.js} +1 -1
- package/public/dist/assets/{dist-CT_X3hVT.js → dist-BZosRD2u.js} +1 -1
- package/public/dist/assets/{dist-4J6YbNXv.js → dist-Bd9PlnQm.js} +1 -1
- package/public/dist/assets/{dist-BcZjyn5g.js → dist-BsT5UdNP.js} +1 -1
- package/public/dist/assets/{dist-BfBhOPR-.js → dist-CIuXW-sc.js} +1 -1
- package/public/dist/assets/{dist-BMiCig3A2.js → dist-CL4PTYWf.js} +1 -1
- package/public/dist/assets/{dist-VyP6-HLb.js → dist-Ch5VAlV9.js} +1 -1
- package/public/dist/assets/dist-ClqO40BE.js +1 -0
- package/public/dist/assets/{dist-c98Tp7bP.js → dist-Cp42cMcI.js} +1 -1
- package/public/dist/assets/{dist-CIlYL1qe.js → dist-CpUK8ypo.js} +1 -1
- package/public/dist/assets/dist-CxeLAw2Y.js +1 -0
- package/public/dist/assets/dist-D2SH8nxa.js +1 -0
- package/public/dist/assets/{dist-84fwQ7bO.js → dist-D6cUXP7K.js} +1 -1
- package/public/dist/assets/{dist-BOCcQAyF.js → dist-D7bCuS3f.js} +1 -1
- package/public/dist/assets/{dist-DMmpfLVP.js → dist-DFYRUAjN.js} +1 -1
- package/public/dist/assets/{dist-DdY6pTJr.js → dist-DZsFVYF4.js} +1 -1
- package/public/dist/assets/{dist-B0p3Eyme.js → dist-Db16ogVk.js} +1 -1
- package/public/dist/assets/{dist-DlnNtr6L.js → dist-DfodGES_.js} +1 -1
- package/public/dist/assets/{dist-DO9so2a2.js → dist-SU-YTAIg.js} +1 -1
- package/public/dist/assets/{dist-DUjXiMLP.js → dist-UYn7T-GH.js} +1 -1
- package/public/dist/assets/{dist-BW1409rz.js → dist-W51oDoeA.js} +1 -1
- package/public/dist/assets/{dist-BimBQZx1.js → dist-eU7TyC86.js} +1 -1
- package/public/dist/assets/dist-l9HH00ip.js +1 -0
- package/public/dist/assets/{dist-BrOPNxdH.js → dist-urPTQzXL.js} +1 -1
- package/public/dist/assets/{dist-AloEV3J52.js → dist-yHP6L0Ty.js} +1 -1
- package/public/dist/assets/{employees-Bbabvbyx.js → employees-CxdghzoD.js} +7 -7
- package/public/dist/assets/{error-normalize-DdvKGLt_.js → error-normalize-5n-zlEQ3.js} +1 -1
- package/public/dist/assets/insert-image-markdown-DIEa-zjk.js +22 -0
- package/public/dist/assets/{manager-DyB2ZUr9.css → manager-DEiyrWDP.css} +1 -1
- package/public/dist/assets/manager-UEXd1_9T.js +25 -0
- package/public/dist/assets/{memory-B_plvcuQ.js → memory-CsMNkYtv.js} +9 -9
- package/public/dist/assets/memory-DXad_DPO.js +1 -0
- package/public/dist/assets/{page-shell-DML_HneX.js → page-shell-D5tbivHH.js} +1 -1
- package/public/dist/assets/{render-DEhbfUAW.js → render-DGQX46ei.js} +2 -2
- package/public/dist/assets/{settings-D7F-_kYG.js → settings-BH213Yv3.js} +14 -14
- package/public/dist/assets/settings-DXT87G2U.js +1 -0
- package/public/dist/assets/settings-client-ajlwI-oK.js +1 -0
- package/public/dist/assets/skills-5o_1v0nz.js +1 -0
- package/public/dist/assets/{skills-DoGJOc0D.js → skills-CQtCtHPA.js} +6 -6
- package/public/dist/assets/slash-commands-D4-hrrmh.js +1 -0
- package/public/dist/assets/{slash-commands-khNFPOyF.js → slash-commands-Dzk1xHWS.js} +1 -1
- package/public/dist/assets/trace-drawer-SRKcfm2S.js +15 -0
- package/public/dist/assets/ui-CdRKN2S6.js +141 -0
- package/public/dist/assets/ui-n43jmg_f.js +1 -0
- package/public/dist/assets/{ws-B2aJ-nD2.js → ws-CTHQFzM1.js} +8 -8
- package/public/dist/index.html +2 -2
- package/public/dist/manager/index.html +2 -2
- package/public/index.html +1 -0
- package/public/js/constants.ts +5 -5
- package/public/js/diagram/iframe-renderer.ts +11 -11
- package/public/js/features/chat.ts +1 -1
- package/public/js/features/memory.ts +10 -10
- package/public/js/features/pending-queue.ts +2 -2
- package/public/js/features/process-block.ts +257 -29
- package/public/js/features/process-step-match.ts +18 -0
- package/public/js/features/settings-cli-status.ts +1 -1
- package/public/js/features/settings-stt.ts +37 -11
- package/public/js/features/settings-templates.ts +2 -2
- package/public/js/features/slash-commands.ts +1 -1
- package/public/js/features/tool-ui.ts +13 -7
- package/public/js/features/trace-drawer.ts +122 -0
- package/public/js/icons.ts +1 -1
- package/public/js/main.ts +28 -28
- package/public/js/provider-icons.ts +1 -1
- package/public/js/render.ts +29 -27
- package/public/js/sanitizer.ts +4 -0
- package/public/js/ui.ts +234 -68
- package/public/js/virtual-scroll.ts +19 -33
- package/public/js/ws.ts +22 -10
- package/public/manager/src/App.tsx +34 -6
- package/public/manager/src/api.ts +55 -1
- package/public/manager/src/components/ActivityTimeline.tsx +1 -1
- package/public/manager/src/components/InstanceDetailPanel.tsx +2 -2
- package/public/manager/src/components/InstanceGroups.tsx +5 -5
- package/public/manager/src/components/InstanceListContent.tsx +1 -1
- package/public/manager/src/dashboard-features.ts +1 -1
- package/public/manager/src/manager-notes.css +49 -2
- package/public/manager/src/notes/MarkdownEditor.tsx +7 -3
- package/public/manager/src/notes/NotesFileTree.tsx +45 -6
- package/public/manager/src/notes/NotesSidebar.tsx +24 -55
- package/public/manager/src/notes/NotesWorkspace.tsx +50 -3
- package/public/manager/src/notes/image-assets/clipboard-images.ts +100 -0
- package/public/manager/src/notes/image-assets/insert-image-markdown.ts +85 -0
- package/public/manager/src/notes/notes-api.ts +2 -0
- package/public/manager/src/notes/notes-types.ts +8 -1
- package/public/manager/src/notes/rendering/MarkdownRenderer.tsx +6 -0
- package/public/manager/src/notes/rendering/markdown-render-security.ts +19 -1
- package/public/manager/src/notes/rich-markdown/paste-policy.ts +6 -2
- package/public/manager/src/notes/rich-markdown/rich-markdown-extension.ts +5 -5
- package/public/manager/src/notes/rich-markdown/rich-widget.ts +8 -8
- package/public/manager/src/notes/useNotesExternalSync.ts +37 -0
- package/public/manager/src/notes/useNotesModel.ts +91 -0
- package/public/manager/src/notes/wysiwyg/MilkdownWysiwygEditor.tsx +157 -17
- package/public/manager/src/notes/wysiwyg/milkdown-block-keymap.ts +14 -2
- package/public/manager/src/notes/wysiwyg/milkdown-code-block-view.ts +20 -20
- package/public/manager/src/notes/wysiwyg/milkdown-heading-source-view.ts +8 -8
- package/public/manager/src/notes/wysiwyg/milkdown-math.ts +25 -25
- package/public/manager/src/settings/pages/Browser.tsx +38 -20
- package/public/manager/src/settings/pages/ModelProvider.tsx +11 -9
- package/public/manager/src/settings/pages/Network.tsx +10 -10
- package/public/manager/src/settings/pages/components/HealthBadge.tsx +12 -12
- package/public/manager/src/settings/pages/components/agent/runtime-employees-helpers.ts +12 -10
- package/public/manager/src/settings/pages/components/employees-helpers.ts +9 -9
- package/public/manager/src/settings/pages/components/heartbeat-helpers.ts +13 -13
- package/public/manager/src/settings/pages/components/memory-helpers.ts +4 -4
- package/public/manager/src/settings/pages/mcp-helpers.ts +7 -7
- package/public/manager/src/settings/settings-client.ts +4 -3
- package/public/manager/src/sync/IframeBridge.tsx +29 -0
- package/public/manager/src/sync/VisibilityBridge.tsx +24 -0
- package/public/manager/src/sync/invalidation-bus.ts +30 -0
- package/public/manager/src/sync/useInvalidationSubscription.ts +19 -0
- package/public/manager/src/types.ts +101 -0
- package/scripts/check-strict-baseline.mjs +255 -0
- package/scripts/claim-audit.mjs +159 -0
- package/scripts/i18n-registry.ts +4 -4
- package/scripts/install-officecli.sh +2 -2
- package/scripts/install-wsl.sh +2 -1
- package/scripts/release-gates.mjs +347 -0
- package/scripts/release-preview.sh +3 -0
- package/scripts/release.sh +4 -0
- package/scripts/smoke/opencode-external-dir-smoke.ts +14 -5
- package/public/dist/assets/Agent-BYdzZwD0.js +0 -1
- package/public/dist/assets/Browser-CkGeczuN.js +0 -1
- package/public/dist/assets/MilkdownWysiwygEditor-B7k9bAey.js +0 -52
- package/public/dist/assets/ModelProvider-CX3Qhowu.js +0 -1
- package/public/dist/assets/Network-DfPLFAvw.js +0 -1
- package/public/dist/assets/app-DLYoRkU9.js +0 -32
- package/public/dist/assets/dist-8zNAQAIa.js +0 -1
- package/public/dist/assets/dist-BgeY6nvK.js +0 -1
- package/public/dist/assets/dist-BzDGGxHQ.js +0 -1
- package/public/dist/assets/dist-D3YKbVi-.js +0 -1
- package/public/dist/assets/manager-CUSgFbMO.js +0 -25
- package/public/dist/assets/markdown-render-security-DJfJPWO-.js +0 -22
- package/public/dist/assets/memory-DQ6dU0qs.js +0 -1
- package/public/dist/assets/settings-DxLPUbpj.js +0 -1
- package/public/dist/assets/settings-client-CGf3uPPf.js +0 -1
- package/public/dist/assets/skills-yMfNYJ8m.js +0 -1
- package/public/dist/assets/slash-commands-CZcwr1W6.js +0 -1
- package/public/dist/assets/ui-04YlOMgn.js +0 -136
- package/public/dist/assets/ui-D6hlMjRq.js +0 -1
- /package/public/dist/assets/{InlineWarn-CqgWEC41.js → InlineWarn-BooBRm7o.js} +0 -0
- /package/public/dist/assets/{agent-meta-puNn13DV.js → agent-meta-DHddpWHQ.js} +0 -0
- /package/public/dist/assets/{fields-DH9JS3mb.js → fields-BtncIZYA.js} +0 -0
- /package/public/dist/assets/{mermaid-loader-6XC0y10y.js → mermaid-loader-BEFIOoJn.js} +0 -0
- /package/public/dist/assets/{path-utils-CtsDDGZg.js → path-utils-BuEEtj9w.js} +0 -0
- /package/public/dist/assets/{w3c-keyname-VSld09PZ.js → w3c-keyname-IiiZScED.js} +0 -0
|
@@ -55,31 +55,31 @@ function safeJob(raw: unknown, index: number): HbJob | null {
|
|
|
55
55
|
if (!raw || typeof raw !== 'object') return null;
|
|
56
56
|
const r = raw as Record<string, unknown>;
|
|
57
57
|
const id =
|
|
58
|
-
typeof r
|
|
59
|
-
? r
|
|
58
|
+
typeof r['id'] === 'string' && r['id'].trim()
|
|
59
|
+
? r['id'].trim()
|
|
60
60
|
: `hb_unknown_${index}`;
|
|
61
|
-
const name = typeof r
|
|
62
|
-
const enabled = r
|
|
63
|
-
const prompt = typeof r
|
|
64
|
-
const sched = r
|
|
61
|
+
const name = typeof r['name'] === 'string' ? r['name'] : '';
|
|
62
|
+
const enabled = r['enabled'] !== false;
|
|
63
|
+
const prompt = typeof r['prompt'] === 'string' ? r['prompt'] : '';
|
|
64
|
+
const sched = r['schedule'];
|
|
65
65
|
let schedule: HbSchedule;
|
|
66
66
|
if (sched && typeof sched === 'object') {
|
|
67
67
|
const s = sched as Record<string, unknown>;
|
|
68
|
-
if (s
|
|
68
|
+
if (s['kind'] === 'cron' && typeof s['cron'] === 'string') {
|
|
69
69
|
schedule = {
|
|
70
70
|
kind: 'cron',
|
|
71
|
-
cron: s
|
|
72
|
-
...(typeof s
|
|
73
|
-
? { timeZone: s
|
|
71
|
+
cron: s['cron'],
|
|
72
|
+
...(typeof s['timeZone'] === 'string' && s['timeZone']
|
|
73
|
+
? { timeZone: s['timeZone'] }
|
|
74
74
|
: {}),
|
|
75
75
|
};
|
|
76
76
|
} else {
|
|
77
|
-
const minutes = typeof s
|
|
77
|
+
const minutes = typeof s['minutes'] === 'number' ? s['minutes'] : 30;
|
|
78
78
|
schedule = {
|
|
79
79
|
kind: 'every',
|
|
80
80
|
minutes,
|
|
81
|
-
...(typeof s
|
|
82
|
-
? { timeZone: s
|
|
81
|
+
...(typeof s['timeZone'] === 'string' && s['timeZone']
|
|
82
|
+
? { timeZone: s['timeZone'] }
|
|
83
83
|
: {}),
|
|
84
84
|
};
|
|
85
85
|
}
|
|
@@ -62,17 +62,17 @@ function unwrapOkData(payload: unknown): unknown {
|
|
|
62
62
|
if (!payload || typeof payload !== 'object') return payload;
|
|
63
63
|
const obj = payload as Record<string, unknown>;
|
|
64
64
|
if (Array.isArray(obj)) return obj;
|
|
65
|
-
if ('data' in obj) return obj
|
|
65
|
+
if ('data' in obj) return obj['data'];
|
|
66
66
|
return payload;
|
|
67
67
|
}
|
|
68
68
|
|
|
69
69
|
function safeMemoryEntry(raw: unknown): MemoryEntry | null {
|
|
70
70
|
if (!raw || typeof raw !== 'object') return null;
|
|
71
71
|
const r = raw as Record<string, unknown>;
|
|
72
|
-
const key = typeof r
|
|
72
|
+
const key = typeof r['key'] === 'string' ? r['key'] : null;
|
|
73
73
|
if (!key) return null;
|
|
74
|
-
const value = typeof r
|
|
75
|
-
const source = typeof r
|
|
74
|
+
const value = typeof r['value'] === 'string' ? r['value'] : '';
|
|
75
|
+
const source = typeof r['source'] === 'string' ? r['source'] : 'manual';
|
|
76
76
|
return { key, value, source };
|
|
77
77
|
}
|
|
78
78
|
|
|
@@ -33,7 +33,7 @@ export function isValidServerName(name: string): boolean {
|
|
|
33
33
|
export function normalizeMcpConfig(raw: unknown): McpConfig {
|
|
34
34
|
if (!raw || typeof raw !== 'object') return { servers: {} };
|
|
35
35
|
const obj = raw as Record<string, unknown>;
|
|
36
|
-
const serversRaw = obj
|
|
36
|
+
const serversRaw = obj['servers'];
|
|
37
37
|
const servers: Record<string, McpServer> = {};
|
|
38
38
|
if (serversRaw && typeof serversRaw === 'object' && !Array.isArray(serversRaw)) {
|
|
39
39
|
for (const [name, value] of Object.entries(serversRaw as Record<string, unknown>)) {
|
|
@@ -46,20 +46,20 @@ export function normalizeMcpConfig(raw: unknown): McpConfig {
|
|
|
46
46
|
export function normalizeServer(raw: unknown): McpServer {
|
|
47
47
|
if (!raw || typeof raw !== 'object') return { command: '' };
|
|
48
48
|
const r = raw as Record<string, unknown>;
|
|
49
|
-
const command = typeof r
|
|
50
|
-
const args = Array.isArray(r
|
|
51
|
-
? r
|
|
49
|
+
const command = typeof r['command'] === 'string' ? r['command'] : '';
|
|
50
|
+
const args = Array.isArray(r['args'])
|
|
51
|
+
? r['args'].filter((v): v is string => typeof v === 'string')
|
|
52
52
|
: undefined;
|
|
53
53
|
const env =
|
|
54
|
-
r
|
|
54
|
+
r['env'] && typeof r['env'] === 'object' && !Array.isArray(r['env'])
|
|
55
55
|
? Object.fromEntries(
|
|
56
|
-
Object.entries(r
|
|
56
|
+
Object.entries(r['env'] as Record<string, unknown>).filter(
|
|
57
57
|
(entry): entry is [string, string] =>
|
|
58
58
|
typeof entry[1] === 'string',
|
|
59
59
|
),
|
|
60
60
|
)
|
|
61
61
|
: undefined;
|
|
62
|
-
const autostart = typeof r
|
|
62
|
+
const autostart = typeof r['autostart'] === 'boolean' ? r['autostart'] : undefined;
|
|
63
63
|
const out: McpServer = { command };
|
|
64
64
|
if (args && args.length > 0) out.args = args;
|
|
65
65
|
if (env && Object.keys(env).length > 0) out.env = env;
|
|
@@ -34,13 +34,14 @@ export function createSettingsClient(port: number): SettingsClient {
|
|
|
34
34
|
const controller = new AbortController();
|
|
35
35
|
const timer = setTimeout(() => controller.abort(), TIMEOUT_MS);
|
|
36
36
|
try {
|
|
37
|
-
const
|
|
37
|
+
const fetchInit: RequestInit = {
|
|
38
38
|
method,
|
|
39
39
|
headers,
|
|
40
|
-
body: body === undefined ? undefined : JSON.stringify(body),
|
|
41
40
|
signal: init?.signal || controller.signal,
|
|
42
41
|
...init,
|
|
43
|
-
}
|
|
42
|
+
};
|
|
43
|
+
if (body !== undefined) fetchInit.body = JSON.stringify(body);
|
|
44
|
+
const response = await fetch(`${base}${path}`, fetchInit);
|
|
44
45
|
if (!response.ok) {
|
|
45
46
|
const detail = await response.text().catch(() => '');
|
|
46
47
|
throw new SettingsRequestError(method, path, response.status, detail);
|
|
@@ -0,0 +1,29 @@
|
|
|
1
|
+
import { useEffect } from 'react';
|
|
2
|
+
import { publishInvalidation, type InvalidationTopic } from './invalidation-bus';
|
|
3
|
+
|
|
4
|
+
const VALID_TOPICS = new Set<InvalidationTopic>(['notes', 'instances']);
|
|
5
|
+
|
|
6
|
+
type IframeInvalidationMessage = {
|
|
7
|
+
type: 'dashboard.invalidate';
|
|
8
|
+
topics: InvalidationTopic[];
|
|
9
|
+
reason: string;
|
|
10
|
+
};
|
|
11
|
+
|
|
12
|
+
function isValid(data: unknown): data is IframeInvalidationMessage {
|
|
13
|
+
const d = data as Partial<IframeInvalidationMessage>;
|
|
14
|
+
return d?.type === 'dashboard.invalidate'
|
|
15
|
+
&& Array.isArray(d.topics)
|
|
16
|
+
&& d.topics.every(t => VALID_TOPICS.has(t as InvalidationTopic));
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
export function IframeBridge(): null {
|
|
20
|
+
useEffect(() => {
|
|
21
|
+
function onMessage(event: MessageEvent): void {
|
|
22
|
+
if (!isValid(event.data)) return;
|
|
23
|
+
publishInvalidation({ ...event.data, source: 'iframe' });
|
|
24
|
+
}
|
|
25
|
+
window.addEventListener('message', onMessage);
|
|
26
|
+
return () => window.removeEventListener('message', onMessage);
|
|
27
|
+
}, []);
|
|
28
|
+
return null;
|
|
29
|
+
}
|
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
import { useEffect, useRef } from 'react';
|
|
2
|
+
import { publishInvalidation } from './invalidation-bus';
|
|
3
|
+
|
|
4
|
+
const MIN_STALE_MS = 30_000;
|
|
5
|
+
|
|
6
|
+
export function VisibilityBridge(): null {
|
|
7
|
+
const lastRefreshRef = useRef(Date.now());
|
|
8
|
+
|
|
9
|
+
useEffect(() => {
|
|
10
|
+
function onVisible(): void {
|
|
11
|
+
if (document.visibilityState !== 'visible') return;
|
|
12
|
+
if (Date.now() - lastRefreshRef.current < MIN_STALE_MS) return;
|
|
13
|
+
lastRefreshRef.current = Date.now();
|
|
14
|
+
publishInvalidation({
|
|
15
|
+
topics: ['notes', 'instances'],
|
|
16
|
+
reason: 'visibility:tab-returned',
|
|
17
|
+
source: 'visibility',
|
|
18
|
+
});
|
|
19
|
+
}
|
|
20
|
+
document.addEventListener('visibilitychange', onVisible);
|
|
21
|
+
return () => document.removeEventListener('visibilitychange', onVisible);
|
|
22
|
+
}, []);
|
|
23
|
+
return null;
|
|
24
|
+
}
|
|
@@ -0,0 +1,30 @@
|
|
|
1
|
+
export type InvalidationTopic = 'notes' | 'instances';
|
|
2
|
+
|
|
3
|
+
export type InvalidationEvent = {
|
|
4
|
+
topics: InvalidationTopic[];
|
|
5
|
+
reason: string;
|
|
6
|
+
source?: 'ui' | 'iframe' | 'visibility';
|
|
7
|
+
sourceId?: string;
|
|
8
|
+
};
|
|
9
|
+
|
|
10
|
+
const target = new EventTarget();
|
|
11
|
+
const EVENT_NAME = 'dashboard:invalidate';
|
|
12
|
+
|
|
13
|
+
export function publishInvalidation(event: InvalidationEvent): void {
|
|
14
|
+
target.dispatchEvent(new CustomEvent<InvalidationEvent>(EVENT_NAME, { detail: event }));
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
export function subscribeInvalidation(
|
|
18
|
+
topic: InvalidationTopic,
|
|
19
|
+
callback: (event: InvalidationEvent) => void,
|
|
20
|
+
ignoreSourceId?: string,
|
|
21
|
+
): () => void {
|
|
22
|
+
function handler(e: Event): void {
|
|
23
|
+
const detail = (e as CustomEvent<InvalidationEvent>).detail;
|
|
24
|
+
if (!detail.topics.includes(topic)) return;
|
|
25
|
+
if (ignoreSourceId && detail.sourceId === ignoreSourceId) return;
|
|
26
|
+
callback(detail);
|
|
27
|
+
}
|
|
28
|
+
target.addEventListener(EVENT_NAME, handler);
|
|
29
|
+
return () => target.removeEventListener(EVENT_NAME, handler);
|
|
30
|
+
}
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
import { useEffect, useRef } from 'react';
|
|
2
|
+
import { subscribeInvalidation, type InvalidationTopic } from './invalidation-bus';
|
|
3
|
+
|
|
4
|
+
export function useInvalidationSubscription(
|
|
5
|
+
topic: InvalidationTopic,
|
|
6
|
+
callback: () => void,
|
|
7
|
+
ignoreSourceId?: string,
|
|
8
|
+
): void {
|
|
9
|
+
const callbackRef = useRef(callback);
|
|
10
|
+
callbackRef.current = callback;
|
|
11
|
+
const timerRef = useRef<ReturnType<typeof setTimeout> | undefined>(undefined);
|
|
12
|
+
|
|
13
|
+
useEffect(() => {
|
|
14
|
+
return subscribeInvalidation(topic, () => {
|
|
15
|
+
clearTimeout(timerRef.current);
|
|
16
|
+
timerRef.current = setTimeout(() => callbackRef.current(), 100);
|
|
17
|
+
}, ignoreSourceId);
|
|
18
|
+
}, [topic, ignoreSourceId]);
|
|
19
|
+
}
|
|
@@ -291,6 +291,20 @@ export type DashboardPutNoteRequest = {
|
|
|
291
291
|
baseRevision?: string;
|
|
292
292
|
};
|
|
293
293
|
|
|
294
|
+
export type DashboardNoteAssetUploadRequest = {
|
|
295
|
+
notePath: string;
|
|
296
|
+
mime: string;
|
|
297
|
+
dataBase64: string;
|
|
298
|
+
};
|
|
299
|
+
|
|
300
|
+
export type DashboardNoteAssetResponse = {
|
|
301
|
+
ok: true;
|
|
302
|
+
path: string;
|
|
303
|
+
markdown: string;
|
|
304
|
+
mime: 'image/png' | 'image/jpeg' | 'image/webp' | 'image/gif';
|
|
305
|
+
size: number;
|
|
306
|
+
};
|
|
307
|
+
|
|
294
308
|
export type DashboardTrashNoteKind = 'file' | 'folder';
|
|
295
309
|
|
|
296
310
|
export type DashboardTrashNoteResponse = {
|
|
@@ -299,3 +313,90 @@ export type DashboardTrashNoteResponse = {
|
|
|
299
313
|
deletedTo: 'os-trash' | 'dashboard-trash';
|
|
300
314
|
restoreHint?: string;
|
|
301
315
|
};
|
|
316
|
+
|
|
317
|
+
export type NoteLinkStatus = 'resolved' | 'missing' | 'ambiguous';
|
|
318
|
+
|
|
319
|
+
export type NoteIndexWarningCode =
|
|
320
|
+
| 'frontmatter_parse_error'
|
|
321
|
+
| 'frontmatter_unsupported_value'
|
|
322
|
+
| 'invalid_wikilink_target'
|
|
323
|
+
| 'note_file_too_large'
|
|
324
|
+
| 'note_symlink_skipped';
|
|
325
|
+
|
|
326
|
+
export type NoteIndexWarning = {
|
|
327
|
+
code: NoteIndexWarningCode;
|
|
328
|
+
path: string;
|
|
329
|
+
message: string;
|
|
330
|
+
line?: number;
|
|
331
|
+
column?: number;
|
|
332
|
+
};
|
|
333
|
+
|
|
334
|
+
export type NoteLinkReason = 'not_found' | 'invalid_target' | 'ambiguous';
|
|
335
|
+
|
|
336
|
+
export type NoteLinkRef = {
|
|
337
|
+
sourcePath: string;
|
|
338
|
+
raw: string;
|
|
339
|
+
target: string;
|
|
340
|
+
displayText?: string;
|
|
341
|
+
heading?: string;
|
|
342
|
+
line: number;
|
|
343
|
+
column: number;
|
|
344
|
+
startOffset: number;
|
|
345
|
+
endOffset: number;
|
|
346
|
+
status: NoteLinkStatus;
|
|
347
|
+
resolvedPath?: string;
|
|
348
|
+
candidatePaths?: string[];
|
|
349
|
+
reason?: NoteLinkReason;
|
|
350
|
+
};
|
|
351
|
+
|
|
352
|
+
export type NoteMetadata = {
|
|
353
|
+
path: string;
|
|
354
|
+
title: string;
|
|
355
|
+
aliases: string[];
|
|
356
|
+
tags: string[];
|
|
357
|
+
created?: string;
|
|
358
|
+
mtimeMs: number;
|
|
359
|
+
size: number;
|
|
360
|
+
revision: string;
|
|
361
|
+
frontmatterError?: string;
|
|
362
|
+
};
|
|
363
|
+
|
|
364
|
+
export type NoteGraphNode = {
|
|
365
|
+
id: string;
|
|
366
|
+
title: string;
|
|
367
|
+
kind: 'note' | 'missing' | 'ambiguous';
|
|
368
|
+
path?: string;
|
|
369
|
+
candidatePaths?: string[];
|
|
370
|
+
};
|
|
371
|
+
|
|
372
|
+
export type NoteGraphEdge = {
|
|
373
|
+
source: string;
|
|
374
|
+
target: string;
|
|
375
|
+
raw: string;
|
|
376
|
+
status: NoteLinkStatus;
|
|
377
|
+
resolvedPath?: string;
|
|
378
|
+
};
|
|
379
|
+
|
|
380
|
+
export type VaultIndexSnapshot = {
|
|
381
|
+
version: number;
|
|
382
|
+
notes: NoteMetadata[];
|
|
383
|
+
outgoingLinks: Record<string, NoteLinkRef[]>;
|
|
384
|
+
backlinks: Record<string, NoteLinkRef[]>;
|
|
385
|
+
unresolvedLinks: NoteLinkRef[];
|
|
386
|
+
graph: { nodes: NoteGraphNode[]; edges: NoteGraphEdge[] };
|
|
387
|
+
errors: NoteIndexWarning[];
|
|
388
|
+
};
|
|
389
|
+
|
|
390
|
+
export type NotesCapability = {
|
|
391
|
+
available: boolean;
|
|
392
|
+
version?: string;
|
|
393
|
+
command?: string;
|
|
394
|
+
reason?: string;
|
|
395
|
+
};
|
|
396
|
+
|
|
397
|
+
export type DashboardNotesCapabilities = {
|
|
398
|
+
ripgrep: NotesCapability;
|
|
399
|
+
git: NotesCapability;
|
|
400
|
+
fileWatching: NotesCapability & { provider?: 'fs.watch' | 'watcher' };
|
|
401
|
+
pdf: NotesCapability;
|
|
402
|
+
};
|
|
@@ -0,0 +1,255 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
// @file scripts/check-strict-baseline.mjs
|
|
3
|
+
// Strict-migration regression gate (AST-aware).
|
|
4
|
+
//
|
|
5
|
+
// Per R2.2 (devlog/_plan/strict-migration/07-pro-review-r2.md), this scanner
|
|
6
|
+
// uses the TypeScript Compiler API to walk source files and count AST nodes
|
|
7
|
+
// of kind `AnyKeyword` plus any-typed `TypeReference` (Array<any>, Promise<any>,
|
|
8
|
+
// Record<string, any>, etc.). Comments and string literals are automatically
|
|
9
|
+
// ignored because they are not type-position AST nodes.
|
|
10
|
+
//
|
|
11
|
+
// Two markers per R2.2.1:
|
|
12
|
+
// @strict-debt(P##) — temporary, must be cleared by phase P##
|
|
13
|
+
// @strict-allow-any(<reason>) — permanently allowed, counted separately
|
|
14
|
+
//
|
|
15
|
+
// Reads docs/migration/strict-baseline.md for frozen counts and exits 1 if
|
|
16
|
+
// any tracked directory's live count exceeds the baseline.
|
|
17
|
+
|
|
18
|
+
import { spawnSync } from 'node:child_process';
|
|
19
|
+
import { readFileSync, readdirSync, statSync } from 'node:fs';
|
|
20
|
+
import { join } from 'node:path';
|
|
21
|
+
import { fileURLToPath } from 'node:url';
|
|
22
|
+
import ts from 'typescript';
|
|
23
|
+
|
|
24
|
+
const REPO_ROOT = process.env.STRICT_BASELINE_ROOT || fileURLToPath(new URL('..', import.meta.url));
|
|
25
|
+
const BASELINE_PATH = join(REPO_ROOT, 'docs/migration/strict-baseline.md');
|
|
26
|
+
|
|
27
|
+
const TRACKED_DIRS = ['src', 'bin', 'lib', 'public/js', 'public/manager/src', 'scripts', 'server.ts', 'types'];
|
|
28
|
+
const STRICT_DEBT_SCAN_PATHS = ['src', 'bin', 'lib', 'scripts', 'server.ts', 'public', 'types'];
|
|
29
|
+
const EXCLUDE_DIR_NAMES = new Set([
|
|
30
|
+
'node_modules', 'dist', 'build', '.git', 'coverage', '__snapshots__',
|
|
31
|
+
]);
|
|
32
|
+
|
|
33
|
+
function isTypeScriptFile(file) {
|
|
34
|
+
return file.endsWith('.ts') || file.endsWith('.tsx');
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
function* walk(dir) {
|
|
38
|
+
let entries;
|
|
39
|
+
try { entries = readdirSync(dir, { withFileTypes: true }); }
|
|
40
|
+
catch { return; }
|
|
41
|
+
for (const e of entries) {
|
|
42
|
+
if (EXCLUDE_DIR_NAMES.has(e.name)) continue;
|
|
43
|
+
const full = join(dir, e.name);
|
|
44
|
+
if (e.isDirectory()) yield* walk(full);
|
|
45
|
+
else if (e.isFile() && isTypeScriptFile(full)) yield full;
|
|
46
|
+
}
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
function* filesForPath(absPath) {
|
|
50
|
+
let stat;
|
|
51
|
+
try { stat = statSync(absPath); } catch { return; }
|
|
52
|
+
if (stat.isDirectory()) {
|
|
53
|
+
yield* walk(absPath);
|
|
54
|
+
return;
|
|
55
|
+
}
|
|
56
|
+
if (stat.isFile() && isTypeScriptFile(absPath)) yield absPath;
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
const ANY_TYPE_REF_NAMES = new Set([
|
|
60
|
+
'Array', 'Promise', 'Record', 'ReadonlyArray', 'Map', 'Set', 'Partial',
|
|
61
|
+
'Required', 'Readonly', 'Pick', 'Omit', 'ReadonlyMap', 'ReadonlySet',
|
|
62
|
+
]);
|
|
63
|
+
|
|
64
|
+
function isAnyKeyword(node) {
|
|
65
|
+
return ts.isToken(node) && node.kind === ts.SyntaxKind.AnyKeyword;
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
// Detect type reference whose own type-arguments include `any`, e.g. Record<string, any>.
|
|
69
|
+
function typeArgsContainAny(node) {
|
|
70
|
+
if (!ts.isTypeReferenceNode(node)) return false;
|
|
71
|
+
const args = node.typeArguments;
|
|
72
|
+
if (!args || args.length === 0) return false;
|
|
73
|
+
return args.some((a) => a.kind === ts.SyntaxKind.AnyKeyword);
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
function leadingMarkerOnLine(src, pos) {
|
|
77
|
+
// Look back to the start of the line that contains `pos`, then look at
|
|
78
|
+
// the lines immediately before for `// @strict-debt(...)` or
|
|
79
|
+
// `// @strict-allow-any(...)`. Allow up to 3 lines of preceding context.
|
|
80
|
+
const before = src.slice(0, pos);
|
|
81
|
+
const idx = before.lastIndexOf('\n');
|
|
82
|
+
const lineStart = idx + 1;
|
|
83
|
+
const lineText = src.slice(lineStart, src.indexOf('\n', pos) >= 0 ? src.indexOf('\n', pos) : src.length);
|
|
84
|
+
|
|
85
|
+
if (/@strict-(debt|allow-any)\b/.test(lineText)) {
|
|
86
|
+
return /@strict-debt\b/.test(lineText) ? 'debt' : 'allow';
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
// check up to 3 lines above (whitespace-only between also fine)
|
|
90
|
+
let cursor = lineStart;
|
|
91
|
+
for (let i = 0; i < 3; i++) {
|
|
92
|
+
if (cursor <= 0) break;
|
|
93
|
+
const prevEnd = cursor - 1;
|
|
94
|
+
const prevStart = src.lastIndexOf('\n', prevEnd - 1) + 1;
|
|
95
|
+
const prev = src.slice(prevStart, prevEnd);
|
|
96
|
+
if (/@strict-debt\b/.test(prev)) return 'debt';
|
|
97
|
+
if (/@strict-allow-any\b/.test(prev)) return 'allow';
|
|
98
|
+
if (prev.trim() !== '' && !/^\s*\/\//.test(prev)) break;
|
|
99
|
+
cursor = prevStart;
|
|
100
|
+
}
|
|
101
|
+
return null;
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
function scanText(src, filename = 'inline.ts') {
|
|
105
|
+
const sf = ts.createSourceFile(filename, src, ts.ScriptTarget.Latest, /*setParentNodes*/ true, filename.endsWith('.tsx') ? ts.ScriptKind.TSX : ts.ScriptKind.TS);
|
|
106
|
+
let any = 0;
|
|
107
|
+
let debt = 0;
|
|
108
|
+
let allow = 0;
|
|
109
|
+
function visit(node) {
|
|
110
|
+
if (isAnyKeyword(node)) {
|
|
111
|
+
const marker = leadingMarkerOnLine(src, node.getStart(sf));
|
|
112
|
+
if (marker === 'debt') debt++;
|
|
113
|
+
else if (marker === 'allow') allow++;
|
|
114
|
+
else any++;
|
|
115
|
+
}
|
|
116
|
+
ts.forEachChild(node, visit);
|
|
117
|
+
}
|
|
118
|
+
visit(sf);
|
|
119
|
+
return { any, debt, allow };
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
function scanFile(file) {
|
|
123
|
+
const src = readFileSync(file, 'utf8');
|
|
124
|
+
return scanText(src, file);
|
|
125
|
+
}
|
|
126
|
+
|
|
127
|
+
function countDir(absDir) {
|
|
128
|
+
const result = { any: 0, debt: 0, allow: 0 };
|
|
129
|
+
for (const file of filesForPath(absDir)) {
|
|
130
|
+
const r = scanFile(file);
|
|
131
|
+
result.any += r.any;
|
|
132
|
+
result.debt += r.debt;
|
|
133
|
+
result.allow += r.allow;
|
|
134
|
+
}
|
|
135
|
+
return result;
|
|
136
|
+
}
|
|
137
|
+
|
|
138
|
+
function findStrictDebtMarkers(absDir) {
|
|
139
|
+
const hits = [];
|
|
140
|
+
for (const file of filesForPath(absDir)) {
|
|
141
|
+
const src = readFileSync(file, 'utf8');
|
|
142
|
+
const lines = src.split(/\r?\n/);
|
|
143
|
+
lines.forEach((line, idx) => {
|
|
144
|
+
if (line.includes('@strict-debt')) hits.push(`${file}:${idx + 1}`);
|
|
145
|
+
});
|
|
146
|
+
}
|
|
147
|
+
return hits;
|
|
148
|
+
}
|
|
149
|
+
|
|
150
|
+
function runTypecheck(args) {
|
|
151
|
+
const npx = process.platform === 'win32' ? 'npx.cmd' : 'npx';
|
|
152
|
+
const result = spawnSync(npx, args, {
|
|
153
|
+
cwd: REPO_ROOT,
|
|
154
|
+
encoding: 'utf8',
|
|
155
|
+
stdio: ['ignore', 'pipe', 'pipe'],
|
|
156
|
+
});
|
|
157
|
+
return {
|
|
158
|
+
ok: result.status === 0,
|
|
159
|
+
status: result.status,
|
|
160
|
+
output: `${result.stdout || ''}${result.stderr || ''}`.trim(),
|
|
161
|
+
};
|
|
162
|
+
}
|
|
163
|
+
|
|
164
|
+
function parseBaseline(text) {
|
|
165
|
+
const lines = text.split(/\r?\n/);
|
|
166
|
+
const start = lines.findIndex((l) => /^##\s+any-shapes baseline\b/i.test(l));
|
|
167
|
+
if (start < 0) throw new Error('baseline section "## any-shapes baseline" not found');
|
|
168
|
+
const out = {};
|
|
169
|
+
for (let i = start + 1; i < lines.length; i++) {
|
|
170
|
+
const line = lines[i].trim();
|
|
171
|
+
if (line.startsWith('## ')) break;
|
|
172
|
+
if (!line.startsWith('|')) continue;
|
|
173
|
+
if (/^\|[\s\-:|]+\|$/.test(line)) continue;
|
|
174
|
+
const cells = line.split('|').slice(1, -1).map((c) => c.trim());
|
|
175
|
+
if (cells.length !== 4) continue;
|
|
176
|
+
if (cells[0].toLowerCase() === 'dir') continue;
|
|
177
|
+
const [dir, any, debt, allow] = cells;
|
|
178
|
+
out[dir] = { any: Number(any), debt: Number(debt), allow: Number(allow) };
|
|
179
|
+
}
|
|
180
|
+
if (Object.keys(out).length === 0) {
|
|
181
|
+
throw new Error('baseline table parsed empty — check column count and separator');
|
|
182
|
+
}
|
|
183
|
+
return out;
|
|
184
|
+
}
|
|
185
|
+
|
|
186
|
+
function fmt(dir, c) { return `| ${dir} | ${c.any} | ${c.debt} | ${c.allow} |`; }
|
|
187
|
+
|
|
188
|
+
function main() {
|
|
189
|
+
const baselineText = readFileSync(BASELINE_PATH, 'utf8');
|
|
190
|
+
const baseline = parseBaseline(baselineText);
|
|
191
|
+
|
|
192
|
+
const live = {};
|
|
193
|
+
for (const dir of TRACKED_DIRS) live[dir] = countDir(join(REPO_ROOT, dir));
|
|
194
|
+
|
|
195
|
+
const regressions = [];
|
|
196
|
+
const debtMarkerHits = [];
|
|
197
|
+
for (const dir of TRACKED_DIRS) {
|
|
198
|
+
const b = baseline[dir];
|
|
199
|
+
const l = live[dir];
|
|
200
|
+
if (!b) { regressions.push(`baseline missing for tracked dir "${dir}"`); continue; }
|
|
201
|
+
if (l.any > b.any) regressions.push(`${dir}.any: live=${l.any} > baseline=${b.any} (+${l.any - b.any})`);
|
|
202
|
+
if (l.debt > b.debt) regressions.push(`${dir}.debt: live=${l.debt} > baseline=${b.debt} (+${l.debt - b.debt})`);
|
|
203
|
+
// allow can grow — permanent contracts; only flag if doc forbids
|
|
204
|
+
}
|
|
205
|
+
for (const path of STRICT_DEBT_SCAN_PATHS) {
|
|
206
|
+
debtMarkerHits.push(...findStrictDebtMarkers(join(REPO_ROOT, path)));
|
|
207
|
+
}
|
|
208
|
+
if (debtMarkerHits.length > 0) {
|
|
209
|
+
regressions.push(`@strict-debt markers are forbidden post-P20: ${debtMarkerHits.join(', ')}`);
|
|
210
|
+
}
|
|
211
|
+
|
|
212
|
+
const header = '| dir | any | debt | allow |';
|
|
213
|
+
const sep = '|-----|----:|-----:|------:|';
|
|
214
|
+
console.log('## strict-baseline live snapshot');
|
|
215
|
+
console.log(header); console.log(sep);
|
|
216
|
+
for (const dir of TRACKED_DIRS) console.log(fmt(dir, live[dir]));
|
|
217
|
+
console.log('');
|
|
218
|
+
console.log('## strict-baseline baseline (frozen)');
|
|
219
|
+
console.log(header); console.log(sep);
|
|
220
|
+
for (const dir of TRACKED_DIRS) {
|
|
221
|
+
const b = baseline[dir] ?? { any: '—', debt: '—', allow: '—' };
|
|
222
|
+
console.log(fmt(dir, b));
|
|
223
|
+
}
|
|
224
|
+
console.log('');
|
|
225
|
+
|
|
226
|
+
console.log('## strict-baseline typecheck gates');
|
|
227
|
+
const rootTypecheck = runTypecheck(['tsc', '--noEmit']);
|
|
228
|
+
console.log(`root: ${rootTypecheck.ok ? 'ok' : `fail (${rootTypecheck.status ?? 'unknown'})`}`);
|
|
229
|
+
if (!rootTypecheck.ok) regressions.push(`root typecheck failed:\n${rootTypecheck.output}`);
|
|
230
|
+
const frontendTypecheck = runTypecheck(['tsc', '--noEmit', '-p', 'tsconfig.frontend.json']);
|
|
231
|
+
console.log(`frontend: ${frontendTypecheck.ok ? 'ok' : `fail (${frontendTypecheck.status ?? 'unknown'})`}`);
|
|
232
|
+
if (!frontendTypecheck.ok) regressions.push(`frontend typecheck failed:\n${frontendTypecheck.output}`);
|
|
233
|
+
console.log('');
|
|
234
|
+
|
|
235
|
+
if (regressions.length > 0) {
|
|
236
|
+
console.error('❌ strict-baseline regression detected:');
|
|
237
|
+
for (const r of regressions) console.error(' - ' + r);
|
|
238
|
+
console.error('');
|
|
239
|
+
console.error('If this regression is intentional (a phase plan accepted it),');
|
|
240
|
+
console.error('lower the baseline in docs/migration/strict-baseline.md in the SAME PR.');
|
|
241
|
+
console.error('To temporarily exempt a single occurrence, annotate it with one of:');
|
|
242
|
+
console.error(' // @strict-debt(P##) — must be cleared by phase P##');
|
|
243
|
+
console.error(' // @strict-allow-any(<reason>) — permanent contract');
|
|
244
|
+
process.exit(1);
|
|
245
|
+
}
|
|
246
|
+
console.log('✅ strict-baseline OK (no regressions in tracked directories).');
|
|
247
|
+
}
|
|
248
|
+
|
|
249
|
+
// Run as CLI when executed directly, not when imported by tests.
|
|
250
|
+
if (import.meta.url === `file://${process.argv[1]}`) {
|
|
251
|
+
main();
|
|
252
|
+
}
|
|
253
|
+
|
|
254
|
+
// Exported for unit tests (R2.2.2). Not part of CLI surface.
|
|
255
|
+
export { scanFile, scanText, parseBaseline, countDir, TRACKED_DIRS };
|