From 4d34d12217456afae48f9d0a625dfb1064c676e0 Mon Sep 17 00:00:00 2001 From: Fabian Meumertzheim Date: Tue, 30 May 2023 12:35:33 +0200 Subject: [PATCH] Allow module extension usages to be isolated If `isolated = True` is specified on `use_extension`, that particular usage will be isolated from all other usages, both in the same and other modules. Module extensions can check whether they are isolated (e.g. in case they can only be used in this way) via `module_ctx.is_isolated`. --- .../bazel/bzlmod/BazelDepGraphFunction.java | 23 ++++++++++++++-- .../bazel/bzlmod/ModuleExtensionContext.java | 10 +++++++ .../lib/bazel/bzlmod/ModuleExtensionId.java | 17 ++++++++++-- .../bazel/bzlmod/ModuleExtensionUsage.java | 4 +++ .../lib/bazel/bzlmod/ModuleFileGlobals.java | 27 ++++++++++++++----- .../bzlmod/BazelDepGraphFunctionTest.java | 12 ++++++--- .../bzlmod/ModuleExtensionResolutionTest.java | 2 +- .../bazel/bzlmod/ModuleFileFunctionTest.java | 3 +++ .../bazel/bzlmod/StarlarkBazelModuleTest.java | 1 + 9 files changed, 84 insertions(+), 15 deletions(-) diff --git a/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/BazelDepGraphFunction.java b/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/BazelDepGraphFunction.java index 14bfbdafc730f0..d6c86f6e72652c 100644 --- a/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/BazelDepGraphFunction.java +++ b/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/BazelDepGraphFunction.java @@ -45,6 +45,7 @@ import java.util.List; import java.util.Map; import java.util.Map.Entry; +import java.util.Optional; import javax.annotation.Nullable; /** @@ -214,7 +215,11 @@ private ImmutableTable getEx try { moduleExtensionId = ModuleExtensionId.create( - labelConverter.convert(usage.getExtensionBzlFile()), usage.getExtensionName()); + labelConverter.convert(usage.getExtensionBzlFile()), + usage.getExtensionName(), + usage.isIsolated() + ? Optional.of(new ModuleExtensionId.Namespace(usage.getUsingModule())) + : Optional.empty()); } catch (LabelSyntaxException e) { throw new BazelDepGraphFunctionException( ExternalDepsException.withCauseAndMessage( @@ -248,7 +253,21 @@ private ImmutableBiMap calculateUniqueNameForUsedExte // not start with a tilde. RepositoryName repository = id.getBzlFileLabel().getRepository(); String nonEmptyRepoPart = repository.isMain() ? "_main" : repository.getName(); - String bestName = nonEmptyRepoPart + "~" + id.getExtensionName(); + // When using a namespace, prefix the extension name with "_" to distinguish the prefix from + // those generated by non-namespaced extension usages. Extension names are identified by their + // Starlark identifier, which in the case of an exported symbol cannot start with "_". + String bestName = + id.getNamespace() + .map( + namespace -> + nonEmptyRepoPart + + "~_" + + id.getExtensionName() + + "~" + + namespace.module.getName() + + "~" + + namespace.module.getVersion()) + .orElse(nonEmptyRepoPart + "~" + id.getExtensionName()); if (extensionUniqueNames.putIfAbsent(bestName, id) == null) { continue; } diff --git a/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/ModuleExtensionContext.java b/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/ModuleExtensionContext.java index 38820bd2a12671..4a79493e6bea9d 100644 --- a/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/ModuleExtensionContext.java +++ b/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/ModuleExtensionContext.java @@ -123,6 +123,16 @@ public boolean isDevDependency(TypeCheckedTag tag) { return tag.isDevDependency(); } + @StarlarkMethod( + name = "is_isolated", + doc = "Whether this particular usage of the extension is isolated from all others, in " + + "particular those in other modules", + structField = true + ) + public boolean isIsolated() { + return extensionId.getNamespace().isPresent(); + } + @StarlarkMethod( name = "extension_metadata", doc = diff --git a/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/ModuleExtensionId.java b/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/ModuleExtensionId.java index 9fef1ef77aaf3c..9f19f4797848a3 100644 --- a/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/ModuleExtensionId.java +++ b/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/ModuleExtensionId.java @@ -17,15 +17,28 @@ import com.google.auto.value.AutoValue; import com.google.devtools.build.lib.cmdline.Label; +import java.util.Optional; /** A unique identifier for a {@link ModuleExtension}. */ @AutoValue public abstract class ModuleExtensionId { + + static final class Namespace { + final ModuleKey module; + + Namespace(ModuleKey module) { + this.module = module; + } + } + public abstract Label getBzlFileLabel(); public abstract String getExtensionName(); - public static ModuleExtensionId create(Label bzlFileLabel, String extensionName) { - return new AutoValue_ModuleExtensionId(bzlFileLabel, extensionName); + public abstract Optional getNamespace(); + + public static ModuleExtensionId create( + Label bzlFileLabel, String extensionName, Optional namespace) { + return new AutoValue_ModuleExtensionId(bzlFileLabel, extensionName, namespace); } } diff --git a/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/ModuleExtensionUsage.java b/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/ModuleExtensionUsage.java index 31cff67ecace10..eac0046eeadfa9 100644 --- a/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/ModuleExtensionUsage.java +++ b/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/ModuleExtensionUsage.java @@ -35,6 +35,8 @@ public abstract class ModuleExtensionUsage { /** The name of the extension. */ public abstract String getExtensionName(); + public abstract boolean isIsolated(); + /** The module that contains this particular extension usage. */ public abstract ModuleKey getUsingModule(); @@ -73,6 +75,8 @@ public abstract static class Builder { public abstract Builder setExtensionName(String value); + public abstract Builder setIsolated(boolean value); + public abstract Builder setUsingModule(ModuleKey value); public abstract Builder setLocation(Location value); diff --git a/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/ModuleFileGlobals.java b/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/ModuleFileGlobals.java index efa2678a21f2d6..76f320b43a57b9 100644 --- a/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/ModuleFileGlobals.java +++ b/src/main/java/com/google/devtools/build/lib/bazel/bzlmod/ModuleFileGlobals.java @@ -432,12 +432,21 @@ public void registerToolchains(boolean devDependency, Sequence toolchainLabel named = true, positional = false, defaultValue = "False"), + @Param( + name = "isolated", + doc = + "If true, this usage of the module extension will be isolated from all other " + + "usages, in particular those in other modules.", + named = true, + positional = false, + defaultValue = "False"), }, useStarlarkThread = true) public ModuleExtensionProxy useExtension( String rawExtensionBzlFile, String extensionName, boolean devDependency, + boolean isolated, StarlarkThread thread) { hadNonModuleCall = true; @@ -445,7 +454,7 @@ public ModuleExtensionProxy useExtension( ModuleExtensionUsageBuilder newUsageBuilder = new ModuleExtensionUsageBuilder( - extensionBzlFile, extensionName, thread.getCallerLocation()); + extensionBzlFile, extensionName, isolated, thread.getCallerLocation()); if (ignoreDevDeps && devDependency) { // This is a no-op proxy. @@ -453,10 +462,12 @@ public ModuleExtensionProxy useExtension( } // Find an existing usage builder corresponding to this extension. - for (ModuleExtensionUsageBuilder usageBuilder : extensionUsageBuilders) { - if (usageBuilder.extensionBzlFile.equals(extensionBzlFile) - && usageBuilder.extensionName.equals(extensionName)) { - return usageBuilder.getProxy(devDependency); + if (!isolated) { + for (ModuleExtensionUsageBuilder usageBuilder : extensionUsageBuilders) { + if (usageBuilder.extensionBzlFile.equals(extensionBzlFile) + && usageBuilder.extensionName.equals(extensionName)) { + return usageBuilder.getProxy(devDependency); + } } } @@ -484,14 +495,17 @@ private String normalizeLabelString(String rawExtensionBzlFile) { class ModuleExtensionUsageBuilder { private final String extensionBzlFile; private final String extensionName; + private final boolean isolated; private final Location location; private final HashBiMap imports; private final ImmutableSet.Builder devImports; private final ImmutableList.Builder tags; - ModuleExtensionUsageBuilder(String extensionBzlFile, String extensionName, Location location) { + ModuleExtensionUsageBuilder( + String extensionBzlFile, String extensionName, boolean isolated, Location location) { this.extensionBzlFile = extensionBzlFile; this.extensionName = extensionName; + this.isolated = isolated; this.location = location; this.imports = HashBiMap.create(); this.devImports = ImmutableSet.builder(); @@ -502,6 +516,7 @@ ModuleExtensionUsage buildUsage() { return ModuleExtensionUsage.builder() .setExtensionBzlFile(extensionBzlFile) .setExtensionName(extensionName) + .setIsolated(isolated) .setUsingModule(module.getKey()) .setLocation(location) .setImports(ImmutableBiMap.copyOf(imports)) diff --git a/src/test/java/com/google/devtools/build/lib/bazel/bzlmod/BazelDepGraphFunctionTest.java b/src/test/java/com/google/devtools/build/lib/bazel/bzlmod/BazelDepGraphFunctionTest.java index 468ac41d6c9228..2e46e898137e62 100644 --- a/src/test/java/com/google/devtools/build/lib/bazel/bzlmod/BazelDepGraphFunctionTest.java +++ b/src/test/java/com/google/devtools/build/lib/bazel/bzlmod/BazelDepGraphFunctionTest.java @@ -64,6 +64,7 @@ import com.google.devtools.build.skyframe.SkyFunctionName; import com.google.devtools.build.skyframe.SkyKey; import com.google.devtools.build.skyframe.SkyValue; +import java.util.Optional; import java.util.concurrent.atomic.AtomicReference; import javax.annotation.Nullable; import net.starlark.java.eval.StarlarkSemantics; @@ -214,6 +215,7 @@ private static ModuleExtensionUsage createModuleExtensionUsage( return ModuleExtensionUsage.builder() .setExtensionBzlFile(bzlFile) .setExtensionName(name) + .setIsolated(false) .setImports(importsBuilder.buildOrThrow()) .setDevImports(ImmutableSet.of()) .setUsingModule(ModuleKey.ROOT) @@ -253,14 +255,16 @@ public void createValue_moduleExtensions() throws Exception { ModuleExtensionId maven = ModuleExtensionId.create( - Label.parseCanonical("@@rules_jvm_external~1.0//:defs.bzl"), "maven"); + Label.parseCanonical("@@rules_jvm_external~1.0//:defs.bzl"), "maven", Optional.empty()); ModuleExtensionId pip = - ModuleExtensionId.create(Label.parseCanonical("@@rules_python~2.0//:defs.bzl"), "pip"); + ModuleExtensionId.create( + Label.parseCanonical("@@rules_python~2.0//:defs.bzl"), "pip", Optional.empty()); ModuleExtensionId myext = - ModuleExtensionId.create(Label.parseCanonical("@@dep~2.0//:defs.bzl"), "myext"); + ModuleExtensionId.create( + Label.parseCanonical("@@dep~2.0//:defs.bzl"), "myext", Optional.empty()); ModuleExtensionId myext2 = ModuleExtensionId.create( - Label.parseCanonical("@@dep~2.0//incredible:conflict.bzl"), "myext"); + Label.parseCanonical("@@dep~2.0//incredible:conflict.bzl"), "myext", Optional.empty()); resolutionFunctionMock.setDepGraph(depGraph); EvaluationResult result = diff --git a/src/test/java/com/google/devtools/build/lib/bazel/bzlmod/ModuleExtensionResolutionTest.java b/src/test/java/com/google/devtools/build/lib/bazel/bzlmod/ModuleExtensionResolutionTest.java index a51d03c9da55da..9a491f54aa42c6 100644 --- a/src/test/java/com/google/devtools/build/lib/bazel/bzlmod/ModuleExtensionResolutionTest.java +++ b/src/test/java/com/google/devtools/build/lib/bazel/bzlmod/ModuleExtensionResolutionTest.java @@ -1890,7 +1890,7 @@ private EvaluationResult evaluateSimpleModuleExtension scratch.file(workspaceRoot.getRelative("BUILD").getPathString()); ModuleExtensionId extensionId = - ModuleExtensionId.create(Label.parseCanonical("//:defs.bzl"), "ext"); + ModuleExtensionId.create(Label.parseCanonical("//:defs.bzl"), "ext", Optional.empty()); reporter.removeHandler(failFastHandler); return evaluator.evaluate( ImmutableList.of(SingleExtensionEvalValue.key(extensionId)), evaluationContext); diff --git a/src/test/java/com/google/devtools/build/lib/bazel/bzlmod/ModuleFileFunctionTest.java b/src/test/java/com/google/devtools/build/lib/bazel/bzlmod/ModuleFileFunctionTest.java index 00fb41c19f64b3..2469ab30dd75b2 100644 --- a/src/test/java/com/google/devtools/build/lib/bazel/bzlmod/ModuleFileFunctionTest.java +++ b/src/test/java/com/google/devtools/build/lib/bazel/bzlmod/ModuleFileFunctionTest.java @@ -476,6 +476,7 @@ public void testModuleExtensions_good() throws Exception { ModuleExtensionUsage.builder() .setExtensionBzlFile("@mymod//:defs.bzl") .setExtensionName("myext1") + .setIsolated(false) .setUsingModule(myMod) .setLocation(Location.fromFileLineColumn("mymod@1.0/MODULE.bazel", 2, 23)) .setImports(ImmutableBiMap.of("repo1", "repo1")) @@ -596,6 +597,7 @@ public void testModuleExtensions_duplicateProxy_asRoot() throws Exception { ModuleExtensionUsage.builder() .setExtensionBzlFile("@//:defs.bzl") .setExtensionName("myext") + .setIsolated(false) .setUsingModule(ModuleKey.ROOT) .setLocation(Location.fromFileLineColumn("/MODULE.bazel", 1, 23)) .setImports( @@ -693,6 +695,7 @@ public void testModuleExtensions_duplicateProxy_asDep() throws Exception { ModuleExtensionUsage.builder() .setExtensionBzlFile("@mymod//:defs.bzl") .setExtensionName("myext") + .setIsolated(false) .setUsingModule(myMod) .setLocation(Location.fromFileLineColumn("mymod@1.0/MODULE.bazel", 5, 23)) .setImports(ImmutableBiMap.of("beta", "beta", "delta", "delta")) 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 aa017278599247..8c3addfdefc593 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 @@ -45,6 +45,7 @@ private static ModuleExtensionUsage.Builder getBaseUsageBuilder() { return ModuleExtensionUsage.builder() .setExtensionBzlFile("//:rje.bzl") .setExtensionName("maven") + .setIsolated(false) .setUsingModule(ModuleKey.ROOT) .setLocation(Location.BUILTIN) .setImports(ImmutableBiMap.of())