blob: 8e71ecca14b9449b6fcc79fcfac7cb6bc57a5252 [file] [log] [blame]
Matthew Treinish1f7b33d2013-10-21 18:07:02 +00001#!/usr/bin/env python
Matthew Treinish1f7b33d2013-10-21 18:07:02 +00002
3# Copyright 2013 IBM Corp.
4#
5# Licensed under the Apache License, Version 2.0 (the "License"); you may
6# not use this file except in compliance with the License. You may obtain
7# a copy of the License at
8#
9# http://www.apache.org/licenses/LICENSE-2.0
10#
11# Unless required by applicable law or agreed to in writing, software
12# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
13# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
14# License for the specific language governing permissions and limitations
15# under the License.
16
Matthew Treinishf0971712014-04-11 20:08:53 +000017import argparse
Matthew Treinishf0971712014-04-11 20:08:53 +000018import os
Brant Knudson28de8d52016-03-18 13:50:02 -050019import re
Matthew Treinish1f7b33d2013-10-21 18:07:02 +000020import sys
David Patersone45aa842015-11-18 16:12:27 -080021import traceback
Matthew Treinish1f7b33d2013-10-21 18:07:02 +000022
David Patersone45aa842015-11-18 16:12:27 -080023from cliff import command
24from oslo_log import log as logging
Matthew Treinish21905512015-07-13 10:33:35 -040025from oslo_serialization import jsonutils as json
Matthew Treinishc795b9e2014-06-09 17:01:10 -040026from six import moves
Matthew Treinishf077dd22015-04-23 09:37:41 -040027from six.moves.urllib import parse as urlparse
Matthew Treinish4f30eb82014-01-07 21:04:49 +000028
Matthew Treinish1f7b33d2013-10-21 18:07:02 +000029from tempest import clients
Andrea Frittoli (andreaf)290b3e12015-10-08 10:25:02 +010030from tempest.common import credentials_factory as credentials
Matthew Treinish1f7b33d2013-10-21 18:07:02 +000031from tempest import config
Jordan Pittier00f25962016-03-18 17:10:07 +010032import tempest.lib.common.http
Matthew Treinish1f7b33d2013-10-21 18:07:02 +000033
34
Sean Dague86bd8422013-12-20 09:56:44 -050035CONF = config.CONF
David Kranzf20ac322014-05-02 16:46:15 -040036CONF_PARSER = None
Matthew Treinish1f7b33d2013-10-21 18:07:02 +000037
David Patersone45aa842015-11-18 16:12:27 -080038LOG = logging.getLogger(__name__)
39
Matthew Treinish1f7b33d2013-10-21 18:07:02 +000040
Matthew Treinishf0971712014-04-11 20:08:53 +000041def _get_config_file():
42 default_config_dir = os.path.join(os.path.abspath(
David Kranzf20ac322014-05-02 16:46:15 -040043 os.path.dirname(os.path.dirname(os.path.dirname(__file__)))), "etc")
Matthew Treinishf0971712014-04-11 20:08:53 +000044 default_config_file = "tempest.conf"
45
46 conf_dir = os.environ.get('TEMPEST_CONFIG_DIR', default_config_dir)
47 conf_file = os.environ.get('TEMPEST_CONFIG', default_config_file)
48 path = os.path.join(conf_dir, conf_file)
Sirushti Murugesan12dc9732016-07-13 22:49:17 +053049 fd = open(path, 'r+')
Matthew Treinishf0971712014-04-11 20:08:53 +000050 return fd
51
52
53def change_option(option, group, value):
David Kranzf20ac322014-05-02 16:46:15 -040054 if not CONF_PARSER.has_section(group):
55 CONF_PARSER.add_section(group)
56 CONF_PARSER.set(group, option, str(value))
Matthew Treinishf0971712014-04-11 20:08:53 +000057
58
59def print_and_or_update(option, group, value, update):
60 print('Config option %s in group %s should be changed to: %s'
61 % (option, group, value))
62 if update:
63 change_option(option, group, value)
64
65
David Kranz0df154d2015-06-02 17:02:27 -040066def contains_version(prefix, versions):
67 return any([x for x in versions if x.startswith(prefix)])
68
69
Matthew Treinishf0971712014-04-11 20:08:53 +000070def verify_glance_api_versions(os, update):
Matthew Treinish99afd072013-10-22 18:03:06 +000071 # Check glance api versions
Julien Danjou99518762016-09-19 17:58:25 +020072 _, versions = os.image_client.get_versions()
David Kranz0df154d2015-06-02 17:02:27 -040073 if CONF.image_feature_enabled.api_v1 != contains_version('v1.', versions):
Nikita Gerasimov35fbdc12015-08-07 19:58:24 +030074 print_and_or_update('api_v1', 'image-feature-enabled',
Matthew Treinishf0971712014-04-11 20:08:53 +000075 not CONF.image_feature_enabled.api_v1, update)
David Kranz0df154d2015-06-02 17:02:27 -040076 if CONF.image_feature_enabled.api_v2 != contains_version('v2.', versions):
Nikita Gerasimov35fbdc12015-08-07 19:58:24 +030077 print_and_or_update('api_v2', 'image-feature-enabled',
Matthew Treinishf0971712014-04-11 20:08:53 +000078 not CONF.image_feature_enabled.api_v2, update)
Matthew Treinish99afd072013-10-22 18:03:06 +000079
80
Brant Knudson28de8d52016-03-18 13:50:02 -050081def _remove_version_project(url_path):
82 # The regex matches strings like /v2.0, /v3/, /v2.1/project-id/
83 return re.sub(r'/v\d+(\.\d+)?(/[^/]+)?', '', url_path)
84
85
Matthew Treinish9b896242014-04-23 21:25:27 +000086def _get_unversioned_endpoint(base_url):
87 endpoint_parts = urlparse.urlparse(base_url)
Brant Knudson28de8d52016-03-18 13:50:02 -050088 new_path = _remove_version_project(endpoint_parts.path)
89 endpoint_parts = endpoint_parts._replace(path=new_path)
90 endpoint = urlparse.urlunparse(endpoint_parts)
Matthew Treinish9b896242014-04-23 21:25:27 +000091 return endpoint
92
93
Matthew Treinish864fe072014-03-02 03:47:26 +000094def _get_api_versions(os, service):
95 client_dict = {
96 'nova': os.servers_client,
97 'keystone': os.identity_client,
Sean McGinnisff5f0c12017-07-07 16:39:24 -050098 'cinder': os.volumes_client_latest,
Matthew Treinish864fe072014-03-02 03:47:26 +000099 }
Ivan Kolodyazhnyb24f9042017-06-12 18:54:18 +0300100 if service != 'keystone' and service != 'cinder':
101 # Since keystone and cinder may be listening on a path,
102 # do not remove the path.
Brant Knudson28de8d52016-03-18 13:50:02 -0500103 client_dict[service].skip_path()
Matthew Treinish9b896242014-04-23 21:25:27 +0000104 endpoint = _get_unversioned_endpoint(client_dict[service].base_url)
Jordan Pittier00f25962016-03-18 17:10:07 +0100105
106 http = tempest.lib.common.http.ClosingHttp(
Daniel Melladocad3f3d2016-08-19 14:17:16 +0000107 CONF.identity.disable_ssl_certificate_validation,
108 CONF.identity.ca_certificates_file)
Jordan Pittier00f25962016-03-18 17:10:07 +0100109
110 __, body = http.request(endpoint, 'GET')
Matthew Treinish864fe072014-03-02 03:47:26 +0000111 client_dict[service].reset_path()
Brant Knudson5a59f872016-03-18 13:07:00 -0500112 try:
113 body = json.loads(body)
114 except ValueError:
115 LOG.error(
116 'Failed to get a JSON response from unversioned endpoint %s '
117 '(versioned endpoint was %s). Response is:\n%s',
118 endpoint, client_dict[service].base_url, body[:100])
119 raise
Matthew Treinish864fe072014-03-02 03:47:26 +0000120 if service == 'keystone':
121 versions = map(lambda x: x['id'], body['versions']['values'])
122 else:
123 versions = map(lambda x: x['id'], body['versions'])
Matthew Treinish09487242015-05-10 12:43:58 -0400124 return list(versions)
Matthew Treinish864fe072014-03-02 03:47:26 +0000125
126
Matthew Treinishf0971712014-04-11 20:08:53 +0000127def verify_keystone_api_versions(os, update):
Matthew Treinish864fe072014-03-02 03:47:26 +0000128 # Check keystone api versions
129 versions = _get_api_versions(os, 'keystone')
David Kranz0df154d2015-06-02 17:02:27 -0400130 if (CONF.identity_feature_enabled.api_v2 !=
131 contains_version('v2.', versions)):
Nikita Gerasimov35fbdc12015-08-07 19:58:24 +0300132 print_and_or_update('api_v2', 'identity-feature-enabled',
Matthew Treinishf0971712014-04-11 20:08:53 +0000133 not CONF.identity_feature_enabled.api_v2, update)
David Kranz0df154d2015-06-02 17:02:27 -0400134 if (CONF.identity_feature_enabled.api_v3 !=
135 contains_version('v3.', versions)):
Nikita Gerasimov35fbdc12015-08-07 19:58:24 +0300136 print_and_or_update('api_v3', 'identity-feature-enabled',
Matthew Treinishf0971712014-04-11 20:08:53 +0000137 not CONF.identity_feature_enabled.api_v3, update)
Matthew Treinish864fe072014-03-02 03:47:26 +0000138
139
Matthew Treinishf0971712014-04-11 20:08:53 +0000140def verify_cinder_api_versions(os, update):
Matthew Treinish2e439632014-03-05 21:53:33 +0000141 # Check cinder api versions
142 versions = _get_api_versions(os, 'cinder')
David Kranz0df154d2015-06-02 17:02:27 -0400143 if (CONF.volume_feature_enabled.api_v1 !=
144 contains_version('v1.', versions)):
Nikita Gerasimov35fbdc12015-08-07 19:58:24 +0300145 print_and_or_update('api_v1', 'volume-feature-enabled',
Matthew Treinishf0971712014-04-11 20:08:53 +0000146 not CONF.volume_feature_enabled.api_v1, update)
David Kranz0df154d2015-06-02 17:02:27 -0400147 if (CONF.volume_feature_enabled.api_v2 !=
148 contains_version('v2.', versions)):
Nikita Gerasimov35fbdc12015-08-07 19:58:24 +0300149 print_and_or_update('api_v2', 'volume-feature-enabled',
Matthew Treinishf0971712014-04-11 20:08:53 +0000150 not CONF.volume_feature_enabled.api_v2, update)
Nikita Gerasimov3b169cc2016-08-10 17:10:42 +0300151 if (CONF.volume_feature_enabled.api_v3 !=
152 contains_version('v3.', versions)):
153 print_and_or_update('api_v3', 'volume-feature-enabled',
154 not CONF.volume_feature_enabled.api_v3, update)
Matthew Treinish2e439632014-03-05 21:53:33 +0000155
156
Adam Gandelman03af5562014-10-07 12:22:48 -0700157def verify_api_versions(os, service, update):
158 verify = {
159 'cinder': verify_cinder_api_versions,
160 'glance': verify_glance_api_versions,
161 'keystone': verify_keystone_api_versions,
Adam Gandelman03af5562014-10-07 12:22:48 -0700162 }
163 if service not in verify:
164 return
165 verify[service](os, update)
166
167
Matthew Treinish8b006d22014-01-07 15:37:20 +0000168def get_extension_client(os, service):
169 extensions_client = {
170 'nova': os.extensions_client,
Ken'ichi Ohmichi52bb8122016-01-26 01:43:06 +0000171 'neutron': os.network_extensions_client,
ghanshyamf29831d2016-12-12 18:45:23 +0900172 'swift': os.capabilities_client,
Ken'ichi Ohmichi8b876dd2017-05-04 14:30:31 -0700173 # NOTE: Cinder v3 API is current and v2 and v1 are deprecated.
174 # V3 extension API is the same as v2, so we reuse the v2 client
175 # for v3 API also.
176 'cinder': os.volumes_v2_extension_client,
Matthew Treinish8b006d22014-01-07 15:37:20 +0000177 }
Ivan Kolodyazhnybcfc32e2015-08-06 13:31:36 +0300178
Matthew Treinish8b006d22014-01-07 15:37:20 +0000179 if service not in extensions_client:
180 print('No tempest extensions client for %s' % service)
caoyueab333022016-01-25 16:45:21 +0800181 sys.exit(1)
Matthew Treinish8b006d22014-01-07 15:37:20 +0000182 return extensions_client[service]
183
184
185def get_enabled_extensions(service):
186 extensions_options = {
187 'nova': CONF.compute_feature_enabled.api_extensions,
Matthew Treinish8b006d22014-01-07 15:37:20 +0000188 'cinder': CONF.volume_feature_enabled.api_extensions,
Matthew Treinish8c6706d2014-01-07 19:28:18 +0000189 'neutron': CONF.network_feature_enabled.api_extensions,
Matthew Treinishc0120ba2014-01-31 20:10:19 +0000190 'swift': CONF.object_storage_feature_enabled.discoverable_apis,
Matthew Treinish8b006d22014-01-07 15:37:20 +0000191 }
192 if service not in extensions_options:
193 print('No supported extensions list option for %s' % service)
caoyueab333022016-01-25 16:45:21 +0800194 sys.exit(1)
Matthew Treinish8b006d22014-01-07 15:37:20 +0000195 return extensions_options[service]
196
197
198def verify_extensions(os, service, results):
199 extensions_client = get_extension_client(os, service)
David Kranz5cf4ba42015-02-10 14:00:50 -0500200 if service != 'swift':
David Kranz34e88122014-12-11 15:24:05 -0500201 resp = extensions_client.list_extensions()
202 else:
ghanshyam17d8a482017-07-21 03:10:48 +0000203 resp = extensions_client.list_capabilities()
Matthew Treinish54176ce2014-12-08 21:28:05 +0000204 # For Nova, Cinder and Neutron we use the alias name rather than the
205 # 'name' field because the alias is considered to be the canonical
206 # name.
Matthew Treinish8b006d22014-01-07 15:37:20 +0000207 if isinstance(resp, dict):
Matthew Treinish54176ce2014-12-08 21:28:05 +0000208 if service == 'swift':
Matthew Treinishc0120ba2014-01-31 20:10:19 +0000209 # Remove Swift general information from extensions list
210 resp.pop('swift')
211 extensions = resp.keys()
Matthew Treinish8c6706d2014-01-07 19:28:18 +0000212 else:
Matthew Treinish54176ce2014-12-08 21:28:05 +0000213 extensions = map(lambda x: x['alias'], resp['extensions'])
Matthew Treinish8c6706d2014-01-07 19:28:18 +0000214
Matthew Treinish8b006d22014-01-07 15:37:20 +0000215 else:
Matthew Treinish54176ce2014-12-08 21:28:05 +0000216 extensions = map(lambda x: x['alias'], resp)
Matthew Treinish09487242015-05-10 12:43:58 -0400217 extensions = list(extensions)
Matthew Treinish8b006d22014-01-07 15:37:20 +0000218 if not results.get(service):
219 results[service] = {}
220 extensions_opt = get_enabled_extensions(service)
221 if extensions_opt[0] == 'all':
Matthew Treinishf0971712014-04-11 20:08:53 +0000222 results[service]['extensions'] = extensions
Matthew Treinish8b006d22014-01-07 15:37:20 +0000223 return results
224 # Verify that all configured extensions are actually enabled
225 for extension in extensions_opt:
226 results[service][extension] = extension in extensions
227 # Verify that there aren't additional extensions enabled that aren't
228 # specified in the config list
229 for extension in extensions:
230 if extension not in extensions_opt:
231 results[service][extension] = False
Matthew Treinish1f7b33d2013-10-21 18:07:02 +0000232 return results
233
234
Matthew Treinishf0971712014-04-11 20:08:53 +0000235def display_results(results, update, replace):
236 update_dict = {
237 'swift': 'object-storage-feature-enabled',
238 'nova': 'compute-feature-enabled',
Matthew Treinishf0971712014-04-11 20:08:53 +0000239 'cinder': 'volume-feature-enabled',
240 'neutron': 'network-feature-enabled',
241 }
Matthew Treinish8b006d22014-01-07 15:37:20 +0000242 for service in results:
243 # If all extensions are specified as being enabled there is no way to
244 # verify this so we just assume this to be true
245 if results[service].get('extensions'):
Matthew Treinishf0971712014-04-11 20:08:53 +0000246 if replace:
247 output_list = results[service].get('extensions')
248 else:
249 output_list = ['all']
250 else:
251 extension_list = get_enabled_extensions(service)
252 output_list = []
253 for extension in results[service]:
254 if not results[service][extension]:
255 if extension in extension_list:
256 print("%s extension: %s should not be included in the "
257 "list of enabled extensions" % (service,
258 extension))
259 else:
260 print("%s extension: %s should be included in the list"
261 " of enabled extensions" % (service, extension))
262 output_list.append(extension)
Matthew Treinish8b006d22014-01-07 15:37:20 +0000263 else:
Matthew Treinishf0971712014-04-11 20:08:53 +0000264 output_list.append(extension)
265 if update:
266 # Sort List
267 output_list.sort()
268 # Convert list to a string
269 output_string = ', '.join(output_list)
270 if service == 'swift':
271 change_option('discoverable_apis', update_dict[service],
272 output_string)
Matthew Treinishf0971712014-04-11 20:08:53 +0000273 else:
274 change_option('api_extensions', update_dict[service],
275 output_string)
Matthew Treinish1f7b33d2013-10-21 18:07:02 +0000276
277
Matthew Treinishf0971712014-04-11 20:08:53 +0000278def check_service_availability(os, update):
Matthew Treinish221bd7f2014-02-07 21:16:09 +0000279 services = []
280 avail_services = []
281 codename_match = {
282 'volume': 'cinder',
283 'network': 'neutron',
284 'image': 'glance',
285 'object_storage': 'swift',
286 'compute': 'nova',
287 'orchestration': 'heat',
Matthew Treinish221bd7f2014-02-07 21:16:09 +0000288 'baremetal': 'ironic',
Matthew Treinish42d50f62014-04-11 19:47:13 +0000289 'identity': 'keystone',
Matthew Treinish221bd7f2014-02-07 21:16:09 +0000290 }
291 # Get catalog list for endpoints to use for validation
David Kranz45714082015-04-01 14:47:33 -0400292 _token, auth_data = os.auth_provider.get_auth()
David Kranz799eee12015-04-08 11:18:19 -0400293 if os.auth_version == 'v2':
294 catalog_key = 'serviceCatalog'
295 else:
296 catalog_key = 'catalog'
297 for entry in auth_data[catalog_key]:
David Kranz45714082015-04-01 14:47:33 -0400298 services.append(entry['type'])
Matthew Treinish221bd7f2014-02-07 21:16:09 +0000299 # Pull all catalog types from config file and compare against endpoint list
300 for cfgname in dir(CONF._config):
301 cfg = getattr(CONF, cfgname)
302 catalog_type = getattr(cfg, 'catalog_type', None)
303 if not catalog_type:
304 continue
305 else:
306 if cfgname == 'identity':
307 # Keystone is a required service for tempest
308 continue
309 if catalog_type not in services:
310 if getattr(CONF.service_available, codename_match[cfgname]):
311 print('Endpoint type %s not found either disable service '
312 '%s or fix the catalog_type in the config file' % (
Matthew Treinish96e9e882014-06-09 18:37:19 -0400313 catalog_type, codename_match[cfgname]))
Matthew Treinishf0971712014-04-11 20:08:53 +0000314 if update:
315 change_option(codename_match[cfgname],
316 'service_available', False)
Matthew Treinish221bd7f2014-02-07 21:16:09 +0000317 else:
318 if not getattr(CONF.service_available,
319 codename_match[cfgname]):
320 print('Endpoint type %s is available, service %s should be'
321 ' set as available in the config file.' % (
Matthew Treinish96e9e882014-06-09 18:37:19 -0400322 catalog_type, codename_match[cfgname]))
Matthew Treinishf0971712014-04-11 20:08:53 +0000323 if update:
324 change_option(codename_match[cfgname],
325 'service_available', True)
David Kranzf20ac322014-05-02 16:46:15 -0400326 # If we are going to enable this we should allow
327 # extension checks.
328 avail_services.append(codename_match[cfgname])
Matthew Treinish221bd7f2014-02-07 21:16:09 +0000329 else:
330 avail_services.append(codename_match[cfgname])
331 return avail_services
Matthew Treinishd44fe032014-01-31 20:07:24 +0000332
333
David Patersone45aa842015-11-18 16:12:27 -0800334def _parser_add_args(parser):
Matthew Treinishf0971712014-04-11 20:08:53 +0000335 parser.add_argument('-u', '--update', action='store_true',
336 help='Update the config file with results from api '
337 'queries. This assumes whatever is set in the '
338 'config file is incorrect. In the case of '
339 'endpoint checks where it could either be the '
340 'incorrect catalog type or the service available '
341 'option the service available option is assumed '
342 'to be incorrect and is thus changed')
343 parser.add_argument('-o', '--output',
344 help="Output file to write an updated config file to. "
345 "This has to be a separate file from the "
346 "original config file. If one isn't specified "
347 "with -u the new config file will be printed to "
348 "STDOUT")
349 parser.add_argument('-r', '--replace-ext', action='store_true',
350 help="If specified the all option will be replaced "
351 "with a full list of extensions")
Matthew Treinishf0971712014-04-11 20:08:53 +0000352
353
David Patersone45aa842015-11-18 16:12:27 -0800354def parse_args():
355 parser = argparse.ArgumentParser()
356 _parser_add_args(parser)
357 opts = parser.parse_args()
358 return opts
359
360
361def main(opts=None):
Matthew Treinish8b006d22014-01-07 15:37:20 +0000362 print('Running config verification...')
David Patersone45aa842015-11-18 16:12:27 -0800363 if opts is None:
364 print("Use of: 'verify-tempest-config' is deprecated, "
365 "please use: 'tempest verify-config'")
366 opts = parse_args()
Matthew Treinishf0971712014-04-11 20:08:53 +0000367 update = opts.update
368 replace = opts.replace_ext
David Kranzf20ac322014-05-02 16:46:15 -0400369 global CONF_PARSER
370
Matthew Treinishf0971712014-04-11 20:08:53 +0000371 if update:
David Kranzf20ac322014-05-02 16:46:15 -0400372 conf_file = _get_config_file()
Janonymous8254a3f2016-09-15 10:38:48 +0530373 CONF_PARSER = moves.configparser.ConfigParser()
David Kranzf20ac322014-05-02 16:46:15 -0400374 CONF_PARSER.optionxform = str
375 CONF_PARSER.readfp(conf_file)
Brad Behle43e4fd82016-04-13 17:15:21 -0500376
377 # Indicate not to create network resources as part of getting credentials
378 net_resources = {
379 'network': False,
380 'router': False,
381 'subnet': False,
382 'dhcp': False
383 }
384 icreds = credentials.get_credentials_provider(
385 'verify_tempest_config', network_resources=net_resources)
David Kranz5fcac942015-05-08 17:43:45 -0400386 try:
Andrea Frittoli (andreaf)848c4a12016-06-09 11:09:02 +0100387 os = clients.Manager(icreds.get_primary_creds().credentials)
David Kranz5fcac942015-05-08 17:43:45 -0400388 services = check_service_availability(os, update)
389 results = {}
390 for service in ['nova', 'cinder', 'neutron', 'swift']:
391 if service not in services:
392 continue
393 results = verify_extensions(os, service, results)
Adam Gandelman03af5562014-10-07 12:22:48 -0700394
David Kranz5fcac942015-05-08 17:43:45 -0400395 # Verify API versions of all services in the keystone catalog and
396 # keystone itself.
397 services.append('keystone')
398 for service in services:
399 verify_api_versions(os, service, update)
Adam Gandelman03af5562014-10-07 12:22:48 -0700400
David Kranz5fcac942015-05-08 17:43:45 -0400401 display_results(results, update, replace)
402 if update:
403 conf_file.close()
zhang.leia4b1cef2016-03-01 10:50:01 +0800404 if opts.output:
405 with open(opts.output, 'w+') as outfile:
406 CONF_PARSER.write(outfile)
David Kranz5fcac942015-05-08 17:43:45 -0400407 finally:
Andrea Frittoli (andreaf)17209bb2015-05-22 10:16:57 -0700408 icreds.clear_creds()
Matthew Treinish1f7b33d2013-10-21 18:07:02 +0000409
410
David Patersone45aa842015-11-18 16:12:27 -0800411class TempestVerifyConfig(command.Command):
412 """Verify your current tempest configuration"""
413
414 def get_parser(self, prog_name):
415 parser = super(TempestVerifyConfig, self).get_parser(prog_name)
416 _parser_add_args(parser)
417 return parser
418
419 def take_action(self, parsed_args):
420 try:
Masayuki Igawa92629432016-06-09 12:28:09 +0900421 main(parsed_args)
David Patersone45aa842015-11-18 16:12:27 -0800422 except Exception:
423 LOG.exception("Failure verifying configuration.")
424 traceback.print_exc()
425 raise
David Patersone45aa842015-11-18 16:12:27 -0800426
Matthew Treinish1f7b33d2013-10-21 18:07:02 +0000427if __name__ == "__main__":
Matthew Treinishf8b816a2014-04-23 20:35:49 +0000428 main()