2016-09-26 01:05:22 +03:00
|
|
|
# Copyright (c) 2016 Mirantis, Inc.
|
|
|
|
# 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.
|
2018-04-23 12:49:37 +02:00
|
|
|
import functools
|
2016-09-26 01:05:22 +03:00
|
|
|
import sys
|
|
|
|
|
2020-04-08 12:53:56 +02:00
|
|
|
import urllib3
|
|
|
|
|
2016-11-16 15:19:11 +03:00
|
|
|
import os_vif
|
2018-03-23 11:46:19 +01:00
|
|
|
from oslo_config import cfg
|
2016-09-26 01:05:22 +03:00
|
|
|
from oslo_log import log as logging
|
2018-04-23 12:49:37 +02:00
|
|
|
from oslo_service import periodic_task
|
2016-09-26 01:05:22 +03:00
|
|
|
from oslo_service import service
|
2018-03-23 11:46:19 +01:00
|
|
|
from stevedore.named import NamedExtensionManager
|
2016-09-26 01:05:22 +03:00
|
|
|
|
|
|
|
from kuryr_kubernetes import clients
|
|
|
|
from kuryr_kubernetes import config
|
2018-08-20 17:49:52 +02:00
|
|
|
from kuryr_kubernetes.controller.drivers import base as drivers
|
2016-10-28 08:39:31 +03:00
|
|
|
from kuryr_kubernetes.controller.handlers import pipeline as h_pipeline
|
2017-12-20 14:15:53 +00:00
|
|
|
from kuryr_kubernetes.controller.managers import health
|
2020-05-06 21:50:30 +00:00
|
|
|
from kuryr_kubernetes.controller.managers import prometheus_exporter as exp
|
2016-12-14 12:58:58 +05:30
|
|
|
from kuryr_kubernetes import objects
|
2018-04-23 12:49:37 +02:00
|
|
|
from kuryr_kubernetes import utils
|
2016-09-26 00:57:07 +03:00
|
|
|
from kuryr_kubernetes import watcher
|
2016-09-26 01:05:22 +03:00
|
|
|
|
2018-03-23 11:46:19 +01:00
|
|
|
|
|
|
|
CONF = cfg.CONF
|
2016-09-26 01:05:22 +03:00
|
|
|
LOG = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
|
2018-03-23 11:46:19 +01:00
|
|
|
def _handler_not_found(names):
|
|
|
|
LOG.exception('Handlers "%s" were not found.', names)
|
|
|
|
LOG.critical('Handlers "%s" were not found.', names)
|
|
|
|
raise SystemExit()
|
|
|
|
|
|
|
|
|
|
|
|
def _handler_not_loaded(manager, entrypoint, exception):
|
|
|
|
LOG.exception('Exception when loading handlers %s.', entrypoint)
|
|
|
|
LOG.critical('Handlers entrypoint "%s" failed to load due to %s.',
|
|
|
|
entrypoint, exception)
|
|
|
|
raise SystemExit()
|
|
|
|
|
|
|
|
|
|
|
|
def _load_kuryr_ctrlr_handlers():
|
|
|
|
configured_handlers = CONF.kubernetes.enabled_handlers
|
|
|
|
LOG.info('Configured handlers: %s', configured_handlers)
|
|
|
|
handlers = NamedExtensionManager(
|
|
|
|
'kuryr_kubernetes.controller.handlers',
|
|
|
|
configured_handlers,
|
|
|
|
invoke_on_load=True,
|
|
|
|
on_missing_entrypoints_callback=_handler_not_found,
|
|
|
|
on_load_failure_callback=_handler_not_loaded)
|
|
|
|
LOG.info('Loaded handlers: %s', handlers.names())
|
|
|
|
ctrlr_handlers = []
|
|
|
|
for handler in handlers.extensions:
|
|
|
|
ctrlr_handlers.append(handler.obj)
|
|
|
|
return ctrlr_handlers
|
|
|
|
|
|
|
|
|
2018-04-23 12:49:37 +02:00
|
|
|
class KuryrK8sServiceMeta(type(service.Service),
|
|
|
|
type(periodic_task.PeriodicTasks)):
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
2020-02-28 12:00:19 +01:00
|
|
|
class KuryrK8sService(service.Service, periodic_task.PeriodicTasks,
|
|
|
|
metaclass=KuryrK8sServiceMeta):
|
2016-09-26 01:05:22 +03:00
|
|
|
"""Kuryr-Kubernetes controller Service."""
|
|
|
|
|
|
|
|
def __init__(self):
|
|
|
|
super(KuryrK8sService, self).__init__()
|
2018-04-23 12:49:37 +02:00
|
|
|
periodic_task.PeriodicTasks.__init__(self, CONF)
|
2016-09-26 01:05:22 +03:00
|
|
|
|
2016-12-14 12:58:58 +05:30
|
|
|
objects.register_locally_defined_vifs()
|
2016-10-28 08:39:31 +03:00
|
|
|
pipeline = h_pipeline.ControllerPipeline(self.tg)
|
2019-08-28 13:38:24 +02:00
|
|
|
self.watcher = watcher.Watcher(pipeline, self.tg)
|
2018-01-19 00:31:25 +00:00
|
|
|
self.health_manager = health.HealthServer()
|
2020-05-06 21:50:30 +00:00
|
|
|
self.exporter = exp.ControllerPrometheusExporter.get_instance()
|
2018-04-23 12:49:37 +02:00
|
|
|
self.current_leader = None
|
|
|
|
self.node_name = utils.get_node_name()
|
2018-03-23 11:46:19 +01:00
|
|
|
|
2021-08-09 02:48:07 +00:00
|
|
|
self.handlers = _load_kuryr_ctrlr_handlers()
|
|
|
|
for handler in self.handlers:
|
2018-03-23 11:46:19 +01:00
|
|
|
self.watcher.add(handler.get_watch_path())
|
|
|
|
pipeline.register(handler)
|
2018-08-20 17:49:52 +02:00
|
|
|
self.pool_driver = drivers.VIFPoolDriver.get_instance(
|
2018-08-27 17:09:38 +09:00
|
|
|
specific_driver='multi_pool')
|
2018-08-20 17:49:52 +02:00
|
|
|
self.pool_driver.set_vif_driver()
|
2016-09-26 00:57:07 +03:00
|
|
|
|
2018-04-23 12:49:37 +02:00
|
|
|
def is_leader(self):
|
|
|
|
return self.current_leader == self.node_name
|
|
|
|
|
2016-09-26 01:05:22 +03:00
|
|
|
def start(self):
|
2017-03-28 09:47:09 +08:00
|
|
|
LOG.info("Service '%s' starting", self.__class__.__name__)
|
2016-09-26 01:05:22 +03:00
|
|
|
super(KuryrK8sService, self).start()
|
2018-04-23 12:49:37 +02:00
|
|
|
|
|
|
|
if not CONF.kubernetes.controller_ha:
|
|
|
|
LOG.info('Running in non-HA mode, starting watcher immediately.')
|
|
|
|
self.watcher.start()
|
2018-08-20 17:49:52 +02:00
|
|
|
self.pool_driver.sync_pools()
|
2018-04-23 12:49:37 +02:00
|
|
|
else:
|
|
|
|
LOG.info('Running in HA mode, watcher will be started later.')
|
2021-08-09 02:48:07 +00:00
|
|
|
f = functools.partial(self.run_periodic_tasks, None)
|
|
|
|
self.tg.add_timer(1, f)
|
2018-04-23 12:49:37 +02:00
|
|
|
|
2020-05-06 21:50:30 +00:00
|
|
|
self.tg.add_thread(self.exporter.run)
|
|
|
|
self.tg.add_thread(self.health_manager.run)
|
2017-03-28 09:47:09 +08:00
|
|
|
LOG.info("Service '%s' started", self.__class__.__name__)
|
2016-09-26 01:05:22 +03:00
|
|
|
|
2018-04-23 12:49:37 +02:00
|
|
|
@periodic_task.periodic_task(spacing=5, run_immediately=True)
|
|
|
|
def monitor_leader(self, context):
|
2021-08-09 02:48:07 +00:00
|
|
|
if not CONF.kubernetes.controller_ha:
|
|
|
|
return
|
2018-04-23 12:49:37 +02:00
|
|
|
leader = utils.get_leader_name()
|
|
|
|
if leader is None:
|
|
|
|
# Error when fetching current leader. We're paranoid, so just to
|
|
|
|
# make sure we won't break anything we'll try to step down.
|
|
|
|
self.on_revoke_leader()
|
|
|
|
elif leader != self.current_leader and leader == self.node_name:
|
|
|
|
# I'm becoming the leader.
|
|
|
|
self.on_become_leader()
|
|
|
|
elif leader != self.current_leader and self.is_leader():
|
|
|
|
# I'm revoked from being the leader.
|
|
|
|
self.on_revoke_leader()
|
|
|
|
elif leader == self.current_leader and self.is_leader():
|
|
|
|
# I continue to be the leader
|
|
|
|
self.on_continue_leader()
|
|
|
|
|
|
|
|
self.current_leader = leader
|
|
|
|
|
|
|
|
def on_become_leader(self):
|
|
|
|
LOG.info('Controller %s becomes the leader, starting watcher.',
|
|
|
|
self.node_name)
|
|
|
|
self.watcher.start()
|
2018-08-20 17:49:52 +02:00
|
|
|
self.pool_driver.sync_pools()
|
2018-04-23 12:49:37 +02:00
|
|
|
|
|
|
|
def on_revoke_leader(self):
|
|
|
|
LOG.info('Controller %s stops being the leader, stopping watcher.',
|
|
|
|
self.node_name)
|
|
|
|
if self.watcher.is_running():
|
|
|
|
self.watcher.stop()
|
|
|
|
|
|
|
|
def on_continue_leader(self):
|
|
|
|
# Just make sure my watcher is running.
|
|
|
|
if not self.watcher.is_running():
|
|
|
|
LOG.warning('Controller %s is the leader, but has watcher '
|
2020-01-03 16:21:13 +08:00
|
|
|
'stopped. Restarting it.', self.node_name)
|
2018-04-23 12:49:37 +02:00
|
|
|
self.watcher.start()
|
|
|
|
|
2016-09-26 01:05:22 +03:00
|
|
|
def wait(self):
|
|
|
|
super(KuryrK8sService, self).wait()
|
2017-03-28 09:47:09 +08:00
|
|
|
LOG.info("Service '%s' stopped", self.__class__.__name__)
|
2016-09-26 01:05:22 +03:00
|
|
|
|
|
|
|
def stop(self, graceful=False):
|
2017-03-28 09:47:09 +08:00
|
|
|
LOG.info("Service '%s' stopping", self.__class__.__name__)
|
2016-09-26 00:57:07 +03:00
|
|
|
self.watcher.stop()
|
2016-09-26 01:05:22 +03:00
|
|
|
super(KuryrK8sService, self).stop(graceful)
|
|
|
|
|
2021-08-09 02:48:07 +00:00
|
|
|
@periodic_task.periodic_task(spacing=600, run_immediately=False)
|
|
|
|
def reconcile_loadbalancers(self, context):
|
|
|
|
LOG.debug("Checking for Kubernetes resources reconciliations")
|
|
|
|
for handler in self.handlers:
|
|
|
|
handler.reconcile()
|
|
|
|
|
2022-04-15 13:11:00 +02:00
|
|
|
@periodic_task.periodic_task(spacing=90, run_immediately=False)
|
2022-05-09 13:56:11 +02:00
|
|
|
def cleanup_dead_resources(self, context):
|
2022-04-15 13:11:00 +02:00
|
|
|
utils.cleanup_dead_ports()
|
2022-05-09 13:56:11 +02:00
|
|
|
utils.cleanup_dead_networks()
|
2022-04-15 13:11:00 +02:00
|
|
|
|
2016-09-26 01:05:22 +03:00
|
|
|
|
|
|
|
def start():
|
2020-04-08 12:53:56 +02:00
|
|
|
urllib3.disable_warnings()
|
2016-09-26 01:05:22 +03:00
|
|
|
config.init(sys.argv[1:])
|
|
|
|
config.setup_logging()
|
|
|
|
clients.setup_clients()
|
2016-11-16 15:19:11 +03:00
|
|
|
os_vif.initialize()
|
2016-09-26 01:05:22 +03:00
|
|
|
kuryrk8s_launcher = service.launch(config.CONF, KuryrK8sService())
|
|
|
|
kuryrk8s_launcher.wait()
|