Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add support for OTEL_TRACE_SAMPLER and OTEL_TRACE_SAMPLER_ARG env variables #1496

Merged
merged 40 commits into from
Jan 19, 2021
Merged
Show file tree
Hide file tree
Changes from 28 commits
Commits
Show all changes
40 commits
Select commit Hold shift + click to select a range
9ef91be
Add known samplers
srikanthccv Dec 17, 2020
755b287
Use get_from_env_or_default for TRACE_SAMPLER
srikanthccv Dec 17, 2020
5a304c1
Update AlwaysOn and AlwaysOff Samplers
srikanthccv Dec 17, 2020
8dbe96b
Fallback rate to 1.0 when arg is not set or unrecognized value given
srikanthccv Dec 17, 2020
663e55e
Add test to ensure default sample is ParentBased(root=AlwaysOn)
srikanthccv Dec 18, 2020
40266d6
Bug fix
srikanthccv Dec 18, 2020
585de43
Update sampler get
srikanthccv Dec 18, 2020
2082d64
Remove OTEL_ from key
srikanthccv Dec 18, 2020
f38effc
Add more tests for sampler env test
srikanthccv Dec 18, 2020
ab510ec
Fix import sort order
srikanthccv Dec 18, 2020
9387dc0
Remove unused import
srikanthccv Dec 18, 2020
745e101
Make samplers private
srikanthccv Dec 18, 2020
cf90495
Add license and make method private
srikanthccv Dec 18, 2020
e45c3a8
Add documnetation for env
srikanthccv Dec 19, 2020
0afb1cc
Fix typo
srikanthccv Dec 19, 2020
7c8e49d
Lint fix
srikanthccv Dec 19, 2020
8f8bbab
Add CHANGELOG entry
srikanthccv Dec 19, 2020
7fb248e
Merge branch 'master' into trace-sampler-env
srikanthccv Dec 19, 2020
ee75656
Merge branch 'master' into trace-sampler-env
srikanthccv Dec 23, 2020
27d8c43
Using safe default instead of null
srikanthccv Dec 27, 2020
986f2fa
Merge branch 'trace-sampler-env' of github.com:lonewolf3739/opentelem…
srikanthccv Dec 27, 2020
a5880b6
Merge branch 'master' into trace-sampler-env
lzchen Jan 5, 2021
cdd37f5
Merge branch 'master' into trace-sampler-env
srikanthccv Jan 6, 2021
bf99448
Resolve review comments
srikanthccv Jan 6, 2021
b0adabd
Fix lint
srikanthccv Jan 6, 2021
e43dd4a
Fix lint
srikanthccv Jan 6, 2021
8654348
Merge branch 'master' into trace-sampler-env
srikanthccv Jan 7, 2021
ac1146e
Merge branch 'master' into trace-sampler-env
Jan 7, 2021
4fc887d
Merge branch 'master' into trace-sampler-env
srikanthccv Jan 12, 2021
498420d
Merge branch 'master' into trace-sampler-env
srikanthccv Jan 12, 2021
2e95894
Fix docs
srikanthccv Jan 12, 2021
d6b262a
Use os.getenv to retrieve trace sampler env vars
srikanthccv Jan 12, 2021
d517fbe
Fix lint
srikanthccv Jan 12, 2021
cf13b5d
Disbale protected member access check for method
srikanthccv Jan 12, 2021
6d75196
Merge branch 'master' into trace-sampler-env
srikanthccv Jan 12, 2021
4c9d3fa
Merge branch 'master' into trace-sampler-env
srikanthccv Jan 15, 2021
67e25ce
Merge branch 'master' into trace-sampler-env
srikanthccv Jan 16, 2021
c762410
Merge branch 'master' into trace-sampler-env
Jan 18, 2021
5fe79b1
Merge branch 'master' into trace-sampler-env
srikanthccv Jan 19, 2021
2817713
Merge branch 'master' into trace-sampler-env
srikanthccv Jan 19, 2021
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 2 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -36,6 +36,8 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
([#1285](https://github.com/open-telemetry/opentelemetry-python/pull/1285))
- Added `__repr__` for `DefaultSpan`, added `trace_flags` to `__repr__` of
`SpanContext` ([#1485](https://github.com/open-telemetry/opentelemetry-python/pull/1485)])
- `opentelemetry-sdk` Add support for OTEL_TRACE_SAMPLER and OTEL_TRACE_SAMPLER_ARG env variables
([#1496](https://github.com/open-telemetry/opentelemetry-python/pull/1496))
### Changed
- `opentelemetry-exporter-zipkin` Updated zipkin exporter status code and error tag
([#1486](https://github.com/open-telemetry/opentelemetry-python/pull/1486))
Expand Down
4 changes: 3 additions & 1 deletion opentelemetry-sdk/src/opentelemetry/sdk/trace/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -56,6 +56,8 @@
SPAN_EVENT_COUNT_LIMIT = Configuration().get("SPAN_EVENT_COUNT_LIMIT", 1000)
SPAN_LINK_COUNT_LIMIT = Configuration().get("SPAN_LINK_COUNT_LIMIT", 1000)
VALID_ATTR_VALUE_TYPES = (bool, str, int, float)
# pylint: disable=protected-access
TRACE_SAMPLER = sampling._get_from_env_or_default()


class SpanProcessor:
Expand Down Expand Up @@ -887,7 +889,7 @@ def use_span(
class TracerProvider(trace_api.TracerProvider):
def __init__(
self,
sampler: sampling.Sampler = sampling.DEFAULT_ON,
sampler: sampling.Sampler = TRACE_SAMPLER,
resource: Resource = Resource.create({}),
shutdown_on_exit: bool = True,
active_span_processor: Union[
Expand Down
79 changes: 79 additions & 0 deletions opentelemetry-sdk/src/opentelemetry/sdk/trace/sampling.py
Original file line number Diff line number Diff line change
Expand Up @@ -59,18 +59,57 @@
# created spans will now be sampled by the TraceIdRatioBased sampler
with trace.get_tracer(__name__).start_as_current_span("Test Span"):
...

The tracer sampler can also be configured via environment variables `OTEL_TRACE_SAMPLER` and `OTEL_TRACE_SAMPLER_ARG` (only if applicable).
The list of known values for `OTEL_TRACE_SAMPLER` are:

* always_on - Sampler that always samples spans, regardless of the parent span's sampling decision.
* always_off - Sampler that never samples spans, regardless of the parent span's sampling decision.
* traceidratio - Sampler that samples probabalistically based on `rate`.
* parentbased_always_on - (default) Sampler that respects its parent span's sampling decision, but otherwise always samples.
* parentbased_always_off - Sampler that respects its parent span's sampling decision, but otherwise never samples.
* parentbased_traceidratio - Sampler that respects its parent span's sampling decision, but otherwise samples probabalistically based on `rate`.

Sampling probability can be set with `OTEL_TRACE_SAMPLER_ARG` if the sampler is `traceidratio` or `parentbased_traceidratio`, when not provided `rate` will be set to 1.0 (maximum rate possible).


Prev example but with environment vairables. Please make sure to set the env `OTEL_TRACE_SAMPLER=traceidratio` and `OTEL_TRACE_SAMPLER_ARG=0.001`.

.. code:: python

from opentelemetry import trace
from opentelemetry.sdk.trace import TracerProvider
from opentelemetry.sdk.trace.export import (
ConsoleSpanExporter,
SimpleExportSpanProcessor,
)

trace.set_tracer_provider(TracerProvider())

# set up an exporter for sampled spans
trace.get_tracer_provider().add_span_processor(
SimpleExportSpanProcessor(ConsoleSpanExporter())
)

# created spans will now be sampled by the TraceIdRatioBased sampler with rate 1/1000.
with trace.get_tracer(__name__).start_as_current_span("Test Span"):
...
"""
import abc
import enum
from logging import getLogger
from types import MappingProxyType
from typing import Optional, Sequence

# pylint: disable=unused-import
from opentelemetry.configuration import Configuration
from opentelemetry.context import Context
from opentelemetry.trace import Link, get_current_span
from opentelemetry.trace.span import TraceState
from opentelemetry.util.types import Attributes

_logger = getLogger(__name__)


class Decision(enum.Enum):
# IsRecording() == false, span will not be recorded and all events and attributes will be dropped.
Expand Down Expand Up @@ -307,3 +346,43 @@ def get_description(self):

DEFAULT_ON = ParentBased(ALWAYS_ON)
"""Sampler that respects its parent span's sampling decision, but otherwise always samples."""


class ParentBasedTraceIdRatio(ParentBased):
"""
Sampler that respects its parent span's sampling decision, but otherwise
samples probabalistically based on `rate`.
"""

def __init__(self, rate: float):
root = TraceIdRatioBased(rate=rate)
super().__init__(root=root)


_KNOWN_SAMPLERS = {
"always_on": ALWAYS_ON,
"always_off": ALWAYS_OFF,
"parentbased_always_on": DEFAULT_ON,
"parentbased_always_off": DEFAULT_OFF,
"traceidratio": TraceIdRatioBased,
"parentbased_traceidratio": ParentBasedTraceIdRatio,
}


def _get_from_env_or_default() -> Sampler:
trace_sampler = (
Configuration().get("TRACE_SAMPLER", "parentbased_always_on").lower()
srikanthccv marked this conversation as resolved.
Show resolved Hide resolved
)
if trace_sampler not in _KNOWN_SAMPLERS:
_logger.warning("Couldn't recognize sampler %s.", trace_sampler)
trace_sampler = "parentbased_always_on"

if trace_sampler in ("traceidratio", "parentbased_traceidratio"):
try:
rate = float(Configuration().TRACE_SAMPLER_ARG)
except ValueError:
_logger.warning("Could not convert TRACE_SAMPLER_ARG to float.")
rate = 1.0
return _KNOWN_SAMPLERS[trace_sampler](rate)

return _KNOWN_SAMPLERS[trace_sampler]
47 changes: 47 additions & 0 deletions opentelemetry-sdk/tests/trace/test_trace.py
Original file line number Diff line number Diff line change
Expand Up @@ -138,6 +138,15 @@ def test_tracer_provider_accepts_concurrent_multi_span_processor(self):


class TestTracerSampling(unittest.TestCase):
def setUp(self):
# pylint: disable=protected-access
Configuration._reset()

def tearDown(self):
# pylint: disable=protected-access
Configuration._reset()
reload(trace)

def test_default_sampler(self):
tracer = new_tracer()

Expand All @@ -158,6 +167,12 @@ def test_default_sampler(self):
trace_api.TraceFlags.SAMPLED,
)

def test_default_sampler_type(self):
tracer_provider = trace.TracerProvider()
self.assertIsInstance(tracer_provider.sampler, sampling.ParentBased)
# pylint: disable=protected-access
self.assertEqual(tracer_provider.sampler._root, sampling.ALWAYS_ON)

def test_sampler_no_sampling(self):
tracer_provider = trace.TracerProvider(sampling.ALWAYS_OFF)
tracer = tracer_provider.get_tracer(__name__)
Expand All @@ -178,6 +193,38 @@ def test_sampler_no_sampling(self):
trace_api.TraceFlags.DEFAULT,
)

@mock.patch.dict("os.environ", {"OTEL_TRACE_SAMPLER": "always_off"})
def test_sampler_with_env(self):
# pylint: disable=protected-access
Configuration._reset()
reload(trace)
tracer_provider = trace.TracerProvider()
self.assertIsInstance(tracer_provider.sampler, sampling.StaticSampler)
self.assertEqual(
tracer_provider.sampler._decision, sampling.Decision.DROP
)

tracer = tracer_provider.get_tracer(__name__)

root_span = tracer.start_span(name="root span", context=None)
# Should be no-op
self.assertIsInstance(root_span, trace_api.DefaultSpan)

@mock.patch.dict(
"os.environ",
{
"OTEL_TRACE_SAMPLER": "parentbased_traceidratio",
"OTEL_TRACE_SAMPLER_ARG": "0.25",
},
)
def test_ratio_sampler_with_env(self):
# pylint: disable=protected-access
Configuration._reset()
reload(trace)
tracer_provider = trace.TracerProvider()
self.assertIsInstance(tracer_provider.sampler, sampling.ParentBased)
self.assertEqual(tracer_provider.sampler._root.rate, 0.25)


class TestSpanCreation(unittest.TestCase):
def test_start_span_invalid_spancontext(self):
Expand Down