Log in user using either email address or username in Django

后端 未结 13 2431
故里飘歌
故里飘歌 2020-12-02 08:34

I am trying to create an auth backend to allow my users to log in using either their email address or their username in Django 1.6 with a custom user model. The backend work

相关标签:
13条回答
  • 2020-12-02 08:54

    After following the advice given to me above and changing AUTHENTICATION_BACKENDS = ['yourapp.yourfile.EmailOrUsernameModelBackend'] I was getting the error Manager isn't available; User has been swapped for 'users.User'. This was caused because I was using the default User model instead of my own custom one. Here is the working code.

    from django.conf import settings
    from django.contrib.auth import get_user_model
    
    class EmailOrUsernameModelBackend(object):
        """
        This is a ModelBacked that allows authentication with either a username or an email address.
    
        """
        def authenticate(self, username=None, password=None):
            if '@' in username:
                kwargs = {'email': username}
            else:
                kwargs = {'username': username}
            try:
                user = get_user_model().objects.get(**kwargs)
                if user.check_password(password):
                    return user
            except User.DoesNotExist:
                return None
    
        def get_user(self, username):
            try:
                return get_user_model().objects.get(pk=username)
            except get_user_model().DoesNotExist:
                return None
    
    0 讨论(0)
  • 2020-12-02 08:55

    I wrote the code for this in 2 simple steps :

    1. VIEWS.py
         if request.method == 'POST':
                userinput = request.POST['username']
    
                try:
                    username = userbase.objects.get(email=userinput).username
                except userbase.DoesNotExist:
                    username = request.POST['username']
                password = request.POST['password']
    
    1. INDEX.html

      I created 2 input fields from which 1st is for username/email. I take whatever input is given and try to search the same data in email column of db, if it matches, I return the username and then try to authenticate , if it doesn't , I use input directly as Username.

    I'm using Django 2.2

    0 讨论(0)
  • 2020-12-02 08:57

    if you are using django-rest-auth then the option to authenticate with email address is built in, and may conflict with the other methods proposed. You just need to add the following to settings.py:

    ACCOUNT_AUTHENTICATION_METHOD = 'username_email'
    ACCOUNT_EMAIL_REQUIRED = False
    ACCOUNT_USERNAME_REQUIRED = False
    
    #Following is added to enable registration with email instead of username
    AUTHENTICATION_BACKENDS = (
     # Needed to login by username in Django admin, regardless of `allauth`
     "django.contrib.auth.backends.ModelBackend",
    
     # `allauth` specific authentication methods, such as login by e-mail
     "allauth.account.auth_backends.AuthenticationBackend",
    )
    

    Django rest auth email instead of username https://django-allauth.readthedocs.io/en/latest/configuration.html

    Note that unless you want to have a single box into which the user can type a username or an email address, you'll have to do some work in the front end to decide whether to send the login request as email, password or username, password. I did a simple test whether the user's entry contained an '@' with a '.' further on. I think somebody deliberately creating a username that looks like an email address - but isn't their email address - is unlikely enough that I'm not supporting it.

    0 讨论(0)
  • 2020-12-02 08:58

    You can simply have a Try block to do this.. use:

    email = request.POST['username'] 
    raw_password = request.POST['password'] 
    
       try:
            account = authenticate(username=MyUserAccount.objects.get(email=email).username,password=raw_password)
            if account is not None:
                login(request, account)
                return redirect('home')
    
        except:
            account = authenticate(username=email, password=raw_password)
            if account is not None:
                login(request, account)
                return redirect('home')
    
    0 讨论(0)
  • 2020-12-02 09:08

    Yet another solution:

    from django.contrib.auth import get_user_model
    from django.contrib.auth.backends import ModelBackend
    from django.db.models import Q
    
    
    class EmailOrUsernameModelBackend(ModelBackend):
        """
        Authentication backend which allows users to authenticate using either their
        username or email address
    
        Source: https://stackoverflow.com/a/35836674/59984
        """
    
        def authenticate(self, request, username=None, password=None, **kwargs):
            # n.b. Django <2.1 does not pass the `request`
    
            user_model = get_user_model()
    
            if username is None:
                username = kwargs.get(user_model.USERNAME_FIELD)
    
            # The `username` field is allows to contain `@` characters so
            # technically a given email address could be present in either field,
            # possibly even for different users, so we'll query for all matching
            # records and test each one.
            users = user_model._default_manager.filter(
                Q(**{user_model.USERNAME_FIELD: username}) | Q(email__iexact=username)
            )
    
            # Test whether any matched user has the provided password:
            for user in users:
                if user.check_password(password):
                    return user
            if not users:
                # Run the default password hasher once to reduce the timing
                # difference between an existing and a non-existing user (see
                # https://code.djangoproject.com/ticket/20760)
                user_model().set_password(password)
    

    Fixes:

    • By default, @ is not prohibited in the username field, so unless custom User model prohibits @ symbol, it can't be used to distinguish between username and email.
    • Technically, there can be two users using the same email, one in the email field, the other in the username. Unless such possibility is restricted, it can lead to either user not being able to authenticate, or unhandled MultipleObjectsReturned exception if UserModel._default_manager.get(Q(username__iexact=username) | Q(email__iexact=username)) is used.
    • Catching any exception with except: is generally bad practice

    Downside - if there are two users, using the same email, one in the username, the other in email, and they have the same password, then it's prone to authenticating the first match. I guess the chances of this is highly unlikely.

    Also note: any of the approaches should enforce unique email field in the User model, since the default User model does not define unique email, which would lead to either unhandled exception in case User.objects.get(email__iexact="...") is used, or authenticating the first match. In any case, using email to login assumes that email is unique.

    0 讨论(0)
  • 2020-12-02 09:09

    Assuming you have blocked/forbidden against the username having an @, and you want to use the django User model.

    if request.method == 'POST':
        form = LoginForm(request.POST)
        if form.is_valid():
            cd=form.cleaned_data
            if '@' in cd['username']:
                username=User.objects.get(email=cd['username']).username
            else:
                username=cd['username']
    
            user = authenticate(username=username,
                                    password=cd['password'])
    
            if user is not None and user.is_active:
                login(request,user)
                return redirect('loggedin')
            else:
                return render(request, 'login.html')
    
    0 讨论(0)
提交回复
热议问题