Files
notifications-admin/app/main/views/organizations.py

367 lines
12 KiB
Python

from collections import OrderedDict
from datetime import datetime
from functools import partial
from flask import flash, redirect, render_template, request, url_for
from flask_login import current_user
from notifications_python_client.errors import HTTPError
from app import current_organization, org_invite_api_client, organizations_client
from app.main import main
from app.main.forms import (
AdminBillingDetailsForm,
AdminNewOrganizationForm,
AdminNotesForm,
AdminOrganizationDomainsForm,
InviteOrgUserForm,
OrganizationOrganizationTypeForm,
RenameOrganizationForm,
SearchByNameForm,
SearchUsersForm,
)
from app.main.views.dashboard import (
get_tuples_of_financial_years,
requested_and_current_financial_year,
)
from app.models.organization import AllOrganizations, Organization
from app.models.user import InvitedOrgUser, User
from app.utils.csv import Spreadsheet
from app.utils.user import user_has_permissions, user_is_platform_admin
@main.route("/organizations", methods=["GET"])
@user_is_platform_admin
def organizations():
return render_template(
"views/organizations/index.html",
organizations=AllOrganizations(),
search_form=SearchByNameForm(),
)
@main.route("/organizations/add", methods=["GET", "POST"])
@user_is_platform_admin
def add_organization():
form = AdminNewOrganizationForm()
if form.validate_on_submit():
try:
return redirect(
url_for(
".organization_settings",
org_id=Organization.create_from_form(form).id,
)
)
except HTTPError as e:
msg = "Organization name already exists"
if e.status_code == 400 and msg in e.message:
form.name.errors.append("This organization name is already in use")
else:
raise e
return render_template("views/organizations/add-organization.html", form=form)
@main.route("/organizations/<uuid:org_id>", methods=["GET"])
@user_has_permissions()
def organization_dashboard(org_id):
year, current_financial_year = requested_and_current_financial_year(request)
services = current_organization.services_and_usage(financial_year=year)["services"]
return render_template(
"views/organizations/organization/index.html",
services=services,
years=get_tuples_of_financial_years(
partial(url_for, ".organization_dashboard", org_id=current_organization.id),
start=current_financial_year - 2,
end=current_financial_year,
),
selected_year=year,
search_form=SearchByNameForm() if len(services) > 7 else None,
**{
f"total_{key}": sum(service[key] for service in services)
for key in ("emails_sent", "sms_cost")
},
download_link=url_for(
".download_organization_usage_report", org_id=org_id, selected_year=year
),
)
@main.route("/organizations/<uuid:org_id>/download-usage-report.csv", methods=["GET"])
@user_has_permissions()
def download_organization_usage_report(org_id):
selected_year = request.args.get("selected_year")
services_usage = current_organization.services_and_usage(
financial_year=selected_year
)["services"]
unit_column_names = OrderedDict(
[
("service_id", "Service ID"),
("service_name", "Service Name"),
("emails_sent", "Emails sent"),
("sms_remainder", "Free text message allowance remaining"),
]
)
monetary_column_names = OrderedDict(
[
("sms_cost", "Spent on text messages ($)"),
]
)
org_usage_data = [
list(unit_column_names.values()) + list(monetary_column_names.values())
] + [
[service[attribute] for attribute in unit_column_names.keys()]
+ [
"{:,.2f}".format(service[attribute])
for attribute in monetary_column_names.keys()
]
for service in services_usage
]
return (
Spreadsheet.from_rows(org_usage_data).as_csv_data,
200,
{
"Content-Type": "text/csv; charset=utf-8",
"Content-Disposition": (
"inline;"
'filename="{} organization usage report for year {}'
' - generated on {}.csv"'.format(
current_organization.name,
selected_year,
datetime.now().strftime("%Y-%m-%dT%H:%M:%S.%fZ"),
)
),
},
)
@main.route("/organizations/<uuid:org_id>/trial-services", methods=["GET"])
@user_is_platform_admin
def organization_trial_mode_services(org_id):
return render_template(
"views/organizations/organization/trial-mode-services.html",
search_form=SearchByNameForm(),
)
@main.route("/organizations/<uuid:org_id>/users", methods=["GET"])
@user_has_permissions()
def manage_org_users(org_id):
return render_template(
"views/organizations/organization/users/index.html",
users=current_organization.team_members,
show_search_box=(len(current_organization.team_members) > 7),
form=SearchUsersForm(),
)
@main.route("/organizations/<uuid:org_id>/users/invite", methods=["GET", "POST"])
@user_has_permissions()
def invite_org_user(org_id):
form = InviteOrgUserForm(inviter_email_address=current_user.email_address)
if form.validate_on_submit():
email_address = form.email_address.data
invited_org_user = InvitedOrgUser.create(current_user.id, org_id, email_address)
flash(
"Invite sent to {}".format(invited_org_user.email_address),
"default_with_tick",
)
return redirect(url_for(".manage_org_users", org_id=org_id))
return render_template(
"views/organizations/organization/users/invite-org-user.html", form=form
)
@main.route("/organizations/<uuid:org_id>/users/<uuid:user_id>", methods=["GET"])
@user_has_permissions()
def edit_organization_user(org_id, user_id):
# The only action that can be done to an org user is to remove them from the org.
# This endpoint is used to get the ID of the user to delete without passing it as a
# query string, but it uses the template for all org team members in order to avoid
# having a page containing a single link.
return render_template(
"views/organizations/organization/users/index.html",
users=current_organization.team_members,
show_search_box=(len(current_organization.team_members) > 7),
form=SearchUsersForm(),
user_to_remove=User.from_id(user_id),
)
@main.route(
"/organizations/<uuid:org_id>/users/<uuid:user_id>/delete", methods=["POST"]
)
@user_has_permissions()
def remove_user_from_organization(org_id, user_id):
organizations_client.remove_user_from_organization(org_id, user_id)
return redirect(url_for(".show_accounts_or_dashboard"))
@main.route(
"/organizations/<uuid:org_id>/cancel-invited-user/<uuid:invited_user_id>",
methods=["GET"],
)
@user_has_permissions()
def cancel_invited_org_user(org_id, invited_user_id):
org_invite_api_client.cancel_invited_user(
org_id=org_id, invited_user_id=invited_user_id
)
invited_org_user = InvitedOrgUser.by_id_and_org_id(org_id, invited_user_id)
flash(
f"Invitation cancelled for {invited_org_user.email_address}",
"default_with_tick",
)
return redirect(url_for("main.manage_org_users", org_id=org_id))
@main.route("/organizations/<uuid:org_id>/settings/", methods=["GET"])
@user_is_platform_admin
def organization_settings(org_id):
return render_template(
"views/organizations/organization/settings/index.html",
)
@main.route("/organizations/<uuid:org_id>/settings/edit-name", methods=["GET", "POST"])
@user_is_platform_admin
def edit_organization_name(org_id):
form = RenameOrganizationForm(name=current_organization.name)
if form.validate_on_submit():
try:
current_organization.update(name=form.name.data)
except HTTPError as http_error:
error_msg = "Organization name already exists"
if http_error.status_code == 400 and error_msg in http_error.message:
form.name.errors.append("This organization name is already in use")
else:
raise http_error
else:
return redirect(url_for(".organization_settings", org_id=org_id))
return render_template(
"views/organizations/organization/settings/edit-name.html",
form=form,
)
@main.route("/organizations/<uuid:org_id>/settings/edit-type", methods=["GET", "POST"])
@user_is_platform_admin
def edit_organization_type(org_id):
form = OrganizationOrganizationTypeForm(
organization_type=current_organization.organization_type
)
if form.validate_on_submit():
current_organization.update(
organization_type=form.organization_type.data,
delete_services_cache=True,
)
return redirect(url_for(".organization_settings", org_id=org_id))
return render_template(
"views/organizations/organization/settings/edit-type.html",
form=form,
)
@main.route(
"/organizations/<uuid:org_id>/settings/edit-organization-domains",
methods=["GET", "POST"],
)
@user_is_platform_admin
def edit_organization_domains(org_id):
form = AdminOrganizationDomainsForm()
if form.validate_on_submit():
try:
organizations_client.update_organization(
org_id,
domains=list(
OrderedDict.fromkeys(
domain.lower() for domain in filter(None, form.domains.data)
)
),
)
except HTTPError as e:
error_message = "Domain already exists"
if e.status_code == 400 and error_message in e.message:
flash("This domain is already in use", "error")
return render_template(
"views/organizations/organization/settings/edit-domains.html",
form=form,
)
else:
raise e
return redirect(url_for(".organization_settings", org_id=org_id))
form.populate(current_organization.domains)
return render_template(
"views/organizations/organization/settings/edit-domains.html",
form=form,
)
@main.route("/organizations/<uuid:org_id>/settings/notes", methods=["GET", "POST"])
@user_is_platform_admin
def edit_organization_notes(org_id):
form = AdminNotesForm(notes=current_organization.notes)
if form.validate_on_submit():
if form.notes.data == current_organization.notes:
return redirect(url_for(".organization_settings", org_id=org_id))
current_organization.update(notes=form.notes.data)
return redirect(url_for(".organization_settings", org_id=org_id))
return render_template(
"views/organizations/organization/settings/edit-organization-notes.html",
form=form,
)
@main.route(
"/organizations/<uuid:org_id>/settings/edit-billing-details",
methods=["GET", "POST"],
)
@user_is_platform_admin
def edit_organization_billing_details(org_id):
form = AdminBillingDetailsForm(
billing_contact_email_addresses=current_organization.billing_contact_email_addresses,
billing_contact_names=current_organization.billing_contact_names,
billing_reference=current_organization.billing_reference,
purchase_order_number=current_organization.purchase_order_number,
notes=current_organization.notes,
)
if form.validate_on_submit():
current_organization.update(
billing_contact_email_addresses=form.billing_contact_email_addresses.data,
billing_contact_names=form.billing_contact_names.data,
billing_reference=form.billing_reference.data,
purchase_order_number=form.purchase_order_number.data,
notes=form.notes.data,
)
return redirect(url_for(".organization_settings", org_id=org_id))
return render_template(
"views/organizations/organization/settings/edit-organization-billing-details.html",
form=form,
)
@main.route("/organizations/<uuid:org_id>/billing")
@user_is_platform_admin
def organization_billing(org_id):
return render_template("views/organizations/organization/billing.html")