
* initial Signed-off-by: Jens Langhammer <jens@goauthentik.io> * add entra mappings Signed-off-by: Jens Langhammer <jens@goauthentik.io> * fix some stuff Signed-off-by: Jens Langhammer <jens@goauthentik.io> * make API endpoints more consistent Signed-off-by: Jens Langhammer <jens@goauthentik.io> * implement more things Signed-off-by: Jens Langhammer <jens@goauthentik.io> * add user tests Signed-off-by: Jens Langhammer <jens@goauthentik.io> * fix most group tests + fix bugs Signed-off-by: Jens Langhammer <jens@goauthentik.io> * more group tests, fix bugs Signed-off-by: Jens Langhammer <jens@goauthentik.io> * fix missing __init__ Signed-off-by: Jens Langhammer <jens@goauthentik.io> * add ui for provisioned users Signed-off-by: Jens Langhammer <jens@goauthentik.io> * fix a bunch of bugs Signed-off-by: Jens Langhammer <jens@goauthentik.io> * add `creating` to property mapping env Signed-off-by: Jens Langhammer <jens@goauthentik.io> * always sync group members Signed-off-by: Jens Langhammer <jens@goauthentik.io> * fix stuff Signed-off-by: Jens Langhammer <jens@goauthentik.io> * fix group membership Signed-off-by: Jens Langhammer <jens@goauthentik.io> * fix some types Signed-off-by: Jens Langhammer <jens@goauthentik.io> * fix tests Signed-off-by: Jens Langhammer <jens@goauthentik.io> * add group member add test Signed-off-by: Jens Langhammer <jens@goauthentik.io> * create sync status component to dedupe Signed-off-by: Jens Langhammer <jens@goauthentik.io> * fix discovery tests Signed-off-by: Jens Langhammer <jens@goauthentik.io> * get rid of more code and fix more issues Signed-off-by: Jens Langhammer <jens@goauthentik.io> * add error handling for auth and transient Signed-off-by: Jens Langhammer <jens@goauthentik.io> * make sure autoretry is on Signed-off-by: Jens Langhammer <jens@goauthentik.io> * format web Signed-off-by: Jens Langhammer <jens@goauthentik.io> * wait for task in signal Signed-off-by: Jens Langhammer <jens@goauthentik.io> * fix tests Signed-off-by: Jens Langhammer <jens@goauthentik.io> * add squashed google migration Signed-off-by: Jens Langhammer <jens@goauthentik.io> --------- Signed-off-by: Jens Langhammer <jens@goauthentik.io>
83 lines
2.6 KiB
Python
83 lines
2.6 KiB
Python
"""Basic outgoing sync Client"""
|
|
|
|
from enum import StrEnum
|
|
from typing import TYPE_CHECKING
|
|
|
|
from django.db import DatabaseError
|
|
from structlog.stdlib import get_logger
|
|
|
|
from authentik.lib.sync.outgoing.exceptions import NotFoundSyncException
|
|
|
|
if TYPE_CHECKING:
|
|
from django.db.models import Model
|
|
|
|
from authentik.lib.sync.outgoing.models import OutgoingSyncProvider
|
|
|
|
|
|
class Direction(StrEnum):
|
|
|
|
add = "add"
|
|
remove = "remove"
|
|
|
|
|
|
class BaseOutgoingSyncClient[
|
|
TModel: "Model", TConnection: "Model", TSchema: dict, TProvider: "OutgoingSyncProvider"
|
|
]:
|
|
"""Basic Outgoing sync client Client"""
|
|
|
|
provider: TProvider
|
|
connection_type: type[TConnection]
|
|
connection_type_query: str
|
|
|
|
can_discover = False
|
|
|
|
def __init__(self, provider: TProvider):
|
|
self.logger = get_logger().bind(provider=provider.name)
|
|
self.provider = provider
|
|
|
|
def create(self, obj: TModel) -> TConnection:
|
|
"""Create object in remote destination"""
|
|
raise NotImplementedError()
|
|
|
|
def update(self, obj: TModel, connection: TConnection):
|
|
"""Update object in remote destination"""
|
|
raise NotImplementedError()
|
|
|
|
def write(self, obj: TModel) -> tuple[TConnection, bool]:
|
|
"""Write object to destination. Uses self.create and self.update, but
|
|
can be overwritten for further logic"""
|
|
connection = self.connection_type.objects.filter(
|
|
provider=self.provider, **{self.connection_type_query: obj}
|
|
).first()
|
|
try:
|
|
if not connection:
|
|
connection = self.create(obj)
|
|
return connection, True
|
|
try:
|
|
self.update(obj, connection)
|
|
return connection, False
|
|
except NotFoundSyncException:
|
|
connection.delete()
|
|
connection = self.create(obj)
|
|
return connection, True
|
|
except DatabaseError as exc:
|
|
self.logger.warning("Failed to write object", obj=obj, exc=exc)
|
|
if connection:
|
|
connection.delete()
|
|
return None, False
|
|
|
|
def delete(self, obj: TModel):
|
|
"""Delete object from destination"""
|
|
raise NotImplementedError()
|
|
|
|
def to_schema(self, obj: TModel, creating: bool) -> TSchema:
|
|
"""Convert object to destination schema"""
|
|
raise NotImplementedError()
|
|
|
|
def discover(self):
|
|
"""Optional method. Can be used to implement a "discovery" where
|
|
upon creation of this provider, this function will be called and can
|
|
pre-link any users/groups in the remote system with the respective
|
|
object in authentik based on a common identifier"""
|
|
raise NotImplementedError()
|