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

com.google.cloud.pubsub.v1.PubsubMessageWrapper Maven / Gradle / Ivy

The newest version!
/*
 * Copyright 2024 Google LLC
 *
 * 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 com.google.cloud.pubsub.v1;

import com.google.pubsub.v1.PubsubMessage;
import com.google.pubsub.v1.SubscriptionName;
import com.google.pubsub.v1.TopicName;
import io.opentelemetry.api.common.Attributes;
import io.opentelemetry.api.trace.Span;
import io.opentelemetry.api.trace.StatusCode;
import io.opentelemetry.api.trace.propagation.W3CTraceContextPropagator;
import io.opentelemetry.context.Context;
import io.opentelemetry.context.propagation.TextMapGetter;
import io.opentelemetry.context.propagation.TextMapSetter;
import io.opentelemetry.semconv.trace.attributes.SemanticAttributes;

/**
 * A wrapper class for a {@link PubsubMessage} object that handles creation and tracking of
 * OpenTelemetry {@link Span} objects for different operations that occur during publishing.
 */
public class PubsubMessageWrapper {
  private PubsubMessage message;

  private final TopicName topicName;
  private final SubscriptionName subscriptionName;

  // Attributes set only for messages received from a streaming pull response.
  private final String ackId;
  private final int deliveryAttempt;

  private static final String PUBLISH_START_EVENT = "publish start";
  private static final String PUBLISH_END_EVENT = "publish end";

  private static final String MODACK_START_EVENT = "modack start";
  private static final String MODACK_END_EVENT = "modack end";
  private static final String NACK_START_EVENT = "nack start";
  private static final String NACK_END_EVENT = "nack end";
  private static final String ACK_START_EVENT = "ack start";
  private static final String ACK_END_EVENT = "ack end";

  private static final String GOOGCLIENT_PREFIX = "googclient_";

  private static final String MESSAGE_RESULT_ATTR_KEY = "messaging.gcp_pubsub.result";

  private Span publisherSpan;
  private Span publishFlowControlSpan;
  private Span publishBatchingSpan;

  private Span subscriberSpan;
  private Span subscribeConcurrencyControlSpan;
  private Span subscribeSchedulerSpan;
  private Span subscribeProcessSpan;

  private PubsubMessageWrapper(Builder builder) {
    this.message = builder.message;
    this.topicName = builder.topicName;
    this.subscriptionName = builder.subscriptionName;
    this.ackId = builder.ackId;
    this.deliveryAttempt = builder.deliveryAttempt;
  }

  static Builder newBuilder(PubsubMessage message, String topicName) {
    return new Builder(message, topicName);
  }

  static Builder newBuilder(
      PubsubMessage message, String subscriptionName, String ackId, int deliveryAttempt) {
    return new Builder(message, subscriptionName, ackId, deliveryAttempt);
  }

  /** Returns the PubsubMessage associated with this wrapper. */
  PubsubMessage getPubsubMessage() {
    return message;
  }

  void setPubsubMessage(PubsubMessage message) {
    this.message = message;
  }

  /** Returns the TopicName for this wrapper as a string. */
  String getTopicName() {
    if (topicName != null) {
      return topicName.getTopic();
    }
    return "";
  }

  String getTopicProject() {
    if (topicName != null) {
      return topicName.getProject();
    }
    return "";
  }

  /** Returns the SubscriptionName for this wrapper as a string. */
  String getSubscriptionName() {
    if (subscriptionName != null) {
      return subscriptionName.getSubscription();
    }
    return "";
  }

  String getSubscriptionProject() {
    if (subscriptionName != null) {
      return subscriptionName.getProject();
    }
    return "";
  }

  String getMessageId() {
    return message.getMessageId();
  }

  String getAckId() {
    return ackId;
  }

  int getDataSize() {
    return message.getData().size();
  }

  String getOrderingKey() {
    return message.getOrderingKey();
  }

  int getDeliveryAttempt() {
    return deliveryAttempt;
  }

  Span getPublisherSpan() {
    return publisherSpan;
  }

  void setPublisherSpan(Span span) {
    this.publisherSpan = span;
  }

  void setPublishFlowControlSpan(Span span) {
    this.publishFlowControlSpan = span;
  }

