@stackmemoryai/stackmemory 0.3.17 → 0.3.18

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 (212) hide show
  1. package/dist/cli/commands/skills.js +15 -2
  2. package/dist/cli/commands/skills.js.map +2 -2
  3. package/dist/cli/index.js +113 -834
  4. package/dist/cli/index.js.map +3 -3
  5. package/dist/core/context/dual-stack-manager.js +1 -1
  6. package/dist/core/context/dual-stack-manager.js.map +1 -1
  7. package/dist/core/context/frame-manager.js +3 -0
  8. package/dist/core/context/frame-manager.js.map +2 -2
  9. package/dist/integrations/claude-code/subagent-client.js +106 -3
  10. package/dist/integrations/claude-code/subagent-client.js.map +2 -2
  11. package/dist/servers/railway/config.js +51 -0
  12. package/dist/servers/railway/config.js.map +7 -0
  13. package/dist/servers/railway/index-enhanced.js +156 -0
  14. package/dist/servers/railway/index-enhanced.js.map +7 -0
  15. package/dist/servers/railway/minimal.js +48 -3
  16. package/dist/servers/railway/minimal.js.map +2 -2
  17. package/dist/servers/railway/storage-test.js +455 -0
  18. package/dist/servers/railway/storage-test.js.map +7 -0
  19. package/dist/skills/claude-skills.js +13 -12
  20. package/dist/skills/claude-skills.js.map +2 -2
  21. package/dist/skills/recursive-agent-orchestrator.js +27 -18
  22. package/dist/skills/recursive-agent-orchestrator.js.map +2 -2
  23. package/dist/skills/unified-rlm-orchestrator.js.map +2 -2
  24. package/package.json +6 -18
  25. package/scripts/README-TESTING.md +186 -0
  26. package/scripts/analyze-cli-security.js +288 -0
  27. package/scripts/archive/add-phase-tasks-to-linear.js +163 -0
  28. package/scripts/archive/analyze-linear-duplicates.js +214 -0
  29. package/scripts/archive/analyze-remaining-duplicates.js +230 -0
  30. package/scripts/archive/analyze-sta-duplicates.js +292 -0
  31. package/scripts/archive/analyze-sta-graphql.js +399 -0
  32. package/scripts/archive/cancel-duplicate-tasks.ts +246 -0
  33. package/scripts/archive/check-all-duplicates.ts +419 -0
  34. package/scripts/archive/clean-duplicate-tasks.js +114 -0
  35. package/scripts/archive/cleanup-duplicate-tasks.ts +286 -0
  36. package/scripts/archive/create-phase-tasks.js +387 -0
  37. package/scripts/archive/delete-linear-duplicates.js +182 -0
  38. package/scripts/archive/delete-remaining-duplicates.js +158 -0
  39. package/scripts/archive/delete-sta-duplicates.js +201 -0
  40. package/scripts/archive/delete-sta-oauth.js +201 -0
  41. package/scripts/archive/export-sta-tasks.js +62 -0
  42. package/scripts/archive/install-auto-sync.js +266 -0
  43. package/scripts/archive/install-chromadb-hooks.sh +133 -0
  44. package/scripts/archive/install-enhanced-clear-hooks.sh +431 -0
  45. package/scripts/archive/install-post-task-hooks.sh +289 -0
  46. package/scripts/archive/install-stackmemory-hooks.sh +420 -0
  47. package/scripts/archive/merge-linear-duplicates-safe.ts +362 -0
  48. package/scripts/archive/merge-linear-duplicates.ts +180 -0
  49. package/scripts/archive/remove-sta-tasks.js +70 -0
  50. package/scripts/archive/setup-background-sync.sh +168 -0
  51. package/scripts/archive/setup-claude-auto-triggers.sh +181 -0
  52. package/scripts/archive/setup-claude-autostart.sh +305 -0
  53. package/scripts/archive/setup-git-hooks.sh +25 -0
  54. package/scripts/archive/setup-linear-oauth.sh +46 -0
  55. package/scripts/archive/setup-mcp.sh +113 -0
  56. package/scripts/archive/setup-railway-deployment.sh +81 -0
  57. package/scripts/auto-handoff.sh +262 -0
  58. package/scripts/background-sync-manager.js +416 -0
  59. package/scripts/benchmark-performance.ts +57 -0
  60. package/scripts/check-redis.ts +48 -0
  61. package/scripts/chromadb-auto-loader.sh +128 -0
  62. package/scripts/chromadb-context-loader.js +479 -0
  63. package/scripts/claude-chromadb-hook.js +460 -0
  64. package/scripts/claude-code-wrapper.sh +66 -0
  65. package/scripts/claude-linear-skill.js +455 -0
  66. package/scripts/claude-pre-commit.sh +302 -0
  67. package/scripts/claude-sm-autostart.js +532 -0
  68. package/scripts/claude-sm-setup.sh +367 -0
  69. package/scripts/claude-with-chromadb.sh +69 -0
  70. package/scripts/claude-worktree-manager.sh +323 -0
  71. package/scripts/claude-worktree-monitor.sh +371 -0
  72. package/scripts/claude-worktree-setup.sh +327 -0
  73. package/scripts/clean-linear-backlog.js +273 -0
  74. package/scripts/cleanup-old-sessions.sh +57 -0
  75. package/scripts/codex-wrapper.sh +88 -0
  76. package/scripts/create-sandbox.sh +269 -0
  77. package/scripts/debug-linear-update.js +174 -0
  78. package/scripts/delete-linear-tasks.js +167 -0
  79. package/scripts/deploy.sh +89 -0
  80. package/scripts/deployment/railway.sh +352 -0
  81. package/scripts/deployment/test-deployment.js +194 -0
  82. package/scripts/detect-and-rehydrate.js +162 -0
  83. package/scripts/detect-and-rehydrate.mjs +165 -0
  84. package/scripts/development/create-demo-tasks.js +143 -0
  85. package/scripts/development/debug-frame-test.js +16 -0
  86. package/scripts/development/demo-auto-sync.js +128 -0
  87. package/scripts/development/fix-all-imports.js +213 -0
  88. package/scripts/development/fix-imports.js +229 -0
  89. package/scripts/development/fix-lint-loop.cjs +103 -0
  90. package/scripts/development/fix-project-id.ts +161 -0
  91. package/scripts/development/fix-strict-mode-issues.ts +291 -0
  92. package/scripts/development/reorganize-structure.sh +228 -0
  93. package/scripts/development/test-persistence-direct.js +148 -0
  94. package/scripts/development/test-persistence.js +114 -0
  95. package/scripts/development/test-tasks.js +93 -0
  96. package/scripts/development/update-imports.js +212 -0
  97. package/scripts/fetch-linear-status.js +125 -0
  98. package/scripts/git-hooks/README.md +310 -0
  99. package/scripts/git-hooks/branch-context-manager.sh +342 -0
  100. package/scripts/git-hooks/post-checkout-stackmemory.sh +63 -0
  101. package/scripts/git-hooks/post-commit-stackmemory.sh +305 -0
  102. package/scripts/git-hooks/pre-commit-stackmemory.sh +275 -0
  103. package/scripts/hooks/cleanup-shell.sh +130 -0
  104. package/scripts/hooks/task-complete.sh +114 -0
  105. package/scripts/initialize.ts +129 -0
  106. package/scripts/install-claude-hooks-auto.js +104 -0
  107. package/scripts/install-claude-hooks.sh +133 -0
  108. package/scripts/install-global.sh +296 -0
  109. package/scripts/install.sh +235 -0
  110. package/scripts/linear-auto-sync.js +262 -0
  111. package/scripts/linear-auto-sync.sh +161 -0
  112. package/scripts/linear-sync-daemon.js +150 -0
  113. package/scripts/linear-task-review.js +237 -0
  114. package/scripts/list-linear-tasks.ts +178 -0
  115. package/scripts/mcp-proxy.js +66 -0
  116. package/scripts/opencode-wrapper.sh +85 -0
  117. package/scripts/publish-local.js +74 -0
  118. package/scripts/query-chromadb.ts +201 -0
  119. package/scripts/railway-env-setup.sh +39 -0
  120. package/scripts/reconcile-local-tasks.js +170 -0
  121. package/scripts/recreate-frames-db.js +89 -0
  122. package/scripts/setup/claude-integration.js +138 -0
  123. package/scripts/setup/configure-alias.js +125 -0
  124. package/scripts/setup/configure-codex-alias.js +161 -0
  125. package/scripts/setup/configure-opencode-alias.js +175 -0
  126. package/scripts/setup-claude-integration.js +204 -0
  127. package/scripts/setup-claude-integration.sh +183 -0
  128. package/scripts/setup.sh +31 -0
  129. package/scripts/show-linear-summary.ts +172 -0
  130. package/scripts/stackmemory-auto-handoff.sh +231 -0
  131. package/scripts/stackmemory-daemon.sh +40 -0
  132. package/scripts/start-linear-sync-daemon.sh +141 -0
  133. package/scripts/start-temporal-paradox.sh +214 -0
  134. package/scripts/status.ts +159 -0
  135. package/scripts/sync-and-clean-tasks.js +258 -0
  136. package/scripts/sync-frames-from-railway.js +228 -0
  137. package/scripts/sync-linear-graphql.js +303 -0
  138. package/scripts/sync-linear-tasks.js +186 -0
  139. package/scripts/test-auto-triggers.sh +57 -0
  140. package/scripts/test-browser-mcp.js +74 -0
  141. package/scripts/test-chromadb-full.js +115 -0
  142. package/scripts/test-chromadb-hooks.sh +28 -0
  143. package/scripts/test-chromadb-sync.ts +245 -0
  144. package/scripts/test-cli-security.js +293 -0
  145. package/scripts/test-hooks-persistence.sh +220 -0
  146. package/scripts/test-installation-scenarios.sh +359 -0
  147. package/scripts/test-installation.sh +224 -0
  148. package/scripts/test-mcp.js +163 -0
  149. package/scripts/test-pre-publish-quick.sh +75 -0
  150. package/scripts/test-quality-gates.sh +263 -0
  151. package/scripts/test-railway-db.js +222 -0
  152. package/scripts/test-redis-storage.ts +490 -0
  153. package/scripts/test-rlm-basic.sh +122 -0
  154. package/scripts/test-rlm-comprehensive.sh +260 -0
  155. package/scripts/test-rlm-e2e.sh +268 -0
  156. package/scripts/test-rlm-simple.js +90 -0
  157. package/scripts/test-rlm.js +110 -0
  158. package/scripts/test-session-handoff.sh +165 -0
  159. package/scripts/test-shell-integration.sh +275 -0
  160. package/scripts/testing/ab-test-runner.ts +508 -0
  161. package/scripts/testing/collect-metrics.ts +457 -0
  162. package/scripts/testing/quick-effectiveness-demo.js +187 -0
  163. package/scripts/testing/real-performance-test.js +422 -0
  164. package/scripts/testing/run-effectiveness-tests.sh +176 -0
  165. package/scripts/testing/scripts/testing/ab-test-runner.js +363 -0
  166. package/scripts/testing/scripts/testing/collect-metrics.js +292 -0
  167. package/scripts/testing/simple-effectiveness-test.js +310 -0
  168. package/scripts/testing/src/core/context/context-bridge.js +253 -0
  169. package/scripts/testing/src/core/context/frame-manager.js +746 -0
  170. package/scripts/testing/src/core/context/shared-context-layer.js +437 -0
  171. package/scripts/testing/src/core/database/database-adapter.js +54 -0
  172. package/scripts/testing/src/core/errors/index.js +291 -0
  173. package/scripts/testing/src/core/errors/recovery.js +268 -0
  174. package/scripts/testing/src/core/monitoring/logger.js +145 -0
  175. package/scripts/testing/src/core/retrieval/context-retriever.js +516 -0
  176. package/scripts/testing/src/core/session/index.js +1 -0
  177. package/scripts/testing/src/core/session/session-manager.js +323 -0
  178. package/scripts/testing/src/core/trace/cli-trace-wrapper.js +140 -0
  179. package/scripts/testing/src/core/trace/db-trace-wrapper.js +251 -0
  180. package/scripts/testing/src/core/trace/debug-trace.js +398 -0
  181. package/scripts/testing/src/core/trace/index.js +120 -0
  182. package/scripts/testing/src/core/trace/linear-api-wrapper.js +204 -0
  183. package/scripts/update-linear-status.js +268 -0
  184. package/scripts/update-linear-tasks-fixed.js +284 -0
  185. package/templates/claude-hooks/hooks.json +5 -0
  186. package/templates/claude-hooks/on-clear.js +56 -0
  187. package/templates/claude-hooks/on-startup.js +56 -0
  188. package/templates/claude-hooks/tool-use-trace.js +67 -0
  189. package/dist/features/tui/components/analytics-panel.js +0 -157
  190. package/dist/features/tui/components/analytics-panel.js.map +0 -7
  191. package/dist/features/tui/components/frame-visualizer.js +0 -377
  192. package/dist/features/tui/components/frame-visualizer.js.map +0 -7
  193. package/dist/features/tui/components/pr-tracker.js +0 -135
  194. package/dist/features/tui/components/pr-tracker.js.map +0 -7
  195. package/dist/features/tui/components/session-monitor.js +0 -299
  196. package/dist/features/tui/components/session-monitor.js.map +0 -7
  197. package/dist/features/tui/components/subagent-fleet.js +0 -395
  198. package/dist/features/tui/components/subagent-fleet.js.map +0 -7
  199. package/dist/features/tui/components/task-board.js +0 -1139
  200. package/dist/features/tui/components/task-board.js.map +0 -7
  201. package/dist/features/tui/index.js +0 -408
  202. package/dist/features/tui/index.js.map +0 -7
  203. package/dist/features/tui/services/data-service.js +0 -641
  204. package/dist/features/tui/services/data-service.js.map +0 -7
  205. package/dist/features/tui/services/linear-task-reader.js +0 -102
  206. package/dist/features/tui/services/linear-task-reader.js.map +0 -7
  207. package/dist/features/tui/services/websocket-client.js +0 -162
  208. package/dist/features/tui/services/websocket-client.js.map +0 -7
  209. package/dist/features/tui/terminal-compat.js +0 -220
  210. package/dist/features/tui/terminal-compat.js.map +0 -7
  211. package/dist/features/tui/types.js +0 -1
  212. package/dist/features/tui/types.js.map +0 -7
