-
Notifications
You must be signed in to change notification settings - Fork 4
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Add notification for third party services #7
Changes from 2 commits
fc6bcb4
5420ceb
3df2275
9d826b5
39d27ef
c0024fa
5175359
3586c1c
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,51 @@ | ||
import json | ||
import logging | ||
import threading | ||
from requests.exceptions import HTTPError, ConnectionError, Timeout, RequestException | ||
from .operations import setup_requests_session | ||
|
||
|
||
class NotifyService: | ||
def __init__(self, url=None, token=None) -> None: | ||
self.url = url | ||
self.token = token | ||
self.session = None | ||
|
||
def send(self, payload): | ||
try: | ||
self.session.post(self.url, data=json.dumps({"payload": payload})) | ||
except HTTPError as e: | ||
logging.error("Invalid response from notify: %s", str(e), exc_info=True) | ||
except ConnectionError as e: | ||
logging.error("Could not connect to notify server: %s", str(e), exc_info=True) | ||
except Timeout as e: | ||
logging.error("Notify server timed out: %s", str(e), exc_info=True) | ||
except RequestException as e: | ||
logging.error("Failed to send notify payload %s", str(e), exc_info=True) | ||
|
||
def notify(self, payload): | ||
if self.url is None or self.url == '': | ||
return | ||
|
||
if self.session is None: | ||
self.session = setup_requests_session(self.token) | ||
|
||
threading.Thread(target=self.send, args=(payload,)).start() | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I wonder... is all this extra complexity really needed? To me it sounds like the whole code in here could be done with a simple def notify(payload):
sess = ...
try:
sess.post(..., timeout=5)
except ...:
pass # log it no threads, no flask extension, no context etc needed ;) There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Sure I thought I might be over-complicating this, but I thought I'd be adding the notify to each of the endpoints, or maybe extending the logic out if required. The reason why I use an extension, I needed to load in the environmental config for the notify URL. So I figured I'd need an extension to load this in? Should I just load from process.env from anywhere? The threading is there as I wanted this to me fire and forget so there would be no slow down to this service if our downstream system was offline or slow to response. The two reasons I'm using a session is that I am able to config the authentication at a parent level using the method you already have, and depending on the traffic in this service, having a common session between requests reduces http overhead. I'm not sure how it cleans itself up though, so I'm happy to simply this if you do not believe there is much traffic? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
Yes, you can read env vars anywhere. Or even put it in the
Fair enough, I don't know how reliable your downstream is. Personally I'd probably just use a low-enough timeout and then only log request-related errors so they don't break the flow.
Actually, I'm not sure if the session actually keeps connections alive etc. or not. Using it for the token is a good idea though! There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Is this call to the microservice blocking, as in, will buttons take longer to load if we put in a downstream call? I expect that the (yet-to-be-built) downstream service may be hosted on the other side of the planet, also I really didn't want to be responsible for an extra 2-5 second load times on some buttons, or prematurely timeout due to distance. That being said, I'm also no python guru, so maybe threads are not the best tool for the job here (due to leaky threads or contention issues)? So if you feel strongly about this I will just change it. Also regarding the use of an extension, I was just following a pattern I'm familiar with from other (non-python) frameworks, so if you want me to simplify it, no problem, I'm just not sure if you comment was more of an observation or a suggestion for change. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. the buttons won't take longer to load since you're not doing anything to the |
||
|
||
|
||
class NotifyExtension: | ||
def __init__(self, context=None, notify_url=None, notify_token=None): | ||
self.url = notify_url | ||
self.token = notify_token | ||
self.service = None | ||
if context is not None: | ||
self.init_app(context) | ||
|
||
def init_app(self, context): | ||
joshpme marked this conversation as resolved.
Show resolved
Hide resolved
|
||
context.config.setdefault('NOTIFY_URL', self.url) | ||
context.config.setdefault('NOTIFY_TOKEN', self.token) | ||
self.service = NotifyService( | ||
context.config['NOTIFY_URL'], | ||
context.config['NOTIFY_TOKEN'] | ||
) | ||
context.extensions['notifier'] = self.service |
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -44,6 +44,12 @@ | |
) | ||
|
||
|
||
def notify(context, payload): | ||
service = context.extensions.get('notifier') | ||
if service is not None: | ||
service.notify(payload) | ||
|
||
|
||
def require_event_token(fn): | ||
@wraps(fn) | ||
def wrapper(*args, **kwargs): | ||
|
@@ -285,6 +291,17 @@ def review_editable( | |
"A new revision %r was submitted for contribution %r", revision_id, contrib_id | ||
) | ||
resp = {} | ||
|
||
notify(current_app, { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. you might want to consider calling this in some of the other methods, such as |
||
"event": event.identifier, | ||
"contrib_id": contrib_id, | ||
"revision_id": revision_id, | ||
"action": action, | ||
"editable_type": editable_type, | ||
"user": user, | ||
"request": request.json | ||
}) | ||
|
||
if action == "accept": | ||
resp = process_accepted_revision(event, revision) | ||
|
||
|
@@ -317,7 +334,7 @@ def remove_editable(event, contrib_id, editable_type): | |
|
||
|
||
@api.route( | ||
"/event/<identifier>/editable/<any(paper,slides,poster):editable_type>/<contrib_id>/<revision_id>/actions", # noqa: E501 | ||
"/event/<identifier>/editable/<any(paper,slides,poster):editable_type>/<contrib_id>/<revision_id>/actions", | ||
methods=("POST",), | ||
) | ||
@use_kwargs(ServiceActionsRequestSchema(unknown=EXCLUDE), location="json") | ||
|
@@ -360,7 +377,7 @@ def get_custom_revision_actions( | |
|
||
|
||
@api.route( | ||
"/event/<identifier>/editable/<any(paper,slides,poster):editable_type>/<contrib_id>/<revision_id>/action", # noqa: E501 | ||
"/event/<identifier>/editable/<any(paper,slides,poster):editable_type>/<contrib_id>/<revision_id>/action", | ||
methods=("POST",), | ||
) | ||
@use_kwargs(ServiceTriggerActionRequestSchema(unknown=EXCLUDE), location="json") | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Use
os.environ[...]
if you expect this to always be presentThere was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This will throw an exception if the keys are not present, where as get will soft return None
I explicitly handle none, plus as these variables can be configured with config instead of environmental variables if that is needed later.
Is there a reason why you prefer to use a dict lookup over get?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I generally prefer to fail early: If required config keys are missing, I want it to fail at startup time. That's something I see immediately because the app is clearly broken.
If the missing value leaks into the config and it only fails later when trying to send a request, there's a good chance I won't notice it until a real request from a user's action arrives.
That said, I think it makes sense to simply disable notifications if these settings are missing. I'd do that by not initializing the extension and silently doing nothing later when the extension is not on the app context (based on your current architecture). Without the extension structure I'd simply return early in the function to send a notification if the config values are None,
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yeah, fair. I'll refactor this now then.