elliot-stack 1.0.30 → 1.0.36

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 (127) hide show
  1. package/LICENSE +21 -21
  2. package/README.md +4 -0
  3. package/bin/install.cjs +981 -950
  4. package/hooks/repo-search-nudge.js +32 -32
  5. package/package.json +1 -1
  6. package/skills/estack-active-learning-tutor/SKILL.md +339 -339
  7. package/skills/estack-better-title/SKILL.md +64 -64
  8. package/skills/estack-better-title/scripts/rename.sh +55 -55
  9. package/skills/estack-chris-voss/SKILL.md +80 -80
  10. package/skills/estack-chris-voss/references/elliot-notes.md +120 -120
  11. package/skills/estack-chris-voss/references/voss-principles.md +210 -210
  12. package/skills/estack-customer-discovery/SKILL.md +60 -60
  13. package/skills/estack-flight-planner/SKILL.md +332 -332
  14. package/skills/estack-flight-planner/references/config_schema.md +156 -156
  15. package/skills/estack-flight-planner/references/flight_history_schema.md +97 -97
  16. package/skills/estack-flight-planner/references/shuttle_schedules.md +98 -98
  17. package/skills/estack-flight-planner/scripts/check_setup.sh +89 -89
  18. package/skills/estack-flight-planner/scripts/fetch_flights.py +99 -99
  19. package/skills/estack-flight-planner/scripts/filter_flights.py +265 -265
  20. package/skills/estack-flight-planner/scripts/pair_shuttles.py +173 -173
  21. package/skills/estack-github-issue-tracker/SKILL.md +322 -322
  22. package/skills/estack-github-issue-tracker/bin/tracker-tools.cjs +1358 -1358
  23. package/skills/estack-github-issue-tracker/references/gh-cli-patterns.md +124 -124
  24. package/skills/estack-github-issue-tracker/references/result-file-schema.md +156 -156
  25. package/skills/estack-github-issue-tracker/references/tracker-schema.md +96 -96
  26. package/skills/estack-github-issue-tracker/tracker-template.md +58 -58
  27. package/skills/estack-leadership-coach/SKILL.md +235 -0
  28. package/skills/estack-leadership-coach/adding-references.md +280 -0
  29. package/skills/estack-leadership-coach/frameworks/delegation/flows/post-mortem.md +120 -0
  30. package/skills/estack-leadership-coach/frameworks/delegation/flows/pre-delegation.md +138 -0
  31. package/skills/estack-leadership-coach/frameworks/delegation/phases/1-intake.md +145 -0
  32. package/skills/estack-leadership-coach/frameworks/delegation/phases/2-trm-assessment.md +119 -0
  33. package/skills/estack-leadership-coach/frameworks/delegation/phases/3-enrollment.md +132 -0
  34. package/skills/estack-leadership-coach/frameworks/delegation/phases/4-build-brief.md +171 -0
  35. package/skills/estack-leadership-coach/frameworks/delegation/phases/5-monitoring.md +134 -0
  36. package/skills/estack-leadership-coach/frameworks/delegation/phases/6-reverse-delegation.md +118 -0
  37. package/skills/estack-leadership-coach/frameworks/delegation/phases/7-diagnose.md +200 -0
  38. package/skills/estack-leadership-coach/references/.source-files/deci-ryan_self-determination-theory__deci-olafsen-ryan-2017-self-determination-theory-in-work-organizations.md +1881 -0
  39. package/skills/estack-leadership-coach/references/.source-files/deci-ryan_self-determination-theory__gagne-deci-2005-self-determination-theory-and-work-motivation.md +2058 -0
  40. package/skills/estack-leadership-coach/references/.source-files/deci-ryan_self-determination-theory__selfdeterminationtheory-org-theory-overview-page.md +61 -0
  41. package/skills/estack-leadership-coach/references/.source-files/gallup_engagement-research__gallup-3-key-insights-into-the-global-workplace-2024.md +57 -0
  42. package/skills/estack-leadership-coach/references/.source-files/gallup_engagement-research__gallup-managers-account-for-70-percent-of-variance-in-employee-engagement-2015.md +40 -0
  43. package/skills/estack-leadership-coach/references/.source-files/gallup_engagement-research__gallup-state-of-the-global-workplace-2026-global-data-summary.md +73 -0
  44. package/skills/estack-leadership-coach/references/.source-files/gallup_engagement-research__gallup-state-of-the-global-workplace-2026-report-landing.md +42 -0
  45. package/skills/estack-leadership-coach/references/.source-files/hormozi-leila_4-stages__leila-hormozi-the-art-of-delegation-blog-post.md +91 -0
  46. package/skills/estack-leadership-coach/references/.source-files/oncken-wass_monkeys-hbr-1974__oncken-wass-management-time-whos-got-the-monkey-hbr-classic-1974.md +969 -0
  47. package/skills/estack-leadership-coach/references/.source-files/sanchez_main-street-millionaire__codie-sanchez-afford-anything-podcast-ep-565-show-notes.md +89 -0
  48. package/skills/estack-leadership-coach/references/.source-files/sullivan_who-not-how__dan-sullivan-impact-filter-tool-and-guide-booklet.md +565 -0
  49. package/skills/estack-leadership-coach/references/.source-files/van-edwards_cues__vanessa-van-edwards-lewis-howes-school-of-greatness-ep-1231-show-notes.md +122 -0
  50. package/skills/estack-leadership-coach/references/.source-files/van-edwards_cues__vanessa-van-edwards-roger-dooley-cues-interview.md +194 -0
  51. package/skills/estack-leadership-coach/references/deci-ryan_self-determination-theory.md +166 -0
  52. package/skills/estack-leadership-coach/references/doerr_measure-what-matters.md +154 -0
  53. package/skills/estack-leadership-coach/references/ferriss_4hww.md +189 -0
  54. package/skills/estack-leadership-coach/references/gallup_engagement-research.md +105 -0
  55. package/skills/estack-leadership-coach/references/gerber_e-myth-revisited.md +118 -0
  56. package/skills/estack-leadership-coach/references/grove_high-output-management.md +95 -0
  57. package/skills/estack-leadership-coach/references/hormozi-alex_followthrough.md +152 -0
  58. package/skills/estack-leadership-coach/references/hormozi-leila_4-stages.md +146 -0
  59. package/skills/estack-leadership-coach/references/oncken-wass_monkeys-hbr-1974.md +128 -0
  60. package/skills/estack-leadership-coach/references/sanchez_main-street-millionaire.md +196 -0
  61. package/skills/estack-leadership-coach/references/sullivan_who-not-how.md +137 -0
  62. package/skills/estack-leadership-coach/references/van-edwards_cues.md +189 -0
  63. package/skills/estack-migrate-claude-session-history/SKILL.md +226 -0
  64. package/skills/estack-migrate-claude-session-history/references/path-encoding.md +55 -0
  65. package/skills/estack-migrate-claude-session-history/references/troubleshooting.md +96 -0
  66. package/skills/estack-migrate-claude-session-history/scripts/migrate-claude-history.js +1123 -0
  67. package/skills/estack-migrate-claude-session-history/scripts/test-append-note.js +48 -0
  68. package/skills/estack-migrate-claude-session-history/scripts/test-validate-migration.py +326 -0
  69. package/skills/estack-migrate-claude-session-history/scripts/validate-migration.py +493 -0
  70. package/skills/estack-pdf-to-md/SKILL.md +180 -0
  71. package/skills/estack-pdf-to-md/scripts/pdf_to_md.py +596 -0
  72. package/skills/estack-productivity-prioritization-coach/SKILL.md +124 -0
  73. package/skills/estack-productivity-prioritization-coach/sources/01-tony-robbins-rpm.md +39 -0
  74. package/skills/estack-productivity-prioritization-coach/sources/02-justin-sung-task-prioritization.md +34 -0
  75. package/skills/estack-prompt-builder-coach/SKILL.md +81 -81
  76. package/skills/estack-prompt-builder-coach/definition-of-done-generator.md +42 -42
  77. package/skills/estack-prompt-builder-coach/prompt-builder.md +37 -37
  78. package/skills/estack-prompt-builder-coach/task-shaper.md +36 -36
  79. package/skills/estack-prompt-builder-coach/vague-ask-auditor.md +37 -37
  80. package/skills/estack-read-claude-session-history/SKILL.md +204 -204
  81. package/skills/estack-read-claude-session-history/references/jsonl-schema.md +126 -126
  82. package/skills/estack-read-claude-session-history/references/modes.md +423 -423
  83. package/skills/estack-read-claude-session-history/references/recipes.md +271 -271
  84. package/skills/estack-read-claude-session-history/scripts/lib/__init__.py +1 -1
  85. package/skills/estack-read-claude-session-history/scripts/lib/parser.py +460 -460
  86. package/skills/estack-read-claude-session-history/scripts/lib/paths.py +234 -234
  87. package/skills/estack-read-claude-session-history/scripts/lib/search.py +179 -179
  88. package/skills/estack-read-claude-session-history/scripts/lib/subagents.py +88 -88
  89. package/skills/estack-read-claude-session-history/scripts/lib/tools.py +144 -144
  90. package/skills/estack-read-claude-session-history/scripts/read_transcript.py +1776 -1776
  91. package/skills/estack-read-claude-session-history/scripts/tests/conftest.py +40 -40
  92. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/README.md +20 -20
  93. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/all-noise.jsonl +4 -4
  94. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/basic-session.jsonl +2 -2
  95. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/engagement-gaps.jsonl +9 -9
  96. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/engagement-noise.jsonl +7 -7
  97. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/engagement-parallel-a.jsonl +3 -3
  98. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/engagement-parallel-b.jsonl +3 -3
  99. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/engagement-waiting.jsonl +5 -5
  100. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/interrupted.jsonl +2 -2
  101. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/multi-compact.jsonl +8 -8
  102. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/pending-user.jsonl +2 -2
  103. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/subagent-no-meta/subagents/agent-aaa.jsonl +2 -2
  104. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/subagent-no-meta.jsonl +2 -2
  105. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/subagent-parent/subagents/agent-xyz123.jsonl +2 -2
  106. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/subagent-parent/subagents/agent-xyz123.meta.json +1 -1
  107. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/subagent-parent.jsonl +4 -4
  108. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/time-spread.jsonl +6 -6
  109. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/timeline-day-test.jsonl +5 -5
  110. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/tool-zoo.jsonl +10 -10
  111. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/truncated.jsonl +2 -2
  112. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/unicode.jsonl +2 -2
  113. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/with-advisor.jsonl +3 -3
  114. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/with-compact.jsonl +5 -5
  115. package/skills/estack-read-claude-session-history/scripts/tests/fixtures/with-thinking.jsonl +2 -2
  116. package/skills/estack-read-claude-session-history/scripts/tests/test_backup_roots.py +56 -56
  117. package/skills/estack-read-claude-session-history/scripts/tests/test_engagement.py +239 -239
  118. package/skills/estack-read-claude-session-history/scripts/tests/test_json_format.py +201 -201
  119. package/skills/estack-read-claude-session-history/scripts/tests/test_modes.py +199 -199
  120. package/skills/estack-read-claude-session-history/scripts/tests/test_parser.py +195 -195
  121. package/skills/estack-read-claude-session-history/scripts/tests/test_paths.py +133 -133
  122. package/skills/estack-read-claude-session-history/scripts/tests/test_search.py +78 -78
  123. package/skills/estack-read-claude-session-history/scripts/tests/test_subagents.py +43 -43
  124. package/skills/estack-read-claude-session-history/scripts/tests/test_timeline.py +179 -179
  125. package/skills/estack-read-claude-session-history/scripts/tests/test_timezone_and_project.py +212 -212
  126. package/skills/estack-read-claude-session-history/scripts/tests/test_tools.py +80 -80
  127. package/skills/estack-repo-search/SKILL.md +3 -1
@@ -1,239 +1,239 @@
1
- """Tests for the engagement mode (attention-time accounting)."""
2
-
3
- import json
4
- import os
5
- import shutil
6
- import subprocess
7
- import sys
8
- from datetime import datetime, timedelta
9
-
10
- import pytest
11
-
12
- import read_transcript as RT
13
- from lib import parser as PR
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 utc_tz():
32
- PR.set_timezone("UTC")
33
- yield
34
- PR.set_timezone(None)
35
-
36
-
37
- def _fake_root(tmp_path, **projects):
38
- """Build a fake projects root: {project_dir_name: [(fixture, uuid), ...]}."""
39
- root = tmp_path / "projects"
40
- for proj_name, files in projects.items():
41
- pd = root / proj_name
42
- pd.mkdir(parents=True)
43
- for src, uuid in files:
44
- shutil.copy(src, pd / f"{uuid}.jsonl")
45
- return root
46
-
47
-
48
- def _build(root, **kw):
49
- defaults = dict(
50
- report_dirs=None,
51
- report_file=None,
52
- since=datetime(2026, 5, 1),
53
- until=datetime(2026, 5, 2),
54
- break_minutes=10,
55
- current_uuid=None,
56
- )
57
- defaults.update(kw)
58
- return RT.build_engagement(root, **defaults)
59
-
60
-
61
- # ── unit: gap math ───────────────────────────────────────────────────────────
62
-
63
- def test_gaps_straddle_threshold(fixtures_dir, tmp_path, utc_tz):
64
- root = _fake_root(
65
- tmp_path, **{"C--proj-a": [(fixtures_dir / "engagement-gaps.jsonl", "aaaa1111")]}
66
- )
67
- data = _build(root)
68
- (s,) = data["sessions"].values()
69
- # 5m + 3m active, 32m break (Claude finished 10:09, reply 10:40 — too late),
70
- # then 5m active = 13m.
71
- assert s["active"] == timedelta(minutes=13)
72
- assert s["user_messages"] == 5
73
- assert len(data["breaks"]) == 1
74
- assert data["breaks"][0][0] == datetime(2026, 5, 1, 10, 8)
75
- assert data["breaks"][0][1] == datetime(2026, 5, 1, 10, 40)
76
-
77
-
78
- def test_waiting_on_claude_credit(fixtures_dir, tmp_path, utc_tz):
79
- root = _fake_root(
80
- tmp_path, **{"C--proj-a": [(fixtures_dir / "engagement-waiting.jsonl", "bbbb2222")]}
81
- )
82
- data = _build(root)
83
- (s,) = data["sessions"].values()
84
- # 32m gap, but Claude's last event was 10:30 and the user replied 10:32
85
- # (2m ≤ 10m) — the whole gap counts as waiting-on-Claude.
86
- assert s["active"] == timedelta(minutes=32)
87
- assert data["breaks"] == []
88
-
89
-
90
- def test_noise_exclusion(fixtures_dir, tmp_path, utc_tz):
91
- root = _fake_root(
92
- tmp_path, **{"C--proj-a": [(fixtures_dir / "engagement-noise.jsonl", "cccc3333")]}
93
- )
94
- data = _build(root)
95
- (s,) = data["sessions"].values()
96
- # Only the two real prompts (10:01, 10:06) count: compact continuation,
97
- # isMeta injections, and tool results are all excluded from the user stream.
98
- assert s["user_messages"] == 2
99
- assert s["first"] == datetime(2026, 5, 1, 10, 1)
100
- assert s["active"] == timedelta(minutes=5)
101
-
102
-
103
- def test_parallel_sessions_never_double_count(fixtures_dir, tmp_path, utc_tz):
104
- root = _fake_root(
105
- tmp_path,
106
- **{
107
- "C--proj-a": [(fixtures_dir / "engagement-parallel-a.jsonl", "aaaa1111")],
108
- "C--proj-b": [(fixtures_dir / "engagement-parallel-b.jsonl", "bbbb2222")],
109
- },
110
- )
111
- data = _build(root)
112
- by_uuid = {s["summary"]["uuid"]: s for s in data["sessions"].values()}
113
- # Stream: 10:00 A, 10:10 B, 10:20 B, 10:30 A. Each segment goes to the
114
- # session of the LATER prompt: B gets 10:00–10:20 (20m), A gets 10:20–10:30
115
- # (10m). Total 30m == wall clock, not the naive 40m.
116
- assert by_uuid["bbbb2222"]["active"] == timedelta(minutes=20)
117
- assert by_uuid["aaaa1111"]["active"] == timedelta(minutes=10)
118
- total = sum((s["active"] for s in data["sessions"].values()), timedelta())
119
- assert total == timedelta(minutes=30)
120
-
121
-
122
- def test_single_message_session(tmp_path, utc_tz):
123
- root = tmp_path / "projects"
124
- pd = root / "C--proj-a"
125
- pd.mkdir(parents=True)
126
- (pd / "dddd4444.jsonl").write_text(
127
- '{"type":"user","timestamp":"2026-05-01T10:00:00Z",'
128
- '"message":{"role":"user","content":"one and done"}}\n',
129
- encoding="utf-8",
130
- )
131
- data = _build(root)
132
- (s,) = data["sessions"].values()
133
- assert s["active"] == timedelta(0)
134
- assert s["user_messages"] == 1
135
-
136
-
137
- def test_report_scope_filters_but_stream_stays_global(fixtures_dir, tmp_path, utc_tz):
138
- root = _fake_root(
139
- tmp_path,
140
- **{
141
- "C--proj-a": [(fixtures_dir / "engagement-parallel-a.jsonl", "aaaa1111")],
142
- "C--proj-b": [(fixtures_dir / "engagement-parallel-b.jsonl", "bbbb2222")],
143
- },
144
- )
145
- data = _build(root, report_dirs=[root / "C--proj-a"])
146
- # Only A is reported, but B's prompts still split the stream — A gets its
147
- # interval-correct 10m, not a naive 30m.
148
- (s,) = data["sessions"].values()
149
- assert s["summary"]["uuid"] == "aaaa1111"
150
- assert s["active"] == timedelta(minutes=10)
151
-
152
-
153
- # ── CLI ──────────────────────────────────────────────────────────────────────
154
-
155
- def test_engagement_cli_text(cli_path, fixtures_dir, tmp_path):
156
- root = _fake_root(
157
- tmp_path, **{"C--proj-a": [(fixtures_dir / "engagement-gaps.jsonl", "aaaa1111")]}
158
- )
159
- r = _run_cli(
160
- cli_path, "--root", str(root), "--tz", "UTC",
161
- "--mode", "engagement", "--date", "2026-05-01",
162
- )
163
- assert r.returncode == 0
164
- assert "13m" in r.stdout
165
- assert "break=10m" in r.stdout
166
- assert "Breaks >10m" in r.stdout
167
-
168
-
169
- def test_engagement_cli_json(cli_path, fixtures_dir, tmp_path):
170
- root = _fake_root(
171
- tmp_path, **{"C--proj-a": [(fixtures_dir / "engagement-gaps.jsonl", "aaaa1111")]}
172
- )
173
- r = _run_cli(
174
- cli_path, "--root", str(root), "--tz", "UTC",
175
- "--mode", "engagement", "--date", "2026-05-01", "--format", "json",
176
- )
177
- assert r.returncode == 0
178
- data = json.loads(r.stdout)
179
- assert data["break_minutes"] == 10
180
- assert data["totals"]["active_minutes"] == 13
181
- assert data["totals"]["sessions"] == 1
182
- s = data["sessions"][0]
183
- assert s["uuid"] == "aaaa1111"
184
- assert s["elapsed_minutes"] == 45
185
- assert s["ratio"] == round(13 / 45, 2)
186
- assert len(data["stream_breaks"]) == 1
187
- assert data["stream_breaks"][0]["minutes"] == 32
188
-
189
-
190
- def test_engagement_cli_file_window_derived(cli_path, fixtures_dir, tmp_path):
191
- root = _fake_root(
192
- tmp_path, **{"C--proj-a": [(fixtures_dir / "engagement-waiting.jsonl", "bbbb2222")]}
193
- )
194
- f = root / "C--proj-a" / "bbbb2222.jsonl"
195
- r = _run_cli(
196
- cli_path, "--root", str(root), "--tz", "UTC",
197
- "--mode", "engagement", "--file", str(f),
198
- )
199
- assert r.returncode == 0
200
- assert "32m" in r.stdout
201
- assert "Prompt gaps" in r.stdout
202
-
203
-
204
- def test_engagement_cli_custom_break(cli_path, fixtures_dir, tmp_path):
205
- root = _fake_root(
206
- tmp_path, **{"C--proj-a": [(fixtures_dir / "engagement-gaps.jsonl", "aaaa1111")]}
207
- )
208
- r = _run_cli(
209
- cli_path, "--root", str(root), "--tz", "UTC",
210
- "--mode", "engagement", "--date", "2026-05-01",
211
- "--break", "1h", "--format", "json",
212
- )
213
- assert r.returncode == 0
214
- data = json.loads(r.stdout)
215
- # 1h threshold swallows the 32m gap — everything is active: 45m.
216
- assert data["totals"]["active_minutes"] == 45
217
- assert data["stream_breaks"] == []
218
-
219
-
220
- def test_engagement_cli_invalid_break(cli_path, tmp_path):
221
- root = tmp_path / "projects"
222
- root.mkdir()
223
- r = _run_cli(
224
- cli_path, "--root", str(root),
225
- "--mode", "engagement", "--break", "soon",
226
- )
227
- assert r.returncode == 1
228
-
229
-
230
- def test_engagement_cli_empty_range(cli_path, fixtures_dir, tmp_path):
231
- root = _fake_root(
232
- tmp_path, **{"C--proj-a": [(fixtures_dir / "engagement-gaps.jsonl", "aaaa1111")]}
233
- )
234
- r = _run_cli(
235
- cli_path, "--root", str(root), "--tz", "UTC",
236
- "--mode", "engagement", "--date", "2020-01-01",
237
- )
238
- assert r.returncode == 0
239
- assert "no user messages" in r.stdout
1
+ """Tests for the engagement mode (attention-time accounting)."""
2
+
3
+ import json
4
+ import os
5
+ import shutil
6
+ import subprocess
7
+ import sys
8
+ from datetime import datetime, timedelta
9
+
10
+ import pytest
11
+
12
+ import read_transcript as RT
13
+ from lib import parser as PR
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 utc_tz():
32
+ PR.set_timezone("UTC")
33
+ yield
34
+ PR.set_timezone(None)
35
+
36
+
37
+ def _fake_root(tmp_path, **projects):
38
+ """Build a fake projects root: {project_dir_name: [(fixture, uuid), ...]}."""
39
+ root = tmp_path / "projects"
40
+ for proj_name, files in projects.items():
41
+ pd = root / proj_name
42
+ pd.mkdir(parents=True)
43
+ for src, uuid in files:
44
+ shutil.copy(src, pd / f"{uuid}.jsonl")
45
+ return root
46
+
47
+
48
+ def _build(root, **kw):
49
+ defaults = dict(
50
+ report_dirs=None,
51
+ report_file=None,
52
+ since=datetime(2026, 5, 1),
53
+ until=datetime(2026, 5, 2),
54
+ break_minutes=10,
55
+ current_uuid=None,
56
+ )
57
+ defaults.update(kw)
58
+ return RT.build_engagement(root, **defaults)
59
+
60
+
61
+ # ── unit: gap math ───────────────────────────────────────────────────────────
62
+
63
+ def test_gaps_straddle_threshold(fixtures_dir, tmp_path, utc_tz):
64
+ root = _fake_root(
65
+ tmp_path, **{"C--proj-a": [(fixtures_dir / "engagement-gaps.jsonl", "aaaa1111")]}
66
+ )
67
+ data = _build(root)
68
+ (s,) = data["sessions"].values()
69
+ # 5m + 3m active, 32m break (Claude finished 10:09, reply 10:40 — too late),
70
+ # then 5m active = 13m.
71
+ assert s["active"] == timedelta(minutes=13)
72
+ assert s["user_messages"] == 5
73
+ assert len(data["breaks"]) == 1
74
+ assert data["breaks"][0][0] == datetime(2026, 5, 1, 10, 8)
75
+ assert data["breaks"][0][1] == datetime(2026, 5, 1, 10, 40)
76
+
77
+
78
+ def test_waiting_on_claude_credit(fixtures_dir, tmp_path, utc_tz):
79
+ root = _fake_root(
80
+ tmp_path, **{"C--proj-a": [(fixtures_dir / "engagement-waiting.jsonl", "bbbb2222")]}
81
+ )
82
+ data = _build(root)
83
+ (s,) = data["sessions"].values()
84
+ # 32m gap, but Claude's last event was 10:30 and the user replied 10:32
85
+ # (2m ≤ 10m) — the whole gap counts as waiting-on-Claude.
86
+ assert s["active"] == timedelta(minutes=32)
87
+ assert data["breaks"] == []
88
+
89
+
90
+ def test_noise_exclusion(fixtures_dir, tmp_path, utc_tz):
91
+ root = _fake_root(
92
+ tmp_path, **{"C--proj-a": [(fixtures_dir / "engagement-noise.jsonl", "cccc3333")]}
93
+ )
94
+ data = _build(root)
95
+ (s,) = data["sessions"].values()
96
+ # Only the two real prompts (10:01, 10:06) count: compact continuation,
97
+ # isMeta injections, and tool results are all excluded from the user stream.
98
+ assert s["user_messages"] == 2
99
+ assert s["first"] == datetime(2026, 5, 1, 10, 1)
100
+ assert s["active"] == timedelta(minutes=5)
101
+
102
+
103
+ def test_parallel_sessions_never_double_count(fixtures_dir, tmp_path, utc_tz):
104
+ root = _fake_root(
105
+ tmp_path,
106
+ **{
107
+ "C--proj-a": [(fixtures_dir / "engagement-parallel-a.jsonl", "aaaa1111")],
108
+ "C--proj-b": [(fixtures_dir / "engagement-parallel-b.jsonl", "bbbb2222")],
109
+ },
110
+ )
111
+ data = _build(root)
112
+ by_uuid = {s["summary"]["uuid"]: s for s in data["sessions"].values()}
113
+ # Stream: 10:00 A, 10:10 B, 10:20 B, 10:30 A. Each segment goes to the
114
+ # session of the LATER prompt: B gets 10:00–10:20 (20m), A gets 10:20–10:30
115
+ # (10m). Total 30m == wall clock, not the naive 40m.
116
+ assert by_uuid["bbbb2222"]["active"] == timedelta(minutes=20)
117
+ assert by_uuid["aaaa1111"]["active"] == timedelta(minutes=10)
118
+ total = sum((s["active"] for s in data["sessions"].values()), timedelta())
119
+ assert total == timedelta(minutes=30)
120
+
121
+
122
+ def test_single_message_session(tmp_path, utc_tz):
123
+ root = tmp_path / "projects"
124
+ pd = root / "C--proj-a"
125
+ pd.mkdir(parents=True)
126
+ (pd / "dddd4444.jsonl").write_text(
127
+ '{"type":"user","timestamp":"2026-05-01T10:00:00Z",'
128
+ '"message":{"role":"user","content":"one and done"}}\n',
129
+ encoding="utf-8",
130
+ )
131
+ data = _build(root)
132
+ (s,) = data["sessions"].values()
133
+ assert s["active"] == timedelta(0)
134
+ assert s["user_messages"] == 1
135
+
136
+
137
+ def test_report_scope_filters_but_stream_stays_global(fixtures_dir, tmp_path, utc_tz):
138
+ root = _fake_root(
139
+ tmp_path,
140
+ **{
141
+ "C--proj-a": [(fixtures_dir / "engagement-parallel-a.jsonl", "aaaa1111")],
142
+ "C--proj-b": [(fixtures_dir / "engagement-parallel-b.jsonl", "bbbb2222")],
143
+ },
144
+ )
145
+ data = _build(root, report_dirs=[root / "C--proj-a"])
146
+ # Only A is reported, but B's prompts still split the stream — A gets its
147
+ # interval-correct 10m, not a naive 30m.
148
+ (s,) = data["sessions"].values()
149
+ assert s["summary"]["uuid"] == "aaaa1111"
150
+ assert s["active"] == timedelta(minutes=10)
151
+
152
+
153
+ # ── CLI ──────────────────────────────────────────────────────────────────────
154
+
155
+ def test_engagement_cli_text(cli_path, fixtures_dir, tmp_path):
156
+ root = _fake_root(
157
+ tmp_path, **{"C--proj-a": [(fixtures_dir / "engagement-gaps.jsonl", "aaaa1111")]}
158
+ )
159
+ r = _run_cli(
160
+ cli_path, "--root", str(root), "--tz", "UTC",
161
+ "--mode", "engagement", "--date", "2026-05-01",
162
+ )
163
+ assert r.returncode == 0
164
+ assert "13m" in r.stdout
165
+ assert "break=10m" in r.stdout
166
+ assert "Breaks >10m" in r.stdout
167
+
168
+
169
+ def test_engagement_cli_json(cli_path, fixtures_dir, tmp_path):
170
+ root = _fake_root(
171
+ tmp_path, **{"C--proj-a": [(fixtures_dir / "engagement-gaps.jsonl", "aaaa1111")]}
172
+ )
173
+ r = _run_cli(
174
+ cli_path, "--root", str(root), "--tz", "UTC",
175
+ "--mode", "engagement", "--date", "2026-05-01", "--format", "json",
176
+ )
177
+ assert r.returncode == 0
178
+ data = json.loads(r.stdout)
179
+ assert data["break_minutes"] == 10
180
+ assert data["totals"]["active_minutes"] == 13
181
+ assert data["totals"]["sessions"] == 1
182
+ s = data["sessions"][0]
183
+ assert s["uuid"] == "aaaa1111"
184
+ assert s["elapsed_minutes"] == 45
185
+ assert s["ratio"] == round(13 / 45, 2)
186
+ assert len(data["stream_breaks"]) == 1
187
+ assert data["stream_breaks"][0]["minutes"] == 32
188
+
189
+
190
+ def test_engagement_cli_file_window_derived(cli_path, fixtures_dir, tmp_path):
191
+ root = _fake_root(
192
+ tmp_path, **{"C--proj-a": [(fixtures_dir / "engagement-waiting.jsonl", "bbbb2222")]}
193
+ )
194
+ f = root / "C--proj-a" / "bbbb2222.jsonl"
195
+ r = _run_cli(
196
+ cli_path, "--root", str(root), "--tz", "UTC",
197
+ "--mode", "engagement", "--file", str(f),
198
+ )
199
+ assert r.returncode == 0
200
+ assert "32m" in r.stdout
201
+ assert "Prompt gaps" in r.stdout
202
+
203
+
204
+ def test_engagement_cli_custom_break(cli_path, fixtures_dir, tmp_path):
205
+ root = _fake_root(
206
+ tmp_path, **{"C--proj-a": [(fixtures_dir / "engagement-gaps.jsonl", "aaaa1111")]}
207
+ )
208
+ r = _run_cli(
209
+ cli_path, "--root", str(root), "--tz", "UTC",
210
+ "--mode", "engagement", "--date", "2026-05-01",
211
+ "--break", "1h", "--format", "json",
212
+ )
213
+ assert r.returncode == 0
214
+ data = json.loads(r.stdout)
215
+ # 1h threshold swallows the 32m gap — everything is active: 45m.
216
+ assert data["totals"]["active_minutes"] == 45
217
+ assert data["stream_breaks"] == []
218
+
219
+
220
+ def test_engagement_cli_invalid_break(cli_path, tmp_path):
221
+ root = tmp_path / "projects"
222
+ root.mkdir()
223
+ r = _run_cli(
224
+ cli_path, "--root", str(root),
225
+ "--mode", "engagement", "--break", "soon",
226
+ )
227
+ assert r.returncode == 1
228
+
229
+
230
+ def test_engagement_cli_empty_range(cli_path, fixtures_dir, tmp_path):
231
+ root = _fake_root(
232
+ tmp_path, **{"C--proj-a": [(fixtures_dir / "engagement-gaps.jsonl", "aaaa1111")]}
233
+ )
234
+ r = _run_cli(
235
+ cli_path, "--root", str(root), "--tz", "UTC",
236
+ "--mode", "engagement", "--date", "2020-01-01",
237
+ )
238
+ assert r.returncode == 0
239
+ assert "no user messages" in r.stdout