@stackmemoryai/stackmemory 0.3.17 → 0.3.19
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/cli/claude-sm.js +51 -5
- package/dist/cli/claude-sm.js.map +2 -2
- package/dist/cli/codex-sm.js +52 -19
- package/dist/cli/codex-sm.js.map +2 -2
- package/dist/cli/commands/db.js +143 -0
- package/dist/cli/commands/db.js.map +7 -0
- package/dist/cli/commands/login.js +50 -0
- package/dist/cli/commands/login.js.map +7 -0
- package/dist/cli/commands/migrate.js +178 -0
- package/dist/cli/commands/migrate.js.map +7 -0
- package/dist/cli/commands/onboard.js +158 -2
- package/dist/cli/commands/onboard.js.map +2 -2
- package/dist/cli/commands/skills.js +15 -2
- package/dist/cli/commands/skills.js.map +2 -2
- package/dist/cli/index.js +118 -834
- package/dist/cli/index.js.map +3 -3
- package/dist/core/context/dual-stack-manager.js +1 -1
- package/dist/core/context/dual-stack-manager.js.map +1 -1
- package/dist/core/context/frame-database.js +1 -0
- package/dist/core/context/frame-database.js.map +2 -2
- package/dist/core/context/frame-manager.js +59 -2
- package/dist/core/context/frame-manager.js.map +2 -2
- package/dist/core/database/database-adapter.js +6 -1
- package/dist/core/database/database-adapter.js.map +2 -2
- package/dist/core/database/sqlite-adapter.js +60 -2
- package/dist/core/database/sqlite-adapter.js.map +2 -2
- package/dist/integrations/claude-code/subagent-client.js +106 -3
- package/dist/integrations/claude-code/subagent-client.js.map +2 -2
- package/dist/servers/railway/config.js +51 -0
- package/dist/servers/railway/config.js.map +7 -0
- package/dist/servers/railway/index-enhanced.js +156 -0
- package/dist/servers/railway/index-enhanced.js.map +7 -0
- package/dist/servers/railway/index.js +843 -82
- package/dist/servers/railway/index.js.map +3 -3
- package/dist/servers/railway/minimal.js +48 -3
- package/dist/servers/railway/minimal.js.map +2 -2
- package/dist/servers/railway/storage-test.js +455 -0
- package/dist/servers/railway/storage-test.js.map +7 -0
- package/dist/skills/claude-skills.js +13 -12
- package/dist/skills/claude-skills.js.map +2 -2
- package/dist/skills/recursive-agent-orchestrator.js +27 -18
- package/dist/skills/recursive-agent-orchestrator.js.map +2 -2
- package/dist/skills/unified-rlm-orchestrator.js.map +2 -2
- package/package.json +13 -21
- package/scripts/README-TESTING.md +186 -0
- package/scripts/analyze-cli-security.js +288 -0
- package/scripts/archive/add-phase-tasks-to-linear.js +163 -0
- package/scripts/archive/analyze-linear-duplicates.js +214 -0
- package/scripts/archive/analyze-remaining-duplicates.js +230 -0
- package/scripts/archive/analyze-sta-duplicates.js +292 -0
- package/scripts/archive/analyze-sta-graphql.js +399 -0
- package/scripts/archive/cancel-duplicate-tasks.ts +246 -0
- package/scripts/archive/check-all-duplicates.ts +419 -0
- package/scripts/archive/clean-duplicate-tasks.js +114 -0
- package/scripts/archive/cleanup-duplicate-tasks.ts +286 -0
- package/scripts/archive/create-phase-tasks.js +387 -0
- package/scripts/archive/delete-linear-duplicates.js +182 -0
- package/scripts/archive/delete-remaining-duplicates.js +158 -0
- package/scripts/archive/delete-sta-duplicates.js +201 -0
- package/scripts/archive/delete-sta-oauth.js +201 -0
- package/scripts/archive/export-sta-tasks.js +62 -0
- package/scripts/archive/install-auto-sync.js +266 -0
- package/scripts/archive/install-chromadb-hooks.sh +133 -0
- package/scripts/archive/install-enhanced-clear-hooks.sh +431 -0
- package/scripts/archive/install-post-task-hooks.sh +289 -0
- package/scripts/archive/install-stackmemory-hooks.sh +420 -0
- package/scripts/archive/merge-linear-duplicates-safe.ts +362 -0
- package/scripts/archive/merge-linear-duplicates.ts +180 -0
- package/scripts/archive/remove-sta-tasks.js +70 -0
- package/scripts/archive/setup-background-sync.sh +168 -0
- package/scripts/archive/setup-claude-auto-triggers.sh +181 -0
- package/scripts/archive/setup-claude-autostart.sh +305 -0
- package/scripts/archive/setup-git-hooks.sh +25 -0
- package/scripts/archive/setup-linear-oauth.sh +46 -0
- package/scripts/archive/setup-mcp.sh +113 -0
- package/scripts/archive/setup-railway-deployment.sh +81 -0
- package/scripts/auto-handoff.sh +262 -0
- package/scripts/background-sync-manager.js +416 -0
- package/scripts/benchmark-performance.ts +57 -0
- package/scripts/check-redis.ts +48 -0
- package/scripts/chromadb-auto-loader.sh +128 -0
- package/scripts/chromadb-context-loader.js +479 -0
- package/scripts/claude-chromadb-hook.js +460 -0
- package/scripts/claude-code-wrapper.sh +66 -0
- package/scripts/claude-linear-skill.js +455 -0
- package/scripts/claude-pre-commit.sh +302 -0
- package/scripts/claude-sm-autostart.js +532 -0
- package/scripts/claude-sm-setup.sh +367 -0
- package/scripts/claude-with-chromadb.sh +69 -0
- package/scripts/claude-worktree-manager.sh +323 -0
- package/scripts/claude-worktree-monitor.sh +371 -0
- package/scripts/claude-worktree-setup.sh +327 -0
- package/scripts/clean-linear-backlog.js +273 -0
- package/scripts/cleanup-old-sessions.sh +57 -0
- package/scripts/codex-wrapper.sh +88 -0
- package/scripts/create-sandbox.sh +269 -0
- package/scripts/debug-linear-update.js +174 -0
- package/scripts/delete-linear-tasks.js +167 -0
- package/scripts/deploy.sh +89 -0
- package/scripts/deployment/railway.sh +352 -0
- package/scripts/deployment/test-deployment.js +194 -0
- package/scripts/detect-and-rehydrate.js +162 -0
- package/scripts/detect-and-rehydrate.mjs +165 -0
- package/scripts/development/create-demo-tasks.js +143 -0
- package/scripts/development/debug-frame-test.js +16 -0
- package/scripts/development/demo-auto-sync.js +128 -0
- package/scripts/development/fix-all-imports.js +213 -0
- package/scripts/development/fix-imports.js +229 -0
- package/scripts/development/fix-lint-loop.cjs +103 -0
- package/scripts/development/fix-project-id.ts +161 -0
- package/scripts/development/fix-strict-mode-issues.ts +291 -0
- package/scripts/development/reorganize-structure.sh +228 -0
- package/scripts/development/test-persistence-direct.js +148 -0
- package/scripts/development/test-persistence.js +114 -0
- package/scripts/development/test-tasks.js +93 -0
- package/scripts/development/update-imports.js +212 -0
- package/scripts/fetch-linear-status.js +125 -0
- package/scripts/git-hooks/README.md +310 -0
- package/scripts/git-hooks/branch-context-manager.sh +342 -0
- package/scripts/git-hooks/post-checkout-stackmemory.sh +63 -0
- package/scripts/git-hooks/post-commit-stackmemory.sh +305 -0
- package/scripts/git-hooks/pre-commit-stackmemory.sh +275 -0
- package/scripts/hooks/cleanup-shell.sh +130 -0
- package/scripts/hooks/task-complete.sh +114 -0
- package/scripts/initialize.ts +129 -0
- package/scripts/install-claude-hooks-auto.js +104 -0
- package/scripts/install-claude-hooks.sh +133 -0
- package/scripts/install-global.sh +296 -0
- package/scripts/install.sh +235 -0
- package/scripts/linear-auto-sync.js +262 -0
- package/scripts/linear-auto-sync.sh +161 -0
- package/scripts/linear-sync-daemon.js +150 -0
- package/scripts/linear-task-review.js +237 -0
- package/scripts/list-linear-tasks.ts +178 -0
- package/scripts/mcp-proxy.js +66 -0
- package/scripts/opencode-wrapper.sh +85 -0
- package/scripts/publish-local.js +74 -0
- package/scripts/query-chromadb.ts +201 -0
- package/scripts/railway-env-setup.sh +39 -0
- package/scripts/reconcile-local-tasks.js +170 -0
- package/scripts/recreate-frames-db.js +89 -0
- package/scripts/setup/claude-integration.js +138 -0
- package/scripts/setup/configure-alias.js +125 -0
- package/scripts/setup/configure-codex-alias.js +161 -0
- package/scripts/setup/configure-opencode-alias.js +175 -0
- package/scripts/setup-claude-integration.js +204 -0
- package/scripts/setup-claude-integration.sh +183 -0
- package/scripts/setup-railway-deployment.sh +37 -0
- package/scripts/setup.sh +31 -0
- package/scripts/show-linear-summary.ts +172 -0
- package/scripts/stackmemory-auto-handoff.sh +231 -0
- package/scripts/stackmemory-daemon.sh +40 -0
- package/scripts/start-linear-sync-daemon.sh +141 -0
- package/scripts/start-temporal-paradox.sh +214 -0
- package/scripts/status.ts +159 -0
- package/scripts/sync-and-clean-tasks.js +258 -0
- package/scripts/sync-frames-from-railway.js +228 -0
- package/scripts/sync-linear-graphql.js +303 -0
- package/scripts/sync-linear-tasks.js +186 -0
- package/scripts/test-auto-triggers.sh +57 -0
- package/scripts/test-browser-mcp.js +74 -0
- package/scripts/test-chromadb-full.js +115 -0
- package/scripts/test-chromadb-hooks.sh +28 -0
- package/scripts/test-chromadb-sync.ts +245 -0
- package/scripts/test-cli-security.js +293 -0
- package/scripts/test-hooks-persistence.sh +220 -0
- package/scripts/test-installation-scenarios.sh +359 -0
- package/scripts/test-installation.sh +224 -0
- package/scripts/test-mcp.js +163 -0
- package/scripts/test-pre-publish-quick.sh +75 -0
- package/scripts/test-quality-gates.sh +263 -0
- package/scripts/test-railway-db.js +222 -0
- package/scripts/test-redis-storage.ts +490 -0
- package/scripts/test-rlm-basic.sh +122 -0
- package/scripts/test-rlm-comprehensive.sh +260 -0
- package/scripts/test-rlm-e2e.sh +268 -0
- package/scripts/test-rlm-simple.js +90 -0
- package/scripts/test-rlm.js +110 -0
- package/scripts/test-session-handoff.sh +165 -0
- package/scripts/test-shell-integration.sh +275 -0
- package/scripts/testing/ab-test-runner.ts +508 -0
- package/scripts/testing/collect-metrics.ts +457 -0
- package/scripts/testing/quick-effectiveness-demo.js +187 -0
- package/scripts/testing/real-performance-test.js +422 -0
- package/scripts/testing/run-effectiveness-tests.sh +176 -0
- package/scripts/testing/scripts/testing/ab-test-runner.js +363 -0
- package/scripts/testing/scripts/testing/collect-metrics.js +292 -0
- package/scripts/testing/simple-effectiveness-test.js +310 -0
- package/scripts/testing/src/core/context/context-bridge.js +253 -0
- package/scripts/testing/src/core/context/frame-manager.js +746 -0
- package/scripts/testing/src/core/context/shared-context-layer.js +437 -0
- package/scripts/testing/src/core/database/database-adapter.js +54 -0
- package/scripts/testing/src/core/errors/index.js +291 -0
- package/scripts/testing/src/core/errors/recovery.js +268 -0
- package/scripts/testing/src/core/monitoring/logger.js +145 -0
- package/scripts/testing/src/core/retrieval/context-retriever.js +516 -0
- package/scripts/testing/src/core/session/index.js +1 -0
- package/scripts/testing/src/core/session/session-manager.js +323 -0
- package/scripts/testing/src/core/trace/cli-trace-wrapper.js +140 -0
- package/scripts/testing/src/core/trace/db-trace-wrapper.js +251 -0
- package/scripts/testing/src/core/trace/debug-trace.js +398 -0
- package/scripts/testing/src/core/trace/index.js +120 -0
- package/scripts/testing/src/core/trace/linear-api-wrapper.js +204 -0
- package/scripts/update-linear-status.js +268 -0
- package/scripts/update-linear-tasks-fixed.js +284 -0
- package/scripts/verify-railway-schema.ts +35 -0
- package/templates/claude-hooks/hooks.json +5 -0
- package/templates/claude-hooks/on-clear.js +56 -0
- package/templates/claude-hooks/on-startup.js +56 -0
- package/templates/claude-hooks/tool-use-trace.js +67 -0
- package/dist/features/tui/components/analytics-panel.js +0 -157
- package/dist/features/tui/components/analytics-panel.js.map +0 -7
- package/dist/features/tui/components/frame-visualizer.js +0 -377
- package/dist/features/tui/components/frame-visualizer.js.map +0 -7
- package/dist/features/tui/components/pr-tracker.js +0 -135
- package/dist/features/tui/components/pr-tracker.js.map +0 -7
- package/dist/features/tui/components/session-monitor.js +0 -299
- package/dist/features/tui/components/session-monitor.js.map +0 -7
- package/dist/features/tui/components/subagent-fleet.js +0 -395
- package/dist/features/tui/components/subagent-fleet.js.map +0 -7
- package/dist/features/tui/components/task-board.js +0 -1139
- package/dist/features/tui/components/task-board.js.map +0 -7
- package/dist/features/tui/index.js +0 -408
- package/dist/features/tui/index.js.map +0 -7
- package/dist/features/tui/services/data-service.js +0 -641
- package/dist/features/tui/services/data-service.js.map +0 -7
- package/dist/features/tui/services/linear-task-reader.js +0 -102
- package/dist/features/tui/services/linear-task-reader.js.map +0 -7
- package/dist/features/tui/services/websocket-client.js +0 -162
- package/dist/features/tui/services/websocket-client.js.map +0 -7
- package/dist/features/tui/terminal-compat.js +0 -220
- package/dist/features/tui/terminal-compat.js.map +0 -7
- package/dist/features/tui/types.js +0 -1
- package/dist/features/tui/types.js.map +0 -7
|
@@ -1,377 +0,0 @@
|
|
|
1
|
-
import blessed from "blessed";
|
|
2
|
-
import { EventEmitter } from "events";
|
|
3
|
-
class FrameVisualizer extends EventEmitter {
|
|
4
|
-
tree;
|
|
5
|
-
// contrib.tree type
|
|
6
|
-
frames;
|
|
7
|
-
rootNodes;
|
|
8
|
-
selectedFrame = null;
|
|
9
|
-
constructor(tree) {
|
|
10
|
-
super();
|
|
11
|
-
this.tree = tree;
|
|
12
|
-
this.frames = /* @__PURE__ */ new Map();
|
|
13
|
-
this.rootNodes = [];
|
|
14
|
-
this.initializeUI();
|
|
15
|
-
}
|
|
16
|
-
initializeUI() {
|
|
17
|
-
this.tree.rows.interactive = true;
|
|
18
|
-
this.tree.rows.mouse = true;
|
|
19
|
-
this.tree.setData({
|
|
20
|
-
extended: true,
|
|
21
|
-
children: {
|
|
22
|
-
Sessions: {
|
|
23
|
-
extended: true,
|
|
24
|
-
children: {
|
|
25
|
-
"Loading...": {}
|
|
26
|
-
}
|
|
27
|
-
}
|
|
28
|
-
}
|
|
29
|
-
});
|
|
30
|
-
this.tree.rows.on("select", (node) => {
|
|
31
|
-
if (node && node.frameId) {
|
|
32
|
-
this.selectFrame(node.frameId);
|
|
33
|
-
}
|
|
34
|
-
});
|
|
35
|
-
this.addLegend();
|
|
36
|
-
}
|
|
37
|
-
addLegend() {
|
|
38
|
-
const legend = `{gray-fg}Tiers: {red-fg}\u25CF Hot{/} {yellow-fg}\u25CF Warm{/} {blue-fg}\u25CF Cold{/}{/}`;
|
|
39
|
-
}
|
|
40
|
-
/**
|
|
41
|
-
* Build tree structure from flat frame data
|
|
42
|
-
*/
|
|
43
|
-
buildFrameTree(frames) {
|
|
44
|
-
const nodeMap = /* @__PURE__ */ new Map();
|
|
45
|
-
const rootNodes = [];
|
|
46
|
-
frames.forEach((frame) => {
|
|
47
|
-
const node = {
|
|
48
|
-
id: frame.id,
|
|
49
|
-
label: this.formatFrameLabel(frame),
|
|
50
|
-
children: [],
|
|
51
|
-
extended: frame.type === "root",
|
|
52
|
-
tier: frame.tier,
|
|
53
|
-
score: frame.score || 0
|
|
54
|
-
};
|
|
55
|
-
nodeMap.set(frame.id, node);
|
|
56
|
-
});
|
|
57
|
-
frames.forEach((frame) => {
|
|
58
|
-
const node = nodeMap.get(frame.id);
|
|
59
|
-
if (!node) return;
|
|
60
|
-
if (frame.parentId) {
|
|
61
|
-
const parent = nodeMap.get(frame.parentId);
|
|
62
|
-
if (parent) {
|
|
63
|
-
parent.children = parent.children || [];
|
|
64
|
-
parent.children.push(node);
|
|
65
|
-
} else {
|
|
66
|
-
rootNodes.push(node);
|
|
67
|
-
}
|
|
68
|
-
} else {
|
|
69
|
-
rootNodes.push(node);
|
|
70
|
-
}
|
|
71
|
-
});
|
|
72
|
-
const sortNodes = (nodes) => {
|
|
73
|
-
nodes.sort((a, b) => b.score - a.score);
|
|
74
|
-
nodes.forEach((node) => {
|
|
75
|
-
if (node.children && node.children.length > 0) {
|
|
76
|
-
sortNodes(node.children);
|
|
77
|
-
}
|
|
78
|
-
});
|
|
79
|
-
};
|
|
80
|
-
sortNodes(rootNodes);
|
|
81
|
-
return rootNodes;
|
|
82
|
-
}
|
|
83
|
-
formatFrameLabel(frame) {
|
|
84
|
-
const tierIcon = this.getTierIcon(frame.tier);
|
|
85
|
-
const typeIcon = this.getTypeIcon(frame.type);
|
|
86
|
-
const timestamp = new Date(frame.timestamp).toLocaleTimeString();
|
|
87
|
-
const tokens = frame.tokenCount ? `[${this.formatTokens(frame.tokenCount)}]` : "";
|
|
88
|
-
const compression = frame.compressionRatio ? `(${frame.compressionRatio.toFixed(1)}x)` : "";
|
|
89
|
-
let label = `${tierIcon} ${typeIcon} ${timestamp} ${tokens} ${compression}`;
|
|
90
|
-
if (frame.digest) {
|
|
91
|
-
const digest = frame.digest.length > 30 ? frame.digest.substring(0, 30) + "..." : frame.digest;
|
|
92
|
-
label += ` - ${digest}`;
|
|
93
|
-
}
|
|
94
|
-
return label;
|
|
95
|
-
}
|
|
96
|
-
getTierIcon(tier) {
|
|
97
|
-
switch (tier) {
|
|
98
|
-
case "hot":
|
|
99
|
-
return "{red-fg}\u25CF{/}";
|
|
100
|
-
case "warm":
|
|
101
|
-
return "{yellow-fg}\u25CF{/}";
|
|
102
|
-
case "cold":
|
|
103
|
-
return "{blue-fg}\u25CF{/}";
|
|
104
|
-
default:
|
|
105
|
-
return "{gray-fg}\u25CB{/}";
|
|
106
|
-
}
|
|
107
|
-
}
|
|
108
|
-
getTypeIcon(type) {
|
|
109
|
-
switch (type) {
|
|
110
|
-
case "root":
|
|
111
|
-
return "\u{1F4C1}";
|
|
112
|
-
case "branch":
|
|
113
|
-
return "\u{1F33F}";
|
|
114
|
-
case "leaf":
|
|
115
|
-
return "\u{1F343}";
|
|
116
|
-
default:
|
|
117
|
-
return "\u{1F4C4}";
|
|
118
|
-
}
|
|
119
|
-
}
|
|
120
|
-
formatTokens(tokens) {
|
|
121
|
-
if (tokens < 1e3) return `${tokens}`;
|
|
122
|
-
if (tokens < 1e6) return `${(tokens / 1e3).toFixed(1)}K`;
|
|
123
|
-
return `${(tokens / 1e6).toFixed(1)}M`;
|
|
124
|
-
}
|
|
125
|
-
/**
|
|
126
|
-
* Convert FrameNode tree to blessed-contrib tree format
|
|
127
|
-
*/
|
|
128
|
-
convertToTreeData(nodes) {
|
|
129
|
-
const children = {};
|
|
130
|
-
const sessionGroups = /* @__PURE__ */ new Map();
|
|
131
|
-
nodes.forEach((node) => {
|
|
132
|
-
const sessionId = this.frames.get(node.id)?.sessionId || "unknown";
|
|
133
|
-
if (!sessionGroups.has(sessionId)) {
|
|
134
|
-
sessionGroups.set(sessionId, []);
|
|
135
|
-
}
|
|
136
|
-
sessionGroups.get(sessionId).push(node);
|
|
137
|
-
});
|
|
138
|
-
sessionGroups.forEach((sessionNodes, sessionId) => {
|
|
139
|
-
const sessionLabel = `Session: ${sessionId.substring(0, 8)}`;
|
|
140
|
-
children[sessionLabel] = {
|
|
141
|
-
extended: true,
|
|
142
|
-
children: this.buildNodeChildren(sessionNodes)
|
|
143
|
-
};
|
|
144
|
-
});
|
|
145
|
-
const stats = this.calculateStatistics();
|
|
146
|
-
children["Statistics"] = {
|
|
147
|
-
extended: false,
|
|
148
|
-
children: {
|
|
149
|
-
[`Total Frames: ${stats.total}`]: {},
|
|
150
|
-
[`Hot Tier: ${stats.hot} (${stats.hotSize})`]: {},
|
|
151
|
-
[`Warm Tier: ${stats.warm} (${stats.warmSize})`]: {},
|
|
152
|
-
[`Cold Tier: ${stats.cold} (${stats.coldSize})`]: {},
|
|
153
|
-
[`Avg Compression: ${stats.avgCompression.toFixed(1)}x`]: {},
|
|
154
|
-
[`Total Tokens: ${this.formatTokens(stats.totalTokens)}`]: {}
|
|
155
|
-
}
|
|
156
|
-
};
|
|
157
|
-
return { extended: true, children };
|
|
158
|
-
}
|
|
159
|
-
buildNodeChildren(nodes) {
|
|
160
|
-
const children = {};
|
|
161
|
-
nodes.forEach((node) => {
|
|
162
|
-
const nodeData = {
|
|
163
|
-
frameId: node.id,
|
|
164
|
-
extended: node.extended
|
|
165
|
-
};
|
|
166
|
-
if (node.children && node.children.length > 0) {
|
|
167
|
-
nodeData.children = this.buildNodeChildren(node.children);
|
|
168
|
-
}
|
|
169
|
-
children[node.label] = nodeData;
|
|
170
|
-
});
|
|
171
|
-
return children;
|
|
172
|
-
}
|
|
173
|
-
calculateStatistics() {
|
|
174
|
-
const frames = Array.from(this.frames.values());
|
|
175
|
-
const stats = {
|
|
176
|
-
total: frames.length,
|
|
177
|
-
hot: 0,
|
|
178
|
-
warm: 0,
|
|
179
|
-
cold: 0,
|
|
180
|
-
hotSize: "0B",
|
|
181
|
-
warmSize: "0B",
|
|
182
|
-
coldSize: "0B",
|
|
183
|
-
totalTokens: 0,
|
|
184
|
-
avgCompression: 0
|
|
185
|
-
};
|
|
186
|
-
let hotBytes = 0;
|
|
187
|
-
let warmBytes = 0;
|
|
188
|
-
let coldBytes = 0;
|
|
189
|
-
let totalCompression = 0;
|
|
190
|
-
let compressionCount = 0;
|
|
191
|
-
frames.forEach((frame) => {
|
|
192
|
-
switch (frame.tier) {
|
|
193
|
-
case "hot":
|
|
194
|
-
stats.hot++;
|
|
195
|
-
hotBytes += frame.tokenCount * 4;
|
|
196
|
-
break;
|
|
197
|
-
case "warm":
|
|
198
|
-
stats.warm++;
|
|
199
|
-
warmBytes += frame.tokenCount * 4;
|
|
200
|
-
break;
|
|
201
|
-
case "cold":
|
|
202
|
-
stats.cold++;
|
|
203
|
-
coldBytes += frame.tokenCount * 4;
|
|
204
|
-
break;
|
|
205
|
-
}
|
|
206
|
-
stats.totalTokens += frame.tokenCount || 0;
|
|
207
|
-
if (frame.compressionRatio) {
|
|
208
|
-
totalCompression += frame.compressionRatio;
|
|
209
|
-
compressionCount++;
|
|
210
|
-
}
|
|
211
|
-
});
|
|
212
|
-
stats.hotSize = this.formatBytes(hotBytes);
|
|
213
|
-
stats.warmSize = this.formatBytes(warmBytes);
|
|
214
|
-
stats.coldSize = this.formatBytes(coldBytes);
|
|
215
|
-
stats.avgCompression = compressionCount > 0 ? totalCompression / compressionCount : 1;
|
|
216
|
-
return stats;
|
|
217
|
-
}
|
|
218
|
-
formatBytes(bytes) {
|
|
219
|
-
if (bytes < 1024) return `${bytes}B`;
|
|
220
|
-
if (bytes < 1048576) return `${(bytes / 1024).toFixed(1)}KB`;
|
|
221
|
-
if (bytes < 1073741824) return `${(bytes / 1048576).toFixed(1)}MB`;
|
|
222
|
-
return `${(bytes / 1073741824).toFixed(1)}GB`;
|
|
223
|
-
}
|
|
224
|
-
update(frames) {
|
|
225
|
-
this.frames.clear();
|
|
226
|
-
frames.forEach((frame) => {
|
|
227
|
-
this.frames.set(frame.id, frame);
|
|
228
|
-
});
|
|
229
|
-
this.rootNodes = this.buildFrameTree(frames);
|
|
230
|
-
const treeData = this.convertToTreeData(this.rootNodes);
|
|
231
|
-
this.tree.setData(treeData);
|
|
232
|
-
const stats = this.calculateStatistics();
|
|
233
|
-
this.tree.screen.render();
|
|
234
|
-
}
|
|
235
|
-
selectFrame(frameId) {
|
|
236
|
-
this.selectedFrame = frameId;
|
|
237
|
-
const frame = this.frames.get(frameId);
|
|
238
|
-
if (frame) {
|
|
239
|
-
this.emit("frame:selected", frame);
|
|
240
|
-
this.showFrameDetails(frame);
|
|
241
|
-
}
|
|
242
|
-
}
|
|
243
|
-
showFrameDetails(frame) {
|
|
244
|
-
const details = blessed.box({
|
|
245
|
-
parent: this.tree.screen,
|
|
246
|
-
top: "center",
|
|
247
|
-
left: "center",
|
|
248
|
-
width: "60%",
|
|
249
|
-
height: "60%",
|
|
250
|
-
content: this.formatFrameDetails(frame),
|
|
251
|
-
tags: true,
|
|
252
|
-
border: {
|
|
253
|
-
type: "line"
|
|
254
|
-
},
|
|
255
|
-
style: {
|
|
256
|
-
border: {
|
|
257
|
-
fg: "yellow"
|
|
258
|
-
}
|
|
259
|
-
},
|
|
260
|
-
scrollable: true,
|
|
261
|
-
keys: true,
|
|
262
|
-
vi: true,
|
|
263
|
-
mouse: true,
|
|
264
|
-
hidden: false,
|
|
265
|
-
label: ` Frame: ${frame.id.substring(0, 8)} `
|
|
266
|
-
});
|
|
267
|
-
details.key(["escape", "q"], () => {
|
|
268
|
-
details.destroy();
|
|
269
|
-
this.tree.screen.render();
|
|
270
|
-
});
|
|
271
|
-
details.key(["m"], () => {
|
|
272
|
-
this.migrateFrame(frame);
|
|
273
|
-
details.destroy();
|
|
274
|
-
this.tree.screen.render();
|
|
275
|
-
});
|
|
276
|
-
details.focus();
|
|
277
|
-
this.tree.screen.render();
|
|
278
|
-
}
|
|
279
|
-
formatFrameDetails(frame) {
|
|
280
|
-
let details = `{bold}Frame ID:{/} ${frame.id}
|
|
281
|
-
`;
|
|
282
|
-
details += `{bold}Session:{/} ${frame.sessionId}
|
|
283
|
-
`;
|
|
284
|
-
details += `{bold}Type:{/} ${frame.type}
|
|
285
|
-
`;
|
|
286
|
-
details += `{bold}Tier:{/} ${this.getTierIcon(frame.tier)} ${frame.tier}
|
|
287
|
-
`;
|
|
288
|
-
details += `{bold}Created:{/} ${new Date(frame.timestamp).toLocaleString()}
|
|
289
|
-
`;
|
|
290
|
-
if (frame.parentId) {
|
|
291
|
-
details += `{bold}Parent:{/} ${frame.parentId.substring(0, 8)}
|
|
292
|
-
`;
|
|
293
|
-
}
|
|
294
|
-
details += `
|
|
295
|
-
{bold}Metrics:{/}
|
|
296
|
-
`;
|
|
297
|
-
details += ` Tokens: ${this.formatTokens(frame.tokenCount)}
|
|
298
|
-
`;
|
|
299
|
-
details += ` Score: ${frame.score || 0}
|
|
300
|
-
`;
|
|
301
|
-
if (frame.compressionRatio) {
|
|
302
|
-
const saved = Math.round((1 - 1 / frame.compressionRatio) * 100);
|
|
303
|
-
details += ` Compression: ${frame.compressionRatio.toFixed(2)}x (${saved}% saved)
|
|
304
|
-
`;
|
|
305
|
-
}
|
|
306
|
-
if (frame.digest) {
|
|
307
|
-
details += `
|
|
308
|
-
{bold}Digest:{/}
|
|
309
|
-
${frame.digest}
|
|
310
|
-
`;
|
|
311
|
-
}
|
|
312
|
-
if (frame.tools && frame.tools.length > 0) {
|
|
313
|
-
details += `
|
|
314
|
-
{bold}Tools Used:{/}
|
|
315
|
-
`;
|
|
316
|
-
frame.tools.forEach((tool) => {
|
|
317
|
-
details += ` \u2022 ${tool}
|
|
318
|
-
`;
|
|
319
|
-
});
|
|
320
|
-
}
|
|
321
|
-
if (frame.inputs && frame.inputs.length > 0) {
|
|
322
|
-
details += `
|
|
323
|
-
{bold}Inputs ({${frame.inputs.length}}):{/}
|
|
324
|
-
`;
|
|
325
|
-
frame.inputs.slice(0, 3).forEach((input) => {
|
|
326
|
-
const preview = input.length > 50 ? input.substring(0, 50) + "..." : input;
|
|
327
|
-
details += ` ${preview}
|
|
328
|
-
`;
|
|
329
|
-
});
|
|
330
|
-
}
|
|
331
|
-
if (frame.outputs && frame.outputs.length > 0) {
|
|
332
|
-
details += `
|
|
333
|
-
{bold}Outputs ({${frame.outputs.length}}):{/}
|
|
334
|
-
`;
|
|
335
|
-
frame.outputs.slice(0, 3).forEach((output) => {
|
|
336
|
-
const preview = output.length > 50 ? output.substring(0, 50) + "..." : output;
|
|
337
|
-
details += ` ${preview}
|
|
338
|
-
`;
|
|
339
|
-
});
|
|
340
|
-
}
|
|
341
|
-
if (frame.children && frame.children.length > 0) {
|
|
342
|
-
details += `
|
|
343
|
-
{bold}Children:{/} ${frame.children.length} frames
|
|
344
|
-
`;
|
|
345
|
-
}
|
|
346
|
-
if (frame.references && frame.references.length > 0) {
|
|
347
|
-
details += `{bold}References:{/} ${frame.references.length} frames
|
|
348
|
-
`;
|
|
349
|
-
}
|
|
350
|
-
details += `
|
|
351
|
-
{gray-fg}[m] Migrate Tier | [d] Delete | [e] Export{/}
|
|
352
|
-
`;
|
|
353
|
-
return details;
|
|
354
|
-
}
|
|
355
|
-
migrateFrame(frame) {
|
|
356
|
-
const tierOrder = ["hot", "warm", "cold"];
|
|
357
|
-
const currentIndex = tierOrder.indexOf(frame.tier);
|
|
358
|
-
const nextTier = tierOrder[(currentIndex + 1) % tierOrder.length];
|
|
359
|
-
this.emit("frame:migrate", {
|
|
360
|
-
frameId: frame.id,
|
|
361
|
-
fromTier: frame.tier,
|
|
362
|
-
toTier: nextTier
|
|
363
|
-
});
|
|
364
|
-
frame.tier = nextTier;
|
|
365
|
-
this.update(Array.from(this.frames.values()));
|
|
366
|
-
}
|
|
367
|
-
focus() {
|
|
368
|
-
this.tree.rows.focus();
|
|
369
|
-
}
|
|
370
|
-
hasFocus() {
|
|
371
|
-
return this.tree.rows === this.tree.screen.focused;
|
|
372
|
-
}
|
|
373
|
-
}
|
|
374
|
-
export {
|
|
375
|
-
FrameVisualizer
|
|
376
|
-
};
|
|
377
|
-
//# sourceMappingURL=frame-visualizer.js.map
|
|
@@ -1,7 +0,0 @@
|
|
|
1
|
-
{
|
|
2
|
-
"version": 3,
|
|
3
|
-
"sources": ["../../../../src/features/tui/components/frame-visualizer.ts"],
|
|
4
|
-
"sourcesContent": ["/**\n * Frame Visualizer Component\n * Displays frame storage hierarchy with tier indicators\n */\n\nimport blessed from 'blessed';\nimport contrib from 'blessed-contrib';\nimport { EventEmitter } from 'events';\nimport type { FrameData, FrameNode } from '../types.js';\n\nexport class FrameVisualizer extends EventEmitter {\n private tree: any; // contrib.tree type\n private frames: Map<string, FrameData>;\n private rootNodes: FrameNode[];\n private selectedFrame: string | null = null;\n\n constructor(tree: any) {\n super();\n this.tree = tree;\n this.frames = new Map();\n this.rootNodes = [];\n this.initializeUI();\n }\n\n private initializeUI(): void {\n // Configure tree widget\n this.tree.rows.interactive = true;\n this.tree.rows.mouse = true;\n\n // Set initial data\n this.tree.setData({\n extended: true,\n children: {\n Sessions: {\n extended: true,\n children: {\n 'Loading...': {},\n },\n },\n },\n });\n\n // Handle selection\n this.tree.rows.on('select', (node: any) => {\n if (node && node.frameId) {\n this.selectFrame(node.frameId);\n }\n });\n\n // Add legend\n this.addLegend();\n }\n\n private addLegend(): void {\n const legend = `{gray-fg}Tiers: {red-fg}\u25CF Hot{/} {yellow-fg}\u25CF Warm{/} {blue-fg}\u25CF Cold{/}{/}`;\n // The tree widget itself doesn't have setLabel, only box widgets do\n // We'll update the label when we update the tree data\n }\n\n /**\n * Build tree structure from flat frame data\n */\n private buildFrameTree(frames: FrameData[]): FrameNode[] {\n const nodeMap = new Map<string, FrameNode>();\n const rootNodes: FrameNode[] = [];\n\n // Create nodes\n frames.forEach((frame) => {\n const node: FrameNode = {\n id: frame.id,\n label: this.formatFrameLabel(frame),\n children: [],\n extended: frame.type === 'root',\n tier: frame.tier,\n score: frame.score || 0,\n };\n nodeMap.set(frame.id, node);\n });\n\n // Build hierarchy\n frames.forEach((frame) => {\n const node = nodeMap.get(frame.id);\n if (!node) return;\n\n if (frame.parentId) {\n const parent = nodeMap.get(frame.parentId);\n if (parent) {\n parent.children = parent.children || [];\n parent.children.push(node);\n } else {\n rootNodes.push(node);\n }\n } else {\n rootNodes.push(node);\n }\n });\n\n // Sort by score and timestamp\n const sortNodes = (nodes: FrameNode[]) => {\n nodes.sort((a, b) => b.score - a.score);\n nodes.forEach((node) => {\n if (node.children && node.children.length > 0) {\n sortNodes(node.children);\n }\n });\n };\n\n sortNodes(rootNodes);\n return rootNodes;\n }\n\n private formatFrameLabel(frame: FrameData): string {\n const tierIcon = this.getTierIcon(frame.tier);\n const typeIcon = this.getTypeIcon(frame.type);\n const timestamp = new Date(frame.timestamp).toLocaleTimeString();\n const tokens = frame.tokenCount\n ? `[${this.formatTokens(frame.tokenCount)}]`\n : '';\n const compression = frame.compressionRatio\n ? `(${frame.compressionRatio.toFixed(1)}x)`\n : '';\n\n let label = `${tierIcon} ${typeIcon} ${timestamp} ${tokens} ${compression}`;\n\n if (frame.digest) {\n const digest =\n frame.digest.length > 30\n ? frame.digest.substring(0, 30) + '...'\n : frame.digest;\n label += ` - ${digest}`;\n }\n\n return label;\n }\n\n private getTierIcon(tier: string): string {\n switch (tier) {\n case 'hot':\n return '{red-fg}\u25CF{/}';\n case 'warm':\n return '{yellow-fg}\u25CF{/}';\n case 'cold':\n return '{blue-fg}\u25CF{/}';\n default:\n return '{gray-fg}\u25CB{/}';\n }\n }\n\n private getTypeIcon(type: string): string {\n switch (type) {\n case 'root':\n return '\uD83D\uDCC1';\n case 'branch':\n return '\uD83C\uDF3F';\n case 'leaf':\n return '\uD83C\uDF43';\n default:\n return '\uD83D\uDCC4';\n }\n }\n\n private formatTokens(tokens: number): string {\n if (tokens < 1000) return `${tokens}`;\n if (tokens < 1000000) return `${(tokens / 1000).toFixed(1)}K`;\n return `${(tokens / 1000000).toFixed(1)}M`;\n }\n\n /**\n * Convert FrameNode tree to blessed-contrib tree format\n */\n private convertToTreeData(nodes: FrameNode[]): any {\n // Blessed-contrib tree expects an object with { extended, children }\n const children: any = {};\n\n // Group by session\n const sessionGroups = new Map<string, FrameNode[]>();\n\n nodes.forEach((node) => {\n const sessionId = this.frames.get(node.id)?.sessionId || 'unknown';\n if (!sessionGroups.has(sessionId)) {\n sessionGroups.set(sessionId, []);\n }\n sessionGroups.get(sessionId)!.push(node);\n });\n\n // Build tree structure\n sessionGroups.forEach((sessionNodes, sessionId) => {\n const sessionLabel = `Session: ${sessionId.substring(0, 8)}`;\n children[sessionLabel] = {\n extended: true,\n children: this.buildNodeChildren(sessionNodes),\n };\n });\n\n // Add statistics\n const stats = this.calculateStatistics();\n children['Statistics'] = {\n extended: false,\n children: {\n [`Total Frames: ${stats.total}`]: {},\n [`Hot Tier: ${stats.hot} (${stats.hotSize})`]: {},\n [`Warm Tier: ${stats.warm} (${stats.warmSize})`]: {},\n [`Cold Tier: ${stats.cold} (${stats.coldSize})`]: {},\n [`Avg Compression: ${stats.avgCompression.toFixed(1)}x`]: {},\n [`Total Tokens: ${this.formatTokens(stats.totalTokens)}`]: {},\n },\n };\n\n return { extended: true, children };\n }\n\n private buildNodeChildren(nodes: FrameNode[]): any {\n const children: any = {};\n\n nodes.forEach((node) => {\n const nodeData: any = {\n frameId: node.id,\n extended: node.extended,\n };\n\n if (node.children && node.children.length > 0) {\n nodeData.children = this.buildNodeChildren(node.children);\n }\n\n children[node.label] = nodeData;\n });\n\n return children;\n }\n\n private calculateStatistics(): any {\n const frames = Array.from(this.frames.values());\n const stats = {\n total: frames.length,\n hot: 0,\n warm: 0,\n cold: 0,\n hotSize: '0B',\n warmSize: '0B',\n coldSize: '0B',\n totalTokens: 0,\n avgCompression: 0,\n };\n\n let hotBytes = 0;\n let warmBytes = 0;\n let coldBytes = 0;\n let totalCompression = 0;\n let compressionCount = 0;\n\n frames.forEach((frame) => {\n // Count by tier\n switch (frame.tier) {\n case 'hot':\n stats.hot++;\n hotBytes += frame.tokenCount * 4; // Rough estimate: 4 bytes per token\n break;\n case 'warm':\n stats.warm++;\n warmBytes += frame.tokenCount * 4;\n break;\n case 'cold':\n stats.cold++;\n coldBytes += frame.tokenCount * 4;\n break;\n }\n\n stats.totalTokens += frame.tokenCount || 0;\n\n if (frame.compressionRatio) {\n totalCompression += frame.compressionRatio;\n compressionCount++;\n }\n });\n\n stats.hotSize = this.formatBytes(hotBytes);\n stats.warmSize = this.formatBytes(warmBytes);\n stats.coldSize = this.formatBytes(coldBytes);\n stats.avgCompression =\n compressionCount > 0 ? totalCompression / compressionCount : 1.0;\n\n return stats;\n }\n\n private formatBytes(bytes: number): string {\n if (bytes < 1024) return `${bytes}B`;\n if (bytes < 1048576) return `${(bytes / 1024).toFixed(1)}KB`;\n if (bytes < 1073741824) return `${(bytes / 1048576).toFixed(1)}MB`;\n return `${(bytes / 1073741824).toFixed(1)}GB`;\n }\n\n public update(frames: FrameData[]): void {\n // Update frame map\n this.frames.clear();\n frames.forEach((frame) => {\n this.frames.set(frame.id, frame);\n });\n\n // Build tree structure\n this.rootNodes = this.buildFrameTree(frames);\n\n // Convert to tree widget format\n const treeData = this.convertToTreeData(this.rootNodes);\n\n // Update tree widget\n this.tree.setData(treeData);\n\n // Update label with stats if the tree has a parent box container\n const stats = this.calculateStatistics();\n // Stats will be shown in the tree data instead\n\n this.tree.screen.render();\n }\n\n private selectFrame(frameId: string): void {\n this.selectedFrame = frameId;\n const frame = this.frames.get(frameId);\n if (frame) {\n this.emit('frame:selected', frame);\n this.showFrameDetails(frame);\n }\n }\n\n private showFrameDetails(frame: FrameData): void {\n const details = blessed.box({\n parent: this.tree.screen,\n top: 'center',\n left: 'center',\n width: '60%',\n height: '60%',\n content: this.formatFrameDetails(frame),\n tags: true,\n border: {\n type: 'line',\n },\n style: {\n border: {\n fg: 'yellow',\n },\n },\n scrollable: true,\n keys: true,\n vi: true,\n mouse: true,\n hidden: false,\n label: ` Frame: ${frame.id.substring(0, 8)} `,\n });\n\n details.key(['escape', 'q'], () => {\n details.destroy();\n this.tree.screen.render();\n });\n\n // Add migrate button\n details.key(['m'], () => {\n this.migrateFrame(frame);\n details.destroy();\n this.tree.screen.render();\n });\n\n details.focus();\n this.tree.screen.render();\n }\n\n private formatFrameDetails(frame: FrameData): string {\n let details = `{bold}Frame ID:{/} ${frame.id}\\n`;\n details += `{bold}Session:{/} ${frame.sessionId}\\n`;\n details += `{bold}Type:{/} ${frame.type}\\n`;\n details += `{bold}Tier:{/} ${this.getTierIcon(frame.tier)} ${frame.tier}\\n`;\n details += `{bold}Created:{/} ${new Date(frame.timestamp).toLocaleString()}\\n`;\n\n if (frame.parentId) {\n details += `{bold}Parent:{/} ${frame.parentId.substring(0, 8)}\\n`;\n }\n\n details += `\\n{bold}Metrics:{/}\\n`;\n details += ` Tokens: ${this.formatTokens(frame.tokenCount)}\\n`;\n details += ` Score: ${frame.score || 0}\\n`;\n\n if (frame.compressionRatio) {\n const saved = Math.round((1 - 1 / frame.compressionRatio) * 100);\n details += ` Compression: ${frame.compressionRatio.toFixed(2)}x (${saved}% saved)\\n`;\n }\n\n if (frame.digest) {\n details += `\\n{bold}Digest:{/}\\n${frame.digest}\\n`;\n }\n\n if (frame.tools && frame.tools.length > 0) {\n details += `\\n{bold}Tools Used:{/}\\n`;\n frame.tools.forEach((tool) => {\n details += ` \u2022 ${tool}\\n`;\n });\n }\n\n if (frame.inputs && frame.inputs.length > 0) {\n details += `\\n{bold}Inputs ({${frame.inputs.length}}):{/}\\n`;\n frame.inputs.slice(0, 3).forEach((input) => {\n const preview =\n input.length > 50 ? input.substring(0, 50) + '...' : input;\n details += ` ${preview}\\n`;\n });\n }\n\n if (frame.outputs && frame.outputs.length > 0) {\n details += `\\n{bold}Outputs ({${frame.outputs.length}}):{/}\\n`;\n frame.outputs.slice(0, 3).forEach((output) => {\n const preview =\n output.length > 50 ? output.substring(0, 50) + '...' : output;\n details += ` ${preview}\\n`;\n });\n }\n\n if (frame.children && frame.children.length > 0) {\n details += `\\n{bold}Children:{/} ${frame.children.length} frames\\n`;\n }\n\n if (frame.references && frame.references.length > 0) {\n details += `{bold}References:{/} ${frame.references.length} frames\\n`;\n }\n\n details += `\\n{gray-fg}[m] Migrate Tier | [d] Delete | [e] Export{/}\\n`;\n\n return details;\n }\n\n private migrateFrame(frame: FrameData): void {\n // Determine next tier\n const tierOrder = ['hot', 'warm', 'cold'];\n const currentIndex = tierOrder.indexOf(frame.tier);\n const nextTier = tierOrder[(currentIndex + 1) % tierOrder.length];\n\n // Emit migration event\n this.emit('frame:migrate', {\n frameId: frame.id,\n fromTier: frame.tier,\n toTier: nextTier,\n });\n\n // Optimistically update UI\n frame.tier = nextTier as 'hot' | 'warm' | 'cold';\n this.update(Array.from(this.frames.values()));\n }\n\n public focus(): void {\n this.tree.rows.focus();\n }\n\n public hasFocus(): boolean {\n return this.tree.rows === this.tree.screen.focused;\n }\n}\n"],
|
|
5
|
-
"mappings": "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",
|
|
6
|
-
"names": []
|
|
7
|
-
}
|
|
@@ -1,135 +0,0 @@
|
|
|
1
|
-
import { EventEmitter } from "events";
|
|
2
|
-
class PRTracker extends EventEmitter {
|
|
3
|
-
list;
|
|
4
|
-
prs;
|
|
5
|
-
issues;
|
|
6
|
-
selectedItem = null;
|
|
7
|
-
viewMode = "prs";
|
|
8
|
-
constructor(list) {
|
|
9
|
-
super();
|
|
10
|
-
this.list = list;
|
|
11
|
-
this.prs = /* @__PURE__ */ new Map();
|
|
12
|
-
this.issues = /* @__PURE__ */ new Map();
|
|
13
|
-
this.initializeUI();
|
|
14
|
-
}
|
|
15
|
-
initializeUI() {
|
|
16
|
-
this.list.on("select", (item, index) => {
|
|
17
|
-
if (this.viewMode === "prs") {
|
|
18
|
-
const prId = Array.from(this.prs.keys())[index];
|
|
19
|
-
this.selectPR(prId);
|
|
20
|
-
} else {
|
|
21
|
-
const issueId = Array.from(this.issues.keys())[index];
|
|
22
|
-
this.selectIssue(issueId);
|
|
23
|
-
}
|
|
24
|
-
});
|
|
25
|
-
this.list.key(["tab"], () => {
|
|
26
|
-
this.toggleView();
|
|
27
|
-
});
|
|
28
|
-
}
|
|
29
|
-
formatPRItem(pr) {
|
|
30
|
-
const status = this.getPRStatusIcon(pr);
|
|
31
|
-
const checks = this.getChecksStatus(pr.checks);
|
|
32
|
-
const reviews = this.getReviewStatus(pr.reviews);
|
|
33
|
-
let item = `${status} #${pr.number}: ${pr.title}
|
|
34
|
-
`;
|
|
35
|
-
item += ` {gray-fg}@${pr.author.login} | ${checks} | ${reviews} | +${pr.additions}/-${pr.deletions}{/}`;
|
|
36
|
-
if (pr.linkedIssues?.length) {
|
|
37
|
-
item += ` | {cyan-fg}\u{1F517}${pr.linkedIssues.length}{/}`;
|
|
38
|
-
}
|
|
39
|
-
return item;
|
|
40
|
-
}
|
|
41
|
-
formatIssueItem(issue) {
|
|
42
|
-
const status = issue.state === "open" ? "{green-fg}\u25CF{/}" : "{gray-fg}\u25CF{/}";
|
|
43
|
-
const assignees = issue.assignees?.map((a) => `@${a}`).join(", ") || "unassigned";
|
|
44
|
-
let item = `${status} #${issue.number}: ${issue.title}
|
|
45
|
-
`;
|
|
46
|
-
item += ` {gray-fg}@${issue.author.login} | ${assignees} | \u{1F4AC}${issue.comments}{/}`;
|
|
47
|
-
return item;
|
|
48
|
-
}
|
|
49
|
-
getPRStatusIcon(pr) {
|
|
50
|
-
if (pr.state === "merged") return "{magenta-fg}\u2B24{/}";
|
|
51
|
-
if (pr.state === "closed") return "{red-fg}\u2B24{/}";
|
|
52
|
-
if (pr.draft) return "{gray-fg}\u25CB{/}";
|
|
53
|
-
return "{green-fg}\u25CF{/}";
|
|
54
|
-
}
|
|
55
|
-
getChecksStatus(checks) {
|
|
56
|
-
if (!checks) return "{gray-fg}no checks{/}";
|
|
57
|
-
if (checks.failed > 0)
|
|
58
|
-
return `{red-fg}\u2717 ${checks.failed}/${checks.total}{/}`;
|
|
59
|
-
if (checks.pending > 0)
|
|
60
|
-
return `{yellow-fg}\u23F3 ${checks.pending}/${checks.total}{/}`;
|
|
61
|
-
return `{green-fg}\u2713 ${checks.passed}/${checks.total}{/}`;
|
|
62
|
-
}
|
|
63
|
-
getReviewStatus(reviews) {
|
|
64
|
-
const approved = reviews.filter((r) => r.state === "approved").length;
|
|
65
|
-
const changes = reviews.filter(
|
|
66
|
-
(r) => r.state === "changes_requested"
|
|
67
|
-
).length;
|
|
68
|
-
if (changes > 0) return `{red-fg}\u{1F44E}${changes}{/}`;
|
|
69
|
-
if (approved > 0) return `{green-fg}\u{1F44D}${approved}{/}`;
|
|
70
|
-
return "{gray-fg}no reviews{/}";
|
|
71
|
-
}
|
|
72
|
-
update(data) {
|
|
73
|
-
if (data.prs) {
|
|
74
|
-
this.prs.clear();
|
|
75
|
-
data.prs.forEach((pr) => this.prs.set(pr.id, pr));
|
|
76
|
-
}
|
|
77
|
-
if (data.issues) {
|
|
78
|
-
this.issues.clear();
|
|
79
|
-
data.issues.forEach((issue) => this.issues.set(issue.id, issue));
|
|
80
|
-
}
|
|
81
|
-
this.refreshDisplay();
|
|
82
|
-
}
|
|
83
|
-
refreshDisplay() {
|
|
84
|
-
let items = [];
|
|
85
|
-
let label = "";
|
|
86
|
-
if (this.viewMode === "prs") {
|
|
87
|
-
items = Array.from(this.prs.values()).map(
|
|
88
|
-
(pr) => this.formatPRItem(pr)
|
|
89
|
-
);
|
|
90
|
-
const open = Array.from(this.prs.values()).filter(
|
|
91
|
-
(pr) => pr.state === "open"
|
|
92
|
-
).length;
|
|
93
|
-
label = ` \u{1F500} Pull Requests (${open}/${this.prs.size}) [Tab] Issues `;
|
|
94
|
-
} else {
|
|
95
|
-
items = Array.from(this.issues.values()).map(
|
|
96
|
-
(issue) => this.formatIssueItem(issue)
|
|
97
|
-
);
|
|
98
|
-
const open = Array.from(this.issues.values()).filter(
|
|
99
|
-
(i) => i.state === "open"
|
|
100
|
-
).length;
|
|
101
|
-
label = ` \u{1F41B} Issues (${open}/${this.issues.size}) [Tab] PRs `;
|
|
102
|
-
}
|
|
103
|
-
this.list.setItems(items);
|
|
104
|
-
if (this.list.parent && typeof this.list.parent.setLabel === "function") {
|
|
105
|
-
this.list.parent.setLabel(label);
|
|
106
|
-
}
|
|
107
|
-
this.list.screen.render();
|
|
108
|
-
}
|
|
109
|
-
toggleView() {
|
|
110
|
-
this.viewMode = this.viewMode === "prs" ? "issues" : "prs";
|
|
111
|
-
this.refreshDisplay();
|
|
112
|
-
}
|
|
113
|
-
selectPR(prId) {
|
|
114
|
-
const pr = this.prs.get(prId);
|
|
115
|
-
if (pr) {
|
|
116
|
-
this.emit("pr:selected", pr);
|
|
117
|
-
}
|
|
118
|
-
}
|
|
119
|
-
selectIssue(issueId) {
|
|
120
|
-
const issue = this.issues.get(issueId);
|
|
121
|
-
if (issue) {
|
|
122
|
-
this.emit("issue:selected", issue);
|
|
123
|
-
}
|
|
124
|
-
}
|
|
125
|
-
focus() {
|
|
126
|
-
this.list.focus();
|
|
127
|
-
}
|
|
128
|
-
hasFocus() {
|
|
129
|
-
return this.list === this.list.screen.focused;
|
|
130
|
-
}
|
|
131
|
-
}
|
|
132
|
-
export {
|
|
133
|
-
PRTracker
|
|
134
|
-
};
|
|
135
|
-
//# sourceMappingURL=pr-tracker.js.map
|
|
@@ -1,7 +0,0 @@
|
|
|
1
|
-
{
|
|
2
|
-
"version": 3,
|
|
3
|
-
"sources": ["../../../../src/features/tui/components/pr-tracker.ts"],
|
|
4
|
-
"sourcesContent": ["/**\n * PR/Issue Tracker Component\n * Displays GitHub PRs and issues with status indicators\n */\n\nimport blessed from 'blessed';\nimport { EventEmitter } from 'events';\nimport type { PRData, IssueData } from '../types.js';\n\nexport class PRTracker extends EventEmitter {\n private list: blessed.Widgets.ListElement;\n private prs: Map<string, PRData>;\n private issues: Map<string, IssueData>;\n private selectedItem: string | null = null;\n private viewMode: 'prs' | 'issues' = 'prs';\n\n constructor(list: blessed.Widgets.ListElement) {\n super();\n this.list = list;\n this.prs = new Map();\n this.issues = new Map();\n this.initializeUI();\n }\n\n private initializeUI(): void {\n this.list.on('select', (item, index) => {\n if (this.viewMode === 'prs') {\n const prId = Array.from(this.prs.keys())[index];\n this.selectPR(prId);\n } else {\n const issueId = Array.from(this.issues.keys())[index];\n this.selectIssue(issueId);\n }\n });\n\n // Toggle between PRs and Issues\n this.list.key(['tab'], () => {\n this.toggleView();\n });\n }\n\n private formatPRItem(pr: PRData): string {\n const status = this.getPRStatusIcon(pr);\n const checks = this.getChecksStatus(pr.checks);\n const reviews = this.getReviewStatus(pr.reviews);\n\n let item = `${status} #${pr.number}: ${pr.title}\\n`;\n item += ` {gray-fg}@${pr.author.login} | ${checks} | ${reviews} | +${pr.additions}/-${pr.deletions}{/}`;\n\n if (pr.linkedIssues?.length) {\n item += ` | {cyan-fg}\uD83D\uDD17${pr.linkedIssues.length}{/}`;\n }\n\n return item;\n }\n\n private formatIssueItem(issue: IssueData): string {\n const status = issue.state === 'open' ? '{green-fg}\u25CF{/}' : '{gray-fg}\u25CF{/}';\n const assignees =\n issue.assignees?.map((a: any) => `@${a}`).join(', ') || 'unassigned';\n\n let item = `${status} #${issue.number}: ${issue.title}\\n`;\n item += ` {gray-fg}@${issue.author.login} | ${assignees} | \uD83D\uDCAC${issue.comments}{/}`;\n\n return item;\n }\n\n private getPRStatusIcon(pr: PRData): string {\n if (pr.state === 'merged') return '{magenta-fg}\u2B24{/}';\n if (pr.state === 'closed') return '{red-fg}\u2B24{/}';\n if (pr.draft) return '{gray-fg}\u25CB{/}';\n return '{green-fg}\u25CF{/}';\n }\n\n private getChecksStatus(checks?: PRData['checks']): string {\n if (!checks) return '{gray-fg}no checks{/}';\n if (checks.failed > 0)\n return `{red-fg}\u2717 ${checks.failed}/${checks.total}{/}`;\n if (checks.pending > 0)\n return `{yellow-fg}\u23F3 ${checks.pending}/${checks.total}{/}`;\n return `{green-fg}\u2713 ${checks.passed}/${checks.total}{/}`;\n }\n\n private getReviewStatus(reviews: PRData['reviews']): string {\n const approved = reviews.filter((r: any) => r.state === 'approved').length;\n const changes = reviews.filter(\n (r: any) => r.state === 'changes_requested'\n ).length;\n\n if (changes > 0) return `{red-fg}\uD83D\uDC4E${changes}{/}`;\n if (approved > 0) return `{green-fg}\uD83D\uDC4D${approved}{/}`;\n return '{gray-fg}no reviews{/}';\n }\n\n public update(data: { prs?: PRData[]; issues?: IssueData[] }): void {\n if (data.prs) {\n this.prs.clear();\n data.prs.forEach((pr) => this.prs.set(pr.id, pr));\n }\n\n if (data.issues) {\n this.issues.clear();\n data.issues.forEach((issue) => this.issues.set(issue.id, issue));\n }\n\n this.refreshDisplay();\n }\n\n private refreshDisplay(): void {\n let items: string[] = [];\n let label = '';\n\n if (this.viewMode === 'prs') {\n items = Array.from(this.prs.values()).map((pr: any) =>\n this.formatPRItem(pr)\n );\n const open = Array.from(this.prs.values()).filter(\n (pr: any) => pr.state === 'open'\n ).length;\n label = ` \uD83D\uDD00 Pull Requests (${open}/${this.prs.size}) [Tab] Issues `;\n } else {\n items = Array.from(this.issues.values()).map((issue: any) =>\n this.formatIssueItem(issue)\n );\n const open = Array.from(this.issues.values()).filter(\n (i: any) => i.state === 'open'\n ).length;\n label = ` \uD83D\uDC1B Issues (${open}/${this.issues.size}) [Tab] PRs `;\n }\n\n this.list.setItems(items);\n if (\n this.list.parent &&\n typeof (this.list.parent as any).setLabel === 'function'\n ) {\n (this.list.parent as any).setLabel(label);\n }\n\n this.list.screen.render();\n }\n\n private toggleView(): void {\n this.viewMode = this.viewMode === 'prs' ? 'issues' : 'prs';\n this.refreshDisplay();\n }\n\n private selectPR(prId: string): void {\n const pr = this.prs.get(prId);\n if (pr) {\n this.emit('pr:selected', pr);\n }\n }\n\n private selectIssue(issueId: string): void {\n const issue = this.issues.get(issueId);\n if (issue) {\n this.emit('issue:selected', issue);\n }\n }\n\n public focus(): void {\n this.list.focus();\n }\n\n public hasFocus(): boolean {\n return this.list === this.list.screen.focused;\n }\n}\n"],
|
|
5
|
-
"mappings": "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",
|
|
6
|
-
"names": []
|
|
7
|
-
}
|