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,133 @@
|
|
|
1
|
+
"""Tests for lib.paths."""
|
|
2
|
+
|
|
3
|
+
import os
|
|
4
|
+
from datetime import datetime, timedelta
|
|
5
|
+
from pathlib import Path
|
|
6
|
+
|
|
7
|
+
import pytest
|
|
8
|
+
|
|
9
|
+
from lib import paths as P
|
|
10
|
+
|
|
11
|
+
|
|
12
|
+
def test_encode_cwd_basic():
|
|
13
|
+
assert P.encode_cwd("C:\\Users\\foo\\bar") == "C--Users-foo-bar"
|
|
14
|
+
|
|
15
|
+
|
|
16
|
+
def test_encode_cwd_spaces():
|
|
17
|
+
assert P.encode_cwd("C:\\Users\\2supe\\Other Claude Code") == "C--Users-2supe-Other-Claude-Code"
|
|
18
|
+
|
|
19
|
+
|
|
20
|
+
def test_encode_cwd_mixed_slashes():
|
|
21
|
+
assert P.encode_cwd("C:/Users/foo bar") == "C--Users-foo-bar"
|
|
22
|
+
|
|
23
|
+
|
|
24
|
+
def test_decode_project_name_strips_prefix():
|
|
25
|
+
enc = "C--Users-elliot-Other-Claude-Code-Personal-Brand-Project"
|
|
26
|
+
decoded = P.decode_project_name(enc)
|
|
27
|
+
assert "Other" in decoded
|
|
28
|
+
assert "Personal" in decoded
|
|
29
|
+
assert "C--Users" not in decoded
|
|
30
|
+
|
|
31
|
+
|
|
32
|
+
def test_decode_project_name_fallback_when_unparseable():
|
|
33
|
+
decoded = P.decode_project_name("totally-weird-thing")
|
|
34
|
+
assert decoded # Non-empty, falls back gracefully
|
|
35
|
+
|
|
36
|
+
|
|
37
|
+
def test_resolve_root_live():
|
|
38
|
+
assert P.resolve_root("live").name == "projects"
|
|
39
|
+
|
|
40
|
+
|
|
41
|
+
def test_resolve_root_mirror():
|
|
42
|
+
r = P.resolve_root("mirror")
|
|
43
|
+
assert "mirror" in str(r)
|
|
44
|
+
assert r.name == "projects"
|
|
45
|
+
|
|
46
|
+
|
|
47
|
+
def test_resolve_root_unknown_relative_raises():
|
|
48
|
+
with pytest.raises(ValueError):
|
|
49
|
+
P.resolve_root("not-a-known-root")
|
|
50
|
+
|
|
51
|
+
|
|
52
|
+
def test_resolve_root_absolute_passes_through(tmp_path: Path):
|
|
53
|
+
r = P.resolve_root(str(tmp_path))
|
|
54
|
+
assert r == tmp_path
|
|
55
|
+
|
|
56
|
+
|
|
57
|
+
def test_current_session_id_unset(monkeypatch):
|
|
58
|
+
monkeypatch.delenv("CLAUDE_SESSION_ID", raising=False)
|
|
59
|
+
assert P.current_session_id() is None
|
|
60
|
+
|
|
61
|
+
|
|
62
|
+
def test_current_session_id_set(monkeypatch):
|
|
63
|
+
monkeypatch.setenv("CLAUDE_SESSION_ID", "abc-123")
|
|
64
|
+
assert P.current_session_id() == "abc-123"
|
|
65
|
+
|
|
66
|
+
|
|
67
|
+
def test_parse_timespec_relative():
|
|
68
|
+
now = datetime.now()
|
|
69
|
+
delta = now - P.parse_timespec("1h")
|
|
70
|
+
assert timedelta(seconds=3590) <= delta <= timedelta(seconds=3610)
|
|
71
|
+
|
|
72
|
+
|
|
73
|
+
def test_parse_timespec_iso_date():
|
|
74
|
+
assert P.parse_timespec("2026-05-01") == datetime(2026, 5, 1)
|
|
75
|
+
|
|
76
|
+
|
|
77
|
+
def test_parse_timespec_yesterday():
|
|
78
|
+
y = P.parse_timespec("yesterday")
|
|
79
|
+
assert y.hour == 0
|
|
80
|
+
assert y.minute == 0
|
|
81
|
+
|
|
82
|
+
|
|
83
|
+
def test_parse_timespec_invalid():
|
|
84
|
+
with pytest.raises(ValueError):
|
|
85
|
+
P.parse_timespec("not-a-time")
|
|
86
|
+
|
|
87
|
+
|
|
88
|
+
def test_list_transcripts_empty(tmp_path: Path):
|
|
89
|
+
assert P.list_transcripts(tmp_path) == []
|
|
90
|
+
|
|
91
|
+
|
|
92
|
+
def test_list_transcripts_excludes_agent_prefix(tmp_path: Path):
|
|
93
|
+
(tmp_path / "real.jsonl").write_text("{}\n")
|
|
94
|
+
(tmp_path / "agent-foo.jsonl").write_text("{}\n")
|
|
95
|
+
files = P.list_transcripts(tmp_path)
|
|
96
|
+
names = [f.name for f in files]
|
|
97
|
+
assert "real.jsonl" in names
|
|
98
|
+
assert "agent-foo.jsonl" not in names
|
|
99
|
+
|
|
100
|
+
|
|
101
|
+
def test_list_transcripts_time_filter(tmp_path: Path):
|
|
102
|
+
old = tmp_path / "old.jsonl"
|
|
103
|
+
new = tmp_path / "new.jsonl"
|
|
104
|
+
old.write_text("{}\n")
|
|
105
|
+
new.write_text("{}\n")
|
|
106
|
+
# Backdate old
|
|
107
|
+
past = datetime.now() - timedelta(days=10)
|
|
108
|
+
os.utime(old, (past.timestamp(), past.timestamp()))
|
|
109
|
+
|
|
110
|
+
since = datetime.now() - timedelta(days=5)
|
|
111
|
+
files = P.list_transcripts(tmp_path, since=since)
|
|
112
|
+
assert [f.name for f in files] == ["new.jsonl"]
|
|
113
|
+
|
|
114
|
+
|
|
115
|
+
def test_list_projects_empty(tmp_path: Path):
|
|
116
|
+
assert P.list_projects(tmp_path) == []
|
|
117
|
+
|
|
118
|
+
|
|
119
|
+
def test_list_subagents_returns_empty_when_no_dir(tmp_path: Path):
|
|
120
|
+
f = tmp_path / "session.jsonl"
|
|
121
|
+
f.write_text("{}\n")
|
|
122
|
+
assert P.list_subagents(f) == []
|
|
123
|
+
|
|
124
|
+
|
|
125
|
+
def test_list_subagents_finds_agent_files(tmp_path: Path):
|
|
126
|
+
f = tmp_path / "session.jsonl"
|
|
127
|
+
f.write_text("{}\n")
|
|
128
|
+
sub_dir = tmp_path / "session" / "subagents"
|
|
129
|
+
sub_dir.mkdir(parents=True)
|
|
130
|
+
(sub_dir / "agent-x.jsonl").write_text("{}\n")
|
|
131
|
+
(sub_dir / "agent-y.jsonl").write_text("{}\n")
|
|
132
|
+
subs = P.list_subagents(f)
|
|
133
|
+
assert len(subs) == 2
|
|
@@ -0,0 +1,78 @@
|
|
|
1
|
+
"""Tests for lib.search."""
|
|
2
|
+
|
|
3
|
+
from datetime import datetime
|
|
4
|
+
from pathlib import Path
|
|
5
|
+
|
|
6
|
+
from lib import search as S
|
|
7
|
+
|
|
8
|
+
|
|
9
|
+
def test_search_session_text(fixtures_dir):
|
|
10
|
+
matches = S.search_session(
|
|
11
|
+
fixtures_dir / "basic-session.jsonl", "Hello", role="both", in_channel="text"
|
|
12
|
+
)
|
|
13
|
+
assert len(matches) >= 1
|
|
14
|
+
|
|
15
|
+
|
|
16
|
+
def test_search_session_assistant_only(fixtures_dir):
|
|
17
|
+
matches = S.search_session(
|
|
18
|
+
fixtures_dir / "basic-session.jsonl", "Hello",
|
|
19
|
+
role="assistant", in_channel="text",
|
|
20
|
+
)
|
|
21
|
+
# "Hello" is in the user message only
|
|
22
|
+
assert len(matches) == 0
|
|
23
|
+
|
|
24
|
+
|
|
25
|
+
def test_search_session_user_only(fixtures_dir):
|
|
26
|
+
matches = S.search_session(
|
|
27
|
+
fixtures_dir / "basic-session.jsonl", "Hello",
|
|
28
|
+
role="user", in_channel="text",
|
|
29
|
+
)
|
|
30
|
+
assert len(matches) >= 1
|
|
31
|
+
|
|
32
|
+
|
|
33
|
+
def test_search_in_tool_use(fixtures_dir):
|
|
34
|
+
matches = S.search_session(
|
|
35
|
+
fixtures_dir / "tool-zoo.jsonl", "ls -la",
|
|
36
|
+
role="both", in_channel="tool_use",
|
|
37
|
+
)
|
|
38
|
+
assert len(matches) >= 1
|
|
39
|
+
assert matches[0].where == "tool_use"
|
|
40
|
+
|
|
41
|
+
|
|
42
|
+
def test_search_in_thinking(fixtures_dir):
|
|
43
|
+
matches = S.search_session(
|
|
44
|
+
fixtures_dir / "with-thinking.jsonl", "step by step",
|
|
45
|
+
role="both", in_channel="thinking",
|
|
46
|
+
)
|
|
47
|
+
assert len(matches) >= 1
|
|
48
|
+
assert matches[0].where == "thinking"
|
|
49
|
+
|
|
50
|
+
|
|
51
|
+
def test_search_no_match(fixtures_dir):
|
|
52
|
+
matches = S.search_session(
|
|
53
|
+
fixtures_dir / "basic-session.jsonl", "this-string-not-present",
|
|
54
|
+
role="both", in_channel="text",
|
|
55
|
+
)
|
|
56
|
+
assert matches == []
|
|
57
|
+
|
|
58
|
+
|
|
59
|
+
def test_search_with_time_filter(fixtures_dir):
|
|
60
|
+
# "Hello" appears in basic-session.jsonl at 2026-05-01T10:00:00Z
|
|
61
|
+
# Excluding that date should yield no matches
|
|
62
|
+
matches = S.search_session(
|
|
63
|
+
fixtures_dir / "basic-session.jsonl", "Hello",
|
|
64
|
+
role="both", in_channel="text",
|
|
65
|
+
since=datetime(2026, 6, 1),
|
|
66
|
+
)
|
|
67
|
+
assert matches == []
|
|
68
|
+
|
|
69
|
+
|
|
70
|
+
def test_search_project(fixtures_dir, tmp_path):
|
|
71
|
+
# Copy 2 fixtures into a fake project dir and search
|
|
72
|
+
import shutil
|
|
73
|
+
pd = tmp_path / "fake-proj"
|
|
74
|
+
pd.mkdir()
|
|
75
|
+
shutil.copy(fixtures_dir / "basic-session.jsonl", pd / "session-a.jsonl")
|
|
76
|
+
shutil.copy(fixtures_dir / "tool-zoo.jsonl", pd / "session-b.jsonl")
|
|
77
|
+
matches = list(S.search_project(pd, "Hello", progress=False))
|
|
78
|
+
assert len(matches) >= 1
|
|
@@ -0,0 +1,43 @@
|
|
|
1
|
+
"""Tests for lib.subagents."""
|
|
2
|
+
|
|
3
|
+
from pathlib import Path
|
|
4
|
+
|
|
5
|
+
from lib import subagents as SA
|
|
6
|
+
from lib import paths as P
|
|
7
|
+
|
|
8
|
+
|
|
9
|
+
def test_load_meta_hit(fixtures_dir):
|
|
10
|
+
agent_file = fixtures_dir / "subagent-parent" / "subagents" / "agent-xyz123.jsonl"
|
|
11
|
+
meta = SA.load_meta(agent_file)
|
|
12
|
+
assert meta["agentType"] == "Explore"
|
|
13
|
+
assert "bug" in meta["description"].lower()
|
|
14
|
+
|
|
15
|
+
|
|
16
|
+
def test_load_meta_miss(fixtures_dir):
|
|
17
|
+
agent_file = fixtures_dir / "subagent-no-meta" / "subagents" / "agent-aaa.jsonl"
|
|
18
|
+
meta = SA.load_meta(agent_file)
|
|
19
|
+
assert meta["agentType"] == "unknown"
|
|
20
|
+
assert meta["description"] == ""
|
|
21
|
+
|
|
22
|
+
|
|
23
|
+
def test_agent_finals(fixtures_dir):
|
|
24
|
+
parent = fixtures_dir / "subagent-parent.jsonl"
|
|
25
|
+
finals = SA.agent_finals(parent)
|
|
26
|
+
assert len(finals) == 1
|
|
27
|
+
agent_id, meta, text = finals[0]
|
|
28
|
+
assert agent_id == "agent-xyz123"
|
|
29
|
+
assert meta["agentType"] == "Explore"
|
|
30
|
+
assert "Found it" in text
|
|
31
|
+
|
|
32
|
+
|
|
33
|
+
def test_agent_finals_no_subagents(fixtures_dir):
|
|
34
|
+
parent = fixtures_dir / "basic-session.jsonl"
|
|
35
|
+
finals = SA.agent_finals(parent)
|
|
36
|
+
assert finals == []
|
|
37
|
+
|
|
38
|
+
|
|
39
|
+
def test_list_subagents(fixtures_dir):
|
|
40
|
+
parent = fixtures_dir / "subagent-parent.jsonl"
|
|
41
|
+
subs = P.list_subagents(parent)
|
|
42
|
+
assert len(subs) == 1
|
|
43
|
+
assert subs[0].stem == "agent-xyz123"
|
|
@@ -0,0 +1,179 @@
|
|
|
1
|
+
"""Tests for the timeline mode (build/render/gap parsing + CLI)."""
|
|
2
|
+
|
|
3
|
+
import json
|
|
4
|
+
import os
|
|
5
|
+
import shutil
|
|
6
|
+
import subprocess
|
|
7
|
+
import sys
|
|
8
|
+
from datetime import datetime, timedelta
|
|
9
|
+
from pathlib import Path
|
|
10
|
+
|
|
11
|
+
import pytest
|
|
12
|
+
|
|
13
|
+
import read_transcript as RT
|
|
14
|
+
|
|
15
|
+
|
|
16
|
+
def _run_cli(cli_path, *args, env_overrides=None):
|
|
17
|
+
env = dict(os.environ)
|
|
18
|
+
env["PYTHONIOENCODING"] = "utf-8"
|
|
19
|
+
if env_overrides:
|
|
20
|
+
env.update(env_overrides)
|
|
21
|
+
return subprocess.run(
|
|
22
|
+
[sys.executable, str(cli_path), *args],
|
|
23
|
+
capture_output=True,
|
|
24
|
+
text=True,
|
|
25
|
+
encoding="utf-8",
|
|
26
|
+
env=env,
|
|
27
|
+
)
|
|
28
|
+
|
|
29
|
+
|
|
30
|
+
@pytest.fixture
|
|
31
|
+
def fake_root(fixtures_dir, tmp_path):
|
|
32
|
+
root = tmp_path / "projects"
|
|
33
|
+
proj = root / "C--fake-proj"
|
|
34
|
+
proj.mkdir(parents=True)
|
|
35
|
+
shutil.copy(fixtures_dir / "timeline-day-test.jsonl", proj / "abc12345.jsonl")
|
|
36
|
+
return root
|
|
37
|
+
|
|
38
|
+
|
|
39
|
+
# ── unit: gap + duration helpers ─────────────────────────────────────────────
|
|
40
|
+
|
|
41
|
+
def test_parse_gap_default():
|
|
42
|
+
assert RT._parse_gap(None) == 15
|
|
43
|
+
|
|
44
|
+
|
|
45
|
+
def test_parse_gap_minutes():
|
|
46
|
+
assert RT._parse_gap("20m") == 20
|
|
47
|
+
assert RT._parse_gap("20") == 20
|
|
48
|
+
|
|
49
|
+
|
|
50
|
+
def test_parse_gap_hours():
|
|
51
|
+
assert RT._parse_gap("1h") == 60
|
|
52
|
+
|
|
53
|
+
|
|
54
|
+
def test_parse_gap_invalid():
|
|
55
|
+
with pytest.raises(ValueError):
|
|
56
|
+
RT._parse_gap("soon")
|
|
57
|
+
|
|
58
|
+
|
|
59
|
+
def test_fmt_dur():
|
|
60
|
+
assert RT._fmt_dur(timedelta(minutes=8)) == "8m"
|
|
61
|
+
assert RT._fmt_dur(timedelta(minutes=72)) == "1h12m"
|
|
62
|
+
assert RT._fmt_dur(timedelta(seconds=30)) == "<1m"
|
|
63
|
+
|
|
64
|
+
|
|
65
|
+
# ── unit: block grouping ─────────────────────────────────────────────────────
|
|
66
|
+
|
|
67
|
+
def test_build_timeline_blocks(fake_root):
|
|
68
|
+
from lib import parser as PR
|
|
69
|
+
PR.set_timezone("UTC")
|
|
70
|
+
try:
|
|
71
|
+
data = RT.build_timeline(
|
|
72
|
+
[fake_root / "C--fake-proj"],
|
|
73
|
+
since=datetime(2026, 5, 1),
|
|
74
|
+
until=datetime(2026, 5, 2),
|
|
75
|
+
gap_minutes=15,
|
|
76
|
+
current_uuid=None,
|
|
77
|
+
)
|
|
78
|
+
finally:
|
|
79
|
+
PR.set_timezone(None)
|
|
80
|
+
blocks = data["blocks"]
|
|
81
|
+
assert len(blocks) == 2
|
|
82
|
+
assert blocks[0]["start"] == datetime(2026, 5, 1, 10, 0)
|
|
83
|
+
assert blocks[0]["end"] == datetime(2026, 5, 1, 10, 8)
|
|
84
|
+
assert blocks[1]["start"] == datetime(2026, 5, 1, 12, 0)
|
|
85
|
+
assert blocks[1]["end"] == datetime(2026, 5, 1, 12, 2)
|
|
86
|
+
|
|
87
|
+
|
|
88
|
+
def test_build_timeline_wide_gap_merges(fake_root):
|
|
89
|
+
from lib import parser as PR
|
|
90
|
+
PR.set_timezone("UTC")
|
|
91
|
+
try:
|
|
92
|
+
data = RT.build_timeline(
|
|
93
|
+
[fake_root / "C--fake-proj"],
|
|
94
|
+
since=datetime(2026, 5, 1),
|
|
95
|
+
until=datetime(2026, 5, 2),
|
|
96
|
+
gap_minutes=180, # 3h gap threshold swallows the 1h52m idle
|
|
97
|
+
current_uuid=None,
|
|
98
|
+
)
|
|
99
|
+
finally:
|
|
100
|
+
PR.set_timezone(None)
|
|
101
|
+
assert len(data["blocks"]) == 1
|
|
102
|
+
|
|
103
|
+
|
|
104
|
+
# ── CLI ──────────────────────────────────────────────────────────────────────
|
|
105
|
+
|
|
106
|
+
def test_timeline_cli_text(cli_path, fake_root):
|
|
107
|
+
r = _run_cli(
|
|
108
|
+
cli_path, "--root", str(fake_root), "--tz", "UTC",
|
|
109
|
+
"--mode", "timeline", "--date", "2026-05-01",
|
|
110
|
+
)
|
|
111
|
+
assert r.returncode == 0
|
|
112
|
+
assert "10:00" in r.stdout
|
|
113
|
+
assert "12:02" in r.stdout
|
|
114
|
+
assert "idle" in r.stdout
|
|
115
|
+
assert "2 block(s)" in r.stdout
|
|
116
|
+
# Timeline makes no attention claim — that's engagement mode's job.
|
|
117
|
+
assert "active" not in r.stdout
|
|
118
|
+
|
|
119
|
+
|
|
120
|
+
def test_timeline_cli_json(cli_path, fake_root):
|
|
121
|
+
r = _run_cli(
|
|
122
|
+
cli_path, "--root", str(fake_root), "--tz", "UTC",
|
|
123
|
+
"--mode", "timeline", "--date", "2026-05-01", "--format", "json",
|
|
124
|
+
)
|
|
125
|
+
assert r.returncode == 0
|
|
126
|
+
data = json.loads(r.stdout)
|
|
127
|
+
assert data["totals"]["blocks"] == 2
|
|
128
|
+
assert data["totals"]["sessions"] == 1
|
|
129
|
+
assert data["totals"]["span_minutes"] == 122 # 10:00 → 12:02
|
|
130
|
+
assert "active_minutes" not in data["totals"]
|
|
131
|
+
assert data["blocks"][0]["start"].endswith("10:00:00")
|
|
132
|
+
assert data["blocks"][0]["sessions"][0]["uuid"] == "abc12345"
|
|
133
|
+
|
|
134
|
+
|
|
135
|
+
def test_timeline_cli_empty_range(cli_path, fake_root):
|
|
136
|
+
r = _run_cli(
|
|
137
|
+
cli_path, "--root", str(fake_root), "--tz", "UTC",
|
|
138
|
+
"--mode", "timeline", "--date", "2020-01-01",
|
|
139
|
+
)
|
|
140
|
+
assert r.returncode == 0
|
|
141
|
+
assert "no activity" in r.stdout
|
|
142
|
+
|
|
143
|
+
|
|
144
|
+
def test_timeline_cli_project_filter(cli_path, fake_root):
|
|
145
|
+
r = _run_cli(
|
|
146
|
+
cli_path, "--root", str(fake_root), "--tz", "UTC",
|
|
147
|
+
"--mode", "timeline", "--date", "2026-05-01", "--project", "fake",
|
|
148
|
+
)
|
|
149
|
+
assert r.returncode == 0
|
|
150
|
+
assert "10:00" in r.stdout
|
|
151
|
+
|
|
152
|
+
|
|
153
|
+
def test_timeline_cli_exclude_current(cli_path, fake_root):
|
|
154
|
+
r = _run_cli(
|
|
155
|
+
cli_path, "--root", str(fake_root), "--tz", "UTC",
|
|
156
|
+
"--mode", "timeline", "--date", "2026-05-01", "--exclude-current",
|
|
157
|
+
env_overrides={"CLAUDE_SESSION_ID": "abc12345"},
|
|
158
|
+
)
|
|
159
|
+
assert r.returncode == 0
|
|
160
|
+
assert "no activity" in r.stdout
|
|
161
|
+
|
|
162
|
+
|
|
163
|
+
def test_journal_cli_exclude_current(cli_path, fake_root):
|
|
164
|
+
r = _run_cli(
|
|
165
|
+
cli_path, "--root", str(fake_root),
|
|
166
|
+
"--mode", "journal", "--since", "2020-01-01", "--all-projects",
|
|
167
|
+
"--exclude-current",
|
|
168
|
+
env_overrides={"CLAUDE_SESSION_ID": "abc12345"},
|
|
169
|
+
)
|
|
170
|
+
assert r.returncode == 0
|
|
171
|
+
assert "abc12345" not in r.stdout
|
|
172
|
+
|
|
173
|
+
|
|
174
|
+
def test_timeline_cli_project_no_match(cli_path, fake_root):
|
|
175
|
+
r = _run_cli(
|
|
176
|
+
cli_path, "--root", str(fake_root), "--tz", "UTC",
|
|
177
|
+
"--mode", "timeline", "--date", "2026-05-01", "--project", "zzz-nope",
|
|
178
|
+
)
|
|
179
|
+
assert r.returncode == 1
|
|
@@ -0,0 +1,212 @@
|
|
|
1
|
+
"""Tests for timezone handling (set_timezone / --tz) and the --project filter."""
|
|
2
|
+
|
|
3
|
+
import json
|
|
4
|
+
import os
|
|
5
|
+
import shutil
|
|
6
|
+
import subprocess
|
|
7
|
+
import sys
|
|
8
|
+
from datetime import datetime
|
|
9
|
+
from pathlib import Path
|
|
10
|
+
|
|
11
|
+
import pytest
|
|
12
|
+
|
|
13
|
+
from lib import parser as PR
|
|
14
|
+
from lib import paths as P
|
|
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
|
+
@pytest.fixture(autouse=True)
|
|
32
|
+
def _reset_tz():
|
|
33
|
+
yield
|
|
34
|
+
PR.set_timezone(None)
|
|
35
|
+
|
|
36
|
+
|
|
37
|
+
# ── set_timezone / _parse_timestamp ─────────────────────────────────────────
|
|
38
|
+
|
|
39
|
+
def test_parse_timestamp_default_is_naive():
|
|
40
|
+
dt = PR._parse_timestamp("2026-05-01T10:00:00Z")
|
|
41
|
+
assert dt is not None
|
|
42
|
+
assert dt.tzinfo is None
|
|
43
|
+
|
|
44
|
+
|
|
45
|
+
def test_parse_timestamp_utc():
|
|
46
|
+
PR.set_timezone("UTC")
|
|
47
|
+
assert PR._parse_timestamp("2026-05-01T10:00:00Z") == datetime(2026, 5, 1, 10, 0)
|
|
48
|
+
|
|
49
|
+
|
|
50
|
+
def test_parse_timestamp_offset():
|
|
51
|
+
PR.set_timezone("+2")
|
|
52
|
+
assert PR._parse_timestamp("2026-05-01T10:00:00Z") == datetime(2026, 5, 1, 12, 0)
|
|
53
|
+
|
|
54
|
+
|
|
55
|
+
def test_parse_timestamp_utc_minus():
|
|
56
|
+
PR.set_timezone("UTC-4")
|
|
57
|
+
assert PR._parse_timestamp("2026-05-01T10:00:00Z") == datetime(2026, 5, 1, 6, 0)
|
|
58
|
+
|
|
59
|
+
|
|
60
|
+
def test_set_timezone_half_hour_offset():
|
|
61
|
+
PR.set_timezone("+05:30")
|
|
62
|
+
assert PR._parse_timestamp("2026-05-01T10:00:00Z") == datetime(2026, 5, 1, 15, 30)
|
|
63
|
+
|
|
64
|
+
|
|
65
|
+
def test_set_timezone_local_resets():
|
|
66
|
+
PR.set_timezone("UTC")
|
|
67
|
+
PR.set_timezone("local")
|
|
68
|
+
assert PR._TARGET_TZ is None
|
|
69
|
+
|
|
70
|
+
|
|
71
|
+
def test_set_timezone_iana():
|
|
72
|
+
try:
|
|
73
|
+
PR.set_timezone("America/New_York")
|
|
74
|
+
except ValueError:
|
|
75
|
+
pytest.skip("IANA tz database not available on this machine")
|
|
76
|
+
dt = PR._parse_timestamp("2026-01-15T10:00:00Z") # EST = UTC-5
|
|
77
|
+
assert dt == datetime(2026, 1, 15, 5, 0)
|
|
78
|
+
|
|
79
|
+
|
|
80
|
+
def test_set_timezone_invalid_raises():
|
|
81
|
+
with pytest.raises(ValueError):
|
|
82
|
+
PR.set_timezone("Mars/Olympus_Mons")
|
|
83
|
+
|
|
84
|
+
|
|
85
|
+
def test_epoch_to_display_utc():
|
|
86
|
+
PR.set_timezone("UTC")
|
|
87
|
+
# 2026-05-01T10:00:00Z as epoch
|
|
88
|
+
epoch = datetime(2026, 5, 1, 10, 0).replace(
|
|
89
|
+
tzinfo=__import__("datetime").timezone.utc
|
|
90
|
+
).timestamp()
|
|
91
|
+
assert PR.epoch_to_display(epoch) == datetime(2026, 5, 1, 10, 0)
|
|
92
|
+
|
|
93
|
+
|
|
94
|
+
def test_cli_tz_invalid(cli_path, fixtures_dir):
|
|
95
|
+
r = _run_cli(
|
|
96
|
+
cli_path, "--file", str(fixtures_dir / "basic-session.jsonl"),
|
|
97
|
+
"--mode", "changelog", "--tz", "Nope/Nowhere",
|
|
98
|
+
)
|
|
99
|
+
assert r.returncode == 1
|
|
100
|
+
assert "timezone" in r.stderr.lower()
|
|
101
|
+
|
|
102
|
+
|
|
103
|
+
def test_display_to_epoch_roundtrip_offset_tz():
|
|
104
|
+
PR.set_timezone("+2")
|
|
105
|
+
epoch = 1772546400.0
|
|
106
|
+
assert PR.display_to_epoch(PR.epoch_to_display(epoch)) == epoch
|
|
107
|
+
|
|
108
|
+
|
|
109
|
+
def test_display_to_epoch_roundtrip_local():
|
|
110
|
+
epoch = 1772546400.0
|
|
111
|
+
assert PR.display_to_epoch(PR.epoch_to_display(epoch)) == epoch
|
|
112
|
+
|
|
113
|
+
|
|
114
|
+
def test_now_display_matches_utc_clock():
|
|
115
|
+
from datetime import timezone as _tzmod
|
|
116
|
+
PR.set_timezone("UTC")
|
|
117
|
+
delta = abs((PR.now_display() - datetime.now(_tzmod.utc).replace(tzinfo=None)).total_seconds())
|
|
118
|
+
assert delta < 5
|
|
119
|
+
|
|
120
|
+
|
|
121
|
+
def test_parse_timespec_now_respects_tz():
|
|
122
|
+
from datetime import timezone as _tzmod
|
|
123
|
+
PR.set_timezone("UTC")
|
|
124
|
+
now_spec = P.parse_timespec("now")
|
|
125
|
+
delta = abs((now_spec - datetime.now(_tzmod.utc).replace(tzinfo=None)).total_seconds())
|
|
126
|
+
assert delta < 5
|
|
127
|
+
|
|
128
|
+
|
|
129
|
+
def test_json_timestamps_are_display_tz(cli_path, fixtures_dir):
|
|
130
|
+
# basic-session events are 10:00:0xZ — with +2 the JSON timestamps read 12:00
|
|
131
|
+
r = _run_cli(
|
|
132
|
+
cli_path, "--file", str(fixtures_dir / "basic-session.jsonl"),
|
|
133
|
+
"--mode", "last", "--format", "json", "--tz", "+2",
|
|
134
|
+
)
|
|
135
|
+
assert r.returncode == 0
|
|
136
|
+
data = json.loads(r.stdout)
|
|
137
|
+
assert data[-1]["timestamp"].startswith("2026-05-01T12:00:05")
|
|
138
|
+
|
|
139
|
+
|
|
140
|
+
def test_cli_changelog_respects_tz(cli_path, fixtures_dir):
|
|
141
|
+
# tool-zoo events are 10:00:0xZ — with +3 they display as 13:00
|
|
142
|
+
r = _run_cli(
|
|
143
|
+
cli_path, "--file", str(fixtures_dir / "tool-zoo.jsonl"),
|
|
144
|
+
"--mode", "changelog", "--tz", "+3",
|
|
145
|
+
)
|
|
146
|
+
assert r.returncode == 0
|
|
147
|
+
assert "13:00:01" in r.stdout
|
|
148
|
+
|
|
149
|
+
|
|
150
|
+
# ── --project filter ─────────────────────────────────────────────────────────
|
|
151
|
+
|
|
152
|
+
@pytest.fixture
|
|
153
|
+
def multi_root(fixtures_dir, tmp_path):
|
|
154
|
+
root = tmp_path / "projects"
|
|
155
|
+
keel = root / "C--Users-x-Keel-Project"
|
|
156
|
+
other = root / "C--Users-x-Other-Claude-Code"
|
|
157
|
+
keel.mkdir(parents=True)
|
|
158
|
+
other.mkdir(parents=True)
|
|
159
|
+
shutil.copy(fixtures_dir / "basic-session.jsonl", keel / "keelsession.jsonl")
|
|
160
|
+
shutil.copy(fixtures_dir / "tool-zoo.jsonl", other / "othersession.jsonl")
|
|
161
|
+
return root
|
|
162
|
+
|
|
163
|
+
|
|
164
|
+
def test_filter_projects_substring(multi_root):
|
|
165
|
+
dirs = P.filter_projects(multi_root, "keel")
|
|
166
|
+
assert len(dirs) == 1
|
|
167
|
+
assert "Keel" in dirs[0].name
|
|
168
|
+
|
|
169
|
+
|
|
170
|
+
def test_filter_projects_spaces_match_hyphens(multi_root):
|
|
171
|
+
dirs = P.filter_projects(multi_root, "Keel Project")
|
|
172
|
+
assert len(dirs) == 1
|
|
173
|
+
|
|
174
|
+
|
|
175
|
+
def test_filter_projects_no_match(multi_root):
|
|
176
|
+
assert P.filter_projects(multi_root, "zzz") == []
|
|
177
|
+
|
|
178
|
+
|
|
179
|
+
def test_cli_list_project_filter(cli_path, multi_root):
|
|
180
|
+
r = _run_cli(
|
|
181
|
+
cli_path, "--root", str(multi_root), "--mode", "list", "--project", "keel",
|
|
182
|
+
)
|
|
183
|
+
assert r.returncode == 0
|
|
184
|
+
assert "keelsess" in r.stdout
|
|
185
|
+
assert "othersess" not in r.stdout
|
|
186
|
+
|
|
187
|
+
|
|
188
|
+
def test_cli_search_project_filter(cli_path, multi_root):
|
|
189
|
+
r = _run_cli(
|
|
190
|
+
cli_path, "--root", str(multi_root), "--mode", "search",
|
|
191
|
+
"--query", "help", "--project", "keel",
|
|
192
|
+
)
|
|
193
|
+
assert r.returncode == 0
|
|
194
|
+
assert "keelsession" in r.stdout
|
|
195
|
+
|
|
196
|
+
|
|
197
|
+
def test_cli_journal_project_filter_json(cli_path, multi_root):
|
|
198
|
+
r = _run_cli(
|
|
199
|
+
cli_path, "--root", str(multi_root), "--mode", "journal",
|
|
200
|
+
"--since", "2020-01-01", "--project", "other claude", "--format", "json",
|
|
201
|
+
)
|
|
202
|
+
assert r.returncode == 0
|
|
203
|
+
data = json.loads(r.stdout)
|
|
204
|
+
assert len(data) == 1
|
|
205
|
+
assert data[0]["uuid"] == "othersession"
|
|
206
|
+
|
|
207
|
+
|
|
208
|
+
def test_cli_project_no_match_exits_1(cli_path, multi_root):
|
|
209
|
+
r = _run_cli(
|
|
210
|
+
cli_path, "--root", str(multi_root), "--mode", "list", "--project", "zzz",
|
|
211
|
+
)
|
|
212
|
+
assert r.returncode == 1
|