diff --git a/CHANGELOG.md b/CHANGELOG.md index 2898e932c..d3d22257b 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -10,7 +10,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 ### Added - You can now add arguments to steps without invalidating the cache. See `Step.SKIP_DEFAULT_ARGUMENTS`. - +- Added `display()` to `ExecutorOutput` for producing a table that summarizes the run ## [v1.1.0](https://github.com/allenai/tango/releases/tag/v1.1.0) - 2022-12-01 diff --git a/tango/executor.py b/tango/executor.py index 91480dcaf..62659ac04 100644 --- a/tango/executor.py +++ b/tango/executor.py @@ -1,7 +1,10 @@ import logging import warnings from dataclasses import dataclass, field -from typing import TYPE_CHECKING, Dict, Optional, Sequence, TypeVar +from typing import TYPE_CHECKING, Dict, List, Optional, Sequence, TypeVar + +from rich.console import Console +from rich.table import Table from .common.logging import log_exception from .common.registrable import Registrable @@ -46,6 +49,50 @@ class ExecutorOutput: not_run: Dict[str, ExecutionMetadata] = field(default_factory=dict) """Steps that were ignored (usually because of failed dependencies).""" + def display(self) -> None: + table = Table(caption_style="") + table.add_column("Step Name", justify="left", style="cyan") + table.add_column("Status", justify="left") + table.add_column("Results", justify="left") + all_steps = dict(self.successful) + all_steps.update(self.failed) + all_steps.update(self.not_run) + for step_name in sorted(all_steps): + status_str: str + result_str: str = "[grey62]N/A[/]" + if step_name in self.failed: + status_str = "[red]\N{ballot x} failed[/]" + execution_metadata = self.failed[step_name] + if execution_metadata.logs_location is not None: + result_str = f"[cyan]{execution_metadata.logs_location}[/]" + elif step_name in self.not_run: + status_str = "[yellow]- not run[/]" + elif step_name in self.successful: + status_str = "[green]\N{check mark} succeeded[/]" + execution_metadata = self.successful[step_name] + if execution_metadata.result_location is not None: + result_str = f"[cyan]{execution_metadata.result_location}[/]" + elif execution_metadata.logs_location is not None: + result_str = f"[cyan]{execution_metadata.logs_location}[/]" + else: + continue + + table.add_row(step_name, status_str, result_str) + + caption_parts: List[str] = [] + if self.failed: + caption_parts.append(f"[red]\N{ballot x}[/] [italic]{len(self.failed)} failed[/]") + if self.successful: + caption_parts.append( + f"[green]\N{check mark}[/] [italic]{len(self.successful)} succeeded[/]" + ) + if self.not_run: + caption_parts.append(f"[italic]{len(self.not_run)} not run[/]") + table.caption = ", ".join(caption_parts) + + console = Console() + console.print(table) + class Executor(Registrable): """