org.apache.aurora.scheduler.base.Numbers.java Source code

Java tutorial

Introduction

Here is the source code for org.apache.aurora.scheduler.base.Numbers.java

Source

/**
 * Licensed 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.aurora.scheduler.base;

import java.util.Set;

import com.google.common.base.Function;
import com.google.common.collect.ContiguousSet;
import com.google.common.collect.DiscreteDomain;
import com.google.common.collect.ImmutableRangeSet;
import com.google.common.collect.ImmutableSet;
import com.google.common.collect.Iterators;
import com.google.common.collect.PeekingIterator;
import com.google.common.collect.Range;
import com.google.common.collect.Sets;

import org.apache.aurora.GuavaUtils;
import org.apache.aurora.scheduler.storage.entities.IRange;
import org.apache.mesos.v1.Protos;

/**
 * Utility class for working with numbers.
 */
public final class Numbers {
    /**
     * Convert {@link com.google.common.collect.Range} to {@link org.apache.mesos.Protos.Value.Range}.
     */
    public static final Function<Range<Integer>, Protos.Value.Range> RANGE_TRANSFORM = input -> Protos.Value.Range
            .newBuilder().setBegin(input.lowerEndpoint()).setEnd(input.upperEndpoint()).build();

    /**
     * Convert {@link org.apache.mesos.v1.Protos.Value.Range} to set of integers.
     */
    public static final Function<Protos.Value.Range, Set<Integer>> RANGE_TO_MEMBERS = range -> ContiguousSet
            .create(Range.closed((int) range.getBegin(), (int) range.getEnd()), DiscreteDomain.integers());

    private Numbers() {
        // Utility class.
    }

    /**
     * Converts a set of integers into a set of contiguous closed ranges that equally represent the
     * input integers.
     * <p>
     * The resulting ranges will be in ascending order.
     * <p>
     * TODO(wfarner): Change this to return a canonicalized RangeSet.
     *
     * @param values Values to transform to ranges.
     * @return Closed ranges with identical members to the input set.
     */
    public static Set<Range<Integer>> toRanges(Iterable<Integer> values) {
        ImmutableSet.Builder<Range<Integer>> builder = ImmutableSet.builder();

        PeekingIterator<Integer> iterator = Iterators.peekingIterator(Sets.newTreeSet(values).iterator());

        // Build ranges until there are no numbers left.
        while (iterator.hasNext()) {
            // Start a new range.
            int start = iterator.next();
            int end = start;
            // Increment the end until the range is non-contiguous.
            while (iterator.hasNext() && iterator.peek() == end + 1) {
                end++;
                iterator.next();
            }

            builder.add(Range.closed(start, end));
        }

        return builder.build();
    }

    /**
     * Convert between range types.
     *
     * @param range Range to convert.
     * @return A closed range from the first to last of {@code range}.
     */
    public static Range<Integer> toRange(IRange range) {
        return Range.closed(range.getFirst(), range.getLast());
    }

    /**
     * Performs {@link #toRange(IRange)} for a collection of ranges, and convert the result to a set
     * of integers.
     *
     * @param ranges Ranges to convert.
     * @return A set representing {@code ranges}.
     */
    public static Set<Integer> rangesToInstanceIds(Iterable<IRange> ranges) {
        ImmutableRangeSet.Builder<Integer> instanceIds = ImmutableRangeSet.builder();
        for (IRange range : ranges) {
            instanceIds.add(toRange(range));
        }

        return instanceIds.build().asSet(DiscreteDomain.integers());
    }

    /**
     * Converts set of instance ranges to a set of {@link IRange}.
     *
     * @param ranges Instance ranges to convert.
     * @return A set of {@link IRange}.
     */
    public static Set<IRange> convertRanges(Set<Range<Integer>> ranges) {
        return ranges.stream()
                .map(range -> IRange
                        .build(new org.apache.aurora.gen.Range(range.lowerEndpoint(), range.upperEndpoint())))
                .collect(GuavaUtils.toImmutableSet());
    }
}