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

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