| // Copyright 2015 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 com.google.devtools.build.lib.bazel.bzlmod.BzlmodTestUtil.createModuleKey; |
| import static com.google.devtools.build.skyframe.EvaluationResultSubjectFactory.assertThatEvaluationResult; |
| import static org.junit.Assert.assertThrows; |
| import static org.junit.Assert.fail; |
| |
| import com.google.common.collect.ImmutableList; |
| import com.google.common.collect.ImmutableMap; |
| import com.google.devtools.build.lib.analysis.util.BuildViewTestCase; |
| import com.google.devtools.build.lib.bazel.bzlmod.BazelLockFileFunction; |
| import com.google.devtools.build.lib.bazel.bzlmod.BazelModuleResolutionFunction; |
| import com.google.devtools.build.lib.bazel.bzlmod.FakeRegistry; |
| import com.google.devtools.build.lib.bazel.bzlmod.ModuleFileFunction; |
| import com.google.devtools.build.lib.bazel.repository.RepositoryOptions.BazelCompatibilityMode; |
| import com.google.devtools.build.lib.bazel.repository.RepositoryOptions.CheckDirectDepsMode; |
| import com.google.devtools.build.lib.bazel.repository.RepositoryOptions.LockfileMode; |
| import com.google.devtools.build.lib.clock.BlazeClock; |
| import com.google.devtools.build.lib.cmdline.BazelModuleContext; |
| import com.google.devtools.build.lib.cmdline.Label; |
| import com.google.devtools.build.lib.packages.RuleVisibility; |
| import com.google.devtools.build.lib.packages.semantics.BuildLanguageOptions; |
| import com.google.devtools.build.lib.pkgcache.PackageOptions; |
| import com.google.devtools.build.lib.pkgcache.PathPackageLocator; |
| import com.google.devtools.build.lib.runtime.QuiescingExecutorsImpl; |
| import com.google.devtools.build.lib.skyframe.BzlLoadFunction.BzlLoadFailedException; |
| import com.google.devtools.build.lib.skyframe.PrecomputedValue.Injected; |
| import com.google.devtools.build.lib.skyframe.util.SkyframeExecutorTestUtils; |
| import com.google.devtools.build.lib.util.io.TimestampGranularityMonitor; |
| import com.google.devtools.build.lib.vfs.DigestHashFunction; |
| 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.PathFragment; |
| import com.google.devtools.build.lib.vfs.Root; |
| import com.google.devtools.build.lib.vfs.RootedPath; |
| 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 com.google.devtools.common.options.Options; |
| import java.io.IOException; |
| import java.io.InputStream; |
| import java.util.UUID; |
| import javax.annotation.Nullable; |
| import net.starlark.java.eval.StarlarkInt; |
| import org.junit.Before; |
| import org.junit.Test; |
| import org.junit.runner.RunWith; |
| import org.junit.runners.JUnit4; |
| |
| /** Tests for BzlLoadFunction. */ |
| @RunWith(JUnit4.class) |
| public class BzlLoadFunctionTest extends BuildViewTestCase { |
| private Path moduleRoot; |
| private FakeRegistry registry; |
| |
| @Override |
| protected FileSystem createFileSystem() { |
| return new CustomInMemoryFs(); |
| } |
| |
| @Before |
| public final void preparePackageLoading() throws Exception { |
| Path alternativeRoot = scratch.dir("/root_2"); |
| PackageOptions packageOptions = Options.getDefaults(PackageOptions.class); |
| packageOptions.defaultVisibility = RuleVisibility.PUBLIC; |
| packageOptions.showLoadingProgress = true; |
| packageOptions.globbingThreads = 7; |
| getSkyframeExecutor() |
| .preparePackageLoading( |
| new PathPackageLocator( |
| outputBase, |
| ImmutableList.of(Root.fromPath(rootDirectory), Root.fromPath(alternativeRoot)), |
| BazelSkyframeExecutorConstants.BUILD_FILES_BY_PRIORITY), |
| packageOptions, |
| Options.getDefaults(BuildLanguageOptions.class), |
| UUID.randomUUID(), |
| ImmutableMap.of(), |
| QuiescingExecutorsImpl.forTesting(), |
| new TimestampGranularityMonitor(BlazeClock.instance())); |
| skyframeExecutor.setActionEnv(ImmutableMap.of()); |
| } |
| |
| @Override |
| protected ImmutableList<Injected> extraPrecomputedValues() { |
| try { |
| moduleRoot = scratch.dir("modules"); |
| } catch (IOException e) { |
| throw new IllegalStateException(e); |
| } |
| registry = FakeRegistry.DEFAULT_FACTORY.newFakeRegistry(moduleRoot.getPathString()); |
| return ImmutableList.of( |
| PrecomputedValue.injected( |
| ModuleFileFunction.REGISTRIES, ImmutableList.of(registry.getUrl())), |
| PrecomputedValue.injected(ModuleFileFunction.IGNORE_DEV_DEPS, false), |
| PrecomputedValue.injected(ModuleFileFunction.MODULE_OVERRIDES, ImmutableMap.of()), |
| PrecomputedValue.injected( |
| BazelModuleResolutionFunction.ALLOWED_YANKED_VERSIONS, ImmutableList.of()), |
| PrecomputedValue.injected( |
| BazelModuleResolutionFunction.CHECK_DIRECT_DEPENDENCIES, CheckDirectDepsMode.WARNING), |
| PrecomputedValue.injected( |
| BazelModuleResolutionFunction.BAZEL_COMPATIBILITY_MODE, BazelCompatibilityMode.ERROR), |
| PrecomputedValue.injected(BazelLockFileFunction.LOCKFILE_MODE, LockfileMode.OFF)); |
| } |
| |
| @Before |
| public void setUpForBzlmod() throws Exception { |
| scratch.file("MODULE.bazel"); |
| } |
| |
| @Test |
| public void testBzlLoadLabels() throws Exception { |
| scratch.file("pkg1/BUILD"); |
| scratch.file("pkg1/ext.bzl"); |
| checkSuccessfulLookup("//pkg1:ext.bzl"); |
| |
| scratch.file("pkg2/BUILD"); |
| scratch.file("pkg2/dir/ext.bzl"); |
| checkSuccessfulLookup("//pkg2:dir/ext.bzl"); |
| |
| scratch.file("dir/pkg3/BUILD"); |
| scratch.file("dir/pkg3/dir/ext.bzl"); |
| checkSuccessfulLookup("//dir/pkg3:dir/ext.bzl"); |
| } |
| |
| @Test |
| public void testBzlLoadLabelsAlternativeRoot() throws Exception { |
| scratch.file("/root_2/pkg4/BUILD"); |
| scratch.file("/root_2/pkg4/ext.bzl"); |
| checkSuccessfulLookup("//pkg4:ext.bzl"); |
| } |
| |
| @Test |
| public void testBzlLoadLabelsMultipleBuildFiles() throws Exception { |
| scratch.file("dir1/BUILD"); |
| scratch.file("dir1/dir2/BUILD"); |
| scratch.file("dir1/dir2/ext.bzl"); |
| checkSuccessfulLookup("//dir1/dir2:ext.bzl"); |
| } |
| |
| @Test |
| public void testLoadFromStarlarkFileInRemoteRepo() throws Exception { |
| 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"); |
| checkSuccessfulLookup("@a_remote_repo//remote_pkg:ext1.bzl"); |
| } |
| |
| @Test |
| public void testLoadRelativeLabel() throws Exception { |
| scratch.file("pkg/BUILD"); |
| scratch.file("pkg/ext1.bzl", "a = 1"); |
| scratch.file("pkg/ext2.bzl", "load(':ext1.bzl', 'a')"); |
| checkSuccessfulLookup("//pkg:ext2.bzl"); |
| } |
| |
| @Test |
| public void testLoadAbsoluteLabel() throws Exception { |
| scratch.file("pkg2/BUILD"); |
| scratch.file("pkg3/BUILD"); |
| scratch.file("pkg2/ext.bzl", "b = 1"); |
| scratch.file("pkg3/ext.bzl", "load('//pkg2:ext.bzl', 'b')"); |
| checkSuccessfulLookup("//pkg3:ext.bzl"); |
| } |
| |
| @Test |
| public void testLoadFromSameAbsoluteLabelTwice() throws Exception { |
| scratch.file("pkg1/BUILD"); |
| scratch.file("pkg2/BUILD"); |
| scratch.file("pkg1/ext.bzl", "a = 1", "b = 2"); |
| scratch.file("pkg2/ext.bzl", "load('//pkg1:ext.bzl', 'a')", "load('//pkg1:ext.bzl', 'b')"); |
| checkSuccessfulLookup("//pkg2:ext.bzl"); |
| } |
| |
| @Test |
| public void testLoadFromSameRelativeLabelTwice() throws Exception { |
| scratch.file("pkg/BUILD"); |
| scratch.file("pkg/ext1.bzl", "a = 1", "b = 2"); |
| scratch.file("pkg/ext2.bzl", "load(':ext1.bzl', 'a')", "load(':ext1.bzl', 'b')"); |
| checkSuccessfulLookup("//pkg:ext2.bzl"); |
| } |
| |
| @Test |
| public void testLoadFromRelativeLabelInSubdir() throws Exception { |
| scratch.file("pkg/BUILD"); |
| scratch.file("pkg/subdir/ext1.bzl", "a = 1"); |
| scratch.file("pkg/subdir/ext2.bzl", "load(':subdir/ext1.bzl', 'a')"); |
| checkSuccessfulLookup("//pkg:subdir/ext2.bzl"); |
| } |
| |
| @Test |
| public void testLoadBadExtension_sclDisabled() throws Exception { |
| setBuildLanguageOptions("--experimental_enable_scl_dialect=false"); |
| |
| scratch.file("pkg/BUILD"); |
| scratch.file("pkg/ext.bzl", "load(':foo.garbage', 'a')"); |
| reporter.removeHandler(failFastHandler); |
| checkFailingLookup("//pkg:ext.bzl", "has invalid load statements"); |
| assertContainsEvent("The label must reference a file with extension \".bzl\""); |
| assertDoesNotContainEvent(".scl"); |
| } |
| |
| @Test |
| public void testLoadBadExtension_sclEnabled() throws Exception { |
| setBuildLanguageOptions("--experimental_enable_scl_dialect=true"); |
| |
| scratch.file("pkg/BUILD"); |
| scratch.file("pkg/ext.bzl", "load(':foo.garbage', 'a')"); |
| reporter.removeHandler(failFastHandler); |
| checkFailingLookup("//pkg:ext.bzl", "has invalid load statements"); |
| assertContainsEvent("The label must reference a file with extension \".bzl\" or \".scl\""); |
| } |
| |
| @Test |
| public void testLoadingSclRequiresExperimentalFlag() throws Exception { |
| setBuildLanguageOptions("--experimental_enable_scl_dialect=false"); |
| |
| scratch.file("pkg/BUILD"); |
| scratch.file("pkg/ext.scl"); |
| reporter.removeHandler(failFastHandler); |
| checkFailingLookup( |
| "//pkg:ext.scl", "loading .scl files requires setting --experimental_enable_scl_dialect"); |
| } |
| |
| @Test |
| public void testCanLoadScl() throws Exception { |
| setBuildLanguageOptions("--experimental_enable_scl_dialect=true"); |
| |
| scratch.file("pkg/BUILD"); |
| scratch.file("pkg/ext.scl"); |
| checkSuccessfulLookup("//pkg:ext.scl"); |
| } |
| |
| @Test |
| public void testCanLoadSclFromBzlAndScl() throws Exception { |
| setBuildLanguageOptions("--experimental_enable_scl_dialect=true"); |
| |
| scratch.file("pkg/BUILD"); |
| scratch.file("pkg/ext1.scl", "a = 1"); |
| // Can use relative load label syntax from ext2a.bzl, but not from ext2b.scl. |
| scratch.file("pkg/ext2a.bzl", "load(':ext1.scl', 'a')"); |
| scratch.file("pkg/ext2b.scl", "load('//pkg:ext1.scl', 'a')"); |
| |
| checkSuccessfulLookup("//pkg:ext2a.bzl"); |
| checkSuccessfulLookup("//pkg:ext2b.scl"); |
| } |
| |
| @Test |
| public void testSclCannotLoadNonSclFiles() throws Exception { |
| setBuildLanguageOptions("--experimental_enable_scl_dialect=true"); |
| |
| scratch.file("pkg/BUILD"); |
| scratch.file("pkg/ext1a.bzl", "a = 1"); |
| scratch.file("pkg/ext1a.garbage", "a = 1"); |
| // Cannot use relative label. |
| scratch.file("pkg/ext2a.scl", "load('//pkg:ext1a.bzl', 'a')"); |
| scratch.file("pkg/ext2b.scl", "load('//pkg:ext1b.garbage', 'a')"); |
| |
| reporter.removeHandler(failFastHandler); |
| checkFailingLookup("//pkg:ext2a.scl", "has invalid load statements"); |
| assertContainsEvent( |
| "The label must reference a file with extension \".scl\" (.scl files cannot load .bzl" |
| + " files)"); |
| eventCollector.clear(); |
| checkFailingLookup("//pkg:ext2b.scl", "has invalid load statements"); |
| assertContainsEvent("The label must reference a file with extension \".scl\""); |
| assertDoesNotContainEvent(".bzl"); |
| } |
| |
| @Test |
| public void testSclCanOnlyLoadLabelsRelativeToDefaultRepoRoot() throws Exception { |
| setBuildLanguageOptions("--experimental_enable_scl_dialect=true"); |
| |
| scratch.file("pkg/BUILD"); |
| scratch.file("pkg/ext1.scl", "load(':foo.scl', 'a')"); |
| scratch.file("pkg/ext2.scl", "load('@repo//:foo.scl', 'a')"); |
| |
| reporter.removeHandler(failFastHandler); |
| checkFailingLookup("//pkg:ext1.scl", "has invalid load statements"); |
| assertContainsEvent("in .scl files, load labels must begin with \"//\""); |
| eventCollector.clear(); |
| checkFailingLookup("//pkg:ext2.scl", "has invalid load statements"); |
| assertContainsEvent("in .scl files, load labels must begin with \"//\""); |
| } |
| |
| @Test |
| public void testSclSupportsStructAndVisibility() throws Exception { |
| setBuildLanguageOptions("--experimental_enable_scl_dialect=true"); |
| |
| scratch.file("pkg/BUILD"); |
| scratch.file( |
| "pkg/ext1.scl", // |
| "visibility('private')", |
| "a = struct()"); |
| scratch.file( |
| "pkg/ext2.scl", // |
| "load('//pkg:ext1.scl', 'a')"); |
| scratch.file("pkg2/BUILD"); |
| scratch.file( |
| "pkg2/ext3.scl", // |
| "load('//pkg:ext1.scl', 'a')"); |
| |
| checkSuccessfulLookup("//pkg:ext2.scl"); |
| reporter.removeHandler(failFastHandler); |
| checkFailingLookup( |
| "//pkg2:ext3.scl", "module //pkg2:ext3.scl contains .bzl load visibility violations"); |
| } |
| |
| @Test |
| public void testSclDoesNotSupportOtherBazelSymbols() throws Exception { |
| setBuildLanguageOptions("--experimental_enable_scl_dialect=true"); |
| |
| scratch.file("pkg/BUILD"); |
| scratch.file( |
| "pkg/ext.scl", // |
| "a = depset([])"); |
| |
| reporter.removeHandler(failFastHandler); |
| checkFailingLookup("//pkg:ext.scl", "compilation of module 'pkg/ext.scl' failed"); |
| assertContainsEvent("name 'depset' is not defined"); |
| } |
| |
| private EvaluationResult<BzlLoadValue> get(SkyKey skyKey) throws Exception { |
| EvaluationResult<BzlLoadValue> result = |
| SkyframeExecutorTestUtils.evaluate( |
| getSkyframeExecutor(), skyKey, /*keepGoing=*/ false, reporter); |
| if (result.hasError()) { |
| fail(result.getError(skyKey).getException().getMessage()); |
| } |
| return result; |
| } |
| |
| private static SkyKey key(String label) { |
| return BzlLoadValue.keyForBuild(Label.parseCanonicalUnchecked(label)); |
| } |
| |
| /** Loads a .bzl with the given label and asserts success. */ |
| private void checkSuccessfulLookup(String label) throws Exception { |
| SkyKey skyKey = key(label); |
| EvaluationResult<BzlLoadValue> result = get(skyKey); |
| // Ensure that the file has been processed by checking its Module for the label field. |
| assertThat(label) |
| .isEqualTo(BazelModuleContext.of(result.get(skyKey).getModule()).label().toString()); |
| } |
| |
| /* Loads a .bzl with the given label and asserts BzlLoadFailedException with the given message. */ |
| private void checkFailingLookup(String label, String expectedMessage) |
| throws InterruptedException { |
| SkyKey skyKey = key(label); |
| EvaluationResult<BzlLoadValue> result = |
| SkyframeExecutorTestUtils.evaluate( |
| getSkyframeExecutor(), skyKey, /*keepGoing=*/ false, reporter); |
| assertThat(result.hasError()).isTrue(); |
| assertThatEvaluationResult(result) |
| .hasErrorEntryForKeyThat(skyKey) |
| .hasExceptionThat() |
| .isInstanceOf(BzlLoadFailedException.class); |
| assertThatEvaluationResult(result) |
| .hasErrorEntryForKeyThat(skyKey) |
| .hasExceptionThat() |
| .hasMessageThat() |
| .contains(expectedMessage); |
| } |
| |
| @Test |
| public void testBzlLoadNoBuildFile() throws Exception { |
| scratch.file("pkg/ext.bzl", ""); |
| SkyKey skyKey = key("//pkg:ext.bzl"); |
| EvaluationResult<BzlLoadValue> result = |
| SkyframeExecutorTestUtils.evaluate( |
| getSkyframeExecutor(), skyKey, /*keepGoing=*/ false, reporter); |
| assertThat(result.hasError()).isTrue(); |
| ErrorInfo errorInfo = result.getError(skyKey); |
| String errorMessage = errorInfo.getException().getMessage(); |
| assertThat(errorMessage) |
| .contains( |
| "Every .bzl file must have a corresponding package, but '//pkg:ext.bzl' does not"); |
| } |
| |
| @Test |
| public void testBzlLoadNoBuildFileForLoad() throws Exception { |
| scratch.file("pkg2/BUILD"); |
| scratch.file("pkg1/ext.bzl", "a = 1"); |
| scratch.file("pkg2/ext.bzl", "load('//pkg1:ext.bzl', 'a')"); |
| SkyKey skyKey = key("//pkg:ext.bzl"); |
| EvaluationResult<BzlLoadValue> result = |
| SkyframeExecutorTestUtils.evaluate( |
| getSkyframeExecutor(), skyKey, /*keepGoing=*/ false, reporter); |
| assertThat(result.hasError()).isTrue(); |
| ErrorInfo errorInfo = result.getError(skyKey); |
| String errorMessage = errorInfo.getException().getMessage(); |
| assertThat(errorMessage).contains("Every .bzl file must have a corresponding package"); |
| } |
| |
| @Test |
| public void testBzlLoadFilenameWithControlChars() throws Exception { |
| scratch.file("pkg/BUILD", ""); |
| scratch.file("pkg/ext.bzl", "load('//pkg:oops\u0000.bzl', 'a')"); |
| SkyKey skyKey = key("//pkg:ext.bzl"); |
| AssertionError e = |
| assertThrows( |
| AssertionError.class, |
| () -> |
| SkyframeExecutorTestUtils.evaluate( |
| getSkyframeExecutor(), skyKey, /*keepGoing=*/ false, reporter)); |
| String errorMessage = e.getMessage(); |
| assertThat(errorMessage) |
| .contains( |
| "invalid target name 'oops<?>.bzl': " |
| + "target names may not contain non-printable characters: '\\x00'"); |
| } |
| |
| @Test |
| public void testLoadFromExternalRepoInWorkspaceFileAllowed() throws Exception { |
| Path p = |
| 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/ext.bzl", "CONST = 17"); |
| |
| RootedPath rootedPath = |
| RootedPath.toRootedPath( |
| Root.fromPath(p.getParentDirectory()), PathFragment.create("WORKSPACE")); |
| |
| SkyKey skyKey = |
| BzlLoadValue.keyForWorkspace( |
| Label.parseCanonicalUnchecked("@a_remote_repo//remote_pkg:ext.bzl"), |
| /* inWorkspace= */ |
| /* workspaceChunk= */ 0, |
| rootedPath); |
| EvaluationResult<BzlLoadValue> result = |
| SkyframeExecutorTestUtils.evaluate( |
| getSkyframeExecutor(), skyKey, /*keepGoing=*/ false, reporter); |
| |
| assertThat(result.hasError()).isFalse(); |
| } |
| |
| @Test |
| public void testLoadFromSubdirInSamePackageIsOk() throws Exception { |
| scratch.file("a/BUILD"); |
| scratch.file("a/a.bzl", "load('//a:b/b.bzl', 'b')"); |
| scratch.file("a/b/b.bzl", "b = 42"); |
| |
| checkSuccessfulLookup("//a:a.bzl"); |
| } |
| |
| @Test |
| public void testLoadMustRespectPackageBoundary_ofSubpkg() throws Exception { |
| scratch.file("a/BUILD"); |
| scratch.file("a/a.bzl", "load('//a:b/b.bzl', 'b')"); |
| scratch.file("a/b/BUILD", ""); |
| scratch.file("a/b/b.bzl", "b = 42"); |
| checkFailingLookup( |
| "//a:a.bzl", |
| "Label '//a:b/b.bzl' is invalid because 'a/b' is a subpackage; perhaps you meant to" |
| + " put the colon here: '//a/b:b.bzl'?"); |
| } |
| |
| @Test |
| public void testLoadMustRespectPackageBoundary_ofSubpkg_relative() throws Exception { |
| scratch.file("a/BUILD"); |
| scratch.file("a/a.bzl", "load('b/b.bzl', 'b')"); |
| scratch.file("a/b/BUILD", ""); |
| scratch.file("a/b/b.bzl", "b = 42"); |
| checkFailingLookup( |
| "//a:a.bzl", |
| "Label '//a:b/b.bzl' is invalid because 'a/b' is a subpackage; perhaps you meant to" |
| + " put the colon here: '//a/b:b.bzl'?"); |
| } |
| |
| @Test |
| public void testLoadMustRespectPackageBoundary_ofIndirectSubpkg() throws Exception { |
| scratch.file("a/BUILD"); |
| scratch.file("a/a.bzl", "load('//a/b:c/c.bzl', 'c')"); |
| scratch.file("a/b/BUILD", ""); |
| scratch.file("a/b/c/BUILD", ""); |
| scratch.file("a/b/c/c.bzl", "c = 42"); |
| checkFailingLookup( |
| "//a:a.bzl", |
| "Label '//a/b:c/c.bzl' is invalid because 'a/b/c' is a subpackage; perhaps you meant" |
| + " to put the colon here: '//a/b/c:c.bzl'?"); |
| } |
| |
| @Test |
| public void testLoadMustRespectPackageBoundary_ofParentPkg() throws Exception { |
| scratch.file("a/b/BUILD"); |
| scratch.file("a/b/b.bzl", "load('//a/c:c/c.bzl', 'c')"); |
| scratch.file("a/BUILD"); |
| scratch.file("a/c/c/c.bzl", "c = 42"); |
| checkFailingLookup( |
| "//a/b:b.bzl", |
| "Label '//a/c:c/c.bzl' is invalid because 'a/c' is not a package; perhaps you meant to " |
| + "put the colon here: '//a:c/c/c.bzl'?"); |
| } |
| |
| @Test |
| public void testBzlVisibility_disabledWithoutFlag() throws Exception { |
| setBuildLanguageOptions("--experimental_bzl_visibility=false"); |
| |
| scratch.file("a/BUILD"); |
| scratch.file( |
| "a/foo.bzl", // |
| "load(\"//b:bar.bzl\", \"x\")"); |
| scratch.file("b/BUILD"); |
| scratch.file( |
| "b/bar.bzl", // |
| "visibility(\"private\")", |
| "x = 1"); |
| |
| reporter.removeHandler(failFastHandler); |
| checkFailingLookup("//a:foo.bzl", "initialization of module 'b/bar.bzl' failed"); |
| assertContainsEvent("Use of `visibility()` requires --experimental_bzl_visibility"); |
| } |
| |
| @Test |
| public void testBzlVisibility_publicExplicit() throws Exception { |
| setBuildLanguageOptions("--experimental_bzl_visibility=true"); |
| |
| scratch.file("a/BUILD"); |
| scratch.file( |
| "a/foo.bzl", // |
| "load(\"//b:bar.bzl\", \"x\")"); |
| scratch.file("b/BUILD"); |
| scratch.file( |
| "b/bar.bzl", // |
| "visibility(\"public\")", |
| "x = 1"); |
| |
| checkSuccessfulLookup("//a:foo.bzl"); |
| assertNoEvents(); |
| } |
| |
| @Test |
| public void testBzlVisibility_publicImplicit() throws Exception { |
| setBuildLanguageOptions("--experimental_bzl_visibility=true"); |
| |
| scratch.file("a/BUILD"); |
| scratch.file( |
| "a/foo.bzl", // |
| "load(\"//b:bar.bzl\", \"x\")"); |
| scratch.file("b/BUILD"); |
| scratch.file( |
| "b/bar.bzl", |
| // No visibility() declaration, defaults to public. |
| "x = 1"); |
| |
| checkSuccessfulLookup("//a:foo.bzl"); |
| assertNoEvents(); |
| } |
| |
| @Test |
| public void testBzlVisibility_privateSamePackage() throws Exception { |
| setBuildLanguageOptions("--experimental_bzl_visibility=true"); |
| |
| scratch.file("a/BUILD"); |
| scratch.file( |
| "a/foo.bzl", // |
| "load(\"//a:bar.bzl\", \"x\")"); |
| scratch.file( |
| "a/bar.bzl", // |
| "visibility(\"private\")", |
| "x = 1"); |
| |
| checkSuccessfulLookup("//a:foo.bzl"); |
| assertNoEvents(); |
| } |
| |
| @Test |
| public void testBzlVisibility_privateDifferentPackage() throws Exception { |
| setBuildLanguageOptions("--experimental_bzl_visibility=true"); |
| |
| scratch.file("a/BUILD"); |
| scratch.file( |
| "a/foo.bzl", // |
| "load(\"//b:bar.bzl\", \"x\")"); |
| scratch.file("b/BUILD"); |
| scratch.file( |
| "b/bar.bzl", // |
| "visibility(\"private\")", |
| "x = 1"); |
| |
| reporter.removeHandler(failFastHandler); |
| checkFailingLookup( |
| "//a:foo.bzl", "module //a:foo.bzl contains .bzl load visibility violations"); |
| assertContainsEvent("Starlark file //b:bar.bzl is not visible for loading from package //a."); |
| } |
| |
| @Test |
| public void testBzlVisibility_emptyListMeansPrivate() throws Exception { |
| setBuildLanguageOptions("--experimental_bzl_visibility=true"); |
| |
| scratch.file("a/BUILD"); |
| scratch.file( |
| "a/foo.bzl", // |
| "load(\"//b:bar.bzl\", \"x\")"); |
| scratch.file("b/BUILD"); |
| scratch.file( |
| "b/bar.bzl", // |
| "visibility([])", |
| "x = 1"); |
| |
| reporter.removeHandler(failFastHandler); |
| checkFailingLookup( |
| "//a:foo.bzl", "module //a:foo.bzl contains .bzl load visibility violations"); |
| assertContainsEvent("Starlark file //b:bar.bzl is not visible for loading from package //a."); |
| } |
| |
| @Test |
| public void testBzlVisibility_publicListElement() throws Exception { |
| setBuildLanguageOptions("--experimental_bzl_visibility=true"); |
| |
| scratch.file("a/BUILD"); |
| scratch.file( |
| "a/foo.bzl", // |
| "load(\"//b:bar.bzl\", \"x\")"); |
| scratch.file("b/BUILD"); |
| scratch.file( |
| "b/bar.bzl", // |
| // Tests "public" as a list item, and alongside other list items. |
| "visibility([\"public\", \"//c\"])", |
| "x = 1"); |
| |
| checkSuccessfulLookup("//a:foo.bzl"); |
| assertNoEvents(); |
| } |
| |
| @Test |
| public void testBzlVisibility_privateListElement() throws Exception { |
| setBuildLanguageOptions("--experimental_bzl_visibility=true"); |
| |
| scratch.file("a1/BUILD"); |
| scratch.file( |
| "a1/foo.bzl", // |
| "load(\"//b:bar.bzl\", \"x\")"); |
| scratch.file("a2/BUILD"); |
| scratch.file( |
| "a2/foo.bzl", // |
| "load(\"//b:bar.bzl\", \"x\")"); |
| scratch.file("b/BUILD"); |
| scratch.file( |
| "b/bar.bzl", // |
| // Tests "private" as a list item, and alongside other list items. |
| "visibility([\"private\", \"//a1\"])", |
| "x = 1"); |
| |
| checkSuccessfulLookup("//a1:foo.bzl"); |
| assertNoEvents(); |
| reporter.removeHandler(failFastHandler); |
| checkFailingLookup( |
| "//a2:foo.bzl", "module //a2:foo.bzl contains .bzl load visibility violations"); |
| assertContainsEvent("Starlark file //b:bar.bzl is not visible for loading from package //a2."); |
| } |
| |
| @Test |
| public void testBzlVisibility_failureInDependency() throws Exception { |
| setBuildLanguageOptions("--experimental_bzl_visibility=true"); |
| |
| scratch.file("a/BUILD"); |
| scratch.file( |
| "a/foo.bzl", // |
| "load(\"//b:bar.bzl\", \"x\")"); |
| scratch.file("b/BUILD"); |
| scratch.file( |
| "b/bar.bzl", // |
| "load(\"//c:baz.bzl\", \"y\")", |
| "visibility(\"public\")", |
| "x = y"); |
| scratch.file("c/BUILD"); |
| scratch.file( |
| "c/baz.bzl", // |
| "visibility(\"private\")", |
| "y = 1"); |
| |
| reporter.removeHandler(failFastHandler); |
| checkFailingLookup( |
| "//a:foo.bzl", |
| "at /workspace/a/foo.bzl:1:6: module //b:bar.bzl contains .bzl load visibility violations"); |
| assertContainsEvent("Starlark file //c:baz.bzl is not visible for loading from package //b."); |
| } |
| |
| @Test |
| public void testBzlVisibility_cannotBeSetInFunction() throws Exception { |
| setBuildLanguageOptions("--experimental_bzl_visibility=true"); |
| |
| scratch.file("a/BUILD"); |
| scratch.file( |
| "a/foo.bzl", // |
| "def helper():", |
| " visibility(\"public\")", |
| "helper()"); |
| |
| reporter.removeHandler(failFastHandler); |
| checkFailingLookup("//a:foo.bzl", "initialization of module 'a/foo.bzl' failed"); |
| assertContainsEvent("load visibility may only be set at the top level"); |
| } |
| |
| @Test |
| public void testBzlVisibility_cannotBeSetTwice() throws Exception { |
| setBuildLanguageOptions("--experimental_bzl_visibility=true"); |
| |
| scratch.file("a/BUILD"); |
| scratch.file( |
| "a/foo.bzl", // |
| "visibility(\"public\")", |
| "visibility(\"public\")"); |
| |
| reporter.removeHandler(failFastHandler); |
| checkFailingLookup("//a:foo.bzl", "initialization of module 'a/foo.bzl' failed"); |
| assertContainsEvent("load visibility may not be set more than once"); |
| } |
| |
| @Test |
| public void testBzlVisibility_enumeratedPackages() throws Exception { |
| setBuildLanguageOptions("--experimental_bzl_visibility=true"); |
| |
| scratch.file("a1/BUILD"); |
| scratch.file( |
| "a1/foo1.bzl", // |
| "load(\"//b:bar.bzl\", \"x\")"); |
| scratch.file("a2/BUILD"); |
| scratch.file( |
| "a2/foo2.bzl", // |
| "load(\"//b:bar.bzl\", \"x\")"); |
| scratch.file("b/BUILD"); |
| scratch.file( |
| "b/bar.bzl", // |
| "visibility([\"//a1\"])", |
| "x = 1"); |
| |
| checkSuccessfulLookup("//a1:foo1.bzl"); |
| assertNoEvents(); |
| |
| reporter.removeHandler(failFastHandler); |
| checkFailingLookup( |
| "//a2:foo2.bzl", "module //a2:foo2.bzl contains .bzl load visibility violations"); |
| assertContainsEvent("Starlark file //b:bar.bzl is not visible for loading from package //a2."); |
| } |
| |
| @Test |
| public void testBzlVisibility_singleEnumeratedPackageAsString() throws Exception { |
| setBuildLanguageOptions("--experimental_bzl_visibility=true"); |
| |
| scratch.file("a1/BUILD"); |
| scratch.file( |
| "a1/foo1.bzl", // |
| "load(\"//b:bar.bzl\", \"x\")"); |
| scratch.file("a2/BUILD"); |
| scratch.file( |
| "a2/foo2.bzl", // |
| "load(\"//b:bar.bzl\", \"x\")"); |
| scratch.file("b/BUILD"); |
| scratch.file( |
| "b/bar.bzl", // |
| // Note: "//a1", not ["//a1"] |
| "visibility(\"//a1\")", |
| "x = 1"); |
| |
| checkSuccessfulLookup("//a1:foo1.bzl"); |
| assertNoEvents(); |
| |
| reporter.removeHandler(failFastHandler); |
| checkFailingLookup( |
| "//a2:foo2.bzl", "module //a2:foo2.bzl contains .bzl load visibility violations"); |
| assertContainsEvent("Starlark file //b:bar.bzl is not visible for loading from package //a2."); |
| } |
| |
| @Test |
| public void testBzlVisibility_enumeratedPackagesMultipleRepos() throws Exception { |
| setBuildLanguageOptions("--experimental_bzl_visibility=true"); |
| |
| // @repo//pkg:foo1.bzl and @//pkg:foo2.bzl both try to access @repo//lib:bar.bzl. Test that when |
| // bar.bzl declares a visibility allowing "//pkg", it means @repo//pkg and *not* @//pkg. |
| scratch.overwriteFile( |
| "WORKSPACE", // |
| "local_repository(", |
| " name = 'repo',", |
| " path = 'repo'", |
| ")"); |
| scratch.file("repo/WORKSPACE"); |
| scratch.file("repo/pkg/BUILD"); |
| scratch.file( |
| "repo/pkg/foo1.bzl", // |
| "load(\"//lib:bar.bzl\", \"x\")"); |
| scratch.file("repo/lib/BUILD"); |
| scratch.file( |
| "repo/lib/bar.bzl", // |
| "visibility([\"//pkg\"])", |
| "x = 1"); |
| scratch.file("pkg/BUILD"); |
| scratch.file( |
| "pkg/foo2.bzl", // |
| "load(\"@repo//lib:bar.bzl\", \"x\")"); |
| |
| checkSuccessfulLookup("@repo//pkg:foo1.bzl"); |
| assertNoEvents(); |
| |
| reporter.removeHandler(failFastHandler); |
| checkFailingLookup( |
| "//pkg:foo2.bzl", "module //pkg:foo2.bzl contains .bzl load visibility violations"); |
| assertContainsEvent( |
| "Starlark file @repo//lib:bar.bzl is not visible for loading from package //pkg."); |
| } |
| |
| // TODO(#16365): This test case can be deleted once --incompatible_package_group_has_public_syntax |
| // is deleted (not just flipped). |
| @Test |
| public void testBzlVisibility_canUsePublicPrivate_regardlessOfFlag() throws Exception { |
| setBuildLanguageOptions( |
| "--experimental_bzl_visibility=true", |
| // Test that we can use "public" and "private" visibility for .bzl files even when the |
| // incompatible flag is disabled. |
| "--incompatible_package_group_has_public_syntax=false"); |
| |
| scratch.file("a/BUILD"); |
| scratch.file( |
| "a/foo1.bzl", // |
| "visibility(\"public\")"); |
| scratch.file( |
| "a/foo2.bzl", // |
| "visibility(\"private\")"); |
| |
| checkSuccessfulLookup("//a:foo1.bzl"); |
| checkSuccessfulLookup("//a:foo2.bzl"); |
| assertNoEvents(); |
| } |
| |
| // TODO(#16324): Once --incompatible_fix_package_group_reporoot_syntax is deleted (not just |
| // flipped), this test case will be redundant with tests for //... in PackageGroupTest. At that |
| // point we'll just delete this test case. |
| @Test |
| public void testBzlVisibility_repoRootSubpackagesIsNotPublic_regardlessOfFlag() throws Exception { |
| setBuildLanguageOptions( |
| "--experimental_bzl_visibility=true", |
| // Test that we get the fixed behavior even when the incompatible flag is disabled. |
| "--incompatible_fix_package_group_reporoot_syntax=false"); |
| |
| scratch.overwriteFile( |
| "WORKSPACE", // |
| "local_repository(", |
| " name = 'repo',", |
| " path = 'repo'", |
| ")"); |
| scratch.file("repo/WORKSPACE"); |
| scratch.file("repo/a/BUILD"); |
| scratch.file( |
| "repo/a/foo.bzl", // |
| "load(\"@//b:bar.bzl\", \"x\")"); |
| scratch.file("b/BUILD"); |
| scratch.file( |
| "b/bar.bzl", // |
| "visibility([\"//...\"])", |
| "x = 1"); |
| |
| reporter.removeHandler(failFastHandler); |
| checkFailingLookup( |
| "@repo//a:foo.bzl", "module @repo//a:foo.bzl contains .bzl load visibility violations"); |
| assertContainsEvent( |
| "Starlark file //b:bar.bzl is not visible for loading from package @repo//a."); |
| } |
| |
| @Test |
| public void testBzlVisibility_disallowsSubpackagesWithoutWildcard() throws Exception { |
| setBuildLanguageOptions("--experimental_bzl_visibility=true"); |
| |
| scratch.file("a/BUILD"); |
| scratch.file( |
| "a/foo1.bzl", // |
| "load(\"//b:bar.bzl\", \"x\")"); |
| scratch.file("a/subpkg/BUILD"); |
| scratch.file( |
| "a/subpkg/foo2.bzl", // |
| "load(\"//b:bar.bzl\", \"x\")"); |
| scratch.file("b/BUILD"); |
| scratch.file( |
| "b/bar.bzl", // |
| "visibility([\"//a\"])", |
| "x = 1"); |
| |
| checkSuccessfulLookup("//a:foo1.bzl"); |
| assertNoEvents(); |
| |
| reporter.removeHandler(failFastHandler); |
| checkFailingLookup( |
| "//a/subpkg:foo2.bzl", |
| "module //a/subpkg:foo2.bzl contains .bzl load visibility violations"); |
| assertContainsEvent( |
| "Starlark file //b:bar.bzl is not visible for loading from package //a/subpkg."); |
| } |
| |
| @Test |
| public void testBzlVisibility_allowsSubpackagesWithWildcard() throws Exception { |
| setBuildLanguageOptions("--experimental_bzl_visibility=true"); |
| |
| scratch.file("a/BUILD"); |
| scratch.file( |
| "a/foo1.bzl", // |
| "load(\"//b:bar.bzl\", \"x\")"); |
| scratch.file("a/subpkg/BUILD"); |
| scratch.file( |
| "a/subpkg/foo2.bzl", // |
| "load(\"//b:bar.bzl\", \"x\")"); |
| scratch.file("b/BUILD"); |
| scratch.file( |
| "b/bar.bzl", // |
| "visibility([\"//a/...\"])", |
| "x = 1"); |
| |
| checkSuccessfulLookup("//a:foo1.bzl"); |
| assertNoEvents(); |
| |
| checkSuccessfulLookup("//a/subpkg:foo2.bzl"); |
| assertNoEvents(); |
| } |
| |
| @Test |
| public void testBzlVisibility_invalid_badType() throws Exception { |
| setBuildLanguageOptions("--experimental_bzl_visibility=true"); |
| |
| scratch.file("a/BUILD"); |
| scratch.file( |
| "a/foo.bzl", // |
| "visibility(123)"); |
| |
| reporter.removeHandler(failFastHandler); |
| checkFailingLookup("//a:foo.bzl", "initialization of module 'a/foo.bzl' failed"); |
| assertContainsEvent("Invalid visibility: got 'int', want string or list of strings"); |
| } |
| |
| @Test |
| public void testBzlVisibility_invalid_badElementType() throws Exception { |
| setBuildLanguageOptions("--experimental_bzl_visibility=true"); |
| |
| scratch.file("a/BUILD"); |
| scratch.file( |
| "a/foo.bzl", // |
| "visibility([\"//a\", 123])"); |
| |
| reporter.removeHandler(failFastHandler); |
| checkFailingLookup("//a:foo.bzl", "initialization of module 'a/foo.bzl' failed"); |
| assertContainsEvent("at index 1 of visibility list, got element of type int, want string"); |
| } |
| |
| @Test |
| public void testBzlVisibility_invalid_packageOutsideRepo() throws Exception { |
| setBuildLanguageOptions("--experimental_bzl_visibility=true"); |
| |
| scratch.file("a/BUILD"); |
| scratch.file( |
| "a/foo.bzl", // |
| "visibility([\"@repo//b\"])"); |
| |
| reporter.removeHandler(failFastHandler); |
| checkFailingLookup("//a:foo.bzl", "initialization of module 'a/foo.bzl' failed"); |
| assertContainsEvent("invalid package name '@repo//b': must start with '//'"); |
| } |
| |
| @Test |
| public void testBzlVisibility_invalid_negationNotSupported() throws Exception { |
| setBuildLanguageOptions("--experimental_bzl_visibility=true"); |
| |
| scratch.file("a/BUILD"); |
| scratch.file( |
| "a/foo.bzl", // |
| "visibility([\"-//a\"])"); |
| |
| reporter.removeHandler(failFastHandler); |
| checkFailingLookup("//a:foo.bzl", "initialization of module 'a/foo.bzl' failed"); |
| assertContainsEvent("Cannot use negative package patterns here"); |
| } |
| |
| @Test |
| public void testBzlVisibility_errorsDemotedToWarningWhenBreakGlassFlagIsSet() throws Exception { |
| setBuildLanguageOptions("--experimental_bzl_visibility=true", "--check_bzl_visibility=false"); |
| |
| scratch.file("a/BUILD"); |
| scratch.file( |
| "a/foo.bzl", // |
| "load(\"//b:bar.bzl\", \"x\")"); |
| scratch.file("b/BUILD"); |
| scratch.file( |
| "b/bar.bzl", // |
| "visibility(\"private\")", |
| "x = 1"); |
| |
| checkSuccessfulLookup("//a:foo.bzl"); |
| assertContainsEvent("Starlark file //b:bar.bzl is not visible for loading from package //a."); |
| assertContainsEvent("Continuing because --nocheck_bzl_visibility is active"); |
| } |
| |
| @Test |
| public void testLoadFromNonExistentRepository_producesMeaningfulError() throws Exception { |
| scratch.file("BUILD", "load(\"@repository//dir:file.bzl\", \"foo\")"); |
| |
| SkyKey skyKey = key("@repository//dir:file.bzl"); |
| EvaluationResult<BzlLoadValue> result = |
| SkyframeExecutorTestUtils.evaluate( |
| getSkyframeExecutor(), skyKey, /*keepGoing=*/ false, reporter); |
| assertThat(result.hasError()).isTrue(); |
| assertThatEvaluationResult(result) |
| .hasErrorEntryForKeyThat(skyKey) |
| .hasExceptionThat() |
| .isInstanceOf(BzlLoadFailedException.class); |
| assertThatEvaluationResult(result) |
| .hasErrorEntryForKeyThat(skyKey) |
| .hasExceptionThat() |
| .hasMessageThat() |
| .contains( |
| "Unable to find package for @repository//dir:file.bzl: The repository '@repository' " |
| + "could not be resolved: Repository '@repository' is not defined."); |
| } |
| |
| @Test |
| public void testLoadBzlFileFromWorkspaceWithRemapping() throws Exception { |
| Path p = |
| scratch.overwriteFile( |
| "WORKSPACE", |
| "local_repository(", |
| " name = 'y',", |
| " path = '/y'", |
| ")", |
| "local_repository(", |
| " name = 'a',", |
| " path = '/a',", |
| " repo_mapping = {'@x' : '@y'}", |
| ")", |
| "load('@a//:a.bzl', 'a_symbol')"); |
| |
| scratch.file("/y/WORKSPACE"); |
| scratch.file("/y/BUILD"); |
| scratch.file("/y/y.bzl", "y_symbol = 5"); |
| |
| scratch.file("/a/WORKSPACE"); |
| scratch.file("/a/BUILD"); |
| scratch.file("/a/a.bzl", "load('@x//:y.bzl', 'y_symbol')", "a_symbol = y_symbol"); |
| |
| Root root = Root.fromPath(p.getParentDirectory()); |
| RootedPath rootedPath = RootedPath.toRootedPath(root, PathFragment.create("WORKSPACE")); |
| |
| SkyKey skyKey = |
| BzlLoadValue.keyForWorkspace(Label.parseCanonicalUnchecked("@a//:a.bzl"), 1, rootedPath); |
| |
| EvaluationResult<BzlLoadValue> result = |
| SkyframeExecutorTestUtils.evaluate( |
| getSkyframeExecutor(), skyKey, /*keepGoing=*/ false, reporter); |
| |
| assertThat(result.get(skyKey).getModule().getGlobals()) |
| .containsEntry("a_symbol", StarlarkInt.of(5)); |
| } |
| |
| @Test |
| public void testLoadBzlFileFromBzlmod() throws Exception { |
| setBuildLanguageOptions("--enable_bzlmod", "--experimental_enable_scl_dialect"); |
| scratch.overwriteFile("MODULE.bazel", "bazel_dep(name='foo',version='1.0')"); |
| registry |
| .addModule( |
| createModuleKey("foo", "1.0"), |
| "module(name='foo',version='1.0')", |
| "bazel_dep(name='bar',version='2.0',repo_name='bar_alias')") |
| .addModule(createModuleKey("bar", "2.0"), "module(name='bar',version='2.0')"); |
| Path fooDir = moduleRoot.getRelative("foo~1.0"); |
| scratch.file(fooDir.getRelative("WORKSPACE").getPathString()); |
| scratch.file(fooDir.getRelative("BUILD").getPathString()); |
| scratch.file( |
| fooDir.getRelative("test.bzl").getPathString(), |
| // Also test that bzlmod .bzl files can load .scl files. |
| "load('@bar_alias//:test.scl', 'haha')", |
| "hoho = haha"); |
| Path barDir = moduleRoot.getRelative("bar~2.0"); |
| scratch.file(barDir.getRelative("WORKSPACE").getPathString()); |
| scratch.file(barDir.getRelative("BUILD").getPathString()); |
| scratch.file(barDir.getRelative("test.scl").getPathString(), "haha = 5"); |
| |
| SkyKey skyKey = BzlLoadValue.keyForBzlmod(Label.parseCanonical("@@foo~1.0//:test.bzl")); |
| EvaluationResult<BzlLoadValue> result = |
| SkyframeExecutorTestUtils.evaluate( |
| getSkyframeExecutor(), skyKey, /*keepGoing=*/ false, reporter); |
| |
| assertThatEvaluationResult(result).hasNoError(); |
| assertThat(result.get(skyKey).getModule().getGlobals()) |
| .containsEntry("hoho", StarlarkInt.of(5)); |
| // Note that we're not testing the case of a non-registry override using @bazel_tools here, but |
| // that is incredibly hard to set up in a unit test. So we should just rely on integration tests |
| // for that. |
| } |
| |
| @Test |
| public void testBuiltinsInjectionFailure() throws Exception { |
| setBuildLanguageOptions("--experimental_builtins_bzl_path=tools/builtins_staging"); |
| scratch.file( |
| "tools/builtins_staging/exports.bzl", |
| "1 // 0 # <-- dynamic error", |
| "exported_toplevels = {}", |
| "exported_rules = {}", |
| "exported_to_java = {}"); |
| scratch.file("pkg/BUILD"); |
| scratch.file("pkg/foo.bzl"); |
| reporter.removeHandler(failFastHandler); |
| |
| SkyKey key = key("//pkg:foo.bzl"); |
| EvaluationResult<BzlLoadValue> result = |
| SkyframeExecutorTestUtils.evaluate( |
| getSkyframeExecutor(), key, /*keepGoing=*/ false, reporter); |
| |
| assertContainsEvent( |
| "File \"/workspace/tools/builtins_staging/exports.bzl\", line 1, column 3, in <toplevel>"); |
| assertContainsEvent("Error: integer division by zero"); |
| Exception ex = result.getError(key).getException(); |
| assertThat(ex) |
| .hasMessageThat() |
| .contains( |
| "Internal error while loading Starlark builtins for //pkg:foo.bzl: Failed to load" |
| + " builtins sources: initialization of module 'exports.bzl' (internal) failed"); |
| } |
| |
| @Test |
| public void testErrorReadingBzlFileIsTransientWhenUsingASTInlining() throws Exception { |
| CustomInMemoryFs fs = (CustomInMemoryFs) fileSystem; |
| scratch.file("a/BUILD"); |
| fs.badPathForRead = scratch.file("a/a1.bzl", "doesntmatter"); |
| |
| SkyKey key = key("//a:a1.bzl"); |
| EvaluationResult<BzlLoadValue> result = |
| SkyframeExecutorTestUtils.evaluate( |
| getSkyframeExecutor(), key, /*keepGoing=*/ false, reporter); |
| assertThatEvaluationResult(result).hasErrorEntryForKeyThat(key).isTransient(); |
| } |
| |
| @Test |
| public void testErrorReadingOtherBzlFileIsPersistentFromPerspectiveOfParent() throws Exception { |
| CustomInMemoryFs fs = (CustomInMemoryFs) fileSystem; |
| scratch.file("a/BUILD"); |
| scratch.file("a/a1.bzl", "load('//a:a2.bzl', 'a2')"); |
| fs.badPathForRead = scratch.file("a/a2.bzl", "doesntmatter"); |
| |
| SkyKey key = key("//a:a1.bzl"); |
| EvaluationResult<BzlLoadValue> result = |
| SkyframeExecutorTestUtils.evaluate( |
| getSkyframeExecutor(), key, /*keepGoing=*/ false, reporter); |
| assertThatEvaluationResult(result).hasErrorEntryForKeyThat(key).isNotTransient(); |
| } |
| |
| @Test |
| public void testErrorStatingBzlFileInFileStateFunctionIsPersistent() throws Exception { |
| CustomInMemoryFs fs = (CustomInMemoryFs) fileSystem; |
| scratch.file("a/BUILD"); |
| fs.badPathForStat = scratch.file("a/a1.bzl", "doesntmatter"); |
| |
| SkyKey key = key("//a:a1.bzl"); |
| EvaluationResult<BzlLoadValue> result = |
| SkyframeExecutorTestUtils.evaluate( |
| getSkyframeExecutor(), key, /*keepGoing=*/ false, reporter); |
| assertThatEvaluationResult(result).hasErrorEntryForKeyThat(key).isNotTransient(); |
| } |
| |
| private static class CustomInMemoryFs extends InMemoryFileSystem { |
| @Nullable private Path badPathForStat; |
| @Nullable private Path badPathForRead; |
| |
| CustomInMemoryFs() { |
| super(DigestHashFunction.SHA256); |
| } |
| |
| @Override |
| public FileStatus statIfFound(PathFragment path, boolean followSymlinks) throws IOException { |
| if (badPathForStat != null && badPathForStat.asFragment().equals(path)) { |
| throw new IOException("bad"); |
| } |
| return super.statIfFound(path, followSymlinks); |
| } |
| |
| @Override |
| protected synchronized InputStream getInputStream(PathFragment path) throws IOException { |
| if (badPathForRead != null && badPathForRead.asFragment().equals(path)) { |
| throw new IOException("bad"); |
| } |
| return super.getInputStream(path); |
| } |
| } |
| } |