aimux-cli 0.1.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 (207) hide show
  1. package/README.md +743 -0
  2. package/bin/aimux +2 -0
  3. package/dist/agent-events.d.ts +20 -0
  4. package/dist/agent-events.js +2 -0
  5. package/dist/agent-events.js.map +1 -0
  6. package/dist/agent-message-parts.d.ts +17 -0
  7. package/dist/agent-message-parts.js +31 -0
  8. package/dist/agent-message-parts.js.map +1 -0
  9. package/dist/agent-output-parser.d.ts +16 -0
  10. package/dist/agent-output-parser.js +229 -0
  11. package/dist/agent-output-parser.js.map +1 -0
  12. package/dist/agent-tracker.d.ts +9 -0
  13. package/dist/agent-tracker.js +144 -0
  14. package/dist/agent-tracker.js.map +1 -0
  15. package/dist/agent-watcher.d.ts +15 -0
  16. package/dist/agent-watcher.js +2 -0
  17. package/dist/agent-watcher.js.map +1 -0
  18. package/dist/attachment-store.d.ts +35 -0
  19. package/dist/attachment-store.js +129 -0
  20. package/dist/attachment-store.js.map +1 -0
  21. package/dist/builtin-metadata-watchers.d.ts +2 -0
  22. package/dist/builtin-metadata-watchers.js +275 -0
  23. package/dist/builtin-metadata-watchers.js.map +1 -0
  24. package/dist/claude-hooks.d.ts +29 -0
  25. package/dist/claude-hooks.js +106 -0
  26. package/dist/claude-hooks.js.map +1 -0
  27. package/dist/config.d.ts +78 -0
  28. package/dist/config.js +172 -0
  29. package/dist/config.js.map +1 -0
  30. package/dist/context/compactor.d.ts +20 -0
  31. package/dist/context/compactor.js +212 -0
  32. package/dist/context/compactor.js.map +1 -0
  33. package/dist/context/context-bridge.d.ts +67 -0
  34. package/dist/context/context-bridge.js +471 -0
  35. package/dist/context/context-bridge.js.map +1 -0
  36. package/dist/context/context-file.d.ts +11 -0
  37. package/dist/context/context-file.js +93 -0
  38. package/dist/context/context-file.js.map +1 -0
  39. package/dist/context/history.d.ts +40 -0
  40. package/dist/context/history.js +108 -0
  41. package/dist/context/history.js.map +1 -0
  42. package/dist/daemon.d.ts +39 -0
  43. package/dist/daemon.js +344 -0
  44. package/dist/daemon.js.map +1 -0
  45. package/dist/dashboard-session-registry.d.ts +47 -0
  46. package/dist/dashboard-session-registry.js +161 -0
  47. package/dist/dashboard-session-registry.js.map +1 -0
  48. package/dist/dashboard-state.d.ts +18 -0
  49. package/dist/dashboard-state.js +26 -0
  50. package/dist/dashboard-state.js.map +1 -0
  51. package/dist/dashboard.d.ts +118 -0
  52. package/dist/dashboard.js +91 -0
  53. package/dist/dashboard.js.map +1 -0
  54. package/dist/debug.d.ts +7 -0
  55. package/dist/debug.js +41 -0
  56. package/dist/debug.js.map +1 -0
  57. package/dist/fast-control.d.ts +45 -0
  58. package/dist/fast-control.js +174 -0
  59. package/dist/fast-control.js.map +1 -0
  60. package/dist/hotkeys.d.ts +44 -0
  61. package/dist/hotkeys.js +118 -0
  62. package/dist/hotkeys.js.map +1 -0
  63. package/dist/http-client.d.ts +10 -0
  64. package/dist/http-client.js +54 -0
  65. package/dist/http-client.js.map +1 -0
  66. package/dist/instance-directory.d.ts +32 -0
  67. package/dist/instance-directory.js +82 -0
  68. package/dist/instance-directory.js.map +1 -0
  69. package/dist/instance-registry.d.ts +38 -0
  70. package/dist/instance-registry.js +208 -0
  71. package/dist/instance-registry.js.map +1 -0
  72. package/dist/key-parser.d.ts +30 -0
  73. package/dist/key-parser.js +272 -0
  74. package/dist/key-parser.js.map +1 -0
  75. package/dist/last-used.d.ts +31 -0
  76. package/dist/last-used.js +93 -0
  77. package/dist/last-used.js.map +1 -0
  78. package/dist/main.d.ts +1 -0
  79. package/dist/main.js +2483 -0
  80. package/dist/main.js.map +1 -0
  81. package/dist/metadata-server.d.ts +268 -0
  82. package/dist/metadata-server.js +1379 -0
  83. package/dist/metadata-server.js.map +1 -0
  84. package/dist/metadata-store.d.ts +80 -0
  85. package/dist/metadata-store.js +87 -0
  86. package/dist/metadata-store.js.map +1 -0
  87. package/dist/multiplexer.d.ts +471 -0
  88. package/dist/multiplexer.js +5714 -0
  89. package/dist/multiplexer.js.map +1 -0
  90. package/dist/notification-context.d.ts +18 -0
  91. package/dist/notification-context.js +68 -0
  92. package/dist/notification-context.js.map +1 -0
  93. package/dist/notifications.d.ts +38 -0
  94. package/dist/notifications.js +111 -0
  95. package/dist/notifications.js.map +1 -0
  96. package/dist/notify.d.ts +10 -0
  97. package/dist/notify.js +62 -0
  98. package/dist/notify.js.map +1 -0
  99. package/dist/orchestration-actions.d.ts +76 -0
  100. package/dist/orchestration-actions.js +310 -0
  101. package/dist/orchestration-actions.js.map +1 -0
  102. package/dist/orchestration-dispatcher.d.ts +22 -0
  103. package/dist/orchestration-dispatcher.js +49 -0
  104. package/dist/orchestration-dispatcher.js.map +1 -0
  105. package/dist/orchestration-routing.d.ts +20 -0
  106. package/dist/orchestration-routing.js +78 -0
  107. package/dist/orchestration-routing.js.map +1 -0
  108. package/dist/orchestration.d.ts +26 -0
  109. package/dist/orchestration.js +110 -0
  110. package/dist/orchestration.js.map +1 -0
  111. package/dist/osc-notifications.d.ts +15 -0
  112. package/dist/osc-notifications.js +180 -0
  113. package/dist/osc-notifications.js.map +1 -0
  114. package/dist/paths.d.ts +55 -0
  115. package/dist/paths.js +259 -0
  116. package/dist/paths.js.map +1 -0
  117. package/dist/plugin-runtime.d.ts +46 -0
  118. package/dist/plugin-runtime.js +180 -0
  119. package/dist/plugin-runtime.js.map +1 -0
  120. package/dist/project-events.d.ts +36 -0
  121. package/dist/project-events.js +63 -0
  122. package/dist/project-events.js.map +1 -0
  123. package/dist/project-scanner.d.ts +38 -0
  124. package/dist/project-scanner.js +243 -0
  125. package/dist/project-scanner.js.map +1 -0
  126. package/dist/project-service-manifest.d.ts +18 -0
  127. package/dist/project-service-manifest.js +56 -0
  128. package/dist/project-service-manifest.js.map +1 -0
  129. package/dist/recency.d.ts +2 -0
  130. package/dist/recency.js +34 -0
  131. package/dist/recency.js.map +1 -0
  132. package/dist/recorder.d.ts +14 -0
  133. package/dist/recorder.js +130 -0
  134. package/dist/recorder.js.map +1 -0
  135. package/dist/session-bootstrap.d.ts +45 -0
  136. package/dist/session-bootstrap.js +436 -0
  137. package/dist/session-bootstrap.js.map +1 -0
  138. package/dist/session-message-history.d.ts +27 -0
  139. package/dist/session-message-history.js +105 -0
  140. package/dist/session-message-history.js.map +1 -0
  141. package/dist/session-runtime.d.ts +44 -0
  142. package/dist/session-runtime.js +56 -0
  143. package/dist/session-runtime.js.map +1 -0
  144. package/dist/session-semantics.d.ts +35 -0
  145. package/dist/session-semantics.js +110 -0
  146. package/dist/session-semantics.js.map +1 -0
  147. package/dist/status-detector.d.ts +17 -0
  148. package/dist/status-detector.js +67 -0
  149. package/dist/status-detector.js.map +1 -0
  150. package/dist/statusline-model.d.ts +103 -0
  151. package/dist/statusline-model.js +177 -0
  152. package/dist/statusline-model.js.map +1 -0
  153. package/dist/task-dispatcher.d.ts +63 -0
  154. package/dist/task-dispatcher.js +210 -0
  155. package/dist/task-dispatcher.js.map +1 -0
  156. package/dist/task-workflow.d.ts +13 -0
  157. package/dist/task-workflow.js +153 -0
  158. package/dist/task-workflow.js.map +1 -0
  159. package/dist/tasks.d.ts +60 -0
  160. package/dist/tasks.js +120 -0
  161. package/dist/tasks.js.map +1 -0
  162. package/dist/team.d.ts +28 -0
  163. package/dist/team.js +91 -0
  164. package/dist/team.js.map +1 -0
  165. package/dist/terminal-host.d.ts +10 -0
  166. package/dist/terminal-host.js +52 -0
  167. package/dist/terminal-host.js.map +1 -0
  168. package/dist/threads.d.ts +61 -0
  169. package/dist/threads.js +200 -0
  170. package/dist/threads.js.map +1 -0
  171. package/dist/tmux-doctor.d.ts +47 -0
  172. package/dist/tmux-doctor.js +112 -0
  173. package/dist/tmux-doctor.js.map +1 -0
  174. package/dist/tmux-runtime-manager.d.ts +164 -0
  175. package/dist/tmux-runtime-manager.js +794 -0
  176. package/dist/tmux-runtime-manager.js.map +1 -0
  177. package/dist/tmux-session-transport.d.ts +31 -0
  178. package/dist/tmux-session-transport.js +115 -0
  179. package/dist/tmux-session-transport.js.map +1 -0
  180. package/dist/tmux-statusline.d.ts +17 -0
  181. package/dist/tmux-statusline.js +166 -0
  182. package/dist/tmux-statusline.js.map +1 -0
  183. package/dist/tool-output-watchers.d.ts +10 -0
  184. package/dist/tool-output-watchers.js +190 -0
  185. package/dist/tool-output-watchers.js.map +1 -0
  186. package/dist/tui/render/box.d.ts +1 -0
  187. package/dist/tui/render/box.js +20 -0
  188. package/dist/tui/render/box.js.map +1 -0
  189. package/dist/tui/render/text.d.ts +8 -0
  190. package/dist/tui/render/text.js +92 -0
  191. package/dist/tui/render/text.js.map +1 -0
  192. package/dist/tui/screens/dashboard-renderers.d.ts +23 -0
  193. package/dist/tui/screens/dashboard-renderers.js +411 -0
  194. package/dist/tui/screens/dashboard-renderers.js.map +1 -0
  195. package/dist/tui/screens/overlay-renderers.d.ts +10 -0
  196. package/dist/tui/screens/overlay-renderers.js +274 -0
  197. package/dist/tui/screens/overlay-renderers.js.map +1 -0
  198. package/dist/tui/screens/subscreen-renderers.d.ts +9 -0
  199. package/dist/tui/screens/subscreen-renderers.js +327 -0
  200. package/dist/tui/screens/subscreen-renderers.js.map +1 -0
  201. package/dist/workflow.d.ts +19 -0
  202. package/dist/workflow.js +111 -0
  203. package/dist/workflow.js.map +1 -0
  204. package/dist/worktree.d.ts +23 -0
  205. package/dist/worktree.js +101 -0
  206. package/dist/worktree.js.map +1 -0
  207. package/package.json +70 -0
