Compare commits
4 Commits
Author | SHA1 | Date | |
---|---|---|---|
3e6867bc17 | |||
a829074584 | |||
25834e8f61 | |||
a62b43b7c4 |
@ -7,8 +7,4 @@ If you want be up to date on security patches, make sure your SimpleLogin image
|
|||||||
|
|
||||||
## Reporting a Vulnerability
|
## Reporting a Vulnerability
|
||||||
|
|
||||||
If you've found a security vulnerability, you can disclose it responsibly by sending a summary to security@simplelogin.io.
|
If you want to report a vulnerability, please take a look at our bug bounty program at https://proton.me/security/bug-bounty.
|
||||||
We will review the potential threat and fix it as fast as we can.
|
|
||||||
|
|
||||||
We are incredibly thankful for people who disclose vulnerabilities, unfortunately we do not have a bounty program in place yet.
|
|
||||||
|
|
||||||
|
@ -3,12 +3,15 @@ from dataclasses import dataclass
|
|||||||
from enum import Enum
|
from enum import Enum
|
||||||
from typing import Optional
|
from typing import Optional
|
||||||
|
|
||||||
|
import arrow
|
||||||
from arrow import Arrow
|
from arrow import Arrow
|
||||||
from newrelic import agent
|
from newrelic import agent
|
||||||
from sqlalchemy import or_
|
from sqlalchemy import or_
|
||||||
|
|
||||||
from app.db import Session
|
from app.db import Session
|
||||||
from app.email_utils import send_welcome_email
|
from app.email_utils import send_welcome_email
|
||||||
|
from app.events.event_dispatcher import EventDispatcher
|
||||||
|
from app.events.generated.event_pb2 import UserPlanChanged, EventContent
|
||||||
from app.partner_user_utils import create_partner_user, create_partner_subscription
|
from app.partner_user_utils import create_partner_user, create_partner_subscription
|
||||||
from app.utils import sanitize_email, canonicalize_email
|
from app.utils import sanitize_email, canonicalize_email
|
||||||
from app.errors import (
|
from app.errors import (
|
||||||
@ -54,6 +57,21 @@ class LinkResult:
|
|||||||
strategy: str
|
strategy: str
|
||||||
|
|
||||||
|
|
||||||
|
def send_user_plan_changed_event(partner_user: PartnerUser) -> Optional[int]:
|
||||||
|
subscription_end = partner_user.user.get_active_subscription_end(
|
||||||
|
include_partner_subscription=False
|
||||||
|
)
|
||||||
|
end_timestamp = None
|
||||||
|
if partner_user.user.lifetime:
|
||||||
|
end_timestamp = arrow.get("2038-01-01").timestamp
|
||||||
|
elif subscription_end:
|
||||||
|
end_timestamp = subscription_end.timestamp
|
||||||
|
event = UserPlanChanged(plan_end_time=end_timestamp)
|
||||||
|
EventDispatcher.send_event(partner_user.user, EventContent(user_plan_change=event))
|
||||||
|
Session.flush()
|
||||||
|
return end_timestamp
|
||||||
|
|
||||||
|
|
||||||
def set_plan_for_partner_user(partner_user: PartnerUser, plan: SLPlan):
|
def set_plan_for_partner_user(partner_user: PartnerUser, plan: SLPlan):
|
||||||
sub = PartnerSubscription.get_by(partner_user_id=partner_user.id)
|
sub = PartnerSubscription.get_by(partner_user_id=partner_user.id)
|
||||||
if plan.type == SLPlanType.Free:
|
if plan.type == SLPlanType.Free:
|
||||||
@ -88,6 +106,8 @@ def set_plan_for_partner_user(partner_user: PartnerUser, plan: SLPlan):
|
|||||||
action=UserAuditLogAction.SubscriptionExtended,
|
action=UserAuditLogAction.SubscriptionExtended,
|
||||||
message="Extended partner subscription",
|
message="Extended partner subscription",
|
||||||
)
|
)
|
||||||
|
Session.flush()
|
||||||
|
send_user_plan_changed_event(partner_user)
|
||||||
Session.commit()
|
Session.commit()
|
||||||
|
|
||||||
|
|
||||||
|
@ -419,9 +419,8 @@ def create_contact_route(alias_id):
|
|||||||
if not data:
|
if not data:
|
||||||
return jsonify(error="request body cannot be empty"), 400
|
return jsonify(error="request body cannot be empty"), 400
|
||||||
|
|
||||||
alias: Alias = Alias.get(alias_id)
|
alias: Optional[Alias] = Alias.get_by(id=alias_id, user_id=g.user.id)
|
||||||
|
if not alias:
|
||||||
if alias.user_id != g.user.id:
|
|
||||||
return jsonify(error="Forbidden"), 403
|
return jsonify(error="Forbidden"), 403
|
||||||
|
|
||||||
contact_address = data.get("contact")
|
contact_address = data.get("contact")
|
||||||
|
@ -10,6 +10,7 @@ from app.events.auth_event import LoginEvent
|
|||||||
from app.extensions import limiter
|
from app.extensions import limiter
|
||||||
from app.log import LOG
|
from app.log import LOG
|
||||||
from app.models import User
|
from app.models import User
|
||||||
|
from app.pw_models import PasswordOracle
|
||||||
from app.utils import sanitize_email, sanitize_next_url, canonicalize_email
|
from app.utils import sanitize_email, sanitize_next_url, canonicalize_email
|
||||||
|
|
||||||
|
|
||||||
@ -43,6 +44,13 @@ def login():
|
|||||||
user = User.get_by(email=email) or User.get_by(email=canonical_email)
|
user = User.get_by(email=email) or User.get_by(email=canonical_email)
|
||||||
|
|
||||||
if not user or not user.check_password(form.password.data):
|
if not user or not user.check_password(form.password.data):
|
||||||
|
if not user:
|
||||||
|
# Do the hash to avoid timing attacks nevertheless
|
||||||
|
dummy_pw = PasswordOracle()
|
||||||
|
dummy_pw.password = (
|
||||||
|
"$2b$12$ZWqpL73h4rGNfLkJohAFAu0isqSw/bX9p/tzpbWRz/To5FAftaW8u"
|
||||||
|
)
|
||||||
|
dummy_pw.check_password(form.password.data)
|
||||||
# Trigger rate limiter
|
# Trigger rate limiter
|
||||||
g.deduct_limit = True
|
g.deduct_limit = True
|
||||||
form.password.data = None
|
form.password.data = None
|
||||||
|
@ -309,6 +309,7 @@ JOB_DELETE_DOMAIN = "delete-domain"
|
|||||||
JOB_SEND_USER_REPORT = "send-user-report"
|
JOB_SEND_USER_REPORT = "send-user-report"
|
||||||
JOB_SEND_PROTON_WELCOME_1 = "proton-welcome-1"
|
JOB_SEND_PROTON_WELCOME_1 = "proton-welcome-1"
|
||||||
JOB_SEND_ALIAS_CREATION_EVENTS = "send-alias-creation-events"
|
JOB_SEND_ALIAS_CREATION_EVENTS = "send-alias-creation-events"
|
||||||
|
JOB_SEND_EVENT_TO_WEBHOOK = "send-event-to-webhook"
|
||||||
|
|
||||||
# for pagination
|
# for pagination
|
||||||
PAGE_LIMIT = 20
|
PAGE_LIMIT = 20
|
||||||
|
@ -16,6 +16,7 @@ from app.utils import sanitize_email
|
|||||||
class ContactCreateError(Enum):
|
class ContactCreateError(Enum):
|
||||||
InvalidEmail = "Invalid email"
|
InvalidEmail = "Invalid email"
|
||||||
NotAllowed = "Your plan does not allow to create contacts"
|
NotAllowed = "Your plan does not allow to create contacts"
|
||||||
|
Unknown = "Unknown error when trying to create contact"
|
||||||
|
|
||||||
|
|
||||||
@dataclass
|
@dataclass
|
||||||
@ -87,8 +88,10 @@ def create_contact(
|
|||||||
return __update_contact_if_needed(contact, name, mail_from)
|
return __update_contact_if_needed(contact, name, mail_from)
|
||||||
# Create the contact
|
# Create the contact
|
||||||
reply_email = generate_reply_email(email, alias)
|
reply_email = generate_reply_email(email, alias)
|
||||||
|
alias_id = alias.id
|
||||||
try:
|
try:
|
||||||
flags = Contact.FLAG_PARTNER_CREATED if from_partner else 0
|
flags = Contact.FLAG_PARTNER_CREATED if from_partner else 0
|
||||||
|
is_invalid_email = email == ""
|
||||||
contact = Contact.create(
|
contact = Contact.create(
|
||||||
user_id=alias.user_id,
|
user_id=alias.user_id,
|
||||||
alias_id=alias.id,
|
alias_id=alias.id,
|
||||||
@ -98,9 +101,10 @@ def create_contact(
|
|||||||
mail_from=mail_from,
|
mail_from=mail_from,
|
||||||
automatic_created=automatic_created,
|
automatic_created=automatic_created,
|
||||||
flags=flags,
|
flags=flags,
|
||||||
invalid_email=email == "",
|
invalid_email=is_invalid_email,
|
||||||
commit=True,
|
commit=True,
|
||||||
)
|
)
|
||||||
|
contact_id = contact.id
|
||||||
if automatic_created:
|
if automatic_created:
|
||||||
trail = ". Automatically created"
|
trail = ". Automatically created"
|
||||||
else:
|
else:
|
||||||
@ -108,17 +112,27 @@ def create_contact(
|
|||||||
emit_alias_audit_log(
|
emit_alias_audit_log(
|
||||||
alias=alias,
|
alias=alias,
|
||||||
action=AliasAuditLogAction.CreateContact,
|
action=AliasAuditLogAction.CreateContact,
|
||||||
message=f"Created contact {contact.id} ({contact.email}){trail}",
|
message=f"Created contact {contact_id} ({email}){trail}",
|
||||||
commit=True,
|
commit=True,
|
||||||
)
|
)
|
||||||
LOG.d(
|
LOG.d(
|
||||||
f"Created contact {contact} for alias {alias} with email {email} invalid_email={contact.invalid_email}"
|
f"Created contact {contact} for alias {alias} with email {email} invalid_email={is_invalid_email}"
|
||||||
)
|
)
|
||||||
|
return ContactCreateResult(contact, created=True, error=None)
|
||||||
except IntegrityError:
|
except IntegrityError:
|
||||||
Session.rollback()
|
Session.rollback()
|
||||||
LOG.info(
|
LOG.info(
|
||||||
f"Contact with email {email} for alias_id {alias.id} already existed, fetching from DB"
|
f"Contact with email {email} for alias_id {alias_id} already existed, fetching from DB"
|
||||||
)
|
)
|
||||||
contact = Contact.get_by(alias_id=alias.id, website_email=email)
|
contact: Optional[Contact] = Contact.get_by(
|
||||||
|
alias_id=alias_id, website_email=email
|
||||||
|
)
|
||||||
|
if contact:
|
||||||
return __update_contact_if_needed(contact, name, mail_from)
|
return __update_contact_if_needed(contact, name, mail_from)
|
||||||
return ContactCreateResult(contact, created=True, error=None)
|
else:
|
||||||
|
LOG.warning(
|
||||||
|
f"Could not find contact with email {email} for alias_id {alias_id} and it should exist"
|
||||||
|
)
|
||||||
|
return ContactCreateResult(
|
||||||
|
None, created=False, error=ContactCreateError.Unknown
|
||||||
|
)
|
||||||
|
@ -1,3 +1,5 @@
|
|||||||
|
import secrets
|
||||||
|
|
||||||
import arrow
|
import arrow
|
||||||
from flask import (
|
from flask import (
|
||||||
render_template,
|
render_template,
|
||||||
@ -163,7 +165,7 @@ def send_reset_password_email(user):
|
|||||||
"""
|
"""
|
||||||
# the activation code is valid for 1h
|
# the activation code is valid for 1h
|
||||||
reset_password_code = ResetPasswordCode.create(
|
reset_password_code = ResetPasswordCode.create(
|
||||||
user_id=user.id, code=random_string(60)
|
user_id=user.id, code=secrets.token_urlsafe(32)
|
||||||
)
|
)
|
||||||
Session.commit()
|
Session.commit()
|
||||||
|
|
||||||
|
@ -1,3 +1,4 @@
|
|||||||
|
import arrow
|
||||||
from flask import render_template, flash, redirect, url_for
|
from flask import render_template, flash, redirect, url_for
|
||||||
from flask_login import login_required, current_user
|
from flask_login import login_required, current_user
|
||||||
from flask_wtf import FlaskForm
|
from flask_wtf import FlaskForm
|
||||||
@ -7,6 +8,8 @@ from app.config import ADMIN_EMAIL
|
|||||||
from app.dashboard.base import dashboard_bp
|
from app.dashboard.base import dashboard_bp
|
||||||
from app.db import Session
|
from app.db import Session
|
||||||
from app.email_utils import send_email
|
from app.email_utils import send_email
|
||||||
|
from app.events.event_dispatcher import EventDispatcher
|
||||||
|
from app.events.generated.event_pb2 import UserPlanChanged, EventContent
|
||||||
from app.models import LifetimeCoupon
|
from app.models import LifetimeCoupon
|
||||||
|
|
||||||
|
|
||||||
@ -40,6 +43,14 @@ def lifetime_licence():
|
|||||||
current_user.lifetime_coupon_id = coupon.id
|
current_user.lifetime_coupon_id = coupon.id
|
||||||
if coupon.paid:
|
if coupon.paid:
|
||||||
current_user.paid_lifetime = True
|
current_user.paid_lifetime = True
|
||||||
|
EventDispatcher.send_event(
|
||||||
|
user=current_user,
|
||||||
|
content=EventContent(
|
||||||
|
user_plan_change=UserPlanChanged(
|
||||||
|
plan_end_time=arrow.get("2038-01-01").timestamp
|
||||||
|
)
|
||||||
|
),
|
||||||
|
)
|
||||||
Session.commit()
|
Session.commit()
|
||||||
|
|
||||||
# notify admin
|
# notify admin
|
||||||
|
70
app/app/jobs/send_event_job.py
Normal file
70
app/app/jobs/send_event_job.py
Normal file
@ -0,0 +1,70 @@
|
|||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import base64
|
||||||
|
from typing import Optional
|
||||||
|
|
||||||
|
import arrow
|
||||||
|
|
||||||
|
from app import config
|
||||||
|
from app.errors import ProtonPartnerNotSetUp
|
||||||
|
from app.events.generated import event_pb2
|
||||||
|
from app.events.generated.event_pb2 import EventContent
|
||||||
|
from app.models import (
|
||||||
|
User,
|
||||||
|
Job,
|
||||||
|
PartnerUser,
|
||||||
|
)
|
||||||
|
from app.proton.utils import get_proton_partner
|
||||||
|
from events.event_sink import EventSink
|
||||||
|
|
||||||
|
|
||||||
|
class SendEventToWebhookJob:
|
||||||
|
def __init__(self, user: User, event: EventContent):
|
||||||
|
self._user: User = user
|
||||||
|
self._event: EventContent = event
|
||||||
|
|
||||||
|
def run(self, sink: EventSink) -> bool:
|
||||||
|
# Check if the current user has a partner_id
|
||||||
|
try:
|
||||||
|
proton_partner_id = get_proton_partner().id
|
||||||
|
except ProtonPartnerNotSetUp:
|
||||||
|
return False
|
||||||
|
|
||||||
|
# It has. Retrieve the information for the PartnerUser
|
||||||
|
partner_user = PartnerUser.get_by(
|
||||||
|
user_id=self._user.id, partner_id=proton_partner_id
|
||||||
|
)
|
||||||
|
if partner_user is None:
|
||||||
|
return True
|
||||||
|
event = event_pb2.Event(
|
||||||
|
user_id=self._user.id,
|
||||||
|
external_user_id=partner_user.external_user_id,
|
||||||
|
partner_id=partner_user.partner_id,
|
||||||
|
content=self._event,
|
||||||
|
)
|
||||||
|
|
||||||
|
serialized = event.SerializeToString()
|
||||||
|
return sink.send_data_to_webhook(serialized)
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def create_from_job(job: Job) -> Optional[SendEventToWebhookJob]:
|
||||||
|
user = User.get(job.payload["user_id"])
|
||||||
|
if not user:
|
||||||
|
return None
|
||||||
|
event_data = base64.b64decode(job.payload["event"])
|
||||||
|
event = event_pb2.EventContent()
|
||||||
|
event.ParseFromString(event_data)
|
||||||
|
|
||||||
|
return SendEventToWebhookJob(user=user, event=event)
|
||||||
|
|
||||||
|
def store_job_in_db(self, run_at: Optional[arrow.Arrow]) -> Job:
|
||||||
|
stub = self._event.SerializeToString()
|
||||||
|
return Job.create(
|
||||||
|
name=config.JOB_SEND_EVENT_TO_WEBHOOK,
|
||||||
|
payload={
|
||||||
|
"user_id": self._user.id,
|
||||||
|
"event": base64.b64encode(stub).decode("utf-8"),
|
||||||
|
},
|
||||||
|
run_at=run_at if run_at is not None else arrow.now(),
|
||||||
|
commit=True,
|
||||||
|
)
|
@ -1,6 +1,5 @@
|
|||||||
import dataclasses
|
import dataclasses
|
||||||
import secrets
|
import secrets
|
||||||
import random
|
|
||||||
from enum import Enum
|
from enum import Enum
|
||||||
from typing import Optional
|
from typing import Optional
|
||||||
import arrow
|
import arrow
|
||||||
@ -37,8 +36,9 @@ class OnlyPaidError(MailboxError):
|
|||||||
|
|
||||||
|
|
||||||
class CannotVerifyError(MailboxError):
|
class CannotVerifyError(MailboxError):
|
||||||
def __init__(self, msg: str):
|
def __init__(self, msg: str, deleted_activation_code: bool = False):
|
||||||
self.msg = msg
|
self.msg = msg
|
||||||
|
self.deleted_activation_code = deleted_activation_code
|
||||||
|
|
||||||
|
|
||||||
MAX_ACTIVATION_TRIES = 3
|
MAX_ACTIVATION_TRIES = 3
|
||||||
@ -196,7 +196,10 @@ def verify_mailbox_code(user: User, mailbox_id: int, code: str) -> Mailbox:
|
|||||||
if activation.tries >= MAX_ACTIVATION_TRIES:
|
if activation.tries >= MAX_ACTIVATION_TRIES:
|
||||||
LOG.i(f"User {user} failed to verify mailbox {mailbox_id} more than 3 times")
|
LOG.i(f"User {user} failed to verify mailbox {mailbox_id} more than 3 times")
|
||||||
clear_activation_codes_for_mailbox(mailbox)
|
clear_activation_codes_for_mailbox(mailbox)
|
||||||
raise CannotVerifyError("Invalid activation code. Please request another code.")
|
raise CannotVerifyError(
|
||||||
|
"Invalid activation code. Please request another code.",
|
||||||
|
deleted_activation_code=True,
|
||||||
|
)
|
||||||
if activation.created_at < arrow.now().shift(minutes=-15):
|
if activation.created_at < arrow.now().shift(minutes=-15):
|
||||||
LOG.i(
|
LOG.i(
|
||||||
f"User {user} failed to verify mailbox {mailbox_id} because code is too old"
|
f"User {user} failed to verify mailbox {mailbox_id} because code is too old"
|
||||||
@ -229,7 +232,7 @@ def generate_activation_code(
|
|||||||
if config.MAILBOX_VERIFICATION_OVERRIDE_CODE:
|
if config.MAILBOX_VERIFICATION_OVERRIDE_CODE:
|
||||||
code = config.MAILBOX_VERIFICATION_OVERRIDE_CODE
|
code = config.MAILBOX_VERIFICATION_OVERRIDE_CODE
|
||||||
else:
|
else:
|
||||||
code = "{:06d}".format(random.randint(1, 999999))
|
code = "{:06d}".format(secrets.randbelow(1000000))[:6]
|
||||||
else:
|
else:
|
||||||
code = secrets.token_urlsafe(16)
|
code = secrets.token_urlsafe(16)
|
||||||
return MailboxActivation.create(
|
return MailboxActivation.create(
|
||||||
|
@ -1,4 +1,3 @@
|
|||||||
import random
|
|
||||||
import re
|
import re
|
||||||
import secrets
|
import secrets
|
||||||
import string
|
import string
|
||||||
@ -32,8 +31,9 @@ def random_words(words: int = 2, numbers: int = 0):
|
|||||||
fields = [secrets.choice(_words) for i in range(words)]
|
fields = [secrets.choice(_words) for i in range(words)]
|
||||||
|
|
||||||
if numbers > 0:
|
if numbers > 0:
|
||||||
digits = "".join([str(random.randint(0, 9)) for i in range(numbers)])
|
digits = [n for n in range(10)]
|
||||||
return "_".join(fields) + digits
|
suffix = "".join([str(secrets.choice(digits)) for i in range(numbers)])
|
||||||
|
return "_".join(fields) + suffix
|
||||||
else:
|
else:
|
||||||
return "_".join(fields)
|
return "_".join(fields)
|
||||||
|
|
||||||
|
39
app/cron.py
39
app/cron.py
@ -286,8 +286,16 @@ def notify_manual_sub_end():
|
|||||||
|
|
||||||
def poll_apple_subscription():
|
def poll_apple_subscription():
|
||||||
"""Poll Apple API to update AppleSubscription"""
|
"""Poll Apple API to update AppleSubscription"""
|
||||||
# todo: only near the end of the subscription
|
for apple_sub in (
|
||||||
for apple_sub in AppleSubscription.all():
|
AppleSubscription.filter(
|
||||||
|
AppleSubscription.expires_date < arrow.now().shift(days=15)
|
||||||
|
)
|
||||||
|
.enable_eagerloads(False)
|
||||||
|
.yield_per(100)
|
||||||
|
):
|
||||||
|
if not apple_sub.is_valid():
|
||||||
|
# Subscription is not valid anymore and hasn't been renewed
|
||||||
|
continue
|
||||||
if not apple_sub.product_id:
|
if not apple_sub.product_id:
|
||||||
LOG.d("Ignore %s", apple_sub)
|
LOG.d("Ignore %s", apple_sub)
|
||||||
continue
|
continue
|
||||||
@ -900,6 +908,24 @@ def check_mailbox_valid_pgp_keys():
|
|||||||
|
|
||||||
|
|
||||||
def check_custom_domain():
|
def check_custom_domain():
|
||||||
|
# Delete custom domains that haven't been verified in a month
|
||||||
|
for custom_domain in (
|
||||||
|
CustomDomain.filter(
|
||||||
|
CustomDomain.verified == False, # noqa: E712
|
||||||
|
CustomDomain.created_at < arrow.now().shift(months=-1),
|
||||||
|
)
|
||||||
|
.enable_eagerloads(False)
|
||||||
|
.yield_per(100)
|
||||||
|
):
|
||||||
|
alias_count = Alias.filter(Alias.custom_domain_id == custom_domain.id).count()
|
||||||
|
if alias_count > 0:
|
||||||
|
LOG.warn(
|
||||||
|
f"Custom Domain {custom_domain} has {alias_count} aliases. Won't delete"
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
LOG.i(f"Deleting unverified old custom domain {custom_domain}")
|
||||||
|
CustomDomain.delete(custom_domain.id)
|
||||||
|
|
||||||
LOG.d("Check verified domain for DNS issues")
|
LOG.d("Check verified domain for DNS issues")
|
||||||
|
|
||||||
for custom_domain in CustomDomain.filter_by(verified=True): # type: CustomDomain
|
for custom_domain in CustomDomain.filter_by(verified=True): # type: CustomDomain
|
||||||
@ -971,7 +997,7 @@ def delete_expired_tokens():
|
|||||||
LOG.d("Delete api to cookie tokens older than %s, nb row %s", max_time, nb_row)
|
LOG.d("Delete api to cookie tokens older than %s, nb row %s", max_time, nb_row)
|
||||||
|
|
||||||
|
|
||||||
async def _hibp_check(api_key, queue):
|
async def _hibp_check(api_key: str, queue: asyncio.Queue):
|
||||||
"""
|
"""
|
||||||
Uses a single API key to check the queue as fast as possible.
|
Uses a single API key to check the queue as fast as possible.
|
||||||
|
|
||||||
@ -990,11 +1016,16 @@ async def _hibp_check(api_key, queue):
|
|||||||
if not alias:
|
if not alias:
|
||||||
continue
|
continue
|
||||||
user = alias.user
|
user = alias.user
|
||||||
if user.disabled or not user.is_paid():
|
if user.disabled or not user.is_premium():
|
||||||
# Mark it as hibp done to skip it as if it had been checked
|
# Mark it as hibp done to skip it as if it had been checked
|
||||||
alias.hibp_last_check = arrow.utcnow()
|
alias.hibp_last_check = arrow.utcnow()
|
||||||
Session.commit()
|
Session.commit()
|
||||||
continue
|
continue
|
||||||
|
if alias.flags & Alias.FLAG_PARTNER_CREATED > 0:
|
||||||
|
# Mark as hibp done
|
||||||
|
alias.hibp_last_check = arrow.utcnow()
|
||||||
|
Session.commit()
|
||||||
|
continue
|
||||||
|
|
||||||
LOG.d("Checking HIBP for %s", alias)
|
LOG.d("Checking HIBP for %s", alias)
|
||||||
|
|
||||||
|
@ -16,13 +16,25 @@ jobs:
|
|||||||
shell: /bin/bash
|
shell: /bin/bash
|
||||||
schedule: "15 2 * * *"
|
schedule: "15 2 * * *"
|
||||||
captureStderr: true
|
captureStderr: true
|
||||||
|
onFailure:
|
||||||
|
retry:
|
||||||
|
maximumRetries: 10
|
||||||
|
initialDelay: 1
|
||||||
|
maximumDelay: 30
|
||||||
|
backoffMultiplier: 2
|
||||||
|
|
||||||
- name: SimpleLogin HIBP check
|
- name: SimpleLogin HIBP check
|
||||||
command: python /code/cron.py -j check_hibp
|
command: python /code/cron.py -j check_hibp
|
||||||
shell: /bin/bash
|
shell: /bin/bash
|
||||||
schedule: "15 3 * * *"
|
schedule: "16 */4 * * *"
|
||||||
captureStderr: true
|
captureStderr: true
|
||||||
concurrencyPolicy: Forbid
|
concurrencyPolicy: Forbid
|
||||||
|
onFailure:
|
||||||
|
retry:
|
||||||
|
maximumRetries: 10
|
||||||
|
initialDelay: 1
|
||||||
|
maximumDelay: 30
|
||||||
|
backoffMultiplier: 2
|
||||||
|
|
||||||
- name: SimpleLogin Notify HIBP breaches
|
- name: SimpleLogin Notify HIBP breaches
|
||||||
command: python /code/cron.py -j notify_hibp
|
command: python /code/cron.py -j notify_hibp
|
||||||
@ -31,6 +43,7 @@ jobs:
|
|||||||
captureStderr: true
|
captureStderr: true
|
||||||
concurrencyPolicy: Forbid
|
concurrencyPolicy: Forbid
|
||||||
|
|
||||||
|
|
||||||
- name: SimpleLogin Delete Logs
|
- name: SimpleLogin Delete Logs
|
||||||
command: python /code/cron.py -j delete_logs
|
command: python /code/cron.py -j delete_logs
|
||||||
shell: /bin/bash
|
shell: /bin/bash
|
||||||
|
@ -177,7 +177,9 @@ from init_app import load_pgp_public_keys
|
|||||||
from server import create_light_app
|
from server import create_light_app
|
||||||
|
|
||||||
|
|
||||||
def get_or_create_contact(from_header: str, mail_from: str, alias: Alias) -> Contact:
|
def get_or_create_contact(
|
||||||
|
from_header: str, mail_from: str, alias: Alias
|
||||||
|
) -> Optional[Contact]:
|
||||||
"""
|
"""
|
||||||
contact_from_header is the RFC 2047 format FROM header
|
contact_from_header is the RFC 2047 format FROM header
|
||||||
"""
|
"""
|
||||||
@ -208,6 +210,8 @@ def get_or_create_contact(from_header: str, mail_from: str, alias: Alias) -> Con
|
|||||||
automatic_created=True,
|
automatic_created=True,
|
||||||
from_partner=False,
|
from_partner=False,
|
||||||
)
|
)
|
||||||
|
if contact_result.error:
|
||||||
|
LOG.w(f"Error creating contact: {contact_result.error.value}")
|
||||||
return contact_result.contact
|
return contact_result.contact
|
||||||
|
|
||||||
|
|
||||||
@ -558,7 +562,7 @@ def handle_forward(envelope, msg: Message, rcpt_to: str) -> List[Tuple[bool, str
|
|||||||
|
|
||||||
if not user.is_active():
|
if not user.is_active():
|
||||||
LOG.w(f"User {user} has been soft deleted")
|
LOG.w(f"User {user} has been soft deleted")
|
||||||
return False, status.E502
|
return [(False, status.E502)]
|
||||||
|
|
||||||
if not user.can_send_or_receive():
|
if not user.can_send_or_receive():
|
||||||
LOG.i(f"User {user} cannot receive emails")
|
LOG.i(f"User {user} cannot receive emails")
|
||||||
@ -579,6 +583,8 @@ def handle_forward(envelope, msg: Message, rcpt_to: str) -> List[Tuple[bool, str
|
|||||||
from_header = get_header_unicode(msg[headers.FROM])
|
from_header = get_header_unicode(msg[headers.FROM])
|
||||||
LOG.d("Create or get contact for from_header:%s", from_header)
|
LOG.d("Create or get contact for from_header:%s", from_header)
|
||||||
contact = get_or_create_contact(from_header, envelope.mail_from, alias)
|
contact = get_or_create_contact(from_header, envelope.mail_from, alias)
|
||||||
|
if not contact:
|
||||||
|
return [(False, status.E504)]
|
||||||
alias = (
|
alias = (
|
||||||
contact.alias
|
contact.alias
|
||||||
) # In case the Session was closed in the get_or_create we re-fetch the alias
|
) # In case the Session was closed in the get_or_create we re-fetch the alias
|
||||||
|
@ -12,6 +12,10 @@ class EventSink(ABC):
|
|||||||
def process(self, event: SyncEvent) -> bool:
|
def process(self, event: SyncEvent) -> bool:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
@abstractmethod
|
||||||
|
def send_data_to_webhook(self, data: bytes) -> bool:
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
class HttpEventSink(EventSink):
|
class HttpEventSink(EventSink):
|
||||||
def process(self, event: SyncEvent) -> bool:
|
def process(self, event: SyncEvent) -> bool:
|
||||||
@ -21,9 +25,16 @@ class HttpEventSink(EventSink):
|
|||||||
|
|
||||||
LOG.info(f"Sending event {event.id} to {EVENT_WEBHOOK}")
|
LOG.info(f"Sending event {event.id} to {EVENT_WEBHOOK}")
|
||||||
|
|
||||||
|
if self.send_data_to_webhook(event.content):
|
||||||
|
LOG.info(f"Event {event.id} sent successfully to webhook")
|
||||||
|
return True
|
||||||
|
|
||||||
|
return False
|
||||||
|
|
||||||
|
def send_data_to_webhook(self, data: bytes) -> bool:
|
||||||
res = requests.post(
|
res = requests.post(
|
||||||
url=EVENT_WEBHOOK,
|
url=EVENT_WEBHOOK,
|
||||||
data=event.content,
|
data=data,
|
||||||
headers={"Content-Type": "application/x-protobuf"},
|
headers={"Content-Type": "application/x-protobuf"},
|
||||||
verify=not EVENT_WEBHOOK_SKIP_VERIFY_SSL,
|
verify=not EVENT_WEBHOOK_SKIP_VERIFY_SSL,
|
||||||
)
|
)
|
||||||
@ -36,7 +47,6 @@ class HttpEventSink(EventSink):
|
|||||||
)
|
)
|
||||||
return False
|
return False
|
||||||
else:
|
else:
|
||||||
LOG.info(f"Event {event.id} sent successfully to webhook")
|
|
||||||
return True
|
return True
|
||||||
|
|
||||||
|
|
||||||
@ -44,3 +54,7 @@ class ConsoleEventSink(EventSink):
|
|||||||
def process(self, event: SyncEvent) -> bool:
|
def process(self, event: SyncEvent) -> bool:
|
||||||
LOG.info(f"Handling event {event.id}")
|
LOG.info(f"Handling event {event.id}")
|
||||||
return True
|
return True
|
||||||
|
|
||||||
|
def send_data_to_webhook(self, data: bytes) -> bool:
|
||||||
|
LOG.info(f"Sending {len(data)} bytes to webhook")
|
||||||
|
return True
|
||||||
|
@ -18,6 +18,7 @@ from app.events.event_dispatcher import PostgresDispatcher
|
|||||||
from app.import_utils import handle_batch_import
|
from app.import_utils import handle_batch_import
|
||||||
from app.jobs.event_jobs import send_alias_creation_events_for_user
|
from app.jobs.event_jobs import send_alias_creation_events_for_user
|
||||||
from app.jobs.export_user_data_job import ExportUserDataJob
|
from app.jobs.export_user_data_job import ExportUserDataJob
|
||||||
|
from app.jobs.send_event_job import SendEventToWebhookJob
|
||||||
from app.log import LOG
|
from app.log import LOG
|
||||||
from app.models import User, Job, BatchImport, Mailbox, CustomDomain, JobState
|
from app.models import User, Job, BatchImport, Mailbox, CustomDomain, JobState
|
||||||
from app.user_audit_log_utils import emit_user_audit_log, UserAuditLogAction
|
from app.user_audit_log_utils import emit_user_audit_log, UserAuditLogAction
|
||||||
@ -300,6 +301,10 @@ def process_job(job: Job):
|
|||||||
send_alias_creation_events_for_user(
|
send_alias_creation_events_for_user(
|
||||||
user, dispatcher=PostgresDispatcher.get()
|
user, dispatcher=PostgresDispatcher.get()
|
||||||
)
|
)
|
||||||
|
elif job.name == config.JOB_SEND_EVENT_TO_WEBHOOK:
|
||||||
|
send_job = SendEventToWebhookJob.create_from_job(job)
|
||||||
|
if send_job:
|
||||||
|
send_job.run()
|
||||||
else:
|
else:
|
||||||
LOG.e("Unknown job name %s", job.name)
|
LOG.e("Unknown job name %s", job.name)
|
||||||
|
|
||||||
|
@ -0,0 +1,28 @@
|
|||||||
|
"""Preserve user id on alias delete
|
||||||
|
|
||||||
|
Revision ID: 4882cc49dde9
|
||||||
|
Revises: 32f25cbf12f6
|
||||||
|
Create Date: 2024-11-06 10:10:40.235991
|
||||||
|
|
||||||
|
"""
|
||||||
|
from alembic import op
|
||||||
|
import sqlalchemy as sa
|
||||||
|
|
||||||
|
|
||||||
|
# revision identifiers, used by Alembic.
|
||||||
|
revision = '4882cc49dde9'
|
||||||
|
down_revision = '32f25cbf12f6'
|
||||||
|
branch_labels = None
|
||||||
|
depends_on = None
|
||||||
|
|
||||||
|
|
||||||
|
def upgrade():
|
||||||
|
op.add_column('deleted_alias', sa.Column('user_id', sa.Integer(), server_default=None, nullable=True))
|
||||||
|
with op.get_context().autocommit_block():
|
||||||
|
op.create_index('ix_deleted_alias_user_id_created_at', 'deleted_alias', ['user_id', 'created_at'], unique=False, postgresql_concurrently=True)
|
||||||
|
|
||||||
|
|
||||||
|
def downgrade():
|
||||||
|
with op.get_context().autocommit_block():
|
||||||
|
op.drop_index('ix_deleted_alias_user_id_created_at', table_name='deleted_alias')
|
||||||
|
op.drop_column('deleted_alias', 'user_id')
|
@ -0,0 +1,28 @@
|
|||||||
|
"""Revert user id on deleted alias
|
||||||
|
|
||||||
|
Revision ID: bc9aa210efa3
|
||||||
|
Revises: 4882cc49dde9
|
||||||
|
Create Date: 2024-11-06 12:44:44.129691
|
||||||
|
|
||||||
|
"""
|
||||||
|
from alembic import op
|
||||||
|
import sqlalchemy as sa
|
||||||
|
|
||||||
|
|
||||||
|
# revision identifiers, used by Alembic.
|
||||||
|
revision = 'bc9aa210efa3'
|
||||||
|
down_revision = '4882cc49dde9'
|
||||||
|
branch_labels = None
|
||||||
|
depends_on = None
|
||||||
|
|
||||||
|
|
||||||
|
def upgrade():
|
||||||
|
with op.get_context().autocommit_block():
|
||||||
|
op.drop_index('ix_deleted_alias_user_id_created_at', table_name='deleted_alias')
|
||||||
|
op.drop_column('deleted_alias', 'user_id')
|
||||||
|
|
||||||
|
|
||||||
|
def downgrade():
|
||||||
|
op.add_column('deleted_alias', sa.Column('user_id', sa.Integer(), server_default=None, nullable=True))
|
||||||
|
with op.get_context().autocommit_block():
|
||||||
|
op.create_index('ix_deleted_alias_user_id_created_at', 'deleted_alias', ['user_id', 'created_at'], unique=False, postgresql_concurrently=True)
|
62
app/oneshot/send_lifetime_user_events.py
Normal file
62
app/oneshot/send_lifetime_user_events.py
Normal file
@ -0,0 +1,62 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
import argparse
|
||||||
|
import time
|
||||||
|
|
||||||
|
import arrow
|
||||||
|
from sqlalchemy import func
|
||||||
|
|
||||||
|
from app.events.event_dispatcher import EventDispatcher
|
||||||
|
from app.events.generated.event_pb2 import UserPlanChanged, EventContent
|
||||||
|
from app.models import PartnerUser, User
|
||||||
|
from app.db import Session
|
||||||
|
|
||||||
|
parser = argparse.ArgumentParser(
|
||||||
|
prog="Backfill alias", description="Send lifetime users to proton"
|
||||||
|
)
|
||||||
|
parser.add_argument(
|
||||||
|
"-s", "--start_pu_id", default=0, type=int, help="Initial partner_user_id"
|
||||||
|
)
|
||||||
|
parser.add_argument(
|
||||||
|
"-e", "--end_pu_id", default=0, type=int, help="Last partner_user_id"
|
||||||
|
)
|
||||||
|
|
||||||
|
args = parser.parse_args()
|
||||||
|
pu_id_start = args.start_pu_id
|
||||||
|
max_pu_id = args.end_pu_id
|
||||||
|
if max_pu_id == 0:
|
||||||
|
max_pu_id = Session.query(func.max(PartnerUser.id)).scalar()
|
||||||
|
|
||||||
|
print(f"Checking partner user {pu_id_start} to {max_pu_id}")
|
||||||
|
step = 1000
|
||||||
|
done = 0
|
||||||
|
start_time = time.time()
|
||||||
|
with_lifetime = 0
|
||||||
|
for batch_start in range(pu_id_start, max_pu_id, step):
|
||||||
|
users = (
|
||||||
|
Session.query(User)
|
||||||
|
.join(PartnerUser, PartnerUser.user_id == User.id)
|
||||||
|
.filter(
|
||||||
|
PartnerUser.id >= batch_start,
|
||||||
|
PartnerUser.id < batch_start + step,
|
||||||
|
User.lifetime == True, # noqa :E712
|
||||||
|
)
|
||||||
|
).all()
|
||||||
|
for user in users:
|
||||||
|
# Just in case the == True cond is wonky
|
||||||
|
if not user.lifetime:
|
||||||
|
continue
|
||||||
|
with_lifetime += 1
|
||||||
|
event = UserPlanChanged(plan_end_time=arrow.get("2038-01-01").timestamp)
|
||||||
|
EventDispatcher.send_event(user, EventContent(user_plan_change=event))
|
||||||
|
Session.flush()
|
||||||
|
Session.commit()
|
||||||
|
elapsed = time.time() - start_time
|
||||||
|
last_batch_id = batch_start + step
|
||||||
|
time_per_alias = elapsed / (last_batch_id)
|
||||||
|
remaining = max_pu_id - last_batch_id
|
||||||
|
time_remaining = remaining / time_per_alias
|
||||||
|
hours_remaining = time_remaining / 60.0
|
||||||
|
print(
|
||||||
|
f"\PartnerUser {batch_start}/{max_pu_id} {with_lifetime} {hours_remaining:.2f} mins remaining"
|
||||||
|
)
|
||||||
|
print(f"With SL lifetime {with_lifetime}")
|
@ -2,10 +2,10 @@
|
|||||||
import argparse
|
import argparse
|
||||||
import time
|
import time
|
||||||
|
|
||||||
|
import arrow
|
||||||
from sqlalchemy import func
|
from sqlalchemy import func
|
||||||
|
|
||||||
from app.events.event_dispatcher import EventDispatcher
|
from app.account_linking import send_user_plan_changed_event
|
||||||
from app.events.generated.event_pb2 import UserPlanChanged, EventContent
|
|
||||||
from app.models import PartnerUser
|
from app.models import PartnerUser
|
||||||
from app.db import Session
|
from app.db import Session
|
||||||
|
|
||||||
@ -30,6 +30,7 @@ step = 100
|
|||||||
updated = 0
|
updated = 0
|
||||||
start_time = time.time()
|
start_time = time.time()
|
||||||
with_premium = 0
|
with_premium = 0
|
||||||
|
with_lifetime = 0
|
||||||
for batch_start in range(pu_id_start, max_pu_id, step):
|
for batch_start in range(pu_id_start, max_pu_id, step):
|
||||||
partner_users = (
|
partner_users = (
|
||||||
Session.query(PartnerUser).filter(
|
Session.query(PartnerUser).filter(
|
||||||
@ -37,18 +38,12 @@ for batch_start in range(pu_id_start, max_pu_id, step):
|
|||||||
)
|
)
|
||||||
).all()
|
).all()
|
||||||
for partner_user in partner_users:
|
for partner_user in partner_users:
|
||||||
subscription_end = partner_user.user.get_active_subscription_end(
|
subscription_end = send_user_plan_changed_event(partner_user)
|
||||||
include_partner_subscription=False
|
if subscription_end is not None:
|
||||||
)
|
if subscription_end > arrow.get("2038-01-01").timestamp:
|
||||||
end_timestamp = None
|
with_lifetime += 1
|
||||||
if subscription_end:
|
else:
|
||||||
with_premium += 1
|
with_premium += 1
|
||||||
end_timestamp = subscription_end.timestamp
|
|
||||||
event = UserPlanChanged(plan_end_time=end_timestamp)
|
|
||||||
EventDispatcher.send_event(
|
|
||||||
partner_user.user, EventContent(user_plan_change=event)
|
|
||||||
)
|
|
||||||
Session.flush()
|
|
||||||
updated += 1
|
updated += 1
|
||||||
Session.commit()
|
Session.commit()
|
||||||
elapsed = time.time() - start_time
|
elapsed = time.time() - start_time
|
||||||
@ -60,4 +55,4 @@ for batch_start in range(pu_id_start, max_pu_id, step):
|
|||||||
print(
|
print(
|
||||||
f"\PartnerUser {batch_start}/{max_pu_id} {updated} {hours_remaining:.2f} mins remaining"
|
f"\PartnerUser {batch_start}/{max_pu_id} {updated} {hours_remaining:.2f} mins remaining"
|
||||||
)
|
)
|
||||||
print(f"With SL premium {with_premium}")
|
print(f"With SL premium {with_premium} lifetime {with_lifetime}")
|
||||||
|
@ -11,6 +11,7 @@
|
|||||||
<th scope="col">Verified</th>
|
<th scope="col">Verified</th>
|
||||||
<th scope="col">Status</th>
|
<th scope="col">Status</th>
|
||||||
<th scope="col">Paid</th>
|
<th scope="col">Paid</th>
|
||||||
|
<th scope="col">Premium</th>
|
||||||
<th>Subscription</th>
|
<th>Subscription</th>
|
||||||
<th>Created At</th>
|
<th>Created At</th>
|
||||||
<th>Updated At</th>
|
<th>Updated At</th>
|
||||||
@ -20,24 +21,31 @@
|
|||||||
<tbody>
|
<tbody>
|
||||||
<tr>
|
<tr>
|
||||||
<td>{{ user.id }}</td>
|
<td>{{ user.id }}</td>
|
||||||
<td><a href="?email={{ user.email }}">{{ user.email }}</a></td>
|
<td>
|
||||||
|
<a href="?email={{ user.email }}">{{ user.email }}</a>
|
||||||
|
</td>
|
||||||
{% if user.activated %}
|
{% if user.activated %}
|
||||||
|
|
||||||
<td class="text-success">Activated</td>
|
<td class="text-success">Activated</td>
|
||||||
{% else %}
|
{% else %}
|
||||||
<td class="text-warning">Pending</td>
|
<td class="text-warning">Pending</td>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
{% if user.disabled %}
|
{% if user.disabled %}
|
||||||
|
|
||||||
<td class="text-danger">Disabled</td>
|
<td class="text-danger">Disabled</td>
|
||||||
{% else %}
|
{% else %}
|
||||||
<td class="text-success">Enabled</td>
|
<td class="text-success">Enabled</td>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
<td>{{ "yes" if user.is_paid() else "No" }}</td>
|
<td>{{ "yes" if user.is_paid() else "No" }}</td>
|
||||||
|
<td>{{ "yes" if user.is_premium() else "No" }}</td>
|
||||||
<td>{{ user.get_active_subscription() }}</td>
|
<td>{{ user.get_active_subscription() }}</td>
|
||||||
<td>{{ user.created_at }}</td>
|
<td>{{ user.created_at }}</td>
|
||||||
<td>{{ user.updated_at }}</td>
|
<td>{{ user.updated_at }}</td>
|
||||||
{% if pu %}
|
{% if pu %}
|
||||||
|
|
||||||
<td><a href="?email={{ pu.partner_email }}">{{ pu.partner_email }}</a></td>
|
<td>
|
||||||
|
<a href="?email={{ pu.partner_email }}">{{ pu.partner_email }}</a>
|
||||||
|
</td>
|
||||||
{% else %}
|
{% else %}
|
||||||
<td>No</td>
|
<td>No</td>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
@ -61,14 +69,13 @@
|
|||||||
</thead>
|
</thead>
|
||||||
<tbody>
|
<tbody>
|
||||||
{% for mailbox in mboxes %}
|
{% for mailbox in mboxes %}
|
||||||
|
|
||||||
<tr>
|
<tr>
|
||||||
<td>{{ mailbox.id }}</td>
|
<td>{{ mailbox.id }}</td>
|
||||||
<td><a href="?email={{ mailbox.email }}">{{ mailbox.email }}</a></td>
|
|
||||||
<td>{{ "Yes" if mailbox.verified else "No" }}</td>
|
|
||||||
<td>
|
<td>
|
||||||
{{ mailbox.created_at }}
|
<a href="?email={{ mailbox.email }}">{{ mailbox.email }}</a>
|
||||||
</td>
|
</td>
|
||||||
|
<td>{{ "Yes" if mailbox.verified else "No" }}</td>
|
||||||
|
<td>{{ mailbox.created_at }}</td>
|
||||||
</tr>
|
</tr>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
</tbody>
|
</tbody>
|
||||||
@ -82,25 +89,20 @@
|
|||||||
<table class="table">
|
<table class="table">
|
||||||
<thead>
|
<thead>
|
||||||
<tr>
|
<tr>
|
||||||
<th>
|
<th>Alias ID</th>
|
||||||
Alias ID
|
<th>Email</th>
|
||||||
</th>
|
<th>Enabled</th>
|
||||||
<th>
|
<th>Created At</th>
|
||||||
Email
|
|
||||||
</th>
|
|
||||||
<th>
|
|
||||||
Enabled
|
|
||||||
</th>
|
|
||||||
<th>
|
|
||||||
Created At
|
|
||||||
</th>
|
|
||||||
</tr>
|
</tr>
|
||||||
</thead>
|
</thead>
|
||||||
<tbody>
|
<tbody>
|
||||||
{% for alias in aliases %}
|
{% for alias in aliases %}
|
||||||
|
|
||||||
<tr>
|
<tr>
|
||||||
<td>{{ alias.id }}</td>
|
<td>{{ alias.id }}</td>
|
||||||
<td><a href="?email={{ alias.email }}">{{ alias.email }}</a></td>
|
<td>
|
||||||
|
<a href="?email={{ alias.email }}">{{ alias.email }}</a>
|
||||||
|
</td>
|
||||||
<td>{{ "Yes" if alias.enabled else "No" }}</td>
|
<td>{{ "Yes" if alias.enabled else "No" }}</td>
|
||||||
<td>{{ alias.created_at }}</td>
|
<td>{{ alias.created_at }}</td>
|
||||||
</tr>
|
</tr>
|
||||||
@ -174,10 +176,13 @@
|
|||||||
</thead>
|
</thead>
|
||||||
<tbody>
|
<tbody>
|
||||||
{% for entry in alias_audit_log %}
|
{% for entry in alias_audit_log %}
|
||||||
|
|
||||||
<tr>
|
<tr>
|
||||||
<td>{{ entry.user_id }}</td>
|
<td>{{ entry.user_id }}</td>
|
||||||
<td>{{ entry.alias_id }}</td>
|
<td>{{ entry.alias_id }}</td>
|
||||||
<td><a href="?email={{ entry.alias_email }}">{{ entry.alias_email }}</a></td>
|
<td>
|
||||||
|
<a href="?email={{ entry.alias_email }}">{{ entry.alias_email }}</a>
|
||||||
|
</td>
|
||||||
<td>{{ entry.action }}</td>
|
<td>{{ entry.action }}</td>
|
||||||
<td>{{ entry.message }}</td>
|
<td>{{ entry.message }}</td>
|
||||||
<td>{{ entry.created_at }}</td>
|
<td>{{ entry.created_at }}</td>
|
||||||
@ -199,8 +204,11 @@
|
|||||||
</thead>
|
</thead>
|
||||||
<tbody>
|
<tbody>
|
||||||
{% for entry in user_audit_log %}
|
{% for entry in user_audit_log %}
|
||||||
|
|
||||||
<tr>
|
<tr>
|
||||||
<td><a href="?email={{ entry.user_email }}">{{ entry.user_email }}</a></td>
|
<td>
|
||||||
|
<a href="?email={{ entry.user_email }}">{{ entry.user_email }}</a>
|
||||||
|
</td>
|
||||||
<td>{{ entry.action }}</td>
|
<td>{{ entry.action }}</td>
|
||||||
<td>{{ entry.message }}</td>
|
<td>{{ entry.message }}</td>
|
||||||
<td>{{ entry.created_at }}</td>
|
<td>{{ entry.created_at }}</td>
|
||||||
@ -224,11 +232,12 @@
|
|||||||
</form>
|
</form>
|
||||||
</div>
|
</div>
|
||||||
{% if data.no_match and email %}
|
{% if data.no_match and email %}
|
||||||
|
|
||||||
<div class="border border-dark border-2 mt-1 mb-2 p-3 alert alert-warning"
|
<div class="border border-dark border-2 mt-1 mb-2 p-3 alert alert-warning"
|
||||||
role="alert">No user, alias or mailbox found for {{ email }}</div>
|
role="alert">No user, alias or mailbox found for {{ email }}</div>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
|
||||||
{% if data.alias %}
|
{% if data.alias %}
|
||||||
|
|
||||||
<div class="border border-dark border-2 mt-1 mb-2 p-3">
|
<div class="border border-dark border-2 mt-1 mb-2 p-3">
|
||||||
<h3 class="mb-3">Found Alias {{ data.alias.email }}</h3>
|
<h3 class="mb-3">Found Alias {{ data.alias.email }}</h3>
|
||||||
{{ list_alias(1,[data.alias]) }}
|
{{ list_alias(1,[data.alias]) }}
|
||||||
@ -237,8 +246,8 @@
|
|||||||
{{ show_user(data.alias.user) }}
|
{{ show_user(data.alias.user) }}
|
||||||
</div>
|
</div>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
|
||||||
{% if data.user %}
|
{% if data.user %}
|
||||||
|
|
||||||
<div class="border border-dark border-2 mt-1 mb-2 p-3">
|
<div class="border border-dark border-2 mt-1 mb-2 p-3">
|
||||||
<h3 class="mb-3">Found User {{ data.user.email }}</h3>
|
<h3 class="mb-3">Found User {{ data.user.email }}</h3>
|
||||||
{{ show_user(data.user) }}
|
{{ show_user(data.user) }}
|
||||||
@ -247,12 +256,14 @@
|
|||||||
</div>
|
</div>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
{% if data.user_audit_log %}
|
{% if data.user_audit_log %}
|
||||||
|
|
||||||
<div class="border border-dark border-2 mt-1 mb-2 p-3">
|
<div class="border border-dark border-2 mt-1 mb-2 p-3">
|
||||||
<h3 class="mb-3">Audit log entries for user {{ data.query }}</h3>
|
<h3 class="mb-3">Audit log entries for user {{ data.query }}</h3>
|
||||||
{{ list_user_audit_log(data.user_audit_log) }}
|
{{ list_user_audit_log(data.user_audit_log) }}
|
||||||
</div>
|
</div>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
{% if data.mailbox_count > 10 %}
|
{% if data.mailbox_count > 10 %}
|
||||||
|
|
||||||
<h3>Found more than 10 mailboxes for {{ email }}. Showing the last 10</h3>
|
<h3>Found more than 10 mailboxes for {{ email }}. Showing the last 10</h3>
|
||||||
{% elif data.mailbox_count > 0 %}
|
{% elif data.mailbox_count > 0 %}
|
||||||
<h3>Found {{ data.mailbox_count }} mailbox(es) for {{ email }}</h3>
|
<h3>Found {{ data.mailbox_count }} mailbox(es) for {{ email }}</h3>
|
||||||
|
@ -43,7 +43,7 @@
|
|||||||
You can change the plan at any moment.
|
You can change the plan at any moment.
|
||||||
<br />
|
<br />
|
||||||
Please note that the new billing cycle starts instantly
|
Please note that the new billing cycle starts instantly
|
||||||
i.e. you will be charged <b>immediately</b> the annual fee ($30) when switching from monthly plan or vice-versa
|
i.e. you will be charged <b>immediately</b> the annual fee ($36) when switching from monthly plan or vice-versa
|
||||||
<b>without pro rata computation </b>.
|
<b>without pro rata computation </b>.
|
||||||
<br />
|
<br />
|
||||||
To change the plan you can also cancel the current one and subscribe a new one <b>by the end</b> of this plan.
|
To change the plan you can also cancel the current one and subscribe a new one <b>by the end</b> of this plan.
|
||||||
|
@ -94,4 +94,3 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
|
||||||
|
@ -91,7 +91,6 @@
|
|||||||
<br />
|
<br />
|
||||||
Some domain registrars (Namecheap, CloudFlare, etc) might also use <em>@</em> for the root domain.
|
Some domain registrars (Namecheap, CloudFlare, etc) might also use <em>@</em> for the root domain.
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{% for record in expected_mx_records %}
|
{% for record in expected_mx_records %}
|
||||||
|
|
||||||
<div class="mb-3 p-3 dns-record">
|
<div class="mb-3 p-3 dns-record">
|
||||||
@ -108,7 +107,6 @@
|
|||||||
data-clipboard-text="{{ record.domain }}">{{ record.domain }}</em>
|
data-clipboard-text="{{ record.domain }}">{{ record.domain }}</em>
|
||||||
</div>
|
</div>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
|
|
||||||
<form method="post" action="#mx-form">
|
<form method="post" action="#mx-form">
|
||||||
{{ csrf_form.csrf_token }}
|
{{ csrf_form.csrf_token }}
|
||||||
<input type="hidden" name="form-name" value="check-mx">
|
<input type="hidden" name="form-name" value="check-mx">
|
||||||
|
@ -22,7 +22,8 @@
|
|||||||
<p>Alternatively you can use your Proton credentials to ensure it's you.</p>
|
<p>Alternatively you can use your Proton credentials to ensure it's you.</p>
|
||||||
</div>
|
</div>
|
||||||
<a class="btn btn-primary btn-block mt-2 proton-button"
|
<a class="btn btn-primary btn-block mt-2 proton-button"
|
||||||
href="{{ url_for('auth.proton_login', next=next) }}" style="max-width: 400px">
|
href="{{ url_for('auth.proton_login', next=next) }}"
|
||||||
|
style="max-width: 400px">
|
||||||
<img class="mr-2" src="/static/images/proton.svg" />
|
<img class="mr-2" src="/static/images/proton.svg" />
|
||||||
Authenticate with Proton
|
Authenticate with Proton
|
||||||
</a>
|
</a>
|
||||||
|
@ -11,7 +11,7 @@
|
|||||||
<div>
|
<div>
|
||||||
<a class="buy-with-crypto"
|
<a class="buy-with-crypto"
|
||||||
data-custom="{{ current_user.id }}"
|
data-custom="{{ current_user.id }}"
|
||||||
href="{{ coinbase_url }}">Extend for 1 year - $30</a>
|
href="{{ coinbase_url }}">Extend for 1 year - $36</a>
|
||||||
<script src="https://commerce.coinbase.com/v1/checkout.js?version=201807"></script>
|
<script src="https://commerce.coinbase.com/v1/checkout.js?version=201807"></script>
|
||||||
</div>
|
</div>
|
||||||
<div class="mt-2">
|
<div class="mt-2">
|
||||||
|
@ -77,6 +77,11 @@
|
|||||||
<div class="text-center mx-md-auto mb-8 mt-6">
|
<div class="text-center mx-md-auto mb-8 mt-6">
|
||||||
<h1>Upgrade to unlock premium features</h1>
|
<h1>Upgrade to unlock premium features</h1>
|
||||||
</div>
|
</div>
|
||||||
|
<div class="alert alert-info">
|
||||||
|
<span class="badge badge-success">new</span> SimpleLogin Premium now includes Proton Pass premium features.
|
||||||
|
<a href="https://simplelogin.io/blog/sl-premium-including-pass-plus/"
|
||||||
|
target="_blank">Learn more ↗</a>
|
||||||
|
</div>
|
||||||
{% if manual_sub %}
|
{% if manual_sub %}
|
||||||
|
|
||||||
<div class="alert alert-info mt-0 mb-6">
|
<div class="alert alert-info mt-0 mb-6">
|
||||||
@ -306,7 +311,7 @@
|
|||||||
<div class="card-body">
|
<div class="card-body">
|
||||||
<div class="text-center">
|
<div class="text-center">
|
||||||
<div class="h3">SimpleLogin Premium</div>
|
<div class="h3">SimpleLogin Premium</div>
|
||||||
<div class="h3 my-3">$30 / year</div>
|
<div class="h3 my-3">$36 / year</div>
|
||||||
<div class="text-center mt-4 mb-6">
|
<div class="text-center mt-4 mb-6">
|
||||||
<button class="btn btn-primary btn-lg w-100"
|
<button class="btn btn-primary btn-lg w-100"
|
||||||
onclick="upgradePaddle({{ PADDLE_YEARLY_PRODUCT_ID }})">Upgrade to Premium</button>
|
onclick="upgradePaddle({{ PADDLE_YEARLY_PRODUCT_ID }})">Upgrade to Premium</button>
|
||||||
@ -471,7 +476,7 @@
|
|||||||
rel="noopener noreferrer">
|
rel="noopener noreferrer">
|
||||||
Upgrade to Premium - cryptocurrency
|
Upgrade to Premium - cryptocurrency
|
||||||
<br />
|
<br />
|
||||||
$30 / year
|
$36 / year
|
||||||
<i class="fe fe-external-link"></i>
|
<i class="fe fe-external-link"></i>
|
||||||
</a>
|
</a>
|
||||||
</div>
|
</div>
|
||||||
|
@ -511,6 +511,19 @@ def test_create_contact_route_invalid_alias(flask_client):
|
|||||||
assert r.status_code == 403
|
assert r.status_code == 403
|
||||||
|
|
||||||
|
|
||||||
|
def test_create_contact_route_non_existing_alias(flask_client):
|
||||||
|
user, api_key = get_new_user_and_api_key()
|
||||||
|
Session.commit()
|
||||||
|
|
||||||
|
r = flask_client.post(
|
||||||
|
url_for("api.create_contact_route", alias_id=99999999),
|
||||||
|
headers={"Authentication": api_key.code},
|
||||||
|
json={"contact": "First Last <first@example.com>"},
|
||||||
|
)
|
||||||
|
|
||||||
|
assert r.status_code == 403
|
||||||
|
|
||||||
|
|
||||||
def test_create_contact_route_free_users(flask_client):
|
def test_create_contact_route_free_users(flask_client):
|
||||||
user, api_key = get_new_user_and_api_key()
|
user, api_key = get_new_user_and_api_key()
|
||||||
|
|
||||||
|
40
app/tests/jobs/test_send_event_to_webhook.py
Normal file
40
app/tests/jobs/test_send_event_to_webhook.py
Normal file
@ -0,0 +1,40 @@
|
|||||||
|
import arrow
|
||||||
|
|
||||||
|
from app import config
|
||||||
|
from app.events.generated.event_pb2 import EventContent, AliasDeleted
|
||||||
|
from app.jobs.send_event_job import SendEventToWebhookJob
|
||||||
|
from app.models import PartnerUser
|
||||||
|
from app.proton.utils import get_proton_partner
|
||||||
|
from events.event_sink import ConsoleEventSink
|
||||||
|
from tests.utils import create_new_user, random_token
|
||||||
|
|
||||||
|
|
||||||
|
def test_serialize_and_deserialize_job():
|
||||||
|
user = create_new_user()
|
||||||
|
alias_id = 34
|
||||||
|
alias_email = "a@b.c"
|
||||||
|
event = EventContent(alias_deleted=AliasDeleted(id=alias_id, email=alias_email))
|
||||||
|
run_at = arrow.now().shift(hours=10)
|
||||||
|
db_job = SendEventToWebhookJob(user, event).store_job_in_db(run_at=run_at)
|
||||||
|
assert db_job.run_at == run_at
|
||||||
|
assert db_job.name == config.JOB_SEND_EVENT_TO_WEBHOOK
|
||||||
|
job = SendEventToWebhookJob.create_from_job(db_job)
|
||||||
|
assert job._user.id == user.id
|
||||||
|
assert job._event.alias_deleted.id == alias_id
|
||||||
|
assert job._event.alias_deleted.email == alias_email
|
||||||
|
|
||||||
|
|
||||||
|
def test_send_event_to_webhook():
|
||||||
|
user = create_new_user()
|
||||||
|
PartnerUser.create(
|
||||||
|
user_id=user.id,
|
||||||
|
partner_id=get_proton_partner().id,
|
||||||
|
external_user_id=random_token(10),
|
||||||
|
flush=True,
|
||||||
|
)
|
||||||
|
alias_id = 34
|
||||||
|
alias_email = "a@b.c"
|
||||||
|
event = EventContent(alias_deleted=AliasDeleted(id=alias_id, email=alias_email))
|
||||||
|
job = SendEventToWebhookJob(user, event)
|
||||||
|
sink = ConsoleEventSink()
|
||||||
|
assert job.run(sink)
|
@ -314,10 +314,13 @@ def test_verify_too_may():
|
|||||||
output = mailbox_utils.create_mailbox(user, random_email())
|
output = mailbox_utils.create_mailbox(user, random_email())
|
||||||
output.activation.tries = mailbox_utils.MAX_ACTIVATION_TRIES
|
output.activation.tries = mailbox_utils.MAX_ACTIVATION_TRIES
|
||||||
Session.commit()
|
Session.commit()
|
||||||
with pytest.raises(mailbox_utils.CannotVerifyError):
|
try:
|
||||||
mailbox_utils.verify_mailbox_code(
|
mailbox_utils.verify_mailbox_code(
|
||||||
user, output.mailbox.id, output.activation.code
|
user, output.mailbox.id, output.activation.code
|
||||||
)
|
)
|
||||||
|
assert False
|
||||||
|
except mailbox_utils.CannotVerifyError as e:
|
||||||
|
assert e.deleted_activation_code
|
||||||
|
|
||||||
|
|
||||||
@mail_sender.store_emails_test_decorator
|
@mail_sender.store_emails_test_decorator
|
||||||
|
Reference in New Issue
Block a user