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

org.apache.cassandra.locator.EndpointsForRange Maven / Gradle / Ivy

Go to download

The Apache Cassandra Project develops a highly scalable second-generation distributed database, bringing together Dynamo's fully distributed design and Bigtable's ColumnFamily-based data model.

There is a newer version: 5.0.2
Show newest version
/*
 * 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.cassandra.locator;

import com.google.common.base.Preconditions;
import org.apache.cassandra.dht.Range;
import org.apache.cassandra.dht.Token;

import java.util.Arrays;
import java.util.Collection;

import static com.google.common.collect.Iterables.all;

/**
 * A ReplicaCollection where all Replica are required to cover a range that fully contains the range() defined in the builder().
 * Endpoints are guaranteed to be unique; on construction, this is enforced unless optionally silenced (in which case
 * only the first occurrence makes the cut).
 */
public class EndpointsForRange extends Endpoints
{
    private final Range range;
    private EndpointsForRange(Range range, ReplicaList list, ReplicaMap byEndpoint)
    {
        super(list, byEndpoint);
        this.range = range;
        assert range != null;
    }

    public Range range()
    {
        return range;
    }

    @Override
    public Builder newBuilder(int initialCapacity)
    {
        return new Builder(range, initialCapacity);
    }

    public EndpointsForToken forToken(Token token)
    {
        if (!range.contains(token))
            throw new IllegalArgumentException(token + " is not contained within " + range);
        return new EndpointsForToken(token, list, byEndpoint);
    }

    @Override
    public EndpointsForRange snapshot()
    {
        return this;
    }

    @Override
    EndpointsForRange snapshot(ReplicaList newList)
    {
        if (newList.isEmpty()) return empty(range);
        ReplicaMap byEndpoint = null;
        if (this.byEndpoint != null && list.isSubList(newList))
            byEndpoint = this.byEndpoint.forSubList(newList);
        return new EndpointsForRange(range, newList, byEndpoint);
    }

    public static class Builder extends EndpointsForRange implements ReplicaCollection.Builder
    {
        boolean built;
        public Builder(Range range) { this(range, 0); }
        public Builder(Range range, int capacity) { this(range, new ReplicaList(capacity)); }
        private Builder(Range range, ReplicaList list) { super(range, list, endpointMap(list)); }

        public EndpointsForRange.Builder add(Replica replica, Conflict ignoreConflict)
        {
            if (built) throw new IllegalStateException();
            Preconditions.checkNotNull(replica);
            if (!replica.range().contains(super.range))
                throw new IllegalArgumentException("Replica " + replica + " does not contain " + super.range);

            if (!super.byEndpoint.internalPutIfAbsent(replica, list.size()))
            {
                switch (ignoreConflict)
                {
                    case DUPLICATE:
                        if (byEndpoint().get(replica.endpoint()).equals(replica))
                            break;
                    case NONE:
                        throw new IllegalArgumentException("Conflicting replica added (expected unique endpoints): "
                                + replica + "; existing: " + byEndpoint().get(replica.endpoint()));
                    case ALL:
                }
                return this;
            }

            list.add(replica);
            return this;
        }

        @Override
        public EndpointsForRange snapshot()
        {
            return snapshot(list.subList(0, list.size()));
        }

        public EndpointsForRange build()
        {
            built = true;
            return new EndpointsForRange(super.range, super.list, super.byEndpoint);
        }
    }

    public static Builder builder(Range range)
    {
        return new Builder(range);
    }
    public static Builder builder(Range range, int capacity)
    {
        return new Builder(range, capacity);
    }

    public static EndpointsForRange empty(Range range)
    {
        return new EndpointsForRange(range, EMPTY_LIST, EMPTY_MAP);
    }

    public static EndpointsForRange of(Replica replica)
    {
        // we only use ArrayList or ArrayList.SubList, to ensure callsites are bimorphic
        ReplicaList one = new ReplicaList(1);
        one.add(replica);
        // we can safely use singletonMap, as we only otherwise use LinkedHashMap
        return new EndpointsForRange(replica.range(), one, endpointMap(one));
    }

    public static EndpointsForRange of(Replica ... replicas)
    {
        return copyOf(Arrays.asList(replicas));
    }

    public static EndpointsForRange copyOf(Collection replicas)
    {
        if (replicas.isEmpty())
            throw new IllegalArgumentException("Collection must be non-empty to copy");
        Range range = replicas.iterator().next().range();
        assert all(replicas, r -> range.equals(r.range()));
        return builder(range, replicas.size()).addAll(replicas).build();
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy