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

io.sentry.protocol.SentryTransaction Maven / Gradle / Ivy

package io.sentry.protocol;

import io.sentry.DateUtils;
import io.sentry.ILogger;
import io.sentry.JsonDeserializer;
import io.sentry.JsonSerializable;
import io.sentry.JsonUnknown;
import io.sentry.ObjectReader;
import io.sentry.ObjectWriter;
import io.sentry.SentryBaseEvent;
import io.sentry.SentryTracer;
import io.sentry.Span;
import io.sentry.SpanContext;
import io.sentry.SpanStatus;
import io.sentry.TracesSamplingDecision;
import io.sentry.metrics.LocalMetricsAggregator;
import io.sentry.util.Objects;
import io.sentry.vendor.gson.stream.JsonToken;
import java.io.IOException;
import java.math.BigDecimal;
import java.math.RoundingMode;
import java.util.ArrayList;
import java.util.Date;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import org.jetbrains.annotations.ApiStatus;
import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;

@ApiStatus.Internal
public final class SentryTransaction extends SentryBaseEvent
    implements JsonUnknown, JsonSerializable {
  /** The transaction name. */
  @SuppressWarnings("UnusedVariable")
  private @Nullable String transaction;

  /** The moment in time when span was started. */
  private @NotNull Double startTimestamp;

  /** The moment in time when span has ended. */
  private @Nullable Double timestamp;

  /** A list of spans within this transaction. Can be empty. */
  private final @NotNull List spans = new ArrayList<>();

  /** The {@code type} property is required in JSON payload sent to Sentry. */
  @SuppressWarnings("UnusedVariable")
  private @NotNull final String type = "transaction";

  private @NotNull final Map measurements = new HashMap<>();

  private @Nullable Map> metricSummaries;

  private @NotNull TransactionInfo transactionInfo;

  private @Nullable Map unknown;

  @SuppressWarnings("deprecation")
  public SentryTransaction(final @NotNull SentryTracer sentryTracer) {
    super(sentryTracer.getEventId());
    Objects.requireNonNull(sentryTracer, "sentryTracer is required");
    // we lose precision here, from potential nanosecond precision down to 10 microsecond precision
    this.startTimestamp = DateUtils.nanosToSeconds(sentryTracer.getStartDate().nanoTimestamp());
    // we lose precision here, from potential nanosecond precision down to 10 microsecond precision
    this.timestamp =
        DateUtils.nanosToSeconds(
            sentryTracer
                .getStartDate()
                .laterDateNanosTimestampByDiff(sentryTracer.getFinishDate()));
    this.transaction = sentryTracer.getName();
    for (final Span span : sentryTracer.getChildren()) {
      if (Boolean.TRUE.equals(span.isSampled())) {
        this.spans.add(new SentrySpan(span));
      }
    }
    final Contexts contexts = this.getContexts();

    contexts.putAll(sentryTracer.getContexts());

    final SpanContext tracerContext = sentryTracer.getSpanContext();
    // tags must be placed on the root of the transaction instead of contexts.trace.tags
    contexts.setTrace(
        new SpanContext(
            tracerContext.getTraceId(),
            tracerContext.getSpanId(),
            tracerContext.getParentSpanId(),
            tracerContext.getOperation(),
            tracerContext.getDescription(),
            tracerContext.getSamplingDecision(),
            tracerContext.getStatus(),
            tracerContext.getOrigin()));
    for (final Map.Entry tag : tracerContext.getTags().entrySet()) {
      this.setTag(tag.getKey(), tag.getValue());
    }

    final Map data = sentryTracer.getData();
    if (data != null) {
      for (final Map.Entry tag : data.entrySet()) {
        this.setExtra(tag.getKey(), tag.getValue());
      }
    }

    this.transactionInfo = new TransactionInfo(sentryTracer.getTransactionNameSource().apiName());

    final @Nullable LocalMetricsAggregator localAggregator =
        sentryTracer.getLocalMetricsAggregator();
    if (localAggregator != null) {
      this.metricSummaries = localAggregator.getSummaries();
    } else {
      this.metricSummaries = null;
    }
  }

  @ApiStatus.Internal
  public SentryTransaction(
      @Nullable String transaction,
      @NotNull Double startTimestamp,
      @Nullable Double timestamp,
      @NotNull List spans,
      @NotNull final Map measurements,
      @Nullable Map> metricsSummaries,
      @NotNull final TransactionInfo transactionInfo) {
    this.transaction = transaction;
    this.startTimestamp = startTimestamp;
    this.timestamp = timestamp;
    this.spans.addAll(spans);
    this.measurements.putAll(measurements);
    for (SentrySpan span : spans) {
      this.measurements.putAll(span.getMeasurements());
    }
    this.transactionInfo = transactionInfo;
    this.metricSummaries = metricsSummaries;
  }

  public @NotNull List getSpans() {
    return spans;
  }

  public boolean isFinished() {
    return this.timestamp != null;
  }

  public @Nullable String getTransaction() {
    return transaction;
  }

  public @NotNull Double getStartTimestamp() {
    return startTimestamp;
  }

  public @Nullable Double getTimestamp() {
    return timestamp;
  }

  public @NotNull String getType() {
    return type;
  }

  public @Nullable SpanStatus getStatus() {
    final SpanContext trace = this.getContexts().getTrace();
    return trace != null ? trace.getStatus() : null;
  }

  public boolean isSampled() {
    final @Nullable TracesSamplingDecision samplingDecsion = getSamplingDecision();
    if (samplingDecsion == null) {
      return false;
    }

    return samplingDecsion.getSampled();
  }

  public @Nullable TracesSamplingDecision getSamplingDecision() {
    final SpanContext trace = this.getContexts().getTrace();
    if (trace == null) {
      return null;
    }

    return trace.getSamplingDecision();
  }

  public @NotNull Map getMeasurements() {
    return measurements;
  }

  public @Nullable Map> getMetricSummaries() {
    return metricSummaries;
  }

  public void setMetricSummaries(final @Nullable Map> metricSummaries) {
    this.metricSummaries = metricSummaries;
  }

  // JsonSerializable

  public static final class JsonKeys {
    public static final String TRANSACTION = "transaction";
    public static final String START_TIMESTAMP = "start_timestamp";
    public static final String TIMESTAMP = "timestamp";
    public static final String SPANS = "spans";
    public static final String TYPE = "type";
    public static final String MEASUREMENTS = "measurements";
    public static final String METRICS_SUMMARY = "_metrics_summary";
    public static final String TRANSACTION_INFO = "transaction_info";
  }

  @Override
  public void serialize(final @NotNull ObjectWriter writer, final @NotNull ILogger logger)
      throws IOException {
    writer.beginObject();
    if (transaction != null) {
      writer.name(JsonKeys.TRANSACTION).value(transaction);
    }
    writer.name(JsonKeys.START_TIMESTAMP).value(logger, doubleToBigDecimal(startTimestamp));
    if (timestamp != null) {
      writer.name(JsonKeys.TIMESTAMP).value(logger, doubleToBigDecimal(timestamp));
    }
    if (!spans.isEmpty()) {
      writer.name(JsonKeys.SPANS).value(logger, spans);
    }
    writer.name(JsonKeys.TYPE).value(type);
    if (!measurements.isEmpty()) {
      writer.name(JsonKeys.MEASUREMENTS).value(logger, measurements);
    }
    if (metricSummaries != null && !metricSummaries.isEmpty()) {
      writer.name(JsonKeys.METRICS_SUMMARY).value(logger, metricSummaries);
    }
    writer.name(JsonKeys.TRANSACTION_INFO).value(logger, transactionInfo);
    new SentryBaseEvent.Serializer().serialize(this, writer, logger);
    if (unknown != null) {
      for (String key : unknown.keySet()) {
        Object value = unknown.get(key);
        writer.name(key);
        writer.value(logger, value);
      }
    }
    writer.endObject();
  }

  private @NotNull BigDecimal doubleToBigDecimal(final @NotNull Double value) {
    return BigDecimal.valueOf(value).setScale(6, RoundingMode.DOWN);
  }

  @Nullable
  @Override
  public Map getUnknown() {
    return unknown;
  }

  @Override
  public void setUnknown(@Nullable Map unknown) {
    this.unknown = unknown;
  }

  public static final class Deserializer implements JsonDeserializer {

    @SuppressWarnings({"unchecked", "JavaUtilDate"})
    @Override
    public @NotNull SentryTransaction deserialize(
        @NotNull ObjectReader reader, @NotNull ILogger logger) throws Exception {
      reader.beginObject();

      // Init with placeholders.
      SentryTransaction transaction =
          new SentryTransaction(
              "",
              0d,
              null,
              new ArrayList<>(),
              new HashMap<>(),
              null,
              new TransactionInfo(TransactionNameSource.CUSTOM.apiName()));
      Map unknown = null;

      SentryBaseEvent.Deserializer baseEventDeserializer = new SentryBaseEvent.Deserializer();

      while (reader.peek() == JsonToken.NAME) {
        final String nextName = reader.nextName();
        switch (nextName) {
          case JsonKeys.TRANSACTION:
            transaction.transaction = reader.nextStringOrNull();
            break;
          case JsonKeys.START_TIMESTAMP:
            try {
              final Double deserializedStartTimestamp = reader.nextDoubleOrNull();
              if (deserializedStartTimestamp != null) {
                transaction.startTimestamp = deserializedStartTimestamp;
              }
            } catch (NumberFormatException e) {
              final Date date = reader.nextDateOrNull(logger);
              if (date != null) {
                transaction.startTimestamp = DateUtils.dateToSeconds(date);
              }
            }
            break;
          case JsonKeys.TIMESTAMP:
            try {
              final Double deserializedTimestamp = reader.nextDoubleOrNull();
              if (deserializedTimestamp != null) {
                transaction.timestamp = deserializedTimestamp;
              }
            } catch (NumberFormatException e) {
              final Date date = reader.nextDateOrNull(logger);
              if (date != null) {
                transaction.timestamp = DateUtils.dateToSeconds(date);
              }
            }
            break;
          case JsonKeys.SPANS:
            List deserializedSpans =
                reader.nextListOrNull(logger, new SentrySpan.Deserializer());
            if (deserializedSpans != null) {
              transaction.spans.addAll(deserializedSpans);
            }
            break;
          case JsonKeys.TYPE:
            reader.nextString(); // No need to assign, as it is final.
            break;
          case JsonKeys.MEASUREMENTS:
            Map deserializedMeasurements =
                reader.nextMapOrNull(logger, new MeasurementValue.Deserializer());
            if (deserializedMeasurements != null) {
              transaction.measurements.putAll(deserializedMeasurements);
            }
            break;
          case SentrySpan.JsonKeys.METRICS_SUMMARY:
            transaction.metricSummaries =
                reader.nextMapOfListOrNull(logger, new MetricSummary.Deserializer());
            break;
          case JsonKeys.TRANSACTION_INFO:
            transaction.transactionInfo =
                new TransactionInfo.Deserializer().deserialize(reader, logger);
            break;
          default:
            if (!baseEventDeserializer.deserializeValue(transaction, nextName, reader, logger)) {
              if (unknown == null) {
                unknown = new ConcurrentHashMap<>();
              }
              reader.nextUnknown(logger, unknown, nextName);
            }
            break;
        }
      }
      transaction.setUnknown(unknown);
      reader.endObject();
      return transaction;
    }
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy