Skip to content

Commit

Permalink
[7.4.0] Add override_repo and inject_repo (#23938)
Browse files Browse the repository at this point in the history
Work towards #19301
Fixes #23580

RELNOTES: `override_repo` and `inject_repo` can be used to override and
inject repos in module extensions.

Closes #23534.

PiperOrigin-RevId: 678139661
Change-Id: Iea7caca949c00e701f056c1037e273fee9740e93

(cherry picked from commit
46341b1)

Fixes #23724
Fixes #23799

Also includes:
* Disallow importing injected repos
(8472c9d)

---------

Co-authored-by: Xùdōng Yáng <[email protected]>
  • Loading branch information
fmeum and Wyverald authored Oct 10, 2024
1 parent 56dd476 commit 067ce4d
Show file tree
Hide file tree
Showing 23 changed files with 925 additions and 57 deletions.
57 changes: 57 additions & 0 deletions site/en/external/extension.md
Original file line number Diff line number Diff line change
Expand Up @@ -176,6 +176,63 @@ several repo visibility rules:
the repo visible to the module instead of the extension-generated repo
of the same name.

### Overriding and injecting module extension repos

The root module can use
[`override_repo`](/rules/lib/globals/module#override_repo) and
[`inject_repo`](/rules/lib/globals/module#inject_repo) to override or inject
module extension repos.

#### Example: Replacing `rules_java`'s `java_tools` with a vendored copy

```python
# MODULE.bazel
local_repository = use_repo_rule("@bazel_tools//tools/build_defs/repo:local.bzl", "local_repository")
local_repository(
name = "my_java_tools",
path = "vendor/java_tools",
)

bazel_dep(name = "rules_java", version = "7.11.1")
java_toolchains = use_extension("@rules_java//java:extension.bzl", "toolchains")

override_repo(java_toolchains, remote_java_tools = "my_java_tools")
```

#### Example: Patch a Go dependency to depend on `@zlib` instead of the system zlib

```python
# MODULE.bazel
bazel_dep(name = "gazelle", version = "0.38.0")
bazel_dep(name = "zlib", version = "1.3.1.bcr.3")

go_deps = use_extension("@gazelle//:extensions.bzl", "go_deps")
go_deps.from_file(go_mod = "//:go.mod")
go_deps.module_override(
patches = [
"//patches:my_module_zlib.patch",
],
path = "example.com/my_module",
)
use_repo(go_deps, ...)

inject_repo(go_deps, "zlib")
```

```diff
# patches/my_module_zlib.patch
--- a/BUILD.bazel
+++ b/BUILD.bazel
@@ -1,6 +1,6 @@
go_binary(
name = "my_module",
importpath = "example.com/my_module",
srcs = ["my_module.go"],
- copts = ["-lz"],
+ cdeps = ["@zlib"],
)
```

## Best practices

This section describes best practices when writing extensions so they are
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -79,15 +79,24 @@ public SkyValue compute(SkyKey skyKey, Environment env)
ImmutableBiMap<String, ModuleExtensionId> extensionUniqueNames =
calculateUniqueNameForUsedExtensionId(extensionUsagesById, starlarkSemantics);

char repoNameSeparator =
starlarkSemantics.getBool(BuildLanguageOptions.INCOMPATIBLE_USE_PLUS_IN_REPO_NAMES)
? '+'
: '~';

return BazelDepGraphValue.create(
depGraph,
canonicalRepoNameLookup,
depGraph.values().stream().map(AbridgedModule::from).collect(toImmutableList()),
extensionUsagesById,
extensionUniqueNames.inverse(),
starlarkSemantics.getBool(BuildLanguageOptions.INCOMPATIBLE_USE_PLUS_IN_REPO_NAMES)
? '+'
: '~');
resolveRepoOverrides(
depGraph,
extensionUsagesById,
extensionUniqueNames.inverse(),
canonicalRepoNameLookup,
repoNameSeparator),
repoNameSeparator);
}

private static ImmutableTable<ModuleExtensionId, ModuleKey, ModuleExtensionUsage>
Expand Down Expand Up @@ -218,6 +227,40 @@ private static String makeUniqueNameCandidate(
+ extensionNameDisambiguator);
}

