blob: 403cc00b00abce4dc041fe719aecf774f2db2eba [file] [log] [blame]
Dennis Dmitriev3ec2e532018-06-08 04:33:34 +03001#!/usr/bin/env python
2
3import argparse
4import os
5import sys
6
7import json
8
9sys.path.append(os.getcwd())
10try:
11 from tcp_tests.managers.jenkins.client import JenkinsClient
12except 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
19EXIT_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
31def 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 Dmitriev2a498732018-12-21 18:30:23 +0200100def print_build_header(build, job_params, build_timeout):
Dennis Dmitriev3ec2e532018-06-08 04:33:34 +0300101 print('\n#############################################################')
102 print('##### Building job [{0}] #{1} (timeout={2}) with the following '
Dennis Dmitriev2a498732018-12-21 18:30:23 +0200103 'parameters:'.format(build[0], build[1], build_timeout))
Dennis Dmitriev3ec2e532018-06-08 04:33:34 +0300104 print('##### ' + '\n##### '.join(
105 [str(key) + ": " + str(val) for key, val in job_params.iteritems()]
106 ))
107 print('#############################################################')
108
109
110def 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 Dmitriev2a498732018-12-21 18:30:23 +0200117def 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 Dmitriev3ec2e532018-06-08 04:33:34 +0300120
121 jenkins = JenkinsClient(
Dennis Dmitriev2a498732018-12-21 18:30:23 +0200122 host=host,
123 username=username,
124 password=password)
Dennis Dmitriev3ec2e532018-06-08 04:33:34 +0300125
Hanna Arhipova874c68f2021-03-29 15:57:19 +0300126 job_params = jenkins.make_defaults_params(job_name)
Dennis Dmitriev2a498732018-12-21 18:30:23 +0200127 if job_parameters is not None: # job_parameters = {}
128 job_params.update(job_parameters)
Dennis Dmitriev3ec2e532018-06-08 04:33:34 +0300129
dtsapikov908b0c02023-04-03 17:41:10 +0400130 job_attempts = 2
131 count = 1
132 while (job_attempts != 0):
133 print('Attempt ' + str(count))
134 count += 1
135 job_attempts -= 1
136 build = jenkins.run_build(job_name,
137 job_params,
138 verbose=verbose,
139 timeout=start_timeout)
Dennis Dmitriev3ec2e532018-06-08 04:33:34 +0300140
dtsapikov908b0c02023-04-03 17:41:10 +0400141 if verbose:
142 print_build_header(build, job_params, build_timeout)
143
144 try:
145 jenkins.wait_end_of_build(
146 name=build[0],
147 build_id=build[1],
148 timeout=build_timeout,
149 interval=1,
150 verbose=verbose,
151 job_output_prefix=job_output_prefix)
152 except Exception as e:
153 print(str(e))
154 raise
155 job_log = jenkins.get_build_output(job_name, build[1])
156 # Workaround for restart jobs that failed by salt-timeout errors
157 # or by sporadic fail in attach disks
158 if ('SaltReqTimeoutError' not in job_log and
159 'not a block device' not in job_log):
160 break
161 else:
162 print('Job returns known infra fail!')
Dennis Dmitriev13e804b2018-10-09 19:25:14 +0300163
Dennis Dmitriev3ec2e532018-06-08 04:33:34 +0300164 result = jenkins.build_info(name=build[0],
165 build_id=build[1])['result']
Dennis Dmitriev2a498732018-12-21 18:30:23 +0200166 if verbose:
167 print_build_footer(build, result, host)
Dennis Dmitriev3ec2e532018-06-08 04:33:34 +0300168
Dennis Dmitriev2a498732018-12-21 18:30:23 +0200169 return result
Dennis Dmitriev3ec2e532018-06-08 04:33:34 +0300170
171
172def main(args=None):
173 parser = load_params()
174 opts = parser.parse_args()
175
176 if opts.host is None or opts.job_name is None:
177 print("JENKINS_URL and a job name are required!")
178 parser.print_help()
179 return 10
180 else:
Dennis Dmitriev2a498732018-12-21 18:30:23 +0200181 result = run_job(
182 opts.host,
183 opts.username,
184 opts.password,
185 opts.job_name,
186 opts.job_parameters,
187 opts.job_output_prefix,
188 opts.start_timeout,
189 opts.build_timeout,
190 opts.verbose)
191 return EXIT_CODES.get(result, 2)
Dennis Dmitriev3ec2e532018-06-08 04:33:34 +0300192
193
194if __name__ == "__main__":
195 sys.exit(main())