@@ -1,299 +0,0 @@
1
- import blessed from "blessed";
2
- import { EventEmitter } from "events";
3
- class SessionMonitor extends EventEmitter {
4
- container;
5
- sessionList;
6
- sessions;
7
- selectedSession = null;
8
- constructor(container) {
9
- super();
10
- this.container = container;
11
- this.sessions = /* @__PURE__ */ new Map();
12
- this.initializeUI();
13
- }
14
- initializeUI() {
15
- this.sessionList = blessed.list({
16
- parent: this.container,
17
- top: 0,
18
- left: 0,
19
- width: "100%",
20
- height: "100%-3",
21
- style: {
22
- selected: {
23
- bg: "blue",
24
- fg: "white",
25
- bold: true
26
- },
27
- item: {
28
- fg: "cyan"
29
- }
30
- },
31
- mouse: true,
32
- keys: true,
33
- vi: true,
34
- scrollable: true,
35
- tags: true
36
- });
37
- const footer = blessed.box({
38
- parent: this.container,
39
- bottom: 0,
40
- left: 0,
41
- width: "100%",
42
- height: 3,
43
- content: "Active: 0 | Idle: 0 | Total: 0",
44
- tags: true,
45
- style: {
46
- fg: "white",
47
- bg: "black"
48
- }
49
- });
50
- this.sessionList.on("select", (item, index) => {
51
- const sessionId = Array.from(this.sessions.keys())[index];
52
- this.selectSession(sessionId);
53
- });
54
- }
55
- /**
56
- * Auto-tag session based on work context
57
- */
58
- autoTagSession(session) {
59
- const tags = [];
60
- if (session.recentActivities) {
61
- const files = session.recentActivities.filter((a) => a.type === "file_edit").map((a) => a.data?.path || "");
62
- if (files.some((f) => f.includes("test"))) tags.push("testing");
63
- if (files.some((f) => f.includes(".tsx") || f.includes(".jsx")))
64
- tags.push("frontend");
65
- if (files.some((f) => f.includes("api") || f.includes("server")))
66
- tags.push("backend");
67
- if (files.some((f) => f.includes("db") || f.includes("migration")))
68
- tags.push("database");
69
- const commands = session.recentActivities.filter((a) => a.type === "command").map((a) => a.data?.command || "");
70
- if (commands.some((c) => c.includes("npm test") || c.includes("jest")))
71
- tags.push("testing");
72
- if (commands.some((c) => c.includes("git"))) tags.push("git-ops");
73
- if (commands.some((c) => c.includes("deploy"))) tags.push("deployment");
74
- if (commands.some((c) => c.includes("debug"))) tags.push("debugging");
75
- }
76
- if (session.linearTask) {
77
- tags.push(`linear:${session.linearTask.identifier}`);
78
- if (session.linearTask.labels) {
79
- tags.push(
80
- ...session.linearTask.labels.map((l) => l.toLowerCase())
81
- );
82
- }
83
- }
84
- if (session.gitBranch) {
85
- const branch = session.gitBranch;
86
- if (branch.includes("feature/")) tags.push("feature");
87
- if (branch.includes("fix/") || branch.includes("bugfix/"))
88
- tags.push("bugfix");
89
- if (branch.includes("refactor/")) tags.push("refactor");
90
- if (branch.includes("docs/")) tags.push("documentation");
91
- }
92
- if (session.agentType) {
93
- tags.push(`agent:${session.agentType}`);
94
- }
95
- return tags.length > 0 ? tags.join(" \u2022 ") : "general";
96
- }
97
- /**
98
- * Generate session display name with context
99
- */
100
- generateSessionName(session) {
101
- const autoTags = this.autoTagSession(session);
102
- const timestamp = new Date(session.startTime).toLocaleTimeString();
103
- let primaryActivity = "Session";
104
- if (session.linearTask) {
105
- primaryActivity = session.linearTask.title || session.linearTask.identifier;
106
- } else if (session.gitBranch && session.gitBranch !== "main" && session.gitBranch !== "master") {
107
- primaryActivity = session.gitBranch.split("/").pop() || session.gitBranch;
108
- } else if (session.primaryFile) {
109
- primaryActivity = session.primaryFile.split("/").pop() || "Code";
110
- }
111
- return `${primaryActivity} [${autoTags}] - ${timestamp}`;
112
- }
113
- /**
114
- * Format session item for display
115
- */
116
- formatSessionItem(session) {
117
- const status = this.getSessionStatus(session);
118
- const statusIcon = this.getStatusIcon(status);
119
- const name = this.generateSessionName(session);
120
- const metrics = this.getSessionMetrics(session);
121
- const contextUsage = Math.round(session.contextUsage * 100);
122
- const contextColor = contextUsage > 80 ? "red" : contextUsage > 60 ? "yellow" : "green";
123
- return `${statusIcon} ${name}
124
- {gray-fg}Tokens: ${metrics.tokens} | Context: {${contextColor}-fg}${contextUsage}%{/} | Duration: ${metrics.duration}{/}`;
125
- }
126
- getSessionStatus(session) {
127
- if (session.error) return "error";
128
- if (session.completed) return "completed";
129
- const now = Date.now();
130
- const lastActivity = session.lastActivity || session.startTime;
131
- const idleTime = now - lastActivity;
132
- if (idleTime > 5 * 60 * 1e3) return "idle";
133
- return "active";
134
- }
135
- getStatusIcon(status) {
136
- switch (status) {
137
- case "active":
138
- return "{green-fg}\u25CF{/}";
139
- case "idle":
140
- return "{yellow-fg}\u25CF{/}";
141
- case "completed":
142
- return "{gray-fg}\u25CF{/}";
143
- case "error":
144
- return "{red-fg}\u25CF{/}";
145
- default:
146
- return "{white-fg}\u25CB{/}";
147
- }
148
- }
149
- getSessionMetrics(session) {
150
- const now = Date.now();
151
- const duration = now - session.startTime;
152
- const hours = Math.floor(duration / (1e3 * 60 * 60));
153
- const minutes = Math.floor(duration % (1e3 * 60 * 60) / (1e3 * 60));
154
- return {
155
- tokens: session.totalTokens || 0,
156
- duration: hours > 0 ? `${hours}h ${minutes}m` : `${minutes}m`,
157
- filesEdited: session.filesEdited?.length || 0,
158
- commandsRun: session.commandsRun || 0,
159
- errorsEncountered: session.errors?.length || 0
160
- };
161
- }
162
- update(sessions) {
163
- this.sessions.clear();
164
- sessions.forEach((session) => {
165
- this.sessions.set(session.id, session);
166
- });
167
- const items = sessions.map(
168
- (session) => this.formatSessionItem(session)
169
- );
170
- this.sessionList.setItems(items);
171
- const active = sessions.filter(
172
- (s) => this.getSessionStatus(s) === "active"
173
- ).length;
174
- const idle = sessions.filter(
175
- (s) => this.getSessionStatus(s) === "idle"
176
- ).length;
177
- const total = sessions.length;
178
- const footer = this.container.children[1];
179
- if (footer) {
180
- footer.setContent(
181
- `{bold}Active:{/} {green-fg}${active}{/} | {bold}Idle:{/} {yellow-fg}${idle}{/} | {bold}Total:{/} ${total}`
182
- );
183
- }
184
- this.container.screen.render();
185
- }
186
- selectSession(sessionId) {
187
- this.selectedSession = sessionId;
188
- const session = this.sessions.get(sessionId);
189
- if (session) {
190
- this.emit("session:selected", session);
191
- this.showSessionDetails(session);
192
- }
193
- }
194
- showSessionDetails(session) {
195
- const details = blessed.box({
196
- parent: this.container.screen,
197
- top: "center",
198
- left: "center",
199
- width: "80%",
200
- height: "80%",
201
- content: this.formatSessionDetails(session),
202
- tags: true,
203
- border: {
204
- type: "line"
205
- },
206
- style: {
207
- border: {
208
- fg: "cyan"
209
- }
210
- },
211
- scrollable: true,
212
- keys: true,
213
- vi: true,
214
- mouse: true,
215
- hidden: false,
216
- label: ` Session: ${this.generateSessionName(session)} `
217
- });
218
- details.key(["escape", "q"], () => {
219
- details.destroy();
220
- this.container.screen.render();
221
- });
222
- details.focus();
223
- this.container.screen.render();
224
- }
225
- formatSessionDetails(session) {
226
- const metrics = this.getSessionMetrics(session);
227
- const status = this.getSessionStatus(session);
228
- const tags = this.autoTagSession(session);
229
- let details = `{bold}Session ID:{/} ${session.id}
230
- `;
231
- details += `{bold}Status:{/} ${status}
232
- `;
233
- details += `{bold}Tags:{/} ${tags}
234
- `;
235
- details += `{bold}Started:{/} ${new Date(session.startTime).toLocaleString()}
236
- `;
237
- if (session.lastActivity) {
238
- details += `{bold}Last Activity:{/} ${new Date(session.lastActivity).toLocaleString()}
239
- `;
240
- }
241
- details += `
242
- {bold}Metrics:{/}
243
- `;
244
- details += ` Tokens Used: ${metrics.tokens}
245
- `;
246
- details += ` Context Usage: ${Math.round(session.contextUsage * 100)}%
247
- `;
248
- details += ` Duration: ${metrics.duration}
249
- `;
250
- details += ` Files Edited: ${metrics.filesEdited}
251
- `;
252
- details += ` Commands Run: ${metrics.commandsRun}
253
- `;
254
- details += ` Errors: ${metrics.errorsEncountered}
255
- `;
256
- if (session.linearTask) {
257
- details += `
258
- {bold}Linear Task:{/}
259
- `;
260
- details += ` ID: ${session.linearTask.identifier}
261
- `;
262
- details += ` Title: ${session.linearTask.title}
263
- `;
264
- details += ` State: ${session.linearTask.state}
265
- `;
266
- }
267
- if (session.gitBranch) {
268
- details += `
269
- {bold}Git:{/}
270
- `;
271
- details += ` Branch: ${session.gitBranch}
272
- `;
273
- if (session.lastCommit) {
274
- details += ` Last Commit: ${session.lastCommit}
275
- `;
276
- }
277
- }
278
- if (session.recentActivities && session.recentActivities.length > 0) {
279
- details += `
280
- {bold}Recent Activities:{/}
281
- `;
282
- session.recentActivities.slice(-10).forEach((activity) => {
283
- details += ` ${new Date(activity.timestamp).toLocaleTimeString()} - ${activity.type}: ${activity.description}
284
- `;
285
- });
286
- }
287
- return details;
288
- }
289
- focus() {
290
- this.sessionList.focus();
291
- }
292
- hasFocus() {
293
- return this.sessionList === this.container.screen.focused;
294
- }
295
- }
296
- export {
297
- SessionMonitor
298
- };
299
- //# sourceMappingURL=session-monitor.js.map
@@ -1,7 +0,0 @@
1
- {
2
- "version": 3,
3
- "sources": ["../../../../src/features/tui/components/session-monitor.ts"],
4
- "sourcesContent": ["/**\n * Session Monitor Component\n * Displays active sessions with auto-tagging based on work context\n */\n\nimport blessed from 'blessed';\nimport { EventEmitter } from 'events';\nimport type { SessionData, SessionMetrics } from '../types.js';\n\nexport class SessionMonitor extends EventEmitter {\n private container: blessed.Widgets.BoxElement;\n private sessionList: blessed.Widgets.ListElement;\n private sessions: Map<string, SessionData>;\n private selectedSession: string | null = null;\n\n constructor(container: blessed.Widgets.BoxElement) {\n super();\n this.container = container;\n this.sessions = new Map();\n this.initializeUI();\n }\n\n private initializeUI(): void {\n // Session list with auto-tagged names\n this.sessionList = blessed.list({\n parent: this.container,\n top: 0,\n left: 0,\n width: '100%',\n height: '100%-3',\n style: {\n selected: {\n bg: 'blue',\n fg: 'white',\n bold: true,\n },\n item: {\n fg: 'cyan',\n },\n },\n mouse: true,\n keys: true,\n vi: true,\n scrollable: true,\n tags: true,\n });\n\n // Summary footer\n const footer = blessed.box({\n parent: this.container,\n bottom: 0,\n left: 0,\n width: '100%',\n height: 3,\n content: 'Active: 0 | Idle: 0 | Total: 0',\n tags: true,\n style: {\n fg: 'white',\n bg: 'black',\n },\n });\n\n this.sessionList.on('select', (item, index) => {\n const sessionId = Array.from(this.sessions.keys())[index];\n this.selectSession(sessionId);\n });\n }\n\n /**\n * Auto-tag session based on work context\n */\n private autoTagSession(session: SessionData): string {\n const tags: string[] = [];\n\n // Analyze recent activities for context\n if (session.recentActivities) {\n // File-based tagging\n const files = session.recentActivities\n .filter((a: any) => a.type === 'file_edit')\n .map((a: any) => a.data?.path || '');\n\n if (files.some((f) => f.includes('test'))) tags.push('testing');\n if (files.some((f) => f.includes('.tsx') || f.includes('.jsx')))\n tags.push('frontend');\n if (files.some((f) => f.includes('api') || f.includes('server')))\n tags.push('backend');\n if (files.some((f) => f.includes('db') || f.includes('migration')))\n tags.push('database');\n\n // Command-based tagging\n const commands = session.recentActivities\n .filter((a: any) => a.type === 'command')\n .map((a: any) => a.data?.command || '');\n\n if (commands.some((c) => c.includes('npm test') || c.includes('jest')))\n tags.push('testing');\n if (commands.some((c) => c.includes('git'))) tags.push('git-ops');\n if (commands.some((c) => c.includes('deploy'))) tags.push('deployment');\n if (commands.some((c) => c.includes('debug'))) tags.push('debugging');\n }\n\n // Task-based tagging from Linear\n if (session.linearTask) {\n tags.push(`linear:${session.linearTask.identifier}`);\n if (session.linearTask.labels) {\n tags.push(\n ...session.linearTask.labels.map((l: any) => l.toLowerCase())\n );\n }\n }\n\n // Branch-based tagging\n if (session.gitBranch) {\n const branch = session.gitBranch;\n if (branch.includes('feature/')) tags.push('feature');\n if (branch.includes('fix/') || branch.includes('bugfix/'))\n tags.push('bugfix');\n if (branch.includes('refactor/')) tags.push('refactor');\n if (branch.includes('docs/')) tags.push('documentation');\n }\n\n // AI agent type tagging\n if (session.agentType) {\n tags.push(`agent:${session.agentType}`);\n }\n\n return tags.length > 0 ? tags.join(' \u2022 ') : 'general';\n }\n\n /**\n * Generate session display name with context\n */\n private generateSessionName(session: SessionData): string {\n const autoTags = this.autoTagSession(session);\n const timestamp = new Date(session.startTime).toLocaleTimeString();\n\n // Create descriptive name based on primary activity\n let primaryActivity = 'Session';\n\n if (session.linearTask) {\n primaryActivity =\n session.linearTask.title || session.linearTask.identifier;\n } else if (\n session.gitBranch &&\n session.gitBranch !== 'main' &&\n session.gitBranch !== 'master'\n ) {\n primaryActivity = session.gitBranch.split('/').pop() || session.gitBranch;\n } else if (session.primaryFile) {\n primaryActivity = session.primaryFile.split('/').pop() || 'Code';\n }\n\n return `${primaryActivity} [${autoTags}] - ${timestamp}`;\n }\n\n /**\n * Format session item for display\n */\n private formatSessionItem(session: SessionData): string {\n const status = this.getSessionStatus(session);\n const statusIcon = this.getStatusIcon(status);\n const name = this.generateSessionName(session);\n const metrics = this.getSessionMetrics(session);\n\n const contextUsage = Math.round(session.contextUsage * 100);\n const contextColor =\n contextUsage > 80 ? 'red' : contextUsage > 60 ? 'yellow' : 'green';\n\n return (\n `${statusIcon} ${name}\\n` +\n ` {gray-fg}Tokens: ${metrics.tokens} | Context: {${contextColor}-fg}${contextUsage}%{/} | Duration: ${metrics.duration}{/}`\n );\n }\n\n private getSessionStatus(\n session: SessionData\n ): 'active' | 'idle' | 'completed' | 'error' {\n if (session.error) return 'error';\n if (session.completed) return 'completed';\n\n const now = Date.now();\n const lastActivity = session.lastActivity || session.startTime;\n const idleTime = now - lastActivity;\n\n if (idleTime > 5 * 60 * 1000) return 'idle'; // 5 minutes\n return 'active';\n }\n\n private getStatusIcon(status: string): string {\n switch (status) {\n case 'active':\n return '{green-fg}\u25CF{/}';\n case 'idle':\n return '{yellow-fg}\u25CF{/}';\n case 'completed':\n return '{gray-fg}\u25CF{/}';\n case 'error':\n return '{red-fg}\u25CF{/}';\n default:\n return '{white-fg}\u25CB{/}';\n }\n }\n\n private getSessionMetrics(session: SessionData): SessionMetrics {\n const now = Date.now();\n const duration = now - session.startTime;\n\n const hours = Math.floor(duration / (1000 * 60 * 60));\n const minutes = Math.floor((duration % (1000 * 60 * 60)) / (1000 * 60));\n\n return {\n tokens: session.totalTokens || 0,\n duration: hours > 0 ? `${hours}h ${minutes}m` : `${minutes}m`,\n filesEdited: session.filesEdited?.length || 0,\n commandsRun: session.commandsRun || 0,\n errorsEncountered: session.errors?.length || 0,\n };\n }\n\n public update(sessions: SessionData[]): void {\n // Update session map\n this.sessions.clear();\n sessions.forEach((session) => {\n this.sessions.set(session.id, session);\n });\n\n // Update list display\n const items = sessions.map((session: any) =>\n this.formatSessionItem(session)\n );\n this.sessionList.setItems(items);\n\n // Update footer statistics\n const active = sessions.filter(\n (s: any) => this.getSessionStatus(s) === 'active'\n ).length;\n const idle = sessions.filter(\n (s: any) => this.getSessionStatus(s) === 'idle'\n ).length;\n const total = sessions.length;\n\n const footer = this.container.children[1] as blessed.Widgets.BoxElement;\n if (footer) {\n footer.setContent(\n `{bold}Active:{/} {green-fg}${active}{/} | ` +\n `{bold}Idle:{/} {yellow-fg}${idle}{/} | ` +\n `{bold}Total:{/} ${total}`\n );\n }\n\n this.container.screen.render();\n }\n\n private selectSession(sessionId: string): void {\n this.selectedSession = sessionId;\n const session = this.sessions.get(sessionId);\n if (session) {\n this.emit('session:selected', session);\n this.showSessionDetails(session);\n }\n }\n\n private showSessionDetails(session: SessionData): void {\n const details = blessed.box({\n parent: this.container.screen,\n top: 'center',\n left: 'center',\n width: '80%',\n height: '80%',\n content: this.formatSessionDetails(session),\n tags: true,\n border: {\n type: 'line',\n },\n style: {\n border: {\n fg: 'cyan',\n },\n },\n scrollable: true,\n keys: true,\n vi: true,\n mouse: true,\n hidden: false,\n label: ` Session: ${this.generateSessionName(session)} `,\n });\n\n details.key(['escape', 'q'], () => {\n details.destroy();\n this.container.screen.render();\n });\n\n details.focus();\n this.container.screen.render();\n }\n\n private formatSessionDetails(session: SessionData): string {\n const metrics = this.getSessionMetrics(session);\n const status = this.getSessionStatus(session);\n const tags = this.autoTagSession(session);\n\n let details = `{bold}Session ID:{/} ${session.id}\\n`;\n details += `{bold}Status:{/} ${status}\\n`;\n details += `{bold}Tags:{/} ${tags}\\n`;\n details += `{bold}Started:{/} ${new Date(session.startTime).toLocaleString()}\\n`;\n\n if (session.lastActivity) {\n details += `{bold}Last Activity:{/} ${new Date(session.lastActivity).toLocaleString()}\\n`;\n }\n\n details += `\\n{bold}Metrics:{/}\\n`;\n details += ` Tokens Used: ${metrics.tokens}\\n`;\n details += ` Context Usage: ${Math.round(session.contextUsage * 100)}%\\n`;\n details += ` Duration: ${metrics.duration}\\n`;\n details += ` Files Edited: ${metrics.filesEdited}\\n`;\n details += ` Commands Run: ${metrics.commandsRun}\\n`;\n details += ` Errors: ${metrics.errorsEncountered}\\n`;\n\n if (session.linearTask) {\n details += `\\n{bold}Linear Task:{/}\\n`;\n details += ` ID: ${session.linearTask.identifier}\\n`;\n details += ` Title: ${session.linearTask.title}\\n`;\n details += ` State: ${session.linearTask.state}\\n`;\n }\n\n if (session.gitBranch) {\n details += `\\n{bold}Git:{/}\\n`;\n details += ` Branch: ${session.gitBranch}\\n`;\n if (session.lastCommit) {\n details += ` Last Commit: ${session.lastCommit}\\n`;\n }\n }\n\n if (session.recentActivities && session.recentActivities.length > 0) {\n details += `\\n{bold}Recent Activities:{/}\\n`;\n session.recentActivities.slice(-10).forEach((activity) => {\n details += ` ${new Date(activity.timestamp).toLocaleTimeString()} - ${activity.type}: ${activity.description}\\n`;\n });\n }\n\n return details;\n }\n\n public focus(): void {\n this.sessionList.focus();\n }\n\n public hasFocus(): boolean {\n return this.sessionList === this.container.screen.focused;\n }\n}\n"],
5
- "mappings": "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",
6
- "names": []
7
- }