mirror of
https://github.com/zebrajr/pytorch.git
synced 2025-12-06 12:20:52 +01:00
This is a temp fix to get trunk green (uses newer stats). The reason the test failed before was because the tests referred to old stats that were deleted after their lifecycle in S3. However, we need to come up with a better way of testing this as this same failure will occur again in a few months. Pull Request resolved: https://github.com/pytorch/pytorch/pull/73082
345 lines
11 KiB
Python
Executable File
345 lines
11 KiB
Python
Executable File
#!/usr/bin/env python3
|
|
|
|
import argparse
|
|
import subprocess
|
|
import sys
|
|
from datetime import datetime, timezone
|
|
from signal import SIG_DFL, SIGPIPE, signal
|
|
from typing import Dict, Iterator, List, Optional, Set, Tuple
|
|
|
|
from tools.stats.s3_stat_parser import (Report, get_cases,
|
|
get_test_stats_summaries)
|
|
|
|
|
|
def get_git_commit_history(
|
|
*,
|
|
path: str,
|
|
ref: str
|
|
) -> List[Tuple[str, datetime]]:
|
|
rc = subprocess.check_output(
|
|
['git', '-C', path, 'log', '--pretty=format:%H %ct', ref],
|
|
).decode("latin-1")
|
|
return [
|
|
(x[0], datetime.fromtimestamp(int(x[1]), tz=timezone.utc))
|
|
for x in [line.split(" ") for line in rc.split("\n")]
|
|
]
|
|
|
|
|
|
def make_column(
|
|
*,
|
|
data: Optional[Report],
|
|
filename: Optional[str],
|
|
suite_name: Optional[str],
|
|
test_name: str,
|
|
digits: int,
|
|
) -> Tuple[str, int]:
|
|
decimals = 3
|
|
num_length = digits + 1 + decimals
|
|
if data:
|
|
cases = get_cases(
|
|
data=data,
|
|
filename=filename,
|
|
suite_name=suite_name,
|
|
test_name=test_name
|
|
)
|
|
if cases:
|
|
case = cases[0]
|
|
status = case['status']
|
|
omitted = len(cases) - 1
|
|
if status:
|
|
return f'{status.rjust(num_length)} ', omitted
|
|
else:
|
|
return f'{case["seconds"]:{num_length}.{decimals}f}s', omitted
|
|
else:
|
|
return f'{"absent".rjust(num_length)} ', 0
|
|
else:
|
|
return ' ' * (num_length + 1), 0
|
|
|
|
|
|
def make_columns(
|
|
*,
|
|
jobs: List[str],
|
|
jsons: Dict[str, Report],
|
|
omitted: Dict[str, int],
|
|
filename: Optional[str],
|
|
suite_name: Optional[str],
|
|
test_name: str,
|
|
digits: int,
|
|
) -> str:
|
|
columns = []
|
|
total_omitted = 0
|
|
total_suites = 0
|
|
for job in jobs:
|
|
data = jsons.get(job)
|
|
column, omitted_suites = make_column(
|
|
data=data,
|
|
filename=filename,
|
|
suite_name=suite_name,
|
|
test_name=test_name,
|
|
digits=digits,
|
|
)
|
|
columns.append(column)
|
|
total_suites += omitted_suites
|
|
if job in omitted:
|
|
total_omitted += omitted[job]
|
|
if total_omitted > 0:
|
|
columns.append(f'({total_omitted} job re-runs omitted)')
|
|
if total_suites > 0:
|
|
columns.append(f'({total_suites} matching suites omitted)')
|
|
return ' '.join(columns)
|
|
|
|
|
|
def make_lines(
|
|
*,
|
|
jobs: Set[str],
|
|
jsons: Dict[str, List[Report]],
|
|
filename: Optional[str],
|
|
suite_name: Optional[str],
|
|
test_name: str,
|
|
) -> List[str]:
|
|
lines = []
|
|
for job, reports in jsons.items():
|
|
for data in reports:
|
|
cases = get_cases(
|
|
data=data,
|
|
filename=filename,
|
|
suite_name=suite_name,
|
|
test_name=test_name,
|
|
)
|
|
if cases:
|
|
case = cases[0]
|
|
status = case['status']
|
|
line = f'{job} {case["seconds"]}s{f" {status}" if status else ""}'
|
|
if len(cases) > 1:
|
|
line += f' ({len(cases) - 1} matching suites omitted)'
|
|
lines.append(line)
|
|
elif job in jobs:
|
|
lines.append(f'{job} (test not found)')
|
|
if lines:
|
|
return lines
|
|
else:
|
|
return ['(no reports in S3)']
|
|
|
|
|
|
def history_lines(
|
|
*,
|
|
commits: List[Tuple[str, datetime]],
|
|
jobs: Optional[List[str]],
|
|
filename: Optional[str],
|
|
suite_name: Optional[str],
|
|
test_name: str,
|
|
delta: int,
|
|
sha_length: int,
|
|
mode: str,
|
|
digits: int,
|
|
) -> Iterator[str]:
|
|
prev_time = datetime.now(tz=timezone.utc)
|
|
for sha, time in commits:
|
|
if (prev_time - time).total_seconds() < delta * 3600:
|
|
continue
|
|
prev_time = time
|
|
if jobs is None:
|
|
summaries = get_test_stats_summaries(sha=sha)
|
|
else:
|
|
summaries = get_test_stats_summaries(sha=sha, jobs=jobs)
|
|
if mode == 'columns':
|
|
assert jobs is not None
|
|
# we assume that get_test_stats_summaries here doesn't
|
|
# return empty lists
|
|
omitted = {
|
|
job: len(l) - 1
|
|
for job, l in summaries.items()
|
|
if len(l) > 1
|
|
}
|
|
lines = [make_columns(
|
|
jobs=jobs,
|
|
jsons={job: l[0] for job, l in summaries.items()},
|
|
omitted=omitted,
|
|
filename=filename,
|
|
suite_name=suite_name,
|
|
test_name=test_name,
|
|
digits=digits,
|
|
)]
|
|
else:
|
|
assert mode == 'multiline'
|
|
lines = make_lines(
|
|
jobs=set(jobs or []),
|
|
jsons=summaries,
|
|
filename=filename,
|
|
suite_name=suite_name,
|
|
test_name=test_name,
|
|
)
|
|
for line in lines:
|
|
yield f"{time:%Y-%m-%d %H:%M:%S}Z {sha[:sha_length]} {line}".rstrip()
|
|
|
|
|
|
class HelpFormatter(
|
|
argparse.ArgumentDefaultsHelpFormatter,
|
|
argparse.RawDescriptionHelpFormatter,
|
|
):
|
|
pass
|
|
|
|
|
|
def description() -> str:
|
|
return r'''
|
|
Display the history of a test.
|
|
|
|
Each line of (non-error) output starts with the timestamp and SHA1 hash
|
|
of the commit it refers to, in this format:
|
|
|
|
YYYY-MM-DD hh:mm:ss 0123456789abcdef0123456789abcdef01234567
|
|
|
|
In multiline mode, each line next includes the name of a CircleCI job,
|
|
followed by the time of the specified test in that job at that commit.
|
|
Example:
|
|
|
|
$ tools/stats/test_history.py --mode=multiline --ref=86a961af879 --sha-length=8 \
|
|
--test=test_composite_compliance_dot_cpu_float32 \
|
|
--job linux-xenial-py3.7-gcc5.4-test-default1 --job linux-xenial-py3.7-gcc7-test-default1
|
|
2022-02-18 15:47:37Z 86a961af linux-xenial-py3.7-gcc5.4-test-default1 0.001s
|
|
2022-02-18 15:47:37Z 86a961af linux-xenial-py3.7-gcc7-test-default1 0.001s
|
|
2022-02-18 15:12:34Z f5e201e4 linux-xenial-py3.7-gcc5.4-test-default1 0.001s
|
|
2022-02-18 15:12:34Z f5e201e4 linux-xenial-py3.7-gcc7-test-default1 0.001s
|
|
2022-02-18 13:14:56Z 1c0df265 linux-xenial-py3.7-gcc5.4-test-default1 0.001s
|
|
2022-02-18 13:14:56Z 1c0df265 linux-xenial-py3.7-gcc7-test-default1 0.001s
|
|
2022-02-18 13:14:56Z e73eaffd (no reports in S3)
|
|
2022-02-18 06:29:12Z 710f12f5 linux-xenial-py3.7-gcc5.4-test-default1 0.001s
|
|
|
|
Another multiline example, this time with the --all flag:
|
|
|
|
$ tools/stats/test_history.py --mode=multiline --all --ref=86a961af879 --delta=12 --sha-length=8 \
|
|
--test=test_composite_compliance_dot_cuda_float32
|
|
2022-02-18 03:49:46Z 69389fb5 linux-bionic-cuda10.2-py3.9-gcc7-test-default1 0.001s skipped
|
|
2022-02-18 03:49:46Z 69389fb5 linux-bionic-cuda10.2-py3.9-gcc7-test-slow1 0.001s skipped
|
|
2022-02-18 03:49:46Z 69389fb5 linux-xenial-cuda11.3-py3.7-gcc7-test-default1 0.001s skipped
|
|
2022-02-18 03:49:46Z 69389fb5 periodic-linux-bionic-cuda11.5-py3.7-gcc7-test-default1 0.001s skipped
|
|
2022-02-18 03:49:46Z 69389fb5 periodic-linux-xenial-cuda10.2-py3-gcc7-slow-gradcheck-test-default1 0.001s skipped
|
|
2022-02-18 03:49:46Z 69389fb5 periodic-linux-xenial-cuda11.1-py3.7-gcc7-debug-test-default1 0.001s skipped
|
|
|
|
In columns mode, the name of the job isn't printed, but the order of the
|
|
columns is guaranteed to match the order of the jobs passed on the
|
|
command line. Example:
|
|
|
|
$ tools/stats/test_history.py --mode=columns --ref=86a961af879 --sha-length=8 \
|
|
--test=test_composite_compliance_dot_cpu_float32 \
|
|
--job linux-xenial-py3.7-gcc5.4-test-default1 --job linux-xenial-py3.7-gcc7-test-default1
|
|
2022-02-18 15:47:37Z 86a961af 0.001s 0.001s
|
|
2022-02-18 15:12:34Z f5e201e4 0.001s 0.001s
|
|
2022-02-18 13:14:56Z 1c0df265 0.001s 0.001s
|
|
2022-02-18 13:14:56Z e73eaffd
|
|
2022-02-18 06:29:12Z 710f12f5 0.001s 0.001s
|
|
2022-02-18 05:20:30Z 51b04f27 0.001s 0.001s
|
|
2022-02-18 03:49:46Z 69389fb5 0.001s 0.001s
|
|
2022-02-18 00:19:12Z 056b6260 0.001s 0.001s
|
|
2022-02-17 23:58:32Z 39fb7714 0.001s 0.001s
|
|
|
|
Minor note: in columns mode, a blank cell means that no report was found
|
|
in S3, while the word "absent" means that a report was found but the
|
|
indicated test was not found in that report.
|
|
'''
|
|
|
|
|
|
def parse_args(raw: List[str]) -> argparse.Namespace:
|
|
parser = argparse.ArgumentParser(
|
|
__file__,
|
|
description=description(),
|
|
formatter_class=HelpFormatter,
|
|
)
|
|
parser.add_argument(
|
|
'--mode',
|
|
choices=['columns', 'multiline'],
|
|
help='output format',
|
|
default='columns',
|
|
)
|
|
parser.add_argument(
|
|
'--pytorch',
|
|
help='path to local PyTorch clone',
|
|
default='.',
|
|
)
|
|
parser.add_argument(
|
|
'--ref',
|
|
help='starting point (most recent Git ref) to display history for',
|
|
default='master',
|
|
)
|
|
parser.add_argument(
|
|
'--delta',
|
|
type=int,
|
|
help='minimum number of hours between commits',
|
|
default=0,
|
|
)
|
|
parser.add_argument(
|
|
'--sha-length',
|
|
type=int,
|
|
help='length of the prefix of the SHA1 hash to show',
|
|
default=40,
|
|
)
|
|
parser.add_argument(
|
|
'--digits',
|
|
type=int,
|
|
help='(columns) number of digits to display before the decimal point',
|
|
default=4,
|
|
)
|
|
parser.add_argument(
|
|
'--all',
|
|
action='store_true',
|
|
help='(multiline) ignore listed jobs, show all jobs for each commit',
|
|
)
|
|
parser.add_argument(
|
|
'--file',
|
|
help='name of the file containing the test',
|
|
)
|
|
parser.add_argument(
|
|
'--suite',
|
|
help='name of the suite containing the test',
|
|
)
|
|
parser.add_argument(
|
|
'--test',
|
|
help='name of the test',
|
|
required=True
|
|
)
|
|
parser.add_argument(
|
|
'--job',
|
|
help='names of jobs to display columns for, in order',
|
|
action='append',
|
|
default=[],
|
|
)
|
|
args = parser.parse_args(raw)
|
|
|
|
args.jobs = None if args.all else args.job
|
|
# We dont allow implicit or empty "--jobs", unless "--all" is specified.
|
|
if args.jobs == []:
|
|
parser.error('No jobs specified.')
|
|
|
|
return args
|
|
|
|
|
|
def run(raw: List[str]) -> Iterator[str]:
|
|
args = parse_args(raw)
|
|
|
|
commits = get_git_commit_history(path=args.pytorch, ref=args.ref)
|
|
|
|
return history_lines(
|
|
commits=commits,
|
|
jobs=args.jobs,
|
|
filename=args.file,
|
|
suite_name=args.suite,
|
|
test_name=args.test,
|
|
delta=args.delta,
|
|
mode=args.mode,
|
|
sha_length=args.sha_length,
|
|
digits=args.digits,
|
|
)
|
|
|
|
|
|
def main() -> None:
|
|
for line in run(sys.argv[1:]):
|
|
print(line, flush=True)
|
|
|
|
|
|
if __name__ == "__main__":
|
|
signal(SIGPIPE, SIG_DFL) # https://stackoverflow.com/a/30091579
|
|
try:
|
|
main()
|
|
except KeyboardInterrupt:
|
|
pass
|