Refactor NestedSetBuilder
Add Order#isCompatible, clean javadoc. Also fix broken @Deprecated-based canary for detecting accidental flattening.
--
PiperOrigin-RevId: 144331341
MOS_MIGRATED_REVID=144331341
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 b2a4f16..be932dd 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 @@
*
* <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 @@
}
/**
- * 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 @@
}
/**
- * Adds a collection of elements to the set.
+ * 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>This is equivalent to invoking {@code add} for every item of the collection in iteration
- * order.
+ * <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 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 @@
* @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 3b207c7..c40546e 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 @@
}
/**
+ * 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 {
diff --git a/src/test/java/com/google/devtools/build/lib/collect/nestedset/ExpanderTestBase.java b/src/test/java/com/google/devtools/build/lib/collect/nestedset/ExpanderTestBase.java
index 8877f71..927d324 100644
--- a/src/test/java/com/google/devtools/build/lib/collect/nestedset/ExpanderTestBase.java
+++ b/src/test/java/com/google/devtools/build/lib/collect/nestedset/ExpanderTestBase.java
@@ -213,7 +213,7 @@
if (ordering != expanderOrder() && ordering != Order.STABLE_ORDER) {
fail(); // An exception was expected.
}
- } catch (IllegalStateException e) {
+ } catch (IllegalArgumentException e) {
if (ordering == expanderOrder() || ordering == Order.STABLE_ORDER) {
fail(); // No exception was expected.
}
diff --git a/src/test/java/com/google/devtools/build/lib/collect/nestedset/NestedSetImplTest.java b/src/test/java/com/google/devtools/build/lib/collect/nestedset/NestedSetImplTest.java
index b72f89c..a5eb38a 100644
--- a/src/test/java/com/google/devtools/build/lib/collect/nestedset/NestedSetImplTest.java
+++ b/src/test/java/com/google/devtools/build/lib/collect/nestedset/NestedSetImplTest.java
@@ -99,7 +99,7 @@
try {
NestedSetBuilder.compileOrder().addTransitive(NestedSetBuilder.linkOrder().build()).build();
fail("Shouldn't be able to include a non-stable order inside a different non-stable order!");
- } catch (IllegalStateException e) {
+ } catch (IllegalArgumentException e) {
// Expected.
}
}
diff --git a/src/test/java/com/google/devtools/build/lib/syntax/SkylarkNestedSetTest.java b/src/test/java/com/google/devtools/build/lib/syntax/SkylarkNestedSetTest.java
index 967940d..b9fd9ac 100644
--- a/src/test/java/com/google/devtools/build/lib/syntax/SkylarkNestedSetTest.java
+++ b/src/test/java/com/google/devtools/build/lib/syntax/SkylarkNestedSetTest.java
@@ -97,7 +97,7 @@
@Test
public void testNsetNestedItemBadOrder() throws Exception {
checkEvalError(
- "LINK_ORDER != COMPILE_ORDER",
+ "Order mismatch: LINK_ORDER != COMPILE_ORDER",
"depset(['a', 'b'], order='compile') + depset(['c', 'd'], order='link')");
}