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

io.sentry.JsonSerializer Maven / Gradle / Ivy

package io.sentry;

import io.sentry.clientreport.ClientReport;
import io.sentry.profilemeasurements.ProfileMeasurement;
import io.sentry.profilemeasurements.ProfileMeasurementValue;
import io.sentry.protocol.App;
import io.sentry.protocol.Browser;
import io.sentry.protocol.Contexts;
import io.sentry.protocol.DebugImage;
import io.sentry.protocol.DebugMeta;
import io.sentry.protocol.Device;
import io.sentry.protocol.Geo;
import io.sentry.protocol.Gpu;
import io.sentry.protocol.MeasurementValue;
import io.sentry.protocol.Mechanism;
import io.sentry.protocol.Message;
import io.sentry.protocol.MetricSummary;
import io.sentry.protocol.OperatingSystem;
import io.sentry.protocol.Request;
import io.sentry.protocol.SdkInfo;
import io.sentry.protocol.SdkVersion;
import io.sentry.protocol.SentryException;
import io.sentry.protocol.SentryPackage;
import io.sentry.protocol.SentryRuntime;
import io.sentry.protocol.SentrySpan;
import io.sentry.protocol.SentryStackFrame;
import io.sentry.protocol.SentryStackTrace;
import io.sentry.protocol.SentryThread;
import io.sentry.protocol.SentryTransaction;
import io.sentry.protocol.User;
import io.sentry.protocol.ViewHierarchy;
import io.sentry.protocol.ViewHierarchyNode;
import io.sentry.rrweb.RRWebBreadcrumbEvent;
import io.sentry.rrweb.RRWebEventType;
import io.sentry.rrweb.RRWebInteractionEvent;
import io.sentry.rrweb.RRWebInteractionMoveEvent;
import io.sentry.rrweb.RRWebMetaEvent;
import io.sentry.rrweb.RRWebSpanEvent;
import io.sentry.rrweb.RRWebVideoEvent;
import io.sentry.util.Objects;
import java.io.BufferedOutputStream;
import java.io.BufferedWriter;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.io.OutputStreamWriter;
import java.io.Reader;
import java.io.StringWriter;
import java.io.Writer;
import java.nio.charset.Charset;
import java.util.Collection;
import java.util.HashMap;
import java.util.Map;
import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;

/**
 * The serializer class that uses manual JSON parsing with the help of vendored GSON reader/writer
 * classes.
 */
public final class JsonSerializer implements ISerializer {

  /** the UTF-8 Charset */
  @SuppressWarnings("CharsetObjectCanBeUsed")
  private static final Charset UTF_8 = Charset.forName("UTF-8");

  /** the SentryOptions */
  private final @NotNull SentryOptions options;

  private final @NotNull Map, JsonDeserializer> deserializersByClass;

  /**
   * All our custom deserializers need to be registered to be used with the deserializer instance. *
   */
  public JsonSerializer(@NotNull SentryOptions options) {
    this.options = options;

    deserializersByClass = new HashMap<>();
    deserializersByClass.put(App.class, new App.Deserializer());
    deserializersByClass.put(Breadcrumb.class, new Breadcrumb.Deserializer());
    deserializersByClass.put(Browser.class, new Browser.Deserializer());
    deserializersByClass.put(Contexts.class, new Contexts.Deserializer());
    deserializersByClass.put(DebugImage.class, new DebugImage.Deserializer());
    deserializersByClass.put(DebugMeta.class, new DebugMeta.Deserializer());
    deserializersByClass.put(Device.class, new Device.Deserializer());
    deserializersByClass.put(
        Device.DeviceOrientation.class, new Device.DeviceOrientation.Deserializer());
    deserializersByClass.put(Gpu.class, new Gpu.Deserializer());
    deserializersByClass.put(MeasurementValue.class, new MeasurementValue.Deserializer());
    deserializersByClass.put(Mechanism.class, new Mechanism.Deserializer());
    deserializersByClass.put(Message.class, new Message.Deserializer());
    deserializersByClass.put(MetricSummary.class, new MetricSummary.Deserializer());
    deserializersByClass.put(OperatingSystem.class, new OperatingSystem.Deserializer());
    deserializersByClass.put(ProfilingTraceData.class, new ProfilingTraceData.Deserializer());
    deserializersByClass.put(
        ProfilingTransactionData.class, new ProfilingTransactionData.Deserializer());
    deserializersByClass.put(ProfileMeasurement.class, new ProfileMeasurement.Deserializer());
    deserializersByClass.put(
        ProfileMeasurementValue.class, new ProfileMeasurementValue.Deserializer());
    deserializersByClass.put(Request.class, new Request.Deserializer());
    deserializersByClass.put(ReplayRecording.class, new ReplayRecording.Deserializer());
    deserializersByClass.put(RRWebBreadcrumbEvent.class, new RRWebBreadcrumbEvent.Deserializer());
    deserializersByClass.put(RRWebEventType.class, new RRWebEventType.Deserializer());
    deserializersByClass.put(RRWebInteractionEvent.class, new RRWebInteractionEvent.Deserializer());
    deserializersByClass.put(
        RRWebInteractionMoveEvent.class, new RRWebInteractionMoveEvent.Deserializer());
    deserializersByClass.put(RRWebMetaEvent.class, new RRWebMetaEvent.Deserializer());
    deserializersByClass.put(RRWebSpanEvent.class, new RRWebSpanEvent.Deserializer());
    deserializersByClass.put(RRWebVideoEvent.class, new RRWebVideoEvent.Deserializer());
    deserializersByClass.put(SdkInfo.class, new SdkInfo.Deserializer());
    deserializersByClass.put(SdkVersion.class, new SdkVersion.Deserializer());
    deserializersByClass.put(SentryEnvelopeHeader.class, new SentryEnvelopeHeader.Deserializer());
    deserializersByClass.put(
        SentryEnvelopeItemHeader.class, new SentryEnvelopeItemHeader.Deserializer());
    deserializersByClass.put(SentryEvent.class, new SentryEvent.Deserializer());
    deserializersByClass.put(SentryException.class, new SentryException.Deserializer());
    deserializersByClass.put(SentryItemType.class, new SentryItemType.Deserializer());
    deserializersByClass.put(SentryLevel.class, new SentryLevel.Deserializer());
    deserializersByClass.put(SentryLockReason.class, new SentryLockReason.Deserializer());
    deserializersByClass.put(SentryPackage.class, new SentryPackage.Deserializer());
    deserializersByClass.put(SentryRuntime.class, new SentryRuntime.Deserializer());
    deserializersByClass.put(SentryReplayEvent.class, new SentryReplayEvent.Deserializer());
    deserializersByClass.put(SentrySpan.class, new SentrySpan.Deserializer());
    deserializersByClass.put(SentryStackFrame.class, new SentryStackFrame.Deserializer());
    deserializersByClass.put(SentryStackTrace.class, new SentryStackTrace.Deserializer());
    deserializersByClass.put(
        SentryAppStartProfilingOptions.class, new SentryAppStartProfilingOptions.Deserializer());
    deserializersByClass.put(SentryThread.class, new SentryThread.Deserializer());
    deserializersByClass.put(SentryTransaction.class, new SentryTransaction.Deserializer());
    deserializersByClass.put(Session.class, new Session.Deserializer());
    deserializersByClass.put(SpanContext.class, new SpanContext.Deserializer());
    deserializersByClass.put(SpanId.class, new SpanId.Deserializer());
    deserializersByClass.put(SpanStatus.class, new SpanStatus.Deserializer());
    deserializersByClass.put(User.class, new User.Deserializer());
    deserializersByClass.put(Geo.class, new Geo.Deserializer());
    deserializersByClass.put(UserFeedback.class, new UserFeedback.Deserializer());
    deserializersByClass.put(ClientReport.class, new ClientReport.Deserializer());
    deserializersByClass.put(ViewHierarchyNode.class, new ViewHierarchyNode.Deserializer());
    deserializersByClass.put(ViewHierarchy.class, new ViewHierarchy.Deserializer());
  }

  // Deserialize

  @SuppressWarnings("unchecked")
  @Override
  public  @Nullable T deserializeCollection(
      @NotNull Reader reader,
      @NotNull Class clazz,
      @Nullable JsonDeserializer elementDeserializer) {
    try (JsonObjectReader jsonObjectReader = new JsonObjectReader(reader)) {
      if (Collection.class.isAssignableFrom(clazz)) {
        if (elementDeserializer == null) {
          // if the object has no known deserializer we do best effort and deserialize it as map
          return (T) jsonObjectReader.nextObjectOrNull();
        }

        return (T) jsonObjectReader.nextListOrNull(options.getLogger(), elementDeserializer);
      } else {
        return (T) jsonObjectReader.nextObjectOrNull();
      }
    } catch (Throwable e) {
      options.getLogger().log(SentryLevel.ERROR, "Error when deserializing", e);
      return null;
    }
  }

