// Copyright 2016 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.skyframe; import static com.google.common.truth.Truth.assertThat; import static org.junit.Assert.assertEquals; import static org.junit.Assert.assertFalse; import static org.junit.Assert.assertTrue; import com.google.common.collect.ImmutableList; import com.google.devtools.build.lib.analysis.util.BuildViewTestCase; import com.google.devtools.build.lib.cmdline.Label; import com.google.devtools.build.lib.cmdline.PackageIdentifier; import com.google.devtools.build.lib.packages.NoSuchPackageException; import com.google.devtools.build.lib.skyframe.util.SkyframeExecutorTestUtils; import com.google.devtools.build.lib.syntax.SkylarkImport; import com.google.devtools.build.lib.testutil.TestConstants; import com.google.devtools.build.lib.vfs.FileStatus; import com.google.devtools.build.lib.vfs.FileSystem; import com.google.devtools.build.lib.vfs.Path; import com.google.devtools.build.lib.vfs.inmemoryfs.InMemoryFileSystem; import com.google.devtools.build.skyframe.ErrorInfo; import com.google.devtools.build.skyframe.EvaluationResult; import com.google.devtools.build.skyframe.SkyKey; import java.io.IOException; import org.junit.Test; import org.junit.runner.RunWith; import org.junit.runners.JUnit4; /** * Unit tests of specific functionality of ASTFileLookupFunction. */ @RunWith(JUnit4.class) public class ASTFileLookupFunctionTest extends BuildViewTestCase { private static class MockFileSystem extends InMemoryFileSystem { boolean statThrowsIoException; @Override public FileStatus stat(Path path, boolean followSymlinks) throws IOException { if (statThrowsIoException && path.asFragment() .getPathString() .equals("/workspace/tools/build_rules/prelude_" + TestConstants.PRODUCT_NAME)) { throw new IOException("bork"); } return super.stat(path, followSymlinks); } } private MockFileSystem mockFS; @Override protected FileSystem createFileSystem() { mockFS = new MockFileSystem(); return mockFS; } @Test public void testPreludeASTFileIsNotMandatory() throws Exception { reporter.removeHandler(failFastHandler); scratch.file( "foo/BUILD", "genrule(name = 'foo',", " outs = ['out.txt'],", " cmd = 'echo hello >@')"); scratch.deleteFile("tools/build_rules/prelude_blaze"); invalidatePackages(); SkyKey skyKey = PackageValue.key(PackageIdentifier.parse("@//foo")); EvaluationResult<PackageValue> result = SkyframeExecutorTestUtils.evaluate( getSkyframeExecutor(), skyKey, /*keepGoing=*/ false, reporter); assertFalse(result.hasError()); assertFalse(result.get(skyKey).getPackage().containsErrors()); } @Test public void testIOExceptionOccursDuringReading() throws Exception { reporter.removeHandler(failFastHandler); scratch.file("/workspace/tools/build_rules/BUILD"); scratch.file( "foo/BUILD", "genrule(name = 'foo',", " outs = ['out.txt'],", " cmd = 'echo hello >@')"); mockFS.statThrowsIoException = true; invalidatePackages(/*alsoConfigs=*/false); // We don't want to fail early on config creation. SkyKey skyKey = PackageValue.key(PackageIdentifier.parse("@//foo")); EvaluationResult<PackageValue> result = SkyframeExecutorTestUtils.evaluate( getSkyframeExecutor(), skyKey, /*keepGoing=*/ false, reporter); assertTrue(result.hasError()); ErrorInfo errorInfo = result.getError(skyKey); Throwable e = errorInfo.getException(); assertEquals(skyKey, errorInfo.getRootCauseOfException()); assertThat(e).isInstanceOf(NoSuchPackageException.class); assertThat(e.getMessage()).contains("bork"); } @Test public void testLoadFromBuildFileInRemoteRepo() throws Exception { scratch.deleteFile("tools/build_rules/prelude_blaze"); scratch.overwriteFile("WORKSPACE", "local_repository(", " name = 'a_remote_repo',", " path = '/a_remote_repo'", ")"); scratch.file("/a_remote_repo/WORKSPACE"); scratch.file("/a_remote_repo/remote_pkg/BUILD", "load(':ext.bzl', 'CONST')"); scratch.file("/a_remote_repo/remote_pkg/ext.bzl", "CONST = 17"); invalidatePackages(/*alsoConfigs=*/false); // Repository shuffling messes with toolchains. SkyKey skyKey = ASTFileLookupValue.key(Label.parseAbsoluteUnchecked("@a_remote_repo//remote_pkg:BUILD")); EvaluationResult<ASTFileLookupValue> result = SkyframeExecutorTestUtils.evaluate( getSkyframeExecutor(), skyKey, /*keepGoing=*/ false, reporter); ImmutableList<SkylarkImport> imports = result.get(skyKey).getAST().getImports(); assertThat(imports).hasSize(1); assertThat(imports.get(0).getImportString()).isEqualTo(":ext.bzl"); } @Test public void testLoadFromSkylarkFileInRemoteRepo() throws Exception { scratch.deleteFile("tools/build_rules/prelude_blaze"); scratch.overwriteFile("WORKSPACE", "local_repository(", " name = 'a_remote_repo',", " path = '/a_remote_repo'", ")"); scratch.file("/a_remote_repo/WORKSPACE"); scratch.file("/a_remote_repo/remote_pkg/BUILD"); scratch.file("/a_remote_repo/remote_pkg/ext1.bzl", "load(':ext2.bzl', 'CONST')"); scratch.file("/a_remote_repo/remote_pkg/ext2.bzl", "CONST = 17"); invalidatePackages(/*alsoConfigs=*/false); // Repository shuffling messes with toolchains. SkyKey skyKey = ASTFileLookupValue.key(Label.parseAbsoluteUnchecked("@a_remote_repo//remote_pkg:ext1.bzl")); EvaluationResult<ASTFileLookupValue> result = SkyframeExecutorTestUtils.evaluate( getSkyframeExecutor(), skyKey, /*keepGoing=*/ false, reporter); ImmutableList<SkylarkImport> imports = result.get(skyKey).getAST().getImports(); assertThat(imports).hasSize(1); assertThat(imports.get(0).getImportString()).isEqualTo(":ext2.bzl"); } @Test public void testLoadWithNonExistentBuildFile() throws Exception { invalidatePackages(); SkyKey skyKey = ASTFileLookupValue.key(Label.parseAbsoluteUnchecked("@a_remote_repo//remote_pkg:BUILD")); EvaluationResult<ASTFileLookupValue> result = SkyframeExecutorTestUtils.evaluate( getSkyframeExecutor(), skyKey, /*keepGoing=*/ false, reporter); assertThat(result.get(skyKey).lookupSuccessful()).isFalse(); assertThat(result.get(skyKey).getErrorMsg()) .contains("Unable to load package for '@a_remote_repo//remote_pkg:BUILD'"); assertThat(result.get(skyKey).getErrorMsg()) .contains("The repository could not be resolved"); } }