triflux 10.3.2 → 10.3.3

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 (55) hide show
  1. package/.claude-plugin/plugin.json +22 -22
  2. package/LICENSE +21 -21
  3. package/README.ko.md +16 -0
  4. package/README.md +8 -0
  5. package/hooks/hook-registry.json +256 -256
  6. package/hub/adaptive-inject.mjs +1 -1
  7. package/hub/assign-callbacks.mjs +120 -120
  8. package/hub/delegator/index.mjs +14 -14
  9. package/hub/delegator/tool-definitions.mjs +35 -35
  10. package/hub/hitl.mjs +143 -143
  11. package/hub/router.mjs +791 -791
  12. package/hub/session-fingerprint.mjs +1 -1
  13. package/hub/team/cli/commands/attach.mjs +37 -37
  14. package/hub/team/cli/commands/debug.mjs +74 -74
  15. package/hub/team/cli/commands/focus.mjs +53 -53
  16. package/hub/team/cli/commands/list.mjs +24 -24
  17. package/hub/team/cli/commands/start/start-in-process.mjs +40 -40
  18. package/hub/team/cli/commands/start/start-mux.mjs +73 -73
  19. package/hub/team/cli/commands/start/start-wt.mjs +69 -69
  20. package/hub/team/cli/commands/tasks.mjs +13 -13
  21. package/hub/team/cli/render.mjs +30 -30
  22. package/hub/team/cli/services/attach-fallback.mjs +54 -54
  23. package/hub/team/cli/services/member-selector.mjs +30 -30
  24. package/hub/team/cli/services/native-control.mjs +116 -116
  25. package/hub/team/cli/services/task-model.mjs +30 -30
  26. package/hub/team/notify.mjs +1 -1
  27. package/hub/team/orchestrator.mjs +161 -161
  28. package/hub/team/session.mjs +611 -611
  29. package/hub/team/shared.mjs +13 -13
  30. package/hub/tray.mjs +368 -368
  31. package/hub/workers/codex-mcp.mjs +507 -507
  32. package/hub/workers/factory.mjs +21 -21
  33. package/mesh/index.mjs +63 -0
  34. package/mesh/mesh-budget.mjs +128 -0
  35. package/mesh/mesh-heartbeat.mjs +100 -0
  36. package/mesh/mesh-protocol.mjs +96 -0
  37. package/mesh/mesh-queue.mjs +165 -0
  38. package/mesh/mesh-registry.mjs +78 -0
  39. package/mesh/mesh-router.mjs +76 -0
  40. package/package.json +2 -1
  41. package/scripts/completions/tfx.bash +47 -47
  42. package/scripts/completions/tfx.fish +44 -44
  43. package/scripts/completions/tfx.zsh +83 -83
  44. package/scripts/hub-ensure.mjs +120 -120
  45. package/scripts/keyword-detector.mjs +272 -272
  46. package/scripts/keyword-rules-expander.mjs +521 -521
  47. package/scripts/lib/mcp-server-catalog.mjs +118 -118
  48. package/scripts/notion-read.mjs +553 -553
  49. package/scripts/test-tfx-route-no-claude-native.mjs +57 -57
  50. package/scripts/tfx-batch-stats.mjs +96 -96
  51. package/skills/.omc/state/agent-replay-8f0e10a9-9693-4410-96f5-a6b07e8ed995.jsonl +0 -1
  52. package/skills/.omc/state/idle-notif-cooldown.json +0 -3
  53. package/skills/.omc/state/last-tool-error.json +0 -7
  54. package/skills/.omc/state/subagent-tracking.json +0 -7
  55. package/skills/tfx-remote-spawn/references/hosts.json +0 -16
