blob: 85979f438a4b76984f12d244fe94e4349e275aba [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 com.google.devtools.build.lib.syntax;
import static com.google.common.truth.Truth.assertThat;
import com.google.common.collect.Iterables;
import com.google.devtools.build.lib.syntax.SkylarkList.MutableList;
import com.google.devtools.build.lib.syntax.util.EvaluationTestCase;
import org.junit.Before;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.junit.runners.JUnit4;
/**
* Tests for MethodLibrary.
*/
@RunWith(JUnit4.class)
public class MethodLibraryTest extends EvaluationTestCase {
private static final String LINE_SEPARATOR = System.lineSeparator();
@Before
public final void setFailFast() throws Exception {
setFailFast(true);
}
@Test
public void testMinWithInvalidArgs() throws Exception {
new SkylarkTest()
.testIfExactError("type 'int' is not iterable", "min(1)")
.testIfExactError("expected at least one item", "min([])");
}
@Test
public void testMinWithString() throws Exception {
new SkylarkTest()
.testStatement("min('abcdefxyz')", "a")
.testStatement("min('test', 'xyz')", "test");
}
@Test
public void testMinWithList() throws Exception {
new BothModesTest()
.testEval("min([4, 5], [1])", "[1]")
.testEval("min([1, 2], [3])", "[1, 2]")
.testEval("min([1, 5], [1, 6], [2, 4], [0, 6])", "[0, 6]")
.testStatement("min([-1])", -1)
.testStatement("min([5, 2, 3])", 2);
}
@Test
public void testMinWithDict() throws Exception {
new BothModesTest().testStatement("min({1: 2, -1 : 3})", -1).testStatement("min({2: None})", 2);
}
@Test
public void testMinWithSet() throws Exception {
new BothModesTest()
.testStatement("min(depset([-1]))", -1)
.testStatement("min(depset([5, 2, 3]))", 2);
}
@Test
public void testMinWithPositionalArguments() throws Exception {
new BothModesTest().testStatement("min(-1, 2)", -1).testStatement("min(5, 2, 3)", 2);
}
@Test
public void testMinWithSameValues() throws Exception {
new BothModesTest()
.testStatement("min(1, 1, 1, 1, 1, 1)", 1)
.testStatement("min([1, 1, 1, 1, 1, 1])", 1);
}
@Test
public void testMinWithDifferentTypes() throws Exception {
new BothModesTest()
.testIfExactError("Cannot compare int with string", "min(1, '2', True)")
.testIfExactError("Cannot compare int with string", "min([1, '2', True])");
}
@Test
public void testMaxWithInvalidArgs() throws Exception {
new BothModesTest()
.testIfExactError("type 'int' is not iterable", "max(1)")
.testIfExactError("expected at least one item", "max([])");
}
@Test
public void testMaxWithString() throws Exception {
new BothModesTest()
.testStatement("max('abcdefxyz')", "z")
.testStatement("max('test', 'xyz')", "xyz");
}
@Test
public void testMaxWithList() throws Exception {
new BothModesTest()
.testEval("max([1, 2], [5])", "[5]")
.testStatement("max([-1])", -1)
.testStatement("max([5, 2, 3])", 5);
}
@Test
public void testMaxWithDict() throws Exception {
new BothModesTest().testStatement("max({1: 2, -1 : 3})", 1).testStatement("max({2: None})", 2);
}
@Test
public void testMaxWithSet() throws Exception {
new BothModesTest()
.testStatement("max(depset([-1]))", -1)
.testStatement("max(depset([5, 2, 3]))", 5);
}
@Test
public void testMaxWithPositionalArguments() throws Exception {
new BothModesTest().testStatement("max(-1, 2)", 2).testStatement("max(5, 2, 3)", 5);
}
@Test
public void testMaxWithSameValues() throws Exception {
new BothModesTest()
.testStatement("max(1, 1, 1, 1, 1, 1)", 1)
.testStatement("max([1, 1, 1, 1, 1, 1])", 1);
}
@Test
public void testMaxWithDifferentTypes() throws Exception {
new BothModesTest()
.testIfExactError("Cannot compare int with string", "max(1, '2', True)")
.testIfExactError("Cannot compare int with string", "max([1, '2', True])");
}
@Test
public void testSplitLines_EmptyLine() throws Exception {
new BothModesTest().testEval("''.splitlines()", "[]").testEval("'\\n'.splitlines()", "['']");
}
@Test
public void testSplitLines_StartsWithLineBreak() throws Exception {
new BothModesTest().testEval("'\\ntest'.splitlines()", "['', 'test']");
}
@Test
public void testSplitLines_EndsWithLineBreak() throws Exception {
new BothModesTest().testEval("'test\\n'.splitlines()", "['test']");
}
@Test
public void testSplitLines_DifferentLineBreaks() throws Exception {
new BothModesTest()
.testEval("'this\\nis\\na\\ntest'.splitlines()", "['this', 'is', 'a', 'test']");
}
@Test
public void testSplitLines_OnlyLineBreaks() throws Exception {
new BothModesTest()
.testEval("'\\n\\n\\n'.splitlines()", "['', '', '']")
.testEval("'\\r\\r\\r'.splitlines()", "['', '', '']")
.testEval("'\\n\\r\\n\\r'.splitlines()", "['', '', '']")
.testEval("'\\r\\n\\r\\n\\r\\n'.splitlines()", "['', '', '']");
}
@Test
public void testSplitLines_EscapedSequences() throws Exception {
new BothModesTest().testEval("'\\n\\\\n\\\\\\n'.splitlines()", "['', '\\\\n\\\\']");
}
@Test
public void testSplitLines_KeepEnds() throws Exception {
new BothModesTest()
.testEval("''.splitlines(True)", "[]")
.testEval("'\\n'.splitlines(True)", "['\\n']")
.testEval(
"'this\\nis\\r\\na\\rtest'.splitlines(True)", "['this\\n', 'is\\r\\n', 'a\\r', 'test']")
.testEval("'\\ntest'.splitlines(True)", "['\\n', 'test']")
.testEval("'test\\n'.splitlines(True)", "['test\\n']")
.testEval("'\\n\\\\n\\\\\\n'.splitlines(True)", "['\\n', '\\\\n\\\\\\n']");
}
@Test
public void testStringIsAlnum() throws Exception {
new BothModesTest()
.testStatement("''.isalnum()", false)
.testStatement("'a0 33'.isalnum()", false)
.testStatement("'1'.isalnum()", true)
.testStatement("'a033'.isalnum()", true);
}
@Test
public void testStringIsDigit() throws Exception {
new BothModesTest()
.testStatement("''.isdigit()", false)
.testStatement("' '.isdigit()", false)
.testStatement("'a'.isdigit()", false)
.testStatement("'0234325.33'.isdigit()", false)
.testStatement("'1'.isdigit()", true)
.testStatement("'033'.isdigit()", true);
}
@Test
public void testStringIsSpace() throws Exception {
new BothModesTest()
.testStatement("''.isspace()", false)
.testStatement("'a'.isspace()", false)
.testStatement("'1'.isspace()", false)
.testStatement("'\\ta\\n'.isspace()", false)
.testStatement("' '.isspace()", true)
.testStatement("'\\t\\n'.isspace()", true);
}
@Test
public void testStringIsLower() throws Exception {
new BothModesTest()
.testStatement("''.islower()", false)
.testStatement("' '.islower()", false)
.testStatement("'1'.islower()", false)
.testStatement("'Almost'.islower()", false)
.testStatement("'abc'.islower()", true)
.testStatement("' \\nabc'.islower()", true)
.testStatement("'abc def\\n'.islower()", true)
.testStatement("'\\ta\\n'.islower()", true);
}
@Test
public void testStringIsUpper() throws Exception {
new BothModesTest()
.testStatement("''.isupper()", false)
.testStatement("' '.isupper()", false)
.testStatement("'1'.isupper()", false)
.testStatement("'aLMOST'.isupper()", false)
.testStatement("'ABC'.isupper()", true)
.testStatement("' \\nABC'.isupper()", true)
.testStatement("'ABC DEF\\n'.isupper()", true)
.testStatement("'\\tA\\n'.isupper()", true);
}
@Test
public void testStringIsTitle() throws Exception {
new BothModesTest()
.testStatement("''.istitle()", false)
.testStatement("' '.istitle()", false)
.testStatement("'134'.istitle()", false)
.testStatement("'almost Correct'.istitle()", false)
.testStatement("'1nope Nope Nope'.istitle()", false)
.testStatement("'NO Way'.istitle()", false)
.testStatement("'T'.istitle()", true)
.testStatement("'Correct'.istitle()", true)
.testStatement("'Very Correct! Yes\\nIndeed1X'.istitle()", true)
.testStatement("'1234Ab Ab'.istitle()", true)
.testStatement("'\\tA\\n'.istitle()", true);
}
@Test
public void testAllWithEmptyValue() throws Exception {
new BothModesTest()
.testStatement("all('')", true)
.testStatement("all([])", true)
.testIfExactError("type 'NoneType' is not iterable", "any(None)");
}
@Test
public void testAllWithPrimitiveType() throws Exception {
new BothModesTest().testStatement("all('test')", true).testIfErrorContains("", "all(1)");
}
@Test
public void testAllWithList() throws Exception {
new BothModesTest()
.testStatement("all([False])", false)
.testStatement("all([True, False])", false)
.testStatement("all([False, False])", false)
.testStatement("all([False, True])", false)
.testStatement("all(['', True])", false)
.testStatement("all([0, True])", false)
.testStatement("all([[], True])", false)
.testStatement("all([True, 't', 1])", true);
}
@Test
public void testAllWithSet() throws Exception {
new BothModesTest()
.testStatement("all(depset([0]))", false)
.testStatement("all(depset([1, 0]))", false)
.testStatement("all(depset([1]))", true);
}
@Test
public void testAllWithDict() throws Exception {
new BothModesTest()
.testStatement("all({1 : None})", true)
.testStatement("all({None : 1})", false);
}
@Test
public void testAnyWithEmptyValue() throws Exception {
new BothModesTest()
.testStatement("any('')", false)
.testStatement("any([])", false)
.testIfExactError("type 'NoneType' is not iterable", "any(None)");
}
@Test
public void testAnyWithPrimitiveType() throws Exception {
new BothModesTest().testStatement("any('test')", true).testIfErrorContains("", "any(1)");
}
@Test
public void testAnyWithList() throws Exception {
new BothModesTest()
.testStatement("any([False])", false)
.testStatement("any([0])", false)
.testStatement("any([''])", false)
.testStatement("any([[]])", false)
.testStatement("any([True, False])", true)
.testStatement("any([False, False])", false)
.testStatement("any([False, '', 0])", false)
.testStatement("any([False, '', 42])", true);
}
@Test
public void testAnyWithSet() throws Exception {
new BothModesTest()
.testStatement("any(depset([0]))", false)
.testStatement("any(depset([1, 0]))", true);
}
@Test
public void testAnyWithDict() throws Exception {
new BothModesTest()
.testStatement("any({1 : None, '' : None})", true)
.testStatement("any({None : 1, '' : 2})", false);
}
@Test
public void testStackTraceLocation() throws Exception {
new SkylarkTest()
.testIfErrorContains(
"Traceback (most recent call last):"
+ LINE_SEPARATOR
+ "\tFile \"\", line 8"
+ LINE_SEPARATOR
+ "\t\tfoo()"
+ LINE_SEPARATOR
+ "\tFile \"\", line 2, in foo"
+ LINE_SEPARATOR
+ "\t\tbar(1)"
+ LINE_SEPARATOR
+ "\tFile \"\", line 7, in bar"
+ LINE_SEPARATOR
+ "\t\t\"test\".index(x)",
"def foo():",
" bar(1)",
"def bar(x):",
" if x == 1:",
" a = x",
" b = 2",
" 'test'.index(x)",
"foo()");
}
@Test
public void testStackTraceWithIf() throws Exception {
new SkylarkTest()
.testIfErrorContains(
"File \"\", line 5"
+ LINE_SEPARATOR
+ "\t\tfoo()"
+ LINE_SEPARATOR
+ "\tFile \"\", line 3, in foo"
+ LINE_SEPARATOR
+ "\t\ts[0]",
"def foo():",
" s = depset()",
" if s[0] == 1:",
" x = 1",
"foo()");
}
@Test
public void testStackTraceSkipBuiltInOnly() throws Exception {
// The error message should not include the stack trace when there is
// only one built-in function.
new BothModesTest()
.testIfExactError(
"argument 'sub' has type 'int', but should be 'string'\n"
+ "in call to builtin method string.index(sub, start, end)",
"'test'.index(1)");
}
@Test
public void testStackTrace() throws Exception {
// Unlike SkylarintegrationTests#testStackTraceErrorInFunction(), this test
// has neither a BUILD nor a bzl file.
new SkylarkTest()
.testIfExactError(
"Traceback (most recent call last):"
+ LINE_SEPARATOR
+ "\tFile \"\", line 6"
+ LINE_SEPARATOR
+ "\t\tfoo()"
+ LINE_SEPARATOR
+ "\tFile \"\", line 2, in foo"
+ LINE_SEPARATOR
+ "\t\tbar(1)"
+ LINE_SEPARATOR
+ "\tFile \"\", line 5, in bar"
+ LINE_SEPARATOR
+ "\t\t\"test\".index(x)"
+ LINE_SEPARATOR
+ "argument 'sub' has type 'int', but should be 'string'\n"
+ "in call to builtin method string.index(sub, start, end)",
"def foo():",
" bar(1)",
"def bar(x):",
" if 1 == 1:",
" 'test'.index(x)",
"foo()");
}
@Test
public void testBuiltinFunctionErrorMessage() throws Exception {
new BothModesTest()
.testIfErrorContains("substring \"z\" not found in \"abc\"", "'abc'.index('z')")
.testIfErrorContains(
"argument 'sub' has type 'int', but should be 'string'\n"
+ "in call to builtin method string.startswith(sub, start, end)",
"'test'.startswith(1)")
.testIfErrorContains(
"expected value of type 'list(object)' for parameter args in dict(), "
+ "but got \"a\" (string)",
"dict('a')");
}
@Test
public void testHasAttr() throws Exception {
new SkylarkTest()
.testStatement("hasattr(depset(), 'union')", Boolean.TRUE)
.testStatement("hasattr('test', 'count')", Boolean.TRUE)
.testStatement("hasattr(dict(a = 1, b = 2), 'items')", Boolean.TRUE)
.testStatement("hasattr({}, 'items')", Boolean.TRUE);
}
@Test
public void testGetAttrMissingField() throws Exception {
new SkylarkTest()
.testIfExactError(
"object of type 'string' has no attribute \"not_there\"",
"getattr('a string', 'not_there')")
.testStatement("getattr('a string', 'not_there', 'use this')", "use this")
.testStatement("getattr('a string', 'not there', None)", Runtime.NONE);
}
@Test
public void testGetAttrWithMethods() throws Exception {
String msg =
"object of type 'string' has no attribute \"count\", however, "
+ "a method of that name exists";
new SkylarkTest()
.testIfExactError(msg, "getattr('a string', 'count')")
.testStatement("getattr('a string', 'count', 'default')", "default");
}
@Test
public void testDir() throws Exception {
new SkylarkTest()
.testStatement(
"str(dir({}))",
"[\"clear\", \"get\", \"items\", \"keys\","
+ " \"pop\", \"popitem\", \"setdefault\", \"update\", \"values\"]");
}
@Test
public void testBoolean() throws Exception {
new BothModesTest().testStatement("False", Boolean.FALSE).testStatement("True", Boolean.TRUE);
}
@Test
public void testBooleanUnsupportedOperationFails() throws Exception {
new BothModesTest()
.testIfErrorContains("unsupported operand type(s) for +: 'bool' and 'bool'", "True + True");
}
@Test
public void testPyStringJoin() throws Exception {
new BothModesTest().testStatement("'-'.join(['a', 'b', 'c'])", "a-b-c");
}
@Test
public void testPyStringGlobalJoin() throws Exception {
new BothModesTest()
.testIfErrorContains("name 'join' is not defined", "join(' ', ['a', 'b', 'c'])")
.testStatement("' '.join(['a', 'b', 'c'])", "a b c");
}
@Test
public void testPyStringJoinCompr() throws Exception {
new BothModesTest()
.testStatement("''.join([(x + '*') for x in ['a', 'b', 'c']])", "a*b*c*")
.testStatement(
"''.join([(y + '*' + z + '|') " + "for y in ['a', 'b', 'c'] for z in ['d', 'e']])",
"a*d|a*e|b*d|b*e|c*d|c*e|");
}
@Test
public void testPyStringLower() throws Exception {
new BothModesTest().testStatement("'Blah Blah'.lower()", "blah blah");
}
@Test
public void testPyStringUpper() throws Exception {
new BothModesTest()
.testStatement("'ein bier'.upper()", "EIN BIER")
.testStatement("''.upper()", "");
}
@Test
public void testPyStringReplace() throws Exception {
new BothModesTest()
.testStatement("'banana'.replace('a', 'e')", "benene")
.testStatement("'banana'.replace('a', '$()')", "b$()n$()n$()")
.testStatement("'banana'.replace('a', '$')", "b$n$n$")
.testStatement("'banana'.replace('a', '\\\\')", "b\\n\\n\\")
.testStatement("'b$()n$()n$()'.replace('$()', '$($())')", "b$($())n$($())n$($())")
.testStatement("'b\\\\n\\\\n\\\\'.replace('\\\\', '$()')", "b$()n$()n$()");
}
@Test
public void testPyStringReplace2() throws Exception {
new BothModesTest().testStatement("'banana'.replace('a', 'e', 2)", "benena");
}
@Test
public void testPyStringSplit() throws Exception {
new BothModesTest().testEval("'h i'.split(' ')", "['h', 'i']");
}
@Test
public void testPyStringSplit2() throws Exception {
new BothModesTest().testEval("'h i p'.split(' ')", "['h', 'i', 'p']");
}
@Test
public void testPyStringSplit3() throws Exception {
new BothModesTest().testEval("'a,e,i,o,u'.split(',', 2)", "['a', 'e', 'i,o,u']");
}
@Test
public void testPyStringSplitNoSep() throws Exception {
new BothModesTest()
.testEval("' 1 2 3 '.split(' ')", "['', '', '1', '', '2', '', '3', '', '']");
}
@Test
public void testPyStringRSplitRegex() throws Exception {
new BothModesTest()
.testEval("'foo/bar.lisp'.rsplit('.')", "['foo/bar', 'lisp']")
.testEval("'foo/bar.?lisp'.rsplit('.?')", "['foo/bar', 'lisp']")
.testEval("'fwe$foo'.rsplit('$')", "['fwe', 'foo']")
.testEval("'windows'.rsplit('\\w')", "['windows']");
}
@Test
public void testPyStringRSplitNoMatch() throws Exception {
new BothModesTest()
.testEval("''.rsplit('o')", "['']")
.testEval("'google'.rsplit('x')", "['google']");
}
@Test
public void testPyStringRSplitSeparator() throws Exception {
new BothModesTest()
.testEval("'xxxxxx'.rsplit('x')", "['', '', '', '', '', '', '']")
.testEval("'xxxxxx'.rsplit('x', 1)", "['xxxxx', '']")
.testEval("'xxxxxx'.rsplit('x', 2)", "['xxxx', '', '']")
.testEval("'xxxxxx'.rsplit('x', 3)", "['xxx', '', '', '']")
.testEval("'xxxxxx'.rsplit('x', 4)", "['xx', '', '', '', '']")
.testEval("'xxxxxx'.rsplit('x', 5)", "['x', '', '', '', '', '']")
.testEval("'xxxxxx'.rsplit('x', 6)", "['', '', '', '', '', '', '']")
.testEval("'xxxxxx'.rsplit('x', 7)", "['', '', '', '', '', '', '']");
}
@Test
public void testPyStringRSplitLongerSep() throws Exception {
new BothModesTest()
.testEval("'abcdabef'.rsplit('ab')", "['', 'cd', 'ef']")
.testEval("'google_or_gogol'.rsplit('go')", "['', 'ogle_or_', '', 'l']");
}
@Test
public void testPyStringRSplitMaxSplit() throws Exception {
new BothModesTest()
.testEval("'google'.rsplit('o')", "['g', '', 'gle']")
.testEval("'google'.rsplit('o')", "['g', '', 'gle']")
.testEval("'google'.rsplit('o', 1)", "['go', 'gle']")
.testEval("'google'.rsplit('o', 2)", "['g', '', 'gle']")
.testEval("'google'.rsplit('o', 3)", "['g', '', 'gle']")
.testEval("'ogooglo'.rsplit('o')", "['', 'g', '', 'gl', '']")
.testEval("'ogooglo'.rsplit('o', 1)", "['ogoogl', '']")
.testEval("'ogooglo'.rsplit('o', 2)", "['ogo', 'gl', '']")
.testEval("'ogooglo'.rsplit('o', 3)", "['og', '', 'gl', '']")
.testEval("'ogooglo'.rsplit('o', 4)", "['', 'g', '', 'gl', '']")
.testEval("'ogooglo'.rsplit('o', 5)", "['', 'g', '', 'gl', '']")
.testEval("'google'.rsplit('google')", "['', '']")
.testEval("'google'.rsplit('google', 1)", "['', '']")
.testEval("'google'.rsplit('google', 2)", "['', '']");
}
@Test
public void testPyStringPartitionEasy() throws Exception {
new BothModesTest()
.testEval("'lawl'.partition('a')", "('l', 'a', 'wl')")
.testEval("'lawl'.rpartition('a')", "('l', 'a', 'wl')");
}
@Test
public void testPyStringPartitionMultipleSep() throws Exception {
new BothModesTest()
.testEval("'google'.partition('o')", "('g', 'o', 'ogle')")
.testEval("'google'.rpartition('o')", "('go', 'o', 'gle')")
.testEval("'xxx'.partition('x')", "('', 'x', 'xx')")
.testEval("'xxx'.rpartition('x')", "('xx', 'x', '')");
}
@Test
public void testPyStringPartitionEmptyInput() throws Exception {
new BothModesTest()
.testEval("''.partition('a')", "('', '', '')")
.testEval("''.rpartition('a')", "('', '', '')");
}
@Test
public void testPyStringPartitionEmptySeparator() throws Exception {
new BothModesTest()
.testIfErrorContains("Empty separator", "'google'.partition('')")
.testIfErrorContains("Empty separator", "'google'.rpartition('')");
}
@Test
public void testPyStringPartitionDefaultSep() throws Exception {
new BothModesTest()
.testEval("'hi this is a test'.partition()", "('hi', ' ', 'this is a test')")
.testEval("'hi this is a test'.rpartition()", "('hi this is a', ' ', 'test')")
.testEval("'google'.partition()", "('google', '', '')")
.testEval("'google'.rpartition()", "('', '', 'google')");
}
@Test
public void testPyStringPartitionNoMatch() throws Exception {
new BothModesTest()
.testEval("'google'.partition('x')", "('google', '', '')")
.testEval("'google'.rpartition('x')", "('', '', 'google')");
}
@Test
public void testPyStringPartitionWordBoundaries() throws Exception {
new BothModesTest()
.testEval("'goog'.partition('g')", "('', 'g', 'oog')")
.testEval("'goog'.rpartition('g')", "('goo', 'g', '')")
.testEval("'plex'.partition('p')", "('', 'p', 'lex')")
.testEval("'plex'.rpartition('p')", "('', 'p', 'lex')")
.testEval("'plex'.partition('x')", "('ple', 'x', '')")
.testEval("'plex'.rpartition('x')", "('ple', 'x', '')");
}
@Test
public void testPyStringPartitionLongSep() throws Exception {
new BothModesTest()
.testEval("'google'.partition('oog')", "('g', 'oog', 'le')")
.testEval("'google'.rpartition('oog')", "('g', 'oog', 'le')")
.testEval(
"'lolgooglolgooglolgooglol'.partition('goog')", "('lol', 'goog', 'lolgooglolgooglol')")
.testEval(
"'lolgooglolgooglolgooglol'.rpartition('goog')",
"('lolgooglolgooglol', 'goog', 'lol')");
}
@Test
public void testPyStringPartitionCompleteString() throws Exception {
new BothModesTest()
.testEval("'google'.partition('google')", "('', 'google', '')")
.testEval("'google'.rpartition('google')", "('', 'google', '')");
}
@Test
public void testPyStringTitle() throws Exception {
new BothModesTest()
.testStatement("'this is a very simple test'.title()", "This Is A Very Simple Test");
new BothModesTest()
.testStatement("'Do We Keep Capital Letters?'.title()", "Do We Keep Capital Letters?");
new BothModesTest()
.testStatement(
"'this isn\\'t just an ol\\' apostrophe test'.title()",
"This Isn'T Just An Ol' Apostrophe Test");
new BothModesTest()
.testStatement(
"'Let us test crazy characters: _bla.exe//foo:bla(test$class)'.title()",
"Let Us Test Crazy Characters: _Bla.Exe//Foo:Bla(Test$Class)");
new BothModesTest().testStatement("'any germans here? äöü'.title()", "Any Germans Here? Äöü");
new BothModesTest()
.testStatement(
"'WE HAve tO lOWERCASE soMEthING heRE, AI?'.title()",
"We Have To Lowercase Something Here, Ai?");
new BothModesTest()
.testStatement("'wh4t ab0ut s0me numb3rs'.title()", "Wh4T Ab0Ut S0Me Numb3Rs");
}
@Test
public void testCapitalize() throws Exception {
new BothModesTest()
.testStatement("'hello world'.capitalize()", "Hello world")
.testStatement("'HELLO WORLD'.capitalize()", "Hello world")
.testStatement("''.capitalize()", "")
.testStatement("'12 lower UPPER 34'.capitalize()", "12 lower upper 34");
}
@Test
public void testPyStringRfind() throws Exception {
new BothModesTest()
.testStatement("'banana'.rfind('na')", 4)
.testStatement("'banana'.rfind('na', 3, 1)", -1)
.testStatement("'aaaa'.rfind('a', 1, 1)", -1)
.testStatement("'aaaa'.rfind('a', 1, 50)", 3)
.testStatement("'aaaa'.rfind('aaaaa')", -1)
.testStatement("'abababa'.rfind('ab', 1)", 4)
.testStatement("'abababa'.rfind('ab', 0)", 4)
.testStatement("'abababa'.rfind('ab', -1)", -1)
.testStatement("'abababa'.rfind('ab', -2)", -1)
.testStatement("'abababa'.rfind('ab', -3)", 4)
.testStatement("'abababa'.rfind('ab', 0, 1)", -1)
.testStatement("'abababa'.rfind('ab', 0, 2)", 0)
.testStatement("'abababa'.rfind('ab', -1000)", 4)
.testStatement("'abababa'.rfind('ab', 1000)", -1)
.testStatement("''.rfind('a', 1)", -1);
}
@Test
public void testPyStringFind() throws Exception {
new BothModesTest()
.testStatement("'banana'.find('na')", 2)
.testStatement("'banana'.find('na', 3, 1)", -1)
.testStatement("'aaaa'.find('a', 1, 1)", -1)
.testStatement("'aaaa'.find('a', 1, 50)", 1)
.testStatement("'aaaa'.find('aaaaa')", -1)
.testStatement("'abababa'.find('ab', 1)", 2)
.testStatement("'abababa'.find('ab', 0)", 0)
.testStatement("'abababa'.find('ab', -1)", -1)
.testStatement("'abababa'.find('ab', -2)", -1)
.testStatement("'abababa'.find('ab', -3)", 4)
.testStatement("'abababa'.find('ab', 0, 1)", -1)
.testStatement("'abababa'.find('ab', 0, 2)", 0)
.testStatement("'abababa'.find('ab', -1000)", 0)
.testStatement("'abababa'.find('ab', 1000)", -1)
.testStatement("''.find('a', 1)", -1);
}
@Test
public void testPyStringIndex() throws Exception {
new BothModesTest()
.testStatement("'banana'.index('na')", 2)
.testStatement("'abababa'.index('ab', 1)", 2)
.testIfErrorContains("substring \"foo\" not found in \"banana\"", "'banana'.index('foo')");
}
@Test
public void testPyStringRIndex() throws Exception {
new BothModesTest()
.testStatement("'banana'.rindex('na')", 4)
.testStatement("'abababa'.rindex('ab', 1)", 4)
.testIfErrorContains("substring \"foo\" not found in \"banana\"", "'banana'.rindex('foo')");
}
@Test
public void testPyStringEndswith() throws Exception {
new BothModesTest()
.testStatement("'Apricot'.endswith('cot')", true)
.testStatement("'a'.endswith('')", true)
.testStatement("''.endswith('')", true)
.testStatement("'Apricot'.endswith('co')", false)
.testStatement("'Apricot'.endswith('co', -1)", false)
.testStatement("'abcd'.endswith('c', -2, -1)", true)
.testStatement("'abcd'.endswith('c', 1, 8)", false)
.testStatement("'abcd'.endswith('d', 1, 8)", true);
}
@Test
public void testPyStringStartswith() throws Exception {
new BothModesTest()
.testStatement("'Apricot'.startswith('Apr')", true)
.testStatement("'Apricot'.startswith('A')", true)
.testStatement("'Apricot'.startswith('')", true)
.testStatement("'Apricot'.startswith('z')", false)
.testStatement("''.startswith('')", true)
.testStatement("''.startswith('a')", false);
}
@Test
public void testPySubstring() throws Exception {
new BothModesTest()
.testStatement("'012345678'[0:-1]", "01234567")
.testStatement("'012345678'[2:4]", "23")
.testStatement("'012345678'[-5:-3]", "45")
.testStatement("'012345678'[2:2]", "")
.testStatement("'012345678'[2:]", "2345678")
.testStatement("'012345678'[:3]", "012")
.testStatement("'012345678'[-1:]", "8")
.testStatement("'012345678'[:]", "012345678")
.testStatement("'012345678'[-1:2]", "")
.testStatement("'012345678'[4:2]", "");
}
@Test
public void testPyStringFormatEscaping() throws Exception {
new BothModesTest()
.testStatement("'{{}}'.format()", "{}")
.testStatement("'{{}}'.format(42)", "{}")
.testStatement("'{{ }}'.format()", "{ }")
.testStatement("'{{ }}'.format(42)", "{ }")
.testStatement("'{{{{}}}}'.format()", "{{}}")
.testStatement("'{{{{}}}}'.format(42)", "{{}}")
.testStatement("'{{0}}'.format(42)", "{0}")
.testStatement("'{{}}'.format(42)", "{}")
.testStatement("'{{{}}}'.format(42)", "{42}")
.testStatement("'{{ '.format(42)", "{ ")
.testStatement("' }}'.format(42)", " }")
.testStatement("'{{ {}'.format(42)", "{ 42")
.testStatement("'{} }}'.format(42)", "42 }")
.testStatement("'{{0}}'.format(42)", "{0}")
.testStatement("'{{{0}}}'.format(42)", "{42}")
.testStatement("'{{ 0'.format(42)", "{ 0")
.testStatement("'0 }}'.format(42)", "0 }")
.testStatement("'{{ {0}'.format(42)", "{ 42")
.testStatement("'{0} }}'.format(42)", "42 }")
.testStatement("'{{test}}'.format(test = 42)", "{test}")
.testStatement("'{{{test}}}'.format(test = 42)", "{42}")
.testStatement("'{{ test'.format(test = 42)", "{ test")
.testStatement("'test }}'.format(test = 42)", "test }")
.testStatement("'{{ {test}'.format(test = 42)", "{ 42")
.testStatement("'{test} }}'.format(test = 42)", "42 }")
.testIfErrorContains("Found '}' without matching '{'", "'{{}'.format(1)")
.testIfErrorContains("Found '}' without matching '{'", "'{}}'.format(1)");
}
@Test
public void testPyStringFormatManualPositionals() throws Exception {
new BothModesTest()
.testStatement(
"'{0}, {1} {2} {3} test'.format('hi', 'this', 'is', 'a')", "hi, this is a test")
.testStatement(
"'{3}, {2} {1} {0} test'.format('a', 'is', 'this', 'hi')", "hi, this is a test")
.testStatement(
"'skip some {0}'.format('arguments', 'obsolete', 'deprecated')", "skip some arguments")
.testStatement(
"'{0} can be reused: {0}'.format('this', 'obsolete')", "this can be reused: this");
}
@Test
public void testPyStringFormatManualPositionalsErrors() throws Exception {
new BothModesTest()
.testIfErrorContains("No replacement found for index 0", "'{0}'.format()")
.testIfErrorContains("No replacement found for index 1", "'{0} and {1}'.format('this')")
.testIfErrorContains(
"No replacement found for index 2", "'{0} and {2}'.format('this', 'that')")
.testIfErrorContains(
"No replacement found for index -1", "'{-0} and {-1}'.format('this', 'that')")
.testIfErrorContains(
"Invalid character ',' inside replacement field",
"'{0,1} and {1}'.format('this', 'that')")
.testIfErrorContains(
"Invalid character '.' inside replacement field",
"'{0.1} and {1}'.format('this', 'that')");
}
@Test
public void testPyStringFormatAutomaticPositionals() throws Exception {
new BothModesTest()
.testStatement("'{}, {} {} {} test'.format('hi', 'this', 'is', 'a')", "hi, this is a test")
.testStatement(
"'skip some {}'.format('arguments', 'obsolete', 'deprecated')", "skip some arguments");
}
@Test
public void testPyStringFormatAutomaticPositionalsError() throws Exception {
new BothModesTest()
.testIfErrorContains("No replacement found for index 0", "'{}'.format()")
.testIfErrorContains("No replacement found for index 1", "'{} and {}'.format('this')");
}
@Test
public void testPyStringFormatMixedFields() throws Exception {
new BothModesTest()
.testStatement("'{test} and {}'.format(2, test = 1)", "1 and 2")
.testStatement("'{test} and {0}'.format(2, test = 1)", "1 and 2")
.testIfErrorContains(
"non-keyword arg after keyword arg", "'{test} and {}'.format(test = 1, 2)")
.testIfErrorContains(
"non-keyword arg after keyword arg", "'{test} and {0}'.format(test = 1, 2)")
.testIfErrorContains(
"Cannot mix manual and automatic numbering of positional fields",
"'{} and {1}'.format(1, 2)")
.testIfErrorContains(
"Cannot mix manual and automatic numbering of positional fields",
"'{1} and {}'.format(1, 2)");
}
@Test
public void testPyStringFormatInvalidFields() throws Exception {
for (char unsupported : new char[] {'.', '[', ']', ','}) {
new BothModesTest()
.testIfErrorContains(
String.format("Invalid character '%c' inside replacement field", unsupported),
String.format("'{test%ctest}'.format(test = 1)", unsupported));
}
new BothModesTest()
.testIfErrorContains("Nested replacement fields are not supported", "'{ {} }'.format(42)");
}
@Test
public void testPyStringFormat() throws Exception {
new BothModesTest()
.testStatement("'abc'.format()", "abc")
.testStatement("'x{key}x'.format(key = 2)", "x2x")
.testStatement("'x{key}x'.format(key = 'abc')", "xabcx")
.testStatement("'{a}{b}{a}{b}'.format(a = 3, b = True)", "3True3True")
.testStatement("'{a}{b}{a}{b}'.format(a = 3, b = True)", "3True3True")
.testStatement("'{s1}{s2}'.format(s1 = ['a'], s2 = 'a')", "[\"a\"]a")
.testIfErrorContains("Missing argument 'b'", "'{a}{b}'.format(a = 5)")
.testStatement("'{a}'.format(a = '$')", "$")
.testStatement("'{a}'.format(a = '$a')", "$a")
.testStatement("'{a}$'.format(a = '$a')", "$a$");
// The test below is using **kwargs, which is not available in BUILD mode.
new SkylarkTest().testStatement("'{(}'.format(**{'(': 2})", "2");
}
@Test
public void testReversedWithInvalidTypes() throws Exception {
new BothModesTest()
.testIfExactError("type 'NoneType' is not iterable", "reversed(None)")
.testIfExactError("type 'int' is not iterable", "reversed(1)")
.testIfExactError(
"Argument to reversed() must be a sequence, not a dictionary.", "reversed({1: 3})");
new SkylarkTest()
.testIfExactError(
"Argument to reversed() must be a sequence, not a depset.", "reversed(depset([1]))");
}
@Test
public void testReversedWithLists() throws Exception {
new BothModesTest()
.testEval("reversed([])", "[]")
.testEval("reversed([1])", "[1]")
.testEval("reversed([1, 2, 3, 4, 5])", "[5, 4, 3, 2, 1]")
.testEval("reversed([[1, 2], 3, 4, [5]])", "[[5], 4, 3, [1, 2]]")
.testEval("reversed([1, 1, 1, 1, 2])", "[2, 1, 1, 1, 1]");
}
@Test
public void testReversedWithStrings() throws Exception {
new BothModesTest()
.testEval("reversed('')", "[]")
.testEval("reversed('a')", "['a']")
.testEval("reversed('abc')", "['c', 'b', 'a']")
.testEval("reversed('__test ')", "[' ', ' ', 't', 's', 'e', 't', '_', '_']")
.testEval("reversed('bbb')", "['b', 'b', 'b']");
}
@Test
public void testReversedNoSideEffects() throws Exception {
new SkylarkTest()
.testEval(
"def foo():\n"
+ " x = ['a', 'b']\n"
+ " y = reversed(x)\n"
+ " y += ['c']\n"
+ " return x\n"
+ "foo()",
"['a', 'b']");
}
@Test
public void testEquivalenceOfReversedAndSlice() throws Exception {
String[] data = new String[] {"[]", "[1]", "[1, 2, 3]"};
for (String toBeReversed : data) {
new BothModesTest()
.testEval(
String.format("reversed(%s)", toBeReversed), String.format("%s[::-1]", toBeReversed));
}
}
@Test
public void testListSlice() throws Exception {
new BothModesTest()
.testEval("[0, 1, 2, 3][0:-1]", "[0, 1, 2]")
.testEval("[0, 1, 2, 3, 4, 5][2:4]", "[2, 3]")
.testEval("[0, 1, 2, 3, 4, 5][-2:-1]", "[4]")
.testEval("[][1:2]", "[]")
.testEval("[0, 1, 2, 3][-10:10]", "[0, 1, 2, 3]");
}
@Test
public void testListSlice_WrongType() throws Exception {
new BothModesTest()
.testIfExactError("slice start must be an integer, not 'a'", "'123'['a'::]")
.testIfExactError("slice end must be an integer, not 'b'", "'123'[:'b':]");
}
@Test
public void testListSliceStep() throws Exception {
new BothModesTest()
.testEval("[1, 2, 3, 4, 5][::1]", "[1, 2, 3, 4, 5]")
.testEval("[1, 2, 3, 4, 5][1::1]", "[2, 3, 4, 5]")
.testEval("[1, 2, 3, 4, 5][:2:1]", "[1, 2]")
.testEval("[1, 2, 3, 4, 5][1:3:1]", "[2, 3]")
.testEval("[1, 2, 3, 4, 5][-4:-2:1]", "[2, 3]")
.testEval("[1, 2, 3, 4, 5][-10:10:1]", "[1, 2, 3, 4, 5]")
.testEval("[1, 2, 3, 4, 5][::42]", "[1]");
}
@Test
public void testListSliceStep_EmptyList() throws Exception {
new BothModesTest().testEval("[][::1]", "[]").testEval("[][::-1]", "[]");
}
@Test
public void testListSliceStep_SkipValues() throws Exception {
new BothModesTest()
.testEval("[1, 2, 3, 4, 5, 6, 7][::3]", "[1, 4, 7]")
.testEval("[1, 2, 3, 4, 5, 6, 7, 8, 9][1:7:3]", "[2, 5]");
}
@Test
public void testListSliceStep_Negative() throws Exception {
new BothModesTest()
.testEval("[1, 2, 3, 4, 5][::-1]", "[5, 4, 3, 2, 1]")
.testEval("[1, 2, 3, 4, 5][4::-1]", "[5, 4, 3, 2, 1]")
.testEval("[1, 2, 3, 4, 5][:0:-1]", "[5, 4, 3, 2]")
.testEval("[1, 2, 3, 4, 5][3:1:-1]", "[4, 3]")
.testEval("[1, 2, 3, 4, 5][::-2]", "[5, 3, 1]")
.testEval("[1, 2, 3, 4, 5][::-10]", "[5]");
}
@Test
public void testListSlice_WrongOrder() throws Exception {
new BothModesTest().testEval("[1, 2, 3][3:1:1]", "[]").testEval("[1, 2, 3][1:3:-1]", "[]");
}
@Test
public void testListSliceStep_InvalidStep() throws Exception {
String msg = "slice step cannot be zero";
new BothModesTest()
.testIfExactError(msg, "[1, 2, 3][::0]")
.testIfExactError(msg, "[1, 2, 3][1::0]")
.testIfExactError(msg, "[1, 2, 3][:3:0]")
.testIfExactError(msg, "[1, 2, 3][1:3:0]");
}
@Test
public void testTupleSlice() throws Exception {
// Not as comprehensive as the tests for slicing lists since the underlying mechanism is the
// same.
new BothModesTest()
.testEval("()[1:2]", "()")
.testEval("()[::1]", "()")
.testEval("(0, 1, 2, 3)[0:-1]", "(0, 1, 2)")
.testEval("(0, 1, 2, 3, 4, 5)[2:4]", "(2, 3)")
.testEval("(0, 1, 2, 3)[-10:10]", "(0, 1, 2, 3)")
.testEval("(1, 2, 3, 4, 5)[-10:10:1]", "(1, 2, 3, 4, 5)")
.testEval("(1, 2, 3, 4, 5, 6, 7, 8, 9)[1:7:3]", "(2, 5)")
.testEval("(1, 2, 3, 4, 5)[::-1]", "(5, 4, 3, 2, 1)")
.testEval("(1, 2, 3, 4, 5)[3:1:-1]", "(4, 3)")
.testEval("(1, 2, 3, 4, 5)[::-2]", "(5, 3, 1)")
.testEval("(1, 2, 3, 4, 5)[::-10]", "(5,)")
.testIfExactError("slice step cannot be zero", "(1, 2, 3)[1::0]");
}
@Test
public void testListSort() throws Exception {
new BothModesTest()
.testEval("sorted([0,1,2,3])", "[0, 1, 2, 3]")
.testEval("sorted([])", "[]")
.testEval("sorted([3, 2, 1, 0])", "[0, 1, 2, 3]")
.testEval("sorted([[1], [], [2], [1, 2]])", "[[], [1], [1, 2], [2]]")
.testEval("sorted([True, False, True])", "[False, True, True]")
.testEval("sorted(['a','x','b','z'])", "[\"a\", \"b\", \"x\", \"z\"]")
.testEval("sorted({1: True, 5: True, 4: False})", "[1, 4, 5]")
.testEval("sorted(depset([1, 5, 4]))", "[1, 4, 5]")
.testIfExactError("Cannot compare function with function", "sorted([sorted, sorted])");
}
@Test
public void testDictionaryCopy() throws Exception {
new BothModesTest()
.setUp("x = {1 : 2}", "y = dict(x)")
.testEval("x[1] == 2 and y[1] == 2", "True");
}
@Test
public void testDictionaryCopyKeyCollision() throws Exception {
new BothModesTest()
.setUp("x = {'test' : 2}", "y = dict(x, test = 3)")
.testEval("y['test']", "3");
}
@Test
public void testDictionaryKeyNotFound() throws Exception {
new BothModesTest()
.testIfErrorContains("key \"0\" not found in dictionary", "{}['0']")
.testIfErrorContains("key 0 not found in dictionary", "{'0': 1, 2: 3, 4: 5}[0]");
}
@Test
public void testListAccessBadIndex() throws Exception {
new BothModesTest()
.testIfErrorContains("indices must be integers, not string", "[[1], [2]]['a']");
}
@Test
public void testDictionaryAccess() throws Exception {
new BothModesTest()
.testEval("{1: ['foo']}[1]", "['foo']")
.testStatement("{'4': 8}['4']", 8)
.testStatement("{'a': 'aa', 'b': 'bb', 'c': 'cc'}['b']", "bb");
}
@Test
public void testDictionaryVariableAccess() throws Exception {
new BothModesTest().setUp("d = {'a' : 1}", "a = d['a']\n").testLookup("a", 1);
}
@Test
public void testStringIndexing() throws Exception {
new BothModesTest()
.testStatement("'somestring'[0]", "s")
.testStatement("'somestring'[1]", "o")
.testStatement("'somestring'[4]", "s")
.testStatement("'somestring'[9]", "g")
.testStatement("'somestring'[-1]", "g")
.testStatement("'somestring'[-2]", "n")
.testStatement("'somestring'[-10]", "s");
}
@Test
public void testStringIndexingOutOfRange() throws Exception {
new BothModesTest()
.testIfErrorContains("index out of range", "'abcdef'[10]")
.testIfErrorContains("index out of range", "'abcdef'[-11]")
.testIfErrorContains("index out of range", "'abcdef'[42]");
}
@Test
public void testStringSlice() throws Exception {
new BothModesTest()
.testStatement("'0123'[0:-1]", "012")
.testStatement("'012345'[2:4]", "23")
.testStatement("'012345'[-2:-1]", "4")
.testStatement("''[1:2]", "")
.testStatement("'012'[1:0]", "")
.testStatement("'0123'[-10:10]", "0123");
}
@Test
public void testStringSlice_WrongType() throws Exception {
new BothModesTest()
.testIfExactError("slice start must be an integer, not 'a'", "'123'['a'::]")
.testIfExactError("slice end must be an integer, not 'b'", "'123'[:'b':]");
}
@Test
public void testStringSliceStep() throws Exception {
new BothModesTest()
.testStatement("'01234'[::1]", "01234")
.testStatement("'01234'[1::1]", "1234")
.testStatement("'01234'[:2:1]", "01")
.testStatement("'01234'[1:3:1]", "12")
.testStatement("'01234'[-4:-2:1]", "12")
.testStatement("'01234'[-10:10:1]", "01234")
.testStatement("'01234'[::42]", "0");
}
@Test
public void testStringSliceStep_EmptyString() throws Exception {
new BothModesTest().testStatement("''[::1]", "").testStatement("''[::-1]", "");
}
@Test
public void testStringSliceStep_SkipValues() throws Exception {
new BothModesTest()
.testStatement("'0123456'[::3]", "036")
.testStatement("'01234567'[1:7:3]", "14");
}
@Test
public void testStringSliceStep_Negative() throws Exception {
new BothModesTest()
.testStatement("'01234'[::-1]", "43210")
.testStatement("'01234'[4::-1]", "43210")
.testStatement("'01234'[:0:-1]", "4321")
.testStatement("'01234'[3:1:-1]", "32")
.testStatement("'01234'[::-2]", "420")
.testStatement("'01234'[::-10]", "4");
}
@Test
public void testStringSliceStep_WrongOrder() throws Exception {
new BothModesTest().testStatement("'123'[3:1:1]", "").testStatement("'123'[1:3:-1]", "");
}
@Test
public void testStringSliceStep_InvalidStep() throws Exception {
String msg = "slice step cannot be zero";
new BothModesTest()
.testIfExactError(msg, "'123'[::0]")
.testIfExactError(msg, "'123'[1::0]")
.testIfExactError(msg, "'123'[:3:0]")
.testIfExactError(msg, "'123'[1:3:0]");
}
@Test
public void testDictionaryCreation() throws Exception {
String expected = "{'a': 1, 'b': 2, 'c': 3}";
new BothModesTest()
.testEval("dict([('a', 1), ('b', 2), ('c', 3)])", expected)
.testEval("dict(a = 1, b = 2, c = 3)", expected)
.testEval("dict([('a', 1)], b = 2, c = 3)", expected);
}
@Test
public void testDictionaryCreationInnerLists() throws Exception {
new BothModesTest().testEval("dict([[1, 2], [3, 4]], a = 5)", "{1: 2, 3: 4, 'a': 5}");
}
@Test
public void testDictionaryCreationEmpty() throws Exception {
new BothModesTest().testEval("dict()", "{}").testEval("dict([])", "{}");
}
@Test
public void testDictionaryCreationDifferentKeyTypes() throws Exception {
String expected = "{'a': 1, 2: 3}";
new BothModesTest()
.testEval("dict([('a', 1), (2, 3)])", expected)
.testEval("dict([(2, 3)], a = 1)", expected);
}
@Test
public void testDictionaryCreationKeyCollision() throws Exception {
String expected = "{'a': 1, 'b': 2, 'c': 3}";
new BothModesTest()
.testEval("dict([('a', 42), ('b', 2), ('a', 1), ('c', 3)])", expected)
.testEval("dict([('a', 42)], a = 1, b = 2, c = 3)", expected);
new SkylarkTest().testEval("dict([('a', 42)], **{'a': 1, 'b': 2, 'c': 3})", expected);
}
@Test
public void testDictionaryCreationInvalidPositional() throws Exception {
new BothModesTest()
.testIfErrorContains(
"expected value of type 'list(object)' for parameter args in dict(), "
+ "but got \"a\" (string)",
"dict('a')")
.testIfErrorContains("cannot convert item #0 to a sequence", "dict(['a'])")
.testIfErrorContains("cannot convert item #0 to a sequence", "dict([('a')])")
.testIfErrorContains("too many (3) positional arguments", "dict((3,4), (3,2), (1,2))")
.testIfErrorContains(
"item #0 has length 3, but exactly two elements are required",
"dict([('a', 'b', 'c')])");
}
@Test
public void testDictionaryValues() throws Exception {
new BothModesTest()
.testEval("{1: 'foo'}.values()", "['foo']")
.testEval("{}.values()", "[]")
.testEval("{True: 3, False: 5}.values()", "[3, 5]")
.testEval("{'a': 5, 'c': 2, 'b': 4, 'd': 3}.values()", "[5, 2, 4, 3]");
// sorted by keys
}
@Test
public void testDictionaryKeys() throws Exception {
new BothModesTest()
.testEval("{1: 'foo'}.keys()", "[1]")
.testEval("{}.keys()", "[]")
.testEval("{True: 3, False: 5}.keys()", "[True, False]")
.testEval(
"{1:'a', 2:'b', 6:'c', 0:'d', 5:'e', 4:'f', 3:'g'}.keys()", "[1, 2, 6, 0, 5, 4, 3]");
}
@Test
public void testDictionaryGet() throws Exception {
new BuildTest()
.testStatement("{1: 'foo'}.get(1)", "foo")
.testStatement("{1: 'foo'}.get(2)", Runtime.NONE)
.testStatement("{1: 'foo'}.get(2, 'a')", "a")
.testStatement("{1: 'foo'}.get(2, default='a')", "a")
.testStatement("{1: 'foo'}.get(2, default=None)", Runtime.NONE);
}
@Test
public void testDictionaryItems() throws Exception {
new BothModesTest()
.testEval("{'a': 'foo'}.items()", "[('a', 'foo')]")
.testEval("{}.items()", "[]")
.testEval("{1: 3, 2: 5}.items()", "[(1, 3), (2, 5)]")
.testEval("{'a': 5, 'c': 2, 'b': 4}.items()", "[('a', 5), ('c', 2), ('b', 4)]");
}
@Test
public void testDictionaryClear() throws Exception {
new SkylarkTest()
.testEval(
"d = {1: 'foo', 2: 'bar', 3: 'baz'}\n"
+ "if len(d) != 3: fail('clear 1')\n"
+ "if d.clear() != None: fail('clear 2')\n"
+ "d",
"{}");
}
@Test
public void testDictionaryPop() throws Exception {
new SkylarkTest()
.testIfErrorContains(
"KeyError: 1",
"d = {1: 'foo', 2: 'bar', 3: 'baz'}\n"
+ "if len(d) != 3: fail('pop 1')\n"
+ "if d.pop(2) != 'bar': fail('pop 2')\n"
+ "if d.pop(3, 'quux') != 'baz': fail('pop 3a')\n"
+ "if d.pop(3, 'quux') != 'quux': fail('pop 3b')\n"
+ "if d.pop(1) != 'foo': fail('pop 1')\n"
+ "if d != {}: fail('pop 0')\n"
+ "d.pop(1)");
}
@Test
public void testDictionaryPopItem() throws Exception {
new SkylarkTest()
.testIfErrorContains(
"popitem(): dictionary is empty",
"d = {2: 'bar', 3: 'baz', 1: 'foo'}\n"
+ "if len(d) != 3: fail('popitem 0')\n"
+ "if d.popitem() != (2, 'bar'): fail('popitem 2')\n"
+ "if d.popitem() != (3, 'baz'): fail('popitem 3')\n"
+ "if d.popitem() != (1, 'foo'): fail('popitem 1')\n"
+ "if d != {}: fail('popitem 4')\n"
+ "d.popitem()");
}
@Test
public void testDictionaryUpdate() throws Exception {
new BothModesTest()
.setUp("foo = {'a': 2}")
.testEval("foo.update({'b': 4}); foo", "{'a': 2, 'b': 4}");
new BothModesTest()
.setUp("foo = {'a': 2}")
.testEval("foo.update({'a': 3, 'b': 4}); foo", "{'a': 3, 'b': 4}");
}
@Test
public void testDictionarySetDefault() throws Exception {
new SkylarkTest()
.testEval(
"d = {2: 'bar', 1: 'foo'}\n"
+ "if len(d) != 2: fail('setdefault 0')\n"
+ "if d.setdefault(1, 'a') != 'foo': fail('setdefault 1')\n"
+ "if d.setdefault(2) != 'bar': fail('setdefault 2')\n"
+ "if d.setdefault(3) != None: fail('setdefault 3')\n"
+ "if d.setdefault(4, 'b') != 'b': fail('setdefault 4')\n"
+ "d",
"{1: 'foo', 2: 'bar', 3: None, 4: 'b'}");
}
@Test
public void testListIndexMethod() throws Exception {
new BothModesTest()
.testStatement("['a', 'b', 'c'].index('a')", 0)
.testStatement("['a', 'b', 'c'].index('b')", 1)
.testStatement("['a', 'b', 'c'].index('c')", 2)
.testStatement("[2, 4, 6].index(4)", 1)
.testStatement("[2, 4, 6].index(4)", 1)
.testStatement("[0, 1, [1]].index([1])", 2)
.testIfErrorContains("item \"a\" not found in list", "[1, 2].index('a')")
.testIfErrorContains("item 0 not found in list", "[].index(0)");
}
@Test
public void testListIndex() throws Exception {
new BothModesTest()
.testStatement("['a', 'b', 'c', 'd'][0]", "a")
.testStatement("['a', 'b', 'c', 'd'][1]", "b")
.testStatement("['a', 'b', 'c', 'd'][-1]", "d")
.testStatement("['a', 'b', 'c', 'd'][-2]", "c")
.testStatement("[0, 1, 2][-3]", 0)
.testStatement("[0, 1, 2][-2]", 1)
.testStatement("[0, 1, 2][-1]", 2)
.testStatement("[0, 1, 2][0]", 0);
}
@Test
public void testListIndexOutOfRange() throws Exception {
new BothModesTest()
.testIfErrorContains(
"index out of range (index is 3, but sequence has 3 elements)", "[0, 1, 2][3]")
.testIfErrorContains(
"index out of range (index is -4, but sequence has 3 elements)", "[0, 1, 2][-4]")
.testIfErrorContains(
"index out of range (index is -2, but sequence has 1 elements)", "[0][-2]")
.testIfErrorContains(
"index out of range (index is 1, but sequence has 1 elements)", "[0][1]")
.testIfErrorContains(
"index out of range (index is 1, but sequence has 0 elements)", "[][1]");
}
@Test
public void testHash() throws Exception {
// We specify the same string hashing algorithm as String.hashCode().
new SkylarkTest()
.testStatement("hash('skylark')", "skylark".hashCode())
.testStatement("hash('google')", "google".hashCode())
.testIfErrorContains(
"argument 'value' has type 'NoneType', but should be 'string'\n"
+ "in call to builtin function hash(value)",
"hash(None)");
}
@Test
public void testRange() throws Exception {
new BothModesTest()
.testStatement("str(range(5))", "[0, 1, 2, 3, 4]")
.testStatement("str(range(0))", "[]")
.testStatement("str(range(1))", "[0]")
.testStatement("str(range(-2))", "[]")
.testStatement("str(range(-3, 2))", "[-3, -2, -1, 0, 1]")
.testStatement("str(range(3, 2))", "[]")
.testStatement("str(range(3, 3))", "[]")
.testStatement("str(range(3, 4))", "[3]")
.testStatement("str(range(3, 5))", "[3, 4]")
.testStatement("str(range(-3, 5, 2))", "[-3, -1, 1, 3]")
.testStatement("str(range(-3, 6, 2))", "[-3, -1, 1, 3, 5]")
.testStatement("str(range(5, 0, -1))", "[5, 4, 3, 2, 1]")
.testStatement("str(range(5, 0, -10))", "[5]")
.testStatement("str(range(0, -3, -2))", "[0, -2]")
.testIfErrorContains("step cannot be 0", "range(2, 3, 0)");
}
@Test
public void testEnumerate() throws Exception {
new BothModesTest()
.testStatement("str(enumerate([]))", "[]")
.testStatement("str(enumerate([5]))", "[(0, 5)]")
.testStatement("str(enumerate([5, 3]))", "[(0, 5), (1, 3)]")
.testStatement("str(enumerate(['a', 'b', 'c']))", "[(0, \"a\"), (1, \"b\"), (2, \"c\")]")
.testStatement("str(enumerate(['a']) + [(1, 'b')])", "[(0, \"a\"), (1, \"b\")]");
}
@Test
public void testEnumerateBadArg() throws Exception {
new BothModesTest()
.testIfErrorContains(
"argument 'list' has type 'string', but should be 'sequence'\n"
+ "in call to builtin function enumerate(list)",
"enumerate('a')");
}
@Test
public void testPyListAppend() throws Exception {
new BuildTest()
.setUp("FOO = ['a', 'b']", "FOO.insert(0, 'c')")
.testLookup("FOO", MutableList.of(env, "c", "a", "b"))
.setUp("FOO.insert(1, 'd')")
.testLookup("FOO", MutableList.of(env, "c", "d", "a", "b"))
.setUp("FOO.insert(4, 'e')")
.testLookup("FOO", MutableList.of(env, "c", "d", "a", "b", "e"))
.setUp("FOO.insert(-10, 'f')")
.testLookup("FOO", MutableList.of(env, "f", "c", "d", "a", "b", "e"))
.setUp("FOO.insert(10, 'g')")
.testLookup("FOO", MutableList.of(env, "f", "c", "d", "a", "b", "e", "g"))
.testIfErrorContains("type 'tuple' has no method insert(int)", "(1, 2).insert(3)");
}
@Test
public void testPyListInsert() throws Exception {
new BuildTest()
.setUp("FOO = ['a', 'b']", "FOO.append('c')")
.testLookup("FOO", MutableList.of(env, "a", "b", "c"))
.testIfErrorContains("type 'tuple' has no method append(int)", "(1, 2).append(3)");
}
@Test
public void testPyListExtend() throws Exception {
new BuildTest()
.setUp("FOO = ['a', 'b']", "FOO.extend(['c', 'd'])", "FOO.extend(('e', 'f'))")
.testLookup("FOO", MutableList.of(env, "a", "b", "c", "d", "e", "f"))
.testIfErrorContains("type 'tuple' has no method extend(list)", "(1, 2).extend([3, 4])")
.testIfErrorContains(
"argument 'items' has type 'int', but should be 'sequence'\n"
+ "in call to builtin method list.extend(items)",
"[1, 2].extend(3)");
}
@Test
public void testListRemove() throws Exception {
new BothModesTest()
.setUp("foo = ['a', 'b', 'c', 'b']", "foo.remove('b')")
.testLookup("foo", MutableList.of(env, "a", "c", "b"))
.setUp("foo.remove('c')")
.testLookup("foo", MutableList.of(env, "a", "b"))
.setUp("foo.remove('a')")
.testLookup("foo", MutableList.of(env, "b"))
.setUp("foo.remove('b')")
.testLookup("foo", MutableList.of(env))
.testIfErrorContains("item 3 not found in list", "[1, 2].remove(3)");
new BothModesTest()
.testIfErrorContains("type 'tuple' has no method remove(int)", "(1, 2).remove(3)");
}
@Test
public void testListPop() throws Exception {
new BothModesTest()
.setUp("li = [2, 3, 4]; ret = li.pop()")
.testLookup("li", MutableList.of(env, 2, 3))
.testLookup("ret", 4);
new BothModesTest()
.setUp("li = [2, 3, 4]; ret = li.pop(-2)")
.testLookup("li", MutableList.of(env, 2, 4))
.testLookup("ret", 3);
new BothModesTest()
.setUp("li = [2, 3, 4]; ret = li.pop(1)")
.testLookup("li", MutableList.of(env, 2, 4))
.testLookup("ret", 3);
new BothModesTest()
.testIfErrorContains(
"index out of range (index is 3, but sequence has 2 elements)", "[1, 2].pop(3)");
new BothModesTest().testIfErrorContains("type 'tuple' has no method pop()", "(1, 2).pop()");
}
@Test
public void testReassignmentOfPrimitivesNotForbiddenByCoreLanguage() throws Exception {
new BuildTest()
.setUp("cc_binary = (['hello.cc'])")
.testIfErrorContains(
"'list' object is not callable",
"cc_binary(name = 'hello', srcs=['hello.cc'], malloc = '//base:system_malloc')");
}
@Test
public void testLenOnString() throws Exception {
new BothModesTest().testStatement("len('abc')", 3);
}
@Test
public void testLenOnList() throws Exception {
new BothModesTest().testStatement("len([1,2,3])", 3);
}
@Test
public void testLenOnDict() throws Exception {
new BothModesTest().testStatement("len({'a' : 1, 'b' : 2})", 2);
}
@Test
public void testLenOnBadType() throws Exception {
new BothModesTest().testIfErrorContains("int is not iterable", "len(1)");
}
@Test
public void testIndexOnFunction() throws Exception {
new BothModesTest()
.testIfErrorContains("type 'function' has no operator [](int)", "len[1]")
.testIfErrorContains("type 'function' has no operator [:](int, int, int)", "len[1:4]");
}
@Test
public void testBool() throws Exception {
new BothModesTest()
.testStatement("bool(1)", Boolean.TRUE)
.testStatement("bool(0)", Boolean.FALSE)
.testStatement("bool([1, 2])", Boolean.TRUE)
.testStatement("bool([])", Boolean.FALSE)
.testStatement("bool(None)", Boolean.FALSE);
}
@Test
public void testStr() throws Exception {
new BothModesTest()
.testStatement("str(1)", "1")
.testStatement("str(-2)", "-2")
.testStatement("str([1, 2])", "[1, 2]")
.testStatement("str(True)", "True")
.testStatement("str(False)", "False")
.testStatement("str(None)", "None")
.testStatement("str(str)", "<function str>");
}
@Test
public void testInt() throws Exception {
new BothModesTest()
.testStatement("int('1')", 1)
.testStatement("int('-1234')", -1234)
.testIfErrorContains("invalid literal for int() with base 10: \"1.5\"", "int('1.5')")
.testIfErrorContains("invalid literal for int() with base 10: \"ab\"", "int('ab')")
.testStatement("int(42)", 42)
.testStatement("int(-1)", -1)
.testStatement("int(True)", 1)
.testStatement("int(False)", 0)
.testIfErrorContains("None is not of type string or int or bool", "int(None)");
}
@Test
public void testIntWithBase() throws Exception {
new BothModesTest()
.testStatement("int('11', 2)", 3)
.testStatement("int('11', 9)", 10)
.testStatement("int('AF', 16)", 175)
.testStatement("int('11', 36)", 37)
.testStatement("int('az', 36)", 395);
}
@Test
public void testIntWithBase_InvalidBase() throws Exception {
new BothModesTest()
.testIfExactError("invalid literal for int() with base 3: \"123\"", "int('123', 3)")
.testIfExactError("invalid literal for int() with base 15: \"FF\"", "int('FF', 15)")
.testIfExactError("int() base must be >= 2 and <= 36", "int('123', -1)")
.testIfExactError("int() base must be >= 2 and <= 36", "int('123', 1)")
.testIfExactError("int() base must be >= 2 and <= 36", "int('123', 37)");
}
@Test
public void testIntWithBase_Prefix() throws Exception {
new BothModesTest()
.testStatement("int('0b11', 0)", 3)
.testStatement("int('0B11', 2)", 3)
.testStatement("int('0o11', 0)", 9)
.testStatement("int('0O11', 8)", 9)
.testStatement("int('0XFF', 0)", 255)
.testStatement("int('0xFF', 16)", 255)
.testIfExactError("invalid literal for int() with base 8: \"0xFF\"", "int('0xFF', 8)");
}
@Test
public void testIntWithBase_NoString() throws Exception {
new BothModesTest()
.testIfExactError("int() can't convert non-string with explicit base", "int(True, 2)")
.testIfExactError("int() can't convert non-string with explicit base", "int(1, 2)");
}
@Test
public void testStrFunction() throws Exception {
new SkylarkTest().testStatement("def foo(x): return x\nstr(foo)", "<function foo>");
}
@Test
public void testType() throws Exception {
new SkylarkTest()
.testStatement("type(1)", "int")
.testStatement("type('a')", "string")
.testStatement("type([1, 2])", "list")
.testStatement("type((1, 2))", "tuple")
.testStatement("type(True)", "bool")
.testStatement("type(None)", "NoneType")
.testStatement("type(str)", "function");
}
// TODO(bazel-team): Move this into a new BazelLibraryTest.java file, or at least out of
// MethodLibraryTest.java.
@Test
public void testSelectFunction() throws Exception {
enableSkylarkMode();
eval("a = select({'a': 1})");
SelectorList result = (SelectorList) lookup("a");
assertThat(((SelectorValue) Iterables.getOnlyElement(result.getElements())).getDictionary())
.containsExactly("a", 1);
}
@Test
public void testCountFunction() throws Exception {
new BothModesTest()
.testStatement("'abc'.count('')", 4)
.testStatement("'abc'.count('a')", 1)
.testStatement("'abc'.count('b')", 1)
.testStatement("'abc'.count('c')", 1)
.testStatement("'abbc'.count('b')", 2)
.testStatement("'aba'.count('a')", 2)
.testStatement("'aaa'.count('aa')", 1)
.testStatement("'aaaa'.count('aa')", 2)
.testStatement("'abc'.count('a', 0)", 1)
.testStatement("'abc'.count('a', 1)", 0)
.testStatement("'abc'.count('c', 0, 3)", 1)
.testStatement("'abc'.count('c', 0, 2)", 0)
.testStatement("'abc'.count('a', -1)", 0)
.testStatement("'abc'.count('c', -1)", 1)
.testStatement("'abc'.count('c', 0, 5)", 1)
.testStatement("'abc'.count('c', 0, -1)", 0)
.testStatement("'abc'.count('a', 0, -1)", 1);
}
@Test
public void testZipFunction() throws Exception {
new BothModesTest()
.testStatement("str(zip())", "[]")
.testStatement("str(zip([1, 2]))", "[(1,), (2,)]")
.testStatement("str(zip([1, 2], ['a', 'b']))", "[(1, \"a\"), (2, \"b\")]")
.testStatement("str(zip([1, 2, 3], ['a', 'b']))", "[(1, \"a\"), (2, \"b\")]")
.testStatement("str(zip([1], [2], [3]))", "[(1, 2, 3)]")
.testStatement("str(zip([1], {2: 'a'}))", "[(1, 2)]")
.testStatement("str(zip([1], []))", "[]")
.testIfErrorContains("type 'int' is not iterable", "zip(123)")
.testIfErrorContains("type 'int' is not iterable", "zip([1], 1)")
.testStatement("str(zip([1], depset([2])))", "[(1, 2)]");
}
@Test
public void testIsAlphaFunction() throws Exception {
new BothModesTest()
.testStatement("''.isalpha()", false)
.testStatement("'abz'.isalpha()", true)
.testStatement("'a1'.isalpha()", false)
.testStatement("'a '.isalpha()", false)
.testStatement("'A'.isalpha()", true)
.testStatement("'AbZ'.isalpha()", true);
}
/**
* Assert that lstrip(), rstrip(), and strip() produce the expected result for a given input
* string and chars argument. If chars is null no argument is passed.
*/
private void checkStrip(
String input, Object chars,
String expLeft, String expRight, String expBoth) throws Exception {
if (chars == null) {
new BothModesTest()
.update("s", input)
.testStatement("s.lstrip()", expLeft)
.testStatement("s.rstrip()", expRight)
.testStatement("s.strip()", expBoth);
} else {
new BothModesTest()
.update("s", input)
.update("chars", chars)
.testStatement("s.lstrip(chars)", expLeft)
.testStatement("s.rstrip(chars)", expRight)
.testStatement("s.strip(chars)", expBoth);
}
}
@Test
public void testStrip() throws Exception {
// Strip nothing.
checkStrip("a b c", "", "a b c", "a b c", "a b c");
checkStrip(" a b c ", "", " a b c ", " a b c ", " a b c ");
// Normal case, found and not found.
checkStrip("abcba", "ba", "cba", "abc", "c");
checkStrip("abc", "xyz", "abc", "abc", "abc");
// Default whitespace.
checkStrip(" a b c ", null, "a b c ", " a b c", "a b c");
checkStrip(" a b c ", Runtime.NONE, "a b c ", " a b c", "a b c");
// Default whitespace with full range of Latin-1 whitespace chars.
String whitespace = "\u0009\n\u000B\u000C\r\u001C\u001D\u001E\u001F\u0020\u0085\u00A0";
checkStrip(
whitespace + "a" + whitespace, null,
"a" + whitespace, whitespace + "a", "a");
checkStrip(
whitespace + "a" + whitespace, Runtime.NONE,
"a" + whitespace, whitespace + "a", "a");
// Empty cases.
checkStrip("", "", "", "", "");
checkStrip("abc", "abc", "", "", "");
checkStrip("", "xyz", "", "", "");
checkStrip("", null, "", "", "");
}
@Test
public void testFail() throws Exception {
new SkylarkTest()
.testIfErrorContains("abc", "fail('abc')")
.testIfErrorContains("18", "fail(18)");
}
@Test
public void testTupleCoercion() throws Exception {
new BothModesTest()
.testStatement("tuple([1, 2]) == (1, 2)", true)
.testStatement("tuple(depset([1, 2])) == (1, 2)", true)
// Depends on current implementation of dict
.testStatement("tuple({1: 'foo', 2: 'bar'}) == (1, 2)", true);
}
}