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

org.apache.cassandra.dht.Range 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.dht;

import java.io.Serializable;
import java.util.*;
import java.util.function.Predicate;

import com.google.common.collect.Iterables;
import org.apache.commons.lang3.ObjectUtils;

import org.apache.cassandra.db.PartitionPosition;
import org.apache.cassandra.utils.Pair;

/**
 * A representation of the range that a node is responsible for on the DHT ring.
 *
 * A Range is responsible for the tokens between (left, right].
 *
 * Used by the partitioner and by map/reduce by-token range scans.
 *
 * Note: this class has a natural ordering that is inconsistent with equals
 */
public class Range> extends AbstractBounds implements Comparable>, Serializable
{
    public static final long serialVersionUID = 1L;

    public Range(T left, T right)
    {
        super(left, right);
    }

    public static > boolean contains(T left, T right, T point)
    {
        if (isWrapAround(left, right))
        {
            /*
             * We are wrapping around, so the interval is (a,b] where a >= b,
             * then we have 3 cases which hold for any given token k:
             * (1) a < k -- return true
             * (2) k <= b -- return true
             * (3) b < k <= a -- return false
             */
            if (point.compareTo(left) > 0)
                return true;
            else
                return right.compareTo(point) >= 0;
        }
        else
        {
            /*
             * This is the range (a, b] where a < b.
             */
            return point.compareTo(left) > 0 && right.compareTo(point) >= 0;
        }
    }

    public boolean contains(Range that)
    {
        if (this.left.equals(this.right))
        {
            // full ring always contains all other ranges
            return true;
        }

        boolean thiswraps = isWrapAround(left, right);
        boolean thatwraps = isWrapAround(that.left, that.right);
        if (thiswraps == thatwraps)
        {
            return left.compareTo(that.left) <= 0 && that.right.compareTo(right) <= 0;
        }
        else if (thiswraps)
        {
            // wrapping might contain non-wrapping
            // that is contained if both its tokens are in one of our wrap segments
            return left.compareTo(that.left) <= 0 || that.right.compareTo(right) <= 0;
        }
        else
        {
            // (thatwraps)
            // non-wrapping cannot contain wrapping
            return false;
        }
    }

    /**
     * Helps determine if a given point on the DHT ring is contained
     * in the range in question.
     * @param point point in question
     * @return true if the point contains within the range else false.
     */
    public boolean contains(T point)
    {
        return contains(left, right, point);
    }

    /**
     * @param that range to check for intersection
     * @return true if the given range intersects with this range.
     */
    public boolean intersects(Range that)
    {
        return intersectionWith(that).size() > 0;
    }

    public boolean intersects(AbstractBounds that)
    {
        // implemented for cleanup compaction membership test, so only Range + Bounds are supported for now
        if (that instanceof Range)
            return intersects((Range) that);
        if (that instanceof Bounds)
            return intersects((Bounds) that);
        throw new UnsupportedOperationException("Intersection is only supported for Bounds and Range objects; found " + that.getClass());
    }

    /**
     * @param that range to check for intersection
     * @return true if the given range intersects with this range.
     */
    public boolean intersects(Bounds that)
    {
        // Same punishment than in Bounds.contains(), we must be carefull if that.left == that.right as
        // as new Range(that.left, that.right) will then cover the full ring which is not what we
        // want.
        return contains(that.left) || (!that.left.equals(that.right) && intersects(new Range(that.left, that.right)));
    }

    public static boolean intersects(Iterable> l, Iterable> r)
    {
        return Iterables.any(l, rng -> rng.intersects(r));
    }

    @SafeVarargs
    public static > Set> rangeSet(Range ... ranges)
    {
        return Collections.unmodifiableSet(new HashSet>(Arrays.asList(ranges)));
    }

    public static > Set> rangeSet(Range range)
    {
        return Collections.singleton(range);
    }

    /**
     * @param that
     * @return the intersection of the two Ranges.  this can be two disjoint Ranges if one is wrapping and one is not.
     * say you have nodes G and M, with query range (D,T]; the intersection is (M-T] and (D-G].
     * If there is no intersection, an empty list is returned.
     */
    public Set> intersectionWith(Range that)
    {
        if (that.contains(this))
            return rangeSet(this);
        if (this.contains(that))
            return rangeSet(that);

        boolean thiswraps = isWrapAround(left, right);
        boolean thatwraps = isWrapAround(that.left, that.right);
        if (!thiswraps && !thatwraps)
        {
            // neither wraps:  the straightforward case.
            if (!(left.compareTo(that.right) < 0 && that.left.compareTo(right) < 0))
                return Collections.emptySet();
            return rangeSet(new Range(ObjectUtils.max(this.left, that.left),
                                         ObjectUtils.min(this.right, that.right)));
        }
        if (thiswraps && thatwraps)
        {
            //both wrap: if the starts are the same, one contains the other, which we have already ruled out.
            assert !this.left.equals(that.left);
            // two wrapping ranges always intersect.
            // since we have already determined that neither this nor that contains the other, we have 2 cases,
            // and mirror images of those case.
            // (1) both of that's (1, 2] endpoints lie in this's (A, B] right segment:
            //  ---------B--------A--1----2------>
            // (2) only that's start endpoint lies in this's right segment:
            //  ---------B----1---A-------2------>
            // or, we have the same cases on the left segement, which we can handle by swapping this and that.
            return this.left.compareTo(that.left) < 0
                   ? intersectionBothWrapping(this, that)
                   : intersectionBothWrapping(that, this);
        }
        if (thiswraps) // this wraps, that does not wrap
            return intersectionOneWrapping(this, that);
        // the last case: this does not wrap, that wraps
        return intersectionOneWrapping(that, this);
    }

    private static > Set> intersectionBothWrapping(Range first, Range that)
    {
        Set> intersection = new HashSet>(2);
        if (that.right.compareTo(first.left) > 0)
            intersection.add(new Range(first.left, that.right));
        intersection.add(new Range(that.left, first.right));
        return Collections.unmodifiableSet(intersection);
    }

    private static > Set> intersectionOneWrapping(Range wrapping, Range other)
    {
        Set> intersection = new HashSet>(2);
        if (other.contains(wrapping.right))
            intersection.add(new Range(other.left, wrapping.right));
        // need the extra compareto here because ranges are asymmetrical; wrapping.left _is not_ contained by the wrapping range
        if (other.contains(wrapping.left) && wrapping.left.compareTo(other.right) < 0)
            intersection.add(new Range(wrapping.left, other.right));
        return Collections.unmodifiableSet(intersection);
    }

    /**
     * Returns the intersection of this range with the provided one, assuming neither are wrapping.
     *
     * @param that the other range to return the intersection with. It must not be wrapping.
     * @return the intersection of {@code this} and {@code that}, or {@code null} if both ranges don't intersect.
     */
    public Range intersectionNonWrapping(Range that)
    {
        assert !isTrulyWrapAround() : "wraparound " + this;
        assert !that.isTrulyWrapAround() : "wraparound " + that;

        if (left.compareTo(that.left) < 0)
        {
            if (right.isMinimum() || (!that.right.isMinimum() && right.compareTo(that.right) >= 0))
                return that;  // this contains that.

            if (right.compareTo(that.left) <= 0)
                return null;  // this is fully before that.

            return new Range<>(that.left, right);
        }
        else
        {
            if (that.right.isMinimum() || (!right.isMinimum() && that.right.compareTo(right) >= 0))
                return this;  // that contains this.

            if (that.right.compareTo(left) <= 0)
                return null;  // that is fully before this.

            return new Range<>(left, that.right);
        }
    }

    public Pair, AbstractBounds> split(T position)
    {
        assert contains(position) || left.equals(position);
        // Check if the split would have no effect on the range
        if (position.equals(left) || position.equals(right))
            return null;

        AbstractBounds lb = new Range(left, position);
        AbstractBounds rb = new Range(position, right);
        return Pair.create(lb, rb);
    }

    public boolean inclusiveLeft()
    {
        return false;
    }

    public boolean inclusiveRight()
    {
        return true;
    }

    public List> unwrap()
    {
        T minValue = right.minValue();
        if (!isWrapAround() || right.equals(minValue))
            return Arrays.asList(this);
        List> unwrapped = new ArrayList>(2);
        unwrapped.add(new Range(left, minValue));
        unwrapped.add(new Range(minValue, right));
        return unwrapped;
    }

