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

org.apache.pulsar.client.impl.conf.ConsumerConfigurationData Maven / Gradle / Ivy

The newest version!
/*
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements.  See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership.  The ASF licenses this file
 * to you 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 org.apache.pulsar.client.impl.conf;

import static org.apache.pulsar.shade.com.google.common.base.Preconditions.checkArgument;
import com.fasterxml.jackson.annotation.JsonIgnore;
import org.apache.pulsar.shade.com.google.common.collect.Sets;
import io.swagger.annotations.ApiModelProperty;
import java.io.Serializable;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.SortedMap;
import java.util.TreeMap;
import java.util.TreeSet;
import java.util.concurrent.TimeUnit;
import java.util.regex.Pattern;
import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;
import org.apache.pulsar.client.api.BatchReceivePolicy;
import org.apache.pulsar.client.api.ConsumerCryptoFailureAction;
import org.apache.pulsar.client.api.ConsumerEventListener;
import org.apache.pulsar.client.api.CryptoKeyReader;
import org.apache.pulsar.client.api.DeadLetterPolicy;
import org.apache.pulsar.client.api.KeySharedPolicy;
import org.apache.pulsar.client.api.MessageCrypto;
import org.apache.pulsar.client.api.MessageListener;
import org.apache.pulsar.client.api.MessageListenerExecutor;
import org.apache.pulsar.client.api.MessagePayloadProcessor;
import org.apache.pulsar.client.api.RedeliveryBackoff;
import org.apache.pulsar.client.api.RegexSubscriptionMode;
import org.apache.pulsar.client.api.SubscriptionInitialPosition;
import org.apache.pulsar.client.api.SubscriptionMode;
import org.apache.pulsar.client.api.SubscriptionType;

@Data
@NoArgsConstructor
@AllArgsConstructor
public class ConsumerConfigurationData implements Serializable, Cloneable {
    private static final long serialVersionUID = 1L;

    @ApiModelProperty(
            name = "topicNames",
            required = true,
            value = "Topic name"
    )
    private Set topicNames = new TreeSet<>();

    @ApiModelProperty(
            name = "topicsPattern",
            value = "The regexp for the topic name(not contains partition suffix)."
    )
    private Pattern topicsPattern;

    @ApiModelProperty(
            name = "subscriptionName",
            value = "Subscription name"
    )
    private String subscriptionName;

    @ApiModelProperty(
            name = "subscriptionType",
            value = "Subscription type.\n"
                    + "Four subscription types are available:\n"
                    + "* Exclusive\n"
                    + "* Failover\n"
                    + "* Shared\n"
                    + "* Key_Shared"
    )
    private SubscriptionType subscriptionType = SubscriptionType.Exclusive;

    private Map subscriptionProperties;

    private SubscriptionMode subscriptionMode = SubscriptionMode.Durable;

    @JsonIgnore
    private transient MessageListenerExecutor messageListenerExecutor;
    @JsonIgnore
    private MessageListener messageListener;

    @JsonIgnore
    private ConsumerEventListener consumerEventListener;

    @ApiModelProperty(
            name = "negativeAckRedeliveryBackoff",
            value = "Interface for custom message is negativeAcked policy. You can specify `RedeliveryBackoff` for a"
                    + " consumer."
    )
    @JsonIgnore
    private RedeliveryBackoff negativeAckRedeliveryBackoff;

    @ApiModelProperty(
            name = "ackTimeoutRedeliveryBackoff",
            value = "Interface for custom message is ackTimeout policy. You can specify `RedeliveryBackoff` for a"
                    + " consumer."
    )
    @JsonIgnore
    private RedeliveryBackoff ackTimeoutRedeliveryBackoff;

    @ApiModelProperty(
            name = "receiverQueueSize",
            value = "Size of a consumer's receiver queue.\n"
                    + "\n"
                    + "For example, the number of messages accumulated by a consumer before an application calls "
                    + "`Receive`.\n"
                    + "\n"
                    + "A value higher than the default value increases consumer throughput, though at the expense of "
                    + "more memory utilization."
    )
    private int receiverQueueSize = 1000;

    @ApiModelProperty(
            name = "acknowledgementsGroupTimeMicros",
            value = "Group a consumer acknowledgment for a specified time.\n"
                    + "\n"
                    + "By default, a consumer uses 100ms grouping time to send out acknowledgments to a broker.\n"
                    + "\n"
                    + "Setting a group time of 0 sends out acknowledgments immediately.\n"
                    + "\n"
                    + "A longer ack group time is more efficient at the expense of a slight increase in message "
                    + "re-deliveries after a failure."
    )
    private long acknowledgementsGroupTimeMicros = TimeUnit.MILLISECONDS.toMicros(100);

    @ApiModelProperty(
            name = "maxAcknowledgmentGroupSize",
            value = "Group a consumer acknowledgment for the number of messages."
    )
    private int maxAcknowledgmentGroupSize = 1000;

    @ApiModelProperty(
            name = "negativeAckRedeliveryDelayMicros",
            value = "Delay to wait before redelivering messages that failed to be processed.\n"
                    + "\n"
                    + "When an application uses {@link Consumer#negativeAcknowledge(Message)}, failed messages are "
                    + "redelivered after a fixed timeout."
    )
    private long negativeAckRedeliveryDelayMicros = TimeUnit.MINUTES.toMicros(1);

    @ApiModelProperty(
            name = "maxTotalReceiverQueueSizeAcrossPartitions",
            value = "The max total receiver queue size across partitions.\n"
                    + "\n"
                    + "This setting reduces the receiver queue size for individual partitions if the total receiver "
                    + "queue size exceeds this value."
    )
    private int maxTotalReceiverQueueSizeAcrossPartitions = 50000;

    @ApiModelProperty(
            name = "consumerName",
            value = "Consumer name"
    )
    private String consumerName = null;

    @ApiModelProperty(
            name = "ackTimeoutMillis",
            value = "Timeout of unacked messages"
    )
    private long ackTimeoutMillis = 0;

    @ApiModelProperty(
            name = "tickDurationMillis",
            value = "Granularity of the ack-timeout redelivery.\n"
                    + "\n"
                    + "Using an higher `tickDurationMillis` reduces the memory overhead to track messages when setting "
                    + "ack-timeout to a bigger value (for example, 1 hour)."
    )
    private long tickDurationMillis = 1000;

    @ApiModelProperty(
            name = "priorityLevel",
            value = "Priority level for a consumer to which a broker gives more priority while dispatching messages "
                    + "in Shared subscription type.\n"
                    + "\n"
                    + "The broker follows descending priorities. For example, 0=max-priority, 1, 2,...\n"
                    + "\n"
                    + "In Shared subscription type, the broker **first dispatches messages to the max priority level "
                    + "consumers if they have permits**. Otherwise, the broker considers next priority level consumers."
                    + "\n\n"
                    + "**Example 1**\n"
                    + "If a subscription has consumerA with `priorityLevel` 0 and consumerB with `priorityLevel` 1,"
                    + " then the broker **only dispatches messages to consumerA until it runs out permits** and then"
                    + " starts dispatching messages to consumerB.\n"
                    + "\n"
                    + "**Example 2**\n"
                    + "Consumer Priority, Level, Permits\n"
                    + "C1, 0, 2\n"
                    + "C2, 0, 1\n"
                    + "C3, 0, 1\n"
                    + "C4, 1, 2\n"
                    + "C5, 1, 1\n"
                    + "\n"
                    + "Order in which a broker dispatches messages to consumers is: C1, C2, C3, C1, C4, C5, C4."
    )
    private int priorityLevel = 0;

    /**
     * @deprecated use {@link #setMaxPendingChunkedMessage(int)}
     */
    @Deprecated
    public void setMaxPendingChuckedMessage(int maxPendingChuckedMessage) {
        this.maxPendingChunkedMessage = maxPendingChuckedMessage;
    }

    /**
     * @deprecated use {@link #getMaxPendingChunkedMessage()}
     */
    @Deprecated
    public int getMaxPendingChuckedMessage() {
        return maxPendingChunkedMessage;
    }

    @ApiModelProperty(
            name = "maxPendingChunkedMessage",
            value = "The maximum size of a queue holding pending chunked messages. When the threshold is reached,"
                    + " the consumer drops pending messages to optimize memory utilization."
    )
    // max pending chunked message to avoid sending incomplete message into the queue and memory
    private int maxPendingChunkedMessage = 10;

    @ApiModelProperty(
            name = "autoAckOldestChunkedMessageOnQueueFull",
            value = "Whether to automatically acknowledge pending chunked messages when the threshold of"
                    + " `maxPendingChunkedMessage` is reached. If set to `false`, these messages will be redelivered"
                    + " by their broker."
    )
    private boolean autoAckOldestChunkedMessageOnQueueFull = false;

    @ApiModelProperty(
            name = "expireTimeOfIncompleteChunkedMessageMillis",
            value = "The time interval to expire incomplete chunks if a consumer fails to receive all the chunks in the"
                    + " specified time period. The default value is 1 minute."
    )
    private long expireTimeOfIncompleteChunkedMessageMillis = TimeUnit.MINUTES.toMillis(1);

    @JsonIgnore
    private CryptoKeyReader cryptoKeyReader = null;

    @JsonIgnore
    private transient MessageCrypto messageCrypto = null;

    @ApiModelProperty(
            name = "cryptoFailureAction",
            value = "Consumer should take action when it receives a message that can not be decrypted.\n"
                    + "* **FAIL**: this is the default option to fail messages until crypto succeeds.\n"
                    + "* **DISCARD**:silently acknowledge and not deliver message to an application.\n"
                    + "* **CONSUME**: deliver encrypted messages to applications. It is the application's"
                    + " responsibility to decrypt the message.\n"
                    + "\n"
                    + "The decompression of message fails.\n"
                    + "\n"
                    + "If messages contain batch messages, a client is not be able to retrieve individual messages in"
                    + " batch.\n"
                    + "\n"
                    + "Delivered encrypted message contains {@link EncryptionContext} which contains encryption and "
                    + "compression information in it using which application can decrypt consumed message payload."
    )
    private ConsumerCryptoFailureAction cryptoFailureAction = ConsumerCryptoFailureAction.FAIL;

    @ApiModelProperty(
            name = "properties",
            value = "A name or value property of this consumer.\n"
                    + "\n"
                    + "`properties` is application defined metadata attached to a consumer.\n"
                    + "\n"
                    + "When getting a topic stats, associate this metadata with the consumer stats for easier "
                    + "identification."
    )
    private SortedMap properties = new TreeMap<>();

    @ApiModelProperty(
            name = "readCompacted",
            value = "If enabling `readCompacted`, a consumer reads messages from a compacted topic rather than reading "
                    + "a full message backlog of a topic.\n"
                    + "\n"
                    + "A consumer only sees the latest value for each key in the compacted topic, up until reaching "
                    + "the point in the topic message when compacting backlog. Beyond that point, send messages as "
                    + "normal.\n"
                    + "\n"
                    + "Only enabling `readCompacted` on subscriptions to persistent topics, which have a single active "
                    + "consumer (like failure or exclusive subscriptions).\n"
                    + "\n"
                    + "Attempting to enable it on subscriptions to non-persistent topics or on shared subscriptions "
                    + "leads to a subscription call throwing a `PulsarClientException`."
    )
    private boolean readCompacted = false;

    @ApiModelProperty(
            name = "subscriptionInitialPosition",
            value = "Initial position at which to set cursor when subscribing to a topic at first time."
    )
    private SubscriptionInitialPosition subscriptionInitialPosition = SubscriptionInitialPosition.Latest;

    @ApiModelProperty(
            name = "patternAutoDiscoveryPeriod",
            value = "Topic auto discovery period when using a pattern for topic's consumer.\n"
                    + "\n"
                    + "The default and minimum value is 1 minute."
    )
    private int patternAutoDiscoveryPeriod = 60;

    @ApiModelProperty(
            name = "regexSubscriptionMode",
            value = "When subscribing to a topic using a regular expression, you can pick a certain type of topics.\n"
                    + "\n"
                    + "* **PersistentOnly**: only subscribe to persistent topics.\n"
                    + "* **NonPersistentOnly**: only subscribe to non-persistent topics.\n"
                    + "* **AllTopics**: subscribe to both persistent and non-persistent topics."
    )
    private RegexSubscriptionMode regexSubscriptionMode = RegexSubscriptionMode.PersistentOnly;

    @ApiModelProperty(
            name = "deadLetterPolicy",
            value = "Dead letter policy for consumers.\n"
                    + "\n"
                    + "By default, some messages are probably redelivered many times, even to the extent that it "
                    + "never stops.\n"
                    + "\n"
                    + "By using the dead letter mechanism, messages have the max redelivery count. **When exceeding the"
                    + " maximum number of redeliveries, messages are sent to the Dead Letter Topic and acknowledged "
                    + "automatically**.\n"
                    + "\n"
                    + "You can enable the dead letter mechanism by setting `deadLetterPolicy`.\n"
                    + "\n"
                    + "**Example**\n"
                    + "```java\n"
                    + "client.newConsumer()\n"
                    + ".deadLetterPolicy(DeadLetterPolicy.builder().maxRedeliverCount(10).build())\n"
                    + ".subscribe();\n"
                    + "```\n"
                    + "Default dead letter topic name is `{TopicName}-{Subscription}-DLQ`.\n"
                    + "\n"
                    + "To set a custom dead letter topic name:\n"
                    + "```java\n"
                    + "client.newConsumer()\n"
                    + ".deadLetterPolicy(DeadLetterPolicy.builder().maxRedeliverCount(10)\n"
                    + ".deadLetterTopic(\"your-topic-name\").build())\n"
                    + ".subscribe();\n"
                    + "```\n"
                    + "When specifying the dead letter policy while not specifying `ackTimeoutMillis`, you can set the"
                    + " ack timeout to 30000 millisecond."
    )
    private transient DeadLetterPolicy deadLetterPolicy;

    private boolean retryEnable = false;

    @JsonIgnore
    private BatchReceivePolicy batchReceivePolicy;

    @ApiModelProperty(
            name = "autoUpdatePartitions",
            value = "If `autoUpdatePartitions` is enabled, a consumer subscribes to partition increasement "
                    + "automatically.\n"
                    + "\n"
                    + "**Note**: this is only for partitioned consumers."
    )
    private boolean autoUpdatePartitions = true;

    private long autoUpdatePartitionsIntervalSeconds = 60;

    @ApiModelProperty(
            name = "replicateSubscriptionState",
            value = "If `replicateSubscriptionState` is enabled, a subscription state is replicated to geo-replicated"
                    + " clusters."
    )
    private boolean replicateSubscriptionState = false;

    private boolean resetIncludeHead = false;

    @JsonIgnore
    private transient KeySharedPolicy keySharedPolicy;

    private boolean batchIndexAckEnabled = false;

    private boolean ackReceiptEnabled = false;

    private boolean poolMessages = false;

    @JsonIgnore
    private transient MessagePayloadProcessor payloadProcessor = null;

    private boolean startPaused = false;

    private boolean autoScaledReceiverQueueSizeEnabled = false;

    private List topicConfigurations = new ArrayList<>();

    public TopicConsumerConfigurationData getMatchingTopicConfiguration(String topicName) {
        return topicConfigurations.stream()
                .filter(topicConf -> topicConf.getTopicNameMatcher().matches(topicName))
                .findFirst()
                .orElseGet(() -> TopicConsumerConfigurationData.ofTopicName(topicName, this));
    }

    public void setTopicConfigurations(List topicConfigurations) {
        checkArgument(topicConfigurations != null, "topicConfigurations should not be null.");
        this.topicConfigurations = topicConfigurations;
    }

    public void setAutoUpdatePartitionsIntervalSeconds(int interval, TimeUnit timeUnit) {
        checkArgument(interval > 0, "interval needs to be > 0");
        this.autoUpdatePartitionsIntervalSeconds = timeUnit.toSeconds(interval);
    }

    @JsonIgnore
    public String getSingleTopic() {
        checkArgument(topicNames.size() == 1, "topicNames needs to be = 1");
        return topicNames.iterator().next();
    }

    public ConsumerConfigurationData clone() {
        try {
            @SuppressWarnings("unchecked")
            ConsumerConfigurationData c = (ConsumerConfigurationData) super.clone();
            c.topicNames = Sets.newTreeSet(this.topicNames);
            c.properties = new TreeMap<>(this.properties);
            return c;
        } catch (CloneNotSupportedException e) {
            throw new RuntimeException("Failed to clone ConsumerConfigurationData");
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy