aboutsummaryrefslogtreecommitdiffhomepage
path: root/src/main
diff options
context:
space:
mode:
authorGravatar Jon Brandvein <brandjon@google.com>2017-01-12 17:01:35 +0000
committerGravatar Marcel Hlopko <hlopko@google.com>2017-01-13 10:49:41 +0000
commit6e95b7d2d2e5d24a344e0b5654e0d38ed04ee701 (patch)
tree777d1d17a251336e55ddb0cdbb843ca738f788c1 /src/main
parentb5ceca9a561eb0e12438adaf91ff56fcb3a00f7e (diff)
Refactor NestedSetBuilder
Add Order#isCompatible, clean javadoc. Also fix broken @Deprecated-based canary for detecting accidental flattening. -- PiperOrigin-RevId: 144331341 MOS_MIGRATED_REVID=144331341
Diffstat (limited to 'src/main')
-rw-r--r--src/main/java/com/google/devtools/build/lib/collect/nestedset/NestedSetBuilder.java81
-rw-r--r--src/main/java/com/google/devtools/build/lib/collect/nestedset/Order.java10
2 files changed, 50 insertions, 41 deletions
diff --git a/src/main/java/com/google/devtools/build/lib/collect/nestedset/NestedSetBuilder.java b/src/main/java/com/google/devtools/build/lib/collect/nestedset/NestedSetBuilder.java
index b2a4f16678..be932dd549 100644
--- a/src/main/java/com/google/devtools/build/lib/collect/nestedset/NestedSetBuilder.java
+++ b/src/main/java/com/google/devtools/build/lib/collect/nestedset/NestedSetBuilder.java
@@ -14,7 +14,6 @@
package com.google.devtools.build.lib.collect.nestedset;
import static com.google.common.collect.Iterables.getOnlyElement;
-import static com.google.common.collect.Iterables.isEmpty;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.Iterables;
@@ -29,7 +28,8 @@ import java.util.concurrent.ConcurrentMap;
*
* <p>The builder supports the standard builder interface (that is, {@code #add}, {@code #addAll}
* and {@code #addTransitive} followed by {@code build}), in addition to shortcut method
- * {@code #wrap}.
+ * {@code #wrap}. Any duplicate elements will be inserted as-is, and pruned later on during the
+ * traversal of the actual NestedSet.
*/
public final class NestedSetBuilder<E> {
@@ -47,19 +47,16 @@ public final class NestedSetBuilder<E> {
}
/**
- * Add an element.
+ * Adds a direct member to the set to be built.
*
- * <p>Preserves ordering of added elements. Discards duplicate values.
- * Throws an exception if a null value is passed in.
+ * <p>The relative left-to-right order of direct members is preserved from the sequence of calls
+ * to {@link #add} and {@link #addAll}. Since the traversal {@link Order} controls whether direct
+ * members appear before or after transitive ones, the interleaving of
+ * {@link #add}/{@link #addAll} with {@link #addTransitive} does not matter.
*
- * <p>The collections of the direct members of the set and the nested sets are
- * kept separate, so the order between multiple add/addAll calls matters,
- * and the order between multiple addTransitive calls matters, but the order
- * between add/addAll and addTransitive does not.
- *
- * @return the builder.
+ * @param element item to add; must not be null
+ * @return the builder
*/
- @SuppressWarnings("unchecked") // B is the type of the concrete subclass
public NestedSetBuilder<E> add(E element) {
Preconditions.checkNotNull(element);
items.add(element);
@@ -67,18 +64,17 @@ public final class NestedSetBuilder<E> {
}
/**
- * Adds a collection of elements to the set.
- *
- * <p>This is equivalent to invoking {@code add} for every item of the collection in iteration
- * order.
+ * Adds a sequence of direct members to the set to be built. Equivalent to invoking {@link #add}
+ * for each item in {@code elements}, in order.
*
- * <p>The collections of the direct members of the set and the nested sets are kept separate, so
- * the order between multiple add/addAll calls matters, and the order between multiple
- * addTransitive calls matters, but the order between add/addAll and addTransitive does not.
+ * <p>The relative left-to-right order of direct members is preserved from the sequence of calls
+ * to {@link #add} and {@link #addAll}. Since the traversal {@link Order} controls whether direct
+ * members appear before or after transitive ones, the interleaving of
+ * {@link #add}/{@link #addAll} with {@link #addTransitive} does not matter.
*
- * @return the builder.
+ * @param elements the sequence of items to add; must not be null
+ * @return the builder
*/
- @SuppressWarnings("unchecked") // B is the type of the concrete subclass
public NestedSetBuilder<E> addAll(Iterable<? extends E> elements) {
Preconditions.checkNotNull(elements);
Iterables.addAll(items, elements);
@@ -89,40 +85,43 @@ public final class NestedSetBuilder<E> {
* @deprecated Use {@link #addTransitive} to avoid excessive memory use.
*/
@Deprecated
- public NestedSetBuilder<E> addAll(NestedSet<E> elements) {
+ public NestedSetBuilder<E> addAll(NestedSet<? extends E> elements) {
// Do not delete this method, or else addAll(Iterable) calls with a NestedSet argument
// will not be flagged.
- Iterable<E> it = elements;
+ Iterable<? extends E> it = elements;
addAll(it);
return this;
}
/**
- * Adds another nested set to this set.
+ * Adds a nested set as a transitive member to the set to be built.
*
- * <p>Preserves ordering of added nested sets. Discards duplicate values. Throws an exception if
- * a null value is passed in.
+ * <p>The relative left-to-right order of transitive members is preserved from the sequence of
+ * calls to {@link #addTransitive}. Since the traversal {@link Order} controls whether direct
+ * members appear before or after transitive ones, the interleaving of
+ * {@link #add}/{@link #addAll} with {@link #addTransitive} does not matter.
*
- * <p>The collections of the direct members of the set and the nested sets are kept separate, so
- * the order between multiple add/addAll calls matters, and the order between multiple
- * addTransitive calls matters, but the order between add/addAll and addTransitive does not.
+ * <p>The {@link Order} of the added set must be compatible with the order of this builder (see
+ * {@link Order#isCompatible}). This is true even if the added set is empty. Strictly speaking, it
+ * is not technically necessary that two nested sets have compatible orders for them to be
+ * combined as part of one larger set. But checking for it helps readability and protects against
+ * bugs. Since {@link Order#STABLE_ORDER} is compatible with everything, it effectively disables
+ * the check. This can be used as an escape hatch to mix and match the set arbitrarily, including
+ * sharing the set as part of multiple other larger sets that have disagreeing orders.
*
- * <p>An error will be thrown if the ordering of {@code subset} is incompatible with the ordering
- * of this set. Either they must match or one must be a {@code STABLE_ORDER} set.
+ * <p>The relative order of the elements of an added set are preserved, unless it has duplicates
+ * or overlaps with other added sets, or its order is different from that of the builder.
*
- * @return the builder.
+ * @param subset the set to add as a transitive member; must not be null
+ * @return the builder
+ * @throws IllegalStateException if the order of {@code subset} is not compatible with the
+ * order of this builder
*/
public NestedSetBuilder<E> addTransitive(NestedSet<? extends E> subset) {
Preconditions.checkNotNull(subset);
- if (subset.getOrder() != order && order != Order.STABLE_ORDER
- && subset.getOrder() != Order.STABLE_ORDER) {
- // Note that this check is not strictly necessary, although keeping the nested set types
- // consistent helps readability and protects against bugs. The polymorphism regarding
- // STABLE_ORDER is allowed in order to be able to, e.g., include an arbitrary nested set in
- // the inputs of an action, or include a nested set that is indifferent to its order in
- // multiple nested sets.
- throw new IllegalStateException(subset.getOrder() + " != " + order);
- }
+ Preconditions.checkArgument(
+ order.isCompatible(subset.getOrder()),
+ "Order mismatch: %s != %s", subset.getOrder(), order);
if (!subset.isEmpty()) {
transitiveSets.add(subset);
}
diff --git a/src/main/java/com/google/devtools/build/lib/collect/nestedset/Order.java b/src/main/java/com/google/devtools/build/lib/collect/nestedset/Order.java
index 3b207c7f13..c40546e9a3 100644
--- a/src/main/java/com/google/devtools/build/lib/collect/nestedset/Order.java
+++ b/src/main/java/com/google/devtools/build/lib/collect/nestedset/Order.java
@@ -146,6 +146,16 @@ public enum Order {
}
/**
+ * Determines whether two orders are considered compatible.
+ *
+ * <p>An order is compatible with itself (reflexivity) and all orders are compatible with
+ * {@link #STABLE_ORDER}; the rest of the combinations are incompatible.
+ */
+ public boolean isCompatible(Order other) {
+ return this == other || this == STABLE_ORDER || other == STABLE_ORDER;
+ }
+
+ /**
* Indexes all possible values by name and stores the results in a {@code ImmutableMap}
*/
static {