5
5
from typing import Dict
6
6
from typing import Generator
7
7
from typing import List
8
+ from typing import Union
8
9
9
10
import attr
10
11
import click
16
17
from _pytask .outcomes import TaskOutcome
17
18
from _pytask .report import CollectionReport
18
19
from _pytask .report import ExecutionReport
20
+ from _pytask .session import Session
19
21
from _pytask .shared import get_first_non_none_value
20
22
from rich .live import Live
21
23
from rich .status import Status
24
+ from rich .style import Style
22
25
from rich .table import Table
23
26
from rich .text import Text
24
27
@@ -78,15 +81,21 @@ def pytask_post_parse(config: dict[str, Any]) -> None:
78
81
79
82
if config ["verbose" ] >= 1 :
80
83
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" ],
85
88
)
86
- config ["pm" ].register (live_execution )
89
+ config ["pm" ].register (live_execution , "live_execution" )
87
90
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 )
90
99
91
100
92
101
@attr .s (eq = False )
@@ -138,25 +147,28 @@ def is_started(self) -> None:
138
147
return self ._live .is_started
139
148
140
149
141
- @attr .s (eq = False )
150
+ @attr .s (eq = False , kw_only = True )
142
151
class LiveExecution :
143
152
"""A class for managing the table displaying task progress during the execution."""
144
153
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 ])
150
159
_reports = attr .ib (factory = list , type = List [Dict [str , Any ]])
160
+ _running_tasks = attr .ib (factory = dict , type = Dict [str , Task ])
151
161
152
162
@hookimpl (hookwrapper = True )
153
163
def pytask_execute_build (self ) -> Generator [None , None , None ]:
154
164
"""Wrap the execution with the live manager and yield a complete table at the
155
165
end."""
156
- self ._live_manager .start ()
166
+ self .live_manager .start ()
157
167
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
+ )
160
172
if table is not None :
161
173
console .print (table )
162
174
@@ -172,7 +184,9 @@ def pytask_execute_task_log_end(self, report: ExecutionReport) -> bool:
172
184
self .update_reports (report )
173
185
return True
174
186
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 :
176
190
"""Generate the table.
177
191
178
192
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:
181
195
if more entries are requested, the list is filled up with completed tasks.
182
196
183
197
"""
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 )
185
199
186
- if self ._verbose < 2 :
200
+ if self .verbose < 2 :
187
201
reports = [
188
202
report
189
203
for report in self ._reports
@@ -210,22 +224,34 @@ def _generate_table(self, reduce_table: bool, sort_table: bool) -> Table | None:
210
224
relevant_reports , key = lambda report : report ["name" ]
211
225
)
212
226
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 )
214
240
table .add_column ("Task" , overflow = "fold" )
215
241
table .add_column ("Outcome" )
216
242
for report in relevant_reports :
217
243
table .add_row (
218
244
format_task_id (
219
245
report ["task" ],
220
- editor_url_scheme = self ._editor_url_scheme ,
246
+ editor_url_scheme = self .editor_url_scheme ,
221
247
short_name = True ,
222
248
),
223
249
Text (report ["outcome" ].symbol , style = report ["outcome" ].style ),
224
250
)
225
251
for task in self ._running_tasks .values ():
226
252
table .add_row (
227
253
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
229
255
),
230
256
"running" ,
231
257
)
@@ -237,11 +263,16 @@ def _generate_table(self, reduce_table: bool, sort_table: bool) -> Table | None:
237
263
return table
238
264
239
265
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 ,
241
270
) -> None :
242
271
"""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 )
245
276
246
277
def update_running_tasks (self , new_running_task : Task ) -> None :
247
278
"""Add a new running task."""
@@ -261,18 +292,18 @@ def update_reports(self, new_report: ExecutionReport) -> None:
261
292
self ._update_table ()
262
293
263
294
264
- @attr .s (eq = False )
295
+ @attr .s (eq = False , kw_only = True )
265
296
class LiveCollection :
266
297
"""A class for managing the live status during the collection."""
267
298
268
- _live_manager = attr .ib (type = LiveManager )
299
+ live_manager = attr .ib (type = LiveManager )
269
300
_n_collected_tasks = attr .ib (default = 0 , type = int )
270
301
_n_errors = attr .ib (default = 0 , type = int )
271
302
272
303
@hookimpl (hookwrapper = True )
273
304
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 ()
276
307
yield
277
308
278
309
@hookimpl
@@ -284,7 +315,7 @@ def pytask_collect_file_log(self, reports: list[CollectionReport]) -> None:
284
315
@hookimpl (hookwrapper = True )
285
316
def pytask_collect_log (self ) -> Generator [None , None , None ]:
286
317
"""Stop the live display when all tasks have been collected."""
287
- self ._live_manager .stop (transient = True )
318
+ self .live_manager .stop (transient = True )
288
319
yield
289
320
290
321
def _update_statistics (self , reports : list [CollectionReport ]) -> None :
@@ -300,7 +331,7 @@ def _update_statistics(self, reports: list[CollectionReport]) -> None:
300
331
def _update_status (self ) -> None :
301
332
"""Update the status."""
302
333
status = self ._generate_status ()
303
- self ._live_manager .update (status )
334
+ self .live_manager .update (status )
304
335
305
336
def _generate_status (self ) -> Status :
306
337
"""Generate the status."""
0 commit comments