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 | |
| 100 | def print_build_header(build, job_params, opts): |
| 101 | print('\n#############################################################') |
| 102 | print('##### Building job [{0}] #{1} (timeout={2}) with the following ' |
| 103 | 'parameters:'.format(build[0], build[1], opts.build_timeout)) |
| 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 | |
| 117 | def run_job(opts): |
| 118 | |
| 119 | jenkins = JenkinsClient( |
| 120 | host=opts.host, |
| 121 | username=opts.username, |
| 122 | password=opts.password) |
| 123 | |
| 124 | job_params = jenkins.make_defults_params(opts.job_name) |
| 125 | job_params.update(opts.job_parameters) |
| 126 | |
| 127 | build = jenkins.run_build(opts.job_name, |
| 128 | job_params, |
| 129 | verbose=opts.verbose, |
| 130 | timeout=opts.start_timeout) |
| 131 | if opts.verbose: |
| 132 | print_build_header(build, job_params, opts) |
| 133 | |
Dennis Dmitriev | 13e804b | 2018-10-09 19:25:14 +0300 | [diff] [blame] | 134 | try: |
| 135 | jenkins.wait_end_of_build( |
| 136 | name=build[0], |
| 137 | build_id=build[1], |
| 138 | timeout=opts.build_timeout, |
| 139 | interval=1, |
| 140 | verbose=opts.verbose, |
| 141 | job_output_prefix=opts.job_output_prefix) |
Dennis Dmitriev | efe5c0b | 2018-10-24 20:35:26 +0300 | [diff] [blame] | 142 | except Exception as e: |
Dennis Dmitriev | 13e804b | 2018-10-09 19:25:14 +0300 | [diff] [blame] | 143 | print(str(e)) |
| 144 | raise |
| 145 | |
Dennis Dmitriev | 3ec2e53 | 2018-06-08 04:33:34 +0300 | [diff] [blame] | 146 | result = jenkins.build_info(name=build[0], |
| 147 | build_id=build[1])['result'] |
| 148 | if opts.verbose: |
| 149 | print_build_footer(build, result, opts.host) |
| 150 | |
| 151 | return EXIT_CODES.get(result, 2) |
| 152 | |
| 153 | |
| 154 | def main(args=None): |
| 155 | parser = load_params() |
| 156 | opts = parser.parse_args() |
| 157 | |
| 158 | if opts.host is None or opts.job_name is None: |
| 159 | print("JENKINS_URL and a job name are required!") |
| 160 | parser.print_help() |
| 161 | return 10 |
| 162 | else: |
| 163 | exit_code = run_job(opts) |
| 164 | return exit_code |
| 165 | |
| 166 | |
| 167 | if __name__ == "__main__": |
| 168 | sys.exit(main()) |