
io.micronaut.tracing.instrument.util.TracingPublisher Maven / Gradle / Ivy
/*
* Copyright 2017-2020 original authors
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package io.micronaut.tracing.instrument.util;
import io.micronaut.core.async.publisher.Publishers;
import io.micronaut.http.MutableHttpResponse;
import io.opentracing.Scope;
import io.opentracing.ScopeManager;
import io.opentracing.Span;
import io.opentracing.Tracer;
import io.opentracing.noop.NoopScopeManager;
import org.reactivestreams.Publisher;
import org.reactivestreams.Subscriber;
import org.reactivestreams.Subscription;
import edu.umd.cs.findbugs.annotations.NonNull;
import java.util.Optional;
import static io.micronaut.tracing.interceptor.TraceInterceptor.logError;
/**
* A reactive streams publisher that traces.
*
* @param the type of element signaled
*
* @author graemerocher
* @since 1.0
*/
@SuppressWarnings("PublisherImplementation")
public class TracingPublisher implements Publisher {
private final Publisher publisher;
private final Tracer tracer;
private final Tracer.SpanBuilder spanBuilder;
private final Span parentSpan;
private final boolean isSingle;
/**
* Creates a new tracing publisher for the given arguments.
*
* @param publisher The target publisher
* @param tracer The tracer
* @param operationName The operation name that should be started
*/
public TracingPublisher(Publisher publisher, Tracer tracer, String operationName) {
this(publisher, tracer, tracer.buildSpan(operationName));
}
/**
* Creates a new tracing publisher for the given arguments. This constructor will just add tracing of the
* existing span if it is present.
*
* @param publisher The target publisher
* @param tracer The tracer
*/
public TracingPublisher(Publisher publisher, Tracer tracer) {
this(publisher, tracer, (Tracer.SpanBuilder) null);
}
/**
* Creates a new tracing publisher for the given arguments.
*
* @param publisher The target publisher
* @param tracer The tracer
* @param spanBuilder The span builder that represents the span that will be created when the publisher is subscribed to
*/
public TracingPublisher(Publisher publisher, Tracer tracer, Tracer.SpanBuilder spanBuilder) {
this(publisher, tracer, spanBuilder, Publishers.isSingle(publisher.getClass()));
}
/**
* Creates a new tracing publisher for the given arguments.
*
* @param publisher The target publisher
* @param tracer The tracer
* @param spanBuilder The span builder that represents the span that will be created when the publisher is subscribed to
* @param isSingle Does the publisher emit a single item
*/
public TracingPublisher(Publisher publisher, Tracer tracer, Tracer.SpanBuilder spanBuilder, boolean isSingle) {
this.publisher = publisher;
this.tracer = tracer;
this.spanBuilder = spanBuilder;
this.parentSpan = tracer.activeSpan();
this.isSingle = isSingle;
if (parentSpan != null && spanBuilder != null) {
spanBuilder.asChildOf(parentSpan);
}
}
@Override
public void subscribe(Subscriber super T> actual) {
Span span;
boolean finishOnClose;
if (spanBuilder != null) {
span = spanBuilder.start();
finishOnClose = true;
} else {
span = parentSpan;
finishOnClose = false;
}
if (span != null) {
final ScopeManager scopeManager = tracer.scopeManager();
try (Scope ignored = scopeManager.activeSpan() != span ? scopeManager.activate(span) : NoopScopeManager.NoopScope.INSTANCE) {
//noinspection SubscriberImplementation
publisher.subscribe(new Subscriber() {
boolean finished = false;
@Override
public void onSubscribe(Subscription s) {
if (scopeManager.activeSpan() != span) {
try (Scope ignored = scopeManager.activate(span)) {
TracingPublisher.this.doOnSubscribe(span);
actual.onSubscribe(s);
}
} else {
TracingPublisher.this.doOnSubscribe(span);
actual.onSubscribe(s);
}
}
@Override
public void onNext(T object) {
boolean finishAfterNext = isSingle && finishOnClose;
try (Scope ignored = scopeManager.activeSpan() != span ? scopeManager.activate(span) : NoopScopeManager.NoopScope.INSTANCE) {
if (object instanceof MutableHttpResponse) {
MutableHttpResponse response = (MutableHttpResponse) object;
Optional> body = response.getBody();
if (body.isPresent()) {
Object o = body.get();
if (Publishers.isConvertibleToPublisher(o)) {
Class> type = o.getClass();
Publisher> resultPublisher = Publishers.convertPublisher(o, Publisher.class);
Publisher> scopedPublisher = new ScopePropagationPublisher(resultPublisher, tracer, span);
response.body(Publishers.convertPublisher(scopedPublisher, type));
}
}
}
TracingPublisher.this.doOnNext(object, span);
actual.onNext(object);
if (isSingle) {
finished = true;
TracingPublisher.this.doOnFinish(span);
}
} finally {
if (finishAfterNext) {
span.finish();
}
}
}
@Override
public void onError(Throwable t) {
try (Scope ignored = scopeManager.activeSpan() != span ? scopeManager.activate(span) : NoopScopeManager.NoopScope.INSTANCE) {
TracingPublisher.this.onError(t, span);
actual.onError(t);
finished = true;
} finally {
if (finishOnClose) {
span.finish();
}
}
}
@Override
public void onComplete() {
try (Scope ignored = scopeManager.activeSpan() != span ? scopeManager.activate(span) : NoopScopeManager.NoopScope.INSTANCE) {
actual.onComplete();
TracingPublisher.this.doOnFinish(span);
} finally {
if (!finished && finishOnClose) {
span.finish();
}
}
}
});
}
} else {
publisher.subscribe(actual);
}
}
/**
* Designed for subclasses to override and implement custom behaviour when an item is emitted.
*
* @param object The object
* @param span The span
*/
protected void doOnNext(@NonNull T object, @NonNull Span span) {
// no-op
}
/**
* Designed for subclasses to override and implement custom on subscribe behaviour.
*
* @param span The span
*/
protected void doOnSubscribe(@NonNull Span span) {
// no-op
}
/**
* Designed for subclasses to override and implement custom on finish behaviour. Fired
* prior to calling {@link Span#finish()}.
*
* @param span The span
*/
@SuppressWarnings("WeakerAccess")
protected void doOnFinish(@NonNull Span span) {
// no-op
}
/**
* Designed for subclasses to override and implement custom on error behaviour.
*
* @param throwable The error
* @param span The span
*/
protected void doOnError(@NonNull Throwable throwable, @NonNull Span span) {
// no-op
}
private void onError(Throwable t, Span span) {
logError(span, t);
doOnError(t, span);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy