2013-11-04 23:16:46 +01:00
|
|
|
|
2016-08-01 13:06:35 +02:00
|
|
|
import logging
|
2017-03-03 19:01:52 +01:00
|
|
|
from typing import Any, Dict, List, Set, Tuple, Optional, Text
|
2016-08-01 13:06:35 +02:00
|
|
|
|
2013-11-04 23:16:46 +01:00
|
|
|
from django.contrib.auth.backends import RemoteUserBackend
|
2013-11-21 01:30:20 +01:00
|
|
|
from django.conf import settings
|
2016-08-08 09:38:50 +02:00
|
|
|
from django.http import HttpResponse
|
2013-11-04 23:42:31 +01:00
|
|
|
import django.contrib.auth
|
|
|
|
|
2016-08-08 09:38:50 +02:00
|
|
|
from django_auth_ldap.backend import LDAPBackend, _LDAPUser
|
2015-10-13 23:08:05 +02:00
|
|
|
from zerver.lib.actions import do_create_user
|
2013-11-21 01:30:20 +01:00
|
|
|
|
2015-10-13 23:08:05 +02:00
|
|
|
from zerver.models import UserProfile, Realm, get_user_profile_by_id, \
|
|
|
|
get_user_profile_by_email, remote_user_to_email, email_to_username, \
|
2017-10-02 08:32:09 +02:00
|
|
|
get_realm
|
2013-08-06 22:51:47 +02:00
|
|
|
|
2014-01-10 23:48:05 +01:00
|
|
|
from apiclient.sample_tools import client as googleapiclient
|
|
|
|
from oauth2client.crypt import AppIdentityError
|
2017-01-21 16:52:59 +01:00
|
|
|
from social_core.backends.github import GithubOAuth2, GithubOrganizationOAuth2, \
|
2016-08-01 13:06:35 +02:00
|
|
|
GithubTeamOAuth2
|
2017-03-07 08:32:40 +01:00
|
|
|
from social_core.exceptions import AuthFailed, SocialAuthBaseException
|
2016-07-20 13:33:27 +02:00
|
|
|
from django.contrib.auth import authenticate
|
2017-02-08 05:04:14 +01:00
|
|
|
from zerver.lib.users import check_full_name
|
|
|
|
from zerver.lib.request import JsonableError
|
2016-10-07 11:10:21 +02:00
|
|
|
from zerver.lib.utils import check_subdomain, get_subdomain
|
2013-08-06 22:51:47 +02:00
|
|
|
|
2017-03-24 10:48:52 +01:00
|
|
|
from social_django.models import DjangoStorage
|
|
|
|
from social_django.strategy import DjangoStrategy
|
|
|
|
|
2016-11-06 23:44:45 +01:00
|
|
|
def pad_method_dict(method_dict):
|
2016-12-08 05:06:51 +01:00
|
|
|
# type: (Dict[Text, bool]) -> Dict[Text, bool]
|
2016-11-06 23:44:45 +01:00
|
|
|
"""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, realm):
|
2016-12-08 05:06:51 +01:00
|
|
|
# type: (List[Text], Optional[Realm]) -> bool
|
2016-11-02 21:41:10 +01:00
|
|
|
if realm is not None:
|
|
|
|
enabled_method_dict = realm.authentication_methods_dict()
|
|
|
|
pad_method_dict(enabled_method_dict)
|
|
|
|
else:
|
|
|
|
enabled_method_dict = dict((method, True) for method in Realm.AUTHENTICATION_FLAGS)
|
|
|
|
pad_method_dict(enabled_method_dict)
|
2016-11-06 23:44:45 +01:00
|
|
|
for supported_backend in django.contrib.auth.get_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
|
2013-11-04 23:42:31 +01:00
|
|
|
return False
|
|
|
|
|
2016-11-07 00:04:59 +01:00
|
|
|
def ldap_auth_enabled(realm=None):
|
|
|
|
# type: (Optional[Realm]) -> bool
|
|
|
|
return auth_enabled_helper([u'LDAP'], realm)
|
|
|
|
|
|
|
|
def email_auth_enabled(realm=None):
|
|
|
|
# type: (Optional[Realm]) -> bool
|
|
|
|
return auth_enabled_helper([u'Email'], realm)
|
|
|
|
|
2016-11-06 23:44:45 +01:00
|
|
|
def password_auth_enabled(realm=None):
|
|
|
|
# type: (Optional[Realm]) -> bool
|
2016-11-07 00:04:59 +01:00
|
|
|
return ldap_auth_enabled(realm) or email_auth_enabled(realm)
|
2015-08-19 02:58:20 +02:00
|
|
|
|
2016-11-06 23:44:45 +01:00
|
|
|
def dev_auth_enabled(realm=None):
|
|
|
|
# type: (Optional[Realm]) -> bool
|
|
|
|
return auth_enabled_helper([u'Dev'], realm)
|
|
|
|
|
|
|
|
def google_auth_enabled(realm=None):
|
|
|
|
# type: (Optional[Realm]) -> bool
|
|
|
|
return auth_enabled_helper([u'Google'], realm)
|
|
|
|
|
|
|
|
def github_auth_enabled(realm=None):
|
|
|
|
# type: (Optional[Realm]) -> bool
|
|
|
|
return auth_enabled_helper([u'GitHub'], realm)
|
2015-08-19 02:58:20 +02:00
|
|
|
|
2017-04-20 21:02:56 +02:00
|
|
|
def any_oauth_backend_enabled(realm=None):
|
|
|
|
# type: (Optional[Realm]) -> bool
|
|
|
|
"""Used by the login page process to determine whether to show the
|
|
|
|
'OR' for login with Google"""
|
|
|
|
return auth_enabled_helper([u'GitHub', u'Google'], realm)
|
|
|
|
|
2017-09-15 16:59:03 +02:00
|
|
|
def require_email_format_usernames(realm=None):
|
|
|
|
# type: (Optional[Realm]) -> bool
|
|
|
|
if ldap_auth_enabled(realm):
|
|
|
|
if settings.LDAP_EMAIL_ATTR or settings.LDAP_APPEND_DOMAIN:
|
|
|
|
return False
|
|
|
|
return True
|
|
|
|
|
2016-04-21 21:07:43 +02:00
|
|
|
def common_get_active_user_by_email(email, return_data=None):
|
2016-12-08 05:06:51 +01:00
|
|
|
# type: (Text, Optional[Dict[str, Any]]) -> Optional[UserProfile]
|
2016-04-21 07:19:08 +02:00
|
|
|
try:
|
|
|
|
user_profile = get_user_profile_by_email(email)
|
|
|
|
except UserProfile.DoesNotExist:
|
|
|
|
return None
|
2016-04-21 21:07:43 +02:00
|
|
|
if not user_profile.is_active:
|
|
|
|
if return_data is not None:
|
|
|
|
return_data['inactive_user'] = True
|
|
|
|
return None
|
|
|
|
if user_profile.realm.deactivated:
|
|
|
|
if return_data is not None:
|
|
|
|
return_data['inactive_realm'] = True
|
2016-04-21 07:19:08 +02:00
|
|
|
return None
|
|
|
|
return user_profile
|
|
|
|
|
2013-11-01 20:22:12 +01:00
|
|
|
class ZulipAuthMixin(object):
|
|
|
|
def get_user(self, user_profile_id):
|
2016-08-08 09:38:50 +02:00
|
|
|
# type: (int) -> Optional[UserProfile]
|
2013-11-01 20:22:12 +01:00
|
|
|
""" Get a UserProfile object from the user_profile_id. """
|
|
|
|
try:
|
|
|
|
return get_user_profile_by_id(user_profile_id)
|
|
|
|
except UserProfile.DoesNotExist:
|
|
|
|
return None
|
|
|
|
|
2016-07-26 12:12:01 +02:00
|
|
|
class SocialAuthMixin(ZulipAuthMixin):
|
2017-06-04 11:38:23 +02:00
|
|
|
auth_backend_name = None # type: Text
|
2016-11-07 00:09:21 +01:00
|
|
|
|
2016-08-08 09:45:52 +02:00
|
|
|
def get_email_address(self, *args, **kwargs):
|
2016-12-08 05:06:51 +01:00
|
|
|
# type: (*Any, **Any) -> Text
|
2016-07-26 12:12:01 +02:00
|
|
|
raise NotImplementedError
|
|
|
|
|
2016-08-08 09:45:52 +02:00
|
|
|
def get_full_name(self, *args, **kwargs):
|
2016-12-08 05:06:51 +01:00
|
|
|
# type: (*Any, **Any) -> Text
|
2016-07-26 12:12:01 +02:00
|
|
|
raise NotImplementedError
|
|
|
|
|
2017-03-24 10:48:52 +01:00
|
|
|
def authenticate(self,
|
|
|
|
realm_subdomain='', # type: Optional[Text]
|
|
|
|
storage=None, # type: Optional[DjangoStorage]
|
2017-04-19 19:05:04 +02:00
|
|
|
strategy=None, # type: Optional[DjangoStrategy]
|
2017-03-24 10:48:52 +01:00
|
|
|
user=None, # type: Optional[Dict[str, Any]]
|
|
|
|
return_data=None, # type: Optional[Dict[str, Any]]
|
|
|
|
response=None, # type: Optional[Dict[str, Any]]
|
|
|
|
backend=None # type: Optional[GithubOAuth2]
|
|
|
|
):
|
|
|
|
# type: (...) -> Optional[UserProfile]
|
|
|
|
"""
|
|
|
|
Django decides which `authenticate` to call by inspecting the
|
|
|
|
arguments. So it's better to create `authenticate` function
|
|
|
|
with well defined arguments.
|
|
|
|
|
|
|
|
Keeping this function separate so that it can easily be
|
|
|
|
overridden.
|
|
|
|
"""
|
|
|
|
if user is None:
|
|
|
|
user = {}
|
|
|
|
|
2017-10-02 08:32:09 +02:00
|
|
|
assert return_data is not None
|
|
|
|
assert response is not None
|
2017-03-24 10:48:52 +01:00
|
|
|
|
|
|
|
return self._common_authenticate(self,
|
|
|
|
realm_subdomain=realm_subdomain,
|
|
|
|
storage=storage,
|
|
|
|
strategy=strategy,
|
|
|
|
user=user,
|
|
|
|
return_data=return_data,
|
|
|
|
response=response,
|
|
|
|
backend=backend)
|
|
|
|
|
|
|
|
def _common_authenticate(self, *args, **kwargs):
|
2016-08-08 09:38:50 +02:00
|
|
|
# type: (*Any, **Any) -> Optional[UserProfile]
|
2016-07-26 12:12:01 +02:00
|
|
|
return_data = kwargs.get('return_data', {})
|
|
|
|
|
|
|
|
email_address = self.get_email_address(*args, **kwargs)
|
|
|
|
if not email_address:
|
2017-03-23 06:02:07 +01:00
|
|
|
return_data['invalid_email'] = True
|
2016-07-26 12:12:01 +02:00
|
|
|
return None
|
|
|
|
|
|
|
|
try:
|
|
|
|
user_profile = get_user_profile_by_email(email_address)
|
|
|
|
except UserProfile.DoesNotExist:
|
|
|
|
return_data["valid_attestation"] = True
|
|
|
|
return None
|
|
|
|
|
|
|
|
if not user_profile.is_active:
|
|
|
|
return_data["inactive_user"] = True
|
|
|
|
return None
|
|
|
|
|
|
|
|
if user_profile.realm.deactivated:
|
|
|
|
return_data["inactive_realm"] = True
|
|
|
|
return None
|
|
|
|
|
2016-07-19 14:35:08 +02:00
|
|
|
if not check_subdomain(kwargs.get("realm_subdomain"),
|
|
|
|
user_profile.realm.subdomain):
|
|
|
|
return_data["invalid_subdomain"] = True
|
|
|
|
return None
|
|
|
|
|
2016-11-07 00:09:21 +01:00
|
|
|
if not auth_enabled_helper([self.auth_backend_name], user_profile.realm):
|
|
|
|
return_data["auth_backend_disabled"] = True
|
|
|
|
return None
|
|
|
|
|
2016-07-26 12:12:01 +02:00
|
|
|
return user_profile
|
|
|
|
|
|
|
|
def process_do_auth(self, user_profile, *args, **kwargs):
|
2016-08-08 09:38:50 +02:00
|
|
|
# type: (UserProfile, *Any, **Any) -> Optional[HttpResponse]
|
2017-03-23 05:01:39 +01:00
|
|
|
# These functions need to be imported here to avoid cyclic
|
2016-07-26 12:12:01 +02:00
|
|
|
# dependency.
|
2016-12-01 13:10:59 +01:00
|
|
|
from zerver.views.auth import (login_or_register_remote_user,
|
|
|
|
redirect_to_subdomain_login_url)
|
2017-01-07 21:46:03 +01:00
|
|
|
from zerver.views.registration import redirect_and_log_into_subdomain
|
2016-07-26 12:12:01 +02:00
|
|
|
|
|
|
|
return_data = kwargs.get('return_data', {})
|
|
|
|
|
|
|
|
inactive_user = return_data.get('inactive_user')
|
|
|
|
inactive_realm = return_data.get('inactive_realm')
|
2016-10-07 11:10:21 +02:00
|
|
|
invalid_subdomain = return_data.get('invalid_subdomain')
|
2017-03-23 06:02:07 +01:00
|
|
|
invalid_email = return_data.get('invalid_email')
|
2016-07-26 12:12:01 +02:00
|
|
|
|
2017-03-24 11:00:58 +01:00
|
|
|
if inactive_user or inactive_realm:
|
2017-03-23 06:02:07 +01:00
|
|
|
# Redirect to login page. We can't send to registration
|
2017-03-24 11:00:58 +01:00
|
|
|
# workflow with these errors. We will redirect to login page.
|
2016-07-26 12:12:01 +02:00
|
|
|
return None
|
|
|
|
|
2017-03-24 11:00:58 +01:00
|
|
|
if invalid_email:
|
|
|
|
# In case of invalid email, we will end up on registration page.
|
|
|
|
# This seems better than redirecting to login page.
|
|
|
|
logging.warning(
|
|
|
|
"{} got invalid email argument.".format(self.auth_backend_name)
|
|
|
|
)
|
2017-03-23 06:02:07 +01:00
|
|
|
|
2016-12-01 13:10:59 +01:00
|
|
|
strategy = self.strategy # type: ignore # This comes from Python Social Auth.
|
|
|
|
request = strategy.request
|
2016-07-26 12:12:01 +02:00
|
|
|
email_address = self.get_email_address(*args, **kwargs)
|
|
|
|
full_name = self.get_full_name(*args, **kwargs)
|
2017-05-05 19:54:36 +02:00
|
|
|
is_signup = strategy.session_get('is_signup') == '1'
|
2016-07-26 12:12:01 +02:00
|
|
|
|
2016-12-01 13:10:59 +01:00
|
|
|
subdomain = strategy.session_get('subdomain')
|
2017-09-30 00:30:55 +02:00
|
|
|
mobile_flow_otp = strategy.session_get('mobile_flow_otp')
|
|
|
|
if not subdomain or mobile_flow_otp is not None:
|
2016-12-01 13:10:59 +01:00
|
|
|
return login_or_register_remote_user(request, email_address,
|
|
|
|
user_profile, full_name,
|
2017-05-05 19:54:36 +02:00
|
|
|
invalid_subdomain=bool(invalid_subdomain),
|
2017-09-30 00:30:55 +02:00
|
|
|
mobile_flow_otp=mobile_flow_otp,
|
2017-05-05 19:54:36 +02:00
|
|
|
is_signup=is_signup)
|
2016-12-01 13:10:59 +01:00
|
|
|
try:
|
|
|
|
realm = Realm.objects.get(string_id=subdomain)
|
|
|
|
except Realm.DoesNotExist:
|
|
|
|
return redirect_to_subdomain_login_url()
|
|
|
|
|
2017-05-05 19:54:36 +02:00
|
|
|
return redirect_and_log_into_subdomain(realm, full_name, email_address,
|
|
|
|
is_signup=is_signup)
|
2016-07-26 12:12:01 +02:00
|
|
|
|
2017-02-28 11:58:03 +01:00
|
|
|
def auth_complete(self, *args, **kwargs):
|
|
|
|
# type: (*Any, **Any) -> Optional[HttpResponse]
|
2017-03-23 05:46:10 +01:00
|
|
|
"""
|
|
|
|
Returning `None` from this function will redirect the browser
|
|
|
|
to the login page.
|
|
|
|
"""
|
2017-02-28 11:58:03 +01:00
|
|
|
try:
|
|
|
|
# Call the auth_complete method of BaseOAuth2 is Python Social Auth
|
2017-07-28 00:59:36 +02:00
|
|
|
return super(SocialAuthMixin, self).auth_complete(*args, **kwargs) # type: ignore # monkey-patching
|
2017-02-28 11:58:03 +01:00
|
|
|
except AuthFailed:
|
|
|
|
return None
|
2017-03-07 08:32:40 +01:00
|
|
|
except SocialAuthBaseException as e:
|
|
|
|
logging.exception(e)
|
|
|
|
return None
|
2017-02-28 11:58:03 +01:00
|
|
|
|
2013-11-21 04:57:23 +01:00
|
|
|
class ZulipDummyBackend(ZulipAuthMixin):
|
|
|
|
"""
|
|
|
|
Used when we want to log you in but we don't know which backend to use.
|
|
|
|
"""
|
2016-11-29 07:22:02 +01:00
|
|
|
|
2016-08-14 04:44:55 +02:00
|
|
|
def authenticate(self, username=None, realm_subdomain=None, use_dummy_backend=False,
|
|
|
|
return_data=None):
|
2016-12-08 05:06:51 +01:00
|
|
|
# type: (Optional[Text], Optional[Text], bool, Optional[Dict[str, Any]]) -> Optional[UserProfile]
|
2017-03-19 23:29:29 +01:00
|
|
|
assert username is not None
|
2013-11-21 04:57:23 +01:00
|
|
|
if use_dummy_backend:
|
2016-07-19 14:35:08 +02:00
|
|
|
user_profile = common_get_active_user_by_email(username)
|
|
|
|
if user_profile is None:
|
|
|
|
return None
|
2016-08-14 04:44:55 +02:00
|
|
|
if not check_subdomain(realm_subdomain, user_profile.realm.subdomain):
|
2017-05-23 23:06:15 +02:00
|
|
|
if return_data is not None:
|
|
|
|
return_data["invalid_subdomain"] = True
|
2016-08-14 04:44:55 +02:00
|
|
|
return None
|
|
|
|
return user_profile
|
2013-11-21 04:57:23 +01:00
|
|
|
return None
|
|
|
|
|
2013-11-01 20:22:12 +01:00
|
|
|
class EmailAuthBackend(ZulipAuthMixin):
|
2013-08-06 22:51:47 +02:00
|
|
|
"""
|
|
|
|
Email Authentication Backend
|
|
|
|
|
|
|
|
Allows a user to sign in using an email/password pair rather than
|
|
|
|
a username/password pair.
|
|
|
|
"""
|
2016-11-29 07:22:02 +01:00
|
|
|
|
2016-07-19 14:35:08 +02:00
|
|
|
def authenticate(self, username=None, password=None, realm_subdomain=None, return_data=None):
|
2016-12-08 05:06:51 +01:00
|
|
|
# type: (Optional[Text], Optional[str], Optional[Text], Optional[Dict[str, Any]]) -> Optional[UserProfile]
|
2013-08-06 22:51:47 +02:00
|
|
|
""" Authenticate a user based on email address as the user name. """
|
|
|
|
if username is None or password is None:
|
|
|
|
# Return immediately. Otherwise we will look for a SQL row with
|
|
|
|
# NULL username. While that's probably harmless, it's needless
|
|
|
|
# exposure.
|
|
|
|
return None
|
|
|
|
|
2016-04-21 21:07:43 +02:00
|
|
|
user_profile = common_get_active_user_by_email(username, return_data=return_data)
|
2016-04-21 07:19:08 +02:00
|
|
|
if user_profile is None:
|
2013-08-06 22:51:47 +02:00
|
|
|
return None
|
2016-04-21 07:19:08 +02:00
|
|
|
if not password_auth_enabled(user_profile.realm):
|
2016-04-21 21:07:43 +02:00
|
|
|
if return_data is not None:
|
|
|
|
return_data['password_auth_disabled'] = True
|
2016-04-21 07:19:08 +02:00
|
|
|
return None
|
2016-11-07 00:04:59 +01:00
|
|
|
if not email_auth_enabled(user_profile.realm):
|
|
|
|
if return_data is not None:
|
|
|
|
return_data['email_auth_disabled'] = True
|
|
|
|
return None
|
2016-04-21 07:19:08 +02:00
|
|
|
if user_profile.check_password(password):
|
2016-07-19 14:35:08 +02:00
|
|
|
if not check_subdomain(realm_subdomain, user_profile.realm.subdomain):
|
2017-05-23 23:06:15 +02:00
|
|
|
if return_data is not None:
|
|
|
|
return_data["invalid_subdomain"] = True
|
2016-07-19 14:35:08 +02:00
|
|
|
return None
|
2016-04-21 07:19:08 +02:00
|
|
|
return user_profile
|
2016-07-19 14:35:08 +02:00
|
|
|
return None
|
2013-08-06 22:51:47 +02:00
|
|
|
|
2014-01-10 23:48:05 +01:00
|
|
|
class GoogleMobileOauth2Backend(ZulipAuthMixin):
|
|
|
|
"""
|
|
|
|
Google Apps authentication for mobile devices
|
|
|
|
|
|
|
|
Allows a user to sign in using a Google-issued OAuth2 token.
|
|
|
|
|
|
|
|
Ref:
|
|
|
|
https://developers.google.com/+/mobile/android/sign-in#server-side_access_for_your_app
|
|
|
|
https://developers.google.com/accounts/docs/CrossClientAuth#offlineAccess
|
|
|
|
|
|
|
|
"""
|
2016-11-29 07:22:02 +01:00
|
|
|
|
2017-03-24 08:45:21 +01:00
|
|
|
def authenticate(self, google_oauth2_token=None, realm_subdomain=None, return_data=None):
|
|
|
|
# type: (Optional[str], Optional[Text], Optional[Dict[str, Any]]) -> Optional[UserProfile]
|
|
|
|
if return_data is None:
|
|
|
|
return_data = {}
|
|
|
|
|
2014-01-10 23:48:05 +01:00
|
|
|
try:
|
|
|
|
token_payload = googleapiclient.verify_id_token(google_oauth2_token, settings.GOOGLE_CLIENT_ID)
|
|
|
|
except AppIdentityError:
|
|
|
|
return None
|
2014-05-14 18:57:09 +02:00
|
|
|
if token_payload["email_verified"] in (True, "true"):
|
2014-01-10 23:48:05 +01:00
|
|
|
try:
|
2016-04-21 07:19:08 +02:00
|
|
|
user_profile = get_user_profile_by_email(token_payload["email"])
|
2014-01-10 23:48:05 +01:00
|
|
|
except UserProfile.DoesNotExist:
|
|
|
|
return_data["valid_attestation"] = True
|
|
|
|
return None
|
2016-04-21 21:07:43 +02:00
|
|
|
if not user_profile.is_active:
|
|
|
|
return_data["inactive_user"] = True
|
|
|
|
return None
|
|
|
|
if user_profile.realm.deactivated:
|
|
|
|
return_data["inactive_realm"] = True
|
2016-04-21 07:19:08 +02:00
|
|
|
return None
|
2016-07-19 14:35:08 +02:00
|
|
|
if not check_subdomain(realm_subdomain, user_profile.realm.subdomain):
|
|
|
|
return_data["invalid_subdomain"] = True
|
|
|
|
return None
|
2016-11-07 00:09:21 +01:00
|
|
|
if not google_auth_enabled(realm=user_profile.realm):
|
|
|
|
return_data["google_auth_disabled"] = True
|
|
|
|
return None
|
2016-04-21 07:19:08 +02:00
|
|
|
return user_profile
|
2014-01-10 23:48:05 +01:00
|
|
|
else:
|
|
|
|
return_data["valid_attestation"] = False
|
2017-03-03 20:30:49 +01:00
|
|
|
return None
|
2014-01-10 23:48:05 +01:00
|
|
|
|
2013-11-04 23:16:46 +01:00
|
|
|
class ZulipRemoteUserBackend(RemoteUserBackend):
|
|
|
|
create_unknown_user = False
|
2016-11-29 07:22:02 +01:00
|
|
|
|
2016-07-19 14:35:08 +02:00
|
|
|
def authenticate(self, remote_user, realm_subdomain=None):
|
2017-05-24 04:21:29 +02:00
|
|
|
# type: (Optional[str], Optional[Text]) -> Optional[UserProfile]
|
2013-11-04 23:16:46 +01:00
|
|
|
if not remote_user:
|
2016-01-26 03:11:31 +01:00
|
|
|
return None
|
2013-11-04 23:16:46 +01:00
|
|
|
|
|
|
|
email = remote_user_to_email(remote_user)
|
2016-11-07 00:09:21 +01:00
|
|
|
user_profile = common_get_active_user_by_email(email)
|
|
|
|
if user_profile is None:
|
|
|
|
return None
|
|
|
|
if not check_subdomain(realm_subdomain, user_profile.realm.subdomain):
|
|
|
|
return None
|
|
|
|
if not auth_enabled_helper([u"RemoteUser"], user_profile.realm):
|
|
|
|
return None
|
|
|
|
return user_profile
|
2015-02-06 18:30:28 +01:00
|
|
|
|
2017-09-27 00:56:34 +02:00
|
|
|
class ZulipLDAPException(_LDAPUser.AuthenticationFailed):
|
2015-10-13 23:08:05 +02:00
|
|
|
pass
|
|
|
|
|
2017-09-22 10:58:12 +02:00
|
|
|
class ZulipLDAPConfigurationError(Exception):
|
|
|
|
pass
|
|
|
|
|
2015-10-13 23:08:05 +02:00
|
|
|
class ZulipLDAPAuthBackendBase(ZulipAuthMixin, LDAPBackend):
|
|
|
|
# Don't use Django LDAP's permissions functions
|
|
|
|
def has_perm(self, user, perm, obj=None):
|
2017-05-24 04:21:29 +02:00
|
|
|
# type: (Optional[UserProfile], Any, Any) -> bool
|
2016-08-08 09:38:50 +02:00
|
|
|
# Using Any type is safe because we are not doing anything with
|
|
|
|
# the arguments.
|
2015-10-13 23:08:05 +02:00
|
|
|
return False
|
2016-11-29 07:22:02 +01:00
|
|
|
|
2015-10-13 23:08:05 +02:00
|
|
|
def has_module_perms(self, user, app_label):
|
2017-05-24 04:21:29 +02:00
|
|
|
# type: (Optional[UserProfile], Optional[str]) -> bool
|
2015-10-13 23:08:05 +02:00
|
|
|
return False
|
2016-11-29 07:22:02 +01:00
|
|
|
|
2015-10-13 23:08:05 +02:00
|
|
|
def get_all_permissions(self, user, obj=None):
|
2017-05-24 04:21:29 +02:00
|
|
|
# type: (Optional[UserProfile], Any) -> Set
|
2016-08-08 09:38:50 +02:00
|
|
|
# Using Any type is safe because we are not doing anything with
|
|
|
|
# the arguments.
|
2015-10-13 23:08:05 +02:00
|
|
|
return set()
|
2016-11-29 07:22:02 +01:00
|
|
|
|
2015-10-13 23:08:05 +02:00
|
|
|
def get_group_permissions(self, user, obj=None):
|
2017-05-24 04:21:29 +02:00
|
|
|
# type: (Optional[UserProfile], Any) -> Set
|
2016-08-08 09:38:50 +02:00
|
|
|
# Using Any type is safe because we are not doing anything with
|
|
|
|
# the arguments.
|
2015-10-13 23:08:05 +02:00
|
|
|
return set()
|
|
|
|
|
2013-11-21 01:30:20 +01:00
|
|
|
def django_to_ldap_username(self, username):
|
2016-12-08 05:06:51 +01:00
|
|
|
# type: (Text) -> Text
|
2015-10-13 22:57:38 +02:00
|
|
|
if settings.LDAP_APPEND_DOMAIN:
|
2015-10-13 23:08:05 +02:00
|
|
|
if not username.endswith("@" + settings.LDAP_APPEND_DOMAIN):
|
|
|
|
raise ZulipLDAPException("Username does not match LDAP domain.")
|
2013-11-21 01:30:20 +01:00
|
|
|
return email_to_username(username)
|
|
|
|
return username
|
2016-11-29 07:22:02 +01:00
|
|
|
|
2013-11-21 01:30:20 +01:00
|
|
|
def ldap_to_django_username(self, username):
|
2016-08-08 09:38:50 +02:00
|
|
|
# type: (str) -> str
|
2015-10-13 22:57:38 +02:00
|
|
|
if settings.LDAP_APPEND_DOMAIN:
|
2013-11-25 17:57:30 +01:00
|
|
|
return "@".join((username, settings.LDAP_APPEND_DOMAIN))
|
2013-11-21 01:30:20 +01:00
|
|
|
return username
|
|
|
|
|
2015-10-13 23:08:05 +02:00
|
|
|
class ZulipLDAPAuthBackend(ZulipLDAPAuthBackendBase):
|
2017-09-22 10:58:12 +02:00
|
|
|
REALM_IS_NONE_ERROR = 1
|
|
|
|
|
2016-07-19 14:35:08 +02:00
|
|
|
def authenticate(self, username, password, realm_subdomain=None, return_data=None):
|
2016-12-08 05:06:51 +01:00
|
|
|
# type: (Text, str, Optional[Text], Optional[Dict[str, Any]]) -> Optional[UserProfile]
|
2015-10-13 23:08:05 +02:00
|
|
|
try:
|
2017-10-02 08:32:09 +02:00
|
|
|
self._realm = get_realm(realm_subdomain)
|
2015-10-13 23:08:05 +02:00
|
|
|
username = self.django_to_ldap_username(username)
|
2016-07-19 14:35:08 +02:00
|
|
|
user_profile = ZulipLDAPAuthBackendBase.authenticate(self, username, password)
|
|
|
|
if user_profile is None:
|
|
|
|
return None
|
|
|
|
if not check_subdomain(realm_subdomain, user_profile.realm.subdomain):
|
|
|
|
return None
|
|
|
|
return user_profile
|
2015-10-13 23:08:05 +02:00
|
|
|
except Realm.DoesNotExist:
|
2017-10-02 08:32:09 +02:00
|
|
|
return None # nocoverage # TODO: this may no longer be possible
|
2017-09-27 21:30:53 +02:00
|
|
|
except ZulipLDAPException:
|
2017-10-02 08:32:09 +02:00
|
|
|
return None # nocoverage # TODO: this may no longer be possible
|
2015-10-13 23:08:05 +02:00
|
|
|
|
2013-11-21 01:30:20 +01:00
|
|
|
def get_or_create_user(self, username, ldap_user):
|
2016-08-08 09:38:50 +02:00
|
|
|
# type: (str, _LDAPUser) -> Tuple[UserProfile, bool]
|
2013-11-21 01:30:20 +01:00
|
|
|
try:
|
2017-09-10 17:25:24 +02:00
|
|
|
if settings.LDAP_EMAIL_ATTR is not None:
|
|
|
|
# Get email from ldap attributes.
|
|
|
|
if settings.LDAP_EMAIL_ATTR not in ldap_user.attrs:
|
|
|
|
raise ZulipLDAPException("LDAP user doesn't have the needed %s attribute" % (settings.LDAP_EMAIL_ATTR,))
|
|
|
|
|
|
|
|
username = ldap_user.attrs[settings.LDAP_EMAIL_ATTR][0]
|
|
|
|
|
2016-04-21 07:19:08 +02:00
|
|
|
user_profile = get_user_profile_by_email(username)
|
|
|
|
if not user_profile.is_active or user_profile.realm.deactivated:
|
|
|
|
raise ZulipLDAPException("Realm has been deactivated")
|
2016-11-07 00:09:21 +01:00
|
|
|
if not ldap_auth_enabled(user_profile.realm):
|
|
|
|
raise ZulipLDAPException("LDAP Authentication is not enabled")
|
2016-04-21 07:19:08 +02:00
|
|
|
return user_profile, False
|
2013-11-21 01:30:20 +01:00
|
|
|
except UserProfile.DoesNotExist:
|
2017-06-21 11:10:56 +02:00
|
|
|
if self._realm is None:
|
2017-09-22 10:58:12 +02:00
|
|
|
raise ZulipLDAPConfigurationError("Realm is None", self.REALM_IS_NONE_ERROR)
|
2016-04-21 07:19:08 +02:00
|
|
|
# No need to check for an inactive user since they don't exist yet
|
2017-01-22 11:21:27 +01:00
|
|
|
if self._realm.deactivated:
|
2016-04-21 07:19:08 +02:00
|
|
|
raise ZulipLDAPException("Realm has been deactivated")
|
2015-10-13 23:08:05 +02:00
|
|
|
|
|
|
|
full_name_attr = settings.AUTH_LDAP_USER_ATTR_MAP["full_name"]
|
2015-10-26 17:17:51 +01:00
|
|
|
short_name = full_name = ldap_user.attrs[full_name_attr][0]
|
2017-02-08 05:04:14 +01:00
|
|
|
try:
|
|
|
|
full_name = check_full_name(full_name)
|
|
|
|
except JsonableError as e:
|
2017-07-20 00:22:36 +02:00
|
|
|
raise ZulipLDAPException(e.msg)
|
2015-10-13 23:08:05 +02:00
|
|
|
if "short_name" in settings.AUTH_LDAP_USER_ATTR_MAP:
|
|
|
|
short_name_attr = settings.AUTH_LDAP_USER_ATTR_MAP["short_name"]
|
2015-10-26 17:17:51 +01:00
|
|
|
short_name = ldap_user.attrs[short_name_attr][0]
|
2013-11-21 01:30:20 +01:00
|
|
|
|
2017-01-22 11:21:27 +01:00
|
|
|
user_profile = do_create_user(username, None, self._realm, full_name, short_name)
|
2016-10-28 14:41:07 +02:00
|
|
|
return user_profile, True
|
2013-11-21 01:30:20 +01:00
|
|
|
|
2015-10-13 23:08:05 +02:00
|
|
|
# Just like ZulipLDAPAuthBackend, but doesn't let you log in.
|
|
|
|
class ZulipLDAPUserPopulator(ZulipLDAPAuthBackendBase):
|
2016-07-19 14:35:08 +02:00
|
|
|
def authenticate(self, username, password, realm_subdomain=None):
|
2016-12-08 05:06:51 +01:00
|
|
|
# type: (Text, str, Optional[Text]) -> None
|
2013-11-21 01:30:20 +01:00
|
|
|
return None
|
2015-08-19 02:58:20 +02:00
|
|
|
|
|
|
|
class DevAuthBackend(ZulipAuthMixin):
|
|
|
|
# Allow logging in as any user without a password.
|
|
|
|
# This is used for convenience when developing Zulip.
|
2016-07-19 14:35:08 +02:00
|
|
|
def authenticate(self, username, realm_subdomain=None, return_data=None):
|
2016-12-08 05:06:51 +01:00
|
|
|
# type: (Text, Optional[Text], Optional[Dict[str, Any]]) -> Optional[UserProfile]
|
2016-11-07 00:09:21 +01:00
|
|
|
user_profile = common_get_active_user_by_email(username, return_data=return_data)
|
|
|
|
if user_profile is None:
|
|
|
|
return None
|
|
|
|
if not dev_auth_enabled(user_profile.realm):
|
|
|
|
return None
|
|
|
|
return user_profile
|
2016-07-20 13:33:27 +02:00
|
|
|
|
2016-07-29 21:47:14 +02:00
|
|
|
class GitHubAuthBackend(SocialAuthMixin, GithubOAuth2):
|
2016-11-07 00:09:21 +01:00
|
|
|
auth_backend_name = u"GitHub"
|
|
|
|
|
2016-07-26 12:12:01 +02:00
|
|
|
def get_email_address(self, *args, **kwargs):
|
2016-12-08 05:06:51 +01:00
|
|
|
# type: (*Any, **Any) -> Optional[Text]
|
2016-07-20 13:33:27 +02:00
|
|
|
try:
|
2016-07-26 12:12:01 +02:00
|
|
|
return kwargs['response']['email']
|
2017-10-02 08:32:09 +02:00
|
|
|
except KeyError: # nocoverage # TODO: investigate
|
2016-07-20 13:33:27 +02:00
|
|
|
return None
|
|
|
|
|
2016-07-26 12:12:01 +02:00
|
|
|
def get_full_name(self, *args, **kwargs):
|
2016-12-08 05:06:51 +01:00
|
|
|
# type: (*Any, **Any) -> Text
|
2017-03-09 09:45:21 +01:00
|
|
|
# In case of any error return an empty string. Name is used by
|
|
|
|
# the registration page to pre-populate the name field. However,
|
|
|
|
# if it is not supplied, our registration process will make sure
|
|
|
|
# that the user enters a valid name.
|
2016-07-20 13:33:27 +02:00
|
|
|
try:
|
2017-03-09 09:45:21 +01:00
|
|
|
name = kwargs['response']['name']
|
2016-07-26 12:12:01 +02:00
|
|
|
except KeyError:
|
2017-03-09 09:45:21 +01:00
|
|
|
name = ''
|
|
|
|
|
|
|
|
if name is None:
|
2016-07-26 12:12:01 +02:00
|
|
|
return ''
|
2016-07-20 13:33:27 +02:00
|
|
|
|
2017-03-09 09:45:21 +01:00
|
|
|
return name
|
|
|
|
|
2016-07-20 13:33:27 +02:00
|
|
|
def do_auth(self, *args, **kwargs):
|
2016-12-14 07:41:17 +01:00
|
|
|
# type: (*Any, **Any) -> Optional[HttpResponse]
|
2017-03-23 05:46:10 +01:00
|
|
|
"""
|
|
|
|
This function is called once the OAuth2 workflow is complete. We
|
|
|
|
override this function to:
|
|
|
|
1. Inject `return_data` and `realm_admin` kwargs. These will
|
|
|
|
be used by `authenticate()` function to make the decision.
|
|
|
|
2. Call the proper `do_auth` function depending on whether
|
|
|
|
we are doing individual, team or organization based GitHub
|
|
|
|
authentication.
|
|
|
|
The actual decision on authentication is done in
|
2017-03-24 10:48:52 +01:00
|
|
|
SocialAuthMixin._common_authenticate().
|
2017-03-23 05:46:10 +01:00
|
|
|
"""
|
2016-07-20 13:33:27 +02:00
|
|
|
kwargs['return_data'] = {}
|
2016-10-07 11:10:21 +02:00
|
|
|
|
2016-10-17 08:22:00 +02:00
|
|
|
request = self.strategy.request
|
2016-10-07 11:10:21 +02:00
|
|
|
kwargs['realm_subdomain'] = get_subdomain(request)
|
|
|
|
|
2016-08-01 13:06:35 +02:00
|
|
|
user_profile = None
|
|
|
|
|
|
|
|
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):
|
2017-02-28 11:58:03 +01:00
|
|
|
try:
|
|
|
|
user_profile = GithubOAuth2.do_auth(self, *args, **kwargs)
|
|
|
|
except AuthFailed:
|
|
|
|
logging.info("User authentication failed.")
|
|
|
|
user_profile = None
|
2016-08-01 13:06:35 +02:00
|
|
|
|
|
|
|
elif (team_id):
|
|
|
|
backend = GithubTeamOAuth2(self.strategy, self.redirect_uri)
|
|
|
|
try:
|
|
|
|
user_profile = backend.do_auth(*args, **kwargs)
|
|
|
|
except AuthFailed:
|
2017-01-12 03:08:29 +01:00
|
|
|
logging.info("User is not member of GitHub team.")
|
2016-08-01 13:06:35 +02:00
|
|
|
user_profile = None
|
|
|
|
|
|
|
|
elif (org_name):
|
|
|
|
backend = GithubOrganizationOAuth2(self.strategy, self.redirect_uri)
|
|
|
|
try:
|
|
|
|
user_profile = backend.do_auth(*args, **kwargs)
|
|
|
|
except AuthFailed:
|
2017-01-12 03:08:29 +01:00
|
|
|
logging.info("User is not member of GitHub organization.")
|
2016-08-01 13:06:35 +02:00
|
|
|
user_profile = None
|
|
|
|
|
2016-07-26 12:12:01 +02:00
|
|
|
return self.process_do_auth(user_profile, *args, **kwargs)
|
2016-11-06 23:44:45 +01:00
|
|
|
|
|
|
|
AUTH_BACKEND_NAME_MAP = {
|
|
|
|
u'Dev': DevAuthBackend,
|
|
|
|
u'Email': EmailAuthBackend,
|
|
|
|
u'GitHub': GitHubAuthBackend,
|
|
|
|
u'Google': GoogleMobileOauth2Backend,
|
|
|
|
u'LDAP': ZulipLDAPAuthBackend,
|
|
|
|
u'RemoteUser': ZulipRemoteUserBackend,
|
2017-06-04 11:38:23 +02:00
|
|
|
} # type: Dict[Text, Any]
|