mirror of
https://github.com/ARM-software/workload-automation.git
synced 2025-01-18 12:06:08 +00:00
commands: Add report command
report provides a summary of a run and an optional list of all jobs in the run, with any events that might have occurred during each job and their current status. report allows an output directory to be specified or will attempt to discover possible output directories within the current directory.
This commit is contained in:
parent
7c6ebfb49c
commit
3f5a31de96
288
wa/commands/report.py
Normal file
288
wa/commands/report.py
Normal file
@ -0,0 +1,288 @@
|
|||||||
|
from collections import Counter
|
||||||
|
from datetime import datetime, timedelta
|
||||||
|
import logging
|
||||||
|
import os
|
||||||
|
|
||||||
|
from wa import Command, settings
|
||||||
|
from wa.framework.configuration.core import Status
|
||||||
|
from wa.framework.output import RunOutput, discover_wa_outputs
|
||||||
|
from wa.utils.doc import underline
|
||||||
|
from wa.utils.log import COLOR_MAP, RESET_COLOR
|
||||||
|
from wa.utils.terminalsize import get_terminal_size
|
||||||
|
|
||||||
|
|
||||||
|
class ReportCommand(Command):
|
||||||
|
|
||||||
|
name = 'report'
|
||||||
|
description = '''
|
||||||
|
Monitor an ongoing run and provide information on its progress.
|
||||||
|
|
||||||
|
Specify the output directory of the run you would like the monitor;
|
||||||
|
alternatively report will attempt to discover wa output directories
|
||||||
|
within the current directory. The output includes run information such as
|
||||||
|
the UUID, start time, duration, project name and a short summary of the
|
||||||
|
run's progress (number of completed jobs, the number of jobs in each
|
||||||
|
different status).
|
||||||
|
|
||||||
|
If verbose output is specified, the output includes a list of all events
|
||||||
|
labelled as not specific to any job, followed by a list of the jobs in the
|
||||||
|
order executed, with their retries (if any), current status and, if the job
|
||||||
|
is finished, a list of events that occurred during that job's execution.
|
||||||
|
|
||||||
|
This is an example of a job status line:
|
||||||
|
|
||||||
|
wk1 (exoplayer) [1] - 2, PARTIAL
|
||||||
|
|
||||||
|
It contains two entries delimited by a comma: the job's descriptor followed
|
||||||
|
by its completion status (``PARTIAL``, in this case). The descriptor
|
||||||
|
consists of the following elements:
|
||||||
|
|
||||||
|
- the job ID (``wk1``)
|
||||||
|
- the job label (which defaults to the workload name) in parentheses
|
||||||
|
- job iteration number in square brakets (``1`` in this case)
|
||||||
|
- a hyphen followed by the retry attempt number.
|
||||||
|
(note: this will only be shown if the job has been retried as least
|
||||||
|
once. If the job has not yet run, or if it completed on the first
|
||||||
|
attempt, the hyphen and retry count -- which in that case would be
|
||||||
|
zero -- will not appear).
|
||||||
|
'''
|
||||||
|
|
||||||
|
def initialize(self, context):
|
||||||
|
self.parser.add_argument('-d', '--directory',
|
||||||
|
help='''
|
||||||
|
Specify the WA output path. report will
|
||||||
|
otherwise attempt to discover output
|
||||||
|
directories in the current directory.
|
||||||
|
''')
|
||||||
|
|
||||||
|
def execute(self, state, args):
|
||||||
|
if args.directory:
|
||||||
|
output_path = args.directory
|
||||||
|
run_output = RunOutput(output_path)
|
||||||
|
else:
|
||||||
|
possible_outputs = list(discover_wa_outputs(os.getcwd()))
|
||||||
|
num_paths = len(possible_outputs)
|
||||||
|
|
||||||
|
if num_paths > 1:
|
||||||
|
print('More than one possible output directory found,'
|
||||||
|
' please choose a path from the following:'
|
||||||
|
)
|
||||||
|
|
||||||
|
for i in range(num_paths):
|
||||||
|
print("{}: {}".format(i, possible_outputs[i].basepath))
|
||||||
|
|
||||||
|
while True:
|
||||||
|
try:
|
||||||
|
select = int(input())
|
||||||
|
except ValueError:
|
||||||
|
print("Please select a valid path number")
|
||||||
|
continue
|
||||||
|
|
||||||
|
if select not in range(num_paths):
|
||||||
|
print("Please select a valid path number")
|
||||||
|
continue
|
||||||
|
break
|
||||||
|
|
||||||
|
run_output = possible_outputs[select]
|
||||||
|
|
||||||
|
else:
|
||||||
|
run_output = possible_outputs[0]
|
||||||
|
|
||||||
|
rm = RunMonitor(run_output)
|
||||||
|
print(rm.generate_output(args.verbose))
|
||||||
|
|
||||||
|
|
||||||
|
class RunMonitor:
|
||||||
|
|
||||||
|
@property
|
||||||
|
def elapsed_time(self):
|
||||||
|
if self._elapsed is None:
|
||||||
|
if self.ro.info.duration is None:
|
||||||
|
self._elapsed = datetime.utcnow() - self.ro.info.start_time
|
||||||
|
else:
|
||||||
|
self._elapsed = self.ro.info.duration
|
||||||
|
return self._elapsed
|
||||||
|
|
||||||
|
@property
|
||||||
|
def job_outputs(self):
|
||||||
|
if self._job_outputs is None:
|
||||||
|
self._job_outputs = {
|
||||||
|
(j_o.id, j_o.label, j_o.iteration): j_o for j_o in self.ro.jobs
|
||||||
|
}
|
||||||
|
return self._job_outputs
|
||||||
|
|
||||||
|
@property
|
||||||
|
def projected_duration(self):
|
||||||
|
elapsed = self.elapsed_time.total_seconds()
|
||||||
|
proj = timedelta(seconds=elapsed * (len(self.jobs) / len(self.segmented['finished'])))
|
||||||
|
return proj - self.elapsed_time
|
||||||
|
|
||||||
|
def __init__(self, ro):
|
||||||
|
self.ro = ro
|
||||||
|
self._elapsed = None
|
||||||
|
self._p_duration = None
|
||||||
|
self._job_outputs = None
|
||||||
|
self._termwidth = None
|
||||||
|
self._fmt = _simple_formatter()
|
||||||
|
self.get_data()
|
||||||
|
|
||||||
|
def get_data(self):
|
||||||
|
self.jobs = [state for label_id, state in self.ro.state.jobs.items()]
|
||||||
|
if self.jobs:
|
||||||
|
rc = self.ro.run_config
|
||||||
|
self.segmented = segment_jobs_by_state(self.jobs,
|
||||||
|
rc.max_retries,
|
||||||
|
rc.retry_on_status
|
||||||
|
)
|
||||||
|
|
||||||
|
def generate_run_header(self):
|
||||||
|
info = self.ro.info
|
||||||
|
|
||||||
|
header = underline('Run Info')
|
||||||
|
header += "UUID: {}\n".format(info.uuid)
|
||||||
|
if info.run_name:
|
||||||
|
header += "Run name: {}\n".format(info.run_name)
|
||||||
|
if info.project:
|
||||||
|
header += "Project: {}\n".format(info.project)
|
||||||
|
if info.project_stage:
|
||||||
|
header += "Project stage: {}\n".format(info.project_stage)
|
||||||
|
|
||||||
|
if info.start_time:
|
||||||
|
duration = _seconds_as_smh(self.elapsed_time.total_seconds())
|
||||||
|
header += ("Start time: {}\n"
|
||||||
|
"Duration: {:02}:{:02}:{:02}\n"
|
||||||
|
).format(info.start_time,
|
||||||
|
duration[2], duration[1], duration[0],
|
||||||
|
)
|
||||||
|
if self.segmented['finished'] and not info.end_time:
|
||||||
|
p_duration = _seconds_as_smh(self.projected_duration.total_seconds())
|
||||||
|
header += "Projected time remaining: {:02}:{:02}:{:02}\n".format(
|
||||||
|
p_duration[2], p_duration[1], p_duration[0]
|
||||||
|
)
|
||||||
|
|
||||||
|
elif self.ro.info.end_time:
|
||||||
|
header += "End time: {}\n".format(info.end_time)
|
||||||
|
|
||||||
|
return header + '\n'
|
||||||
|
|
||||||
|
def generate_job_summary(self):
|
||||||
|
total = len(self.jobs)
|
||||||
|
num_fin = len(self.segmented['finished'])
|
||||||
|
|
||||||
|
summary = underline('Job Summary')
|
||||||
|
summary += 'Total: {}, Completed: {} ({}%)\n'.format(
|
||||||
|
total, num_fin, (num_fin / total) * 100
|
||||||
|
) if total > 0 else 'No jobs created\n'
|
||||||
|
|
||||||
|
ctr = Counter()
|
||||||
|
for run_state, jobs in ((k, v) for k, v in self.segmented.items() if v):
|
||||||
|
if run_state == 'finished':
|
||||||
|
ctr.update([job.status.name.lower() for job in jobs])
|
||||||
|
else:
|
||||||
|
ctr[run_state] += len(jobs)
|
||||||
|
|
||||||
|
return summary + ', '.join(
|
||||||
|
[str(count) + ' ' + self._fmt.highlight_keyword(status) for status, count in ctr.items()]
|
||||||
|
) + '\n\n'
|
||||||
|
|
||||||
|
def generate_job_detail(self):
|
||||||
|
detail = underline('Job Detail')
|
||||||
|
for job in self.jobs:
|
||||||
|
detail += ('{} ({}) [{}]{}, {}\n').format(
|
||||||
|
job.id,
|
||||||
|
job.label,
|
||||||
|
job.iteration,
|
||||||
|
' - ' + str(job.retries)if job.retries else '',
|
||||||
|
self._fmt.highlight_keyword(str(job.status))
|
||||||
|
)
|
||||||
|
|
||||||
|
job_output = self.job_outputs[(job.id, job.label, job.iteration)]
|
||||||
|
for event in job_output.events:
|
||||||
|
detail += self._fmt.fit_term_width(
|
||||||
|
'\t{}\n'.format(event.summary)
|
||||||
|
)
|
||||||
|
return detail
|
||||||
|
|
||||||
|
def generate_run_detail(self):
|
||||||
|
detail = underline('Run Events') if self.ro.events else ''
|
||||||
|
|
||||||
|
for event in self.ro.events:
|
||||||
|
detail += '{}\n'.format(event.summary)
|
||||||
|
|
||||||
|
return detail + '\n'
|
||||||
|
|
||||||
|
def generate_output(self, verbose):
|
||||||
|
if not self.jobs:
|
||||||
|
return 'No jobs found in output directory\n'
|
||||||
|
|
||||||
|
output = self.generate_run_header()
|
||||||
|
output += self.generate_job_summary()
|
||||||
|
|
||||||
|
if verbose:
|
||||||
|
output += self.generate_run_detail()
|
||||||
|
output += self.generate_job_detail()
|
||||||
|
|
||||||
|
return output
|
||||||
|
|
||||||
|
|
||||||
|
def _seconds_as_smh(seconds):
|
||||||
|
seconds = int(seconds)
|
||||||
|
hours = seconds // 3600
|
||||||
|
minutes = (seconds % 3600) // 60
|
||||||
|
seconds = seconds % 60
|
||||||
|
return seconds, minutes, hours
|
||||||
|
|
||||||
|
|
||||||
|
def segment_jobs_by_state(jobstates, max_retries, retry_status):
|
||||||
|
finished_states = [
|
||||||
|
Status.PARTIAL, Status.FAILED,
|
||||||
|
Status.ABORTED, Status.OK, Status.SKIPPED
|
||||||
|
]
|
||||||
|
|
||||||
|
segmented = {
|
||||||
|
'finished': [], 'other': [], 'running': [],
|
||||||
|
'pending': [], 'uninitialized': []
|
||||||
|
}
|
||||||
|
|
||||||
|
for jobstate in jobstates:
|
||||||
|
if (jobstate.status in retry_status) and jobstate.retries < max_retries:
|
||||||
|
segmented['running'].append(jobstate)
|
||||||
|
elif jobstate.status in finished_states:
|
||||||
|
segmented['finished'].append(jobstate)
|
||||||
|
elif jobstate.status == Status.RUNNING:
|
||||||
|
segmented['running'].append(jobstate)
|
||||||
|
elif jobstate.status == Status.PENDING:
|
||||||
|
segmented['pending'].append(jobstate)
|
||||||
|
elif jobstate.status == Status.NEW:
|
||||||
|
segmented['uninitialized'].append(jobstate)
|
||||||
|
else:
|
||||||
|
segmented['other'].append(jobstate)
|
||||||
|
|
||||||
|
return segmented
|
||||||
|
|
||||||
|
|
||||||
|
class _simple_formatter:
|
||||||
|
color_map = {
|
||||||
|
'running': COLOR_MAP[logging.INFO],
|
||||||
|
'partial': COLOR_MAP[logging.WARNING],
|
||||||
|
'failed': COLOR_MAP[logging.CRITICAL],
|
||||||
|
'aborted': COLOR_MAP[logging.ERROR]
|
||||||
|
}
|
||||||
|
|
||||||
|
def __init__(self):
|
||||||
|
self.termwidth = get_terminal_size()[0]
|
||||||
|
self.color = settings.logging['color']
|
||||||
|
|
||||||
|
def fit_term_width(self, text):
|
||||||
|
text = text.expandtabs()
|
||||||
|
if len(text) <= self.termwidth:
|
||||||
|
return text
|
||||||
|
else:
|
||||||
|
return text[0:self.termwidth - 4] + " ...\n"
|
||||||
|
|
||||||
|
def highlight_keyword(self, kw):
|
||||||
|
if not self.color or kw not in _simple_formatter.color_map:
|
||||||
|
return kw
|
||||||
|
|
||||||
|
color = _simple_formatter.color_map[kw.lower()]
|
||||||
|
return '{}{}{}'.format(color, kw, RESET_COLOR)
|
Loading…
x
Reference in New Issue
Block a user