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.
- package/LICENSE +21 -21
- package/bin/install.cjs +981 -981
- package/hooks/repo-search-nudge.js +32 -32
- package/package.json +1 -1
- package/skills/estack-active-learning-tutor/SKILL.md +339 -339
- package/skills/estack-better-title/SKILL.md +64 -64
- package/skills/estack-better-title/scripts/rename.sh +55 -55
- package/skills/estack-chris-voss/SKILL.md +80 -80
- package/skills/estack-chris-voss/references/elliot-notes.md +120 -120
- package/skills/estack-chris-voss/references/voss-principles.md +210 -210
- package/skills/estack-customer-discovery/SKILL.md +60 -60
- package/skills/estack-flight-planner/SKILL.md +332 -332
- package/skills/estack-flight-planner/references/config_schema.md +156 -156
- package/skills/estack-flight-planner/references/flight_history_schema.md +97 -97
- package/skills/estack-flight-planner/references/shuttle_schedules.md +98 -98
- package/skills/estack-flight-planner/scripts/check_setup.sh +89 -89
- package/skills/estack-flight-planner/scripts/fetch_flights.py +99 -99
- package/skills/estack-flight-planner/scripts/filter_flights.py +265 -265
- package/skills/estack-flight-planner/scripts/pair_shuttles.py +173 -173
- package/skills/estack-github-issue-tracker/SKILL.md +322 -322
- package/skills/estack-github-issue-tracker/bin/tracker-tools.cjs +1358 -1358
- package/skills/estack-github-issue-tracker/references/gh-cli-patterns.md +124 -124
- package/skills/estack-github-issue-tracker/references/result-file-schema.md +156 -156
- package/skills/estack-github-issue-tracker/references/tracker-schema.md +96 -96
- package/skills/estack-github-issue-tracker/tracker-template.md +58 -58
- package/skills/estack-leadership-coach/SKILL.md +1 -1
- package/skills/estack-leadership-coach/adding-references.md +1 -1
- package/skills/estack-migrate-claude-session-history/SKILL.md +15 -2
- package/skills/estack-pdf-to-md/SKILL.md +1 -2
- package/skills/estack-prompt-builder-coach/SKILL.md +81 -81
- package/skills/estack-prompt-builder-coach/definition-of-done-generator.md +42 -42
- package/skills/estack-prompt-builder-coach/prompt-builder.md +37 -37
- package/skills/estack-prompt-builder-coach/task-shaper.md +36 -36
- package/skills/estack-prompt-builder-coach/vague-ask-auditor.md +37 -37
- package/skills/estack-read-claude-session-history/SKILL.md +228 -204
- package/skills/estack-read-claude-session-history/references/jsonl-schema.md +126 -126
- package/skills/estack-read-claude-session-history/references/modes.md +455 -423
- package/skills/estack-read-claude-session-history/references/recipes.md +300 -271
- package/skills/estack-read-claude-session-history/scripts/lib/__init__.py +1 -1
- package/skills/estack-read-claude-session-history/scripts/lib/parser.py +460 -460
- package/skills/estack-read-claude-session-history/scripts/lib/paths.py +234 -234
- package/skills/estack-read-claude-session-history/scripts/lib/search.py +179 -179
- package/skills/estack-read-claude-session-history/scripts/lib/subagents.py +88 -88
- package/skills/estack-read-claude-session-history/scripts/lib/tools.py +144 -144
- package/skills/estack-read-claude-session-history/scripts/read_transcript.py +1914 -1776
- package/skills/estack-read-claude-session-history/scripts/tests/conftest.py +40 -40
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/README.md +20 -20
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/all-noise.jsonl +4 -4
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/basic-session.jsonl +2 -2
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/engagement-gaps.jsonl +9 -9
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/engagement-noise.jsonl +7 -7
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/engagement-parallel-a.jsonl +3 -3
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/engagement-parallel-b.jsonl +3 -3
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/engagement-waiting.jsonl +5 -5
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/interrupted.jsonl +2 -2
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/multi-compact.jsonl +8 -8
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/pending-user.jsonl +2 -2
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/subagent-no-meta/subagents/agent-aaa.jsonl +2 -2
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/subagent-no-meta.jsonl +2 -2
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/subagent-parent/subagents/agent-xyz123.jsonl +2 -2
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/subagent-parent/subagents/agent-xyz123.meta.json +1 -1
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/subagent-parent.jsonl +4 -4
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/time-spread.jsonl +6 -6
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/timeline-day-test.jsonl +5 -5
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/tool-usage-parent/subagents/agent-sub1.jsonl +3 -0
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/tool-usage-parent.jsonl +3 -0
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/tool-zoo.jsonl +10 -10
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/truncated.jsonl +2 -2
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/unicode.jsonl +2 -2
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/with-advisor.jsonl +3 -3
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/with-compact.jsonl +5 -5
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/with-thinking.jsonl +2 -2
- package/skills/estack-read-claude-session-history/scripts/tests/test_backup_roots.py +56 -56
- package/skills/estack-read-claude-session-history/scripts/tests/test_engagement.py +239 -239
- package/skills/estack-read-claude-session-history/scripts/tests/test_json_format.py +201 -201
- package/skills/estack-read-claude-session-history/scripts/tests/test_modes.py +323 -199
- package/skills/estack-read-claude-session-history/scripts/tests/test_parser.py +195 -195
- package/skills/estack-read-claude-session-history/scripts/tests/test_paths.py +133 -133
- package/skills/estack-read-claude-session-history/scripts/tests/test_search.py +78 -78
- package/skills/estack-read-claude-session-history/scripts/tests/test_subagents.py +43 -43
- package/skills/estack-read-claude-session-history/scripts/tests/test_timeline.py +179 -179
- package/skills/estack-read-claude-session-history/scripts/tests/test_timezone_and_project.py +212 -212
- 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
|
|
114
|
-
r = _run_cli(
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
assert r.
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
"--
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
assert
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
|
|
142
|
-
|
|
143
|
-
|
|
144
|
-
|
|
145
|
-
|
|
146
|
-
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
|
|
158
|
-
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
|
|
163
|
-
|
|
164
|
-
|
|
165
|
-
|
|
166
|
-
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
|
|
170
|
-
|
|
171
|
-
|
|
172
|
-
|
|
173
|
-
|
|
174
|
-
|
|
175
|
-
|
|
176
|
-
|
|
177
|
-
|
|
178
|
-
|
|
179
|
-
|
|
180
|
-
|
|
181
|
-
|
|
182
|
-
|
|
183
|
-
|
|
184
|
-
|
|
185
|
-
|
|
186
|
-
|
|
187
|
-
|
|
188
|
-
|
|
189
|
-
|
|
190
|
-
|
|
191
|
-
|
|
192
|
-
|
|
193
|
-
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
|
|
197
|
-
|
|
198
|
-
|
|
199
|
-
|
|
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()
|