io.opentracing.mock.MockSpan Maven / Gradle / Ivy
/*
* Copyright 2016-2017 The OpenTracing 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.opentracing.mock;
import io.opentracing.References;
import java.util.ArrayList;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.concurrent.atomic.AtomicLong;
import io.opentracing.Span;
import io.opentracing.SpanContext;
/**
* MockSpans are created via MockTracer.buildSpan(...), but they are also returned via calls to
* MockTracer.finishedSpans(). They provide accessors to all Span state.
*
* @see MockTracer#finishedSpans()
*/
public final class MockSpan implements Span {
// A simple-as-possible (consecutive for repeatability) id generator.
private static AtomicLong nextId = new AtomicLong(0);
private final MockTracer mockTracer;
private MockContext context;
private final long parentId; // 0 if there's no parent.
private final long startMicros;
private boolean finished;
private long finishMicros;
private final Map tags;
private final List logEntries = new ArrayList<>();
private String operationName;
private final List references;
private final List errors = new ArrayList<>();
public String operationName() {
return this.operationName;
}
@Override
public MockSpan setOperationName(String operationName) {
finishedCheck("Setting operationName {%s} on already finished span", operationName);
this.operationName = operationName;
return this;
}
/**
* @return the spanId of the Span's first {@value References#CHILD_OF} reference, or the first reference of any type, or 0 if no reference exists.
*
* @see MockContext#spanId()
* @see MockSpan#references()
*/
public long parentId() {
return parentId;
}
public long startMicros() {
return startMicros;
}
/**
* @return the finish time of the Span; only valid after a call to finish().
*/
public long finishMicros() {
assert finishMicros > 0 : "must call finish() before finishMicros()";
return finishMicros;
}
/**
* @return a copy of all tags set on this Span.
*/
public Map tags() {
return new HashMap<>(this.tags);
}
/**
* @return a copy of all log entries added to this Span.
*/
public List logEntries() {
return new ArrayList<>(this.logEntries);
}
/**
* @return a copy of exceptions thrown by this class (e.g. adding a tag after span is finished).
*/
public List generatedErrors() {
return new ArrayList<>(errors);
}
public List references() {
return new ArrayList<>(references);
}
@Override
public synchronized MockContext context() {
return this.context;
}
@Override
public void finish() {
this.finish(nowMicros());
}
@Override
public synchronized void finish(long finishMicros) {
finishedCheck("Finishing already finished span");
this.finishMicros = finishMicros;
this.mockTracer.appendFinishedSpan(this);
this.finished = true;
}
@Override
public MockSpan setTag(String key, String value) {
return setObjectTag(key, value);
}
@Override
public MockSpan setTag(String key, boolean value) {
return setObjectTag(key, value);
}
@Override
public MockSpan setTag(String key, Number value) {
return setObjectTag(key, value);
}
private synchronized MockSpan setObjectTag(String key, Object value) {
finishedCheck("Adding tag {%s:%s} to already finished span", key, value);
tags.put(key, value);
return this;
}
@Override
public final Span log(Map fields) {
return log(nowMicros(), fields);
}
@Override
public final synchronized MockSpan log(long timestampMicros, Map fields) {
finishedCheck("Adding logs %s at %d to already finished span", fields, timestampMicros);
this.logEntries.add(new LogEntry(timestampMicros, fields));
return this;
}
@Override
public MockSpan log(String event) {
return this.log(nowMicros(), event);
}
@Override
public MockSpan log(long timestampMicroseconds, String event) {
return this.log(timestampMicroseconds, Collections.singletonMap("event", event));
}
@Override
public synchronized Span setBaggageItem(String key, String value) {
finishedCheck("Adding baggage {%s:%s} to already finished span", key, value);
this.context = this.context.withBaggageItem(key, value);
return this;
}
@Override
public synchronized String getBaggageItem(String key) {
return this.context.getBaggageItem(key);
}
/**
* MockContext implements a Dapper-like opentracing.SpanContext with a trace- and span-id.
*
* Note that parent ids are part of the MockSpan, not the MockContext (since they do not need to propagate
* between processes).
*/
public static final class MockContext implements SpanContext {
private final long traceId;
private final Map baggage;
private final long spanId;
/**
* A package-protected constructor to create a new MockContext. This should only be called by MockSpan and/or
* MockTracer.
*
* @param baggage the MockContext takes ownership of the baggage parameter
*
* @see MockContext#withBaggageItem(String, String)
*/
public MockContext(long traceId, long spanId, Map baggage) {
this.baggage = baggage;
this.traceId = traceId;
this.spanId = spanId;
}
public String getBaggageItem(String key) { return this.baggage.get(key); }
public long traceId() { return traceId; }
public long spanId() { return spanId; }
/**
* Create and return a new (immutable) MockContext with the added baggage item.
*/
public MockContext withBaggageItem(String key, String val) {
Map newBaggage = new HashMap<>(this.baggage);
newBaggage.put(key, val);
return new MockContext(this.traceId, this.spanId, newBaggage);
}
@Override
public Iterable> baggageItems() {
return baggage.entrySet();
}
}
public static final class LogEntry {
private final long timestampMicros;
private final Map fields;
public LogEntry(long timestampMicros, Map fields) {
this.timestampMicros = timestampMicros;
this.fields = fields;
}
public long timestampMicros() {
return timestampMicros;
}
public Map fields() {
return fields;
}
}
public static final class Reference {
private final MockContext context;
private final String referenceType;
public Reference(MockContext context, String referenceType) {
this.context = context;
this.referenceType = referenceType;
}
public MockContext getContext() {
return context;
}
public String getReferenceType() {
return referenceType;
}
@Override
public boolean equals(Object o) {
if (this == o) return true;
if (o == null || getClass() != o.getClass()) return false;
Reference reference = (Reference) o;
return Objects.equals(context, reference.context) &&
Objects.equals(referenceType, reference.referenceType);
}
@Override
public int hashCode() {
return Objects.hash(context, referenceType);
}
}
MockSpan(MockTracer tracer, String operationName, long startMicros, Map initialTags, List refs) {
this.mockTracer = tracer;
this.operationName = operationName;
this.startMicros = startMicros;
if (initialTags == null) {
this.tags = new HashMap<>();
} else {
this.tags = new HashMap<>(initialTags);
}
if(refs == null) {
this.references = Collections.emptyList();
} else {
this.references = new ArrayList<>(refs);
}
MockContext parent = findPreferredParentRef(this.references);
if (parent == null) {
// We're a root Span.
this.context = new MockContext(nextId(), nextId(), new HashMap());
this.parentId = 0;
} else {
// We're a child Span.
this.context = new MockContext(parent.traceId, nextId(), mergeBaggages(this.references));
this.parentId = parent.spanId;
}
}
private static MockContext findPreferredParentRef(List references) {
if(references.isEmpty()) {
return null;
}
for (Reference reference : references) {
if (References.CHILD_OF.equals(reference.getReferenceType())) {
return reference.getContext();
}
}
return references.get(0).getContext();
}
private static Map mergeBaggages(List references) {
Map baggage = new HashMap<>();
for(Reference ref : references) {
if(ref.getContext().baggage != null) {
baggage.putAll(ref.getContext().baggage);
}
}
return baggage;
}
static long nextId() {
return nextId.addAndGet(1);
}
static long nowMicros() {
return System.currentTimeMillis() * 1000;
}
private synchronized void finishedCheck(String format, Object... args) {
if (finished) {
RuntimeException ex = new IllegalStateException(String.format(format, args));
errors.add(ex);
throw ex;
}
}
@Override
public String toString() {
return "{" +
"traceId:" + context.traceId() +
", spanId:" + context.spanId() +
", parentId:" + parentId +
", operationName:\"" + operationName + "\"}";
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy