outposts: ensure all Service Connection state updates are done by the task

This commit is contained in:
Jens Langhammer 2020-12-15 23:39:52 +01:00
parent d9956e1e9c
commit 42005e7def
2 changed files with 21 additions and 13 deletions

View file

@ -35,6 +35,7 @@ from authentik.lib.models import InheritanceForeignKey
from authentik.lib.sentry import SentryIgnoredException from authentik.lib.sentry import SentryIgnoredException
from authentik.lib.utils.template import render_to_string from authentik.lib.utils.template import render_to_string
from authentik.outposts.docker_tls import DockerInlineTLS from authentik.outposts.docker_tls import DockerInlineTLS
from authentik.outposts.tasks import outpost_service_connection_state
OUR_VERSION = parse(__version__) OUR_VERSION = parse(__version__)
OUTPOST_HELLO_INTERVAL = 10 OUTPOST_HELLO_INTERVAL = 10
@ -113,17 +114,22 @@ class OutpostServiceConnection(models.Model):
objects = InheritanceManager() objects = InheritanceManager()
@property
def state_key(self) -> str:
"""Key used to save connection state in cache"""
return f"outpost_service_connection_{self.pk.hex}"
@property @property
def state(self) -> OutpostServiceConnectionState: def state(self) -> OutpostServiceConnectionState:
"""Get state of service connection""" """Get state of service connection"""
state_key = f"outpost_service_connection_{self.pk.hex}" state = cache.get(self.state_key, None)
state = cache.get(state_key, None)
if not state: if not state:
state = self._get_state() outpost_service_connection_state.delay(self.pk)
cache.set(state_key, state, timeout=0) return OutpostServiceConnectionState("", False)
return state return state
def _get_state(self) -> OutpostServiceConnectionState: def fetch_state(self) -> OutpostServiceConnectionState:
"""Fetch current Service Connection state"""
raise NotImplementedError raise NotImplementedError
@property @property
@ -203,7 +209,7 @@ class DockerServiceConnection(OutpostServiceConnection):
raise ServiceConnectionInvalid from exc raise ServiceConnectionInvalid from exc
return client return client
def _get_state(self) -> OutpostServiceConnectionState: def fetch_state(self) -> OutpostServiceConnectionState:
try: try:
client = self.client() client = self.client()
return OutpostServiceConnectionState( return OutpostServiceConnectionState(
@ -239,7 +245,7 @@ class KubernetesServiceConnection(OutpostServiceConnection):
def __str__(self) -> str: def __str__(self) -> str:
return f"Kubernetes Service-Connection {self.name}" return f"Kubernetes Service-Connection {self.name}"
def _get_state(self) -> OutpostServiceConnectionState: def fetch_state(self) -> OutpostServiceConnectionState:
try: try:
client = self.client() client = self.client()
api_instance = VersionApi(client) api_instance = VersionApi(client)

View file

@ -35,21 +35,23 @@ def outpost_controller_all():
@CELERY_APP.task() @CELERY_APP.task()
def outpost_service_connection_state(state_pk: Any): def outpost_service_connection_state(connection_pk: Any):
"""Update cached state of a service connection""" """Update cached state of a service connection"""
connection: OutpostServiceConnection = ( connection: OutpostServiceConnection = (
OutpostServiceConnection.objects.filter(pk=state_pk).select_subclasses().first() OutpostServiceConnection.objects.filter(pk=connection_pk)
.select_subclasses()
.first()
) )
cache.delete(f"outpost_service_connection_{connection.pk.hex}") cache.delete(f"outpost_service_connection_{connection.pk.hex}")
_ = connection.state state = connection.fetch_state()
cache.set(connection.state_key, state, timeout=0)
@CELERY_APP.task(bind=True, base=MonitoredTask) @CELERY_APP.task(bind=True, base=MonitoredTask)
def outpost_service_connection_monitor(self: MonitoredTask): def outpost_service_connection_monitor(self: MonitoredTask):
"""Regularly check the state of Outpost Service Connections""" """Regularly check the state of Outpost Service Connections"""
for connection in OutpostServiceConnection.objects.select_subclasses(): for connection in OutpostServiceConnection.objects.all():
cache.delete(f"outpost_service_connection_{connection.pk.hex}") outpost_service_connection_state.delay(connection.pk)
_ = connection.state
self.set_status(TaskResult(TaskResultStatus.SUCCESSFUL)) self.set_status(TaskResult(TaskResultStatus.SUCCESSFUL))