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

com.microsoft.azure.eventprocessorhost.PartitionScanner 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.ArrayList;
import java.util.Collections;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Random;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.function.Consumer;

class PartitionScanner extends Closable {
    private static final Logger TRACE_LOGGER = LoggerFactory.getLogger(PartitionScanner.class);
    private static final Random RANDOMIZER = new Random();
    private final HostContext hostContext;
    private final Consumer addPump;
    
    // Populated by getAllLeaseStates()
    private List allLeaseStates = null;
    
    // Values populated by sortLeasesAndCalculateDesiredCount
    private int desiredCount;
    private int unownedCount; // updated by acquireExpiredInChunksParallel
    private final ConcurrentHashMap leasesOwnedByOthers; // updated by acquireExpiredInChunksParallel

    PartitionScanner(HostContext hostContext, Consumer addPump, Closable parent) {
        super(parent);

        this.hostContext = hostContext;
        this.addPump = addPump;

        this.desiredCount = 0;
        this.unownedCount = 0;
        this.leasesOwnedByOthers = new ConcurrentHashMap();
    }

    public CompletableFuture scan(boolean isFirst) {
        return getAllLeaseStates()
                .thenCompose((unused) -> {
                    throwIfClosingOrClosed("PartitionScanner is shutting down");
                    int ourLeasesCount = sortLeasesAndCalculateDesiredCount(isFirst);
                    return acquireExpiredInChunksParallel(0, this.desiredCount - ourLeasesCount);
                })
                .thenApplyAsync((remainingNeeded) -> {
                    throwIfClosingOrClosed("PartitionScanner is shutting down");
                    ArrayList stealThese = new ArrayList();
                    if (remainingNeeded > 0) {
                        TRACE_LOGGER.debug(this.hostContext.withHost("Looking to steal: " + remainingNeeded));
                        stealThese = findLeasesToSteal(remainingNeeded);
                    }
                    return stealThese;
                }, this.hostContext.getExecutor())
                .thenCompose((stealThese) -> {
                    throwIfClosingOrClosed("PartitionScanner is shutting down");
                    return stealLeases(stealThese);
                })
                .handleAsync((didSteal, e) -> {
                    if ((e != null) && !(e instanceof ClosingException)) {
                        StringBuilder outAction = new StringBuilder();
                        Exception notifyWith = (Exception) LoggingUtils.unwrapException(e, outAction);
                        TRACE_LOGGER.warn(this.hostContext.withHost("Exception scanning leases"), notifyWith);
                        this.hostContext.getEventProcessorOptions().notifyOfException(this.hostContext.getHostName(), notifyWith, outAction.toString(),
                                ExceptionReceivedEventArgs.NO_ASSOCIATED_PARTITION);
                        didSteal = false;
                    }
                    return didSteal;
                }, this.hostContext.getExecutor());
    }

    private CompletableFuture getAllLeaseStates() {
        throwIfClosingOrClosed("PartitionScanner is shutting down");
        return this.hostContext.getLeaseManager().getAllLeases()
                .thenAcceptAsync((states) -> {
                    throwIfClosingOrClosed("PartitionScanner is shutting down");
                    this.allLeaseStates = states;
                    Collections.sort(this.allLeaseStates);
                }, this.hostContext.getExecutor());
    }

    // NONBLOCKING
    private int sortLeasesAndCalculateDesiredCount(boolean isFirst) {
        TRACE_LOGGER.debug(this.hostContext.withHost("Accounting input: allLeaseStates size is " + this.allLeaseStates.size()));

        HashSet uniqueOwners = new HashSet();
        uniqueOwners.add(this.hostContext.getHostName());
        int ourLeasesCount = 0;
        this.unownedCount = 0;
        for (BaseLease info : this.allLeaseStates) {
            boolean ownedByUs = info.getIsOwned() && info.getOwner() != null && (info.getOwner().compareTo(this.hostContext.getHostName()) == 0);
            if (info.getIsOwned() && info.getOwner() != null) {
                uniqueOwners.add(info.getOwner());
            } else {
                this.unownedCount++;
            }
            if (ownedByUs) {
                ourLeasesCount++;
            } else if (info.getIsOwned()) {
                this.leasesOwnedByOthers.put(info.getPartitionId(), info);
            }
        }
        int hostCount = uniqueOwners.size();
        int countPerHost = this.allLeaseStates.size() / hostCount;
        this.desiredCount = isFirst ? 1 : countPerHost;
        if (!isFirst && (this.unownedCount > 0) && (this.unownedCount < hostCount) && ((this.allLeaseStates.size() % hostCount) != 0)) {
            // Distribute leftovers.
            this.desiredCount++;
        }

        ArrayList sortedHosts = new ArrayList(uniqueOwners);
        Collections.sort(sortedHosts);
        int hostOrdinal = -1;
        int startingPoint = 0;
        if (isFirst) {
            // If the entire system is starting up, the list of hosts is probably not complete and we can't really
            // compute a meaningful hostOrdinal. But we only want hostOrdinal to calculate startingPoint. Instead,
            // just randomly select a startingPoint.
            startingPoint = PartitionScanner.RANDOMIZER.nextInt(this.allLeaseStates.size());
        } else {
            for (hostOrdinal = 0; hostOrdinal < sortedHosts.size(); hostOrdinal++) {
                if (sortedHosts.get(hostOrdinal).compareTo(this.hostContext.getHostName()) == 0) {
                    break;
                }
            }
            startingPoint = countPerHost * hostOrdinal;
        }
        // Rotate allLeaseStates
        TRACE_LOGGER.debug(this.hostContext.withHost("Host ordinal: " + hostOrdinal + "  Rotating leases to start at " + startingPoint));
        if (startingPoint != 0) {
            ArrayList rotatedList = new ArrayList(this.allLeaseStates.size());
            for (int j = 0; j < this.allLeaseStates.size(); j++) {
                rotatedList.add(this.allLeaseStates.get((j + startingPoint) % this.allLeaseStates.size()));
            }
            this.allLeaseStates = rotatedList;
        }

        TRACE_LOGGER.debug(this.hostContext.withHost("Host count is " + hostCount + "  Desired owned count is " + this.desiredCount));
        TRACE_LOGGER.debug(this.hostContext.withHost("ourLeasesCount " + ourLeasesCount + "  leasesOwnedByOthers " + this.leasesOwnedByOthers.size()
                + " unowned " + unownedCount));

        return ourLeasesCount;
    }