package/dist/team.js ADDED
@@ -0,0 +1,91 @@
1
+ import { existsSync, readFileSync, writeFileSync, mkdirSync } from "node:fs";
2
+ import { getLocalAimuxDir, getProjectTeamPath, getGlobalTeamPath, getGlobalAimuxDir } from "./paths.js";
3
+ const DEFAULT_TEAM_CONFIG = {
4
+ roles: {
5
+ coder: {
6
+ description: "Implements features and fixes bugs",
7
+ reviewedBy: "reviewer",
8
+ },
9
+ reviewer: {
10
+ description: "Reviews code changes, approves or requests changes",
11
+ canEdit: true,
12
+ },
13
+ },
14
+ defaultRole: "coder",
15
+ };
16
+ function cloneJson(value) {
17
+ return JSON.parse(JSON.stringify(value));
18
+ }
19
+ export function getDefaultTeamConfig() {
20
+ return cloneJson(DEFAULT_TEAM_CONFIG);
21
+ }
22
+ /**
23
+ * Load team config: project .aimux/team.json → global ~/.aimux/team.json → built-in defaults.
24
+ */
25
+ export function loadTeamConfig() {
26
+ const projectPath = getProjectTeamPath();
27
+ if (existsSync(projectPath)) {
28
+ try {
29
+ return JSON.parse(readFileSync(projectPath, "utf-8"));
30
+ }
31
+ catch {
32
+ // Fall through
33
+ }
34
+ }
35
+ const globalPath = getGlobalTeamPath();
36
+ if (existsSync(globalPath)) {
37
+ try {
38
+ return JSON.parse(readFileSync(globalPath, "utf-8"));
39
+ }
40
+ catch {
41
+ // Fall through
42
+ }
43
+ }
44
+ return cloneJson(DEFAULT_TEAM_CONFIG);
45
+ }
46
+ /**
47
+ * Save team config at the project level.
48
+ */
49
+ export function saveTeamConfig(config) {
50
+ const dir = getLocalAimuxDir();
51
+ if (!existsSync(dir)) {
52
+ mkdirSync(dir, { recursive: true });
53
+ }
54
+ writeFileSync(getProjectTeamPath(), JSON.stringify(config, null, 2) + "\n");
55
+ }
56
+ /**
57
+ * Save team config at the global level (~/.aimux/team.json).
58
+ */
59
+ export function saveGlobalTeamConfig(config) {
60
+ const dir = getGlobalAimuxDir();
61
+ if (!existsSync(dir)) {
62
+ mkdirSync(dir, { recursive: true });
63
+ }
64
+ writeFileSync(getGlobalTeamPath(), JSON.stringify(config, null, 2) + "\n");
65
+ }
66
+ /**
67
+ * Build a role-specific preamble string for injection into an agent's system prompt.
68
+ */
69
+ export function buildRolePreamble(role, config) {
70
+ const roleConfig = config.roles[role];
71
+ if (!roleConfig)
72
+ return "";
73
+ const lines = [
74
+ `You are assigned the "${role}" role in this aimux team.`,
75
+ `Role: ${roleConfig.description}`,
76
+ ];
77
+ if (roleConfig.reviewedBy) {
78
+ lines.push(`Your work will be reviewed by the "${roleConfig.reviewedBy}" role.`);
79
+ }
80
+ if (roleConfig.canEdit === false) {
81
+ lines.push("You should NOT directly edit code files.");
82
+ }
83
+ const otherRoles = Object.entries(config.roles)
84
+ .filter(([name]) => name !== role)
85
+ .map(([name, rc]) => ` - ${name}: ${rc.description}`);
86
+ if (otherRoles.length > 0) {
87
+ lines.push("", "Other team roles:", ...otherRoles);
88
+ }
89
+ return lines.join("\n");
90
+ }
91
+ //# sourceMappingURL=team.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"team.js","sourceRoot":"","sources":["../src/team.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,10 @@
1
+ export declare class TerminalHost {
2
+ private rawModeWas;
3
+ private terminalRestored;
4
+ private inAlternateScreen;
5
+ enterRawMode(): void;
6
+ enterAlternateScreen(clear?: boolean): void;
7
+ exitAlternateScreen(): void;
8
+ exitRawMode(): void;
9
+ restoreTerminalState(): void;
10
+ }
@@ -0,0 +1,52 @@
1
+ export class TerminalHost {
2
+ rawModeWas;
3
+ terminalRestored = false;
4
+ inAlternateScreen = false;
5
+ enterRawMode() {
6
+ if (process.stdin.isTTY) {
7
+ this.rawModeWas = process.stdin.isRaw;
8
+ process.stdin.setRawMode(true);
9
+ process.stdin.resume();
10
+ }
11
+ process.stdout.write("\x1b[?1004h");
12
+ }
13
+ enterAlternateScreen(clear = false) {
14
+ if (!this.inAlternateScreen) {
15
+ process.stdout.write("\x1b[?1049h");
16
+ this.inAlternateScreen = true;
17
+ }
18
+ if (clear) {
19
+ process.stdout.write("\x1b[2J\x1b[H");
20
+ }
21
+ }
22
+ exitAlternateScreen() {
23
+ if (!this.inAlternateScreen)
24
+ return;
25
+ process.stdout.write("\x1b[?1049l");
26
+ this.inAlternateScreen = false;
27
+ }
28
+ exitRawMode() {
29
+ if (process.stdin.isTTY) {
30
+ process.stdin.setRawMode(this.rawModeWas ?? false);
31
+ process.stdin.pause();
32
+ }
33
+ }
34
+ restoreTerminalState() {
35
+ if (this.terminalRestored)
36
+ return;
37
+ this.terminalRestored = true;
38
+ try {
39
+ this.exitRawMode();
40
+ process.stdout.write("\x1b[0m" +
41
+ "\x1b[?25h" +
42
+ "\x1b[?1l" +
43
+ "\x1b>" +
44
+ "\x1b[?1000l\x1b[?1002l\x1b[?1003l\x1b[?1004l\x1b[?1005l\x1b[?1006l\x1b[?1015l" +
45
+ "\x1b[?2004l" +
46
+ "\x1b[?1049l");
47
+ this.inAlternateScreen = false;
48
+ }
49
+ catch { }
50
+ }
51
+ }
52
+ //# sourceMappingURL=terminal-host.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"terminal-host.js","sourceRoot":"","sources":["../src/terminal-host.ts"],"names":[],"mappings":"AAAA,MAAM,OAAO,YAAY;IACf,UAAU,CAAsB;IAChC,gBAAgB,GAAG,KAAK,CAAC;IACzB,iBAAiB,GAAG,KAAK,CAAC;IAElC,YAAY;QACV,IAAI,OAAO,CAAC,KAAK,CAAC,KAAK,EAAE,CAAC;YACxB,IAAI,CAAC,UAAU,GAAG,OAAO,CAAC,KAAK,CAAC,KAAK,CAAC;YACtC,OAAO,CAAC,KAAK,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC;YAC/B,OAAO,CAAC,KAAK,CAAC,MAAM,EAAE,CAAC;QACzB,CAAC;QACD,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,aAAa,CAAC,CAAC;IACtC,CAAC;IAED,oBAAoB,CAAC,KAAK,GAAG,KAAK;QAChC,IAAI,CAAC,IAAI,CAAC,iBAAiB,EAAE,CAAC;YAC5B,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,aAAa,CAAC,CAAC;YACpC,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC;QAChC,CAAC;QACD,IAAI,KAAK,EAAE,CAAC;YACV,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,eAAe,CAAC,CAAC;QACxC,CAAC;IACH,CAAC;IAED,mBAAmB;QACjB,IAAI,CAAC,IAAI,CAAC,iBAAiB;YAAE,OAAO;QACpC,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,aAAa,CAAC,CAAC;QACpC,IAAI,CAAC,iBAAiB,GAAG,KAAK,CAAC;IACjC,CAAC;IAED,WAAW;QACT,IAAI,OAAO,CAAC,KAAK,CAAC,KAAK,EAAE,CAAC;YACxB,OAAO,CAAC,KAAK,CAAC,UAAU,CAAC,IAAI,CAAC,UAAU,IAAI,KAAK,CAAC,CAAC;YACnD,OAAO,CAAC,KAAK,CAAC,KAAK,EAAE,CAAC;QACxB,CAAC;IACH,CAAC;IAED,oBAAoB;QAClB,IAAI,IAAI,CAAC,gBAAgB;YAAE,OAAO;QAClC,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC;QAE7B,IAAI,CAAC;YACH,IAAI,CAAC,WAAW,EAAE,CAAC;YACnB,OAAO,CAAC,MAAM,CAAC,KAAK,CAClB,SAAS;gBACP,WAAW;gBACX,UAAU;gBACV,OAAO;gBACP,+EAA+E;gBAC/E,aAAa;gBACb,aAAa,CAChB,CAAC;YACF,IAAI,CAAC,iBAAiB,GAAG,KAAK,CAAC;QACjC,CAAC;QAAC,MAAM,CAAC,CAAA,CAAC;IACZ,CAAC;CACF"}
@@ -0,0 +1,61 @@
1
+ export type ThreadKind = "conversation" | "task" | "review" | "handoff" | "user";
2
+ export type ThreadStatus = "open" | "waiting" | "blocked" | "done" | "abandoned";
3
+ export type MessageKind = "request" | "reply" | "status" | "decision" | "handoff" | "note";
4
+ export interface OrchestrationThread {
5
+ id: string;
6
+ title: string;
7
+ kind: ThreadKind;
8
+ createdAt: string;
9
+ updatedAt: string;
10
+ createdBy: string;
11
+ participants: string[];
12
+ status: ThreadStatus;
13
+ owner?: string;
14
+ waitingOn?: string[];
15
+ worktreePath?: string;
16
+ taskId?: string;
17
+ relatedPlanIds?: string[];
18
+ lastMessageId?: string;
19
+ unreadBy?: string[];
20
+ tags?: string[];
21
+ }
22
+ export interface OrchestrationMessage {
23
+ id: string;
24
+ threadId: string;
25
+ ts: string;
26
+ from: string;
27
+ to?: string[];
28
+ kind: MessageKind;
29
+ body: string;
30
+ taskId?: string;
31
+ planId?: string;
32
+ metadata?: Record<string, string | number | boolean | null>;
33
+ deliveredTo?: string[];
34
+ deliveredAt?: string;
35
+ }
36
+ export interface ThreadSummary {
37
+ thread: OrchestrationThread;
38
+ latestMessage?: OrchestrationMessage;
39
+ }
40
+ export declare function createThread(input: Omit<OrchestrationThread, "id" | "createdAt" | "updatedAt" | "status"> & Partial<Pick<OrchestrationThread, "id" | "status">>): OrchestrationThread;
41
+ export declare function readThread(threadId: string): OrchestrationThread | undefined;
42
+ export declare function updateThread(threadId: string, updater: (current: OrchestrationThread) => OrchestrationThread): OrchestrationThread | undefined;
43
+ export declare function listThreads(): OrchestrationThread[];
44
+ export declare function appendMessage(threadId: string, input: Omit<OrchestrationMessage, "id" | "threadId" | "ts"> & Partial<Pick<OrchestrationMessage, "id" | "ts">>): OrchestrationMessage;
45
+ export declare function readMessages(threadId: string): OrchestrationMessage[];
46
+ export declare function updateMessage(threadId: string, messageId: string, updater: (current: OrchestrationMessage) => OrchestrationMessage): OrchestrationMessage | undefined;
47
+ export declare function markMessageDelivered(threadId: string, messageId: string, recipient: string): OrchestrationMessage | undefined;
48
+ export declare function markThreadSeen(threadId: string, sessionId: string): OrchestrationThread | undefined;
49
+ export declare function setThreadStatus(threadId: string, status: ThreadStatus, input?: {
50
+ owner?: string;
51
+ waitingOn?: string[];
52
+ }): OrchestrationThread | undefined;
53
+ export declare function openTaskThread(taskId: string, input: {
54
+ title: string;
55
+ createdBy: string;
56
+ participants: string[];
57
+ worktreePath?: string;
58
+ kind?: Extract<ThreadKind, "task" | "review">;
59
+ }): OrchestrationThread;
60
+ export declare function listThreadsForParticipant(participantId: string): OrchestrationThread[];
61
+ export declare function listThreadSummaries(participantId?: string): ThreadSummary[];
@@ -0,0 +1,200 @@
1
+ import { appendFileSync, existsSync, mkdirSync, readFileSync, readdirSync, writeFileSync } from "node:fs";
2
+ import { join } from "node:path";
3
+ import { getThreadsDir } from "./paths.js";
4
+ function threadsDir() {
5
+ const dir = getThreadsDir();
6
+ if (!existsSync(dir))
7
+ mkdirSync(dir, { recursive: true });
8
+ return dir;
9
+ }
10
+ function threadPath(threadId) {
11
+ return join(threadsDir(), `${threadId}.json`);
12
+ }
13
+ function messagesPath(threadId) {
14
+ return join(threadsDir(), `${threadId}.jsonl`);
15
+ }
16
+ function nowIso() {
17
+ return new Date().toISOString();
18
+ }
19
+ function randomId(prefix) {
20
+ return `${prefix}-${Math.random().toString(36).slice(2, 10)}`;
21
+ }
22
+ export function createThread(input) {
23
+ const thread = {
24
+ id: input.id ?? randomId("thread"),
25
+ title: input.title,
26
+ kind: input.kind,
27
+ createdAt: nowIso(),
28
+ updatedAt: nowIso(),
29
+ createdBy: input.createdBy,
30
+ participants: [...new Set(input.participants)],
31
+ status: input.status ?? "open",
32
+ owner: input.owner,
33
+ waitingOn: input.waitingOn,
34
+ worktreePath: input.worktreePath,
35
+ taskId: input.taskId,
36
+ relatedPlanIds: input.relatedPlanIds,
37
+ lastMessageId: input.lastMessageId,
38
+ unreadBy: input.unreadBy,
39
+ tags: input.tags,
40
+ };
41
+ writeFileSync(threadPath(thread.id), JSON.stringify(thread, null, 2) + "\n");
42
+ return thread;
43
+ }
44
+ export function readThread(threadId) {
45
+ if (!existsSync(threadPath(threadId)))
46
+ return undefined;
47
+ try {
48
+ return JSON.parse(readFileSync(threadPath(threadId), "utf-8"));
49
+ }
50
+ catch {
51
+ return undefined;
52
+ }
53
+ }
54
+ export function updateThread(threadId, updater) {
55
+ const current = readThread(threadId);
56
+ if (!current)
57
+ return undefined;
58
+ const next = {
59
+ ...updater(current),
60
+ id: current.id,
61
+ createdAt: current.createdAt,
62
+ updatedAt: nowIso(),
63
+ };
64
+ writeFileSync(threadPath(threadId), JSON.stringify(next, null, 2) + "\n");
65
+ return next;
66
+ }
67
+ export function listThreads() {
68
+ try {
69
+ return readdirSync(threadsDir())
70
+ .filter((name) => name.endsWith(".json"))
71
+ .map((name) => {
72
+ try {
73
+ return JSON.parse(readFileSync(join(threadsDir(), name), "utf-8"));
74
+ }
75
+ catch {
76
+ return undefined;
77
+ }
78
+ })
79
+ .filter((value) => Boolean(value))
80
+ .sort((a, b) => (a.updatedAt < b.updatedAt ? 1 : a.updatedAt > b.updatedAt ? -1 : 0));
81
+ }
82
+ catch {
83
+ return [];
84
+ }
85
+ }
86
+ export function appendMessage(threadId, input) {
87
+ const message = {
88
+ id: input.id ?? randomId("msg"),
89
+ threadId,
90
+ ts: input.ts ?? nowIso(),
91
+ from: input.from,
92
+ to: input.to,
93
+ kind: input.kind,
94
+ body: input.body,
95
+ taskId: input.taskId,
96
+ planId: input.planId,
97
+ metadata: input.metadata,
98
+ };
99
+ appendFileSync(messagesPath(threadId), JSON.stringify(message) + "\n");
100
+ updateThread(threadId, (current) => ({
101
+ ...current,
102
+ lastMessageId: message.id,
103
+ unreadBy: [...new Set((current.participants ?? []).filter((id) => id !== message.from))],
104
+ }));
105
+ return message;
106
+ }
107
+ export function readMessages(threadId) {
108
+ if (!existsSync(messagesPath(threadId)))
109
+ return [];
110
+ try {
111
+ return readFileSync(messagesPath(threadId), "utf-8")
112
+ .split("\n")
113
+ .map((line) => line.trim())
114
+ .filter(Boolean)
115
+ .map((line) => {
116
+ try {
117
+ return JSON.parse(line);
118
+ }
119
+ catch {
120
+ return undefined;
121
+ }
122
+ })
123
+ .filter((value) => Boolean(value));
124
+ }
125
+ catch {
126
+ return [];
127
+ }
128
+ }
129
+ export function updateMessage(threadId, messageId, updater) {
130
+ const messages = readMessages(threadId);
131
+ if (messages.length === 0)
132
+ return undefined;
133
+ let updated;
134
+ const nextMessages = messages.map((message) => {
135
+ if (message.id !== messageId)
136
+ return message;
137
+ updated = {
138
+ ...updater(message),
139
+ id: message.id,
140
+ threadId: message.threadId,
141
+ ts: message.ts,
142
+ };
143
+ return updated;
144
+ });
145
+ if (!updated)
146
+ return undefined;
147
+ writeFileSync(messagesPath(threadId), nextMessages.map((message) => JSON.stringify(message)).join("\n") + "\n");
148
+ return updated;
149
+ }
150
+ export function markMessageDelivered(threadId, messageId, recipient) {
151
+ return updateMessage(threadId, messageId, (current) => {
152
+ const deliveredTo = [...new Set([...(current.deliveredTo ?? []), recipient])];
153
+ return {
154
+ ...current,
155
+ deliveredTo,
156
+ deliveredAt: new Date().toISOString(),
157
+ };
158
+ });
159
+ }
160
+ export function markThreadSeen(threadId, sessionId) {
161
+ return updateThread(threadId, (current) => ({
162
+ ...current,
163
+ unreadBy: (current.unreadBy ?? []).filter((id) => id !== sessionId),
164
+ }));
165
+ }
166
+ export function setThreadStatus(threadId, status, input) {
167
+ return updateThread(threadId, (current) => ({
168
+ ...current,
169
+ status,
170
+ owner: input?.owner ?? current.owner,
171
+ waitingOn: input?.waitingOn ?? (status === "done" || status === "abandoned" ? [] : current.waitingOn),
172
+ }));
173
+ }
174
+ export function openTaskThread(taskId, input) {
175
+ const existing = listThreads().find((thread) => thread.taskId === taskId);
176
+ if (existing)
177
+ return existing;
178
+ return createThread({
179
+ title: input.title,
180
+ kind: input.kind ?? "task",
181
+ createdBy: input.createdBy,
182
+ participants: input.participants,
183
+ taskId,
184
+ worktreePath: input.worktreePath,
185
+ });
186
+ }
187
+ export function listThreadsForParticipant(participantId) {
188
+ return listThreads().filter((thread) => thread.participants.includes(participantId));
189
+ }
190
+ export function listThreadSummaries(participantId) {
191
+ const threads = participantId ? listThreadsForParticipant(participantId) : listThreads();
192
+ return threads.map((thread) => {
193
+ const messages = readMessages(thread.id);
194
+ return {
195
+ thread,
196
+ latestMessage: messages[messages.length - 1],
197
+ };
198
+ });
199
+ }
200
+ //# sourceMappingURL=threads.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"threads.js","sourceRoot":"","sources":["../src/threads.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,47 @@
1
+ import { TmuxRuntimeManager } from "./tmux-runtime-manager.js";
2
+ export interface TmuxDoctorOptions {
3
+ projectRoot: string;
4
+ sessionName?: string;
5
+ windowId?: string;
6
+ env?: NodeJS.ProcessEnv;
7
+ }
8
+ export interface TmuxDoctorCheck {
9
+ expected: string;
10
+ observed: string | null;
11
+ ok: boolean;
12
+ }
13
+ export interface TmuxDoctorReport {
14
+ env: {
15
+ term: string | null;
16
+ termProgram: string | null;
17
+ insideTmux: boolean;
18
+ };
19
+ tmux: {
20
+ available: boolean;
21
+ version: string | null;
22
+ currentClientSession: string | null;
23
+ currentWindowId: string | null;
24
+ currentWindowName: string | null;
25
+ };
26
+ managedSession: {
27
+ sessionName: string;
28
+ exists: boolean;
29
+ options: Record<string, TmuxDoctorCheck>;
30
+ terminalFeatures: Record<string, TmuxDoctorCheck>;
31
+ };
32
+ activeWindow: null | {
33
+ windowId: string;
34
+ windowName: string | null;
35
+ tool: string | null;
36
+ options: Record<string, TmuxDoctorCheck>;
37
+ };
38
+ managedWindows: Array<{
39
+ windowId: string;
40
+ windowIndex: number;
41
+ windowName: string;
42
+ tool: string;
43
+ allowPassthrough: string | null;
44
+ }>;
45
+ }
46
+ export declare function buildTmuxDoctorReport(tmux: TmuxRuntimeManager, { projectRoot, sessionName, windowId, env }: TmuxDoctorOptions): TmuxDoctorReport;
47
+ export declare function renderTmuxDoctorReport(report: TmuxDoctorReport): string;
@@ -0,0 +1,112 @@
1
+ import { MANAGED_TMUX_AGENT_WINDOW_OPTIONS, MANAGED_TMUX_SESSION_OPTIONS, MANAGED_TMUX_TERMINAL_FEATURES, } from "./tmux-runtime-manager.js";
2
+ function buildCheck(expected, observed) {
3
+ return { expected, observed, ok: observed === expected };
4
+ }
5
+ export function buildTmuxDoctorReport(tmux, { projectRoot, sessionName, windowId, env = process.env }) {
6
+ const available = tmux.isAvailable();
7
+ const insideTmux = tmux.isInsideTmux(env);
8
+ const currentClientSession = available && insideTmux ? tmux.currentClientSession() : null;
9
+ const resolvedSessionName = sessionName ??
10
+ (insideTmux && currentClientSession && tmux.isManagedSessionName(currentClientSession)
11
+ ? currentClientSession
12
+ : tmux.getProjectSession(projectRoot).sessionName);
13
+ const currentWindowId = available ? (windowId ?? (insideTmux ? tmux.displayMessage("#{window_id}") : null)) : null;
14
+ const currentWindowName = available ? (insideTmux ? tmux.displayMessage("#{window_name}") : null) : null;
15
+ const sessionExists = available ? tmux.hasSession(resolvedSessionName) : false;
16
+ const sessionOptions = {
17
+ prefix: buildCheck(MANAGED_TMUX_SESSION_OPTIONS.prefix, sessionExists ? tmux.getSessionOption(resolvedSessionName, "prefix") : null),
18
+ prefix2: buildCheck(MANAGED_TMUX_SESSION_OPTIONS.prefix2, sessionExists ? tmux.getSessionOption(resolvedSessionName, "prefix2") : null),
19
+ mouse: buildCheck(MANAGED_TMUX_SESSION_OPTIONS.mouse, sessionExists ? tmux.getSessionOption(resolvedSessionName, "mouse") : null),
20
+ "extended-keys": buildCheck(MANAGED_TMUX_SESSION_OPTIONS.extendedKeys, sessionExists ? tmux.getSessionOption(resolvedSessionName, "extended-keys") : null),
21
+ "extended-keys-format": buildCheck(MANAGED_TMUX_SESSION_OPTIONS.extendedKeysFormat, sessionExists ? tmux.getSessionOption(resolvedSessionName, "extended-keys-format") : null),
22
+ };
23
+ const featureText = sessionExists ? tmux.getSessionOption(resolvedSessionName, "terminal-features") : null;
24
+ const featureSet = new Set(featureText
25
+ ?.split("\n")
26
+ .map((entry) => entry.trim())
27
+ .filter(Boolean) ?? []);
28
+ const terminalFeatures = Object.fromEntries(MANAGED_TMUX_TERMINAL_FEATURES.map((feature) => [
29
+ feature,
30
+ { expected: "present", observed: featureSet.has(feature) ? "present" : null, ok: featureSet.has(feature) },
31
+ ]));
32
+ const activeWindow = available && currentWindowId
33
+ ? {
34
+ windowId: currentWindowId,
35
+ windowName: currentWindowName,
36
+ tool: tmux.getWindowOption(currentWindowId, "@aimux-tool"),
37
+ options: {
38
+ "allow-passthrough": buildCheck(MANAGED_TMUX_AGENT_WINDOW_OPTIONS.allowPassthrough, tmux.getWindowOption(currentWindowId, "allow-passthrough")),
39
+ },
40
+ }
41
+ : null;
42
+ const managedWindows = sessionExists
43
+ ? tmux.listManagedWindows(resolvedSessionName).map(({ target, metadata }) => ({
44
+ windowId: target.windowId,
45
+ windowIndex: target.windowIndex,
46
+ windowName: target.windowName,
47
+ tool: metadata.toolConfigKey,
48
+ allowPassthrough: tmux.getWindowOption(target, "allow-passthrough"),
49
+ }))
50
+ : [];
51
+ return {
52
+ env: {
53
+ term: env.TERM || null,
54
+ termProgram: env.TERM_PROGRAM || null,
55
+ insideTmux,
56
+ },
57
+ tmux: {
58
+ available,
59
+ version: available ? tmux.getVersion() : null,
60
+ currentClientSession,
61
+ currentWindowId,
62
+ currentWindowName,
63
+ },
64
+ managedSession: {
65
+ sessionName: resolvedSessionName,
66
+ exists: sessionExists,
67
+ options: sessionOptions,
68
+ terminalFeatures,
69
+ },
70
+ activeWindow,
71
+ managedWindows,
72
+ };
73
+ }
74
+ export function renderTmuxDoctorReport(report) {
75
+ const lines = [
76
+ "Tmux Doctor",
77
+ ` TERM: ${report.env.term ?? "(unset)"}`,
78
+ ` TERM_PROGRAM: ${report.env.termProgram ?? "(unset)"}`,
79
+ ` inside tmux: ${report.env.insideTmux ? "yes" : "no"}`,
80
+ ` tmux available: ${report.tmux.available ? "yes" : "no"}`,
81
+ ` tmux version: ${report.tmux.version ?? "(unavailable)"}`,
82
+ ` current client session: ${report.tmux.currentClientSession ?? "(none)"}`,
83
+ ` managed session: ${report.managedSession.sessionName}`,
84
+ ` managed session exists: ${report.managedSession.exists ? "yes" : "no"}`,
85
+ " managed session options:",
86
+ ];
87
+ for (const [key, check] of Object.entries(report.managedSession.options)) {
88
+ lines.push(` ${key}: ${check.observed ?? "(missing)"} (expected ${check.expected}) [${check.ok ? "ok" : "mismatch"}]`);
89
+ }
90
+ lines.push(" managed terminal features:");
91
+ for (const [feature, check] of Object.entries(report.managedSession.terminalFeatures)) {
92
+ lines.push(` ${feature}: ${check.ok ? "present" : "missing"} [${check.ok ? "ok" : "mismatch"}]`);
93
+ }
94
+ if (report.activeWindow) {
95
+ lines.push(` active window: ${report.activeWindow.windowId} (${report.activeWindow.windowName ?? "unknown"})`);
96
+ lines.push(` @aimux-tool: ${report.activeWindow.tool ?? "(unset)"}`);
97
+ for (const [key, check] of Object.entries(report.activeWindow.options)) {
98
+ lines.push(` ${key}: ${check.observed ?? "(missing)"} (expected ${check.expected}) [${check.ok ? "ok" : "mismatch"}]`);
99
+ }
100
+ }
101
+ else {
102
+ lines.push(" active window: (none)");
103
+ }
104
+ if (report.managedWindows.length > 0) {
105
+ lines.push(" managed windows:");
106
+ for (const window of report.managedWindows) {
107
+ lines.push(` ${window.windowId} ${window.windowName} tool=${window.tool} allow-passthrough=${window.allowPassthrough ?? "(unset)"}`);
108
+ }
109
+ }
110
+ return lines.join("\n");
111
+ }
112
+ //# sourceMappingURL=tmux-doctor.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"tmux-doctor.js","sourceRoot":"","sources":["../src/tmux-doctor.ts"],"names":[],"mappings":"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"}