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

com.bazaarvoice.emodb.databus.core.DefaultDatabus Maven / Gradle / Ivy

package com.bazaarvoice.emodb.databus.core;

import com.bazaarvoice.emodb.common.dropwizard.lifecycle.LifeCycleRegistry;
import com.bazaarvoice.emodb.common.dropwizard.time.ClockTicker;
import com.bazaarvoice.emodb.common.uuid.TimeUUIDs;
import com.bazaarvoice.emodb.databus.ChannelNames;
import com.bazaarvoice.emodb.databus.DefaultJoinFilter;
import com.bazaarvoice.emodb.databus.MasterFanoutPartitions;
import com.bazaarvoice.emodb.databus.QueueDrainExecutorService;
import com.bazaarvoice.emodb.databus.SystemIdentity;
import com.bazaarvoice.emodb.databus.api.Event;
import com.bazaarvoice.emodb.databus.api.MoveSubscriptionStatus;
import com.bazaarvoice.emodb.databus.api.Names;
import com.bazaarvoice.emodb.databus.api.PollResult;
import com.bazaarvoice.emodb.databus.api.ReplaySubscriptionStatus;
import com.bazaarvoice.emodb.databus.api.Subscription;
import com.bazaarvoice.emodb.databus.api.UnauthorizedSubscriptionException;
import com.bazaarvoice.emodb.databus.api.UnknownMoveException;
import com.bazaarvoice.emodb.databus.api.UnknownReplayException;
import com.bazaarvoice.emodb.databus.api.UnknownSubscriptionException;
import com.bazaarvoice.emodb.databus.auth.DatabusAuthorizer;
import com.bazaarvoice.emodb.databus.db.SubscriptionDAO;
import com.bazaarvoice.emodb.databus.model.OwnedSubscription;
import com.bazaarvoice.emodb.event.api.EventData;
import com.bazaarvoice.emodb.event.api.EventSink;
import com.bazaarvoice.emodb.event.core.SizeCacheKey;
import com.bazaarvoice.emodb.job.api.JobHandler;
import com.bazaarvoice.emodb.job.api.JobHandlerRegistry;
import com.bazaarvoice.emodb.job.api.JobIdentifier;
import com.bazaarvoice.emodb.job.api.JobRequest;
import com.bazaarvoice.emodb.job.api.JobService;
import com.bazaarvoice.emodb.job.api.JobStatus;
import com.bazaarvoice.emodb.sor.api.Coordinate;
import com.bazaarvoice.emodb.sor.api.Intrinsic;
import com.bazaarvoice.emodb.sor.api.ReadConsistency;
import com.bazaarvoice.emodb.sor.api.UnknownPlacementException;
import com.bazaarvoice.emodb.sor.api.UnknownTableException;
import com.bazaarvoice.emodb.sor.condition.Condition;
import com.bazaarvoice.emodb.sor.condition.Conditions;
import com.bazaarvoice.emodb.sor.core.DataProvider;
import com.bazaarvoice.emodb.sor.core.DatabusEventWriter;
import com.bazaarvoice.emodb.sor.core.DatabusEventWriterRegistry;
import com.bazaarvoice.emodb.sor.core.UpdateRef;
import com.bazaarvoice.emodb.sortedq.core.ReadOnlyQueueException;
import com.codahale.metrics.Meter;
import com.codahale.metrics.MetricRegistry;
import com.google.common.annotations.VisibleForTesting;
import com.google.common.base.Stopwatch;
import com.google.common.base.Supplier;
import com.google.common.base.Ticker;
import com.google.common.cache.Cache;
import com.google.common.cache.CacheBuilder;
import com.google.common.cache.CacheLoader;
import com.google.common.cache.LoadingCache;
import com.google.common.collect.AbstractIterator;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableMap;
import com.google.common.collect.ImmutableMultimap;
import com.google.common.collect.ImmutableSet;
import com.google.common.collect.Iterators;
import com.google.common.collect.Lists;
import com.google.common.collect.Maps;
import com.google.common.collect.Ordering;
import com.google.common.primitives.Ints;
import com.google.inject.Inject;
import io.dropwizard.lifecycle.Managed;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import javax.annotation.Nullable;
import java.nio.ByteBuffer;
import java.time.Clock;
import java.time.Duration;
import java.util.Collection;
import java.util.Collections;
import java.util.Comparator;
import java.util.Date;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.UUID;
import java.util.concurrent.ConcurrentMap;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.TimeUnit;
import java.util.stream.Collectors;
import java.util.stream.StreamSupport;

import static com.google.common.base.Preconditions.checkArgument;
import static com.google.common.base.Preconditions.checkState;
import static java.util.Objects.requireNonNull;

public class DefaultDatabus implements OwnerAwareDatabus, DatabusEventWriter, Managed {

    private static final Logger _log = LoggerFactory.getLogger(DefaultDatabus.class);

    /**
     * How long should poll loop, searching for events before giving up and returning.
     */
    private static final Duration MAX_POLL_TIME = Duration.ofMillis(100);

    /**
     * How long should the app wait before querying the data store again when it finds an unknown change?
     */
    private static final Duration RECENT_UNKNOWN_RETRY = Duration.ofMillis(400);

    /**
     * How old does an event need to be before we stop aggressively looking for it?
     */
    private static final Duration STALE_UNKNOWN_AGE = Duration.ofSeconds(2);

    /**
     * Don't merge too many duplicate events together to avoid event keys getting unreasonably long.
     */
    private static final int MAX_EVENTS_TO_CONSOLIDATE = 1000;

    /* How many items are to be fetched in each try for draining the queue. */
    private static final int MAX_ITEMS_TO_FETCH_FOR_QUEUE_DRAINING = 100;

    /* This is how long we submit tasks to drain the queue for each subscription from one poll request */
    private static final Duration MAX_QUEUE_DRAIN_TIME_FOR_A_SUBSCRIPTION = Duration.ofMinutes(1);

    /* We don't allow subscriptions to be created beyond this number. */
    private static final Duration MAX_SUBSCRIPTION_TTL = Duration.ofDays(365 * 10);
    /* We don't allow subscriptions to be created with event TTLs beyond this number. */
    public static final Duration MAX_EVENT_TTL = Duration.ofDays(365);

    private final DatabusEventWriterRegistry _eventWriterRegistry;
    private final SubscriptionDAO _subscriptionDao;
    private final DatabusEventStore _eventStore;
    private final DataProvider _dataProvider;
    private final SubscriptionEvaluator _subscriptionEvaluator;
    private final JobService _jobService;
    private final DatabusAuthorizer _databusAuthorizer;
    private final String _systemOwnerId;
    private final PartitionSelector _masterPartitionSelector;
    private final List _masterFanoutChannels;
    private final Meter _peekedMeter;
    private final Meter _polledMeter;
    private final Meter _renewedMeter;
    private final Meter _ackedMeter;
    private final Meter _recentUnknownMeter;
    private final Meter _staleUnknownMeter;
    private final Meter _redundantMeter;
    private final Meter _discardedMeter;
    private final Meter _consolidatedMeter;
    private final Meter _unownedSubscriptionMeter;
    private final Meter _drainQueueAsyncMeter;
    private final Meter _drainQueueTaskMeter;
    private final Meter _drainQueueRedundantMeter;
    private final LoadingCache> _eventSizeCache;
    private final Supplier _defaultJoinFilterCondition;
    private final Ticker _ticker;
    private final Clock _clock;
    private ExecutorService _drainService;
    private ConcurrentMap _drainedSubscriptionsMap = Maps.newConcurrentMap();

    @Inject
    public DefaultDatabus(LifeCycleRegistry lifeCycle, DatabusEventWriterRegistry eventWriterRegistry,
                          DataProvider dataProvider, SubscriptionDAO subscriptionDao, DatabusEventStore eventStore,
                          SubscriptionEvaluator subscriptionEvaluator, JobService jobService,
                          JobHandlerRegistry jobHandlerRegistry, DatabusAuthorizer databusAuthorizer,
                          @SystemIdentity String systemOwnerId,
                          @DefaultJoinFilter Supplier defaultJoinFilterCondition,
                          @QueueDrainExecutorService ExecutorService drainService,
                          @MasterFanoutPartitions int masterPartitions,
                          @MasterFanoutPartitions PartitionSelector masterPartitionSelector,
                          MetricRegistry metricRegistry, Clock clock) {
        _eventWriterRegistry = eventWriterRegistry;
        _subscriptionDao = subscriptionDao;
        _eventStore = eventStore;
        _dataProvider = dataProvider;
        _subscriptionEvaluator = subscriptionEvaluator;
        _jobService = jobService;
        _databusAuthorizer = databusAuthorizer;
        _systemOwnerId = systemOwnerId;
        _defaultJoinFilterCondition = defaultJoinFilterCondition;
        _drainService = requireNonNull(drainService, "drainService");
        _masterPartitionSelector = masterPartitionSelector;
        _ticker = ClockTicker.getTicker(clock);
        _clock = clock;
        _peekedMeter = newEventMeter("peeked", metricRegistry);
        _polledMeter = newEventMeter("polled", metricRegistry);
        _renewedMeter = newEventMeter("renewed", metricRegistry);
        _ackedMeter = newEventMeter("acked", metricRegistry);
        _recentUnknownMeter = newEventMeter("recent-unknown", metricRegistry);
        _staleUnknownMeter = newEventMeter("stale-unknown", metricRegistry);
        _redundantMeter = newEventMeter("redundant", metricRegistry);
        _discardedMeter = newEventMeter("discarded", metricRegistry);
        _consolidatedMeter = newEventMeter("consolidated", metricRegistry);
        _unownedSubscriptionMeter = newEventMeter("unowned", metricRegistry);
        _drainQueueAsyncMeter = newEventMeter("drainQueueAsync", metricRegistry);
        _drainQueueTaskMeter = newEventMeter("drainQueueTask", metricRegistry);
        _drainQueueRedundantMeter = newEventMeter("drainQueueRedundant", metricRegistry);
        _eventSizeCache = CacheBuilder.newBuilder()
                .expireAfterWrite(15, TimeUnit.SECONDS)
                .maximumSize(2000)
                .ticker(_ticker)
                .build(new CacheLoader>() {
                    @Override
                    public Map.Entry load(SizeCacheKey key)
                            throws Exception {
                        return Maps.immutableEntry(internalEventCountUpTo(key.channelName, key.limitAsked), key.limitAsked);
                    }
                });
        lifeCycle.manage(this);

        ImmutableList.Builder masterFanoutChannels = ImmutableList.builder();
        for (int partition=0; partition < masterPartitions; partition++) {
            masterFanoutChannels.add(ChannelNames.getMasterFanoutChannel(partition));
        }
        _masterFanoutChannels = masterFanoutChannels.build();

        requireNonNull(jobHandlerRegistry, "jobHandlerRegistry");
        registerMoveSubscriptionJobHandler(jobHandlerRegistry);
        registerReplaySubscriptionJobHandler(jobHandlerRegistry);
    }

    private void registerMoveSubscriptionJobHandler(JobHandlerRegistry jobHandlerRegistry) {
        jobHandlerRegistry.addHandler(
                MoveSubscriptionJob.INSTANCE,
                new Supplier>() {
                    @Override
                    public JobHandler get() {
                        return new JobHandler() {
                            @Override
                            public MoveSubscriptionResult run(MoveSubscriptionRequest request)
                                    throws Exception {
                                try {
                                    // Last chance to verify the subscriptions' owner before doing anything mutative
                                    checkSubscriptionOwner(request.getOwnerId(), request.getFrom());
                                    checkSubscriptionOwner(request.getOwnerId(), request.getTo());

                                    _eventStore.move(request.getFrom(), request.getTo());
                                } catch (ReadOnlyQueueException e) {
                                    // The from queue is not owned by this server.
                                    return notOwner();
                                }
                                return new MoveSubscriptionResult(new Date());
                            }
                        };
                    }
                });
    }

    private void registerReplaySubscriptionJobHandler(JobHandlerRegistry jobHandlerRegistry) {
        jobHandlerRegistry.addHandler(
                ReplaySubscriptionJob.INSTANCE,
                new Supplier>() {
                    @Override
                    public JobHandler get() {
                        return new JobHandler() {
                            @Override
                            public ReplaySubscriptionResult run(ReplaySubscriptionRequest request)
                                    throws Exception {
                                try {
                                    // Last chance to verify the subscription's owner before doing anything mutative
                                    checkSubscriptionOwner(request.getOwnerId(), request.getSubscription());

                                    replay(request.getSubscription(), request.getSince());
                                } catch (ReadOnlyQueueException e) {
                                    // The subscription is not owned by this server.
                                    return notOwner();
                                }
                                // Make sure that we have completed the replay request in time.
                                // If replay took more than the replay TTL time, we should fail it as
                                // there could be some events that were expired before we could replay it
                                if (request.getSince() != null && request.getSince().toInstant()
                                        .plus(DatabusChannelConfiguration.REPLAY_TTL)
                                        .isBefore(_clock.instant())) {
                                    // Uh-oh we were too late in replaying, and could have lost some events
                                    throw new ReplayTooLateException();
                                }
                                return new ReplaySubscriptionResult(new Date());
                            }
                        };
                    }
                });
    }

    private void createDatabusReplaySubscription() {
        // Create a master databus replay subscription where the events expire every 50 hours (2 days + 2 hours)
        subscribe(_systemOwnerId, ChannelNames.getMasterReplayChannel(), Conditions.alwaysTrue(),
                Duration.ofDays(3650), DatabusChannelConfiguration.REPLAY_TTL, false);
    }

    private Meter newEventMeter(String name, MetricRegistry metricRegistry) {
        return metricRegistry.meter(getMetricName(name));
    }

    private String getMetricName(String name) {
        return MetricRegistry.name("bv.emodb.databus", "DefaultDatabus", name);
    }

    @VisibleForTesting
    protected ConcurrentMap getDrainedSubscriptionsMap() {
        return _drainedSubscriptionsMap;
    }

    @Override
    public void start()
            throws Exception {
        // Create a databus replay subscription
        createDatabusReplaySubscription();
        _eventWriterRegistry.registerDatabusEventWriter(this);
    }

    @Override
    public void stop()
            throws Exception {
    }

    @Override
    public Iterator listSubscriptions(final String ownerId, @Nullable String fromSubscriptionExclusive, long limit) {
        checkArgument(limit > 0, "Limit must be >0");

        // We always have all the subscriptions cached in memory so fetch them all.
        Iterable allSubscriptions = _subscriptionDao.getAllSubscriptions();

        return StreamSupport.stream(allSubscriptions.spliterator(), false)
                // Ignore subscriptions not accessible by the owner.
                .filter((subscription) -> _databusAuthorizer.owner(ownerId).canAccessSubscription(subscription))
                // Sort them by name.  They're stored sorted in Cassandra so this should be a no-op, but
                // do the sort anyway so we're not depending on internals of the subscription DAO.
                .sorted((left, right) -> left.getName().compareTo(right.getName()))
                // Apply the "from" parameter
                .filter(subscription -> fromSubscriptionExclusive == null || subscription.getName().compareTo(fromSubscriptionExclusive) > 0)
                // Apply the "limit" parameter (be careful to avoid overflow when limit == Long.MAX_VALUE).
                .limit(limit)
                // Necessary to make generics work
                .map(subscription -> (Subscription) subscription)
                .iterator();
    }

    @Override
    public void subscribe(String ownerId, String subscription, Condition tableFilter, Duration subscriptionTtl, Duration eventTtl) {
        subscribe(ownerId, subscription, tableFilter, subscriptionTtl, eventTtl, true);
    }

    @Override
    public void subscribe(String ownerId, String subscription, Condition tableFilter, Duration subscriptionTtl,
                          Duration eventTtl, boolean includeDefaultJoinFilter) {
        // This call should be deprecated soon.
        checkLegalSubscriptionName(subscription);
        checkSubscriptionOwner(ownerId, subscription);
        requireNonNull(tableFilter, "tableFilter");
        checkArgument(subscriptionTtl.compareTo(Duration.ZERO) > 0, "SubscriptionTtl must be >0");
        checkArgument(subscriptionTtl.compareTo(MAX_SUBSCRIPTION_TTL) <= 0, "Subscription TTL duration limit is 10 years. The value cannot go beyond that.");
        checkArgument(eventTtl.compareTo(Duration.ZERO) > 0, "EventTtl must be >0");
        checkArgument(eventTtl.compareTo(MAX_EVENT_TTL) <= 0, "Event TTL duration limit is 365 days. The value cannot go beyond that.");
        SubscriptionConditionValidator.checkAllowed(tableFilter);

        if (includeDefaultJoinFilter) {
            // If the default join filter condition is set (that is, isn't "alwaysTrue()") then add it to the filter
            Condition defaultJoinFilterCondition = _defaultJoinFilterCondition.get();
            if (!Conditions.alwaysTrue().equals(defaultJoinFilterCondition)) {
                if (tableFilter.equals(Conditions.alwaysTrue())) {
                    tableFilter = defaultJoinFilterCondition;
                } else {
                    tableFilter = Conditions.and(tableFilter, defaultJoinFilterCondition);
                }
            }
        }

        // except for resetting the ttl, recreating a subscription that already exists has no effect.
        // assume that multiple servers that manage the same subscriptions can each attempt to create
        // the subscription at startup.
        _subscriptionDao.insertSubscription(ownerId, subscription, tableFilter, subscriptionTtl, eventTtl);
    }

    @Override
    public void unsubscribe(String ownerId, String subscription) {
        checkLegalSubscriptionName(subscription);
        checkSubscriptionOwner(ownerId, subscription);

        _subscriptionDao.deleteSubscription(subscription);
        _eventStore.purge(subscription);
    }

    @Override
    public Subscription getSubscription(String ownerId, String name)
            throws UnknownSubscriptionException {
        checkLegalSubscriptionName(name);

        OwnedSubscription subscription = getSubscriptionByName(name);
        checkSubscriptionOwner(ownerId, subscription);

        return subscription;
    }

    private OwnedSubscription getSubscriptionByName(String name) {
        OwnedSubscription subscription = _subscriptionDao.getSubscription(name);
        if (subscription == null) {
            throw new UnknownSubscriptionException(name);
        }
        return subscription;
    }

    @Override
    public void writeEvents(Collection refs) {
        ImmutableMultimap.Builder eventIds = ImmutableMultimap.builder();
        for (UpdateRef ref : refs) {
            int partition = _masterPartitionSelector.getPartition(ref.getKey());
            eventIds.put(_masterFanoutChannels.get(partition), UpdateRefSerializer.toByteBuffer(ref));
        }
        _eventStore.addAll(eventIds.build());
    }
    
    @Override
    public long getEventCount(String ownerId, String subscription) {
        return getEventCountUpTo(ownerId, subscription, Long.MAX_VALUE);
    }

    @Override
    public long getEventCountUpTo(String ownerId, String subscription, long limit) {
        checkSubscriptionOwner(ownerId, subscription);

        // We get the size from cache as a tuple of size, and the limit used to estimate that size
        // So, the key is the size, and value is the limit used to estimate the size
        SizeCacheKey sizeCacheKey = new SizeCacheKey(subscription, limit);
        Map.Entry size = _eventSizeCache.getUnchecked(sizeCacheKey);
        if (size.getValue() >= limit) { // We have the same or better estimate
            return size.getKey();
        }
        // User wants a better estimate than what our cache has, so we need to invalidate this key and load a new size
        _eventSizeCache.invalidate(sizeCacheKey);
        return _eventSizeCache.getUnchecked(sizeCacheKey).getKey();
    }

    private long internalEventCountUpTo(String subscription, long limit) {
        checkLegalSubscriptionName(subscription);
        checkArgument(limit > 0, "Limit must be >0");

        return _eventStore.getSizeEstimate(subscription, limit);
    }

    @Override
    public long getClaimCount(String ownerId, String subscription) {
        checkLegalSubscriptionName(subscription);
        checkSubscriptionOwner(ownerId, subscription);

        return _eventStore.getClaimCount(subscription);
    }

    @Override
    public Iterator peek(String ownerId, final String subscription, int limit) {
        checkLegalSubscriptionName(subscription);
        checkArgument(limit > 0, "Limit must be >0");
        checkSubscriptionOwner(ownerId, subscription);

        PollResult result = peekOrPoll(subscription, null, limit);
        return result.getEventIterator();
    }

    @Override
    public PollResult poll(String ownerId, final String subscription, final Duration claimTtl, int limit) {
        checkLegalSubscriptionName(subscription);
        checkArgument(claimTtl.compareTo(Duration.ZERO) >= 0, "ClaimTtl must be >=0");
        checkArgument(limit > 0, "Limit must be >0");
        checkSubscriptionOwner(ownerId, subscription);

        return peekOrPoll(subscription, claimTtl, limit);
    }

    /** Implements peek() or poll() based on whether claimTtl is null or non-null. */
    private PollResult peekOrPoll(String subscription, @Nullable Duration claimTtl, int limit) {
        int remaining = limit;
        Map rawEvents = ImmutableMap.of();
        Map uniqueItems = Maps.newHashMap();
        boolean isPeek = claimTtl == null;
        boolean repeatable = !isPeek && claimTtl.toMillis() > 0;
        boolean eventsAvailableForNextPoll = false;
        boolean noMaxPollTimeOut = true;
        int itemsDiscarded = 0;
        Meter eventMeter = isPeek ? _peekedMeter : _polledMeter;

        // Reading raw events from the event store is a significantly faster operation than resolving the events into
        // databus poll events.  This is because the former sequentially loads small event references while the latter
        // requires reading and resolving the effectively random objects associated with those references from the data
        // store.
        //
        // To make the process more efficient this method first polls for "limit" raw events from the event store.
        // Then, up to the first 10 of those raw events are resolved synchronously with a time limit of MAX_POLL_TIME.
        // Any remaining raw events are resolved lazily as the event list is consumed by the caller.  This makes the
        // return time for this method faster and more predictable while supporting polls for more events than can
        // be resolved within MAX_POLL_TIME.  This is especially beneficial for REST clients which may otherwise time
        // out while waiting for "limit" events to be read and resolved.

        Stopwatch stopwatch = Stopwatch.createStarted(_ticker);
        int padding = 0;
        do {
            if (remaining == 0) {
                break;  // Don't need any more events.
            }

            // Query the databus event store.  Consolidate multiple events that refer to the same item.
            ConsolidatingEventSink sink = new ConsolidatingEventSink(remaining + padding);
            boolean more = isPeek ?
                    _eventStore.peek(subscription, sink) :
                    _eventStore.poll(subscription, claimTtl, sink);
            rawEvents = sink.getEvents();

            if (rawEvents.isEmpty()) {
                // No events to be had.
                eventsAvailableForNextPoll = more;
                break;
            }

            // Resolve the raw events in batches of 10 until at least one response item is found for a maximum time of MAX_POLL_TIME.
            do {
                int batchItemsDiscarded = resolvePeekOrPollEvents(subscription, rawEvents, Math.min(10, remaining),
                        (coord, item) -> {
                            // Check whether we've already added this piece of content to the poll result.  If so, consolidate
                            // the two together to reduce the amount of work a client must do.  Note that the previous item
                            // would be from a previous batch of events and it's possible that we have read two different
                            // versions of the same item of content.  This will prefer the most recent.
                            Item previousItem = uniqueItems.get(coord);
                            if (previousItem != null && previousItem.consolidateWith(item)) {
                                _consolidatedMeter.mark();
                            } else {
                                // We have found a new item of content to return!
                                uniqueItems.put(coord, item);
                            }
                        });
                remaining = limit - uniqueItems.size();
                itemsDiscarded += batchItemsDiscarded;
            } while (!rawEvents.isEmpty() && remaining > 0 && stopwatch.elapsed(TimeUnit.MILLISECONDS) < MAX_POLL_TIME.toMillis());

            // There are more events for the next poll if either the event store explicitly said so or if, due to padding,
            // we got more events than "limit", in which case we're likely to unclaim at last one.
            eventsAvailableForNextPoll = more || rawEvents.size() + uniqueItems.size() > limit;
            if (!more) {
                // There are no more events to be had, so exit now
                break;
            }

            // Note: Due to redundant/unknown events, it's possible that the 'events' list is empty even though, if we
            // tried again, we'd find more events.  Try again a few times, but not for more than MAX_POLL_TIME so clients
            // don't timeout the request.  This helps move through large amounts of redundant deltas relatively quickly
            // while also putting a bound on the total amount of work done by a single call to poll().
            padding = 10;
        } while (repeatable && (noMaxPollTimeOut = stopwatch.elapsed(TimeUnit.MILLISECONDS) < MAX_POLL_TIME.toMillis()));

        Iterator events;
        int approximateSize;
        if (uniqueItems.isEmpty()) {
            // Either there were no raw events or all events found were for redundant or unknown changes.  It's possible
            // that eventually there will be more events, but to prevent a lengthy delay iterating the remaining events
            // quit now and return an empty result.  The caller can always poll again to try to pick up any more events,
            // and if necessary an async drain will kick off a few lines down to assist in clearing the redundant update
            // wasteland.
            events = Collections.emptyIterator();
            approximateSize = 0;

            // If there are still more unresolved events claimed then unclaim them now
            if (repeatable && !rawEvents.isEmpty()) {
                unclaim(subscription, rawEvents.values());
            }
        } else if (rawEvents.isEmpty()) {
            // All events have been resolved
            events = toEvents(uniqueItems.values()).iterator();
            approximateSize = uniqueItems.size();
            eventMeter.mark(approximateSize);
        } else {
            // Return an event list which contains the first events which were resolved synchronously plus the
            // remaining events from the peek or poll which will be resolved lazily in batches of 25.

            final Map deferredRawEvents = Maps.newLinkedHashMap(rawEvents);
            final int initialDeferredLimit = remaining;

            Iterator deferredEvents = new AbstractIterator() {
                private Iterator currentBatch = Collections.emptyIterator();
                private int remaining = initialDeferredLimit;

                @Override
                protected Event computeNext() {
                    Event next = null;

                    if (currentBatch.hasNext()) {
                        next = currentBatch.next();
                    } else if (!deferredRawEvents.isEmpty() && remaining > 0) {
                        // Resolve the next batch of events
                        try {
                            final List items = Lists.newArrayList();
                            do {
                                resolvePeekOrPollEvents(subscription, deferredRawEvents, Math.min(remaining, 25),
                                        (coord, item) -> {
                                            // Unlike with the original batch the deferred batch's events are always
                                            // already de-duplicated by coordinate, so there is no need to maintain
                                            // a coordinate-to-item uniqueness map.
                                            items.add(item);
                                        });

                                if (!items.isEmpty()) {
                                    remaining -= items.size();
                                    currentBatch = toEvents(items).iterator();
                                    next = currentBatch.next();
                                }
                            } while (next == null && !deferredRawEvents.isEmpty() && remaining > 0);
                        } catch (Exception e) {
                            // Don't fail; the caller has already received some events.  Just cut the result stream short
                            // now and throw back any remaining events for a future poll.
                            _log.warn("Failed to load additional events during peek/poll for subscription {}", subscription, e);
                        }
                    }

                    if (next != null) {
                        return next;
                    }

                    if (!deferredRawEvents.isEmpty()) {
                        // If we padded the number of raw events it's possible there are more than the caller actually
                        // requested.  Release the extra padded events now.
                        try {
                            unclaim(subscription, deferredRawEvents.values());
                        } catch (Exception e) {
                            // Don't fail, just log a warning.  The claims will eventually time out on their own.
                            _log.warn("Failed to unclaim {} events from subscription {}", deferredRawEvents.size(), subscription, e);
                        }
                    }

                    // Update the metric for the actual number of events returned
                    eventMeter.mark(limit - remaining);

                    return endOfData();
                }
            };

            events = Iterators.concat(toEvents(uniqueItems.values()).iterator(), deferredEvents);
            approximateSize = uniqueItems.size() + deferredRawEvents.size();
        }

        // Try draining the queue asynchronously if there are still more events available and more redundant events were
        // discarded than resolved items found so far.
        // Doing this only in the poll case for now.
        if (repeatable && eventsAvailableForNextPoll && itemsDiscarded > uniqueItems.size()) {
            drainQueueAsync(subscription);
        }

        return new PollResult(events, approximateSize, eventsAvailableForNextPoll);
    }

