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

com.google.cloud.pubsub.v1.OpenTelemetryPubsubTracer 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.common.AttributesBuilder;
import io.opentelemetry.api.trace.Span;
import io.opentelemetry.api.trace.SpanBuilder;
import io.opentelemetry.api.trace.SpanKind;
import io.opentelemetry.api.trace.StatusCode;
import io.opentelemetry.api.trace.Tracer;
import io.opentelemetry.context.Context;
import io.opentelemetry.semconv.trace.attributes.SemanticAttributes;
import java.util.List;

public class OpenTelemetryPubsubTracer {
  private final Tracer tracer;
  private boolean enabled = false;

  private static final String PUBLISH_FLOW_CONTROL_SPAN_NAME = "publisher flow control";
  private static final String PUBLISH_BATCHING_SPAN_NAME = "publisher batching";
  private static final String SUBSCRIBE_CONCURRENCY_CONTROL_SPAN_NAME =
      "subscriber concurrency control";
  private static final String SUBSCRIBE_SCHEDULER_SPAN_NAME = "subscriber scheduler";

  private static final String MESSAGE_SIZE_ATTR_KEY = "messaging.message.body.size";
  private static final String ORDERING_KEY_ATTR_KEY = "messaging.gcp_pubsub.message.ordering_key";
  private static final String MESSAGE_ACK_ID_ATTR_KEY = "messaging.gcp_pubsub.message.ack_id";
  private static final String MESSAGE_EXACTLY_ONCE_ATTR_KEY =
      "messaging.gcp_pubsub.message.exactly_once_delivery";
  private static final String MESSAGE_DELIVERY_ATTEMPT_ATTR_KEY =
      "messaging.gcp_pubsub.message.delivery_attempt";
  private static final String ACK_DEADLINE_ATTR_KEY = "messaging.gcp_pubsub.message.ack_deadline";
  private static final String RECEIPT_MODACK_ATTR_KEY = "messaging.gcp_pubsub.is_receipt_modack";
  private static final String PROJECT_ATTR_KEY = "gcp.project_id";
  private static final String PUBLISH_RPC_SPAN_SUFFIX = " publish";

  private static final String MESSAGING_SYSTEM_VALUE = "gcp_pubsub";

  OpenTelemetryPubsubTracer(Tracer tracer, boolean enableOpenTelemetry) {
    this.tracer = tracer;
    if (this.tracer != null && enableOpenTelemetry) {
      this.enabled = true;
    }
  }

  /** Populates attributes that are common the publisher parent span and publish RPC span. */
  private static final AttributesBuilder createCommonSpanAttributesBuilder(
      String destinationName, String projectName, String codeFunction, String operation) {
    AttributesBuilder attributesBuilder =
        Attributes.builder()
            .put(SemanticAttributes.MESSAGING_SYSTEM, MESSAGING_SYSTEM_VALUE)
            .put(SemanticAttributes.MESSAGING_DESTINATION_NAME, destinationName)
            .put(PROJECT_ATTR_KEY, projectName)
            .put(SemanticAttributes.CODE_FUNCTION, codeFunction);
    if (operation != null) {
      attributesBuilder.put(SemanticAttributes.MESSAGING_OPERATION, operation);
    }

    return attributesBuilder;
  }

  private Span startChildSpan(String name, Span parent) {
    return tracer.spanBuilder(name).setParent(Context.current().with(parent)).startSpan();
  }

  /**
   * Creates and starts the parent span with the appropriate span attributes and injects the span
   * context into the {@link PubsubMessage} attributes.
   */
  void startPublisherSpan(PubsubMessageWrapper message) {
    if (!enabled) {
      return;
    }
    AttributesBuilder attributesBuilder =
        createCommonSpanAttributesBuilder(
            message.getTopicName(), message.getTopicProject(), "publish", "create");

    attributesBuilder.put(MESSAGE_SIZE_ATTR_KEY, message.getDataSize());
    if (!message.getOrderingKey().isEmpty()) {
      attributesBuilder.put(ORDERING_KEY_ATTR_KEY, message.getOrderingKey());
    }

    Span publisherSpan =
        tracer
            .spanBuilder(message.getTopicName() + " create")
            .setSpanKind(SpanKind.PRODUCER)
            .setAllAttributes(attributesBuilder.build())
            .startSpan();

    message.setPublisherSpan(publisherSpan);
    if (publisherSpan.getSpanContext().isValid()) {
      message.injectSpanContext();
    }
  }

  void endPublisherSpan(PubsubMessageWrapper message) {
    if (!enabled) {
      return;
    }
    message.endPublisherSpan();
  }

  void setPublisherMessageIdSpanAttribute(PubsubMessageWrapper message, String messageId) {
    if (!enabled) {
      return;
    }
    message.setPublisherMessageIdSpanAttribute(messageId);
  }

  /** Creates a span for publish-side flow control as a child of the parent publisher span. */
  void startPublishFlowControlSpan(PubsubMessageWrapper message) {
    if (!enabled) {
      return;
    }
    Span publisherSpan = message.getPublisherSpan();
    if (publisherSpan != null)
      message.setPublishFlowControlSpan(
          startChildSpan(PUBLISH_FLOW_CONTROL_SPAN_NAME, publisherSpan));
  }

  void endPublishFlowControlSpan(PubsubMessageWrapper message) {
    if (!enabled) {
      return;
    }
    message.endPublishFlowControlSpan();
  }

  void setPublishFlowControlSpanException(PubsubMessageWrapper message, Throwable t) {
    if (!enabled) {
      return;
    }
    message.setPublishFlowControlSpanException(t);
  }

  /** Creates a span for publish message batching as a child of the parent publisher span. */
  void startPublishBatchingSpan(PubsubMessageWrapper message) {
    if (!enabled) {
      return;
    }
    Span publisherSpan = message.getPublisherSpan();
    if (publisherSpan != null) {
      message.setPublishBatchingSpan(startChildSpan(PUBLISH_BATCHING_SPAN_NAME, publisherSpan));
    }
  }

  void endPublishBatchingSpan(PubsubMessageWrapper message) {
    if (!enabled) {
      return;
    }
    message.endPublishBatchingSpan();
  }

  /**
   * Creates, starts, and returns a publish RPC span for the given message batch. Bi-directional
   * links with the publisher parent span are created for sampled messages in the batch.
   */
  Span startPublishRpcSpan(String topic, List messages) {
    if (!enabled) {
      return null;
    }
    TopicName topicName = TopicName.parse(topic);
    Attributes attributes =
        createCommonSpanAttributesBuilder(
                topicName.getTopic(), topicName.getProject(), "publishCall", "publish")
            .put(SemanticAttributes.MESSAGING_BATCH_MESSAGE_COUNT, messages.size())
            .build();
    SpanBuilder publishRpcSpanBuilder =
        tracer
            .spanBuilder(topicName.getTopic() + PUBLISH_RPC_SPAN_SUFFIX)
            .setSpanKind(SpanKind.CLIENT)
            .setAllAttributes(attributes);
    Attributes linkAttributes =
        Attributes.builder().put(SemanticAttributes.MESSAGING_OPERATION, "publish").build();
    for (PubsubMessageWrapper message : messages) {
      if (message.getPublisherSpan().getSpanContext().isSampled())
        publishRpcSpanBuilder.addLink(message.getPublisherSpan().getSpanContext(), linkAttributes);
    }
    Span publishRpcSpan = publishRpcSpanBuilder.startSpan();

    for (PubsubMessageWrapper message : messages) {
      if (publishRpcSpan.getSpanContext().isSampled()) {
        message.getPublisherSpan().addLink(publishRpcSpan.getSpanContext(), linkAttributes);
        message.addPublishStartEvent();
      }
    }
    return publishRpcSpan;
  }

  /** Ends the given publish RPC span if it exists. */
  void endPublishRpcSpan(Span publishRpcSpan) {
    if (!enabled) {
      return;
    }
    if (publishRpcSpan != null) {
      publishRpcSpan.end();
    }
  }

  /**
   * Sets an error status and records an exception when an exception is thrown when publishing the
   * message batch.
   */
  void setPublishRpcSpanException(Span publishRpcSpan, Throwable t) {
    if (!enabled) {
      return;
    }
    if (publishRpcSpan != null) {
      publishRpcSpan.setStatus(StatusCode.ERROR, "Exception thrown on publish RPC.");
      publishRpcSpan.recordException(t);
      publishRpcSpan.end();
    }
  }

  void startSubscriberSpan(PubsubMessageWrapper message, boolean exactlyOnceDeliveryEnabled) {
    if (!enabled) {
      return;
    }
    AttributesBuilder attributesBuilder =
        createCommonSpanAttributesBuilder(
            message.getSubscriptionName(), message.getSubscriptionProject(), "onResponse", null);

    attributesBuilder
        .put(SemanticAttributes.MESSAGING_MESSAGE_ID, message.getMessageId())
        .put(MESSAGE_SIZE_ATTR_KEY, message.getDataSize())
        .put(MESSAGE_ACK_ID_ATTR_KEY, message.getAckId())
        .put(MESSAGE_EXACTLY_ONCE_ATTR_KEY, exactlyOnceDeliveryEnabled);
    if (!message.getOrderingKey().isEmpty()) {
      attributesBuilder.put(ORDERING_KEY_ATTR_KEY, message.getOrderingKey());
    }
    if (message.getDeliveryAttempt() > 0) {
      attributesBuilder.put(MESSAGE_DELIVERY_ATTEMPT_ATTR_KEY, message.getDeliveryAttempt());
    }
    Attributes attributes = attributesBuilder.build();
    Context publisherSpanContext = message.extractSpanContext(attributes);
    message.setPublisherSpan(Span.fromContextOrNull(publisherSpanContext));
    message.setSubscriberSpan(
        tracer
            .spanBuilder(message.getSubscriptionName() + " subscribe")
            .setSpanKind(SpanKind.CONSUMER)
            .setParent(publisherSpanContext)
            .setAllAttributes(attributes)
            .startSpan());
  }

  void endSubscriberSpan(PubsubMessageWrapper message) {
    if (!enabled) {
      return;
    }
    message.endSubscriberSpan();
  }

  void setSubscriberSpanExpirationResult(PubsubMessageWrapper message) {
    if (!enabled) {
      return;
    }
    message.setSubscriberSpanExpirationResult();
  }

  void setSubscriberSpanException(PubsubMessageWrapper message, Throwable t, String exception) {
    if (!enabled) {
      return;
    }
    message.setSubscriberSpanException(t, exception);
  }

  /** Creates a span for subscribe concurrency control as a child of the parent subscriber span. */
  void startSubscribeConcurrencyControlSpan(PubsubMessageWrapper message) {
    if (!enabled) {
      return;
    }
    Span subscriberSpan = message.getSubscriberSpan();
    if (subscriberSpan != null) {
      message.setSubscribeConcurrencyControlSpan(
          startChildSpan(SUBSCRIBE_CONCURRENCY_CONTROL_SPAN_NAME, subscriberSpan));
    }
  }

  void endSubscribeConcurrencyControlSpan(PubsubMessageWrapper message) {
    if (!enabled) {
      return;
    }
    message.endSubscribeConcurrencyControlSpan();
  }

  void setSubscribeConcurrencyControlSpanException(PubsubMessageWrapper message, Throwable t) {
    if (!enabled) {
      return;
    }
    message.setSubscribeConcurrencyControlSpanException(t);
  }

  /**
   * Creates a span for subscribe ordering key scheduling as a child of the parent subscriber span.
   */
  void startSubscribeSchedulerSpan(PubsubMessageWrapper message) {
    if (!enabled) {
      return;
    }
    Span subscriberSpan = message.getSubscriberSpan();
    if (subscriberSpan != null) {
      message.setSubscribeSchedulerSpan(
          startChildSpan(SUBSCRIBE_SCHEDULER_SPAN_NAME, subscriberSpan));
    }
  }

  void endSubscribeSchedulerSpan(PubsubMessageWrapper message) {
    if (!enabled) {
      return;
    }
    message.endSubscribeSchedulerSpan();
  }

  /** Creates a span for subscribe message processing as a child of the parent subscriber span. */
  void startSubscribeProcessSpan(PubsubMessageWrapper message) {
    if (!enabled) {
      return;
    }
    Span subscriberSpan = message.getSubscriberSpan();
    if (subscriberSpan != null) {
      Span subscribeProcessSpan =
          startChildSpan(message.getSubscriptionName() + " process", subscriberSpan);
      subscribeProcessSpan.setAttribute(
          SemanticAttributes.MESSAGING_SYSTEM, MESSAGING_SYSTEM_VALUE);
      Span publisherSpan = message.getPublisherSpan();
      if (publisherSpan != null) {
        subscribeProcessSpan.addLink(publisherSpan.getSpanContext());
      }
      message.setSubscribeProcessSpan(subscribeProcessSpan);
    }
  }

