blob: f62d268503e71705af31d1f267c01a03ca841223 [file] [log] [blame]
// Copyright 2006 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 net.starlark.java.eval;
import static com.google.common.truth.Truth.assertThat;
import static org.junit.Assert.assertThrows;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableMap;
import java.math.BigInteger;
import javax.annotation.Nullable;
import net.starlark.java.annot.StarlarkBuiltin;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.junit.runners.JUnit4;
/**
* Test properties of the evaluator's datatypes and utility functions without actually creating any
* parse trees.
*/
@RunWith(JUnit4.class)
public final class EvalUtilsTest {
private static StarlarkList<Object> makeList(@Nullable Mutability mu) {
return StarlarkList.of(mu, 1, 2, 3);
}
private static Dict<Object, Object> makeDict(@Nullable Mutability mu) {
return Dict.of(mu, 1, 1, 2, 2);
}
/** MockClassA */
@StarlarkBuiltin(name = "MockClassA", doc = "MockClassA")
public static class MockClassA implements StarlarkValue {}
/** MockClassB */
public static class MockClassB extends MockClassA {
}
@Test
public void testDataTypeNames() throws Exception {
assertThat(Starlark.type("foo")).isEqualTo("string");
assertThat(Starlark.type(StarlarkInt.of(3))).isEqualTo("int");
assertThat(Starlark.type(Tuple.of(1, 2, 3))).isEqualTo("tuple");
assertThat(Starlark.type(makeList(null))).isEqualTo("list");
assertThat(Starlark.type(makeDict(null))).isEqualTo("dict");
assertThat(Starlark.type(Starlark.NONE)).isEqualTo("NoneType");
assertThat(Starlark.type(new MockClassA())).isEqualTo("MockClassA");
assertThat(Starlark.type(new MockClassB())).isEqualTo("MockClassA");
// Not legal values, but relied upon by (e.g.) ApiExporter
// when formatting the result type of an annotated method.
// Other types accepted by fromJava are treated similarly.
assertThat(Starlark.type(3)).isEqualTo("int");
assertThat(Starlark.type(ImmutableList.of())).isEqualTo("List");
assertThat(Starlark.type(ImmutableMap.of())).isEqualTo("Map");
}
@Test
public void testDatatypeMutabilityPrimitive() throws Exception {
assertThat(Starlark.isImmutable("foo")).isTrue();
assertThat(Starlark.isImmutable(StarlarkInt.of(3))).isTrue();
}
@Test
public void testDatatypeMutabilityShallow() throws Exception {
assertThat(
Starlark.isImmutable(Tuple.of(StarlarkInt.of(1), StarlarkInt.of(2), StarlarkInt.of(3))))
.isTrue();
assertThat(Starlark.isImmutable(makeList(null))).isTrue();
assertThat(Starlark.isImmutable(makeDict(null))).isTrue();
Mutability mu = Mutability.create("test");
assertThat(Starlark.isImmutable(makeList(mu))).isFalse();
assertThat(Starlark.isImmutable(makeDict(mu))).isFalse();
}
@Test
public void testDatatypeMutabilityDeep() throws Exception {
Mutability mu = Mutability.create("test");
assertThat(Starlark.isImmutable(Tuple.of(makeList(null)))).isTrue();
assertThat(Starlark.isImmutable(Tuple.of(makeList(mu)))).isFalse();
}
@Test
public void testComparatorWithDifferentTypes() throws Exception {
Mutability mu = Mutability.create("test");
StarlarkValue myValue = new StarlarkValue() {};
Object[] objects = {
"1",
StarlarkInt.of(2),
true,
Starlark.NONE,
Tuple.of(StarlarkInt.of(1), StarlarkInt.of(2), StarlarkInt.of(3)),
Tuple.of("1", "2", "3"),
StarlarkList.of(mu, StarlarkInt.of(1), StarlarkInt.of(2), StarlarkInt.of(3)),
StarlarkList.of(mu, "1", "2", "3"),
Dict.of(mu, "key", StarlarkInt.of(123)),
Dict.of(mu, StarlarkInt.of(123), "value"),
myValue,
};
for (int i = 0; i < objects.length; ++i) {
for (int j = 0; j < objects.length; ++j) {
if (i != j) {
Object first = objects[i];
Object second = objects[j];
assertThrows(ClassCastException.class, () -> Starlark.ORDERING.compare(first, second));
}
}
}
}
@Test
public void testComparatorWithNones() throws Exception {
assertThrows(
ClassCastException.class, () -> Starlark.ORDERING.compare(Starlark.NONE, Starlark.NONE));
}
@Test
public void testLen() {
assertThat(Starlark.len("abc")).isEqualTo(3);
assertThat(Starlark.len(Tuple.of(StarlarkInt.of(1), StarlarkInt.of(2), StarlarkInt.of(3))))
.isEqualTo(3);
assertThat(
Starlark.len(
StarlarkList.of(null, StarlarkInt.of(1), StarlarkInt.of(2), StarlarkInt.of(3))))
.isEqualTo(3);
assertThat(Starlark.len(Dict.of(null, "one", StarlarkInt.of(1), "two", StarlarkInt.of(2))))
.isEqualTo(2);
assertThat(Starlark.len(true)).isEqualTo(-1);
assertThrows(IllegalArgumentException.class, () -> Starlark.len(this));
}
@Test
public void testIntConstructor() throws Exception {
// small values are cached
assertThat(StarlarkInt.of(-1)).isSameInstanceAs(StarlarkInt.of(-1));
assertThat(StarlarkInt.of(0)).isSameInstanceAs(StarlarkInt.ZERO);
assertThat(StarlarkInt.of(123)).isSameInstanceAs(StarlarkInt.of(123));
// int32
assertThat(StarlarkInt.of(0).getClass().getSimpleName()).isEqualTo("Int32");
assertThat(StarlarkInt.of(Integer.MAX_VALUE).getClass().getSimpleName()).isEqualTo("Int32");
assertThat(StarlarkInt.of(Integer.MIN_VALUE).getClass().getSimpleName()).isEqualTo("Int32");
// int64
assertThat(StarlarkInt.of((long) Integer.MAX_VALUE + 1).getClass().getSimpleName())
.isEqualTo("Int64");
assertThat(StarlarkInt.of((long) Integer.MIN_VALUE - 1).getClass().getSimpleName())
.isEqualTo("Int64");
assertThat(StarlarkInt.of(Long.MAX_VALUE).getClass().getSimpleName()).isEqualTo("Int64");
assertThat(StarlarkInt.of(Long.MIN_VALUE).getClass().getSimpleName()).isEqualTo("Int64");
// big
assertThat(StarlarkInt.of(new BigInteger("7fffffffffffffff", 16)).getClass().getSimpleName())
.isEqualTo("Int64"); // (max long)
assertThat(StarlarkInt.of(new BigInteger("8000000000000000", 16)).getClass().getSimpleName())
.isEqualTo("Big");
assertThat(StarlarkInt.of(new BigInteger("8000000000000001", 16)).getClass().getSimpleName())
.isEqualTo("Big");
assertThat(StarlarkInt.of(new BigInteger("-7fffffffffffffff", 16)).getClass().getSimpleName())
.isEqualTo("Int64");
assertThat(StarlarkInt.of(new BigInteger("-8000000000000000", 16)).getClass().getSimpleName())
.isEqualTo("Int64"); // (min long)
assertThat(StarlarkInt.of(new BigInteger("-8000000000000001", 16)).getClass().getSimpleName())
.isEqualTo("Big");
}
}