tribunal-kit 4.4.0 → 4.4.2

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 (90) hide show
  1. package/.agent/agents/api-architect.md +66 -66
  2. package/.agent/agents/db-latency-auditor.md +216 -216
  3. package/.agent/agents/precedence-reviewer.md +250 -250
  4. package/.agent/agents/resilience-reviewer.md +88 -88
  5. package/.agent/agents/schema-reviewer.md +67 -67
  6. package/.agent/agents/throughput-optimizer.md +299 -299
  7. package/.agent/agents/ui-ux-auditor.md +292 -292
  8. package/.agent/agents/vitals-reviewer.md +223 -223
  9. package/.agent/history/architecture-graph.yaml +32 -1
  10. package/.agent/history/graph-cache.json +66 -19
  11. package/.agent/history/snapshots/bin__tribunal-kit.js.json +19 -0
  12. package/.agent/history/snapshots/eslint.config.js.json +9 -0
  13. package/.agent/history/snapshots/migrate_refs.js.json +3 -3
  14. package/.agent/history/snapshots/scripts__changelog.js.json +2 -1
  15. package/.agent/history/snapshots/scripts__sync-version.js.json +2 -1
  16. package/.agent/history/snapshots/scripts__validate-payload.js.json +1 -0
  17. package/.agent/history/snapshots/test__integration__bridges.test.js.json +2 -1
  18. package/.agent/history/snapshots/test__integration__init.test.js.json +1 -0
  19. package/.agent/history/snapshots/test__integration__routing.test.js.json +1 -0
  20. package/.agent/history/snapshots/test__integration__swarm_dispatcher.test.js.json +2 -1
  21. package/.agent/history/snapshots/test__integration__wave2.test.js.json +2 -1
  22. package/.agent/history/snapshots/test__unit__args.test.js.json +11 -1
  23. package/.agent/history/snapshots/test__unit__case_law_manager.test.js.json +1 -0
  24. package/.agent/history/snapshots/test__unit__context_broker.test.js.json +11 -0
  25. package/.agent/history/snapshots/test__unit__copyDir.test.js.json +11 -1
  26. package/.agent/history/snapshots/test__unit__graph_tools.test.js.json +1 -0
  27. package/.agent/history/snapshots/test__unit__inner_loop_validator.test.js.json +11 -0
  28. package/.agent/history/snapshots/test__unit__selfInstall.test.js.json +11 -1
  29. package/.agent/history/snapshots/test__unit__semver.test.js.json +11 -1
  30. package/.agent/history/snapshots/test__unit__swarm_dispatcher.test.js.json +1 -0
  31. package/.agent/scripts/_colors.js +154 -2
  32. package/.agent/scripts/_utils.js +205 -3
  33. package/.agent/scripts/append_flow.js +72 -72
  34. package/.agent/scripts/auto_preview.js +197 -197
  35. package/.agent/scripts/bundle_analyzer.js +90 -119
  36. package/.agent/scripts/case_law_manager.js +18 -13
  37. package/.agent/scripts/checklist.js +100 -88
  38. package/.agent/scripts/colors.js +7 -13
  39. package/.agent/scripts/compress_skills.js +141 -141
  40. package/.agent/scripts/consolidate_skills.js +149 -149
  41. package/.agent/scripts/context_broker.js +605 -609
  42. package/.agent/scripts/deep_compress.js +150 -150
  43. package/.agent/scripts/dependency_analyzer.js +68 -106
  44. package/.agent/scripts/graph_builder.js +341 -311
  45. package/.agent/scripts/graph_visualizer.js +390 -384
  46. package/.agent/scripts/graph_zoom.js +6 -4
  47. package/.agent/scripts/inner_loop_validator.js +445 -465
  48. package/.agent/scripts/lint_runner.js +27 -28
  49. package/.agent/scripts/minify_context.js +100 -100
  50. package/.agent/scripts/mutation_runner.js +280 -280
  51. package/.agent/scripts/patch_skills_meta.js +156 -156
  52. package/.agent/scripts/patch_skills_output.js +244 -244
  53. package/.agent/scripts/schema_validator.js +280 -297
  54. package/.agent/scripts/security_scan.js +37 -64
  55. package/.agent/scripts/session_manager.js +270 -276
  56. package/.agent/scripts/skill_evolution.js +637 -644
  57. package/.agent/scripts/skill_integrator.js +307 -313
  58. package/.agent/scripts/strengthen_skills.js +193 -193
  59. package/.agent/scripts/strip_tribunal.js +47 -47
  60. package/.agent/scripts/swarm_dispatcher.js +360 -360
  61. package/.agent/scripts/test_runner.js +32 -39
  62. package/.agent/scripts/utils.js +10 -25
  63. package/.agent/scripts/verify_all.js +84 -92
  64. package/.agent/skills/app-builder/templates/astro-static/TEMPLATE.md +1 -1
  65. package/.agent/skills/app-builder/templates/chrome-extension/TEMPLATE.md +1 -1
  66. package/.agent/skills/app-builder/templates/cli-tool/TEMPLATE.md +1 -1
  67. package/.agent/skills/app-builder/templates/electron-desktop/TEMPLATE.md +1 -1
  68. package/.agent/skills/app-builder/templates/express-api/TEMPLATE.md +1 -1
  69. package/.agent/skills/app-builder/templates/flutter-app/TEMPLATE.md +1 -1
  70. package/.agent/skills/app-builder/templates/monorepo-turborepo/TEMPLATE.md +1 -1
  71. package/.agent/skills/app-builder/templates/nextjs-fullstack/TEMPLATE.md +1 -1
  72. package/.agent/skills/app-builder/templates/nextjs-saas/TEMPLATE.md +1 -1
  73. package/.agent/skills/app-builder/templates/nextjs-static/TEMPLATE.md +1 -1
  74. package/.agent/skills/app-builder/templates/nuxt-app/TEMPLATE.md +1 -1
  75. package/.agent/skills/app-builder/templates/python-fastapi/TEMPLATE.md +1 -1
  76. package/.agent/skills/app-builder/templates/react-native-app/TEMPLATE.md +1 -1
  77. package/.agent/skills/doc.md +1 -1
  78. package/.agent/skills/knowledge-graph/SKILL.md +52 -52
  79. package/.agent/skills/ui-ux-pro-max/SKILL.md +562 -562
  80. package/.agent/workflows/generate.md +183 -183
  81. package/.agent/workflows/tribunal-speed.md +183 -183
  82. package/README.md +1 -1
  83. package/bin/tribunal-kit.js +76 -87
  84. package/package.json +6 -3
  85. package/scripts/changelog.js +167 -167
  86. package/scripts/sync-version.js +81 -81
  87. package/.agent/history/architecture-explorer.html +0 -352
  88. package/.agent/scripts/__pycache__/_colors.cpython-311.pyc +0 -0
  89. package/.agent/scripts/__pycache__/_utils.cpython-311.pyc +0 -0
  90. package/.agent/scripts/__pycache__/case_law_manager.cpython-311.pyc +0 -0
@@ -1,276 +1,270 @@
1
- #!/usr/bin/env node
2
- /**
3
- * session_manager.js — Agent session state tracking for multi-conversation work.
4
- *
5
- * Usage:
6
- * node .agent/scripts/session_manager.js save "working on auth"
7
- * node .agent/scripts/session_manager.js load
8
- * node .agent/scripts/session_manager.js show
9
- * node .agent/scripts/session_manager.js clear
10
- * node .agent/scripts/session_manager.js status
11
- * node .agent/scripts/session_manager.js tag <label>
12
- * node .agent/scripts/session_manager.js list [--all]
13
- * node .agent/scripts/session_manager.js export [--stdout]
14
- */
15
-
16
- 'use strict';
17
-
18
- const fs = require('fs');
19
- const path = require('path');
20
-
21
- const STATE_FILE = ".agent_session.json";
22
-
23
- const GREEN = "\x1b[92m";
24
- const YELLOW = "\x1b[93m";
25
- const BLUE = "\x1b[94m";
26
- const CYAN = "\x1b[96m";
27
- const RED = "\x1b[91m";
28
- const BOLD = "\x1b[1m";
29
- const RESET = "\x1b[0m";
30
-
31
- const VALID_COMMANDS = new Set(["save", "load", "show", "clear", "status", "tag", "list", "export"]);
32
- const LIST_PAGE_SIZE = 10;
33
-
34
- function loadState() {
35
- if (!fs.existsSync(STATE_FILE)) return {};
36
- try {
37
- const content = fs.readFileSync(STATE_FILE, 'utf8');
38
- return JSON.parse(content);
39
- } catch {
40
- return {};
41
- }
42
- }
43
-
44
- function saveState(state) {
45
- fs.writeFileSync(STATE_FILE, JSON.stringify(state, null, 2), 'utf8');
46
- }
47
-
48
- function cmdSave(note) {
49
- const state = loadState();
50
- const entry = {
51
- timestamp: new Date().toISOString(),
52
- note: note,
53
- session: (state.history || []).length + 1,
54
- tags: []
55
- };
56
- if (!state.history) state.history = [];
57
- state.history.push(entry);
58
- state.current = entry;
59
- saveState(state);
60
-
61
- console.log(`${GREEN}✅ Session saved:${RESET} ${note}`);
62
- console.log(` Time: ${entry.timestamp}`);
63
- console.log(` Session: #${entry.session}`);
64
- }
65
-
66
- function cmdLoad() {
67
- const state = loadState();
68
- const current = state.current;
69
- if (!current) {
70
- console.log(`${YELLOW}No active session — use 'save' first.${RESET}`);
71
- return;
72
- }
73
- const tagsStr = (current.tags || []).join(", ") || "none";
74
- console.log(`${BOLD}Current session:${RESET}`);
75
- console.log(` Session: #${current.session}`);
76
- console.log(` Time: ${current.timestamp}`);
77
- console.log(` Note: ${current.note}`);
78
- console.log(` Tags: ${tagsStr}`);
79
- }
80
-
81
- function cmdShow() {
82
- const state = loadState();
83
- const history = state.history || [];
84
- if (!history.length) {
85
- console.log(`${YELLOW}No session history.${RESET}`);
86
- return;
87
- }
88
- console.log(`${BOLD}Session History (${history.length} total):${RESET}`);
89
- const recent = history.slice(-10).reverse();
90
- for (const entry of recent) {
91
- const tagsStr = (entry.tags || []).join(", ") || "";
92
- const tagsDisplay = tagsStr ? ` [${tagsStr}]` : "";
93
- console.log(`\n ${BLUE}#${entry.session}${RESET} — ${entry.timestamp.slice(0, 16)}${tagsDisplay}`);
94
- console.log(` ${entry.note}`);
95
- }
96
- }
97
-
98
- function cmdClear() {
99
- if (fs.existsSync(STATE_FILE)) {
100
- fs.unlinkSync(STATE_FILE);
101
- console.log(`${GREEN}✅ Session state cleared.${RESET}`);
102
- } else {
103
- console.log(`${YELLOW}No session file found nothing to clear.${RESET}`);
104
- }
105
- }
106
-
107
- function cmdStatus() {
108
- const state = loadState();
109
- const history = state.history || [];
110
- const current = state.current;
111
-
112
- if (!history.length) {
113
- console.log(`${YELLOW}No session history — use 'save' to start tracking.${RESET}`);
114
- return;
115
- }
116
-
117
- const total = history.length;
118
- const recent = history.slice(-3).reverse();
119
-
120
- console.log(`\n${BOLD}${CYAN}━━━ Session Status ━━━━━━━━━━━━━━━━━━━━━━━━${RESET}`);
121
- console.log(` Total sessions: ${total}`);
122
- if (current) {
123
- console.log(` Active: #${current.session} ${current.note.slice(0, 60)}`);
124
- }
125
- console.log(`\n${BOLD} Last 3 sessions:${RESET}`);
126
- for (const entry of recent) {
127
- const tagsStr = (entry.tags || []).join(", ") || "";
128
- const tagsDisplay = tagsStr ? ` [${tagsStr}]` : "";
129
- const ts = entry.timestamp.slice(0, 16);
130
- console.log(` ${BLUE}#${entry.session}${RESET} ${ts}${tagsDisplay}`);
131
- console.log(` ${entry.note.slice(0, 70)}`);
132
- }
133
- console.log(`${CYAN}━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━${RESET}\n`);
134
- }
135
-
136
- function cmdTag(label) {
137
- if (!label) {
138
- console.log(`${RED}Error: provide a tag label. Example: node session_manager.js tag v2-feature${RESET}`);
139
- process.exit(1);
140
- }
141
- const state = loadState();
142
- const current = state.current;
143
- if (!current) {
144
- console.log(`${YELLOW}No active session — use 'save' first before tagging.${RESET}`);
145
- process.exit(1);
146
- }
147
- if (!current.tags) current.tags = [];
148
- if (current.tags.includes(label)) {
149
- console.log(`${YELLOW}Tag '${label}' already exists on session #${current.session}.${RESET}`);
150
- return;
151
- }
152
-
153
- current.tags.push(label);
154
- state.current = current;
155
-
156
- if (state.history) {
157
- for (const entry of state.history) {
158
- if (entry.session === current.session) {
159
- if (!entry.tags) entry.tags = [];
160
- if (!entry.tags.includes(label)) {
161
- entry.tags.push(label);
162
- }
163
- break;
164
- }
165
- }
166
- }
167
-
168
- saveState(state);
169
- console.log(`${GREEN}✅ Tagged session #${current.session} with '${label}'.${RESET}`);
170
- }
171
-
172
- function cmdList(showAll) {
173
- const state = loadState();
174
- const history = state.history || [];
175
- if (!history.length) {
176
- console.log(`${YELLOW}No session history.${RESET}`);
177
- return;
178
- }
179
-
180
- const total = history.length;
181
- const pageSize = showAll ? total : LIST_PAGE_SIZE;
182
- const recent = history.slice().reverse().slice(0, pageSize);
183
-
184
- console.log(`\n${BOLD}${CYAN}━━━ Session List (${total} total, showing ${recent.length}) ━━━━━━━${RESET}`);
185
-
186
- for (const entry of recent) {
187
- const tagsStr = (entry.tags || []).join(", ");
188
- const tagsDisplay = tagsStr ? ` [${YELLOW}${tagsStr}${RESET}]` : "";
189
- const ts = entry.timestamp.slice(0, 16);
190
- console.log(`\n ${BOLD}${BLUE}#${entry.session}${RESET} ${ts}${tagsDisplay}`);
191
- console.log(` ${entry.note}`);
192
- }
193
-
194
- if (!showAll && total > pageSize) {
195
- const remaining = total - pageSize;
196
- console.log(`\n ${YELLOW}... ${remaining} older session(s) not shown. Use '--all' to see all.${RESET}`);
197
- }
198
- console.log(`${CYAN}━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━${RESET}\n`);
199
- }
200
-
201
- function cmdExport(toStdout) {
202
- const state = loadState();
203
- const history = state.history || [];
204
- if (!history.length) {
205
- console.log(`${YELLOW}No session history to export.${RESET}`);
206
- return;
207
- }
208
-
209
- const lines = ["# Session Export\n"];
210
- lines.push(`Generated: ${new Date().toISOString().slice(0, 16)}\n`);
211
- lines.push(`Total sessions: ${history.length}\n\n---\n`);
212
-
213
- const reversed = history.slice().reverse();
214
- for (const entry of reversed) {
215
- const sessionNum = entry.session || "?";
216
- const ts = (entry.timestamp || "").slice(0, 16);
217
- const note = entry.note || "";
218
- const tags = entry.tags || [];
219
- const tagsStr = tags.length ? `\n**Tags:** ${tags.join(", ")}` : "";
220
-
221
- lines.push(`## Session #${sessionNum} — ${ts}\n`);
222
- lines.push(`${note}${tagsStr}\n\n---\n`);
223
- }
224
-
225
- const content = lines.join("\n");
226
- if (toStdout) {
227
- console.log(content);
228
- } else {
229
- const exportPath = path.resolve("session_export.md");
230
- fs.writeFileSync(exportPath, content, "utf8");
231
- console.log(`${GREEN}✅ Exported ${history.length} sessions to${RESET} ${exportPath}`);
232
- }
233
- }
234
-
235
- function main() {
236
- const args = process.argv.slice(2);
237
- if (!args.length) {
238
- console.log(`Usage: node session_manager.js [save <note>|load|show|clear|status|tag <label>|list [--all]|export [--stdout]]`);
239
- process.exit(1);
240
- }
241
-
242
- const cmd = args[0].toLowerCase();
243
-
244
- if (!VALID_COMMANDS.has(cmd)) {
245
- console.log(`${RED}Unknown command: '${cmd}'${RESET}`);
246
- console.log(`Valid commands: ${[...VALID_COMMANDS].sort().join(', ')}`);
247
- process.exit(1);
248
- }
249
-
250
- if (cmd === "save") {
251
- let note = args.slice(1).join(" ").trim();
252
- if (!note) note = `session ${new Date().toISOString().slice(0,16).replace('T', ' ')}`;
253
- cmdSave(note);
254
- } else if (cmd === "load") {
255
- cmdLoad();
256
- } else if (cmd === "show") {
257
- cmdShow();
258
- } else if (cmd === "clear") {
259
- cmdClear();
260
- } else if (cmd === "status") {
261
- cmdStatus();
262
- } else if (cmd === "tag") {
263
- const label = args.slice(1).join(" ").trim();
264
- cmdTag(label);
265
- } else if (cmd === "list") {
266
- const showAll = args.includes("--all");
267
- cmdList(showAll);
268
- } else if (cmd === "export") {
269
- const toStdout = args.includes("--stdout");
270
- cmdExport(toStdout);
271
- }
272
- }
273
-
274
- if (require.main === module) {
275
- main();
276
- }
1
+ #!/usr/bin/env node
2
+ /**
3
+ * session_manager.js — Agent session state tracking for multi-conversation work.
4
+ *
5
+ * Usage:
6
+ * node .agent/scripts/session_manager.js save "working on auth"
7
+ * node .agent/scripts/session_manager.js load
8
+ * node .agent/scripts/session_manager.js show
9
+ * node .agent/scripts/session_manager.js clear
10
+ * node .agent/scripts/session_manager.js status
11
+ * node .agent/scripts/session_manager.js tag <label>
12
+ * node .agent/scripts/session_manager.js list [--all]
13
+ * node .agent/scripts/session_manager.js export [--stdout]
14
+ */
15
+
16
+ 'use strict';
17
+
18
+ const fs = require('fs');
19
+ const path = require('path');
20
+
21
+ const STATE_FILE = ".agent_session.json";
22
+
23
+ const { GREEN, YELLOW, BLUE, CYAN, RED, BOLD, RESET } = require('./_colors');
24
+
25
+ const VALID_COMMANDS = new Set(["save", "load", "show", "clear", "status", "tag", "list", "export"]);
26
+ const LIST_PAGE_SIZE = 10;
27
+
28
+ function loadState() {
29
+ if (!fs.existsSync(STATE_FILE)) return {};
30
+ try {
31
+ const content = fs.readFileSync(STATE_FILE, 'utf8');
32
+ return JSON.parse(content);
33
+ } catch {
34
+ return {};
35
+ }
36
+ }
37
+
38
+ function saveState(state) {
39
+ fs.writeFileSync(STATE_FILE, JSON.stringify(state, null, 2), 'utf8');
40
+ }
41
+
42
+ function cmdSave(note) {
43
+ const state = loadState();
44
+ const entry = {
45
+ timestamp: new Date().toISOString(),
46
+ note: note,
47
+ session: (state.history || []).length + 1,
48
+ tags: []
49
+ };
50
+ if (!state.history) state.history = [];
51
+ state.history.push(entry);
52
+ state.current = entry;
53
+ saveState(state);
54
+
55
+ console.log(`${GREEN}✅ Session saved:${RESET} ${note}`);
56
+ console.log(` Time: ${entry.timestamp}`);
57
+ console.log(` Session: #${entry.session}`);
58
+ }
59
+
60
+ function cmdLoad() {
61
+ const state = loadState();
62
+ const current = state.current;
63
+ if (!current) {
64
+ console.log(`${YELLOW}No active session — use 'save' first.${RESET}`);
65
+ return;
66
+ }
67
+ const tagsStr = (current.tags || []).join(", ") || "none";
68
+ console.log(`${BOLD}Current session:${RESET}`);
69
+ console.log(` Session: #${current.session}`);
70
+ console.log(` Time: ${current.timestamp}`);
71
+ console.log(` Note: ${current.note}`);
72
+ console.log(` Tags: ${tagsStr}`);
73
+ }
74
+
75
+ function cmdShow() {
76
+ const state = loadState();
77
+ const history = state.history || [];
78
+ if (!history.length) {
79
+ console.log(`${YELLOW}No session history.${RESET}`);
80
+ return;
81
+ }
82
+ console.log(`${BOLD}Session History (${history.length} total):${RESET}`);
83
+ const recent = history.slice(-10).reverse();
84
+ for (const entry of recent) {
85
+ const tagsStr = (entry.tags || []).join(", ") || "";
86
+ const tagsDisplay = tagsStr ? ` [${tagsStr}]` : "";
87
+ console.log(`\n ${BLUE}#${entry.session}${RESET} — ${entry.timestamp.slice(0, 16)}${tagsDisplay}`);
88
+ console.log(` ${entry.note}`);
89
+ }
90
+ }
91
+
92
+ function cmdClear() {
93
+ if (fs.existsSync(STATE_FILE)) {
94
+ fs.unlinkSync(STATE_FILE);
95
+ console.log(`${GREEN}✅ Session state cleared.${RESET}`);
96
+ } else {
97
+ console.log(`${YELLOW}No session file found — nothing to clear.${RESET}`);
98
+ }
99
+ }
100
+
101
+ function cmdStatus() {
102
+ const state = loadState();
103
+ const history = state.history || [];
104
+ const current = state.current;
105
+
106
+ if (!history.length) {
107
+ console.log(`${YELLOW}No session history — use 'save' to start tracking.${RESET}`);
108
+ return;
109
+ }
110
+
111
+ const total = history.length;
112
+ const recent = history.slice(-3).reverse();
113
+
114
+ console.log(`\n${BOLD}${CYAN}━━━ Session Status ━━━━━━━━━━━━━━━━━━━━━━━━${RESET}`);
115
+ console.log(` Total sessions: ${total}`);
116
+ if (current) {
117
+ console.log(` Active: #${current.session} ${current.note.slice(0, 60)}`);
118
+ }
119
+ console.log(`\n${BOLD} Last 3 sessions:${RESET}`);
120
+ for (const entry of recent) {
121
+ const tagsStr = (entry.tags || []).join(", ") || "";
122
+ const tagsDisplay = tagsStr ? ` [${tagsStr}]` : "";
123
+ const ts = entry.timestamp.slice(0, 16);
124
+ console.log(` ${BLUE}#${entry.session}${RESET} ${ts}${tagsDisplay}`);
125
+ console.log(` ${entry.note.slice(0, 70)}`);
126
+ }
127
+ console.log(`${CYAN}━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━${RESET}\n`);
128
+ }
129
+
130
+ function cmdTag(label) {
131
+ if (!label) {
132
+ console.log(`${RED}Error: provide a tag label. Example: node session_manager.js tag v2-feature${RESET}`);
133
+ process.exit(1);
134
+ }
135
+ const state = loadState();
136
+ const current = state.current;
137
+ if (!current) {
138
+ console.log(`${YELLOW}No active session use 'save' first before tagging.${RESET}`);
139
+ process.exit(1);
140
+ }
141
+ if (!current.tags) current.tags = [];
142
+ if (current.tags.includes(label)) {
143
+ console.log(`${YELLOW}Tag '${label}' already exists on session #${current.session}.${RESET}`);
144
+ return;
145
+ }
146
+
147
+ current.tags.push(label);
148
+ state.current = current;
149
+
150
+ if (state.history) {
151
+ for (const entry of state.history) {
152
+ if (entry.session === current.session) {
153
+ if (!entry.tags) entry.tags = [];
154
+ if (!entry.tags.includes(label)) {
155
+ entry.tags.push(label);
156
+ }
157
+ break;
158
+ }
159
+ }
160
+ }
161
+
162
+ saveState(state);
163
+ console.log(`${GREEN}✅ Tagged session #${current.session} with '${label}'.${RESET}`);
164
+ }
165
+
166
+ function cmdList(showAll) {
167
+ const state = loadState();
168
+ const history = state.history || [];
169
+ if (!history.length) {
170
+ console.log(`${YELLOW}No session history.${RESET}`);
171
+ return;
172
+ }
173
+
174
+ const total = history.length;
175
+ const pageSize = showAll ? total : LIST_PAGE_SIZE;
176
+ const recent = history.slice().reverse().slice(0, pageSize);
177
+
178
+ console.log(`\n${BOLD}${CYAN}━━━ Session List (${total} total, showing ${recent.length}) ━━━━━━━${RESET}`);
179
+
180
+ for (const entry of recent) {
181
+ const tagsStr = (entry.tags || []).join(", ");
182
+ const tagsDisplay = tagsStr ? ` [${YELLOW}${tagsStr}${RESET}]` : "";
183
+ const ts = entry.timestamp.slice(0, 16);
184
+ console.log(`\n ${BOLD}${BLUE}#${entry.session}${RESET} ${ts}${tagsDisplay}`);
185
+ console.log(` ${entry.note}`);
186
+ }
187
+
188
+ if (!showAll && total > pageSize) {
189
+ const remaining = total - pageSize;
190
+ console.log(`\n ${YELLOW}... ${remaining} older session(s) not shown. Use '--all' to see all.${RESET}`);
191
+ }
192
+ console.log(`${CYAN}━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━${RESET}\n`);
193
+ }
194
+
195
+ function cmdExport(toStdout) {
196
+ const state = loadState();
197
+ const history = state.history || [];
198
+ if (!history.length) {
199
+ console.log(`${YELLOW}No session history to export.${RESET}`);
200
+ return;
201
+ }
202
+
203
+ const lines = ["# Session Export\n"];
204
+ lines.push(`Generated: ${new Date().toISOString().slice(0, 16)}\n`);
205
+ lines.push(`Total sessions: ${history.length}\n\n---\n`);
206
+
207
+ const reversed = history.slice().reverse();
208
+ for (const entry of reversed) {
209
+ const sessionNum = entry.session || "?";
210
+ const ts = (entry.timestamp || "").slice(0, 16);
211
+ const note = entry.note || "";
212
+ const tags = entry.tags || [];
213
+ const tagsStr = tags.length ? `\n**Tags:** ${tags.join(", ")}` : "";
214
+
215
+ lines.push(`## Session #${sessionNum} ${ts}\n`);
216
+ lines.push(`${note}${tagsStr}\n\n---\n`);
217
+ }
218
+
219
+ const content = lines.join("\n");
220
+ if (toStdout) {
221
+ console.log(content);
222
+ } else {
223
+ const exportPath = path.resolve("session_export.md");
224
+ fs.writeFileSync(exportPath, content, "utf8");
225
+ console.log(`${GREEN}✅ Exported ${history.length} sessions to${RESET} ${exportPath}`);
226
+ }
227
+ }
228
+
229
+ function main() {
230
+ const args = process.argv.slice(2);
231
+ if (!args.length) {
232
+ console.log(`Usage: node session_manager.js [save <note>|load|show|clear|status|tag <label>|list [--all]|export [--stdout]]`);
233
+ process.exit(1);
234
+ }
235
+
236
+ const cmd = args[0].toLowerCase();
237
+
238
+ if (!VALID_COMMANDS.has(cmd)) {
239
+ console.log(`${RED}Unknown command: '${cmd}'${RESET}`);
240
+ console.log(`Valid commands: ${[...VALID_COMMANDS].sort().join(', ')}`);
241
+ process.exit(1);
242
+ }
243
+
244
+ if (cmd === "save") {
245
+ let note = args.slice(1).join(" ").trim();
246
+ if (!note) note = `session ${new Date().toISOString().slice(0,16).replace('T', ' ')}`;
247
+ cmdSave(note);
248
+ } else if (cmd === "load") {
249
+ cmdLoad();
250
+ } else if (cmd === "show") {
251
+ cmdShow();
252
+ } else if (cmd === "clear") {
253
+ cmdClear();
254
+ } else if (cmd === "status") {
255
+ cmdStatus();
256
+ } else if (cmd === "tag") {
257
+ const label = args.slice(1).join(" ").trim();
258
+ cmdTag(label);
259
+ } else if (cmd === "list") {
260
+ const showAll = args.includes("--all");
261
+ cmdList(showAll);
262
+ } else if (cmd === "export") {
263
+ const toStdout = args.includes("--stdout");
264
+ cmdExport(toStdout);
265
+ }
266
+ }
267
+
268
+ if (require.main === module) {
269
+ main();
270
+ }