    /**
     * Tells if the given range is a wrap around.
     */
    public static > boolean isWrapAround(T left, T right)
    {
       return left.compareTo(right) >= 0;
    }

    /**
     * Checks if the range truly wraps around.
     *
     * This exists only because {@link #isWrapAround()} is a tad dumb and return true if right is the minimum token,
     * no matter what left is, but for most intent and purposes, such range doesn't truly warp around (unwrap produces
     * the identity in this case).
     * 

* Also note that it could be that the remaining uses of {@link #isWrapAround()} could be replaced by this method, * but that is to be checked carefully at some other time (Sylvain). *

* The one thing this method guarantees is that if it's true, then {@link #unwrap()} will return a list with * exactly 2 ranges, never one. */ public boolean isTrulyWrapAround() { return isTrulyWrapAround(left, right); } public static > boolean isTrulyWrapAround(T left, T right) { return isWrapAround(left, right) && !right.isMinimum(); } /** * Tells if the given range covers the entire ring */ private static > boolean isFull(T left, T right) { return left.equals(right); } /** * Note: this class has a natural ordering that is inconsistent with equals */ public int compareTo(Range rhs) { boolean lhsWrap = isWrapAround(left, right); boolean rhsWrap = isWrapAround(rhs.left, rhs.right); // if one of the two wraps, that's the smaller one. if (lhsWrap != rhsWrap) return Boolean.compare(!lhsWrap, !rhsWrap); // otherwise compare by right. return right.compareTo(rhs.right); } /** * Subtracts a portion of this range. * @param contained The range to subtract from this. It must be totally * contained by this range. * @return A List of the Ranges left after subtracting contained * from this. */ private List> subtractContained(Range contained) { // both ranges cover the entire ring, their difference is an empty set if(isFull(left, right) && isFull(contained.left, contained.right)) { return Collections.emptyList(); } // a range is subtracted from another range that covers the entire ring if(isFull(left, right)) { return Collections.singletonList(new Range<>(contained.right, contained.left)); } List> difference = new ArrayList<>(2); if (!left.equals(contained.left)) difference.add(new Range(left, contained.left)); if (!right.equals(contained.right)) difference.add(new Range(contained.right, right)); return difference; } public Set> subtract(Range rhs) { return rhs.differenceToFetch(this); } public Set> subtractAll(Collection> ranges) { Set> result = new HashSet<>(); result.add(this); for(Range range : ranges) { result = substractAllFromToken(result, range); } return result; } private static > Set> substractAllFromToken(Set> ranges, Range subtract) { Set> result = new HashSet<>(); for(Range range : ranges) { result.addAll(range.subtract(subtract)); } return result; } public static > Set> subtract(Collection> ranges, Collection> subtract) { Set> result = new HashSet<>(); for (Range range : ranges) { result.addAll(range.subtractAll(subtract)); } return result; } /** * Calculate set of the difference ranges of given two ranges * (as current (A, B] and rhs is (C, D]) * which node will need to fetch when moving to a given new token * * @param rhs range to calculate difference * @return set of difference ranges */ public Set> differenceToFetch(Range rhs) { Set> result; Set> intersectionSet = this.intersectionWith(rhs); if (intersectionSet.isEmpty()) { result = new HashSet>(); result.add(rhs); } else { @SuppressWarnings("unchecked") Range[] intersections = new Range[intersectionSet.size()]; intersectionSet.toArray(intersections); if (intersections.length == 1) { result = new HashSet>(rhs.subtractContained(intersections[0])); } else { // intersections.length must be 2 Range first = intersections[0]; Range second = intersections[1]; List> temp = rhs.subtractContained(first); // Because there are two intersections, subtracting only one of them // will yield a single Range. Range single = temp.get(0); result = new HashSet>(single.subtractContained(second)); } } return result; } public static > boolean isInRanges(T token, Iterable> ranges) { assert ranges != null; for (Range range : ranges) { if (range.contains(token)) { return true; } } return false; } @Override public boolean equals(Object o) { if (!(o instanceof Range)) return false; Range rhs = (Range)o; return left.equals(rhs.left) && right.equals(rhs.right); } @Override public String toString() { return "(" + left + "," + right + "]"; } protected String getOpeningString() { return "("; } protected String getClosingString() { return "]"; } public boolean isStartInclusive() { return false; } public boolean isEndInclusive() { return true; } public List asList() { ArrayList ret = new ArrayList(2); ret.add(left.toString()); ret.add(right.toString()); return ret; } public boolean isWrapAround() { return isWrapAround(left, right); } /** * @return A copy of the given list of with all ranges unwrapped, sorted by left bound and with overlapping bounds merged. */ public static > List> normalize(Collection> ranges) { // unwrap all List> output = new ArrayList>(ranges.size()); for (Range range : ranges) output.addAll(range.unwrap()); // sort by left Collections.sort(output, new Comparator>() { public int compare(Range b1, Range b2) { return b1.left.compareTo(b2.left); } }); // deoverlap return deoverlap(output); } /** * Given a list of unwrapped ranges sorted by left position, return an * equivalent list of ranges but with no overlapping ranges. */ public static > List> deoverlap(List> ranges) { if (ranges.isEmpty()) return ranges; List> output = new ArrayList>(); Iterator> iter = ranges.iterator(); Range current = iter.next(); T min = current.left.minValue(); while (iter.hasNext()) { // If current goes to the end of the ring, we're done if (current.right.equals(min)) { // If one range is the full range, we return only that if (current.left.equals(min)) return Collections.>singletonList(current); output.add(new Range(current.left, min)); return output; } Range next = iter.next(); // if next left is equal to current right, we do not intersect per se, but replacing (A, B] and (B, C] by (A, C] is // legit, and since this avoid special casing and will result in more "optimal" ranges, we do the transformation if (next.left.compareTo(current.right) <= 0) { // We do overlap // (we've handled current.right.equals(min) already) if (next.right.equals(min) || current.right.compareTo(next.right) < 0) current = new Range(current.left, next.right); } else { output.add(current); current = next; } } output.add(current); return output; } public AbstractBounds withNewRight(T newRight) { return new Range(left, newRight); } public static > List> sort(Collection> ranges) { List> output = new ArrayList<>(ranges.size()); for (Range r : ranges) output.addAll(r.unwrap()); // sort by left Collections.sort(output, new Comparator>() { public int compare(Range b1, Range b2) { return b1.left.compareTo(b2.left); } }); return output; } /** * Compute a range of keys corresponding to a given range of token. */ public static Range makeRowRange(Token left, Token right) { return new Range(left.maxKeyBound(), right.maxKeyBound()); } public static Range makeRowRange(Range tokenBounds) { return makeRowRange(tokenBounds.left, tokenBounds.right); } /** * Helper class to check if a token is contained within a given collection of ranges */ public static class OrderedRangeContainmentChecker implements Predicate { private final Iterator> normalizedRangesIterator; private Token lastToken = null; private Range currentRange; public OrderedRangeContainmentChecker(Collection> ranges) { normalizedRangesIterator = normalize(ranges).iterator(); assert normalizedRangesIterator.hasNext(); currentRange = normalizedRangesIterator.next(); } /** * Returns true if the ranges given in the constructor contains the token, false otherwise. * * The tokens passed to this method must be in increasing order * * @param t token to check, must be larger than or equal to the last token passed * @return true if the token is contained within the ranges given to the constructor. */ @Override public boolean test(Token t) { assert lastToken == null || lastToken.compareTo(t) <= 0; lastToken = t; while (true) { if (t.compareTo(currentRange.left) <= 0) return false; else if (t.compareTo(currentRange.right) <= 0 || currentRange.right.compareTo(currentRange.left) <= 0) return true; if (!normalizedRangesIterator.hasNext()) return false; currentRange = normalizedRangesIterator.next(); } } } public static > void assertNormalized(List> ranges) { Range lastRange = null; for (Range range : ranges) { if (lastRange == null) { lastRange = range; } else if (lastRange.left.compareTo(range.left) >= 0 || lastRange.intersects(range)) { throw new AssertionError(String.format("Ranges aren't properly normalized. lastRange %s, range %s, compareTo %d, intersects %b, all ranges %s%n", lastRange, range, lastRange.compareTo(range), lastRange.intersects(range), ranges)); } } } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy