Skip to content

Commit

Permalink
Rewrite symlinks for vendored repositories
Browse files Browse the repository at this point in the history
To make sure symlinks work correctly, Bazel uses the following strategy to rewrite symlinks in the vendored source:

  - Create a symlink `<vendor_dir>/bazel-external` that points to `$(bazel info output_base)/external`. It is refreshed by every Bazel command automatically.
  - For the vendored source, rewrite all symlinks that originally point to a path under `$(bazel info output_base)/external` to a relative path under `<vendor_dir>/bazel-external`.

Fixes bazelbuild#22303

Closes bazelbuild#22723.

PiperOrigin-RevId: 644349481
Change-Id: I853ac0ea5405f0cf58431e988d727e690cbbb013
  • Loading branch information
meteorcloudy committed Jun 18, 2024
1 parent e2a0396 commit 32fff52
Show file tree
Hide file tree
Showing 8 changed files with 277 additions and 41 deletions.
59 changes: 57 additions & 2 deletions site/en/external/vendor.md
Original file line number Diff line number Diff line change
Expand Up @@ -141,8 +141,8 @@ always excluded from vendoring.

Bazel fetches external dependencies of a project under `$(bazel info
output_base)/external`. Vendoring external dependencies means moving out
relevant files and directories to a given vendor directory and use the vendored
source for later builds.
relevant files and directories to the given vendor directory and use the
vendored source for later builds.

The content being vendored includes:

Expand All @@ -160,3 +160,58 @@ is pinned in the VENDOR.bazel file. If a user does change the vendored source
without pinning the repo, the changed vendored source will be used, but it will
be overwritten if its existing marker file is
outdated and the repo is vendored again.

### Vendor registry files {:#vendor-registry-files}

Bazel has to perform the Bazel module resolution in order to fetch external
dependencies, which may require accessing registry files through internet. To
achieve offline build, Bazel vendors all registry files fetched from
network under the `<vendor_dir>/_registries` directory.

### Vendor symlinks {:#vendor-symlinks}

External repositories may contain symlinks pointing to other files or
directories. To make sure symlinks work correctly, Bazel uses the following
strategy to rewrite symlinks in the vendored source:

- Create a symlink `<vendor_dir>/bazel-external` that points to `$(bazel info
output_base)/external`. It is refreshed by every Bazel command
automatically.
- For the vendored source, rewrite all symlinks that originally point to a
path under `$(bazel info output_base)/external` to a relative path under
`<vendor_dir>/bazel-external`.

For example, if the original symlink is

```none
<vendor_dir>/repo_foo~/link => $(bazel info output_base)/external/repo_bar~/file
```

It will be rewritten to

```none
<vendor_dir>/repo_foo~/link => ../../bazel-external/repo_bar~/file
```

where

```none
<vendor_dir>/bazel-external => $(bazel info output_base)/external # This might be new if output base is changed
```

Since `<vendor_dir>/bazel-external` is generated by Bazel automatically, it's
recommended to add it to `.gitignore` or equivalent to avoid checking it in.

With this strategy, symlinks in the vendored source should work correctly even
after the vendored source is moved to another location or the bazel output base
is changed.

Note: symlinks that point to an absolute path outside of $(bazel info
output_base)/external are not rewritten. Therefore, it could still break
cross-machine compatibility.

Note: On Windows, vendoring symlinks only works with
[`--windows_enable_symlinks`][windows_enable_symlinks]
flag enabled.

