Skip to content

Commit

Permalink
tracing: Delegate envoy tracing decision to tracer to determine if sh…
Browse files Browse the repository at this point in the history
…ould override sampling (#2787)

Signed-off-by: Gary Brown <gary@brownuk.com>
  • Loading branch information
objectiser authored and mattklein123 committed Mar 27, 2018
1 parent 2429797 commit 85fe233
Show file tree
Hide file tree
Showing 19 changed files with 304 additions and 136 deletions.
2 changes: 1 addition & 1 deletion bazel/repository_locations.bzl
Original file line number Diff line number Diff line change
Expand Up @@ -39,7 +39,7 @@ REPOSITORY_LOCATIONS = dict(
remote = "https://github.com/grpc/grpc.git",
),
io_opentracing_cpp = dict(
commit = "f3c1f42601d13504c68e2bc81c60604f0de055dd",
commit = "f6be24043e00baa2a25e0c1bb8793433d44ecc8b",
remote = "https://github.com/opentracing/opentracing-cpp",
),
com_lightstep_tracer_cpp = dict(
Expand Down
30 changes: 28 additions & 2 deletions include/envoy/tracing/http_tracer.h
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,30 @@ namespace Tracing {

enum class OperationName { Ingress, Egress };

/**
* The reasons why trace sampling may or may not be performed.
*/
enum class Reason {
// Not sampled based on supplied request id.
NotTraceableRequestId,
// Not sampled due to being a health check.
HealthCheck,
// Sampling enabled.
Sampling,
// Sampling forced by the service.
ServiceForced,
// Sampling forced by the client.
ClientForced,
};

/**
* The decision regarding whether traces should be sampled, and the reason for it.
*/
struct Decision {
Reason reason;
bool traced;
};

/**
* Tracing configuration, it carries additional data needed to populate the span.
*/
Expand Down Expand Up @@ -88,7 +112,8 @@ class Driver {
* Start driver specific span.
*/
virtual SpanPtr startSpan(const Config& config, Http::HeaderMap& request_headers,
const std::string& operation_name, SystemTime start_time) PURE;
const std::string& operation_name, SystemTime start_time,
const Tracing::Decision tracing_decision) PURE;
};

typedef std::unique_ptr<Driver> DriverPtr;
Expand All @@ -102,7 +127,8 @@ class HttpTracer {
virtual ~HttpTracer() {}

virtual SpanPtr startSpan(const Config& config, Http::HeaderMap& request_headers,
const RequestInfo::RequestInfo& request_info) PURE;
const RequestInfo::RequestInfo& request_info,
const Tracing::Decision tracing_decision) PURE;
};

typedef std::unique_ptr<HttpTracer> HttpTracerPtr;
Expand Down
2 changes: 1 addition & 1 deletion source/common/access_log/access_log_impl.cc
Original file line number Diff line number Diff line change
Expand Up @@ -74,7 +74,7 @@ bool TraceableRequestFilter::evaluate(const RequestInfo::RequestInfo& info,
const Http::HeaderMap& request_headers) {
Tracing::Decision decision = Tracing::HttpTracerUtility::isTracing(info, request_headers);

return decision.is_tracing && decision.reason == Tracing::Reason::ServiceForced;
return decision.traced && decision.reason == Tracing::Reason::ServiceForced;
}

bool StatusCodeFilter::evaluate(const RequestInfo::RequestInfo& info, const Http::HeaderMap&) {
Expand Down
7 changes: 4 additions & 3 deletions source/common/http/conn_manager_impl.cc
Original file line number Diff line number Diff line change
Expand Up @@ -607,12 +607,13 @@ void ConnectionManagerImpl::ActiveStream::traceRequest() {
ConnectionManagerImpl::chargeTracingStats(tracing_decision.reason,
connection_manager_.config_.tracingStats());

if (!tracing_decision.is_tracing) {
active_span_ = connection_manager_.tracer_.startSpan(*this, *request_headers_, request_info_,
tracing_decision);

if (!active_span_) {
return;
}

active_span_ = connection_manager_.tracer_.startSpan(*this, *request_headers_, request_info_);

// TODO: Need to investigate the following code based on the cached route, as may
// be broken in the case a filter changes the route.

Expand Down
3 changes: 3 additions & 0 deletions source/common/tracing/dynamic_opentracing_driver_impl.h
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,9 @@ class DynamicOpenTracingDriver : public OpenTracingDriver {
return OpenTracingDriver::PropagationMode::TracerNative;
}

protected:
bool useTagForSamplingDecision() override { return true; }

private:
opentracing::DynamicTracingLibraryHandle library_handle_;
std::shared_ptr<opentracing::Tracer> tracer_;
Expand Down
7 changes: 4 additions & 3 deletions source/common/tracing/http_tracer_impl.cc
Original file line number Diff line number Diff line change
Expand Up @@ -168,16 +168,17 @@ HttpTracerImpl::HttpTracerImpl(DriverPtr&& driver, const LocalInfo::LocalInfo& l
: driver_(std::move(driver)), local_info_(local_info) {}

SpanPtr HttpTracerImpl::startSpan(const Config& config, Http::HeaderMap& request_headers,
const RequestInfo::RequestInfo& request_info) {
const RequestInfo::RequestInfo& request_info,
const Tracing::Decision tracing_decision) {
std::string span_name = HttpTracerUtility::toString(config.operationName());

if (config.operationName() == OperationName::Egress) {
span_name.append(" ");
span_name.append(request_headers.Host()->value().c_str());
}

SpanPtr active_span =
driver_->startSpan(config, request_headers, span_name, request_info.startTime());
SpanPtr active_span = driver_->startSpan(config, request_headers, span_name,
request_info.startTime(), tracing_decision);
if (active_span) {
active_span->setTag(Tracing::Tags::get().COMPONENT, Tracing::Tags::get().PROXY);
active_span->setTag(Tracing::Tags::get().NODE_ID, local_info_.nodeName());
Expand Down
19 changes: 4 additions & 15 deletions source/common/tracing/http_tracer_impl.h
Original file line number Diff line number Diff line change
Expand Up @@ -14,19 +14,6 @@
namespace Envoy {
namespace Tracing {

enum class Reason {
NotTraceableRequestId,
HealthCheck,
Sampling,
ServiceForced,
ClientForced,
};

struct Decision {
Reason reason;
bool is_tracing;
};

/**
* Tracing tag names.
*/
Expand Down Expand Up @@ -143,7 +130,8 @@ class NullSpan : public Span {
class HttpNullTracer : public HttpTracer {
public:
// Tracing::HttpTracer
SpanPtr startSpan(const Config&, Http::HeaderMap&, const RequestInfo::RequestInfo&) override {
SpanPtr startSpan(const Config&, Http::HeaderMap&, const RequestInfo::RequestInfo&,
const Tracing::Decision) override {
return SpanPtr{new NullSpan()};
}
};
Expand All @@ -154,7 +142,8 @@ class HttpTracerImpl : public HttpTracer {

// Tracing::HttpTracer
SpanPtr startSpan(const Config& config, Http::HeaderMap& request_headers,
const RequestInfo::RequestInfo& request_info) override;
const RequestInfo::RequestInfo& request_info,
const Tracing::Decision tracing_decision) override;

private:
DriverPtr driver_;
Expand Down
3 changes: 3 additions & 0 deletions source/common/tracing/lightstep_tracer_impl.h
Original file line number Diff line number Diff line change
Expand Up @@ -61,6 +61,9 @@ class LightStepDriver : public OpenTracingDriver {
opentracing::Tracer& tracer() override;
PropagationMode propagationMode() const override { return propagation_mode_; }

protected:
bool useTagForSamplingDecision() override { return false; }

private:
class LightStepTransporter : public lightstep::AsyncTransporter, Http::AsyncClient::Callbacks {
public:
Expand Down
18 changes: 15 additions & 3 deletions source/common/tracing/opentracing_driver_impl.cc
Original file line number Diff line number Diff line change
Expand Up @@ -128,7 +128,13 @@ OpenTracingDriver::OpenTracingDriver(Stats::Store& stats)
: tracer_stats_{OPENTRACING_TRACER_STATS(POOL_COUNTER_PREFIX(stats, "tracing.opentracing."))} {}

SpanPtr OpenTracingDriver::startSpan(const Config&, Http::HeaderMap& request_headers,
const std::string& operation_name, SystemTime start_time) {
const std::string& operation_name, SystemTime start_time,
const Tracing::Decision tracing_decision) {
// If tracing decision is no, and sampling decision is not communicated via tags, then
// return a null span to indicate that tracing is not being performed.
if (!tracing_decision.traced && !useTagForSamplingDecision()) {
return nullptr;
}
const PropagationMode propagation_mode = this->propagationMode();
const opentracing::Tracer& tracer = this->tracer();
std::unique_ptr<opentracing::Span> active_span;
Expand Down Expand Up @@ -164,8 +170,14 @@ SpanPtr OpenTracingDriver::startSpan(const Config&, Http::HeaderMap& request_hea
tracerStats().span_context_extraction_error_.inc();
}
}
active_span = tracer.StartSpan(operation_name, {opentracing::ChildOf(parent_span_ctx.get()),
opentracing::StartTimestamp(start_time)});
opentracing::StartSpanOptions options;
options.references.emplace_back(opentracing::SpanReferenceType::ChildOfRef,
parent_span_ctx.get());
options.start_system_timestamp = start_time;
if (!tracing_decision.traced) {
options.tags.emplace_back(opentracing::ext::sampling_priority, 0);
}
active_span = tracer.StartSpanWithOptions(operation_name, options);
RELEASE_ASSERT(active_span != nullptr);
return SpanPtr{new OpenTracingSpan{*this, std::move(active_span)}};
}
Expand Down
8 changes: 7 additions & 1 deletion source/common/tracing/opentracing_driver_impl.h
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,9 @@
#include "envoy/tracing/http_tracer.h"

#include "common/common/logger.h"
#include "common/singleton/const_singleton.h"

#include "opentracing/ext/tags.h"
#include "opentracing/tracer.h"

namespace Envoy {
Expand Down Expand Up @@ -49,7 +51,8 @@ class OpenTracingDriver : public Driver, protected Logger::Loggable<Logger::Id::

// Tracer::TracingDriver
SpanPtr startSpan(const Config& config, Http::HeaderMap& request_headers,
const std::string& operation_name, SystemTime start_time) override;
const std::string& operation_name, SystemTime start_time,
const Tracing::Decision tracing_decision) override;

virtual opentracing::Tracer& tracer() PURE;

Expand All @@ -64,6 +67,9 @@ class OpenTracingDriver : public Driver, protected Logger::Loggable<Logger::Id::

OpenTracingTracerStats& tracerStats() { return tracer_stats_; }

protected:
virtual bool useTagForSamplingDecision() PURE;

private:
OpenTracingTracerStats tracer_stats_;
};
Expand Down
5 changes: 4 additions & 1 deletion source/common/tracing/zipkin/zipkin_tracer_impl.cc
Original file line number Diff line number Diff line change
Expand Up @@ -76,7 +76,8 @@ Driver::Driver(const Json::Object& config, Upstream::ClusterManager& cluster_man
}

Tracing::SpanPtr Driver::startSpan(const Tracing::Config& config, Http::HeaderMap& request_headers,
const std::string&, SystemTime start_time) {
const std::string&, SystemTime start_time,
const Tracing::Decision tracing_decision) {
Tracer& tracer = *tls_->getTyped<TlsTracer>().tracer_;
SpanPtr new_zipkin_span;
bool sampled(true);
Expand All @@ -86,6 +87,8 @@ Tracing::SpanPtr Driver::startSpan(const Tracing::Config& config, Http::HeaderMa
// zipkin tracers may still use that value, although should be 0 or 1.
absl::string_view xb3_sampled = request_headers.XB3Sampled()->value().getStringView();
sampled = xb3_sampled == ZipkinCoreConstants::get().SAMPLED || xb3_sampled == "true";
} else {
sampled = tracing_decision.traced;
}

if (request_headers.XB3TraceId() && request_headers.XB3SpanId()) {
Expand Down
3 changes: 2 additions & 1 deletion source/common/tracing/zipkin/zipkin_tracer_impl.h
Original file line number Diff line number Diff line change
Expand Up @@ -99,7 +99,8 @@ class Driver : public Tracing::Driver {
* ("ingress" or "egress") passed by the caller.
*/
Tracing::SpanPtr startSpan(const Tracing::Config&, Http::HeaderMap& request_headers,
const std::string&, SystemTime start_time) override;
const std::string&, SystemTime start_time,
const Tracing::Decision tracing_decision) override;

// Getters to return the ZipkinDriver's key members.
Upstream::ClusterManager& clusterManager() { return cm_; }
Expand Down
69 changes: 37 additions & 32 deletions test/common/http/conn_manager_impl_test.cc
Original file line number Diff line number Diff line change
Expand Up @@ -520,13 +520,14 @@ TEST_F(HttpConnectionManagerImplTest, StartAndFinishSpanNormalFlow) {
setup(false, "");

NiceMock<Tracing::MockSpan>* span = new NiceMock<Tracing::MockSpan>();
EXPECT_CALL(tracer_, startSpan_(_, _, _))
.WillOnce(Invoke([&](const Tracing::Config& config, const HeaderMap&,
const RequestInfo::RequestInfo&) -> Tracing::Span* {
EXPECT_EQ(Tracing::OperationName::Ingress, config.operationName());
EXPECT_CALL(tracer_, startSpan_(_, _, _, _))
.WillOnce(
Invoke([&](const Tracing::Config& config, const HeaderMap&,
const RequestInfo::RequestInfo&, const Tracing::Decision) -> Tracing::Span* {
EXPECT_EQ(Tracing::OperationName::Ingress, config.operationName());

return span;
}));
return span;
}));
// No decorator.
EXPECT_CALL(*route_config_provider_.route_config_->route_, decorator())
.WillRepeatedly(Return(nullptr));
Expand Down Expand Up @@ -586,13 +587,14 @@ TEST_F(HttpConnectionManagerImplTest, StartAndFinishSpanNormalFlowIngressDecorat
setup(false, "");

NiceMock<Tracing::MockSpan>* span = new NiceMock<Tracing::MockSpan>();
EXPECT_CALL(tracer_, startSpan_(_, _, _))
.WillOnce(Invoke([&](const Tracing::Config& config, const HeaderMap&,
const RequestInfo::RequestInfo&) -> Tracing::Span* {
EXPECT_EQ(Tracing::OperationName::Ingress, config.operationName());
EXPECT_CALL(tracer_, startSpan_(_, _, _, _))
.WillOnce(
Invoke([&](const Tracing::Config& config, const HeaderMap&,
const RequestInfo::RequestInfo&, const Tracing::Decision) -> Tracing::Span* {
EXPECT_EQ(Tracing::OperationName::Ingress, config.operationName());

return span;
}));
return span;
}));
route_config_provider_.route_config_->route_->decorator_.operation_ = "testOp";
EXPECT_CALL(*route_config_provider_.route_config_->route_, decorator()).Times(4);
EXPECT_CALL(route_config_provider_.route_config_->route_->decorator_, apply(_))
Expand Down Expand Up @@ -647,13 +649,14 @@ TEST_F(HttpConnectionManagerImplTest, StartAndFinishSpanNormalFlowIngressDecorat
setup(false, "");

NiceMock<Tracing::MockSpan>* span = new NiceMock<Tracing::MockSpan>();
EXPECT_CALL(tracer_, startSpan_(_, _, _))
.WillOnce(Invoke([&](const Tracing::Config& config, const HeaderMap&,
const RequestInfo::RequestInfo&) -> Tracing::Span* {
EXPECT_EQ(Tracing::OperationName::Ingress, config.operationName());
EXPECT_CALL(tracer_, startSpan_(_, _, _, _))
.WillOnce(
Invoke([&](const Tracing::Config& config, const HeaderMap&,
const RequestInfo::RequestInfo&, const Tracing::Decision) -> Tracing::Span* {
EXPECT_EQ(Tracing::OperationName::Ingress, config.operationName());

return span;
}));
return span;
}));
route_config_provider_.route_config_->route_->decorator_.operation_ = "initOp";
EXPECT_CALL(*route_config_provider_.route_config_->route_, decorator()).Times(4);
EXPECT_CALL(route_config_provider_.route_config_->route_->decorator_, apply(_))
Expand Down Expand Up @@ -713,13 +716,14 @@ TEST_F(HttpConnectionManagerImplTest, StartAndFinishSpanNormalFlowEgressDecorato
{Tracing::OperationName::Egress, {LowerCaseString(":method")}}));

NiceMock<Tracing::MockSpan>* span = new NiceMock<Tracing::MockSpan>();
EXPECT_CALL(tracer_, startSpan_(_, _, _))
.WillOnce(Invoke([&](const Tracing::Config& config, const HeaderMap&,
const RequestInfo::RequestInfo&) -> Tracing::Span* {
EXPECT_EQ(Tracing::OperationName::Egress, config.operationName());
EXPECT_CALL(tracer_, startSpan_(_, _, _, _))
.WillOnce(
Invoke([&](const Tracing::Config& config, const HeaderMap&,
const RequestInfo::RequestInfo&, const Tracing::Decision) -> Tracing::Span* {
EXPECT_EQ(Tracing::OperationName::Egress, config.operationName());

return span;
}));
return span;
}));
route_config_provider_.route_config_->route_->decorator_.operation_ = "testOp";
EXPECT_CALL(*route_config_provider_.route_config_->route_, decorator()).Times(4);
EXPECT_CALL(route_config_provider_.route_config_->route_->decorator_, apply(_))
Expand Down Expand Up @@ -779,13 +783,14 @@ TEST_F(HttpConnectionManagerImplTest, StartAndFinishSpanNormalFlowEgressDecorato
{Tracing::OperationName::Egress, {LowerCaseString(":method")}}));

NiceMock<Tracing::MockSpan>* span = new NiceMock<Tracing::MockSpan>();
EXPECT_CALL(tracer_, startSpan_(_, _, _))
.WillOnce(Invoke([&](const Tracing::Config& config, const HeaderMap&,
const RequestInfo::RequestInfo&) -> Tracing::Span* {
EXPECT_EQ(Tracing::OperationName::Egress, config.operationName());
EXPECT_CALL(tracer_, startSpan_(_, _, _, _))
.WillOnce(
Invoke([&](const Tracing::Config& config, const HeaderMap&,
const RequestInfo::RequestInfo&, const Tracing::Decision) -> Tracing::Span* {
EXPECT_EQ(Tracing::OperationName::Egress, config.operationName());

return span;
}));
return span;
}));
route_config_provider_.route_config_->route_->decorator_.operation_ = "initOp";
EXPECT_CALL(*route_config_provider_.route_config_->route_, decorator()).Times(4);
EXPECT_CALL(route_config_provider_.route_config_->route_->decorator_, apply(_))
Expand Down Expand Up @@ -962,7 +967,7 @@ TEST_F(HttpConnectionManagerImplTest, DoNotStartSpanIfTracingIsNotEnabled) {
// Disable tracing.
tracing_config_.reset();

EXPECT_CALL(tracer_, startSpan_(_, _, _)).Times(0);
EXPECT_CALL(tracer_, startSpan_(_, _, _, _)).Times(0);
ON_CALL(runtime_.snapshot_, featureEnabled("tracing.global_enabled", 100, _))
.WillByDefault(Return(true));

Expand Down Expand Up @@ -1000,7 +1005,7 @@ TEST_F(HttpConnectionManagerImplTest, StartSpanOnlyHealthCheckRequest) {

NiceMock<Tracing::MockSpan>* span = new NiceMock<Tracing::MockSpan>();

EXPECT_CALL(tracer_, startSpan_(_, _, _)).WillOnce(Return(span));
EXPECT_CALL(tracer_, startSpan_(_, _, _, _)).WillOnce(Return(span));
EXPECT_CALL(*span, finishSpan()).Times(0);

EXPECT_CALL(runtime_.snapshot_, featureEnabled("tracing.global_enabled", 100, _))
Expand Down
3 changes: 2 additions & 1 deletion test/common/tracing/dynamic_opentracing_driver_impl_test.cc
Original file line number Diff line number Diff line change
Expand Up @@ -70,7 +70,8 @@ TEST_F(DynamicOpenTracingDriverTest, InitializeDriver) {
TEST_F(DynamicOpenTracingDriverTest, FlushSpans) {
setupValidDriver();

SpanPtr first_span = driver_->startSpan(config_, request_headers_, operation_name_, start_time_);
SpanPtr first_span = driver_->startSpan(config_, request_headers_, operation_name_, start_time_,
{Reason::Sampling, true});
first_span->finishSpan();
driver_->tracer().Close();

Expand Down
Loading

0 comments on commit 85fe233

Please sign in to comment.