Skip to content

Commit 339c635

Browse files
authored
Merge branch 'open-telemetry:main' into main
2 parents ce1f0b1 + e49806e commit 339c635

File tree

10 files changed

+324
-31
lines changed

10 files changed

+324
-31
lines changed

CHANGELOG.md

Lines changed: 7 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,8 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
2323
([#2879](https://github.com/open-telemetry/opentelemetry-python-contrib/pull/2879))
2424
- `opentelemetry-instrumentation-redis` Add additional attributes for methods create_index and search, rename those spans
2525
([#2635](https://github.com/open-telemetry/opentelemetry-python-contrib/pull/2635))
26+
- `opentelemetry-instrumentation` Add support for string based dotted module paths in unwrap
27+
([#2919](https://github.com/open-telemetry/opentelemetry-python-contrib/pull/2919))
2628

2729
### Fixed
2830

@@ -82,6 +84,9 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
8284
([#2753](https://github.com/open-telemetry/opentelemetry-python-contrib/pull/2753))
8385
- `opentelemetry-instrumentation-grpc` Fix grpc supported version
8486
([#2845](https://github.com/open-telemetry/opentelemetry-python-contrib/pull/2845))
87+
- `opentelemetry-instrumentation-asyncio` fix `AttributeError` in
88+
`AsyncioInstrumentor.trace_to_thread` when `func` is a `functools.partial` instance
89+
([#2911](https://github.com/open-telemetry/opentelemetry-python-contrib/pull/2911))
8590

8691
## Version 1.26.0/0.47b0 (2024-07-23)
8792

@@ -414,6 +419,8 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
414419
([#1879](https://github.com/open-telemetry/opentelemetry-python-contrib/pull/1879))
415420
- Add optional distro and configurator selection for auto-instrumentation
416421
([#1823](https://github.com/open-telemetry/opentelemetry-python-contrib/pull/1823))
422+
- `opentelemetry-instrumentation-django` - Add option to add Opentelemetry middleware at specific position in middleware chain
423+
([#2912]https://github.com/open-telemetry/opentelemetry-python-contrib/pull/2912)
417424

418425
### Added
419426

instrumentation/opentelemetry-instrumentation-asyncio/src/opentelemetry/instrumentation/asyncio/__init__.py

Lines changed: 7 additions & 5 deletions
Original file line numberDiff line numberDiff line change
@@ -78,6 +78,7 @@ def func():
7878
"""
7979

8080
import asyncio
81+
import functools
8182
import sys
8283
from asyncio import futures
8384
from timeit import default_timer
@@ -231,14 +232,15 @@ def wrap_taskgroup_create_task(method, instance, args, kwargs) -> None:
231232
def trace_to_thread(self, func: callable):
232233
"""Trace a function."""
233234
start = default_timer()
235+
func_name = getattr(func, "__name__", None)
236+
if func_name is None and isinstance(func, functools.partial):
237+
func_name = func.func.__name__
234238
span = (
235-
self._tracer.start_span(
236-
f"{ASYNCIO_PREFIX} to_thread-" + func.__name__
237-
)
238-
if func.__name__ in self._to_thread_name_to_trace
239+
self._tracer.start_span(f"{ASYNCIO_PREFIX} to_thread-" + func_name)
240+
if func_name in self._to_thread_name_to_trace
239241
else None
240242
)
241-
attr = {"type": "to_thread", "name": func.__name__}
243+
attr = {"type": "to_thread", "name": func_name}
242244
exception = None
243245
try:
244246
attr["state"] = "finished"

instrumentation/opentelemetry-instrumentation-asyncio/tests/test_asyncio_to_thread.py

Lines changed: 35 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,7 @@
1212
# See the License for the specific language governing permissions and
1313
# limitations under the License.
1414
import asyncio
15+
import functools
1516
import sys
1617
from unittest import skipIf
1718
from unittest.mock import patch
@@ -72,3 +73,37 @@ async def to_thread():
7273
for point in metric.data.data_points:
7374
self.assertEqual(point.attributes["type"], "to_thread")
7475
self.assertEqual(point.attributes["name"], "multiply")
76+
77+
@skipIf(
78+
sys.version_info < (3, 9), "to_thread is only available in Python 3.9+"
79+
)
80+
def test_to_thread_partial_func(self):
81+
def multiply(x, y):
82+
return x * y
83+
84+
double = functools.partial(multiply, 2)
85+
86+
async def to_thread():
87+
result = await asyncio.to_thread(double, 3)
88+
assert result == 6
89+
90+
with self._tracer.start_as_current_span("root"):
91+
asyncio.run(to_thread())
92+
spans = self.memory_exporter.get_finished_spans()
93+
94+
self.assertEqual(len(spans), 2)
95+
assert spans[0].name == "asyncio to_thread-multiply"
96+
for metric in (
97+
self.memory_metrics_reader.get_metrics_data()
98+
.resource_metrics[0]
99+
.scope_metrics[0]
100+
.metrics
101+
):
102+
if metric.name == "asyncio.process.duration":
103+
for point in metric.data.data_points:
104+
self.assertEqual(point.attributes["type"], "to_thread")
105+
self.assertEqual(point.attributes["name"], "multiply")
106+
if metric.name == "asyncio.process.created":
107+
for point in metric.data.data_points:
108+
self.assertEqual(point.attributes["type"], "to_thread")
109+
self.assertEqual(point.attributes["name"], "multiply")

instrumentation/opentelemetry-instrumentation-django/src/opentelemetry/instrumentation/django/__init__.py

Lines changed: 34 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -285,6 +285,30 @@ def _get_django_middleware_setting() -> str:
285285
return "MIDDLEWARE"
286286

287287

288+
def _get_django_otel_middleware_position(
289+
middleware_length, default_middleware_position=0
290+
):
291+
otel_position = environ.get("OTEL_PYTHON_DJANGO_MIDDLEWARE_POSITION")
292+
try:
293+
middleware_position = int(otel_position)
294+
except (ValueError, TypeError):
295+
_logger.debug(
296+
"Invalid OTEL_PYTHON_DJANGO_MIDDLEWARE_POSITION value: (%s). Using default position: %d.",
297+
otel_position,
298+
default_middleware_position,
299+
)
300+
middleware_position = default_middleware_position
301+
302+
if middleware_position < 0 or middleware_position > middleware_length:
303+
_logger.debug(
304+
"Middleware position %d is out of range (0-%d). Using 0 as the position",
305+
middleware_position,
306+
middleware_length,
307+
)
308+
middleware_position = 0
309+
return middleware_position
310+
311+
288312
class DjangoInstrumentor(BaseInstrumentor):
289313
"""An instrumentor for Django
290314
@@ -388,10 +412,18 @@ def _instrument(self, **kwargs):
388412

389413
is_sql_commentor_enabled = kwargs.pop("is_sql_commentor_enabled", None)
390414

415+
middleware_position = _get_django_otel_middleware_position(
416+
len(settings_middleware), kwargs.pop("middleware_position", 0)
417+
)
418+
391419
if is_sql_commentor_enabled:
392-
settings_middleware.insert(0, self._sql_commenter_middleware)
420+
settings_middleware.insert(
421+
middleware_position, self._sql_commenter_middleware
422+
)
393423

394-
settings_middleware.insert(0, self._opentelemetry_middleware)
424+
settings_middleware.insert(
425+
middleware_position, self._opentelemetry_middleware
426+
)
395427

396428
setattr(settings, _middleware_setting, settings_middleware)
397429

instrumentation/opentelemetry-instrumentation-django/tests/test_middleware.py

Lines changed: 40 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -157,6 +157,46 @@ def tearDownClass(cls):
157157
super().tearDownClass()
158158
conf.settings = conf.LazySettings()
159159

160+
def test_middleware_added_at_position(self):
161+
_django_instrumentor.uninstrument()
162+
if DJANGO_2_0:
163+
middleware = conf.settings.MIDDLEWARE
164+
else:
165+
middleware = conf.settings.MIDDLEWARE_CLASSES
166+
# adding two dummy middlewares
167+
temprory_middelware = "django.utils.deprecation.MiddlewareMixin"
168+
middleware.append(temprory_middelware)
169+
middleware.append(temprory_middelware)
170+
171+
middleware_position = 1
172+
_django_instrumentor.instrument(
173+
middleware_position=middleware_position
174+
)
175+
self.assertEqual(
176+
middleware[middleware_position],
177+
"opentelemetry.instrumentation.django.middleware.otel_middleware._DjangoMiddleware",
178+
)
179+
180+
def test_middleware_added_at_position_if_wrong_position(self):
181+
_django_instrumentor.uninstrument()
182+
if DJANGO_2_0:
183+
middleware = conf.settings.MIDDLEWARE
184+
else:
185+
middleware = conf.settings.MIDDLEWARE_CLASSES
186+
# adding middleware
187+
temprory_middelware = "django.utils.deprecation.MiddlewareMixin"
188+
middleware.append(temprory_middelware)
189+
middleware_position = (
190+
756 # wrong position out of bound of middleware length
191+
)
192+
_django_instrumentor.instrument(
193+
middleware_position=middleware_position
194+
)
195+
self.assertEqual(
196+
middleware[0],
197+
"opentelemetry.instrumentation.django.middleware.otel_middleware._DjangoMiddleware",
198+
)
199+
160200
def test_templated_route_get(self):
161201
Client().get("/route/2020/template/")
162202

instrumentation/opentelemetry-instrumentation-django/tests/test_sqlcommenter.py

Lines changed: 31 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -72,6 +72,37 @@ def test_middleware_added(self, sqlcommenter_middleware):
7272
in middleware
7373
)
7474

75+
@patch(
76+
"opentelemetry.instrumentation.django.middleware.sqlcommenter_middleware.SqlCommenter"
77+
)
78+
def test_middleware_added_at_position(self, sqlcommenter_middleware):
79+
_django_instrumentor.uninstrument()
80+
if DJANGO_2_0:
81+
middleware = conf.settings.MIDDLEWARE
82+
else:
83+
middleware = conf.settings.MIDDLEWARE_CLASSES
84+
85+
# adding two dummy middlewares
86+
temprory_middelware = "django.utils.deprecation.MiddlewareMixin"
87+
middleware.append(temprory_middelware)
88+
middleware.append(temprory_middelware)
89+
90+
middleware_position = 1
91+
_django_instrumentor.instrument(
92+
is_sql_commentor_enabled=True,
93+
middleware_position=middleware_position,
94+
)
95+
instance = sqlcommenter_middleware.return_value
96+
instance.get_response = HttpResponse()
97+
self.assertEqual(
98+
middleware[middleware_position],
99+
"opentelemetry.instrumentation.django.middleware.otel_middleware._DjangoMiddleware",
100+
)
101+
self.assertEqual(
102+
middleware[middleware_position + 1],
103+
"opentelemetry.instrumentation.django.middleware.sqlcommenter_middleware.SqlCommenter",
104+
)
105+
75106
@patch(
76107
"opentelemetry.instrumentation.django.middleware.sqlcommenter_middleware._get_opentelemetry_values"
77108
)

opentelemetry-instrumentation/src/opentelemetry/instrumentation/bootstrap.py

Lines changed: 27 additions & 11 deletions
Original file line numberDiff line numberDiff line change
@@ -22,12 +22,15 @@
2222
SubprocessError,
2323
check_call,
2424
)
25+
from typing import Optional
2526

2627
from packaging.requirements import Requirement
2728

2829
from opentelemetry.instrumentation.bootstrap_gen import (
29-
default_instrumentations,
30-
libraries,
30+
default_instrumentations as gen_default_instrumentations,
31+
)
32+
from opentelemetry.instrumentation.bootstrap_gen import (
33+
libraries as gen_libraries,
3134
)
3235
from opentelemetry.instrumentation.version import __version__
3336
from opentelemetry.util._importlib_metadata import (
@@ -75,7 +78,7 @@ def _sys_pip_install(package):
7578
print(error)
7679

7780

78-
def _pip_check():
81+
def _pip_check(libraries):
7982
"""Ensures none of the instrumentations have dependency conflicts.
8083
Clean check reported as:
8184
'No broken requirements found.'
@@ -113,7 +116,7 @@ def _is_installed(req):
113116
return True
114117

115118

116-
def _find_installed_libraries():
119+
def _find_installed_libraries(default_instrumentations, libraries):
117120
for lib in default_instrumentations:
118121
yield lib
119122

@@ -122,18 +125,25 @@ def _find_installed_libraries():
122125
yield lib["instrumentation"]
123126

124127

125-
def _run_requirements():
128+
def _run_requirements(default_instrumentations, libraries):
126129
logger.setLevel(logging.ERROR)
127-
print("\n".join(_find_installed_libraries()))
130+
print(
131+
"\n".join(
132+
_find_installed_libraries(default_instrumentations, libraries)
133+
)
134+
)
128135

129136

130-
def _run_install():
131-
for lib in _find_installed_libraries():
137+
def _run_install(default_instrumentations, libraries):
138+
for lib in _find_installed_libraries(default_instrumentations, libraries):
132139
_sys_pip_install(lib)
133-
_pip_check()
140+
_pip_check(libraries)
134141

135142

136-
def run() -> None:
143+
def run(
144+
default_instrumentations: Optional[list] = None,
145+
libraries: Optional[list] = None,
146+
) -> None:
137147
action_install = "install"
138148
action_requirements = "requirements"
139149

@@ -163,8 +173,14 @@ def run() -> None:
163173
)
164174
args = parser.parse_args()
165175

176+
if libraries is None:
177+
libraries = gen_libraries
178+
179+
if default_instrumentations is None:
180+
default_instrumentations = gen_default_instrumentations
181+
166182
cmd = {
167183
action_install: _run_install,
168184
action_requirements: _run_requirements,
169185
}[args.action]
170-
cmd()
186+
cmd(default_instrumentations, libraries)

opentelemetry-instrumentation/src/opentelemetry/instrumentation/utils.py

Lines changed: 21 additions & 3 deletions
Original file line numberDiff line numberDiff line change
@@ -14,8 +14,9 @@
1414

1515
import urllib.parse
1616
from contextlib import contextmanager
17+
from importlib import import_module
1718
from re import escape, sub
18-
from typing import Dict, Iterable, Sequence
19+
from typing import Dict, Iterable, Sequence, Union
1920

2021
from wrapt import ObjectProxy
2122

@@ -80,13 +81,30 @@ def http_status_to_status_code(
8081
return StatusCode.ERROR
8182

8283

83-
def unwrap(obj, attr: str):
84+
def unwrap(obj: Union[object, str], attr: str):
8485
"""Given a function that was wrapped by wrapt.wrap_function_wrapper, unwrap it
8586
87+
The object containing the function to unwrap may be passed as dotted module path string.
88+
8689
Args:
87-
obj: Object that holds a reference to the wrapped function
90+
obj: Object that holds a reference to the wrapped function or dotted import path as string
8891
attr (str): Name of the wrapped function
8992
"""
93+
if isinstance(obj, str):
94+
try:
95+
module_path, class_name = obj.rsplit(".", 1)
96+
except ValueError as exc:
97+
raise ImportError(
98+
f"Cannot parse '{obj}' as dotted import path"
99+
) from exc
100+
module = import_module(module_path)
101+
try:
102+
obj = getattr(module, class_name)
103+
except AttributeError as exc:
104+
raise ImportError(
105+
f"Cannot import '{class_name}' from '{module}'"
106+
) from exc
107+
90108
func = getattr(obj, attr, None)
91109
if func and isinstance(func, ObjectProxy) and hasattr(func, "__wrapped__"):
92110
setattr(obj, attr, func.__wrapped__)

0 commit comments

Comments
 (0)