From a1fff4bcfae282a884a0f36a0bc93902867e876f Mon Sep 17 00:00:00 2001 From: salma-samy Date: Tue, 18 Jul 2023 04:42:43 -0700 Subject: [PATCH 1/4] Add 'environ' attribute to module extensions and update lockfile - Add 'environ' to module extensions - Store the variables and their values in the lockfile - Compare the variables and values with the lockfile and re-evaluate or error if they differ PiperOrigin-RevId: 548964193 Change-Id: Ic2d52fe3332e93095c414d8bca4c9b4312bca8c2 # Conflicts: # src/main/java/com/google/devtools/build/lib/bazel/bzlmod/ModuleExtension.java # src/main/java/com/google/devtools/build/lib/bazel/bzlmod/SingleExtensionEvalFunction.java # src/main/java/com/google/devtools/build/lib/bazel/repository/starlark/StarlarkRepositoryModule.java # src/main/java/com/google/devtools/build/lib/skyframe/ActionEnvironmentFunction.java # src/main/java/com/google/devtools/build/lib/skyframe/ClientEnvironmentFunction.java # src/main/java/com/google/devtools/build/lib/starlarkbuildapi/repository/RepositoryModuleApi.java # src/main/java/com/google/devtools/build/skydoc/fakebuildapi/repository/FakeRepositoryModule.java # src/test/java/com/google/devtools/build/lib/bazel/bzlmod/StarlarkBazelModuleTest.java # src/test/py/bazel/bzlmod/bazel_lockfile_test.py --- .../devtools/build/lib/bazel/bzlmod/BUILD | 1 + .../lib/bazel/bzlmod/BazelLockFileValue.java | 22 +- .../bazel/bzlmod/LockFileModuleExtension.java | 9 +- .../lib/bazel/bzlmod/ModuleExtension.java | 13 +- .../bzlmod/SingleExtensionEvalFunction.java | 26 ++- .../build/lib/bazel/bzlmod/TagClass.java | 6 +- .../starlark/StarlarkRepositoryModule.java | 14 +- .../rules/repository/RepositoryFunction.java | 52 +++-- .../skyframe/ActionEnvironmentFunction.java | 19 +- .../skyframe/ClientEnvironmentFunction.java | 3 +- .../repository/FakeRepositoryModule.java | 3 +- .../bazel/bzlmod/StarlarkBazelModuleTest.java | 2 + .../py/bazel/bzlmod/bazel_lockfile_test.py | 207 ++++++++++++------ 13 files changed, 259 insertions(+), 118 deletions(-) diff --git a/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/BUILD b/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/BUILD index 1d5e560b921c7d..dd307527d1ba3c 100644 --- a/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/BUILD +++ b/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/BUILD @@ -55,6 +55,7 @@ java_library( ], deps = [ ":common", + "//src/main/java/com/google/devtools/build/docgen/annot", "//src/main/java/com/google/devtools/build/lib/cmdline", "//src/main/java/com/google/devtools/build/lib/events", "//src/main/java/com/google/devtools/build/lib/packages", diff --git a/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/BazelLockFileValue.java b/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/BazelLockFileValue.java index c0d0c06aed41e4..415c31cb62af2f 100644 --- a/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/BazelLockFileValue.java +++ b/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/BazelLockFileValue.java @@ -114,20 +114,28 @@ public ImmutableList getModuleExtensionDiff( ImmutableMap lockedExtensionUsages, ModuleExtensionId extensionId, byte[] transitiveDigest, + ImmutableMap envVariables, ImmutableMap extensionUsages) { ImmutableList.Builder extDiff = new ImmutableList.Builder<>(); if (lockedExtension == null) { - extDiff.add("The module extension '" + extensionId + "' does not exist in the lockfile"); - } else { - if (!Arrays.equals(transitiveDigest, lockedExtension.getBzlTransitiveDigest())) { + return extDiff + .add("The module extension '" + extensionId + "' does not exist in the lockfile") + .build(); + } + if (!Arrays.equals(transitiveDigest, lockedExtension.getBzlTransitiveDigest())) { extDiff.add( "The implementation of the extension '" + extensionId + "' or one of its transitive .bzl files has changed"); - } - if (!extensionUsages.equals(lockedExtensionUsages)) { - extDiff.add("The usages of the extension named '" + extensionId + "' has changed"); - } + } + if (!extensionUsages.equals(lockedExtensionUsages)) { + extDiff.add("The usages of the extension '" + extensionId + "' has changed"); + } + if (!envVariables.equals(lockedExtension.getEnvVariables())) { + extDiff.add( + "The environment variables the extension '" + + extensionId + + "' depends on (or their values) have changed"); } return extDiff.build(); } diff --git a/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/LockFileModuleExtension.java b/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/LockFileModuleExtension.java index 5431d6cf0fabe6..84b9581d74ebf2 100644 --- a/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/LockFileModuleExtension.java +++ b/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/LockFileModuleExtension.java @@ -31,10 +31,15 @@ public abstract class LockFileModuleExtension implements Postable { @SuppressWarnings("mutable") public abstract byte[] getBzlTransitiveDigest(); + public abstract ImmutableMap getEnvVariables(); + public abstract ImmutableMap getGeneratedRepoSpecs(); public static LockFileModuleExtension create( - byte[] transitiveDigest, ImmutableMap generatedRepoSpecs) { - return new AutoValue_LockFileModuleExtension(transitiveDigest, generatedRepoSpecs); + byte[] transitiveDigest, + ImmutableMap envVariables, + ImmutableMap generatedRepoSpecs) { + return new AutoValue_LockFileModuleExtension( + transitiveDigest, envVariables, generatedRepoSpecs); } } diff --git a/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/ModuleExtension.java b/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/ModuleExtension.java index a55de9583b2545..d796e4e6db8ccb 100644 --- a/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/ModuleExtension.java +++ b/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/ModuleExtension.java @@ -16,21 +16,28 @@ import com.google.auto.value.AutoValue; import com.google.common.base.Preconditions; +import com.google.common.collect.ImmutableList; import com.google.common.collect.ImmutableMap; +import com.google.devtools.build.docgen.annot.DocCategory; import com.google.devtools.build.lib.cmdline.Label; import com.google.devtools.build.lib.events.EventHandler; import com.google.devtools.build.lib.packages.StarlarkExportable; import javax.annotation.Nullable; import net.starlark.java.annot.StarlarkBuiltin; import net.starlark.java.eval.StarlarkCallable; +import net.starlark.java.eval.StarlarkValue; import net.starlark.java.syntax.Location; /** * A module extension object, which can be used to perform arbitrary logic in order to create repos * or register toolchains and execution platforms. */ +@StarlarkBuiltin( + name = "module_extension", + category = DocCategory.BUILTIN, + doc = "A module extension declared using the module_extension function.") @AutoValue -public abstract class ModuleExtension { +public abstract class ModuleExtension implements StarlarkValue { public abstract String getName(); public abstract StarlarkCallable getImplementation(); @@ -43,6 +50,8 @@ public abstract class ModuleExtension { public abstract Location getLocation(); + public abstract ImmutableList getEnvVariables(); + public static Builder builder() { return new AutoValue_ModuleExtension.Builder(); } @@ -63,6 +72,8 @@ public abstract static class Builder { public abstract Builder setTagClasses(ImmutableMap value); + public abstract Builder setEnvVariables(ImmutableList value); + public abstract ModuleExtension build(); } diff --git a/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/SingleExtensionEvalFunction.java b/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/SingleExtensionEvalFunction.java index 8bb26e5a45f26a..b371ab4838115f 100644 --- a/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/SingleExtensionEvalFunction.java +++ b/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/SingleExtensionEvalFunction.java @@ -32,6 +32,7 @@ import com.google.devtools.build.lib.cmdline.RepositoryName; import com.google.devtools.build.lib.events.Event; import com.google.devtools.build.lib.rules.repository.NeedsSkyframeRestartException; +import com.google.devtools.build.lib.rules.repository.RepositoryFunction; import com.google.devtools.build.lib.runtime.ProcessWrapper; import com.google.devtools.build.lib.runtime.RepositoryRemoteExecutor; import com.google.devtools.build.lib.server.FailureDetails.ExternalDeps; @@ -144,9 +145,16 @@ public SkyValue compute(SkyKey skyKey, Environment env) Transience.PERSISTENT); } - // Check the lockfile first for that module extension + ModuleExtension extension = ((ModuleExtension.InStarlark) exported).get(); + ImmutableMap extensionEnvVars = + RepositoryFunction.getEnvVarValues(env, extension.getEnvVariables()); + if (extensionEnvVars == null) { + return null; + } byte[] bzlTransitiveDigest = BazelModuleContext.of(bzlLoadValue.getModule()).bzlTransitiveDigest(); + + // Check the lockfile first for that module extension LockfileMode lockfileMode = BazelLockFileFunction.LOCKFILE_MODE.get(env); if (!lockfileMode.equals(LockfileMode.OFF)) { BazelLockFileValue lockfile = (BazelLockFileValue) env.getValue(BazelLockFileValue.KEY); @@ -155,14 +163,18 @@ public SkyValue compute(SkyKey skyKey, Environment env) } SingleExtensionEvalValue singleExtensionEvalValue = tryGettingValueFromLockFile( - extensionId, usagesValue, bzlTransitiveDigest, lockfileMode, lockfile); + extensionId, + extensionEnvVars, + usagesValue, + bzlTransitiveDigest, + lockfileMode, + lockfile); if (singleExtensionEvalValue != null) { return singleExtensionEvalValue; } } // Run that extension! - ModuleExtension extension = ((ModuleExtension.InStarlark) exported).get(); ImmutableMap generatedRepoSpecs = runModuleExtension( extensionId, extension, usagesValue, bzlLoadValue.getModule(), starlarkSemantics, env); @@ -177,7 +189,8 @@ public SkyValue compute(SkyKey skyKey, Environment env) .post( ModuleExtensionResolutionEvent.create( extensionId, - LockFileModuleExtension.create(bzlTransitiveDigest, generatedRepoSpecs))); + LockFileModuleExtension.create( + bzlTransitiveDigest, extensionEnvVars, generatedRepoSpecs))); } return createSingleExtentionValue(generatedRepoSpecs, usagesValue); } @@ -185,6 +198,7 @@ public SkyValue compute(SkyKey skyKey, Environment env) @Nullable private SingleExtensionEvalValue tryGettingValueFromLockFile( ModuleExtensionId extensionId, + ImmutableMap envVariables, SingleExtensionUsagesValue usagesValue, byte[] bzlTransitiveDigest, LockfileMode lockfileMode, @@ -205,7 +219,8 @@ private SingleExtensionEvalValue tryGettingValueFromLockFile( // If we have the extension, check if the implementation and usage haven't changed if (lockedExtension != null && Arrays.equals(bzlTransitiveDigest, lockedExtension.getBzlTransitiveDigest()) - && usagesValue.getExtensionUsages().equals(lockedExtensionUsages)) { + && usagesValue.getExtensionUsages().equals(lockedExtensionUsages) + && envVariables.equals(lockedExtension.getEnvVariables())) { return createSingleExtentionValue(lockedExtension.getGeneratedRepoSpecs(), usagesValue); } else if (lockfileMode.equals(LockfileMode.ERROR)) { ImmutableList extDiff = @@ -214,6 +229,7 @@ private SingleExtensionEvalValue tryGettingValueFromLockFile( lockedExtensionUsages, extensionId, bzlTransitiveDigest, + envVariables, usagesValue.getExtensionUsages()); throw new SingleExtensionEvalFunctionException( ExternalDepsException.withMessage( diff --git a/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/TagClass.java b/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/TagClass.java index f2bdc7c1e93e8d..7c6f8030103d61 100644 --- a/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/TagClass.java +++ b/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/TagClass.java @@ -17,6 +17,7 @@ import com.google.auto.value.AutoValue; import com.google.common.collect.ImmutableList; import com.google.common.collect.ImmutableMap; +import com.google.devtools.build.docgen.annot.DocCategory; import com.google.devtools.build.lib.packages.Attribute; import net.starlark.java.annot.StarlarkBuiltin; import net.starlark.java.eval.StarlarkValue; @@ -25,7 +26,10 @@ /** * Represents a tag class, which is a "class" of {@link Tag}s that share the same attribute schema. */ -@StarlarkBuiltin(name = "tag_class", doc = "Defines a schema of attributes for a tag.") +@StarlarkBuiltin( + name = "tag_class", + category = DocCategory.BUILTIN, + doc = "Defines a schema of attributes for a tag.") @AutoValue public abstract class TagClass implements StarlarkValue { /** The list of attributes of this tag class. */ diff --git a/src/main/java/com/google/devtools/build/lib/bazel/repository/starlark/StarlarkRepositoryModule.java b/src/main/java/com/google/devtools/build/lib/bazel/repository/starlark/StarlarkRepositoryModule.java index 3dd3e9d8f6d67e..cc67eca5b0437e 100644 --- a/src/main/java/com/google/devtools/build/lib/bazel/repository/starlark/StarlarkRepositoryModule.java +++ b/src/main/java/com/google/devtools/build/lib/bazel/repository/starlark/StarlarkRepositoryModule.java @@ -296,6 +296,15 @@ public void failWithIncompatibleUseCcConfigureFromRulesCc(StarlarkThread thread) "A description of the module extension that can be extracted by documentation" + " generating tools.", named = true, + positional = false), + @Param( + name = "environ", + defaultValue = "[]", + doc = + "Provides a list of environment variable that this module extension depends on. If " + + "an environment variable in that list changes, the extension will be " + + "re-evaluated.", + named = true, positional = false) }, useStarlarkThread = true) @@ -303,10 +312,11 @@ public Object moduleExtension( StarlarkCallable implementation, Dict tagClasses, // Dict String doc, + Sequence environ, // StarlarkThread thread) throws EvalException { ModuleExtension.InStarlark inStarlark = new ModuleExtension.InStarlark(); - inStarlark + return inStarlark .getBuilder() .setImplementation(implementation) .setTagClasses( @@ -314,8 +324,8 @@ public Object moduleExtension( .setDoc(doc) .setDefinitionEnvironmentLabel( BazelModuleContext.of(Module.ofInnermostEnclosingStarlarkFunction(thread)).label()) + .setEnvVariables(ImmutableList.copyOf(Sequence.cast(environ, String.class, "environ"))) .setLocation(thread.getCallerLocation()); - return inStarlark; } @StarlarkMethod( diff --git a/src/main/java/com/google/devtools/build/lib/rules/repository/RepositoryFunction.java b/src/main/java/com/google/devtools/build/lib/rules/repository/RepositoryFunction.java index 55084b91d90d3a..26d04ae28c2288 100644 --- a/src/main/java/com/google/devtools/build/lib/rules/repository/RepositoryFunction.java +++ b/src/main/java/com/google/devtools/build/lib/rules/repository/RepositoryFunction.java @@ -20,6 +20,7 @@ import com.google.common.annotations.VisibleForTesting; import com.google.common.base.Preconditions; import com.google.common.collect.ImmutableList; +import com.google.common.collect.ImmutableMap; import com.google.common.io.BaseEncoding; import com.google.devtools.build.lib.actions.FileValue; import com.google.devtools.build.lib.analysis.BlazeDirectories; @@ -50,6 +51,7 @@ import com.google.devtools.build.skyframe.SkyFunctionException.Transience; import com.google.devtools.build.skyframe.SkyKey; import java.io.IOException; +import java.util.Collection; import java.util.LinkedHashMap; import java.util.Map; import java.util.Objects; @@ -170,9 +172,9 @@ public abstract RepositoryDirectoryValue.Builder fetch( throws InterruptedException, RepositoryFunctionException; @SuppressWarnings("unchecked") - private static Iterable getEnviron(Rule rule) { + private static Collection getEnviron(Rule rule) { if (rule.isAttrDefined("$environ", Type.STRING_LIST)) { - return (Iterable) rule.getAttr("$environ"); + return (Collection) rule.getAttr("$environ"); } return ImmutableList.of(); } @@ -183,7 +185,7 @@ private static Iterable getEnviron(Rule rule) { * is needed. */ public boolean verifyMarkerData(Rule rule, Map markerData, Environment env) - throws InterruptedException, RepositoryFunctionException { + throws InterruptedException { return verifyEnvironMarkerData(markerData, env, getEnviron(rule)) && verifyMarkerDataForFiles(rule, markerData, env) && verifySemanticsMarkerData(markerData, env); @@ -288,33 +290,37 @@ public static RootedPath getRootedPathFromLabel(Label label, Environment env) protected Map declareEnvironmentDependencies( Map markerData, Environment env, Iterable keys) throws InterruptedException { - Map environ = ActionEnvironmentFunction.getEnvironmentView(env, keys); + ImmutableMap envDep = getEnvVarValues(env, keys); + if (envDep == null) { + return null; + } + // Add the dependencies to the marker file + keys.forEach(key -> markerData.put("ENV:" + key, envDep.get(key))); + return envDep; + } - // Returns true if there is a null value and we need to wait for some dependencies. + @Nullable + public static ImmutableMap getEnvVarValues(Environment env, Iterable keys) + throws InterruptedException { + ImmutableMap environ = ActionEnvironmentFunction.getEnvironmentView(env, keys); if (environ == null) { return null; } - Map repoEnvOverride = PrecomputedValue.REPO_ENV.get(env); if (repoEnvOverride == null) { return null; } // Only depend on --repo_env values that are specified in the "environ" attribute. - Map repoEnv = new LinkedHashMap(environ); + ImmutableMap.Builder repoEnv = ImmutableMap.builder(); + repoEnv.putAll(environ); for (String key : keys) { String value = repoEnvOverride.get(key); if (value != null) { repoEnv.put(key, value); } } - - // Add the dependencies to the marker file - for (Map.Entry value : repoEnv.entrySet()) { - markerData.put("ENV:" + value.getKey(), value.getValue()); - } - - return repoEnv; + return repoEnv.buildKeepingLast(); } /** @@ -323,9 +329,9 @@ protected Map declareEnvironmentDependencies( * Environment)} function to verify the values for environment variables. */ protected boolean verifyEnvironMarkerData( - Map markerData, Environment env, Iterable keys) + Map markerData, Environment env, Collection keys) throws InterruptedException { - Map environ = ActionEnvironmentFunction.getEnvironmentView(env, keys); + ImmutableMap environ = ActionEnvironmentFunction.getEnvironmentView(env, keys); if (env.valuesMissing()) { return false; // Returns false so caller knows to return immediately } @@ -346,17 +352,18 @@ protected boolean verifyEnvironMarkerData( // Verify that all environment variable in the marker file are also in keys for (String key : markerData.keySet()) { - if (key.startsWith("ENV:") && !repoEnv.containsKey(key.substring(4))) { + if (key.startsWith("ENV:") && !keys.contains(key.substring(4))) { return false; } } + // Now verify the values of the marker data - for (Map.Entry value : repoEnv.entrySet()) { - if (!markerData.containsKey("ENV:" + value.getKey())) { + for (String key : keys) { + if (!markerData.containsKey("ENV:" + key)) { return false; } - String markerValue = markerData.get("ENV:" + value.getKey()); - if (!Objects.equals(markerValue, value.getValue())) { + String markerValue = markerData.get("ENV:" + key); + if (!Objects.equals(markerValue, repoEnv.get(key))) { return false; } } @@ -439,8 +446,7 @@ protected static String getPathAttr(Rule rule) throws RepositoryFunctionExceptio } @VisibleForTesting - protected static PathFragment getTargetPath(String userDefinedPath, Path workspace) - throws RepositoryFunctionException { + protected static PathFragment getTargetPath(String userDefinedPath, Path workspace) { PathFragment pathFragment = PathFragment.create(userDefinedPath); return workspace.getRelative(pathFragment).asFragment(); } diff --git a/src/main/java/com/google/devtools/build/lib/skyframe/ActionEnvironmentFunction.java b/src/main/java/com/google/devtools/build/lib/skyframe/ActionEnvironmentFunction.java index 25d12db34afbcd..4edf0fb1b31078 100644 --- a/src/main/java/com/google/devtools/build/lib/skyframe/ActionEnvironmentFunction.java +++ b/src/main/java/com/google/devtools/build/lib/skyframe/ActionEnvironmentFunction.java @@ -15,6 +15,7 @@ package com.google.devtools.build.lib.skyframe; import com.google.common.collect.ImmutableList; +import com.google.common.collect.ImmutableMap; import com.google.common.collect.Interner; import com.google.devtools.build.lib.bugreport.BugReport; import com.google.devtools.build.lib.concurrent.BlazeInterners; @@ -25,8 +26,6 @@ import com.google.devtools.build.skyframe.SkyKey; import com.google.devtools.build.skyframe.SkyValue; import com.google.devtools.build.skyframe.SkyframeLookupResult; -import java.util.Collections; -import java.util.LinkedHashMap; import java.util.Map; import javax.annotation.Nullable; @@ -47,7 +46,7 @@ public SkyValue compute(SkyKey skyKey, Environment env) throws InterruptedExcept return env.getValue(ClientEnvironmentFunction.key(key)); } - /** @return the SkyKey to invoke this function for the environment variable {@code variable}. */ + /** Returns the SkyKey to invoke this function for the environment variable {@code variable}. */ public static Key key(String variable) { return Key.create(variable); } @@ -78,8 +77,8 @@ public SkyFunctionName functionName() { * if and only if some dependencies from Skyframe still need to be resolved. */ @Nullable - public static Map getEnvironmentView(Environment env, Iterable keys) - throws InterruptedException { + public static ImmutableMap getEnvironmentView( + Environment env, Iterable keys) throws InterruptedException { ImmutableList.Builder skyframeKeysBuilder = ImmutableList.builder(); for (String key : keys) { skyframeKeysBuilder.add(key(key)); @@ -89,8 +88,8 @@ public static Map getEnvironmentView(Environment env, Iterable result = new LinkedHashMap<>(); + + ImmutableMap.Builder result = ImmutableMap.builder(); for (SkyKey key : skyframeKeys) { ClientEnvironmentValue value = (ClientEnvironmentValue) values.get(key); if (value == null) { @@ -99,8 +98,10 @@ public static Map getEnvironmentView(Environment env, Iterable { @@ -61,7 +62,7 @@ public ClientEnvironmentFunction(AtomicReference> clientEnv) @Nullable @Override - public SkyValue compute(SkyKey key, Environment env) throws InterruptedException { + public SkyValue compute(SkyKey key, Environment env) { return new ClientEnvironmentValue(clientEnv.get().get((String) key.argument())); } } diff --git a/src/main/java/com/google/devtools/build/skydoc/fakebuildapi/repository/FakeRepositoryModule.java b/src/main/java/com/google/devtools/build/skydoc/fakebuildapi/repository/FakeRepositoryModule.java index 8f9edc31cb8be9..2cbf475276304a 100644 --- a/src/main/java/com/google/devtools/build/skydoc/fakebuildapi/repository/FakeRepositoryModule.java +++ b/src/main/java/com/google/devtools/build/skydoc/fakebuildapi/repository/FakeRepositoryModule.java @@ -72,7 +72,6 @@ public StarlarkCallable repositoryRule( String doc, StarlarkThread thread) throws EvalException { - List attrInfos; ImmutableMap.Builder attrsMapBuilder = ImmutableMap.builder(); if (attrs != null && attrs != Starlark.NONE) { attrsMapBuilder.putAll(Dict.cast(attrs, String.class, FakeDescriptor.class, "attrs")); @@ -80,7 +79,7 @@ public StarlarkCallable repositoryRule( attrsMapBuilder.put("name", IMPLICIT_NAME_ATTRIBUTE_DESCRIPTOR); attrsMapBuilder.put("repo_mapping", IMPLICIT_REPO_MAPPING_ATTRIBUTE_DESCRIPTOR); - attrInfos = + List attrInfos = attrsMapBuilder.build().entrySet().stream() .filter(entry -> !entry.getKey().startsWith("_")) .map(entry -> entry.getValue().asAttributeInfo(entry.getKey())) diff --git a/src/test/java/com/google/devtools/build/lib/bazel/bzlmod/StarlarkBazelModuleTest.java b/src/test/java/com/google/devtools/build/lib/bazel/bzlmod/StarlarkBazelModuleTest.java index f20e7c092d8c62..6e0dd26752ed64 100644 --- a/src/test/java/com/google/devtools/build/lib/bazel/bzlmod/StarlarkBazelModuleTest.java +++ b/src/test/java/com/google/devtools/build/lib/bazel/bzlmod/StarlarkBazelModuleTest.java @@ -24,6 +24,7 @@ import static org.junit.Assert.assertThrows; import com.google.common.collect.ImmutableBiMap; +import com.google.common.collect.ImmutableList; import com.google.common.collect.ImmutableMap; import com.google.common.collect.ImmutableSet; import com.google.devtools.build.lib.cmdline.Label; @@ -61,6 +62,7 @@ private static ModuleExtension.Builder getBaseExtensionBuilder() { .setName("maven") .setDoc("") .setLocation(Location.BUILTIN) + .setEnvVariables(ImmutableList.of()) .setDefinitionEnvironmentLabel(Label.parseAbsoluteUnchecked("//:rje.bzl")) .setImplementation(() -> "maven"); } diff --git a/src/test/py/bazel/bzlmod/bazel_lockfile_test.py b/src/test/py/bazel/bzlmod/bazel_lockfile_test.py index f145c633d301b5..6ecac08c607210 100644 --- a/src/test/py/bazel/bzlmod/bazel_lockfile_test.py +++ b/src/test/py/bazel/bzlmod/bazel_lockfile_test.py @@ -282,23 +282,27 @@ def testModuleExtension(self): ' for dep in mod.tags.dep:', ' print("Name:", dep.name, ", Versions:", dep.versions)', '', - ( - '_dep = tag_class(attrs={"name": attr.string(), "versions":' - ' attr.string_list()})' - ), + '_dep = tag_class(attrs={"name": attr.string(), "versions":', + ' attr.string_list()})', 'lockfile_ext = module_extension(', ' implementation=_module_ext_impl,', ' tag_classes={"dep": _dep},', + ' environ=["GREEN_TREES", "NOT_SET"],', ')', ], ) - _, _, stderr = self.RunBazel(['build', '@hello//:all']) + # Only set one env var, to make sure null variables don't crash + _, _, stderr = self.RunBazel( + ['build', '@hello//:all'], env_add={'GREEN_TREES': 'In the city'} + ) self.assertIn('Hello from the other side!', ''.join(stderr)) self.assertIn('Name: bmbm , Versions: ["v1", "v2"]', ''.join(stderr)) self.RunBazel(['shutdown']) - _, _, stderr = self.RunBazel(['build', '@hello//:all']) + _, _, stderr = self.RunBazel( + ['build', '@hello//:all'], env_add={'GREEN_TREES': 'In the city'} + ) self.assertNotIn('Hello from the other side!', ''.join(stderr)) def testIsolatedModuleExtension(self): @@ -403,13 +407,10 @@ def testUpdateModuleExtension(self): ' ctx.file("WORKSPACE")', ' ctx.file("BUILD", "filegroup(name=\\"lala\\")")', 'repo_rule = repository_rule(implementation = _repo_rule_impl)', - 'def _module_ext_impl(ctx):', + 'def _mod_ext_impl(ctx):', ' print("Hello from the other side!")', ' repo_rule(name= "hello")', - ( - 'lockfile_ext = module_extension(implementation =' - ' _module_ext_impl)' - ), + 'lockfile_ext = module_extension(implementation = _mod_ext_impl)', ], ) _, _, stderr = self.RunBazel(['build', '@hello//:all']) @@ -434,13 +435,10 @@ def testUpdateModuleExtension(self): ' ctx.file("WORKSPACE")', ' ctx.file("BUILD", "filegroup(name=\\"lala\\")")', 'repo_rule = repository_rule(implementation = _repo_rule_impl)', - 'def _module_ext_impl(ctx):', + 'def _mod_ext_impl(ctx):', ' print("Hello from the other town!")', ' repo_rule(name= "hello")', - ( - 'lockfile_ext = module_extension(implementation =' - ' _module_ext_impl)' - ), + 'lockfile_ext = module_extension(implementation = _mod_ext_impl)', ], ) _, _, stderr = self.RunBazel(['build', '@hello//:all']) @@ -468,13 +466,10 @@ def testUpdateModuleExtensionErrorMode(self): ' ctx.file("WORKSPACE")', ' ctx.file("BUILD", "filegroup(name=\\"lala\\")")', 'repo_rule = repository_rule(implementation = _repo_rule_impl)', - 'def _module_ext_impl(ctx):', + 'def _mod_ext_impl(ctx):', ' print("Hello from the other side!")', ' repo_rule(name= "hello")', - ( - 'lockfile_ext = module_extension(implementation =' - ' _module_ext_impl)' - ), + 'lockfile_ext = module_extension(implementation = _mod_ext_impl)', ], ) _, _, stderr = self.RunBazel(['build', '@hello//:all']) @@ -489,13 +484,10 @@ def testUpdateModuleExtensionErrorMode(self): ' ctx.file("WORKSPACE")', ' ctx.file("BUILD", "filegroup(name=\\"lalo\\")")', 'repo_rule = repository_rule(implementation = _repo_rule_impl)', - 'def _module_ext_impl(ctx):', + 'def _mod_ext_impl(ctx):', ' print("Hello from the other town!")', ' repo_rule(name= "hello")', - ( - 'lockfile_ext = module_extension(implementation =' - ' _module_ext_impl)' - ), + 'lockfile_ext = module_extension(implementation = _mod_ext_impl)', ], ) @@ -554,42 +546,42 @@ def testRemoveModuleExtensionsNotUsed(self): def testNoAbsoluteRootModuleFilePath(self): self.ScratchFile( - 'MODULE.bazel', - [ - 'ext = use_extension("extension.bzl", "ext")', - 'ext.dep(generate = True)', - 'use_repo(ext, ext_hello = "hello")', - 'other_ext = use_extension("extension.bzl", "other_ext")', - 'other_ext.dep(generate = False)', - 'use_repo(other_ext, other_ext_hello = "hello")', - ], + 'MODULE.bazel', + [ + 'ext = use_extension("extension.bzl", "ext")', + 'ext.dep(generate = True)', + 'use_repo(ext, ext_hello = "hello")', + 'other_ext = use_extension("extension.bzl", "other_ext")', + 'other_ext.dep(generate = False)', + 'use_repo(other_ext, other_ext_hello = "hello")', + ], ) self.ScratchFile('BUILD.bazel') self.ScratchFile( - 'extension.bzl', - [ - 'def _repo_rule_impl(ctx):', - ' ctx.file("WORKSPACE")', - ' ctx.file("BUILD", "filegroup(name=\'lala\')")', - '', - 'repo_rule = repository_rule(implementation=_repo_rule_impl)', - '', - 'def _module_ext_impl(ctx):', - ' for mod in ctx.modules:', - ' for dep in mod.tags.dep:', - ' if dep.generate:', - ' repo_rule(name="hello")', - '', - '_dep = tag_class(attrs={"generate": attr.bool()})', - 'ext = module_extension(', - ' implementation=_module_ext_impl,', - ' tag_classes={"dep": _dep},', - ')', - 'other_ext = module_extension(', - ' implementation=_module_ext_impl,', - ' tag_classes={"dep": _dep},', - ')', - ], + 'extension.bzl', + [ + 'def _repo_rule_impl(ctx):', + ' ctx.file("WORKSPACE")', + ' ctx.file("BUILD", "filegroup(name=\'lala\')")', + '', + 'repo_rule = repository_rule(implementation=_repo_rule_impl)', + '', + 'def _module_ext_impl(ctx):', + ' for mod in ctx.modules:', + ' for dep in mod.tags.dep:', + ' if dep.generate:', + ' repo_rule(name="hello")', + '', + '_dep = tag_class(attrs={"generate": attr.bool()})', + 'ext = module_extension(', + ' implementation=_module_ext_impl,', + ' tag_classes={"dep": _dep},', + ')', + 'other_ext = module_extension(', + ' implementation=_module_ext_impl,', + ' tag_classes={"dep": _dep},', + ')', + ], ) # Paths to module files in error message always use forward slashes as @@ -602,16 +594,101 @@ def testNoAbsoluteRootModuleFilePath(self): self.RunBazel(['shutdown']) exit_code, _, stderr = self.RunBazel( - ['build', '--nobuild', '@other_ext_hello//:all'], allow_failure=True + ['build', '--nobuild', '@other_ext_hello//:all'], allow_failure=True ) self.AssertNotExitCode(exit_code, 0, stderr) + self.assertIn( + ( + 'ERROR: module extension "other_ext" from "//:extension.bzl" does ' + 'not generate repository "hello", yet it is imported as ' + '"other_ext_hello" in the usage at ' + + module_file_path + + ':4:26' + ), + stderr, + ) + + def testModuleExtensionEnvVariable(self): + self.ScratchFile( + 'MODULE.bazel', + [ + 'lockfile_ext = use_extension("extension.bzl", "lockfile_ext")', + 'use_repo(lockfile_ext, "hello")', + ], + ) + self.ScratchFile('BUILD.bazel') + self.ScratchFile( + 'extension.bzl', + [ + 'def _repo_rule_impl(ctx):', + ' ctx.file("WORKSPACE")', + ' ctx.file("BUILD", "filegroup(name=\\"lala\\")")', + 'repo_rule = repository_rule(implementation = _repo_rule_impl)', + 'def _module_ext_impl(ctx):', + ' print("Hello from the other side!")', + ' repo_rule(name= "hello")', + 'lockfile_ext = module_extension(', + ' implementation = _module_ext_impl,', + ' environ = ["SET_ME"]', + ')', + ], + ) + _, _, stderr = self.RunBazel( + ['build', '@hello//:all'], env_add={'SET_ME': 'High in sky'} + ) + self.assertIn('Hello from the other side!', ''.join(stderr)) + # Run with same value, no evaluated + _, _, stderr = self.RunBazel( + ['build', '@hello//:all'], env_add={'SET_ME': 'High in sky'} + ) + self.assertNotIn('Hello from the other side!', ''.join(stderr)) + # Run with different value, will be re-evaluated + _, _, stderr = self.RunBazel( + ['build', '@hello//:all'], env_add={'SET_ME': 'Down to earth'} + ) + self.assertIn('Hello from the other side!', ''.join(stderr)) + + def testChangeEnvVariableInErrorMode(self): + # If environ is set up in module extension, it should be re-evaluated if its + # value changed + self.ScratchFile( + 'MODULE.bazel', + [ + 'lockfile_ext = use_extension("extension.bzl", "lockfile_ext")', + 'use_repo(lockfile_ext, "hello")', + ], + ) + self.ScratchFile('BUILD.bazel') + self.ScratchFile( + 'extension.bzl', + [ + 'def _repo_rule_impl(ctx):', + ' ctx.file("WORKSPACE")', + ' ctx.file("BUILD", "filegroup(name=\\"lala\\")")', + 'repo_rule = repository_rule(implementation = _repo_rule_impl)', + 'def _module_ext_impl(ctx):', + ' print("Hello from the other side!")', + ' repo_rule(name= "hello")', + 'lockfile_ext = module_extension(', + ' implementation = _module_ext_impl,', + ' environ = ["SET_ME"]', + ')', + ], + ) + self.RunBazel(['build', '@hello//:all'], env_add={'SET_ME': 'High in sky'}) + exit_code, _, stderr = self.RunBazel( + ['build', '--lockfile_mode=error', '@hello//:all'], + env_add={'SET_ME': 'Down to earth'}, + allow_failure=True, + ) + self.AssertExitCode(exit_code, 48, stderr) self.assertIn( ( - 'ERROR: module extension "other_ext" from "//:extension.bzl" does ' - 'not generate repository "hello", yet it is imported as ' - '"other_ext_hello" in the usage at ' - + module_file_path - + ':4:26' + 'ERROR: Lock file is no longer up-to-date because: The environment' + ' variables the extension' + " 'ModuleExtensionId{bzlFileLabel=//:extension.bzl," + " extensionName=lockfile_ext, isolationKey=Optional.empty}' depends" + ' on (or their values) have changed' ), stderr, ) From 25856b3f9947b0668d5bb72e28dbf3a11fb68d80 Mon Sep 17 00:00:00 2001 From: salma-samy Date: Thu, 3 Aug 2023 03:07:12 +0200 Subject: [PATCH 2/4] Remove extra changes --- .../java/com/google/devtools/build/lib/bazel/bzlmod/BUILD | 1 - .../devtools/build/lib/bazel/bzlmod/ModuleExtension.java | 5 ----- .../google/devtools/build/lib/bazel/bzlmod/TagClass.java | 6 +----- .../bazel/repository/starlark/StarlarkRepositoryModule.java | 3 ++- 4 files changed, 3 insertions(+), 12 deletions(-) diff --git a/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/BUILD b/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/BUILD index dd307527d1ba3c..1d5e560b921c7d 100644 --- a/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/BUILD +++ b/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/BUILD @@ -55,7 +55,6 @@ java_library( ], deps = [ ":common", - "//src/main/java/com/google/devtools/build/docgen/annot", "//src/main/java/com/google/devtools/build/lib/cmdline", "//src/main/java/com/google/devtools/build/lib/events", "//src/main/java/com/google/devtools/build/lib/packages", diff --git a/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/ModuleExtension.java b/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/ModuleExtension.java index d796e4e6db8ccb..99219efde8f595 100644 --- a/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/ModuleExtension.java +++ b/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/ModuleExtension.java @@ -18,7 +18,6 @@ import com.google.common.base.Preconditions; import com.google.common.collect.ImmutableList; import com.google.common.collect.ImmutableMap; -import com.google.devtools.build.docgen.annot.DocCategory; import com.google.devtools.build.lib.cmdline.Label; import com.google.devtools.build.lib.events.EventHandler; import com.google.devtools.build.lib.packages.StarlarkExportable; @@ -32,10 +31,6 @@ * A module extension object, which can be used to perform arbitrary logic in order to create repos * or register toolchains and execution platforms. */ -@StarlarkBuiltin( - name = "module_extension", - category = DocCategory.BUILTIN, - doc = "A module extension declared using the module_extension function.") @AutoValue public abstract class ModuleExtension implements StarlarkValue { public abstract String getName(); diff --git a/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/TagClass.java b/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/TagClass.java index 7c6f8030103d61..f2bdc7c1e93e8d 100644 --- a/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/TagClass.java +++ b/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/TagClass.java @@ -17,7 +17,6 @@ import com.google.auto.value.AutoValue; import com.google.common.collect.ImmutableList; import com.google.common.collect.ImmutableMap; -import com.google.devtools.build.docgen.annot.DocCategory; import com.google.devtools.build.lib.packages.Attribute; import net.starlark.java.annot.StarlarkBuiltin; import net.starlark.java.eval.StarlarkValue; @@ -26,10 +25,7 @@ /** * Represents a tag class, which is a "class" of {@link Tag}s that share the same attribute schema. */ -@StarlarkBuiltin( - name = "tag_class", - category = DocCategory.BUILTIN, - doc = "Defines a schema of attributes for a tag.") +@StarlarkBuiltin(name = "tag_class", doc = "Defines a schema of attributes for a tag.") @AutoValue public abstract class TagClass implements StarlarkValue { /** The list of attributes of this tag class. */ diff --git a/src/main/java/com/google/devtools/build/lib/bazel/repository/starlark/StarlarkRepositoryModule.java b/src/main/java/com/google/devtools/build/lib/bazel/repository/starlark/StarlarkRepositoryModule.java index cc67eca5b0437e..540b556c070f2e 100644 --- a/src/main/java/com/google/devtools/build/lib/bazel/repository/starlark/StarlarkRepositoryModule.java +++ b/src/main/java/com/google/devtools/build/lib/bazel/repository/starlark/StarlarkRepositoryModule.java @@ -316,7 +316,7 @@ public Object moduleExtension( StarlarkThread thread) throws EvalException { ModuleExtension.InStarlark inStarlark = new ModuleExtension.InStarlark(); - return inStarlark + inStarlark .getBuilder() .setImplementation(implementation) .setTagClasses( @@ -326,6 +326,7 @@ public Object moduleExtension( BazelModuleContext.of(Module.ofInnermostEnclosingStarlarkFunction(thread)).label()) .setEnvVariables(ImmutableList.copyOf(Sequence.cast(environ, String.class, "environ"))) .setLocation(thread.getCallerLocation()); + return inStarlark; } @StarlarkMethod( From 0ba107bc3aa109953827caf41082476503cac22d Mon Sep 17 00:00:00 2001 From: salma-samy Date: Tue, 1 Aug 2023 09:18:35 -0700 Subject: [PATCH 3/4] Re-run extension if its files change fixes https://github.com/bazelbuild/bazel/issues/19068 PiperOrigin-RevId: 552823534 Change-Id: I87256b2cf954b932e24c70e22386020599f21a6f --- .../lib/bazel/bzlmod/BazelLockFileValue.java | 17 +-- .../lib/bazel/bzlmod/GsonTypeAdapterUtil.java | 14 +++ .../bazel/bzlmod/LockFileModuleExtension.java | 6 +- .../bzlmod/SingleExtensionEvalFunction.java | 114 +++++++++++++++--- .../py/bazel/bzlmod/bazel_lockfile_test.py | 42 +++++++ 5 files changed, 170 insertions(+), 23 deletions(-) diff --git a/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/BazelLockFileValue.java b/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/BazelLockFileValue.java index 415c31cb62af2f..26c674fb6bc14f 100644 --- a/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/BazelLockFileValue.java +++ b/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/BazelLockFileValue.java @@ -109,25 +109,26 @@ public ImmutableList getModuleAndFlagsDiff( return moduleDiff.build(); } + /** Returns the differences between an extension and its locked data */ public ImmutableList getModuleExtensionDiff( - LockFileModuleExtension lockedExtension, - ImmutableMap lockedExtensionUsages, ModuleExtensionId extensionId, byte[] transitiveDigest, + boolean filesChanged, ImmutableMap envVariables, - ImmutableMap extensionUsages) { + ImmutableMap extensionUsages, + ImmutableMap lockedExtensionUsages) { + LockFileModuleExtension lockedExtension = getModuleExtensions().get(extensionId); + ImmutableList.Builder extDiff = new ImmutableList.Builder<>(); - if (lockedExtension == null) { - return extDiff - .add("The module extension '" + extensionId + "' does not exist in the lockfile") - .build(); - } if (!Arrays.equals(transitiveDigest, lockedExtension.getBzlTransitiveDigest())) { extDiff.add( "The implementation of the extension '" + extensionId + "' or one of its transitive .bzl files has changed"); } + if (filesChanged) { + extDiff.add("One or more files the extension '" + extensionId + "' is using have changed"); + } if (!extensionUsages.equals(lockedExtensionUsages)) { extDiff.add("The usages of the extension '" + extensionId + "' has changed"); } diff --git a/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/GsonTypeAdapterUtil.java b/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/GsonTypeAdapterUtil.java index f0190f4c9fb66f..334484716f8bc2 100644 --- a/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/GsonTypeAdapterUtil.java +++ b/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/GsonTypeAdapterUtil.java @@ -91,6 +91,19 @@ public ModuleKey read(JsonReader jsonReader) throws IOException { } }; + public static final TypeAdapter