|
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