|
5 | 5 | if MYPY: |
6 | 6 | from typing import Any |
7 | 7 |
|
8 | | - from django.urls.resolvers import ResolverMatch |
9 | 8 |
|
10 | | - |
11 | | -def patch_resolver(): |
| 9 | +def patch_views(): |
12 | 10 | # type: () -> None |
13 | | - try: |
14 | | - from django.urls.resolvers import URLResolver |
15 | | - except ImportError: |
16 | | - try: |
17 | | - from django.urls.resolvers import RegexURLResolver as URLResolver |
18 | | - except ImportError: |
19 | | - from django.core.urlresolvers import RegexURLResolver as URLResolver |
20 | 11 |
|
| 12 | + from django.core.handlers.base import BaseHandler |
21 | 13 | from sentry_sdk.integrations.django import DjangoIntegration |
22 | 14 |
|
23 | | - old_resolve = URLResolver.resolve |
24 | | - |
25 | | - def resolve(self, path): |
26 | | - # type: (URLResolver, Any) -> ResolverMatch |
27 | | - hub = Hub.current |
28 | | - integration = hub.get_integration(DjangoIntegration) |
29 | | - |
30 | | - if integration is None or not integration.middleware_spans: |
31 | | - return old_resolve(self, path) |
| 15 | + old_make_view_atomic = BaseHandler.make_view_atomic |
32 | 16 |
|
33 | | - return _wrap_resolver_match(hub, old_resolve(self, path)) |
| 17 | + @_functools.wraps(old_make_view_atomic) |
| 18 | + def sentry_patched_make_view_atomic(self, *args, **kwargs): |
| 19 | + # type: (Any, *Any, **Any) -> Any |
| 20 | + callback = old_make_view_atomic(self, *args, **kwargs) |
34 | 21 |
|
35 | | - URLResolver.resolve = resolve |
| 22 | + # XXX: The wrapper function is created for every request. Find more |
| 23 | + # efficient way to wrap views (or build a cache?) |
36 | 24 |
|
| 25 | + hub = Hub.current |
| 26 | + integration = hub.get_integration(DjangoIntegration) |
37 | 27 |
|
38 | | -def _wrap_resolver_match(hub, resolver_match): |
39 | | - # type: (Hub, ResolverMatch) -> ResolverMatch |
40 | | - |
41 | | - # XXX: The wrapper function is created for every request. Find more |
42 | | - # efficient way to wrap views (or build a cache?) |
43 | | - |
44 | | - old_callback = resolver_match.func |
| 28 | + if integration is not None and integration.middleware_spans: |
45 | 29 |
|
46 | | - # Explicitly forward `csrf_exempt` in case it is not an attribute in |
47 | | - # callback.__dict__, but rather a class attribute (on a class |
48 | | - # implementing __call__) such as this: |
49 | | - # |
50 | | - # class Foo(object): |
51 | | - # csrf_exempt = True |
52 | | - # |
53 | | - # def __call__(self, request): ... |
54 | | - # |
55 | | - # We have had this in the Sentry codebase (for no good reason, but |
56 | | - # nevertheless we broke user code) |
57 | | - assigned = _functools.WRAPPER_ASSIGNMENTS + ("csrf_exempt",) |
| 30 | + @_functools.wraps(callback) |
| 31 | + def sentry_wrapped_callback(request, *args, **kwargs): |
| 32 | + # type: (Any, *Any, **Any) -> Any |
| 33 | + with hub.start_span( |
| 34 | + op="django.view", description=request.resolver_match.view_name |
| 35 | + ): |
| 36 | + return callback(request, *args, **kwargs) |
58 | 37 |
|
59 | | - @_functools.wraps(old_callback, assigned=assigned) |
60 | | - def callback(*args, **kwargs): |
61 | | - # type: (*Any, **Any) -> Any |
62 | | - with hub.start_span(op="django.view", description=resolver_match.view_name): |
63 | | - return old_callback(*args, **kwargs) |
| 38 | + else: |
| 39 | + sentry_wrapped_callback = callback |
64 | 40 |
|
65 | | - resolver_match.func = callback |
| 41 | + return sentry_wrapped_callback |
66 | 42 |
|
67 | | - return resolver_match |
| 43 | + BaseHandler.make_view_atomic = sentry_patched_make_view_atomic |
0 commit comments