blob: a833dafb8e277493db570540286a0620c862ebd5 [file] [log] [blame]
// Copyright 2015 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.ImmutableMap;
import com.google.common.collect.ImmutableSet;
import com.google.common.collect.Lists;
import com.google.common.collect.Maps;
import com.google.devtools.build.lib.vfs.FileSystem;
import com.google.devtools.build.lib.vfs.Path;
import com.google.devtools.common.options.Converter;
import com.google.devtools.common.options.Option;
import com.google.devtools.common.options.OptionDocumentationCategory;
import com.google.devtools.common.options.OptionEffectTag;
import com.google.devtools.common.options.OptionsBase;
import com.google.devtools.common.options.OptionsParsingException;
import java.io.IOException;
import java.util.ArrayList;
import java.util.List;
/** Options for sandboxed execution. */
public class SandboxOptions extends OptionsBase {
/**
* A converter for customized path mounting pair from the parameter list of a bazel command
* invocation. Pairs are expected to have the form 'source:target'.
*/
public static final class MountPairConverter
implements Converter<ImmutableMap.Entry<String, String>> {
@Override
public ImmutableMap.Entry<String, String> convert(String input) throws OptionsParsingException {
List<String> paths = Lists.newArrayList();
for (String path : input.split("(?<!\\\\):")) { // Split on ':' but not on '\:'
if (path != null && !path.trim().isEmpty()) {
paths.add(path.replace("\\:", ":"));
} else {
throw new OptionsParsingException(
"Input "
+ input
+ " contains one or more empty paths. "
+ "Input must be a single path to mount inside the sandbox or "
+ "a mounting pair in the form of 'source:target'");
}
}
if (paths.size() < 1 || paths.size() > 2) {
throw new OptionsParsingException(
"Input must be a single path to mount inside the sandbox or "
+ "a mounting pair in the form of 'source:target'");
}
return paths.size() == 1
? Maps.immutableEntry(paths.get(0), paths.get(0))
: Maps.immutableEntry(paths.get(0), paths.get(1));
}
@Override
public String getTypeDescription() {
return "a single path or a 'source:target' pair";
}
}
@Option(
name = "ignore_unsupported_sandboxing",
defaultValue = "false",
category = "strategy",
documentationCategory = OptionDocumentationCategory.UNCATEGORIZED,
effectTags = {OptionEffectTag.UNKNOWN},
help = "Do not print a warning when sandboxed execution is not supported on this system."
)
public boolean ignoreUnsupportedSandboxing;
@Option(
name = "sandbox_debug",
defaultValue = "false",
category = "strategy",
documentationCategory = OptionDocumentationCategory.UNCATEGORIZED,
effectTags = {OptionEffectTag.UNKNOWN},
help =
"Let the sandbox print debug information on execution. This might help developers of "
+ "Bazel or Skylark rules with debugging failures due to missing input files, etc."
)
public boolean sandboxDebug;
@Option(
name = "experimental_sandbox_base",
defaultValue = "",
category = "strategy",
documentationCategory = OptionDocumentationCategory.UNCATEGORIZED,
effectTags = {OptionEffectTag.UNKNOWN},
help =
"Lets the sandbox create its sandbox directories underneath this path. Specify a path "
+ "on tmpfs (like /run/shm) to possibly improve performance a lot when your build / "
+ "tests have many input files. Note: You need enough RAM and free space on the tmpfs "
+ "to hold output and intermediate files generated by running actions."
)
public String sandboxBase;
@Option(
name = "sandbox_fake_hostname",
defaultValue = "false",
category = "strategy",
documentationCategory = OptionDocumentationCategory.UNCATEGORIZED,
effectTags = {OptionEffectTag.UNKNOWN},
help = "Change the current hostname to 'localhost' for sandboxed actions."
)
public boolean sandboxFakeHostname;
@Option(
name = "sandbox_fake_username",
defaultValue = "false",
category = "strategy",
documentationCategory = OptionDocumentationCategory.UNCATEGORIZED,
effectTags = {OptionEffectTag.UNKNOWN},
help = "Change the current username to 'nobody' for sandboxed actions."
)
public boolean sandboxFakeUsername;
@Option(
name = "sandbox_block_path",
allowMultiple = true,
defaultValue = "",
category = "config",
documentationCategory = OptionDocumentationCategory.UNCATEGORIZED,
effectTags = {OptionEffectTag.UNKNOWN},
help = "For sandboxed actions, disallow access to this path."
)
public List<String> sandboxBlockPath;
@Option(
name = "sandbox_tmpfs_path",
allowMultiple = true,
defaultValue = "",
category = "config",
documentationCategory = OptionDocumentationCategory.UNCATEGORIZED,
effectTags = {OptionEffectTag.UNKNOWN},
help =
"For sandboxed actions, mount an empty, writable directory at this path"
+ " (if supported by the sandboxing implementation, ignored otherwise)."
)
public List<String> sandboxTmpfsPath;
@Option(
name = "sandbox_writable_path",
allowMultiple = true,
defaultValue = "",
category = "config",
documentationCategory = OptionDocumentationCategory.UNCATEGORIZED,
effectTags = {OptionEffectTag.UNKNOWN},
help =
"For sandboxed actions, make an existing directory writable in the sandbox"
+ " (if supported by the sandboxing implementation, ignored otherwise)."
)
public List<String> sandboxWritablePath;
@Option(
name = "sandbox_add_mount_pair",
allowMultiple = true,
converter = MountPairConverter.class,
defaultValue = "",
category = "config",
documentationCategory = OptionDocumentationCategory.UNCATEGORIZED,
effectTags = {OptionEffectTag.UNKNOWN},
help = "Add additional path pair to mount in sandbox."
)
public List<ImmutableMap.Entry<String, String>> sandboxAdditionalMounts;
public ImmutableSet<Path> getInaccessiblePaths(FileSystem fs) {
List<Path> inaccessiblePaths = new ArrayList<>();
for (String path : sandboxBlockPath) {
Path blockedPath = fs.getPath(path);
try {
inaccessiblePaths.add(blockedPath.resolveSymbolicLinks());
} catch (IOException e) {
// It's OK to block access to an invalid symlink. In this case we'll just make the symlink
// itself inaccessible, instead of the target, though.
inaccessiblePaths.add(blockedPath);
}
}
return ImmutableSet.copyOf(inaccessiblePaths);
}
}