2011-07-31 23:55:18 +02:00
|
|
|
from collections import defaultdict
|
|
|
|
|
2011-08-19 23:44:54 +02:00
|
|
|
from flask import *
|
2011-07-30 15:46:53 +02:00
|
|
|
from flaskext.mail import Mail, Message
|
2011-07-23 18:45:40 +02:00
|
|
|
|
|
|
|
# local modules
|
|
|
|
from models import db, Project, Person, Bill
|
2011-09-11 05:25:42 +02:00
|
|
|
from forms import (get_billform_for, ProjectForm, AuthenticationForm, BillForm,
|
|
|
|
MemberForm, InviteForm, CreateArchiveForm)
|
|
|
|
from utils import Redirect303
|
2011-07-23 18:45:40 +02:00
|
|
|
|
2011-09-09 21:21:37 +02:00
|
|
|
"""
|
|
|
|
The blueprint for the web interface.
|
2011-07-23 18:45:40 +02:00
|
|
|
|
2011-09-09 21:21:37 +02:00
|
|
|
Contains all the interaction logic with the end user (except forms which
|
|
|
|
are directly handled in the forms module.
|
2011-08-21 21:21:52 +02:00
|
|
|
|
2011-09-09 21:21:37 +02:00
|
|
|
Basically, this blueprint takes care of the authentication and provides
|
|
|
|
some shortcuts to make your life better when coding (see `pull_project`
|
|
|
|
and `add_project_id` for a quick overview
|
|
|
|
"""
|
2011-08-21 21:21:52 +02:00
|
|
|
|
2011-09-09 21:21:37 +02:00
|
|
|
main = Blueprint("main", __name__)
|
|
|
|
mail = Mail()
|
2011-08-21 21:21:52 +02:00
|
|
|
|
2011-09-09 21:21:37 +02:00
|
|
|
@main.url_defaults
|
2011-08-21 01:42:10 +02:00
|
|
|
def add_project_id(endpoint, values):
|
2011-09-09 19:57:28 +02:00
|
|
|
"""Add the project id to the url calls if it is expected.
|
|
|
|
|
|
|
|
This is to not carry it everywhere in the templates.
|
|
|
|
"""
|
2011-08-21 01:42:10 +02:00
|
|
|
if 'project_id' in values or not hasattr(g, 'project'):
|
|
|
|
return
|
2011-09-09 21:21:37 +02:00
|
|
|
if current_app.url_map.is_endpoint_expecting(endpoint, 'project_id'):
|
2011-08-21 01:42:10 +02:00
|
|
|
values['project_id'] = g.project.id
|
|
|
|
|
2011-09-09 21:21:37 +02:00
|
|
|
@main.url_value_preprocessor
|
2011-08-21 01:42:10 +02:00
|
|
|
def pull_project(endpoint, values):
|
2011-09-09 19:57:28 +02:00
|
|
|
"""When a request contains a project_id value, transform it directly
|
|
|
|
into a project by checking the credentials are stored in session.
|
|
|
|
|
|
|
|
If not, redirect the user to an authentication form
|
|
|
|
"""
|
2011-08-21 02:23:53 +02:00
|
|
|
if endpoint == "authenticate":
|
|
|
|
return
|
2011-08-21 01:42:10 +02:00
|
|
|
if not values:
|
|
|
|
values = {}
|
|
|
|
project_id = values.pop('project_id', None)
|
|
|
|
if project_id:
|
|
|
|
project = Project.query.get(project_id)
|
|
|
|
if not project:
|
2011-09-09 21:21:37 +02:00
|
|
|
raise Redirect303(url_for(".create_project", project_id=project_id))
|
2011-08-21 01:42:10 +02:00
|
|
|
if project.id in session and session[project.id] == project.password:
|
|
|
|
# add project into kwargs and call the original function
|
|
|
|
g.project = project
|
|
|
|
else:
|
|
|
|
# redirect to authentication page
|
2011-08-21 22:35:01 +02:00
|
|
|
raise Redirect303(
|
2011-09-09 21:21:37 +02:00
|
|
|
url_for(".authenticate", project_id=project_id))
|
2011-07-29 15:44:15 +02:00
|
|
|
|
2011-09-09 21:21:37 +02:00
|
|
|
@main.route("/authenticate", methods=["GET", "POST"])
|
2011-08-21 22:20:50 +02:00
|
|
|
def authenticate(project_id=None):
|
2011-09-09 19:57:28 +02:00
|
|
|
"""Authentication form"""
|
2011-07-23 18:45:40 +02:00
|
|
|
form = AuthenticationForm()
|
2011-08-21 22:20:50 +02:00
|
|
|
if not form.id.data and request.args['project_id']:
|
|
|
|
form.id.data = request.args['project_id']
|
|
|
|
project_id = form.id.data
|
2011-07-30 01:32:55 +02:00
|
|
|
project = Project.query.get(project_id)
|
2011-08-09 17:29:44 +02:00
|
|
|
create_project = False # We don't want to create the project by default
|
2011-07-30 01:32:55 +02:00
|
|
|
if not project:
|
2011-08-09 17:29:44 +02:00
|
|
|
# But if the user try to connect to an unexisting project, we will
|
|
|
|
# propose him a link to the creation form.
|
|
|
|
create_project = project_id
|
2011-07-30 01:32:55 +02:00
|
|
|
|
2011-08-09 19:28:50 +02:00
|
|
|
else:
|
|
|
|
# if credentials are already in session, redirect
|
|
|
|
if project_id in session and project.password == session[project_id]:
|
2011-08-20 00:28:58 +02:00
|
|
|
setattr(g, 'project', project)
|
2011-09-09 21:21:37 +02:00
|
|
|
return redirect(url_for(".list_bills"))
|
2011-08-09 19:28:50 +02:00
|
|
|
|
|
|
|
# else process the form
|
|
|
|
if request.method == "POST":
|
|
|
|
if form.validate():
|
|
|
|
if not form.password.data == project.password:
|
|
|
|
form.errors['password'] = ["The password is not the right one"]
|
|
|
|
else:
|
|
|
|
# maintain a list of visited projects
|
|
|
|
if "projects" not in session:
|
|
|
|
session["projects"] = []
|
|
|
|
# add the project on the top of the list
|
|
|
|
session["projects"].insert(0, (project_id, project.name))
|
|
|
|
session[project_id] = form.password.data
|
|
|
|
session.update()
|
2011-08-19 23:44:54 +02:00
|
|
|
setattr(g, 'project', project)
|
2011-09-09 21:21:37 +02:00
|
|
|
return redirect(url_for(".list_bills"))
|
2011-07-29 15:44:15 +02:00
|
|
|
|
2011-08-09 17:29:44 +02:00
|
|
|
return render_template("authenticate.html", form=form,
|
|
|
|
create_project=create_project)
|
2011-07-23 19:11:24 +02:00
|
|
|
|
2011-09-09 21:21:37 +02:00
|
|
|
@main.route("/")
|
2011-08-21 01:42:10 +02:00
|
|
|
def home():
|
|
|
|
project_form = ProjectForm()
|
|
|
|
auth_form = AuthenticationForm()
|
|
|
|
return render_template("home.html", project_form=project_form,
|
|
|
|
auth_form=auth_form, session=session)
|
|
|
|
|
2011-09-09 21:21:37 +02:00
|
|
|
@main.route("/create", methods=["GET", "POST"])
|
2011-07-23 19:11:24 +02:00
|
|
|
def create_project():
|
2011-07-26 16:03:00 +02:00
|
|
|
form = ProjectForm()
|
2011-07-23 19:11:24 +02:00
|
|
|
if request.method == "GET" and 'project_id' in request.values:
|
|
|
|
form.name.data = request.values['project_id']
|
|
|
|
|
|
|
|
if request.method == "POST":
|
2011-09-11 05:25:42 +02:00
|
|
|
# At first, we don't want the user to bother with the identifier
|
|
|
|
# so it will automatically be missing because not displayed into the form
|
|
|
|
# Thus we fill it with the same value as the filled name, the validation will
|
|
|
|
# take care of the slug
|
|
|
|
if not form.id.data:
|
|
|
|
form.id.data = form.name.data
|
2011-07-23 19:11:24 +02:00
|
|
|
if form.validate():
|
|
|
|
# save the object in the db
|
|
|
|
project = form.save()
|
|
|
|
db.session.add(project)
|
|
|
|
db.session.commit()
|
|
|
|
|
|
|
|
# create the session object (authenticate)
|
|
|
|
session[project.id] = project.password
|
|
|
|
session.update()
|
|
|
|
|
|
|
|
# redirect the user to the next step (invite)
|
2011-09-09 21:21:37 +02:00
|
|
|
return redirect(url_for(".invite", project_id=project.id))
|
2011-07-23 19:11:24 +02:00
|
|
|
|
|
|
|
return render_template("create_project.html", form=form)
|
|
|
|
|
2011-09-09 21:21:37 +02:00
|
|
|
@main.route("/exit")
|
2011-08-09 23:48:13 +02:00
|
|
|
def exit():
|
2011-07-29 15:44:15 +02:00
|
|
|
# delete the session
|
2011-07-30 01:32:55 +02:00
|
|
|
session.clear()
|
2011-09-09 21:21:37 +02:00
|
|
|
return redirect(url_for(".home"))
|
2011-07-29 15:44:15 +02:00
|
|
|
|
2011-09-09 21:21:37 +02:00
|
|
|
@main.route("/demo")
|
2011-08-21 03:27:59 +02:00
|
|
|
def demo():
|
2011-09-09 19:57:28 +02:00
|
|
|
"""
|
|
|
|
Authenticate the user for the demonstration project and redirect him to
|
|
|
|
the bills list for this project.
|
|
|
|
|
|
|
|
Create a demo project if it doesnt exists yet (or has been deleted)
|
|
|
|
"""
|
2011-08-21 03:27:59 +02:00
|
|
|
project = Project.query.get("demo")
|
2011-08-21 20:54:20 +02:00
|
|
|
if not project:
|
|
|
|
project = Project(id="demo", name=u"demonstration", password="demo",
|
|
|
|
contact_email="demo@notmyidea.org")
|
|
|
|
db.session.add(project)
|
|
|
|
db.session.commit()
|
2011-08-21 03:27:59 +02:00
|
|
|
session[project.id] = project.password
|
2011-09-09 21:21:37 +02:00
|
|
|
return redirect(url_for(".list_bills", project_id=project.id))
|
2011-08-21 03:27:59 +02:00
|
|
|
|
2011-09-09 21:21:37 +02:00
|
|
|
@main.route("/<project_id>/invite", methods=["GET", "POST"])
|
2011-08-19 23:44:54 +02:00
|
|
|
def invite():
|
2011-09-09 19:57:28 +02:00
|
|
|
"""Send invitations for this particular project"""
|
2011-07-30 15:46:53 +02:00
|
|
|
|
|
|
|
form = InviteForm()
|
|
|
|
|
|
|
|
if request.method == "POST":
|
|
|
|
if form.validate():
|
|
|
|
# send the email
|
|
|
|
|
2011-08-19 23:44:54 +02:00
|
|
|
message_body = render_template("invitation_mail")
|
2011-07-30 15:46:53 +02:00
|
|
|
|
2011-08-09 23:48:13 +02:00
|
|
|
message_title = "You have been invited to share your"\
|
2011-08-19 23:44:54 +02:00
|
|
|
+ " expenses for %s" % g.project.name
|
2011-07-30 15:46:53 +02:00
|
|
|
msg = Message(message_title,
|
|
|
|
body=message_body,
|
2011-08-09 23:48:13 +02:00
|
|
|
recipients=[email.strip()
|
|
|
|
for email in form.emails.data.split(",")])
|
2011-07-30 15:46:53 +02:00
|
|
|
mail.send(msg)
|
2011-08-21 21:55:47 +02:00
|
|
|
flash("You invitations have been sent")
|
2011-09-09 21:21:37 +02:00
|
|
|
return redirect(url_for(".list_bills"))
|
2011-07-30 15:46:53 +02:00
|
|
|
|
2011-08-19 23:44:54 +02:00
|
|
|
return render_template("send_invites.html", form=form)
|
2011-07-23 19:11:24 +02:00
|
|
|
|
2011-09-09 21:21:37 +02:00
|
|
|
@main.route("/<project_id>/")
|
2011-08-19 23:44:54 +02:00
|
|
|
def list_bills():
|
2011-09-09 19:57:28 +02:00
|
|
|
bills = g.project.get_bills()
|
2011-07-23 19:11:24 +02:00
|
|
|
return render_template("list_bills.html",
|
2011-08-19 23:44:54 +02:00
|
|
|
bills=bills, member_form=MemberForm(g.project),
|
2011-09-11 05:25:42 +02:00
|
|
|
bill_form=get_billform_for(request, g.project)
|
2011-07-31 23:55:18 +02:00
|
|
|
)
|
2011-07-23 20:36:13 +02:00
|
|
|
|
2011-09-09 21:21:37 +02:00
|
|
|
@main.route("/<project_id>/members/add", methods=["GET", "POST"])
|
2011-08-19 23:44:54 +02:00
|
|
|
def add_member():
|
2011-07-23 20:36:13 +02:00
|
|
|
# FIXME manage form errors on the list_bills page
|
2011-08-19 23:44:54 +02:00
|
|
|
form = MemberForm(g.project)
|
2011-07-23 20:36:13 +02:00
|
|
|
if request.method == "POST":
|
|
|
|
if form.validate():
|
2011-08-22 23:19:00 +02:00
|
|
|
# if the user is already bound to the project, just reactivate him
|
|
|
|
person = Person.query.filter(Person.name == form.name.data)\
|
|
|
|
.filter(Project.id == g.project.id).all()
|
|
|
|
if person:
|
|
|
|
person[0].activated = True
|
|
|
|
db.session.commit()
|
2011-08-22 23:29:10 +02:00
|
|
|
flash("%s is part of this project again" % person[0].name)
|
2011-09-09 21:21:37 +02:00
|
|
|
return redirect(url_for(".list_bills"))
|
2011-08-22 23:19:00 +02:00
|
|
|
|
2011-08-19 23:44:54 +02:00
|
|
|
db.session.add(Person(name=form.name.data, project=g.project))
|
2011-07-23 20:36:13 +02:00
|
|
|
db.session.commit()
|
2011-09-09 21:21:37 +02:00
|
|
|
return redirect(url_for(".list_bills"))
|
2011-08-19 23:44:54 +02:00
|
|
|
return render_template("add_member.html", form=form)
|
2011-07-23 18:45:40 +02:00
|
|
|
|
2011-09-09 21:21:37 +02:00
|
|
|
@main.route("/<project_id>/members/<member_id>/reactivate", methods=["GET",])
|
2011-08-25 18:25:58 +02:00
|
|
|
def reactivate(member_id):
|
|
|
|
person = Person.query.filter(Person.id == member_id)\
|
|
|
|
.filter(Project.id == g.project.id).all()
|
|
|
|
if person:
|
|
|
|
person[0].activated = True
|
|
|
|
db.session.commit()
|
|
|
|
flash("%s is part of this project again" % person[0].name)
|
2011-09-09 21:21:37 +02:00
|
|
|
return redirect(url_for(".list_bills"))
|
2011-08-25 18:25:58 +02:00
|
|
|
|
|
|
|
|
2011-09-09 21:21:37 +02:00
|
|
|
@main.route("/<project_id>/members/<member_id>/delete", methods=["GET", "POST"])
|
2011-08-19 23:44:54 +02:00
|
|
|
def remove_member(member_id):
|
2011-09-09 19:57:28 +02:00
|
|
|
member = g.project.remove_member(member_id)
|
|
|
|
if member.activated == False:
|
|
|
|
flash("User '%s' has been desactivated" % member.name)
|
|
|
|
else:
|
|
|
|
flash("User '%s' has been removed" % member.name)
|
|
|
|
|
2011-09-09 21:21:37 +02:00
|
|
|
return redirect(url_for(".list_bills"))
|
2011-07-31 23:55:18 +02:00
|
|
|
|
2011-09-09 21:21:37 +02:00
|
|
|
@main.route("/<project_id>/add", methods=["GET", "POST"])
|
2011-08-19 23:44:54 +02:00
|
|
|
def add_bill():
|
2011-09-11 05:25:42 +02:00
|
|
|
form = get_billform_for(request, g.project)
|
2011-07-23 18:45:40 +02:00
|
|
|
if request.method == 'POST':
|
|
|
|
if form.validate():
|
2011-08-10 12:59:30 +02:00
|
|
|
bill = Bill()
|
|
|
|
db.session.add(form.save(bill))
|
2011-07-23 18:45:40 +02:00
|
|
|
db.session.commit()
|
2011-07-31 00:41:28 +02:00
|
|
|
|
2011-08-10 01:05:25 +02:00
|
|
|
flash("The bill has been added")
|
2011-09-09 21:21:37 +02:00
|
|
|
return redirect(url_for('.list_bills'))
|
2011-07-23 18:45:40 +02:00
|
|
|
|
2011-08-19 23:44:54 +02:00
|
|
|
return render_template("add_bill.html", form=form)
|
2011-07-23 18:45:40 +02:00
|
|
|
|
|
|
|
|
2011-09-09 21:21:37 +02:00
|
|
|
@main.route("/<project_id>/delete/<int:bill_id>")
|
2011-08-19 23:44:54 +02:00
|
|
|
def delete_bill(bill_id):
|
2011-08-10 12:59:30 +02:00
|
|
|
bill = Bill.query.get_or_404(bill_id)
|
|
|
|
db.session.delete(bill)
|
|
|
|
db.session.commit()
|
|
|
|
flash("The bill has been deleted")
|
|
|
|
|
2011-09-09 21:21:37 +02:00
|
|
|
return redirect(url_for('.list_bills'))
|
2011-08-10 12:59:30 +02:00
|
|
|
|
|
|
|
|
2011-09-09 21:21:37 +02:00
|
|
|
@main.route("/<project_id>/edit/<int:bill_id>", methods=["GET", "POST"])
|
2011-08-19 23:44:54 +02:00
|
|
|
def edit_bill(bill_id):
|
2011-08-10 12:59:30 +02:00
|
|
|
bill = Bill.query.get_or_404(bill_id)
|
2011-09-11 05:25:42 +02:00
|
|
|
form = get_billform_for(request, g.project, set_default=False)
|
2011-08-10 12:59:30 +02:00
|
|
|
if request.method == 'POST' and form.validate():
|
|
|
|
form.save(bill)
|
|
|
|
db.session.commit()
|
|
|
|
|
|
|
|
flash("The bill has been modified")
|
2011-09-09 21:21:37 +02:00
|
|
|
return redirect(url_for('.list_bills'))
|
2011-08-10 12:59:30 +02:00
|
|
|
|
|
|
|
form.fill(bill)
|
2011-08-21 01:42:10 +02:00
|
|
|
return render_template("add_bill.html", form=form, edit=True)
|
2011-08-10 12:59:30 +02:00
|
|
|
|
2011-09-09 21:21:37 +02:00
|
|
|
@main.route("/<project_id>/compute")
|
2011-08-19 23:44:54 +02:00
|
|
|
def compute_bills():
|
2011-07-23 18:45:40 +02:00
|
|
|
"""Compute the sum each one have to pay to each other and display it"""
|
2011-08-19 23:44:54 +02:00
|
|
|
return render_template("compute_bills.html")
|
2011-07-23 18:45:40 +02:00
|
|
|
|
2011-09-09 21:21:37 +02:00
|
|
|
@main.route("/<project_id>/archives/create")
|
2011-09-09 19:14:19 +02:00
|
|
|
def create_archive():
|
|
|
|
form = CreateArchiveForm()
|
|
|
|
if request.method == "POST":
|
|
|
|
if form.validate():
|
|
|
|
pass
|
|
|
|
flash("The data from XX to XX has been archived")
|
2011-07-23 18:45:40 +02:00
|
|
|
|
2011-09-09 19:14:19 +02:00
|
|
|
return render_template("create_archive.html", form=form)
|