    /**
     * Resolves events found during a peek or poll and converts them into items.  No more than limit
     * events are read, not including events which are skipped because they come from dropped tables.
     *
     * Any events for content which has not yet been replicated to the local data center are excluded and set to retry
     * in RECENT_UNKNOWN_RETRY.  Any events for redundant changes are automatically deleted.
     *
     * To make use of this method more efficient it is not idempotent.  This method has the following side effect:
     *
     * 
    *
  1. All events processed are removed from rawEvents.
  2. *
* * Finally, this method returns the number of redundant events that were found and deleted, false otherwise. */ private int resolvePeekOrPollEvents(String subscription, Map rawEvents, int limit, ResolvedItemSink sink) { Map eventOrder = Maps.newHashMap(); List eventIdsToDiscard = Lists.newArrayList(); List recentUnknownEventIds = Lists.newArrayList(); int remaining = limit; int itemsDiscarded = 0; DataProvider.AnnotatedGet annotatedGet = _dataProvider.prepareGetAnnotated(ReadConsistency.STRONG); Iterator> rawEventIterator = rawEvents.entrySet().iterator(); while (rawEventIterator.hasNext() && remaining != 0) { Map.Entry entry = rawEventIterator.next(); Coordinate coord = entry.getKey(); // Query the table/key pair. try { annotatedGet.add(coord.getTable(), coord.getId()); remaining -= 1; } catch (UnknownTableException | UnknownPlacementException e) { // It's likely the table or facade was dropped since the event was queued. Discard the events. EventList list = entry.getValue(); for (Pair pair : list.getEventAndChangeIds()) { eventIdsToDiscard.add(pair.first()); } _discardedMeter.mark(list.size()); } // Keep track of the order in which we received the events from the EventStore. eventOrder.put(coord, eventOrder.size()); } Iterator readResultIter = annotatedGet.execute(); // Loop through the results of the data store query. while (readResultIter.hasNext()) { DataProvider.AnnotatedContent readResult = readResultIter.next(); // Get the JSON System of Record entity for this piece of content Map content = readResult.getContent(); // Find the original event IDs that correspond to this piece of content Coordinate coord = Coordinate.fromJson(content); EventList eventList = rawEvents.get(coord); // Get all databus event tags for the original event(s) for this coordinate List> tags = eventList.getTags(); Item item = null; // Loop over the Databus events for this piece of content. Usually there's just one, but not always... for (Pair eventData : eventList.getEventAndChangeIds()) { String eventId = eventData.first(); UUID changeId = eventData.second(); // Has the content replicated yet? If not, abandon the event and we'll try again when the claim expires. if (readResult.isChangeDeltaPending(changeId)) { if (isRecent(changeId)) { recentUnknownEventIds.add(eventId); _recentUnknownMeter.mark(); } else { _staleUnknownMeter.mark(); } continue; } // Is the change redundant? If so, no need to fire databus events for it. Ack it now. if (readResult.isChangeDeltaRedundant(changeId)) { eventIdsToDiscard.add(eventId); _redundantMeter.mark(); continue; } Item eventItem = new Item(eventId, eventOrder.get(coord), content, tags); if (item == null) { item = eventItem; } else if (item.consolidateWith(eventItem)) { _consolidatedMeter.mark(); } else { sink.accept(coord, item); item = eventItem; } } if (item != null) { sink.accept(coord, item); } } // Reduce the claim length on recent unknown IDs so we look for them again soon. if (!recentUnknownEventIds.isEmpty()) { _eventStore.renew(subscription, recentUnknownEventIds, RECENT_UNKNOWN_RETRY, false); } // Ack events we never again want to see. if ((itemsDiscarded = eventIdsToDiscard.size()) != 0) { _eventStore.delete(subscription, eventIdsToDiscard, true); } // Remove all coordinates from rawEvents which were processed by this method for (Coordinate coord : eventOrder.keySet()) { rawEvents.remove(coord); } return itemsDiscarded; } /** * Simple interface for the event sink in {@link #resolvePeekOrPollEvents(String, Map, int, ResolvedItemSink)} */ private interface ResolvedItemSink { void accept(Coordinate coordinate, Item item); } /** * Converts a collection of Items to Events. */ private List toEvents(Collection items) { if (items.isEmpty()) { return ImmutableList.of(); } // Sort the items to match the order of their events in an attempt to get first-in-first-out. return items.stream().sorted().map(Item::toEvent).collect(Collectors.toList()); } /** * Convenience method to unclaim all of the events from a collection of event lists. This is to unclaim excess * events when a padded poll returns more events than the requested limit. */ private void unclaim(String subscription, Collection eventLists) { List eventIdsToUnclaim = Lists.newArrayList(); for (EventList unclaimEvents : eventLists) { for (Pair eventAndChangeId : unclaimEvents.getEventAndChangeIds()) { eventIdsToUnclaim.add(eventAndChangeId.first()); } } _eventStore.renew(subscription, eventIdsToUnclaim, Duration.ZERO, false); } private boolean isRecent(UUID changeId) { return _clock.millis() - TimeUUIDs.getTimeMillis(changeId) < STALE_UNKNOWN_AGE.toMillis(); } @Override public void renew(String ownerId, String subscription, Collection eventKeys, Duration claimTtl) { checkLegalSubscriptionName(subscription); requireNonNull(eventKeys, "eventKeys"); checkArgument(claimTtl.compareTo(Duration.ZERO) >= 0, "ClaimTtl must be >=0"); checkSubscriptionOwner(ownerId, subscription); _eventStore.renew(subscription, EventKeyFormat.decodeAll(eventKeys), claimTtl, true); _renewedMeter.mark(eventKeys.size()); } @Override public void acknowledge(String ownerId, String subscription, Collection eventKeys) { checkLegalSubscriptionName(subscription); requireNonNull(eventKeys, "eventKeys"); checkSubscriptionOwner(ownerId, subscription); _eventStore.delete(subscription, EventKeyFormat.decodeAll(eventKeys), true); _ackedMeter.mark(eventKeys.size()); } @Override public String replayAsync(String ownerId, String subscription) { return replayAsyncSince(ownerId, subscription, null); } @Override public String replayAsyncSince(String ownerId, String subscription, Date since) { checkLegalSubscriptionName(subscription); checkSubscriptionOwner(ownerId, subscription); JobIdentifier jobId = _jobService.submitJob( new JobRequest<>(ReplaySubscriptionJob.INSTANCE, new ReplaySubscriptionRequest(ownerId, subscription, since))); return jobId.toString(); } public void replay(String subscription, Date since) { // Make sure since is within Replay TTL checkState(since == null || since.toInstant().plus(DatabusChannelConfiguration.REPLAY_TTL).isAfter(_clock.instant()), "Since timestamp is outside the replay TTL."); String source = ChannelNames.getMasterReplayChannel(); final OwnedSubscription destination = getSubscriptionByName(subscription); _eventStore.copy(source, subscription, (eventDataBytes) -> _subscriptionEvaluator.matches(destination, eventDataBytes, since), since); } @Override public ReplaySubscriptionStatus getReplayStatus(String ownerId, String reference) { requireNonNull(reference, "reference"); JobIdentifier jobId; try { jobId = JobIdentifier.fromString(reference, ReplaySubscriptionJob.INSTANCE); } catch (IllegalArgumentException e) { // The reference is illegal and therefore cannot match any replay jobs. throw new UnknownReplayException(reference); } JobStatus status = _jobService.getJobStatus(jobId); if (status == null) { throw new UnknownReplayException(reference); } ReplaySubscriptionRequest request = status.getRequest(); if (request == null) { throw new IllegalStateException("Replay request details not found: " + jobId); } checkSubscriptionOwner(ownerId, request.getSubscription()); switch (status.getStatus()) { case FINISHED: return new ReplaySubscriptionStatus(request.getSubscription(), ReplaySubscriptionStatus.Status.COMPLETE); case FAILED: return new ReplaySubscriptionStatus(request.getSubscription(), ReplaySubscriptionStatus.Status.ERROR); default: return new ReplaySubscriptionStatus(request.getSubscription(), ReplaySubscriptionStatus.Status.IN_PROGRESS); } } @Override public String moveAsync(String ownerId, String from, String to) { checkLegalSubscriptionName(from); checkLegalSubscriptionName(to); checkSubscriptionOwner(ownerId, from); checkSubscriptionOwner(ownerId, to); JobIdentifier jobId = _jobService.submitJob(new JobRequest<>( MoveSubscriptionJob.INSTANCE, new MoveSubscriptionRequest(ownerId, from, to))); return jobId.toString(); } @Override public MoveSubscriptionStatus getMoveStatus(String ownerId, String reference) { requireNonNull(reference, "reference"); JobIdentifier jobId; try { jobId = JobIdentifier.fromString(reference, MoveSubscriptionJob.INSTANCE); } catch (IllegalArgumentException e) { // The reference is illegal and therefore cannot match any move jobs. throw new UnknownMoveException(reference); } JobStatus status = _jobService.getJobStatus(jobId); if (status == null) { throw new UnknownMoveException(reference); } MoveSubscriptionRequest request = status.getRequest(); if (request == null) { throw new IllegalStateException("Move request details not found: " + jobId); } checkSubscriptionOwner(ownerId, request.getFrom()); switch (status.getStatus()) { case FINISHED: return new MoveSubscriptionStatus(request.getFrom(), request.getTo(), MoveSubscriptionStatus.Status.COMPLETE); case FAILED: return new MoveSubscriptionStatus(request.getFrom(), request.getTo(), MoveSubscriptionStatus.Status.ERROR); default: return new MoveSubscriptionStatus(request.getFrom(), request.getTo(), MoveSubscriptionStatus.Status.IN_PROGRESS); } } @Override public void injectEvent(String ownerId, String subscription, String table, String key) { // Pick a changeId UUID that's guaranteed to be older than the compaction cutoff so poll()'s calls to // AnnotatedContent.isChangeDeltaPending() and isChangeDeltaRedundant() will always return false. checkSubscriptionOwner(ownerId, subscription); UpdateRef ref = new UpdateRef(table, key, TimeUUIDs.minimumUuid(), ImmutableSet.of()); _eventStore.add(subscription, UpdateRefSerializer.toByteBuffer(ref)); } @Override public void unclaimAll(String ownerId, String subscription) { checkLegalSubscriptionName(subscription); checkSubscriptionOwner(ownerId, subscription); _eventStore.unclaimAll(subscription); } @Override public void purge(String ownerId, String subscription) { checkLegalSubscriptionName(subscription); checkSubscriptionOwner(ownerId, subscription); _eventStore.purge(subscription); } private void checkLegalSubscriptionName(String subscription) { checkArgument(Names.isLegalSubscriptionName(subscription), "Subscription name must be a lowercase ASCII string between 1 and 255 characters in length. " + "Allowed punctuation characters are -.:@_ and the subscription name may not start with a single underscore character. " + "An example of a valid subscription name would be 'polloi:review'."); } private void checkSubscriptionOwner(String ownerId, String subscription) { // Verify the subscription either doesn't exist or is already owned by the same owner. In practice this is // predominantly cached by SubscriptionDAO so performance should be good. checkSubscriptionOwner(ownerId, _subscriptionDao.getSubscription(subscription)); } private void checkSubscriptionOwner(String ownerId, OwnedSubscription subscription) { requireNonNull(ownerId, "ownerId"); if (subscription != null) { // Grandfather-in subscriptions created before ownership was introduced. This should be a temporary issue // since the subscriptions will need to renew at some point or expire. if (subscription.getOwnerId() == null) { _unownedSubscriptionMeter.mark(); } else if (!_databusAuthorizer.owner(ownerId).canAccessSubscription(subscription)) { throw new UnauthorizedSubscriptionException("Not subscriber", subscription.getName()); } } } @VisibleForTesting protected void drainQueueAsync(String subscription) { try { boolean notDrainingNow = _drainedSubscriptionsMap.putIfAbsent(subscription, 0L) == null; if (notDrainingNow) { _log.info("Starting the draining process for subscription: {}.", subscription); _drainQueueAsyncMeter.mark(); // submit a task to drain the queue. try { submitDrainServiceTask(subscription, MAX_ITEMS_TO_FETCH_FOR_QUEUE_DRAINING, null); } catch (Exception e) { // Failed to submit the task. This is unlikely, but just in case clear the draining marker _drainedSubscriptionsMap.remove(subscription); } } else { _log.debug("Draining for subscription: {} was already started from a previous poll.", subscription); } } catch (Exception e) { _log.error("Encountered exception while draining the queue for subscription: {}.", subscription, e); } } private void submitDrainServiceTask(String subscription, int itemsToFetch, @Nullable Cache knownNonRedundantEvents) { _drainQueueTaskMeter.mark(); _drainService.submit(new Runnable() { @Override public void run() { doDrainQueue(subscription, itemsToFetch, knownNonRedundantEvents != null ? knownNonRedundantEvents : CacheBuilder.newBuilder().maximumSize(1000).build()); } }); } private void doDrainQueue(String subscription, int itemsToFetch, Cache knownNonRedundantEvents) { boolean anyRedundantItemFound = false; Stopwatch stopwatch = Stopwatch.createStarted(_ticker); ConsolidatingEventSink sink = new ConsolidatingEventSink(itemsToFetch); boolean more = _eventStore.peek(subscription, sink); Map rawEvents = sink.getEvents(); if (rawEvents.isEmpty()) { _drainedSubscriptionsMap.remove(subscription); return; // queue is empty. } List eventIdsToDiscard = Lists.newArrayList(); // Query the events from the data store in batch to reduce latency. DataProvider.AnnotatedGet annotatedGet = _dataProvider.prepareGetAnnotated(ReadConsistency.STRONG); for (Map.Entry entry : rawEvents.entrySet()) { Coordinate coord = entry.getKey(); // If we've determined on a previous iteration that all change IDs returned for this coordinate are // not redundant then skip it. boolean anyUnverifiedEvents = entry.getValue().getEventAndChangeIds().stream() .map(Pair::first) .anyMatch(eventId -> knownNonRedundantEvents.getIfPresent(eventId) == null); if (anyUnverifiedEvents) { // Query the table/key pair. try { annotatedGet.add(coord.getTable(), coord.getId()); } catch (UnknownTableException | UnknownPlacementException e) { // It's likely the table or facade was dropped since the event was queued. Discard the events. EventList list = entry.getValue(); for (Pair pair : list.getEventAndChangeIds()) { eventIdsToDiscard.add(pair.first()); } } } } Iterator readResultIter = annotatedGet.execute(); // Loop through the results of the data store query. while (readResultIter.hasNext()) { DataProvider.AnnotatedContent readResult = readResultIter.next(); // Get the JSON System of Record entity for this piece of content Map content = readResult.getContent(); // Find the original event IDs that correspond to this piece of content Coordinate coord = Coordinate.fromJson(content); EventList eventList = rawEvents.get(coord); // Loop over the Databus events for this piece of content. Usually there's just one, but not always... for (Pair eventData : eventList.getEventAndChangeIds()) { String eventId = eventData.first(); UUID changeId = eventData.second(); // Is the change redundant? if (readResult.isChangeDeltaRedundant(changeId)) { anyRedundantItemFound = true; eventIdsToDiscard.add(eventId); } else { knownNonRedundantEvents.put(eventId, Boolean.TRUE); } } } // delete the events we never again want to see. if (!eventIdsToDiscard.isEmpty()) { _drainQueueRedundantMeter.mark(eventIdsToDiscard.size()); _eventStore.delete(subscription, eventIdsToDiscard, true); } long totalSubscriptionDrainTime = _drainedSubscriptionsMap.getOrDefault(subscription, 0L) + stopwatch.elapsed(TimeUnit.MILLISECONDS); // submit a new task for next batch if any the found items in this batch are redundant and there are more items available on the queue. // Also right now, we are only giving MAX_QUEUE_DRAIN_TIME_FOR_A_SUBSCRIPTION time for draining for a subscription for each poll. This is because there is no guarantee that the local server // remains as the owner of the subscription through out. The right solution here is to check if the local service still owns the subscription for each task submission. // But, MAX_QUEUE_DRAIN_TIME_FOR_A_SUBSCRIPTION may be OK as we can easily expect subsequent polls from the clients which will trigger these tasks again. if (anyRedundantItemFound && more && totalSubscriptionDrainTime < MAX_QUEUE_DRAIN_TIME_FOR_A_SUBSCRIPTION.toMillis()) { _drainedSubscriptionsMap.replace(subscription, totalSubscriptionDrainTime); submitDrainServiceTask(subscription, itemsToFetch, knownNonRedundantEvents); } else { _drainedSubscriptionsMap.remove(subscription); } } /** * EventStore sink that doesn't count adjacent events for the same table/key against the peek/poll limit. */ private class ConsolidatingEventSink implements EventSink { private final Map _eventMap = Maps.newLinkedHashMap(); private final int _limit; ConsolidatingEventSink(int limit) { _limit = limit; } @Override public int remaining() { // Go a bit past the desired limit to maximize opportunity to consolidate events. Otherwise, for example, // if limit was 1, we'd stop after the first event and not consolidate anything. return _limit - _eventMap.size() + 1; } @Override public Status accept(EventData rawEvent) { // Parse the raw event data into table/key/changeId/tags. UpdateRef ref = UpdateRefSerializer.fromByteBuffer(rawEvent.getData()); // Consolidate events that refer to the same item. Coordinate contentKey = Coordinate.of(ref.getTable(), ref.getKey()); EventList eventList = _eventMap.get(contentKey); if (eventList == null) { if (_eventMap.size() == _limit) { return Status.REJECTED_STOP; } _eventMap.put(contentKey, eventList = new EventList()); } eventList.add(rawEvent.getId(), ref.getChangeId(), ref.getTags()); if (eventList.size() == MAX_EVENTS_TO_CONSOLIDATE) { return Status.ACCEPTED_STOP; } return Status.ACCEPTED_CONTINUE; } Map getEvents() { return _eventMap; } } private static class EventList { private final List> _eventAndChangeIds = Lists.newArrayList(); private List> _tags; void add(String eventId, UUID changeId, Set tags) { _eventAndChangeIds.add(Pair.of(eventId, changeId)); _tags = sortedTagUnion(_tags, tags); } List> getEventAndChangeIds() { return _eventAndChangeIds; } List> getTags() { return _tags; } int size() { return _eventAndChangeIds.size(); } } private static class Item implements Comparable { private final List _consolidatedEventIds; private final int _sortIndex; private Map _content; private List> _tags; Item(String eventId, int sortIndex, Map content, List> tags) { _consolidatedEventIds = Lists.newArrayList(eventId); _sortIndex = sortIndex; _content = content; _tags = tags; } boolean consolidateWith(Item other) { if (_consolidatedEventIds.size() >= MAX_EVENTS_TO_CONSOLIDATE) { return false; } // We'll construct an event key that combines all the duplicate events. Unfortunately we can't discard/ack // the extra events at this time, subtle race conditions result that can cause clients to miss events. _consolidatedEventIds.addAll(other._consolidatedEventIds); // Pick the newest version of the content. There's no reason to return stale stuff. if (Intrinsic.getVersion(_content) < Intrinsic.getVersion(other._content)) { _content = other._content; } // Combine tags from the other event for (List tagList : other._tags) { // The contents of "tagList" are already sorted, no need to sort again. _tags = sortedTagUnion(_tags, tagList); } return true; } Event toEvent() { Collections.sort(_consolidatedEventIds); // Tags are already sorted return new Event(EventKeyFormat.encode(_consolidatedEventIds), _content, _tags); } @Override public int compareTo(Item item) { return Ints.compare(_sortIndex, item._sortIndex); } } // Conserve memory by having a singleton set to represent a set containing a single empty set of tags private final static List> EMPTY_TAGS = ImmutableList.>of(ImmutableList.of()); // Simple comparator for comparing lists of pre-sorted lists private final static Comparator> TAG_LIST_COMPARATOR = new Comparator>() { @Override public int compare(List o1, List o2) { int l1 = o1.size(); int l2 = o2.size(); int l = Math.min(l1, l2); for (int i = 0; i < l; i++) { int c = o1.get(i).compareTo(o2.get(i)); if (c != 0) { return c; } } // Shorter list sorts first return Integer.compare(l1, l2); } }; /** * Produces a list-of-lists for all unique tag sets for a given databus event. existingTags must either * be null or the result of a previous call to {@link #sortedTagUnion(java.util.List, java.util.List)} or * {@link #sortedTagUnion(java.util.List, java.util.Set)}. */ private static List> sortedTagUnion(@Nullable List> existingTags, Set newTagSet) { return sortedTagUnion(existingTags, asSortedList(newTagSet)); } /** * Produces a list-of-lists for all unique tag sets for a given databus event. existingTags must either * be null or the result of a previous call to {@link #sortedTagUnion(java.util.List, java.util.List)} or * {@link #sortedTagUnion(java.util.List, java.util.Set)}. Additionally the contents of sortedNewTagList * must already be sorted prior to this method call. */ private static List> sortedTagUnion(@Nullable List> existingTags, List sortedNewTagList) { // Optimize for the common case where for each coordinate there is exactly one event or that all events // use the same tags (typically the empty set) if (existingTags == null) { if (sortedNewTagList.isEmpty()) { return EMPTY_TAGS; } existingTags = Lists.newArrayListWithCapacity(3); existingTags.add(sortedNewTagList); return existingTags; } int insertionPoint = Collections.binarySearch(existingTags, sortedNewTagList, TAG_LIST_COMPARATOR); if (insertionPoint >= 0) { // Existing tags already includes this set of tags return existingTags; } if (existingTags == EMPTY_TAGS) { // Can't update the default empty tag set. Make a copy. existingTags = Lists.newArrayListWithCapacity(3); existingTags.addAll(EMPTY_TAGS); } insertionPoint = -insertionPoint - 1; existingTags.add(insertionPoint, sortedNewTagList); return existingTags; } private static List asSortedList(Set tags) { switch (tags.size()) { case 0: return ImmutableList.of(); case 1: return ImmutableList.of(tags.iterator().next()); default: return Ordering.natural().immutableSortedCopy(tags); } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy