First shot at an AdminLTE dashboard
This commit is contained in:
1
admin/freeposte/views/__init__.py
Normal file
1
admin/freeposte/views/__init__.py
Normal file
@@ -0,0 +1 @@
|
||||
from freeposte.views import base, account, admin, domains, users, aliases
|
||||
18
admin/freeposte/views/admin.py
Normal file
18
admin/freeposte/views/admin.py
Normal file
@@ -0,0 +1,18 @@
|
||||
from freeposte import app, db, models, forms, utils
|
||||
from flask.ext import login as flask_login
|
||||
|
||||
import os
|
||||
import flask
|
||||
|
||||
|
||||
@app.route('/status', methods=['GET'])
|
||||
@flask_login.login_required
|
||||
def status():
|
||||
utils.require_global_admin()
|
||||
return flask.render_template('admin/status.html')
|
||||
|
||||
|
||||
@app.route('/admins', methods=['GET'])
|
||||
@flask_login.login_required
|
||||
def admins():
|
||||
return flask.render_template('admin/admins.html')
|
||||
64
admin/freeposte/views/aliases.py
Normal file
64
admin/freeposte/views/aliases.py
Normal file
@@ -0,0 +1,64 @@
|
||||
from freeposte import app, db, models, forms, utils
|
||||
from flask.ext import login as flask_login
|
||||
|
||||
import os
|
||||
import flask
|
||||
|
||||
|
||||
@app.route('/alias/list/<domain_name>', methods=['GET'])
|
||||
@flask_login.login_required
|
||||
def alias_list(domain_name):
|
||||
domain = utils.get_domain_admin(domain_name)
|
||||
return flask.render_template('alias/list.html', domain=domain)
|
||||
|
||||
|
||||
@app.route('/alias/create/<domain_name>', methods=['GET', 'POST'])
|
||||
@flask_login.login_required
|
||||
def alias_create(domain_name):
|
||||
domain = utils.get_domain_admin(domain_name)
|
||||
if len(domain.aliases) >= domain.max_aliases:
|
||||
flask.flash('Too many aliases for domain %s' % domain, 'error')
|
||||
return flask.redirect(flask.url_for('alias_list', domain_name=domain.name))
|
||||
form = forms.AliasCreateForm()
|
||||
if form.validate_on_submit():
|
||||
for address in domain.users + domain.aliases:
|
||||
if address.localpart == form.localpart.data:
|
||||
flask.flash('Address %s is already used' % address, 'error')
|
||||
break
|
||||
else:
|
||||
alias = models.Alias(
|
||||
localpart=form.localpart.data, domain=domain,
|
||||
destination=form.destination.data
|
||||
)
|
||||
db.session.add(alias)
|
||||
db.session.commit()
|
||||
flask.flash('Alias %s created' % alias)
|
||||
return flask.redirect(
|
||||
flask.url_for('alias_list', domain_name=domain.name))
|
||||
return flask.render_template('alias/create.html',
|
||||
domain=domain, form=form)
|
||||
|
||||
|
||||
@app.route('/alias/edit/<alias>', methods=['GET', 'POST'])
|
||||
@flask_login.login_required
|
||||
def alias_edit(alias):
|
||||
alias = utils.get_alias(alias)
|
||||
form = forms.AliasEditForm()
|
||||
if form.validate_on_submit():
|
||||
alias.destination = form.destination.data
|
||||
db.session.add(alias)
|
||||
db.session.commit()
|
||||
flask.flash('Alias %s updated' % alias)
|
||||
return flask.redirect(
|
||||
flask.url_for('alias_list', domain_name=alias.domain.name))
|
||||
return flask.render_template('alias/edit.html', form=form, alias=alias)
|
||||
|
||||
|
||||
@app.route('/alias/delete/<alias>', methods=['GET'])
|
||||
@flask_login.login_required
|
||||
def alias_delete(alias):
|
||||
alias = utils.get_alias(alias)
|
||||
db.session.delete(alias)
|
||||
db.session.commit()
|
||||
flask.flash('Alias %s deleted' % alias)
|
||||
return flask.redirect(flask.url_for('alias_list', domain_name=alias.domain.name))
|
||||
31
admin/freeposte/views/base.py
Normal file
31
admin/freeposte/views/base.py
Normal file
@@ -0,0 +1,31 @@
|
||||
from freeposte import app, db, models, forms
|
||||
from flask.ext import login as flask_login
|
||||
|
||||
import os
|
||||
import flask
|
||||
|
||||
|
||||
@app.route('/', methods=["GET"])
|
||||
@flask_login.login_required
|
||||
def index():
|
||||
return flask.redirect(flask.url_for('user_settings'))
|
||||
|
||||
|
||||
@app.route('/login', methods=['GET', 'POST'])
|
||||
def login():
|
||||
form = forms.LoginForm()
|
||||
if form.validate_on_submit():
|
||||
user = models.User.login(form.email.data, form.pw.data)
|
||||
if user:
|
||||
flask_login.login_user(user)
|
||||
return flask.redirect(flask.url_for('index'))
|
||||
else:
|
||||
flask.flash('Wrong e-mail address or password', 'error')
|
||||
return flask.render_template('login.html', form=form)
|
||||
|
||||
|
||||
@app.route('/logout', methods=['GET'])
|
||||
@flask_login.login_required
|
||||
def logout():
|
||||
flask_login.logout_user()
|
||||
return flask.redirect(flask.url_for('index'))
|
||||
62
admin/freeposte/views/domains.py
Normal file
62
admin/freeposte/views/domains.py
Normal file
@@ -0,0 +1,62 @@
|
||||
from freeposte import app, db, models, forms, utils
|
||||
from flask.ext import login as flask_login
|
||||
|
||||
import os
|
||||
import flask
|
||||
|
||||
|
||||
@app.route('/domain', methods=['GET'])
|
||||
@flask_login.login_required
|
||||
def domain_list():
|
||||
return flask.render_template('domain/list.html')
|
||||
|
||||
|
||||
@app.route('/domain/create', methods=['GET', 'POST'])
|
||||
@flask_login.login_required
|
||||
def domain_create():
|
||||
utils.require_global_admin()
|
||||
form = forms.DomainCreateForm()
|
||||
if form.validate_on_submit():
|
||||
if models.Domain.query.filter_by(name=form.name.data).first():
|
||||
flask.flash('Domain %s is already used' % form.name.data, 'error')
|
||||
else:
|
||||
domain = models.Domain(name=form.name.data)
|
||||
db.session.add(domain)
|
||||
db.session.commit()
|
||||
flask.flash('Domain %s created' % domain)
|
||||
return flask.redirect(flask.url_for('domain_list'))
|
||||
return flask.render_template('domain/create.html', form=form)
|
||||
|
||||
|
||||
@app.route('/domain/edit/<domain_name>', methods=['GET', 'POST'])
|
||||
@flask_login.login_required
|
||||
def domain_edit(domain_name):
|
||||
utils.require_global_admin()
|
||||
domain = utils.get_domain_admin(domain_name)
|
||||
form = forms.DomainEditForm()
|
||||
if form.validate_on_submit():
|
||||
domain.max_users = form.max_users.data
|
||||
domain.max_aliases = form.max_aliases.data
|
||||
db.session.add(domain)
|
||||
db.session.commit()
|
||||
flask.flash('Domain %s saved' % domain)
|
||||
return flask.redirect(flask.url_for('domain_list'))
|
||||
return flask.render_template('domain/edit.html', form=form,
|
||||
domain=domain)
|
||||
|
||||
|
||||
@app.route('/domain/delete/<domain_name>', methods=['GET'])
|
||||
@flask_login.login_required
|
||||
def domain_delete(domain_name):
|
||||
utils.require_global_admin()
|
||||
domain = utils.get_domain_admin(domain_name)
|
||||
db.session.delete(domain)
|
||||
db.session.commit()
|
||||
flask.flash('Domain %s deleted' % domain)
|
||||
return flask.redirect(flask.url_for('domain_list'))
|
||||
|
||||
|
||||
@app.route('/domain/admins/<domain_name>', methods=['GET'])
|
||||
@flask_login.login_required
|
||||
def domain_admins(domain_name):
|
||||
domain = utils.get_domain_admin(domain_name)
|
||||
120
admin/freeposte/views/users.py
Normal file
120
admin/freeposte/views/users.py
Normal file
@@ -0,0 +1,120 @@
|
||||
from freeposte import app, db, models, forms, utils
|
||||
from flask.ext import login as flask_login
|
||||
|
||||
import os
|
||||
import flask
|
||||
|
||||
|
||||
@app.route('/user/list/<domain_name>', methods=['GET'])
|
||||
@flask_login.login_required
|
||||
def user_list(domain_name):
|
||||
domain = utils.get_domain_admin(domain_name)
|
||||
return flask.render_template('user/list.html', domain=domain)
|
||||
|
||||
|
||||
@app.route('/user/create/<domain_name>', methods=['GET', 'POST'])
|
||||
@flask_login.login_required
|
||||
def user_create(domain_name):
|
||||
domain = utils.get_domain_admin(domain_name)
|
||||
if len(domain.users) >= domain.max_users:
|
||||
flask.flash('Too many users for domain %s' % domain, 'error')
|
||||
return flask.redirect(flask.url_for('user_list', domain_name=domain.name))
|
||||
form = forms.UserCreateForm()
|
||||
if form.validate_on_submit():
|
||||
for address in domain.users + domain.aliases:
|
||||
if address.localpart == form.localpart.data:
|
||||
flask.flash('Address %s is already used' % address, 'error')
|
||||
break
|
||||
else:
|
||||
user = models.User(localpart=form.localpart.data, domain=domain)
|
||||
user.set_password(form.pw.data)
|
||||
db.session.add(user)
|
||||
db.session.commit()
|
||||
flask.flash('User %s created' % user)
|
||||
return flask.redirect(
|
||||
flask.url_for('user_list', domain_name=domain.name))
|
||||
return flask.render_template('user/create.html',
|
||||
domain=domain, form=form)
|
||||
|
||||
|
||||
@app.route('/user/edit/<user_email>', methods=['GET', 'POST'])
|
||||
@flask_login.login_required
|
||||
def user_edit(user_email):
|
||||
user = utils.get_user(user_email, True)
|
||||
form = forms.UserEditForm()
|
||||
if form.validate_on_submit():
|
||||
user.quota_bytes = form.quota.data
|
||||
db.session.add(user)
|
||||
db.session.commit()
|
||||
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)
|
||||
|
||||
|
||||
@app.route('/user/delete/<user_email>', methods=['GET'])
|
||||
@flask_login.login_required
|
||||
def user_delete(user_email):
|
||||
user = utils.get_user(user_email, True)
|
||||
db.session.delete(user)
|
||||
db.session.commit()
|
||||
flask.flash('User %s deleted' % user)
|
||||
return flask.redirect(flask.url_for('user_list', domain_name=user.domain.name))
|
||||
|
||||
|
||||
@app.route('/user/settings', methods=['GET', 'POST'], defaults={'user_email': None})
|
||||
@app.route('/user/usersettings/<user_email>', methods=['GET', 'POST'])
|
||||
@flask_login.login_required
|
||||
def user_settings(user_email):
|
||||
return flask.render_template('user/settings.html')
|
||||
|
||||
|
||||
@app.route('/user/password', methods=['GET', 'POST'], defaults={'user_email': None})
|
||||
@app.route('/user/password/<user_email>', methods=['GET', 'POST'])
|
||||
@flask_login.login_required
|
||||
def user_password(user_email):
|
||||
user = utils.get_user(user_email)
|
||||
form = forms.UserPasswordForm()
|
||||
if form.validate_on_submit():
|
||||
if form.pw.data != form.pw2.data:
|
||||
flask.flash('Passwords do not match', 'error')
|
||||
else:
|
||||
user.set_password(form.pw.data)
|
||||
db.session.add(user)
|
||||
db.session.commit()
|
||||
flask.flash('Password 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/password.html', form=form, user=user)
|
||||
|
||||
|
||||
@app.route('/user/forward', methods=['GET', 'POST'], defaults={'user_email': None})
|
||||
@app.route('/user/forward/<user_email>', methods=['GET', 'POST'])
|
||||
@flask_login.login_required
|
||||
def user_forward(user_email):
|
||||
user = utils.get_user(user_email)
|
||||
form = forms.UserForwardForm()
|
||||
if form.validate_on_submit():
|
||||
user.forward = form.forward.data
|
||||
db.session.add(user)
|
||||
db.session.commit()
|
||||
flask.flash('Forward destination 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/forward.html', form=form, user=useré)
|
||||
|
||||
|
||||
@app.route('/user/vacation', methods=['GET', 'POST'], defaults={'user_email': None})
|
||||
@app.route('/user/vacation<user_email>', methods=['GET', 'POST'])
|
||||
@flask_login.login_required
|
||||
def user_vacation(user_email):
|
||||
return flask.render_template('user/vacation.html')
|
||||
|
||||
|
||||
@app.route('/user/fetchmail', methods=['GET', 'POST'], defaults={'user_email': None})
|
||||
@app.route('/user/fetchmail/<user_email>', methods=['GET', 'POST'])
|
||||
@flask_login.login_required
|
||||
def user_fetchmail(user_email):
|
||||
return flask.render_template('user/fetchmail.html')
|
||||
Reference in New Issue
Block a user