  @SuppressWarnings("unchecked")
  @Override
  public  @Nullable T deserialize(@NotNull Reader reader, @NotNull Class clazz) {
    try (JsonObjectReader jsonObjectReader = new JsonObjectReader(reader)) {
      JsonDeserializer deserializer = deserializersByClass.get(clazz);
      if (deserializer != null) {
        Object object = deserializer.deserialize(jsonObjectReader, options.getLogger());
        return clazz.cast(object);
      } else if (isKnownPrimitive(clazz)) {
        return (T) jsonObjectReader.nextObjectOrNull();
      } else {
        return null; // No way to deserialize objects we don't know about.
      }
    } catch (Exception e) {
      options.getLogger().log(SentryLevel.ERROR, "Error when deserializing", e);
      return null;
    }
  }

  @Override
  public @Nullable SentryEnvelope deserializeEnvelope(@NotNull InputStream inputStream) {
    Objects.requireNonNull(inputStream, "The InputStream object is required.");
    try {
      return options.getEnvelopeReader().read(inputStream);
    } catch (IOException e) {
      options.getLogger().log(SentryLevel.ERROR, "Error deserializing envelope.", e);
      return null;
    }
  }

  // Serialize

  @Override
  public  void serialize(@NotNull T entity, @NotNull Writer writer) throws IOException {
    Objects.requireNonNull(entity, "The entity is required.");
    Objects.requireNonNull(writer, "The Writer object is required.");

    if (options.getLogger().isEnabled(SentryLevel.DEBUG)) {
      String serialized = serializeToString(entity, options.isEnablePrettySerializationOutput());
      options.getLogger().log(SentryLevel.DEBUG, "Serializing object: %s", serialized);
    }
    JsonObjectWriter jsonObjectWriter = new JsonObjectWriter(writer, options.getMaxDepth());
    jsonObjectWriter.value(options.getLogger(), entity);
    writer.flush();
  }

  /**
   * Serializes an envelope to an OutputStream
   *
   * @param envelope the envelope
   * @param outputStream will not be closed automatically
   * @throws Exception an exception
   */
  @Override
  public void serialize(@NotNull SentryEnvelope envelope, @NotNull OutputStream outputStream)
      throws Exception {
    Objects.requireNonNull(envelope, "The SentryEnvelope object is required.");
    Objects.requireNonNull(outputStream, "The Stream object is required.");

    // we do not want to close these as we would also close the stream that was passed in
    final BufferedOutputStream bufferedOutputStream = new BufferedOutputStream(outputStream);
    final Writer writer = new BufferedWriter(new OutputStreamWriter(bufferedOutputStream, UTF_8));

    try {
      envelope
          .getHeader()
          .serialize(new JsonObjectWriter(writer, options.getMaxDepth()), options.getLogger());
      writer.write("\n");

      for (final SentryEnvelopeItem item : envelope.getItems()) {
        try {
          // When this throws we don't write anything and continue with the next item.
          final byte[] data = item.getData();

          item.getHeader()
              .serialize(new JsonObjectWriter(writer, options.getMaxDepth()), options.getLogger());
          writer.write("\n");
          writer.flush();

          outputStream.write(data);

          writer.write("\n");
        } catch (Exception exception) {
          options
              .getLogger()
              .log(SentryLevel.ERROR, "Failed to create envelope item. Dropping it.", exception);
        }
      }
    } finally {
      writer.flush();
    }
  }

  @Override
  public @NotNull String serialize(@NotNull Map data) throws Exception {
    return serializeToString(data, false);
  }

  // Helper

  private @NotNull String serializeToString(Object object, boolean pretty) throws IOException {
    StringWriter stringWriter = new StringWriter();
    JsonObjectWriter jsonObjectWriter = new JsonObjectWriter(stringWriter, options.getMaxDepth());
    if (pretty) {
      jsonObjectWriter.setIndent("\t");
    }
    jsonObjectWriter.value(options.getLogger(), object);
    return stringWriter.toString();
  }

  private  boolean isKnownPrimitive(final @NotNull Class clazz) {
    return clazz.isArray()
        || Collection.class.isAssignableFrom(clazz)
        || String.class.isAssignableFrom(clazz)
        || Map.class.isAssignableFrom(clazz);
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy