|  | // 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. | 
|  |  | 
|  | #include "src/main/tools/linux-sandbox-options.h" | 
|  |  | 
|  | #include <errno.h> | 
|  | #include <sched.h> | 
|  | #include <stdarg.h> | 
|  | #include <stdbool.h> | 
|  | #include <stdio.h> | 
|  | #include <stdlib.h> | 
|  | #include <string.h> | 
|  | #include <sys/wait.h> | 
|  | #include <unistd.h> | 
|  | #include <fstream> | 
|  | #include <iostream> | 
|  | #include <memory> | 
|  | #include <string> | 
|  | #include <vector> | 
|  |  | 
|  | #include "src/main/tools/logging.h" | 
|  | #include "src/main/tools/process-tools.h" | 
|  |  | 
|  | using std::ifstream; | 
|  | using std::unique_ptr; | 
|  | using std::vector; | 
|  |  | 
|  | struct Options opt; | 
|  |  | 
|  | // Print out a usage error. argc and argv are the argument counter and vector, | 
|  | // fmt is a format, string for the error message to print. | 
|  | static void Usage(char *program_name, const char *fmt, ...) { | 
|  | va_list ap; | 
|  | va_start(ap, fmt); | 
|  | vfprintf(stderr, fmt, ap); | 
|  | va_end(ap); | 
|  |  | 
|  | fprintf(stderr, "\nUsage: %s -- command arg1 @args\n", program_name); | 
|  | fprintf(stderr, | 
|  | "\nPossible arguments:\n" | 
|  | "  -W <working-dir>  working directory (uses current directory if " | 
|  | "not specified)\n" | 
|  | "  -T <timeout>  timeout after which the child process will be " | 
|  | "terminated with SIGTERM\n" | 
|  | "  -t <timeout>  in case timeout occurs, how long to wait before " | 
|  | "killing the child with SIGKILL\n" | 
|  | "  -i  on receipt of a SIGINT, forward it to the child process as a " | 
|  | "SIGTERM first and then as a SIGKILL after the -T timeout\n" | 
|  | "  -l <file>  redirect stdout to a file\n" | 
|  | "  -L <file>  redirect stderr to a file\n" | 
|  | "  -w <file>  make a file or directory writable for the sandboxed " | 
|  | "process\n" | 
|  | "  -e <dir>  mount an empty tmpfs on a directory\n" | 
|  | "  -M/-m <source/target>  directory to mount inside the sandbox\n" | 
|  | "    Multiple directories can be specified and each of them will be " | 
|  | "mounted readonly.\n" | 
|  | "    The -M option specifies which directory to mount, the -m option " | 
|  | "specifies where to\n" | 
|  | "  -S <file>  if set, write stats in protobuf format to a file\n" | 
|  | "  -H  if set, make hostname in the sandbox equal to 'localhost'\n" | 
|  | "  -n  if set, create a new network namespace\n" | 
|  | "  -N  if set, create a new network namespace with loopback\n" | 
|  | "        Only one of -n and -N may be specified.\n" | 
|  | "  -R  if set, make the uid/gid be root\n" | 
|  | "  -U  if set, make the uid/gid be nobody\n" | 
|  | "  -P  if set, make the gid be tty and make /dev/pts writable\n" | 
|  | "  -D <debug-file> if set, debug info will be printed to this file\n" | 
|  | "  -p  if set, the process is persistent and ignores parent thread " | 
|  | "death signals\n" | 
|  | "  -C <dir> if set, put all subprocesses inside this cgroup.\n" | 
|  | "  -h <sandbox-dir>  if set, chroot to sandbox-dir and only " | 
|  | " mount whats been specified with -M/-m for improved hermeticity. " | 
|  | " The working-dir should be a folder inside the sandbox-dir\n" | 
|  | "  @FILE  read newline-separated arguments from FILE\n" | 
|  | "  --  command to run inside sandbox, followed by arguments\n"); | 
|  | exit(EXIT_FAILURE); | 
|  | } | 
|  |  | 
|  | static void ValidateIsAbsolutePath(char *path, char *program_name, char flag) { | 
|  | if (path[0] != '/') { | 
|  | Usage(program_name, "The -%c option must be used with absolute paths only.", | 
|  | flag); | 
|  | } | 
|  | } | 
|  |  | 
|  | // Parses command line flags from an argv array and puts the results into an | 
|  | // Options structure passed in as an argument. | 
|  | static void ParseCommandLine(unique_ptr<vector<char *>> args) { | 
|  | extern char *optarg; | 
|  | extern int optind, optopt; | 
|  | int c; | 
|  | bool source_specified = false; | 
|  | while ((c = getopt(args->size(), args->data(), | 
|  | ":W:T:t:il:L:w:e:M:m:S:h:pC:HnNRUPD:")) != -1) { | 
|  | if (c != 'M' && c != 'm') source_specified = false; | 
|  | switch (c) { | 
|  | case 'W': | 
|  | if (opt.working_dir.empty()) { | 
|  | ValidateIsAbsolutePath(optarg, args->front(), static_cast<char>(c)); | 
|  | opt.working_dir.assign(optarg); | 
|  | } else { | 
|  | Usage(args->front(), | 
|  | "Multiple working directories (-W) specified, expected one."); | 
|  | } | 
|  | break; | 
|  | case 'T': | 
|  | if (sscanf(optarg, "%d", &opt.timeout_secs) != 1 || | 
|  | opt.timeout_secs < 0) { | 
|  | Usage(args->front(), "Invalid timeout (-T) value: %s", optarg); | 
|  | } | 
|  | break; | 
|  | case 't': | 
|  | if (sscanf(optarg, "%d", &opt.kill_delay_secs) != 1 || | 
|  | opt.kill_delay_secs < 0) { | 
|  | Usage(args->front(), "Invalid kill delay (-t) value: %s", optarg); | 
|  | } | 
|  | break; | 
|  | case 'i': | 
|  | opt.sigint_sends_sigterm = true; | 
|  | break; | 
|  | case 'p': | 
|  | opt.persistent_process = true; | 
|  | break; | 
|  | case 'l': | 
|  | if (opt.stdout_path.empty()) { | 
|  | opt.stdout_path.assign(optarg); | 
|  | } else { | 
|  | Usage(args->front(), | 
|  | "Cannot redirect stdout to more than one destination."); | 
|  | } | 
|  | break; | 
|  | case 'L': | 
|  | if (opt.stderr_path.empty()) { | 
|  | opt.stderr_path.assign(optarg); | 
|  | } else { | 
|  | Usage(args->front(), | 
|  | "Cannot redirect stderr to more than one destination."); | 
|  | } | 
|  | break; | 
|  | case 'w': | 
|  | ValidateIsAbsolutePath(optarg, args->front(), static_cast<char>(c)); | 
|  | opt.writable_files.emplace_back(optarg); | 
|  | break; | 
|  | case 'e': | 
|  | ValidateIsAbsolutePath(optarg, args->front(), static_cast<char>(c)); | 
|  | opt.tmpfs_dirs.emplace_back(optarg); | 
|  | break; | 
|  | case 'M': | 
|  | ValidateIsAbsolutePath(optarg, args->front(), static_cast<char>(c)); | 
|  | // Add the current source path to both source and target lists | 
|  | opt.bind_mount_sources.emplace_back(optarg); | 
|  | opt.bind_mount_targets.emplace_back(optarg); | 
|  | source_specified = true; | 
|  | break; | 
|  | case 'm': | 
|  | ValidateIsAbsolutePath(optarg, args->front(), static_cast<char>(c)); | 
|  | if (!source_specified) { | 
|  | Usage(args->front(), | 
|  | "The -m option must be strictly preceded by an -M option."); | 
|  | } | 
|  | opt.bind_mount_targets.pop_back(); | 
|  | opt.bind_mount_targets.emplace_back(optarg); | 
|  | source_specified = false; | 
|  | break; | 
|  | case 'S': | 
|  | if (opt.stats_path.empty()) { | 
|  | opt.stats_path.assign(optarg); | 
|  | } else { | 
|  | Usage(args->front(), | 
|  | "Cannot write stats to more than one destination."); | 
|  | } | 
|  | break; | 
|  | case 'h': | 
|  | opt.hermetic = true; | 
|  | if (opt.sandbox_root.empty()) { | 
|  | std::string sandbox_root(optarg); | 
|  | // Make sure that the sandbox_root path has no trailing slash. | 
|  | if (sandbox_root.back() == '/') { | 
|  | ValidateIsAbsolutePath(optarg, args->front(), static_cast<char>(c)); | 
|  | opt.sandbox_root.assign(sandbox_root, 0, sandbox_root.length() - 1); | 
|  | if (opt.sandbox_root.back() == '/') { | 
|  | Usage(args->front(), | 
|  | "Sandbox root path should not have trailing slashes"); | 
|  | } | 
|  | } else { | 
|  | opt.sandbox_root.assign(sandbox_root); | 
|  | } | 
|  | } else { | 
|  | Usage(args->front(), | 
|  | "Multiple sandbox roots (-s) specified, expected one."); | 
|  | } | 
|  | break; | 
|  | case 'H': | 
|  | opt.fake_hostname = true; | 
|  | break; | 
|  | case 'n': | 
|  | if (opt.create_netns == NETNS_WITH_LOOPBACK) { | 
|  | Usage(args->front(), "Only one of -n and -N may be specified."); | 
|  | } | 
|  | opt.create_netns = NETNS; | 
|  | break; | 
|  | case 'N': | 
|  | if (opt.create_netns == NETNS) { | 
|  | Usage(args->front(), "Only one of -n and -N may be specified."); | 
|  | } | 
|  | opt.create_netns = NETNS_WITH_LOOPBACK; | 
|  | break; | 
|  | case 'R': | 
|  | if (opt.fake_username) { | 
|  | Usage(args->front(), | 
|  | "The -R option cannot be used at the same time us the -U " | 
|  | "option."); | 
|  | } | 
|  | opt.fake_root = true; | 
|  | break; | 
|  | case 'U': | 
|  | if (opt.fake_root) { | 
|  | Usage(args->front(), | 
|  | "The -U option cannot be used at the same time us the -R " | 
|  | "option."); | 
|  | } | 
|  | opt.fake_username = true; | 
|  | break; | 
|  | case 'C': | 
|  | ValidateIsAbsolutePath(optarg, args->front(), static_cast<char>(c)); | 
|  | opt.cgroups_dirs.emplace_back(optarg); | 
|  | break; | 
|  | case 'P': | 
|  | opt.enable_pty = true; | 
|  | break; | 
|  | case 'D': | 
|  | if (opt.debug_path.empty()) { | 
|  | ValidateIsAbsolutePath(optarg, args->front(), static_cast<char>(c)); | 
|  | opt.debug_path.assign(optarg); | 
|  | } else { | 
|  | Usage(args->front(), | 
|  | "Cannot write debug output to more than one file."); | 
|  | } | 
|  | break; | 
|  | case '?': | 
|  | Usage(args->front(), "Unrecognized argument: -%c (%d)", optopt, optind); | 
|  | break; | 
|  | case ':': | 
|  | Usage(args->front(), "Flag -%c requires an argument", optopt); | 
|  | break; | 
|  | } | 
|  | } | 
|  |  | 
|  | if (!opt.working_dir.empty() && !opt.sandbox_root.empty() && | 
|  | opt.working_dir.find(opt.sandbox_root) == std::string::npos) { | 
|  | Usage(args->front(), | 
|  | "working-dir %s (-W) should be a " | 
|  | "subdirectory of sandbox-dir %s (-h)", | 
|  | opt.working_dir.c_str(), opt.sandbox_root.c_str()); | 
|  | } | 
|  | if (optind < static_cast<int>(args->size())) { | 
|  | if (opt.args.empty()) { | 
|  | opt.args.assign(args->begin() + optind, args->end()); | 
|  | } else { | 
|  | Usage(args->front(), "Merging commands not supported."); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | // Expands a single argument, expanding options @filename to read in the content | 
|  | // of the file and add it to the list of processed arguments. | 
|  | static unique_ptr<vector<char *>> ExpandArgument( | 
|  | unique_ptr<vector<char *>> expanded, char *arg) { | 
|  | if (arg[0] == '@') { | 
|  | const char *filename = arg + 1;  // strip off the '@'. | 
|  | ifstream f(filename); | 
|  |  | 
|  | if (!f.is_open()) { | 
|  | DIE("opening argument file %s failed", filename); | 
|  | } | 
|  |  | 
|  | for (std::string line; std::getline(f, line);) { | 
|  | if (!line.empty()) { | 
|  | expanded = ExpandArgument(std::move(expanded), strdup(line.c_str())); | 
|  | } | 
|  | } | 
|  |  | 
|  | if (f.bad()) { | 
|  | DIE("error while reading from argument file %s", filename); | 
|  | } | 
|  | } else { | 
|  | expanded->push_back(arg); | 
|  | } | 
|  |  | 
|  | return expanded; | 
|  | } | 
|  |  | 
|  | // Pre-processes an argument list, expanding options @filename to read in the | 
|  | // content of the file and add it to the list of arguments. Stops expanding | 
|  | // arguments once it encounters "--". | 
|  | static unique_ptr<vector<char *>> ExpandArguments(const vector<char *> &args) { | 
|  | unique_ptr<vector<char *>> expanded(new vector<char *>()); | 
|  | expanded->reserve(args.size()); | 
|  | for (auto arg = args.begin(); arg != args.end(); ++arg) { | 
|  | if (strcmp(*arg, "--") != 0) { | 
|  | expanded = ExpandArgument(std::move(expanded), *arg); | 
|  | } else { | 
|  | expanded->insert(expanded->end(), arg, args.end()); | 
|  | break; | 
|  | } | 
|  | } | 
|  | return expanded; | 
|  | } | 
|  |  | 
|  | void ParseOptions(int argc, char *argv[]) { | 
|  | vector<char *> args(argv, argv + argc); | 
|  | ParseCommandLine(ExpandArguments(args)); | 
|  |  | 
|  | if (opt.args.empty()) { | 
|  | Usage(args.front(), "No command specified."); | 
|  | } | 
|  |  | 
|  | if (opt.working_dir.empty()) { | 
|  | char *working_dir = getcwd(nullptr, 0); | 
|  | if (working_dir == nullptr) { | 
|  | DIE("getcwd"); | 
|  | } | 
|  | opt.working_dir = working_dir; | 
|  | free(working_dir); | 
|  | } | 
|  | } |