All Downloads are FREE. Search and download functionalities are using the official Maven repository.

io.micrometer.tracing.test.simple.SimpleTracer Maven / Gradle / Ivy

/**
 * Copyright 2024 the original author or 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 *

* https://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.micrometer.tracing.test.simple; import io.micrometer.tracing.*; import java.util.Collections; import java.util.Deque; import java.util.List; import java.util.Map; import java.util.concurrent.ConcurrentHashMap; import java.util.concurrent.LinkedBlockingDeque; /** * A test tracer implementation. Puts started span in a list. * * @author Marcin Grzejszczak * @since 1.0.0 */ public class SimpleTracer implements Tracer { private static final Map traceContextToSpans = new ConcurrentHashMap<>(); private static final ThreadLocal scopedSpans = new ThreadLocal<>(); private final SimpleCurrentTraceContext currentTraceContext; final SimpleBaggageManager simpleBaggageManager = new SimpleBaggageManager(this); private final Deque spans = new LinkedBlockingDeque<>(); /** * Creates a new instance of {@link SimpleTracer}. */ public SimpleTracer() { this.currentTraceContext = new SimpleCurrentTraceContext(this); } @Override public SimpleSpan nextSpan(Span parent) { SimpleSpan span = simpleSpan(parent); this.spans.add(span); return span; } private SimpleSpan simpleSpan(Span parent) { Map baggageFromParent = Collections.emptyMap(); SimpleSpan span = new SimpleSpan(); boolean hasParent = parent != null; if (hasParent) { span.context().setParentId(parent.context().spanId()); baggageFromParent = simpleBaggageManager.getAllBaggageForCtx(parent.context()); } String traceId = hasParent ? parent.context().traceId() : span.context().generateId(); span.context().setTraceId(traceId); span.context().setSpanId(hasParent ? span.context().generateId() : traceId); span.context().addParentBaggage(baggageFromParent); return span; } /** * Returns a single reported span. * @return a single reported span * @throws AssertionError when there are 0 or more than 1 spans * @throws AssertionError when span hasn't been started and stopped */ public SimpleSpan onlySpan() { assertTrue(this.spans.size() == 1, "There must be only one span"); SimpleSpan span = this.spans.getFirst(); assertTrue(span.getStartTimestamp().toEpochMilli() > 0, "Span must be started"); assertTrue(span.getEndTimestamp().toEpochMilli() > 0, "Span must be finished"); return span; } private void assertTrue(boolean condition, String text) { if (!condition) { throw new AssertionError(text); } } /** * Returns the last reported span. * @return the last reported span * @throws AssertionError when there are 0 spans * @throws AssertionError when span hasn't been started */ public SimpleSpan lastSpan() { assertTrue(!this.spans.isEmpty(), "There must be at least one span"); SimpleSpan span = this.spans.getLast(); assertTrue(span.getStartTimestamp().toEpochMilli() > 0, "Span must be started"); return span; } @Override public SimpleSpanInScope withSpan(Span span) { return new SimpleSpanInScope(this.currentTraceContext.newScope(span != null ? span.context() : null)); } @Override public SimpleSpanCustomizer currentSpanCustomizer() { return new SimpleSpanCustomizer(this); } @Override public SimpleSpan currentSpan() { return scopedSpans.get(); } @Override public SimpleSpan nextSpan() { SimpleSpan span = simpleSpan(currentSpan()); this.spans.add(span); return span; } @Override public ScopedSpan startScopedSpan(String name) { return new SimpleScopedSpan(this).name(name); } @Override public SimpleSpanBuilder spanBuilder() { return new SimpleSpanBuilder(this); } @Override public TraceContext.Builder traceContextBuilder() { return new SimpleTraceContextBuilder(); } @Override public SimpleCurrentTraceContext currentTraceContext() { return this.currentTraceContext; } @Override public Map getAllBaggage() { return this.simpleBaggageManager.getAllBaggage(); } @Override public Baggage getBaggage(String name) { return this.simpleBaggageManager.getBaggage(name); } @Override public Baggage getBaggage(TraceContext traceContext, String name) { return this.simpleBaggageManager.getBaggage(traceContext, name); } @Override public Map getAllBaggage(TraceContext traceContext) { return this.simpleBaggageManager.getAllBaggageForCtx(traceContext); } @Override public Baggage createBaggage(String name) { return this.simpleBaggageManager.createBaggage(name); } @Override public Baggage createBaggage(String name, String value) { return this.simpleBaggageManager.createBaggage(name, value); } @Override public BaggageInScope createBaggageInScope(String name, String value) { return this.simpleBaggageManager.createBaggageInScope(name, value); } @Override public BaggageInScope createBaggageInScope(TraceContext traceContext, String name, String value) { return this.simpleBaggageManager.createBaggageInScope(traceContext, name, value); } @Override public List getBaggageFields() { return this.simpleBaggageManager.getBaggageFields(); } /** * Created spans. * @return all created spans */ public Deque getSpans() { return spans; } /** * Binds the given {@link Span} to the given {@link TraceContext}. * @param traceContext the traceContext to use to bind this span to * @param span the span that needs to be bounded to the traceContext */ static void bindSpanToTraceContext(TraceContext traceContext, SimpleSpan span) { traceContextToSpans.put(traceContext, span); } /** * Returns the {@link Span} that is bounded to the given {@link TraceContext}. * @param traceContext the traceContext to use to fetch the span * @return the span that is bounded to the given traceContext (null if none) */ static SimpleSpan getSpanForTraceContext(TraceContext traceContext) { return traceContextToSpans.get(traceContext); } static SimpleSpan getCurrentSpan() { return scopedSpans.get(); } static void resetCurrentSpan() { scopedSpans.remove(); } static void setCurrentSpan(SimpleSpan simpleSpan) { scopedSpans.set(simpleSpan); } static void setCurrentSpan(TraceContext context) { scopedSpans.set(context != null ? getSpanForTraceContext(context) : null); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy