triflux 3.3.0-dev.7 → 4.0.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.
Files changed (91) hide show
  1. package/README.ko.md +108 -199
  2. package/README.md +108 -199
  3. package/bin/triflux.mjs +2415 -1762
  4. package/hooks/keyword-rules.json +361 -354
  5. package/hooks/pipeline-stop.mjs +5 -2
  6. package/hub/assign-callbacks.mjs +136 -136
  7. package/hub/bridge.mjs +734 -708
  8. package/hub/delegator/contracts.mjs +38 -0
  9. package/hub/delegator/index.mjs +14 -0
  10. package/hub/delegator/schema/delegator-tools.schema.json +250 -0
  11. package/hub/delegator/service.mjs +302 -0
  12. package/hub/delegator/tool-definitions.mjs +35 -0
  13. package/hub/hitl.mjs +67 -67
  14. package/hub/paths.mjs +28 -0
  15. package/hub/pipe.mjs +589 -561
  16. package/hub/pipeline/state.mjs +23 -0
  17. package/hub/public/dashboard.html +349 -0
  18. package/hub/public/tray-icon.ico +0 -0
  19. package/hub/public/tray-icon.png +0 -0
  20. package/hub/router.mjs +782 -782
  21. package/hub/schema.sql +40 -40
  22. package/hub/server.mjs +810 -637
  23. package/hub/store.mjs +706 -706
  24. package/hub/team/cli/commands/attach.mjs +37 -0
  25. package/hub/team/cli/commands/control.mjs +43 -0
  26. package/hub/team/cli/commands/debug.mjs +74 -0
  27. package/hub/team/cli/commands/focus.mjs +53 -0
  28. package/hub/team/cli/commands/interrupt.mjs +36 -0
  29. package/hub/team/cli/commands/kill.mjs +37 -0
  30. package/hub/team/cli/commands/list.mjs +24 -0
  31. package/hub/team/cli/commands/send.mjs +37 -0
  32. package/hub/team/cli/commands/start/index.mjs +87 -0
  33. package/hub/team/cli/commands/start/parse-args.mjs +32 -0
  34. package/hub/team/cli/commands/start/start-in-process.mjs +40 -0
  35. package/hub/team/cli/commands/start/start-mux.mjs +73 -0
  36. package/hub/team/cli/commands/start/start-wt.mjs +69 -0
  37. package/hub/team/cli/commands/status.mjs +87 -0
  38. package/hub/team/cli/commands/stop.mjs +31 -0
  39. package/hub/team/cli/commands/task.mjs +30 -0
  40. package/hub/team/cli/commands/tasks.mjs +13 -0
  41. package/hub/team/{cli.mjs → cli/help.mjs} +38 -99
  42. package/hub/team/cli/index.mjs +39 -0
  43. package/hub/team/cli/manifest.mjs +28 -0
  44. package/hub/team/cli/render.mjs +30 -0
  45. package/hub/team/cli/services/attach-fallback.mjs +54 -0
  46. package/hub/team/cli/services/hub-client.mjs +171 -0
  47. package/hub/team/cli/services/member-selector.mjs +30 -0
  48. package/hub/team/cli/services/native-control.mjs +115 -0
  49. package/hub/team/cli/services/runtime-mode.mjs +60 -0
  50. package/hub/team/cli/services/state-store.mjs +34 -0
  51. package/hub/team/cli/services/task-model.mjs +30 -0
  52. package/hub/team/native-supervisor.mjs +69 -63
  53. package/hub/team/native.mjs +367 -266
  54. package/hub/team/nativeProxy.mjs +217 -173
  55. package/hub/team/pane.mjs +149 -149
  56. package/hub/team/psmux.mjs +946 -946
  57. package/hub/team/session.mjs +608 -608
  58. package/hub/team/staleState.mjs +369 -299
  59. package/hub/tools.mjs +107 -107
  60. package/hub/tray.mjs +332 -0
  61. package/hub/workers/claude-worker.mjs +446 -446
  62. package/hub/workers/codex-mcp.mjs +414 -414
  63. package/hub/workers/delegator-mcp.mjs +1045 -1045
  64. package/hub/workers/factory.mjs +21 -21
  65. package/hub/workers/gemini-worker.mjs +349 -349
  66. package/hub/workers/interface.mjs +41 -41
  67. package/package.json +61 -60
  68. package/scripts/__tests__/keyword-detector.test.mjs +234 -234
  69. package/scripts/hub-ensure.mjs +102 -101
  70. package/scripts/keyword-detector.mjs +272 -272
  71. package/scripts/keyword-rules-expander.mjs +521 -521
  72. package/scripts/lib/keyword-rules.mjs +168 -168
  73. package/scripts/lib/mcp-filter.mjs +642 -642
  74. package/scripts/lib/mcp-server-catalog.mjs +118 -118
  75. package/scripts/mcp-check.mjs +126 -126
  76. package/scripts/preflight-cache.mjs +19 -0
  77. package/scripts/run.cjs +62 -62
  78. package/scripts/setup.mjs +68 -31
  79. package/scripts/test-tfx-route-no-claude-native.mjs +57 -57
  80. package/scripts/tfx-route-worker.mjs +161 -161
  81. package/scripts/tfx-route.sh +1360 -1326
  82. package/skills/tfx-auto/SKILL.md +196 -196
  83. package/skills/tfx-auto-codex/SKILL.md +77 -77
  84. package/skills/tfx-multi/SKILL.md +378 -378
  85. package/hub/team/cli-team-common.mjs +0 -348
  86. package/hub/team/cli-team-control.mjs +0 -393
  87. package/hub/team/cli-team-start.mjs +0 -516
  88. package/hub/team/cli-team-status.mjs +0 -283
  89. package/skills/auto-verify/SKILL.md +0 -145
  90. package/skills/manage-skills/SKILL.md +0 -192
  91. package/skills/verify-implementation/SKILL.md +0 -138
