org.apache.pulsar.client.impl.PatternConsumerUpdateQueue Maven / Gradle / Ivy
/*
* 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;
import org.apache.pulsar.shade.com.google.common.annotations.VisibleForTesting;
import edu.umd.cs.findbugs.annotations.SuppressFBWarnings;
import java.util.Collection;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.LinkedBlockingQueue;
import java.util.concurrent.TimeUnit;
import lombok.extern.slf4j.Slf4j;
import org.apache.pulsar.shade.org.apache.commons.lang3.tuple.Pair;
/**
* Used to make all tasks that will modify subscriptions will be executed one by one, and skip the unnecessary updating.
*
* So far, four three scenarios that will modify subscriptions:
* 1. When start pattern consumer.
* 2. After topic list watcher reconnected, it will call {@link PatternMultiTopicsConsumerImpl#recheckTopicsChange()}.
* this scenario only exists in the version >= 2.11 (both client-version and broker version are >= 2.11).
* 3. A scheduled task will call {@link PatternMultiTopicsConsumerImpl#recheckTopicsChange()}, this scenario only
* exists in the version < 2.11.
* 4. The topics change events will trigger a
* {@link PatternMultiTopicsConsumerImpl#topicsChangeListener#onTopicsRemoved(Collection)} or
* {@link PatternMultiTopicsConsumerImpl#topicsChangeListener#onTopicsAdded(Collection)}.
*
* When you are using this client connect to the broker whose version >= 2.11, there are three scenarios: [1, 2, 4].
* When you are using this client connect to the broker whose version < 2.11, there is only one scenario: [3] and all
* the event will run in the same thread.
*/
@Slf4j
@SuppressFBWarnings("EI_EXPOSE_REP2")
public class PatternConsumerUpdateQueue {
private static final Pair> RECHECK_OP =
Pair.of(UpdateSubscriptionType.RECHECK, null);
private final LinkedBlockingQueue>> pendingTasks;
private final PatternMultiTopicsConsumerImpl patternConsumer;
private final PatternMultiTopicsConsumerImpl.TopicsChangedListener topicsChangeListener;
/**
* Whether there is a task is in progress, this variable is used to confirm whether a next-task triggering is
* needed.
*/
private Pair> taskInProgress = null;
/**
* Whether there is a recheck task in queue.
* - Since recheck task will do all changes, it can be used to compress multiple tasks to one.
* - To avoid skipping the newest changes, once the recheck task is starting to work, this variable will be set
* to "false".
*/
private boolean recheckTaskInQueue = false;
private volatile long lastRecheckTaskStartingTimestamp = 0;
private boolean closed;
public PatternConsumerUpdateQueue(PatternMultiTopicsConsumerImpl patternConsumer) {
this(patternConsumer, patternConsumer.topicsChangeListener);
}
/** This constructor is only for test. **/
@VisibleForTesting
public PatternConsumerUpdateQueue(PatternMultiTopicsConsumerImpl patternConsumer,
PatternMultiTopicsConsumerImpl.TopicsChangedListener topicsChangeListener) {
this.patternConsumer = patternConsumer;
this.topicsChangeListener = topicsChangeListener;
this.pendingTasks = new LinkedBlockingQueue<>();
// To avoid subscribing and topics changed events execute concurrently, let the change events starts after the
// subscribing task.
doAppend(Pair.of(UpdateSubscriptionType.CONSUMER_INIT, null));
}
synchronized void appendTopicsAddedOp(Collection topics) {
if (topics == null || topics.isEmpty()) {
return;
}
doAppend(Pair.of(UpdateSubscriptionType.TOPICS_ADDED, topics));
}
synchronized void appendTopicsRemovedOp(Collection topics) {
if (topics == null || topics.isEmpty()) {
return;
}
doAppend(Pair.of(UpdateSubscriptionType.TOPICS_REMOVED, topics));
}
synchronized void appendRecheckOp() {
doAppend(RECHECK_OP);
}
synchronized void doAppend(Pair> task) {
if (log.isDebugEnabled()) {
log.debug("Pattern consumer [{}] try to append task. {} {}", patternConsumer.getSubscription(),
task.getLeft(), task.getRight() == null ? "" : task.getRight());
}
// Once there is a recheck task in queue, it means other tasks can be skipped.
if (recheckTaskInQueue) {
return;
}
// Once there are too many tasks in queue, compress them as a recheck task.
if (pendingTasks.size() >= 30 && !task.getLeft().equals(UpdateSubscriptionType.RECHECK)) {
appendRecheckOp();
return;
}
pendingTasks.add(task);
if (task.getLeft().equals(UpdateSubscriptionType.RECHECK)) {
recheckTaskInQueue = true;
}
// If no task is in-progress, trigger a task execution.
if (taskInProgress == null) {
triggerNextTask();
}
}
synchronized void triggerNextTask() {
if (closed) {
return;
}
final Pair> task = pendingTasks.poll();
// No pending task.
if (task == null) {
taskInProgress = null;
return;
}
// If there is a recheck task in queue, skip others and only call the recheck task.
if (recheckTaskInQueue && !task.getLeft().equals(UpdateSubscriptionType.RECHECK)) {
triggerNextTask();
return;
}
// Execute pending task.
CompletableFuture newTaskFuture = null;
switch (task.getLeft()) {
case CONSUMER_INIT: {
newTaskFuture = patternConsumer.getSubscribeFuture().thenAccept(__ -> {}).exceptionally(ex -> {
// If the subscribe future was failed, the consumer will be closed.
synchronized (PatternConsumerUpdateQueue.this) {
this.closed = true;
patternConsumer.closeAsync().exceptionally(ex2 -> {
log.error("Pattern consumer failed to close, this error may left orphan consumers."
+ " Subscription: {}", patternConsumer.getSubscription());
return null;
});
}
return null;
});
break;
}
case TOPICS_ADDED: {
newTaskFuture = topicsChangeListener.onTopicsAdded(task.getRight());
break;
}
case TOPICS_REMOVED: {
newTaskFuture = topicsChangeListener.onTopicsRemoved(task.getRight());
break;
}
case RECHECK: {
recheckTaskInQueue = false;
lastRecheckTaskStartingTimestamp = System.currentTimeMillis();
newTaskFuture = patternConsumer.recheckTopicsChange();
break;
}
default: {
throw new RuntimeException("Un-support UpdateSubscriptionType");
}
}
if (log.isDebugEnabled()) {
log.debug("Pattern consumer [{}] starting task. {} {} ", patternConsumer.getSubscription(),
task.getLeft(), task.getRight() == null ? "" : task.getRight());
}
// Trigger next pending task.
taskInProgress = Pair.of(task.getLeft(), newTaskFuture);
newTaskFuture.thenAccept(ignore -> {
if (log.isDebugEnabled()) {
log.debug("Pattern consumer [{}] task finished. {} {} ", patternConsumer.getSubscription(),
task.getLeft(), task.getRight() == null ? "" : task.getRight());
}
triggerNextTask();
}).exceptionally(ex -> {
/**
* Once a updating fails, trigger a delayed new recheck task to guarantee all things is correct.
* - Skip if there is already a recheck task in queue.
* - Skip if the last recheck task has been executed after the current time.
*/
log.error("Pattern consumer [{}] task finished. {} {}. But it failed", patternConsumer.getSubscription(),
task.getLeft(), task.getRight() == null ? "" : task.getRight(), ex);
// Skip if there is already a recheck task in queue.
synchronized (PatternConsumerUpdateQueue.this) {
if (recheckTaskInQueue || PatternConsumerUpdateQueue.this.closed) {
return null;
}
}
// Skip if the last recheck task has been executed after the current time.
long failedTime = System.currentTimeMillis();
patternConsumer.getClient().timer().newTimeout(timeout -> {
if (lastRecheckTaskStartingTimestamp <= failedTime) {
appendRecheckOp();
}
}, 10, TimeUnit.SECONDS);
triggerNextTask();
return null;
});
}
public synchronized CompletableFuture cancelAllAndWaitForTheRunningTask() {
this.closed = true;
if (taskInProgress == null) {
return CompletableFuture.completedFuture(null);
}
// If the in-progress task is consumer init task, it means nothing is in-progress.
if (taskInProgress.getLeft().equals(UpdateSubscriptionType.CONSUMER_INIT)) {
return CompletableFuture.completedFuture(null);
}
return taskInProgress.getRight().thenAccept(__ -> {}).exceptionally(ex -> null);
}
private enum UpdateSubscriptionType {
/** A marker that indicates the consumer's subscribe task.**/
CONSUMER_INIT,
/** Triggered by {@link PatternMultiTopicsConsumerImpl#topicsChangeListener}.**/
TOPICS_ADDED,
/** Triggered by {@link PatternMultiTopicsConsumerImpl#topicsChangeListener}.**/
TOPICS_REMOVED,
/** A fully check for pattern consumer. **/
RECHECK;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy