Skip to content

Commit ca10b51

Browse files
committedJun 21, 2018
POC Tracing support
Implements a new tracing support for uvloop adding the pair methods `start_tracing` and `stop_tracing` that allows the user to start or stop the tracing. The user must provide a valid `uvloop.tracing.Tracer` implementation that would be used internally by uvloop to create spans, each span must also meet the `uvloop.tracing.Span` class contract. This POC only implements the creation of spans during underlying calls to the `getaddrinfo` function. This POC relates to the conversation in MagicStack#163.
1 parent 4d6621f commit ca10b51

File tree

10 files changed

+211
-3
lines changed

10 files changed

+211
-3
lines changed
 

‎tests/test_dns.py

+55
Original file line numberDiff line numberDiff line change
@@ -1,10 +1,14 @@
11
import asyncio
22
import socket
3+
import sys
34
import unittest
45

56
from uvloop import _testbase as tb
67

78

9+
PY37 = sys.version_info >= (3, 7, 0)
10+
11+
812
class BaseTestDNS:
913

1014
def _test_getaddrinfo(self, *args, **kwargs):
@@ -177,6 +181,57 @@ async def run():
177181
finally:
178182
self.loop.close()
179183

184+
@unittest.skipUnless(PY37, 'requires Python 3.7')
185+
def test_getaddrinfo_tracing(self):
186+
from time import monotonic
187+
from uvloop import start_tracing, stop_tracing
188+
from uvloop.tracing import Tracer, Span
189+
190+
class DummySpan(Span):
191+
def __init__(self, name, parent=None):
192+
self.name = name
193+
self.parent = parent
194+
self.start_time = monotonic()
195+
self.finish_time = None
196+
self.children = []
197+
self.tags = {}
198+
199+
def set_tag(self, key, value):
200+
self.tags[key] = value
201+
202+
def finish(self, finish_time=None):
203+
self.finish_time = finish_time or monotonic()
204+
205+
@property
206+
def is_finished(self):
207+
return self.finish_time is not None
208+
209+
210+
class DummyTracer(Tracer):
211+
def start_span(self, name, parent_span):
212+
span = DummySpan(name, parent_span)
213+
parent_span.children.append(span)
214+
return span
215+
216+
root_span = DummySpan('root')
217+
start_tracing(DummyTracer(), root_span)
218+
self.loop.run_until_complete(
219+
self.loop.getaddrinfo('example.com', 80)
220+
)
221+
root_span.finish()
222+
assert root_span.children
223+
assert root_span.children[0].name == 'getaddrinfo'
224+
assert root_span.children[0].tags['host'] == b'example.com'
225+
assert root_span.children[0].tags['port'] == b'80'
226+
assert root_span.children[0].is_finished
227+
assert root_span.children[0].start_time < root_span.children[0].finish_time
228+
229+
stop_tracing()
230+
self.loop.run_until_complete(
231+
self.loop.getaddrinfo('example.com', 80)
232+
)
233+
assert len(root_span.children) == 1
234+
180235

181236
class Test_AIO_DNS(BaseTestDNS, tb.AIOTestCase):
182237
pass

‎uvloop/__init__.py

+8-2
Original file line numberDiff line numberDiff line change
@@ -1,15 +1,21 @@
11
import asyncio
22

33
from asyncio.events import BaseDefaultEventLoopPolicy as __BasePolicy
4+
from sys import version_info
45

56
from . import includes as __includes # NOQA
67
from . import _patch # NOQA
78
from .loop import Loop as __BaseLoop # NOQA
89

10+
PY37 = version_info >= (3, 7, 0)
911

10-
__version__ = '0.11.0.dev0'
11-
__all__ = ('new_event_loop', 'EventLoopPolicy')
12+
if PY37:
13+
from .loop import start_tracing, stop_tracing
14+
__all__ = ('new_event_loop', 'EventLoopPolicy', 'start_tracing', 'stop_tracing')
15+
else:
16+
__all__ = ('new_event_loop', 'EventLoopPolicy')
1217

18+
__version__ = '0.11.0.dev0'
1319

1420
class Loop(__BaseLoop, asyncio.AbstractEventLoop):
1521
pass

‎uvloop/dns.pyx

+11
Original file line numberDiff line numberDiff line change
@@ -249,6 +249,7 @@ cdef class AddrInfoRequest(UVRequest):
249249
cdef:
250250
system.addrinfo hints
251251
object callback
252+
object context
252253
uv.uv_getaddrinfo_t _req_data
253254

254255
def __cinit__(self, Loop loop,
@@ -278,6 +279,11 @@ cdef class AddrInfoRequest(UVRequest):
278279
callback(ex)
279280
return
280281

282+
if PY37:
283+
self.context = <object>PyContext_CopyCurrent()
284+
else:
285+
self.context = None
286+
281287
memset(&self.hints, 0, sizeof(system.addrinfo))
282288
self.hints.ai_flags = flags
283289
self.hints.ai_family = family
@@ -336,6 +342,9 @@ cdef void __on_addrinfo_resolved(uv.uv_getaddrinfo_t *resolver,
336342
object callback = request.callback
337343
AddrInfo ai
338344

345+
if PY37:
346+
PyContext_Enter(<PyContext*>request.context)
347+
339348
try:
340349
if status < 0:
341350
callback(convert_error(status))
@@ -347,6 +356,8 @@ cdef void __on_addrinfo_resolved(uv.uv_getaddrinfo_t *resolver,
347356
loop._handle_exception(ex)
348357
finally:
349358
request.on_done()
359+
if PY37:
360+
PyContext_Exit(<PyContext*>request.context)
350361

351362

352363
cdef void __on_nameinfo_resolved(uv.uv_getnameinfo_t* req,

‎uvloop/includes/compat.h

+8
Original file line numberDiff line numberDiff line change
@@ -35,6 +35,10 @@ typedef struct {
3535
PyObject_HEAD
3636
} PyContext;
3737

38+
typedef struct {
39+
PyObject_HEAD
40+
} PyContextVar;
41+
3842
PyContext * PyContext_CopyCurrent(void) {
3943
abort();
4044
return NULL;
@@ -49,4 +53,8 @@ int PyContext_Exit(PyContext *ctx) {
4953
abort();
5054
return -1;
5155
}
56+
57+
int PyContextVar_Get(PyContextVar *var, PyObject *default_value, PyObject **value) {
58+
return -1;
59+
}
5260
#endif

‎uvloop/includes/python.pxd

+4
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,10 @@ cdef extern from "Python.h":
1717

1818
cdef extern from "includes/compat.h":
1919
ctypedef struct PyContext
20+
ctypedef struct PyContextVar
21+
ctypedef struct PyObject
2022
PyContext* PyContext_CopyCurrent() except NULL
2123
int PyContext_Enter(PyContext *) except -1
2224
int PyContext_Exit(PyContext *) except -1
25+
int PyContextVar_Get(
26+
PyContextVar *var, object default_value, PyObject **value) except -1

‎uvloop/loop.pxd

+2
Original file line numberDiff line numberDiff line change
@@ -230,3 +230,5 @@ include "request.pxd"
230230
include "handles/udp.pxd"
231231

232232
include "server.pxd"
233+
234+
include "tracing.pxd"

‎uvloop/loop.pyx

+18-1
Original file line numberDiff line numberDiff line change
@@ -18,7 +18,9 @@ from .includes.python cimport PY_VERSION_HEX, \
1818
PyContext, \
1919
PyContext_CopyCurrent, \
2020
PyContext_Enter, \
21-
PyContext_Exit
21+
PyContext_Exit, \
22+
PyContextVar, \
23+
PyContextVar_Get
2224

2325
from libc.stdint cimport uint64_t
2426
from libc.string cimport memset, strerror, memcpy
@@ -821,13 +823,26 @@ cdef class Loop:
821823
except Exception as ex:
822824
if not fut.cancelled():
823825
fut.set_exception(ex)
826+
824827
else:
825828
if not fut.cancelled():
826829
fut.set_result(data)
830+
827831
else:
828832
if not fut.cancelled():
829833
fut.set_exception(result)
830834

835+
traced_context = __traced_context()
836+
if traced_context:
837+
traced_context.current_span().finish()
838+
839+
traced_context = __traced_context()
840+
if traced_context:
841+
traced_context.start_span(
842+
"getaddrinfo",
843+
tags={'host': host, 'port': port}
844+
)
845+
831846
AddrInfoRequest(self, host, port, family, type, proto, flags, callback)
832847
return fut
833848

@@ -2976,6 +2991,8 @@ include "handles/udp.pyx"
29762991

29772992
include "server.pyx"
29782993

2994+
include "tracing.pyx"
2995+
29792996

29802997
# Used in UVProcess
29812998
cdef vint __atfork_installed = 0

‎uvloop/tracing.pxd

+10
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
cdef class TracedContext:
2+
cdef:
3+
object _tracer
4+
object _span
5+
object _root_span
6+
7+
cdef object start_span(self, name, tags=?)
8+
cdef object current_span(self)
9+
10+
cdef TracedContext __traced_context()

‎uvloop/tracing.py

+24
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,24 @@
1+
import abc
2+
3+
class Span(abc.ABC):
4+
5+
@abc.abstractmethod
6+
def set_tag(self, key, value):
7+
"""Tag the span with an arbitrary key and value."""
8+
9+
@abc.abstractmethod
10+
def finish(self, finish_time=None):
11+
"""Indicate that the work represented by this span
12+
has been completed or terminated."""
13+
14+
@abc.abstractproperty
15+
def is_finished(self):
16+
"""Return True if the current span is already finished."""
17+
18+
19+
class Tracer(abc.ABC):
20+
21+
@abc.abstractmethod
22+
def start_span(self, name, parent_span):
23+
"""Start a new Span with a specific name. The parent of the span
24+
will be also passed as a paramter."""

‎uvloop/tracing.pyx

+71
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,71 @@
1+
from contextlib import contextmanager
2+
3+
if PY37:
4+
import contextvars
5+
__traced_ctx = contextvars.ContextVar('__traced_ctx', default=None)
6+
else:
7+
__traced_ctx = None
8+
9+
10+
cdef class TracedContext:
11+
def __cinit__(self, tracer, root_span):
12+
self._tracer = tracer
13+
self._root_span = root_span
14+
self._span = None
15+
16+
cdef object start_span(self, name, tags=None):
17+
parent_span = self._span if self._span else self._root_span
18+
span = self._tracer.start_span(name, parent_span)
19+
20+
if tags:
21+
for key, value in tags.items():
22+
span.set_tag(key, value)
23+
24+
self._span = span
25+
return self._span
26+
27+
cdef object current_span(self):
28+
return self._span
29+
30+
31+
cdef inline TracedContext __traced_context():
32+
cdef:
33+
PyObject* traced_context = NULL
34+
35+
if not PY37:
36+
return
37+
38+
PyContextVar_Get(<PyContextVar*> __traced_ctx, None, &traced_context)
39+
40+
if <object>traced_context is None:
41+
return
42+
return <TracedContext>traced_context
43+
44+
45+
def start_tracing(tracer, root_span):
46+
if not PY37:
47+
raise RuntimeError(
48+
"tracing only supported by Python 3.7 or newer versions")
49+
50+
traced_context = __traced_ctx.get(None)
51+
if traced_context is not None:
52+
raise RuntimeError("Tracing already started")
53+
54+
traced_context = TracedContext(tracer, root_span)
55+
__traced_ctx.set(traced_context)
56+
57+
58+
def stop_tracing():
59+
if not PY37:
60+
raise RuntimeError(
61+
"tracing only supported by Python 3.7 or newer versions")
62+
63+
traced_context = __traced_context()
64+
if traced_context is None:
65+
return
66+
67+
span = traced_context.current_span()
68+
if span and not span.is_finished:
69+
span.finish()
70+
71+
__traced_ctx.set(None)

0 commit comments

Comments
 (0)
Please sign in to comment.