private static ImmutableTable<ModuleExtensionId, String, RepositoryName> resolveRepoOverrides(
ImmutableMap<ModuleKey, Module> depGraph,
ImmutableTable<ModuleExtensionId, ModuleKey, ModuleExtensionUsage> extensionUsagesTable,
ImmutableMap<ModuleExtensionId, String> extensionUniqueNames,
ImmutableBiMap<RepositoryName, ModuleKey> canonicalRepoNameLookup,
char repoNameSeparator) {
RepositoryMapping rootModuleMappingWithoutOverrides =
BazelDepGraphValue.getRepositoryMapping(
ModuleKey.ROOT,
depGraph,
extensionUsagesTable,
extensionUniqueNames,
canonicalRepoNameLookup,
// ModuleFileFunction ensures that repos that override other repos are not themselves
// overridden, so we can safely pass an empty table here instead of resolving chains
// of overrides.
ImmutableTable.of(),
repoNameSeparator);
ImmutableTable.Builder<ModuleExtensionId, String, RepositoryName> repoOverridesBuilder =
ImmutableTable.builder();
for (var extensionId : extensionUsagesTable.rowKeySet()) {
var rootUsage = extensionUsagesTable.row(extensionId).get(ModuleKey.ROOT);
if (rootUsage != null) {
for (var override : rootUsage.getRepoOverrides().entrySet()) {
repoOverridesBuilder.put(
extensionId,
override.getKey(),
rootModuleMappingWithoutOverrides.get(override.getValue().overridingRepoName()));
}
}
}
return repoOverridesBuilder.buildOrThrow();
}

