// 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 static org.junit.Assert.assertEquals; import static org.junit.Assert.assertNull; import static org.junit.Assert.fail; import com.google.common.collect.Sets; import com.google.devtools.build.lib.events.Location; import com.google.devtools.build.lib.syntax.util.EvaluationTestCase; import com.google.devtools.build.lib.vfs.PathFragment; import org.junit.Test; import org.junit.runner.RunWith; import org.junit.runners.JUnit4; /** * Tests of Environment. */ @RunWith(JUnit4.class) public class EnvironmentTest extends EvaluationTestCase { @Override public Environment newEnvironment() { return newBuildEnvironment(); } // Test the API directly @Test public void testLookupAndUpdate() throws Exception { assertNull(lookup("foo")); update("foo", "bar"); assertEquals("bar", lookup("foo")); } @Test public void testHasVariable() throws Exception { assertThat(getEnvironment().hasVariable("VERSION")).isFalse(); update("VERSION", 42); assertThat(getEnvironment().hasVariable("VERSION")).isTrue(); } @Test public void testDoubleUpdateSucceeds() throws Exception { update("VERSION", 42); assertEquals(42, lookup("VERSION")); update("VERSION", 43); assertEquals(43, lookup("VERSION")); } // Test assign through interpreter, lookup through API: @Test public void testAssign() throws Exception { assertNull(lookup("foo")); eval("foo = 'bar'"); assertEquals("bar", lookup("foo")); } // Test update through API, reference through interpreter: @Test public void testReference() throws Exception { setFailFast(false); try { eval("foo"); fail(); } catch (EvalException e) { assertThat(e).hasMessage("name 'foo' is not defined"); } update("foo", "bar"); assertEquals("bar", eval("foo")); } // Test assign and reference through interpreter: @Test public void testAssignAndReference() throws Exception { setFailFast(false); try { eval("foo"); fail(); } catch (EvalException e) { assertThat(e).hasMessage("name 'foo' is not defined"); } eval("foo = 'bar'"); assertEquals("bar", eval("foo")); } @Test public void testGetVariableNames() throws Exception { Environment outerEnv; Environment innerEnv; try (Mutability mut = Mutability.create("outer")) { outerEnv = Environment.builder(mut) .setGlobals(Environment.DEFAULT_GLOBALS) .build() .update("foo", "bar") .update("wiz", 3); } try (Mutability mut = Mutability.create("inner")) { innerEnv = Environment.builder(mut) .setGlobals(outerEnv.getGlobals()).build() .update("foo", "bat") .update("quux", 42); } assertEquals( Sets.newHashSet( "foo", "wiz", "False", "None", "True", "-", "all", "any", "bool", "dict", "dir", "enumerate", "fail", "getattr", "hasattr", "hash", "int", "len", "list", "max", "min", "print", "range", "repr", "reversed", "sorted", "str", "tuple", "zip"), outerEnv.getVariableNames()); assertEquals( Sets.newHashSet( "foo", "wiz", "quux", "False", "None", "True", "-", "all", "any", "bool", "dict", "dir", "enumerate", "fail", "getattr", "hasattr", "hash", "int", "len", "list", "max", "min", "print", "range", "repr", "reversed", "sorted", "str", "tuple", "zip"), innerEnv.getVariableNames()); } @Test public void testToString() throws Exception { update("subject", new StringLiteral("Hello, 'world'.")); update("from", new StringLiteral("Java")); assertThat(getEnvironment().toString()).isEqualTo("<Environment[test]>"); } @Test public void testBindToNullThrowsException() throws Exception { try { update("some_name", null); fail(); } catch (NullPointerException e) { assertThat(e).hasMessage("update(value == null)"); } } @Test public void testFrozen() throws Exception { Environment env; try (Mutability mutability = Mutability.create("testFrozen")) { env = Environment.builder(mutability) .setGlobals(Environment.DEFAULT_GLOBALS) .setEventHandler(Environment.FAIL_FAST_HANDLER) .build(); env.update("x", 1); assertEquals(1, env.lookup("x")); env.update("y", 2); assertEquals(2, env.lookup("y")); assertEquals(1, env.lookup("x")); env.update("x", 3); assertEquals(3, env.lookup("x")); } try { // This update to an existing variable should fail because the environment was frozen. env.update("x", 4); throw new Exception("failed to fail"); // not an AssertionError like fail() } catch (AssertionError e) { assertThat(e).hasMessage("Can't update x to 4 in frozen environment"); } try { // This update to a new variable should also fail because the environment was frozen. env.update("newvar", 5); throw new Exception("failed to fail"); // not an AssertionError like fail() } catch (AssertionError e) { assertThat(e).hasMessage("Can't update newvar to 5 in frozen environment"); } } @Test public void testLocked() throws Exception { final Mutability mutability = Mutability.create("testLocked"); class DummyFreezable implements Mutability.Freezable { @Override public Mutability mutability() { return mutability; } } DummyFreezable dummy = new DummyFreezable(); Location locA = Location.fromPathFragment(PathFragment.create("/a")); Location locB = Location.fromPathFragment(PathFragment.create("/b")); Environment env = Environment.builder(mutability).build(); // Acquire two locks, release two locks, check along the way. assertThat(mutability.isLocked(dummy)).isFalse(); mutability.lock(dummy, locA); assertThat(mutability.isLocked(dummy)).isTrue(); mutability.lock(dummy, locB); assertThat(mutability.isLocked(dummy)).isTrue(); assertThat(mutability.getLockLocations(dummy)).containsExactly(locA, locB); mutability.unlock(dummy, locA); assertThat(mutability.isLocked(dummy)).isTrue(); try { Mutability.checkMutable(dummy, env); fail("Able to mutate locked object"); } catch (Mutability.MutabilityException e) { assertThat(e).hasMessage("trying to mutate a locked object (is it currently being iterated " + "over by a for loop or comprehension?)\n" + "Object locked at the following location(s): /b:1"); } try { mutability.unlock(dummy, locA); fail("Able to unlock object with wrong location"); } catch (AssertionError e) { assertThat(e).hasMessage("trying to unlock an object for a location at which " + "it was not locked (/a:1)"); } mutability.unlock(dummy, locB); assertThat(mutability.isLocked(dummy)).isFalse(); Mutability.checkMutable(dummy, env); // Acquire, then freeze. mutability.lock(dummy, locA); mutability.freeze(); assertThat(mutability.isLocked(dummy)).isFalse(); try { Mutability.checkMutable(dummy, env); fail("Able to mutate locked object"); } catch (Mutability.MutabilityException e) { assertThat(e).hasMessage("trying to mutate a frozen object"); } } @Test public void testReadOnly() throws Exception { Environment env = newSkylarkEnvironment() .setup("special_var", 42) .update("global_var", 666); // We don't even get a runtime exception trying to modify these, // because we get compile-time exceptions even before we reach runtime! try { BuildFileAST.eval(env, "special_var = 41"); throw new AssertionError("failed to fail"); } catch (EvalException e) { assertThat(e.getMessage()).contains("Variable special_var is read only"); } try { BuildFileAST.eval(env, "def foo(x): x += global_var; global_var = 36; return x", "foo(1)"); throw new AssertionError("failed to fail"); } catch (EvalExceptionWithStackTrace e) { assertThat(e.getMessage()).contains("Variable 'global_var' is referenced before assignment. " + "The variable is defined in the global scope."); } } }