Dennis Dmitriev | 3ec2e53 | 2018-06-08 04:33:34 +0300 | [diff] [blame] | 1 | #!/usr/bin/env python |
| 2 | |
| 3 | import argparse |
| 4 | import os |
| 5 | import sys |
| 6 | |
| 7 | import json |
| 8 | |
| 9 | sys.path.append(os.getcwd()) |
| 10 | try: |
| 11 | from tcp_tests.managers.jenkins.client import JenkinsClient |
| 12 | except ImportError: |
| 13 | print("ImportError: Run the application from the tcp-qa directory or " |
| 14 | "set the PYTHONPATH environment variable to directory which contains" |
| 15 | " ./tcp_tests") |
| 16 | sys.exit(1) |
| 17 | |
| 18 | |
| 19 | EXIT_CODES = { |
| 20 | "SUCCESS": 0, |
| 21 | # 1 - python runtime execution error |
| 22 | # 2 - job unknown status |
| 23 | "FAILURE": 3, |
| 24 | "UNSTABLE": 4, |
| 25 | "ABORTED": 5, |
| 26 | "DISABLED": 6 |
| 27 | # 10 - invalid cli options |
| 28 | } |
| 29 | |
| 30 | |
| 31 | def load_params(): |
| 32 | """ |
| 33 | Parse CLI arguments and environment variables |
| 34 | |
| 35 | Returns: ArgumentParser instance |
| 36 | """ |
| 37 | env_host = os.environ.get('JENKINS_URL', None) |
| 38 | env_username = os.environ.get('JENKINS_USER', None) |
| 39 | env_password = os.environ.get('JENKINS_PASS', None) |
| 40 | env_start_timeout = os.environ.get('JENKINS_START_TIMEOUT', 1800) |
| 41 | env_build_timeout = os.environ.get('JENKINS_BUILD_TIMEOUT', 3600 * 4) |
| 42 | |
| 43 | parser = argparse.ArgumentParser(description=( |
| 44 | 'Host, username and password may be specified either by the command ' |
| 45 | 'line arguments or using environment variables: JENKINS_URL, ' |
| 46 | 'JENKINS_USER, JENKINS_PASS, JENKINS_START_TIMEOUT, ' |
| 47 | 'JENKINS_BUILD_TIMEOUT. \nCommand line arguments have the highest ' |
| 48 | 'priority, after that the environment variables are used as defaults.' |
| 49 | )) |
| 50 | parser.add_argument('--host', |
| 51 | metavar='JENKINS_URL', |
| 52 | help='Jenkins Host', |
| 53 | default=env_host) |
| 54 | parser.add_argument('--username', |
| 55 | metavar='JENKINS_USER', |
| 56 | help='Jenkins Username', default=env_username) |
| 57 | parser.add_argument('--password', |
| 58 | metavar='JENKINS_PASS', |
| 59 | help='Jenkins Password or API token', |
| 60 | default=env_password) |
| 61 | parser.add_argument('--start-timeout', |
| 62 | metavar='JENKINS_START_TIMEOUT', |
| 63 | help='Timeout waiting until build is started', |
| 64 | default=env_start_timeout, |
| 65 | type=int) |
| 66 | parser.add_argument('--build-timeout', |
| 67 | metavar='JENKINS_BUILD_TIMEOUT', |
| 68 | help='Timeout waiting until build is finished', |
| 69 | default=env_build_timeout, |
| 70 | type=int) |
| 71 | parser.add_argument('--job-name', |
| 72 | help='Jenkins job name to run', |
| 73 | default=None) |
| 74 | parser.add_argument('--job-parameters', |
| 75 | metavar='json-dict', |
| 76 | help=('Job parameters to use instead of default ' |
| 77 | 'values, as a json string, for example: ' |
| 78 | '--job-parameters=\'{"SALT_MASTER_URL": ' |
| 79 | '"http://localhost:6969"}\''), |
| 80 | default={}, type=json.loads) |
| 81 | parser.add_argument('--job-output-prefix', |
| 82 | help=('Jenkins job output prefix for each line in the ' |
| 83 | 'output, if --verbose is enabled. Useful for the' |
| 84 | ' pipelines that use multiple different runs of ' |
| 85 | 'jobs. The string is a template for python ' |
| 86 | 'format() function where the following arguments' |
| 87 | ' are allowed: job_name, build_number. ' |
| 88 | 'Example: --job-output-prefix=\"[ {job_name} ' |
| 89 | '#{build_number}, core ]\"'), |
| 90 | default='', |
| 91 | type=str) |
| 92 | parser.add_argument('--verbose', |
| 93 | action='store_const', |
| 94 | const=True, |
| 95 | help='Show build console output', |
| 96 | default=False) |
| 97 | return parser |
| 98 | |
| 99 | |
Dennis Dmitriev | 2a49873 | 2018-12-21 18:30:23 +0200 | [diff] [blame] | 100 | def print_build_header(build, job_params, build_timeout): |
Dennis Dmitriev | 3ec2e53 | 2018-06-08 04:33:34 +0300 | [diff] [blame] | 101 | print('\n#############################################################') |
| 102 | print('##### Building job [{0}] #{1} (timeout={2}) with the following ' |
Dennis Dmitriev | 2a49873 | 2018-12-21 18:30:23 +0200 | [diff] [blame] | 103 | 'parameters:'.format(build[0], build[1], build_timeout)) |
Dennis Dmitriev | 3ec2e53 | 2018-06-08 04:33:34 +0300 | [diff] [blame] | 104 | print('##### ' + '\n##### '.join( |
| 105 | [str(key) + ": " + str(val) for key, val in job_params.iteritems()] |
| 106 | )) |
| 107 | print('#############################################################') |
| 108 | |
| 109 | |
| 110 | def print_build_footer(build, result, url): |
| 111 | print('\n\n#############################################################') |
| 112 | print('##### Completed job [{0}] #{1} at {2}: {3}' |
| 113 | .format(build[0], build[1], url, result)) |
| 114 | print('#############################################################\n') |
| 115 | |
| 116 | |
Dennis Dmitriev | 2a49873 | 2018-12-21 18:30:23 +0200 | [diff] [blame] | 117 | def run_job(host, username, password, |
| 118 | job_name, job_parameters=None, job_output_prefix='', |
| 119 | start_timeout=1800, build_timeout=3600 * 4, verbose=False): |
Dennis Dmitriev | 3ec2e53 | 2018-06-08 04:33:34 +0300 | [diff] [blame] | 120 | |
| 121 | jenkins = JenkinsClient( |
Dennis Dmitriev | 2a49873 | 2018-12-21 18:30:23 +0200 | [diff] [blame] | 122 | host=host, |
| 123 | username=username, |
| 124 | password=password) |
Dennis Dmitriev | 3ec2e53 | 2018-06-08 04:33:34 +0300 | [diff] [blame] | 125 | |
Hanna Arhipova | 874c68f | 2021-03-29 15:57:19 +0300 | [diff] [blame] | 126 | job_params = jenkins.make_defaults_params(job_name) |
Dennis Dmitriev | 2a49873 | 2018-12-21 18:30:23 +0200 | [diff] [blame] | 127 | if job_parameters is not None: # job_parameters = {} |
| 128 | job_params.update(job_parameters) |
Dennis Dmitriev | 3ec2e53 | 2018-06-08 04:33:34 +0300 | [diff] [blame] | 129 | |
dtsapikov | 908b0c0 | 2023-04-03 17:41:10 +0400 | [diff] [blame] | 130 | job_attempts = 2 |
Anna Arhipova | db63a45 | 2023-06-15 09:43:25 +0200 | [diff] [blame] | 131 | for attempt in range(1, job_attempts+1): |
| 132 | print('Attempt ' + str(attempt)) |
dtsapikov | 908b0c0 | 2023-04-03 17:41:10 +0400 | [diff] [blame] | 133 | build = jenkins.run_build(job_name, |
| 134 | job_params, |
| 135 | verbose=verbose, |
| 136 | timeout=start_timeout) |
Dennis Dmitriev | 3ec2e53 | 2018-06-08 04:33:34 +0300 | [diff] [blame] | 137 | |
dtsapikov | 908b0c0 | 2023-04-03 17:41:10 +0400 | [diff] [blame] | 138 | if verbose: |
| 139 | print_build_header(build, job_params, build_timeout) |
| 140 | |
| 141 | try: |
| 142 | jenkins.wait_end_of_build( |
| 143 | name=build[0], |
| 144 | build_id=build[1], |
| 145 | timeout=build_timeout, |
| 146 | interval=1, |
| 147 | verbose=verbose, |
| 148 | job_output_prefix=job_output_prefix) |
| 149 | except Exception as e: |
| 150 | print(str(e)) |
| 151 | raise |
Anna Arhipova | db63a45 | 2023-06-15 09:43:25 +0200 | [diff] [blame] | 152 | result = jenkins.build_info(name=build[0], |
| 153 | build_id=build[1])['result'] |
| 154 | if verbose: |
| 155 | print_build_footer(build, result, host) |
| 156 | |
| 157 | if result == "SUCCESS": |
| 158 | break |
| 159 | |
dtsapikov | 908b0c0 | 2023-04-03 17:41:10 +0400 | [diff] [blame] | 160 | job_log = jenkins.get_build_output(job_name, build[1]) |
| 161 | # Workaround for restart jobs that failed by salt-timeout errors |
| 162 | # or by sporadic fail in attach disks |
Anna Arhipova | db63a45 | 2023-06-15 09:43:25 +0200 | [diff] [blame] | 163 | if ('SaltReqTimeoutError' in job_log or |
| 164 | 'not a block device' in job_log): |
dtsapikov | 908b0c0 | 2023-04-03 17:41:10 +0400 | [diff] [blame] | 165 | print('Job returns known infra fail!') |
Anna Arhipova | db63a45 | 2023-06-15 09:43:25 +0200 | [diff] [blame] | 166 | continue |
| 167 | else: |
| 168 | # do not retry if it's not known issue |
| 169 | break |
Dennis Dmitriev | 3ec2e53 | 2018-06-08 04:33:34 +0300 | [diff] [blame] | 170 | |
Dennis Dmitriev | 2a49873 | 2018-12-21 18:30:23 +0200 | [diff] [blame] | 171 | return result |
Dennis Dmitriev | 3ec2e53 | 2018-06-08 04:33:34 +0300 | [diff] [blame] | 172 | |
| 173 | |
| 174 | def main(args=None): |
| 175 | parser = load_params() |
| 176 | opts = parser.parse_args() |
| 177 | |
| 178 | if opts.host is None or opts.job_name is None: |
| 179 | print("JENKINS_URL and a job name are required!") |
| 180 | parser.print_help() |
| 181 | return 10 |
| 182 | else: |
Dennis Dmitriev | 2a49873 | 2018-12-21 18:30:23 +0200 | [diff] [blame] | 183 | result = run_job( |
| 184 | opts.host, |
| 185 | opts.username, |
| 186 | opts.password, |
| 187 | opts.job_name, |
| 188 | opts.job_parameters, |
| 189 | opts.job_output_prefix, |
| 190 | opts.start_timeout, |
| 191 | opts.build_timeout, |
| 192 | opts.verbose) |
| 193 | return EXIT_CODES.get(result, 2) |
Dennis Dmitriev | 3ec2e53 | 2018-06-08 04:33:34 +0300 | [diff] [blame] | 194 | |
| 195 | |
| 196 | if __name__ == "__main__": |
| 197 | sys.exit(main()) |