mirror of
https://github.com/simple-login/app.git
synced 2025-10-06 13:26:51 +08:00
Add job to send events to webhook (#2282)
* Add job to send events to webhook * Apply comments --------- Co-authored-by: Carlos Quintana <74399022+cquintana92@users.noreply.github.com>
This commit is contained in:
parent
9646f84b79
commit
ccd687f091
5 changed files with 132 additions and 2 deletions
|
@ -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
|
||||||
|
|
70
app/jobs/send_event_job.py
Normal file
70
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,
|
||||||
|
)
|
|
@ -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)
|
||||||
|
|
||||||
|
|
40
tests/jobs/test_send_event_to_webhook.py
Normal file
40
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)
|
Loading…
Add table
Reference in a new issue