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

com.microsoft.azure.eventprocessorhost.PumpManager Maven / Gradle / Ivy

// Copyright (c) Microsoft Corporation. All rights reserved.
// Licensed under the MIT License.

package com.microsoft.azure.eventprocessorhost;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.util.concurrent.CompletableFuture;
import java.util.concurrent.ConcurrentHashMap;
import java.util.function.Consumer;


class PumpManager extends Closable implements Consumer {
    private static final Logger TRACE_LOGGER = LoggerFactory.getLogger(PumpManager.class);
    protected final HostContext hostContext;
    protected ConcurrentHashMap pumpStates; // protected for testability

    PumpManager(HostContext hostContext, Closable parent) {
        super(parent);

        this.hostContext = hostContext;

        this.pumpStates = new ConcurrentHashMap();
    }

    public void addPump(CompleteLease lease) {
        if (getIsClosingOrClosed()) {
            TRACE_LOGGER.info(this.hostContext.withHostAndPartition(lease, "Shutting down, not creating new pump"));
            return;
        }

        PartitionPump capturedPump = this.pumpStates.get(lease.getPartitionId()); // CONCURRENTHASHTABLE
        if (capturedPump != null) {
            // There already is a pump. This should never happen and it's not harmless if it does. If we get here,
            // it implies that the existing pump is a zombie which is not renewing its lease.
            TRACE_LOGGER.error(this.hostContext.withHostAndPartition(lease, "throwing away zombie pump"));
            // Shutdown should remove the pump from the hashmap, but we don't know what state this pump is in so
            // remove it manually. ConcurrentHashMap specifies that removing an item that doesn't exist is a safe no-op.
            this.pumpStates.remove(lease.getPartitionId());
            // Call shutdown to try to clean up, but do not wait.
            capturedPump.shutdown(CloseReason.Shutdown);
        }

        TRACE_LOGGER.info(this.hostContext.withHostAndPartition(lease, "creating new pump"));
        PartitionPump newPartitionPump = createNewPump(lease);
        this.pumpStates.put(lease.getPartitionId(), newPartitionPump);
        newPartitionPump.startPump();
    }

    // Callback used by pumps during pump shutdown. 
    @Override
    public void accept(String partitionId) {
        // These are fast, non-blocking actions.
        this.pumpStates.remove(partitionId);
        removingPumpTestHook(partitionId);
    }

    // Separated out so that tests can override and substitute their own pump class.
    protected PartitionPump createNewPump(CompleteLease lease) {
        return new PartitionPump(this.hostContext, lease, this, this);
    }

    public CompletableFuture removePump(String partitionId, final CloseReason reason) {
        CompletableFuture retval = CompletableFuture.completedFuture(null);
        PartitionPump capturedPump = this.pumpStates.get(partitionId); // CONCURRENTHASHTABLE
        if (capturedPump != null) {
            TRACE_LOGGER.info(this.hostContext.withHostAndPartition(partitionId,
                    "closing pump for reason " + reason.toString()));
            retval = capturedPump.shutdown(reason);
        } else {
            // Shouldn't get here but not really harmful, so just trace.
            TRACE_LOGGER.debug(this.hostContext.withHostAndPartition(partitionId,
                    "no pump found to remove for partition " + partitionId));
        }
        return retval;
    }

    public CompletableFuture removeAllPumps(CloseReason reason) {
        setClosing();

        CompletableFuture[] futures = new CompletableFuture[this.pumpStates.size()];
        int i = 0;
        for (String partitionId : this.pumpStates.keySet()) {
            futures[i++] = removePump(partitionId, reason);
        }

        return CompletableFuture.allOf(futures).whenCompleteAsync((empty, e) -> {
            setClosed();
        }, this.hostContext.getExecutor());
    }

    protected void removingPumpTestHook(String partitionId) {
        // For test use. MUST BE FAST, NON-BLOCKING.
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy