blob: 3b5d5dd3220ba6e9077d5bbd6fb5e88e1ccad8bb [file] [log] [blame]
// 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 "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/numbers.h"
#include "src/main/cpp/util/strings.h"
#include "src/main/cpp/util/port.h"
using blaze_util::die;
namespace blaze {
using std::string;
using std::vector;
const char kServerPidFile[] = "server.pid.txt";
const unsigned int kPostShutdownGracePeriodSeconds = 60;
const unsigned int kPostKillGracePeriodSeconds = 10;
string MakeAbsolute(const string &p) {
string path = ConvertPath(p);
if (path.empty()) {
return blaze_util::GetCwd();
}
if (blaze_util::IsDevNull(path.c_str()) || blaze_util::IsAbsolute(path)) {
return path;
}
return blaze_util::JoinPath(blaze_util::GetCwd(), path);
}
const char* GetUnaryOption(const char *arg,
const char *next_arg,
const char *key) {
const char *value = blaze_util::var_strprefix(arg, key);
if (value == NULL) {
return NULL;
} else if (value[0] == '=') {
return value + 1;
} else if (value[0]) {
return NULL; // 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 == NULL) {
return false;
} else if (value[0] == '=') {
die(blaze_exit_code::BAD_ARGV,
"In argument '%s': option '%s' does not take a value.", arg, key);
} else if (value[0]) {
return false; // trailing garbage in key name
}
return true;
}
const char* SearchUnaryOption(const vector<string>& args,
const char *key) {
if (args.empty()) {
return NULL;
}
vector<string>::size_type i = 0;
for (; i < args.size() - 1; ++i) {
if (args[i] == "--") {
return NULL;
}
const char* result = GetUnaryOption(args[i].c_str(),
args[i + 1].c_str(),
key);
if (result != NULL) {
return result;
}
}
return GetUnaryOption(args[i].c_str(), NULL, key);
}
static bool SearchNullaryOption(const vector<string>& args,
const char *key,
const bool include_positional_params) {
for (vector<string>::size_type i = 0; i < args.size(); i++) {
if (args[i] == "--") {
if (!include_positional_params) {
return false;
}
continue;
}
if (GetNullaryOption(args[i].c_str(), key)) {
return true;
}
}
return false;
}
bool SearchNullaryOption(const vector<string>& args, const char *key) {
return SearchNullaryOption(args, key, false);
}
bool VerboseLogging() { return !GetEnv("VERBOSE_BLAZE_CLIENT").empty(); }
// Read the Jvm version from a file descriptor. The read fd
// should contains a similar output as the java -version output.
string ReadJvmVersion(const string& version_string) {
// try to look out for 'version "'
static const string version_pattern = "version \"";
size_t found = version_string.find(version_pattern);
if (found != string::npos) {
found += version_pattern.size();
// If we found "version \"", process until next '"'
size_t end = version_string.find("\"", found);
if (end == string::npos) { // consider end of string as a '"'
end = version_string.size();
}
return version_string.substr(found, end - found);
}
return "";
}
bool CheckJavaVersionIsAtLeast(const string &jvm_version,
const string &version_spec) {
vector<string> jvm_version_vect = blaze_util::Split(jvm_version, '.');
int jvm_version_size = static_cast<int>(jvm_version_vect.size());
vector<string> version_spec_vect = blaze_util::Split(version_spec, '.');
int version_spec_size = static_cast<int>(version_spec_vect.size());
int i;
for (i = 0; i < jvm_version_size && i < version_spec_size; i++) {
int jvm = blaze_util::strto32(jvm_version_vect[i].c_str(), NULL, 10);
int spec = blaze_util::strto32(version_spec_vect[i].c_str(), NULL, 10);
if (jvm > spec) {
return true;
} else if (jvm < spec) {
return false;
}
}
if (i < version_spec_size) {
for (; i < version_spec_size; i++) {
if (version_spec_vect[i] != "0") {
return false;
}
}
}
return true;
}
bool IsArg(const string& arg) {
return blaze_util::starts_with(arg, "-") && (arg != "--help")
&& (arg != "-help") && (arg != "-h");
}
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 string& 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;
}
static bool is_debug_log_enabled = false;
void SetDebugLog(bool enabled) { is_debug_log_enabled = enabled; }
void debug_log(const char *format, ...) {
if (!is_debug_log_enabled) {
return;
}
fprintf(stderr, "CLIENT: ");
va_list arglist;
va_start(arglist, format);
vfprintf(stderr, format, arglist);
va_end(arglist);
fprintf(stderr, "%s", "\n");
fflush(stderr);
}
} // namespace blaze