![gcp-cherry-pick-bot[bot]](/assets/img/avatar_default.png)
sources/ldap: fix missing search attribute (#11125) * unrelated * sources/ldap: fix ldap sync not requesting uniqueness attribute * check object_uniqueness_field for none --------- Signed-off-by: Jens Langhammer <jens@goauthentik.io> Co-authored-by: Jens L. <jens@goauthentik.io>
109 lines
4.4 KiB
Python
109 lines
4.4 KiB
Python
"""Sync LDAP Users into authentik"""
|
|
|
|
from collections.abc import Generator
|
|
|
|
from django.core.exceptions import FieldError
|
|
from django.db.utils import IntegrityError
|
|
from ldap3 import ALL_ATTRIBUTES, ALL_OPERATIONAL_ATTRIBUTES, SUBTREE
|
|
|
|
from authentik.core.expression.exceptions import (
|
|
PropertyMappingExpressionException,
|
|
SkipObjectException,
|
|
)
|
|
from authentik.core.models import User
|
|
from authentik.core.sources.mapper import SourceMapper
|
|
from authentik.events.models import Event, EventAction
|
|
from authentik.lib.sync.outgoing.exceptions import StopSync
|
|
from authentik.sources.ldap.models import LDAP_UNIQUENESS, LDAPSource, flatten
|
|
from authentik.sources.ldap.sync.base import BaseLDAPSynchronizer
|
|
from authentik.sources.ldap.sync.vendor.freeipa import FreeIPA
|
|
from authentik.sources.ldap.sync.vendor.ms_ad import MicrosoftActiveDirectory
|
|
|
|
|
|
class UserLDAPSynchronizer(BaseLDAPSynchronizer):
|
|
"""Sync LDAP Users into authentik"""
|
|
|
|
def __init__(self, source: LDAPSource):
|
|
super().__init__(source)
|
|
self.mapper = SourceMapper(source)
|
|
self.manager = self.mapper.get_manager(User, ["ldap", "dn"])
|
|
|
|
@staticmethod
|
|
def name() -> str:
|
|
return "users"
|
|
|
|
def get_objects(self, **kwargs) -> Generator:
|
|
if not self._source.sync_users:
|
|
self.message("User syncing is disabled for this Source")
|
|
return iter(())
|
|
return self.search_paginator(
|
|
search_base=self.base_dn_users,
|
|
search_filter=self._source.user_object_filter,
|
|
search_scope=SUBTREE,
|
|
attributes=[
|
|
ALL_ATTRIBUTES,
|
|
ALL_OPERATIONAL_ATTRIBUTES,
|
|
self._source.object_uniqueness_field,
|
|
],
|
|
**kwargs,
|
|
)
|
|
|
|
def sync(self, page_data: list) -> int:
|
|
"""Iterate over all LDAP Users and create authentik_core.User instances"""
|
|
if not self._source.sync_users:
|
|
self.message("User syncing is disabled for this Source")
|
|
return -1
|
|
user_count = 0
|
|
for user in page_data:
|
|
if "attributes" not in user:
|
|
continue
|
|
attributes = user.get("attributes", {})
|
|
user_dn = flatten(user.get("entryDN", user.get("dn")))
|
|
if not attributes.get(self._source.object_uniqueness_field):
|
|
self.message(
|
|
f"Uniqueness field not found/not set in attributes: '{user_dn}'",
|
|
attributes=attributes.keys(),
|
|
dn=user_dn,
|
|
)
|
|
continue
|
|
uniq = flatten(attributes[self._source.object_uniqueness_field])
|
|
try:
|
|
defaults = {
|
|
k: flatten(v)
|
|
for k, v in self.mapper.build_object_properties(
|
|
object_type=User,
|
|
manager=self.manager,
|
|
user=None,
|
|
request=None,
|
|
dn=user_dn,
|
|
ldap=attributes,
|
|
).items()
|
|
}
|
|
self._logger.debug("Writing user with attributes", **defaults)
|
|
if "username" not in defaults:
|
|
raise IntegrityError("Username was not set by propertymappings")
|
|
ak_user, created = User.update_or_create_attributes(
|
|
{f"attributes__{LDAP_UNIQUENESS}": uniq}, defaults
|
|
)
|
|
except PropertyMappingExpressionException as exc:
|
|
raise StopSync(exc, None, exc.mapping) from exc
|
|
except SkipObjectException:
|
|
continue
|
|
except (IntegrityError, FieldError, TypeError, AttributeError) as exc:
|
|
Event.new(
|
|
EventAction.CONFIGURATION_ERROR,
|
|
message=(
|
|
f"Failed to create user: {str(exc)} "
|
|
"To merge new user with existing user, set the user's "
|
|
f"Attribute '{LDAP_UNIQUENESS}' to '{uniq}'"
|
|
),
|
|
source=self._source,
|
|
dn=user_dn,
|
|
).save()
|
|
else:
|
|
self._logger.debug("Synced User", user=ak_user.username, created=created)
|
|
user_count += 1
|
|
MicrosoftActiveDirectory(self._source).sync(attributes, ak_user, created)
|
|
FreeIPA(self._source).sync(attributes, ak_user, created)
|
|
return user_count
|