@@ -0,0 +1,37 @@
1
+ import { attachSession } from "../../session.mjs";
2
+ import { DIM, RESET } from "../../shared.mjs";
3
+ import { buildManualAttachCommand, launchAttachInWindowsTerminal, wantsWtAttachFallback } from "../services/attach-fallback.mjs";
4
+ import { isNativeMode, isTeamAlive, isWtMode } from "../services/runtime-mode.mjs";
5
+ import { loadTeamState } from "../services/state-store.mjs";
6
+ import { fail, ok, warn } from "../render.mjs";
7
+
8
+ export async function teamAttach(args = []) {
9
+ const state = loadTeamState();
10
+ if (!state || !isTeamAlive(state)) {
11
+ console.log(`\n ${DIM}활성 팀 세션 없음${RESET}\n`);
12
+ return;
13
+ }
14
+ if (isNativeMode(state)) {
15
+ console.log(`\n ${DIM}in-process 모드는 별도 attach가 없습니다.${RESET}\n ${DIM}상태 확인: tfx multi status${RESET}\n`);
16
+ return;
17
+ }
18
+ if (isWtMode(state)) {
19
+ console.log(`\n ${DIM}wt 모드는 attach 개념이 없습니다 (Windows Terminal pane가 독립 실행됨).${RESET}\n ${DIM}재실행/정리는: tfx multi stop${RESET}\n`);
20
+ return;
21
+ }
22
+
23
+ try {
24
+ attachSession(state.sessionName);
25
+ } catch (error) {
26
+ const allowWt = wantsWtAttachFallback(args);
27
+ if (allowWt && await launchAttachInWindowsTerminal(state.sessionName)) {
28
+ warn(`현재 터미널에서 attach 실패: ${error.message}`);
29
+ ok("Windows Terminal split-pane로 attach 재시도 창을 열었습니다.");
30
+ console.log(` ${DIM}수동 attach 명령: ${buildManualAttachCommand(state.sessionName)}${RESET}\n`);
31
+ return;
32
+ }
33
+ fail(`attach 실패: ${error.message}`);
34
+ warn(allowWt ? "WT 분할창 attach 자동 검증 실패 (session_attached 증가 없음)" : "자동 WT 분할은 기본 비활성입니다. 필요 시 --wt 옵션으로 실행하세요.");
35
+ console.log(` ${DIM}수동 attach 명령: ${buildManualAttachCommand(state.sessionName)}${RESET}\n`);
36
+ }
37
+ }
@@ -0,0 +1,43 @@
1
+ import { injectPrompt, sendKeys } from "../../pane.mjs";
2
+ import { DIM, RESET, WHITE } from "../../shared.mjs";
3
+ import { publishLeadControl } from "../services/hub-client.mjs";
4
+ import { resolveMember } from "../services/member-selector.mjs";
5
+ import { nativeRequest } from "../services/native-control.mjs";
6
+ import { isNativeMode, isTeamAlive, isWtMode } from "../services/runtime-mode.mjs";
7
+ import { loadTeamState } from "../services/state-store.mjs";
8
+ import { ok, warn } from "../render.mjs";
9
+
10
+ export async function teamControl(args = []) {
11
+ const state = loadTeamState();
12
+ if (!state || !isTeamAlive(state)) {
13
+ console.log(`\n ${DIM}활성 팀 세션 없음${RESET}\n`);
14
+ return;
15
+ }
16
+
17
+ const member = resolveMember(state, args[0]);
18
+ const command = String(args[1] || "").toLowerCase();
19
+ const reason = args.slice(2).join(" ");
20
+ if (!member || !new Set(["interrupt", "stop", "pause", "resume"]).has(command)) {
21
+ console.log(`\n 사용법: ${WHITE}tfx multi control <lead|이름|번호> <interrupt|stop|pause|resume> [사유]${RESET}\n`);
22
+ return;
23
+ }
24
+ if (isWtMode(state)) {
25
+ console.log(`\n \x1b[33m⚠\x1b[0m wt 모드는 Hub direct/control 주입 경로가 비활성입니다.\n ${DIM}수동 제어: 해당 pane에서 직접 명령/인터럽트를 수행하세요.${RESET}\n`);
26
+ return;
27
+ }
28
+
29
+ let directOk = false;
30
+ if (isNativeMode(state)) {
31
+ directOk = !!(await nativeRequest(state, "/control", { member: member.name, command, reason }))?.ok;
32
+ } else {
33
+ injectPrompt(member.pane, `[LEAD CONTROL] command=${command}${reason ? ` reason=${reason}` : ""}`);
34
+ if (command === "interrupt") sendKeys(member.pane, "C-c");
35
+ directOk = true;
36
+ }
37
+
38
+ const published = await publishLeadControl(state, member, command, reason);
39
+ if (directOk && published) ok(`${member.name} 제어 전송 (${command}, direct + hub)`);
40
+ else if (directOk) ok(`${member.name} 제어 전송 (${command}, direct only)`);
41
+ else warn(`${member.name} 제어 전송 실패 (${command})`);
42
+ console.log("");
43
+ }
@@ -0,0 +1,74 @@
1
+ import { AMBER, BOLD, DIM, RESET } from "../../shared.mjs";
2
+ import {
3
+ capturePaneOutput,
4
+ detectMultiplexer,
5
+ getSessionAttachedCount,
6
+ hasWindowsTerminal,
7
+ hasWindowsTerminalSession,
8
+ listSessions,
9
+ } from "../../session.mjs";
10
+ import { getHubInfo, nativeGetStatus } from "../services/hub-client.mjs";
11
+ import { isNativeMode, isTeamAlive, isWtMode } from "../services/runtime-mode.mjs";
12
+ import { loadTeamState, TEAM_PROFILE } from "../services/state-store.mjs";
13
+ import { formatCompletionSuffix } from "../render.mjs";
14
+
15
+ export async function teamDebug(args = []) {
16
+ const state = loadTeamState();
17
+ const flagIndex = args.findIndex((arg) => arg === "--lines" || arg === "-n");
18
+ const lines = flagIndex === -1 ? 20 : Math.max(3, parseInt(args[flagIndex + 1] || "20", 10) || 20);
19
+ const hub = await getHubInfo();
20
+
21
+ console.log(`\n ${AMBER}${BOLD}⬡ Team Debug${RESET}\n`);
22
+ console.log(` platform: ${process.platform}`);
23
+ console.log(` node: ${process.version}`);
24
+ console.log(` tty: stdout=${!!process.stdout.isTTY}, stdin=${!!process.stdin.isTTY}`);
25
+ console.log(` mux: ${detectMultiplexer() || "none"}`);
26
+ console.log(` hub-pid: ${hub ? `${hub.pid}` : "-"}`);
27
+ console.log(` hub-url: ${hub?.url || "-"}`);
28
+ const sessions = listSessions();
29
+ console.log(` sessions: ${sessions.length ? sessions.join(", ") : "-"}`);
30
+
31
+ if (!state) {
32
+ console.log(`\n ${DIM}team-state 없음 (활성 세션 없음)${RESET}\n`);
33
+ return;
34
+ }
35
+
36
+ console.log(`\n ${BOLD}state${RESET}`);
37
+ console.log(` session: ${state.sessionName}`);
38
+ console.log(` profile: ${state.profile || TEAM_PROFILE}`);
39
+ console.log(` mode: ${state.teammateMode || "tmux"}`);
40
+ console.log(` lead: ${state.lead}`);
41
+ console.log(` agents: ${(state.agents || []).join(", ")}`);
42
+ console.log(` alive: ${isTeamAlive(state) ? "yes" : "no"}`);
43
+ console.log(` attached: ${getSessionAttachedCount(state.sessionName) ?? "-"}`);
44
+
45
+ if (isWtMode(state)) {
46
+ console.log(`\n ${BOLD}wt-session${RESET}`);
47
+ console.log(` window: ${state?.wt?.windowId ?? 0}`);
48
+ console.log(` layout: ${state?.wt?.layout || state?.layout || "-"}`);
49
+ console.log(` panes: ${state?.wt?.paneCount ?? (state.members || []).length}`);
50
+ console.log(` wt.exe: ${hasWindowsTerminal() ? "yes" : "no"}`);
51
+ console.log(` WT_SESSION:${hasWindowsTerminalSession() ? "yes" : "no"}`);
52
+ console.log("");
53
+ return;
54
+ }
55
+
56
+ if (isNativeMode(state)) {
57
+ console.log(`\n ${BOLD}native-members${RESET}`);
58
+ const members = (await nativeGetStatus(state))?.data?.members || [];
59
+ if (!members.length) console.log(` ${DIM}(no data)${RESET}`);
60
+ for (const member of members) console.log(` - ${member.name}: ${member.status}${formatCompletionSuffix(member)}${member.lastPreview ? ` ${DIM}${member.lastPreview}${RESET}` : ""}`);
61
+ console.log("");
62
+ return;
63
+ }
64
+
65
+ console.log(`\n ${BOLD}pane-tail${RESET} ${DIM}(last ${lines} lines)${RESET}`);
66
+ if (!(state.members || []).length) console.log(` ${DIM}(members 없음)${RESET}`);
67
+ for (const member of state.members || []) {
68
+ console.log(`\n [${member.name}] ${member.pane}`);
69
+ for (const line of (capturePaneOutput(member.pane, lines) || "(empty)").split("\n").slice(-lines)) {
70
+ console.log(` ${line}`);
71
+ }
72
+ }
73
+ console.log("");
74
+ }
@@ -0,0 +1,53 @@
1
+ import { attachSession, focusPane, focusWtPane } from "../../session.mjs";
2
+ import { DIM, RESET, WHITE } from "../../shared.mjs";
3
+ import { buildManualAttachCommand, launchAttachInWindowsTerminal, wantsWtAttachFallback } from "../services/attach-fallback.mjs";
4
+ import { resolveMember } from "../services/member-selector.mjs";
5
+ import { isNativeMode, isTeamAlive, isWtMode } from "../services/runtime-mode.mjs";
6
+ import { loadTeamState } from "../services/state-store.mjs";
7
+ import { fail, ok, warn } from "../render.mjs";
8
+
9
+ export async function teamFocus(args = []) {
10
+ const state = loadTeamState();
11
+ if (!state || !isTeamAlive(state)) {
12
+ console.log(`\n ${DIM}활성 팀 세션 없음${RESET}\n`);
13
+ return;
14
+ }
15
+ if (isNativeMode(state)) {
16
+ console.log(`\n ${DIM}in-process 모드는 focus/attach 개념이 없습니다.${RESET}\n ${DIM}직접 지시: tfx multi send <대상> "메시지"${RESET}\n`);
17
+ return;
18
+ }
19
+
20
+ const member = resolveMember(state, args[0]);
21
+ if (!member) {
22
+ console.log(`\n 사용법: ${WHITE}tfx multi focus <lead|이름|번호>${RESET}\n`);
23
+ return;
24
+ }
25
+
26
+ if (isWtMode(state)) {
27
+ const paneIndex = Number(/^wt:(\d+)$/.exec(member.pane || "")?.[1]);
28
+ if (!Number.isFinite(paneIndex)) {
29
+ console.log(`\n ${DIM}wt pane 인덱스 파싱 실패: ${member.pane}${RESET}\n`);
30
+ return;
31
+ }
32
+ if (focusWtPane(paneIndex, { layout: state?.wt?.layout || state?.layout || "1xN" })) ok(`${member.name} pane 포커스 이동 (wt)`);
33
+ else warn("wt pane 포커스 이동 실패 (WT_SESSION/wt.exe 상태 확인 필요)");
34
+ console.log("");
35
+ return;
36
+ }
37
+
38
+ focusPane(member.pane, { zoom: false });
39
+ try {
40
+ attachSession(state.sessionName);
41
+ } catch (error) {
42
+ const allowWt = wantsWtAttachFallback(args);
43
+ if (allowWt && await launchAttachInWindowsTerminal(state.sessionName)) {
44
+ warn(`현재 터미널에서 attach 실패: ${error.message}`);
45
+ ok("Windows Terminal split-pane로 attach 재시도 창을 열었습니다.");
46
+ console.log(` ${DIM}수동 attach 명령: ${buildManualAttachCommand(state.sessionName)}${RESET}\n`);
47
+ return;
48
+ }
49
+ fail(`attach 실패: ${error.message}`);
50
+ warn(allowWt ? "WT 분할창 attach 자동 검증 실패 (session_attached 증가 없음)" : "자동 WT 분할은 기본 비활성입니다. 필요 시 --wt 옵션으로 실행하세요.");
51
+ console.log(` ${DIM}수동 attach 명령: ${buildManualAttachCommand(state.sessionName)}${RESET}\n`);
52
+ }
53
+ }
@@ -0,0 +1,36 @@
1
+ import { sendKeys } from "../../pane.mjs";
2
+ import { DIM, RESET, WHITE } from "../../shared.mjs";
3
+ import { resolveMember } from "../services/member-selector.mjs";
4
+ import { nativeRequest } from "../services/native-control.mjs";
5
+ import { isNativeMode, isTeamAlive, isWtMode } from "../services/runtime-mode.mjs";
6
+ import { loadTeamState } from "../services/state-store.mjs";
7
+ import { ok, warn } from "../render.mjs";
8
+
9
+ export async function teamInterrupt(args = []) {
10
+ const state = loadTeamState();
11
+ if (!state || !isTeamAlive(state)) {
12
+ console.log(`\n ${DIM}활성 팀 세션 없음${RESET}\n`);
13
+ return;
14
+ }
15
+
16
+ const member = resolveMember(state, args[0] || "lead");
17
+ if (!member) {
18
+ console.log(`\n 사용법: ${WHITE}tfx multi interrupt <lead|이름|번호>${RESET}\n`);
19
+ return;
20
+ }
21
+ if (isWtMode(state)) {
22
+ console.log(`\n \x1b[33m⚠\x1b[0m wt 모드에서는 pane stdin 주입이 지원되지 않아 interrupt를 자동 전송할 수 없습니다.\n ${DIM}수동으로 해당 pane에서 Ctrl+C를 입력하세요.${RESET}\n`);
23
+ return;
24
+ }
25
+
26
+ if (isNativeMode(state)) {
27
+ const result = await nativeRequest(state, "/interrupt", { member: member.name });
28
+ (result?.ok ? ok : warn)(`${member.name} ${result?.ok ? "인터럽트 전송" : "인터럽트 실패"}`);
29
+ console.log("");
30
+ return;
31
+ }
32
+
33
+ sendKeys(member.pane, "C-c");
34
+ ok(`${member.name} 인터럽트 전송`);
35
+ console.log("");
36
+ }
@@ -0,0 +1,37 @@
1
+ import { closeWtSession, killSession, listSessions } from "../../session.mjs";
2
+ import { DIM, RESET } from "../../shared.mjs";
3
+ import { nativeRequest } from "../services/native-control.mjs";
4
+ import { isNativeMode, isTeamAlive, isWtMode } from "../services/runtime-mode.mjs";
5
+ import { clearTeamState, loadTeamState } from "../services/state-store.mjs";
6
+ import { ok } from "../render.mjs";
7
+
8
+ export async function teamKill() {
9
+ const state = loadTeamState();
10
+ if (state && isNativeMode(state) && isTeamAlive(state)) {
11
+ await nativeRequest(state, "/stop", {});
12
+ try { process.kill(state.native.supervisorPid, "SIGTERM"); } catch {}
13
+ clearTeamState();
14
+ ok(`종료: ${state.sessionName}`);
15
+ console.log("");
16
+ return;
17
+ }
18
+ if (state && isWtMode(state)) {
19
+ const closed = closeWtSession({ layout: state?.wt?.layout || state?.layout || "1xN", paneCount: state?.wt?.paneCount ?? (state.members || []).length });
20
+ clearTeamState();
21
+ ok(`종료: ${state.sessionName}${closed ? ` (${closed} panes closed)` : ""}`);
22
+ console.log("");
23
+ return;
24
+ }
25
+
26
+ const sessions = listSessions();
27
+ if (!sessions.length) {
28
+ console.log(`\n ${DIM}활성 팀 세션 없음${RESET}\n`);
29
+ return;
30
+ }
31
+ for (const session of sessions) {
32
+ killSession(session);
33
+ ok(`종료: ${session}`);
34
+ }
35
+ clearTeamState();
36
+ console.log("");
37
+ }
@@ -0,0 +1,24 @@
1
+ import { AMBER, BOLD, DIM, GREEN, RESET } from "../../shared.mjs";
2
+ import { listSessions } from "../../session.mjs";
3
+ import { isNativeMode, isTeamAlive, isWtMode } from "../services/runtime-mode.mjs";
4
+ import { loadTeamState } from "../services/state-store.mjs";
5
+
6
+ export function teamList() {
7
+ const state = loadTeamState();
8
+ if (state && isTeamAlive(state) && (isNativeMode(state) || isWtMode(state))) {
9
+ console.log(`\n ${AMBER}${BOLD}⬡ 팀 세션 목록${RESET}\n`);
10
+ console.log(` ${GREEN}●${RESET} ${state.sessionName} ${DIM}(${isNativeMode(state) ? "in-process" : "wt"})${RESET}`);
11
+ console.log("");
12
+ return;
13
+ }
14
+
15
+ const sessions = listSessions();
16
+ if (!sessions.length) {
17
+ console.log(`\n ${DIM}활성 팀 세션 없음${RESET}\n`);
18
+ return;
19
+ }
20
+
21
+ console.log(`\n ${AMBER}${BOLD}⬡ 팀 세션 목록${RESET}\n`);
22
+ for (const session of sessions) console.log(` ${GREEN}●${RESET} ${session}`);
23
+ console.log("");
24
+ }
@@ -0,0 +1,37 @@
1
+ import { injectPrompt } from "../../pane.mjs";
2
+ import { DIM, RESET, WHITE } from "../../shared.mjs";
3
+ import { resolveMember } from "../services/member-selector.mjs";
4
+ import { nativeRequest } from "../services/native-control.mjs";
5
+ import { isNativeMode, isTeamAlive, isWtMode } from "../services/runtime-mode.mjs";
6
+ import { loadTeamState } from "../services/state-store.mjs";
7
+ import { ok, warn } from "../render.mjs";
8
+
9
+ export async function teamSend(args = []) {
10
+ const state = loadTeamState();
11
+ if (!state || !isTeamAlive(state)) {
12
+ console.log(`\n ${DIM}활성 팀 세션 없음${RESET}\n`);
13
+ return;
14
+ }
15
+
16
+ const member = resolveMember(state, args[0]);
17
+ const message = args.slice(1).join(" ");
18
+ if (!member || !message) {
19
+ console.log(`\n 사용법: ${WHITE}tfx multi send <lead|이름|번호> "메시지"${RESET}\n`);
20
+ return;
21
+ }
22
+ if (isWtMode(state)) {
23
+ console.log(`\n \x1b[33m⚠\x1b[0m wt 모드는 pane 프롬프트 자동 주입(send)이 지원되지 않습니다.\n ${DIM}수동 전달: 선택한 pane에 직접 붙여넣으세요.${RESET}\n`);
24
+ return;
25
+ }
26
+
27
+ if (isNativeMode(state)) {
28
+ const result = await nativeRequest(state, "/send", { member: member.name, text: message });
29
+ (result?.ok ? ok : warn)(`${member.name}${result?.ok ? "에 메시지 주입 완료" : " 메시지 주입 실패"}`);
30
+ console.log("");
31
+ return;
32
+ }
33
+
34
+ injectPrompt(member.pane, message);
35
+ ok(`${member.name}에 메시지 주입 완료`);
36
+ console.log("");
37
+ }
@@ -0,0 +1,87 @@
1
+ import { decomposeTask } from "../../../orchestrator.mjs";
2
+ import { hasWindowsTerminal, hasWindowsTerminalSession } from "../../../session.mjs";
3
+ import { AMBER, BOLD, DIM, GREEN, RED, RESET, WHITE } from "../../../shared.mjs";
4
+ import { getDefaultHubUrl, getHubInfo, startHubDaemon } from "../../services/hub-client.mjs";
5
+ import { ensureTmuxOrExit } from "../../services/runtime-mode.mjs";
6
+ import { saveTeamState } from "../../services/state-store.mjs";
7
+ import { fail, ok, warn } from "../../render.mjs";
8
+ import { parseTeamArgs } from "./parse-args.mjs";
9
+ import { startInProcessTeam } from "./start-in-process.mjs";
10
+ import { startMuxTeam } from "./start-mux.mjs";
11
+ import { startWtTeam } from "./start-wt.mjs";
12
+
13
+ function printStartUsage() {
14
+ console.log(`\n ${AMBER}${BOLD}⬡ tfx multi${RESET}\n`);
15
+ console.log(` 사용법: ${WHITE}tfx multi "작업 설명"${RESET}`);
16
+ console.log(` ${WHITE}tfx multi --agents codex,gemini --lead claude "작업"${RESET}`);
17
+ console.log(` ${WHITE}tfx multi --teammate-mode psmux "작업"${RESET} ${DIM}(Windows psmux 네이티브)${RESET}`);
18
+ console.log(` ${WHITE}tfx multi --teammate-mode wt "작업"${RESET} ${DIM}(Windows Terminal split-pane)${RESET}`);
19
+ console.log(` ${WHITE}tfx multi --teammate-mode in-process "작업"${RESET} ${DIM}(mux 불필요)${RESET}\n`);
20
+ }
21
+
22
+ function printWorkerPreview(agents, subtasks) {
23
+ for (let index = 0; index < subtasks.length; index += 1) {
24
+ const preview = subtasks[index].length > 44 ? `${subtasks[index].slice(0, 44)}…` : subtasks[index];
25
+ console.log(` ${DIM}[${agents[index]}-${index + 1}] ${preview}${RESET}`);
26
+ }
27
+ console.log("");
28
+ }
29
+
30
+ function renderTmuxInstallHelp() {
31
+ console.log(`\n ${RED}${BOLD}tmux 미발견${RESET}\n`);
32
+ console.log(" 현재 선택한 모드는 tmux 기반 팀세션이 필요합니다.\n");
33
+ console.log(` 설치:\n WSL2: ${WHITE}wsl sudo apt install tmux${RESET}\n macOS: ${WHITE}brew install tmux${RESET}\n Linux: ${WHITE}apt install tmux${RESET}\n`);
34
+ console.log(` Windows에서는 WSL2를 권장합니다:\n 1. ${WHITE}wsl --install${RESET}\n 2. ${WHITE}wsl sudo apt install tmux${RESET}\n 3. ${WHITE}tfx multi "작업"${RESET}\n`);
35
+ }
36
+
37
+ export { parseTeamArgs };
38
+
39
+ export async function teamStart(args = []) {
40
+ const { agents, lead, layout, teammateMode, task } = parseTeamArgs(args);
41
+ if (!task) return printStartUsage();
42
+
43
+ console.log(`\n ${AMBER}${BOLD}⬡ tfx multi${RESET}\n`);
44
+ let hub = await getHubInfo();
45
+ if (!hub) {
46
+ process.stdout.write(" Hub 시작 중...");
47
+ try { hub = await startHubDaemon(); } catch (error) { if (error?.code === "HUB_SERVER_MISSING") fail("hub/server.mjs 없음 — hub 모듈이 설치되지 않음"); }
48
+ console.log(` ${hub ? `${GREEN}✓${RESET}` : `${RED}✗${RESET}`}`);
49
+ if (!hub) warn("Hub 시작 실패 — 수동으로 실행: tfx hub start");
50
+ } else ok(`Hub: ${DIM}${hub.url}${RESET}`);
51
+
52
+ const sessionId = `tfx-multi-${Date.now().toString(36).slice(-4)}${Math.random().toString(36).slice(2, 6)}`;
53
+ const subtasks = decomposeTask(task, agents.length);
54
+ const hubUrl = hub?.url || getDefaultHubUrl();
55
+ let effectiveMode = teammateMode;
56
+ if (effectiveMode === "wt" && !hasWindowsTerminal()) { warn("wt.exe 미발견 — in-process 모드로 자동 fallback"); effectiveMode = "in-process"; }
57
+ if (effectiveMode === "wt" && !hasWindowsTerminalSession()) { warn("WT_SESSION 미감지(Windows Terminal 외부) — in-process 모드로 자동 fallback"); effectiveMode = "in-process"; }
58
+
59
+ console.log(` 세션: ${WHITE}${sessionId}${RESET}`);
60
+ console.log(` 모드: ${effectiveMode}`);
61
+ console.log(` 리드: ${AMBER}${lead}${RESET}`);
62
+ console.log(` 워커: ${agents.map((agent) => `${AMBER}${agent}${RESET}`).join(", ")}`);
63
+ printWorkerPreview(agents, subtasks);
64
+
65
+ if (effectiveMode === "tmux") {
66
+ try { ensureTmuxOrExit(); } catch { return renderTmuxInstallHelp(); }
67
+ }
68
+
69
+ const state = effectiveMode === "in-process"
70
+ ? await startInProcessTeam({ sessionId, task, lead, agents, subtasks, hubUrl })
71
+ : effectiveMode === "wt"
72
+ ? await startWtTeam({ sessionId, task, lead, agents, subtasks, layout, hubUrl })
73
+ : await startMuxTeam({ sessionId, task, lead, agents, subtasks, layout, hubUrl, teammateMode: effectiveMode });
74
+
75
+ if (!state) return fail("in-process supervisor 시작 실패");
76
+ saveTeamState(state);
77
+ if (typeof state.postSave === "function") state.postSave();
78
+ if (effectiveMode === "in-process") {
79
+ ok("네이티브 in-process 팀 시작 완료");
80
+ console.log(` ${DIM}tmux 없이 실행됨 (직접 CLI 프로세스)${RESET}`);
81
+ console.log(` ${DIM}제어: tfx multi send/control/tasks/status${RESET}\n`);
82
+ } else if (effectiveMode === "wt") {
83
+ ok("Windows Terminal wt 팀 시작 완료");
84
+ console.log(` ${DIM}현재 pane 기준으로 ${state.layout} 분할 생성됨${RESET}`);
85
+ console.log(` ${DIM}wt 모드는 자동 프롬프트 주입/Hub direct 제어(send/control)가 제한됩니다.${RESET}\n`);
86
+ }
87
+ }
@@ -0,0 +1,32 @@
1
+ import { normalizeLayout, normalizeTeammateMode } from "../../services/runtime-mode.mjs";
2
+
3
+ export function parseTeamArgs(args = []) {
4
+ let agents = ["codex", "gemini"];
5
+ let lead = "claude";
6
+ let layout = "2x2";
7
+ let teammateMode = "auto";
8
+ const taskParts = [];
9
+
10
+ for (let index = 0; index < args.length; index += 1) {
11
+ const current = args[index];
12
+ if (current === "--agents" && args[index + 1]) {
13
+ agents = args[++index].split(",").map((value) => value.trim().toLowerCase()).filter(Boolean);
14
+ } else if (current === "--lead" && args[index + 1]) {
15
+ lead = args[++index].trim().toLowerCase();
16
+ } else if (current === "--layout" && args[index + 1]) {
17
+ layout = args[++index];
18
+ } else if ((current === "--teammate-mode" || current === "--mode") && args[index + 1]) {
19
+ teammateMode = args[++index];
20
+ } else if (!current.startsWith("-")) {
21
+ taskParts.push(current);
22
+ }
23
+ }
24
+
25
+ return {
26
+ agents,
27
+ lead,
28
+ layout: normalizeLayout(layout),
29
+ teammateMode: normalizeTeammateMode(teammateMode),
30
+ task: taskParts.join(" ").trim(),
31
+ };
32
+ }
@@ -0,0 +1,40 @@
1
+ import { startNativeSupervisor } from "../../services/native-control.mjs";
2
+ import { buildTasks } from "../../services/task-model.mjs";
3
+
4
+ export async function startInProcessTeam({ sessionId, task, lead, agents, subtasks, hubUrl }) {
5
+ const { runtime, members } = await startNativeSupervisor({
6
+ sessionId,
7
+ task,
8
+ lead,
9
+ agents,
10
+ subtasks,
11
+ hubUrl,
12
+ });
13
+
14
+ if (!runtime?.controlUrl) return null;
15
+
16
+ return {
17
+ sessionName: sessionId,
18
+ task,
19
+ lead,
20
+ agents,
21
+ layout: "native",
22
+ teammateMode: "in-process",
23
+ startedAt: Date.now(),
24
+ hubUrl,
25
+ members: members.map((member, index) => ({
26
+ role: member.role,
27
+ name: member.name,
28
+ cli: member.cli,
29
+ agentId: member.agentId,
30
+ pane: `native:${index}`,
31
+ subtask: member.subtask || null,
32
+ })),
33
+ panes: {},
34
+ tasks: buildTasks(subtasks, members.filter((member) => member.role === "worker")),
35
+ native: {
36
+ controlUrl: runtime.controlUrl,
37
+ supervisorPid: runtime.supervisorPid,
38
+ },
39
+ };
40
+ }
@@ -0,0 +1,73 @@
1
+ import { join } from "node:path";
2
+
3
+ import { buildCliCommand, startCliInPane } from "../../../pane.mjs";
4
+ import { orchestrate } from "../../../orchestrator.mjs";
5
+ import { attachSession, configureTeammateKeybindings, createSession } from "../../../session.mjs";
6
+ import { BOLD, DIM, GREEN, RESET } from "../../../shared.mjs";
7
+ import { toAgentId } from "../../services/member-selector.mjs";
8
+ import { PKG_ROOT, TEAM_PROFILE } from "../../services/state-store.mjs";
9
+ import { buildTasks } from "../../services/task-model.mjs";
10
+ import { ok, warn } from "../../render.mjs";
11
+
12
+ export async function startMuxTeam({ sessionId, task, lead, agents, subtasks, layout, hubUrl, teammateMode }) {
13
+ const paneCount = agents.length + 1;
14
+ const effectiveLayout = paneCount <= 4 ? layout : (layout === "Nx1" ? "Nx1" : "1xN");
15
+ console.log(` 레이아웃: ${effectiveLayout} (${paneCount} panes)`);
16
+
17
+ const session = createSession(sessionId, { layout: effectiveLayout, paneCount });
18
+ const leadTarget = session.panes[0];
19
+ startCliInPane(leadTarget, buildCliCommand(lead));
20
+
21
+ const members = [{ role: "lead", name: "lead", cli: lead, pane: leadTarget, agentId: toAgentId(lead, leadTarget) }];
22
+ const assignments = [];
23
+ for (let index = 0; index < agents.length; index += 1) {
24
+ const cli = agents[index];
25
+ const pane = session.panes[index + 1];
26
+ startCliInPane(pane, buildCliCommand(cli));
27
+ const worker = { role: "worker", name: `${cli}-${index + 1}`, cli, pane, subtask: subtasks[index], agentId: toAgentId(cli, pane) };
28
+ members.push(worker);
29
+ assignments.push({ target: pane, cli, subtask: subtasks[index] });
30
+ }
31
+
32
+ ok("CLI 초기화 대기 (3초)...");
33
+ await new Promise((resolve) => setTimeout(resolve, 3000));
34
+ await orchestrate(sessionId, assignments, { hubUrl, teammateMode, lead: { target: leadTarget, cli: lead, task } });
35
+ ok("리드/워커 프롬프트 주입 완료");
36
+
37
+ return {
38
+ sessionName: sessionId,
39
+ task,
40
+ lead,
41
+ agents,
42
+ layout: effectiveLayout,
43
+ teammateMode,
44
+ startedAt: Date.now(),
45
+ hubUrl,
46
+ members,
47
+ panes: Object.fromEntries(members.map((member) => [member.pane, {
48
+ role: member.role,
49
+ name: member.name,
50
+ cli: member.cli,
51
+ agentId: member.agentId,
52
+ subtask: member.subtask || null,
53
+ }])),
54
+ tasks: buildTasks(subtasks, members.filter((member) => member.role === "worker")),
55
+ postSave() {
56
+ const profilePrefix = TEAM_PROFILE === "team" ? "" : `TFX_TEAM_PROFILE=${TEAM_PROFILE} `;
57
+ const taskListCommand = `${profilePrefix}${process.execPath} ${join(PKG_ROOT, "bin", "triflux.mjs")} team tasks`;
58
+ configureTeammateKeybindings(sessionId, { inProcess: false, taskListCommand });
59
+ console.log(`\n ${GREEN}${BOLD}팀 세션 준비 완료${RESET}`);
60
+ console.log(` ${DIM}Shift+Down: 다음 팀메이트 전환${RESET}`);
61
+ console.log(` ${DIM}Shift+Tab / Shift+Left: 이전 팀메이트 전환${RESET}`);
62
+ console.log(` ${DIM}Escape: 현재 팀메이트 인터럽트${RESET}`);
63
+ console.log(` ${DIM}Ctrl+T: 태스크 목록${RESET}`);
64
+ console.log(` ${DIM}참고: Shift+Up은 Claude Code 미지원 (scroll-up 충돌). Shift+Tab 사용${RESET}`);
65
+ console.log(` ${DIM}Ctrl+B → D: 세션 분리 (백그라운드)${RESET}\n`);
66
+ if (process.stdout.isTTY && process.stdin.isTTY) attachSession(sessionId);
67
+ else {
68
+ warn("TTY 미지원 환경이라 자동 attach를 생략함");
69
+ console.log(` ${DIM}수동 연결: tfx multi attach${RESET}\n`);
70
+ }
71
+ },
72
+ };
73
+ }
@@ -0,0 +1,69 @@
1
+ import { createWtSession } from "../../../session.mjs";
2
+ import { buildCliCommand } from "../../../pane.mjs";
3
+ import { toAgentId } from "../../services/member-selector.mjs";
4
+ import { buildTasks } from "../../services/task-model.mjs";
5
+ import { warn } from "../../render.mjs";
6
+
7
+ export async function startWtTeam({ sessionId, task, lead, agents, subtasks, layout, hubUrl }) {
8
+ const paneCount = agents.length + 1;
9
+ const effectiveLayout = layout === "Nx1" ? "Nx1" : "1xN";
10
+ if (layout !== effectiveLayout) warn(`wt 모드에서 ${layout} 레이아웃은 미지원 — ${effectiveLayout}로 대체`);
11
+ console.log(` 레이아웃: ${effectiveLayout} (${paneCount} panes)`);
12
+
13
+ const session = createWtSession(sessionId, {
14
+ layout: effectiveLayout,
15
+ paneCommands: [
16
+ { title: `${sessionId}-lead`, command: buildCliCommand(lead) },
17
+ ...agents.map((cli, index) => ({
18
+ title: `${sessionId}-${cli}-${index + 1}`,
19
+ command: buildCliCommand(cli),
20
+ })),
21
+ ],
22
+ });
23
+
24
+ const members = [
25
+ {
26
+ role: "lead",
27
+ name: "lead",
28
+ cli: lead,
29
+ pane: session.panes[0] || "wt:0",
30
+ agentId: toAgentId(lead, session.panes[0] || "wt:0"),
31
+ },
32
+ ...agents.map((cli, index) => {
33
+ const pane = session.panes[index + 1] || `wt:${index + 1}`;
34
+ return {
35
+ role: "worker",
36
+ name: `${cli}-${index + 1}`,
37
+ cli,
38
+ pane,
39
+ subtask: subtasks[index],
40
+ agentId: toAgentId(cli, pane),
41
+ };
42
+ }),
43
+ ];
44
+
45
+ return {
46
+ sessionName: sessionId,
47
+ task,
48
+ lead,
49
+ agents,
50
+ layout: effectiveLayout,
51
+ teammateMode: "wt",
52
+ startedAt: Date.now(),
53
+ hubUrl,
54
+ members,
55
+ panes: Object.fromEntries(members.map((member) => [member.pane, {
56
+ role: member.role,
57
+ name: member.name,
58
+ cli: member.cli,
59
+ agentId: member.agentId,
60
+ subtask: member.subtask || null,
61
+ }])),
62
+ tasks: buildTasks(subtasks, members.filter((member) => member.role === "worker")),
63
+ wt: {
64
+ windowId: 0,
65
+ layout: effectiveLayout,
66
+ paneCount: session.paneCount,
67
+ },
68
+ };
69
+ }