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

io.micrometer.tracing.reporter.wavefront.WavefrontSpanHandler Maven / Gradle / Ivy

/**
 * Copyright 2022 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.reporter.wavefront;

import com.wavefront.internal.reporter.WavefrontInternalReporter;
import com.wavefront.sdk.common.NamedThreadFactory;
import com.wavefront.sdk.common.Pair;
import com.wavefront.sdk.common.WavefrontSender;
import com.wavefront.sdk.common.application.ApplicationTags;
import com.wavefront.sdk.entities.tracing.SpanLog;
import io.micrometer.common.util.StringUtils;
import io.micrometer.common.util.internal.logging.InternalLogger;
import io.micrometer.common.util.internal.logging.InternalLoggerFactory;
import io.micrometer.tracing.TraceContext;
import io.micrometer.tracing.exporter.FinishedSpan;

import java.io.Closeable;
import java.io.IOException;
import java.time.temporal.ChronoUnit;
import java.util.*;
import java.util.concurrent.*;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicLong;
import java.util.stream.Collectors;

import static com.wavefront.internal.SpanDerivedMetricsUtils.*;
import static com.wavefront.sdk.common.Constants.*;

/**
 * This converts a span recorded by Micrometer Tracing and invokes
 * {@link WavefrontSender#sendSpan}.
 *
 * 

* This uses a combination of conversion approaches from Wavefront projects: *

    *
  • https://github.com/wavefrontHQ/wavefront-opentracing-sdk-java
  • *
  • https://github.com/wavefrontHQ/wavefront-proxy
  • *
* *

* On conflict, we make a comment and prefer wavefront-opentracing-sdk-java. The rationale * is wavefront-opentracing-sdk-java uses the same {@link WavefrontSender#sendSpan} * library, so it is easier to reason with. This policy can be revisited by future * maintainers. * *

* Note:UUID conversions follow the same conventions used in practice in * Wavefront. Ex. * https://github.com/wavefrontHQ/wavefront-opentracing-sdk-java/blob/6babf2ff95daa37452e1e8c35ae54b58b6abb50f/src/main/java/com/wavefront/opentracing/propagation/JaegerWavefrontPropagator.java#L191-L204 * While in practice this is not a problem, it is worth mentioning that this convention * will only result in RFC 4122 timestamp (version 1) format by accident. In other words, * don't call {@link UUID#timestamp()} on UUIDs converted here, or in other Wavefront * code, as it might throw an exception. * * @since 1.0.0 */ public class WavefrontSpanHandler implements Runnable, Closeable { private static final InternalLogger LOG = InternalLoggerFactory.getInstance(WavefrontSpanHandler.class); // https://github.com/wavefrontHQ/wavefront-proxy/blob/3dd1fa11711a04de2d9d418e2269f0f9fb464f36/proxy/src/main/java/com/wavefront/agent/listeners/tracing/ZipkinPortUnificationHandler.java#L114-L114 private static final String DEFAULT_SPAN_NAME = "defaultOperation"; private static final String DEFAULT_SOURCE = "micrometer-tracing"; private static final String WAVEFRONT_GENERATED_COMPONENT = "wavefront-generated"; private static final int LONG_BYTES = Long.SIZE / Byte.SIZE; private static final int BYTE_BASE16 = 2; private static final int LONG_BASE16 = BYTE_BASE16 * LONG_BYTES; private static final int TRACE_ID_HEX_SIZE = 2 * LONG_BASE16; private static final String ALPHABET = "0123456789abcdef"; private static final int ASCII_CHARACTERS = 128; private static final byte[] DECODING = buildDecodingArray(); private final LinkedBlockingQueue spanBuffer; private final WavefrontSender wavefrontSender; private final WavefrontInternalReporter wfInternalReporter; private final Set traceDerivedCustomTagKeys; private final Thread sendingThread; private final Set, String>> discoveredHeartbeatMetrics; private final ScheduledExecutorService heartbeatMetricsScheduledExecutorService; private final String source; private final List> defaultTags; private final Set defaultTagKeys; private final ApplicationTags applicationTags; private final SpanMetrics spanMetrics; private final AtomicBoolean stop = new AtomicBoolean(); private final AtomicLong spansDropped = new AtomicLong(); /** * Creates a new instance of {@link WavefrontSpanHandler}. * @param maxQueueSize maximal span queue size * @param wavefrontSender wavefront server * @param spanMetrics span metrics * @param source source of metrics and spans * @param applicationTags additional application tags * @param redMetricsCustomTagKeys RED metrics custom tag keys */ public WavefrontSpanHandler(int maxQueueSize, WavefrontSender wavefrontSender, SpanMetrics spanMetrics, String source, ApplicationTags applicationTags, Set redMetricsCustomTagKeys) { this.wavefrontSender = wavefrontSender; this.applicationTags = applicationTags; this.discoveredHeartbeatMetrics = ConcurrentHashMap.newKeySet(); this.spanMetrics = spanMetrics; this.heartbeatMetricsScheduledExecutorService = Executors.newScheduledThreadPool(1, new NamedThreadFactory("micrometer-heart-beater").setDaemon(true)); // Emit Heartbeats Metrics every 1 min. heartbeatMetricsScheduledExecutorService.scheduleAtFixedRate(() -> { try { reportHeartbeats(wavefrontSender, discoveredHeartbeatMetrics, WAVEFRONT_GENERATED_COMPONENT); } catch (IOException e) { LOG.warn("Cannot report heartbeat metric to wavefront"); } }, 1, 60, TimeUnit.SECONDS); this.traceDerivedCustomTagKeys = new HashSet<>(redMetricsCustomTagKeys); // Start the reporter wfInternalReporter = new WavefrontInternalReporter.Builder().prefixedWith(TRACING_DERIVED_PREFIX) .withSource(DEFAULT_SOURCE) .reportMinuteDistribution() .build(wavefrontSender); wfInternalReporter.start(1, TimeUnit.MINUTES); this.source = source; this.defaultTags = createDefaultTags(applicationTags); this.defaultTagKeys = defaultTags.stream().map(p -> p._1).collect(Collectors.toSet()); this.defaultTagKeys.add(SOURCE_KEY); this.spanBuffer = new LinkedBlockingQueue<>(maxQueueSize); spanMetrics.registerQueueSize(spanBuffer); spanMetrics.registerQueueRemainingCapacity(spanBuffer); this.sendingThread = new Thread(this, "wavefrontSpanReporter"); this.sendingThread.setDaemon(true); this.sendingThread.start(); } private static byte[] buildDecodingArray() { byte[] decoding = new byte[ASCII_CHARACTERS]; Arrays.fill(decoding, (byte) -1); for (int i = 0; i < ALPHABET.length(); i++) { char c = ALPHABET.charAt(i); decoding[c] = (byte) i; } return decoding; } /** * Returns the {@code long} value whose base16 representation is stored in the first * 16 chars of {@code chars} starting from the {@code offset}. * @param chars the base16 representation of the {@code long}. */ private static long longFromBase16String(CharSequence chars) { int offset = 0; return (decodeByte(chars.charAt(offset), chars.charAt(offset + 1)) & 0xFFL) << 56 | (decodeByte(chars.charAt(offset + 2), chars.charAt(offset + 3)) & 0xFFL) << 48 | (decodeByte(chars.charAt(offset + 4), chars.charAt(offset + 5)) & 0xFFL) << 40 | (decodeByte(chars.charAt(offset + 6), chars.charAt(offset + 7)) & 0xFFL) << 32 | (decodeByte(chars.charAt(offset + 8), chars.charAt(offset + 9)) & 0xFFL) << 24 | (decodeByte(chars.charAt(offset + 10), chars.charAt(offset + 11)) & 0xFFL) << 16 | (decodeByte(chars.charAt(offset + 12), chars.charAt(offset + 13)) & 0xFFL) << 8 | (decodeByte(chars.charAt(offset + 14), chars.charAt(offset + 15)) & 0xFFL); } private static byte decodeByte(char hi, char lo) { int decoded = DECODING[hi] << 4 | DECODING[lo]; return (byte) decoded; } // https://github.com/wavefrontHQ/wavefront-proxy/blob/3dd1fa11711a04de2d9d418e2269f0f9fb464f36/proxy/src/main/java/com/wavefront/agent/listeners/tracing/ZipkinPortUnificationHandler.java#L397-L402 static List convertAnnotationsToSpanLogs(FinishedSpan span) { return span.getEvents() .stream() .map(entry -> new SpanLog(entry.getKey(), Collections.singletonMap("annotation", entry.getValue()))) .collect(Collectors.toList()); } // https://github.com/wavefrontHQ/wavefront-opentracing-sdk-java/blob/f1f08d8daf7b692b9b61dcd5bc24ca6befa8e710/src/main/java/com/wavefront/opentracing/WavefrontTracer.java#L275-L280 static List> createDefaultTags(ApplicationTags applicationTags) { List> result = new ArrayList<>(); result.add(Pair.of(APPLICATION_TAG_KEY, applicationTags.getApplication())); result.add(Pair.of(SERVICE_TAG_KEY, applicationTags.getService())); result.add(Pair.of(CLUSTER_TAG_KEY, applicationTags.getCluster() == null ? NULL_TAG_VAL : applicationTags.getCluster())); result.add( Pair.of(SHARD_TAG_KEY, applicationTags.getShard() == null ? NULL_TAG_VAL : applicationTags.getShard())); if (applicationTags.getCustomTags() != null) { applicationTags.getCustomTags().forEach((k, v) -> result.add(Pair.of(k, v))); } return result; } /** * Exact same behavior as WavefrontSpanReporter. * https://github.com/wavefrontHQ/wavefront-opentracing-sdk-java/blob/f1f08d8daf7b692b9b61dcd5bc24ca6befa8e710/src/main/java/com/wavefront/opentracing/reporting/WavefrontSpanReporter.java#L163-L179 * @param context trace context * @param span reported span * @return should other handler be ran */ public boolean end(TraceContext context, FinishedSpan span) { this.spanMetrics.reportReceived(); if (stop.get()) { // A span is being reported, but close() has already been called this.spanMetrics.reportDropped(); long dropped = this.spansDropped.incrementAndGet(); if (LOG.isWarnEnabled()) { LOG.warn("Trying to send span after close() have been called, dropping span " + span); LOG.warn("Total spans dropped: " + dropped); } } else { if (!spanBuffer.offer(new SpanToSend(context, span))) { this.spanMetrics.reportDropped(); long dropped = this.spansDropped.incrementAndGet(); if (LOG.isWarnEnabled()) { LOG.warn("Buffer full, dropping span: " + span); LOG.warn("Total spans dropped: " + dropped); } } } return true; // regardless of error, other handlers should run } List> getDefaultTags() { return Collections.unmodifiableList(this.defaultTags); } private String padLeftWithZeros(String string, int length) { if (string.length() >= length) { return string; } else { StringBuilder sb = new StringBuilder(length); for (int i = string.length(); i < length; i++) { sb.append('0'); } return sb.append(string).toString(); } } private void send(TraceContext context, FinishedSpan span) { String traceIdString = padLeftWithZeros(context.traceId(), TRACE_ID_HEX_SIZE); String traceIdHigh = traceIdString.substring(0, traceIdString.length() / 2); String traceIdLow = traceIdString.substring(traceIdString.length() / 2); UUID traceId = new UUID(longFromBase16String(traceIdHigh), longFromBase16String(traceIdLow)); UUID spanId = new UUID(0L, longFromBase16String(context.spanId())); // NOTE: wavefront-opentracing-sdk-java and wavefront-proxy differ, but we prefer // the former. // https://github.com/wavefrontHQ/wavefront-opentracing-sdk-java/blob/f1f08d8daf7b692b9b61dcd5bc24ca6befa8e710/src/main/java/com/wavefront/opentracing/reporting/WavefrontSpanReporter.java#L187-L190 // https://github.com/wavefrontHQ/wavefront-proxy/blob/3dd1fa11711a04de2d9d418e2269f0f9fb464f36/proxy/src/main/java/com/wavefront/agent/listeners/tracing/ZipkinPortUnificationHandler.java#L248-L252 List parents = null; String parentId = context.parentId(); if (!StringUtils.isEmpty(parentId) && longFromBase16String(parentId) != 0L) { parents = Collections.singletonList(new UUID(0L, longFromBase16String(parentId))); } List followsFrom = null; // https://github.com/wavefrontHQ/wavefront-proxy/blob/3dd1fa11711a04de2d9d418e2269f0f9fb464f36/proxy/src/main/java/com/wavefront/agent/listeners/tracing/ZipkinPortUnificationHandler.java#L344-L345 String name = span.getName(); if (name == null) { name = DEFAULT_SPAN_NAME; } // Start and duration become 0L if unset. Any positive duration rounds up to 1 // millis. long startMillis = span.getStartTimestamp().toEpochMilli(); long finishMillis = span.getEndTimestamp().toEpochMilli(); long durationMicros = ChronoUnit.MICROS.between(span.getStartTimestamp(), span.getEndTimestamp()); long durationMillis = startMillis != 0 && finishMillis != 0L ? Math.max(finishMillis - startMillis, 1L) : 0L; List spanLogs = convertAnnotationsToSpanLogs(span); TagList tags = new TagList(defaultTagKeys, defaultTags, span); try { wavefrontSender.sendSpan(name, startMillis, durationMillis, source, traceId, spanId, parents, followsFrom, tags, spanLogs); } catch (IOException | RuntimeException t) { if (LOG.isDebugEnabled()) { LOG.debug("error sending span " + context, t); } this.spanMetrics.reportErrors(); } // report stats irrespective of span sampling. if (wfInternalReporter != null) { // report converted metrics/histograms from the span try { discoveredHeartbeatMetrics.add(reportWavefrontGeneratedData(wfInternalReporter, name, applicationTags.getApplication(), applicationTags.getService(), applicationTags.getCluster() == null ? NULL_TAG_VAL : applicationTags.getCluster(), applicationTags.getShard() == null ? NULL_TAG_VAL : applicationTags.getShard(), source, tags.componentTagValue, tags.isError, durationMicros, traceDerivedCustomTagKeys, tags)); } catch (RuntimeException t) { if (LOG.isDebugEnabled()) { LOG.debug("error sending span RED metrics " + context, t); } this.spanMetrics.reportErrors(); } } } @Override public void run() { while (!stop.get()) { try { SpanToSend spanToSend = spanBuffer.take(); if (spanToSend == DeathPill.INSTANCE) { LOG.info("reporting thread stopping"); return; } send(spanToSend.getTraceContext(), spanToSend.getFinishedSpan()); } catch (InterruptedException ex) { if (LOG.isInfoEnabled()) { LOG.info("reporting thread interrupted"); } } catch (Throwable ex) { LOG.warn("Error processing buffer", ex); } } } @Override public void close() { if (!stop.compareAndSet(false, true)) { // Ignore multiple stop calls return; } try { // This will release the thread if it's waiting in BlockingQueue#take() spanBuffer.offer(DeathPill.INSTANCE); // wait for 5 secs max to send remaining spans sendingThread.join(5000); sendingThread.interrupt(); heartbeatMetricsScheduledExecutorService.shutdownNow(); } catch (InterruptedException ex) { // no-op } try { // It seems WavefrontClient does not support graceful shutdown, so we need to // flush manually, and // send should not be called after flush wavefrontSender.flush(); wavefrontSender.close(); } catch (IOException e) { LOG.warn("Unable to close Wavefront Client", e); } } private static class SpanToSend { private final TraceContext traceContext; private final FinishedSpan finishedSpan; SpanToSend(TraceContext traceContext, FinishedSpan finishedSpan) { this.traceContext = traceContext; this.finishedSpan = finishedSpan; } TraceContext getTraceContext() { return traceContext; } FinishedSpan getFinishedSpan() { return finishedSpan; } } /** * Gets queued into {@link #spanBuffer} if {@link #close()} is called and will lead * the sender thread to stop. */ private static class DeathPill extends SpanToSend { static final DeathPill INSTANCE = new DeathPill(); private DeathPill() { super(null, null); } } /** * Extracted for test isolation and as parsing otherwise implies multiple-returns or * scanning later. * *

* Ex. {@code SpanDerivedMetricsUtils#reportWavefrontGeneratedData} needs tags * separately from the component tag and error status. */ static final class TagList extends ArrayList> { String componentTagValue = NULL_TAG_VAL; boolean isError; // See explanation here: // https://github.com/openzipkin/brave/pull/1221 TagList(Set defaultTagKeys, List> defaultTags, FinishedSpan span) { super(defaultTags.size() + span.getTags().size()); boolean debug = false; // OTel doesn't have a notion of debug boolean hasAnnotations = span.getEvents().size() > 0; isError = span.getError() != null; addAll(defaultTags); for (Map.Entry tag : span.getTags().entrySet()) { String lowerCaseKey = tag.getKey().toLowerCase(Locale.ROOT); if (lowerCaseKey.equals(ERROR_TAG_KEY)) { isError = true; continue; // We later replace whatever the potentially empty value was // with "true" } if (tag.getValue().isEmpty()) { continue; } if (defaultTagKeys.contains(lowerCaseKey)) { continue; } if (lowerCaseKey.equals(DEBUG_TAG_KEY)) { debug = true; // This tag is set out-of-band continue; } if (lowerCaseKey.equals(COMPONENT_TAG_KEY)) { componentTagValue = tag.getValue(); } add(Pair.of(tag.getKey(), tag.getValue())); } // Check for span.error() for uncaught exception in request mapping and add it // to Wavefront span tag if (isError) { add(Pair.of("error", "true")); } // https://github.com/wavefrontHQ/wavefront-proxy/blob/3dd1fa11711a04de2d9d418e2269f0f9fb464f36/proxy/src/main/java/com/wavefront/agent/listeners/tracing/ZipkinPortUnificationHandler.java#L300-L303 if (debug) { add(Pair.of(DEBUG_TAG_KEY, "true")); } // https://github.com/wavefrontHQ/wavefront-proxy/blob/3dd1fa11711a04de2d9d418e2269f0f9fb464f36/proxy/src/main/java/com/wavefront/agent/listeners/tracing/ZipkinPortUnificationHandler.java#L254-L266 if (span.getKind() != null) { String kind = span.getKind().toString().toLowerCase(); add(Pair.of("span.kind", kind)); if (hasAnnotations) { add(Pair.of("_spanSecondaryId", kind)); } } // https://github.com/wavefrontHQ/wavefront-proxy/blob/3dd1fa11711a04de2d9d418e2269f0f9fb464f36/proxy/src/main/java/com/wavefront/agent/listeners/tracing/ZipkinPortUnificationHandler.java#L329-L332 if (hasAnnotations) { add(Pair.of(SPAN_LOG_KEY, "true")); } // https://github.com/wavefrontHQ/wavefront-proxy/blob/3dd1fa11711a04de2d9d418e2269f0f9fb464f36/proxy/src/main/java/com/wavefront/agent/listeners/tracing/ZipkinPortUnificationHandler.java#L324-L327 if (span.getLocalIp() != null) { String localIp = span.getLocalIp(); String version = localIp.contains(":") ? "ipv6" : "ipv4"; add(Pair.of(version, localIp)); } } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy