| // Copyright 2014 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/cpp/blaze_util.h" | 
 |  | 
 | #include <fcntl.h> | 
 | #include <stdarg.h> | 
 | #include <stdio.h> | 
 | #include <stdlib.h> | 
 | #include <string.h> | 
 |  | 
 | #include <algorithm> | 
 | #include <cassert> | 
 | #include <iostream> | 
 |  | 
 | #include "src/main/cpp/blaze_util_platform.h" | 
 | #include "src/main/cpp/util/errors.h" | 
 | #include "src/main/cpp/util/exit_code.h" | 
 | #include "src/main/cpp/util/file.h" | 
 | #include "src/main/cpp/util/logging.h" | 
 | #include "src/main/cpp/util/numbers.h" | 
 | #include "src/main/cpp/util/path_platform.h" | 
 | #include "src/main/cpp/util/port.h" | 
 | #include "src/main/cpp/util/strings.h" | 
 |  | 
 | #include "absl/base/log_severity.h" | 
 | #include "absl/log/globals.h" | 
 | #include "absl/log/initialize.h" | 
 |  | 
 | namespace blaze { | 
 |  | 
 | using std::map; | 
 | using std::string; | 
 | using std::vector; | 
 |  | 
 | const char kServerPidFile[] = "server.pid.txt"; | 
 |  | 
 | const unsigned int kPostShutdownGracePeriodSeconds = 60; | 
 |  | 
 | const unsigned int kPostKillGracePeriodSeconds = 10; | 
 |  | 
 | const char* GetUnaryOption(const char *arg, | 
 |                            const char *next_arg, | 
 |                            const char *key) { | 
 |   const char *value = blaze_util::var_strprefix(arg, key); | 
 |   if (value == nullptr) { | 
 |     return nullptr; | 
 |   } else if (value[0] == '=') { | 
 |     return value + 1; | 
 |   } else if (value[0]) { | 
 |     return nullptr;  // trailing garbage in key name | 
 |   } | 
 |  | 
 |   return next_arg; | 
 | } | 
 |  | 
 | bool GetNullaryOption(const char *arg, const char *key) { | 
 |   const char *value = blaze_util::var_strprefix(arg, key); | 
 |   if (value == nullptr) { | 
 |     return false; | 
 |   } else if (value[0] == '=') { | 
 |     BAZEL_DIE(blaze_exit_code::BAD_ARGV) | 
 |         << "In argument '" << arg << "': option '" << key | 
 |         << "' does not take a value."; | 
 |   } else if (value[0]) { | 
 |     return false;  // trailing garbage in key name | 
 |   } | 
 |  | 
 |   return true; | 
 | } | 
 |  | 
 | std::vector<std::string> GetAllUnaryOptionValues( | 
 |     const vector<string>& args, const char* key, | 
 |     const char* ignore_after_value) { | 
 |   vector<std::string> values; | 
 |   for (vector<string>::size_type i = 0; i < args.size(); ++i) { | 
 |     if (args[i] == "--") { | 
 |       // "--" means all remaining args aren't options | 
 |       return values; | 
 |     } | 
 |  | 
 |     const char* next_arg = args[std::min(i + 1, args.size() - 1)].c_str(); | 
 |     const char* result = GetUnaryOption(args[i].c_str(), next_arg, key); | 
 |     if (result != nullptr) { | 
 |       // 'key' was found and 'result' has its value. | 
 |       values.push_back(result); | 
 |  | 
 |       if (ignore_after_value != nullptr && | 
 |           strcmp(result, ignore_after_value) == 0) { | 
 |         break; | 
 |       } | 
 |     } | 
 |  | 
 |     // This is a pointer comparison, so equality means that the result must be | 
 |     // from the next arg instead of happening to match the value from | 
 |     // "--key=<value>" string, in which case we need to advance the index to | 
 |     // skip the next arg for later iterations. | 
 |     if (result == next_arg) { | 
 |       i++; | 
 |     } | 
 |   } | 
 |  | 
 |   return values; | 
 | } | 
 |  | 
 | bool SearchNullaryOption(const vector<string>& args, | 
 |                          const string& flag_name, | 
 |                          const bool default_value) { | 
 |   const string positive_flag = "--" + flag_name; | 
 |   const string negative_flag = "--no" + flag_name; | 
 |   bool result = default_value; | 
 |   for (vector<string>::size_type i = 0; i < args.size(); i++) { | 
 |     if (args[i] == "--") { | 
 |       break; | 
 |     } | 
 |     if (GetNullaryOption(args[i].c_str(), positive_flag.c_str())) { | 
 |       result = true; | 
 |     } else if (GetNullaryOption(args[i].c_str(), negative_flag.c_str())) { | 
 |       result = false; | 
 |     } | 
 |   } | 
 |   return result; | 
 | } | 
 |  | 
 | bool IsArg(const string& arg) { | 
 |   return blaze_util::starts_with(arg, "-") && (arg != "--help") | 
 |       && (arg != "-help") && (arg != "-h"); | 
 | } | 
 |  | 
 | std::string AbsolutePathFromFlag(const std::string& value) { | 
 |   if (value.empty()) { | 
 |     return blaze_util::GetCwd(); | 
 |   } else if (!value.empty() && value[0] == '~') { | 
 |     return blaze_util::JoinPath(GetHomeDir(), value.substr(1)); | 
 |   } else { | 
 |     return blaze_util::MakeAbsolute(value); | 
 |   } | 
 | } | 
 |  | 
 | void LogWait(unsigned int elapsed_seconds, unsigned int wait_seconds) { | 
 |   SigPrintf("WARNING: Waiting for server process to terminate " | 
 |             "(waited %d seconds, waiting at most %d)\n", | 
 |             elapsed_seconds, wait_seconds); | 
 | } | 
 |  | 
 | bool AwaitServerProcessTermination(int pid, const blaze_util::Path& output_base, | 
 |                                    unsigned int wait_seconds) { | 
 |   uint64_t st = GetMillisecondsMonotonic(); | 
 |   const unsigned int first_seconds = 5; | 
 |   bool logged_first = false; | 
 |   const unsigned int second_seconds = 10; | 
 |   bool logged_second = false; | 
 |   const unsigned int third_seconds = 30; | 
 |   bool logged_third = false; | 
 |  | 
 |   while (VerifyServerProcess(pid, output_base)) { | 
 |     TrySleep(100); | 
 |     uint64_t elapsed_millis = GetMillisecondsMonotonic() - st; | 
 |     if (!logged_first && elapsed_millis > first_seconds * 1000) { | 
 |       LogWait(first_seconds, wait_seconds); | 
 |       logged_first = true; | 
 |     } | 
 |     if (!logged_second && elapsed_millis > second_seconds * 1000) { | 
 |       LogWait(second_seconds, wait_seconds); | 
 |       logged_second = true; | 
 |     } | 
 |     if (!logged_third && elapsed_millis > third_seconds * 1000) { | 
 |       LogWait(third_seconds, wait_seconds); | 
 |       logged_third = true; | 
 |     } | 
 |     if (elapsed_millis > wait_seconds * 1000) { | 
 |       SigPrintf("INFO: Waited %d seconds for server process (pid=%d) to" | 
 |                 " terminate.\n", | 
 |                 wait_seconds, pid); | 
 |       return false; | 
 |     } | 
 |   } | 
 |   return true; | 
 | } | 
 |  | 
 | void SetDebugLog(blaze_util::LoggingDetail detail) { | 
 |   if (detail == blaze_util::LOGGINGDETAIL_DEBUG) { | 
 |     blaze_util::SetLoggingDetail(blaze_util::LOGGINGDETAIL_DEBUG, &std::cerr); | 
 |     absl::SetStderrThreshold(absl::LogSeverityAtLeast::kInfo); | 
 |   } else { | 
 |     blaze_util::SetLoggingDetail(detail, nullptr); | 
 |  | 
 |     // Disable absl debug logging, since that gets printed to stderr due to us | 
 |     // not setting up a log file. We don't use absl but one of our dependencies | 
 |     // might (as of 2024Q2, gRPC does). | 
 |     // | 
 |     // Future improvements to this approach: | 
 |     // * Disable absl logging ASAP, not just here after handling | 
 |     //   --client_debug=false. | 
 |     // * Use the same approach for handling --client_debug=true that we do for | 
 |     //   BAZEL_LOG of first redirecting all messages to an inmemory string, and | 
 |     //   then writing that string to stderr. We could use a absl::LogSink to | 
 |     //   achieve this. | 
 |     absl::InitializeLog(); | 
 |     absl::SetStderrThreshold(absl::LogSeverityAtLeast::kInfinity); | 
 |   } | 
 | } | 
 |  | 
 | bool IsRunningWithinTest() { return ExistsEnv("TEST_TMPDIR"); } | 
 |  | 
 | void WithEnvVars::SetEnvVars(const map<string, EnvVarValue>& vars) { | 
 |   for (const auto& var : vars) { | 
 |     switch (var.second.action) { | 
 |       case EnvVarAction::UNSET: | 
 |         UnsetEnv(var.first); | 
 |         break; | 
 |  | 
 |       case EnvVarAction::SET: | 
 |         SetEnv(var.first, var.second.value); | 
 |         break; | 
 |  | 
 |       default: | 
 |         assert(false); | 
 |     } | 
 |   } | 
 | } | 
 |  | 
 | WithEnvVars::WithEnvVars(const map<string, EnvVarValue>& vars) { | 
 |   for (const auto& v : vars) { | 
 |     if (ExistsEnv(v.first)) { | 
 |       _old_values[v.first] = EnvVarValue(EnvVarAction::SET, GetEnv(v.first)); | 
 |     } else { | 
 |       _old_values[v.first] = EnvVarValue(EnvVarAction::UNSET, ""); | 
 |     } | 
 |   } | 
 |  | 
 |   SetEnvVars(vars); | 
 | } | 
 |  | 
 | WithEnvVars::~WithEnvVars() { | 
 |   SetEnvVars(_old_values); | 
 | } | 
 |  | 
 | }  // namespace blaze |