blob: 00ebec6fe23dabf7039de48e62f23ed7a0c53d1b [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
100def 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
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
117def 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
134 jenkins.wait_end_of_build(
135 name=build[0],
136 build_id=build[1],
137 timeout=opts.build_timeout,
138 interval=1,
139 verbose=opts.verbose,
140 job_output_prefix=opts.job_output_prefix)
141 result = jenkins.build_info(name=build[0],
142 build_id=build[1])['result']
143 if opts.verbose:
144 print_build_footer(build, result, opts.host)
145
146 return EXIT_CODES.get(result, 2)
147
148
149def main(args=None):
150 parser = load_params()
151 opts = parser.parse_args()
152
153 if opts.host is None or opts.job_name is None:
154 print("JENKINS_URL and a job name are required!")
155 parser.print_help()
156 return 10
157 else:
158 exit_code = run_job(opts)
159 return exit_code
160
161
162if __name__ == "__main__":
163 sys.exit(main())