elliot-stack 1.0.18 → 1.0.20
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/README.md +11 -0
- package/bin/install.cjs +134 -49
- package/package.json +1 -1
- package/skills/estack-read-claude-session-history/SKILL.md +203 -0
- package/skills/estack-read-claude-session-history/references/jsonl-schema.md +126 -0
- package/skills/estack-read-claude-session-history/references/modes.md +423 -0
- package/skills/estack-read-claude-session-history/references/recipes.md +271 -0
- package/skills/estack-read-claude-session-history/scripts/lib/__init__.py +1 -0
- package/skills/estack-read-claude-session-history/scripts/lib/parser.py +460 -0
- package/skills/estack-read-claude-session-history/scripts/lib/paths.py +234 -0
- package/skills/estack-read-claude-session-history/scripts/lib/search.py +179 -0
- package/skills/estack-read-claude-session-history/scripts/lib/subagents.py +88 -0
- package/skills/estack-read-claude-session-history/scripts/lib/tools.py +144 -0
- package/skills/estack-read-claude-session-history/scripts/read_transcript.py +1776 -0
- package/skills/estack-read-claude-session-history/scripts/tests/conftest.py +40 -0
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/README.md +20 -0
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/all-noise.jsonl +4 -0
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/basic-session.jsonl +2 -0
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/engagement-gaps.jsonl +9 -0
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/engagement-noise.jsonl +7 -0
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/engagement-parallel-a.jsonl +3 -0
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/engagement-parallel-b.jsonl +3 -0
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/engagement-waiting.jsonl +5 -0
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/interrupted.jsonl +2 -0
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/multi-compact.jsonl +8 -0
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/pending-user.jsonl +2 -0
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/subagent-no-meta/subagents/agent-aaa.jsonl +2 -0
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/subagent-no-meta.jsonl +2 -0
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/subagent-parent/subagents/agent-xyz123.jsonl +2 -0
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/subagent-parent/subagents/agent-xyz123.meta.json +1 -0
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/subagent-parent.jsonl +4 -0
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/time-spread.jsonl +6 -0
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/timeline-day-test.jsonl +5 -0
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/tool-zoo.jsonl +10 -0
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/truncated.jsonl +3 -0
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/unicode.jsonl +2 -0
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/with-advisor.jsonl +3 -0
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/with-compact.jsonl +5 -0
- package/skills/estack-read-claude-session-history/scripts/tests/fixtures/with-thinking.jsonl +2 -0
- package/skills/estack-read-claude-session-history/scripts/tests/test_backup_roots.py +56 -0
- package/skills/estack-read-claude-session-history/scripts/tests/test_engagement.py +239 -0
- package/skills/estack-read-claude-session-history/scripts/tests/test_json_format.py +201 -0
- package/skills/estack-read-claude-session-history/scripts/tests/test_modes.py +199 -0
- package/skills/estack-read-claude-session-history/scripts/tests/test_parser.py +195 -0
- package/skills/estack-read-claude-session-history/scripts/tests/test_paths.py +133 -0
- package/skills/estack-read-claude-session-history/scripts/tests/test_search.py +78 -0
- package/skills/estack-read-claude-session-history/scripts/tests/test_subagents.py +43 -0
- package/skills/estack-read-claude-session-history/scripts/tests/test_timeline.py +179 -0
- package/skills/estack-read-claude-session-history/scripts/tests/test_timezone_and_project.py +212 -0
- package/skills/estack-read-claude-session-history/scripts/tests/test_tools.py +80 -0
|
@@ -0,0 +1,201 @@
|
|
|
1
|
+
"""Tests for --format json across modes."""
|
|
2
|
+
|
|
3
|
+
import json
|
|
4
|
+
import os
|
|
5
|
+
import shutil
|
|
6
|
+
import subprocess
|
|
7
|
+
import sys
|
|
8
|
+
from pathlib import Path
|
|
9
|
+
|
|
10
|
+
import pytest
|
|
11
|
+
|
|
12
|
+
|
|
13
|
+
def _run_cli(cli_path, *args, env_overrides=None):
|
|
14
|
+
env = dict(os.environ)
|
|
15
|
+
env["PYTHONIOENCODING"] = "utf-8"
|
|
16
|
+
if env_overrides:
|
|
17
|
+
env.update(env_overrides)
|
|
18
|
+
return subprocess.run(
|
|
19
|
+
[sys.executable, str(cli_path), *args],
|
|
20
|
+
capture_output=True,
|
|
21
|
+
text=True,
|
|
22
|
+
encoding="utf-8",
|
|
23
|
+
env=env,
|
|
24
|
+
)
|
|
25
|
+
|
|
26
|
+
|
|
27
|
+
def _json(r):
|
|
28
|
+
assert r.returncode == 0, r.stderr
|
|
29
|
+
return json.loads(r.stdout)
|
|
30
|
+
|
|
31
|
+
|
|
32
|
+
def test_last_json(cli_path, fixtures_dir):
|
|
33
|
+
data = _json(_run_cli(
|
|
34
|
+
cli_path, "--file", str(fixtures_dir / "basic-session.jsonl"),
|
|
35
|
+
"--mode", "last", "--format", "json",
|
|
36
|
+
))
|
|
37
|
+
assert isinstance(data, list)
|
|
38
|
+
assert data[-1]["n_from_end"] == 1
|
|
39
|
+
assert "Here is help" in data[-1]["text"]
|
|
40
|
+
|
|
41
|
+
|
|
42
|
+
def test_json_alias_flag(cli_path, fixtures_dir):
|
|
43
|
+
data = _json(_run_cli(
|
|
44
|
+
cli_path, "--file", str(fixtures_dir / "basic-session.jsonl"),
|
|
45
|
+
"--mode", "last", "--json",
|
|
46
|
+
))
|
|
47
|
+
assert isinstance(data, list)
|
|
48
|
+
|
|
49
|
+
|
|
50
|
+
def test_advisor_json(cli_path, fixtures_dir):
|
|
51
|
+
data = _json(_run_cli(
|
|
52
|
+
cli_path, "--file", str(fixtures_dir / "with-advisor.jsonl"),
|
|
53
|
+
"--mode", "advisor", "--format", "json",
|
|
54
|
+
))
|
|
55
|
+
assert data == ["The advisor says do X then Y."]
|
|
56
|
+
|
|
57
|
+
|
|
58
|
+
def test_pre_compact_json(cli_path, fixtures_dir):
|
|
59
|
+
data = _json(_run_cli(
|
|
60
|
+
cli_path, "--file", str(fixtures_dir / "with-compact.jsonl"),
|
|
61
|
+
"--mode", "pre-compact", "--format", "json",
|
|
62
|
+
))
|
|
63
|
+
assert data["found_compact"] is True
|
|
64
|
+
assert any("First answer" in m["text"] for m in data["messages"])
|
|
65
|
+
|
|
66
|
+
|
|
67
|
+
def test_dump_json(cli_path, fixtures_dir):
|
|
68
|
+
data = _json(_run_cli(
|
|
69
|
+
cli_path, "--file", str(fixtures_dir / "basic-session.jsonl"),
|
|
70
|
+
"--mode", "dump", "--format", "json",
|
|
71
|
+
))
|
|
72
|
+
assert isinstance(data, list)
|
|
73
|
+
assert data[0]["role"] == "user"
|
|
74
|
+
|
|
75
|
+
|
|
76
|
+
def test_debug_json(cli_path, fixtures_dir):
|
|
77
|
+
data = _json(_run_cli(
|
|
78
|
+
cli_path, "--file", str(fixtures_dir / "with-advisor.jsonl"),
|
|
79
|
+
"--mode", "debug", "--format", "json",
|
|
80
|
+
))
|
|
81
|
+
assert data["advisor_blocks"] == 1
|
|
82
|
+
assert "entry_types" in data
|
|
83
|
+
|
|
84
|
+
|
|
85
|
+
def test_brief_json(cli_path, fixtures_dir):
|
|
86
|
+
data = _json(_run_cli(
|
|
87
|
+
cli_path, "--file", str(fixtures_dir / "tool-zoo.jsonl"),
|
|
88
|
+
"--mode", "brief", "--format", "json",
|
|
89
|
+
))
|
|
90
|
+
assert data["exists"] is True
|
|
91
|
+
assert data["tool_counts"]["Bash"] == 1
|
|
92
|
+
assert isinstance(data["files_touched"], list)
|
|
93
|
+
|
|
94
|
+
|
|
95
|
+
def test_changelog_json(cli_path, fixtures_dir):
|
|
96
|
+
data = _json(_run_cli(
|
|
97
|
+
cli_path, "--file", str(fixtures_dir / "tool-zoo.jsonl"),
|
|
98
|
+
"--mode", "changelog", "--format", "json",
|
|
99
|
+
))
|
|
100
|
+
assert len(data) == 9
|
|
101
|
+
assert data[0]["tool"] == "Bash"
|
|
102
|
+
assert "input" not in data[0]
|
|
103
|
+
|
|
104
|
+
|
|
105
|
+
def test_tool_calls_json_includes_input(cli_path, fixtures_dir):
|
|
106
|
+
data = _json(_run_cli(
|
|
107
|
+
cli_path, "--file", str(fixtures_dir / "tool-zoo.jsonl"),
|
|
108
|
+
"--mode", "tool-calls", "--format", "json", "--tool", "Bash",
|
|
109
|
+
))
|
|
110
|
+
assert len(data) == 1
|
|
111
|
+
assert data[0]["input"]["command"] == "ls -la /tmp"
|
|
112
|
+
|
|
113
|
+
|
|
114
|
+
def test_file_edits_json(cli_path, fixtures_dir):
|
|
115
|
+
data = _json(_run_cli(
|
|
116
|
+
cli_path, "--file", str(fixtures_dir / "tool-zoo.jsonl"),
|
|
117
|
+
"--mode", "file-edits", "--format", "json",
|
|
118
|
+
))
|
|
119
|
+
paths = [row["path"] for row in data]
|
|
120
|
+
assert any("foo.py" in p for p in paths)
|
|
121
|
+
|
|
122
|
+
|
|
123
|
+
def test_subagent_finals_json(cli_path, fixtures_dir):
|
|
124
|
+
data = _json(_run_cli(
|
|
125
|
+
cli_path, "--file", str(fixtures_dir / "subagent-parent.jsonl"),
|
|
126
|
+
"--mode", "subagent-finals", "--format", "json",
|
|
127
|
+
))
|
|
128
|
+
assert data[0]["agentType"] == "Explore"
|
|
129
|
+
assert "Found it" in data[0]["text"]
|
|
130
|
+
|
|
131
|
+
|
|
132
|
+
def test_search_json_single_file(cli_path, fixtures_dir):
|
|
133
|
+
data = _json(_run_cli(
|
|
134
|
+
cli_path, "--file", str(fixtures_dir / "basic-session.jsonl"),
|
|
135
|
+
"--mode", "search", "--query", "help", "--format", "json",
|
|
136
|
+
))
|
|
137
|
+
assert len(data) == 1
|
|
138
|
+
assert data[0]["role"] == "assistant"
|
|
139
|
+
|
|
140
|
+
|
|
141
|
+
def test_list_json(cli_path, fixtures_dir, tmp_path):
|
|
142
|
+
fake_root = tmp_path / "projects"
|
|
143
|
+
fake_proj = fake_root / "C--fake-proj"
|
|
144
|
+
fake_proj.mkdir(parents=True)
|
|
145
|
+
shutil.copy(fixtures_dir / "basic-session.jsonl", fake_proj / "abc.jsonl")
|
|
146
|
+
data = _json(_run_cli(
|
|
147
|
+
cli_path, "--root", str(fake_root), "--all-projects",
|
|
148
|
+
"--mode", "list", "--format", "json",
|
|
149
|
+
))
|
|
150
|
+
assert len(data) == 1
|
|
151
|
+
assert data[0]["uuid"] == "abc"
|
|
152
|
+
|
|
153
|
+
|
|
154
|
+
def test_journal_json(cli_path, fixtures_dir, tmp_path):
|
|
155
|
+
fake_root = tmp_path / "projects"
|
|
156
|
+
fake_proj = fake_root / "C--fake-proj"
|
|
157
|
+
fake_proj.mkdir(parents=True)
|
|
158
|
+
shutil.copy(fixtures_dir / "tool-zoo.jsonl", fake_proj / "def.jsonl")
|
|
159
|
+
data = _json(_run_cli(
|
|
160
|
+
cli_path, "--root", str(fake_root), "--all-projects",
|
|
161
|
+
"--mode", "journal", "--since", "2020-01-01", "--format", "json",
|
|
162
|
+
))
|
|
163
|
+
assert data[0]["uuid"] == "def"
|
|
164
|
+
assert data[0]["tool_counts"]["Bash"] == 1
|
|
165
|
+
|
|
166
|
+
|
|
167
|
+
def test_count_json(cli_path, fixtures_dir, tmp_path):
|
|
168
|
+
fake_root = tmp_path / "projects"
|
|
169
|
+
fake_proj = fake_root / "C--fake-proj"
|
|
170
|
+
fake_proj.mkdir(parents=True)
|
|
171
|
+
shutil.copy(fixtures_dir / "basic-session.jsonl", fake_proj / "abc.jsonl")
|
|
172
|
+
data = _json(_run_cli(
|
|
173
|
+
cli_path, "--root", str(fake_root), "--all-projects",
|
|
174
|
+
"--mode", "count", "--query", "help", "--format", "json",
|
|
175
|
+
))
|
|
176
|
+
assert data == {"sessions": 1, "messages": 2, "matches": 1}
|
|
177
|
+
|
|
178
|
+
|
|
179
|
+
def test_count_text_unchanged(cli_path, fixtures_dir, tmp_path):
|
|
180
|
+
fake_root = tmp_path / "projects"
|
|
181
|
+
fake_proj = fake_root / "C--fake-proj"
|
|
182
|
+
fake_proj.mkdir(parents=True)
|
|
183
|
+
shutil.copy(fixtures_dir / "basic-session.jsonl", fake_proj / "abc.jsonl")
|
|
184
|
+
r = _run_cli(
|
|
185
|
+
cli_path, "--root", str(fake_root), "--all-projects",
|
|
186
|
+
"--mode", "count", "--query", "help",
|
|
187
|
+
)
|
|
188
|
+
assert r.returncode == 0
|
|
189
|
+
assert r.stdout.strip() == "1"
|
|
190
|
+
assert "1 sessions" in r.stderr
|
|
191
|
+
|
|
192
|
+
|
|
193
|
+
def test_diff_json(cli_path, fixtures_dir):
|
|
194
|
+
data = _json(_run_cli(
|
|
195
|
+
cli_path, "--mode", "diff",
|
|
196
|
+
"--file-a", str(fixtures_dir / "basic-session.jsonl"),
|
|
197
|
+
"--file-b", str(fixtures_dir / "with-thinking.jsonl"),
|
|
198
|
+
"--format", "json",
|
|
199
|
+
))
|
|
200
|
+
sources = {m["source"] for m in data["messages"]}
|
|
201
|
+
assert sources == {"A", "B"}
|
|
@@ -0,0 +1,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_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()
|
|
@@ -0,0 +1,195 @@
|
|
|
1
|
+
"""Tests for lib.parser."""
|
|
2
|
+
|
|
3
|
+
import json
|
|
4
|
+
from datetime import datetime
|
|
5
|
+
from pathlib import Path
|
|
6
|
+
|
|
7
|
+
import pytest
|
|
8
|
+
|
|
9
|
+
from lib import parser as PR
|
|
10
|
+
|
|
11
|
+
|
|
12
|
+
def _load(fixtures_dir, name):
|
|
13
|
+
return PR.parse_lines(fixtures_dir / name)
|
|
14
|
+
|
|
15
|
+
|
|
16
|
+
def test_parse_basic(fixtures_dir):
|
|
17
|
+
lines = _load(fixtures_dir, "basic-session.jsonl")
|
|
18
|
+
assert len(lines) == 2
|
|
19
|
+
assert lines[0]["type"] == "user"
|
|
20
|
+
|
|
21
|
+
|
|
22
|
+
def test_get_messages_basic(fixtures_dir):
|
|
23
|
+
lines = _load(fixtures_dir, "basic-session.jsonl")
|
|
24
|
+
msgs = PR.get_messages(lines)
|
|
25
|
+
assert len(msgs) == 2
|
|
26
|
+
assert msgs[0]["role"] == "user"
|
|
27
|
+
assert msgs[1]["role"] == "assistant"
|
|
28
|
+
|
|
29
|
+
|
|
30
|
+
def test_compact_marker_single(fixtures_dir):
|
|
31
|
+
lines = _load(fixtures_dir, "with-compact.jsonl")
|
|
32
|
+
msgs = PR.get_messages(lines)
|
|
33
|
+
compact = [m for m in msgs if m["is_compact"]]
|
|
34
|
+
assert len(compact) == 1
|
|
35
|
+
|
|
36
|
+
|
|
37
|
+
def test_compact_marker_multiple(fixtures_dir):
|
|
38
|
+
lines = _load(fixtures_dir, "multi-compact.jsonl")
|
|
39
|
+
msgs = PR.get_messages(lines)
|
|
40
|
+
compact = [m for m in msgs if m["is_compact"]]
|
|
41
|
+
assert len(compact) == 2
|
|
42
|
+
|
|
43
|
+
|
|
44
|
+
def test_compact_marker_absent(fixtures_dir):
|
|
45
|
+
lines = _load(fixtures_dir, "basic-session.jsonl")
|
|
46
|
+
msgs = PR.get_messages(lines)
|
|
47
|
+
assert all(not m["is_compact"] for m in msgs)
|
|
48
|
+
|
|
49
|
+
|
|
50
|
+
def test_extract_text_blocks_string():
|
|
51
|
+
assert PR.extract_text_blocks("hello") == ["hello"]
|
|
52
|
+
|
|
53
|
+
|
|
54
|
+
def test_extract_text_blocks_empty_string():
|
|
55
|
+
assert PR.extract_text_blocks("") == []
|
|
56
|
+
assert PR.extract_text_blocks(" ") == []
|
|
57
|
+
|
|
58
|
+
|
|
59
|
+
def test_extract_text_blocks_array():
|
|
60
|
+
blocks = [{"type": "text", "text": "hi"}, {"type": "tool_use", "name": "X"}]
|
|
61
|
+
assert PR.extract_text_blocks(blocks) == ["hi"]
|
|
62
|
+
|
|
63
|
+
|
|
64
|
+
def test_extract_text_blocks_with_thinking():
|
|
65
|
+
blocks = [
|
|
66
|
+
{"type": "thinking", "thinking": "reasoning..."},
|
|
67
|
+
{"type": "text", "text": "answer"},
|
|
68
|
+
]
|
|
69
|
+
out = PR.extract_text_blocks(blocks, include_thinking=True)
|
|
70
|
+
assert any("THINKING" in t for t in out)
|
|
71
|
+
assert "answer" in out
|
|
72
|
+
|
|
73
|
+
|
|
74
|
+
def test_extract_text_blocks_with_tool_use():
|
|
75
|
+
blocks = [{"type": "tool_use", "name": "Bash", "input": {"command": "ls"}}]
|
|
76
|
+
out = PR.extract_text_blocks(blocks, include_tool_use=True)
|
|
77
|
+
assert any("TOOL_USE Bash" in t for t in out)
|
|
78
|
+
|
|
79
|
+
|
|
80
|
+
def test_extract_advisor():
|
|
81
|
+
blocks = [{"type": "advisor_tool_result", "content": {"text": "advice"}}]
|
|
82
|
+
out = PR.extract_text_blocks(blocks)
|
|
83
|
+
assert any("[ADVISOR]" in t for t in out)
|
|
84
|
+
assert any("advice" in t for t in out)
|
|
85
|
+
|
|
86
|
+
|
|
87
|
+
def test_classify_entry_user():
|
|
88
|
+
obj = {"type": "user", "message": {"role": "user", "content": "hi"}}
|
|
89
|
+
assert PR.classify_entry(obj) == "user"
|
|
90
|
+
|
|
91
|
+
|
|
92
|
+
def test_classify_entry_compact():
|
|
93
|
+
obj = {
|
|
94
|
+
"type": "user",
|
|
95
|
+
"message": {"role": "user", "content": PR.COMPACT_MARKER + " more"},
|
|
96
|
+
}
|
|
97
|
+
assert PR.classify_entry(obj) == "compact"
|
|
98
|
+
|
|
99
|
+
|
|
100
|
+
def test_classify_entry_assistant():
|
|
101
|
+
obj = {"type": "assistant", "message": {"role": "assistant", "content": []}}
|
|
102
|
+
assert PR.classify_entry(obj) == "assistant"
|
|
103
|
+
|
|
104
|
+
|
|
105
|
+
def test_classify_entry_noise():
|
|
106
|
+
obj = {"type": "ai-title", "aiTitle": "X"}
|
|
107
|
+
assert PR.classify_entry(obj) == "title"
|
|
108
|
+
|
|
109
|
+
|
|
110
|
+
def test_all_noise_fixture_yields_no_messages(fixtures_dir):
|
|
111
|
+
lines = _load(fixtures_dir, "all-noise.jsonl")
|
|
112
|
+
msgs = PR.get_messages(lines)
|
|
113
|
+
assert msgs == []
|
|
114
|
+
|
|
115
|
+
|
|
116
|
+
def test_filter_by_role(fixtures_dir):
|
|
117
|
+
lines = _load(fixtures_dir, "basic-session.jsonl")
|
|
118
|
+
msgs = PR.get_messages(lines)
|
|
119
|
+
user_only = PR.filter_by_role(msgs, "user")
|
|
120
|
+
assert all(m["role"] == "user" for m in user_only)
|
|
121
|
+
assert len(user_only) == 1
|
|
122
|
+
|
|
123
|
+
|
|
124
|
+
def test_filter_by_time(fixtures_dir):
|
|
125
|
+
lines = _load(fixtures_dir, "time-spread.jsonl")
|
|
126
|
+
msgs = PR.get_messages(lines)
|
|
127
|
+
since = datetime(2026, 5, 1)
|
|
128
|
+
filtered = PR.filter_by_time(msgs, since=since, until=None)
|
|
129
|
+
assert all(
|
|
130
|
+
PR._parse_timestamp(m["timestamp"]).replace(tzinfo=None) >= since
|
|
131
|
+
for m in filtered if m["timestamp"]
|
|
132
|
+
)
|
|
133
|
+
|
|
134
|
+
|
|
135
|
+
def test_truncated_line_dropped(fixtures_dir, capsys):
|
|
136
|
+
# Should not raise, and warning should be on stderr
|
|
137
|
+
lines = _load(fixtures_dir, "truncated.jsonl")
|
|
138
|
+
# 2 valid + 1 truncated = 2 valid records
|
|
139
|
+
assert len(lines) == 2
|
|
140
|
+
|
|
141
|
+
|
|
142
|
+
def test_unicode_roundtrip(fixtures_dir):
|
|
143
|
+
lines = _load(fixtures_dir, "unicode.jsonl")
|
|
144
|
+
msgs = PR.get_messages(lines)
|
|
145
|
+
user_text = msgs[0]["texts"][0]
|
|
146
|
+
assert "🌍" in user_text or "你好" in user_text
|
|
147
|
+
|
|
148
|
+
|
|
149
|
+
def test_infer_status_clean(fixtures_dir):
|
|
150
|
+
lines = _load(fixtures_dir, "basic-session.jsonl")
|
|
151
|
+
mtime = (fixtures_dir / "basic-session.jsonl").stat().st_mtime
|
|
152
|
+
status = PR.infer_status(lines, mtime, current_session_id=None, session_uuid=None)
|
|
153
|
+
assert status == "clean"
|
|
154
|
+
|
|
155
|
+
|
|
156
|
+
def test_infer_status_pending_user(fixtures_dir):
|
|
157
|
+
lines = _load(fixtures_dir, "pending-user.jsonl")
|
|
158
|
+
mtime = (fixtures_dir / "pending-user.jsonl").stat().st_mtime
|
|
159
|
+
status = PR.infer_status(lines, mtime, current_session_id=None, session_uuid=None)
|
|
160
|
+
assert status == "pending-user"
|
|
161
|
+
|
|
162
|
+
|
|
163
|
+
def test_infer_status_interrupted(fixtures_dir):
|
|
164
|
+
lines = _load(fixtures_dir, "interrupted.jsonl")
|
|
165
|
+
mtime = (fixtures_dir / "interrupted.jsonl").stat().st_mtime
|
|
166
|
+
status = PR.infer_status(lines, mtime, current_session_id=None, session_uuid=None)
|
|
167
|
+
assert status == "interrupted"
|
|
168
|
+
|
|
169
|
+
|
|
170
|
+
def test_infer_status_active(fixtures_dir):
|
|
171
|
+
import time
|
|
172
|
+
# Touch the fixture to make it fresh, then set CLAUDE_SESSION_ID to its stem
|
|
173
|
+
f = fixtures_dir / "basic-session.jsonl"
|
|
174
|
+
mtime = time.time()
|
|
175
|
+
lines = _load(fixtures_dir, "basic-session.jsonl")
|
|
176
|
+
status = PR.infer_status(
|
|
177
|
+
lines, mtime, current_session_id="basic-session", session_uuid="basic-session"
|
|
178
|
+
)
|
|
179
|
+
assert status == "active"
|
|
180
|
+
|
|
181
|
+
|
|
182
|
+
def test_session_summary_fields(fixtures_dir):
|
|
183
|
+
s = PR.session_summary(fixtures_dir / "tool-zoo.jsonl")
|
|
184
|
+
assert s["exists"]
|
|
185
|
+
assert s["msg_count"] > 0
|
|
186
|
+
assert "Bash" in s["tool_counts"]
|
|
187
|
+
assert "Read" in s["tool_counts"]
|
|
188
|
+
|
|
189
|
+
|
|
190
|
+
def test_parse_lines_cache(fixtures_dir):
|
|
191
|
+
f = fixtures_dir / "basic-session.jsonl"
|
|
192
|
+
a = PR.parse_lines(f)
|
|
193
|
+
b = PR.parse_lines(f)
|
|
194
|
+
# Same object thanks to cache
|
|
195
|
+
assert a is b
|