# Copyright 2011 OpenStack Foundation.
# All Rights Reserved.
#
#    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.

from __future__ import annotations

import logging
import os
import typing as ty

from oslo_middleware.healthcheck import opts
from oslo_middleware.healthcheck import pluginbase

if ty.TYPE_CHECKING:
    from oslo_config import cfg

LOG = logging.getLogger(__name__)


class DisableByFilesPortsHealthcheck(pluginbase.HealthcheckBaseExtension):
    """DisableByFilesPorts healthcheck middleware plugin

    This plugin checks presence of a file that is provided for a application
    running on a certain port to report if the service is unavailable
    or not.

    Example of middleware configuration:

    .. code-block:: ini

      [filter:healthcheck]
      paste.filter_factory = oslo_middleware:Healthcheck.factory
      path = /healthcheck
      backends = disable_by_files_ports
      disable_by_file_paths = 5000:/var/run/keystone/healthcheck_disable, \
            35357:/var/run/keystone/admin_healthcheck_disable
      # set to True to enable detailed output, False is the default
      detailed = False
    """

    def __init__(
        self,
        oslo_conf: cfg.ConfigOpts,
        conf: dict[str, ty.Any],
    ) -> None:
        super().__init__(oslo_conf, conf)
        self.oslo_conf.register_opts(
            opts.DISABLE_BY_FILES_OPTS, group='healthcheck'
        )
        self.status_files: dict[int, str] = {}
        paths = self._conf_get('disable_by_file_paths')
        self.status_files.update(self._iter_paths_ports(paths))

    @staticmethod
    def _iter_paths_ports(
        paths: list[str],
    ) -> ty.Generator[tuple[int, str], None, None]:
        for port_path in paths:
            port_path = port_path.strip()
            if port_path:
                # On windows, drive letters are followed by colons,
                # which makes split() return 3 elements in this case
                port, path = port_path.split(":", 1)
                yield (int(port), path)

    def healthcheck(self, server_port: int) -> pluginbase.HealthcheckResult:
        path = self.status_files.get(server_port)
        if not path:
            LOG.warning(
                'DisableByFilesPorts healthcheck middleware '
                'enabled without disable_by_file_paths set '
                'for port %s',
                server_port,
            )
            return pluginbase.HealthcheckResult(available=True, reason="OK")
        else:
            if not os.path.exists(path):
                return pluginbase.HealthcheckResult(
                    available=True, reason="OK"
                )
            else:
                return pluginbase.HealthcheckResult(
                    available=False, reason="DISABLED BY FILE"
                )


class DisableByFileHealthcheck(pluginbase.HealthcheckBaseExtension):
    """DisableByFile healthcheck middleware plugin

    This plugin checks presence of a file to report if the service
    is unavailable or not.

    Example of middleware configuration:

    .. code-block:: ini

      [filter:healthcheck]
      paste.filter_factory = oslo_middleware:Healthcheck.factory
      path = /healthcheck
      backends = disable_by_file
      disable_by_file_path = /var/run/nova/healthcheck_disable
      # set to True to enable detailed output, False is the default
      detailed = False
    """

    def __init__(
        self,
        oslo_conf: cfg.ConfigOpts,
        conf: dict[str, ty.Any],
    ) -> None:
        super().__init__(oslo_conf, conf)
        self.oslo_conf.register_opts(
            opts.DISABLE_BY_FILE_OPTS, group='healthcheck'
        )

    def healthcheck(self, server_port: int) -> pluginbase.HealthcheckResult:
        path = self._conf_get('disable_by_file_path')
        if not path:
            LOG.warning(
                'DisableByFile healthcheck middleware enabled '
                'without disable_by_file_path set'
            )
            return pluginbase.HealthcheckResult(
                available=True,
                reason="OK",
                details="No 'disable_by_file_path' configuration value"
                " specified",
            )
        elif not os.path.exists(path):
            return pluginbase.HealthcheckResult(
                available=True,
                reason="OK",
                details=f"Path '{path}' was not found",
            )
        else:
            return pluginbase.HealthcheckResult(
                available=False,
                reason="DISABLED BY FILE",
                details=f"Path '{path}' was found",
            )