@@ -1,120 +1,120 @@
1
- #!/usr/bin/env node
2
- // SessionStart 훅에서 호출되는 Hub 보장 스크립트.
3
- // - /status 기반 헬스체크
4
- // - 비정상 시 Hub를 detached로 기동
5
-
6
- import { existsSync, readFileSync } from "fs";
7
- import { join, dirname } from "path";
8
- import { homedir } from "os";
9
- import { spawn } from "child_process";
10
- import { fileURLToPath } from "url";
11
-
12
- const LOOPBACK_HOSTS = new Set(["127.0.0.1", "localhost", "::1"]);
13
- const PLUGIN_ROOT = dirname(dirname(fileURLToPath(import.meta.url)));
14
- const HUB_PID_FILE = join(homedir(), ".claude", "cache", "tfx-hub", "hub.pid");
15
-
16
- function formatHostForUrl(host) {
17
- return host.includes(":") ? `[${host}]` : host;
18
- }
19
-
20
- function buildHubBaseUrl(host, port) {
21
- return `http://${formatHostForUrl(host)}:${port}`;
22
- }
23
-
24
- function resolveHubTarget() {
25
- const envPortRaw = Number(process.env.TFX_HUB_PORT || "");
26
- const envPort = Number.isFinite(envPortRaw) && envPortRaw > 0 ? envPortRaw : null;
27
- const target = {
28
- host: "127.0.0.1",
29
- port: envPort || 27888,
30
- };
31
-
32
- if (existsSync(HUB_PID_FILE)) {
33
- try {
34
- const info = JSON.parse(readFileSync(HUB_PID_FILE, "utf8"));
35
- if (!envPort) {
36
- const pidPort = Number(info?.port);
37
- if (Number.isFinite(pidPort) && pidPort > 0) target.port = pidPort;
38
- }
39
- if (typeof info?.host === "string") {
40
- const host = info.host.trim();
41
- if (LOOPBACK_HOSTS.has(host)) target.host = host;
42
- }
43
- } catch {
44
- // ignore parse errors and use env/default
45
- }
46
- }
47
-
48
- return target;
49
- }
50
-
51
- async function isHubHealthy(host, port) {
52
- try {
53
- const res = await fetch(`${buildHubBaseUrl(host, port)}/status`, {
54
- signal: AbortSignal.timeout(1000),
55
- });
56
- if (!res.ok) return false;
57
- const data = await res.json();
58
- return data?.hub?.state === "healthy";
59
- } catch {
60
- return false;
61
- }
62
- }
63
-
64
- function startHubDetached(port) {
65
- const serverPath = join(PLUGIN_ROOT, "hub", "server.mjs");
66
- if (!existsSync(serverPath)) return false;
67
-
68
- try {
69
- const env = { ...process.env, TFX_HUB_PORT: String(port) };
70
- if (process.platform === "win32") {
71
- // Windows: cmd.exe /c start /b → 완전 독립 프로세스 트리 생성
72
- // hook timeout 시 프로세스 트리 킬에서 살아남음
73
- const child = spawn("cmd.exe", ["/c", "start", "/b", "", process.execPath, serverPath], {
74
- env,
75
- stdio: "ignore",
76
- windowsHide: true,
77
- });
78
- child.unref();
79
- } else {
80
- const child = spawn(process.execPath, [serverPath], {
81
- env,
82
- detached: true,
83
- stdio: "ignore",
84
- });
85
- child.unref();
86
- }
87
- return true;
88
- } catch {
89
- return false;
90
- }
91
- }
92
-
93
- /** Hub 기동 후 ready 상태까지 대기 (최대 maxWaitMs) */
94
- async function waitForHubReady(host, port, maxWaitMs = 5000) {
95
- const interval = 250;
96
- const deadline = Date.now() + maxWaitMs;
97
- while (Date.now() < deadline) {
98
- if (await isHubHealthy(host, port)) return true;
99
- await new Promise((r) => setTimeout(r, interval));
100
- }
101
- return false;
102
- }
103
-
104
- const { host, port } = resolveHubTarget();
105
- if (!(await isHubHealthy(host, port))) {
106
- const started = startHubDetached(port);
107
- if (started) {
108
- const ready = await waitForHubReady(host, port, 3000);
109
- if (ready) {
110
- process.stdout.write("hub: ok");
111
- } else {
112
- // fire-and-forget: hub이 아직 기동 중일 수 있음 — 에러가 아닌 경고
113
- process.stdout.write("hub: starting");
114
- }
115
- } else {
116
- process.stderr.write("[hub-ensure] hub 시작 실패");
117
- }
118
- } else {
119
- process.stdout.write("hub: ok");
120
- }
1
+ #!/usr/bin/env node
2
+ // SessionStart 훅에서 호출되는 Hub 보장 스크립트.
3
+ // - /status 기반 헬스체크
4
+ // - 비정상 시 Hub를 detached로 기동
5
+
6
+ import { existsSync, readFileSync } from "fs";
7
+ import { join, dirname } from "path";
8
+ import { homedir } from "os";
9
+ import { spawn } from "child_process";
10
+ import { fileURLToPath } from "url";
11
+
12
+ const LOOPBACK_HOSTS = new Set(["127.0.0.1", "localhost", "::1"]);
13
+ const PLUGIN_ROOT = dirname(dirname(fileURLToPath(import.meta.url)));
14
+ const HUB_PID_FILE = join(homedir(), ".claude", "cache", "tfx-hub", "hub.pid");
15
+
16
+ function formatHostForUrl(host) {
17
+ return host.includes(":") ? `[${host}]` : host;
18
+ }
19
+
20
+ function buildHubBaseUrl(host, port) {
21
+ return `http://${formatHostForUrl(host)}:${port}`;
22
+ }
23
+
24
+ function resolveHubTarget() {
25
+ const envPortRaw = Number(process.env.TFX_HUB_PORT || "");
26
+ const envPort = Number.isFinite(envPortRaw) && envPortRaw > 0 ? envPortRaw : null;
27
+ const target = {
28
+ host: "127.0.0.1",
29
+ port: envPort || 27888,
30
+ };
31
+
32
+ if (existsSync(HUB_PID_FILE)) {
33
+ try {
34
+ const info = JSON.parse(readFileSync(HUB_PID_FILE, "utf8"));
35
+ if (!envPort) {
36
+ const pidPort = Number(info?.port);
37
+ if (Number.isFinite(pidPort) && pidPort > 0) target.port = pidPort;
38
+ }
39
+ if (typeof info?.host === "string") {
40
+ const host = info.host.trim();
41
+ if (LOOPBACK_HOSTS.has(host)) target.host = host;
42
+ }
43
+ } catch {
44
+ // ignore parse errors and use env/default
45
+ }
46
+ }
47
+
48
+ return target;
49
+ }
50
+
51
+ async function isHubHealthy(host, port) {
52
+ try {
53
+ const res = await fetch(`${buildHubBaseUrl(host, port)}/status`, {
54
+ signal: AbortSignal.timeout(1000),
55
+ });
56
+ if (!res.ok) return false;
57
+ const data = await res.json();
58
+ return data?.hub?.state === "healthy";
59
+ } catch {
60
+ return false;
61
+ }
62
+ }
63
+
64
+ function startHubDetached(port) {
65
+ const serverPath = join(PLUGIN_ROOT, "hub", "server.mjs");
66
+ if (!existsSync(serverPath)) return false;
67
+
68
+ try {
69
+ const env = { ...process.env, TFX_HUB_PORT: String(port) };
70
+ if (process.platform === "win32") {
71
+ // Windows: cmd.exe /c start /b → 완전 독립 프로세스 트리 생성
72
+ // hook timeout 시 프로세스 트리 킬에서 살아남음
73
+ const child = spawn("cmd.exe", ["/c", "start", "/b", "", process.execPath, serverPath], {
74
+ env,
75
+ stdio: "ignore",
76
+ windowsHide: true,
77
+ });
78
+ child.unref();
79
+ } else {
80
+ const child = spawn(process.execPath, [serverPath], {
81
+ env,
82
+ detached: true,
83
+ stdio: "ignore",
84
+ });
85
+ child.unref();
86
+ }
87
+ return true;
88
+ } catch {
89
+ return false;
90
+ }
91
+ }
92
+
93
+ /** Hub 기동 후 ready 상태까지 대기 (최대 maxWaitMs) */
94
+ async function waitForHubReady(host, port, maxWaitMs = 5000) {
95
+ const interval = 250;
96
+ const deadline = Date.now() + maxWaitMs;
97
+ while (Date.now() < deadline) {
98
+ if (await isHubHealthy(host, port)) return true;
99
+ await new Promise((r) => setTimeout(r, interval));
100
+ }
101
+ return false;
102
+ }
103
+
104
+ const { host, port } = resolveHubTarget();
105
+ if (!(await isHubHealthy(host, port))) {
106
+ const started = startHubDetached(port);
107
+ if (started) {
108
+ const ready = await waitForHubReady(host, port, 3000);
109
+ if (ready) {
110
+ process.stdout.write("hub: ok");
111
+ } else {
112
+ // fire-and-forget: hub이 아직 기동 중일 수 있음 — 에러가 아닌 경고
113
+ process.stdout.write("hub: starting");
114
+ }
115
+ } else {
116
+ process.stderr.write("[hub-ensure] hub 시작 실패");
117
+ }
118
+ } else {
119
+ process.stdout.write("hub: ok");
120
+ }