static class BazelDepGraphFunctionException extends SkyFunctionException {
BazelDepGraphFunctionException(ExternalDepsException e, Transience transience) {
super(e, transience);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -43,13 +43,15 @@ public static BazelDepGraphValue create(
ImmutableList<AbridgedModule> abridgedModules,
ImmutableTable<ModuleExtensionId, ModuleKey, ModuleExtensionUsage> extensionUsagesTable,
ImmutableMap<ModuleExtensionId, String> extensionUniqueNames,
ImmutableTable<ModuleExtensionId, String, RepositoryName> repoOverrides,
char repoNameSeparator) {
return new AutoValue_BazelDepGraphValue(
depGraph,
ImmutableBiMap.copyOf(canonicalRepoNameLookup),
abridgedModules,
extensionUsagesTable,
extensionUniqueNames,
repoOverrides,
repoNameSeparator);
}

Expand All @@ -75,6 +77,7 @@ public static BazelDepGraphValue createEmptyDepGraph() {
ImmutableList.of(),
ImmutableTable.of(),
ImmutableMap.of(),
ImmutableTable.of(),
'+');
}

Expand Down Expand Up @@ -107,6 +110,12 @@ public static BazelDepGraphValue createEmptyDepGraph() {
*/
public abstract ImmutableMap<ModuleExtensionId, String> getExtensionUniqueNames();

/**
* For each module extension, a mapping from the name of the repo exported by the extension to the
* canonical name of the repo that should override it (if any).
*/
public abstract ImmutableTable<ModuleExtensionId, String, RepositoryName> getRepoOverrides();

/** The character to use to separate the different segments of a canonical repo name. */
public abstract char getRepoNameSeparator();

Expand All @@ -115,22 +124,45 @@ public static BazelDepGraphValue createEmptyDepGraph() {
* module deps and module extensions.
*/
public final RepositoryMapping getFullRepoMapping(ModuleKey key) {
return getRepositoryMapping(
key,
getDepGraph(),
getExtensionUsagesTable(),
getExtensionUniqueNames(),
getCanonicalRepoNameLookup(),
getRepoOverrides(),
getRepoNameSeparator());
}

static RepositoryMapping getRepositoryMapping(
ModuleKey key,
ImmutableMap<ModuleKey, Module> depGraph,
ImmutableTable<ModuleExtensionId, ModuleKey, ModuleExtensionUsage> extensionUsagesTable,
ImmutableMap<ModuleExtensionId, String> extensionUniqueNames,
ImmutableBiMap<RepositoryName, ModuleKey> canonicalRepoNameLookup,
ImmutableTable<ModuleExtensionId, String, RepositoryName> repoOverrides,
char repoNameSeparator) {
ImmutableMap.Builder<String, RepositoryName> mapping = ImmutableMap.builder();
for (Map.Entry<ModuleExtensionId, ModuleExtensionUsage> extIdAndUsage :
getExtensionUsagesTable().column(key).entrySet()) {
extensionUsagesTable.column(key).entrySet()) {
ModuleExtensionId extensionId = extIdAndUsage.getKey();
ModuleExtensionUsage usage = extIdAndUsage.getValue();
String repoNamePrefix = getExtensionUniqueNames().get(extensionId) + getRepoNameSeparator();
String repoNamePrefix = extensionUniqueNames.get(extensionId) + repoNameSeparator;
for (ModuleExtensionUsage.Proxy proxy : usage.getProxies()) {
for (Map.Entry<String, String> entry : proxy.getImports().entrySet()) {
String canonicalRepoName = repoNamePrefix + entry.getValue();
mapping.put(entry.getKey(), RepositoryName.createUnvalidated(canonicalRepoName));
RepositoryName defaultCanonicalRepoName =
RepositoryName.createUnvalidated(repoNamePrefix + entry.getValue());
mapping.put(
entry.getKey(),
repoOverrides
.row(extensionId)
.getOrDefault(entry.getValue(), defaultCanonicalRepoName));
}
}
}
return getDepGraph()
return depGraph
.get(key)
.getRepoMappingWithBazelDepsOnly(getCanonicalRepoNameLookup().inverse())
.getRepoMappingWithBazelDepsOnly(canonicalRepoNameLookup.inverse())
.withAdditionalMappings(mapping.buildOrThrow());
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -66,6 +66,7 @@ record RepoRuleCall(
private final String repoPrefix;
private final PackageIdentifier basePackageId;
private final RepositoryMapping baseRepoMapping;
private final ImmutableMap<String, RepositoryName> repoOverrides;
private final BlazeDirectories directories;
private final ExtendedEventHandler eventHandler;
private final Map<String, RepoRuleCall> deferredRepos = new LinkedHashMap<>();
Expand All @@ -75,12 +76,14 @@ public ModuleExtensionEvalStarlarkThreadContext(
String repoPrefix,
PackageIdentifier basePackageId,
RepositoryMapping baseRepoMapping,
ImmutableMap<String, RepositoryName> repoOverrides,
BlazeDirectories directories,
ExtendedEventHandler eventHandler) {
this.extensionId = extensionId;
this.repoPrefix = repoPrefix;
this.basePackageId = basePackageId;
this.baseRepoMapping = baseRepoMapping;
this.repoOverrides = repoOverrides;
this.directories = directories;
this.eventHandler = eventHandler;
}
Expand Down Expand Up @@ -127,13 +130,15 @@ public ImmutableMap<String, RepoSpec> createRepos(StarlarkSemantics starlarkSema
// Make it possible to refer to extension repos in the label attributes of another extension
// repo. Wrapping a label in Label(...) ensures that it is evaluated with respect to the
// containing module's repo mapping instead.
var extensionRepos =
ImmutableMap.Builder<String, RepositoryName> entries = ImmutableMap.builder();
entries.putAll(baseRepoMapping.entries());
entries.putAll(
Maps.asMap(
deferredRepos.keySet(),
apparentName -> RepositoryName.createUnvalidated(repoPrefix + apparentName));
apparentName -> RepositoryName.createUnvalidated(repoPrefix + apparentName)));
entries.putAll(repoOverrides);
RepositoryMapping fullRepoMapping =
RepositoryMapping.create(extensionRepos, baseRepoMapping.ownerRepo())
.withAdditionalMappings(baseRepoMapping);
RepositoryMapping.create(entries.buildKeepingLast(), baseRepoMapping.ownerRepo());
// LINT.ThenChange(//src/main/java/com/google/devtools/build/lib/bazel/bzlmod/ModuleExtensionRepoMappingEntriesFunction.java)

ImmutableMap.Builder<String, RepoSpec> repoSpecs = ImmutableMap.builder();
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -74,6 +74,7 @@ private ModuleExtensionRepoMappingEntriesValue computeRepoMappingEntries(
ImmutableMap.Builder<String, RepositoryName> entries = ImmutableMap.builder();
entries.putAll(bazelDepGraphValue.getFullRepoMapping(moduleKey).entries());
entries.putAll(extensionEvalValue.getCanonicalRepoNameToInternalNames().inverse());
entries.putAll(bazelDepGraphValue.getRepoOverrides().row(extensionId));
return ModuleExtensionRepoMappingEntriesValue.create(entries.buildKeepingLast(), moduleKey);
// LINT.ThenChange(//src/main/java/com/google/devtools/build/lib/bazel/bzlmod/ModuleExtensionEvalStarlarkThreadContext.java)
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,7 @@
import com.google.auto.value.AutoValue;
import com.google.common.collect.ImmutableBiMap;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableMap;
import com.google.devtools.build.lib.vfs.PathFragment;
import com.google.errorprone.annotations.CanIgnoreReturnValue;
import com.ryanharter.auto.value.gson.GenerateTypeAdapter;
Expand Down Expand Up @@ -130,6 +131,24 @@ public final boolean getHasNonDevUseExtension() {
return getProxies().stream().anyMatch(p -> !p.isDevDependency());
}

/**
* Represents a repo that overrides another repo within the scope of the extension.
*
* @param overridingRepoName The apparent name of the overriding repo in the root module.
* @param mustExist Whether this override should apply to an existing repo.
* @param location The location of the {@code override_repo} or {@code inject_repo} call.
*/
@GenerateTypeAdapter
public record RepoOverride(String overridingRepoName, boolean mustExist, Location location) {}

/**
* Contains information about overrides that apply to repos generated by this extension. Keyed by
* the extension-local repo name.
*
* <p>This is only non-empty for root module usages.
*/
public abstract ImmutableMap<String, RepoOverride> getRepoOverrides();

public abstract Builder toBuilder();

public static Builder builder() {
Expand All @@ -152,6 +171,11 @@ ModuleExtensionUsage trimForEvaluation() {
// Extension implementation functions do not see the imports, they are only validated
// against the set of generated repos in a validation step that comes afterward.
.setProxies(ImmutableList.of())
// Tracked in SingleExtensionUsagesValue instead, using canonical instead of apparent names.
// Whether this override must apply to an existing repo as well as its source location also
// don't influence the evaluation of the extension as they are checked in
// SingleExtensionFunction.
.setRepoOverrides(ImmutableMap.of())
.build();
}

Expand Down Expand Up @@ -185,6 +209,9 @@ public Builder addTag(Tag value) {
return this;
}

@CanIgnoreReturnValue
public abstract Builder setRepoOverrides(ImmutableMap<String, RepoOverride> repoOverrides);

public abstract ModuleExtensionUsage build();
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -435,7 +435,7 @@ public static RootModuleFileValue evaluateRootModuleFile(
try {
module = moduleThreadContext.buildModule(/* registry= */ null);
} catch (EvalException e) {
eventHandler.handle(Event.error(e.getMessageWithStack()));
eventHandler.handle(Event.error(e.getInnermostLocation(), e.getMessageWithStack()));
throw errorf(Code.BAD_MODULE, "error executing MODULE.bazel file for the root module");
}
for (ModuleExtensionUsage usage : module.getExtensionUsages()) {
Expand Down Expand Up @@ -521,7 +521,7 @@ private static ModuleThreadContext execModuleFile(
});
compiledRootModuleFile.runOnThread(thread);
} catch (EvalException e) {
eventHandler.handle(Event.error(e.getMessageWithStack()));
eventHandler.handle(Event.error(e.getInnermostLocation(), e.getMessageWithStack()));
throw errorf(Code.BAD_MODULE, "error executing MODULE.bazel file for %s", moduleKey);
}
return context;
Expand Down
Loading

0 comments on commit 067ce4d

Please sign in to comment.