blob: a0d84206e8a7fa719ad5ab3ab19a524eeec07006 [file] [log] [blame]
// Copyright 2015 The Bazel Authors. All rights reserved.
//
// 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 com.google.devtools.build.lib.syntax;
import static com.google.common.truth.Truth.assertThat;
import static org.junit.Assert.assertEquals;
import com.google.common.collect.ImmutableList;
import com.google.devtools.build.lib.collect.nestedset.Order;
import com.google.devtools.build.lib.syntax.SkylarkList.Tuple;
import com.google.devtools.build.lib.syntax.util.EvaluationTestCase;
import java.util.Arrays;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.junit.runners.JUnit4;
/**
* Tests for SkylarkNestedSet.
*/
@RunWith(JUnit4.class)
public class SkylarkNestedSetTest extends EvaluationTestCase {
@Test
public void testNsetBuilder() throws Exception {
eval("n = set(order='stable')");
assertThat(lookup("n")).isInstanceOf(SkylarkNestedSet.class);
}
@Test
public void testNsetOrder() throws Exception {
eval("n = set(['a', 'b'], order='compile')");
assertEquals(Order.COMPILE_ORDER, get("n").getSet(String.class).getOrder());
}
@Test
public void testEmptyNsetGenericType() throws Exception {
eval("n = set()");
assertEquals(SkylarkType.TOP, get("n").getContentType());
}
@Test
public void testFunctionReturnsNset() throws Exception {
eval("def func():",
" n = set()",
" n += ['a']",
" return n",
"s = func()");
assertEquals(ImmutableList.of("a"), get("s").toCollection());
}
@Test
public void testNsetTwoReferences() throws Exception {
eval("def func():",
" n1 = set()",
" n1 += ['a']",
" n2 = n1",
" n2 += ['b']",
" return n1",
"n = func()");
assertEquals(ImmutableList.of("a"), get("n").toCollection());
}
@Test
public void testNsetNestedItem() throws Exception {
eval("def func():",
" n1 = set()",
" n2 = set()",
" n1 += ['a']",
" n2 += ['b']",
" n1 += n2",
" return n1",
"n = func()");
assertEquals(ImmutableList.of("b", "a"), get("n").toCollection());
}
@Test
public void testNsetNestedItemBadOrder() throws Exception {
checkEvalError("LINK_ORDER != COMPILE_ORDER",
"set(['a', 'b'], order='compile') + set(['c', 'd'], order='link')");
}
@Test
public void testNsetItemList() throws Exception {
eval("def func():",
" n = set()",
" n += ['a', 'b']",
" return n",
"n = func()");
assertEquals(ImmutableList.of("a", "b"), get("n").toCollection());
}
@Test
public void testNsetFuncParamNoSideEffects() throws Exception {
eval("def func1(n):",
" n += ['b']",
"def func2():",
" n = set()",
" n += ['a']",
" func1(n)",
" return n",
"n = func2()");
assertEquals(ImmutableList.of("a"), get("n").toCollection());
}
@Test
public void testNsetTransitiveOrdering() throws Exception {
eval("def func():",
" na = set(['a'], order='compile')",
" nb = set(['b'], order='compile')",
" nc = set(['c'], order='compile') + na",
" return set() + nb + nc",
"n = func()");
// The iterator lists the Transitive sets first
assertEquals(ImmutableList.of("b", "a", "c"), get("n").toCollection());
}
@Test
public void testNsetOrdering() throws Exception {
eval("def func():",
" na = set()",
" na += [4]",
" na += [2, 4]",
" na += [3, 4, 5]",
" return na",
"n = func()");
// The iterator lists the Transitive sets first
assertEquals(ImmutableList.of(4, 2, 3, 5), get("n").toCollection());
}
@Test
public void testNsetBadOrder() throws Exception {
checkEvalError("Invalid order: non_existing", "set(order='non_existing')");
}
@Test
public void testNsetBadRightOperand() throws Exception {
checkEvalError("cannot add value of type 'string' to a set", "l = ['a']\n" + "set() + l[0]");
}
@Test
public void testNsetToString() throws Exception {
eval("s = set() + [2, 4, 6] + [3, 4, 5]",
"x = str(s)");
assertEquals("set([2, 4, 6, 3, 5])", lookup("x"));
}
@Test
public void testNsetToStringWithOrder() throws Exception {
eval("s = set(order = 'link') + [2, 4, 6] + [3, 4, 5]",
"x = str(s)");
assertEquals("set([2, 4, 6, 3, 5], order = \"link\")", lookup("x"));
}
@SuppressWarnings("unchecked")
private SkylarkNestedSet get(String varname) throws Exception {
return (SkylarkNestedSet) lookup(varname);
}
@Test
public void testSetOrderCompatibility() throws Exception {
// Two sets are compatible if
// (a) both have the same order or
// (b) at least one order is "stable"
for (Order first : Order.values()) {
SkylarkNestedSet s1 = new SkylarkNestedSet(first, Tuple.of("1", "11"), null);
for (Order second : Order.values()) {
SkylarkNestedSet s2 = new SkylarkNestedSet(second, Tuple.of("2", "22"), null);
boolean compatible = true;
try {
new SkylarkNestedSet(s1, s2, null);
} catch (Exception ex) {
compatible = false;
}
assertThat(compatible).isEqualTo(areOrdersCompatible(first, second));
}
}
}
private boolean areOrdersCompatible(Order first, Order second) {
return first == Order.STABLE_ORDER || second == Order.STABLE_ORDER || first == second;
}
@Test
public void testSetOrderComplexUnion() throws Exception {
// {1, 11, {2, 22}, {3, 33}, {4, 44}}
List<String> preOrder = Arrays.asList("1", "11", "2", "22", "3", "33", "4", "44");
List<String> postOrder = Arrays.asList("2", "22", "3", "33", "4", "44", "1", "11");
MergeStrategy strategy = new MergeStrategy() {
@Override
public SkylarkNestedSet merge(SkylarkNestedSet[] sets) throws Exception {
SkylarkNestedSet union = new SkylarkNestedSet(sets[0], sets[1], null);
union = new SkylarkNestedSet(union, sets[2], null);
union = new SkylarkNestedSet(union, sets[3], null);
return union;
}
};
runComplexOrderTest(strategy, preOrder, postOrder);
}
@Test
public void testSetOrderBalancedTree() throws Exception {
// {{1, 11, {2, 22}}, {3, 33, {4, 44}}}
List<String> preOrder = Arrays.asList("1", "11", "2", "22", "3", "33", "4", "44");
List<String> postOrder = Arrays.asList("2", "22", "4", "44", "3", "33", "1", "11");
MergeStrategy strategy = new MergeStrategy() {
@Override
public SkylarkNestedSet merge(SkylarkNestedSet[] sets) throws Exception {
SkylarkNestedSet leftUnion = new SkylarkNestedSet(sets[0], sets[1], null);
SkylarkNestedSet rightUnion = new SkylarkNestedSet(sets[2], sets[3], null);
SkylarkNestedSet union = new SkylarkNestedSet(leftUnion, rightUnion, null);
return union;
}
};
runComplexOrderTest(strategy, preOrder, postOrder);
}
@Test
public void testSetOrderManyLevelsOfNesting() throws Exception {
// {1, 11, {2, 22, {3, 33, {4, 44}}}}
List<String> preOrder = Arrays.asList("1", "11", "2", "22", "3", "33", "4", "44");
List<String> postOrder = Arrays.asList("4", "44", "3", "33", "2", "22", "1", "11");
MergeStrategy strategy = new MergeStrategy() {
@Override
public SkylarkNestedSet merge(SkylarkNestedSet[] sets) throws Exception {
SkylarkNestedSet union = new SkylarkNestedSet(sets[2], sets[3], null);
union = new SkylarkNestedSet(sets[1], union, null);
union = new SkylarkNestedSet(sets[0], union, null);
return union;
}
};
runComplexOrderTest(strategy, preOrder, postOrder);
}
private interface MergeStrategy {
SkylarkNestedSet merge(SkylarkNestedSet[] sets) throws Exception;
}
private void runComplexOrderTest(
MergeStrategy strategy, List<String> preOrder, List<String> postOrder) throws Exception {
Map<Order, List<String>> expected = createExpectedMap(preOrder, postOrder);
for (Order order : Order.values()) {
SkylarkNestedSet union = strategy.merge(makeFourSets(order));
assertThat(union.toCollection()).containsExactlyElementsIn(expected.get(order)).inOrder();
}
}
private Map<Order, List<String>> createExpectedMap(
List<String> preOrder, List<String> postOrder) {
Map<Order, List<String>> expected = new HashMap<>();
for (Order order : Order.values()) {
expected.put(order, isPostOrder(order) ? postOrder : preOrder);
}
return expected;
}
private boolean isPostOrder(Order order) {
return order == Order.STABLE_ORDER || order == Order.COMPILE_ORDER;
}
private SkylarkNestedSet[] makeFourSets(Order order) throws Exception {
return new SkylarkNestedSet[] {
new SkylarkNestedSet(order, Tuple.of("1", "11"), null),
new SkylarkNestedSet(order, Tuple.of("2", "22"), null),
new SkylarkNestedSet(order, Tuple.of("3", "33"), null),
new SkylarkNestedSet(order, Tuple.of("4", "44"), null)};
}
}