1
0
mirror of https://github.com/Mailu/Mailu.git synced 2024-12-16 10:59:53 +02:00
Mailu/core/admin/mailu/ui/views/users.py

193 lines
8.7 KiB
Python
Raw Normal View History

2022-11-03 17:19:44 +02:00
from mailu import models, utils
from mailu.ui import ui, access, forms
from flask import current_app as app
2016-03-19 21:37:48 +02:00
import flask
2016-08-13 20:51:54 +02:00
import flask_login
import wtforms
import wtforms_components
2016-03-19 21:37:48 +02:00
@ui.route('/user/list/<domain_name>', methods=['GET'])
@access.domain_admin(models.Domain, 'domain_name')
2016-03-19 21:37:48 +02:00
def user_list(domain_name):
domain = models.Domain.query.get(domain_name) or flask.abort(404)
2016-03-19 21:37:48 +02:00
return flask.render_template('user/list.html', domain=domain)
@ui.route('/user/create/<domain_name>', methods=['GET', 'POST'])
@access.domain_admin(models.Domain, 'domain_name')
2016-03-19 21:37:48 +02:00
def user_create(domain_name):
domain = models.Domain.query.get(domain_name) or flask.abort(404)
if not domain.max_users == -1 and len(domain.users) >= domain.max_users:
2016-03-19 21:37:48 +02:00
flask.flash('Too many users for domain %s' % domain, 'error')
return flask.redirect(
flask.url_for('.user_list', domain_name=domain.name))
form = forms.UserForm()
2019-01-09 13:03:47 +02:00
form.pw.validators = [wtforms.validators.DataRequired()]
2023-02-23 14:54:37 +02:00
form.quota_bytes.default = app.config['DEFAULT_QUOTA']
if domain.max_quota_bytes:
form.quota_bytes.validators = [
wtforms.validators.NumberRange(max=domain.max_quota_bytes)]
if form.quota_bytes.default > domain.max_quota_bytes:
form.quota_bytes.default = domain.max_quota_bytes
2016-03-19 21:37:48 +02:00
if form.validate_on_submit():
2022-11-03 17:19:44 +02:00
if msg := utils.isBadOrPwned(form):
flask.flash(msg, "error")
2022-10-31 20:41:40 +02:00
return flask.render_template('user/create.html',
domain=domain, form=form)
if domain.has_email(form.localpart.data):
2016-05-01 21:12:08 +02:00
flask.flash('Email is already used', 'error')
2016-03-19 21:37:48 +02:00
else:
2016-05-01 21:12:08 +02:00
user = models.User(domain=domain)
form.populate_obj(user)
2016-03-19 21:37:48 +02:00
user.set_password(form.pw.data)
models.db.session.add(user)
models.db.session.commit()
2017-11-10 12:56:43 +02:00
user.send_welcome()
2016-03-19 21:37:48 +02:00
flask.flash('User %s created' % user)
return flask.redirect(
flask.url_for('.user_list', domain_name=domain.name))
2023-02-24 07:51:41 +02:00
form.process()
2016-03-19 21:37:48 +02:00
return flask.render_template('user/create.html',
domain=domain, form=form)
@ui.route('/user/edit/<path:user_email>', methods=['GET', 'POST'])
@access.domain_admin(models.User, 'user_email')
2016-03-19 21:37:48 +02:00
def user_edit(user_email):
user = models.User.query.get(user_email) or flask.abort(404)
# Handle the case where user quota is more than allowed
max_quota_bytes = user.domain.max_quota_bytes
if max_quota_bytes and user.quota_bytes > max_quota_bytes:
max_quota_bytes = user.quota_bytes
# Create the form
form = forms.UserForm(obj=user)
wtforms_components.read_only(form.localpart)
form.localpart.validators = []
if max_quota_bytes:
form.quota_bytes.validators = [
wtforms.validators.NumberRange(max=max_quota_bytes)]
2016-03-19 21:37:48 +02:00
if form.validate_on_submit():
2022-11-20 10:10:39 +02:00
if form.pw.data:
if msg := utils.isBadOrPwned(form):
flask.flash(msg, "error")
return flask.render_template('user/edit.html', form=form, user=user,
domain=user.domain, max_quota_bytes=max_quota_bytes)
2016-05-01 21:12:08 +02:00
form.populate_obj(user)
2016-03-22 22:15:57 +02:00
if form.pw.data:
2023-08-11 09:03:42 +02:00
user.set_password(form.pw.data, keep_only_session=flask.session)
models.db.session.commit()
2016-03-19 21:37:48 +02:00
flask.flash('User %s updated' % user)
return flask.redirect(
flask.url_for('.user_list', domain_name=user.domain.name))
return flask.render_template('user/edit.html', form=form, user=user,
domain=user.domain, max_quota_bytes=max_quota_bytes)
2016-03-19 21:37:48 +02:00
@ui.route('/user/settings', methods=['GET', 'POST'], defaults={'user_email': None})
@ui.route('/user/usersettings/<path:user_email>', methods=['GET', 'POST'])
@access.owner(models.User, 'user_email')
2016-03-19 21:37:48 +02:00
def user_settings(user_email):
user_email_or_current = user_email or flask_login.current_user.email
user = models.User.query.get(user_email_or_current) or flask.abort(404)
2016-03-20 12:14:27 +02:00
form = forms.UserSettingsForm(obj=user)
2022-11-14 20:05:41 +02:00
utils.formatCSVField(form.forward_destination)
2016-03-20 12:09:06 +02:00
if form.validate_on_submit():
2018-12-28 22:03:57 +02:00
form.forward_destination.data = form.forward_destination.data.replace(" ","").split(",")
2016-05-01 21:12:08 +02:00
form.populate_obj(user)
models.db.session.commit()
2018-12-28 22:03:57 +02:00
form.forward_destination.data = ", ".join(form.forward_destination.data)
2016-03-20 12:09:06 +02:00
flask.flash('Settings updated for %s' % user)
if user_email:
return flask.redirect(
flask.url_for('.user_list', domain_name=user.domain.name))
2016-03-20 12:09:06 +02:00
return flask.render_template('user/settings.html', form=form, user=user)
2016-03-19 21:37:48 +02:00
def _process_password_change(form, user_email):
user_email_or_current = user_email or flask_login.current_user.email
user = models.User.query.get(user_email_or_current) or flask.abort(404)
2016-03-19 21:37:48 +02:00
if form.validate_on_submit():
if form.pw.data != form.pw2.data:
flask.flash('Passwords do not match', 'error')
2023-04-02 16:41:12 +02:00
elif user_email or models.User.login(user_email_or_current, form.current_pw.data):
2022-11-03 17:19:44 +02:00
if msg := utils.isBadOrPwned(form):
flask.flash(msg, "error")
2022-10-31 20:41:40 +02:00
return flask.render_template('user/password.html', form=form, user=user)
flask.session.regenerate()
2023-08-11 09:03:42 +02:00
user.set_password(form.pw.data, keep_only_session=flask.session)
models.db.session.commit()
2016-03-19 21:37:48 +02:00
flask.flash('Password updated for %s' % user)
if user_email:
return flask.redirect(flask.url_for('.user_list',
domain_name=user.domain.name))
else:
flask.flash('Wrong current password', 'error')
2016-03-19 21:37:48 +02:00
return flask.render_template('user/password.html', form=form, user=user)
@ui.route('/user/password', methods=['GET', 'POST'], defaults={'user_email': None})
@access.owner(models.User, 'user_email')
def user_password_change(user_email):
return _process_password_change(forms.UserPasswordChangeForm(), user_email)
@ui.route('/user/password/<path:user_email>', methods=['GET', 'POST'])
@access.domain_admin(models.User, 'user_email')
def user_password(user_email):
return _process_password_change(forms.UserPasswordForm(), user_email)
2016-03-19 21:37:48 +02:00
@ui.route('/user/reply', methods=['GET', 'POST'], defaults={'user_email': None})
@ui.route('/user/reply/<path:user_email>', methods=['GET', 'POST'])
@access.owner(models.User, 'user_email')
def user_reply(user_email):
user_email_or_current = user_email or flask_login.current_user.email
user = models.User.query.get(user_email_or_current) or flask.abort(404)
2016-03-20 12:14:27 +02:00
form = forms.UserReplyForm(obj=user)
if form.validate_on_submit():
2016-05-01 21:12:08 +02:00
form.populate_obj(user)
models.db.session.commit()
flask.flash('Auto-reply message updated for %s' % user)
if user_email:
return flask.redirect(
flask.url_for('.user_list', domain_name=user.domain.name))
return flask.render_template('user/reply.html', form=form, user=user)
2017-12-03 13:01:25 +02:00
@ui.route('/user/signup', methods=['GET', 'POST'])
@ui.route('/user/signup/<domain_name>', methods=['GET', 'POST'])
def user_signup(domain_name=None):
available_domains = {
domain.name: domain
for domain in models.Domain.query.filter_by(signup_enabled=True).all()
if domain.max_users == -1 or len(domain.users) < domain.max_users
2017-12-03 13:01:25 +02:00
}
if not available_domains:
flask.flash('No domain available for registration')
if not domain_name:
return flask.render_template('user/signup_domain.html',
available_domains=available_domains)
domain = available_domains.get(domain_name) or flask.abort(404)
2017-12-03 18:53:33 +02:00
quota_bytes = domain.max_quota_bytes or app.config['DEFAULT_QUOTA']
if app.config['RECAPTCHA_PUBLIC_KEY'] == "" or app.config['RECAPTCHA_PRIVATE_KEY'] == "":
form = forms.UserSignupForm()
else:
form = forms.UserSignupFormCaptcha()
2017-12-03 13:01:25 +02:00
if form.validate_on_submit():
if domain.has_email(form.localpart.data) or models.Alias.resolve(form.localpart.data, domain_name):
2017-12-03 13:01:25 +02:00
flask.flash('Email is already used', 'error')
else:
2022-11-03 17:19:44 +02:00
if msg := utils.isBadOrPwned(form):
flask.flash(msg, "error")
2022-10-31 20:41:40 +02:00
return flask.render_template('user/signup.html', domain=domain, form=form)
flask.session.regenerate()
2017-12-03 13:01:25 +02:00
user = models.User(domain=domain)
form.populate_obj(user)
2023-08-12 08:07:10 +02:00
user.change_pw_next_login = True
2017-12-03 13:01:25 +02:00
user.set_password(form.pw.data)
user.quota_bytes = quota_bytes
models.db.session.add(user)
models.db.session.commit()
2017-12-03 13:01:25 +02:00
user.send_welcome()
flask.flash('Successfully signed up %s' % user)
return flask.redirect(flask.url_for('.index'))
return flask.render_template('user/signup.html', domain=domain, form=form)