| // 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", |
| "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", |
| "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(env.lookup("x"), 1); |
| env.update("y", 2); |
| assertEquals(env.lookup("y"), 2); |
| assertEquals(env.lookup("x"), 1); |
| env.update("x", 3); |
| assertEquals(env.lookup("x"), 3); |
| } |
| 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(new PathFragment("/a")); |
| Location locB = Location.fromPathFragment(new PathFragment("/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 (IllegalArgumentException e) { |
| assertThat(e.getMessage()).contains("ERROR 1:1: 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."); |
| } |
| } |
| } |