2023-12-18 08:07:22 +00:00
import argparse
2023-12-18 12:44:11 +00:00
import concurrent . futures
2023-12-18 08:07:22 +00:00
import json
2024-01-04 15:35:09 +00:00
import logging
2023-12-18 08:07:22 +00:00
import os
import re
import subprocess
import sys
2024-07-14 09:39:31 +00:00
import time
2024-06-04 16:59:10 +00:00
from dataclasses import dataclass
2024-02-06 12:39:34 +00:00
from pathlib import Path
2024-06-02 16:25:14 +00:00
from typing import Any , Dict , List , Optional
2023-12-18 08:07:22 +00:00
import docker_images_helper
2024-01-30 17:44:52 +00:00
import upload_result_helper
from build_check import get_release_or_pr
2024-09-18 20:04:50 +00:00
from ci_buddy import CIBuddy
from ci_cache import CiCache
2024-06-10 09:18:03 +00:00
from ci_config import CI
2024-05-16 16:23:35 +00:00
from ci_metadata import CiMetadata
2024-09-18 20:04:50 +00:00
from ci_settings import CiSettings
from ci_utils import GH , Envs , Utils
2024-01-30 17:44:52 +00:00
from clickhouse_helper import (
CiLogsCredentials ,
ClickHouseHelper ,
2024-02-27 12:59:01 +00:00
InsertException ,
2024-01-30 17:44:52 +00:00
get_instance_id ,
get_instance_type ,
prepare_tests_results_for_clickhouse ,
)
2023-12-18 12:45:15 +00:00
from commit_status_helper import (
CommitStatusData ,
2024-01-04 15:35:09 +00:00
RerunHelper ,
2023-12-18 12:45:15 +00:00
format_description ,
get_commit ,
2024-09-18 20:04:50 +00:00
get_commit_filtered_statuses ,
2024-01-04 15:35:09 +00:00
post_commit_status ,
2023-12-18 12:45:15 +00:00
set_status_comment ,
)
2024-06-02 16:25:14 +00:00
from digest_helper import DockerDigester
2024-09-18 20:04:50 +00:00
from env_helper import GITHUB_REPOSITORY , GITHUB_RUN_ID , IS_CI , REPO_COPY , TEMP_PATH
2023-12-18 08:07:22 +00:00
from get_robot_token import get_best_robot_token
2023-12-18 12:44:11 +00:00
from git_helper import GIT_PREFIX , Git
from git_helper import Runner as GitRunner
2024-04-17 12:56:00 +00:00
from github_helper import GitHub
2023-12-18 08:07:22 +00:00
from pr_info import PRInfo
2024-07-15 14:32:49 +00:00
from report import (
ERROR ,
2024-09-18 20:04:50 +00:00
FAIL ,
GITHUB_JOB_API_URL ,
JOB_FINISHED_TEST_NAME ,
JOB_STARTED_TEST_NAME ,
OK ,
2024-07-15 14:32:49 +00:00
PENDING ,
SUCCESS ,
BuildResult ,
JobReport ,
TestResult ,
)
2023-12-18 12:44:11 +00:00
from s3_helper import S3Helper
2024-08-06 19:39:41 +00:00
from stopwatch import Stopwatch
2024-09-18 20:04:50 +00:00
from tee_popen import TeePopen
2023-12-18 08:07:22 +00:00
from version_helper import get_version_from_repo
2024-03-11 18:49:45 +00:00
# pylint: disable=too-many-lines
2023-12-18 08:07:22 +00:00
def get_check_name ( check_name : str , batch : int , num_batches : int ) - > str :
res = check_name
if num_batches > 1 :
res = f " { check_name } [ { batch + 1 } / { num_batches } ] "
return res
def parse_args ( parser : argparse . ArgumentParser ) - > argparse . Namespace :
2024-05-21 10:57:05 +00:00
parser . add_argument (
" --cancel-previous-run " ,
action = " store_true " ,
help = " Action that cancels previous running PR workflow if PR added into the Merge Queue " ,
)
2024-05-24 09:36:28 +00:00
parser . add_argument (
" --set-pending-status " ,
action = " store_true " ,
help = " Action to set needed pending statuses in the beginning of CI workflow, e.g. for Sync wf " ,
)
2023-12-18 08:07:22 +00:00
parser . add_argument (
" --configure " ,
action = " store_true " ,
help = " Action that configures ci run. Calculates digests, checks job to be executed, generates json output " ,
)
2024-07-24 17:51:34 +00:00
parser . add_argument (
" --workflow " ,
default = " " ,
type = str ,
help = " Workflow Name, to be provided with --configure for workflow-specific CI runs " ,
)
2023-12-18 08:07:22 +00:00
parser . add_argument (
" --update-gh-statuses " ,
action = " store_true " ,
2024-05-21 10:57:05 +00:00
help = " Action that recreate success GH statuses for jobs that finished successfully in past and will be "
" skipped this time " ,
2023-12-18 08:07:22 +00:00
)
parser . add_argument (
" --pre " ,
action = " store_true " ,
2024-05-16 11:58:19 +00:00
help = " Action that executes prerequisites for the job provided in --job-name " ,
2023-12-18 08:07:22 +00:00
)
parser . add_argument (
" --run " ,
action = " store_true " ,
2024-05-21 10:57:05 +00:00
help = " Action that executes run action for specified --job-name. run_command must be configured for a given "
" job name. " ,
2023-12-18 08:07:22 +00:00
)
parser . add_argument (
" --post " ,
action = " store_true " ,
help = " Action that executes post actions for the job provided in --job-name " ,
)
parser . add_argument (
" --mark-success " ,
action = " store_true " ,
2023-12-18 16:06:10 +00:00
help = " Action that marks job provided in --job-name (with batch provided in --batch) as successful " ,
2023-12-18 08:07:22 +00:00
)
parser . add_argument (
" --job-name " ,
default = " " ,
type = str ,
help = " Job name as in config " ,
)
2024-01-04 15:35:09 +00:00
parser . add_argument (
" --run-command " ,
default = " " ,
type = str ,
help = " A run command to run in --run action. Will override run_command from a job config if any " ,
)
2023-12-18 08:07:22 +00:00
parser . add_argument (
" --batch " ,
default = - 1 ,
type = int ,
help = " Current batch number (required for --mark-success), -1 or omit for single-batch job " ,
)
parser . add_argument (
" --infile " ,
default = " " ,
type = str ,
help = " Input json file or json string with ci run config " ,
)
parser . add_argument (
" --outfile " ,
default = " " ,
type = str ,
required = False ,
2023-12-18 16:06:10 +00:00
help = " output file to write json result to, if not set - stdout " ,
2023-12-18 08:07:22 +00:00
)
parser . add_argument (
" --pretty " ,
action = " store_true " ,
default = False ,
2023-12-18 16:06:10 +00:00
help = " makes json output pretty formatted " ,
2023-12-18 08:07:22 +00:00
)
parser . add_argument (
" --skip-docker " ,
action = " store_true " ,
default = False ,
help = " skip fetching docker data from dockerhub, used in --configure action (for debugging) " ,
)
parser . add_argument (
" --docker-digest-or-latest " ,
action = " store_true " ,
default = False ,
help = " temporary hack to fallback to latest if image with digest as a tag is not on docker hub " ,
)
parser . add_argument (
" --skip-jobs " ,
action = " store_true " ,
default = False ,
2024-05-16 11:58:19 +00:00
help = " skip fetching data about job runs, used in --configure action (for debugging and nightly ci) " ,
2023-12-18 08:07:22 +00:00
)
2024-02-09 12:21:11 +00:00
parser . add_argument (
" --force " ,
action = " store_true " ,
default = False ,
help = " Used with --run, force the job to run, omitting the ci cache " ,
)
2024-02-02 17:10:47 +00:00
# FIXME: remove, not used
2023-12-18 08:07:22 +00:00
parser . add_argument (
" --rebuild-all-binaries " ,
action = " store_true " ,
default = False ,
2024-05-21 10:57:05 +00:00
help = " [DEPRECATED. to be removed, once no wf use it] will create run config without skipping build jobs in "
" any case, used in --configure action (for release branches) " ,
2023-12-18 08:07:22 +00:00
)
2024-01-04 15:35:09 +00:00
parser . add_argument (
" --commit-message " ,
default = " " ,
help = " debug option to test commit message processing " ,
)
2023-12-18 08:07:22 +00:00
return parser . parse_args ( )
2024-02-02 17:10:47 +00:00
# FIXME: rewrite the docker job as regular reusable_test job and move interaction with docker hub inside job script
# that way run config will be more clean, workflow more generic and less api calls to dockerhub
2023-12-18 08:07:22 +00:00
def check_missing_images_on_dockerhub (
image_name_tag : Dict [ str , str ] , arch : Optional [ str ] = None
) - > Dict [ str , str ] :
"""
Checks missing images on dockerhub .
Works concurrently for all given images .
Docker must be logged in .
"""
def run_docker_command (
image : str , image_digest : str , arch : Optional [ str ] = None
) - > Dict :
"""
aux command for fetching single docker manifest
"""
command = [
" docker " ,
" manifest " ,
" inspect " ,
f " { image } : { image_digest } " if not arch else f " { image } : { image_digest } - { arch } " ,
]
process = subprocess . run (
command ,
stdout = subprocess . PIPE ,
stderr = subprocess . PIPE ,
text = True ,
check = False ,
)
return {
" image " : image ,
" image_digest " : image_digest ,
" arch " : arch ,
" stdout " : process . stdout ,
" stderr " : process . stderr ,
" return_code " : process . returncode ,
}
result : Dict [ str , str ] = { }
with concurrent . futures . ThreadPoolExecutor ( ) as executor :
futures = [
executor . submit ( run_docker_command , image , tag , arch )
for image , tag in image_name_tag . items ( )
]
responses = [
future . result ( ) for future in concurrent . futures . as_completed ( futures )
]
for resp in responses :
name , stdout , stderr , digest , arch = (
resp [ " image " ] ,
resp [ " stdout " ] ,
resp [ " stderr " ] ,
resp [ " image_digest " ] ,
resp [ " arch " ] ,
)
if stderr :
if stderr . startswith ( " no such manifest " ) :
result [ name ] = digest
else :
print ( f " Error: Unknown error: { stderr } , { name } , { arch } " )
elif stdout :
if " mediaType " in stdout :
pass
else :
print ( f " Error: Unknown response: { stdout } " )
assert False , " FIXME "
else :
print ( f " Error: No response for { name } , { digest } , { arch } " )
assert False , " FIXME "
return result
2024-07-15 14:32:49 +00:00
def _pre_action ( s3 , job_name , batch , indata , pr_info ) :
no_cache = CiSettings . create_from_run_config ( indata ) . no_ci_cache
2024-07-11 11:37:26 +00:00
print ( " Clear dmesg " )
2024-07-13 18:15:47 +00:00
Utils . clear_dmesg ( )
2024-02-02 17:10:47 +00:00
CommitStatusData . cleanup ( )
JobReport . cleanup ( )
BuildResult . cleanup ( )
ci_cache = CiCache ( s3 , indata [ " jobs_data " ] [ " digests " ] )
2024-05-30 19:21:40 +00:00
# for release/master branches reports must be from the same branch
2024-05-31 14:40:23 +00:00
report_prefix = " "
if pr_info . is_master or pr_info . is_release :
2024-07-24 17:51:34 +00:00
# do not set report prefix for scheduled or dispatched wf (in case it started from feature branch while
# testing), otherwise reports won't be found
if not ( pr_info . is_scheduled or pr_info . is_dispatched ) :
2024-08-03 08:40:12 +00:00
report_prefix = Utils . normalize_string ( pr_info . head_ref )
2024-02-04 19:12:37 +00:00
print (
f " Use report prefix [ { report_prefix } ], pr_num [ { pr_info . number } ], head_ref [ { pr_info . head_ref } ] "
)
2024-02-02 17:10:47 +00:00
reports_files = ci_cache . download_build_reports ( file_prefix = report_prefix )
2024-05-31 08:57:31 +00:00
ci_cache . dump_run_config ( indata )
2024-07-15 14:32:49 +00:00
to_be_skipped = False
skip_status = SUCCESS
# check if job was run already
if CI . is_build_job ( job_name ) :
# this is a build job - check if a build report is present
build_result = (
BuildResult . load_any ( job_name , pr_info . number , pr_info . head_ref )
if not no_cache
else None
)
if build_result :
if build_result . status == SUCCESS :
to_be_skipped = True
else :
print (
" Build report found but status is unsuccessful - will try to rerun "
)
print ( " ::group::Build Report " )
print ( build_result . as_json ( ) )
print ( " ::endgroup:: " )
else :
# this is a test job - check if GH commit status or cache record is present
commit = get_commit ( GitHub ( get_best_robot_token ( ) , per_page = 100 ) , pr_info . sha )
# rerun helper check
# FIXME: Find a way to identify if job restarted manually (by developer) or by automatic workflow restart (died spot-instance)
# disable rerun check for the former
if job_name not in (
CI . JobNames . BUILD_CHECK ,
) : # we might want to rerun build report job
rerun_helper = RerunHelper ( commit , _get_ext_check_name ( job_name ) )
2024-09-03 09:44:25 +00:00
if rerun_helper . is_already_finished_by_status ( ) :
print (
f " WARNING: Rerunning job with GH status, rerun triggered by { Envs . GITHUB_ACTOR } "
)
2024-07-15 14:32:49 +00:00
status = rerun_helper . get_finished_status ( )
assert status
print ( " ::group::Commit Status " )
print ( status )
print ( " ::endgroup:: " )
to_be_skipped = True
skip_status = status . state
# ci cache check
2024-08-23 00:51:27 +00:00
if not to_be_skipped and not no_cache and not Utils . is_job_triggered_manually ( ) :
2024-07-15 14:32:49 +00:00
ci_cache = CiCache ( s3 , indata [ " jobs_data " ] [ " digests " ] ) . update ( )
job_config = CI . get_job_config ( job_name )
if ci_cache . is_successful (
job_name ,
batch ,
job_config . num_batches ,
job_config . required_on_release_branch ,
) :
print ( " CICache record has be found - job will be skipped " )
job_status = ci_cache . get_successful (
job_name , batch , job_config . num_batches
)
assert job_status , " BUG "
_create_gh_status (
commit ,
job_name ,
batch ,
job_config . num_batches ,
job_status ,
)
to_be_skipped = True
# skip_status = SUCCESS already there
2024-08-02 07:23:40 +00:00
GH . print_in_group ( " Commit Status Data " , job_status )
2024-07-15 14:32:49 +00:00
2024-08-06 19:39:41 +00:00
# create dummy report
jr = JobReport . create_dummy ( status = skip_status , job_skipped = to_be_skipped )
2024-07-15 14:32:49 +00:00
jr . dump ( )
if not to_be_skipped :
print ( " push start record to ci db " )
prepared_events = prepare_tests_results_for_clickhouse (
pr_info ,
[ TestResult ( JOB_STARTED_TEST_NAME , OK ) ] ,
SUCCESS ,
0.0 ,
JobReport . get_start_time_from_current ( ) ,
" " ,
_get_ext_check_name ( job_name ) ,
)
ClickHouseHelper ( ) . insert_events_into (
db = " default " , table = " checks " , events = prepared_events
)
2024-02-02 17:10:47 +00:00
print ( f " Pre action done. Report files [ { reports_files } ] have been downloaded " )
def _mark_success_action (
s3 : S3Helper ,
indata : Dict [ str , Any ] ,
pr_info : PRInfo ,
job : str ,
batch : int ,
) - > None :
ci_cache = CiCache ( s3 , indata [ " jobs_data " ] [ " digests " ] )
2024-06-10 09:18:03 +00:00
job_config = CI . get_job_config ( job )
2024-02-02 17:10:47 +00:00
num_batches = job_config . num_batches
# if batch is not provided - set to 0
batch = 0 if batch == - 1 else batch
assert (
0 < = batch < num_batches
) , f " --batch must be provided and in range [0, { num_batches } ) for { job } "
# FIXME: find generic design for propagating and handling job status (e.g. stop using statuses in GH api)
# now job ca be build job w/o status data, any other job that exit with 0 with or w/o status data
2024-06-10 09:18:03 +00:00
if CI . is_build_job ( job ) :
2024-02-29 15:39:27 +00:00
# there is no CommitStatus for build jobs
# create dummy status relying on JobReport
2024-02-02 17:10:47 +00:00
# FIXME: consider creating commit status for build jobs too, to treat everything the same way
2024-02-29 15:39:27 +00:00
job_report = JobReport . load ( ) if JobReport . exist ( ) else None
if job_report and job_report . status == SUCCESS :
CommitStatusData (
SUCCESS ,
" dummy description " ,
" dummy_url " ,
pr_num = pr_info . number ,
sha = pr_info . sha ,
) . dump_status ( )
2024-02-02 17:10:47 +00:00
job_status = None
if CommitStatusData . exist ( ) :
# normal scenario
job_status = CommitStatusData . load_status ( )
else :
# apparently exit after rerun-helper check
# do nothing, exit without failure
print ( f " ERROR: no status file for job [ { job } ] " )
2024-10-01 15:36:39 +00:00
if job_config . run_by_labels or not job_config . has_digest ( ) :
2024-07-13 10:48:48 +00:00
print ( f " Job [ { job } ] has no digest or run by label in CI - do not cache " )
2024-02-02 17:10:47 +00:00
else :
2024-04-17 20:23:41 +00:00
if pr_info . is_master :
2024-02-02 17:10:47 +00:00
pass
# delete method is disabled for ci_cache. need it?
# pending enabled for master branch jobs only
# ci_cache.delete_pending(job, batch, num_batches, release_branch=True)
if job_status and job_status . is_ok ( ) :
ci_cache . push_successful (
2024-06-02 16:25:14 +00:00
job , batch , num_batches , job_status , pr_info . is_release
2024-02-02 17:10:47 +00:00
)
print ( f " Job [ { job } ] is ok " )
2024-02-04 19:12:37 +00:00
elif job_status and not job_status . is_ok ( ) :
ci_cache . push_failed (
2024-06-02 16:25:14 +00:00
job , batch , num_batches , job_status , pr_info . is_release
2024-02-04 19:12:37 +00:00
)
print ( f " Job [ { job } ] is failed with status [ { job_status . status } ] " )
else :
job_status = CommitStatusData (
description = " dummy description " , status = ERROR , report_url = " dummy url "
)
ci_cache . push_failed (
2024-06-02 16:25:14 +00:00
job , batch , num_batches , job_status , pr_info . is_release
2024-02-04 19:12:37 +00:00
)
print ( f " No CommitStatusData for [ { job } ], push dummy failure to ci_cache " )
2024-02-02 17:10:47 +00:00
def _print_results ( result : Any , outfile : Optional [ str ] , pretty : bool = False ) - > None :
if outfile :
2024-02-26 17:46:15 +00:00
with open ( outfile , " w " , encoding = " utf-8 " ) as f :
2024-02-02 17:10:47 +00:00
if isinstance ( result , str ) :
print ( result , file = f )
elif isinstance ( result , dict ) :
print ( json . dumps ( result , indent = 2 if pretty else None ) , file = f )
else :
raise AssertionError ( f " Unexpected type for ' res ' : { type ( result ) } " )
else :
if isinstance ( result , str ) :
print ( result )
elif isinstance ( result , dict ) :
print ( json . dumps ( result , indent = 2 if pretty else None ) )
else :
raise AssertionError ( f " Unexpected type for ' res ' : { type ( result ) } " )
2024-02-05 17:09:43 +00:00
def _configure_docker_jobs ( docker_digest_or_latest : bool ) - > Dict :
2024-01-04 15:35:09 +00:00
print ( " ::group::Docker images check " )
2023-12-18 08:07:22 +00:00
# generate docker jobs data
docker_digester = DockerDigester ( )
imagename_digest_dict = (
docker_digester . get_all_digests ( )
) # 'image name - digest' mapping
images_info = docker_images_helper . get_images_info ( )
2024-02-05 17:09:43 +00:00
# FIXME: we need login as docker manifest inspect goes directly to one of the *.docker.com hosts instead of "registry-mirrors" : ["http://dockerhub-proxy.dockerhub-proxy-zone:5000"]
# find if it's possible to use the setting of /etc/docker/daemon.json (https://github.com/docker/cli/issues/4484#issuecomment-1688095463)
docker_images_helper . docker_login ( )
missing_multi_dict = check_missing_images_on_dockerhub ( imagename_digest_dict )
missing_multi = list ( missing_multi_dict )
missing_amd64 = [ ]
missing_aarch64 = [ ]
if not docker_digest_or_latest :
2024-05-16 11:58:19 +00:00
# look for missing arm and amd images only among missing multi-arch manifests @missing_multi_dict
2024-02-05 17:09:43 +00:00
# to avoid extra dockerhub api calls
missing_amd64 = list (
check_missing_images_on_dockerhub ( missing_multi_dict , " amd64 " )
)
# FIXME: WA until full arm support: skip not supported arm images
missing_aarch64 = list (
check_missing_images_on_dockerhub (
{
im : digest
for im , digest in missing_multi_dict . items ( )
if not images_info [ im ] [ " only_amd64 " ]
} ,
" aarch64 " ,
2023-12-18 08:07:22 +00:00
)
2024-02-05 17:09:43 +00:00
)
2023-12-18 08:07:22 +00:00
else :
2024-02-05 17:09:43 +00:00
if missing_multi :
assert False , f " Missing images [ { missing_multi } ], cannot proceed "
2024-01-04 15:35:09 +00:00
print ( " ::endgroup:: " )
2023-12-18 08:07:22 +00:00
return {
" images " : imagename_digest_dict ,
" missing_aarch64 " : missing_aarch64 ,
" missing_amd64 " : missing_amd64 ,
" missing_multi " : missing_multi ,
}
def _configure_jobs (
s3 : S3Helper ,
2024-02-02 17:10:47 +00:00
pr_info : PRInfo ,
2024-06-02 16:25:14 +00:00
ci_settings : CiSettings ,
2024-06-07 14:37:05 +00:00
skip_jobs : bool ,
2024-07-24 17:51:34 +00:00
workflow_name : str = " " ,
2024-06-10 09:18:03 +00:00
dry_run : bool = False ,
2024-06-02 16:25:14 +00:00
) - > CiCache :
"""
returns CICache instance with configured job ' s data
: param s3 :
: param pr_info :
: param ci_settings :
: return :
"""
2023-12-18 08:07:22 +00:00
2024-06-02 16:25:14 +00:00
# get all jobs
2024-06-07 14:37:05 +00:00
if not skip_jobs :
2024-06-10 09:18:03 +00:00
job_configs = CI . get_workflow_jobs_with_configs (
2024-06-07 14:37:05 +00:00
is_mq = pr_info . is_merge_queue ,
is_docs_only = pr_info . has_changes_in_documentation_only ( ) ,
is_master = pr_info . is_master ,
2024-06-10 09:18:03 +00:00
is_pr = pr_info . is_pr ,
2024-07-24 17:51:34 +00:00
workflow_name = workflow_name ,
2024-06-07 14:37:05 +00:00
)
else :
job_configs = { }
2024-02-21 14:55:48 +00:00
2024-07-24 17:51:34 +00:00
if not workflow_name :
# filter jobs in accordance with ci settings
job_configs = ci_settings . apply (
job_configs ,
pr_info . is_release ,
is_pr = pr_info . is_pr ,
is_mq = pr_info . is_merge_queue ,
labels = pr_info . labels ,
)
# add all job batches to job's to_do batches
for _job , job_config in job_configs . items ( ) :
batches = [ ]
for batch in range ( job_config . num_batches ) :
batches . append ( batch )
job_config . batches = batches
2024-02-21 14:55:48 +00:00
2024-06-02 16:25:14 +00:00
# check jobs in ci cache
ci_cache = CiCache . calc_digests_and_create (
2024-06-07 14:37:05 +00:00
s3 ,
job_configs ,
2024-06-10 09:18:03 +00:00
cache_enabled = not ci_settings . no_ci_cache and not skip_jobs and IS_CI ,
dry_run = dry_run ,
2024-03-26 14:43:53 +00:00
)
2024-06-02 16:25:14 +00:00
ci_cache . update ( )
ci_cache . apply ( job_configs , is_release = pr_info . is_release )
2024-02-07 19:36:28 +00:00
2024-06-02 16:25:14 +00:00
return ci_cache
2023-12-18 08:07:22 +00:00
2024-06-18 11:17:42 +00:00
def _generate_ci_stage_config (
jobs_data : Dict [ str , Any ] , non_blocking_mode : bool = False
) - > Dict [ str , Dict [ str , Any ] ] :
2024-02-26 14:52:05 +00:00
"""
populates GH Actions ' workflow with real jobs
2024-05-16 11:58:19 +00:00
" Builds_1 " : [ { " job_name " : NAME , " runner_type " : RUNNER_TYPE } ]
" Tests_1 " : [ { " job_name " : NAME , " runner_type " : RUNNER_TYPE } ]
2024-02-26 14:52:05 +00:00
. . .
"""
result = { } # type: Dict[str, Any]
stages_to_do = [ ]
2024-06-02 16:25:14 +00:00
for job in jobs_data :
2024-06-18 11:17:42 +00:00
stage_type = CI . get_job_ci_stage ( job , non_blocking_ci = non_blocking_mode )
2024-06-10 09:18:03 +00:00
if stage_type == CI . WorkflowStages . NA :
2024-02-26 14:52:05 +00:00
continue
if stage_type not in result :
result [ stage_type ] = [ ]
stages_to_do . append ( stage_type )
result [ stage_type ] . append (
2024-06-10 09:18:03 +00:00
{ " job_name " : job , " runner_type " : CI . JOB_CONFIGS [ job ] . runner_type }
2024-02-26 14:52:05 +00:00
)
result [ " stages_to_do " ] = stages_to_do
return result
2024-02-04 19:12:37 +00:00
def _create_gh_status (
commit : Any , job : str , batch : int , num_batches : int , job_status : CommitStatusData
) - > None :
print ( f " Going to re-create GH status for job [ { job } ] " )
assert job_status . status == SUCCESS , " BUG! "
commit . create_status (
state = job_status . status ,
target_url = job_status . report_url ,
description = format_description (
f " Reused from [ { job_status . pr_num } - { job_status . sha [ 0 : 8 ] } ]: "
f " { job_status . description } "
) ,
context = get_check_name ( job , batch = batch , num_batches = num_batches ) ,
)
2024-02-02 17:10:47 +00:00
def _update_gh_statuses_action ( indata : Dict , s3 : S3Helper ) - > None :
2024-06-02 16:25:14 +00:00
if CiSettings . create_from_run_config ( indata ) . no_ci_cache :
2024-01-04 15:35:09 +00:00
print ( " CI cache is disabled - skip restoring commit statuses from CI cache " )
return
2023-12-18 08:07:22 +00:00
job_digests = indata [ " jobs_data " ] [ " digests " ]
2024-02-04 19:12:37 +00:00
jobs_to_skip = indata [ " jobs_data " ] [ " jobs_to_skip " ]
jobs_to_do = indata [ " jobs_data " ] [ " jobs_to_do " ]
ci_cache = CiCache ( s3 , job_digests ) . update ( ) . fetch_records_data ( ) . print_status ( )
2024-02-02 17:10:47 +00:00
2023-12-18 08:07:22 +00:00
# create GH status
pr_info = PRInfo ( )
2024-04-17 12:56:00 +00:00
commit = get_commit ( GitHub ( get_best_robot_token ( ) , per_page = 100 ) , pr_info . sha )
2023-12-18 08:07:22 +00:00
2024-02-04 19:12:37 +00:00
def _concurrent_create_status ( job : str , batch : int , num_batches : int ) - > None :
2024-02-02 17:10:47 +00:00
job_status = ci_cache . get_successful ( job , batch , num_batches )
if not job_status :
return
2024-02-04 19:12:37 +00:00
_create_gh_status ( commit , job , batch , num_batches , job_status )
2023-12-18 08:07:22 +00:00
with concurrent . futures . ThreadPoolExecutor ( ) as executor :
futures = [ ]
for job in job_digests :
2024-02-08 19:09:28 +00:00
if job not in jobs_to_skip and job not in jobs_to_do :
2024-02-04 19:12:37 +00:00
# no need to create status for job that are not supposed to be executed
continue
2024-06-10 09:18:03 +00:00
if CI . is_build_job ( job ) :
2023-12-18 08:07:22 +00:00
# no GH status for build jobs
continue
2024-06-10 09:18:03 +00:00
job_config = CI . get_job_config ( job )
2024-01-29 17:13:32 +00:00
if not job_config :
# there might be a new job that does not exist on this branch - skip it
continue
for batch in range ( job_config . num_batches ) :
2024-02-04 19:12:37 +00:00
future = executor . submit (
2024-01-29 17:13:32 +00:00
_concurrent_create_status , job , batch , job_config . num_batches
2024-02-04 19:12:37 +00:00
)
2023-12-18 08:07:22 +00:00
futures . append ( future )
done , _ = concurrent . futures . wait ( futures )
for future in done :
try :
_ = future . result ( )
except Exception as e :
raise e
print ( " Going to update overall CI report " )
2024-07-14 09:39:31 +00:00
for retry in range ( 2 ) :
try :
set_status_comment ( commit , pr_info )
break
except Exception as e :
print (
f " WARNING: Failed to update CI Running status, attempt [ { retry + 1 } ], exception [ { e } ] "
)
time . sleep ( 1 )
else :
print ( " ERROR: All retry attempts failed. " )
2023-12-18 08:07:22 +00:00
print ( " ... CI report update - done " )
2024-03-21 17:20:08 +00:00
def _fetch_commit_tokens ( message : str , pr_info : PRInfo ) - > List [ str ] :
2024-03-21 22:20:33 +00:00
pattern = r " (#|- \ [x \ ] +<!---)( \ w+) "
2024-03-21 17:20:08 +00:00
matches = [ match [ - 1 ] for match in re . findall ( pattern , message ) ]
2024-02-07 19:36:28 +00:00
res = [
match
for match in matches
2024-06-10 09:18:03 +00:00
if match in CI . Tags or match . startswith ( " job_ " ) or match . startswith ( " batch_ " )
2024-02-07 19:36:28 +00:00
]
2024-05-16 11:58:19 +00:00
print ( f " CI modifiers from commit message: [ { res } ] " )
2024-03-21 17:20:08 +00:00
res_2 = [ ]
2024-04-17 20:23:41 +00:00
if pr_info . is_pr :
2024-03-21 17:20:08 +00:00
matches = [ match [ - 1 ] for match in re . findall ( pattern , pr_info . body ) ]
res_2 = [
match
for match in matches
2024-06-10 09:18:03 +00:00
if match in CI . Tags
2024-04-24 15:28:48 +00:00
or match . startswith ( " job_ " )
or match . startswith ( " batch_ " )
2024-03-21 17:20:08 +00:00
]
2024-05-16 11:58:19 +00:00
print ( f " CI modifiers from PR body: [ { res_2 } ] " )
2024-03-21 17:20:08 +00:00
return list ( set ( res + res_2 ) )
2023-12-18 08:07:22 +00:00
2024-01-04 15:35:09 +00:00
def _upload_build_artifacts (
pr_info : PRInfo ,
build_name : str ,
2024-02-02 17:10:47 +00:00
ci_cache : CiCache ,
2024-01-04 15:35:09 +00:00
job_report : JobReport ,
s3 : S3Helper ,
s3_destination : str ,
2024-05-30 19:21:40 +00:00
upload_binary : bool ,
2024-01-04 15:35:09 +00:00
) - > str :
# There are ugly artifacts for the performance test. FIXME:
s3_performance_path = " / " . join (
(
get_release_or_pr ( pr_info , get_version_from_repo ( ) ) [ 1 ] ,
pr_info . sha ,
2024-08-03 08:40:12 +00:00
Utils . normalize_string ( build_name ) ,
2024-01-04 15:35:09 +00:00
" performance.tar.zst " ,
)
)
performance_urls = [ ]
assert job_report . build_dir_for_upload , " Must be set for build job "
performance_path = Path ( job_report . build_dir_for_upload ) / " performance.tar.zst "
2024-05-30 19:21:40 +00:00
if upload_binary :
if performance_path . exists ( ) :
performance_urls . append (
s3 . upload_build_file_to_s3 ( performance_path , s3_performance_path )
)
print (
" Uploaded performance.tar.zst to %s , now delete to avoid duplication " ,
performance_urls [ 0 ] ,
)
performance_path . unlink ( )
build_urls = (
s3 . upload_build_directory_to_s3 (
Path ( job_report . build_dir_for_upload ) ,
s3_destination ,
keep_dirs_in_s3_path = False ,
upload_symlinks = False ,
)
+ performance_urls
2024-01-04 15:35:09 +00:00
)
2024-05-30 19:21:40 +00:00
print ( " ::notice ::Build URLs: {} " . format ( " \n " . join ( build_urls ) ) )
else :
build_urls = [ ]
print ( " ::notice ::No binaries will be uploaded for this job " )
2024-01-04 15:35:09 +00:00
log_path = Path ( job_report . additional_files [ 0 ] )
log_url = " "
if log_path . exists ( ) :
log_url = s3 . upload_build_file_to_s3 (
log_path , s3_destination + " / " + log_path . name
)
print ( f " ::notice ::Log URL: { log_url } " )
2024-05-30 19:21:40 +00:00
# generate and upload a build report
2024-01-04 15:35:09 +00:00
build_result = BuildResult (
build_name ,
log_url ,
build_urls ,
job_report . version ,
job_report . status ,
int ( job_report . duration ) ,
GITHUB_JOB_API_URL ( ) ,
head_ref = pr_info . head_ref ,
2024-07-29 19:07:24 +00:00
# PRInfo fetches pr number for release branches as well - set pr_number to 0 for release
# so that build results are not mistakenly treated as feature branch builds
pr_number = pr_info . number if pr_info . is_pr else 0 ,
2024-01-04 15:35:09 +00:00
)
2024-02-02 17:10:47 +00:00
report_url = ci_cache . upload_build_report ( build_result )
print ( f " Report file has been uploaded to [ { report_url } ] " )
2024-01-04 15:35:09 +00:00
2024-05-16 11:58:19 +00:00
# Upload master head's binaries
2024-06-10 09:18:03 +00:00
static_bin_name = CI . get_build_config ( build_name ) . static_binary_name
2024-04-17 20:23:41 +00:00
if pr_info . is_master and static_bin_name :
2024-01-04 15:35:09 +00:00
# Full binary with debug info:
s3_path_full = " / " . join ( ( pr_info . base_ref , static_bin_name , " clickhouse-full " ) )
binary_full = Path ( job_report . build_dir_for_upload ) / " clickhouse "
url_full = s3 . upload_build_file_to_s3 ( binary_full , s3_path_full )
print ( f " ::notice ::Binary static URL (with debug info): { url_full } " )
# Stripped binary without debug info:
s3_path_compact = " / " . join ( ( pr_info . base_ref , static_bin_name , " clickhouse " ) )
binary_compact = Path ( job_report . build_dir_for_upload ) / " clickhouse-stripped "
url_compact = s3 . upload_build_file_to_s3 ( binary_compact , s3_path_compact )
print ( f " ::notice ::Binary static URL (compact): { url_compact } " )
return log_url
def _upload_build_profile_data (
pr_info : PRInfo ,
build_name : str ,
job_report : JobReport ,
git_runner : GitRunner ,
ch_helper : ClickHouseHelper ,
) - > None :
ci_logs_credentials = CiLogsCredentials ( Path ( " /dev/null " ) )
2024-06-04 13:25:55 +00:00
if not ci_logs_credentials . host :
2024-06-04 13:37:26 +00:00
logging . info ( " Unknown CI logs host, skip uploading build profile data " )
return
2024-01-04 15:35:09 +00:00
2024-06-04 13:37:26 +00:00
if not pr_info . number == 0 :
logging . info ( " Skipping uploading build profile data for PRs " )
2024-06-04 13:25:55 +00:00
return
2024-01-04 15:35:09 +00:00
2024-06-04 13:25:55 +00:00
instance_type = get_instance_type ( )
instance_id = get_instance_id ( )
auth = {
" X-ClickHouse-User " : " ci " ,
" X-ClickHouse-Key " : ci_logs_credentials . password ,
}
url = f " https:// { ci_logs_credentials . host } / "
profiles_dir = Path ( TEMP_PATH ) / " profiles_source "
profiles_dir . mkdir ( parents = True , exist_ok = True )
print (
" Processing profile JSON files from %s " ,
Path ( REPO_COPY ) / " build_docker " ,
)
git_runner (
" ./utils/prepare-time-trace/prepare-time-trace.sh "
f " build_docker { profiles_dir . absolute ( ) } "
)
profile_data_file = Path ( TEMP_PATH ) / " profile.json "
with open ( profile_data_file , " wb " ) as profile_fd :
for profile_source in profiles_dir . iterdir ( ) :
if profile_source . name not in (
" binary_sizes.txt " ,
" binary_symbols.txt " ,
) :
with open ( profiles_dir / profile_source , " rb " ) as ps_fd :
profile_fd . write ( ps_fd . read ( ) )
2024-01-04 15:35:09 +00:00
2024-06-04 13:25:55 +00:00
@dataclass
class FileQuery :
file : Path
query : str
profile_query = f """ INSERT INTO build_time_trace
(
pull_request_number ,
commit_sha ,
check_start_time ,
check_name ,
instance_type ,
instance_id ,
file ,
library ,
time ,
pid ,
tid ,
ph ,
ts ,
dur ,
cat ,
name ,
detail ,
count ,
avgMs ,
args_name
)
SELECT { pr_info . number } , ' {pr_info.sha} ' , ' {job_report.start_time} ' , ' {build_name} ' , ' {instance_type} ' , ' {instance_id} ' , *
FROM input ( '
file String ,
library String ,
time DateTime64 ( 6 ) ,
pid UInt32 ,
tid UInt32 ,
ph String ,
ts UInt64 ,
dur UInt64 ,
cat String ,
name String ,
detail String ,
count UInt64 ,
avgMs UInt64 ,
args_name String ' )
FORMAT JSONCompactEachRow """
binary_sizes_query = f """ INSERT INTO binary_sizes
(
pull_request_number ,
commit_sha ,
check_start_time ,
check_name ,
instance_type ,
instance_id ,
file ,
size
)
SELECT { pr_info . number } , ' {pr_info.sha} ' , ' {job_report.start_time} ' , ' {build_name} ' , ' {instance_type} ' , ' {instance_id} ' , file , size
FROM input ( ' size UInt64, file String ' )
SETTINGS format_regexp = ' ^ \\ s*( \\ d+) (.+)$ '
FORMAT Regexp """
binary_symbols_query = f """ INSERT INTO binary_symbols
(
pull_request_number ,
commit_sha ,
check_start_time ,
check_name ,
instance_type ,
instance_id ,
file ,
address ,
size ,
type ,
symbol
)
SELECT { pr_info . number } , ' {pr_info.sha} ' , ' {job_report.start_time} ' , ' {build_name} ' , ' {instance_type} ' , ' {instance_id} ' ,
file , reinterpretAsUInt64 ( reverse ( unhex ( address ) ) ) , reinterpretAsUInt64 ( reverse ( unhex ( size ) ) ) , type , symbol
FROM input ( ' file String, address String, size String, type String, symbol String ' )
SETTINGS format_regexp = ' ^([^ ]+) ([0-9a-fA-F]+)(?: ([0-9a-fA-F]+))? (.) (.+)$ '
FORMAT Regexp """
files_queries = (
FileQuery (
2024-01-04 15:35:09 +00:00
profile_data_file ,
2024-06-04 13:25:55 +00:00
profile_query ,
) ,
FileQuery (
profiles_dir / " binary_sizes.txt " ,
binary_sizes_query ,
) ,
FileQuery (
profiles_dir / " binary_symbols.txt " ,
binary_symbols_query ,
) ,
)
for fq in files_queries :
logging . info (
2024-06-04 13:37:26 +00:00
" Uploading profile data, path: %s , size: %s , query: \n %s " ,
2024-06-04 13:25:55 +00:00
fq . file ,
fq . file . stat ( ) . st_size ,
fq . query ,
2024-01-04 15:35:09 +00:00
)
2024-02-27 12:59:01 +00:00
try :
2024-06-04 13:25:55 +00:00
ch_helper . insert_file ( url , auth , fq . query , fq . file , timeout = 5 )
2024-02-27 12:59:01 +00:00
except InsertException :
logging . error ( " Failed to insert profile data for the build, continue " )
2024-01-04 15:35:09 +00:00
2024-03-15 18:08:41 +00:00
def _add_build_to_version_history (
pr_info : PRInfo ,
job_report : JobReport ,
version : str ,
2024-03-18 17:45:23 +00:00
docker_tag : str ,
2024-03-15 18:08:41 +00:00
ch_helper : ClickHouseHelper ,
) - > None :
# with some probability we will not silently break this logic
2024-03-25 14:09:04 +00:00
assert pr_info . sha and pr_info . commit_html_url and pr_info . head_ref and version
2024-03-15 18:08:41 +00:00
data = {
" check_start_time " : job_report . start_time ,
" pull_request_number " : pr_info . number ,
" pull_request_url " : pr_info . pr_html_url ,
" commit_sha " : pr_info . sha ,
" commit_url " : pr_info . commit_html_url ,
" version " : version ,
2024-03-18 17:45:23 +00:00
" docker_tag " : docker_tag ,
2024-03-25 14:09:04 +00:00
" git_ref " : pr_info . head_ref ,
2024-03-15 18:08:41 +00:00
}
2024-03-26 15:09:16 +00:00
print ( f " ::notice ::Log Adding record to versions history: { data } " )
2024-03-15 18:08:41 +00:00
2024-03-25 14:09:04 +00:00
ch_helper . insert_event_into ( db = " default " , table = " version_history " , event = data )
2024-03-15 18:08:41 +00:00
2024-01-04 15:35:09 +00:00
def _run_test ( job_name : str , run_command : str ) - > int :
assert (
2024-06-10 09:18:03 +00:00
run_command or CI . get_job_config ( job_name ) . run_command
2024-01-04 15:35:09 +00:00
) , " Run command must be provided as input argument or be configured in job config "
2024-05-03 10:14:33 +00:00
env = os . environ . copy ( )
2024-06-10 09:18:03 +00:00
timeout = CI . get_job_config ( job_name ) . timeout or None
2024-03-28 14:00:04 +00:00
2024-01-04 15:35:09 +00:00
if not run_command :
run_command = " / " . join (
2024-06-10 09:18:03 +00:00
( os . path . dirname ( __file__ ) , CI . get_job_config ( job_name ) . run_command )
2024-01-04 15:35:09 +00:00
)
if " .py " in run_command and not run_command . startswith ( " python " ) :
run_command = " python3 " + run_command
print ( " Use run command from a job config " )
else :
print ( " Use run command from the workflow " )
2024-05-03 10:14:33 +00:00
env [ " CHECK_NAME " ] = job_name
2024-08-07 21:10:40 +00:00
env [ " MAX_RUN_TIME " ] = str ( timeout or 0 )
2024-01-04 15:35:09 +00:00
print ( f " Going to start run command [ { run_command } ] " )
2024-05-16 16:23:35 +00:00
stopwatch = Stopwatch ( )
job_log = Path ( TEMP_PATH ) / " job_log.txt "
with TeePopen ( run_command , job_log , env , timeout ) as process :
2024-08-06 19:39:41 +00:00
print ( f " Job process started, pid [ { process . process . pid } ] " )
2024-05-16 16:23:35 +00:00
retcode = process . wait ( )
if retcode != 0 :
print ( f " Run action failed for: [ { job_name } ] with exit code [ { retcode } ] " )
2024-08-06 19:39:41 +00:00
if process . timeout_exceeded :
print ( f " Job timed out: [ { job_name } ] exit code [ { retcode } ] " )
assert JobReport . exist ( ) , " JobReport real or dummy must be present "
jr = JobReport . load ( )
if jr . dummy :
print (
2024-09-26 14:05:34 +00:00
" ERROR: Run action failed with timeout and did not generate JobReport - update dummy report with execution time "
2024-08-06 19:39:41 +00:00
)
jr . test_results = [ TestResult . create_check_timeout_expired ( ) ]
jr . duration = stopwatch . duration_seconds
2024-08-15 11:11:10 +00:00
jr . additional_files + = [ job_log ]
2024-05-16 16:23:35 +00:00
print ( f " Run action done for: [ { job_name } ] " )
return retcode
2024-01-04 15:35:09 +00:00
def _get_ext_check_name ( check_name : str ) - > str :
run_by_hash_num = int ( os . getenv ( " RUN_BY_HASH_NUM " , " 0 " ) )
run_by_hash_total = int ( os . getenv ( " RUN_BY_HASH_TOTAL " , " 0 " ) )
if run_by_hash_total > 1 :
check_name_with_group = (
check_name + f " [ { run_by_hash_num + 1 } / { run_by_hash_total } ] "
)
else :
check_name_with_group = check_name
return check_name_with_group
2024-08-02 18:28:38 +00:00
def _cancel_pr_workflow (
s3 : S3Helper , pr_number : int , cancel_sync : bool = False
) - > None :
2024-05-23 14:36:24 +00:00
wf_data = CiMetadata ( s3 , pr_number ) . fetch_meta ( )
if not cancel_sync :
if not wf_data . run_id :
print ( f " ERROR: FIX IT: Run id has not been found PR [ { pr_number } ]! " )
else :
print (
f " Canceling PR workflow run_id: [ { wf_data . run_id } ], pr: [ { pr_number } ] "
)
2024-05-23 20:35:21 +00:00
GitHub . cancel_wf ( GITHUB_REPOSITORY , wf_data . run_id , get_best_robot_token ( ) )
2024-05-21 10:57:05 +00:00
else :
2024-05-23 14:36:24 +00:00
if not wf_data . sync_pr_run_id :
print ( " WARNING: Sync PR run id has not been found " )
else :
print ( f " Canceling sync PR workflow run_id: [ { wf_data . sync_pr_run_id } ] " )
GitHub . cancel_wf (
" ClickHouse/clickhouse-private " ,
wf_data . sync_pr_run_id ,
2024-05-23 20:35:21 +00:00
get_best_robot_token ( ) ,
2024-05-23 14:36:24 +00:00
)
2024-05-21 10:57:05 +00:00
2024-05-24 09:36:28 +00:00
def _set_pending_statuses ( pr_info : PRInfo ) - > None :
commit = get_commit ( GitHub ( get_best_robot_token ( ) , per_page = 100 ) , pr_info . sha )
try :
2024-06-17 06:33:31 +00:00
found = False
statuses = get_commit_filtered_statuses ( commit )
for commit_status in statuses :
if commit_status . context == CI . StatusNames . SYNC :
print (
f " Sync status found [ { commit_status . state } ], [ { commit_status . description } ] - won ' t be overwritten "
)
found = True
break
if not found :
print ( " Set Sync status to pending " )
commit . create_status (
state = PENDING ,
target_url = " " ,
description = CI . SyncState . PENDING ,
context = CI . StatusNames . SYNC ,
)
2024-05-24 09:36:28 +00:00
except Exception as ex :
print ( f " ERROR: failed to set GH commit status, ex: { ex } " )
2023-12-18 08:07:22 +00:00
def main ( ) - > int :
2024-01-04 15:35:09 +00:00
logging . basicConfig ( level = logging . INFO )
2023-12-18 08:07:22 +00:00
exit_code = 0
parser = argparse . ArgumentParser (
formatter_class = argparse . ArgumentDefaultsHelpFormatter ,
)
args = parse_args ( parser )
2024-01-22 08:26:46 +00:00
if args . mark_success or args . pre or args . run :
2023-12-18 08:07:22 +00:00
assert args . infile , " Run config must be provided via --infile "
assert args . job_name , " Job name must be provided via --job-name "
indata : Optional [ Dict [ str , Any ] ] = None
if args . infile :
2024-02-26 17:46:15 +00:00
if os . path . isfile ( args . infile ) :
with open ( args . infile , encoding = " utf-8 " ) as jfd :
indata = json . load ( jfd )
else :
indata = json . loads ( args . infile )
2023-12-18 08:07:22 +00:00
assert indata and isinstance ( indata , dict ) , " Invalid --infile json "
result : Dict [ str , Any ] = { }
s3 = S3Helper ( )
2024-01-04 15:35:09 +00:00
pr_info = PRInfo ( )
git_runner = GitRunner ( set_cwd_to_git_root = True )
2023-12-18 08:07:22 +00:00
2024-01-04 15:35:09 +00:00
### CONFIGURE action: start
2023-12-18 08:07:22 +00:00
if args . configure :
2024-06-10 09:18:03 +00:00
if IS_CI and pr_info . is_pr :
2024-05-21 10:57:05 +00:00
# store meta on s3 (now we need it only for PRs)
2024-05-23 14:36:24 +00:00
meta = CiMetadata ( s3 , pr_info . number , pr_info . head_ref )
2024-05-21 10:57:05 +00:00
meta . run_id = int ( GITHUB_RUN_ID )
meta . push_meta ( )
2024-06-02 16:25:14 +00:00
ci_settings = CiSettings . create_from_pr_message (
2024-03-26 15:56:41 +00:00
args . commit_message or None , update_from_api = True
)
2024-03-26 14:43:53 +00:00
2024-06-10 09:18:03 +00:00
if ci_settings . no_merge_commit and IS_CI :
2024-03-26 14:43:53 +00:00
git_runner . run ( f " { GIT_PREFIX } checkout { pr_info . sha } " )
2023-12-18 08:07:22 +00:00
2024-02-02 17:10:47 +00:00
git_ref = git_runner . run ( f " { GIT_PREFIX } rev-parse HEAD " )
2023-12-18 08:07:22 +00:00
# let's get CH version
version = get_version_from_repo ( git = Git ( True ) ) . string
print ( f " Got CH version for this commit: [ { version } ] " )
docker_data = (
2024-02-05 17:09:43 +00:00
_configure_docker_jobs ( args . docker_digest_or_latest )
2023-12-18 08:07:22 +00:00
if not args . skip_docker
else { }
)
2024-06-02 16:25:14 +00:00
ci_cache = _configure_jobs (
s3 ,
pr_info ,
ci_settings ,
2024-06-07 14:37:05 +00:00
args . skip_jobs ,
2024-07-24 17:51:34 +00:00
args . workflow ,
2023-12-18 08:07:22 +00:00
)
2024-02-04 19:12:37 +00:00
2024-07-14 09:47:18 +00:00
ci_cache . print_status ( )
2024-07-13 14:13:56 +00:00
if IS_CI and pr_info . is_pr and not ci_settings . no_ci_cache :
2024-07-11 20:19:50 +00:00
ci_cache . filter_out_not_affected_jobs ( )
2024-07-14 09:47:18 +00:00
ci_cache . print_status ( )
2024-07-12 10:29:34 +00:00
2024-06-10 09:18:03 +00:00
if IS_CI and not pr_info . is_merge_queue :
2024-02-04 19:12:37 +00:00
2024-10-03 07:26:11 +00:00
if pr_info . is_master and pr_info . is_push_event :
2024-06-10 09:18:03 +00:00
print ( " Release/master: CI Cache add pending records for all todo jobs " )
2024-06-02 16:25:14 +00:00
ci_cache . push_pending_all ( pr_info . is_release )
2024-02-02 17:10:47 +00:00
2024-10-03 07:26:11 +00:00
if pr_info . is_master or pr_info . is_pr :
# - wait for pending jobs to be finished, await_jobs is a long blocking call
# - don't wait for release CI because some jobs may not be present there
# and we may wait until timeout in vain
ci_cache . await_pending_jobs ( pr_info . is_release )
2024-07-21 10:46:58 +00:00
2023-12-18 08:07:22 +00:00
# conclude results
result [ " git_ref " ] = git_ref
result [ " version " ] = version
2024-06-10 09:18:03 +00:00
result [ " build " ] = ci_cache . job_digests [ CI . BuildNames . PACKAGE_RELEASE ]
result [ " docs " ] = ci_cache . job_digests [ CI . JobNames . DOCS_CHECK ]
2024-06-02 16:25:14 +00:00
result [ " ci_settings " ] = ci_settings . as_dict ( )
2024-03-11 13:35:05 +00:00
if not args . skip_jobs :
2024-06-18 11:17:42 +00:00
result [ " stages_data " ] = _generate_ci_stage_config (
ci_cache . jobs_to_do , ci_settings . woolen_wolfdog
)
2024-06-07 14:37:05 +00:00
result [ " jobs_data " ] = {
" jobs_to_do " : list ( ci_cache . jobs_to_do ) ,
" jobs_to_skip " : ci_cache . jobs_to_skip ,
" digests " : ci_cache . job_digests ,
" jobs_params " : {
job : { " batches " : config . batches , " num_batches " : config . num_batches }
for job , config in ci_cache . jobs_to_do . items ( )
} ,
}
2023-12-18 08:07:22 +00:00
result [ " docker_data " ] = docker_data
2024-01-04 15:35:09 +00:00
### CONFIGURE action: end
2023-12-18 08:07:22 +00:00
2024-01-04 15:35:09 +00:00
### PRE action: start
2023-12-18 08:07:22 +00:00
elif args . pre :
2024-01-04 15:35:09 +00:00
assert indata , " Run config must be provided via --infile "
2024-07-15 14:32:49 +00:00
_pre_action ( s3 , args . job_name , args . batch , indata , pr_info )
2023-12-18 08:07:22 +00:00
2024-01-04 15:35:09 +00:00
### RUN action: start
2023-12-18 08:07:22 +00:00
elif args . run :
2024-01-04 15:35:09 +00:00
assert indata
2024-07-15 14:32:49 +00:00
job_report = JobReport . load ( )
2024-01-04 15:35:09 +00:00
check_name = args . job_name
check_name_with_group = _get_ext_check_name ( check_name )
print (
f " Check if rerun for name: [ { check_name } ], extended name [ { check_name_with_group } ] "
2023-12-18 08:07:22 +00:00
)
2024-02-02 17:10:47 +00:00
2024-07-15 14:32:49 +00:00
if job_report . job_skipped and not args . force :
2023-12-18 08:07:22 +00:00
print (
2024-07-15 14:32:49 +00:00
f " Commit status or Build Report is already present - job will be skipped with status: [ { job_report . status } ] "
2023-12-18 08:07:22 +00:00
)
2024-07-15 14:32:49 +00:00
if job_report . status == SUCCESS :
2024-01-04 15:35:09 +00:00
exit_code = 0
else :
exit_code = 1
else :
exit_code = _run_test ( check_name , args . run_command )
2024-07-13 18:15:47 +00:00
job_report = JobReport . load ( ) if JobReport . exist ( ) else None
assert (
job_report
) , " BUG. There must be job report either real report, or pre-report if job was killed "
job_report . exit_code = exit_code
job_report . dump ( )
2024-01-04 15:35:09 +00:00
### RUN action: end
2023-12-18 08:07:22 +00:00
2024-01-04 15:35:09 +00:00
### POST action: start
2023-12-18 08:07:22 +00:00
elif args . post :
2024-01-04 15:35:09 +00:00
job_report = JobReport . load ( ) if JobReport . exist ( ) else None
2024-07-13 18:15:47 +00:00
assert (
job_report
) , " BUG. There must be job report either real report, or pre-report if job was killed "
2024-07-15 14:32:49 +00:00
error_description = " "
2024-08-06 19:39:41 +00:00
if not job_report . dummy :
2024-07-13 18:15:47 +00:00
# it's a real job report
2024-01-04 15:35:09 +00:00
ch_helper = ClickHouseHelper ( )
check_url = " "
2024-06-10 09:18:03 +00:00
if CI . is_build_job ( args . job_name ) :
2024-02-02 17:10:47 +00:00
assert (
indata
2024-02-04 19:12:37 +00:00
) , f " --infile with config must be provided for POST action of a build type job [ { args . job_name } ] "
2024-05-30 19:21:40 +00:00
# upload binaries only for normal builds in PRs
upload_binary = (
not pr_info . is_pr
2024-06-10 09:18:03 +00:00
or CI . get_job_ci_stage ( args . job_name ) == CI . WorkflowStages . BUILDS_1
2024-06-02 16:25:14 +00:00
or CiSettings . create_from_run_config ( indata ) . upload_all
2024-05-30 19:21:40 +00:00
)
2024-01-04 15:35:09 +00:00
build_name = args . job_name
s3_path_prefix = " / " . join (
(
get_release_or_pr ( pr_info , get_version_from_repo ( ) ) [ 0 ] ,
pr_info . sha ,
build_name ,
)
)
log_url = _upload_build_artifacts (
pr_info ,
build_name ,
2024-02-02 17:10:47 +00:00
ci_cache = CiCache ( s3 , indata [ " jobs_data " ] [ " digests " ] ) ,
2024-01-04 15:35:09 +00:00
job_report = job_report ,
s3 = s3 ,
s3_destination = s3_path_prefix ,
2024-05-30 19:21:40 +00:00
upload_binary = upload_binary ,
2024-01-04 15:35:09 +00:00
)
2024-06-04 13:25:01 +00:00
_upload_build_profile_data (
pr_info , build_name , job_report , git_runner , ch_helper
)
2024-01-04 15:35:09 +00:00
check_url = log_url
else :
# test job
2024-05-03 15:45:39 +00:00
gh = GitHub ( get_best_robot_token ( ) , per_page = 100 )
2024-01-04 15:35:09 +00:00
additional_urls = [ ]
s3_path_prefix = " / " . join (
(
get_release_or_pr ( pr_info , get_version_from_repo ( ) ) [ 0 ] ,
pr_info . sha ,
2024-08-03 08:40:12 +00:00
Utils . normalize_string (
2024-01-04 15:35:09 +00:00
job_report . check_name or _get_ext_check_name ( args . job_name )
) ,
)
)
if job_report . build_dir_for_upload :
additional_urls = s3 . upload_build_directory_to_s3 (
Path ( job_report . build_dir_for_upload ) ,
s3_path_prefix ,
keep_dirs_in_s3_path = False ,
upload_symlinks = False ,
)
if job_report . test_results or job_report . additional_files :
check_url = upload_result_helper . upload_results (
s3 ,
pr_info . number ,
pr_info . sha ,
2024-11-12 10:07:57 +00:00
pr_info . head_ref ,
2024-01-04 15:35:09 +00:00
job_report . test_results ,
job_report . additional_files ,
2024-02-20 21:19:30 +00:00
job_report . check_name or _get_ext_check_name ( args . job_name ) ,
2024-01-04 15:35:09 +00:00
additional_urls = additional_urls or None ,
)
2024-05-03 15:45:39 +00:00
commit = get_commit ( gh , pr_info . sha )
2024-01-04 15:35:09 +00:00
post_commit_status (
commit ,
job_report . status ,
check_url ,
format_description ( job_report . description ) ,
2024-02-20 21:19:30 +00:00
job_report . check_name or _get_ext_check_name ( args . job_name ) ,
2024-01-04 15:35:09 +00:00
pr_info ,
dump_to_file = True ,
)
print ( f " Job report url: [ { check_url } ] " )
prepared_events = prepare_tests_results_for_clickhouse (
pr_info ,
job_report . test_results ,
job_report . status ,
job_report . duration ,
job_report . start_time ,
check_url or " " ,
2024-02-20 21:19:30 +00:00
job_report . check_name or _get_ext_check_name ( args . job_name ) ,
2023-12-18 08:07:22 +00:00
)
2024-01-04 15:35:09 +00:00
ch_helper . insert_events_into (
db = " default " , table = " checks " , events = prepared_events
2023-12-18 08:07:22 +00:00
)
2024-03-15 18:08:41 +00:00
2024-03-18 17:45:23 +00:00
if " DockerServerImage " in args . job_name and indata is not None :
2024-03-15 18:08:41 +00:00
_add_build_to_version_history (
2024-03-18 17:45:23 +00:00
pr_info ,
job_report ,
indata [ " version " ] ,
indata [ " build " ] ,
ch_helper ,
2024-03-15 18:08:41 +00:00
)
2024-07-13 16:49:23 +00:00
elif job_report . job_skipped :
print ( f " Skipped after rerun check { [ args . job_name ] } - do nothing " )
2024-07-15 14:32:49 +00:00
else :
2024-09-26 14:05:34 +00:00
print ( " ERROR: Job was killed - generate evidence " )
2024-07-13 18:15:47 +00:00
job_report . update_duration ( )
2024-07-15 11:59:57 +00:00
ret_code = os . getenv ( " JOB_EXIT_CODE " , " " )
if ret_code :
try :
job_report . exit_code = int ( ret_code )
except ValueError :
pass
2024-07-13 18:15:47 +00:00
if Utils . is_killed_with_oom ( ) :
print ( " WARNING: OOM while job execution " )
2024-07-18 11:16:35 +00:00
print ( subprocess . run ( " sudo dmesg -T " , check = False ) )
2024-07-15 14:32:49 +00:00
error_description = f " Out Of Memory, exit_code { job_report . exit_code } "
2024-07-13 18:15:47 +00:00
else :
2024-07-15 14:32:49 +00:00
error_description = f " Unknown, exit_code { job_report . exit_code } "
2024-07-16 15:07:49 +00:00
CIBuddy ( ) . post_job_error (
2024-07-15 14:32:49 +00:00
error_description + f " after { int ( job_report . duration ) } s " ,
job_name = _get_ext_check_name ( args . job_name ) ,
)
2024-07-12 18:32:01 +00:00
if CI . is_test_job ( args . job_name ) :
gh = GitHub ( get_best_robot_token ( ) , per_page = 100 )
commit = get_commit ( gh , pr_info . sha )
2024-08-15 11:11:10 +00:00
check_url = " "
if job_report . test_results or job_report . additional_files :
check_url = upload_result_helper . upload_results (
s3 ,
pr_info . number ,
pr_info . sha ,
2024-11-12 10:07:57 +00:00
pr_info . head_ref ,
2024-08-15 11:11:10 +00:00
job_report . test_results ,
job_report . additional_files ,
job_report . check_name or _get_ext_check_name ( args . job_name ) ,
)
2024-07-12 18:32:01 +00:00
post_commit_status (
commit ,
ERROR ,
2024-08-15 11:11:10 +00:00
check_url ,
2024-07-15 14:32:49 +00:00
" Error: " + error_description ,
2024-07-13 14:13:56 +00:00
_get_ext_check_name ( args . job_name ) ,
2024-07-12 18:32:01 +00:00
pr_info ,
dump_to_file = True ,
)
2024-07-15 14:32:49 +00:00
if not job_report . job_skipped :
print ( " push finish record to ci db " )
prepared_events = prepare_tests_results_for_clickhouse (
pr_info ,
[
TestResult (
JOB_FINISHED_TEST_NAME ,
FAIL if error_description else OK ,
raw_logs = error_description or None ,
)
] ,
SUCCESS if not error_description else ERROR ,
0.0 ,
JobReport . get_start_time_from_current ( ) ,
" " ,
_get_ext_check_name ( args . job_name ) ,
)
ClickHouseHelper ( ) . insert_events_into (
db = " default " , table = " checks " , events = prepared_events
)
2024-01-04 15:35:09 +00:00
### POST action: end
2023-12-18 08:07:22 +00:00
2024-01-04 15:35:09 +00:00
### MARK SUCCESS action: start
2023-12-18 08:07:22 +00:00
elif args . mark_success :
assert indata , " Run config must be provided via --infile "
2024-02-02 17:10:47 +00:00
_mark_success_action ( s3 , indata , pr_info , args . job_name , args . batch )
2024-01-04 15:35:09 +00:00
### UPDATE GH STATUSES action: start
elif args . update_gh_statuses :
assert indata , " Run config must be provided via --infile "
2024-02-02 17:10:47 +00:00
_update_gh_statuses_action ( indata = indata , s3 = s3 )
2023-12-18 08:07:22 +00:00
2024-08-02 18:16:44 +00:00
### CANCEL THE PREVIOUS WORKFLOW RUN
2024-05-21 10:57:05 +00:00
elif args . cancel_previous_run :
2024-05-23 14:36:24 +00:00
if pr_info . is_merge_queue :
2024-08-02 18:16:44 +00:00
_cancel_pr_workflow ( s3 , pr_info . merged_pr )
2024-05-23 14:36:24 +00:00
elif pr_info . is_pr :
2024-08-02 18:16:44 +00:00
_cancel_pr_workflow ( s3 , pr_info . number , cancel_sync = True )
2024-05-23 14:36:24 +00:00
else :
assert False , " BUG! Not supported scenario "
2024-05-21 10:57:05 +00:00
2024-05-24 09:36:28 +00:00
### SET PENDING STATUS
2024-05-24 09:46:50 +00:00
elif args . set_pending_status :
2024-05-24 09:36:28 +00:00
if pr_info . is_pr :
_set_pending_statuses ( pr_info )
else :
assert False , " BUG! Not supported scenario "
2024-01-04 15:35:09 +00:00
### print results
2024-02-02 17:10:47 +00:00
_print_results ( result , args . outfile , args . pretty )
2023-12-18 08:07:22 +00:00
return exit_code
if __name__ == " __main__ " :
sys . exit ( main ( ) )