elliot-stack 1.0.36 → 1.0.38

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 (83) hide show
  1. package/LICENSE +21 -21
  2. package/bin/install.cjs +981 -981
  3. package/hooks/repo-search-nudge.js +32 -32
  4. package/package.json +1 -1
  5. package/skills/estack-active-learning-tutor/SKILL.md +339 -339
  6. package/skills/estack-better-title/SKILL.md +64 -64
  7. package/skills/estack-better-title/scripts/rename.sh +55 -55
  8. package/skills/estack-chris-voss/SKILL.md +80 -80
  9. package/skills/estack-chris-voss/references/elliot-notes.md +120 -120
  10. package/skills/estack-chris-voss/references/voss-principles.md +210 -210
  11. package/skills/estack-customer-discovery/SKILL.md +60 -60
  12. package/skills/estack-flight-planner/SKILL.md +332 -332
  13. package/skills/estack-flight-planner/references/config_schema.md +156 -156
  14. package/skills/estack-flight-planner/references/flight_history_schema.md +97 -97
  15. package/skills/estack-flight-planner/references/shuttle_schedules.md +98 -98
  16. package/skills/estack-flight-planner/scripts/check_setup.sh +89 -89
  17. package/skills/estack-flight-planner/scripts/fetch_flights.py +99 -99
  18. package/skills/estack-flight-planner/scripts/filter_flights.py +265 -265
  19. package/skills/estack-flight-planner/scripts/pair_shuttles.py +173 -173
  20. package/skills/estack-github-issue-tracker/SKILL.md +322 -322
  21. package/skills/estack-github-issue-tracker/bin/tracker-tools.cjs +1358 -1358
  22. package/skills/estack-github-issue-tracker/references/gh-cli-patterns.md +124 -124
  23. package/skills/estack-github-issue-tracker/references/result-file-schema.md +156 -156
  24. package/skills/estack-github-issue-tracker/references/tracker-schema.md +96 -96
  25. package/skills/estack-github-issue-tracker/tracker-template.md +58 -58
  26. package/skills/estack-leadership-coach/SKILL.md +1 -1
  27. package/skills/estack-leadership-coach/adding-references.md +1 -1
  28. package/skills/estack-migrate-claude-session-history/SKILL.md +15 -2
  29. package/skills/estack-pdf-to-md/SKILL.md +1 -2
  30. package/skills/estack-prompt-builder-coach/SKILL.md +81 -81
  31. package/skills/estack-prompt-builder-coach/definition-of-done-generator.md +42 -42
  32. package/skills/estack-prompt-builder-coach/prompt-builder.md +37 -37
  33. package/skills/estack-prompt-builder-coach/task-shaper.md +36 -36
  34. package/skills/estack-prompt-builder-coach/vague-ask-auditor.md +37 -37
  35. package/skills/estack-read-claude-session-history/SKILL.md +228 -204
  36. package/skills/estack-read-claude-session-history/references/jsonl-schema.md +126 -126
  37. package/skills/estack-read-claude-session-history/references/modes.md +455 -423
  38. package/skills/estack-read-claude-session-history/references/recipes.md +300 -271
  39. package/skills/estack-read-claude-session-history/scripts/lib/__init__.py +1 -1
  40. package/skills/estack-read-claude-session-history/scripts/lib/parser.py +460 -460
  41. package/skills/estack-read-claude-session-history/scripts/lib/paths.py +234 -234
  42. package/skills/estack-read-claude-session-history/scripts/lib/search.py +179 -179
  43. package/skills/estack-read-claude-session-history/scripts/lib/subagents.py +88 -88
  44. package/skills/estack-read-claude-session-history/scripts/lib/tools.py +144 -144
  45. package/skills/estack-read-claude-session-history/scripts/read_transcript.py +1914 -1776
  46. package/skills/estack-read-claude-session-history/scripts/tests/conftest.py +40 -40
  47. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/README.md +20 -20
  48. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/all-noise.jsonl +4 -4
  49. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/basic-session.jsonl +2 -2
  50. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/engagement-gaps.jsonl +9 -9
  51. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/engagement-noise.jsonl +7 -7
  52. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/engagement-parallel-a.jsonl +3 -3
  53. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/engagement-parallel-b.jsonl +3 -3
  54. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/engagement-waiting.jsonl +5 -5
  55. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/interrupted.jsonl +2 -2
  56. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/multi-compact.jsonl +8 -8
  57. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/pending-user.jsonl +2 -2
  58. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/subagent-no-meta/subagents/agent-aaa.jsonl +2 -2
  59. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/subagent-no-meta.jsonl +2 -2
  60. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/subagent-parent/subagents/agent-xyz123.jsonl +2 -2
  61. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/subagent-parent/subagents/agent-xyz123.meta.json +1 -1
  62. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/subagent-parent.jsonl +4 -4
  63. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/time-spread.jsonl +6 -6
  64. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/timeline-day-test.jsonl +5 -5
  65. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/tool-usage-parent/subagents/agent-sub1.jsonl +3 -0
  66. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/tool-usage-parent.jsonl +3 -0
  67. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/tool-zoo.jsonl +10 -10
  68. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/truncated.jsonl +2 -2
  69. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/unicode.jsonl +2 -2
  70. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/with-advisor.jsonl +3 -3
  71. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/with-compact.jsonl +5 -5
  72. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/with-thinking.jsonl +2 -2
  73. package/skills/estack-read-claude-session-history/scripts/tests/test_backup_roots.py +56 -56
  74. package/skills/estack-read-claude-session-history/scripts/tests/test_engagement.py +239 -239
  75. package/skills/estack-read-claude-session-history/scripts/tests/test_json_format.py +201 -201
  76. package/skills/estack-read-claude-session-history/scripts/tests/test_modes.py +323 -199
  77. package/skills/estack-read-claude-session-history/scripts/tests/test_parser.py +195 -195
  78. package/skills/estack-read-claude-session-history/scripts/tests/test_paths.py +133 -133
  79. package/skills/estack-read-claude-session-history/scripts/tests/test_search.py +78 -78
  80. package/skills/estack-read-claude-session-history/scripts/tests/test_subagents.py +43 -43
  81. package/skills/estack-read-claude-session-history/scripts/tests/test_timeline.py +179 -179
  82. package/skills/estack-read-claude-session-history/scripts/tests/test_timezone_and_project.py +212 -212
  83. package/skills/estack-read-claude-session-history/scripts/tests/test_tools.py +80 -80
@@ -1,199 +1,323 @@
1
- """End-to-end CLI tests via subprocess.run.
2
-
3
- Exercises mode dispatch + argument parsing. Library-level behavior is
4
- covered by the unit tests in test_paths/parser/tools/search/subagents.
5
- """
6
-
7
- import json
8
- import os
9
- import shutil
10
- import subprocess
11
- import sys
12
- from pathlib import Path
13
-
14
- import pytest
15
-
16
-
17
- def _run_cli(cli_path, *args, env_overrides=None):
18
- env = dict(os.environ)
19
- env["PYTHONIOENCODING"] = "utf-8"
20
- if env_overrides:
21
- env.update(env_overrides)
22
- return subprocess.run(
23
- [sys.executable, str(cli_path), *args],
24
- capture_output=True,
25
- text=True,
26
- encoding="utf-8",
27
- env=env,
28
- )
29
-
30
-
31
- def test_help(cli_path):
32
- r = _run_cli(cli_path, "--help")
33
- assert r.returncode == 0
34
- assert "--mode" in r.stdout
35
-
36
-
37
- def test_last_mode(cli_path, fixtures_dir):
38
- r = _run_cli(cli_path, "--file", str(fixtures_dir / "basic-session.jsonl"), "--mode", "last")
39
- assert r.returncode == 0
40
- assert "Here is help" in r.stdout
41
-
42
-
43
- def test_advisor_mode(cli_path, fixtures_dir):
44
- r = _run_cli(cli_path, "--file", str(fixtures_dir / "with-advisor.jsonl"), "--mode", "advisor")
45
- assert r.returncode == 0
46
- assert "advisor" in r.stdout.lower()
47
-
48
-
49
- def test_pre_compact_mode(cli_path, fixtures_dir):
50
- r = _run_cli(cli_path, "--file", str(fixtures_dir / "with-compact.jsonl"), "--mode", "pre-compact")
51
- assert r.returncode == 0
52
- assert "Pre-compact" in r.stdout or "First answer" in r.stdout
53
-
54
-
55
- def test_debug_mode(cli_path, fixtures_dir):
56
- r = _run_cli(cli_path, "--file", str(fixtures_dir / "basic-session.jsonl"), "--mode", "debug")
57
- assert r.returncode == 0
58
- assert "Entry type distribution" in r.stdout
59
-
60
-
61
- def test_brief_mode(cli_path, fixtures_dir):
62
- r = _run_cli(cli_path, "--file", str(fixtures_dir / "tool-zoo.jsonl"), "--mode", "brief")
63
- assert r.returncode == 0
64
- body = r.stdout
65
- # 6 lines expected
66
- assert "intent:" in body
67
- assert "last:" in body
68
- assert "edits:" in body
69
- assert "tools:" in body
70
- assert "subagents:" in body
71
-
72
-
73
- def test_brief_with_include_subagents(cli_path, fixtures_dir):
74
- r = _run_cli(
75
- cli_path, "--file", str(fixtures_dir / "subagent-parent.jsonl"),
76
- "--mode", "brief", "--include-subagents",
77
- )
78
- assert r.returncode == 0
79
- assert "subagent" in r.stdout.lower()
80
- assert "Found it" in r.stdout
81
-
82
-
83
- def test_changelog(cli_path, fixtures_dir):
84
- r = _run_cli(cli_path, "--file", str(fixtures_dir / "tool-zoo.jsonl"), "--mode", "changelog")
85
- assert r.returncode == 0
86
- assert "Bash" in r.stdout
87
- assert "Read" in r.stdout
88
-
89
-
90
- def test_file_edits(cli_path, fixtures_dir):
91
- r = _run_cli(cli_path, "--file", str(fixtures_dir / "tool-zoo.jsonl"), "--mode", "file-edits")
92
- assert r.returncode == 0
93
- assert "foo.py" in r.stdout
94
- assert "bar.py" in r.stdout
95
-
96
-
97
- def test_tool_calls(cli_path, fixtures_dir):
98
- r = _run_cli(cli_path, "--file", str(fixtures_dir / "tool-zoo.jsonl"), "--mode", "tool-calls")
99
- assert r.returncode == 0
100
- assert "Bash" in r.stdout
101
-
102
-
103
- def test_tool_calls_filter(cli_path, fixtures_dir):
104
- r = _run_cli(
105
- cli_path, "--file", str(fixtures_dir / "tool-zoo.jsonl"),
106
- "--mode", "tool-calls", "--tool", "Bash",
107
- )
108
- assert r.returncode == 0
109
- assert "Bash" in r.stdout
110
- assert "Glob" not in r.stdout
111
-
112
-
113
- def test_subagent_list(cli_path, fixtures_dir):
114
- r = _run_cli(
115
- cli_path, "--file", str(fixtures_dir / "subagent-parent.jsonl"),
116
- "--mode", "subagent-list",
117
- )
118
- assert r.returncode == 0
119
- assert "agent-xyz123" in r.stdout
120
-
121
-
122
- def test_subagent_finals(cli_path, fixtures_dir):
123
- r = _run_cli(
124
- cli_path, "--file", str(fixtures_dir / "subagent-parent.jsonl"),
125
- "--mode", "subagent-finals",
126
- )
127
- assert r.returncode == 0
128
- assert "Found it" in r.stdout
129
-
130
-
131
- def test_diff_mode(cli_path, fixtures_dir):
132
- r = _run_cli(
133
- cli_path, "--mode", "diff",
134
- "--file-a", str(fixtures_dir / "basic-session.jsonl"),
135
- "--file-b", str(fixtures_dir / "with-thinking.jsonl"),
136
- )
137
- assert r.returncode == 0
138
- assert "A>" in r.stdout
139
- assert "B>" in r.stdout
140
-
141
-
142
- def test_lookup_no_match(cli_path, fixtures_dir, tmp_path):
143
- # Point --root at an empty dir so lookup definitely misses
144
- r = _run_cli(cli_path, "--root", str(tmp_path), "--mode", "lookup", "--uuid", "nope")
145
- assert r.returncode == 1
146
-
147
-
148
- def test_list_legacy_format(cli_path, fixtures_dir, tmp_path):
149
- # Build a fake project root + cwd that matches encoding
150
- fake_root = tmp_path / "projects"
151
- fake_proj = fake_root / "C--fake-proj"
152
- fake_proj.mkdir(parents=True)
153
- shutil.copy(fixtures_dir / "basic-session.jsonl", fake_proj / "abc.jsonl")
154
- r = _run_cli(cli_path, "--root", str(fake_root), "--cwd", "C:\\fake\\proj", "--list")
155
- assert r.returncode == 0
156
- assert "abc.jsonl" in r.stdout
157
-
158
-
159
- def test_exclude_current(cli_path, fixtures_dir, tmp_path):
160
- fake_root = tmp_path / "projects"
161
- fake_proj = fake_root / "C--fake-proj"
162
- fake_proj.mkdir(parents=True)
163
- shutil.copy(fixtures_dir / "basic-session.jsonl", fake_proj / "abc.jsonl")
164
- shutil.copy(fixtures_dir / "tool-zoo.jsonl", fake_proj / "def.jsonl")
165
- r = _run_cli(
166
- cli_path, "--root", str(fake_root), "--cwd", "C:\\fake\\proj",
167
- "--mode", "list", "--exclude-current",
168
- env_overrides={"CLAUDE_SESSION_ID": "abc"},
169
- )
170
- assert r.returncode == 0
171
- assert "def" in r.stdout
172
- assert "abc" not in r.stdout
173
-
174
-
175
- def test_dump_large_file_degrades(cli_path, fixtures_dir, tmp_path):
176
- # Build a 6MB padded fixture by writing many valid lines
177
- big = tmp_path / "big.jsonl"
178
- line = '{"type":"user","timestamp":"2026-05-01T10:00:00Z","message":{"role":"user","content":"'
179
- pad = "x" * 1000 + '"}}\n'
180
- with open(big, "w", encoding="utf-8") as f:
181
- # Each line ~1KB → write ~7000 to hit 6MB+
182
- for _ in range(7000):
183
- f.write(line + pad)
184
- r = _run_cli(cli_path, "--file", str(big), "--mode", "dump")
185
- assert r.returncode == 0
186
- assert "degraded" in r.stderr.lower()
187
-
188
-
189
- def test_dump_large_file_force(cli_path, tmp_path):
190
- big = tmp_path / "big.jsonl"
191
- line = '{"type":"user","timestamp":"2026-05-01T10:00:00Z","message":{"role":"user","content":"'
192
- pad = "x" * 1000 + '"}}\n'
193
- with open(big, "w", encoding="utf-8") as f:
194
- for _ in range(7000):
195
- f.write(line + pad)
196
- r = _run_cli(cli_path, "--file", str(big), "--mode", "dump", "--force-dump")
197
- assert r.returncode == 0
198
- # No degrade note when forced
199
- assert "degraded" not in r.stderr.lower()
1
+ """End-to-end CLI tests via subprocess.run.
2
+
3
+ Exercises mode dispatch + argument parsing. Library-level behavior is
4
+ covered by the unit tests in test_paths/parser/tools/search/subagents.
5
+ """
6
+
7
+ import json
8
+ import os
9
+ import shutil
10
+ import subprocess
11
+ import sys
12
+ from pathlib import Path
13
+
14
+ import pytest
15
+
16
+
17
+ def _run_cli(cli_path, *args, env_overrides=None):
18
+ env = dict(os.environ)
19
+ env["PYTHONIOENCODING"] = "utf-8"
20
+ if env_overrides:
21
+ env.update(env_overrides)
22
+ return subprocess.run(
23
+ [sys.executable, str(cli_path), *args],
24
+ capture_output=True,
25
+ text=True,
26
+ encoding="utf-8",
27
+ env=env,
28
+ )
29
+
30
+
31
+ def test_help(cli_path):
32
+ r = _run_cli(cli_path, "--help")
33
+ assert r.returncode == 0
34
+ assert "--mode" in r.stdout
35
+
36
+
37
+ def test_last_mode(cli_path, fixtures_dir):
38
+ r = _run_cli(cli_path, "--file", str(fixtures_dir / "basic-session.jsonl"), "--mode", "last")
39
+ assert r.returncode == 0
40
+ assert "Here is help" in r.stdout
41
+
42
+
43
+ def test_advisor_mode(cli_path, fixtures_dir):
44
+ r = _run_cli(cli_path, "--file", str(fixtures_dir / "with-advisor.jsonl"), "--mode", "advisor")
45
+ assert r.returncode == 0
46
+ assert "advisor" in r.stdout.lower()
47
+
48
+
49
+ def test_pre_compact_mode(cli_path, fixtures_dir):
50
+ r = _run_cli(cli_path, "--file", str(fixtures_dir / "with-compact.jsonl"), "--mode", "pre-compact")
51
+ assert r.returncode == 0
52
+ assert "Pre-compact" in r.stdout or "First answer" in r.stdout
53
+
54
+
55
+ def test_debug_mode(cli_path, fixtures_dir):
56
+ r = _run_cli(cli_path, "--file", str(fixtures_dir / "basic-session.jsonl"), "--mode", "debug")
57
+ assert r.returncode == 0
58
+ assert "Entry type distribution" in r.stdout
59
+
60
+
61
+ def test_brief_mode(cli_path, fixtures_dir):
62
+ r = _run_cli(cli_path, "--file", str(fixtures_dir / "tool-zoo.jsonl"), "--mode", "brief")
63
+ assert r.returncode == 0
64
+ body = r.stdout
65
+ # 6 lines expected
66
+ assert "intent:" in body
67
+ assert "last:" in body
68
+ assert "edits:" in body
69
+ assert "tools:" in body
70
+ assert "subagents:" in body
71
+
72
+
73
+ def test_brief_with_include_subagents(cli_path, fixtures_dir):
74
+ r = _run_cli(
75
+ cli_path, "--file", str(fixtures_dir / "subagent-parent.jsonl"),
76
+ "--mode", "brief", "--include-subagents",
77
+ )
78
+ assert r.returncode == 0
79
+ assert "subagent" in r.stdout.lower()
80
+ assert "Found it" in r.stdout
81
+
82
+
83
+ def test_changelog(cli_path, fixtures_dir):
84
+ r = _run_cli(cli_path, "--file", str(fixtures_dir / "tool-zoo.jsonl"), "--mode", "changelog")
85
+ assert r.returncode == 0
86
+ assert "Bash" in r.stdout
87
+ assert "Read" in r.stdout
88
+
89
+
90
+ def test_file_edits(cli_path, fixtures_dir):
91
+ r = _run_cli(cli_path, "--file", str(fixtures_dir / "tool-zoo.jsonl"), "--mode", "file-edits")
92
+ assert r.returncode == 0
93
+ assert "foo.py" in r.stdout
94
+ assert "bar.py" in r.stdout
95
+
96
+
97
+ def test_tool_calls(cli_path, fixtures_dir):
98
+ r = _run_cli(cli_path, "--file", str(fixtures_dir / "tool-zoo.jsonl"), "--mode", "tool-calls")
99
+ assert r.returncode == 0
100
+ assert "Bash" in r.stdout
101
+
102
+
103
+ def test_tool_calls_filter(cli_path, fixtures_dir):
104
+ r = _run_cli(
105
+ cli_path, "--file", str(fixtures_dir / "tool-zoo.jsonl"),
106
+ "--mode", "tool-calls", "--tool", "Bash",
107
+ )
108
+ assert r.returncode == 0
109
+ assert "Bash" in r.stdout
110
+ assert "Glob" not in r.stdout
111
+
112
+
113
+ def test_tool_usage_file(cli_path, fixtures_dir):
114
+ r = _run_cli(cli_path, "--file", str(fixtures_dir / "tool-zoo.jsonl"), "--mode", "tool-usage")
115
+ assert r.returncode == 0
116
+ assert "Tool calls" in r.stdout
117
+ assert "Bash" in r.stdout
118
+ assert "Skill" in r.stdout
119
+ # Skill calls are sub-tallied by the actual skill name (input.skill).
120
+ assert "using-superpowers" in r.stdout
121
+
122
+
123
+ def test_tool_usage_skill_filter(cli_path, fixtures_dir):
124
+ r = _run_cli(
125
+ cli_path, "--file", str(fixtures_dir / "tool-zoo.jsonl"),
126
+ "--mode", "tool-usage", "--tool", "Skill",
127
+ )
128
+ assert r.returncode == 0
129
+ assert "Skill" in r.stdout
130
+ assert "using-superpowers" in r.stdout
131
+ # Filtering to Skill must drop every other tool.
132
+ assert "Bash" not in r.stdout
133
+ assert "Glob" not in r.stdout
134
+
135
+
136
+ def test_tool_usage_json(cli_path, fixtures_dir):
137
+ r = _run_cli(
138
+ cli_path, "--file", str(fixtures_dir / "tool-zoo.jsonl"),
139
+ "--mode", "tool-usage", "--format", "json",
140
+ )
141
+ assert r.returncode == 0
142
+ data = json.loads(r.stdout)
143
+ assert data["total"] == 9
144
+ assert data["sessions"] == 1
145
+ tools = {t["tool"]: t["count"] for t in data["tools"]}
146
+ assert tools["Skill"] == 1
147
+ assert tools["Bash"] == 1
148
+ skills = {s["skill"]: s["count"] for s in data["skills"]}
149
+ assert skills == {"using-superpowers": 1}
150
+
151
+
152
+ def test_tool_usage_scope_aggregates(cli_path, fixtures_dir, tmp_path):
153
+ # Two sessions in one project → counts add up across files.
154
+ fake_root = tmp_path / "projects"
155
+ fake_proj = fake_root / "C--fake-proj"
156
+ fake_proj.mkdir(parents=True)
157
+ shutil.copy(fixtures_dir / "tool-zoo.jsonl", fake_proj / "a.jsonl")
158
+ shutil.copy(fixtures_dir / "tool-zoo.jsonl", fake_proj / "b.jsonl")
159
+ r = _run_cli(
160
+ cli_path, "--root", str(fake_root), "--cwd", "C:\\fake\\proj",
161
+ "--mode", "tool-usage", "--format", "json",
162
+ )
163
+ assert r.returncode == 0
164
+ data = json.loads(r.stdout)
165
+ assert data["sessions"] == 2
166
+ assert data["total"] == 18
167
+ skills = {s["skill"]: s["count"] for s in data["skills"]}
168
+ assert skills == {"using-superpowers": 2}
169
+
170
+
171
+ def test_tool_usage_missing_file(cli_path, tmp_path):
172
+ r = _run_cli(cli_path, "--file", str(tmp_path / "nope.jsonl"), "--mode", "tool-usage")
173
+ assert r.returncode == 1
174
+
175
+
176
+ def test_tool_usage_until_keeps_in_window_calls(cli_path, fixtures_dir, tmp_path):
177
+ # The tool-zoo calls are stamped 2026-05-01T10:00:0x. Copy the fixture, then
178
+ # bump its mtime far past --until. A naive mtime filter would drop the file
179
+ # and report zero; per-call timestamp filtering must still count the calls.
180
+ import os
181
+ fake_root = tmp_path / "projects"
182
+ fake_proj = fake_root / "C--fake-proj"
183
+ fake_proj.mkdir(parents=True)
184
+ target = fake_proj / "a.jsonl"
185
+ shutil.copy(fixtures_dir / "tool-zoo.jsonl", target)
186
+ future = 1_900_000_000 # ~2030, well after the --until bound below
187
+ os.utime(target, (future, future))
188
+ r = _run_cli(
189
+ cli_path, "--root", str(fake_root), "--cwd", "C:\\fake\\proj",
190
+ "--mode", "tool-usage", "--until", "2026-05-02", "--format", "json",
191
+ )
192
+ assert r.returncode == 0
193
+ data = json.loads(r.stdout)
194
+ assert data["total"] == 9, data # all 9 calls are inside the window
195
+ assert data["sessions"] == 1
196
+
197
+
198
+ def test_tool_usage_exclude_current_drops_session(cli_path, fixtures_dir, tmp_path):
199
+ fake_root = tmp_path / "projects"
200
+ fake_proj = fake_root / "C--fake-proj"
201
+ fake_proj.mkdir(parents=True)
202
+ shutil.copy(fixtures_dir / "tool-zoo.jsonl", fake_proj / "cur.jsonl")
203
+ shutil.copy(fixtures_dir / "tool-zoo.jsonl", fake_proj / "other.jsonl")
204
+ # Without exclusion: 2 sessions, 18 calls. Excluding "cur" leaves 1 session, 9.
205
+ r = _run_cli(
206
+ cli_path, "--root", str(fake_root), "--cwd", "C:\\fake\\proj",
207
+ "--mode", "tool-usage", "--exclude-current", "--format", "json",
208
+ env_overrides={"CLAUDE_SESSION_ID": "cur"},
209
+ )
210
+ assert r.returncode == 0
211
+ data = json.loads(r.stdout)
212
+ assert data["sessions"] == 1
213
+ assert data["total"] == 9
214
+
215
+
216
+ def test_tool_usage_include_subagents(cli_path, fixtures_dir):
217
+ # Parent calls: Skill(commit) + Agent. Subagent calls: Skill(estack-repo-search) + Bash.
218
+ # --include-subagents must fold the subagent's calls in, without counting the
219
+ # subagent as its own session.
220
+ parent = fixtures_dir / "tool-usage-parent.jsonl"
221
+ without = json.loads(_run_cli(
222
+ cli_path, "--file", str(parent), "--mode", "tool-usage", "--format", "json",
223
+ ).stdout)
224
+ assert without["total"] == 2
225
+ assert without["sessions"] == 1
226
+ assert {s["skill"] for s in without["skills"]} == {"commit"}
227
+
228
+ with_sub = json.loads(_run_cli(
229
+ cli_path, "--file", str(parent), "--mode", "tool-usage",
230
+ "--include-subagents", "--format", "json",
231
+ ).stdout)
232
+ assert with_sub["total"] == 4 # +Skill(estack-repo-search) +Bash
233
+ assert with_sub["sessions"] == 1 # subagent is not a separate session
234
+ assert {s["skill"] for s in with_sub["skills"]} == {"commit", "estack-repo-search"}
235
+
236
+
237
+ def test_subagent_list(cli_path, fixtures_dir):
238
+ r = _run_cli(
239
+ cli_path, "--file", str(fixtures_dir / "subagent-parent.jsonl"),
240
+ "--mode", "subagent-list",
241
+ )
242
+ assert r.returncode == 0
243
+ assert "agent-xyz123" in r.stdout
244
+
245
+
246
+ def test_subagent_finals(cli_path, fixtures_dir):
247
+ r = _run_cli(
248
+ cli_path, "--file", str(fixtures_dir / "subagent-parent.jsonl"),
249
+ "--mode", "subagent-finals",
250
+ )
251
+ assert r.returncode == 0
252
+ assert "Found it" in r.stdout
253
+
254
+
255
+ def test_diff_mode(cli_path, fixtures_dir):
256
+ r = _run_cli(
257
+ cli_path, "--mode", "diff",
258
+ "--file-a", str(fixtures_dir / "basic-session.jsonl"),
259
+ "--file-b", str(fixtures_dir / "with-thinking.jsonl"),
260
+ )
261
+ assert r.returncode == 0
262
+ assert "A>" in r.stdout
263
+ assert "B>" in r.stdout
264
+
265
+
266
+ def test_lookup_no_match(cli_path, fixtures_dir, tmp_path):
267
+ # Point --root at an empty dir so lookup definitely misses
268
+ r = _run_cli(cli_path, "--root", str(tmp_path), "--mode", "lookup", "--uuid", "nope")
269
+ assert r.returncode == 1
270
+
271
+
272
+ def test_list_legacy_format(cli_path, fixtures_dir, tmp_path):
273
+ # Build a fake project root + cwd that matches encoding
274
+ fake_root = tmp_path / "projects"
275
+ fake_proj = fake_root / "C--fake-proj"
276
+ fake_proj.mkdir(parents=True)
277
+ shutil.copy(fixtures_dir / "basic-session.jsonl", fake_proj / "abc.jsonl")
278
+ r = _run_cli(cli_path, "--root", str(fake_root), "--cwd", "C:\\fake\\proj", "--list")
279
+ assert r.returncode == 0
280
+ assert "abc.jsonl" in r.stdout
281
+
282
+
283
+ def test_exclude_current(cli_path, fixtures_dir, tmp_path):
284
+ fake_root = tmp_path / "projects"
285
+ fake_proj = fake_root / "C--fake-proj"
286
+ fake_proj.mkdir(parents=True)
287
+ shutil.copy(fixtures_dir / "basic-session.jsonl", fake_proj / "abc.jsonl")
288
+ shutil.copy(fixtures_dir / "tool-zoo.jsonl", fake_proj / "def.jsonl")
289
+ r = _run_cli(
290
+ cli_path, "--root", str(fake_root), "--cwd", "C:\\fake\\proj",
291
+ "--mode", "list", "--exclude-current",
292
+ env_overrides={"CLAUDE_SESSION_ID": "abc"},
293
+ )
294
+ assert r.returncode == 0
295
+ assert "def" in r.stdout
296
+ assert "abc" not in r.stdout
297
+
298
+
299
+ def test_dump_large_file_degrades(cli_path, fixtures_dir, tmp_path):
300
+ # Build a 6MB padded fixture by writing many valid lines
301
+ big = tmp_path / "big.jsonl"
302
+ line = '{"type":"user","timestamp":"2026-05-01T10:00:00Z","message":{"role":"user","content":"'
303
+ pad = "x" * 1000 + '"}}\n'
304
+ with open(big, "w", encoding="utf-8") as f:
305
+ # Each line ~1KB → write ~7000 to hit 6MB+
306
+ for _ in range(7000):
307
+ f.write(line + pad)
308
+ r = _run_cli(cli_path, "--file", str(big), "--mode", "dump")
309
+ assert r.returncode == 0
310
+ assert "degraded" in r.stderr.lower()
311
+
312
+
313
+ def test_dump_large_file_force(cli_path, tmp_path):
314
+ big = tmp_path / "big.jsonl"
315
+ line = '{"type":"user","timestamp":"2026-05-01T10:00:00Z","message":{"role":"user","content":"'
316
+ pad = "x" * 1000 + '"}}\n'
317
+ with open(big, "w", encoding="utf-8") as f:
318
+ for _ in range(7000):
319
+ f.write(line + pad)
320
+ r = _run_cli(cli_path, "--file", str(big), "--mode", "dump", "--force-dump")
321
+ assert r.returncode == 0
322
+ # No degrade note when forced
323
+ assert "degraded" not in r.stderr.lower()