@jingyi0605/codingns 0.2.0 → 0.3.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +44 -0
- package/bin/codingns.mjs +918 -55
- package/dist/public/assets/{TerminalPage-BlbQuWi1.js → TerminalPage-Dfw1QUqW.js} +19 -19
- package/dist/public/assets/index-DR2rPNi7.css +1 -0
- package/dist/public/assets/index-DTOruahn.js +114 -0
- package/dist/public/index.html +2 -2
- package/dist/server/config/env.d.ts +3 -0
- package/dist/server/config/env.js +35 -3
- package/dist/server/config/env.js.map +1 -1
- package/dist/server/modules/assistant-capability/assistant-capability-controller.d.ts +89 -0
- package/dist/server/modules/assistant-capability/assistant-capability-controller.js +138 -0
- package/dist/server/modules/assistant-capability/assistant-capability-controller.js.map +1 -0
- package/dist/server/modules/assistant-capability/assistant-capability-service.d.ts +115 -0
- package/dist/server/modules/assistant-capability/assistant-capability-service.js +241 -0
- package/dist/server/modules/assistant-capability/assistant-capability-service.js.map +1 -0
- package/dist/server/modules/butler/butler-control-session-service.d.ts +3 -1
- package/dist/server/modules/butler/butler-control-session-service.js +96 -33
- package/dist/server/modules/butler/butler-control-session-service.js.map +1 -1
- package/dist/server/modules/butler/butler-follow-up-scheduler.d.ts +9 -0
- package/dist/server/modules/butler/butler-follow-up-scheduler.js +47 -11
- package/dist/server/modules/butler/butler-follow-up-scheduler.js.map +1 -1
- package/dist/server/modules/butler/butler-follow-up-service.d.ts +7 -1
- package/dist/server/modules/butler/butler-follow-up-service.js +10 -0
- package/dist/server/modules/butler/butler-follow-up-service.js.map +1 -1
- package/dist/server/modules/butler/butler-session-service.d.ts +3 -1
- package/dist/server/modules/butler/butler-session-service.js +82 -16
- package/dist/server/modules/butler/butler-session-service.js.map +1 -1
- package/dist/server/modules/butler/butler-session-summary-service.d.ts +8 -1
- package/dist/server/modules/butler/butler-session-summary-service.js +34 -7
- package/dist/server/modules/butler/butler-session-summary-service.js.map +1 -1
- package/dist/server/modules/butler/context-aggregator.js +44 -13
- package/dist/server/modules/butler/context-aggregator.js.map +1 -1
- package/dist/server/modules/butler/patrol-scheduler.d.ts +9 -0
- package/dist/server/modules/butler/patrol-scheduler.js +63 -9
- package/dist/server/modules/butler/patrol-scheduler.js.map +1 -1
- package/dist/server/modules/butler/session-summary-scheduler.d.ts +9 -0
- package/dist/server/modules/butler/session-summary-scheduler.js +47 -11
- package/dist/server/modules/butler/session-summary-scheduler.js.map +1 -1
- package/dist/server/modules/debug-target/debug-runtime-reconciliation-scheduler.d.ts +38 -0
- package/dist/server/modules/debug-target/debug-runtime-reconciliation-scheduler.js +99 -0
- package/dist/server/modules/debug-target/debug-runtime-reconciliation-scheduler.js.map +1 -0
- package/dist/server/modules/debug-target/debug-target-controller.d.ts +70 -0
- package/dist/server/modules/debug-target/debug-target-controller.js +113 -0
- package/dist/server/modules/debug-target/debug-target-controller.js.map +1 -0
- package/dist/server/modules/debug-target/debug-target-service.d.ts +105 -0
- package/dist/server/modules/debug-target/debug-target-service.js +1644 -0
- package/dist/server/modules/debug-target/debug-target-service.js.map +1 -0
- package/dist/server/modules/debug-target/framework-compatibility-matrix.d.ts +4 -0
- package/dist/server/modules/debug-target/framework-compatibility-matrix.js +45 -0
- package/dist/server/modules/debug-target/framework-compatibility-matrix.js.map +1 -0
- package/dist/server/modules/debug-target/launch-adapter-registry.d.ts +25 -0
- package/dist/server/modules/debug-target/launch-adapter-registry.js +445 -0
- package/dist/server/modules/debug-target/launch-adapter-registry.js.map +1 -0
- package/dist/server/modules/file/file-constants.d.ts +1 -0
- package/dist/server/modules/file/file-constants.js +1 -0
- package/dist/server/modules/file/file-constants.js.map +1 -1
- package/dist/server/modules/file/file-content-service.d.ts +2 -1
- package/dist/server/modules/file/file-content-service.js +53 -0
- package/dist/server/modules/file/file-content-service.js.map +1 -1
- package/dist/server/modules/file/file-controller.js +12 -3
- package/dist/server/modules/file/file-controller.js.map +1 -1
- package/dist/server/modules/file/file-preview-link-service.js +6 -37
- package/dist/server/modules/file/file-preview-link-service.js.map +1 -1
- package/dist/server/modules/file/file-preview-service.d.ts +6 -12
- package/dist/server/modules/file/file-preview-service.js +114 -28
- package/dist/server/modules/file/file-preview-service.js.map +1 -1
- package/dist/server/modules/file/file-preview-types.d.ts +37 -0
- package/dist/server/modules/file/file-preview-types.js +84 -0
- package/dist/server/modules/file/file-preview-types.js.map +1 -0
- package/dist/server/modules/git/commit-orchestrator.d.ts +4 -1
- package/dist/server/modules/git/commit-orchestrator.js +18 -1
- package/dist/server/modules/git/commit-orchestrator.js.map +1 -1
- package/dist/server/modules/git/git-auth.d.ts +25 -0
- package/dist/server/modules/git/git-auth.js +88 -0
- package/dist/server/modules/git/git-auth.js.map +1 -0
- package/dist/server/modules/git/git-controller.d.ts +12 -0
- package/dist/server/modules/git/git-controller.js +18 -1
- package/dist/server/modules/git/git-controller.js.map +1 -1
- package/dist/server/modules/git/git-read-service.d.ts +3 -1
- package/dist/server/modules/git/git-read-service.js +119 -2
- package/dist/server/modules/git/git-read-service.js.map +1 -1
- package/dist/server/modules/git/git-remote-credential-service.d.ts +9 -0
- package/dist/server/modules/git/git-remote-credential-service.js +76 -0
- package/dist/server/modules/git/git-remote-credential-service.js.map +1 -0
- package/dist/server/modules/git/git-write-service.d.ts +5 -2
- package/dist/server/modules/git/git-write-service.js +33 -17
- package/dist/server/modules/git/git-write-service.js.map +1 -1
- package/dist/server/modules/git/types.d.ts +26 -0
- package/dist/server/modules/git/workspace-repo-guard.js +3 -2
- package/dist/server/modules/git/workspace-repo-guard.js.map +1 -1
- package/dist/server/modules/provider/codex-model-options.d.ts +3 -1
- package/dist/server/modules/provider/codex-model-options.js +4 -1
- package/dist/server/modules/provider/codex-model-options.js.map +1 -1
- package/dist/server/modules/provider/opencode-model-options.d.ts +3 -1
- package/dist/server/modules/provider/opencode-model-options.js +5 -1
- package/dist/server/modules/provider/opencode-model-options.js.map +1 -1
- package/dist/server/modules/provider/provider-discovery-helper-client.d.ts +24 -0
- package/dist/server/modules/provider/provider-discovery-helper-client.js +14 -0
- package/dist/server/modules/provider/provider-discovery-helper-client.js.map +1 -1
- package/dist/server/modules/provider/provider-discovery-helper-process.js +54 -0
- package/dist/server/modules/provider/provider-discovery-helper-process.js.map +1 -1
- package/dist/server/modules/sessions/session-activity-authority-service.js +13 -1
- package/dist/server/modules/sessions/session-activity-authority-service.js.map +1 -1
- package/dist/server/modules/sessions/session-activity-inspector.js +21 -5
- package/dist/server/modules/sessions/session-activity-inspector.js.map +1 -1
- package/dist/server/modules/sessions/session-controller.d.ts +5 -0
- package/dist/server/modules/sessions/session-controller.js +16 -0
- package/dist/server/modules/sessions/session-controller.js.map +1 -1
- package/dist/server/modules/sessions/session-history-service.d.ts +27 -7
- package/dist/server/modules/sessions/session-history-service.js +439 -81
- package/dist/server/modules/sessions/session-history-service.js.map +1 -1
- package/dist/server/modules/sessions/session-live-runtime-service.d.ts +2 -1
- package/dist/server/modules/sessions/session-live-runtime-service.js +12 -0
- package/dist/server/modules/sessions/session-live-runtime-service.js.map +1 -1
- package/dist/server/modules/skills/skill-controller.d.ts +23 -0
- package/dist/server/modules/skills/skill-controller.js +35 -0
- package/dist/server/modules/skills/skill-controller.js.map +1 -0
- package/dist/server/modules/skills/skill-manager-service.d.ts +86 -0
- package/dist/server/modules/skills/skill-manager-service.js +557 -0
- package/dist/server/modules/skills/skill-manager-service.js.map +1 -0
- package/dist/server/modules/skills/skill-reconciler.d.ts +21 -0
- package/dist/server/modules/skills/skill-reconciler.js +99 -0
- package/dist/server/modules/skills/skill-reconciler.js.map +1 -0
- package/dist/server/modules/skills/skill-sync-planner.d.ts +8 -0
- package/dist/server/modules/skills/skill-sync-planner.js +20 -0
- package/dist/server/modules/skills/skill-sync-planner.js.map +1 -0
- package/dist/server/modules/skills/skill-target-adapter.d.ts +34 -0
- package/dist/server/modules/skills/skill-target-adapter.js +65 -0
- package/dist/server/modules/skills/skill-target-adapter.js.map +1 -0
- package/dist/server/modules/tailscale/tailscale-controller.d.ts +15 -0
- package/dist/server/modules/tailscale/tailscale-controller.js +33 -0
- package/dist/server/modules/tailscale/tailscale-controller.js.map +1 -0
- package/dist/server/modules/tailscale/tailscale-helper-client.d.ts +41 -0
- package/dist/server/modules/tailscale/tailscale-helper-client.js +135 -0
- package/dist/server/modules/tailscale/tailscale-helper-client.js.map +1 -0
- package/dist/server/modules/tailscale/tailscale-helper-process.d.ts +1 -0
- package/dist/server/modules/tailscale/tailscale-helper-process.js +327 -0
- package/dist/server/modules/tailscale/tailscale-helper-process.js.map +1 -0
- package/dist/server/modules/tailscale/tailscale-manager.d.ts +41 -0
- package/dist/server/modules/tailscale/tailscale-manager.js +259 -0
- package/dist/server/modules/tailscale/tailscale-manager.js.map +1 -0
- package/dist/server/modules/tailscale/tailscale-service.d.ts +43 -0
- package/dist/server/modules/tailscale/tailscale-service.js +201 -0
- package/dist/server/modules/tailscale/tailscale-service.js.map +1 -0
- package/dist/server/modules/tasks/event-loop-monitor.d.ts +21 -0
- package/dist/server/modules/tasks/event-loop-monitor.js +64 -0
- package/dist/server/modules/tasks/event-loop-monitor.js.map +1 -0
- package/dist/server/modules/tasks/observability-controller.d.ts +30 -0
- package/dist/server/modules/tasks/observability-controller.js +44 -0
- package/dist/server/modules/tasks/observability-controller.js.map +1 -0
- package/dist/server/modules/tasks/observability-service.d.ts +32 -0
- package/dist/server/modules/tasks/observability-service.js +104 -0
- package/dist/server/modules/tasks/observability-service.js.map +1 -0
- package/dist/server/modules/tasks/scheduler-metrics.d.ts +41 -0
- package/dist/server/modules/tasks/scheduler-metrics.js +92 -0
- package/dist/server/modules/tasks/scheduler-metrics.js.map +1 -0
- package/dist/server/modules/tasks/task-activity-log.d.ts +39 -0
- package/dist/server/modules/tasks/task-activity-log.js +43 -0
- package/dist/server/modules/tasks/task-activity-log.js.map +1 -0
- package/dist/server/modules/tasks/task-helper-client.d.ts +11 -0
- package/dist/server/modules/tasks/task-helper-client.js +132 -0
- package/dist/server/modules/tasks/task-helper-client.js.map +1 -0
- package/dist/server/modules/tasks/task-helper-process-handlers.d.ts +16 -0
- package/dist/server/modules/tasks/task-helper-process-handlers.js +14 -0
- package/dist/server/modules/tasks/task-helper-process-handlers.js.map +1 -0
- package/dist/server/modules/tasks/task-helper-process.d.ts +1 -0
- package/dist/server/modules/tasks/task-helper-process.js +49 -0
- package/dist/server/modules/tasks/task-helper-process.js.map +1 -0
- package/dist/server/modules/tasks/task-lane-executors.d.ts +2 -0
- package/dist/server/modules/tasks/task-lane-executors.js +15 -0
- package/dist/server/modules/tasks/task-lane-executors.js.map +1 -0
- package/dist/server/modules/tasks/task-manager.d.ts +15 -0
- package/dist/server/modules/tasks/task-manager.js +36 -0
- package/dist/server/modules/tasks/task-manager.js.map +1 -0
- package/dist/server/modules/tasks/task-metrics.d.ts +9 -0
- package/dist/server/modules/tasks/task-metrics.js +81 -0
- package/dist/server/modules/tasks/task-metrics.js.map +1 -0
- package/dist/server/modules/tasks/task-registry.d.ts +7 -0
- package/dist/server/modules/tasks/task-registry.js +21 -0
- package/dist/server/modules/tasks/task-registry.js.map +1 -0
- package/dist/server/modules/tasks/task-scheduler.d.ts +31 -0
- package/dist/server/modules/tasks/task-scheduler.js +473 -0
- package/dist/server/modules/tasks/task-scheduler.js.map +1 -0
- package/dist/server/modules/tasks/task-types.d.ts +106 -0
- package/dist/server/modules/tasks/task-types.js +23 -0
- package/dist/server/modules/tasks/task-types.js.map +1 -0
- package/dist/server/modules/terminal/command-template-service.d.ts +4 -0
- package/dist/server/modules/terminal/command-template-service.js +5 -3
- package/dist/server/modules/terminal/command-template-service.js.map +1 -1
- package/dist/server/modules/terminal/runtime/terminal-log-spooler.d.ts +7 -3
- package/dist/server/modules/terminal/runtime/terminal-log-spooler.js +95 -15
- package/dist/server/modules/terminal/runtime/terminal-log-spooler.js.map +1 -1
- package/dist/server/modules/terminal/runtime/terminal-log-writer-client.d.ts +21 -0
- package/dist/server/modules/terminal/runtime/terminal-log-writer-client.js +144 -0
- package/dist/server/modules/terminal/runtime/terminal-log-writer-client.js.map +1 -0
- package/dist/server/modules/terminal/runtime/terminal-log-writer-process.d.ts +1 -0
- package/dist/server/modules/terminal/runtime/terminal-log-writer-process.js +187 -0
- package/dist/server/modules/terminal/runtime/terminal-log-writer-process.js.map +1 -0
- package/dist/server/modules/terminal/terminal-service.d.ts +12 -0
- package/dist/server/modules/terminal/terminal-service.js +34 -17
- package/dist/server/modules/terminal/terminal-service.js.map +1 -1
- package/dist/server/modules/workbench/workbench-service.d.ts +23 -2
- package/dist/server/modules/workbench/workbench-service.js +126 -15
- package/dist/server/modules/workbench/workbench-service.js.map +1 -1
- package/dist/server/modules/workbench/workspace-panel-snapshot-service.d.ts +5 -1
- package/dist/server/modules/workbench/workspace-panel-snapshot-service.js +88 -19
- package/dist/server/modules/workbench/workspace-panel-snapshot-service.js.map +1 -1
- package/dist/server/modules/workspace/workspace-code-composition.d.ts +2 -0
- package/dist/server/modules/workspace/workspace-code-composition.js +154 -0
- package/dist/server/modules/workspace/workspace-code-composition.js.map +1 -0
- package/dist/server/modules/workspace/workspace-controller.d.ts +14 -0
- package/dist/server/modules/workspace/workspace-controller.js +19 -0
- package/dist/server/modules/workspace/workspace-controller.js.map +1 -1
- package/dist/server/modules/workspace/workspace-service.d.ts +21 -14
- package/dist/server/modules/workspace/workspace-service.js +183 -234
- package/dist/server/modules/workspace/workspace-service.js.map +1 -1
- package/dist/server/modules/worktree/worktree-cleanup-service.d.ts +35 -0
- package/dist/server/modules/worktree/worktree-cleanup-service.js +210 -0
- package/dist/server/modules/worktree/worktree-cleanup-service.js.map +1 -0
- package/dist/server/modules/worktree/worktree-controller.d.ts +44 -0
- package/dist/server/modules/worktree/worktree-controller.js +40 -0
- package/dist/server/modules/worktree/worktree-controller.js.map +1 -0
- package/dist/server/modules/worktree/worktree-manager.d.ts +34 -0
- package/dist/server/modules/worktree/worktree-manager.js +292 -0
- package/dist/server/modules/worktree/worktree-manager.js.map +1 -0
- package/dist/server/modules/worktree/worktree-merge-service.d.ts +52 -0
- package/dist/server/modules/worktree/worktree-merge-service.js +293 -0
- package/dist/server/modules/worktree/worktree-merge-service.js.map +1 -0
- package/dist/server/modules/worktree/worktree-sync-service.d.ts +23 -0
- package/dist/server/modules/worktree/worktree-sync-service.js +166 -0
- package/dist/server/modules/worktree/worktree-sync-service.js.map +1 -0
- package/dist/server/routes/assistant.d.ts +3 -0
- package/dist/server/routes/assistant.js +15 -0
- package/dist/server/routes/assistant.js.map +1 -0
- package/dist/server/routes/debug-targets.d.ts +3 -0
- package/dist/server/routes/debug-targets.js +15 -0
- package/dist/server/routes/debug-targets.js.map +1 -0
- package/dist/server/routes/git.js +2 -0
- package/dist/server/routes/git.js.map +1 -1
- package/dist/server/routes/observability.d.ts +3 -0
- package/dist/server/routes/observability.js +7 -0
- package/dist/server/routes/observability.js.map +1 -0
- package/dist/server/routes/skills.d.ts +3 -0
- package/dist/server/routes/skills.js +7 -0
- package/dist/server/routes/skills.js.map +1 -0
- package/dist/server/routes/system.d.ts +3 -0
- package/dist/server/routes/system.js +9 -0
- package/dist/server/routes/system.js.map +1 -0
- package/dist/server/routes/workspaces.js +2 -0
- package/dist/server/routes/workspaces.js.map +1 -1
- package/dist/server/routes/worktrees.d.ts +3 -0
- package/dist/server/routes/worktrees.js +8 -0
- package/dist/server/routes/worktrees.js.map +1 -0
- package/dist/server/server/create-server.d.ts +46 -0
- package/dist/server/server/create-server.js +141 -12
- package/dist/server/server/create-server.js.map +1 -1
- package/dist/server/shared/utils/command-availability.d.ts +1 -0
- package/dist/server/shared/utils/command-availability.js +26 -3
- package/dist/server/shared/utils/command-availability.js.map +1 -1
- package/dist/server/shared/utils/secret-box.d.ts +2 -0
- package/dist/server/shared/utils/secret-box.js +29 -0
- package/dist/server/shared/utils/secret-box.js.map +1 -0
- package/dist/server/shared/utils/terminal-debug-log.js +5 -3
- package/dist/server/shared/utils/terminal-debug-log.js.map +1 -1
- package/dist/server/storage/repositories/ai-fallback-edit-repository.d.ts +11 -0
- package/dist/server/storage/repositories/ai-fallback-edit-repository.js +118 -0
- package/dist/server/storage/repositories/ai-fallback-edit-repository.js.map +1 -0
- package/dist/server/storage/repositories/debug-runtime-session-repository.d.ts +11 -0
- package/dist/server/storage/repositories/debug-runtime-session-repository.js +100 -0
- package/dist/server/storage/repositories/debug-runtime-session-repository.js.map +1 -0
- package/dist/server/storage/repositories/debug-service-repository.d.ts +9 -0
- package/dist/server/storage/repositories/debug-service-repository.js +99 -0
- package/dist/server/storage/repositories/debug-service-repository.js.map +1 -0
- package/dist/server/storage/repositories/debug-target-repository.d.ts +11 -0
- package/dist/server/storage/repositories/debug-target-repository.js +100 -0
- package/dist/server/storage/repositories/debug-target-repository.js.map +1 -0
- package/dist/server/storage/repositories/framework-analysis-result-repository.d.ts +9 -0
- package/dist/server/storage/repositories/framework-analysis-result-repository.js +98 -0
- package/dist/server/storage/repositories/framework-analysis-result-repository.js.map +1 -0
- package/dist/server/storage/repositories/git-remote-credential-repository.d.ts +9 -0
- package/dist/server/storage/repositories/git-remote-credential-repository.js +51 -0
- package/dist/server/storage/repositories/git-remote-credential-repository.js.map +1 -0
- package/dist/server/storage/repositories/instance-tailscale-repository.d.ts +10 -0
- package/dist/server/storage/repositories/instance-tailscale-repository.js +112 -0
- package/dist/server/storage/repositories/instance-tailscale-repository.js.map +1 -0
- package/dist/server/storage/repositories/managed-skill-repository.d.ts +11 -0
- package/dist/server/storage/repositories/managed-skill-repository.js +102 -0
- package/dist/server/storage/repositories/managed-skill-repository.js.map +1 -0
- package/dist/server/storage/repositories/port-lease-repository.d.ts +12 -0
- package/dist/server/storage/repositories/port-lease-repository.js +124 -0
- package/dist/server/storage/repositories/port-lease-repository.js.map +1 -0
- package/dist/server/storage/repositories/runtime-binding-repository.d.ts +10 -0
- package/dist/server/storage/repositories/runtime-binding-repository.js +89 -0
- package/dist/server/storage/repositories/runtime-binding-repository.js.map +1 -0
- package/dist/server/storage/repositories/skill-target-binding-repository.d.ts +10 -0
- package/dist/server/storage/repositories/skill-target-binding-repository.js +77 -0
- package/dist/server/storage/repositories/skill-target-binding-repository.js.map +1 -0
- package/dist/server/storage/repositories/terminal-command-template-repository.js +77 -4
- package/dist/server/storage/repositories/terminal-command-template-repository.js.map +1 -1
- package/dist/server/storage/repositories/terminal-instance-repository.js +89 -7
- package/dist/server/storage/repositories/terminal-instance-repository.js.map +1 -1
- package/dist/server/storage/repositories/workspace-navigation-state-repository.d.ts +9 -0
- package/dist/server/storage/repositories/workspace-navigation-state-repository.js +49 -0
- package/dist/server/storage/repositories/workspace-navigation-state-repository.js.map +1 -0
- package/dist/server/storage/repositories/workspace-repository.d.ts +7 -1
- package/dist/server/storage/repositories/workspace-repository.js +32 -8
- package/dist/server/storage/repositories/workspace-repository.js.map +1 -1
- package/dist/server/storage/repositories/workspace-worktree-repository.d.ts +13 -0
- package/dist/server/storage/repositories/workspace-worktree-repository.js +158 -0
- package/dist/server/storage/repositories/workspace-worktree-repository.js.map +1 -0
- package/dist/server/storage/sqlite/client.js +311 -0
- package/dist/server/storage/sqlite/client.js.map +1 -1
- package/dist/server/storage/sqlite/schema.sql +303 -0
- package/dist/server/types/domain.d.ts +303 -0
- package/dist/server/ws/workbench-ws-hub.js +33 -9
- package/dist/server/ws/workbench-ws-hub.js.map +1 -1
- package/dist/server/ws/ws-server.js +4 -4
- package/dist/server/ws/ws-server.js.map +1 -1
- package/node_modules/@codingns/session-sync-core/dist/providers/claude-code.js +18 -6
- package/node_modules/@codingns/session-sync-core/dist/providers/claude-code.js.map +1 -1
- package/node_modules/@codingns/session-sync-core/dist/providers/codex.js +14 -2
- package/node_modules/@codingns/session-sync-core/dist/providers/codex.js.map +1 -1
- package/node_modules/@codingns/session-sync-core/dist/providers/opencode.js +25 -1
- package/node_modules/@codingns/session-sync-core/dist/providers/opencode.js.map +1 -1
- package/node_modules/@codingns/session-sync-core/dist/types.d.ts +6 -0
- package/package.json +1 -1
- package/dist/public/assets/index-1VIm8lVL.css +0 -1
- package/dist/public/assets/index-Dti93O2S.js +0 -109
|
@@ -0,0 +1,43 @@
|
|
|
1
|
+
import type Database from "better-sqlite3";
|
|
2
|
+
import type { InstanceTailscaleRepository } from "../../storage/repositories/instance-tailscale-repository.js";
|
|
3
|
+
import type { TailscalePhase } from "../../types/domain.js";
|
|
4
|
+
import type { TailscaleManager } from "./tailscale-manager.js";
|
|
5
|
+
export interface TailscaleConfigUpdateInput {
|
|
6
|
+
controlServerUrl?: string | null;
|
|
7
|
+
hostname?: string | null;
|
|
8
|
+
}
|
|
9
|
+
export interface InstanceTailscaleStatusDto {
|
|
10
|
+
enabled: boolean;
|
|
11
|
+
controlServerUrl: string | null;
|
|
12
|
+
hostname: string | null;
|
|
13
|
+
phase: TailscalePhase;
|
|
14
|
+
connected: boolean;
|
|
15
|
+
loginUrl: string | null;
|
|
16
|
+
accountName: string | null;
|
|
17
|
+
tailnetFqdn: string | null;
|
|
18
|
+
tailnetIpv4: string | null;
|
|
19
|
+
tailnetIpv6: string | null;
|
|
20
|
+
reachableBaseUrl: string | null;
|
|
21
|
+
lastError: string | null;
|
|
22
|
+
observedAt: string | null;
|
|
23
|
+
updatedAt: string | null;
|
|
24
|
+
}
|
|
25
|
+
export declare class TailscaleService {
|
|
26
|
+
private readonly db;
|
|
27
|
+
private readonly repository;
|
|
28
|
+
private readonly manager;
|
|
29
|
+
private readonly defaultStateDir;
|
|
30
|
+
constructor(db: Database.Database, repository: InstanceTailscaleRepository, manager: TailscaleManager, options: {
|
|
31
|
+
defaultStateDir?: string;
|
|
32
|
+
databasePath: string;
|
|
33
|
+
});
|
|
34
|
+
restoreOnStartup(): Promise<void>;
|
|
35
|
+
getStatus(): Promise<InstanceTailscaleStatusDto>;
|
|
36
|
+
updateConfig(input: TailscaleConfigUpdateInput): Promise<InstanceTailscaleStatusDto>;
|
|
37
|
+
enable(): Promise<InstanceTailscaleStatusDto>;
|
|
38
|
+
disable(): Promise<InstanceTailscaleStatusDto>;
|
|
39
|
+
login(): Promise<InstanceTailscaleStatusDto>;
|
|
40
|
+
logout(): Promise<InstanceTailscaleStatusDto>;
|
|
41
|
+
private readStateSnapshot;
|
|
42
|
+
private buildStatusDto;
|
|
43
|
+
}
|
|
@@ -0,0 +1,201 @@
|
|
|
1
|
+
import path from "node:path";
|
|
2
|
+
import { AppError } from "../../shared/errors/app-error.js";
|
|
3
|
+
import { nowIso } from "../../shared/utils/time.js";
|
|
4
|
+
export class TailscaleService {
|
|
5
|
+
db;
|
|
6
|
+
repository;
|
|
7
|
+
manager;
|
|
8
|
+
defaultStateDir;
|
|
9
|
+
constructor(db, repository, manager, options) {
|
|
10
|
+
this.db = db;
|
|
11
|
+
this.repository = repository;
|
|
12
|
+
this.manager = manager;
|
|
13
|
+
this.defaultStateDir = options.defaultStateDir
|
|
14
|
+
?? path.join(path.dirname(options.databasePath), "tailscale-state");
|
|
15
|
+
}
|
|
16
|
+
async restoreOnStartup() {
|
|
17
|
+
const snapshot = this.readStateSnapshot();
|
|
18
|
+
if (!snapshot.hasPersistedConfig) {
|
|
19
|
+
return;
|
|
20
|
+
}
|
|
21
|
+
if (!snapshot.config.enabled) {
|
|
22
|
+
this.manager.getStatusSync(snapshot.config);
|
|
23
|
+
return;
|
|
24
|
+
}
|
|
25
|
+
await this.manager.restore(snapshot.config);
|
|
26
|
+
}
|
|
27
|
+
async getStatus() {
|
|
28
|
+
const snapshot = this.readStateSnapshot();
|
|
29
|
+
return this.buildStatusDto(snapshot, await this.manager.getStatus(snapshot.config));
|
|
30
|
+
}
|
|
31
|
+
async updateConfig(input) {
|
|
32
|
+
const snapshot = this.readStateSnapshot();
|
|
33
|
+
const timestamp = nowIso();
|
|
34
|
+
const nextConfig = {
|
|
35
|
+
...snapshot.config,
|
|
36
|
+
controlServerUrl: input.controlServerUrl !== undefined
|
|
37
|
+
? normalizeControlServerUrl(input.controlServerUrl)
|
|
38
|
+
: snapshot.config.controlServerUrl,
|
|
39
|
+
hostname: input.hostname !== undefined
|
|
40
|
+
? normalizeHostname(input.hostname)
|
|
41
|
+
: snapshot.config.hostname,
|
|
42
|
+
updatedAt: timestamp
|
|
43
|
+
};
|
|
44
|
+
const transaction = this.db.transaction(() => {
|
|
45
|
+
this.repository.upsertConfig(nextConfig);
|
|
46
|
+
if (nextConfig.enabled) {
|
|
47
|
+
// 这里只同步状态镜像,不主动发起新的登录动作。
|
|
48
|
+
}
|
|
49
|
+
});
|
|
50
|
+
transaction();
|
|
51
|
+
if (nextConfig.enabled) {
|
|
52
|
+
await this.manager.syncConfig(nextConfig);
|
|
53
|
+
}
|
|
54
|
+
return await this.getStatus();
|
|
55
|
+
}
|
|
56
|
+
async enable() {
|
|
57
|
+
const snapshot = this.readStateSnapshot();
|
|
58
|
+
const timestamp = nowIso();
|
|
59
|
+
const nextConfig = {
|
|
60
|
+
...snapshot.config,
|
|
61
|
+
enabled: true,
|
|
62
|
+
updatedAt: timestamp
|
|
63
|
+
};
|
|
64
|
+
this.repository.upsertConfig(nextConfig);
|
|
65
|
+
const status = await this.manager.enable(nextConfig);
|
|
66
|
+
return this.buildStatusDto({
|
|
67
|
+
config: nextConfig,
|
|
68
|
+
hasPersistedConfig: true
|
|
69
|
+
}, status);
|
|
70
|
+
}
|
|
71
|
+
async disable() {
|
|
72
|
+
const snapshot = this.readStateSnapshot();
|
|
73
|
+
const timestamp = nowIso();
|
|
74
|
+
const nextConfig = {
|
|
75
|
+
...snapshot.config,
|
|
76
|
+
enabled: false,
|
|
77
|
+
updatedAt: timestamp
|
|
78
|
+
};
|
|
79
|
+
this.repository.upsertConfig(nextConfig);
|
|
80
|
+
const status = await this.manager.disable(nextConfig);
|
|
81
|
+
return this.buildStatusDto({
|
|
82
|
+
config: nextConfig,
|
|
83
|
+
hasPersistedConfig: true
|
|
84
|
+
}, status);
|
|
85
|
+
}
|
|
86
|
+
async login() {
|
|
87
|
+
const snapshot = this.readStateSnapshot();
|
|
88
|
+
if (!snapshot.config.enabled) {
|
|
89
|
+
throw new AppError({
|
|
90
|
+
statusCode: 409,
|
|
91
|
+
errorCode: "TAILSCALE_NOT_ENABLED",
|
|
92
|
+
detail: "当前实例尚未启用 Tailscale"
|
|
93
|
+
});
|
|
94
|
+
}
|
|
95
|
+
const status = await this.manager.requestLogin(snapshot.config);
|
|
96
|
+
return this.buildStatusDto(snapshot, status);
|
|
97
|
+
}
|
|
98
|
+
async logout() {
|
|
99
|
+
const snapshot = this.readStateSnapshot();
|
|
100
|
+
const status = await this.manager.logout(snapshot.config);
|
|
101
|
+
return this.buildStatusDto(snapshot, status);
|
|
102
|
+
}
|
|
103
|
+
readStateSnapshot() {
|
|
104
|
+
const persistedConfig = this.repository.findConfig();
|
|
105
|
+
return {
|
|
106
|
+
config: persistedConfig
|
|
107
|
+
?? {
|
|
108
|
+
enabled: false,
|
|
109
|
+
controlServerUrl: null,
|
|
110
|
+
hostname: null,
|
|
111
|
+
stateDir: this.defaultStateDir,
|
|
112
|
+
updatedAt: nowIso()
|
|
113
|
+
},
|
|
114
|
+
hasPersistedConfig: persistedConfig !== null
|
|
115
|
+
};
|
|
116
|
+
}
|
|
117
|
+
buildStatusDto(snapshot, effectiveStatus) {
|
|
118
|
+
return {
|
|
119
|
+
enabled: snapshot.config.enabled,
|
|
120
|
+
controlServerUrl: snapshot.config.controlServerUrl,
|
|
121
|
+
hostname: snapshot.config.hostname,
|
|
122
|
+
phase: effectiveStatus.phase,
|
|
123
|
+
connected: effectiveStatus.connected,
|
|
124
|
+
loginUrl: effectiveStatus.loginUrl,
|
|
125
|
+
accountName: effectiveStatus.accountName,
|
|
126
|
+
tailnetFqdn: effectiveStatus.tailnetFqdn,
|
|
127
|
+
tailnetIpv4: effectiveStatus.tailnetIpv4,
|
|
128
|
+
tailnetIpv6: effectiveStatus.tailnetIpv6,
|
|
129
|
+
reachableBaseUrl: effectiveStatus.reachableBaseUrl,
|
|
130
|
+
lastError: effectiveStatus.lastError,
|
|
131
|
+
observedAt: effectiveStatus.observedAt,
|
|
132
|
+
updatedAt: snapshot.hasPersistedConfig ? snapshot.config.updatedAt : null
|
|
133
|
+
};
|
|
134
|
+
}
|
|
135
|
+
}
|
|
136
|
+
function normalizeControlServerUrl(value) {
|
|
137
|
+
if (value === null) {
|
|
138
|
+
return null;
|
|
139
|
+
}
|
|
140
|
+
const normalized = value.trim();
|
|
141
|
+
if (normalized.length === 0) {
|
|
142
|
+
return null;
|
|
143
|
+
}
|
|
144
|
+
let parsed;
|
|
145
|
+
try {
|
|
146
|
+
parsed = new URL(normalized);
|
|
147
|
+
}
|
|
148
|
+
catch {
|
|
149
|
+
throw new AppError({
|
|
150
|
+
statusCode: 400,
|
|
151
|
+
errorCode: "INVALID_INPUT",
|
|
152
|
+
detail: "controlServerUrl 必须是合法的 http 或 https 地址",
|
|
153
|
+
field: "controlServerUrl"
|
|
154
|
+
});
|
|
155
|
+
}
|
|
156
|
+
if (parsed.protocol !== "http:" && parsed.protocol !== "https:") {
|
|
157
|
+
throw new AppError({
|
|
158
|
+
statusCode: 400,
|
|
159
|
+
errorCode: "INVALID_INPUT",
|
|
160
|
+
detail: "controlServerUrl 只允许使用 http 或 https 协议",
|
|
161
|
+
field: "controlServerUrl"
|
|
162
|
+
});
|
|
163
|
+
}
|
|
164
|
+
if (parsed.username || parsed.password || parsed.search || parsed.hash) {
|
|
165
|
+
throw new AppError({
|
|
166
|
+
statusCode: 400,
|
|
167
|
+
errorCode: "INVALID_INPUT",
|
|
168
|
+
detail: "controlServerUrl 不能包含账号、查询参数或 hash",
|
|
169
|
+
field: "controlServerUrl"
|
|
170
|
+
});
|
|
171
|
+
}
|
|
172
|
+
const pathname = parsed.pathname === "/" ? "" : parsed.pathname.replace(/\/+$/, "");
|
|
173
|
+
return `${parsed.protocol}//${parsed.host}${pathname}`;
|
|
174
|
+
}
|
|
175
|
+
function normalizeHostname(value) {
|
|
176
|
+
if (value === null) {
|
|
177
|
+
return null;
|
|
178
|
+
}
|
|
179
|
+
const normalized = value.trim();
|
|
180
|
+
if (normalized.length === 0) {
|
|
181
|
+
return null;
|
|
182
|
+
}
|
|
183
|
+
if (normalized.length > 63) {
|
|
184
|
+
throw new AppError({
|
|
185
|
+
statusCode: 400,
|
|
186
|
+
errorCode: "INVALID_INPUT",
|
|
187
|
+
detail: "hostname 最长不能超过 63 个字符",
|
|
188
|
+
field: "hostname"
|
|
189
|
+
});
|
|
190
|
+
}
|
|
191
|
+
if (!/^[a-zA-Z0-9](?:[a-zA-Z0-9-]*[a-zA-Z0-9])?$/.test(normalized)) {
|
|
192
|
+
throw new AppError({
|
|
193
|
+
statusCode: 400,
|
|
194
|
+
errorCode: "INVALID_INPUT",
|
|
195
|
+
detail: "hostname 只允许字母、数字和连字符,且不能以连字符开头或结尾",
|
|
196
|
+
field: "hostname"
|
|
197
|
+
});
|
|
198
|
+
}
|
|
199
|
+
return normalized;
|
|
200
|
+
}
|
|
201
|
+
//# sourceMappingURL=tailscale-service.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"tailscale-service.js","sourceRoot":"","sources":["../../../../src/modules/tailscale/tailscale-service.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
export interface EventLoopDelaySnapshot {
|
|
2
|
+
readonly enabled: boolean;
|
|
3
|
+
readonly resolutionMs: number;
|
|
4
|
+
readonly minMs: number;
|
|
5
|
+
readonly maxMs: number;
|
|
6
|
+
readonly meanMs: number;
|
|
7
|
+
readonly stddevMs: number;
|
|
8
|
+
readonly p50Ms: number;
|
|
9
|
+
readonly p95Ms: number;
|
|
10
|
+
readonly p99Ms: number;
|
|
11
|
+
}
|
|
12
|
+
export declare class EventLoopMonitor {
|
|
13
|
+
private readonly resolutionMs;
|
|
14
|
+
private histogram;
|
|
15
|
+
constructor(resolutionMs?: number);
|
|
16
|
+
observe(): EventLoopDelaySnapshot;
|
|
17
|
+
start(): void;
|
|
18
|
+
stop(): void;
|
|
19
|
+
reset(): void;
|
|
20
|
+
dispose(): void;
|
|
21
|
+
}
|
|
@@ -0,0 +1,64 @@
|
|
|
1
|
+
import { monitorEventLoopDelay } from "node:perf_hooks";
|
|
2
|
+
export class EventLoopMonitor {
|
|
3
|
+
resolutionMs;
|
|
4
|
+
histogram = null;
|
|
5
|
+
constructor(resolutionMs = 20) {
|
|
6
|
+
this.resolutionMs = resolutionMs;
|
|
7
|
+
// 默认不启动采样,只有显式打开调试观测时才开始收集。
|
|
8
|
+
}
|
|
9
|
+
observe() {
|
|
10
|
+
if (!this.histogram) {
|
|
11
|
+
return {
|
|
12
|
+
enabled: false,
|
|
13
|
+
resolutionMs: this.resolutionMs,
|
|
14
|
+
minMs: 0,
|
|
15
|
+
maxMs: 0,
|
|
16
|
+
meanMs: 0,
|
|
17
|
+
stddevMs: 0,
|
|
18
|
+
p50Ms: 0,
|
|
19
|
+
p95Ms: 0,
|
|
20
|
+
p99Ms: 0
|
|
21
|
+
};
|
|
22
|
+
}
|
|
23
|
+
return {
|
|
24
|
+
enabled: true,
|
|
25
|
+
resolutionMs: this.resolutionMs,
|
|
26
|
+
minMs: toMilliseconds(this.histogram.min),
|
|
27
|
+
maxMs: toMilliseconds(this.histogram.max),
|
|
28
|
+
meanMs: toMilliseconds(this.histogram.mean),
|
|
29
|
+
stddevMs: toMilliseconds(this.histogram.stddev),
|
|
30
|
+
p50Ms: toMilliseconds(this.histogram.percentile(50)),
|
|
31
|
+
p95Ms: toMilliseconds(this.histogram.percentile(95)),
|
|
32
|
+
p99Ms: toMilliseconds(this.histogram.percentile(99))
|
|
33
|
+
};
|
|
34
|
+
}
|
|
35
|
+
start() {
|
|
36
|
+
if (this.histogram) {
|
|
37
|
+
return;
|
|
38
|
+
}
|
|
39
|
+
this.histogram = monitorEventLoopDelay({
|
|
40
|
+
resolution: this.resolutionMs
|
|
41
|
+
});
|
|
42
|
+
this.histogram.enable();
|
|
43
|
+
}
|
|
44
|
+
stop() {
|
|
45
|
+
if (!this.histogram) {
|
|
46
|
+
return;
|
|
47
|
+
}
|
|
48
|
+
this.histogram.disable();
|
|
49
|
+
this.histogram = null;
|
|
50
|
+
}
|
|
51
|
+
reset() {
|
|
52
|
+
this.histogram?.reset();
|
|
53
|
+
}
|
|
54
|
+
dispose() {
|
|
55
|
+
this.stop();
|
|
56
|
+
}
|
|
57
|
+
}
|
|
58
|
+
function toMilliseconds(value) {
|
|
59
|
+
if (!Number.isFinite(value)) {
|
|
60
|
+
return 0;
|
|
61
|
+
}
|
|
62
|
+
return value / 1_000_000;
|
|
63
|
+
}
|
|
64
|
+
//# sourceMappingURL=event-loop-monitor.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"event-loop-monitor.js","sourceRoot":"","sources":["../../../../src/modules/tasks/event-loop-monitor.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,qBAAqB,EAAE,MAAM,iBAAiB,CAAC;AAcxD,MAAM,OAAO,gBAAgB;IAGE;IAFrB,SAAS,GAAoD,IAAI,CAAC;IAE1E,YAA6B,eAAe,EAAE;QAAjB,iBAAY,GAAZ,YAAY,CAAK;QAC5C,4BAA4B;IAC9B,CAAC;IAED,OAAO;QACL,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,CAAC;YACpB,OAAO;gBACL,OAAO,EAAE,KAAK;gBACd,YAAY,EAAE,IAAI,CAAC,YAAY;gBAC/B,KAAK,EAAE,CAAC;gBACR,KAAK,EAAE,CAAC;gBACR,MAAM,EAAE,CAAC;gBACT,QAAQ,EAAE,CAAC;gBACX,KAAK,EAAE,CAAC;gBACR,KAAK,EAAE,CAAC;gBACR,KAAK,EAAE,CAAC;aACT,CAAC;QACJ,CAAC;QAED,OAAO;YACL,OAAO,EAAE,IAAI;YACb,YAAY,EAAE,IAAI,CAAC,YAAY;YAC/B,KAAK,EAAE,cAAc,CAAC,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC;YACzC,KAAK,EAAE,cAAc,CAAC,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC;YACzC,MAAM,EAAE,cAAc,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC;YAC3C,QAAQ,EAAE,cAAc,CAAC,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC;YAC/C,KAAK,EAAE,cAAc,CAAC,IAAI,CAAC,SAAS,CAAC,UAAU,CAAC,EAAE,CAAC,CAAC;YACpD,KAAK,EAAE,cAAc,CAAC,IAAI,CAAC,SAAS,CAAC,UAAU,CAAC,EAAE,CAAC,CAAC;YACpD,KAAK,EAAE,cAAc,CAAC,IAAI,CAAC,SAAS,CAAC,UAAU,CAAC,EAAE,CAAC,CAAC;SACrD,CAAC;IACJ,CAAC;IAED,KAAK;QACH,IAAI,IAAI,CAAC,SAAS,EAAE,CAAC;YACnB,OAAO;QACT,CAAC;QAED,IAAI,CAAC,SAAS,GAAG,qBAAqB,CAAC;YACrC,UAAU,EAAE,IAAI,CAAC,YAAY;SAC9B,CAAC,CAAC;QACH,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,CAAC;IAC1B,CAAC;IAED,IAAI;QACF,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,CAAC;YACpB,OAAO;QACT,CAAC;QAED,IAAI,CAAC,SAAS,CAAC,OAAO,EAAE,CAAC;QACzB,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC;IACxB,CAAC;IAED,KAAK;QACH,IAAI,CAAC,SAAS,EAAE,KAAK,EAAE,CAAC;IAC1B,CAAC;IAED,OAAO;QACL,IAAI,CAAC,IAAI,EAAE,CAAC;IACd,CAAC;CACF;AAED,SAAS,cAAc,CAAC,KAAa;IACnC,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,KAAK,CAAC,EAAE,CAAC;QAC5B,OAAO,CAAC,CAAC;IACX,CAAC;IAED,OAAO,KAAK,GAAG,SAAS,CAAC;AAC3B,CAAC"}
|
|
@@ -0,0 +1,30 @@
|
|
|
1
|
+
import type { FastifyReply, FastifyRequest } from "fastify";
|
|
2
|
+
import type { RuntimeObservabilityService } from "./observability-service.js";
|
|
3
|
+
interface RuntimeObservabilityQuery {
|
|
4
|
+
sessionId?: string;
|
|
5
|
+
activityLimit?: string;
|
|
6
|
+
}
|
|
7
|
+
interface RuntimeObservabilitySessionBody {
|
|
8
|
+
ttlMs?: number;
|
|
9
|
+
}
|
|
10
|
+
interface RuntimeObservabilitySessionParams {
|
|
11
|
+
sessionId: string;
|
|
12
|
+
}
|
|
13
|
+
export declare class ObservabilityController {
|
|
14
|
+
private readonly runtimeObservabilityService;
|
|
15
|
+
constructor(runtimeObservabilityService: RuntimeObservabilityService);
|
|
16
|
+
readonly createRuntimeSession: (request: FastifyRequest<{
|
|
17
|
+
Body: RuntimeObservabilitySessionBody;
|
|
18
|
+
}>, reply: FastifyReply) => Promise<void>;
|
|
19
|
+
readonly heartbeatRuntimeSession: (request: FastifyRequest<{
|
|
20
|
+
Params: RuntimeObservabilitySessionParams;
|
|
21
|
+
Body: RuntimeObservabilitySessionBody;
|
|
22
|
+
}>, reply: FastifyReply) => Promise<void>;
|
|
23
|
+
readonly closeRuntimeSession: (request: FastifyRequest<{
|
|
24
|
+
Params: RuntimeObservabilitySessionParams;
|
|
25
|
+
}>, reply: FastifyReply) => Promise<void>;
|
|
26
|
+
readonly getRuntimeSnapshot: (request: FastifyRequest<{
|
|
27
|
+
Querystring: RuntimeObservabilityQuery;
|
|
28
|
+
}>, reply: FastifyReply) => Promise<void>;
|
|
29
|
+
}
|
|
30
|
+
export {};
|
|
@@ -0,0 +1,44 @@
|
|
|
1
|
+
import { AppError } from "../../shared/errors/app-error.js";
|
|
2
|
+
export class ObservabilityController {
|
|
3
|
+
runtimeObservabilityService;
|
|
4
|
+
constructor(runtimeObservabilityService) {
|
|
5
|
+
this.runtimeObservabilityService = runtimeObservabilityService;
|
|
6
|
+
}
|
|
7
|
+
createRuntimeSession = async (request, reply) => {
|
|
8
|
+
assertAuthenticated(request);
|
|
9
|
+
reply.send(this.runtimeObservabilityService.openSession(request.body?.ttlMs));
|
|
10
|
+
};
|
|
11
|
+
heartbeatRuntimeSession = async (request, reply) => {
|
|
12
|
+
assertAuthenticated(request);
|
|
13
|
+
reply.send(this.runtimeObservabilityService.touchSession(request.params.sessionId, request.body?.ttlMs));
|
|
14
|
+
};
|
|
15
|
+
closeRuntimeSession = async (request, reply) => {
|
|
16
|
+
assertAuthenticated(request);
|
|
17
|
+
this.runtimeObservabilityService.closeSession(request.params.sessionId);
|
|
18
|
+
reply.code(204).send();
|
|
19
|
+
};
|
|
20
|
+
getRuntimeSnapshot = async (request, reply) => {
|
|
21
|
+
assertAuthenticated(request);
|
|
22
|
+
const sessionId = request.query.sessionId?.trim();
|
|
23
|
+
if (!sessionId) {
|
|
24
|
+
throw new AppError({
|
|
25
|
+
statusCode: 400,
|
|
26
|
+
errorCode: "INVALID_INPUT",
|
|
27
|
+
detail: "缺少观测会话 sessionId",
|
|
28
|
+
field: "sessionId"
|
|
29
|
+
});
|
|
30
|
+
}
|
|
31
|
+
const activityLimit = Number.parseInt(request.query.activityLimit ?? "100", 10);
|
|
32
|
+
reply.send(this.runtimeObservabilityService.observe(sessionId, activityLimit));
|
|
33
|
+
};
|
|
34
|
+
}
|
|
35
|
+
function assertAuthenticated(request) {
|
|
36
|
+
if (!request.auth?.user.userId) {
|
|
37
|
+
throw new AppError({
|
|
38
|
+
statusCode: 401,
|
|
39
|
+
errorCode: "UNAUTHORIZED",
|
|
40
|
+
detail: "当前请求缺少有效登录态"
|
|
41
|
+
});
|
|
42
|
+
}
|
|
43
|
+
}
|
|
44
|
+
//# sourceMappingURL=observability-controller.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"observability-controller.js","sourceRoot":"","sources":["../../../../src/modules/tasks/observability-controller.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,QAAQ,EAAE,MAAM,kCAAkC,CAAC;AAgB5D,MAAM,OAAO,uBAAuB;IACL;IAA7B,YAA6B,2BAAwD;QAAxD,gCAA2B,GAA3B,2BAA2B,CAA6B;IAAG,CAAC;IAEhF,oBAAoB,GAAG,KAAK,EACnC,OAAkE,EAClE,KAAmB,EACJ,EAAE;QACjB,mBAAmB,CAAC,OAAO,CAAC,CAAC;QAC7B,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,2BAA2B,CAAC,WAAW,CAAC,OAAO,CAAC,IAAI,EAAE,KAAK,CAAC,CAAC,CAAC;IAChF,CAAC,CAAC;IAEO,uBAAuB,GAAG,KAAK,EACtC,OAGE,EACF,KAAmB,EACJ,EAAE;QACjB,mBAAmB,CAAC,OAAO,CAAC,CAAC;QAC7B,KAAK,CAAC,IAAI,CACR,IAAI,CAAC,2BAA2B,CAAC,YAAY,CAC3C,OAAO,CAAC,MAAM,CAAC,SAAS,EACxB,OAAO,CAAC,IAAI,EAAE,KAAK,CACpB,CACF,CAAC;IACJ,CAAC,CAAC;IAEO,mBAAmB,GAAG,KAAK,EAClC,OAAsE,EACtE,KAAmB,EACJ,EAAE;QACjB,mBAAmB,CAAC,OAAO,CAAC,CAAC;QAC7B,IAAI,CAAC,2BAA2B,CAAC,YAAY,CAAC,OAAO,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC;QACxE,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,IAAI,EAAE,CAAC;IACzB,CAAC,CAAC;IAEO,kBAAkB,GAAG,KAAK,EACjC,OAAmE,EACnE,KAAmB,EACJ,EAAE;QACjB,mBAAmB,CAAC,OAAO,CAAC,CAAC;QAE7B,MAAM,SAAS,GAAG,OAAO,CAAC,KAAK,CAAC,SAAS,EAAE,IAAI,EAAE,CAAC;QAElD,IAAI,CAAC,SAAS,EAAE,CAAC;YACf,MAAM,IAAI,QAAQ,CAAC;gBACjB,UAAU,EAAE,GAAG;gBACf,SAAS,EAAE,eAAe;gBAC1B,MAAM,EAAE,kBAAkB;gBAC1B,KAAK,EAAE,WAAW;aACnB,CAAC,CAAC;QACL,CAAC;QAED,MAAM,aAAa,GAAG,MAAM,CAAC,QAAQ,CAAC,OAAO,CAAC,KAAK,CAAC,aAAa,IAAI,KAAK,EAAE,EAAE,CAAC,CAAC;QAChF,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,2BAA2B,CAAC,OAAO,CAAC,SAAS,EAAE,aAAa,CAAC,CAAC,CAAC;IACjF,CAAC,CAAC;CACH;AAED,SAAS,mBAAmB,CAAC,OAAuB;IAClD,IAAI,CAAC,OAAO,CAAC,IAAI,EAAE,IAAI,CAAC,MAAM,EAAE,CAAC;QAC/B,MAAM,IAAI,QAAQ,CAAC;YACjB,UAAU,EAAE,GAAG;YACf,SAAS,EAAE,cAAc;YACzB,MAAM,EAAE,aAAa;SACtB,CAAC,CAAC;IACL,CAAC;AACH,CAAC"}
|
|
@@ -0,0 +1,32 @@
|
|
|
1
|
+
import type { TaskMetricsSnapshot } from "./task-types.js";
|
|
2
|
+
import type { SchedulerMetricsSnapshot } from "./scheduler-metrics.js";
|
|
3
|
+
import type { EventLoopDelaySnapshot, EventLoopMonitor } from "./event-loop-monitor.js";
|
|
4
|
+
import type { TaskActivityLog, TaskActivityRecord } from "./task-activity-log.js";
|
|
5
|
+
export interface RuntimeObservabilitySessionLease {
|
|
6
|
+
readonly sessionId: string;
|
|
7
|
+
readonly expiresAt: string;
|
|
8
|
+
readonly ttlMs: number;
|
|
9
|
+
}
|
|
10
|
+
export interface RuntimeObservabilitySnapshot {
|
|
11
|
+
readonly observedAt: string;
|
|
12
|
+
readonly session: RuntimeObservabilitySessionLease;
|
|
13
|
+
readonly backgroundTasks: TaskMetricsSnapshot;
|
|
14
|
+
readonly recentTaskActivities: TaskActivityRecord[];
|
|
15
|
+
readonly schedulers: SchedulerMetricsSnapshot;
|
|
16
|
+
readonly eventLoop: EventLoopDelaySnapshot;
|
|
17
|
+
}
|
|
18
|
+
export declare class RuntimeObservabilityService {
|
|
19
|
+
private readonly getTaskMetrics;
|
|
20
|
+
private readonly getSchedulerMetrics;
|
|
21
|
+
private readonly eventLoopMonitor;
|
|
22
|
+
private readonly taskActivityLog;
|
|
23
|
+
private readonly sessions;
|
|
24
|
+
constructor(getTaskMetrics: () => TaskMetricsSnapshot, getSchedulerMetrics: () => SchedulerMetricsSnapshot, eventLoopMonitor: EventLoopMonitor, taskActivityLog: TaskActivityLog);
|
|
25
|
+
hasActiveSession(): boolean;
|
|
26
|
+
openSession(ttlMs?: number): RuntimeObservabilitySessionLease;
|
|
27
|
+
touchSession(sessionId: string, ttlMs?: number): RuntimeObservabilitySessionLease;
|
|
28
|
+
closeSession(sessionId: string): void;
|
|
29
|
+
observe(sessionId: string, activityLimit?: number): RuntimeObservabilitySnapshot;
|
|
30
|
+
private pruneExpiredSessions;
|
|
31
|
+
private syncCollectors;
|
|
32
|
+
}
|
|
@@ -0,0 +1,104 @@
|
|
|
1
|
+
import { createId } from "../../shared/utils/id.js";
|
|
2
|
+
import { nowIso } from "../../shared/utils/time.js";
|
|
3
|
+
import { AppError } from "../../shared/errors/app-error.js";
|
|
4
|
+
const DEFAULT_SESSION_TTL_MS = 20_000;
|
|
5
|
+
const MIN_SESSION_TTL_MS = 5_000;
|
|
6
|
+
const MAX_SESSION_TTL_MS = 120_000;
|
|
7
|
+
export class RuntimeObservabilityService {
|
|
8
|
+
getTaskMetrics;
|
|
9
|
+
getSchedulerMetrics;
|
|
10
|
+
eventLoopMonitor;
|
|
11
|
+
taskActivityLog;
|
|
12
|
+
sessions = new Map();
|
|
13
|
+
constructor(getTaskMetrics, getSchedulerMetrics, eventLoopMonitor, taskActivityLog) {
|
|
14
|
+
this.getTaskMetrics = getTaskMetrics;
|
|
15
|
+
this.getSchedulerMetrics = getSchedulerMetrics;
|
|
16
|
+
this.eventLoopMonitor = eventLoopMonitor;
|
|
17
|
+
this.taskActivityLog = taskActivityLog;
|
|
18
|
+
}
|
|
19
|
+
hasActiveSession() {
|
|
20
|
+
this.pruneExpiredSessions();
|
|
21
|
+
return this.sessions.size > 0;
|
|
22
|
+
}
|
|
23
|
+
openSession(ttlMs) {
|
|
24
|
+
this.pruneExpiredSessions();
|
|
25
|
+
const normalizedTtlMs = normalizeSessionTtlMs(ttlMs);
|
|
26
|
+
const sessionId = createId();
|
|
27
|
+
const expiresAtMs = Date.now() + normalizedTtlMs;
|
|
28
|
+
this.sessions.set(sessionId, {
|
|
29
|
+
expiresAtMs,
|
|
30
|
+
ttlMs: normalizedTtlMs
|
|
31
|
+
});
|
|
32
|
+
this.syncCollectors();
|
|
33
|
+
return buildLease(sessionId, expiresAtMs, normalizedTtlMs);
|
|
34
|
+
}
|
|
35
|
+
touchSession(sessionId, ttlMs) {
|
|
36
|
+
this.pruneExpiredSessions();
|
|
37
|
+
const session = this.sessions.get(sessionId);
|
|
38
|
+
if (!session) {
|
|
39
|
+
throw new AppError({
|
|
40
|
+
statusCode: 404,
|
|
41
|
+
errorCode: "OBSERVABILITY_SESSION_NOT_FOUND",
|
|
42
|
+
detail: "观测会话不存在或已过期"
|
|
43
|
+
});
|
|
44
|
+
}
|
|
45
|
+
const normalizedTtlMs = normalizeSessionTtlMs(ttlMs ?? session.ttlMs);
|
|
46
|
+
session.ttlMs = normalizedTtlMs;
|
|
47
|
+
session.expiresAtMs = Date.now() + normalizedTtlMs;
|
|
48
|
+
this.sessions.set(sessionId, session);
|
|
49
|
+
this.syncCollectors();
|
|
50
|
+
return buildLease(sessionId, session.expiresAtMs, normalizedTtlMs);
|
|
51
|
+
}
|
|
52
|
+
closeSession(sessionId) {
|
|
53
|
+
this.pruneExpiredSessions();
|
|
54
|
+
this.sessions.delete(sessionId);
|
|
55
|
+
this.syncCollectors();
|
|
56
|
+
}
|
|
57
|
+
observe(sessionId, activityLimit = 100) {
|
|
58
|
+
const session = this.touchSession(sessionId);
|
|
59
|
+
return {
|
|
60
|
+
observedAt: nowIso(),
|
|
61
|
+
session,
|
|
62
|
+
backgroundTasks: this.getTaskMetrics(),
|
|
63
|
+
recentTaskActivities: this.taskActivityLog.list(activityLimit),
|
|
64
|
+
schedulers: this.getSchedulerMetrics(),
|
|
65
|
+
eventLoop: this.eventLoopMonitor.observe()
|
|
66
|
+
};
|
|
67
|
+
}
|
|
68
|
+
pruneExpiredSessions() {
|
|
69
|
+
const now = Date.now();
|
|
70
|
+
let changed = false;
|
|
71
|
+
for (const [sessionId, session] of this.sessions.entries()) {
|
|
72
|
+
if (session.expiresAtMs > now) {
|
|
73
|
+
continue;
|
|
74
|
+
}
|
|
75
|
+
this.sessions.delete(sessionId);
|
|
76
|
+
changed = true;
|
|
77
|
+
}
|
|
78
|
+
if (changed) {
|
|
79
|
+
this.syncCollectors();
|
|
80
|
+
}
|
|
81
|
+
}
|
|
82
|
+
syncCollectors() {
|
|
83
|
+
if (this.sessions.size > 0) {
|
|
84
|
+
this.eventLoopMonitor.start();
|
|
85
|
+
return;
|
|
86
|
+
}
|
|
87
|
+
this.taskActivityLog.clear();
|
|
88
|
+
this.eventLoopMonitor.stop();
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
function normalizeSessionTtlMs(value) {
|
|
92
|
+
if (!value || !Number.isFinite(value)) {
|
|
93
|
+
return DEFAULT_SESSION_TTL_MS;
|
|
94
|
+
}
|
|
95
|
+
return Math.min(MAX_SESSION_TTL_MS, Math.max(MIN_SESSION_TTL_MS, Math.floor(value)));
|
|
96
|
+
}
|
|
97
|
+
function buildLease(sessionId, expiresAtMs, ttlMs) {
|
|
98
|
+
return {
|
|
99
|
+
sessionId,
|
|
100
|
+
expiresAt: new Date(expiresAtMs).toISOString(),
|
|
101
|
+
ttlMs
|
|
102
|
+
};
|
|
103
|
+
}
|
|
104
|
+
//# sourceMappingURL=observability-service.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"observability-service.js","sourceRoot":"","sources":["../../../../src/modules/tasks/observability-service.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,41 @@
|
|
|
1
|
+
export interface SchedulerDurationStatsSnapshot {
|
|
2
|
+
readonly count: number;
|
|
3
|
+
readonly total: number;
|
|
4
|
+
readonly max: number;
|
|
5
|
+
readonly min: number | null;
|
|
6
|
+
readonly avg: number;
|
|
7
|
+
}
|
|
8
|
+
export interface SchedulerMetricSnapshot {
|
|
9
|
+
readonly tickTotal: number;
|
|
10
|
+
readonly idleTickTotal: number;
|
|
11
|
+
readonly errorTotal: number;
|
|
12
|
+
readonly taskCountTotal: number;
|
|
13
|
+
readonly durationMs: SchedulerDurationStatsSnapshot;
|
|
14
|
+
readonly lastTickAt: string | null;
|
|
15
|
+
readonly lastDurationMs: number | null;
|
|
16
|
+
readonly lastTaskCount: number;
|
|
17
|
+
readonly lastIdle: boolean;
|
|
18
|
+
readonly lastErrorCount: number;
|
|
19
|
+
readonly nextDelayMs: number | null;
|
|
20
|
+
readonly idleStreak: number;
|
|
21
|
+
}
|
|
22
|
+
export interface SchedulerMetricsSnapshot {
|
|
23
|
+
readonly schedulers: Readonly<Record<string, SchedulerMetricSnapshot>>;
|
|
24
|
+
}
|
|
25
|
+
export interface SchedulerTickObservation {
|
|
26
|
+
readonly schedulerName: string;
|
|
27
|
+
readonly referenceAt: string;
|
|
28
|
+
readonly durationMs: number;
|
|
29
|
+
readonly taskCount: number;
|
|
30
|
+
readonly idle: boolean;
|
|
31
|
+
readonly errorCount?: number;
|
|
32
|
+
readonly nextDelayMs?: number | null;
|
|
33
|
+
readonly idleStreak?: number;
|
|
34
|
+
}
|
|
35
|
+
export declare class SchedulerMetrics {
|
|
36
|
+
private readonly schedulers;
|
|
37
|
+
recordTick(observation: SchedulerTickObservation): void;
|
|
38
|
+
observe(): SchedulerMetricsSnapshot;
|
|
39
|
+
private getState;
|
|
40
|
+
}
|
|
41
|
+
export declare function resolveAdaptiveSchedulerDelayMs(baseIntervalMs: number, maxIntervalMs: number, idleStreak: number): number;
|