/** * 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()); } }