Skip to content

Commit 8637853

Browse files
authored
Add a counter under the execution table to show completed tasks. (#252)
1 parent 37b9ddd commit 8637853

File tree

3 files changed

+94
-35
lines changed

3 files changed

+94
-35
lines changed

docs/source/changes.md

+2
Original file line numberDiff line numberDiff line change
@@ -34,6 +34,8 @@ releases are available on [PyPI](https://pypi.org/project/pytask) and
3434
- {pull}`245` adds choices on the command line to the help pages as metavars and show
3535
defaults.
3636
- {pull}`246` formalizes choices for {class}`click.Choice` to {class}`enum.Enum`.
37+
- {pull}`248` adds a counter at the bottom of the execution table to show how many tasks
38+
have been processed.
3739

3840
## 0.1.9 - 2022-02-23
3941

src/_pytask/live.py

+62-31
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@
55
from typing import Dict
66
from typing import Generator
77
from typing import List
8+
from typing import Union
89

910
import attr
1011
import click
@@ -16,9 +17,11 @@
1617
from _pytask.outcomes import TaskOutcome
1718
from _pytask.report import CollectionReport
1819
from _pytask.report import ExecutionReport
20+
from _pytask.session import Session
1921
from _pytask.shared import get_first_non_none_value
2022
from rich.live import Live
2123
from rich.status import Status
24+
from rich.style import Style
2225
from rich.table import Table
2326
from rich.text import Text
2427

@@ -78,15 +81,21 @@ def pytask_post_parse(config: dict[str, Any]) -> None:
7881

7982
if config["verbose"] >= 1:
8083
live_execution = LiveExecution(
81-
live_manager,
82-
config["n_entries_in_table"],
83-
config["verbose"],
84-
config["editor_url_scheme"],
84+
live_manager=live_manager,
85+
n_entries_in_table=config["n_entries_in_table"],
86+
verbose=config["verbose"],
87+
editor_url_scheme=config["editor_url_scheme"],
8588
)
86-
config["pm"].register(live_execution)
89+
config["pm"].register(live_execution, "live_execution")
8790

88-
live_collection = LiveCollection(live_manager)
89-
config["pm"].register(live_collection)
91+
live_collection = LiveCollection(live_manager=live_manager)
92+
config["pm"].register(live_collection, "live_collection")
93+
94+
95+
@hookimpl(tryfirst=True)
96+
def pytask_execute_build(session: Session) -> None:
97+
live_execution = session.config["pm"].get_plugin("live_execution")
98+
live_execution.n_tasks = len(session.tasks)
9099

91100

92101
@attr.s(eq=False)
@@ -138,25 +147,28 @@ def is_started(self) -> None:
138147
return self._live.is_started
139148

140149

141-
@attr.s(eq=False)
150+
@attr.s(eq=False, kw_only=True)
142151
class LiveExecution:
143152
"""A class for managing the table displaying task progress during the execution."""
144153

145-
_live_manager = attr.ib(type=LiveManager)
146-
_n_entries_in_table = attr.ib(type=int)
147-
_verbose = attr.ib(type=int)
148-
_editor_url_scheme = attr.ib(type=str)
149-
_running_tasks = attr.ib(factory=dict, type=Dict[str, Task])
154+
live_manager = attr.ib(type=LiveManager)
155+
n_entries_in_table = attr.ib(type=int)
156+
verbose = attr.ib(type=int)
157+
editor_url_scheme = attr.ib(type=str)
158+
n_tasks = attr.ib(default="x", type=Union[int, str])
150159
_reports = attr.ib(factory=list, type=List[Dict[str, Any]])
160+
_running_tasks = attr.ib(factory=dict, type=Dict[str, Task])
151161

152162
@hookimpl(hookwrapper=True)
153163
def pytask_execute_build(self) -> Generator[None, None, None]:
154164
"""Wrap the execution with the live manager and yield a complete table at the
155165
end."""
156-
self._live_manager.start()
166+
self.live_manager.start()
157167
yield
158-
self._live_manager.stop(transient=True)
159-
table = self._generate_table(reduce_table=False, sort_table=True)
168+
self.live_manager.stop(transient=True)
169+
table = self._generate_table(
170+
reduce_table=False, sort_table=True, add_caption=False
171+
)
160172
if table is not None:
161173
console.print(table)
162174

@@ -172,7 +184,9 @@ def pytask_execute_task_log_end(self, report: ExecutionReport) -> bool:
172184
self.update_reports(report)
173185
return True
174186

175-
def _generate_table(self, reduce_table: bool, sort_table: bool) -> Table | None:
187+
def _generate_table(
188+
self, reduce_table: bool, sort_table: bool, add_caption: bool
189+
) -> Table | None:
176190
"""Generate the table.
177191
178192
First, display all completed tasks and, then, all running tasks.
@@ -181,9 +195,9 @@ def _generate_table(self, reduce_table: bool, sort_table: bool) -> Table | None:
181195
if more entries are requested, the list is filled up with completed tasks.
182196
183197
"""
184-
n_reports_to_display = self._n_entries_in_table - len(self._running_tasks)
198+
n_reports_to_display = self.n_entries_in_table - len(self._running_tasks)
185199

186-
if self._verbose < 2:
200+
if self.verbose < 2:
187201
reports = [
188202
report
189203
for report in self._reports
@@ -210,22 +224,34 @@ def _generate_table(self, reduce_table: bool, sort_table: bool) -> Table | None:
210224
relevant_reports, key=lambda report: report["name"]
211225
)
212226

213-
table = Table()
227+
if add_caption:
228+
caption_kwargs = {
229+
"caption": Text(
230+
f"Completed: {len(self._reports)}/{self.n_tasks}",
231+
style=Style(dim=True, italic=False),
232+
),
233+
"caption_justify": "right",
234+
"caption_style": None,
235+
}
236+
else:
237+
caption_kwargs = {}
238+
239+
table = Table(**caption_kwargs)
214240
table.add_column("Task", overflow="fold")
215241
table.add_column("Outcome")
216242
for report in relevant_reports:
217243
table.add_row(
218244
format_task_id(
219245
report["task"],
220-
editor_url_scheme=self._editor_url_scheme,
246+
editor_url_scheme=self.editor_url_scheme,
221247
short_name=True,
222248
),
223249
Text(report["outcome"].symbol, style=report["outcome"].style),
224250
)
225251
for task in self._running_tasks.values():
226252
table.add_row(
227253
format_task_id(
228-
task, editor_url_scheme=self._editor_url_scheme, short_name=True
254+
task, editor_url_scheme=self.editor_url_scheme, short_name=True
229255
),
230256
"running",
231257
)
@@ -237,11 +263,16 @@ def _generate_table(self, reduce_table: bool, sort_table: bool) -> Table | None:
237263
return table
238264

239265
def _update_table(
240-
self, reduce_table: bool = True, sort_table: bool = False
266+
self,
267+
reduce_table: bool = True,
268+
sort_table: bool = False,
269+
add_caption: bool = True,
241270
) -> None:
242271
"""Regenerate the table."""
243-
table = self._generate_table(reduce_table=reduce_table, sort_table=sort_table)
244-
self._live_manager.update(table)
272+
table = self._generate_table(
273+
reduce_table=reduce_table, sort_table=sort_table, add_caption=add_caption
274+
)
275+
self.live_manager.update(table)
245276

246277
def update_running_tasks(self, new_running_task: Task) -> None:
247278
"""Add a new running task."""
@@ -261,18 +292,18 @@ def update_reports(self, new_report: ExecutionReport) -> None:
261292
self._update_table()
262293

263294

264-
@attr.s(eq=False)
295+
@attr.s(eq=False, kw_only=True)
265296
class LiveCollection:
266297
"""A class for managing the live status during the collection."""
267298

268-
_live_manager = attr.ib(type=LiveManager)
299+
live_manager = attr.ib(type=LiveManager)
269300
_n_collected_tasks = attr.ib(default=0, type=int)
270301
_n_errors = attr.ib(default=0, type=int)
271302

272303
@hookimpl(hookwrapper=True)
273304
def pytask_collect(self) -> Generator[None, None, None]:
274-
"""Start the status of the cllection."""
275-
self._live_manager.start()
305+
"""Start the status of the collection."""
306+
self.live_manager.start()
276307
yield
277308

278309
@hookimpl
@@ -284,7 +315,7 @@ def pytask_collect_file_log(self, reports: list[CollectionReport]) -> None:
284315
@hookimpl(hookwrapper=True)
285316
def pytask_collect_log(self) -> Generator[None, None, None]:
286317
"""Stop the live display when all tasks have been collected."""
287-
self._live_manager.stop(transient=True)
318+
self.live_manager.stop(transient=True)
288319
yield
289320

290321
def _update_statistics(self, reports: list[CollectionReport]) -> None:
@@ -300,7 +331,7 @@ def _update_statistics(self, reports: list[CollectionReport]) -> None:
300331
def _update_status(self) -> None:
301332
"""Update the status."""
302333
status = self._generate_status()
303-
self._live_manager.update(status)
334+
self.live_manager.update(status)
304335

305336
def _generate_status(self) -> Status:
306337
"""Generate the status."""

tests/test_live.py

+30-4
Original file line numberDiff line numberDiff line change
@@ -65,7 +65,12 @@ def test_live_execution_sequentially(capsys, tmp_path):
6565
task.short_name = "task_module.py::task_example"
6666

6767
live_manager = LiveManager()
68-
live = LiveExecution(live_manager, 20, 1, "no_link")
68+
live = LiveExecution(
69+
live_manager=live_manager,
70+
n_entries_in_table=20,
71+
verbose=1,
72+
editor_url_scheme="no_link",
73+
)
6974

7075
live_manager.start()
7176
live.update_running_tasks(task)
@@ -77,6 +82,7 @@ def test_live_execution_sequentially(capsys, tmp_path):
7782
assert "Outcome" not in captured.out
7883
assert "task_module.py::task_example" not in captured.out
7984
assert "running" not in captured.out
85+
assert "Completed: 0/x" not in captured.out
8086

8187
live_manager.resume()
8288
live_manager.start()
@@ -88,6 +94,7 @@ def test_live_execution_sequentially(capsys, tmp_path):
8894
assert "Outcome" in captured.out
8995
assert "task_module.py::task_example" in captured.out
9096
assert "running" in captured.out
97+
assert "Completed: 0/x" in captured.out
9198

9299
live_manager.start()
93100

@@ -104,6 +111,7 @@ def test_live_execution_sequentially(capsys, tmp_path):
104111
assert "task_module.py::task_example" in captured.out
105112
assert "running" not in captured.out
106113
assert TaskOutcome.SUCCESS.symbol in captured.out
114+
assert "Completed: 1/x" in captured.out
107115

108116

109117
@pytest.mark.unit
@@ -115,7 +123,12 @@ def test_live_execution_displays_skips_and_persists(capsys, tmp_path, verbose, o
115123
task.short_name = "task_module.py::task_example"
116124

117125
live_manager = LiveManager()
118-
live = LiveExecution(live_manager, 20, verbose, "no_link")
126+
live = LiveExecution(
127+
live_manager=live_manager,
128+
n_entries_in_table=20,
129+
verbose=verbose,
130+
editor_url_scheme="no_link",
131+
)
119132

120133
live_manager.start()
121134
live.update_running_tasks(task)
@@ -159,7 +172,13 @@ def test_live_execution_displays_subset_of_table(capsys, tmp_path, n_entries_in_
159172
running_task.short_name = "task_module.py::task_running"
160173

161174
live_manager = LiveManager()
162-
live = LiveExecution(live_manager, n_entries_in_table, 1, "no_link")
175+
live = LiveExecution(
176+
live_manager=live_manager,
177+
n_entries_in_table=n_entries_in_table,
178+
verbose=1,
179+
editor_url_scheme="no_link",
180+
n_tasks=2,
181+
)
163182

164183
live_manager.start()
165184
live.update_running_tasks(running_task)
@@ -170,6 +189,7 @@ def test_live_execution_displays_subset_of_table(capsys, tmp_path, n_entries_in_
170189
assert "Outcome" in captured.out
171190
assert "::task_running" in captured.out
172191
assert " running " in captured.out
192+
assert "Completed: 0/2" in captured.out
173193

174194
completed_task = Task(base_name="task_completed", path=path, function=lambda x: x)
175195
completed_task.short_name = "task_module.py::task_completed"
@@ -188,6 +208,7 @@ def test_live_execution_displays_subset_of_table(capsys, tmp_path, n_entries_in_
188208
assert "Outcome" in captured.out
189209
assert "::task_running" in captured.out
190210
assert " running " in captured.out
211+
assert "Completed: 1/2" in captured.out
191212

192213
if n_entries_in_table == 1:
193214
assert "task_module.py::task_completed" not in captured.out
@@ -204,7 +225,12 @@ def test_live_execution_skips_do_not_crowd_out_displayed_tasks(capsys, tmp_path)
204225
task.short_name = "task_module.py::task_example"
205226

206227
live_manager = LiveManager()
207-
live = LiveExecution(live_manager, 20, 1, "no_link")
228+
live = LiveExecution(
229+
live_manager=live_manager,
230+
n_entries_in_table=20,
231+
verbose=1,
232+
editor_url_scheme="no_link",
233+
)
208234

209235
live_manager.start()
210236
live.update_running_tasks(task)

0 commit comments

Comments
 (0)