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
|
@@ -1,19 +1,33 @@
|
|
|
1
|
-
import {
|
|
1
|
+
import {
|
|
2
|
+
copyFileSync,
|
|
3
|
+
existsSync,
|
|
4
|
+
mkdirSync,
|
|
5
|
+
readFileSync,
|
|
6
|
+
writeFileSync,
|
|
7
|
+
} from "node:fs";
|
|
2
8
|
import { homedir } from "node:os";
|
|
3
9
|
import { join } from "node:path";
|
|
4
10
|
|
|
5
11
|
const DEFAULT_GEMINI_PROFILES = {
|
|
6
12
|
model: "gemini-3.1-pro-preview",
|
|
7
13
|
profiles: {
|
|
8
|
-
pro31: {
|
|
9
|
-
|
|
14
|
+
pro31: {
|
|
15
|
+
model: "gemini-3.1-pro-preview",
|
|
16
|
+
hint: "3.1 Pro — 플래그십 (1M ctx, 멀티모달)",
|
|
17
|
+
},
|
|
18
|
+
flash3: {
|
|
19
|
+
model: "gemini-3-flash-preview",
|
|
20
|
+
hint: "3.0 Flash — 빠른 응답, 비용 효율",
|
|
21
|
+
},
|
|
10
22
|
pro25: { model: "gemini-2.5-pro", hint: "2.5 Pro — 안정 (추론 강화)" },
|
|
11
23
|
flash25: { model: "gemini-2.5-flash", hint: "2.5 Flash — 경량 범용" },
|
|
12
24
|
lite25: { model: "gemini-2.5-flash-lite", hint: "2.5 Flash Lite — 최경량" },
|
|
13
25
|
},
|
|
14
26
|
};
|
|
15
27
|
|
|
16
|
-
const DEFAULT_PROFILE_COUNT = Object.keys(
|
|
28
|
+
const DEFAULT_PROFILE_COUNT = Object.keys(
|
|
29
|
+
DEFAULT_GEMINI_PROFILES.profiles,
|
|
30
|
+
).length;
|
|
17
31
|
|
|
18
32
|
function ensureGeminiProfiles({
|
|
19
33
|
geminiDir = join(homedir(), ".gemini"),
|
|
@@ -23,7 +37,11 @@ function ensureGeminiProfiles({
|
|
|
23
37
|
if (!existsSync(geminiDir)) mkdirSync(geminiDir, { recursive: true });
|
|
24
38
|
|
|
25
39
|
if (!existsSync(profilesPath)) {
|
|
26
|
-
writeFileSync(
|
|
40
|
+
writeFileSync(
|
|
41
|
+
profilesPath,
|
|
42
|
+
JSON.stringify(DEFAULT_GEMINI_PROFILES, null, 2) + "\n",
|
|
43
|
+
{ encoding: "utf8", mode: 0o600 },
|
|
44
|
+
);
|
|
27
45
|
return {
|
|
28
46
|
ok: true,
|
|
29
47
|
created: true,
|
|
@@ -37,8 +55,14 @@ function ensureGeminiProfiles({
|
|
|
37
55
|
try {
|
|
38
56
|
cfg = JSON.parse(readFileSync(profilesPath, "utf8"));
|
|
39
57
|
} catch {
|
|
40
|
-
try {
|
|
41
|
-
|
|
58
|
+
try {
|
|
59
|
+
copyFileSync(profilesPath, profilesPath + `.bak.${Date.now()}`);
|
|
60
|
+
} catch {}
|
|
61
|
+
writeFileSync(
|
|
62
|
+
profilesPath,
|
|
63
|
+
JSON.stringify(DEFAULT_GEMINI_PROFILES, null, 2) + "\n",
|
|
64
|
+
{ encoding: "utf8", mode: 0o600 },
|
|
65
|
+
);
|
|
42
66
|
return {
|
|
43
67
|
ok: true,
|
|
44
68
|
created: true,
|
|
@@ -49,10 +73,17 @@ function ensureGeminiProfiles({
|
|
|
49
73
|
}
|
|
50
74
|
|
|
51
75
|
if (!cfg || typeof cfg !== "object" || Array.isArray(cfg)) cfg = {};
|
|
52
|
-
if (
|
|
76
|
+
if (
|
|
77
|
+
!cfg.profiles ||
|
|
78
|
+
typeof cfg.profiles !== "object" ||
|
|
79
|
+
Array.isArray(cfg.profiles)
|
|
80
|
+
)
|
|
81
|
+
cfg.profiles = {};
|
|
53
82
|
|
|
54
83
|
let added = 0;
|
|
55
|
-
for (const [name, value] of Object.entries(
|
|
84
|
+
for (const [name, value] of Object.entries(
|
|
85
|
+
DEFAULT_GEMINI_PROFILES.profiles,
|
|
86
|
+
)) {
|
|
56
87
|
if (!cfg.profiles[name]) {
|
|
57
88
|
cfg.profiles[name] = value;
|
|
58
89
|
added++;
|
|
@@ -61,7 +92,10 @@ function ensureGeminiProfiles({
|
|
|
61
92
|
if (!cfg.model) cfg.model = DEFAULT_GEMINI_PROFILES.model;
|
|
62
93
|
|
|
63
94
|
if (added > 0) {
|
|
64
|
-
writeFileSync(profilesPath, JSON.stringify(cfg, null, 2) + "\n", {
|
|
95
|
+
writeFileSync(profilesPath, JSON.stringify(cfg, null, 2) + "\n", {
|
|
96
|
+
encoding: "utf8",
|
|
97
|
+
mode: 0o600,
|
|
98
|
+
});
|
|
65
99
|
}
|
|
66
100
|
|
|
67
101
|
return {
|
package/scripts/lib/handoff.mjs
CHANGED
|
@@ -18,7 +18,9 @@ function runCommand(command, cwd, executor = execSync) {
|
|
|
18
18
|
}
|
|
19
19
|
|
|
20
20
|
function parseStatusLines(statusText) {
|
|
21
|
-
const lines = String(statusText || "")
|
|
21
|
+
const lines = String(statusText || "")
|
|
22
|
+
.split(/\r?\n/u)
|
|
23
|
+
.filter(Boolean);
|
|
22
24
|
const changedFiles = [];
|
|
23
25
|
const status = [];
|
|
24
26
|
|
|
@@ -28,7 +30,9 @@ function parseStatusLines(statusText) {
|
|
|
28
30
|
|
|
29
31
|
const rawStatus = normalized.slice(0, 2).trim() || "??";
|
|
30
32
|
const rawPath = normalized.slice(3).trim();
|
|
31
|
-
const filePath = rawPath.includes(" -> ")
|
|
33
|
+
const filePath = rawPath.includes(" -> ")
|
|
34
|
+
? rawPath.split(" -> ").at(-1)?.trim()
|
|
35
|
+
: rawPath;
|
|
32
36
|
if (!filePath) continue;
|
|
33
37
|
|
|
34
38
|
changedFiles.push(filePath);
|
|
@@ -82,22 +86,30 @@ function formatAheadBehind(value) {
|
|
|
82
86
|
|
|
83
87
|
export function collectHandoffContext(options = {}) {
|
|
84
88
|
const cwd = resolve(options.cwd || process.cwd());
|
|
85
|
-
const executor =
|
|
89
|
+
const executor =
|
|
90
|
+
typeof options.commandRunner === "function"
|
|
91
|
+
? options.commandRunner
|
|
92
|
+
: execSync;
|
|
86
93
|
const target = options.target === "local" ? "local" : "remote";
|
|
87
94
|
const decisions = normalizeDecisions(options.decisions, options.decisionFile);
|
|
88
95
|
const generatedAt = options.generatedAt || new Date().toISOString();
|
|
89
96
|
const claudeMdPaths = Array.isArray(options.claudeMdPaths)
|
|
90
97
|
? options.claudeMdPaths
|
|
91
98
|
: findAllClaudeMdPaths({
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
99
|
+
cwd,
|
|
100
|
+
homeDir: options.homeDir,
|
|
101
|
+
});
|
|
95
102
|
|
|
96
103
|
const gitRoot = runCommand("git rev-parse --show-toplevel", cwd, executor);
|
|
97
|
-
const branch =
|
|
104
|
+
const branch =
|
|
105
|
+
runCommand("git rev-parse --abbrev-ref HEAD", cwd, executor) || null;
|
|
98
106
|
const shortStatus = runCommand("git status --short", cwd, executor);
|
|
99
107
|
const diffStat = runCommand("git diff --stat --no-color", cwd, executor);
|
|
100
|
-
const upstreamRaw = runCommand(
|
|
108
|
+
const upstreamRaw = runCommand(
|
|
109
|
+
"git rev-list --left-right --count @{upstream}...HEAD",
|
|
110
|
+
cwd,
|
|
111
|
+
executor,
|
|
112
|
+
);
|
|
101
113
|
const parsedStatus = parseStatusLines(shortStatus);
|
|
102
114
|
|
|
103
115
|
return {
|
|
@@ -122,15 +134,19 @@ export function buildHandoffPrompt(context) {
|
|
|
122
134
|
const upstream = safeContext.upstream
|
|
123
135
|
? `ahead ${safeContext.upstream.ahead}, behind ${safeContext.upstream.behind}`
|
|
124
136
|
: "unknown";
|
|
125
|
-
const changedFiles =
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
|
|
137
|
+
const changedFiles =
|
|
138
|
+
safeContext.changedFiles.length > 0
|
|
139
|
+
? safeContext.changedFiles.map((file) => `- ${file}`).join("\n")
|
|
140
|
+
: "- 변경 파일 없음";
|
|
141
|
+
const decisions =
|
|
142
|
+
safeContext.decisions.length > 0
|
|
143
|
+
? safeContext.decisions.map((decision) => `- ${decision}`).join("\n")
|
|
144
|
+
: "- 명시된 결정사항 없음";
|
|
145
|
+
const claudeMdList =
|
|
146
|
+
Array.isArray(safeContext.claudeMdPaths) &&
|
|
147
|
+
safeContext.claudeMdPaths.length > 0
|
|
148
|
+
? safeContext.claudeMdPaths.map((path) => `- ${path}`).join("\n")
|
|
149
|
+
: "- 자동 탐지된 CLAUDE.md 없음";
|
|
134
150
|
const diffStat = safeContext.diffStat || "(diff stat 없음)";
|
|
135
151
|
|
|
136
152
|
return [
|
|
@@ -1,10 +1,12 @@
|
|
|
1
1
|
export function nudge(message) {
|
|
2
|
-
process.stdout.write(
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
2
|
+
process.stdout.write(
|
|
3
|
+
JSON.stringify({
|
|
4
|
+
hookSpecificOutput: {
|
|
5
|
+
hookEventName: "PreToolUse",
|
|
6
|
+
additionalContext: message,
|
|
7
|
+
},
|
|
8
|
+
}),
|
|
9
|
+
);
|
|
8
10
|
process.exit(0);
|
|
9
11
|
}
|
|
10
12
|
|
|
@@ -28,21 +28,31 @@ function normalizeRule(rule) {
|
|
|
28
28
|
if (!rule || typeof rule !== "object") return null;
|
|
29
29
|
if (typeof rule.id !== "string" || !rule.id.trim()) return null;
|
|
30
30
|
if (!Array.isArray(rule.patterns) || rule.patterns.length === 0) return null;
|
|
31
|
-
if (typeof rule.priority !== "number" || !Number.isFinite(rule.priority))
|
|
31
|
+
if (typeof rule.priority !== "number" || !Number.isFinite(rule.priority))
|
|
32
|
+
return null;
|
|
32
33
|
|
|
33
34
|
const patterns = rule.patterns.map(normalizePattern).filter(Boolean);
|
|
34
35
|
if (patterns.length === 0) return null;
|
|
35
36
|
|
|
36
|
-
const skill =
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
37
|
+
const skill =
|
|
38
|
+
typeof rule.skill === "string" && rule.skill.trim()
|
|
39
|
+
? rule.skill.trim()
|
|
40
|
+
: null;
|
|
41
|
+
const action =
|
|
42
|
+
typeof rule.action === "string" && rule.action.trim()
|
|
43
|
+
? rule.action.trim()
|
|
44
|
+
: null;
|
|
45
|
+
const mcpRoute =
|
|
46
|
+
typeof rule.mcp_route === "string" && VALID_MCP_ROUTES.has(rule.mcp_route)
|
|
47
|
+
? rule.mcp_route
|
|
48
|
+
: null;
|
|
41
49
|
|
|
42
50
|
if (!skill && !mcpRoute && !action) return null;
|
|
43
51
|
|
|
44
52
|
const supersedes = Array.isArray(rule.supersedes)
|
|
45
|
-
? rule.supersedes
|
|
53
|
+
? rule.supersedes
|
|
54
|
+
.filter((id) => typeof id === "string" && id.trim())
|
|
55
|
+
.map((id) => id.trim())
|
|
46
56
|
: [];
|
|
47
57
|
|
|
48
58
|
const state = normalizeState(rule.state);
|
|
@@ -57,7 +67,7 @@ function normalizeRule(rule) {
|
|
|
57
67
|
supersedes,
|
|
58
68
|
exclusive: rule.exclusive === true,
|
|
59
69
|
state,
|
|
60
|
-
mcp_route: mcpRoute
|
|
70
|
+
mcp_route: mcpRoute,
|
|
61
71
|
};
|
|
62
72
|
}
|
|
63
73
|
|
|
@@ -81,26 +91,40 @@ export function loadRules(rulesPath) {
|
|
|
81
91
|
|
|
82
92
|
// pattern.source / flags를 RegExp로 컴파일
|
|
83
93
|
export function compileRules(rules) {
|
|
84
|
-
return rules
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
94
|
+
return rules
|
|
95
|
+
.map((rule) => {
|
|
96
|
+
try {
|
|
97
|
+
return {
|
|
98
|
+
...rule,
|
|
99
|
+
compiledPatterns: rule.patterns.map(
|
|
100
|
+
(p) => new RegExp(p.source, p.flags),
|
|
101
|
+
),
|
|
102
|
+
};
|
|
103
|
+
} catch (error) {
|
|
104
|
+
logRuleError(`정규식 컴파일 실패: ${rule.id}`, error);
|
|
105
|
+
return null;
|
|
106
|
+
}
|
|
107
|
+
})
|
|
108
|
+
.filter(Boolean);
|
|
92
109
|
}
|
|
93
110
|
|
|
94
111
|
// 입력 텍스트에서 매칭된 규칙 목록 반환
|
|
95
112
|
export function matchRules(compiledRules, cleanText) {
|
|
96
|
-
if (
|
|
113
|
+
if (
|
|
114
|
+
!Array.isArray(compiledRules) ||
|
|
115
|
+
typeof cleanText !== "string" ||
|
|
116
|
+
!cleanText
|
|
117
|
+
) {
|
|
97
118
|
return [];
|
|
98
119
|
}
|
|
99
120
|
|
|
100
121
|
const matches = [];
|
|
101
122
|
|
|
102
123
|
for (const rule of compiledRules) {
|
|
103
|
-
if (
|
|
124
|
+
if (
|
|
125
|
+
!Array.isArray(rule.compiledPatterns) ||
|
|
126
|
+
rule.compiledPatterns.length === 0
|
|
127
|
+
) {
|
|
104
128
|
continue;
|
|
105
129
|
}
|
|
106
130
|
|
|
@@ -119,7 +143,7 @@ export function matchRules(compiledRules, cleanText) {
|
|
|
119
143
|
supersedes: rule.supersedes || [],
|
|
120
144
|
exclusive: rule.exclusive === true,
|
|
121
145
|
state: rule.state || null,
|
|
122
|
-
mcp_route: rule.mcp_route || null
|
|
146
|
+
mcp_route: rule.mcp_route || null,
|
|
123
147
|
});
|
|
124
148
|
}
|
|
125
149
|
|
package/scripts/lib/logger.mjs
CHANGED
|
@@ -27,17 +27,17 @@
|
|
|
27
27
|
* error — 작업 실패 (CLI 실행 실패, MCP 연결 끊김)
|
|
28
28
|
* fatal — 프로세스 위협 (DB 연결 불가, 포트 충돌)
|
|
29
29
|
*/
|
|
30
|
-
import pino from
|
|
30
|
+
import pino from "pino";
|
|
31
31
|
|
|
32
|
-
const isDev = process.env.NODE_ENV !==
|
|
32
|
+
const isDev = process.env.NODE_ENV !== "production";
|
|
33
33
|
|
|
34
34
|
export const logger = pino({
|
|
35
|
-
level: process.env.LOG_LEVEL || (isDev ?
|
|
35
|
+
level: process.env.LOG_LEVEL || (isDev ? "debug" : "info"),
|
|
36
36
|
|
|
37
37
|
// 모든 로그에 포함되는 기본 필드
|
|
38
38
|
base: {
|
|
39
|
-
service: process.env.SERVICE_NAME ||
|
|
40
|
-
env: process.env.NODE_ENV ||
|
|
39
|
+
service: process.env.SERVICE_NAME || "triflux",
|
|
40
|
+
env: process.env.NODE_ENV || "development",
|
|
41
41
|
},
|
|
42
42
|
|
|
43
43
|
// 레벨을 대문자로 출력 (AI 파싱 용이)
|
|
@@ -51,18 +51,18 @@ export const logger = pino({
|
|
|
51
51
|
// 민감정보 자동 필터링
|
|
52
52
|
redact: {
|
|
53
53
|
paths: [
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
54
|
+
"password",
|
|
55
|
+
"token",
|
|
56
|
+
"apiKey",
|
|
57
|
+
"secret",
|
|
58
|
+
"authorization",
|
|
59
|
+
"*.password",
|
|
60
|
+
"*.token",
|
|
61
|
+
"*.apiKey",
|
|
62
|
+
"*.secret",
|
|
63
|
+
"req.headers.authorization",
|
|
64
|
+
"req.headers.cookie",
|
|
65
|
+
"hubToken",
|
|
66
66
|
],
|
|
67
67
|
remove: true,
|
|
68
68
|
},
|
|
@@ -70,11 +70,11 @@ export const logger = pino({
|
|
|
70
70
|
// 개발 환경: 컬러 콘솔 출력
|
|
71
71
|
transport: isDev
|
|
72
72
|
? {
|
|
73
|
-
target:
|
|
73
|
+
target: "pino-pretty",
|
|
74
74
|
options: {
|
|
75
75
|
colorize: true,
|
|
76
|
-
translateTime:
|
|
77
|
-
ignore:
|
|
76
|
+
translateTime: "yyyy-mm-dd HH:MM:ss",
|
|
77
|
+
ignore: "pid,hostname",
|
|
78
78
|
},
|
|
79
79
|
}
|
|
80
80
|
: undefined,
|
|
@@ -92,14 +92,14 @@ export function createModuleLogger(module) {
|
|
|
92
92
|
}
|
|
93
93
|
|
|
94
94
|
// 정상 종료 시 버퍼 flush 보장
|
|
95
|
-
process.on(
|
|
96
|
-
logger.fatal({ err },
|
|
95
|
+
process.on("uncaughtException", (err) => {
|
|
96
|
+
logger.fatal({ err }, "process.uncaught_exception");
|
|
97
97
|
logger.flush();
|
|
98
98
|
process.exit(1);
|
|
99
99
|
});
|
|
100
100
|
|
|
101
|
-
process.on(
|
|
102
|
-
logger.fatal({ reason: String(reason) },
|
|
101
|
+
process.on("unhandledRejection", (reason) => {
|
|
102
|
+
logger.fatal({ reason: String(reason) }, "process.unhandled_rejection");
|
|
103
103
|
logger.flush();
|
|
104
104
|
process.exit(1);
|
|
105
105
|
});
|