-
-
Notifications
You must be signed in to change notification settings - Fork 5
/
Copy path__init__.py
100 lines (84 loc) · 3.34 KB
/
__init__.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
import asyncio
import logging
import sys
from aiohttp import web
import raven
import raven_aiohttp
from raven.conf import setup_logging
from raven.handlers.logging import SentryHandler
@web.middleware
class SentryMiddleware:
def __init__(
self,
sentry_kwargs=None,
*,
install_excepthook=True,
loop=None,
patch_logging=False,
sentry_log_level=logging.ERROR
):
if sentry_kwargs is None:
sentry_kwargs = {}
sentry_kwargs = {
"transport": raven_aiohttp.AioHttpTransport,
"enable_breadcrumbs": False,
# by default, do not let raven.Client install its own excepthook
"install_sys_hook": not install_excepthook,
**sentry_kwargs,
}
self.client = raven.Client(**sentry_kwargs)
# Setup Sentry logger - https://docs.sentry.io/clients/python/integrations/#logging
if patch_logging:
handler = SentryHandler(client=self.client)
handler.setLevel(sentry_log_level)
setup_logging(handler)
if install_excepthook:
self.update_excepthook(loop)
def update_excepthook(self, loop=None):
"""Update sys.excepthook so that it closes the Sentry transport.
If a custom event loop is provided,
it will be used for closing the Sentry transport
derived from raven_aiohttp.AioHttpTransportBase
instead of the event loop for the current context
defined by the current event loop policy.
"""
original_excepthook = sys.excepthook
def aiohttp_transport_excepthook(*exc_info):
"""An aiohttp-transport-friendly excepthook.
It closes the transport, which delivers the messages."""
self.client.captureException(exc_info=exc_info, level="fatal")
transport = self.client.remote.get_transport()
if isinstance(transport, raven_aiohttp.AioHttpTransportBase):
event_loop = loop
if event_loop is None:
event_loop = asyncio.get_event_loop()
# wait for Sentry transport to send the outstanding messages
event_loop.run_until_complete(transport.close())
# the idea of using the original excepthook
# was taken from raven-python repository:
# https://github.com/getsentry/raven-python/blob/f6d79c3b/raven/base.py#L280
original_excepthook(*exc_info)
sys.excepthook = aiohttp_transport_excepthook
async def __call__(self, request, handler):
try:
return await handler(request)
except:
extra_data = await self.get_extra_data(request)
self.client.captureException(data=extra_data)
raise
async def get_extra_data(self, request):
data = {
"request": {
"query_string": request.query_string,
"cookies": request.headers.get("Cookie", ""),
"headers": dict(request.headers),
"url": request.path,
"method": request.method,
}
}
if request.transport:
data["request"]["env"] = {
"REMOTE_ADDR": request.transport.get_extra_info("peername")[0]
}
data["request"]["scheme"] = request.scheme
return data