    // NONBLOCKING
    // Returns a CompletableFuture as a convenience for the caller
    private CompletableFuture> findExpiredLeases(int startAt, int endAt) {
        final ArrayList expiredLeases = new ArrayList();
        TRACE_LOGGER.debug(this.hostContext.withHost("Finding expired leases from '" + this.allLeaseStates.get(startAt).getPartitionId() + "'[" + startAt + "] up to '"
                + ((endAt < this.allLeaseStates.size()) ? this.allLeaseStates.get(endAt).getPartitionId() : "end") + "'[" + endAt + "]"));

        for (BaseLease info : this.allLeaseStates.subList(startAt, endAt)) {
            if (!info.getIsOwned()) {
                expiredLeases.add(info);
            }
        }

        TRACE_LOGGER.debug(this.hostContext.withHost("Found in range: " + expiredLeases.size()));
        return CompletableFuture.completedFuture(expiredLeases);
    }

    private CompletableFuture acquireExpiredInChunksParallel(int startAt, int needed) {
        throwIfClosingOrClosed("PartitionScanner is shutting down");

        CompletableFuture resultFuture = CompletableFuture.completedFuture(needed);
        if (startAt < this.allLeaseStates.size()) {
            TRACE_LOGGER.debug(this.hostContext.withHost("Examining chunk at '" + this.allLeaseStates.get(startAt).getPartitionId() + "'[" + startAt + "] need " + needed));
        } else {
            TRACE_LOGGER.debug(this.hostContext.withHost("Examining chunk skipping, startAt is off end: " + startAt));
        }

        if ((needed > 0) && (this.unownedCount > 0) && (startAt < this.allLeaseStates.size())) {
            final AtomicInteger runningNeeded = new AtomicInteger(needed);
            final int endAt = Math.min(startAt + needed, this.allLeaseStates.size());

            resultFuture = findExpiredLeases(startAt, endAt)
                    .thenCompose((getThese) -> {
                        throwIfClosingOrClosed("PartitionScanner is shutting down");
                        CompletableFuture acquireFuture = CompletableFuture.completedFuture(null);
                        if (getThese.size() > 0) {
                            ArrayList> getFutures = new ArrayList>();
                            for (BaseLease info : getThese) {
                                throwIfClosingOrClosed("PartitionScanner is shutting down");
                                final AcquisitionHolder holder = new AcquisitionHolder();
                                CompletableFuture getOneFuture = this.hostContext.getLeaseManager().getLease(info.getPartitionId())
                                        .thenCompose((lease) -> {
                                            throwIfClosingOrClosed("PartitionScanner is shutting down");
                                            holder.setAcquiredLease(lease);
                                            return this.hostContext.getLeaseManager().acquireLease(lease);
                                        })
                                        .thenAcceptAsync((acquired) -> {
                                            throwIfClosingOrClosed("PartitionScanner is shutting down");
                                            if (acquired) {
                                                runningNeeded.decrementAndGet();
                                                TRACE_LOGGER.debug(this.hostContext.withHostAndPartition(holder.getAcquiredLease().getPartitionId(), "Acquired unowned/expired"));
                                                if (this.leasesOwnedByOthers.containsKey(holder.getAcquiredLease().getPartitionId())) {
                                                    this.leasesOwnedByOthers.remove(holder.getAcquiredLease().getPartitionId());
                                                    this.unownedCount--;
                                                }
                                                this.addPump.accept(holder.getAcquiredLease());
                                            } else {
                                                this.leasesOwnedByOthers.put(holder.getAcquiredLease().getPartitionId(), holder.getAcquiredLease());
                                            }
                                        }, this.hostContext.getExecutor());
                                getFutures.add(getOneFuture);
                            }
                            CompletableFuture[] dummy = new CompletableFuture[getFutures.size()];
                            acquireFuture = CompletableFuture.allOf(getFutures.toArray(dummy));
                        }
                        return acquireFuture;
                    })
                    .handleAsync((empty, e) -> {
                        // log/notify if exception occurred, then swallow exception and continue with next chunk
                        if ((e != null) && !(e instanceof ClosingException)) {
                            Exception notifyWith = (Exception) LoggingUtils.unwrapException(e, null);
                            TRACE_LOGGER.warn(this.hostContext.withHost("Failure getting/acquiring lease, continuing"), notifyWith);
                            this.hostContext.getEventProcessorOptions().notifyOfException(this.hostContext.getHostName(), notifyWith,
                                    EventProcessorHostActionStrings.CHECKING_LEASES, ExceptionReceivedEventArgs.NO_ASSOCIATED_PARTITION);
                        }
                        return null;
                    }, this.hostContext.getExecutor())
                    .thenCompose((unused) -> acquireExpiredInChunksParallel(endAt, runningNeeded.get()));
        } else {
            TRACE_LOGGER.debug(this.hostContext.withHost("Short circuit: needed is 0, unowned is 0, or off end"));
        }

        return resultFuture;
    }

