mirror of
https://github.com/ethereum/solidity
synced 2023-10-03 13:03:40 +00:00
Merge pull request #13097 from ethereum/skipping-ext-benchmark-diff
Skipping benchmark diff when previous run failed
This commit is contained in:
commit
95f22db175
@ -1257,12 +1257,17 @@ jobs:
|
|||||||
pr_id=$(echo "$CIRCLE_PULL_REQUEST" | sed 's|\(.*\)\/||')
|
pr_id=$(echo "$CIRCLE_PULL_REQUEST" | sed 's|\(.*\)\/||')
|
||||||
scripts_dir=../../../scripts
|
scripts_dir=../../../scripts
|
||||||
|
|
||||||
"${scripts_dir}/externalTests/download_benchmarks.py" --base-of-pr "$pr_id"
|
# Our main goal here is to provide new benchmarks, the diff is optional. When benchmarks from
|
||||||
|
# the previous run are not available for whatever reason, we still succeed and just skip the diff.
|
||||||
|
# download_benchmarks.py exits with status 2 in that case.
|
||||||
|
if "${scripts_dir}/externalTests/download_benchmarks.py" --base-of-pr "$pr_id" || [[ $? == 2 ]]; then
|
||||||
|
echo 'export SKIP_BENCHMARK_DIFF=true' >> $BASH_ENV
|
||||||
|
fi
|
||||||
fi
|
fi
|
||||||
- run:
|
- run:
|
||||||
name: Diff benchmarks
|
name: Diff benchmarks
|
||||||
command: |
|
command: |
|
||||||
if [[ $CIRCLE_PULL_REQUEST != "" ]]; then
|
if [[ $CIRCLE_PULL_REQUEST != "" && $SKIP_BENCHMARK_DIFF != "true" ]]; then
|
||||||
cd reports/externalTests/
|
cd reports/externalTests/
|
||||||
mkdir diff/
|
mkdir diff/
|
||||||
scripts_dir=../../scripts
|
scripts_dir=../../scripts
|
||||||
|
@ -11,7 +11,28 @@ import requests
|
|||||||
class APIHelperError(Exception):
|
class APIHelperError(Exception):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
class DataUnavailable(APIHelperError):
|
class JobNotSuccessful(APIHelperError):
|
||||||
|
def __init__(self, name: str, status: str):
|
||||||
|
assert status != 'success'
|
||||||
|
|
||||||
|
self.name = name
|
||||||
|
self.status = status
|
||||||
|
self.job_finished = (status in ['failed', 'blocked'])
|
||||||
|
|
||||||
|
if status == 'not_running':
|
||||||
|
message = f"Job {name} has not started yet."
|
||||||
|
elif status == 'blocked':
|
||||||
|
message = f"Job {name} will not run because one of its dependencies failed."
|
||||||
|
elif status == 'running':
|
||||||
|
message = f"Job {name} is still running."
|
||||||
|
elif status == 'failed':
|
||||||
|
message = f"Job {name} failed."
|
||||||
|
else:
|
||||||
|
message = f"Job {name} did not finish successfully. Current status: {status}."
|
||||||
|
|
||||||
|
super().__init__(message)
|
||||||
|
|
||||||
|
class JobMissing(APIHelperError):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
class InvalidResponse(APIHelperError):
|
class InvalidResponse(APIHelperError):
|
||||||
@ -145,13 +166,10 @@ class CircleCI:
|
|||||||
def job(self, workflow_id: str, name: str, require_success: bool = False) -> dict:
|
def job(self, workflow_id: str, name: str, require_success: bool = False) -> dict:
|
||||||
jobs = self.jobs(workflow_id)
|
jobs = self.jobs(workflow_id)
|
||||||
if name not in jobs:
|
if name not in jobs:
|
||||||
raise DataUnavailable(f"Job {name} is not present in the workflow.")
|
raise JobMissing(f"Job {name} is not present in the workflow.")
|
||||||
|
|
||||||
if require_success and jobs[name]['status'] != 'success':
|
if require_success and jobs[name]['status'] != 'success':
|
||||||
raise DataUnavailable(
|
raise JobNotSuccessful(name, jobs[name]['status'])
|
||||||
f"Job {name} has failed or is still running. "
|
|
||||||
f"Current status: {jobs[name]['status']}."
|
|
||||||
)
|
|
||||||
|
|
||||||
return jobs[name]
|
return jobs[name]
|
||||||
|
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
#!/usr/bin/env python3
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
from argparse import ArgumentParser, Namespace
|
from argparse import ArgumentParser, Namespace
|
||||||
|
from enum import Enum, unique
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
from typing import Mapping, Optional
|
from typing import Mapping, Optional
|
||||||
import sys
|
import sys
|
||||||
@ -13,10 +14,18 @@ SCRIPTS_DIR = Path(__file__).parent.parent
|
|||||||
sys.path.insert(0, str(SCRIPTS_DIR))
|
sys.path.insert(0, str(SCRIPTS_DIR))
|
||||||
|
|
||||||
from common.git_helpers import git_current_branch, git_commit_hash
|
from common.git_helpers import git_current_branch, git_commit_hash
|
||||||
from common.rest_api_helpers import APIHelperError, CircleCI, Github, download_file
|
from common.rest_api_helpers import APIHelperError, JobNotSuccessful, CircleCI, Github, download_file
|
||||||
# pragma pylint: enable=import-error,wrong-import-position
|
# pragma pylint: enable=import-error,wrong-import-position
|
||||||
|
|
||||||
|
|
||||||
|
@unique
|
||||||
|
class Status(Enum):
|
||||||
|
OK = 0 # Benchmarks downloaded successfully
|
||||||
|
ERROR = 1 # Error in the script, bad API response, unexpected data, etc.
|
||||||
|
NO_BENCHMARK = 2 # Benchmark collector job did not finish successfully and/or benchmark artifacts are missing.
|
||||||
|
PENDING = 3 # Benchmark collector job has not finished yet.
|
||||||
|
|
||||||
|
|
||||||
def process_commandline() -> Namespace:
|
def process_commandline() -> Namespace:
|
||||||
script_description = (
|
script_description = (
|
||||||
"Downloads benchmark results attached as artifacts to the c_ext_benchmarks job on CircleCI. "
|
"Downloads benchmark results attached as artifacts to the c_ext_benchmarks job on CircleCI. "
|
||||||
@ -76,14 +85,16 @@ def download_benchmark_artifact(
|
|||||||
commit_hash: str,
|
commit_hash: str,
|
||||||
overwrite: bool,
|
overwrite: bool,
|
||||||
silent: bool = False
|
silent: bool = False
|
||||||
):
|
) -> bool:
|
||||||
if not silent:
|
if not silent:
|
||||||
print(f"Downloading artifact: {benchmark_name}-{branch}-{commit_hash[:8]}.json.")
|
print(f"Downloading artifact: {benchmark_name}-{branch}-{commit_hash[:8]}.json.")
|
||||||
|
|
||||||
artifact_path = f'reports/externalTests/{benchmark_name}.json'
|
artifact_path = f'reports/externalTests/{benchmark_name}.json'
|
||||||
|
|
||||||
if artifact_path not in artifacts:
|
if artifact_path not in artifacts:
|
||||||
raise RuntimeError(f"Missing artifact: {artifact_path}.")
|
if not silent:
|
||||||
|
print(f"Missing artifact: {artifact_path}.")
|
||||||
|
return False
|
||||||
|
|
||||||
download_file(
|
download_file(
|
||||||
artifacts[artifact_path]['url'],
|
artifacts[artifact_path]['url'],
|
||||||
@ -91,6 +102,8 @@ def download_benchmark_artifact(
|
|||||||
overwrite,
|
overwrite,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
return True
|
||||||
|
|
||||||
|
|
||||||
def download_benchmarks(
|
def download_benchmarks(
|
||||||
branch: Optional[str],
|
branch: Optional[str],
|
||||||
@ -100,7 +113,7 @@ def download_benchmarks(
|
|||||||
overwrite: bool = False,
|
overwrite: bool = False,
|
||||||
debug_requests: bool = False,
|
debug_requests: bool = False,
|
||||||
silent: bool = False,
|
silent: bool = False,
|
||||||
):
|
) -> Status:
|
||||||
github = Github('ethereum/solidity', debug_requests)
|
github = Github('ethereum/solidity', debug_requests)
|
||||||
circleci = CircleCI('ethereum/solidity', debug_requests)
|
circleci = CircleCI('ethereum/solidity', debug_requests)
|
||||||
|
|
||||||
@ -141,32 +154,41 @@ def download_benchmarks(
|
|||||||
|
|
||||||
artifacts = circleci.artifacts(int(benchmark_collector_job['job_number']))
|
artifacts = circleci.artifacts(int(benchmark_collector_job['job_number']))
|
||||||
|
|
||||||
download_benchmark_artifact(artifacts, 'summarized-benchmarks', branch, actual_commit_hash, overwrite, silent)
|
got_summary = download_benchmark_artifact(artifacts, 'summarized-benchmarks', branch, actual_commit_hash, overwrite, silent)
|
||||||
download_benchmark_artifact(artifacts, 'all-benchmarks', branch, actual_commit_hash, overwrite, silent)
|
got_full = download_benchmark_artifact(artifacts, 'all-benchmarks', branch, actual_commit_hash, overwrite, silent)
|
||||||
|
|
||||||
|
return Status.OK if got_summary and got_full else Status.NO_BENCHMARK
|
||||||
|
|
||||||
|
|
||||||
def main():
|
def main():
|
||||||
try:
|
try:
|
||||||
options = process_commandline()
|
options = process_commandline()
|
||||||
download_benchmarks(
|
return download_benchmarks(
|
||||||
options.branch,
|
options.branch,
|
||||||
options.pull_request_id,
|
options.pull_request_id,
|
||||||
options.base_of_pr,
|
options.base_of_pr,
|
||||||
options.ignore_commit_hash,
|
options.ignore_commit_hash,
|
||||||
options.overwrite,
|
options.overwrite,
|
||||||
options.debug_requests,
|
options.debug_requests,
|
||||||
)
|
).value
|
||||||
|
except JobNotSuccessful as exception:
|
||||||
return 0
|
print(f"[ERROR] {exception}", file=sys.stderr)
|
||||||
|
if not exception.job_finished:
|
||||||
|
print("Please wait for the workflow to finish and try again.", file=sys.stderr)
|
||||||
|
return Status.PENDING.value
|
||||||
|
else:
|
||||||
|
print("Benchmarks from this run of the pipeline are not available.", file=sys.stderr)
|
||||||
|
return Status.NO_BENCHMARK.value
|
||||||
except APIHelperError as exception:
|
except APIHelperError as exception:
|
||||||
print(f"[ERROR] {exception}", file=sys.stderr)
|
print(f"[ERROR] {exception}", file=sys.stderr)
|
||||||
return 1
|
return Status.ERROR.value
|
||||||
except requests.exceptions.HTTPError as exception:
|
except requests.exceptions.HTTPError as exception:
|
||||||
print(f"[ERROR] {exception}", file=sys.stderr)
|
print(f"[ERROR] {exception}", file=sys.stderr)
|
||||||
return 1
|
return Status.ERROR.value
|
||||||
except RuntimeError as exception:
|
except RuntimeError as exception:
|
||||||
print(f"[ERROR] {exception}", file=sys.stderr)
|
print(f"[ERROR] {exception}", file=sys.stderr)
|
||||||
return 1
|
return Status.ERROR.value
|
||||||
|
|
||||||
|
|
||||||
if __name__ == '__main__':
|
if __name__ == '__main__':
|
||||||
sys.exit(main())
|
sys.exit(main())
|
||||||
|
Loading…
Reference in New Issue
Block a user