[windows_enable_symlinks]: /reference/command-line-reference#flag--windows_enable_symlinks
2 changes: 2 additions & 0 deletions src/main/java/com/google/devtools/build/lib/bazel/BUILD
Original file line number Diff line number Diff line change
Expand Up @@ -34,6 +34,7 @@ java_library(
"//src/main/java/com/google/devtools/build/lib/bazel/bzlmod:resolution",
"//src/main/java/com/google/devtools/build/lib/bazel/bzlmod:resolution_impl",
"//src/main/java/com/google/devtools/build/lib/bazel/bzlmod:tidy_impl",
"//src/main/java/com/google/devtools/build/lib/bazel/bzlmod:vendor",
"//src/main/java/com/google/devtools/build/lib/bazel/commands",
"//src/main/java/com/google/devtools/build/lib/bazel/repository",
"//src/main/java/com/google/devtools/build/lib/bazel/repository:repository_options",
Expand All @@ -57,6 +58,7 @@ java_library(
"//src/main/java/com/google/devtools/build/lib/starlarkbuildapi/repository",
"//src/main/java/com/google/devtools/build/lib/util:abrupt_exit_exception",
"//src/main/java/com/google/devtools/build/lib/util:detailed_exit_code",
"//src/main/java/com/google/devtools/build/lib/util:os",
"//src/main/java/com/google/devtools/build/lib/vfs",
"//src/main/java/com/google/devtools/build/lib/vfs:pathfragment",
"//src/main/java/com/google/devtools/common/options",
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -55,6 +55,7 @@
import com.google.devtools.build.lib.bazel.bzlmod.SingleExtensionFunction;
import com.google.devtools.build.lib.bazel.bzlmod.SingleExtensionUsagesFunction;
import com.google.devtools.build.lib.bazel.bzlmod.VendorFileFunction;
import com.google.devtools.build.lib.bazel.bzlmod.VendorManager;
import com.google.devtools.build.lib.bazel.bzlmod.YankedVersionsFunction;
import com.google.devtools.build.lib.bazel.bzlmod.YankedVersionsUtil;
import com.google.devtools.build.lib.bazel.commands.FetchCommand;
Expand Down Expand Up @@ -82,6 +83,7 @@
import com.google.devtools.build.lib.bazel.rules.android.AndroidSdkRepositoryFunction;
import com.google.devtools.build.lib.bazel.rules.android.AndroidSdkRepositoryRule;
import com.google.devtools.build.lib.clock.Clock;
import com.google.devtools.build.lib.cmdline.LabelConstants;
import com.google.devtools.build.lib.cmdline.RepositoryName;
import com.google.devtools.build.lib.events.Event;
import com.google.devtools.build.lib.pkgcache.PackageOptions;
Expand Down Expand Up @@ -114,7 +116,9 @@
import com.google.devtools.build.lib.starlarkbuildapi.repository.RepositoryBootstrap;
import com.google.devtools.build.lib.util.AbruptExitException;
import com.google.devtools.build.lib.util.DetailedExitCode;
import com.google.devtools.build.lib.util.OS;
import com.google.devtools.build.lib.vfs.FileSystem;
import com.google.devtools.build.lib.vfs.FileSystemUtils;
import com.google.devtools.build.lib.vfs.Path;
import com.google.devtools.build.lib.vfs.PathFragment;
import com.google.devtools.build.lib.vfs.Root;
Expand Down Expand Up @@ -507,6 +511,33 @@ public void beforeCommand(CommandEnvironment env) throws AbruptExitException {
vendorDirectory =
Optional.ofNullable(repoOptions.vendorDirectory)
.map(vendorDirectory -> env.getWorkspace().getRelative(vendorDirectory));

if (vendorDirectory.isPresent()) {
try {
Path externalRoot =
env.getOutputBase().getRelative(LabelConstants.EXTERNAL_PATH_PREFIX);
FileSystemUtils.ensureSymbolicLink(
vendorDirectory.get().getChild(VendorManager.EXTERNAL_ROOT_SYMLINK_NAME),
externalRoot);
if (OS.getCurrent() == OS.WINDOWS) {
// On Windows, symlinks are resolved differently.
// Given <external>/repo_foo/link,
// where <external>/repo_foo points to <vendor dir>/repo_foo in vendor mode
// and repo_foo/link points to a relative path ../bazel-external/repo_bar/data.
// Windows won't resolve `repo_foo` before resolving `link`, which causes
// <external>/repo_foo/link to be resolved to <external>/bazel-external/repo_bar/data
// To work around this, we create a symlink <external>/bazel-external -> <external>.
FileSystemUtils.ensureSymbolicLink(
externalRoot.getChild(VendorManager.EXTERNAL_ROOT_SYMLINK_NAME), externalRoot);
}
} catch (IOException e) {
env.getReporter()
.handle(
Event.error(
"Failed to create symlink to external repo root under vendor directory: "
+ e.getMessage()));
}
}
}

if (repoOptions.registries != null && !repoOptions.registries.isEmpty()) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -52,7 +52,10 @@ java_library(
deps = [
"//src/main/java/com/google/devtools/build/lib/bazel/repository/downloader",
"//src/main/java/com/google/devtools/build/lib/cmdline",
"//src/main/java/com/google/devtools/build/lib/profiler",
"//src/main/java/com/google/devtools/build/lib/util:os",
"//src/main/java/com/google/devtools/build/lib/vfs",
"//src/main/java/com/google/devtools/build/lib/vfs:pathfragment",
"//third_party:guava",
],
)
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -20,12 +20,19 @@
import com.google.common.hash.Hasher;
import com.google.devtools.build.lib.bazel.repository.downloader.Checksum;
import com.google.devtools.build.lib.cmdline.RepositoryName;
import com.google.devtools.build.lib.profiler.Profiler;
import com.google.devtools.build.lib.profiler.ProfilerTask;
import com.google.devtools.build.lib.profiler.SilentCloseable;
import com.google.devtools.build.lib.util.OS;
import com.google.devtools.build.lib.vfs.FileSystemUtils;
import com.google.devtools.build.lib.vfs.Path;
import com.google.devtools.build.lib.vfs.PathFragment;
import java.io.IOException;
import java.io.UnsupportedEncodingException;
import java.net.URL;
import java.net.URLDecoder;
import java.nio.file.Files;
import java.util.Collection;
import java.util.Locale;
import java.util.Objects;

Expand All @@ -34,6 +41,8 @@ public class VendorManager {

private static final String REGISTRIES_DIR = "_registries";

public static final String EXTERNAL_ROOT_SYMLINK_NAME = "bazel-external";

private final Path vendorDirectory;

public VendorManager(Path vendorDirectory) {
Expand All @@ -56,38 +65,88 @@ public void vendorRepos(Path externalRepoRoot, ImmutableList<RepositoryName> rep
}

for (RepositoryName repo : reposToVendor) {
Path repoUnderExternal = externalRepoRoot.getChild(repo.getName());
Path repoUnderVendor = vendorDirectory.getChild(repo.getName());
// This could happen when running the vendor command twice without changing anything.
if (repoUnderExternal.isSymbolicLink()
&& repoUnderExternal.resolveSymbolicLinks().equals(repoUnderVendor)) {
continue;
}
try (SilentCloseable c =
Profiler.instance().profile(ProfilerTask.REPOSITORY_VENDOR, repo.toString())) {
Path repoUnderExternal = externalRepoRoot.getChild(repo.getName());
Path repoUnderVendor = vendorDirectory.getChild(repo.getName());
// This could happen when running the vendor command twice without changing anything.
if (repoUnderExternal.isSymbolicLink()
&& repoUnderExternal.resolveSymbolicLinks().equals(repoUnderVendor)) {
continue;
}

// At this point, the repo should exist under external dir, but check if the vendor src is
// already up-to-date.
Path markerUnderExternal = externalRepoRoot.getChild(repo.getMarkerFileName());
Path markerUnderVendor = vendorDirectory.getChild(repo.getMarkerFileName());
if (isRepoUpToDate(markerUnderVendor, markerUnderExternal)) {
continue;
}

// At this point, the repo should exist under external dir, but check if the vendor src is
// already up-to-date.
Path markerUnderExternal = externalRepoRoot.getChild(repo.getMarkerFileName());
Path markerUnderVendor = vendorDirectory.getChild(repo.getMarkerFileName());
if (isRepoUpToDate(markerUnderVendor, markerUnderExternal)) {
continue;
// Actually vendor the repo:
// 1. Clean up existing marker file and vendor dir.
markerUnderVendor.delete();
repoUnderVendor.deleteTree();
repoUnderVendor.createDirectory();
// 2. Move the marker file to a temporary one under vendor dir.
Path tMarker = vendorDirectory.getChild(repo.getMarkerFileName() + ".tmp");
FileSystemUtils.moveFile(markerUnderExternal, tMarker);
// 3. Move the external repo to vendor dir. It's fine if this step fails or is interrupted,
// because the marker file under external is gone anyway.
FileSystemUtils.moveTreesBelow(repoUnderExternal, repoUnderVendor);
// 4. Re-plant symlinks pointing a path under the external root to a relative path
// to make sure the vendor src keep working after being moved or output base changed
replantSymlinks(repoUnderVendor, externalRepoRoot);
// 5. Rename the temporary marker file after the move is done.
tMarker.renameTo(markerUnderVendor);
// 6. Leave a symlink in external dir to keep things working.
repoUnderExternal.deleteTree();
FileSystemUtils.ensureSymbolicLink(repoUnderExternal, repoUnderVendor);
}
}
}

// Actually vendor the repo:
// 1. Clean up existing marker file and vendor dir.
markerUnderVendor.delete();
repoUnderVendor.deleteTree();
repoUnderVendor.createDirectory();
// 2. Move the marker file to a temporary one under vendor dir.
Path tMarker = vendorDirectory.getChild(repo.getMarkerFileName() + ".tmp");
FileSystemUtils.moveFile(markerUnderExternal, tMarker);
// 3. Move the external repo to vendor dir. It's fine if this step fails or is interrupted,
// because the marker file under external is gone anyway.
FileSystemUtils.moveTreesBelow(repoUnderExternal, repoUnderVendor);
// 4. Rename to temporary marker file after the move is done.
tMarker.renameTo(markerUnderVendor);
// 5. Leave a symlink in external dir.
repoUnderExternal.deleteTree();
FileSystemUtils.ensureSymbolicLink(repoUnderExternal, repoUnderVendor);
/**
* Replants the symlinks under the specified repository directory.
*
* <p>Re-write symlinks that originally pointing to a path under the external root to a relative
* path pointing to an external root symlink under the vendor directory.
*
* @param repoUnderVendor The path to the repository directory under the vendor directory.
* @param externalRepoRoot The path to the root of external repositories.
* @throws IOException If an I/O error occurs while replanting the symlinks.
*/
private void replantSymlinks(Path repoUnderVendor, Path externalRepoRoot) throws IOException {
try {
Collection<Path> symlinks =
FileSystemUtils.traverseTree(repoUnderVendor, Path::isSymbolicLink);
Path externalSymlinkUnderVendor = vendorDirectory.getChild(EXTERNAL_ROOT_SYMLINK_NAME);
FileSystemUtils.ensureSymbolicLink(externalSymlinkUnderVendor, externalRepoRoot);
for (Path symlink : symlinks) {
PathFragment target = symlink.readSymbolicLink();
if (!target.startsWith(externalRepoRoot.asFragment())) {
// TODO: print a warning for absolute symlinks?
continue;
}
PathFragment newTarget =
PathFragment.create(
"../".repeat(symlink.relativeTo(vendorDirectory).segmentCount() - 1))
.getRelative(EXTERNAL_ROOT_SYMLINK_NAME)
.getRelative(target.relativeTo(externalRepoRoot.asFragment()));
if (OS.getCurrent() == OS.WINDOWS) {
// On Windows, FileSystemUtils.ensureSymbolicLink always resolves paths to absolute path.
// Use Files.createSymbolicLink here instead to preserve relative target path.
symlink.delete();
Files.createSymbolicLink(
java.nio.file.Path.of(symlink.getPathString()),
java.nio.file.Path.of(newTarget.getPathString()));
} else {
FileSystemUtils.ensureSymbolicLink(symlink, newTarget);
}
}
} catch (IOException e) {
throw new IOException(
String.format("Failed to rewrite symlinks under %s: ", repoUnderVendor), e);
}
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -99,6 +99,7 @@ public enum ProfilerTask {
CONFLICT_CHECK("Conflict checking"),
DYNAMIC_LOCK("Acquiring dynamic execution output lock", Threshold.FIFTY_MILLIS),
REPOSITORY_FETCH("Fetching repository"),
REPOSITORY_VENDOR("Vendoring repository"),
UNKNOWN("Unknown event");

private static class Threshold {
Expand Down
Loading

0 comments on commit 32fff52

Please sign in to comment.