Merge pull request #532 from mouse-reeve/email-login

Allow users to log in with email or username
This commit is contained in:
Mouse Reeve 2021-01-18 11:45:02 -08:00 committed by GitHub
commit da81e25022
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
5 changed files with 46 additions and 16 deletions

View file

@ -0,0 +1,22 @@
# Generated by Django 3.0.7 on 2021-01-18 19:00
from django.db import migrations, models
class Migration(migrations.Migration):
dependencies = [
('bookwyrm', '0036_annualgoal'),
]
operations = [
migrations.AlterModelOptions(
name='shelfbook',
options={'ordering': ('-created_date',)},
),
migrations.AlterField(
model_name='user',
name='email',
field=models.EmailField(max_length=254, unique=True),
),
]

View file

@ -25,6 +25,7 @@ from . import fields, Review
class User(OrderedCollectionPageMixin, AbstractUser): class User(OrderedCollectionPageMixin, AbstractUser):
''' a user who wants to read books ''' ''' a user who wants to read books '''
username = fields.UsernameField() username = fields.UsernameField()
email = models.EmailField(unique=True)
key_pair = fields.OneToOneField( key_pair = fields.OneToOneField(
'KeyPair', 'KeyPair',

View file

@ -8,40 +8,40 @@
{% endif %} {% endif %}
<form name="edit-profile" action="/edit-profile/" method="post" enctype="multipart/form-data"> <form name="edit-profile" action="/edit-profile/" method="post" enctype="multipart/form-data">
{% csrf_token %} {% csrf_token %}
<p class="block"> <div class="block">
<label class="label" for="id_avatar">Avatar:</label> <label class="label" for="id_avatar">Avatar:</label>
{{ form.avatar }} {{ form.avatar }}
{% for error in form.avatar.errors %} {% for error in form.avatar.errors %}
<p class="help is-danger">{{ error | escape }}</p> <p class="help is-danger">{{ error | escape }}</p>
{% endfor %} {% endfor %}
</p> </div>
<p class="block"> <div class="block">
<label class="label" for="id_name">Display name:</label> <label class="label" for="id_name">Display name:</label>
{{ form.name }} {{ form.name }}
{% for error in form.name.errors %} {% for error in form.name.errors %}
<p class="help is-danger">{{ error | escape }}</p> <p class="help is-danger">{{ error | escape }}</p>
{% endfor %} {% endfor %}
</p> </div>
<p class="block"> <div class="block">
<label class="label" for="id_summary">Summary:</label> <label class="label" for="id_summary">Summary:</label>
{{ form.summary }} {{ form.summary }}
{% for error in form.summary.errors %} {% for error in form.summary.errors %}
<p class="help is-danger">{{ error | escape }}</p> <p class="help is-danger">{{ error | escape }}</p>
{% endfor %} {% endfor %}
</p> </div>
<p class="block"> <div class="block">
<label class="label" for="id_email">Email address:</label> <label class="label" for="id_email">Email address:</label>
{{ form.email }} {{ form.email }}
{% for error in form.email.errors %} {% for error in form.email.errors %}
<p class="help is-danger">{{ error | escape }}</p> <p class="help is-danger">{{ error | escape }}</p>
{% endfor %} {% endfor %}
</p> </div>
<p class="block"> <div class="block">
<label class="checkbox label" for="id_manually_approves_followers"> <label class="checkbox label" for="id_manually_approves_followers">
Manually approve followers: Manually approve followers:
{{ form.manually_approves_followers }} {{ form.manually_approves_followers }}
</label> </label>
</p> </div>
<button class="button is-primary" type="submit">Save</button> <button class="button is-primary" type="submit">Save</button>
</form> </form>
</div> </div>
@ -50,14 +50,14 @@
<h2 class="title">Change password</h2> <h2 class="title">Change password</h2>
<form name="edit-profile" action="/change-password/" method="post" enctype="multipart/form-data"> <form name="edit-profile" action="/change-password/" method="post" enctype="multipart/form-data">
{% csrf_token %} {% csrf_token %}
<p class="block"> <div class="block">
<label class="label" for="id_password">New password:</label> <label class="label" for="id_password">New password:</label>
<input type="password" name="password" maxlength="128" class="input" required="" id="id_password"> <input type="password" name="password" maxlength="128" class="input" required="" id="id_password">
</p> </div>
<p class="block"> <div class="block">
<label class="label" for="id_confirm_password">Confirm password:</label> <label class="label" for="id_confirm_password">Confirm password:</label>
<input type="password" name="confirm-password" maxlength="128" class="input" required="" id="id_confirm_password"> <input type="password" name="confirm-password" maxlength="128" class="input" required="" id="id_confirm_password">
</p> </div>
<button class="button is-primary" type="submit">Change password</button> <button class="button is-primary" type="submit">Change password</button>
</form> </form>
</div> </div>

View file

@ -32,6 +32,13 @@ class Login(View):
login_form = forms.LoginForm(request.POST) login_form = forms.LoginForm(request.POST)
localname = login_form.data['localname'] localname = login_form.data['localname']
if '@' in localname: # looks like an email address to me
email = localname
try:
username = models.User.objects.get(email=email)
except models.User.DoesNotExist: # maybe it's a full username?
username = localname
else:
username = '%s@%s' % (localname, DOMAIN) username = '%s@%s' % (localname, DOMAIN)
password = login_form.data['password'] password = login_form.data['password']
user = authenticate(request, username=username, password=password) user = authenticate(request, username=username, password=password)

View file

@ -185,4 +185,4 @@ class EditUser(View):
user.save() user.save()
broadcast(user, user.to_update_activity(user)) broadcast(user, user.to_update_activity(user))
return redirect('/user/%s' % request.user.localname) return redirect(user.local_path)