  void setPublishBatchingSpan(Span span) {
    this.publishBatchingSpan = span;
  }

  Span getSubscriberSpan() {
    return subscriberSpan;
  }

  void setSubscriberSpan(Span span) {
    this.subscriberSpan = span;
  }

  void setSubscribeConcurrencyControlSpan(Span span) {
    this.subscribeConcurrencyControlSpan = span;
  }

  void setSubscribeSchedulerSpan(Span span) {
    this.subscribeSchedulerSpan = span;
  }

  void setSubscribeProcessSpan(Span span) {
    this.subscribeProcessSpan = span;
  }

  /** Creates a publish start event that is tied to the publish RPC span time. */
  void addPublishStartEvent() {
    if (publisherSpan != null) {
      publisherSpan.addEvent(PUBLISH_START_EVENT);
    }
  }

  /**
   * Sets the message ID attribute in the publisher parent span. This is called after the publish
   * RPC returns with a message ID.
   */
  void setPublisherMessageIdSpanAttribute(String messageId) {
    if (publisherSpan != null) {
      publisherSpan.setAttribute(SemanticAttributes.MESSAGING_MESSAGE_ID, messageId);
    }
  }

  /** Ends the publisher parent span if it exists. */
  void endPublisherSpan() {
    if (publisherSpan != null) {
      publisherSpan.addEvent(PUBLISH_END_EVENT);
      publisherSpan.end();
    }
  }

  /** Ends the publish flow control span if it exists. */
  void endPublishFlowControlSpan() {
    if (publishFlowControlSpan != null) {
      publishFlowControlSpan.end();
    }
  }

  /** Ends the publish batching span if it exists. */
  void endPublishBatchingSpan() {
    if (publishBatchingSpan != null) {
      publishBatchingSpan.end();
    }
  }

  /**
   * Sets an error status and records an exception when an exception is thrown during flow control.
   */
  void setPublishFlowControlSpanException(Throwable t) {
    if (publishFlowControlSpan != null) {
      publishFlowControlSpan.setStatus(
          StatusCode.ERROR, "Exception thrown during publish flow control.");
      publishFlowControlSpan.recordException(t);
      endAllPublishSpans();
    }
  }

  /**
   * Creates start and end events for ModAcks, Nacks, and Acks that are tied to the corresponding
   * RPC span start and end times.
   */
  void addModAckStartEvent() {
    if (subscriberSpan != null) {
      subscriberSpan.addEvent(MODACK_START_EVENT);
    }
  }

  void addModAckEndEvent() {
    if (subscriberSpan != null) {
      subscriberSpan.addEvent(MODACK_END_EVENT);
    }
  }

  void addNackStartEvent() {
    if (subscriberSpan != null) {
      subscriberSpan.addEvent(NACK_START_EVENT);
    }
  }

  void addNackEndEvent() {
    if (subscriberSpan != null) {
      subscriberSpan.addEvent(NACK_END_EVENT);
    }
  }

  void addAckStartEvent() {
    if (subscriberSpan != null) {
      subscriberSpan.addEvent(ACK_START_EVENT);
    }
  }

  void addAckEndEvent() {
    if (subscriberSpan != null) {
      subscriberSpan.addEvent(ACK_END_EVENT);
    }
  }

  /** Ends the subscriber parent span if exists. */
  void endSubscriberSpan() {
    if (subscriberSpan != null) {
      subscriberSpan.end();
    }
  }

  /** Ends the subscribe concurreny control span if exists. */
  void endSubscribeConcurrencyControlSpan() {
    if (subscribeConcurrencyControlSpan != null) {
      subscribeConcurrencyControlSpan.end();
    }
  }

  /** Ends the subscribe scheduler span if exists. */
  void endSubscribeSchedulerSpan() {
    if (subscribeSchedulerSpan != null) {
      subscribeSchedulerSpan.end();
    }
  }

  /**
   * Ends the subscribe process span if it exists, creates an event with the appropriate result, and
   * sets the result on the parent subscriber span.
   */
  void endSubscribeProcessSpan(String action) {
    if (subscribeProcessSpan != null) {
      subscribeProcessSpan.addEvent(action + " called");
      subscribeProcessSpan.end();
      subscriberSpan.setAttribute(MESSAGE_RESULT_ATTR_KEY, action);
    }
  }

