mirror of
https://github.com/zulip/zulip.git
synced 2025-10-24 00:23:49 +00:00
Due to the atomic(savepoint=False) here, an LDAP sync exception while
syncing a single user breaks the whole sync_ldap_user_data transaction,
preventing it from successfully syncing other users.
Fixes a regression introduced in
1eecbad381
Closes #35291.
3662 lines
150 KiB
Python
3662 lines
150 KiB
Python
# Documentation for Zulip's authentication backends is split across a few places:
|
|
#
|
|
# * https://zulip.readthedocs.io/en/latest/production/authentication-methods.html and
|
|
# zproject/prod_settings_template.py have user-level configuration documentation.
|
|
# * https://zulip.readthedocs.io/en/latest/development/authentication.html
|
|
# has developer-level documentation, especially on testing authentication backends
|
|
# in the Zulip development environment.
|
|
#
|
|
# Django upstream's documentation for authentication backends is also
|
|
# helpful background. The most important detail to understand for
|
|
# reading this file is that the Django authenticate() function will
|
|
# call the authenticate methods of all backends registered in
|
|
# settings.AUTHENTICATION_BACKENDS that have a function signature
|
|
# matching the args/kwargs passed in the authenticate() call.
|
|
import binascii
|
|
import json
|
|
import logging
|
|
from abc import ABC, abstractmethod
|
|
from collections.abc import Callable
|
|
from dataclasses import dataclass
|
|
from email.headerregistry import Address
|
|
from typing import Any, TypedDict, TypeVar, cast
|
|
|
|
import magic
|
|
import orjson
|
|
from decorator import decorator
|
|
from django.conf import settings
|
|
from django.contrib.auth import authenticate, get_backends, logout
|
|
from django.contrib.auth.backends import RemoteUserBackend
|
|
from django.contrib.staticfiles.storage import staticfiles_storage
|
|
from django.core.exceptions import ValidationError
|
|
from django.core.validators import validate_email
|
|
from django.db import transaction
|
|
from django.dispatch import Signal, receiver
|
|
from django.http import HttpRequest, HttpResponse, HttpResponseRedirect
|
|
from django.shortcuts import render
|
|
from django.urls import reverse
|
|
from django.utils.translation import gettext as _
|
|
from django_auth_ldap.backend import LDAPBackend, _LDAPUser, _LDAPUserGroups, ldap_error
|
|
from django_auth_ldap.config import LDAPSettings
|
|
from lxml.etree import XMLSyntaxError
|
|
from onelogin.saml2 import compat as onelogin_saml2_compat
|
|
from onelogin.saml2.auth import OneLogin_Saml2_Auth
|
|
from onelogin.saml2.errors import OneLogin_Saml2_Error, OneLogin_Saml2_ValidationError
|
|
from onelogin.saml2.logout_request import OneLogin_Saml2_Logout_Request
|
|
from onelogin.saml2.logout_response import OneLogin_Saml2_Logout_Response
|
|
from onelogin.saml2.response import OneLogin_Saml2_Response
|
|
from onelogin.saml2.settings import OneLogin_Saml2_Settings
|
|
from onelogin.saml2.utils import OneLogin_Saml2_Utils
|
|
from onelogin.saml2.xml_utils import OneLogin_Saml2_XML
|
|
from requests import HTTPError
|
|
from social_core.backends.apple import AppleIdAuth
|
|
from social_core.backends.azuread import AzureADOAuth2
|
|
from social_core.backends.base import BaseAuth
|
|
from social_core.backends.github import GithubOAuth2, GithubOrganizationOAuth2, GithubTeamOAuth2
|
|
from social_core.backends.gitlab import GitLabOAuth2
|
|
from social_core.backends.google import GoogleOAuth2
|
|
from social_core.backends.open_id_connect import OpenIdConnectAuth
|
|
from social_core.backends.saml import SAMLAuth, SAMLIdentityProvider
|
|
from social_core.exceptions import (
|
|
AuthCanceled,
|
|
AuthFailed,
|
|
AuthMissingParameter,
|
|
AuthStateForbidden,
|
|
SocialAuthBaseException,
|
|
)
|
|
from social_core.pipeline.partial import partial
|
|
from social_django.utils import load_backend, load_strategy
|
|
from typing_extensions import override
|
|
from zxcvbn import zxcvbn
|
|
|
|
from zerver.actions.create_user import do_create_user, do_reactivate_user
|
|
from zerver.actions.custom_profile_fields import do_update_user_custom_profile_data_if_changed
|
|
from zerver.actions.user_groups import (
|
|
bulk_add_members_to_user_groups,
|
|
bulk_remove_members_from_user_groups,
|
|
)
|
|
from zerver.actions.user_settings import do_change_user_delivery_email, do_regenerate_api_key
|
|
from zerver.actions.users import do_change_user_role, do_deactivate_user
|
|
from zerver.lib.avatar import avatar_url, is_avatar_new
|
|
from zerver.lib.avatar_hash import user_avatar_content_hash
|
|
from zerver.lib.dev_ldap_directory import init_fakeldap
|
|
from zerver.lib.email_validation import email_allowed_for_realm, validate_email_not_already_in_realm
|
|
from zerver.lib.exceptions import JsonableError
|
|
from zerver.lib.mobile_auth_otp import is_valid_otp
|
|
from zerver.lib.rate_limiter import RateLimitedObject, client_is_exempt_from_rate_limiting
|
|
from zerver.lib.redis_utils import get_dict_from_redis, get_redis_client, put_dict_in_redis
|
|
from zerver.lib.request import RequestNotes
|
|
from zerver.lib.sessions import delete_user_sessions
|
|
from zerver.lib.subdomains import get_subdomain
|
|
from zerver.lib.types import ProfileDataElementUpdateDict
|
|
from zerver.lib.users import check_full_name, validate_user_custom_profile_field
|
|
from zerver.models import (
|
|
CustomProfileField,
|
|
NamedUserGroup,
|
|
PreregistrationRealm,
|
|
PreregistrationUser,
|
|
Realm,
|
|
UserGroupMembership,
|
|
UserProfile,
|
|
)
|
|
from zerver.models.custom_profile_fields import custom_profile_fields_for_realm
|
|
from zerver.models.groups import SystemGroups
|
|
from zerver.models.realms import (
|
|
DisposableEmailError,
|
|
DomainNotAllowedForRealmError,
|
|
EmailContainsPlusError,
|
|
get_realm,
|
|
supported_auth_backends,
|
|
)
|
|
from zerver.models.users import (
|
|
ExternalAuthID,
|
|
PasswordTooWeakError,
|
|
get_user_by_delivery_email,
|
|
get_user_profile_by_id,
|
|
remote_user_to_email,
|
|
)
|
|
from zproject.settings_types import OIDCIdPConfigDict
|
|
|
|
redis_client = get_redis_client()
|
|
|
|
|
|
def all_default_backend_names() -> list[str]:
|
|
if not settings.BILLING_ENABLED or settings.DEVELOPMENT:
|
|
# If billing isn't enabled, it's a self-hosted server
|
|
# and has access to all authentication backends.
|
|
#
|
|
# In DEVELOPMENT, we have BILLING_ENABLED=True, but
|
|
# nonetheless we want to enable all backends by default
|
|
# for convenience - we shouldn't add additional steps to the
|
|
# process of setting up a backend for testing.
|
|
return list(AUTH_BACKEND_NAME_MAP.keys())
|
|
|
|
# By default, only enable backends that are available without requiring a plan.
|
|
return [
|
|
name
|
|
for name, backend in AUTH_BACKEND_NAME_MAP.items()
|
|
if backend.available_for_cloud_plans is None
|
|
]
|
|
|
|
|
|
# This first batch of methods is used by other code in Zulip to check
|
|
# whether a given authentication backend is enabled for a given realm.
|
|
# In each case, we both needs to check at the server level (via
|
|
# `settings.AUTHENTICATION_BACKENDS`, queried via
|
|
# `django.contrib.auth.get_backends`) and at the realm level (via the
|
|
# `RealmAuthenticationMethod` table).
|
|
def pad_method_dict(method_dict: dict[str, bool]) -> dict[str, bool]:
|
|
"""Pads an authentication methods dict to contain all auth backends
|
|
supported by the software, regardless of whether they are
|
|
configured on this server"""
|
|
for key in AUTH_BACKEND_NAME_MAP:
|
|
if key not in method_dict:
|
|
method_dict[key] = False
|
|
return method_dict
|
|
|
|
|
|
def auth_enabled_helper(
|
|
backends_to_check: list[str],
|
|
realm: Realm | None,
|
|
realm_authentication_methods: dict[str, bool] | None = None,
|
|
) -> bool:
|
|
"""
|
|
realm_authentication_methods can be passed if already fetched to avoid
|
|
a database query.
|
|
"""
|
|
if realm is not None:
|
|
if realm_authentication_methods is not None:
|
|
# Copy the dict to avoid mutating the original if it was passed in as argument.
|
|
enabled_method_dict = realm_authentication_methods.copy()
|
|
else:
|
|
enabled_method_dict = realm.authentication_methods_dict()
|
|
else:
|
|
enabled_method_dict = dict.fromkeys(AUTH_BACKEND_NAME_MAP, True)
|
|
|
|
pad_method_dict(enabled_method_dict)
|
|
for supported_backend in supported_auth_backends():
|
|
for backend_name in backends_to_check:
|
|
backend = AUTH_BACKEND_NAME_MAP[backend_name]
|
|
if enabled_method_dict[backend_name] and isinstance(supported_backend, backend):
|
|
return True
|
|
return False
|
|
|
|
|
|
def ldap_auth_enabled(
|
|
realm: Realm | None = None, realm_authentication_methods: dict[str, bool] | None = None
|
|
) -> bool:
|
|
return auth_enabled_helper(["LDAP"], realm, realm_authentication_methods)
|
|
|
|
|
|
def email_auth_enabled(
|
|
realm: Realm | None = None, realm_authentication_methods: dict[str, bool] | None = None
|
|
) -> bool:
|
|
return auth_enabled_helper(["Email"], realm, realm_authentication_methods)
|
|
|
|
|
|
def password_auth_enabled(
|
|
realm: Realm | None = None, realm_authentication_methods: dict[str, bool] | None = None
|
|
) -> bool:
|
|
return ldap_auth_enabled(realm, realm_authentication_methods) or email_auth_enabled(
|
|
realm, realm_authentication_methods
|
|
)
|
|
|
|
|
|
def dev_auth_enabled(
|
|
realm: Realm | None = None, realm_authentication_methods: dict[str, bool] | None = None
|
|
) -> bool:
|
|
return auth_enabled_helper(["Dev"], realm, realm_authentication_methods)
|
|
|
|
|
|
def google_auth_enabled(
|
|
realm: Realm | None = None, realm_authentication_methods: dict[str, bool] | None = None
|
|
) -> bool:
|
|
return auth_enabled_helper(["Google"], realm, realm_authentication_methods)
|
|
|
|
|
|
def github_auth_enabled(
|
|
realm: Realm | None = None, realm_authentication_methods: dict[str, bool] | None = None
|
|
) -> bool:
|
|
return auth_enabled_helper(["GitHub"], realm, realm_authentication_methods)
|
|
|
|
|
|
def gitlab_auth_enabled(
|
|
realm: Realm | None = None, realm_authentication_methods: dict[str, bool] | None = None
|
|
) -> bool:
|
|
return auth_enabled_helper(["GitLab"], realm, realm_authentication_methods)
|
|
|
|
|
|
def apple_auth_enabled(
|
|
realm: Realm | None = None, realm_authentication_methods: dict[str, bool] | None = None
|
|
) -> bool:
|
|
return auth_enabled_helper(["Apple"], realm, realm_authentication_methods)
|
|
|
|
|
|
def saml_auth_enabled(
|
|
realm: Realm | None = None, realm_authentication_methods: dict[str, bool] | None = None
|
|
) -> bool:
|
|
return auth_enabled_helper(["SAML"], realm, realm_authentication_methods)
|
|
|
|
|
|
def require_email_format_usernames(realm: Realm | None = None) -> bool:
|
|
if ldap_auth_enabled(realm) and (settings.LDAP_EMAIL_ATTR or settings.LDAP_APPEND_DOMAIN):
|
|
return False
|
|
return True
|
|
|
|
|
|
def is_user_active(user_profile: UserProfile, return_data: dict[str, Any] | None = None) -> bool:
|
|
if user_profile.realm.deactivated:
|
|
if return_data is not None:
|
|
return_data["inactive_realm"] = True
|
|
return False
|
|
if not user_profile.is_active:
|
|
if return_data is not None:
|
|
if user_profile.is_mirror_dummy:
|
|
# Record whether it's a mirror dummy account
|
|
return_data["is_mirror_dummy"] = True
|
|
return_data["inactive_user"] = True
|
|
return_data["inactive_user_id"] = user_profile.id
|
|
return False
|
|
|
|
return True
|
|
|
|
|
|
def common_get_active_user(
|
|
email: str, realm: Realm, return_data: dict[str, Any] | None = None
|
|
) -> UserProfile | None:
|
|
"""This is the core common function used by essentially all
|
|
authentication backends to check if there's an active user account
|
|
with a given email address in the organization, handling both
|
|
user-level and realm-level deactivation correctly.
|
|
"""
|
|
try:
|
|
user_profile = get_user_by_delivery_email(email, realm)
|
|
except UserProfile.DoesNotExist:
|
|
# If the user doesn't have an account in the target realm, we
|
|
# check whether they might have an account in another realm,
|
|
# and if so, provide a helpful error message via
|
|
# `invalid_subdomain`.
|
|
if not UserProfile.objects.filter(delivery_email__iexact=email).exists():
|
|
return None
|
|
if return_data is not None:
|
|
return_data["invalid_subdomain"] = True
|
|
return_data["matching_user_ids_in_different_realms"] = list(
|
|
UserProfile.objects.filter(delivery_email__iexact=email).values("realm_id", "id")
|
|
)
|
|
return None
|
|
if not is_user_active(user_profile, return_data):
|
|
return None
|
|
|
|
return user_profile
|
|
|
|
|
|
def is_subdomain_in_allowed_subdomains_list(subdomain: str, allowed_subdomains: list[str]) -> bool:
|
|
if subdomain in allowed_subdomains:
|
|
return True
|
|
|
|
# The root subdomain is a special case, as sending an
|
|
# empty string in the list of values of the attribute may
|
|
# not be viable. So, any of the ROOT_SUBDOMAIN_ALIASES can
|
|
# be used to signify the user is authorized for the root
|
|
# subdomain.
|
|
if (
|
|
subdomain == Realm.SUBDOMAIN_FOR_ROOT_DOMAIN
|
|
and not settings.ROOT_DOMAIN_LANDING_PAGE
|
|
and any(alias in allowed_subdomains for alias in settings.ROOT_SUBDOMAIN_ALIASES)
|
|
):
|
|
return True
|
|
|
|
return False
|
|
|
|
|
|
AuthFuncT = TypeVar("AuthFuncT", bound=Callable[..., UserProfile | None])
|
|
|
|
|
|
class RateLimitedAuthenticationByUsername(RateLimitedObject):
|
|
def __init__(self, username: str) -> None:
|
|
self.username = username
|
|
super().__init__()
|
|
|
|
@override
|
|
def key(self) -> str:
|
|
return f"{type(self).__name__}:{self.username}"
|
|
|
|
@override
|
|
def rules(self) -> list[tuple[int, int]]:
|
|
return settings.RATE_LIMITING_RULES["authenticate_by_username"]
|
|
|
|
|
|
def rate_limit_authentication_by_username(request: HttpRequest, username: str) -> None:
|
|
RateLimitedAuthenticationByUsername(username).rate_limit_request(request)
|
|
|
|
|
|
def auth_rate_limiting_already_applied(request: HttpRequest) -> bool:
|
|
request_notes = RequestNotes.get_notes(request)
|
|
|
|
return any(
|
|
isinstance(r.entity, RateLimitedAuthenticationByUsername)
|
|
for r in request_notes.ratelimits_applied
|
|
)
|
|
|
|
|
|
# Django's authentication mechanism uses introspection on the various authenticate() functions
|
|
# defined by backends, so we need a decorator that doesn't break function signatures.
|
|
# @decorator does this for us.
|
|
# The usual @wraps from functools breaks signatures, so it can't be used here.
|
|
@decorator
|
|
def custom_auth_decorator(auth_func: AuthFuncT, *args: Any, **kwargs: Any) -> UserProfile | None:
|
|
custom_auth_wrapper_func = settings.CUSTOM_AUTHENTICATION_WRAPPER_FUNCTION
|
|
if custom_auth_wrapper_func is None:
|
|
return auth_func(*args, **kwargs)
|
|
else:
|
|
return custom_auth_wrapper_func(auth_func, *args, **kwargs)
|
|
|
|
|
|
@decorator
|
|
def rate_limit_auth(auth_func: AuthFuncT, *args: Any, **kwargs: Any) -> UserProfile | None:
|
|
if not settings.RATE_LIMITING_AUTHENTICATE:
|
|
return auth_func(*args, **kwargs)
|
|
|
|
request = args[1]
|
|
username = kwargs["username"]
|
|
if RequestNotes.get_notes(request).client is None or not client_is_exempt_from_rate_limiting(
|
|
request
|
|
):
|
|
# Django cycles through enabled authentication backends until one succeeds,
|
|
# or all of them fail. If multiple backends are tried like this, we only want
|
|
# to execute rate_limit_authentication_* once, on the first attempt:
|
|
if auth_rate_limiting_already_applied(request):
|
|
pass
|
|
else:
|
|
# Apply rate limiting. If this request is above the limit,
|
|
# RateLimitedError will be raised, interrupting the authentication process.
|
|
# From there, the code calling authenticate() can either catch the exception
|
|
# and handle it on its own, or it will be processed by RateLimitMiddleware.
|
|
rate_limit_authentication_by_username(request, username)
|
|
|
|
result = auth_func(*args, **kwargs)
|
|
if result is not None:
|
|
# Authentication succeeded, clear the rate-limiting record.
|
|
RateLimitedAuthenticationByUsername(username).clear_history()
|
|
|
|
return result
|
|
|
|
|
|
@decorator
|
|
def log_auth_attempts(auth_func: AuthFuncT, *args: Any, **kwargs: Any) -> UserProfile | None:
|
|
result = auth_func(*args, **kwargs)
|
|
|
|
backend_instance = args[0]
|
|
request = args[1]
|
|
username = kwargs["username"]
|
|
realm = kwargs["realm"]
|
|
return_data = kwargs["return_data"]
|
|
|
|
log_auth_attempt(
|
|
backend_instance.logger,
|
|
request,
|
|
realm,
|
|
username,
|
|
succeeded=result is not None,
|
|
return_data=return_data,
|
|
)
|
|
|
|
return result
|
|
|
|
|
|
def log_auth_attempt(
|
|
logger: logging.Logger,
|
|
request: HttpRequest,
|
|
realm: Realm,
|
|
username: str,
|
|
succeeded: bool,
|
|
return_data: dict[str, Any],
|
|
) -> None:
|
|
ip_addr = request.META.get("REMOTE_ADDR")
|
|
outcome = "success" if succeeded else "failed"
|
|
logger.info(
|
|
"Authentication attempt from %s: subdomain=%s;username=%s;outcome=%s;return_data=%s",
|
|
ip_addr,
|
|
realm.subdomain,
|
|
username,
|
|
outcome,
|
|
return_data,
|
|
)
|
|
|
|
|
|
class ZulipAuthMixin:
|
|
"""This common mixin is used to override Django's default behavior for
|
|
looking up a logged-in user by ID to use a version that fetches
|
|
from memcached before checking the database (avoiding a database
|
|
query in most cases).
|
|
"""
|
|
|
|
name = "undefined"
|
|
_logger: logging.Logger | None = None
|
|
|
|
# Describes which plans gives access to this authentication method on zulipchat.com.
|
|
# None means the backend is available regardless of the plan.
|
|
# Otherwise, it should be a list of Realm.plan_type values that give access to the backend.
|
|
available_for_cloud_plans: list[int] | None = None
|
|
|
|
@property
|
|
def logger(self) -> logging.Logger:
|
|
if self._logger is None:
|
|
self._logger = logging.getLogger(f"zulip.auth.{self.name}")
|
|
return self._logger
|
|
|
|
def get_user(self, user_profile_id: int) -> UserProfile | None:
|
|
"""Override the Django method for getting a UserProfile object from
|
|
the user_profile_id,."""
|
|
try:
|
|
return get_user_profile_by_id(user_profile_id)
|
|
except UserProfile.DoesNotExist:
|
|
return None
|
|
|
|
|
|
class ZulipDummyBackend(ZulipAuthMixin):
|
|
"""Used when we want to log you in without checking any
|
|
authentication (i.e. new user registration or when otherwise
|
|
authentication has already been checked earlier in the process).
|
|
|
|
We ensure that this backend only ever successfully authenticates
|
|
when explicitly requested by including the use_dummy_backend kwarg.
|
|
"""
|
|
|
|
name = "dummy"
|
|
|
|
@custom_auth_decorator
|
|
def authenticate(
|
|
self,
|
|
request: HttpRequest | None = None,
|
|
*,
|
|
username: str,
|
|
realm: Realm,
|
|
use_dummy_backend: bool = False,
|
|
return_data: dict[str, Any] | None = None,
|
|
) -> UserProfile | None:
|
|
if use_dummy_backend:
|
|
return common_get_active_user(username, realm, return_data)
|
|
return None
|
|
|
|
|
|
def check_password_strength(password: str) -> bool:
|
|
"""
|
|
Returns True if the password is strong enough,
|
|
False otherwise.
|
|
"""
|
|
if len(password) < settings.PASSWORD_MIN_LENGTH:
|
|
return False
|
|
|
|
if password == "":
|
|
# zxcvbn throws an exception when passed the empty string, so
|
|
# we need a special case for the empty string password here.
|
|
return False
|
|
|
|
if (
|
|
int(zxcvbn(password, max_length=settings.PASSWORD_MAX_LENGTH)["guesses"])
|
|
< settings.PASSWORD_MIN_GUESSES
|
|
):
|
|
return False
|
|
|
|
return True
|
|
|
|
|
|
class EmailAuthBackend(ZulipAuthMixin):
|
|
"""
|
|
Email+Password authentication backend (the default).
|
|
|
|
Allows a user to sign in using an email/password pair.
|
|
"""
|
|
|
|
name = "email"
|
|
|
|
@rate_limit_auth
|
|
@log_auth_attempts
|
|
@custom_auth_decorator
|
|
def authenticate(
|
|
self,
|
|
request: HttpRequest,
|
|
*,
|
|
username: str,
|
|
password: str,
|
|
realm: Realm,
|
|
return_data: dict[str, Any] | None = None,
|
|
) -> UserProfile | None:
|
|
"""Authenticate a user based on email address as the user name."""
|
|
if not password_auth_enabled(realm):
|
|
if return_data is not None:
|
|
return_data["password_auth_disabled"] = True
|
|
return None
|
|
if not email_auth_enabled(realm):
|
|
if return_data is not None:
|
|
return_data["email_auth_disabled"] = True
|
|
return None
|
|
if password == "":
|
|
# Never allow an empty password. This is defensive code;
|
|
# a user having password "" should only be possible
|
|
# through a bug somewhere else.
|
|
return None
|
|
|
|
user_profile = common_get_active_user(username, realm, return_data=return_data)
|
|
if user_profile is None:
|
|
return None
|
|
|
|
try:
|
|
is_password_correct = user_profile.check_password(password)
|
|
except PasswordTooWeakError:
|
|
# In some rare cases when password hasher is changed and the user has
|
|
# a weak password, PasswordTooWeakError will be raised.
|
|
self.logger.info(
|
|
"User %s password can't be rehashed due to being too weak.", user_profile.id
|
|
)
|
|
if return_data is not None:
|
|
return_data["password_reset_needed"] = True
|
|
return None
|
|
else:
|
|
# Since we can't communicate the situation via return_data,
|
|
# we have to raise an error - a silent failure would not be right
|
|
# because the password actually is correct, just can't be re-hashed.
|
|
raise JsonableError(_("You need to reset your password."))
|
|
|
|
if is_password_correct:
|
|
return user_profile
|
|
return None
|
|
|
|
|
|
def is_valid_email(email: str) -> bool:
|
|
try:
|
|
validate_email(email)
|
|
except ValidationError:
|
|
return False
|
|
return True
|
|
|
|
|
|
def check_ldap_config() -> None:
|
|
if not settings.LDAP_APPEND_DOMAIN:
|
|
# Email search needs to be configured in this case.
|
|
assert settings.AUTH_LDAP_USERNAME_ATTR and settings.AUTH_LDAP_REVERSE_EMAIL_SEARCH
|
|
|
|
# These two are alternatives approaches to deactivating users based on an ldap attribute
|
|
# and thus don't make sense to have enabled together.
|
|
assert not (
|
|
settings.AUTH_LDAP_USER_ATTR_MAP.get("userAccountControl")
|
|
and settings.AUTH_LDAP_USER_ATTR_MAP.get("deactivated")
|
|
)
|
|
|
|
|
|
def ldap_should_sync_active_status() -> bool:
|
|
if "userAccountControl" in settings.AUTH_LDAP_USER_ATTR_MAP:
|
|
return True
|
|
|
|
if "deactivated" in settings.AUTH_LDAP_USER_ATTR_MAP:
|
|
return True
|
|
|
|
return False
|
|
|
|
|
|
def find_ldap_users_by_email(email: str) -> list[_LDAPUser]:
|
|
"""
|
|
Returns list of _LDAPUsers matching the email search
|
|
"""
|
|
return LDAPReverseEmailSearch().search_for_users(email)
|
|
|
|
|
|
def email_belongs_to_ldap(realm: Realm, email: str) -> bool:
|
|
"""Used to make determinations on whether a user's email address is
|
|
managed by LDAP. For environments using both LDAP and
|
|
Email+Password authentication, we do not allow EmailAuthBackend
|
|
authentication for email addresses managed by LDAP (to avoid a
|
|
security issue where one create separate credentials for an LDAP
|
|
user), and this function is used to enforce that rule.
|
|
"""
|
|
if not ldap_auth_enabled(realm):
|
|
return False
|
|
|
|
check_ldap_config()
|
|
if settings.LDAP_APPEND_DOMAIN:
|
|
# Check if the email ends with LDAP_APPEND_DOMAIN
|
|
return Address(addr_spec=email).domain.lower() == settings.LDAP_APPEND_DOMAIN
|
|
|
|
# If we don't have an LDAP domain, we have to do a lookup for the email.
|
|
if find_ldap_users_by_email(email):
|
|
return True
|
|
else:
|
|
return False
|
|
|
|
|
|
ldap_logger = logging.getLogger("zulip.ldap")
|
|
|
|
|
|
class LDAPReverseEmailSearch(_LDAPUser):
|
|
"""
|
|
This class is a workaround - we want to use
|
|
django-auth-ldap to query the ldap directory for
|
|
users with the specified email address, but it doesn't
|
|
provide an API for that or an isolated class for handling
|
|
the connection. Because connection-handling is tightly integrated
|
|
into the _LDAPUser class, we have to make this strange inheritance here,
|
|
in order to be able to comfortably have an ldap connection and make search
|
|
queries.
|
|
|
|
We may be able to get rid of this in the future if we can get
|
|
https://github.com/django-auth-ldap/django-auth-ldap/pull/150 merged upstream.
|
|
"""
|
|
|
|
def __init__(self) -> None:
|
|
# Superclass __init__ requires a username argument - it doesn't actually
|
|
# impact anything for us in this class, given its very limited use
|
|
# for only making a search query, so we pass an empty string.
|
|
super().__init__(LDAPBackend(), username="")
|
|
|
|
def search_for_users(self, email: str) -> list[_LDAPUser]:
|
|
search = settings.AUTH_LDAP_REVERSE_EMAIL_SEARCH
|
|
USERNAME_ATTR = settings.AUTH_LDAP_USERNAME_ATTR
|
|
|
|
assert search is not None
|
|
results = search.execute(self.connection, {"email": email})
|
|
|
|
ldap_users = []
|
|
for result in results:
|
|
user_dn, user_attrs = result
|
|
username = user_attrs[USERNAME_ATTR][0]
|
|
ldap_user = _LDAPUser(self.backend, username=username)
|
|
ldap_user._user_dn = user_dn
|
|
ldap_user._user_attrs = user_attrs
|
|
|
|
ldap_users.append(ldap_user)
|
|
|
|
return ldap_users
|
|
|
|
|
|
class ZulipLDAPError(_LDAPUser.AuthenticationFailed):
|
|
"""Since this inherits from _LDAPUser.AuthenticationFailed, these will
|
|
be caught and logged at debug level inside django-auth-ldap's authenticate()"""
|
|
|
|
|
|
class NoMatchingLDAPUserError(ZulipLDAPError):
|
|
pass
|
|
|
|
|
|
class OutsideLDAPDomainError(NoMatchingLDAPUserError):
|
|
pass
|
|
|
|
|
|
class ZulipLDAPConfigurationError(Exception):
|
|
pass
|
|
|
|
|
|
LDAP_USER_ACCOUNT_CONTROL_DISABLED_MASK = 2
|
|
|
|
|
|
def ldap_external_auth_id_sync_enabled() -> bool:
|
|
external_auth_id_attr = settings.AUTH_LDAP_USER_ATTR_MAP.get("unique_account_id")
|
|
if external_auth_id_attr is None:
|
|
return False
|
|
|
|
if not isinstance(external_auth_id_attr, str):
|
|
raise AssertionError(
|
|
"unique_account_id in AUTH_LDAP_USER_ATTR_MAP set to invalid value! must be a string."
|
|
)
|
|
|
|
return True
|
|
|
|
|
|
class ZulipLDAPSettings(LDAPSettings):
|
|
NON_SYNCABLE_ATTRS = [
|
|
"unique_account_id",
|
|
"avatar",
|
|
"userAccountControl",
|
|
"deactivated",
|
|
"org_membership",
|
|
]
|
|
|
|
def __init__(
|
|
self, prefix: str = "AUTH_LDAP_", defaults: dict[object, object] | None = None
|
|
) -> None:
|
|
"""
|
|
django-auth-ldap populate_user() codepath iterates through USER_ATTR_MAP dict, and calls
|
|
setattr(user, key, value) for each key, value pair. The challenge here is that Zulip adds some custom
|
|
mappings in AUTH_LDAP_USER_ATTR_MAP - for which the keys are special keywords (e.g. "unique_account_id")
|
|
with a meaning understood by Zulip's code for handling them - and aren't actual UserProfile attributes
|
|
that can be set in a valid way.
|
|
|
|
To avoid ending up with strange UserProfile objects, which have attributes set on them which shouldn't
|
|
exist at all, we have to patch this settings class - to drop such special keywords from USER_ATTR_MAP
|
|
for the purposes of django-auth-ldap's internal code.
|
|
"""
|
|
if defaults is None: # nocoverage
|
|
defaults = dict()
|
|
|
|
super().__init__(prefix, defaults)
|
|
self.USER_ATTR_MAP: dict[object, object] = {
|
|
key: value
|
|
for key, value in self.USER_ATTR_MAP.items()
|
|
if key not in self.NON_SYNCABLE_ATTRS
|
|
}
|
|
|
|
|
|
class ZulipLDAPAuthBackendBase(ZulipAuthMixin, LDAPBackend):
|
|
"""Common code between LDAP authentication (ZulipLDAPAuthBackend) and
|
|
using LDAP just to sync user data (ZulipLDAPUserPopulator).
|
|
|
|
To fully understand our LDAP backend, you may want to skim
|
|
django_auth_ldap/backend.py from the upstream django-auth-ldap
|
|
library. It's not a lot of code, and searching around in that
|
|
file makes the flow for LDAP authentication clear.
|
|
"""
|
|
|
|
name = "ldap"
|
|
|
|
def __init__(self) -> None:
|
|
# Used to initialize a fake LDAP directly for both manual
|
|
# and automated testing in a development environment where
|
|
# there is no actual LDAP server.
|
|
if settings.DEVELOPMENT and settings.FAKE_LDAP_MODE: # nocoverage
|
|
init_fakeldap()
|
|
|
|
check_ldap_config()
|
|
|
|
@property
|
|
def settings(self) -> ZulipLDAPSettings:
|
|
# See ZulipLDAPSettings for explanation of why we need to patch this
|
|
# to substitute our overridden LDAPSettings class.
|
|
self._settings: ZulipLDAPSettings | None
|
|
if self._settings is None:
|
|
self._settings = ZulipLDAPSettings(self.settings_prefix, self.default_settings)
|
|
|
|
return self._settings
|
|
|
|
# Disable django-auth-ldap's permissions functions -- we don't use
|
|
# the standard Django user/group permissions system because they
|
|
# are prone to performance issues.
|
|
def has_perm(self, user: UserProfile | None, perm: Any, obj: Any = None) -> bool:
|
|
return False
|
|
|
|
def has_module_perms(self, user: UserProfile | None, app_label: str | None) -> bool:
|
|
return False
|
|
|
|
def get_all_permissions(self, user: UserProfile | None, obj: Any = None) -> set[Any]:
|
|
return set()
|
|
|
|
def get_group_permissions(self, user: UserProfile | None, obj: Any = None) -> set[Any]:
|
|
return set()
|
|
|
|
def django_to_ldap_username(self, username: str) -> str:
|
|
"""
|
|
Translates django username (user_profile.delivery_email or whatever the user typed in the login
|
|
field when authenticating via the LDAP backend) into LDAP username.
|
|
Guarantees that the username it returns actually has an entry in the LDAP directory.
|
|
Raises NoMatchingLDAPUserError if that's not possible.
|
|
"""
|
|
result = username
|
|
if settings.LDAP_APPEND_DOMAIN:
|
|
if is_valid_email(username):
|
|
address = Address(addr_spec=username)
|
|
if address.domain != settings.LDAP_APPEND_DOMAIN:
|
|
raise OutsideLDAPDomainError(
|
|
f"Email {username} does not match LDAP domain {settings.LDAP_APPEND_DOMAIN}."
|
|
)
|
|
result = address.username
|
|
else:
|
|
# We can use find_ldap_users_by_email
|
|
if is_valid_email(username):
|
|
email_search_result = find_ldap_users_by_email(username)
|
|
if not email_search_result:
|
|
result = username
|
|
elif len(email_search_result) == 1:
|
|
return email_search_result[0]._username
|
|
elif len(email_search_result) > 1:
|
|
# This is possible, but strange, so worth logging a warning about.
|
|
# We can't translate the email to a unique username,
|
|
# so we don't do anything else here.
|
|
logging.warning("Multiple users with email %s found in LDAP.", username)
|
|
result = username
|
|
|
|
if _LDAPUser(self, result).attrs is None:
|
|
# Check that there actually is an LDAP entry matching the result username
|
|
# we want to return. Otherwise, raise an exception.
|
|
error_message = (
|
|
"No LDAP user matching django_to_ldap_username result: {}. Input username: {}"
|
|
)
|
|
raise NoMatchingLDAPUserError(
|
|
error_message.format(result, username),
|
|
)
|
|
|
|
return result
|
|
|
|
def user_email_from_ldapuser(self, username: str, ldap_user: _LDAPUser) -> str:
|
|
if hasattr(ldap_user, "_username"):
|
|
# In tests, we sometimes pass a simplified _LDAPUser without _username attr,
|
|
# and with the intended username in the username argument.
|
|
username = ldap_user._username
|
|
|
|
if settings.LDAP_APPEND_DOMAIN:
|
|
return Address(username=username, domain=settings.LDAP_APPEND_DOMAIN).addr_spec
|
|
|
|
if settings.LDAP_EMAIL_ATTR is not None:
|
|
# Get email from LDAP attributes.
|
|
if settings.LDAP_EMAIL_ATTR not in ldap_user.attrs:
|
|
raise ZulipLDAPError(
|
|
f"LDAP user doesn't have the needed {settings.LDAP_EMAIL_ATTR} attribute"
|
|
)
|
|
else:
|
|
return ldap_user.attrs[settings.LDAP_EMAIL_ATTR][0]
|
|
|
|
return username
|
|
|
|
def ldap_to_django_username(self, username: str) -> str:
|
|
"""
|
|
This is called inside django_auth_ldap with only one role:
|
|
to convert _LDAPUser._username to django username (so in Zulip, the email)
|
|
and pass that as "username" argument to get_or_build_user(username, ldapuser).
|
|
In many cases, the email is stored in the _LDAPUser's attributes, so it can't be
|
|
constructed just from the username. We choose to do nothing in this function,
|
|
and our overrides of get_or_build_user() obtain that username from the _LDAPUser
|
|
object on their own, through our user_email_from_ldapuser function.
|
|
"""
|
|
return username
|
|
|
|
def sync_avatar_from_ldap(self, user: UserProfile, ldap_user: _LDAPUser) -> None:
|
|
if "avatar" not in settings.AUTH_LDAP_USER_ATTR_MAP:
|
|
return
|
|
|
|
# We do local imports here to avoid import loops
|
|
from io import BytesIO
|
|
|
|
from zerver.actions.user_settings import do_change_avatar_fields
|
|
from zerver.lib.upload import upload_avatar_image
|
|
|
|
avatar_attr_name = settings.AUTH_LDAP_USER_ATTR_MAP["avatar"]
|
|
if avatar_attr_name not in ldap_user.attrs: # nocoverage
|
|
# If this specific user doesn't have e.g. a
|
|
# thumbnailPhoto set in LDAP, just skip that user.
|
|
return
|
|
|
|
ldap_avatar = ldap_user.attrs[avatar_attr_name][0]
|
|
|
|
avatar_changed = is_avatar_new(ldap_avatar, user)
|
|
if not avatar_changed:
|
|
# Don't do work to replace the avatar with itself.
|
|
return
|
|
|
|
# Structurally, to make the S3 backend happy, we need to
|
|
# provide a Content-Type; since that isn't specified in
|
|
# any metadata, we auto-detect it.
|
|
content_type = magic.from_buffer(ldap_avatar[:1024], mime=True)
|
|
if content_type.startswith("image/"):
|
|
upload_avatar_image(BytesIO(ldap_avatar), user, content_type=content_type)
|
|
do_change_avatar_fields(user, UserProfile.AVATAR_FROM_USER, acting_user=None)
|
|
# Update avatar hash.
|
|
user.avatar_hash = user_avatar_content_hash(ldap_avatar)
|
|
user.save(update_fields=["avatar_hash"])
|
|
else:
|
|
logging.warning("Could not parse %s field for user %s", avatar_attr_name, user.id)
|
|
|
|
def is_user_disabled_in_ldap(self, ldap_user: _LDAPUser) -> bool:
|
|
"""Implements checks for whether a user has been
|
|
disabled in the LDAP server being integrated with
|
|
Zulip."""
|
|
if "userAccountControl" in settings.AUTH_LDAP_USER_ATTR_MAP:
|
|
account_control_value = ldap_user.attrs[
|
|
settings.AUTH_LDAP_USER_ATTR_MAP["userAccountControl"]
|
|
][0]
|
|
return bool(int(account_control_value) & LDAP_USER_ACCOUNT_CONTROL_DISABLED_MASK)
|
|
|
|
assert "deactivated" in settings.AUTH_LDAP_USER_ATTR_MAP
|
|
attr_value = ldap_user.attrs[settings.AUTH_LDAP_USER_ATTR_MAP["deactivated"]][0]
|
|
|
|
# In the LDAP specification, a Boolean attribute should be
|
|
# *exactly* either "TRUE" or "FALSE". However,
|
|
# https://www.freeipa.org/page/V4/User_Life-Cycle_Management suggests
|
|
# that FreeIPA at least documents using Yes/No for booleans.
|
|
true_values = ["TRUE", "YES"]
|
|
false_values = ["FALSE", "NO"]
|
|
attr_value_upper = attr_value.upper()
|
|
assert attr_value_upper in true_values or attr_value_upper in false_values, (
|
|
f"Invalid value '{attr_value}' in the LDAP attribute mapped to deactivated"
|
|
)
|
|
return attr_value_upper in true_values
|
|
|
|
def is_account_realm_access_forbidden(self, ldap_user: _LDAPUser, realm: Realm) -> bool:
|
|
realm_access_control = settings.AUTH_LDAP_ADVANCED_REALM_ACCESS_CONTROL
|
|
# org_membership takes priority over AUTH_LDAP_ADVANCED_REALM_ACCESS_CONTROL.
|
|
if "org_membership" in settings.AUTH_LDAP_USER_ATTR_MAP:
|
|
org_membership_attr = settings.AUTH_LDAP_USER_ATTR_MAP["org_membership"]
|
|
allowed_orgs: list[str] = ldap_user.attrs.get(org_membership_attr, [])
|
|
if is_subdomain_in_allowed_subdomains_list(realm.subdomain, allowed_orgs):
|
|
return False
|
|
# If Advanced is not configured, forbid access
|
|
if realm_access_control is None:
|
|
return True
|
|
|
|
# If neither setting is configured, allow access.
|
|
if realm_access_control is None:
|
|
return False
|
|
if realm.subdomain not in realm_access_control:
|
|
# If a realm is not configured in this setting, it shouldn't
|
|
# be affected by it - therefore, allow access.
|
|
return False
|
|
|
|
# With settings.AUTH_LDAP_ADVANCED_REALM_ACCESS_CONTROL, we
|
|
# allow access if and only if one of the entries for the
|
|
# target subdomain matches the user's LDAP attributes.
|
|
|
|
# Make sure the format of the setting makes sense.
|
|
assert isinstance(realm_access_control[realm.subdomain], list)
|
|
assert len(realm_access_control[realm.subdomain]) > 0
|
|
|
|
# Go through every "or" check
|
|
for attribute_group in realm_access_control[realm.subdomain]:
|
|
access = True
|
|
for attribute in attribute_group:
|
|
if not (
|
|
attribute in ldap_user.attrs
|
|
and attribute_group[attribute] in ldap_user.attrs[attribute]
|
|
):
|
|
access = False
|
|
if access:
|
|
return False
|
|
|
|
return True
|
|
|
|
@classmethod
|
|
def get_mapped_name(cls, ldap_user: _LDAPUser) -> str:
|
|
"""Constructs the user's Zulip full_name from the LDAP data"""
|
|
if "full_name" in settings.AUTH_LDAP_USER_ATTR_MAP:
|
|
full_name_attr = settings.AUTH_LDAP_USER_ATTR_MAP["full_name"]
|
|
full_name = ldap_user.attrs[full_name_attr][0]
|
|
elif all(key in settings.AUTH_LDAP_USER_ATTR_MAP for key in ["first_name", "last_name"]):
|
|
first_name_attr = settings.AUTH_LDAP_USER_ATTR_MAP["first_name"]
|
|
last_name_attr = settings.AUTH_LDAP_USER_ATTR_MAP["last_name"]
|
|
first_name = ldap_user.attrs[first_name_attr][0]
|
|
last_name = ldap_user.attrs[last_name_attr][0]
|
|
full_name = f"{first_name} {last_name}"
|
|
else:
|
|
raise ZulipLDAPError("Missing required mapping for user's full name")
|
|
|
|
return full_name
|
|
|
|
def sync_full_name_from_ldap(self, user_profile: UserProfile, ldap_user: _LDAPUser) -> None:
|
|
from zerver.actions.user_settings import do_change_full_name
|
|
|
|
full_name = self.get_mapped_name(ldap_user)
|
|
if full_name != user_profile.full_name:
|
|
try:
|
|
full_name = check_full_name(
|
|
full_name_raw=full_name, user_profile=user_profile, realm=user_profile.realm
|
|
)
|
|
except JsonableError as e:
|
|
raise ZulipLDAPError(e.msg)
|
|
do_change_full_name(user_profile, full_name, None)
|
|
|
|
def sync_custom_profile_fields_from_ldap(
|
|
self, user_profile: UserProfile, ldap_user: _LDAPUser
|
|
) -> None:
|
|
values_by_var_name: dict[str, int | str | list[int]] = {}
|
|
for attr, ldap_attr in settings.AUTH_LDAP_USER_ATTR_MAP.items():
|
|
if not attr.startswith("custom_profile_field__"):
|
|
continue
|
|
var_name = attr.split("custom_profile_field__")[1]
|
|
try:
|
|
value = ldap_user.attrs[ldap_attr][0]
|
|
except KeyError:
|
|
# If this user doesn't have this field set then ignore this
|
|
# field and continue syncing other fields. `django-auth-ldap`
|
|
# automatically logs error about missing field.
|
|
continue
|
|
values_by_var_name[var_name] = value
|
|
|
|
try:
|
|
sync_user_profile_custom_fields(user_profile, values_by_var_name)
|
|
except SyncUserError as e:
|
|
raise ZulipLDAPError(str(e)) from e
|
|
|
|
def sync_groups_from_ldap(self, user_profile: UserProfile, ldap_user: _LDAPUser) -> None:
|
|
if user_profile.realm.string_id not in settings.LDAP_SYNCHRONIZED_GROUPS_BY_REALM:
|
|
return
|
|
|
|
configured_group_names = set(
|
|
settings.LDAP_SYNCHRONIZED_GROUPS_BY_REALM[user_profile.realm.string_id]
|
|
)
|
|
|
|
try:
|
|
intended_group_names = set(ldap_user.group_names) & configured_group_names
|
|
sync_groups(
|
|
all_group_names=configured_group_names,
|
|
intended_group_names=intended_group_names,
|
|
user_profile=user_profile,
|
|
logger=ldap_logger,
|
|
)
|
|
except Exception as e:
|
|
raise ZulipLDAPError(str(e)) from e
|
|
|
|
|
|
class ZulipLDAPAuthBackend(ZulipLDAPAuthBackendBase):
|
|
REALM_IS_NONE_ERROR = 1
|
|
|
|
@rate_limit_auth
|
|
@log_auth_attempts
|
|
@custom_auth_decorator
|
|
def authenticate(
|
|
self,
|
|
request: HttpRequest | None = None,
|
|
*,
|
|
username: str,
|
|
password: str,
|
|
realm: Realm,
|
|
prereg_realm: PreregistrationRealm | None = None,
|
|
prereg_user: PreregistrationUser | None = None,
|
|
return_data: dict[str, Any] | None = None,
|
|
) -> UserProfile | None:
|
|
self._realm = realm
|
|
self._prereg_user = prereg_user
|
|
self._prereg_realm = prereg_realm
|
|
if not ldap_auth_enabled(realm):
|
|
return None
|
|
|
|
try:
|
|
# We want to pass the user's LDAP username into
|
|
# authenticate() below. If an email address was entered
|
|
# in the login form, we need to use
|
|
# django_to_ldap_username to translate the email address
|
|
# to the user's LDAP username before calling the
|
|
# django-auth-ldap authenticate().
|
|
username = self.django_to_ldap_username(username)
|
|
except NoMatchingLDAPUserError as e:
|
|
ldap_logger.debug("%s: %s", type(self).__name__, e)
|
|
if return_data is not None:
|
|
return_data["no_matching_ldap_user"] = True
|
|
return None
|
|
|
|
# Now we would want to run the django-auth-ldap authenticate
|
|
# function. We're, however, forced to fork it here in order
|
|
# to use our own ZulipLDAPUser class instead of its _LDAPUser.
|
|
#
|
|
# This will check the username/password pair
|
|
# against the LDAP database, and assuming those are correct,
|
|
# end up calling `self.get_or_build_user` with the
|
|
# authenticated user's data from LDAP.
|
|
|
|
if not password:
|
|
self.logger.debug("Rejecting empty password for %s", username)
|
|
return None
|
|
|
|
ldap_user = ZulipLDAPUser(self, username=username.strip(), request=request, realm=realm)
|
|
user = self.authenticate_ldap_user(ldap_user, password)
|
|
|
|
return user
|
|
|
|
def get_and_sync_user_profile_by_external_auth_id(
|
|
self, external_auth_id: str, email: str, return_data: dict[str, Any]
|
|
) -> UserProfile | None:
|
|
# If we're using External Auth ID based auth, then the lookup by
|
|
# external_auth_id takes precedence over lookup by email.
|
|
try:
|
|
user_profile: UserProfile | None = ExternalAuthID.objects.get(
|
|
external_auth_method_name=self.name,
|
|
external_auth_id=external_auth_id,
|
|
realm=self._realm,
|
|
).user
|
|
except ExternalAuthID.DoesNotExist:
|
|
user_profile = common_get_active_user(email, self._realm, return_data)
|
|
if user_profile is not None:
|
|
# If we found an account with the matching email, but no ExternalAuthID, then this account hasn't
|
|
# yet had its external_auth_id stored - we need to do this now.
|
|
ExternalAuthID.objects.create(
|
|
external_auth_method_name=self.name,
|
|
external_auth_id=external_auth_id,
|
|
realm=self._realm,
|
|
user=user_profile,
|
|
)
|
|
return user_profile
|
|
|
|
assert user_profile is not None
|
|
# We found a user with the matching external_auth_id. Now we need to do a seemingly redundant
|
|
# re-fetching via common_get_active_user - as that's the core function for securely fetching
|
|
# a user for login, in authentication contexts. It's responsible for the relevant security
|
|
# checks (such as the account being active) - we don't attempt to duplicate these checks
|
|
# here independently.
|
|
user_profile = common_get_active_user(user_profile.delivery_email, self._realm, return_data)
|
|
if user_profile is None:
|
|
return None
|
|
|
|
# We need to ensure the email address of the Zulip account remains synced with what's in the ldap
|
|
# directory. That's the point of external_auth_id-based authentication.
|
|
if user_profile.delivery_email != email:
|
|
# We intentionally do a case-sensitive comparison, despite emails in Zulip being
|
|
# case-insensitive in auth contexts. We want to support the case of sync tweaking just
|
|
# capitalization of the email address.
|
|
self.logger.info(
|
|
"User %s, logged in via ExternalAuthId %s, has mismatched email. Syncing: %s => %s",
|
|
user_profile.id,
|
|
external_auth_id,
|
|
user_profile.delivery_email,
|
|
email,
|
|
)
|
|
if (
|
|
UserProfile.objects.filter(realm=self._realm, delivery_email__iexact=email)
|
|
# The EXISTS query has a somewhat strange shape because we need
|
|
# to again consider the edge case where we might be simply
|
|
# updating capitalization of the email for the user. In such a
|
|
# situation, a "conflict" on (realm, delivery_email__iexact) is
|
|
# not an actual conflict.
|
|
.exclude(id=user_profile.id)
|
|
.exists()
|
|
):
|
|
self.logger.warning(
|
|
"Can't sync email for user %s: another user exists with target email %s",
|
|
user_profile.id,
|
|
email,
|
|
)
|
|
else:
|
|
do_change_user_delivery_email(user_profile, email, acting_user=None)
|
|
|
|
return user_profile
|
|
|
|
def get_or_build_user(
|
|
self, username: str, ldap_user: "ZulipLDAPUser"
|
|
) -> tuple[UserProfile, bool]:
|
|
"""The main function of our authentication backend extension of
|
|
django-auth-ldap. When this is called (from `authenticate`),
|
|
django-auth-ldap will already have verified that the provided
|
|
username and password match those in the LDAP database.
|
|
|
|
This function's responsibility is to check (1) whether the
|
|
email address for this user obtained from LDAP has an active
|
|
account in this Zulip realm. If so, it will log them in.
|
|
|
|
Otherwise, to provide a seamless single sign-on experience
|
|
with LDAP, this function can automatically create a new Zulip
|
|
user account in the realm (assuming the realm is configured to
|
|
allow that email address to sign up).
|
|
"""
|
|
return_data: dict[str, Any] = {}
|
|
|
|
email = self.user_email_from_ldapuser(username, ldap_user)
|
|
|
|
external_auth_id: str | None = None
|
|
if ldap_external_auth_id_sync_enabled():
|
|
external_auth_id = ldap_user.get_external_auth_id()
|
|
|
|
if self.is_account_realm_access_forbidden(ldap_user, self._realm):
|
|
raise ZulipLDAPError("User not allowed to access realm")
|
|
|
|
if ldap_should_sync_active_status():
|
|
ldap_disabled = self.is_user_disabled_in_ldap(ldap_user)
|
|
if ldap_disabled:
|
|
# Treat disabled users as deactivated in Zulip.
|
|
return_data["inactive_user"] = True
|
|
raise ZulipLDAPError("User has been deactivated")
|
|
|
|
if ldap_external_auth_id_sync_enabled() and external_auth_id:
|
|
user_profile = self.get_and_sync_user_profile_by_external_auth_id(
|
|
external_auth_id, email, return_data
|
|
)
|
|
else:
|
|
user_profile = common_get_active_user(email, self._realm, return_data)
|
|
|
|
if user_profile is not None:
|
|
# An existing user, successfully authed; return it.
|
|
return user_profile, False
|
|
|
|
if return_data.get("inactive_realm"):
|
|
# This happens if there is a user account in a deactivated realm
|
|
raise ZulipLDAPError("Realm has been deactivated")
|
|
if return_data.get("inactive_user"):
|
|
raise ZulipLDAPError("User has been deactivated")
|
|
# An invalid_subdomain `return_data` value here is ignored,
|
|
# since that just means we're trying to create an account in a
|
|
# second realm on the server (`ldap_auth_enabled(realm)` would
|
|
# have been false if this user wasn't meant to have an account
|
|
# in this second realm).
|
|
if self._realm.deactivated:
|
|
# This happens if no account exists, but the realm is
|
|
# deactivated, so we shouldn't create a new user account
|
|
raise ZulipLDAPError("Realm has been deactivated")
|
|
|
|
try:
|
|
validate_email(email)
|
|
except ValidationError:
|
|
error_message = f"{email} is not a valid email address."
|
|
# This indicates a misconfiguration of ldap settings
|
|
# or a malformed email value in the ldap directory,
|
|
# so we should log a warning about this before failing.
|
|
self.logger.warning(error_message)
|
|
raise ZulipLDAPError(error_message)
|
|
|
|
# Makes sure that email domain hasn't be restricted for this
|
|
# realm. The main thing here is email_allowed_for_realm; but
|
|
# we also call validate_email_not_already_in_realm just for consistency,
|
|
# even though its checks were already done above.
|
|
try:
|
|
email_allowed_for_realm(email, self._realm)
|
|
validate_email_not_already_in_realm(
|
|
self._realm, email, allow_inactive_mirror_dummies=True
|
|
)
|
|
except DomainNotAllowedForRealmError:
|
|
raise ZulipLDAPError("This email domain isn't allowed in this organization.")
|
|
except (DisposableEmailError, EmailContainsPlusError):
|
|
raise ZulipLDAPError("Email validation failed.")
|
|
|
|
# We have valid LDAP credentials; time to create an account.
|
|
full_name = self.get_mapped_name(ldap_user)
|
|
try:
|
|
full_name = check_full_name(
|
|
full_name_raw=full_name, user_profile=None, realm=self._realm
|
|
)
|
|
except JsonableError as e:
|
|
raise ZulipLDAPError(e.msg)
|
|
|
|
opts: dict[str, Any] = {}
|
|
if self._prereg_user:
|
|
invited_as = self._prereg_user.invited_as
|
|
opts["prereg_user"] = self._prereg_user
|
|
|
|
opts["role"] = invited_as
|
|
opts["realm_creation"] = False
|
|
# TODO: Ideally, we should add a mechanism for the user
|
|
# entering which default stream groups they've selected in
|
|
# the LDAP flow.
|
|
opts["default_stream_groups"] = []
|
|
|
|
if self._prereg_realm:
|
|
opts["prereg_realm"] = self._prereg_realm
|
|
opts["realm_creation"] = True
|
|
opts["role"] = UserProfile.ROLE_REALM_OWNER
|
|
opts["default_stream_groups"] = []
|
|
|
|
if ldap_external_auth_id_sync_enabled():
|
|
opts["external_auth_id_dict"] = {self.name: external_auth_id}
|
|
|
|
user_profile = do_create_user(
|
|
email,
|
|
None,
|
|
self._realm,
|
|
full_name,
|
|
tos_version=UserProfile.TOS_VERSION_BEFORE_FIRST_LOGIN,
|
|
acting_user=None,
|
|
**opts,
|
|
)
|
|
self.sync_avatar_from_ldap(user_profile, ldap_user)
|
|
self.sync_custom_profile_fields_from_ldap(user_profile, ldap_user)
|
|
|
|
return user_profile, True
|
|
|
|
|
|
class ZulipLDAPUser(_LDAPUser):
|
|
"""
|
|
This is an extension of the _LDAPUser class, with a realm attribute
|
|
attached to it. It's purpose is to call its inherited method
|
|
populate_user() which will sync the LDAP data with the corresponding
|
|
UserProfile. The realm attribute serves to uniquely identify the UserProfile
|
|
in case the LDAP user is registered to multiple realms.
|
|
"""
|
|
|
|
def __init__(self, *args: Any, **kwargs: Any) -> None:
|
|
self.realm: Realm = kwargs["realm"]
|
|
del kwargs["realm"]
|
|
|
|
super().__init__(*args, **kwargs)
|
|
|
|
def get_external_auth_id(self) -> str:
|
|
"""
|
|
The user's ldap DN is usually the best choice for the external_auth_id in LDAP
|
|
authentication, as DNs are unique within an ldap directory.
|
|
However, we also support configuring a custom attribute to use instead.
|
|
"""
|
|
attr_name = settings.AUTH_LDAP_USER_ATTR_MAP.get("unique_account_id")
|
|
if attr_name is None:
|
|
raise AssertionError("unique_account_id is not configured in AUTH_LDAP_USER_ATTR_MAP")
|
|
|
|
if attr_name.lower() == "dn":
|
|
return self.dn
|
|
|
|
external_auth_id = self.attrs[attr_name][0]
|
|
assert isinstance(external_auth_id, str)
|
|
return external_auth_id
|
|
|
|
# We intentionally want to create a transaction savepoint here. This only
|
|
# runs when syncing a user with LDAP during login or sync_ldap_user_data job
|
|
# and the performance cost should be negligible.
|
|
# A savepoint is needed because sync_ldap_user_data runs inside a large transaction
|
|
# syncing all users. LDAP sync exceptions can propagate through here, to be caught
|
|
# and logged up the callstack.
|
|
# When that occurs, we actually want to roll back this subtransaction, while allowing
|
|
# sync_ldap_user_data to go on to sync other users, without breaking its outermost
|
|
# transactions - which savepoint=False would do.
|
|
@transaction.atomic(savepoint=True) # intentional use of savepoint=True
|
|
def _get_or_create_user(self, force_populate: bool = False) -> UserProfile:
|
|
# This function is responsible for the core logic of syncing
|
|
# a user's data with ldap - run in both populate_user codepath
|
|
# and just-in-time user creation upon first login via LDAP.
|
|
#
|
|
# To ensure we don't end up with corrupted database state,
|
|
# we need to run these operations atomically.
|
|
return super()._get_or_create_user(force_populate=force_populate)
|
|
|
|
def _populate_user(self) -> None:
|
|
"""
|
|
Populates our User object with information from the LDAP directory.
|
|
"""
|
|
assert isinstance(self._user, UserProfile)
|
|
user_profile = self._user
|
|
original_role = user_profile.role
|
|
|
|
# _populate_user() will make whatever changes to the user's attributes
|
|
# that it needs - possibly changing the user's role multiple times e.g.
|
|
# as it cycles through various role setters in AUTH_LDAP_USER_FLAGS_BY_GROUP.
|
|
#
|
|
# For that reason, we want to only look at the final role value after
|
|
# it is executed. This is the actual change (if any) that should take place.
|
|
# This allows us to call do_change_user_role only once.
|
|
super()._populate_user()
|
|
if user_profile.role != original_role:
|
|
# Change the role properly, updating system groups.
|
|
updated_role = user_profile.role
|
|
user_profile.role = original_role
|
|
do_change_user_role(user_profile, updated_role, acting_user=None)
|
|
|
|
def _get_groups(self) -> _LDAPUserGroups:
|
|
groups = super()._get_groups()
|
|
if settings.DEVELOPMENT:
|
|
# HACK: We force the loading of all group dns, to coerce
|
|
# django-auth-ldap into caching the result - which
|
|
# makes it use this result when running is_member_of
|
|
# to check if the user is a member of a specific group.
|
|
#
|
|
# When this result isn't already loaded, django-auth-ldap
|
|
# runs a different ldap query, attempting to query for membership
|
|
# in the specific member group - which doesn't work in our fakeldap
|
|
# setup for mocking LDAP in tests and development.
|
|
#
|
|
# This hack is necessary for us to be able to test
|
|
# AUTH_LDAP_USER_FLAGS_BY_GROUP functionality.
|
|
groups.get_group_dns()
|
|
|
|
return groups
|
|
|
|
|
|
class ZulipLDAPUserPopulator(ZulipLDAPAuthBackendBase):
|
|
"""Just like ZulipLDAPAuthBackend, but doesn't let you log in. Used
|
|
for syncing data like names, avatars, and custom profile fields
|
|
from LDAP in `manage.py sync_ldap_user_data` as well as in
|
|
registration for organizations that use a different SSO solution
|
|
for managing login (often via RemoteUserBackend).
|
|
"""
|
|
|
|
def authenticate(
|
|
self,
|
|
request: HttpRequest | None = None,
|
|
*,
|
|
username: str,
|
|
password: str,
|
|
realm: Realm,
|
|
return_data: dict[str, Any] | None = None,
|
|
) -> UserProfile | None:
|
|
return None
|
|
|
|
def get_or_build_user(
|
|
self, username: str, ldap_user: ZulipLDAPUser
|
|
) -> tuple[UserProfile, bool]:
|
|
"""This is used only in non-authentication contexts such as:
|
|
./manage.py sync_ldap_user_data
|
|
"""
|
|
# Obtain the django username from the ldap_user object:
|
|
username = self.user_email_from_ldapuser(username, ldap_user)
|
|
|
|
# We set the built flag (which tells django-auth-ldap whether the user object
|
|
# was taken from the database or freshly built) to False - because in this codepath
|
|
# the user we're syncing of course already has to exist in the database.
|
|
user = get_user_by_delivery_email(username, ldap_user.realm)
|
|
built = False
|
|
# Synchronise the UserProfile with its LDAP attributes:
|
|
if ldap_should_sync_active_status():
|
|
user_disabled_in_ldap = self.is_user_disabled_in_ldap(ldap_user)
|
|
if user_disabled_in_ldap:
|
|
if user.is_active:
|
|
ldap_logger.info(
|
|
"Deactivating user %s because they are disabled in LDAP.",
|
|
user.delivery_email,
|
|
)
|
|
do_deactivate_user(user, acting_user=None)
|
|
# Do an early return to avoid trying to sync additional data.
|
|
return (user, built)
|
|
elif not user.is_active:
|
|
ldap_logger.info(
|
|
"Reactivating user %s because they are not disabled in LDAP.",
|
|
user.delivery_email,
|
|
)
|
|
do_reactivate_user(user, acting_user=None)
|
|
|
|
self.sync_avatar_from_ldap(user, ldap_user)
|
|
self.sync_full_name_from_ldap(user, ldap_user)
|
|
self.sync_custom_profile_fields_from_ldap(user, ldap_user)
|
|
self.sync_groups_from_ldap(user, ldap_user)
|
|
return (user, built)
|
|
|
|
|
|
class PopulateUserLDAPError(ZulipLDAPError):
|
|
pass
|
|
|
|
|
|
@receiver(ldap_error, sender=ZulipLDAPUserPopulator)
|
|
def catch_ldap_error(signal: Signal, **kwargs: Any) -> None:
|
|
"""
|
|
Inside django_auth_ldap populate_user(), if LDAPError is raised,
|
|
e.g. due to invalid connection credentials, the function catches it
|
|
and emits a signal (ldap_error) to communicate this error to others.
|
|
We normally don't use signals, but here there's no choice, so in this function
|
|
we essentially convert the signal to a normal exception that will properly
|
|
propagate out of django_auth_ldap internals.
|
|
"""
|
|
# The exception message can contain the password (if it was invalid),
|
|
# so it seems better not to log that, and only use the original exception's name here.
|
|
raise PopulateUserLDAPError(type(kwargs["exception"]).__name__)
|
|
|
|
|
|
def sync_user_from_ldap(user_profile: UserProfile, logger: logging.Logger) -> bool:
|
|
backend = ZulipLDAPUserPopulator()
|
|
try:
|
|
ldap_username = backend.django_to_ldap_username(user_profile.delivery_email)
|
|
except NoMatchingLDAPUserError:
|
|
if (
|
|
settings.ONLY_LDAP
|
|
if settings.LDAP_DEACTIVATE_NON_MATCHING_USERS is None
|
|
else settings.LDAP_DEACTIVATE_NON_MATCHING_USERS
|
|
):
|
|
do_deactivate_user(user_profile, acting_user=None)
|
|
logger.info("Deactivated non-matching user: %s", user_profile.delivery_email)
|
|
return True
|
|
elif user_profile.is_active:
|
|
logger.warning("Did not find %s in LDAP.", user_profile.delivery_email)
|
|
return False
|
|
|
|
# What one would expect to see like to do here is just a call to
|
|
# `backend.populate_user`, which in turn just creates the
|
|
# `_LDAPUser` object and calls `ldap_user.populate_user()` on
|
|
# that. Unfortunately, that will produce incorrect results in the
|
|
# case that the server has multiple Zulip users in different
|
|
# realms associated with a single LDAP user, because
|
|
# `django-auth-ldap` isn't implemented with the possibility of
|
|
# multiple realms on different subdomains in mind.
|
|
#
|
|
# To address this, we construct a version of the _LDAPUser class
|
|
# extended to store the realm of the target user, and call its
|
|
# `.populate_user` function directly.
|
|
#
|
|
# Ideally, we'd contribute changes to `django-auth-ldap` upstream
|
|
# making this flow possible in a more directly supported fashion.
|
|
updated_user = ZulipLDAPUser(backend, ldap_username, realm=user_profile.realm).populate_user()
|
|
if updated_user:
|
|
logger.info("Updated %s.", user_profile.delivery_email)
|
|
return True
|
|
|
|
raise PopulateUserLDAPError(f"populate_user unexpectedly returned {updated_user}")
|
|
|
|
|
|
# Quick tool to test whether you're correctly authenticating to LDAP
|
|
def query_ldap(email: str) -> list[str]:
|
|
values = []
|
|
backend = next(
|
|
(backend for backend in get_backends() if isinstance(backend, LDAPBackend)), None
|
|
)
|
|
if backend is not None:
|
|
try:
|
|
ldap_username = backend.django_to_ldap_username(email)
|
|
except NoMatchingLDAPUserError as e:
|
|
values.append(f"No such user found: {e}")
|
|
return values
|
|
|
|
ldap_attrs = _LDAPUser(backend, ldap_username).attrs
|
|
|
|
for django_field, ldap_field in settings.AUTH_LDAP_USER_ATTR_MAP.items():
|
|
value = ldap_attrs.get(ldap_field, ["LDAP field not present"])[0]
|
|
if django_field == "avatar" and isinstance(value, bytes):
|
|
value = "(An avatar image file)"
|
|
values.append(f"{django_field}: {value}")
|
|
if settings.LDAP_EMAIL_ATTR is not None:
|
|
values.append("{}: {}".format("email", ldap_attrs[settings.LDAP_EMAIL_ATTR][0]))
|
|
else:
|
|
values.append("LDAP backend not configured on this server.")
|
|
return values
|
|
|
|
|
|
class DevAuthBackend(ZulipAuthMixin):
|
|
"""Allow logging in as any user without a password. This is used for
|
|
convenience when developing Zulip, and is disabled in production."""
|
|
|
|
name = "dev"
|
|
|
|
def authenticate(
|
|
self,
|
|
request: HttpRequest | None = None,
|
|
*,
|
|
dev_auth_username: str,
|
|
realm: Realm,
|
|
return_data: dict[str, Any] | None = None,
|
|
) -> UserProfile | None:
|
|
if not dev_auth_enabled(realm):
|
|
return None
|
|
return common_get_active_user(dev_auth_username, realm, return_data=return_data)
|
|
|
|
|
|
class ExternalAuthMethodDictT(TypedDict):
|
|
name: str
|
|
display_name: str
|
|
display_icon: str | None
|
|
login_url: str
|
|
signup_url: str
|
|
|
|
|
|
class ExternalAuthMethod(ABC):
|
|
"""
|
|
To register a backend as an external_authentication_method, it should
|
|
subclass ExternalAuthMethod and define its dict_representation
|
|
classmethod, and finally use the external_auth_method class decorator to
|
|
get added to the EXTERNAL_AUTH_METHODS list.
|
|
"""
|
|
|
|
auth_backend_name = "undeclared"
|
|
name = "undeclared"
|
|
display_icon: str | None = None
|
|
|
|
# Used to determine how to order buttons on login form, backend with
|
|
# higher sort order are displayed first.
|
|
sort_order = 0
|
|
|
|
@classmethod
|
|
@abstractmethod
|
|
def dict_representation(cls, realm: Realm | None = None) -> list[ExternalAuthMethodDictT]:
|
|
"""
|
|
Method returning dictionaries representing the authentication methods
|
|
corresponding to the backend that subclasses this. The documentation
|
|
for the external_authentication_methods field of the /server_settings endpoint
|
|
explains the details of these dictionaries.
|
|
This returns a list, because one backend can support configuring multiple methods,
|
|
that are all serviced by that backend - our SAML backend is an example of that.
|
|
"""
|
|
|
|
|
|
EXTERNAL_AUTH_METHODS: list[type[ExternalAuthMethod]] = []
|
|
|
|
|
|
def external_auth_method(cls: type[ExternalAuthMethod]) -> type[ExternalAuthMethod]:
|
|
assert issubclass(cls, ExternalAuthMethod)
|
|
|
|
EXTERNAL_AUTH_METHODS.append(cls)
|
|
return cls
|
|
|
|
|
|
# We want to be able to store this data in Redis, so it has to be easy to serialize.
|
|
# That's why we avoid having fields that could pose a problem for that.
|
|
class ExternalAuthDataDict(TypedDict, total=False):
|
|
subdomain: str
|
|
full_name: str
|
|
email: str
|
|
role: int | None
|
|
is_signup: bool
|
|
is_realm_creation: bool
|
|
redirect_to: str
|
|
mobile_flow_otp: str | None
|
|
desktop_flow_otp: str | None
|
|
multiuse_object_key: str
|
|
full_name_validated: bool
|
|
group_memberships_sync_map: dict[str, bool] | None
|
|
# The mobile app doesn't actually use a session, so this
|
|
# data is not applicable there.
|
|
params_to_store_in_authenticated_session: dict[str, str]
|
|
|
|
|
|
class ExternalAuthResult:
|
|
LOGIN_KEY_PREFIX = "login_key_"
|
|
LOGIN_KEY_FORMAT = LOGIN_KEY_PREFIX + "{token}"
|
|
LOGIN_KEY_EXPIRATION_SECONDS = 15
|
|
LOGIN_TOKEN_LENGTH = UserProfile.API_KEY_LENGTH
|
|
|
|
def __init__(
|
|
self,
|
|
*,
|
|
user_profile: UserProfile | None = None,
|
|
data_dict: ExternalAuthDataDict | None = None,
|
|
request: HttpRequest | None = None,
|
|
login_token: str | None = None,
|
|
delete_stored_data: bool = True,
|
|
) -> None:
|
|
if data_dict is None:
|
|
data_dict = {}
|
|
|
|
if login_token is not None:
|
|
assert (not data_dict) and (user_profile is None), (
|
|
"Passing in data_dict or user_profile with login_token is disallowed."
|
|
)
|
|
assert request is not None, "Passing in request with login_token is required."
|
|
self.instantiate_with_token(request, login_token, delete_stored_data)
|
|
else:
|
|
self.data_dict = data_dict.copy()
|
|
self.user_profile = user_profile
|
|
|
|
if self.user_profile is not None:
|
|
# Ensure data inconsistent with the user_profile wasn't passed in inside the data_dict argument.
|
|
assert (
|
|
"full_name" not in data_dict
|
|
or data_dict["full_name"] == self.user_profile.full_name
|
|
)
|
|
assert (
|
|
"email" not in data_dict
|
|
or data_dict["email"].lower() == self.user_profile.delivery_email.lower()
|
|
)
|
|
# Update these data_dict fields to ensure consistency with self.user_profile. This is mostly
|
|
# defensive code, but is useful in these scenarios:
|
|
# 1. user_profile argument was passed in, and no full_name or email_data in the data_dict arg.
|
|
# 2. We're instantiating from the login_token and the user has changed their full_name since
|
|
# the data was stored under the token.
|
|
self.data_dict["full_name"] = self.user_profile.full_name
|
|
self.data_dict["email"] = self.user_profile.delivery_email
|
|
|
|
if "subdomain" not in self.data_dict:
|
|
self.data_dict["subdomain"] = self.user_profile.realm.subdomain
|
|
if not self.user_profile.is_mirror_dummy:
|
|
self.data_dict["is_signup"] = False
|
|
|
|
def store_data(self) -> str:
|
|
key = put_dict_in_redis(
|
|
redis_client,
|
|
self.LOGIN_KEY_FORMAT,
|
|
self.data_dict,
|
|
expiration_seconds=self.LOGIN_KEY_EXPIRATION_SECONDS,
|
|
token_length=self.LOGIN_TOKEN_LENGTH,
|
|
)
|
|
token = key.split(self.LOGIN_KEY_PREFIX, 1)[1] # remove the prefix
|
|
return token
|
|
|
|
def instantiate_with_token(
|
|
self, request: HttpRequest, token: str, delete_stored_data: bool = True
|
|
) -> None:
|
|
key = self.LOGIN_KEY_FORMAT.format(token=token)
|
|
data = get_dict_from_redis(redis_client, self.LOGIN_KEY_FORMAT, key)
|
|
if data is None or None in [data.get("email"), data.get("subdomain")]:
|
|
raise self.InvalidTokenError
|
|
|
|
if delete_stored_data:
|
|
redis_client.delete(key)
|
|
|
|
self.data_dict = cast(ExternalAuthDataDict, data)
|
|
|
|
# Here we refetch the UserProfile object (if any) for this
|
|
# ExternalAuthResult. Using authenticate() will re-check for
|
|
# (unlikely) races like the realm or user having been deactivated
|
|
# between generating this ExternalAuthResult and accessing it.
|
|
#
|
|
# In theory, we should return_data here so the caller can do
|
|
# more customized error messages for those unlikely races, but
|
|
# it's likely not worth implementing.
|
|
realm = get_realm(data["subdomain"])
|
|
auth_result = authenticate(
|
|
request=request, username=data["email"], realm=realm, use_dummy_backend=True
|
|
)
|
|
if auth_result is not None:
|
|
assert isinstance(auth_result, UserProfile)
|
|
self.user_profile = auth_result
|
|
|
|
class InvalidTokenError(Exception):
|
|
pass
|
|
|
|
|
|
class SyncUserError(Exception):
|
|
pass
|
|
|
|
|
|
def sync_user_profile_custom_fields(
|
|
user_profile: UserProfile, custom_field_name_to_value: dict[str, Any]
|
|
) -> None:
|
|
fields_by_var_name: dict[str, CustomProfileField] = {}
|
|
custom_profile_fields = custom_profile_fields_for_realm(user_profile.realm.id)
|
|
for field in custom_profile_fields:
|
|
var_name = "_".join(field.name.lower().split(" "))
|
|
fields_by_var_name[var_name] = field
|
|
|
|
existing_values = {}
|
|
for data in user_profile.profile_data():
|
|
var_name = "_".join(data["name"].lower().split(" "))
|
|
existing_values[var_name] = data["value"]
|
|
|
|
profile_data: list[ProfileDataElementUpdateDict] = []
|
|
for var_name, value in custom_field_name_to_value.items():
|
|
try:
|
|
field = fields_by_var_name[var_name]
|
|
except KeyError:
|
|
raise SyncUserError(f"Custom profile field with name {var_name} not found.")
|
|
if existing_values.get(var_name) == value:
|
|
continue
|
|
try:
|
|
validate_user_custom_profile_field(user_profile.realm.id, field, value)
|
|
except ValidationError as error:
|
|
raise SyncUserError(f"Invalid data for {var_name} field: {error.message}")
|
|
profile_data.append(
|
|
{
|
|
"id": field.id,
|
|
"value": value,
|
|
}
|
|
)
|
|
do_update_user_custom_profile_data_if_changed(user_profile, profile_data)
|
|
|
|
|
|
@external_auth_method
|
|
class ZulipRemoteUserBackend(ZulipAuthMixin, RemoteUserBackend, ExternalAuthMethod):
|
|
"""Authentication backend that reads the Apache REMOTE_USER variable.
|
|
Used primarily in enterprise environments with an SSO solution
|
|
that has an Apache REMOTE_USER integration. For manual testing, see
|
|
|
|
https://zulip.readthedocs.io/en/latest/production/authentication-methods.html
|
|
|
|
See also remote_user_sso in zerver/views/auth.py.
|
|
"""
|
|
|
|
auth_backend_name = "RemoteUser"
|
|
name = "remoteuser"
|
|
display_icon = None
|
|
# If configured, this backend should have its button near the top of the list.
|
|
sort_order = 9000
|
|
|
|
create_unknown_user = False
|
|
|
|
@override
|
|
def authenticate( # type: ignore[override] # authenticate has an incompatible signature with ModelBackend and BaseBackend
|
|
self,
|
|
request: HttpRequest | None = None,
|
|
*,
|
|
remote_user: str,
|
|
realm: Realm,
|
|
return_data: dict[str, Any] | None = None,
|
|
) -> UserProfile | None:
|
|
if not auth_enabled_helper(["RemoteUser"], realm):
|
|
return None
|
|
|
|
email = remote_user_to_email(remote_user)
|
|
return common_get_active_user(email, realm, return_data=return_data)
|
|
|
|
@classmethod
|
|
@override
|
|
def dict_representation(cls, realm: Realm | None = None) -> list[ExternalAuthMethodDictT]:
|
|
return [
|
|
dict(
|
|
name=cls.name,
|
|
display_name="SSO",
|
|
display_icon=cls.display_icon,
|
|
# The user goes to the same URL for both login and signup:
|
|
login_url=reverse("start-login-sso"),
|
|
signup_url=reverse("start-login-sso"),
|
|
)
|
|
]
|
|
|
|
|
|
def redirect_to_signup(realm: Realm) -> HttpResponseRedirect:
|
|
signup_url = reverse("register")
|
|
redirect_url = realm.url + signup_url
|
|
return HttpResponseRedirect(redirect_url)
|
|
|
|
|
|
def redirect_to_login(realm: Realm) -> HttpResponseRedirect:
|
|
login_url = reverse("login_page", kwargs={"template_name": "zerver/login.html"})
|
|
redirect_url = realm.url + login_url
|
|
return HttpResponseRedirect(redirect_url)
|
|
|
|
|
|
def redirect_deactivated_user_to_login(realm: Realm, email: str) -> HttpResponseRedirect:
|
|
# Specifying the template name makes sure that the user is not redirected to dev_login in case of
|
|
# a deactivated account on a test server.
|
|
redirect_url = realm.url + reverse(
|
|
"login_page", kwargs={"template_name": "zerver/login.html"}, query={"is_deactivated": email}
|
|
)
|
|
return HttpResponseRedirect(redirect_url)
|
|
|
|
|
|
@transaction.atomic(savepoint=False)
|
|
def sync_groups_for_prereg_user(
|
|
prereg_user: PreregistrationUser, group_memberships_sync_map: dict[str, bool]
|
|
) -> None:
|
|
assert prereg_user.realm is not None
|
|
realm = prereg_user.realm
|
|
|
|
group_names_to_ensure_member = [
|
|
group_name for group_name, is_member in group_memberships_sync_map.items() if is_member
|
|
]
|
|
group_names_to_ensure_not_member = [
|
|
group_name for group_name, is_member in group_memberships_sync_map.items() if not is_member
|
|
]
|
|
|
|
groups_to_ensure_member = list(
|
|
NamedUserGroup.objects.filter(realm=realm, name__in=group_names_to_ensure_member)
|
|
)
|
|
groups_to_ensure_not_member = list(
|
|
NamedUserGroup.objects.filter(realm=realm, name__in=group_names_to_ensure_not_member)
|
|
)
|
|
|
|
prereg_user.groups.add(*groups_to_ensure_member)
|
|
prereg_user.groups.remove(*groups_to_ensure_not_member)
|
|
|
|
final_group_names = set(prereg_user.groups.all().values_list("name", flat=True))
|
|
|
|
stringified_dict = json.dumps(group_memberships_sync_map, sort_keys=True)
|
|
logging.info(
|
|
"Synced user groups for PreregistrationUser %s in %s: %s. Final groups set: %s",
|
|
prereg_user.id,
|
|
realm.id,
|
|
stringified_dict,
|
|
final_group_names,
|
|
)
|
|
|
|
|
|
def sync_groups(
|
|
all_group_names: set[str],
|
|
intended_group_names: set[str],
|
|
user_profile: UserProfile,
|
|
logger: logging.Logger,
|
|
) -> None:
|
|
"""
|
|
Ensure that for this user:
|
|
- They are in every NamedUserGroup named in intended_names.
|
|
- They are not in any NamedUserGroup in all_group_names minus intended_names.
|
|
The idea is that intended_group_names is the set of names of groups to which
|
|
the user should belong, within the universe specified by all_group_names.
|
|
"""
|
|
for system_group_name in SystemGroups.GROUP_DISPLAY_NAME_MAP:
|
|
# system groups are not allowed to be synced.
|
|
assert system_group_name not in all_group_names
|
|
assert system_group_name not in intended_group_names
|
|
|
|
user_id = user_profile.id
|
|
realm = user_profile.realm
|
|
logger.debug("Starting group sync for user %s in realm %s", user_id, realm.string_id)
|
|
|
|
existing = set(
|
|
UserGroupMembership.objects.filter(
|
|
user_group__realm=realm,
|
|
user_group__named_user_group__name__in=all_group_names,
|
|
user_group__named_user_group__is_system_group=False,
|
|
user_profile__id=user_id,
|
|
).values_list("user_group__named_user_group__name", flat=True)
|
|
)
|
|
logger.debug(
|
|
"intended groups for user <%s>: %s; current groups: %s",
|
|
user_id,
|
|
intended_group_names,
|
|
existing,
|
|
)
|
|
|
|
to_add = intended_group_names - existing
|
|
to_remove = existing - intended_group_names
|
|
|
|
if to_add:
|
|
logger.debug("Adding user %s to groups %s", user_id, to_add)
|
|
add_groups = list(NamedUserGroup.objects.filter(name__in=to_add, realm=realm))
|
|
bulk_add_members_to_user_groups(add_groups, [user_id], acting_user=None)
|
|
|
|
if to_remove:
|
|
logger.debug("Removing user %s from groups %s", user_id, to_remove)
|
|
remove_groups = list(NamedUserGroup.objects.filter(name__in=to_remove, realm=realm))
|
|
bulk_remove_members_from_user_groups(remove_groups, [user_id], acting_user=None)
|
|
|
|
logger.debug("Finished group sync for user %s", user_id)
|
|
|
|
|
|
@dataclass
|
|
class SocialAuthSyncNewUserInfo:
|
|
role: int | None
|
|
group_memberships_sync_map: dict[str, bool]
|
|
|
|
|
|
def social_auth_sync_user_attributes(
|
|
realm: Realm, user_profile: UserProfile | None, extra_attrs: dict[str, Any], backend: Any
|
|
) -> SocialAuthSyncNewUserInfo | None:
|
|
"""
|
|
Syncs user attributes based on the SOCIAL_AUTH_SYNC_ATTRS_DICT setting.
|
|
Only supports:
|
|
1. Syncing the role. This is plumbed through to user creation, so can be
|
|
used to immediately create new users with their role set based on an attribute
|
|
provided by the IdP.
|
|
2. Syncing custom attributes. This isn't supported for user creation,
|
|
so they'll only be synced during the user's next login, not during
|
|
signup.
|
|
"""
|
|
# This is only supported for SAML right now, though the design
|
|
# is meant to be easy to extend this to other backends if desired.
|
|
# Unlike LDAP or SCIM, this hook can only do syncing during the authentication
|
|
# flow, as that's when the data is provided and we don't have a way to query
|
|
# for it otherwise.
|
|
if backend.name != "saml":
|
|
assert not extra_attrs
|
|
return None
|
|
|
|
attrs_by_backend = settings.SOCIAL_AUTH_SYNC_ATTRS_DICT.get(realm.subdomain, {})
|
|
attrs_config = attrs_by_backend.get(backend.name, {})
|
|
external_group_name_to_zulip_group_name: dict[str, str] = {}
|
|
groups_config_list = cast(list[str | tuple[str, str]], attrs_config.get("groups", []))
|
|
if groups_config_list:
|
|
# Group sync is only supported for SAML for the foreseeable time.
|
|
assert backend.name == "saml"
|
|
|
|
for group_name in groups_config_list:
|
|
# the objects in the config are either straight-forward group names
|
|
# or tuples (<saml group name>, <zulip group name>) indicating the
|
|
# obvious mapping.
|
|
if isinstance(group_name, str):
|
|
external_group_name_to_zulip_group_name[group_name] = group_name
|
|
else:
|
|
saml_group_name, zulip_group_name = group_name
|
|
external_group_name_to_zulip_group_name[saml_group_name] = zulip_group_name
|
|
|
|
should_sync_groups = bool(external_group_name_to_zulip_group_name)
|
|
|
|
profile_field_name_to_attr_name = cast(
|
|
dict[str, str], {key: value for key, value in attrs_config.items() if key != "groups"}
|
|
)
|
|
|
|
if should_sync_groups:
|
|
syncable_group_names = set(external_group_name_to_zulip_group_name.values())
|
|
|
|
# pop zulip_groups from extra_attrs, so that only user attribute sync
|
|
# values remain there.
|
|
# zulip_groups being absent should be treated as if no group memberships
|
|
# are desired. That's because Okta doesn't send the SAML attribute at all
|
|
# if the user has no group memberships. Thus we treat this absence as
|
|
# an empty list.
|
|
all_received_group_names = extra_attrs.pop("zulip_groups", [])
|
|
external_group_names = [
|
|
name
|
|
for name in all_received_group_names
|
|
# Ignore group names which aren't configured.
|
|
if name in external_group_name_to_zulip_group_name
|
|
]
|
|
intended_group_names = {
|
|
external_group_name_to_zulip_group_name[external_name]
|
|
for external_name in external_group_names
|
|
}
|
|
|
|
# It's important to log the information about what we received in the request and what Zulip groups
|
|
# that was translated to based on the configuration - otherwise debugging a misconfiguration would be
|
|
# a very painful process.
|
|
#
|
|
# Notably, it's generally expected for the list of received groups to be shorter than the "translated"
|
|
# list of intended Zulip groups. The expected way for admins to configure what is sent in zulip_groups
|
|
# is to send *all* the groups the user belongs to in their user directory. That is very likely to contain
|
|
# some groups that are irrelevant to their Zulip setup and thus shouldn't be translated into any
|
|
# Zulip group memberships.
|
|
# For example, Okta sends the Everyone group, which is just a built-in group inside of Okta described
|
|
# in their documentation as
|
|
# "catch-all group where every single user in the Okta instance automatically lands".
|
|
# That obviously will be irrelevant to Zulip group memberships in almost all configurations.
|
|
user_string = (
|
|
f"<user:{user_profile.id}>" if user_profile is not None else "<new user signup>"
|
|
)
|
|
backend.logger.info(
|
|
"social_auth_sync_user_attributes:%s: received group names: %s|intended Zulip groups: %s. group mapping used: %s",
|
|
user_string,
|
|
sorted(all_received_group_names),
|
|
sorted(intended_group_names),
|
|
external_group_name_to_zulip_group_name,
|
|
)
|
|
|
|
should_sync_user_attrs = extra_attrs and profile_field_name_to_attr_name
|
|
if not (should_sync_user_attrs or should_sync_groups):
|
|
return None
|
|
|
|
user_id = None
|
|
if user_profile is not None:
|
|
user_id = user_profile.id
|
|
|
|
custom_profile_field_name_to_value = {}
|
|
new_role = None
|
|
for field_name, attr_name in profile_field_name_to_attr_name.items():
|
|
if field_name == "role":
|
|
attr_value = extra_attrs.get(attr_name)
|
|
if not attr_value:
|
|
continue
|
|
if attr_value not in UserProfile.ROLE_API_NAME_TO_ID:
|
|
backend.logger.warning(
|
|
"Ignoring unsupported role value %s for user %s in SOCIAL_AUTH_SYNC_ATTRS_DICT",
|
|
attr_value,
|
|
user_id,
|
|
)
|
|
continue
|
|
new_role = UserProfile.ROLE_API_NAME_TO_ID[attr_value]
|
|
elif field_name.startswith("custom__"):
|
|
attr_value = extra_attrs.get(attr_name)
|
|
if attr_value is None:
|
|
continue
|
|
custom_profile_field_name_to_value[field_name.removeprefix("custom__")] = attr_value
|
|
else:
|
|
backend.logger.warning(
|
|
"Ignoring unsupported UserProfile field %s in SOCIAL_AUTH_SYNC_ATTRS_DICT",
|
|
field_name,
|
|
)
|
|
|
|
if user_profile is None:
|
|
# We don't support user creation with custom profile fields, so just
|
|
# return role and group memberships so that it can be plumbed through to the signup flow.
|
|
if new_role is not None:
|
|
backend.logger.info(
|
|
"Returning role %s for user creation", UserProfile.ROLE_ID_TO_API_NAME[new_role]
|
|
)
|
|
if should_sync_groups:
|
|
group_memberships_sync_map = {
|
|
group_name: (group_name in intended_group_names)
|
|
for group_name in syncable_group_names
|
|
}
|
|
else:
|
|
group_memberships_sync_map = {}
|
|
|
|
return SocialAuthSyncNewUserInfo(
|
|
role=new_role, group_memberships_sync_map=group_memberships_sync_map
|
|
)
|
|
|
|
# Based on the information collected above, sync what's needed for the user_profile.
|
|
old_role = user_profile.role
|
|
if new_role is not None and old_role != new_role:
|
|
do_change_user_role(user_profile, new_role, acting_user=None)
|
|
backend.logger.info(
|
|
"Set role %s for user %s", UserProfile.ROLE_ID_TO_API_NAME[new_role], user_profile.id
|
|
)
|
|
|
|
try:
|
|
sync_user_profile_custom_fields(user_profile, custom_profile_field_name_to_value)
|
|
except SyncUserError as e:
|
|
backend.logger.warning(
|
|
"Exception while syncing custom profile fields for user %s: %s",
|
|
user_profile.id,
|
|
str(e),
|
|
)
|
|
|
|
if should_sync_groups:
|
|
sync_groups(
|
|
all_group_names=syncable_group_names,
|
|
intended_group_names=intended_group_names,
|
|
user_profile=user_profile,
|
|
logger=backend.logger,
|
|
)
|
|
|
|
return None
|
|
|
|
|
|
def social_associate_user_helper(
|
|
backend: "SocialAuthMixin", return_data: dict[str, Any], *args: Any, **kwargs: Any
|
|
) -> HttpResponse | UserProfile | None:
|
|
"""Responsible for doing the Zulip account lookup and validation parts
|
|
of the Zulip social auth pipeline (similar to the authenticate()
|
|
methods in most other auth backends in this file).
|
|
|
|
Returns a UserProfile object for successful authentication, and None otherwise.
|
|
"""
|
|
subdomain = backend.strategy.session_get("subdomain")
|
|
try:
|
|
realm = get_realm(subdomain)
|
|
except Realm.DoesNotExist:
|
|
return_data["invalid_realm"] = True
|
|
return None
|
|
return_data["realm_id"] = realm.id
|
|
return_data["realm_string_id"] = realm.string_id
|
|
|
|
if not auth_enabled_helper([backend.auth_backend_name], realm):
|
|
return_data["auth_backend_disabled"] = True
|
|
return None
|
|
|
|
if "auth_failed_reason" in kwargs.get("response", {}):
|
|
return_data["social_auth_failed_reason"] = kwargs["response"]["auth_failed_reason"]
|
|
return None
|
|
elif hasattr(backend, "get_verified_emails"):
|
|
# Some social backends, like GitHubAuthBackend, don't
|
|
# guarantee that the `details` data is validated (i.e., it's
|
|
# possible users can put any string they want in the "email"
|
|
# field of the `details` object). For those backends, we have
|
|
# custom per-backend code to properly fetch only verified
|
|
# email addresses from the appropriate third-party API.
|
|
verified_emails = backend.get_verified_emails(realm, *args, **kwargs)
|
|
verified_emails_length = len(verified_emails)
|
|
if verified_emails_length == 0:
|
|
# TODO: Provide a nice error message screen to the user
|
|
# for this case, rather than just logging a warning.
|
|
backend.logger.warning(
|
|
"Social auth (%s) failed because user has no verified emails",
|
|
backend.auth_backend_name,
|
|
)
|
|
return_data["email_not_verified"] = True
|
|
return None
|
|
|
|
if verified_emails_length == 1:
|
|
chosen_email = verified_emails[0]
|
|
else:
|
|
chosen_email = backend.strategy.request_data().get("email")
|
|
|
|
if not chosen_email:
|
|
avatars = {} # Dict[str, str]
|
|
existing_account_emails = []
|
|
for email in verified_emails:
|
|
existing_account = common_get_active_user(email, realm, {})
|
|
if existing_account is not None:
|
|
existing_account_emails.append(email)
|
|
avatars[email] = avatar_url(existing_account)
|
|
|
|
if (
|
|
len(existing_account_emails) != 1
|
|
or backend.strategy.session_get("is_signup") == "1"
|
|
):
|
|
unverified_emails = []
|
|
if hasattr(backend, "get_unverified_emails"):
|
|
unverified_emails = backend.get_unverified_emails(realm, *args, **kwargs)
|
|
return render(
|
|
backend.strategy.request,
|
|
"zerver/social_auth_select_email.html",
|
|
context={
|
|
"primary_email": verified_emails[0],
|
|
"verified_non_primary_emails": verified_emails[1:],
|
|
"unverified_emails": unverified_emails,
|
|
"backend": "github",
|
|
"avatar_urls": avatars,
|
|
},
|
|
)
|
|
else:
|
|
chosen_email = existing_account_emails[0]
|
|
|
|
try:
|
|
validate_email(chosen_email)
|
|
except ValidationError:
|
|
return_data["invalid_email"] = True
|
|
return None
|
|
|
|
if chosen_email not in verified_emails:
|
|
# If a user edits the submit value for the choose email form, we might
|
|
# end up with a wrong email associated with the account. The below code
|
|
# takes care of that.
|
|
backend.logger.warning(
|
|
"Social auth (%s) failed because user has no verified"
|
|
" emails associated with the account",
|
|
backend.auth_backend_name,
|
|
)
|
|
return_data["email_not_associated"] = True
|
|
return None
|
|
|
|
validated_email = chosen_email
|
|
else:
|
|
try:
|
|
validate_email(kwargs["details"].get("email"))
|
|
except ValidationError:
|
|
return_data["invalid_email"] = True
|
|
return None
|
|
validated_email = kwargs["details"].get("email")
|
|
|
|
if not validated_email: # nocoverage
|
|
# This code path isn't used with GitHubAuthBackend, but may be relevant for other
|
|
# social auth backends.
|
|
return_data["invalid_email"] = True
|
|
return None
|
|
|
|
return_data["valid_attestation"] = True
|
|
return_data["validated_email"] = validated_email
|
|
user_profile = common_get_active_user(validated_email, realm, return_data)
|
|
|
|
full_name = kwargs["details"].get("fullname")
|
|
first_name = kwargs["details"].get("first_name")
|
|
last_name = kwargs["details"].get("last_name")
|
|
|
|
if all(name is None for name in [full_name, first_name, last_name]) and backend.name not in [
|
|
"apple",
|
|
"saml",
|
|
"oidc",
|
|
]:
|
|
# (1) Apple authentication provides the user's name only the very first time a user tries to log in.
|
|
# So if the user aborts login or otherwise is doing this the second time,
|
|
# we won't have any name data.
|
|
# (2) Some SAML or OIDC IdPs may not send any name value if the user doesn't
|
|
# have them set in the IdP's directory.
|
|
#
|
|
# The name will just default to the empty string in the code below.
|
|
|
|
# We need custom code here for any social auth backends
|
|
# that don't provide name details feature.
|
|
raise AssertionError("Social auth backend doesn't provide name")
|
|
|
|
if full_name:
|
|
return_data["full_name"] = full_name
|
|
else:
|
|
# Some authentications methods like Apple and SAML send
|
|
# first name and last name as separate attributes. In that case
|
|
# we construct the full name from them.
|
|
# strip removes the unnecessary ' '
|
|
return_data["full_name"] = f"{first_name or ''} {last_name or ''}".strip()
|
|
|
|
return_data["extra_attrs"] = kwargs["details"].get("extra_attrs", {})
|
|
|
|
return user_profile
|
|
|
|
|
|
@partial
|
|
def social_auth_associate_user(
|
|
backend: BaseAuth, *args: Any, **kwargs: Any
|
|
) -> HttpResponse | dict[str, Any]:
|
|
"""A simple wrapper function to reformat the return data from
|
|
social_associate_user_helper as a dictionary. The
|
|
python-social-auth infrastructure will then pass those values into
|
|
later stages of settings.SOCIAL_AUTH_PIPELINE, such as
|
|
social_auth_finish, as kwargs.
|
|
"""
|
|
assert isinstance(backend, SocialAuthMixin)
|
|
partial_token = backend.strategy.request_data().get("partial_token")
|
|
return_data: dict[str, Any] = {}
|
|
user_profile = social_associate_user_helper(backend, return_data, *args, **kwargs)
|
|
|
|
if isinstance(user_profile, HttpResponse):
|
|
return user_profile
|
|
else:
|
|
return {
|
|
"user_profile": user_profile,
|
|
"return_data": return_data,
|
|
"partial_token": partial_token,
|
|
"partial_backend_name": backend,
|
|
}
|
|
|
|
|
|
def social_auth_finish(
|
|
backend: Any, details: dict[str, Any], response: HttpResponse, *args: Any, **kwargs: Any
|
|
) -> HttpResponse | None:
|
|
"""Given the determination in social_auth_associate_user for whether
|
|
the user should be authenticated, this takes care of actually
|
|
logging in the user (if appropriate) and redirecting the browser
|
|
to the appropriate next page depending on the situation. Read the
|
|
comments below as well as login_or_register_remote_user in
|
|
`zerver/views/auth.py` for the details on how that dispatch works.
|
|
"""
|
|
from zerver.views.auth import login_or_register_remote_user, redirect_and_log_into_subdomain
|
|
|
|
user_profile = kwargs["user_profile"]
|
|
return_data = kwargs["return_data"]
|
|
|
|
no_verified_email = return_data.get("email_not_verified")
|
|
auth_backend_disabled = return_data.get("auth_backend_disabled")
|
|
inactive_user = return_data.get("inactive_user")
|
|
inactive_realm = return_data.get("inactive_realm")
|
|
invalid_realm = return_data.get("invalid_realm")
|
|
invalid_email = return_data.get("invalid_email")
|
|
auth_failed_reason = return_data.get("social_auth_failed_reason")
|
|
email_not_associated = return_data.get("email_not_associated")
|
|
|
|
if invalid_realm:
|
|
# User has passed an invalid subdomain param - this shouldn't happen in the normal flow,
|
|
# unless the user manually edits the param. In any case, it's most appropriate to just take
|
|
# them to find_account, as there isn't even an appropriate subdomain to take them to the login
|
|
# form on.
|
|
return HttpResponseRedirect(reverse("find_account"))
|
|
|
|
realm = Realm.objects.get(id=return_data["realm_id"])
|
|
if auth_backend_disabled or inactive_realm or no_verified_email or email_not_associated:
|
|
# Redirect to login page. We can't send to registration
|
|
# workflow with these errors. We will redirect to login page.
|
|
return redirect_to_login(realm)
|
|
if inactive_user:
|
|
backend.logger.info(
|
|
"Failed login attempt for deactivated account: %s@%s",
|
|
return_data["inactive_user_id"],
|
|
return_data["realm_string_id"],
|
|
)
|
|
return redirect_deactivated_user_to_login(realm, return_data["validated_email"])
|
|
|
|
if invalid_email:
|
|
# In case of invalid email, we will end up on registration page.
|
|
# This seems better than redirecting to login page.
|
|
backend.logger.warning(
|
|
"%s got invalid email argument.",
|
|
backend.auth_backend_name,
|
|
)
|
|
return redirect_to_signup(realm)
|
|
|
|
if auth_failed_reason:
|
|
backend.logger.info(auth_failed_reason)
|
|
return redirect_to_login(realm)
|
|
|
|
# Structurally, all the cases where we don't have an authenticated
|
|
# email for the user should be handled above; this assertion helps
|
|
# prevent any violations of that contract from resulting in a user
|
|
# being incorrectly authenticated.
|
|
assert return_data.get("valid_attestation") is True
|
|
|
|
strategy = backend.strategy
|
|
full_name_validated = backend.full_name_validated
|
|
email_address = return_data["validated_email"]
|
|
full_name = return_data["full_name"]
|
|
redirect_to = strategy.session_get("next")
|
|
multiuse_object_key = strategy.session_get("multiuse_object_key", "")
|
|
|
|
mobile_flow_otp = strategy.session_get("mobile_flow_otp")
|
|
desktop_flow_otp = strategy.session_get("desktop_flow_otp")
|
|
validate_otp_params(mobile_flow_otp, desktop_flow_otp)
|
|
|
|
if user_profile is None or user_profile.is_mirror_dummy:
|
|
is_signup = strategy.session_get("is_signup") == "1" or backend.should_auto_signup()
|
|
else:
|
|
is_signup = False
|
|
|
|
extra_attrs = return_data.get("extra_attrs", {})
|
|
social_auth_sync_new_user_info = social_auth_sync_user_attributes(
|
|
realm, user_profile, extra_attrs, backend
|
|
)
|
|
|
|
if user_profile:
|
|
# This call to authenticate() is just to get to invoke the custom_auth_decorator logic.
|
|
# Social auth backends don't work via authenticate() in the same way as normal backends,
|
|
# so we can't just wrap their authenticate() methods. But the decorator is applied on
|
|
# ZulipDummyBackend.authenticate(), so we can invoke it here to trigger the custom logic.
|
|
#
|
|
# Note: We're only doing in the case where we already have a user_profile, meaning the
|
|
# account already exists and the user is just logging in. The new account registration case
|
|
# is handled in the registration codepath.
|
|
validated_user_profile = authenticate(
|
|
request=strategy.request,
|
|
username=user_profile.delivery_email,
|
|
realm=realm,
|
|
use_dummy_backend=True,
|
|
)
|
|
if validated_user_profile is None or validated_user_profile != user_profile:
|
|
# Log this as as a failure to authenticate via the social backend, since that's
|
|
# the correct way to think about this. ZulipDummyBackend is just an implementation
|
|
# tool, not an actual backend a user could be authenticating through.
|
|
log_auth_attempt(
|
|
backend.logger,
|
|
strategy.request,
|
|
realm,
|
|
username=email_address,
|
|
succeeded=False,
|
|
return_data={},
|
|
)
|
|
return redirect_to_login(realm)
|
|
|
|
# At this point, we have now confirmed that the user has
|
|
# demonstrated control over the target email address.
|
|
#
|
|
# The next step is to call login_or_register_remote_user, but
|
|
# there are two code paths here because of an optimization to save
|
|
# a redirect on mobile and desktop.
|
|
|
|
# Authentication failures happen on the external provider's side, so we don't get to log those,
|
|
# but we should log the successes at least.
|
|
log_auth_attempt(
|
|
backend.logger,
|
|
strategy.request,
|
|
realm,
|
|
username=email_address,
|
|
succeeded=True,
|
|
return_data={},
|
|
)
|
|
|
|
data_dict = ExternalAuthDataDict(
|
|
subdomain=realm.subdomain,
|
|
is_signup=is_signup,
|
|
redirect_to=redirect_to,
|
|
multiuse_object_key=multiuse_object_key,
|
|
full_name_validated=full_name_validated,
|
|
mobile_flow_otp=mobile_flow_otp,
|
|
desktop_flow_otp=desktop_flow_otp,
|
|
params_to_store_in_authenticated_session=backend.get_params_to_store_in_authenticated_session(),
|
|
)
|
|
if user_profile is None:
|
|
data_dict.update(
|
|
dict(
|
|
full_name=full_name,
|
|
email=email_address,
|
|
)
|
|
)
|
|
if social_auth_sync_new_user_info is not None:
|
|
data_dict.update(
|
|
dict(
|
|
role=social_auth_sync_new_user_info.role,
|
|
group_memberships_sync_map=social_auth_sync_new_user_info.group_memberships_sync_map,
|
|
)
|
|
)
|
|
|
|
result = ExternalAuthResult(user_profile=user_profile, data_dict=data_dict)
|
|
|
|
if mobile_flow_otp or desktop_flow_otp:
|
|
if user_profile is not None and not user_profile.is_mirror_dummy:
|
|
# For mobile and desktop app authentication, login_or_register_remote_user
|
|
# will redirect to a special zulip:// URL that is handled by
|
|
# the app after a successful authentication; so we can
|
|
# redirect directly from here, saving a round trip over what
|
|
# we need to do to create session cookies on the right domain
|
|
# in the web login flow (below).
|
|
return login_or_register_remote_user(strategy.request, result)
|
|
else:
|
|
# The user needs to register, so we need to go the realm's
|
|
# subdomain for that.
|
|
pass
|
|
|
|
# If this authentication code were executing on
|
|
# subdomain.zulip.example.com, we would just call
|
|
# login_or_register_remote_user as in the mobile code path.
|
|
# However, because third-party SSO providers generally don't allow
|
|
# wildcard addresses in their redirect URLs, for multi-realm
|
|
# servers, we will have just completed authentication on e.g.
|
|
# auth.zulip.example.com (depending on
|
|
# settings.SOCIAL_AUTH_SUBDOMAIN), which cannot store cookies on
|
|
# the subdomain.zulip.example.com domain. So instead we serve a
|
|
# redirect (encoding the authentication result data in a
|
|
# cryptographically signed token) to a route on
|
|
# subdomain.zulip.example.com that will verify the signature and
|
|
# then call login_or_register_remote_user.
|
|
return redirect_and_log_into_subdomain(result)
|
|
|
|
|
|
class SocialAuthMixin(ZulipAuthMixin, ExternalAuthMethod, BaseAuth):
|
|
# Whether we expect that the full_name value obtained by the
|
|
# social backend is definitely how the user should be referred to
|
|
# in Zulip, which in turn determines whether we should always show
|
|
# a registration form in the event with a default value of the
|
|
# user's name when using this social backend so they can change
|
|
# it. For social backends like SAML that are expected to be a
|
|
# central database, this should be True; for backends like GitHub
|
|
# where the user might not have a name set or have it set to
|
|
# something other than the name they will prefer to use in Zulip,
|
|
# it should be False.
|
|
full_name_validated = False
|
|
|
|
standard_relay_params = [*settings.SOCIAL_AUTH_FIELDS_STORED_IN_SESSION, "next"]
|
|
|
|
@override
|
|
def auth_complete(self, *args: Any, **kwargs: Any) -> HttpResponse | None:
|
|
"""This is a small wrapper around the core `auth_complete` method of
|
|
python-social-auth, designed primarily to prevent 500s for
|
|
exceptions in the social auth code from situations that are
|
|
really user errors. Returning `None` from this function will
|
|
redirect the browser to the login page.
|
|
"""
|
|
try:
|
|
# Call the auth_complete method of social_core.backends.oauth.BaseOAuth2
|
|
return super().auth_complete(*args, **kwargs)
|
|
except (AuthFailed, HTTPError) as e:
|
|
# When a user's social authentication fails (e.g. because
|
|
# they did something funny with reloading in the middle of
|
|
# the flow or the IdP is unreliable and returns a bad http response),
|
|
# don't throw a 500, just send them back to the
|
|
# login page and record the event at the info log level.
|
|
self.logger.info("%s: %s", type(e).__name__, e)
|
|
return None
|
|
except SocialAuthBaseException as e:
|
|
# Other python-social-auth exceptions are likely
|
|
# interesting enough that we should log a warning.
|
|
self.logger.warning("%s", e)
|
|
return None
|
|
|
|
def should_auto_signup(self) -> bool:
|
|
return False
|
|
|
|
def get_params_to_store_in_authenticated_session(self) -> dict[str, str]:
|
|
"""
|
|
Specifies a dict of keys:values to be saved in the user's session
|
|
after successfully authenticating.
|
|
"""
|
|
return {"social_auth_backend": self.name}
|
|
|
|
@classmethod
|
|
@override
|
|
def dict_representation(cls, realm: Realm | None = None) -> list[ExternalAuthMethodDictT]:
|
|
return [
|
|
dict(
|
|
name=cls.name,
|
|
display_name=cls.auth_backend_name,
|
|
display_icon=cls.display_icon,
|
|
login_url=reverse("login-social", args=(cls.name,)),
|
|
signup_url=reverse("signup-social", args=(cls.name,)),
|
|
)
|
|
]
|
|
|
|
|
|
@external_auth_method
|
|
class GitHubAuthBackend(SocialAuthMixin, GithubOAuth2):
|
|
name = "github"
|
|
auth_backend_name = "GitHub"
|
|
sort_order = 100
|
|
display_icon = staticfiles_storage.url("images/authentication_backends/github-icon.png")
|
|
|
|
def get_all_associated_email_objects(self, *args: Any, **kwargs: Any) -> list[dict[str, Any]]:
|
|
access_token = kwargs["response"]["access_token"]
|
|
try:
|
|
emails = self._user_data(access_token, "/emails")
|
|
except (HTTPError, json.JSONDecodeError): # nocoverage
|
|
# We don't really need an explicit test for this code
|
|
# path, since the outcome will be the same as any other
|
|
# case without any verified emails
|
|
emails = []
|
|
return emails
|
|
|
|
def get_unverified_emails(self, realm: Realm, *args: Any, **kwargs: Any) -> list[str]:
|
|
return [
|
|
email_obj["email"]
|
|
for email_obj in self.get_usable_email_objects(realm, *args, **kwargs)
|
|
if not email_obj.get("verified")
|
|
]
|
|
|
|
def get_verified_emails(self, realm: Realm, *args: Any, **kwargs: Any) -> list[str]:
|
|
# We only let users log in using email addresses that are
|
|
# verified by GitHub, because the whole point is for the user
|
|
# to demonstrate that they control the target email address.
|
|
verified_emails: list[str] = []
|
|
for email_obj in [
|
|
obj
|
|
for obj in self.get_usable_email_objects(realm, *args, **kwargs)
|
|
if obj.get("verified")
|
|
]:
|
|
# social_associate_user_helper assumes that the first email in
|
|
# verified_emails is primary.
|
|
if email_obj.get("primary"):
|
|
verified_emails.insert(0, email_obj["email"])
|
|
else:
|
|
verified_emails.append(email_obj["email"])
|
|
|
|
return verified_emails
|
|
|
|
def get_usable_email_objects(
|
|
self, realm: Realm, *args: Any, **kwargs: Any
|
|
) -> list[dict[str, Any]]:
|
|
# We disallow creation of new accounts with
|
|
# @noreply.github.com/@users.noreply.github.com email
|
|
# addresses, because structurally, we only want to allow email
|
|
# addresses that can receive emails, and those cannot.
|
|
|
|
# However, if an account with this address already exists in
|
|
# the realm (which could happen e.g. as a result of data
|
|
# import from another chat tool), we will allow signing in to
|
|
# it.
|
|
email_objs = self.get_all_associated_email_objects(*args, **kwargs)
|
|
return [
|
|
email
|
|
for email in email_objs
|
|
if (
|
|
not email["email"].endswith("@users.noreply.github.com")
|
|
or common_get_active_user(email["email"], realm) is not None
|
|
)
|
|
]
|
|
|
|
@override
|
|
def user_data(self, access_token: str, *args: Any, **kwargs: Any) -> dict[str, str]:
|
|
"""This patched user_data function lets us combine together the 3
|
|
social auth backends into a single Zulip backend for GitHub OAuth2"""
|
|
team_id = settings.SOCIAL_AUTH_GITHUB_TEAM_ID
|
|
org_name = settings.SOCIAL_AUTH_GITHUB_ORG_NAME
|
|
|
|
if team_id is None and org_name is None:
|
|
# I believe this can't raise AuthFailed, so we don't try to catch it here.
|
|
return super().user_data(
|
|
access_token,
|
|
*args,
|
|
**kwargs,
|
|
)
|
|
elif team_id is not None:
|
|
team_backend = GithubTeamOAuth2(self.strategy, self.redirect_uri)
|
|
try:
|
|
return team_backend.user_data(access_token, *args, **kwargs)
|
|
except AuthFailed:
|
|
return dict(auth_failed_reason="GitHub user is not member of required team")
|
|
elif org_name is not None:
|
|
organization_backend = GithubOrganizationOAuth2(self.strategy, self.redirect_uri)
|
|
try:
|
|
return organization_backend.user_data(access_token, *args, **kwargs)
|
|
except AuthFailed:
|
|
return dict(auth_failed_reason="GitHub user is not member of required organization")
|
|
|
|
raise AssertionError("Invalid configuration")
|
|
|
|
|
|
@external_auth_method
|
|
class AzureADAuthBackend(SocialAuthMixin, AzureADOAuth2):
|
|
# The upstream implementation uses the outdated /oauth2/authorize
|
|
# API (instead of the v2.0 API), which doesn't allow us to authenticate
|
|
# users with just a personal Microsoft account. v2.0 API is required.
|
|
# This requires us to override the default URLs to use it as well
|
|
# as adjust the requested scopes, to match this new API.
|
|
AUTHORIZATION_URL = "{base_url}/oauth2/v2.0/authorize"
|
|
ACCESS_TOKEN_URL = "{base_url}/oauth2/v2.0/token"
|
|
DEFAULT_SCOPE = ["User.Read profile openid email"]
|
|
|
|
sort_order = 50
|
|
name = "azuread-oauth2"
|
|
auth_backend_name = "AzureAD"
|
|
display_icon = staticfiles_storage.url("images/authentication_backends/azuread-icon.png")
|
|
|
|
available_for_cloud_plans = [
|
|
Realm.PLAN_TYPE_STANDARD,
|
|
Realm.PLAN_TYPE_STANDARD_FREE,
|
|
Realm.PLAN_TYPE_PLUS,
|
|
]
|
|
|
|
|
|
@external_auth_method
|
|
class GitLabAuthBackend(SocialAuthMixin, GitLabOAuth2):
|
|
sort_order = 75
|
|
name = "gitlab"
|
|
auth_backend_name = "GitLab"
|
|
display_icon = staticfiles_storage.url("images/authentication_backends/gitlab-icon.png")
|
|
|
|
# Note: GitLab as of early 2020 supports having multiple email
|
|
# addresses connected with a GitLab account, and we could access
|
|
# those emails, but its APIs don't indicate which of those email
|
|
# addresses were verified, so we cannot use them for
|
|
# authentication like we do for the GitHub integration. Instead,
|
|
# we just use the primary email address, which is always verified.
|
|
# (No code is required to do so, as that's the default behavior).
|
|
|
|
|
|
@external_auth_method
|
|
class GoogleAuthBackend(SocialAuthMixin, GoogleOAuth2):
|
|
sort_order = 150
|
|
auth_backend_name = "Google"
|
|
name = "google"
|
|
display_icon = staticfiles_storage.url("images/authentication_backends/googl_e-icon.png")
|
|
|
|
def get_verified_emails(self, *args: Any, **kwargs: Any) -> list[str]:
|
|
verified_emails: list[str] = []
|
|
details = kwargs["response"]
|
|
email_verified = details.get("email_verified")
|
|
if email_verified:
|
|
verified_emails.append(details["email"])
|
|
return verified_emails
|
|
|
|
|
|
@external_auth_method
|
|
class AppleAuthBackend(SocialAuthMixin, AppleIdAuth):
|
|
"""
|
|
Authentication backend for "Sign in with Apple". This supports two flows:
|
|
1. The web flow, usable in a browser, like our other social auth methods.
|
|
It is a slightly modified Oauth2 authorization flow, where the response
|
|
returning the access_token also contains a JWT id_token containing the user's
|
|
identity, signed with Apple's private keys.
|
|
https://developer.apple.com/documentation/sign_in_with_apple/tokenresponse
|
|
2. The native flow, intended for users on an Apple device. In the native flow,
|
|
the device handles authentication of the user with Apple's servers and ends up
|
|
with the JWT id_token (like in the web flow). The client-side details aren't
|
|
relevant to us; the app should simply send the id_token as a param to the
|
|
/complete/apple/ endpoint, together with native_flow=true and any other
|
|
appropriate params, such as mobile_flow_otp.
|
|
"""
|
|
|
|
sort_order = 10
|
|
name = "apple"
|
|
auth_backend_name = "Apple"
|
|
display_icon = staticfiles_storage.url("images/authentication_backends/apple-icon.png")
|
|
|
|
# Apple only sends `name` in its response the first time a user
|
|
# tries to sign up, so we won't have it in consecutive attempts.
|
|
# But if Apple does send us the user's name, it will be validated,
|
|
# so it's appropriate to set full_name_validated here.
|
|
full_name_validated = True
|
|
REDIS_EXPIRATION_SECONDS = 60 * 10
|
|
|
|
SCOPE_SEPARATOR = "%20" # https://github.com/python-social-auth/social-core/issues/470
|
|
|
|
@classmethod
|
|
def check_config(cls) -> bool:
|
|
obligatory_apple_settings_list = [
|
|
settings.SOCIAL_AUTH_APPLE_TEAM,
|
|
settings.SOCIAL_AUTH_APPLE_SERVICES_ID,
|
|
settings.SOCIAL_AUTH_APPLE_KEY,
|
|
settings.SOCIAL_AUTH_APPLE_SECRET,
|
|
]
|
|
if any(not setting for setting in obligatory_apple_settings_list):
|
|
return False
|
|
|
|
return True
|
|
|
|
def is_native_flow(self) -> bool:
|
|
return self.strategy.request_data().get("native_flow", False)
|
|
|
|
# This method replaces a method from python-social-auth; it is adapted to store
|
|
# the state_token data in Redis.
|
|
@override
|
|
def get_or_create_state(self) -> str:
|
|
"""Creates the Oauth2 state parameter in first step of the flow,
|
|
before redirecting the user to the IdP (aka Apple).
|
|
|
|
Apple will send the user back to us with a POST
|
|
request. Normally, we rely on being able to store certain
|
|
parameters in the user's session and use them after the
|
|
redirect. But because we've configured our session cookies to
|
|
use the Django default of in SameSite Lax mode, the browser
|
|
won't send the session cookies to our server in delivering the
|
|
POST request coming from Apple.
|
|
|
|
To work around this, we replace python-social-auth's default
|
|
session-based storage with storing the parameters in Redis
|
|
under a random token derived from the state. That will allow
|
|
us to validate the state and retrieve the params after the
|
|
redirect - by querying Redis for the key derived from the
|
|
state sent in the POST redirect.
|
|
"""
|
|
request_data = self.strategy.request_data().dict()
|
|
data_to_store = {
|
|
key: request_data[key] for key in self.standard_relay_params if key in request_data
|
|
}
|
|
|
|
# Generate a random string of 32 alphanumeric characters.
|
|
state = self.state_token()
|
|
put_dict_in_redis(
|
|
redis_client,
|
|
"apple_auth_{token}",
|
|
data_to_store,
|
|
self.REDIS_EXPIRATION_SECONDS,
|
|
token=state,
|
|
)
|
|
return state
|
|
|
|
@override
|
|
def validate_state(self) -> str | None:
|
|
"""
|
|
This method replaces a method from python-social-auth; it is
|
|
adapted to retrieve the data stored in Redis, save it in
|
|
the session so that it can be accessed by the social pipeline.
|
|
"""
|
|
request_state = self.get_request_state()
|
|
|
|
if not request_state:
|
|
self.logger.info("Sign in with Apple failed: missing state parameter.")
|
|
raise AuthMissingParameter(self, "state")
|
|
|
|
formatted_request_state = "apple_auth_" + request_state
|
|
redis_data = get_dict_from_redis(
|
|
redis_client, "apple_auth_{token}", formatted_request_state
|
|
)
|
|
if redis_data is None:
|
|
self.logger.info("Sign in with Apple failed: bad state token.")
|
|
raise AuthStateForbidden(self)
|
|
|
|
for param, value in redis_data.items():
|
|
if param in self.standard_relay_params:
|
|
self.strategy.session_set(param, value)
|
|
return request_state
|
|
|
|
@override
|
|
def get_user_details(self, response: dict[str, Any]) -> dict[str, Any]:
|
|
"""
|
|
Overridden to correctly grab the user's name from the request params,
|
|
as current upstream code expects it in the id_token and Apple changed
|
|
the API.
|
|
Taken from https://github.com/python-social-auth/social-core/pull/483
|
|
TODO: Remove this when the PR is merged.
|
|
"""
|
|
name = response.get("name") or {}
|
|
name = json.loads(self.data.get("user", "{}")).get("name", {})
|
|
fullname, first_name, last_name = self.get_user_names(
|
|
fullname="", first_name=name.get("firstName", ""), last_name=name.get("lastName", "")
|
|
)
|
|
email = response.get("email", "")
|
|
# prevent updating User with empty strings
|
|
user_details = {
|
|
"fullname": fullname or None,
|
|
"first_name": first_name or None,
|
|
"last_name": last_name or None,
|
|
"email": email,
|
|
}
|
|
user_details["username"] = email
|
|
|
|
return user_details
|
|
|
|
@override
|
|
def auth_complete(self, *args: Any, **kwargs: Any) -> HttpResponse | None:
|
|
if not self.is_native_flow():
|
|
# The default implementation in python-social-auth is the browser flow.
|
|
return super().auth_complete(*args, **kwargs)
|
|
|
|
# We handle the Apple's native flow on our own. In this flow,
|
|
# before contacting the server, the client obtains an id_token
|
|
# from Apple directly, and then sends that to /complete/apple/
|
|
# (the endpoint handled by this function), together with any
|
|
# other desired parameters from self.standard_relay_params.
|
|
#
|
|
# What we'd like to do with the payload is just pass it into
|
|
# the common code path for the web flow. In the web flow,
|
|
# before sending a request to Apple, python-social-auth sets
|
|
# various values about the intended authentication in the
|
|
# session, before the redirect.
|
|
#
|
|
# Thus, we need to set those session variables here, before
|
|
# processing the id_token we received using the common do_auth.
|
|
request_data = self.strategy.request_data()
|
|
if "id_token" not in request_data:
|
|
raise JsonableError(_("Missing id_token parameter"))
|
|
|
|
for param in self.standard_relay_params:
|
|
self.strategy.session_set(param, request_data.get(param))
|
|
|
|
# We should get the subdomain from the hostname of the request.
|
|
self.strategy.session_set("subdomain", get_subdomain(self.strategy.request))
|
|
|
|
try:
|
|
# Things are now ready to be handled by the superclass code. It will
|
|
# validate the id_token and push appropriate user data to the social pipeline.
|
|
result = self.do_auth(request_data["id_token"], *args, **kwargs)
|
|
return result
|
|
except (AuthFailed, AuthCanceled) as e:
|
|
# AuthFailed is a general "failure" exception from
|
|
# python-social-auth that we should convert to None return
|
|
# value here to avoid getting tracebacks.
|
|
#
|
|
# AuthCanceled is raised in the Apple backend
|
|
# implementation in python-social-auth in certain cases,
|
|
# though AuthFailed would have been more correct.
|
|
#
|
|
# We have an open PR to python-social-auth to clean this up.
|
|
self.logger.info("/complete/apple/: %s", str(e))
|
|
return None
|
|
|
|
|
|
class ZulipSAMLIdentityProvider(SAMLIdentityProvider):
|
|
@override
|
|
def get_user_details(self, attributes: dict[str, Any]) -> dict[str, Any]:
|
|
"""
|
|
Overridden to support plumbing of additional Attributes
|
|
from the SAMLResponse.
|
|
"""
|
|
result = super().get_user_details(attributes)
|
|
|
|
extra_attr_names = self.conf.get("extra_attrs", [])
|
|
result["extra_attrs"] = {}
|
|
|
|
if (groups_list := attributes.get("zulip_groups")) is not None:
|
|
result["extra_attrs"]["zulip_groups"] = groups_list
|
|
|
|
for extra_attr_name in extra_attr_names:
|
|
result["extra_attrs"][extra_attr_name] = self.get_attr(
|
|
attributes=attributes, conf_key=None, default_attribute=extra_attr_name
|
|
)
|
|
|
|
return result
|
|
|
|
|
|
class SAMLDocument:
|
|
"""
|
|
Parent class, subclassed by SAMLRequest and SAMLResponse,
|
|
for wrapping the fiddly logic of handling these SAML XML documents.
|
|
"""
|
|
|
|
SAML_PARSING_EXCEPTIONS = (
|
|
OneLogin_Saml2_Error,
|
|
OneLogin_Saml2_ValidationError,
|
|
binascii.Error,
|
|
XMLSyntaxError,
|
|
)
|
|
|
|
def __init__(self, encoded_saml_message: str, backend: "SAMLAuthBackend") -> None:
|
|
"""
|
|
encoded_saml_message is the base64-encoded XML string that's received
|
|
in the SAMLRequest or SAMLResponse params. The underlying XML
|
|
can be either deflated or not, both cases should be handled fine by the class.
|
|
|
|
backend is an instance of the SAMLAuthBackend class, which is handling
|
|
the HTTP request in which the SAMLRequest or SAMLResponse was delivered.
|
|
"""
|
|
self.encoded_saml_message = encoded_saml_message
|
|
self.backend = backend
|
|
|
|
self._decoded_saml_message: str | None = None
|
|
|
|
@property
|
|
def logger(self) -> logging.Logger:
|
|
return self.backend.logger
|
|
|
|
@property
|
|
def decoded_saml_message(self) -> str:
|
|
"""
|
|
Returns the decoded SAMLRequest/SAMLResponse.
|
|
"""
|
|
if self._decoded_saml_message is None:
|
|
# This logic is taken from how
|
|
# python3-saml handles decoding received SAMLRequest
|
|
# and SAMLResponse params.
|
|
self._decoded_saml_message = onelogin_saml2_compat.to_string(
|
|
OneLogin_Saml2_Utils.decode_base64_and_inflate(
|
|
self.encoded_saml_message, ignore_zip=True
|
|
)
|
|
)
|
|
|
|
return self._decoded_saml_message
|
|
|
|
def document_type(self) -> str:
|
|
"""
|
|
Returns whether the instance is a SAMLRequest or SAMLResponse.
|
|
"""
|
|
return type(self).__name__
|
|
|
|
def get_issuing_idp(self) -> str | None:
|
|
"""
|
|
Given a SAMLResponse or SAMLRequest, returns which of the configured IdPs
|
|
is declared as the issuer.
|
|
This value MUST NOT be trusted as the true issuer!
|
|
The signatures are not validated, so it can be tampered with by the user.
|
|
That's not a problem for this function,
|
|
and true validation happens later in the underlying libraries, but it's important
|
|
to note this detail. The purpose of this function is merely as a helper to figure out which
|
|
of the configured IdPs' information to use for parsing and validating the request.
|
|
"""
|
|
|
|
issuers = self.get_issuers()
|
|
|
|
for idp_name, idp_config in settings.SOCIAL_AUTH_SAML_ENABLED_IDPS.items():
|
|
if idp_config["entity_id"] in issuers:
|
|
return idp_name
|
|
|
|
return None
|
|
|
|
@abstractmethod
|
|
def get_issuers(self) -> list[str]:
|
|
"""
|
|
Returns a list of the issuers of the SAML document.
|
|
"""
|
|
|
|
|
|
class SAMLRequest(SAMLDocument):
|
|
@override
|
|
def get_issuers(self) -> list[str]:
|
|
config = self.backend.generate_saml_config()
|
|
saml_settings = OneLogin_Saml2_Settings(config, sp_validation_only=True)
|
|
|
|
try:
|
|
# The only valid SAMLRequest we can receive is a LogoutRequest.
|
|
logout_request_xml = OneLogin_Saml2_Logout_Request(
|
|
saml_settings, self.encoded_saml_message
|
|
).get_xml()
|
|
issuers = [OneLogin_Saml2_Logout_Request.get_issuer(logout_request_xml)]
|
|
return issuers
|
|
except self.SAML_PARSING_EXCEPTIONS as e:
|
|
self.logger.error("Error parsing SAMLRequest: %s", str(e))
|
|
return []
|
|
|
|
|
|
class SAMLResponse(SAMLDocument):
|
|
@override
|
|
def get_issuers(self) -> list[str]:
|
|
config = self.backend.generate_saml_config()
|
|
saml_settings = OneLogin_Saml2_Settings(config, sp_validation_only=True)
|
|
|
|
try:
|
|
if not self.is_logout_response():
|
|
resp = OneLogin_Saml2_Response(
|
|
settings=saml_settings, response=self.encoded_saml_message
|
|
)
|
|
return resp.get_issuers()
|
|
else:
|
|
logout_response = OneLogin_Saml2_Logout_Response(
|
|
settings=saml_settings, response=self.encoded_saml_message
|
|
)
|
|
return logout_response.get_issuer()
|
|
except self.SAML_PARSING_EXCEPTIONS as e:
|
|
self.logger.error("Error parsing SAMLResponse: %s", str(e))
|
|
return []
|
|
|
|
def get_session_index(self) -> str | None:
|
|
"""
|
|
Returns the SessionIndex from the SAMLResponse.
|
|
"""
|
|
config = self.backend.generate_saml_config()
|
|
saml_settings = OneLogin_Saml2_Settings(config, sp_validation_only=True)
|
|
|
|
try:
|
|
resp = OneLogin_Saml2_Response(
|
|
settings=saml_settings, response=self.encoded_saml_message
|
|
)
|
|
return resp.get_session_index()
|
|
except self.SAML_PARSING_EXCEPTIONS as e:
|
|
self.logger.error("Error parsing SAMLResponse: %s", str(e))
|
|
return None
|
|
|
|
def is_logout_response(self) -> bool:
|
|
"""
|
|
Checks whether the SAMLResponse is a LogoutResponse based on some
|
|
basic XML parsing.
|
|
"""
|
|
|
|
try:
|
|
parsed_xml = OneLogin_Saml2_XML.to_etree(self.decoded_saml_message)
|
|
return bool(OneLogin_Saml2_XML.query(parsed_xml, "/samlp:LogoutResponse"))
|
|
except self.SAML_PARSING_EXCEPTIONS:
|
|
return False
|
|
|
|
|
|
@external_auth_method
|
|
class SAMLAuthBackend(SocialAuthMixin, SAMLAuth):
|
|
auth_backend_name = "SAML"
|
|
REDIS_EXPIRATION_SECONDS = 60 * 15
|
|
|
|
name = "saml"
|
|
# Organization which go through the trouble of setting up SAML are most likely
|
|
# to have it as their main authentication method, so it seems appropriate to have
|
|
# SAML buttons at the top.
|
|
sort_order = 9999
|
|
# There's no common default logo for SAML authentication.
|
|
display_icon = None
|
|
|
|
# The full_name provided by the IdP is very likely the standard
|
|
# employee directory name for the user, and thus what they and
|
|
# their organization want to use in Zulip. So don't unnecessarily
|
|
# provide a registration flow prompt for them to set their name.
|
|
full_name_validated = True
|
|
|
|
available_for_cloud_plans = [Realm.PLAN_TYPE_PLUS]
|
|
|
|
def __init__(self, *args: Any, **kwargs: Any) -> None:
|
|
if settings.SAML_REQUIRE_LIMIT_TO_SUBDOMAINS:
|
|
idps_without_limit_to_subdomains = [
|
|
idp_name
|
|
for idp_name, idp_dict in settings.SOCIAL_AUTH_SAML_ENABLED_IDPS.items()
|
|
if "limit_to_subdomains" not in idp_dict
|
|
]
|
|
if idps_without_limit_to_subdomains:
|
|
self.logger.error(
|
|
"SAML_REQUIRE_LIMIT_TO_SUBDOMAINS is enabled and the following IdPs don't have"
|
|
" limit_to_subdomains specified and will be ignored: %r",
|
|
idps_without_limit_to_subdomains,
|
|
)
|
|
for idp_name in idps_without_limit_to_subdomains:
|
|
del settings.SOCIAL_AUTH_SAML_ENABLED_IDPS[idp_name]
|
|
super().__init__(*args, **kwargs)
|
|
|
|
@override
|
|
def get_idp(self, idp_name: str) -> ZulipSAMLIdentityProvider:
|
|
"""Given the name of an IdP, get a SAMLIdentityProvider instance.
|
|
Forked to use our subclass of SAMLIdentityProvider for more flexibility."""
|
|
idp_config = self.setting("ENABLED_IDPS")[idp_name]
|
|
return ZulipSAMLIdentityProvider(idp_name, **idp_config)
|
|
|
|
@override
|
|
def auth_url(self) -> str:
|
|
"""Get the URL to which we must redirect in order to
|
|
authenticate the user. Overriding the original SAMLAuth.auth_url.
|
|
Runs when someone accesses the /login/saml/ endpoint."""
|
|
try:
|
|
idp_name = self.strategy.request_data()["idp"]
|
|
auth = self._create_saml_auth(idp=self.get_idp(idp_name))
|
|
except KeyError as e:
|
|
# If the above raise KeyError, it means invalid or no idp was specified,
|
|
# we should log that and redirect to the login page.
|
|
self.logger.info("/login/saml/ : Bad idp param: KeyError: %s.", str(e))
|
|
return reverse("login_page", kwargs={"template_name": "zerver/login.html"})
|
|
|
|
# This where we change things. We need to pass some params
|
|
# (`mobile_flow_otp`, `next`, etc.) through RelayState, which
|
|
# then the IdP will pass back to us so we can read those
|
|
# parameters in the final part of the authentication flow, at
|
|
# the /complete/saml/ endpoint.
|
|
#
|
|
# To protect against network eavesdropping of these
|
|
# parameters, we send just a random token to the IdP in
|
|
# RelayState, which is used as a key into our Redis data store
|
|
# for fetching the actual parameters after the IdP has
|
|
# returned a successful authentication.
|
|
params_to_relay = self.standard_relay_params
|
|
request_data = self.strategy.request_data().dict()
|
|
data_to_relay = {key: request_data[key] for key in params_to_relay if key in request_data}
|
|
relay_state = orjson.dumps({"state_token": self.put_data_in_redis(data_to_relay)}).decode()
|
|
|
|
return auth.login(return_to=relay_state)
|
|
|
|
@classmethod
|
|
def put_data_in_redis(cls, data_to_relay: dict[str, Any]) -> str:
|
|
return put_dict_in_redis(
|
|
redis_client,
|
|
"saml_token_{token}",
|
|
data_to_store=data_to_relay,
|
|
expiration_seconds=cls.REDIS_EXPIRATION_SECONDS,
|
|
)
|
|
|
|
@classmethod
|
|
def get_data_from_redis(cls, key: str) -> dict[str, Any] | None:
|
|
data = None
|
|
if key.startswith("saml_token_"):
|
|
# Safety if statement, to not allow someone to poke around arbitrary Redis keys here.
|
|
data = get_dict_from_redis(redis_client, "saml_token_{token}", key)
|
|
|
|
return data
|
|
|
|
def get_relayed_params(self) -> dict[str, Any]:
|
|
request_data = self.strategy.request_data()
|
|
if "RelayState" not in request_data:
|
|
return {}
|
|
|
|
relay_state = request_data["RelayState"]
|
|
try:
|
|
data = orjson.loads(relay_state)
|
|
if "state_token" in data:
|
|
# SP-initiated sign in. We stored relevant information in the first
|
|
# step of the flow
|
|
return self.get_data_from_redis(data["state_token"]) or {}
|
|
else:
|
|
# IdP-initiated sign in. Right now we only support transporting subdomain through json in
|
|
# RelayState, but this format is nice in that it allows easy extensibility here.
|
|
return {"subdomain": data.get("subdomain")}
|
|
except orjson.JSONDecodeError:
|
|
return {}
|
|
|
|
def choose_subdomain(self, relayed_params: dict[str, Any]) -> str | None:
|
|
subdomain = relayed_params.get("subdomain")
|
|
if subdomain is not None:
|
|
return subdomain
|
|
|
|
# If not specified otherwise, the intended subdomain for this
|
|
# authentication attempt is the subdomain of the request.
|
|
request_subdomain = get_subdomain(self.strategy.request)
|
|
try:
|
|
# We only want to do a basic sanity-check here for whether
|
|
# this subdomain has a realm one could try to authenticate
|
|
# to. True validation of whether the realm is active, the
|
|
# IdP is appropriate for the subdomain, etc. happens
|
|
# elsewhere in the flow and we shouldn't duplicate such
|
|
# logic here.
|
|
get_realm(request_subdomain)
|
|
except Realm.DoesNotExist:
|
|
return None
|
|
else:
|
|
return request_subdomain
|
|
|
|
@override
|
|
def _check_entitlements(
|
|
self, idp: SAMLIdentityProvider, attributes: dict[str, list[str]]
|
|
) -> None:
|
|
"""
|
|
Below is the docstring from the social_core SAML backend.
|
|
|
|
Additional verification of a SAML response before
|
|
authenticating the user.
|
|
|
|
Subclasses can override this method if they need custom
|
|
validation code, such as requiring the presence of an
|
|
eduPersonEntitlement.
|
|
|
|
raise social_core.exceptions.AuthForbidden if the user should not
|
|
be authenticated, or do nothing to allow the login pipeline to
|
|
continue.
|
|
"""
|
|
org_membership_attribute = idp.conf.get("attr_org_membership", None)
|
|
if org_membership_attribute is None:
|
|
return
|
|
|
|
subdomain = self.strategy.session_get("subdomain")
|
|
entitlements: str | list[str] = attributes.get(org_membership_attribute, [])
|
|
if isinstance(entitlements, str): # nocoverage
|
|
# This shouldn't happen as we'd always expect a list from this attribute even
|
|
# if it only has one element, but it's safer to have this defensive code.
|
|
entitlements = [
|
|
entitlements,
|
|
]
|
|
assert isinstance(entitlements, list)
|
|
|
|
if is_subdomain_in_allowed_subdomains_list(subdomain, entitlements):
|
|
return
|
|
|
|
error_msg = (
|
|
f"SAML user from IdP {idp.name} rejected due to missing entitlement for subdomain"
|
|
f" '{subdomain}'. User entitlements: {entitlements}."
|
|
)
|
|
raise AuthFailed(self, error_msg)
|
|
|
|
def process_logout_request(self, subdomain: str, idp_name: str) -> HttpResponse | None:
|
|
"""
|
|
We don't use process_logout, because we need to customize
|
|
the way of revoking sessions and introduce NameID validation.
|
|
|
|
The python-social-auth and python3-saml implementations expect a simple
|
|
callback function without arguments, to delete the session. We're not
|
|
happy with that for two reasons:
|
|
1. These implementations don't look at the NameID in the LogoutRequest, which
|
|
is not quite correct, as a LogoutRequest to log out user X can be delivered
|
|
through any means, and doesn't need a session to be valid.
|
|
E.g. a backchannel logout request sent by the IdP wouldn't have a session cookie.
|
|
Also, hypothetically, a LogoutRequest to log out user Y shouldn't log out user X, even if the
|
|
request is made with a session cookie belonging to user X.
|
|
2. We want to revoke all sessions for the user, not just the current session
|
|
of the request, so after validating the LogoutRequest, we need to identify
|
|
the user by the NameID, do some validation and then revoke all sessions.
|
|
|
|
TODO: This does not return a LogoutResponse in case of failure, like the spec requires.
|
|
https://github.com/zulip/zulip/issues/20076 is the related issue with more detail
|
|
on how to implement the desired behavior.
|
|
"""
|
|
idp = self.get_idp(idp_name)
|
|
auth = self._create_saml_auth(idp)
|
|
|
|
# We only want to accept signed LogoutResponses - or potentially anyone
|
|
# would be able to create a LogoutResponse to get an arbitrary user logged out.
|
|
patch_saml_auth_require_messages_signed(auth)
|
|
|
|
# This validates the LogoutRequest and prepares the response
|
|
# (the URL to which to redirect the client to convey the response to the IdP)
|
|
# but is a no-op otherwise because keep_local_session=True keeps it from
|
|
# doing anything else. We want to take care of revoking session on our own.
|
|
url = auth.process_slo(keep_local_session=True)
|
|
errors = auth.get_errors()
|
|
if errors:
|
|
self.logger.info("/complete/saml/: LogoutRequest failed: %s", errors)
|
|
return None
|
|
|
|
logout_request_xml = auth.get_last_request_xml()
|
|
name_id = OneLogin_Saml2_Logout_Request.get_nameid(logout_request_xml)
|
|
try:
|
|
validate_email(name_id)
|
|
except ValidationError:
|
|
self.logger.info(
|
|
"/complete/saml/: LogoutRequest failed: NameID is not a valid email address: %s",
|
|
name_id,
|
|
)
|
|
return None
|
|
|
|
return_data: dict[str, Any] = {}
|
|
|
|
realm = get_realm(subdomain)
|
|
user_profile = common_get_active_user(name_id, realm, return_data)
|
|
if user_profile is None:
|
|
self.logger.info(
|
|
"/complete/saml/: LogoutRequest failed: No user with email specified in NameID found in realm %s. return_data=%s",
|
|
realm.id,
|
|
return_data,
|
|
)
|
|
return None
|
|
|
|
self.logger.info(
|
|
"/complete/saml/: LogoutRequest triggered deletion of all session for user %s",
|
|
user_profile.id,
|
|
)
|
|
delete_user_sessions(user_profile)
|
|
do_regenerate_api_key(user_profile, user_profile)
|
|
|
|
return HttpResponseRedirect(url)
|
|
|
|
@override
|
|
def auth_complete(self, *args: Any, **kwargs: Any) -> HttpResponse | None:
|
|
"""
|
|
Additional ugly wrapping on top of auth_complete in SocialAuthMixin.
|
|
We handle two things for processing SAMLResponses here:
|
|
1. Working around bad RelayState or SAMLResponse parameters in the request.
|
|
Both parameters should be present if the user came to /complete/saml/ through
|
|
the IdP as intended. The errors can happen if someone simply types the endpoint into
|
|
their browsers, or generally tries messing with it in some ways.
|
|
|
|
2. The first part of our SAML authentication flow will encode important parameters
|
|
into the RelayState. We need to read them and set those values in the session,
|
|
and then change the RelayState param to the idp_name, because that's what
|
|
SAMLAuth.auth_complete() expects.
|
|
|
|
Additionally, this handles incoming LogoutRequests for IdP-initiated logout.
|
|
"""
|
|
|
|
encoded_saml_request = self.strategy.request_data().get("SAMLRequest")
|
|
encoded_saml_response = self.strategy.request_data().get("SAMLResponse")
|
|
if encoded_saml_response is None and encoded_saml_request is None:
|
|
self.logger.info("/complete/saml/: No SAMLResponse or SAMLRequest in request.")
|
|
return None
|
|
elif encoded_saml_request is not None:
|
|
saml_document: SAMLDocument = SAMLRequest(encoded_saml_request, self)
|
|
elif encoded_saml_response is not None:
|
|
saml_document = SAMLResponse(encoded_saml_response, self)
|
|
|
|
relayed_params = self.get_relayed_params()
|
|
|
|
subdomain = self.choose_subdomain(relayed_params)
|
|
if subdomain is None:
|
|
error_msg = (
|
|
"/complete/saml/: Can't figure out subdomain for this %s. relayed_params: %s"
|
|
)
|
|
self.logger.info(error_msg, saml_document.document_type(), relayed_params)
|
|
return None
|
|
|
|
idp_name = saml_document.get_issuing_idp()
|
|
if idp_name is None:
|
|
self.logger.info(
|
|
"/complete/saml/: No valid IdP as issuer of the %s.", saml_document.document_type()
|
|
)
|
|
return None
|
|
|
|
idp_valid = self.validate_idp_for_subdomain(idp_name, subdomain)
|
|
if not idp_valid:
|
|
error_msg = (
|
|
"/complete/saml/: Authentication request with IdP %s but this provider is not"
|
|
" enabled for this subdomain %s."
|
|
)
|
|
self.logger.info(error_msg, idp_name, subdomain)
|
|
return None
|
|
|
|
# We have to branch here to do different things depending on the kind
|
|
# of SAMLRequest/SAMLResponse we received. We do just basic heuristics here
|
|
# to choose the right branch, since it's not our intent to do proper validation now.
|
|
# We end up calling the appropriate process_*() function, which takes care of validation
|
|
# in the python3-saml library, ensuring it received the correct kind of XML document
|
|
# and finishes processing it.
|
|
# (1) We received a SAMLRequest - the only SAMLRequest we accept is a LogoutRequest,
|
|
# so we call process_logout_request().
|
|
# (2) We received a SAMLResponse and it looks like a LogoutResponse - we call
|
|
# process_logout_response()
|
|
# (3) We received a SAMLResponse that's not a LogoutResponse. We proceed to treat it
|
|
# as an authentication response. We don't do anything security-sensitive here, just some setup
|
|
# before calling the super().auth_complete() method, which is where the actual validation
|
|
# and authentication will happen.
|
|
#
|
|
# If for any reason, an XML document that doesn't match the expected type is passed
|
|
# to these *_process() functions, it will be rejected.
|
|
if isinstance(saml_document, SAMLRequest):
|
|
return self.process_logout_request(subdomain, idp_name)
|
|
elif isinstance(saml_document, SAMLResponse) and saml_document.is_logout_response():
|
|
return SAMLSPInitiatedLogout.process_logout_response(saml_document, idp_name)
|
|
|
|
# IMPORTANT: The saml_document has not yet been validated at this point. We are
|
|
# assuming it is to be treated as an authentication SAMLResponse, but it will only
|
|
# be validated in the super().auth_complete() call below - and code until then
|
|
# must not assume trust in the data.
|
|
assert isinstance(saml_document, SAMLResponse)
|
|
|
|
result = None
|
|
try:
|
|
params = relayed_params.copy()
|
|
params["subdomain"] = subdomain
|
|
for param, value in params.items():
|
|
if param in self.standard_relay_params:
|
|
self.strategy.session_set(param, value)
|
|
|
|
# We want the IdP name to be accessible from the social pipeline.
|
|
self.strategy.session_set("saml_idp_name", idp_name)
|
|
session_index = saml_document.get_session_index()
|
|
if session_index is None:
|
|
# In general IdPs will always provide a SessionIndex, but we can't know
|
|
# if some providers might not send it, so we allow it but log the event.
|
|
self.logger.info(
|
|
"/complete/saml/: IdP did not provide SessionIndex in the SAMLResponse."
|
|
)
|
|
self.strategy.session_set("saml_session_index", session_index)
|
|
|
|
# super().auth_complete expects to have RelayState set to the idp_name,
|
|
# so we need to replace this param.
|
|
post_params = self.strategy.request.POST.copy()
|
|
post_params["RelayState"] = idp_name
|
|
self.strategy.request.POST = post_params
|
|
|
|
# Call the auth_complete method of SocialAuthMixIn
|
|
result = super().auth_complete(*args, **kwargs)
|
|
except SAMLResponse.SAML_PARSING_EXCEPTIONS:
|
|
# These can be raised if SAMLResponse is missing or badly formatted.
|
|
self.logger.info("/complete/saml/: error while parsing SAMLResponse:", exc_info=True)
|
|
# Fall through to returning None.
|
|
finally:
|
|
# We need a finally: block to ensure we don't keep around information in the session
|
|
# if the authentication failed.
|
|
if result is None:
|
|
for param in [*self.standard_relay_params, "saml_idp_name", "saml_session_index"]:
|
|
# If an attacker managed to eavesdrop on the RelayState token,
|
|
# they may pass it here to the endpoint with an invalid SAMLResponse.
|
|
# We remove these potentially sensitive parameters that we have set in the session
|
|
# earlier, to avoid leaking their values.
|
|
self.strategy.session_set(param, None)
|
|
|
|
return result
|
|
|
|
@classmethod
|
|
def validate_idp_for_subdomain(cls, idp_name: str, subdomain: str) -> bool:
|
|
idp_dict = settings.SOCIAL_AUTH_SAML_ENABLED_IDPS.get(idp_name)
|
|
if idp_dict is None:
|
|
raise AssertionError(f"IdP: {idp_name} not found")
|
|
if "limit_to_subdomains" in idp_dict and subdomain not in idp_dict["limit_to_subdomains"]:
|
|
return False
|
|
|
|
return True
|
|
|
|
@classmethod
|
|
def check_config(cls) -> bool:
|
|
obligatory_saml_settings_list = [
|
|
settings.SOCIAL_AUTH_SAML_SP_ENTITY_ID,
|
|
settings.SOCIAL_AUTH_SAML_ORG_INFO,
|
|
settings.SOCIAL_AUTH_SAML_TECHNICAL_CONTACT,
|
|
settings.SOCIAL_AUTH_SAML_SUPPORT_CONTACT,
|
|
settings.SOCIAL_AUTH_SAML_ENABLED_IDPS,
|
|
]
|
|
if any(not setting for setting in obligatory_saml_settings_list):
|
|
return False
|
|
|
|
return True
|
|
|
|
@classmethod
|
|
@override
|
|
def dict_representation(cls, realm: Realm | None = None) -> list[ExternalAuthMethodDictT]:
|
|
result: list[ExternalAuthMethodDictT] = []
|
|
for idp_name, idp_dict in settings.SOCIAL_AUTH_SAML_ENABLED_IDPS.items():
|
|
if realm and not cls.validate_idp_for_subdomain(idp_name, realm.subdomain):
|
|
continue
|
|
if realm is None and "limit_to_subdomains" in idp_dict:
|
|
# If queried without a realm, only return IdPs that can be used on all realms.
|
|
continue
|
|
|
|
saml_dict: ExternalAuthMethodDictT = dict(
|
|
name=f"saml:{idp_name}",
|
|
display_name=idp_dict.get("display_name", cls.auth_backend_name),
|
|
display_icon=idp_dict.get("display_icon", cls.display_icon),
|
|
login_url=reverse("login-social", args=("saml", idp_name)),
|
|
signup_url=reverse("signup-social", args=("saml", idp_name)),
|
|
)
|
|
result.append(saml_dict)
|
|
|
|
return result
|
|
|
|
@override
|
|
def should_auto_signup(self) -> bool:
|
|
"""
|
|
This function is meant to be called in the social pipeline or later,
|
|
as it requires (validated) information about the IdP name to have
|
|
already been store in the session.
|
|
"""
|
|
idp_name = self.strategy.session_get("saml_idp_name")
|
|
assert isinstance(idp_name, str)
|
|
auto_signup = settings.SOCIAL_AUTH_SAML_ENABLED_IDPS[idp_name].get("auto_signup", False)
|
|
assert isinstance(auto_signup, bool)
|
|
return auto_signup
|
|
|
|
@override
|
|
def get_params_to_store_in_authenticated_session(self) -> dict[str, str]:
|
|
idp_name = self.strategy.session_get("saml_idp_name")
|
|
saml_session_index = self.strategy.session_get("saml_session_index")
|
|
|
|
return {
|
|
"social_auth_backend": f"saml:{idp_name}",
|
|
"saml_session_index": saml_session_index,
|
|
}
|
|
|
|
|
|
def patch_saml_auth_require_messages_signed(auth: OneLogin_Saml2_Auth) -> None:
|
|
"""
|
|
wantMessagesSigned controls whether requests processed by this saml auth
|
|
object need to be signed. The default of False is often not acceptable,
|
|
because we don't want anyone to be able to submit such a request.
|
|
Callers should use this to enforce the requirement of signatures.
|
|
"""
|
|
|
|
auth.get_settings().get_security_data()["wantMessagesSigned"] = True
|
|
# Defensive code to confirm the setting change above is successful,
|
|
# to catch API changes in python3-saml that would make the change not
|
|
# be applied to the actual settings of `auth` - e.g. due to us only
|
|
# receiving a copy of the dict.
|
|
assert auth.get_settings().get_security_data()["wantMessagesSigned"] is True
|
|
|
|
|
|
@external_auth_method
|
|
class GenericOpenIdConnectBackend(SocialAuthMixin, OpenIdConnectAuth):
|
|
name = "oidc"
|
|
auth_backend_name = "OpenID Connect"
|
|
sort_order = 100
|
|
|
|
# Hack: We don't yet support multiple IdPs, but we want this
|
|
# module to import if nothing has been configured yet.
|
|
settings_dict: OIDCIdPConfigDict
|
|
[settings_dict] = settings.SOCIAL_AUTH_OIDC_ENABLED_IDPS.values() or [OIDCIdPConfigDict()]
|
|
|
|
display_icon: str | None = settings_dict.get("display_icon", None)
|
|
display_name: str = settings_dict.get("display_name", "OIDC")
|
|
|
|
full_name_validated = getattr(settings, "SOCIAL_AUTH_OIDC_FULL_NAME_VALIDATED", False)
|
|
|
|
# Discovery endpoint for the superclass to read all the appropriate
|
|
# configuration from.
|
|
OIDC_ENDPOINT = settings_dict.get("oidc_url")
|
|
|
|
@override
|
|
def get_key_and_secret(self) -> tuple[str, str]:
|
|
client_id = self.settings_dict.get("client_id", "")
|
|
assert isinstance(client_id, str)
|
|
secret = self.settings_dict.get("secret", "")
|
|
assert isinstance(secret, str)
|
|
return client_id, secret
|
|
|
|
@classmethod
|
|
def check_config(cls) -> bool:
|
|
if len(settings.SOCIAL_AUTH_OIDC_ENABLED_IDPS.keys()) != 1:
|
|
# Only one IdP supported for now.
|
|
return False
|
|
|
|
mandatory_config_keys = ["oidc_url", "client_id", "secret"]
|
|
[idp_config_dict] = settings.SOCIAL_AUTH_OIDC_ENABLED_IDPS.values()
|
|
if not all(idp_config_dict.get(key) for key in mandatory_config_keys):
|
|
return False
|
|
|
|
return True
|
|
|
|
@classmethod
|
|
@override
|
|
def dict_representation(cls, realm: Realm | None = None) -> list[ExternalAuthMethodDictT]:
|
|
return [
|
|
dict(
|
|
name=f"oidc:{cls.name}",
|
|
display_name=cls.display_name,
|
|
display_icon=cls.display_icon,
|
|
login_url=reverse("login-social", args=(cls.name,)),
|
|
signup_url=reverse("signup-social", args=(cls.name,)),
|
|
)
|
|
]
|
|
|
|
@override
|
|
def should_auto_signup(self) -> bool:
|
|
result = self.settings_dict.get("auto_signup", False)
|
|
assert isinstance(result, bool)
|
|
return result
|
|
|
|
|
|
def validate_otp_params(
|
|
mobile_flow_otp: str | None = None, desktop_flow_otp: str | None = None
|
|
) -> None:
|
|
for otp in [mobile_flow_otp, desktop_flow_otp]:
|
|
if otp is not None and not is_valid_otp(otp):
|
|
raise JsonableError(_("Invalid OTP"))
|
|
|
|
if mobile_flow_otp and desktop_flow_otp:
|
|
raise JsonableError(_("Can't use both mobile_flow_otp and desktop_flow_otp together."))
|
|
|
|
|
|
class SAMLSPInitiatedLogout:
|
|
@classmethod
|
|
def get_logged_in_user_idp(cls, request: HttpRequest) -> str | None:
|
|
"""
|
|
Information about the authentication method which was used for
|
|
this session is stored in social_auth_backend session attribute.
|
|
If SAML was used, this extracts the IdP name and returns it.
|
|
"""
|
|
# Some asserts to ensure this doesn't get called incorrectly:
|
|
assert hasattr(request, "user")
|
|
assert isinstance(request.user, UserProfile)
|
|
|
|
authentication_method = request.session.get("social_auth_backend", "")
|
|
if not authentication_method.startswith("saml:"):
|
|
return None
|
|
|
|
return authentication_method.split("saml:")[1]
|
|
|
|
@classmethod
|
|
def get_logged_in_user_session_index(cls, request: HttpRequest) -> str | None:
|
|
"""
|
|
During SAML authentication, we obtain the SessionIndex value provided
|
|
by the IdP and save it in the session. This function can be used
|
|
to retrieve it.
|
|
"""
|
|
# Some asserts to ensure this doesn't get called incorrectly:
|
|
assert hasattr(request, "user")
|
|
assert isinstance(request.user, UserProfile)
|
|
|
|
session_index = request.session.get("saml_session_index")
|
|
return session_index
|
|
|
|
@classmethod
|
|
def slo_request_to_idp(cls, request: HttpRequest, return_to: str | None = None) -> HttpResponse:
|
|
"""
|
|
Generates the redirect to the IdP's SLO endpoint with
|
|
the appropriately generated LogoutRequest. This should only be called
|
|
on requests with a session that was indeed obtained via SAML.
|
|
"""
|
|
|
|
user_profile = request.user
|
|
assert isinstance(user_profile, UserProfile)
|
|
|
|
realm = user_profile.realm
|
|
assert saml_auth_enabled(realm)
|
|
|
|
complete_url = reverse("social:complete", args=("saml",))
|
|
saml_backend = load_backend(load_strategy(request), "saml", complete_url)
|
|
|
|
idp_name = cls.get_logged_in_user_idp(request)
|
|
if idp_name is None:
|
|
raise AssertionError("User not logged in via SAML")
|
|
session_index = cls.get_logged_in_user_session_index(request)
|
|
|
|
idp = saml_backend.get_idp(idp_name)
|
|
auth = saml_backend._create_saml_auth(idp)
|
|
slo_url = auth.logout(
|
|
name_id=user_profile.delivery_email, return_to=return_to, session_index=session_index
|
|
)
|
|
|
|
return HttpResponseRedirect(slo_url)
|
|
|
|
@classmethod
|
|
def process_logout_response(cls, logout_response: SAMLResponse, idp_name: str) -> HttpResponse:
|
|
"""
|
|
Validates the LogoutResponse and logs out the user if successful,
|
|
finishing the SP-initiated logout flow.
|
|
"""
|
|
idp = logout_response.backend.get_idp(idp_name)
|
|
auth = logout_response.backend._create_saml_auth(idp)
|
|
auth.process_slo(keep_local_session=True)
|
|
errors = auth.get_errors()
|
|
if errors:
|
|
# These errors should essentially only happen in case of misconfiguration,
|
|
# so we give a json error response with the direct error codes from python3-saml.
|
|
# They're informative but generic enough to not leak any sensitive information.
|
|
raise JsonableError(f"LogoutResponse error: {errors}")
|
|
|
|
logout(logout_response.backend.strategy.request)
|
|
return HttpResponseRedirect(settings.LOGIN_URL)
|
|
|
|
|
|
def get_external_method_dicts(realm: Realm | None = None) -> list[ExternalAuthMethodDictT]:
|
|
"""
|
|
Returns a list of dictionaries that represent social backends, sorted
|
|
in the order in which they should be displayed.
|
|
"""
|
|
result: list[ExternalAuthMethodDictT] = []
|
|
for backend in EXTERNAL_AUTH_METHODS:
|
|
# EXTERNAL_AUTH_METHODS is already sorted in the correct order,
|
|
# so we don't need to worry about sorting here.
|
|
if auth_enabled_helper([backend.auth_backend_name], realm):
|
|
result.extend(backend.dict_representation(realm))
|
|
|
|
return result
|
|
|
|
|
|
AUTH_BACKEND_NAME_MAP: dict[str, Any] = {
|
|
"Dev": DevAuthBackend,
|
|
"Email": EmailAuthBackend,
|
|
"LDAP": ZulipLDAPAuthBackend,
|
|
}
|
|
|
|
for external_method in EXTERNAL_AUTH_METHODS:
|
|
AUTH_BACKEND_NAME_MAP[external_method.auth_backend_name] = external_method
|
|
|
|
EXTERNAL_AUTH_METHODS = sorted(EXTERNAL_AUTH_METHODS, key=lambda x: x.sort_order, reverse=True)
|
|
|
|
# Provide this alternative name for backwards compatibility with
|
|
# installations that had the old backend enabled.
|
|
GoogleMobileOauth2Backend = GoogleAuthBackend
|