    // NONBLOCKING
    private ArrayList findLeasesToSteal(int stealAsk) {
        // Generate a map of hostnames and owned counts.
        HashMap hostOwns = new HashMap();
        for (BaseLease info : this.leasesOwnedByOthers.values()) {
            if (hostOwns.containsKey(info.getOwner())) {
                int newCount = hostOwns.get(info.getOwner()) + 1;
                hostOwns.put(info.getOwner(), newCount);
            } else {
                hostOwns.put(info.getOwner(), 1);
            }
        }

        // Extract hosts which own more than the desired count
        ArrayList bigOwners = new ArrayList();
        for (Map.Entry pair : hostOwns.entrySet()) {
            if (pair.getValue() > this.desiredCount) {
                bigOwners.add(pair.getKey());
                TRACE_LOGGER.debug(this.hostContext.withHost("Big owner " + pair.getKey() + " has " + pair.getValue()));
            }
        }

        ArrayList stealInfos = new ArrayList();

        if (bigOwners.size() > 0) {
            // Randomly pick one of the big owners
            String bigVictim = bigOwners.get(PartitionScanner.RANDOMIZER.nextInt(bigOwners.size()));
            int victimExtra = hostOwns.get(bigVictim) - this.desiredCount - 1;
            int stealCount = Math.min(victimExtra, stealAsk);
            TRACE_LOGGER.debug(this.hostContext.withHost("Stealing " + stealCount + " from " + bigVictim));

            // Grab stealCount partitions owned by bigVictim and return the infos.
            for (BaseLease candidate : this.allLeaseStates) {
                if (candidate.getOwner() != null && candidate.getOwner().compareTo(bigVictim) == 0) {
                    stealInfos.add(candidate);
                    if (stealInfos.size() >= stealCount) {
                        break;
                    }
                }
            }
        } else {
            TRACE_LOGGER.debug(this.hostContext.withHost("No big owners found, skipping steal"));
        }

        return stealInfos;
    }

    private CompletableFuture stealLeases(List stealThese) {
        CompletableFuture allSteals = CompletableFuture.completedFuture(false);

        if (stealThese.size() > 0) {
            ArrayList> steals = new ArrayList>();
            for (BaseLease info : stealThese) {
                throwIfClosingOrClosed("PartitionScanner is shutting down");

                final AcquisitionHolder holder = new AcquisitionHolder();
                CompletableFuture oneSteal = this.hostContext.getLeaseManager().getLease(info.getPartitionId())
                    .thenCompose((lease) -> {
                        throwIfClosingOrClosed("PartitionScanner is shutting down");
                        holder.setAcquiredLease(lease);
                        return this.hostContext.getLeaseManager().acquireLease(lease);
                    })
                    .thenAcceptAsync((acquired) -> {
                        throwIfClosingOrClosed("PartitionScanner is shutting down");
                        if (acquired) {
                            TRACE_LOGGER.debug(this.hostContext.withHostAndPartition(holder.getAcquiredLease().getPartitionId(), "Stole lease"));
                            this.addPump.accept(holder.getAcquiredLease());
                        }
                    }, this.hostContext.getExecutor());
                steals.add(oneSteal);
            }

            CompletableFuture[] dummy = new CompletableFuture[steals.size()];
            allSteals = CompletableFuture.allOf(steals.toArray(dummy)).thenApplyAsync((empty) -> true, this.hostContext.getExecutor());
        }

        return allSteals;
    }

    private static class AcquisitionHolder {
        private CompleteLease acquiredLease;

        void setAcquiredLease(CompleteLease l) {
            this.acquiredLease = l;
        }

        CompleteLease getAcquiredLease() {
            return this.acquiredLease;
        }
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy