blob: 46cc169defa9c598c066f740c2f5cb7805c21bab [file] [log] [blame]
// Copyright 2016 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.sandbox;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableMap;
import com.google.common.flogger.GoogleLogger;
import com.google.devtools.build.lib.exec.TreeDeleter;
import com.google.devtools.build.lib.sandbox.SandboxHelpers.SandboxInputs;
import com.google.devtools.build.lib.sandbox.SandboxHelpers.SandboxOutputs;
import com.google.devtools.build.lib.util.CommandDescriptionForm;
import com.google.devtools.build.lib.util.CommandFailureUtils;
import com.google.devtools.build.lib.vfs.FileStatus;
import com.google.devtools.build.lib.vfs.FileSystemUtils;
import com.google.devtools.build.lib.vfs.Path;
import com.google.devtools.build.lib.vfs.Symlinks;
import java.io.IOException;
import java.util.Collection;
import java.util.Optional;
import java.util.Set;
import javax.annotation.Nullable;
/**
* Creates an execRoot for a Spawn that contains input files as hardlinks to their original
* destination.
*/
public class HardlinkedSandboxedSpawn extends AbstractContainerizingSandboxedSpawn {
private static final GoogleLogger logger = GoogleLogger.forEnclosingClass();
private boolean sandboxDebug = false;
@Nullable private final ImmutableList<String> interactiveDebugArguments;
public HardlinkedSandboxedSpawn(
Path sandboxPath,
Path sandboxExecRoot,
ImmutableList<String> arguments,
ImmutableMap<String, String> environment,
SandboxInputs inputs,
SandboxOutputs outputs,
Set<Path> writableDirs,
TreeDeleter treeDeleter,
@Nullable Path sandboxDebugPath,
@Nullable Path statisticsPath,
boolean sandboxDebug,
@Nullable ImmutableList<String> interactiveDebugArguments,
String mnemonic) {
super(
sandboxPath,
sandboxExecRoot,
arguments,
environment,
inputs,
outputs,
writableDirs,
treeDeleter,
sandboxDebugPath,
statisticsPath,
mnemonic);
this.sandboxDebug = sandboxDebug;
this.interactiveDebugArguments = interactiveDebugArguments;
}
@Override
protected void copyFile(Path source, Path target) throws IOException {
hardLinkRecursive(source, target);
}
/**
* Recursively creates hardlinks for all files in {@code source} path, in {@code target} path.
* Symlinks are resolved. If files is located on another disk, hardlink will fail and a copy will
* be made instead. Throws IllegalArgumentException if source path is a subdirectory of target
* path.
*/
private void hardLinkRecursive(Path source, Path target) throws IOException {
FileStatus stat = source.stat(Symlinks.NOFOLLOW);
if (stat.isSymbolicLink()) {
source = source.resolveSymbolicLinks();
stat = source.stat();
}
if (stat.isFile()) {
try {
source.createHardLink(target);
} catch (IOException e) {
if (sandboxDebug) {
logger.atInfo().log(
"File %s could not be hardlinked, file will be copied instead.", source);
}
FileSystemUtils.copyFile(source, target);
}
} else if (stat.isDirectory()) {
if (source.startsWith(target)) {
throw new IllegalArgumentException(source + " is a subdirectory of " + target);
}
target.createDirectory();
Collection<Path> entries = source.getDirectoryEntries();
for (Path entry : entries) {
Path toPath = target.getChild(entry.getBaseName());
hardLinkRecursive(entry, toPath);
}
}
}
@Override
public Optional<String> getInteractiveDebugInstructions() {
if (interactiveDebugArguments == null) {
return Optional.empty();
}
return Optional.of(
"Run this command to start an interactive shell in an identical sandboxed environment:\n"
+ CommandFailureUtils.describeCommand(
CommandDescriptionForm.COMPLETE,
/* prettyPrintArgs= */ false,
interactiveDebugArguments,
getEnvironment(),
/* environmentVariablesToClear= */ null,
/* cwd= */ sandboxExecRoot.getPathString(),
/* configurationChecksum= */ null,
/* executionPlatformLabel= */ null,
/* spawnRunner= */ null));
}
}