1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176
|
Configuring Trace Sampling
==========================
If instrumented services are producing a higher volume of tracing data than is
desired, then the services can be configured to send tracing data for only a
subset of processed requests. This is called trace sampling.
By default, the rate at which instrumented services sample traces is governed by
the Datadog Agent, which dynamically adjusts the sampling rates of its clients
in order to reach a [configured target number][1] of traces per second.
For fine-grained control over trace sampling, instrumented services can be
configured with _sampling rules_. What follows is a description of how
trace sampling may be configured in the Datadog C++ tracing library.
Sampling Rules
--------------
It is the _first_ service in a trace (the "root service") that determines
whether the trace will be sent to Datadog. Subsequent services in the trace
follow whichever decision was made by the root service.
The root service may define rules that assign different sampling rates to
different kinds of traces. In these rules, traces are distinguished by the
"service" and "operation name" associated with the root span. Typically, the
root span of a service is always associated with the same "service" and
"operation name." However, services acting as hosts to multiple services may
produce different "service" spans for different requests.
For example, consider the following array of rules:
```json
[
{"service": "usersvc", "name": "healthcheck", "sample_rate": 0.0},
{"service": "usersvc", "sample_rate": 0.5},
{"service": "authsvc", "sample_rate": 1.0},
{"sample_rate": 0.1}
]
```
These rules stipulate the following trace sampling behavior:
- `usersvc` requests whose operation name is `healthcheck` are never sampled.
- Other `usersvc` requests are sampled 50% of the time.
- `authsvc` requests are sampling 100% of the time.
- All other requests are sampled 10% of the time.
`sample_rate` is a probability. Its minimum value is zero, indicating "never,"
and its maximum value is one, indicating "always."
Note that the sampling behavior stipulated by sampling rules is relevant only
if the tracer being configured is the _first_ in the trace.
When a trace is created, its root span is evaluated against each sampling rule
in order. The first rule that matches determines the probability that the
trace will be sampled. If no rule matches, then the trace is subject to the
sampling rates governed by the Datadog Agent, as explained above.
Sampling rules can be configured programmatically in `std::string
TracerOptions::sampling_rules` or via the environment variable
`DD_TRACE_SAMPLING_RULES`. In either case, the rules are expressed as a JSON
array of objects. Each object supports the following properties:
```
[{
"service": <the root span's service name, or any if absent>,
"name": <the root span's operation name, or any if absent>,
"sample_rate": <the probability of sampling the trace, or 1.0 if absent>
}, ...]
```
`DD_TRACE_SAMPLE_RATE`
----------------------
Setting a (numeric) value for the `DD_TRACE_SAMPLE_RATE` environment variable
effectively appends a sampling rule to the tracer's array of sampling rules:
```
[
...,
{"sample_rate": $DD_TRACE_SAMPLE_RATE
]
```
Now there is a sampling rule that matches _any_ trace, and so traces that do
not match an earlier sampling rule are subject to the configured sampling rate.
Note that using `DD_TRACE_SAMPLE_RATE` means that the Datadog Agent no longer
governs the sampling rate of any traces produced by the tracer. The implicit
"catch-all" rule, with the configured sampling rate, always takes precedence
over the Agent-based fallback.
`double TracerOptions::sample_rate`
-----------------------------------
This configuration option has the same meaning as the `DD_TRACE_SAMPLE_RATE`
environment variable. Note that the environment variable overrides the
`TracerOptions` field if both are specified.
`DD_TRACE_RATE_LIMIT`
---------------------
Sampling rules (and, by extension, `DD_TRACE_SAMPLE_RATE`) specify the
_probability_ that a trace will be sampled, but they do not specify the maximum
number of traces that may be produced by the tracer in a given time period.
`DD_TRACE_RATE_LIMIT` is the maximum number of traces, per second, that may be
sampled by the tracer on account of sampling rules or `DD_TRACE_SAMPLE_RATE`.
The limit applies globally across all applicable traces, i.e. there is not a
separate limit for each sampling rule.
`DD_TRACE_RATE_LIMIT` is a floating point number, but is usually specified as an integer, e.g.
```shell
export DD_TRACE_RATE_LIMIT=200
```
for a limit of 200 traces per second.
If this limit is not configured, its default value is 100 traces per second.
Note that this limit applies separately to each tracer. If the instrumented
service spawns multiple processes, then each process contains its own tracer,
and each tracer is separately subject to the configured rate limit. For
example, if [nginx][2] is configured with `DD_TRACE_RATE_LIMIT=200` and also
spawns eight worker processes, then the actual limit overall is `200 * 8 =
1600` traces per second.
`double TracerOptions::sampling_limit_per_second`
-------------------------------------------------
This configuration option has the same meaning as the `DD_TRACE_RATE_LIMIT`
environment variable. Note that the environment variable overrides the
`TracerOptions` field if both are specified.
Span Sampling
-------------
_Note: Span sampling requires version 7.40 of the Datadog Agent or a more recent version._
Span sampling is used to select spans to keep even when the enclosing
trace is dropped.
Similar to _trace_ sampling rules, _span_ sampling rules are configured as a
JSON array of object, where each object may contain the following properties:
```
[{
"service": <matches the span's service name, or any if absent>,
"name": <matches the span's operation name, or any if absent>,
"sample_rate": <the probability of sampling matching spans, or 1.0 if absent>,
"max_per_second": <limit in spans sampled by this rule each second, or unlimited if absent>
}, ...]
```
The `service` and `name` are glob patterns, where "glob" here means:
- `*` matches any substring, including the empty string,
- `?` matches exactly one of any character, and
- any other character matches exactly one of itself.
Span sampling rules are examined only when the enclosing trace is to be
dropped.
The first span sampling rule that matches a span is used to make a span
sampling decision for that span. If the decision is "keep," then the span is
sent to Datadog despite the enclosing trace having been dropped.
Span sampling rules can be configured [directly][3] or [in a file][4].
For example, consider the following span sampling rules:
```shell
export DD_SPAN_SAMPLING_RULES='[
{"service": "router", "name": "rack.request", "max_per_second": 2000},
{"service": "classic-mysql", "name": "mysql2.*"},
{"service": "authn?", "sample_rate": 0.5}
]'
```
These rules state:
- When a trace is dropped, keep spans whose service name is `router` and whose
operation name is `rack.request`, but keep at most 2000 such spans per
second.
- When a trace is dropped, keep spans whose service name is `classic-mysql` and
whose operation name begins with `mysql2.`.
- When a trace is dropped, keep 50% of spans whose service name is `authn`
followed by another character, e.g. `authny`, `authnj`.
[1]: https://docs.datadoghq.com/tracing/trace_ingestion/mechanisms/?tab=environmentvariables#in-the-agent
[2]: https://docs.datadoghq.com/tracing/setup_overview/proxy_setup/?tab=nginx
[3]: configuration.md#span-sampling-rules
[4]: configuration.md#span-sampling-rules-file
|