  /** Sets an exception on the subscriber span during Ack/ModAck/Nack failures */
  void setSubscriberSpanException(Throwable t, String exception) {
    if (subscriberSpan != null) {
      subscriberSpan.setStatus(StatusCode.ERROR, exception);
      subscriberSpan.recordException(t);
      endAllSubscribeSpans();
    }
  }

  /** Sets result of the parent subscriber span to expired and ends its. */
  void setSubscriberSpanExpirationResult() {
    if (subscriberSpan != null) {
      subscriberSpan.setAttribute(MESSAGE_RESULT_ATTR_KEY, "expired");
      endSubscriberSpan();
    }
  }

  /**
   * Sets an error status and records an exception when an exception is thrown subscriber
   * concurrency control.
   */
  void setSubscribeConcurrencyControlSpanException(Throwable t) {
    if (subscribeConcurrencyControlSpan != null) {
      subscribeConcurrencyControlSpan.setStatus(
          StatusCode.ERROR, "Exception thrown during subscribe concurrency control.");
      subscribeConcurrencyControlSpan.recordException(t);
      endAllSubscribeSpans();
    }
  }

  /** Ends all publisher-side spans associated with this message wrapper. */
  private void endAllPublishSpans() {
    endPublishFlowControlSpan();
    endPublishBatchingSpan();
    endPublisherSpan();
  }

  /** Ends all subscriber-side spans associated with this message wrapper. */
  private void endAllSubscribeSpans() {
    endSubscribeConcurrencyControlSpan();
    endSubscribeSchedulerSpan();
    endSubscriberSpan();
  }

  /**
   * Injects the span context into the attributes of a Pub/Sub message for propagation to the
   * subscriber client.
   */
  void injectSpanContext() {
    TextMapSetter injectMessageAttributes =
        new TextMapSetter() {
          @Override
          public void set(PubsubMessageWrapper carrier, String key, String value) {
            PubsubMessage newMessage =
                PubsubMessage.newBuilder(carrier.message)
                    .putAttributes(GOOGCLIENT_PREFIX + key, value)
                    .build();
            carrier.message = newMessage;
          }
        };
    W3CTraceContextPropagator.getInstance()
        .inject(Context.current().with(publisherSpan), this, injectMessageAttributes);
  }

  /**
   * Extracts the span context from the attributes of a Pub/Sub message and creates the parent
   * subscriber span using that context.
   */
  Context extractSpanContext(Attributes attributes) {
    TextMapGetter extractMessageAttributes =
        new TextMapGetter() {
          @Override
          public String get(PubsubMessageWrapper carrier, String key) {
            return carrier.message.getAttributesOrDefault(GOOGCLIENT_PREFIX + key, "");
          }

          public Iterable keys(PubsubMessageWrapper carrier) {
            return carrier.message.getAttributesMap().keySet();
          }
        };
    Context context =
        W3CTraceContextPropagator.getInstance()
            .extract(Context.current(), this, extractMessageAttributes);
    return context;
  }

  /** Builder of {@link PubsubMessageWrapper PubsubMessageWrapper}. */
  static final class Builder {
    private PubsubMessage message = null;
    private TopicName topicName = null;
    private SubscriptionName subscriptionName = null;
    private String ackId = null;
    private int deliveryAttempt = 0;

    public Builder(PubsubMessage message, String topicName) {
      this.message = message;
      if (topicName != null) {
        this.topicName = TopicName.parse(topicName);
      }
    }

    public Builder(
        PubsubMessage message, String subscriptionName, String ackId, int deliveryAttempt) {
      this.message = message;
      if (subscriptionName != null) {
        this.subscriptionName = SubscriptionName.parse(subscriptionName);
      }
      this.ackId = ackId;
      this.deliveryAttempt = deliveryAttempt;
    }

    public Builder(
        PubsubMessage message,
        SubscriptionName subscriptionName,
        String ackId,
        int deliveryAttempt) {
      this.message = message;
      this.subscriptionName = subscriptionName;
      this.ackId = ackId;
      this.deliveryAttempt = deliveryAttempt;
    }

    public PubsubMessageWrapper build() {
      return new PubsubMessageWrapper(this);
    }
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy