2020-05-23 16:40:28 +02:00
|
|
|
from smtplib import SMTPRecipientsRefused
|
|
|
|
|
2021-10-13 11:40:15 +02:00
|
|
|
import arrow
|
2020-05-23 19:18:35 +02:00
|
|
|
from flask import g
|
2020-05-23 16:09:06 +02:00
|
|
|
from flask import jsonify
|
|
|
|
from flask import request
|
|
|
|
|
|
|
|
from app.api.base import api_bp, require_api_auth
|
2021-10-13 11:40:15 +02:00
|
|
|
from app.config import JOB_DELETE_MAILBOX
|
2020-05-23 16:09:06 +02:00
|
|
|
from app.dashboard.views.mailbox import send_verification_email
|
2020-05-23 16:40:28 +02:00
|
|
|
from app.dashboard.views.mailbox_detail import verify_mailbox_change
|
2021-10-12 14:36:47 +02:00
|
|
|
from app.db import Session
|
2020-05-23 16:09:06 +02:00
|
|
|
from app.email_utils import (
|
|
|
|
mailbox_already_used,
|
2020-10-15 16:05:47 +02:00
|
|
|
email_can_be_used_as_mailbox,
|
2020-11-22 12:18:31 +01:00
|
|
|
is_valid_email,
|
2020-05-23 16:09:06 +02:00
|
|
|
)
|
2021-10-13 11:40:15 +02:00
|
|
|
from app.log import LOG
|
|
|
|
from app.models import Mailbox, Job
|
2021-03-31 14:41:32 +02:00
|
|
|
from app.utils import sanitize_email
|
2020-05-23 16:09:06 +02:00
|
|
|
|
|
|
|
|
2020-11-03 11:22:01 +01:00
|
|
|
def mailbox_to_dict(mailbox: Mailbox):
|
|
|
|
return {
|
|
|
|
"id": mailbox.id,
|
|
|
|
"email": mailbox.email,
|
|
|
|
"verified": mailbox.verified,
|
|
|
|
"default": mailbox.user.default_mailbox_id == mailbox.id,
|
|
|
|
"creation_timestamp": mailbox.created_at.timestamp,
|
|
|
|
"nb_alias": mailbox.nb_alias(),
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-05-23 16:09:06 +02:00
|
|
|
@api_bp.route("/mailboxes", methods=["POST"])
|
|
|
|
@require_api_auth
|
|
|
|
def create_mailbox():
|
|
|
|
"""
|
|
|
|
Create a new mailbox. User needs to verify the mailbox via an activation email.
|
|
|
|
Input:
|
|
|
|
email: in body
|
|
|
|
Output:
|
2020-11-03 11:22:01 +01:00
|
|
|
the new mailbox dict
|
2020-05-23 16:09:06 +02:00
|
|
|
"""
|
|
|
|
user = g.user
|
2021-01-11 12:29:40 +01:00
|
|
|
mailbox_email = sanitize_email(request.get_json().get("email"))
|
2020-05-23 16:09:06 +02:00
|
|
|
|
2021-06-02 17:17:28 +02:00
|
|
|
if not user.is_premium():
|
|
|
|
return jsonify(error=f"Only premium plan can add additional mailbox"), 400
|
|
|
|
|
2020-11-22 12:18:31 +01:00
|
|
|
if not is_valid_email(mailbox_email):
|
|
|
|
return jsonify(error=f"{mailbox_email} invalid"), 400
|
|
|
|
elif mailbox_already_used(mailbox_email, user):
|
2020-05-23 16:09:06 +02:00
|
|
|
return jsonify(error=f"{mailbox_email} already used"), 400
|
2020-10-15 16:05:47 +02:00
|
|
|
elif not email_can_be_used_as_mailbox(mailbox_email):
|
2020-05-23 16:09:06 +02:00
|
|
|
return (
|
|
|
|
jsonify(
|
|
|
|
error=f"{mailbox_email} cannot be used. Please note a mailbox cannot "
|
|
|
|
f"be a disposable email address"
|
|
|
|
),
|
|
|
|
400,
|
|
|
|
)
|
|
|
|
else:
|
|
|
|
new_mailbox = Mailbox.create(email=mailbox_email, user_id=user.id)
|
2021-10-12 14:36:47 +02:00
|
|
|
Session.commit()
|
2020-05-23 16:09:06 +02:00
|
|
|
|
|
|
|
send_verification_email(user, new_mailbox)
|
|
|
|
|
|
|
|
return (
|
2020-11-03 11:22:01 +01:00
|
|
|
jsonify(mailbox_to_dict(new_mailbox)),
|
2020-05-23 16:09:06 +02:00
|
|
|
201,
|
|
|
|
)
|
2020-05-23 16:18:12 +02:00
|
|
|
|
|
|
|
|
|
|
|
@api_bp.route("/mailboxes/<mailbox_id>", methods=["DELETE"])
|
|
|
|
@require_api_auth
|
|
|
|
def delete_mailbox(mailbox_id):
|
|
|
|
"""
|
|
|
|
Delete mailbox
|
|
|
|
Input:
|
|
|
|
mailbox_id: in url
|
|
|
|
Output:
|
|
|
|
200 if deleted successfully
|
|
|
|
|
|
|
|
"""
|
|
|
|
user = g.user
|
|
|
|
mailbox = Mailbox.get(mailbox_id)
|
|
|
|
|
|
|
|
if not mailbox or mailbox.user_id != user.id:
|
|
|
|
return jsonify(error="Forbidden"), 403
|
|
|
|
|
|
|
|
if mailbox.id == user.default_mailbox_id:
|
|
|
|
return jsonify(error="You cannot delete the default mailbox"), 400
|
|
|
|
|
2021-10-13 11:40:15 +02:00
|
|
|
# Schedule delete account job
|
|
|
|
LOG.w("schedule delete mailbox job for %s", mailbox)
|
|
|
|
Job.create(
|
|
|
|
name=JOB_DELETE_MAILBOX,
|
|
|
|
payload={"mailbox_id": mailbox.id},
|
|
|
|
run_at=arrow.now(),
|
|
|
|
commit=True,
|
|
|
|
)
|
2020-05-23 16:18:12 +02:00
|
|
|
|
|
|
|
return jsonify(deleted=True), 200
|
2020-05-23 16:26:26 +02:00
|
|
|
|
|
|
|
|
|
|
|
@api_bp.route("/mailboxes/<mailbox_id>", methods=["PUT"])
|
|
|
|
@require_api_auth
|
|
|
|
def update_mailbox(mailbox_id):
|
|
|
|
"""
|
|
|
|
Update mailbox
|
|
|
|
Input:
|
|
|
|
mailbox_id: in url
|
2020-05-23 16:40:28 +02:00
|
|
|
(optional) default: in body. Set a mailbox as the default mailbox.
|
|
|
|
(optional) email: in body. Change a mailbox email.
|
2020-05-23 16:43:48 +02:00
|
|
|
(optional) cancel_email_change: in body. Cancel mailbox email change.
|
2020-05-23 16:26:26 +02:00
|
|
|
Output:
|
|
|
|
200 if updated successfully
|
|
|
|
|
|
|
|
"""
|
|
|
|
user = g.user
|
|
|
|
mailbox = Mailbox.get(mailbox_id)
|
|
|
|
|
|
|
|
if not mailbox or mailbox.user_id != user.id:
|
|
|
|
return jsonify(error="Forbidden"), 403
|
|
|
|
|
|
|
|
data = request.get_json() or {}
|
|
|
|
changed = False
|
|
|
|
if "default" in data:
|
|
|
|
is_default = data.get("default")
|
|
|
|
if is_default:
|
2020-11-03 12:43:01 +01:00
|
|
|
if not mailbox.verified:
|
|
|
|
return (
|
|
|
|
jsonify(
|
|
|
|
error="Unverified mailbox cannot be used as default mailbox"
|
|
|
|
),
|
|
|
|
400,
|
|
|
|
)
|
2020-05-23 16:26:26 +02:00
|
|
|
user.default_mailbox_id = mailbox.id
|
|
|
|
changed = True
|
|
|
|
|
2020-05-23 16:40:28 +02:00
|
|
|
if "email" in data:
|
2021-01-11 12:29:40 +01:00
|
|
|
new_email = sanitize_email(data.get("email"))
|
2020-05-23 16:40:28 +02:00
|
|
|
|
|
|
|
if mailbox_already_used(new_email, user):
|
|
|
|
return jsonify(error=f"{new_email} already used"), 400
|
2020-10-15 16:05:47 +02:00
|
|
|
elif not email_can_be_used_as_mailbox(new_email):
|
2020-05-23 16:40:28 +02:00
|
|
|
return (
|
|
|
|
jsonify(
|
|
|
|
error=f"{new_email} cannot be used. Please note a mailbox cannot "
|
|
|
|
f"be a disposable email address"
|
|
|
|
),
|
|
|
|
400,
|
|
|
|
)
|
|
|
|
|
|
|
|
try:
|
|
|
|
verify_mailbox_change(user, mailbox, new_email)
|
|
|
|
except SMTPRecipientsRefused:
|
|
|
|
return jsonify(error=f"Incorrect mailbox, please recheck {new_email}"), 400
|
|
|
|
else:
|
|
|
|
mailbox.new_email = new_email
|
|
|
|
changed = True
|
|
|
|
|
2020-05-23 16:43:48 +02:00
|
|
|
if "cancel_email_change" in data:
|
|
|
|
cancel_email_change = data.get("cancel_email_change")
|
|
|
|
if cancel_email_change:
|
|
|
|
mailbox.new_email = None
|
|
|
|
changed = True
|
|
|
|
|
2020-05-23 16:26:26 +02:00
|
|
|
if changed:
|
2021-10-12 14:36:47 +02:00
|
|
|
Session.commit()
|
2020-05-23 16:26:26 +02:00
|
|
|
|
2020-05-23 16:40:28 +02:00
|
|
|
return jsonify(updated=True), 200
|
2020-05-23 16:46:10 +02:00
|
|
|
|
|
|
|
|
|
|
|
@api_bp.route("/mailboxes", methods=["GET"])
|
|
|
|
@require_api_auth
|
|
|
|
def get_mailboxes():
|
|
|
|
"""
|
2020-11-03 12:14:13 +01:00
|
|
|
Get verified mailboxes
|
2020-05-23 16:46:10 +02:00
|
|
|
Output:
|
2020-11-03 12:14:13 +01:00
|
|
|
- mailboxes: list of mailbox dict
|
2020-05-23 16:46:10 +02:00
|
|
|
"""
|
|
|
|
user = g.user
|
|
|
|
|
|
|
|
return (
|
2020-11-03 11:22:01 +01:00
|
|
|
jsonify(mailboxes=[mailbox_to_dict(mb) for mb in user.mailboxes()]),
|
2020-05-23 16:46:10 +02:00
|
|
|
200,
|
|
|
|
)
|
2020-11-03 12:14:13 +01:00
|
|
|
|
|
|
|
|
|
|
|
@api_bp.route("/v2/mailboxes", methods=["GET"])
|
|
|
|
@require_api_auth
|
|
|
|
def get_mailboxes_v2():
|
|
|
|
"""
|
|
|
|
Get all mailboxes - including unverified mailboxes
|
|
|
|
Output:
|
|
|
|
- mailboxes: list of mailbox dict
|
|
|
|
"""
|
|
|
|
user = g.user
|
|
|
|
mailboxes = []
|
|
|
|
|
2021-10-12 14:36:47 +02:00
|
|
|
for mailbox in Mailbox.filter_by(user_id=user.id):
|
2020-11-03 12:14:13 +01:00
|
|
|
mailboxes.append(mailbox)
|
|
|
|
|
|
|
|
return (
|
|
|
|
jsonify(mailboxes=[mailbox_to_dict(mb) for mb in mailboxes]),
|
|
|
|
200,
|
|
|
|
)
|