| #!/usr/bin/python |
| # Copyright 2018 Mirantis, Inc. |
| # |
| # Licensed under the Apache License, Version 2.0 (the "License"); |
| # you may not use this file except in compliance with the License. |
| # You may obtain a copy of the License at |
| # |
| # http://www.apache.org/licenses/LICENSE-2.0 |
| # |
| # Unless required by applicable law or agreed to in writing, software |
| # distributed under the License is distributed on an "AS IS" BASIS, |
| # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| # See the License for the specific language governing permissions and |
| # limitations under the License. |
| import logging |
| import os |
| import subprocess |
| import time |
| |
| # Import Salt Libs |
| import salt.utils.http |
| |
| |
| MODULE_NAME = 'contrail_health' |
| LOG = logging.getLogger(__name__) |
| |
| |
| def __virtual__(): |
| ''' |
| Only load this module if contrail-status or doctrail utility |
| (in case of containerized contrail version) is available. |
| ''' |
| if _is_cmd_available('contrail-status') or _is_cmd_available('doctrail'): |
| return MODULE_NAME |
| return False |
| |
| |
| def _is_cmd_available(cmd_name): |
| try: |
| with open(os.devnull) as devnull: |
| subprocess.Popen( |
| [cmd_name], stdout=devnull, stderr=devnull |
| ).communicate() |
| except OSError as e: |
| if e.errno == os.errno.ENOENT: |
| return False |
| return True |
| |
| |
| def get_services_status(): |
| cs_out = None |
| |
| if _is_cmd_available('contrail-status'): |
| LOG.info('Trying to get status of contrail services ' |
| 'using contrail-status utility on host ...') |
| try: |
| cs_out = str(subprocess.check_output(['contrail-status'])) |
| except subprocess.CalledProcessError as e: |
| LOG.warn('Status of contrail services cannot be checked ' |
| 'by contrail-status utility from host') |
| if cs_out is None and _is_cmd_available('doctrail'): |
| LOG.info('Trying to get status of contrail services inside containers ' |
| 'using doctrail utility ...') |
| try: |
| cs_out = str(subprocess.check_output( |
| ['doctrail', 'all', 'contrail-status']) |
| ) |
| except subprocess.CalledProcessError as e: |
| LOG.warn('Status of contrail services inside containers cannot ' |
| 'be checked by contrail-status utility via doctrail cmd') |
| |
| status_map = {} |
| |
| if cs_out: |
| for line in cs_out.split('\n'): |
| line_list = line.split() |
| if (not line.startswith("==") and "FOR NODE" not in line and |
| len(line_list) >= 2): |
| status_map[line_list[0].split(":")[0]] = line_list[1] |
| else: |
| LOG.error('Status of contrail services cannot be checked ' |
| 'by {0} module.'.format(MODULE_NAME)) |
| |
| return status_map |
| |
| ''' |
| Check status of Contail API service on Virtual IP which is defined by pillars. |
| |
| CLI Example: |
| |
| .. code-block:: bash |
| |
| salt 'ntw01*' contrail_health.get_api_status [wait_for=300] \\ |
| [tries=20] |
| |
| wait_for |
| Number of seconds how long to wait for API response. |
| |
| tries |
| Number of tries. After each unsuccessful try will sleep for \\ |
| (wait_for/tries). |
| ''' |
| |
| |
| def get_api_status(wait_for=180, tries=20): |
| api_host = __pillar__['opencontrail'].get('client', {}).get('api', {}).\ |
| get('host', {}) |
| api_port = __pillar__['opencontrail']['client']['api']['port'] |
| use_ssl = __pillar__['opencontrail'].get('client', {}).get('api', {}).\ |
| get('use_ssl', {}) |
| api_protocol = 'https' if use_ssl is True else 'http' |
| |
| for t in range(0, tries): |
| try: |
| data = salt.utils.http.query( |
| "{0}://{1}:{2}".format(api_protocol, api_host, api_port), |
| backend='requests', status=True |
| ) |
| if data['status'] == 200: |
| return True |
| except Exception as e: |
| LOG.error(e.message) |
| time.sleep(int(wait_for / tries)) |
| |
| return False |