From a4f87917779f8461755357af14eb57b8e8b5d817 Mon Sep 17 00:00:00 2001 From: rimu <3310831+rimu@users.noreply.github.com> Date: Sat, 26 Aug 2023 15:41:11 +1200 Subject: [PATCH] user registration and email verification --- README.md | 9 + app/__init__.py | 3 + app/auth/email.py | 8 + app/auth/forms.py | 12 +- app/auth/recaptcha3.py | 4 +- app/auth/routes.py | 34 ++- app/auth/util.py | 7 + app/main/routes.py | 7 +- app/models.py | 1 + app/static/fonts/feather/feather.svg | 276 ++++++++++++++++++ app/static/fonts/feather/feather.ttf | Bin 0 -> 65112 bytes app/static/fonts/feather/feather.woff | Bin 0 -> 65188 bytes app/static/scss/_colours.scss | 4 + app/static/scss/_controls.scss | 64 ++++ app/static/scss/_dark.scss | 34 +++ app/static/scss/_mixins.scss | 20 ++ app/static/scss/_typography.scss | 153 ++++++++++ app/static/structure.css | 267 ++++++++++++++++- app/static/structure.scss | 53 +++- app/static/styles.css | 253 ++++++++++++++++ app/static/styles.scss | 51 ++++ app/templates/auth/login.html | 1 + app/templates/base.html | 5 +- app/templates/email/close_account.html | 3 + app/templates/email/close_account.txt | 3 + app/templates/email/reset_password.html | 12 + app/templates/email/reset_password.txt | 11 + app/templates/email/verification.html | 11 + app/templates/email/verification.txt | 9 + app/templates/email/welcome.html | 13 + app/templates/email/welcome.txt | 14 + app/templates/index.html | 6 + config.py | 4 +- ...e4cef1698c_user_email_addr_verification.py | 32 ++ requirements.txt | 1 + 35 files changed, 1368 insertions(+), 17 deletions(-) create mode 100644 app/auth/util.py create mode 100755 app/static/fonts/feather/feather.svg create mode 100755 app/static/fonts/feather/feather.ttf create mode 100755 app/static/fonts/feather/feather.woff create mode 100644 app/static/scss/_colours.scss create mode 100644 app/static/scss/_controls.scss create mode 100644 app/static/scss/_dark.scss create mode 100644 app/static/scss/_mixins.scss create mode 100644 app/static/scss/_typography.scss create mode 100644 app/templates/email/close_account.html create mode 100644 app/templates/email/close_account.txt create mode 100644 app/templates/email/reset_password.html create mode 100644 app/templates/email/reset_password.txt create mode 100644 app/templates/email/verification.html create mode 100644 app/templates/email/verification.txt create mode 100644 app/templates/email/welcome.html create mode 100644 app/templates/email/welcome.txt create mode 100644 app/templates/index.html create mode 100644 migrations/versions/93e4cef1698c_user_email_addr_verification.py diff --git a/README.md b/README.md index 22c8e222..3f4552d8 100644 --- a/README.md +++ b/README.md @@ -5,3 +5,12 @@ A lemmy/kbin clone written in Python with Flask. - Clean, simple code. - Easy setup, easy to manage - few dependencies and extra software required. - GPL. + +## Contributing + +Join our Maxtrix channel. + +Report bugs in our issue queue. + +If submitting a substantial PR, communicate your intentions to the maintainer(s) before investing your effort into coding. +This is to ensure your contribution will fit well with the rest of the code base. \ No newline at end of file diff --git a/app/__init__.py b/app/__init__.py index 8a4a0340..0909cd47 100644 --- a/app/__init__.py +++ b/app/__init__.py @@ -50,6 +50,9 @@ def create_app(config_class=Config): from app.activitypub import bp as activitypub_bp app.register_blueprint(activitypub_bp) + from app.auth import bp as auth_bp + app.register_blueprint(auth_bp, url_prefix='/auth') + def get_resource_as_string(name, charset='utf-8'): with app.open_resource(name) as f: return f.read().decode(charset) diff --git a/app/auth/email.py b/app/auth/email.py index ea1fadfe..3c24f91b 100644 --- a/app/auth/email.py +++ b/app/auth/email.py @@ -20,3 +20,11 @@ def send_welcome_email(user): recipients=[user.email], text_body=render_template('email/welcome.txt', user=user), html_body=render_template('email/welcome.html', user=user)) + + +def send_verification_email(user): + send_email(_('Please verify your email address'), + sender='PyFedi ', + recipients=[user.email], + text_body=render_template('email/verification.txt', user=user), + html_body=render_template('email/verification.html', user=user)) \ No newline at end of file diff --git a/app/auth/forms.py b/app/auth/forms.py index 6e2e3437..2d9ec1d2 100644 --- a/app/auth/forms.py +++ b/app/auth/forms.py @@ -2,7 +2,7 @@ from flask_wtf import FlaskForm from wtforms import StringField, PasswordField, SubmitField, HiddenField from wtforms.validators import ValidationError, DataRequired, Email, EqualTo, Length from flask_babel import _, lazy_gettext as _l -from app.models import User +from app.models import User, Community from app.auth.recaptcha3 import Recaptcha3Field @@ -22,13 +22,21 @@ class RegistrationForm(FlaskForm): EqualTo('password')]) recaptcha = Recaptcha3Field(action="TestAction", execute_on_load=True) - submit = SubmitField(_l('Sign up')) + submit = SubmitField(_l('Register')) def validate_real_email(self, email): user = User.query.filter_by(email=email.data).first() if user is not None: raise ValidationError(_('An account with this email address already exists.')) + def validate_user_name(self, user_name): + user = User.query.filter_by(user_name=user_name.data).first() + if user is not None: + raise ValidationError(_('An account with this user name already exists.')) + community = Community.query.filter_by(name=user_name.data).first() + if community is not None: + raise ValidationError(_('A community with this name exists so it cannot be used for a user.')) + class ResetPasswordRequestForm(FlaskForm): email = StringField(_l('Email'), validators=[DataRequired(), Email()]) diff --git a/app/auth/recaptcha3.py b/app/auth/recaptcha3.py index fbd5cd02..6a7ea5bd 100644 --- a/app/auth/recaptcha3.py +++ b/app/auth/recaptcha3.py @@ -1,14 +1,12 @@ import logging import requests -from flask import Markup, current_app, json, request +from flask import Markup, current_app, request from wtforms import ValidationError from wtforms.fields import HiddenField from wtforms.widgets import HiddenInput logger = logging.getLogger(__name__) -JSONEncoder = json.JSONEncoder - RECAPTCHA_TEMPLATE = '''