  void endSubscribeProcessSpan(PubsubMessageWrapper message, String action) {
    if (!enabled) {
      return;
    }
    message.endSubscribeProcessSpan(action);
  }

  /**
   * Creates, starts, and returns spans for ModAck, Nack, and Ack RPC requests. Bi-directional links
   * to parent subscribe span for sampled messages are added.
   */
  Span startSubscribeRpcSpan(
      String subscription,
      String rpcOperation,
      List messages,
      int ackDeadline,
      boolean isReceiptModack) {
    if (!enabled) {
      return null;
    }
    String codeFunction = rpcOperation == "ack" ? "sendAckOperations" : "sendModAckOperations";
    SubscriptionName subscriptionName = SubscriptionName.parse(subscription);
    AttributesBuilder attributesBuilder =
        createCommonSpanAttributesBuilder(
                subscriptionName.getSubscription(),
                subscriptionName.getProject(),
                codeFunction,
                rpcOperation)
            .put(SemanticAttributes.MESSAGING_BATCH_MESSAGE_COUNT, messages.size());

    // Ack deadline and receipt modack are specific to the modack operation
    if (rpcOperation == "modack") {
      attributesBuilder
          .put(ACK_DEADLINE_ATTR_KEY, ackDeadline)
          .put(RECEIPT_MODACK_ATTR_KEY, isReceiptModack);
    }

    SpanBuilder rpcSpanBuilder =
        tracer
            .spanBuilder(subscriptionName.getSubscription() + " " + rpcOperation)
            .setSpanKind(SpanKind.CLIENT)
            .setAllAttributes(attributesBuilder.build());
    Attributes linkAttributes =
        Attributes.builder().put(SemanticAttributes.MESSAGING_OPERATION, rpcOperation).build();
    for (PubsubMessageWrapper message : messages) {
      if (message.getSubscriberSpan().getSpanContext().isSampled()) {
        rpcSpanBuilder.addLink(message.getSubscriberSpan().getSpanContext(), linkAttributes);
      }
    }
    Span rpcSpan = rpcSpanBuilder.startSpan();

    for (PubsubMessageWrapper message : messages) {
      if (rpcSpan.getSpanContext().isSampled()) {
        message.getSubscriberSpan().addLink(rpcSpan.getSpanContext(), linkAttributes);
        switch (rpcOperation) {
          case "ack":
            message.addAckStartEvent();
            break;
          case "modack":
            message.addModAckStartEvent();
            break;
          case "nack":
            message.addNackStartEvent();
            break;
        }
      }
    }
    return rpcSpan;
  }

  /** Ends the given subscribe RPC span if it exists. */
  void endSubscribeRpcSpan(Span rpcSpan) {
    if (!enabled) {
      return;
    }
    if (rpcSpan != null) {
      rpcSpan.end();
    }
  }

  /**
   * Sets an error status and records an exception when an exception is thrown when handling a
   * subscribe-side RPC.
   */
  void setSubscribeRpcSpanException(Span rpcSpan, boolean isModack, int ackDeadline, Throwable t) {
    if (!enabled) {
      return;
    }
    if (rpcSpan != null) {
      String operation = !isModack ? "ack" : (ackDeadline == 0 ? "nack" : "modack");
      rpcSpan.setStatus(StatusCode.ERROR, "Exception thrown on " + operation + " RPC.");
      rpcSpan.recordException(t);
      rpcSpan.end();
    }
  }

  /** Adds the appropriate subscribe-side RPC end event. */
  void addEndRpcEvent(
      PubsubMessageWrapper message, boolean rpcSampled, boolean isModack, int ackDeadline) {
    if (!enabled || !rpcSampled) {
      return;
    }
    if (!isModack) {
      message.addAckEndEvent();
    } else if (ackDeadline == 0) {
      message.addNackEndEvent();
    } else {
      message.addModAckEndEvent();
    }
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy