2020-10-02 16:18:33 +02:00
|
|
|
#!/usr/bin/env python3
|
|
|
|
# Copyright 2020 Google LLC
|
|
|
|
#
|
|
|
|
# Licensed under the the Apache License v2.0 with LLVM Exceptions (the "License");
|
|
|
|
# you may not use this file except in compliance with the License.
|
|
|
|
# You may obtain a copy of the License at
|
|
|
|
#
|
|
|
|
# https://llvm.org/LICENSE.txt
|
|
|
|
#
|
|
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
|
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
# See the License for the specific language governing permissions and
|
|
|
|
# limitations under the License.
|
|
|
|
|
2020-10-09 11:05:19 +02:00
|
|
|
import io
|
2020-10-02 16:18:33 +02:00
|
|
|
import json
|
2020-10-09 11:05:19 +02:00
|
|
|
import logging
|
2020-10-02 16:18:33 +02:00
|
|
|
import os
|
2021-08-11 16:20:03 +02:00
|
|
|
from typing import List, Set, Dict
|
2020-10-02 16:18:33 +02:00
|
|
|
|
2020-10-09 11:05:19 +02:00
|
|
|
from exec_utils import watch_shell
|
|
|
|
import yaml
|
2020-10-02 16:18:33 +02:00
|
|
|
|
|
|
|
|
|
|
|
def generic_linux(projects: str, check_diff: bool) -> List:
|
|
|
|
if os.getenv('ph_skip_linux') is not None:
|
|
|
|
return []
|
2020-12-09 17:23:01 +01:00
|
|
|
scripts_refspec = os.getenv("ph_scripts_refspec", "main")
|
2020-10-02 16:18:33 +02:00
|
|
|
no_cache = os.getenv('ph_no_cache') is not None
|
|
|
|
log_level = os.getenv('ph_log_level', 'WARNING')
|
|
|
|
linux_agents = {'queue': 'linux'}
|
|
|
|
t = os.getenv('ph_linux_agents')
|
|
|
|
if t is not None:
|
|
|
|
linux_agents = json.loads(t)
|
|
|
|
commands = [
|
|
|
|
'set -euo pipefail',
|
|
|
|
'ccache --clear' if no_cache else '',
|
|
|
|
'ccache --zero-stats',
|
|
|
|
'ccache --show-config',
|
|
|
|
'mkdir -p artifacts',
|
|
|
|
'dpkg -l >> artifacts/packages.txt',
|
2020-10-12 16:25:23 +02:00
|
|
|
*checkout_scripts('linux', scripts_refspec),
|
2020-10-02 16:18:33 +02:00
|
|
|
'set +e',
|
|
|
|
]
|
2020-10-09 12:50:44 +02:00
|
|
|
|
2020-10-02 16:18:33 +02:00
|
|
|
if check_diff:
|
|
|
|
commands.extend([
|
2021-10-04 16:44:40 +02:00
|
|
|
'$${SRC}/scripts/premerge_checks.py --check-clang-format '
|
2020-11-25 15:29:50 +01:00
|
|
|
f'--projects="{projects}" --log-level={log_level}',
|
2020-10-02 16:18:33 +02:00
|
|
|
])
|
|
|
|
else:
|
|
|
|
commands.extend([
|
2021-04-26 20:10:15 +02:00
|
|
|
f'$${{SRC}}/scripts/premerge_checks.py --projects="{projects}" --log-level={log_level}'
|
2020-10-02 16:18:33 +02:00
|
|
|
])
|
|
|
|
commands.extend([
|
2021-04-26 20:10:15 +02:00
|
|
|
'EXIT_STATUS=$$?',
|
2020-10-02 16:18:33 +02:00
|
|
|
'echo "--- ccache stats"',
|
|
|
|
'ccache --print-stats',
|
2021-04-26 20:10:15 +02:00
|
|
|
'exit $$EXIT_STATUS',
|
2020-10-02 16:18:33 +02:00
|
|
|
])
|
|
|
|
|
|
|
|
linux_buld_step = {
|
2020-11-25 15:29:50 +01:00
|
|
|
'label': ':linux: x64 debian',
|
2020-10-02 16:18:33 +02:00
|
|
|
'key': 'linux',
|
|
|
|
'commands': commands,
|
2020-11-25 15:29:50 +01:00
|
|
|
'artifact_paths': ['artifacts/**/*', '*_result.json', 'build/test-results.xml'],
|
2020-10-02 16:18:33 +02:00
|
|
|
'agents': linux_agents,
|
|
|
|
'timeout_in_minutes': 120,
|
|
|
|
'retry': {'automatic': [
|
|
|
|
{'exit_status': -1, 'limit': 2}, # Agent lost
|
|
|
|
{'exit_status': 255, 'limit': 2}, # Forced agent shutdown
|
|
|
|
]},
|
|
|
|
}
|
|
|
|
return [linux_buld_step]
|
|
|
|
|
|
|
|
|
2021-09-03 15:15:11 +02:00
|
|
|
def bazel(modified_files: Set[str], force: bool = False) -> List:
|
2021-07-28 15:50:57 +02:00
|
|
|
if os.getenv('ph_skip_bazel') is not None:
|
|
|
|
logging.info('bazel build is skipped as "ph_skip_bazel" is set')
|
|
|
|
return []
|
|
|
|
updated_build = any(s.startswith('utils/bazel/') for s in modified_files)
|
2021-09-03 15:15:11 +02:00
|
|
|
if not force:
|
|
|
|
if updated_build:
|
|
|
|
logging.info('files in utils/bazel/ modified, will trigger bazel build')
|
|
|
|
else:
|
|
|
|
user_projects = os.getenv('ph_user_project_slugs', '').split(',')
|
|
|
|
if 'bazel_build' not in user_projects:
|
|
|
|
logging.info('bazel build is skipped as "bazel_build" is not listed in user projects and no files in '
|
|
|
|
'utils/bazel/ are modified')
|
|
|
|
return []
|
2021-07-28 14:44:35 -07:00
|
|
|
agents = {'queue': 'llvm-bazel-premerge'}
|
2021-07-28 15:50:57 +02:00
|
|
|
t = os.getenv('ph_bazel_agents')
|
|
|
|
if t is not None:
|
|
|
|
agents = json.loads(t)
|
2021-10-04 16:44:40 +02:00
|
|
|
|
2021-07-28 15:50:57 +02:00
|
|
|
return [{
|
|
|
|
'label': ':bazel: bazel',
|
|
|
|
'key': 'bazel',
|
|
|
|
'commands': [
|
|
|
|
'set -eu',
|
|
|
|
'cd utils/bazel',
|
2022-05-12 14:49:02 -07:00
|
|
|
'bazel query //... + @llvm-project//... | xargs bazel test --config=generic_clang --config=rbe --copt=-Werror --host_copt=-Werror --test_output=errors --test_tag_filters=-nobuildkite --build_tag_filters=-nobuildkite',
|
2021-07-28 15:50:57 +02:00
|
|
|
],
|
|
|
|
'agents': agents,
|
|
|
|
'timeout_in_minutes': 120,
|
|
|
|
'retry': {'automatic': [
|
|
|
|
{'exit_status': -1, 'limit': 2}, # Agent lost
|
|
|
|
{'exit_status': 255, 'limit': 2}, # Forced agent shutdown
|
|
|
|
]},
|
|
|
|
}]
|
|
|
|
|
|
|
|
|
2020-10-02 16:18:33 +02:00
|
|
|
def generic_windows(projects: str) -> List:
|
|
|
|
if os.getenv('ph_skip_windows') is not None:
|
|
|
|
return []
|
2020-12-09 17:23:01 +01:00
|
|
|
scripts_refspec = os.getenv("ph_scripts_refspec", "main")
|
2020-10-02 16:18:33 +02:00
|
|
|
no_cache = os.getenv('ph_no_cache') is not None
|
|
|
|
log_level = os.getenv('ph_log_level', 'WARNING')
|
2021-04-26 20:10:15 +02:00
|
|
|
clear_sccache = 'powershell -command "sccache --stop-server; echo $$env:SCCACHE_DIR; ' \
|
|
|
|
'Remove-Item -Recurse -Force -ErrorAction Ignore $$env:SCCACHE_DIR; ' \
|
2020-10-02 16:18:33 +02:00
|
|
|
'sccache --start-server"'
|
|
|
|
win_agents = {'queue': 'windows'}
|
|
|
|
t = os.getenv('ph_windows_agents')
|
|
|
|
if t is not None:
|
|
|
|
win_agents = json.loads(t)
|
|
|
|
windows_buld_step = {
|
2020-11-25 15:29:50 +01:00
|
|
|
'label': ':windows: x64 windows',
|
2020-10-02 16:18:33 +02:00
|
|
|
'key': 'windows',
|
|
|
|
'commands': [
|
|
|
|
clear_sccache if no_cache else '',
|
|
|
|
'sccache --zero-stats',
|
2020-10-12 16:25:23 +02:00
|
|
|
*checkout_scripts('windows', scripts_refspec),
|
2020-10-02 16:18:33 +02:00
|
|
|
|
|
|
|
'powershell -command "'
|
2020-11-25 15:29:50 +01:00
|
|
|
f'%SRC%/scripts/premerge_checks.py --projects=\'{projects}\' --log-level={log_level}; '
|
2021-04-26 20:10:15 +02:00
|
|
|
'$$exit=$$?;'
|
2020-10-02 16:18:33 +02:00
|
|
|
'sccache --show-stats;'
|
2021-04-26 20:10:15 +02:00
|
|
|
'if ($$exit) {'
|
2020-10-02 16:18:33 +02:00
|
|
|
' echo success;'
|
|
|
|
' exit 0; } '
|
|
|
|
'else {'
|
|
|
|
' echo failure;'
|
|
|
|
' exit 1;'
|
|
|
|
'}"',
|
|
|
|
],
|
2020-11-25 15:29:50 +01:00
|
|
|
'artifact_paths': ['artifacts/**/*', '*_result.json', 'build/test-results.xml'],
|
2020-10-02 16:18:33 +02:00
|
|
|
'agents': win_agents,
|
2021-06-01 15:26:00 +02:00
|
|
|
'timeout_in_minutes': 150,
|
2020-10-02 16:18:33 +02:00
|
|
|
'retry': {'automatic': [
|
|
|
|
{'exit_status': -1, 'limit': 2}, # Agent lost
|
|
|
|
{'exit_status': 255, 'limit': 2}, # Forced agent shutdown
|
|
|
|
]},
|
|
|
|
}
|
|
|
|
return [windows_buld_step]
|
2020-10-09 11:05:19 +02:00
|
|
|
|
|
|
|
|
2021-08-11 16:20:03 +02:00
|
|
|
def from_shell_output(command, **kwargs) -> []:
|
2020-10-09 11:05:19 +02:00
|
|
|
"""
|
|
|
|
Executes shell command and parses stdout as multidoc yaml file, see
|
|
|
|
https://buildkite.com/docs/agent/v3/cli-pipeline#pipeline-format.
|
|
|
|
:param command: command, may include env variables
|
|
|
|
:return: all 'steps' that defined in the result ("env" section is ignored).
|
|
|
|
Non-zero exit code and malformed YAML produces empty result.
|
|
|
|
"""
|
|
|
|
path = os.path.expandvars(command)
|
|
|
|
logging.debug(f'invoking "{path}"')
|
|
|
|
out = io.BytesIO()
|
|
|
|
err = io.BytesIO()
|
2021-08-11 16:20:03 +02:00
|
|
|
rc = watch_shell(out.write, err.write, path, **kwargs)
|
2020-10-09 11:05:19 +02:00
|
|
|
logging.debug(f'exit code: {rc}, stdout: "{out.getvalue().decode()}", stderr: "{err.getvalue().decode()}"')
|
|
|
|
steps = []
|
|
|
|
if rc != 0:
|
|
|
|
logging.error(
|
|
|
|
f'{path} returned non-zero code {rc}, stdout: "{out.getvalue().decode()}", stderr: "{err.getvalue().decode()}"')
|
|
|
|
return steps
|
|
|
|
try:
|
|
|
|
for part in yaml.safe_load_all(out.getvalue()):
|
|
|
|
part.setdefault('steps', [])
|
|
|
|
steps.extend(part['steps'])
|
|
|
|
except yaml.YAMLError as e:
|
|
|
|
logging.error(f'''"{path}" produced malformed YAML, exception:
|
|
|
|
{e}
|
|
|
|
|
|
|
|
stdout: >>>{out.getvalue().decode()}>>>''')
|
|
|
|
return steps
|
2020-10-12 16:25:23 +02:00
|
|
|
|
|
|
|
|
|
|
|
def checkout_scripts(target_os: str, scripts_refspec: str) -> []:
|
|
|
|
if target_os == 'windows':
|
|
|
|
return [
|
|
|
|
'set SRC=%BUILDKITE_BUILD_PATH%/llvm-premerge-checks',
|
|
|
|
'rm -rf %SRC%',
|
|
|
|
'git clone --depth 1 https://github.com/google/llvm-premerge-checks.git %SRC%',
|
|
|
|
'cd %SRC%',
|
|
|
|
f'git fetch origin "{scripts_refspec}":x',
|
|
|
|
'git checkout x',
|
|
|
|
'echo llvm-premerge-checks commit:',
|
|
|
|
'git rev-parse HEAD',
|
2020-11-25 15:29:50 +01:00
|
|
|
'pip install -q -r %SRC%/scripts/requirements.txt',
|
2020-10-12 16:25:23 +02:00
|
|
|
'cd %BUILDKITE_BUILD_CHECKOUT_PATH%',
|
|
|
|
]
|
|
|
|
return [
|
2021-04-26 20:10:15 +02:00
|
|
|
'export SRC=$${BUILDKITE_BUILD_PATH}/llvm-premerge-checks',
|
|
|
|
'rm -rf $${SRC}',
|
|
|
|
'git clone --depth 1 https://github.com/google/llvm-premerge-checks.git "$${SRC}"',
|
|
|
|
'cd $${SRC}',
|
2020-10-12 16:25:23 +02:00
|
|
|
f'git fetch origin "{scripts_refspec}":x',
|
|
|
|
'git checkout x',
|
|
|
|
'echo "llvm-premerge-checks commit"',
|
|
|
|
'git rev-parse HEAD',
|
2021-04-26 20:10:15 +02:00
|
|
|
'pip install -q -r $${SRC}/scripts/requirements.txt',
|
|
|
|
'cd "$$BUILDKITE_BUILD_CHECKOUT_PATH"',
|
2020-10-12 16:25:23 +02:00
|
|
|
]
|
2021-08-11 16:20:03 +02:00
|
|
|
|
|
|
|
|
|
|
|
def extend_dict(target: Dict, extra: Dict) -> Dict:
|
|
|
|
if not target:
|
|
|
|
return extra
|
|
|
|
for k in extra:
|
|
|
|
if k in target:
|
|
|
|
continue
|
|
|
|
target[k] = extra[k]
|
|
|
|
return target
|
|
|
|
|
|
|
|
|
|
|
|
def extend_steps_env(steps: List[Dict], env: Dict):
|
|
|
|
for s in steps:
|
|
|
|
if 'commands' in s:
|
|
|
|
s['env'] = extend_dict(s.get('env'), env)
|
|
|
|
if 'build' in s:
|
|
|
|
s['build']['env'] = extend_dict(s['build'].get('env'), env)
|