triflux 10.3.4 → 10.7.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/LICENSE +21 -21
- package/bin/tfx-doctor-tui.mjs +1 -1
- package/bin/tfx-doctor.mjs +6 -1
- package/bin/tfx-profile.mjs +1 -1
- package/bin/tfx-setup-tui.mjs +1 -1
- package/bin/tfx-setup.mjs +6 -1
- package/bin/triflux.mjs +2396 -1140
- package/hooks/agent-route-guard.mjs +12 -8
- package/hooks/cross-review-tracker.mjs +21 -8
- package/hooks/error-context.mjs +19 -7
- package/hooks/hook-adaptive-collector.mjs +18 -16
- package/hooks/hook-manager.mjs +93 -32
- package/hooks/hook-orchestrator.mjs +108 -24
- package/hooks/hook-registry.json +11 -0
- package/hooks/keyword-rules.json +6 -10
- package/hooks/lib/resolve-root.mjs +1 -1
- package/hooks/mcp-config-watcher.mjs +6 -2
- package/hooks/pipeline-stop.mjs +3 -6
- package/hooks/safety-guard.mjs +99 -28
- package/hooks/session-start-fast.mjs +143 -0
- package/hooks/subagent-verifier.mjs +5 -4
- package/hub/account-broker.mjs +256 -60
- package/hub/adaptive-diagnostic.mjs +75 -48
- package/hub/adaptive-inject.mjs +95 -57
- package/hub/adaptive-memory.mjs +156 -42
- package/hub/adaptive.mjs +60 -31
- package/hub/assign-callbacks.mjs +67 -30
- package/hub/bridge.mjs +0 -1
- package/hub/cli-adapter-base.mjs +200 -48
- package/hub/codex-adapter.mjs +76 -96
- package/hub/codex-compat.mjs +3 -3
- package/hub/codex-preflight.mjs +63 -37
- package/hub/delegator/contracts.mjs +19 -23
- package/hub/delegator/index.mjs +3 -3
- package/hub/delegator/service.mjs +88 -64
- package/hub/delegator/tool-definitions.mjs +5 -5
- package/hub/fullcycle.mjs +33 -17
- package/hub/gemini-adapter.mjs +69 -94
- package/hub/hitl.mjs +89 -30
- package/hub/intent.mjs +161 -38
- package/hub/lib/cache-guard.mjs +43 -17
- package/hub/lib/mcp-response-cache.mjs +66 -32
- package/hub/lib/memory-store.mjs +285 -111
- package/hub/lib/path-utils.mjs +35 -37
- package/hub/lib/process-utils.mjs +106 -37
- package/hub/lib/spawn-trace.mjs +527 -0
- package/hub/lib/ssh-command.mjs +34 -4
- package/hub/lib/ssh-retry.mjs +5 -1
- package/hub/lib/uuidv7.mjs +4 -3
- package/hub/memory-doctor.mjs +266 -106
- package/hub/middleware/request-logger.mjs +61 -34
- package/hub/paths.mjs +9 -9
- package/hub/pipeline/gates/confidence.mjs +34 -15
- package/hub/pipeline/gates/consensus.mjs +27 -15
- package/hub/pipeline/gates/index.mjs +7 -3
- package/hub/pipeline/gates/selfcheck.mjs +57 -19
- package/hub/pipeline/index.mjs +77 -42
- package/hub/pipeline/state.mjs +10 -10
- package/hub/pipeline/transitions.mjs +40 -23
- package/hub/platform.mjs +57 -48
- package/hub/promote-penalties.mjs +25 -7
- package/hub/quality/deslop.mjs +70 -49
- package/hub/research.mjs +32 -25
- package/hub/router.mjs +240 -107
- package/hub/routing/complexity.mjs +132 -29
- package/hub/routing/index.mjs +17 -12
- package/hub/routing/q-learning.mjs +76 -28
- package/hub/server.mjs +4 -4
- package/hub/session-fingerprint.mjs +126 -60
- package/hub/state.mjs +84 -43
- package/hub/store-adapter.mjs +59 -26
- package/hub/store.mjs +356 -153
- package/hub/team/agent-map.json +22 -7
- package/hub/team/ansi.mjs +186 -122
- package/hub/team/backend.mjs +28 -10
- package/hub/team/cli/commands/attach.mjs +29 -9
- package/hub/team/cli/commands/control.mjs +29 -8
- package/hub/team/cli/commands/debug.mjs +32 -11
- package/hub/team/cli/commands/focus.mjs +38 -11
- package/hub/team/cli/commands/interrupt.mjs +18 -6
- package/hub/team/cli/commands/kill.mjs +16 -5
- package/hub/team/cli/commands/list.mjs +11 -4
- package/hub/team/cli/commands/send.mjs +19 -6
- package/hub/team/cli/commands/start/index.mjs +154 -31
- package/hub/team/cli/commands/start/parse-args.mjs +38 -11
- package/hub/team/cli/commands/start/start-headless.mjs +112 -36
- package/hub/team/cli/commands/start/start-in-process.mjs +12 -2
- package/hub/team/cli/commands/start/start-mux.mjs +70 -21
- package/hub/team/cli/commands/start/start-wt.mjs +29 -12
- package/hub/team/cli/commands/status.mjs +43 -14
- package/hub/team/cli/commands/stop.mjs +11 -4
- package/hub/team/cli/commands/task.mjs +8 -3
- package/hub/team/cli/commands/tasks.mjs +1 -1
- package/hub/team/cli/index.mjs +2 -2
- package/hub/team/cli/manifest.mjs +38 -8
- package/hub/team/cli/render.mjs +30 -8
- package/hub/team/cli/services/attach-fallback.mjs +31 -11
- package/hub/team/cli/services/hub-client.mjs +42 -14
- package/hub/team/cli/services/member-selector.mjs +11 -4
- package/hub/team/cli/services/native-control.mjs +48 -21
- package/hub/team/cli/services/runtime-mode.mjs +2 -1
- package/hub/team/cli/services/state-store.mjs +25 -8
- package/hub/team/cli/services/task-model.mjs +16 -6
- package/hub/team/conductor-mesh-bridge.mjs +24 -23
- package/hub/team/conductor.mjs +8 -4
- package/hub/team/dashboard-anchor.mjs +4 -5
- package/hub/team/dashboard-layout.mjs +3 -1
- package/hub/team/dashboard-open.mjs +41 -21
- package/hub/team/dashboard.mjs +76 -28
- package/hub/team/event-log.mjs +18 -10
- package/hub/team/handoff.mjs +31 -15
- package/hub/team/headless.mjs +2 -1
- package/hub/team/health-probe.mjs +69 -54
- package/hub/team/launcher-template.mjs +16 -13
- package/hub/team/native-supervisor.mjs +65 -21
- package/hub/team/native.mjs +74 -35
- package/hub/team/nativeProxy.mjs +184 -113
- package/hub/team/notify.mjs +119 -76
- package/hub/team/orchestrator.mjs +9 -4
- package/hub/team/pane.mjs +12 -7
- package/hub/team/process-cleanup.mjs +25 -16
- package/hub/team/psmux.mjs +491 -201
- package/hub/team/remote-probe.mjs +68 -52
- package/hub/team/remote-session.mjs +117 -59
- package/hub/team/remote-watcher.mjs +61 -33
- package/hub/team/routing.mjs +51 -25
- package/hub/team/runtime-strategy.mjs +3 -1
- package/hub/team/session.mjs +98 -34
- package/hub/team/staleState.mjs +72 -30
- package/hub/team/swarm-locks.mjs +15 -13
- package/hub/team/swarm-planner.mjs +32 -21
- package/hub/team/swarm-reconciler.mjs +48 -23
- package/hub/team/tui-lite.mjs +266 -68
- package/hub/team/tui-remote-adapter.mjs +14 -10
- package/hub/team/tui-viewer.mjs +99 -43
- package/hub/team/tui.mjs +708 -271
- package/hub/team/worktree-lifecycle.mjs +152 -58
- package/hub/team/wt-manager.mjs +24 -14
- package/hub/token-mode.mjs +71 -71
- package/hub/tray.mjs +66 -23
- package/hub/workers/claude-worker.mjs +162 -118
- package/hub/workers/codex-mcp.mjs +192 -141
- package/hub/workers/delegator-mcp.mjs +507 -333
- package/hub/workers/factory.mjs +8 -8
- package/hub/workers/gemini-worker.mjs +115 -84
- package/hub/workers/interface.mjs +6 -1
- package/hub/workers/worker-utils.mjs +21 -14
- package/hud/colors.mjs +27 -9
- package/hud/constants.mjs +162 -26
- package/hud/context-monitor.mjs +82 -41
- package/hud/hud-qos-status.mjs +129 -49
- package/hud/mission-board.mjs +6 -3
- package/hud/providers/claude.mjs +226 -115
- package/hud/providers/codex.mjs +62 -22
- package/hud/providers/gemini.mjs +168 -56
- package/hud/renderers.mjs +384 -119
- package/hud/terminal.mjs +101 -31
- package/hud/utils.mjs +78 -38
- package/mesh/index.mjs +11 -5
- package/mesh/mesh-budget.mjs +18 -9
- package/mesh/mesh-heartbeat.mjs +1 -1
- package/mesh/mesh-queue.mjs +3 -5
- package/mesh/mesh-router.mjs +5 -4
- package/package.json +2 -1
- package/scripts/__tests__/gen-skill-docs.test.mjs +36 -7
- package/scripts/__tests__/keyword-detector.test.mjs +77 -28
- package/scripts/__tests__/mcp-guard-engine.test.mjs +58 -20
- package/scripts/__tests__/remote-spawn-transfer.test.mjs +30 -19
- package/scripts/__tests__/remote-spawn.test.mjs +10 -4
- package/scripts/__tests__/session-start-fast.test.mjs +36 -0
- package/scripts/__tests__/skill-template.test.mjs +98 -50
- package/scripts/__tests__/smoke.test.mjs +1 -1
- package/scripts/__tests__/spawn-trace.test.mjs +102 -0
- package/scripts/__tests__/tfx-doctor-diagnose.test.mjs +48 -0
- package/scripts/cache-doctor.mjs +11 -4
- package/scripts/cache-warmup.mjs +96 -37
- package/scripts/claudemd-sync.mjs +27 -17
- package/scripts/codex-gateway-preflight.mjs +52 -37
- package/scripts/codex-mcp-gateway-sync.mjs +59 -39
- package/scripts/completions/tfx.bash +47 -47
- package/scripts/completions/tfx.fish +44 -44
- package/scripts/completions/tfx.zsh +83 -83
- package/scripts/config-audit.mjs +232 -0
- package/scripts/convert-to-tmpl.mjs +54 -0
- package/scripts/cross-review-gate.mjs +35 -12
- package/scripts/cross-review-tracker.mjs +21 -8
- package/scripts/demo.mjs +35 -17
- package/scripts/doctor-diagnose.mjs +284 -0
- package/scripts/gen-skill-docs.mjs +7 -2
- package/scripts/gen-skill-manifest.mjs +2 -1
- package/scripts/headless-guard.mjs +86 -48
- package/scripts/hub-ensure.mjs +45 -26
- package/scripts/keyword-detector.mjs +41 -20
- package/scripts/keyword-rules-expander.mjs +47 -30
- package/scripts/lib/claudemd-scanner.mjs +6 -1
- package/scripts/lib/context.mjs +3 -3
- package/scripts/lib/cross-review-utils.mjs +6 -3
- package/scripts/lib/env-probe.mjs +47 -28
- package/scripts/lib/gemini-profiles.mjs +44 -10
- package/scripts/lib/handoff.mjs +33 -17
- package/scripts/lib/hook-utils.mjs +8 -6
- package/scripts/lib/keyword-rules.mjs +43 -19
- package/scripts/lib/logger.mjs +24 -24
- package/scripts/lib/mcp-filter.mjs +377 -239
- package/scripts/lib/mcp-guard-engine.mjs +194 -79
- package/scripts/lib/mcp-manifest.mjs +23 -13
- package/scripts/lib/mcp-server-catalog.mjs +300 -63
- package/scripts/lib/psmux-info.mjs +11 -6
- package/scripts/lib/remote-spawn-transfer.mjs +44 -14
- package/scripts/lib/skill-template.mjs +30 -7
- package/scripts/mcp-check.mjs +58 -39
- package/scripts/mcp-gateway-config.mjs +83 -39
- package/scripts/mcp-gateway-ensure.mjs +43 -35
- package/scripts/mcp-gateway-integration-test.mjs +70 -58
- package/scripts/mcp-gateway-start.mjs +126 -60
- package/scripts/mcp-gateway-verify.mjs +24 -22
- package/scripts/mcp-safety-guard.mjs +44 -11
- package/scripts/notion-read.mjs +199 -84
- package/scripts/pack.mjs +94 -89
- package/scripts/preflight-cache.mjs +27 -10
- package/scripts/preinstall.mjs +42 -13
- package/scripts/remote-spawn.mjs +309 -94
- package/scripts/run.cjs +8 -5
- package/scripts/session-spawn-helper.mjs +130 -39
- package/scripts/session-stale-cleanup.mjs +123 -0
- package/scripts/setup.mjs +941 -492
- package/scripts/test-lock.mjs +20 -7
- package/scripts/test-tfx-route-no-claude-native.mjs +16 -12
- package/scripts/tfx-batch-stats.mjs +32 -11
- package/scripts/tfx-gate-activate.mjs +11 -4
- package/scripts/tfx-route-post.mjs +87 -20
- package/scripts/tfx-route-worker.mjs +57 -51
- package/scripts/tfx-route.sh +41 -124
- package/scripts/tmp-cleanup.mjs +21 -7
- package/scripts/token-snapshot.mjs +204 -85
- package/skills/.omc/state/agent-replay-8f0e10a9-9693-4410-96f5-a6b07e8ed995.jsonl +1 -0
- package/skills/.omc/state/idle-notif-cooldown.json +3 -0
- package/skills/.omc/state/last-tool-error.json +7 -0
- package/skills/.omc/state/subagent-tracking.json +7 -0
- package/skills/_templates/base.md +1 -6
- package/skills/merge-worktree/SKILL.md.tmpl +144 -0
- package/skills/shared/telemetry-segment.md +6 -0
- package/skills/star-prompt/SKILL.md.tmpl +222 -0
- package/skills/tfx-analysis/SKILL.md.tmpl +107 -0
- package/skills/tfx-analysis/skill.json +1 -6
- package/skills/tfx-auto/SKILL.md +1 -0
- package/skills/tfx-auto-codex/SKILL.md.tmpl +106 -0
- package/skills/tfx-auto-codex/skill.json +1 -3
- package/skills/tfx-autopilot/SKILL.md.tmpl +116 -0
- package/skills/tfx-autopilot/skill.json +1 -5
- package/skills/tfx-autoresearch/SKILL.md.tmpl +136 -0
- package/skills/tfx-autoroute/SKILL.md.tmpl +189 -0
- package/skills/tfx-autoroute/skill.json +1 -7
- package/skills/tfx-codex/SKILL.md +1 -0
- package/skills/tfx-codex/skill.json +1 -3
- package/skills/tfx-codex-swarm/SKILL.md.tmpl +16 -0
- package/skills/tfx-codex-swarm/evals/evals.json +1 -1
- package/skills/tfx-codex-swarm/skill.json +1 -4
- package/skills/tfx-codex-swarm-workspace/iteration-1/benchmark.json +54 -12
- package/skills/tfx-codex-swarm-workspace/iteration-1/full-swarm-all-prds/with_skill/grading.json +35 -7
- package/skills/tfx-codex-swarm-workspace/iteration-1/full-swarm-all-prds/without_skill/grading.json +35 -7
- package/skills/tfx-codex-swarm-workspace/iteration-1/implicit-swarm-no-keywords/with_skill/grading.json +25 -5
- package/skills/tfx-codex-swarm-workspace/iteration-1/implicit-swarm-no-keywords/without_skill/grading.json +25 -5
- package/skills/tfx-codex-swarm-workspace/iteration-1/selective-spawn-with-override/with_skill/grading.json +20 -4
- package/skills/tfx-codex-swarm-workspace/iteration-1/selective-spawn-with-override/without_skill/grading.json +16 -4
- package/skills/tfx-consensus/SKILL.md.tmpl +146 -0
- package/skills/tfx-debate/SKILL.md.tmpl +192 -0
- package/skills/tfx-debate/skill.json +1 -7
- package/skills/tfx-deep-analysis/SKILL.md.tmpl +228 -0
- package/skills/tfx-deep-analysis/skill.json +1 -5
- package/skills/tfx-deep-interview/SKILL.md.tmpl +203 -0
- package/skills/tfx-deep-plan/SKILL.md.tmpl +282 -0
- package/skills/tfx-deep-qa/SKILL.md.tmpl +165 -0
- package/skills/tfx-deep-qa/skill.json +1 -6
- package/skills/tfx-deep-research/SKILL.md.tmpl +217 -0
- package/skills/tfx-deep-review/SKILL.md.tmpl +179 -0
- package/skills/tfx-doctor/SKILL.md +21 -0
- package/skills/tfx-doctor/SKILL.md.tmpl +172 -0
- package/skills/tfx-doctor/skill.json +1 -3
- package/skills/tfx-find/SKILL.md +1 -0
- package/skills/tfx-forge/SKILL.md.tmpl +187 -0
- package/skills/tfx-fullcycle/SKILL.md.tmpl +286 -0
- package/skills/tfx-fullcycle/skill.json +1 -6
- package/skills/tfx-gemini/SKILL.md.tmpl +91 -0
- package/skills/tfx-gemini/skill.json +1 -3
- package/skills/tfx-hooks/SKILL.md.tmpl +216 -0
- package/skills/tfx-hooks/skill.json +1 -3
- package/skills/tfx-hub/SKILL.md.tmpl +212 -0
- package/skills/tfx-hub/skill.json +1 -3
- package/skills/tfx-index/SKILL.md +1 -0
- package/skills/tfx-index/skill.json +1 -6
- package/skills/tfx-interview/SKILL.md.tmpl +285 -0
- package/skills/tfx-multi/SKILL.md.tmpl +183 -0
- package/skills/tfx-multi/skill.json +1 -3
- package/skills/tfx-panel/SKILL.md.tmpl +189 -0
- package/skills/tfx-panel/skill.json +1 -7
- package/skills/tfx-persist/SKILL.md.tmpl +270 -0
- package/skills/tfx-persist/skill.json +1 -7
- package/skills/tfx-plan/SKILL.md +1 -0
- package/skills/tfx-plan/skill.json +1 -6
- package/skills/tfx-profile/SKILL.md.tmpl +239 -0
- package/skills/tfx-profile/skill.json +1 -3
- package/skills/tfx-prune/SKILL.md.tmpl +200 -0
- package/skills/tfx-prune/skill.json +1 -7
- package/skills/tfx-psmux-rules/SKILL.md.tmpl +326 -0
- package/skills/tfx-psmux-rules/skill.json +1 -4
- package/skills/tfx-qa/SKILL.md +1 -0
- package/skills/tfx-qa/skill.json +1 -6
- package/skills/tfx-ralph/SKILL.md.tmpl +28 -0
- package/skills/tfx-ralph/skill.json +1 -4
- package/skills/tfx-remote-setup/SKILL.md.tmpl +576 -0
- package/skills/tfx-remote-setup/skill.json +1 -3
- package/skills/tfx-remote-spawn/SKILL.md.tmpl +263 -0
- package/skills/tfx-remote-spawn/references/hosts.json +16 -0
- package/skills/tfx-remote-spawn/skill.json +1 -4
- package/skills/tfx-research/SKILL.md +1 -0
- package/skills/tfx-review/SKILL.md +1 -0
- package/skills/tfx-review/skill.json +1 -6
- package/skills/tfx-setup/SKILL.md.tmpl +504 -0
- package/skills/tfx-setup/skill.json +1 -3
- package/skills/tfx-swarm/SKILL.md +22 -0
- package/skills/tfx-swarm/SKILL.md.tmpl +218 -0
- package/tui/codex-profile.mjs +88 -33
- package/tui/core.mjs +45 -15
- package/tui/doctor.mjs +75 -28
- package/tui/gemini-profile.mjs +74 -29
- package/tui/monitor-data.mjs +8 -4
- package/tui/monitor.mjs +71 -27
- package/tui/setup.mjs +133 -42
|
@@ -0,0 +1,232 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
// scripts/config-audit.mjs — 설정 정적 보안/성능 감사
|
|
3
|
+
//
|
|
4
|
+
// triflux doctor --audit 또는 단독 실행.
|
|
5
|
+
// settings.json, CLAUDE.md, MCP, 훅 설정을 스캔하여 위험/성능 이슈 탐지.
|
|
6
|
+
//
|
|
7
|
+
// 출력: JSON (--json) 또는 마크다운 테이블
|
|
8
|
+
|
|
9
|
+
import { existsSync, readFileSync, readdirSync } from "node:fs";
|
|
10
|
+
import { homedir } from "node:os";
|
|
11
|
+
import { join, basename } from "node:path";
|
|
12
|
+
|
|
13
|
+
const HOME = homedir();
|
|
14
|
+
const CLAUDE_DIR = join(HOME, ".claude");
|
|
15
|
+
const JSON_OUTPUT = process.argv.includes("--json");
|
|
16
|
+
|
|
17
|
+
// ── 결과 수집 ──
|
|
18
|
+
const findings = [];
|
|
19
|
+
|
|
20
|
+
function addFinding(category, severity, message, detail = "") {
|
|
21
|
+
findings.push({ category, severity, message, detail });
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
// ── 1. settings.json 감사 ──
|
|
25
|
+
function auditSettings() {
|
|
26
|
+
const settingsPath = join(CLAUDE_DIR, "settings.json");
|
|
27
|
+
if (!existsSync(settingsPath)) {
|
|
28
|
+
addFinding("settings", "info", "settings.json 없음", settingsPath);
|
|
29
|
+
return;
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
let settings;
|
|
33
|
+
try {
|
|
34
|
+
settings = JSON.parse(readFileSync(settingsPath, "utf8"));
|
|
35
|
+
} catch {
|
|
36
|
+
addFinding("settings", "warn", "settings.json 파싱 실패", settingsPath);
|
|
37
|
+
return;
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
// 권한 모드
|
|
41
|
+
const mode = settings.permissions?.defaultMode;
|
|
42
|
+
if (mode === "bypassPermissions" || mode === "acceptEdits") {
|
|
43
|
+
addFinding("settings", "warn", `defaultMode: "${mode}" — 도구 승인 없이 실행됨`, "보안 리뷰 시 주의");
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
// 환경 변수 내 시크릿 패턴
|
|
47
|
+
const env = settings.env || {};
|
|
48
|
+
const SECRET_PATTERNS = [
|
|
49
|
+
/(?:api[_-]?key|secret|token|password|credential)s?\s*[:=]/i,
|
|
50
|
+
/(?:sk-|ghp_|gho_|github_pat_|xoxb-|xoxp-)/i,
|
|
51
|
+
/ANTHROPIC_API_KEY/i,
|
|
52
|
+
/OPENAI_API_KEY/i,
|
|
53
|
+
];
|
|
54
|
+
for (const [key, value] of Object.entries(env)) {
|
|
55
|
+
for (const pattern of SECRET_PATTERNS) {
|
|
56
|
+
if (pattern.test(key) || pattern.test(String(value))) {
|
|
57
|
+
addFinding("settings", "critical", `env에 시크릿 패턴 감지: ${key}`, "환경변수로 분리 권장");
|
|
58
|
+
break;
|
|
59
|
+
}
|
|
60
|
+
}
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
// 훅 timeout 검사
|
|
64
|
+
const hooks = settings.hooks || {};
|
|
65
|
+
let totalHooks = 0;
|
|
66
|
+
let longTimeouts = 0;
|
|
67
|
+
for (const [event, matchers] of Object.entries(hooks)) {
|
|
68
|
+
for (const matcher of Array.isArray(matchers) ? matchers : []) {
|
|
69
|
+
for (const hook of matcher.hooks || []) {
|
|
70
|
+
totalHooks++;
|
|
71
|
+
if (hook.timeout && hook.timeout > 15) {
|
|
72
|
+
longTimeouts++;
|
|
73
|
+
addFinding("hooks", "warn", `${event} 훅 timeout ${hook.timeout}s (>15s)`, hook.command?.slice(0, 80));
|
|
74
|
+
}
|
|
75
|
+
}
|
|
76
|
+
}
|
|
77
|
+
}
|
|
78
|
+
if (totalHooks > 10) {
|
|
79
|
+
addFinding("hooks", "info", `settings.json에 훅 ${totalHooks}개 등록`, "SessionStart 성능에 영향 가능");
|
|
80
|
+
}
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
// ── 2. CLAUDE.md 시크릿 스캔 ──
|
|
84
|
+
function auditClaudeMd() {
|
|
85
|
+
const candidates = [
|
|
86
|
+
join(process.cwd(), "CLAUDE.md"),
|
|
87
|
+
join(CLAUDE_DIR, "CLAUDE.md"),
|
|
88
|
+
];
|
|
89
|
+
|
|
90
|
+
for (const mdPath of candidates) {
|
|
91
|
+
if (!existsSync(mdPath)) continue;
|
|
92
|
+
let content;
|
|
93
|
+
try {
|
|
94
|
+
content = readFileSync(mdPath, "utf8");
|
|
95
|
+
} catch { continue; }
|
|
96
|
+
|
|
97
|
+
const SECRET_LINE_PATTERNS = [
|
|
98
|
+
/(?:api[_-]?key|secret|password)\s*[:=]\s*["']?\S{8,}/i,
|
|
99
|
+
/(?:^|[\s"'=:])(?:sk-|ghp_|gho_|github_pat_|xoxb-|xoxp-)[A-Za-z0-9_-]{10,}/,
|
|
100
|
+
/-----BEGIN (?:RSA |EC )?PRIVATE KEY-----/,
|
|
101
|
+
];
|
|
102
|
+
|
|
103
|
+
const lines = content.split("\n");
|
|
104
|
+
for (let i = 0; i < lines.length; i++) {
|
|
105
|
+
for (const pattern of SECRET_LINE_PATTERNS) {
|
|
106
|
+
if (pattern.test(lines[i])) {
|
|
107
|
+
addFinding("claude-md", "critical", `CLAUDE.md:${i + 1} 시크릿 패턴 감지`, basename(mdPath));
|
|
108
|
+
break;
|
|
109
|
+
}
|
|
110
|
+
}
|
|
111
|
+
}
|
|
112
|
+
|
|
113
|
+
// 길이 경고
|
|
114
|
+
const tokenEstimate = Math.ceil(Buffer.byteLength(content, "utf8") / 4);
|
|
115
|
+
if (tokenEstimate > 3000) {
|
|
116
|
+
addFinding("claude-md", "warn", `CLAUDE.md ~${tokenEstimate} 토큰 (>3000)`, `${basename(mdPath)} — 컨텍스트 로트 위험`);
|
|
117
|
+
}
|
|
118
|
+
}
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
// ── 3. MCP 설정 감사 ──
|
|
122
|
+
function auditMcp() {
|
|
123
|
+
const mcpPaths = [
|
|
124
|
+
join(CLAUDE_DIR, "mcp_servers.json"),
|
|
125
|
+
join(CLAUDE_DIR, ".mcp.json"),
|
|
126
|
+
join(process.cwd(), ".mcp.json"),
|
|
127
|
+
];
|
|
128
|
+
|
|
129
|
+
let totalServers = 0;
|
|
130
|
+
let stdioCount = 0;
|
|
131
|
+
|
|
132
|
+
for (const mcpPath of mcpPaths) {
|
|
133
|
+
if (!existsSync(mcpPath)) continue;
|
|
134
|
+
let config;
|
|
135
|
+
try {
|
|
136
|
+
config = JSON.parse(readFileSync(mcpPath, "utf8"));
|
|
137
|
+
} catch { continue; }
|
|
138
|
+
|
|
139
|
+
const servers = config.mcpServers || config.servers || config;
|
|
140
|
+
for (const [name, def] of Object.entries(servers)) {
|
|
141
|
+
if (!def || typeof def !== "object") continue;
|
|
142
|
+
totalServers++;
|
|
143
|
+
|
|
144
|
+
if (def.type === "stdio" || def.command) {
|
|
145
|
+
stdioCount++;
|
|
146
|
+
}
|
|
147
|
+
|
|
148
|
+
// 위험 패턴: 쉘 실행, eval, 알 수 없는 npx 패키지
|
|
149
|
+
const cmd = String(def.command || "");
|
|
150
|
+
if (/\beval\b|\bexec\b.*sh\b|\bcurl\b.*\|\s*(?:bash|sh)\b/i.test(cmd)) {
|
|
151
|
+
addFinding("mcp", "critical", `MCP "${name}": 위험한 명령 패턴`, cmd.slice(0, 100));
|
|
152
|
+
}
|
|
153
|
+
}
|
|
154
|
+
}
|
|
155
|
+
|
|
156
|
+
if (totalServers > 10) {
|
|
157
|
+
addFinding("mcp", "warn", `MCP 서버 ${totalServers}개 등록 (>10)`, "컨텍스트 윈도우 압박 가능");
|
|
158
|
+
}
|
|
159
|
+
if (stdioCount > 5) {
|
|
160
|
+
addFinding("mcp", "info", `stdio MCP ${stdioCount}개 — 프로세스 spawn 부하`, "불필요한 서버 비활성화 권장");
|
|
161
|
+
}
|
|
162
|
+
}
|
|
163
|
+
|
|
164
|
+
// ── 4. 훅 레지스트리 감사 ──
|
|
165
|
+
function auditHookRegistry() {
|
|
166
|
+
const registryPath = join(process.cwd(), "hooks", "hook-registry.json");
|
|
167
|
+
if (!existsSync(registryPath)) return;
|
|
168
|
+
|
|
169
|
+
let registry;
|
|
170
|
+
try {
|
|
171
|
+
registry = JSON.parse(readFileSync(registryPath, "utf8"));
|
|
172
|
+
} catch { return; }
|
|
173
|
+
|
|
174
|
+
const events = registry.events || {};
|
|
175
|
+
let blockingCount = 0;
|
|
176
|
+
let externalCount = 0;
|
|
177
|
+
|
|
178
|
+
for (const [event, hooks] of Object.entries(events)) {
|
|
179
|
+
for (const hook of hooks) {
|
|
180
|
+
if (hook.blocking) blockingCount++;
|
|
181
|
+
if (hook.source !== "triflux" && hook.source !== "omc") externalCount++;
|
|
182
|
+
|
|
183
|
+
// 외부 훅의 명령에 위험 패턴
|
|
184
|
+
if (hook.source !== "triflux") {
|
|
185
|
+
const cmd = String(hook.command || "");
|
|
186
|
+
if (/\brm\b|\bdel\b|\bformat\b|\bgit\s+push\b/i.test(cmd)) {
|
|
187
|
+
addFinding("hooks", "warn", `외부 훅 "${hook.id}": 위험 명령 패턴`, cmd.slice(0, 80));
|
|
188
|
+
}
|
|
189
|
+
}
|
|
190
|
+
}
|
|
191
|
+
}
|
|
192
|
+
|
|
193
|
+
if (blockingCount > 5) {
|
|
194
|
+
addFinding("hooks", "info", `blocking 훅 ${blockingCount}개 — 도구 실행 지연 가능`, "불필요한 blocking 해제 검토");
|
|
195
|
+
}
|
|
196
|
+
if (externalCount > 0) {
|
|
197
|
+
addFinding("hooks", "info", `외부 훅 ${externalCount}개 등록`, "출처 확인 권장");
|
|
198
|
+
}
|
|
199
|
+
}
|
|
200
|
+
|
|
201
|
+
// ── 실행 ──
|
|
202
|
+
auditSettings();
|
|
203
|
+
auditClaudeMd();
|
|
204
|
+
auditMcp();
|
|
205
|
+
auditHookRegistry();
|
|
206
|
+
|
|
207
|
+
// ── 출력 ──
|
|
208
|
+
const summary = {
|
|
209
|
+
total: findings.length,
|
|
210
|
+
critical: findings.filter(f => f.severity === "critical").length,
|
|
211
|
+
warn: findings.filter(f => f.severity === "warn").length,
|
|
212
|
+
info: findings.filter(f => f.severity === "info").length,
|
|
213
|
+
};
|
|
214
|
+
|
|
215
|
+
if (JSON_OUTPUT) {
|
|
216
|
+
console.log(JSON.stringify({ summary, findings }, null, 2));
|
|
217
|
+
} else {
|
|
218
|
+
const SEVERITY_ICON = { critical: "🔴", warn: "🟡", info: "🔵" };
|
|
219
|
+
console.log("\n ⬡ triflux config audit\n");
|
|
220
|
+
if (findings.length === 0) {
|
|
221
|
+
console.log(" ✅ 이슈 없음\n");
|
|
222
|
+
} else {
|
|
223
|
+
console.log(` | 심각도 | 카테고리 | 이슈 | 상세 |`);
|
|
224
|
+
console.log(` |--------|----------|------|------|`);
|
|
225
|
+
for (const f of findings) {
|
|
226
|
+
console.log(` | ${SEVERITY_ICON[f.severity] || "⚪"} ${f.severity} | ${f.category} | ${f.message} | ${f.detail} |`);
|
|
227
|
+
}
|
|
228
|
+
console.log(`\n 합계: critical=${summary.critical} warn=${summary.warn} info=${summary.info}\n`);
|
|
229
|
+
}
|
|
230
|
+
}
|
|
231
|
+
|
|
232
|
+
process.exit(summary.critical > 0 ? 1 : 0);
|
|
@@ -0,0 +1,54 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
/**
|
|
3
|
+
* Converts SKILL.md files to SKILL.md.tmpl format.
|
|
4
|
+
* - Replaces skill name in title with {{SKILL_NAME}}
|
|
5
|
+
* - Removes expanded base block (ARGUMENTS + Telemetry)
|
|
6
|
+
* - Inserts {{> base}} after the title line
|
|
7
|
+
*/
|
|
8
|
+
import { existsSync, readdirSync, readFileSync, writeFileSync } from "fs";
|
|
9
|
+
import { join } from "path";
|
|
10
|
+
|
|
11
|
+
const SKILLS_DIR = join(import.meta.dirname, "..", "skills");
|
|
12
|
+
|
|
13
|
+
const BASE_BLOCK_RE =
|
|
14
|
+
/\n> \*\*ARGUMENTS 처리\*\*.*?\n> 워크플로우의 첫 단계.*?기존 절차대로.*?\n\n> \*\*Telemetry\*\*\n>\n> - Skill:.*?\n> - Description:.*?\n> - Session:.*?\n> - Errors:.*?\n/s;
|
|
15
|
+
|
|
16
|
+
let converted = 0;
|
|
17
|
+
let skipped = 0;
|
|
18
|
+
|
|
19
|
+
for (const name of readdirSync(SKILLS_DIR)) {
|
|
20
|
+
const dir = join(SKILLS_DIR, name);
|
|
21
|
+
const skillMd = join(dir, "SKILL.md");
|
|
22
|
+
const tmplMd = join(dir, "SKILL.md.tmpl");
|
|
23
|
+
|
|
24
|
+
if (name.startsWith("_")) continue;
|
|
25
|
+
if (!existsSync(skillMd)) continue;
|
|
26
|
+
if (existsSync(tmplMd)) {
|
|
27
|
+
skipped++;
|
|
28
|
+
continue;
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
let content = readFileSync(skillMd, "utf8");
|
|
32
|
+
|
|
33
|
+
// Replace skill name in title: # tfx-foo — Title -> # {{SKILL_NAME}} — Title
|
|
34
|
+
content = content.replace(
|
|
35
|
+
new RegExp(`^(# )${name.replace(/[.*+?^${}()|[\]\\]/g, "\\$&")}( —)`, "m"),
|
|
36
|
+
"$1{{SKILL_NAME}}$2",
|
|
37
|
+
);
|
|
38
|
+
|
|
39
|
+
// Remove expanded base block if present
|
|
40
|
+
if (BASE_BLOCK_RE.test(content)) {
|
|
41
|
+
content = content.replace(BASE_BLOCK_RE, "\n\n{{> base}}\n\n");
|
|
42
|
+
} else {
|
|
43
|
+
// Insert {{> base}} after the title line
|
|
44
|
+
content = content.replace(/^(# .+\n)/, "$1\n{{> base}}\n");
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
writeFileSync(tmplMd, content);
|
|
48
|
+
converted++;
|
|
49
|
+
console.log(`✅ ${name}`);
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
console.log(
|
|
53
|
+
`\nConverted: ${converted}, Skipped (already has .tmpl): ${skipped}`,
|
|
54
|
+
);
|
|
@@ -2,17 +2,17 @@
|
|
|
2
2
|
|
|
3
3
|
import { existsSync, readFileSync, unlinkSync } from "node:fs";
|
|
4
4
|
import { join } from "node:path";
|
|
5
|
-
import { nudge, deny } from "./lib/hook-utils.mjs";
|
|
6
5
|
import {
|
|
7
|
-
|
|
8
|
-
parseJson,
|
|
6
|
+
expectedReviewer,
|
|
9
7
|
nowSec,
|
|
8
|
+
parseJson,
|
|
9
|
+
readStdin,
|
|
10
10
|
resolveBaseDir,
|
|
11
|
-
shouldTrackPath,
|
|
12
|
-
expectedReviewer,
|
|
13
11
|
SESSION_TTL_SEC,
|
|
14
12
|
STATE_REL_PATH,
|
|
13
|
+
shouldTrackPath,
|
|
15
14
|
} from "./lib/cross-review-utils.mjs";
|
|
15
|
+
import { deny, nudge } from "./lib/hook-utils.mjs";
|
|
16
16
|
|
|
17
17
|
function loadState(statePath) {
|
|
18
18
|
if (!existsSync(statePath)) return null;
|
|
@@ -84,17 +84,37 @@ async function main() {
|
|
|
84
84
|
|
|
85
85
|
// tracker가 설정한 self_approved 플래그 명시적 체크
|
|
86
86
|
if (meta.self_approved === true) {
|
|
87
|
-
selfApproved.push({
|
|
87
|
+
selfApproved.push({
|
|
88
|
+
path,
|
|
89
|
+
author,
|
|
90
|
+
reviewer: meta.reviewer || author,
|
|
91
|
+
expectedReviewer: requiredReviewer,
|
|
92
|
+
});
|
|
88
93
|
continue;
|
|
89
94
|
}
|
|
90
95
|
|
|
91
96
|
if (reviewed && reviewer && reviewer === author) {
|
|
92
|
-
selfApproved.push({
|
|
97
|
+
selfApproved.push({
|
|
98
|
+
path,
|
|
99
|
+
author,
|
|
100
|
+
reviewer,
|
|
101
|
+
expectedReviewer: requiredReviewer,
|
|
102
|
+
});
|
|
93
103
|
continue;
|
|
94
104
|
}
|
|
95
105
|
|
|
96
|
-
if (
|
|
97
|
-
|
|
106
|
+
if (
|
|
107
|
+
reviewed &&
|
|
108
|
+
requiredReviewer &&
|
|
109
|
+
reviewer &&
|
|
110
|
+
reviewer !== requiredReviewer
|
|
111
|
+
) {
|
|
112
|
+
selfApproved.push({
|
|
113
|
+
path,
|
|
114
|
+
author,
|
|
115
|
+
reviewer,
|
|
116
|
+
expectedReviewer: requiredReviewer,
|
|
117
|
+
});
|
|
98
118
|
continue;
|
|
99
119
|
}
|
|
100
120
|
|
|
@@ -105,18 +125,21 @@ async function main() {
|
|
|
105
125
|
|
|
106
126
|
if (selfApproved.length > 0) {
|
|
107
127
|
const lines = selfApproved
|
|
108
|
-
.map(
|
|
128
|
+
.map(
|
|
129
|
+
(item) =>
|
|
130
|
+
` * ${item.path} (author=${item.author}, reviewer=${item.reviewer}, required=${item.expectedReviewer || "n/a"})`,
|
|
131
|
+
)
|
|
109
132
|
.join("\n");
|
|
110
133
|
deny(
|
|
111
134
|
`[cross-review] self-approve 차단: 동일/비허용 reviewer가 감지되었습니다.\n${lines}\n` +
|
|
112
|
-
|
|
135
|
+
"규칙: author=claude -> reviewer=codex, author=codex -> reviewer=claude",
|
|
113
136
|
);
|
|
114
137
|
}
|
|
115
138
|
|
|
116
139
|
if (pending.length > 0) {
|
|
117
140
|
nudge(
|
|
118
141
|
`[cross-review] git commit 전에 교차 검증이 필요합니다.\n${summarizePending(pending)}\n` +
|
|
119
|
-
|
|
142
|
+
"규칙: author=claude -> reviewer=codex, author=codex -> reviewer=claude",
|
|
120
143
|
);
|
|
121
144
|
}
|
|
122
145
|
|
|
@@ -1,16 +1,22 @@
|
|
|
1
1
|
#!/usr/bin/env node
|
|
2
2
|
|
|
3
|
-
import {
|
|
3
|
+
import {
|
|
4
|
+
existsSync,
|
|
5
|
+
mkdirSync,
|
|
6
|
+
readFileSync,
|
|
7
|
+
unlinkSync,
|
|
8
|
+
writeFileSync,
|
|
9
|
+
} from "node:fs";
|
|
4
10
|
import { dirname, isAbsolute, join, relative } from "node:path";
|
|
5
11
|
import {
|
|
6
|
-
|
|
7
|
-
parseJson,
|
|
12
|
+
expectedReviewer,
|
|
8
13
|
nowSec,
|
|
14
|
+
parseJson,
|
|
15
|
+
readStdin,
|
|
9
16
|
resolveBaseDir,
|
|
10
|
-
shouldTrackPath,
|
|
11
|
-
expectedReviewer,
|
|
12
17
|
SESSION_TTL_SEC,
|
|
13
18
|
STATE_REL_PATH,
|
|
19
|
+
shouldTrackPath,
|
|
14
20
|
} from "./lib/cross-review-utils.mjs";
|
|
15
21
|
|
|
16
22
|
function resolveStatePath(baseDir) {
|
|
@@ -39,7 +45,8 @@ function loadState(statePath) {
|
|
|
39
45
|
}
|
|
40
46
|
|
|
41
47
|
return {
|
|
42
|
-
files:
|
|
48
|
+
files:
|
|
49
|
+
parsed?.files && typeof parsed.files === "object" ? parsed.files : {},
|
|
43
50
|
session_start: sessionStart,
|
|
44
51
|
};
|
|
45
52
|
} catch {
|
|
@@ -69,7 +76,8 @@ function normalizePath(filePath, baseDir) {
|
|
|
69
76
|
|
|
70
77
|
function extractFilePath(toolInput) {
|
|
71
78
|
if (!toolInput || typeof toolInput !== "object") return "";
|
|
72
|
-
const candidate =
|
|
79
|
+
const candidate =
|
|
80
|
+
toolInput.file_path ?? toolInput.path ?? toolInput.filePath ?? "";
|
|
73
81
|
return typeof candidate === "string" ? candidate : "";
|
|
74
82
|
}
|
|
75
83
|
|
|
@@ -81,7 +89,12 @@ function extractCandidatePaths(payload, baseDir) {
|
|
|
81
89
|
const trimmed = value.trim();
|
|
82
90
|
if (!trimmed || /\s/.test(trimmed)) return false;
|
|
83
91
|
if (trimmed.length > 260) return false;
|
|
84
|
-
if (
|
|
92
|
+
if (
|
|
93
|
+
!trimmed.includes(".") &&
|
|
94
|
+
!trimmed.includes("/") &&
|
|
95
|
+
!trimmed.includes("\\")
|
|
96
|
+
)
|
|
97
|
+
return false;
|
|
85
98
|
return /^[./\\A-Za-z0-9_-]/.test(trimmed);
|
|
86
99
|
};
|
|
87
100
|
|
package/scripts/demo.mjs
CHANGED
|
@@ -3,14 +3,6 @@
|
|
|
3
3
|
import childProcess from "node:child_process";
|
|
4
4
|
import { parseArgs } from "node:util";
|
|
5
5
|
|
|
6
|
-
const { values: flags } = parseArgs({
|
|
7
|
-
options: {
|
|
8
|
-
"dry-run": { type: "boolean", default: false },
|
|
9
|
-
keep: { type: "boolean", default: false },
|
|
10
|
-
},
|
|
11
|
-
strict: false,
|
|
12
|
-
});
|
|
13
|
-
|
|
14
6
|
const SESSION_NAME = "triflux-demo";
|
|
15
7
|
|
|
16
8
|
const WORKERS = [
|
|
@@ -46,7 +38,10 @@ const WORKERS = [
|
|
|
46
38
|
export function checkPsmux(opts = {}) {
|
|
47
39
|
if (opts.dryRun) return false;
|
|
48
40
|
try {
|
|
49
|
-
childProcess.execFileSync("psmux", ["-V"], {
|
|
41
|
+
childProcess.execFileSync("psmux", ["-V"], {
|
|
42
|
+
encoding: "utf8",
|
|
43
|
+
stdio: "pipe",
|
|
44
|
+
});
|
|
50
45
|
return true;
|
|
51
46
|
} catch {
|
|
52
47
|
return false;
|
|
@@ -60,9 +55,19 @@ export function createDemoSession(sessionName, opts = {}) {
|
|
|
60
55
|
console.log(`[dry-run] psmux split-window -h -t ${sessionName}`);
|
|
61
56
|
return;
|
|
62
57
|
}
|
|
63
|
-
childProcess.execFileSync("psmux", ["new-session", "-d", "-s", sessionName], {
|
|
64
|
-
|
|
65
|
-
|
|
58
|
+
childProcess.execFileSync("psmux", ["new-session", "-d", "-s", sessionName], {
|
|
59
|
+
stdio: "pipe",
|
|
60
|
+
});
|
|
61
|
+
childProcess.execFileSync(
|
|
62
|
+
"psmux",
|
|
63
|
+
["split-window", "-h", "-t", sessionName],
|
|
64
|
+
{ stdio: "pipe" },
|
|
65
|
+
);
|
|
66
|
+
childProcess.execFileSync(
|
|
67
|
+
"psmux",
|
|
68
|
+
["split-window", "-h", "-t", sessionName],
|
|
69
|
+
{ stdio: "pipe" },
|
|
70
|
+
);
|
|
66
71
|
}
|
|
67
72
|
|
|
68
73
|
export function simulateWorker(pane, agentName, messages, opts = {}) {
|
|
@@ -70,11 +75,19 @@ export function simulateWorker(pane, agentName, messages, opts = {}) {
|
|
|
70
75
|
for (const msg of messages) {
|
|
71
76
|
const escapedMsg = msg.replace(/'/g, "'\\''");
|
|
72
77
|
if (opts.dryRun) {
|
|
73
|
-
console.log(
|
|
78
|
+
console.log(
|
|
79
|
+
`[dry-run] psmux send-keys -t ${sessionName}:0.${pane} "echo '${escapedMsg}'" Enter`,
|
|
80
|
+
);
|
|
74
81
|
} else {
|
|
75
82
|
childProcess.execFileSync(
|
|
76
83
|
"psmux",
|
|
77
|
-
[
|
|
84
|
+
[
|
|
85
|
+
"send-keys",
|
|
86
|
+
"-t",
|
|
87
|
+
`${sessionName}:0.${pane}`,
|
|
88
|
+
`echo '${escapedMsg}'`,
|
|
89
|
+
"Enter",
|
|
90
|
+
],
|
|
78
91
|
{ stdio: "pipe" },
|
|
79
92
|
);
|
|
80
93
|
}
|
|
@@ -102,7 +115,9 @@ export function cleanup(sessionName, opts = {}) {
|
|
|
102
115
|
return;
|
|
103
116
|
}
|
|
104
117
|
try {
|
|
105
|
-
childProcess.execFileSync("psmux", ["kill-session", "-t", sessionName], {
|
|
118
|
+
childProcess.execFileSync("psmux", ["kill-session", "-t", sessionName], {
|
|
119
|
+
stdio: "pipe",
|
|
120
|
+
});
|
|
106
121
|
} catch {
|
|
107
122
|
// session may already be gone
|
|
108
123
|
}
|
|
@@ -143,7 +158,7 @@ async function main() {
|
|
|
143
158
|
if (!opts.dryRun) {
|
|
144
159
|
await wait(2000);
|
|
145
160
|
}
|
|
146
|
-
|
|
161
|
+
|
|
147
162
|
showSummary();
|
|
148
163
|
|
|
149
164
|
if (!opts.keep) {
|
|
@@ -155,7 +170,10 @@ async function main() {
|
|
|
155
170
|
// Normalize both paths to forward-slash for cross-platform comparison
|
|
156
171
|
function isDirectExec() {
|
|
157
172
|
if (!process.argv[1]) return false;
|
|
158
|
-
const scriptPath = new URL(import.meta.url).pathname.replace(
|
|
173
|
+
const scriptPath = new URL(import.meta.url).pathname.replace(
|
|
174
|
+
/^\/([A-Za-z]:)/,
|
|
175
|
+
"$1",
|
|
176
|
+
);
|
|
159
177
|
const argv1 = process.argv[1].replace(/\\/g, "/");
|
|
160
178
|
const norm = scriptPath.replace(/\\/g, "/");
|
|
161
179
|
return argv1 === norm || argv1.endsWith(norm);
|