2015-03-19 18:41:01 -04:00
|
|
|
|
// Copyright 2015 The Crashpad 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.
|
|
|
|
|
|
2016-11-15 14:12:23 -05:00
|
|
|
|
#include <errno.h>
|
2015-03-19 18:41:01 -04:00
|
|
|
|
#include <getopt.h>
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
#include <string.h>
|
2015-08-05 16:34:50 -04:00
|
|
|
|
#include <strings.h>
|
2015-03-19 18:41:01 -04:00
|
|
|
|
#include <sys/types.h>
|
|
|
|
|
#include <time.h>
|
|
|
|
|
|
2016-04-25 12:13:07 -07:00
|
|
|
|
#include <memory>
|
2015-03-19 18:41:01 -04:00
|
|
|
|
#include <string>
|
2015-12-09 17:36:32 -05:00
|
|
|
|
#include <utility>
|
2015-03-19 18:41:01 -04:00
|
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
|
|
#include "base/files/file_path.h"
|
2015-05-06 10:28:07 -07:00
|
|
|
|
#include "base/logging.h"
|
2016-01-06 12:22:50 -05:00
|
|
|
|
#include "base/macros.h"
|
2015-03-19 18:41:01 -04:00
|
|
|
|
#include "base/numerics/safe_conversions.h"
|
2015-05-06 10:28:07 -07:00
|
|
|
|
#include "base/strings/utf_string_conversions.h"
|
|
|
|
|
#include "build/build_config.h"
|
2015-03-19 18:41:01 -04:00
|
|
|
|
#include "client/crash_report_database.h"
|
|
|
|
|
#include "client/settings.h"
|
|
|
|
|
#include "tools/tool_support.h"
|
2015-04-03 18:57:01 -04:00
|
|
|
|
#include "util/file/file_io.h"
|
|
|
|
|
#include "util/file/file_reader.h"
|
2015-03-19 18:41:01 -04:00
|
|
|
|
#include "util/misc/uuid.h"
|
2016-11-15 13:50:32 -05:00
|
|
|
|
#include "util/stdlib/string_number_conversion.h"
|
2015-03-19 18:41:01 -04:00
|
|
|
|
|
|
|
|
|
namespace crashpad {
|
|
|
|
|
namespace {
|
|
|
|
|
|
2015-05-06 10:28:07 -07:00
|
|
|
|
void Usage(const base::FilePath& me) {
|
2015-03-19 18:41:01 -04:00
|
|
|
|
fprintf(stderr,
|
2015-05-06 10:28:07 -07:00
|
|
|
|
"Usage: %" PRFilePath " [OPTION]... PID\n"
|
2015-03-19 18:41:01 -04:00
|
|
|
|
"Operate on Crashpad crash report databases.\n"
|
|
|
|
|
"\n"
|
2015-10-08 13:10:02 -04:00
|
|
|
|
" --create allow database at PATH to be created\n"
|
2015-03-19 18:41:01 -04:00
|
|
|
|
" -d, --database=PATH operate on the crash report database at PATH\n"
|
|
|
|
|
" --show-client-id show the client ID\n"
|
|
|
|
|
" --show-uploads-enabled show whether uploads are enabled\n"
|
|
|
|
|
" --show-last-upload-attempt-time\n"
|
|
|
|
|
" show the last-upload-attempt time\n"
|
|
|
|
|
" --show-pending-reports show reports eligible for upload\n"
|
|
|
|
|
" --show-completed-reports show reports not eligible for upload\n"
|
|
|
|
|
" --show-all-report-info with --show-*-reports, show more information\n"
|
|
|
|
|
" --show-report=UUID show report stored under UUID\n"
|
|
|
|
|
" --set-uploads-enabled=BOOL enable or disable uploads\n"
|
|
|
|
|
" --set-last-upload-attempt-time=TIME\n"
|
|
|
|
|
" set the last-upload-attempt time to TIME\n"
|
2015-08-07 13:57:05 -04:00
|
|
|
|
" --new-report=PATH submit a new report at PATH, or - for stdin\n"
|
2015-03-19 18:41:01 -04:00
|
|
|
|
" --utc show and set UTC times instead of local\n"
|
|
|
|
|
" --help display this help and exit\n"
|
|
|
|
|
" --version output version information and exit\n",
|
2015-05-06 10:28:07 -07:00
|
|
|
|
me.value().c_str());
|
2015-03-19 18:41:01 -04:00
|
|
|
|
ToolSupport::UsageTail(me);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
struct Options {
|
|
|
|
|
std::vector<UUID> show_reports;
|
2015-04-03 18:57:01 -04:00
|
|
|
|
std::vector<base::FilePath> new_report_paths;
|
2015-03-19 18:41:01 -04:00
|
|
|
|
const char* database;
|
|
|
|
|
const char* set_last_upload_attempt_time_string;
|
|
|
|
|
time_t set_last_upload_attempt_time;
|
2015-10-08 13:10:02 -04:00
|
|
|
|
bool create;
|
2015-03-19 18:41:01 -04:00
|
|
|
|
bool show_client_id;
|
|
|
|
|
bool show_uploads_enabled;
|
|
|
|
|
bool show_last_upload_attempt_time;
|
|
|
|
|
bool show_pending_reports;
|
|
|
|
|
bool show_completed_reports;
|
|
|
|
|
bool show_all_report_info;
|
|
|
|
|
bool set_uploads_enabled;
|
|
|
|
|
bool has_set_uploads_enabled;
|
|
|
|
|
bool utc;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// Converts |string| to |boolean|, returning true if a conversion could be
|
|
|
|
|
// performed, and false without setting |boolean| if no conversion could be
|
|
|
|
|
// performed. Various string representations of a boolean are recognized
|
|
|
|
|
// case-insensitively.
|
|
|
|
|
bool StringToBool(const char* string, bool* boolean) {
|
|
|
|
|
const char* const kFalseWords[] = {
|
|
|
|
|
"0",
|
|
|
|
|
"false",
|
|
|
|
|
"no",
|
|
|
|
|
"off",
|
|
|
|
|
"disabled",
|
|
|
|
|
"clear",
|
|
|
|
|
};
|
|
|
|
|
const char* const kTrueWords[] = {
|
|
|
|
|
"1",
|
|
|
|
|
"true",
|
|
|
|
|
"yes",
|
|
|
|
|
"on",
|
|
|
|
|
"enabled",
|
|
|
|
|
"set",
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
for (size_t index = 0; index < arraysize(kFalseWords); ++index) {
|
2015-08-05 16:34:50 -04:00
|
|
|
|
if (strcasecmp(string, kFalseWords[index]) == 0) {
|
2015-03-19 18:41:01 -04:00
|
|
|
|
*boolean = false;
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (size_t index = 0; index < arraysize(kTrueWords); ++index) {
|
2015-08-05 16:34:50 -04:00
|
|
|
|
if (strcasecmp(string, kTrueWords[index]) == 0) {
|
2015-03-19 18:41:01 -04:00
|
|
|
|
*boolean = true;
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Converts |boolean| to a string, either "true" or "false".
|
|
|
|
|
std::string BoolToString(bool boolean) {
|
|
|
|
|
return std::string(boolean ? "true" : "false");
|
|
|
|
|
}
|
|
|
|
|
|
2016-11-15 13:50:32 -05:00
|
|
|
|
// Converts |string| to |out_time|, returning true if a conversion could be
|
2015-03-19 18:41:01 -04:00
|
|
|
|
// performed, and false without setting |boolean| if no conversion could be
|
|
|
|
|
// performed. Various time formats are recognized, including several string
|
2016-11-15 13:50:32 -05:00
|
|
|
|
// representations and a numeric time_t representation. The special |string|
|
|
|
|
|
// "never" is recognized as converted to a |out_time| value of 0; "now" is
|
|
|
|
|
// converted to the current time. |utc|, when true, causes |string| to be
|
|
|
|
|
// interpreted as a UTC time rather than a local time when the time zone is
|
|
|
|
|
// ambiguous.
|
|
|
|
|
bool StringToTime(const char* string, time_t* out_time, bool utc) {
|
2015-08-05 16:34:50 -04:00
|
|
|
|
if (strcasecmp(string, "never") == 0) {
|
2016-11-15 13:50:32 -05:00
|
|
|
|
*out_time = 0;
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (strcasecmp(string, "now") == 0) {
|
2016-11-15 14:12:23 -05:00
|
|
|
|
errno = 0;
|
|
|
|
|
PCHECK(time(out_time) != -1 || errno == 0);
|
2015-03-19 18:41:01 -04:00
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const char* end = string + strlen(string);
|
|
|
|
|
|
|
|
|
|
const char* const kFormats[] = {
|
|
|
|
|
"%Y-%m-%d %H:%M:%S %Z",
|
|
|
|
|
"%Y-%m-%d %H:%M:%S",
|
|
|
|
|
"%+",
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
for (size_t index = 0; index < arraysize(kFormats); ++index) {
|
|
|
|
|
tm time_tm;
|
|
|
|
|
const char* strptime_result = strptime(string, kFormats[index], &time_tm);
|
|
|
|
|
if (strptime_result == end) {
|
2016-11-15 13:50:32 -05:00
|
|
|
|
time_t test_out_time;
|
2015-03-19 18:41:01 -04:00
|
|
|
|
if (utc) {
|
2016-11-15 13:50:32 -05:00
|
|
|
|
test_out_time = timegm(&time_tm);
|
2015-03-19 18:41:01 -04:00
|
|
|
|
} else {
|
2016-11-15 13:50:32 -05:00
|
|
|
|
test_out_time = mktime(&time_tm);
|
2015-03-19 18:41:01 -04:00
|
|
|
|
}
|
|
|
|
|
|
2016-11-15 13:50:32 -05:00
|
|
|
|
// mktime() is supposed to set errno in the event of an error, but support
|
|
|
|
|
// for this is spotty, so there’s no way to distinguish between a true
|
|
|
|
|
// time_t of -1 (1969-12-31 23:59:59 UTC) and an error. Assume error.
|
|
|
|
|
//
|
|
|
|
|
// See 10.11.5 Libc-1082.50.1/stdtime/FreeBSD/localtime.c and
|
|
|
|
|
// glibc-2.24/time/mktime.c, which don’t set errno or save and restore
|
|
|
|
|
// errno. Post-Android 7.1.0 Bionic is even more hopeless, setting errno
|
|
|
|
|
// whenever the time conversion returns -1, even for valid input. See
|
|
|
|
|
// libc/tzcode/localtime.c mktime(). Windows seems to get it right: see
|
|
|
|
|
// 10.0.14393 SDK Source/ucrt/time/mktime.cpp.
|
|
|
|
|
if (test_out_time != -1) {
|
|
|
|
|
*out_time = test_out_time;
|
|
|
|
|
return true;
|
|
|
|
|
}
|
2015-03-19 18:41:01 -04:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2016-11-15 13:50:32 -05:00
|
|
|
|
int64_t int64_result;
|
|
|
|
|
if (StringToNumber(string, &int64_result) &&
|
|
|
|
|
base::IsValueInRangeForNumericType<time_t>(int64_result)) {
|
|
|
|
|
*out_time = int64_result;
|
2015-03-19 18:41:01 -04:00
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2016-11-15 13:50:32 -05:00
|
|
|
|
// Converts |out_time| to a string, and returns it. |utc| determines whether the
|
|
|
|
|
// converted time will reference local time or UTC. If |out_time| is 0, the
|
2015-03-19 18:41:01 -04:00
|
|
|
|
// string "never" will be returned as a special case.
|
2016-11-15 13:50:32 -05:00
|
|
|
|
std::string TimeToString(time_t out_time, bool utc) {
|
|
|
|
|
if (out_time == 0) {
|
2015-03-19 18:41:01 -04:00
|
|
|
|
return std::string("never");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
tm time_tm;
|
|
|
|
|
if (utc) {
|
2016-11-15 13:50:32 -05:00
|
|
|
|
PCHECK(gmtime_r(&out_time, &time_tm));
|
2015-03-19 18:41:01 -04:00
|
|
|
|
} else {
|
2016-11-15 13:50:32 -05:00
|
|
|
|
PCHECK(localtime_r(&out_time, &time_tm));
|
2015-03-19 18:41:01 -04:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
char string[64];
|
|
|
|
|
CHECK_NE(
|
|
|
|
|
strftime(string, arraysize(string), "%Y-%m-%d %H:%M:%S %Z", &time_tm),
|
|
|
|
|
0u);
|
|
|
|
|
|
|
|
|
|
return std::string(string);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Shows information about a single |report|. |space_count| is the number of
|
|
|
|
|
// spaces to print before each line that is printed. |utc| determines whether
|
|
|
|
|
// times should be shown in UTC or the local time zone.
|
|
|
|
|
void ShowReport(const CrashReportDatabase::Report& report,
|
|
|
|
|
size_t space_count,
|
|
|
|
|
bool utc) {
|
|
|
|
|
std::string spaces(space_count, ' ');
|
|
|
|
|
|
2015-05-06 10:28:07 -07:00
|
|
|
|
printf("%sPath: %" PRFilePath "\n",
|
|
|
|
|
spaces.c_str(),
|
|
|
|
|
report.file_path.value().c_str());
|
2015-03-19 18:41:01 -04:00
|
|
|
|
if (!report.id.empty()) {
|
|
|
|
|
printf("%sRemote ID: %s\n", spaces.c_str(), report.id.c_str());
|
|
|
|
|
}
|
|
|
|
|
printf("%sCreation time: %s\n",
|
|
|
|
|
spaces.c_str(),
|
|
|
|
|
TimeToString(report.creation_time, utc).c_str());
|
|
|
|
|
printf("%sUploaded: %s\n",
|
|
|
|
|
spaces.c_str(),
|
|
|
|
|
BoolToString(report.uploaded).c_str());
|
|
|
|
|
printf("%sLast upload attempt time: %s\n",
|
|
|
|
|
spaces.c_str(),
|
|
|
|
|
TimeToString(report.last_upload_attempt_time, utc).c_str());
|
|
|
|
|
printf("%sUpload attempts: %d\n", spaces.c_str(), report.upload_attempts);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Shows information about a vector of |reports|. |space_count| is the number of
|
|
|
|
|
// spaces to print before each line that is printed. |options| will be consulted
|
|
|
|
|
// to determine whether to show expanded information
|
|
|
|
|
// (options.show_all_report_info) and what time zone to use when showing
|
|
|
|
|
// expanded information (options.utc).
|
|
|
|
|
void ShowReports(const std::vector<CrashReportDatabase::Report>& reports,
|
|
|
|
|
size_t space_count,
|
|
|
|
|
const Options& options) {
|
|
|
|
|
std::string spaces(space_count, ' ');
|
|
|
|
|
const char* colon = options.show_all_report_info ? ":" : "";
|
|
|
|
|
|
|
|
|
|
for (const CrashReportDatabase::Report& report : reports) {
|
|
|
|
|
printf("%s%s%s\n", spaces.c_str(), report.uuid.ToString().c_str(), colon);
|
|
|
|
|
if (options.show_all_report_info) {
|
|
|
|
|
ShowReport(report, space_count + 2, options.utc);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int DatabaseUtilMain(int argc, char* argv[]) {
|
2015-05-06 11:09:31 -07:00
|
|
|
|
const base::FilePath argv0(
|
|
|
|
|
ToolSupport::CommandLineArgumentToFilePathStringType(argv[0]));
|
|
|
|
|
const base::FilePath me(argv0.BaseName());
|
2015-03-19 18:41:01 -04:00
|
|
|
|
|
|
|
|
|
enum OptionFlags {
|
|
|
|
|
// “Short” (single-character) options.
|
|
|
|
|
kOptionDatabase = 'd',
|
|
|
|
|
|
|
|
|
|
// Long options without short equivalents.
|
|
|
|
|
kOptionLastChar = 255,
|
2015-10-08 13:10:02 -04:00
|
|
|
|
kOptionCreate,
|
2015-03-19 18:41:01 -04:00
|
|
|
|
kOptionShowClientID,
|
|
|
|
|
kOptionShowUploadsEnabled,
|
|
|
|
|
kOptionShowLastUploadAttemptTime,
|
|
|
|
|
kOptionShowPendingReports,
|
|
|
|
|
kOptionShowCompletedReports,
|
|
|
|
|
kOptionShowAllReportInfo,
|
|
|
|
|
kOptionShowReport,
|
|
|
|
|
kOptionSetUploadsEnabled,
|
|
|
|
|
kOptionSetLastUploadAttemptTime,
|
2015-04-03 18:57:01 -04:00
|
|
|
|
kOptionNewReport,
|
2015-03-19 18:41:01 -04:00
|
|
|
|
kOptionUTC,
|
|
|
|
|
|
|
|
|
|
// Standard options.
|
|
|
|
|
kOptionHelp = -2,
|
|
|
|
|
kOptionVersion = -3,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const option long_options[] = {
|
2015-10-08 13:10:02 -04:00
|
|
|
|
{"create", no_argument, nullptr, kOptionCreate},
|
2015-03-19 18:41:01 -04:00
|
|
|
|
{"database", required_argument, nullptr, kOptionDatabase},
|
|
|
|
|
{"show-client-id", no_argument, nullptr, kOptionShowClientID},
|
|
|
|
|
{"show-uploads-enabled", no_argument, nullptr, kOptionShowUploadsEnabled},
|
|
|
|
|
{"show-last-upload-attempt-time",
|
|
|
|
|
no_argument,
|
|
|
|
|
nullptr,
|
|
|
|
|
kOptionShowLastUploadAttemptTime},
|
|
|
|
|
{"show-pending-reports", no_argument, nullptr, kOptionShowPendingReports},
|
|
|
|
|
{"show-completed-reports",
|
|
|
|
|
no_argument,
|
|
|
|
|
nullptr,
|
|
|
|
|
kOptionShowCompletedReports},
|
|
|
|
|
{"show-all-report-info", no_argument, nullptr, kOptionShowAllReportInfo},
|
|
|
|
|
{"show-report", required_argument, nullptr, kOptionShowReport},
|
|
|
|
|
{"set-uploads-enabled",
|
|
|
|
|
required_argument,
|
|
|
|
|
nullptr,
|
|
|
|
|
kOptionSetUploadsEnabled},
|
|
|
|
|
{"set-last-upload-attempt-time",
|
|
|
|
|
required_argument,
|
|
|
|
|
nullptr,
|
|
|
|
|
kOptionSetLastUploadAttemptTime},
|
2015-04-03 18:57:01 -04:00
|
|
|
|
{"new-report", required_argument, nullptr, kOptionNewReport},
|
2015-03-19 18:41:01 -04:00
|
|
|
|
{"utc", no_argument, nullptr, kOptionUTC},
|
|
|
|
|
{"help", no_argument, nullptr, kOptionHelp},
|
|
|
|
|
{"version", no_argument, nullptr, kOptionVersion},
|
|
|
|
|
{nullptr, 0, nullptr, 0},
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
Options options = {};
|
|
|
|
|
|
|
|
|
|
int opt;
|
|
|
|
|
while ((opt = getopt_long(argc, argv, "d:", long_options, nullptr)) != -1) {
|
|
|
|
|
switch (opt) {
|
2015-10-08 13:10:02 -04:00
|
|
|
|
case kOptionCreate: {
|
|
|
|
|
options.create = true;
|
|
|
|
|
break;
|
|
|
|
|
}
|
2015-03-19 18:41:01 -04:00
|
|
|
|
case kOptionDatabase: {
|
|
|
|
|
options.database = optarg;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case kOptionShowClientID: {
|
|
|
|
|
options.show_client_id = true;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case kOptionShowUploadsEnabled: {
|
|
|
|
|
options.show_uploads_enabled = true;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case kOptionShowLastUploadAttemptTime: {
|
|
|
|
|
options.show_last_upload_attempt_time = true;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case kOptionShowPendingReports: {
|
|
|
|
|
options.show_pending_reports = true;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case kOptionShowCompletedReports: {
|
|
|
|
|
options.show_completed_reports = true;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case kOptionShowAllReportInfo: {
|
|
|
|
|
options.show_all_report_info = true;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case kOptionShowReport: {
|
|
|
|
|
UUID uuid;
|
|
|
|
|
if (!uuid.InitializeFromString(optarg)) {
|
|
|
|
|
ToolSupport::UsageHint(me, "--show-report requires a UUID");
|
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
|
}
|
|
|
|
|
options.show_reports.push_back(uuid);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case kOptionSetUploadsEnabled: {
|
|
|
|
|
if (!StringToBool(optarg, &options.set_uploads_enabled)) {
|
|
|
|
|
ToolSupport::UsageHint(me, "--set-uploads-enabled requires a BOOL");
|
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
|
}
|
|
|
|
|
options.has_set_uploads_enabled = true;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case kOptionSetLastUploadAttemptTime: {
|
|
|
|
|
options.set_last_upload_attempt_time_string = optarg;
|
|
|
|
|
break;
|
|
|
|
|
}
|
2015-04-03 18:57:01 -04:00
|
|
|
|
case kOptionNewReport: {
|
2015-05-06 11:09:31 -07:00
|
|
|
|
options.new_report_paths.push_back(base::FilePath(
|
|
|
|
|
ToolSupport::CommandLineArgumentToFilePathStringType(optarg)));
|
2015-04-03 18:57:01 -04:00
|
|
|
|
break;
|
|
|
|
|
}
|
2015-03-19 18:41:01 -04:00
|
|
|
|
case kOptionUTC: {
|
|
|
|
|
options.utc = true;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case kOptionHelp: {
|
|
|
|
|
Usage(me);
|
|
|
|
|
return EXIT_SUCCESS;
|
|
|
|
|
}
|
|
|
|
|
case kOptionVersion: {
|
|
|
|
|
ToolSupport::Version(me);
|
|
|
|
|
return EXIT_SUCCESS;
|
|
|
|
|
}
|
|
|
|
|
default: {
|
|
|
|
|
ToolSupport::UsageHint(me, nullptr);
|
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
argc -= optind;
|
|
|
|
|
argv += optind;
|
|
|
|
|
|
|
|
|
|
if (!options.database) {
|
|
|
|
|
ToolSupport::UsageHint(me, "--database is required");
|
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// This conversion couldn’t happen in the option-processing loop above because
|
|
|
|
|
// it depends on options.utc, which may have been set after
|
|
|
|
|
// options.set_last_upload_attempt_time_string.
|
|
|
|
|
if (options.set_last_upload_attempt_time_string) {
|
|
|
|
|
if (!StringToTime(options.set_last_upload_attempt_time_string,
|
|
|
|
|
&options.set_last_upload_attempt_time,
|
|
|
|
|
options.utc)) {
|
|
|
|
|
ToolSupport::UsageHint(me,
|
|
|
|
|
"--set-last-upload-attempt-time requires a TIME");
|
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2015-04-03 18:57:01 -04:00
|
|
|
|
// --new-report is treated as a show operation because it produces output.
|
2015-03-19 18:41:01 -04:00
|
|
|
|
const size_t show_operations = options.show_client_id +
|
|
|
|
|
options.show_uploads_enabled +
|
|
|
|
|
options.show_last_upload_attempt_time +
|
|
|
|
|
options.show_pending_reports +
|
|
|
|
|
options.show_completed_reports +
|
2015-04-03 18:57:01 -04:00
|
|
|
|
options.show_reports.size() +
|
|
|
|
|
options.new_report_paths.size();
|
2015-03-19 18:41:01 -04:00
|
|
|
|
const size_t set_operations =
|
|
|
|
|
options.has_set_uploads_enabled +
|
|
|
|
|
(options.set_last_upload_attempt_time_string != nullptr);
|
|
|
|
|
|
2015-10-08 13:10:02 -04:00
|
|
|
|
if ((options.create ? 1 : 0) + show_operations + set_operations == 0) {
|
2015-03-19 18:41:01 -04:00
|
|
|
|
ToolSupport::UsageHint(me, "nothing to do");
|
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
|
}
|
|
|
|
|
|
2016-04-25 12:13:07 -07:00
|
|
|
|
std::unique_ptr<CrashReportDatabase> database;
|
2015-10-08 13:10:02 -04:00
|
|
|
|
base::FilePath database_path = base::FilePath(
|
|
|
|
|
ToolSupport::CommandLineArgumentToFilePathStringType(options.database));
|
|
|
|
|
if (options.create) {
|
|
|
|
|
database = CrashReportDatabase::Initialize(database_path);
|
|
|
|
|
} else {
|
|
|
|
|
database = CrashReportDatabase::InitializeWithoutCreating(database_path);
|
|
|
|
|
}
|
2015-03-19 18:41:01 -04:00
|
|
|
|
if (!database) {
|
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Settings* settings = database->GetSettings();
|
|
|
|
|
|
|
|
|
|
// Handle the “show” options before the “set” options so that when they’re
|
|
|
|
|
// specified together, the “show” option reflects the initial state.
|
|
|
|
|
|
|
|
|
|
if (options.show_client_id) {
|
|
|
|
|
UUID client_id;
|
|
|
|
|
if (!settings->GetClientID(&client_id)) {
|
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const char* prefix = (show_operations > 1) ? "Client ID: " : "";
|
|
|
|
|
|
|
|
|
|
printf("%s%s\n", prefix, client_id.ToString().c_str());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (options.show_uploads_enabled) {
|
|
|
|
|
bool uploads_enabled;
|
|
|
|
|
if (!settings->GetUploadsEnabled(&uploads_enabled)) {
|
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const char* prefix = (show_operations > 1) ? "Uploads enabled: " : "";
|
|
|
|
|
|
|
|
|
|
printf("%s%s\n", prefix, BoolToString(uploads_enabled).c_str());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (options.show_last_upload_attempt_time) {
|
|
|
|
|
time_t last_upload_attempt_time;
|
|
|
|
|
if (!settings->GetLastUploadAttemptTime(&last_upload_attempt_time)) {
|
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const char* prefix =
|
|
|
|
|
(show_operations > 1) ? "Last upload attempt time: " : "";
|
|
|
|
|
|
|
|
|
|
printf("%s%s (%ld)\n",
|
|
|
|
|
prefix,
|
|
|
|
|
TimeToString(last_upload_attempt_time, options.utc).c_str(),
|
2015-05-06 10:28:07 -07:00
|
|
|
|
static_cast<long>(last_upload_attempt_time));
|
2015-03-19 18:41:01 -04:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (options.show_pending_reports) {
|
|
|
|
|
std::vector<CrashReportDatabase::Report> pending_reports;
|
|
|
|
|
if (database->GetPendingReports(&pending_reports) !=
|
|
|
|
|
CrashReportDatabase::kNoError) {
|
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (show_operations > 1) {
|
|
|
|
|
printf("Pending reports:\n");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ShowReports(pending_reports, show_operations > 1 ? 2 : 0, options);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (options.show_completed_reports) {
|
|
|
|
|
std::vector<CrashReportDatabase::Report> completed_reports;
|
|
|
|
|
if (database->GetCompletedReports(&completed_reports) !=
|
|
|
|
|
CrashReportDatabase::kNoError) {
|
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (show_operations > 1) {
|
|
|
|
|
printf("Completed reports:\n");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ShowReports(completed_reports, show_operations > 1 ? 2 : 0, options);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (const UUID& uuid : options.show_reports) {
|
|
|
|
|
CrashReportDatabase::Report report;
|
|
|
|
|
CrashReportDatabase::OperationStatus status =
|
|
|
|
|
database->LookUpCrashReport(uuid, &report);
|
|
|
|
|
if (status == CrashReportDatabase::kNoError) {
|
|
|
|
|
if (show_operations > 1) {
|
|
|
|
|
printf("Report %s:\n", uuid.ToString().c_str());
|
|
|
|
|
}
|
|
|
|
|
ShowReport(report, show_operations > 1 ? 2 : 0, options.utc);
|
|
|
|
|
} else if (status == CrashReportDatabase::kReportNotFound) {
|
|
|
|
|
// If only asked to do one thing, a failure to find the single requested
|
|
|
|
|
// report should result in a failure exit status.
|
|
|
|
|
if (show_operations + set_operations == 1) {
|
2015-05-06 10:28:07 -07:00
|
|
|
|
fprintf(
|
|
|
|
|
stderr, "%" PRFilePath ": Report not found\n", me.value().c_str());
|
2015-03-19 18:41:01 -04:00
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
|
}
|
|
|
|
|
printf("Report %s not found\n", uuid.ToString().c_str());
|
|
|
|
|
} else {
|
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (options.has_set_uploads_enabled &&
|
|
|
|
|
!settings->SetUploadsEnabled(options.set_uploads_enabled)) {
|
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (options.set_last_upload_attempt_time_string &&
|
|
|
|
|
!settings->SetLastUploadAttemptTime(
|
|
|
|
|
options.set_last_upload_attempt_time)) {
|
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
|
}
|
|
|
|
|
|
2017-03-16 16:15:54 -04:00
|
|
|
|
bool used_stdin = false;
|
2015-04-03 18:57:01 -04:00
|
|
|
|
for (const base::FilePath new_report_path : options.new_report_paths) {
|
2016-04-25 12:13:07 -07:00
|
|
|
|
std::unique_ptr<FileReaderInterface> file_reader;
|
2015-08-07 13:57:05 -04:00
|
|
|
|
|
|
|
|
|
if (new_report_path.value() == FILE_PATH_LITERAL("-")) {
|
2017-03-16 16:15:54 -04:00
|
|
|
|
if (used_stdin) {
|
|
|
|
|
fprintf(stderr,
|
|
|
|
|
"%" PRFilePath
|
|
|
|
|
": Only one --new-report may be read from standard input\n",
|
|
|
|
|
me.value().c_str());
|
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
|
}
|
|
|
|
|
used_stdin = true;
|
|
|
|
|
file_reader.reset(new WeakFileHandleFileReader(
|
|
|
|
|
StdioFileHandle(StdioStream::kStandardInput)));
|
2015-08-07 13:57:05 -04:00
|
|
|
|
} else {
|
2016-04-25 12:13:07 -07:00
|
|
|
|
std::unique_ptr<FileReader> file_path_reader(new FileReader());
|
2015-08-07 13:57:05 -04:00
|
|
|
|
if (!file_path_reader->Open(new_report_path)) {
|
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
|
}
|
|
|
|
|
|
2015-12-09 17:36:32 -05:00
|
|
|
|
file_reader = std::move(file_path_reader);
|
2015-04-03 18:57:01 -04:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
CrashReportDatabase::NewReport* new_report;
|
|
|
|
|
CrashReportDatabase::OperationStatus status =
|
|
|
|
|
database->PrepareNewCrashReport(&new_report);
|
|
|
|
|
if (status != CrashReportDatabase::kNoError) {
|
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
CrashReportDatabase::CallErrorWritingCrashReport
|
|
|
|
|
call_error_writing_crash_report(database.get(), new_report);
|
|
|
|
|
|
|
|
|
|
char buf[4096];
|
2015-10-22 16:14:18 -07:00
|
|
|
|
FileOperationResult read_result;
|
2015-08-07 13:57:05 -04:00
|
|
|
|
do {
|
|
|
|
|
read_result = file_reader->Read(buf, sizeof(buf));
|
|
|
|
|
if (read_result < 0) {
|
2015-04-03 18:57:01 -04:00
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
|
}
|
2015-08-07 13:57:05 -04:00
|
|
|
|
if (read_result > 0 &&
|
|
|
|
|
!LoggingWriteFile(new_report->handle, buf, read_result)) {
|
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
|
}
|
Make file_io reads more rational and predictable
ReadFile() attempted to continue reading after a short read. In most
cases, this is fine. However, ReadFile() would keep trying to fill a
partially-filled buffer until experiencing a 0-length read(), signaling
end-of-file. For certain weird file descriptors like terminal input, EOF
is an ephemeral condition, and attempting to read beyond EOF doesn’t
actually return 0 (EOF) provided that they remain open, it will block
waiting for more input. Consequently, ReadFile() and anything based on
ReadFile() had an undocumented and quirky interface, which was that any
short read that it returned (not an underlying short read) actually
indicated EOF.
This facet of ReadFile() was unexpected, so it’s being removed. The new
behavior is that ReadFile() will return an underlying short read. The
behavior of FileReaderInterface::Read() is updated in accordance with
this change.
Upon experiencing a short read, the caller can determine the best
action. Most callers were already prepared for this behavior. Outside of
util/file, only crashpad_database_util properly implemented EOF
detection according to previous semantics, and adapting it to new
semantics is trivial.
Callers who require an exact-length read can use the new
ReadFileExactly(), or the newly renamed LoggingReadFileExactly() or
CheckedReadFileExactly(). These functions will retry following a short
read. The renamed functions were previously called LoggingReadFile() and
CheckedReadFile(), but those names implied that they were simply
wrapping ReadFile(), which is not the case. They wrapped ReadFile() and
further, insisted on a full read. Since ReadFile()’s semantics are now
changing but these functions’ are not, they’re now even more distinct
from ReadFile(), and must be renamed to avoid confusion.
Test: *
Change-Id: I06b77e0d6ad8719bd2eb67dab93a8740542dd908
Reviewed-on: https://chromium-review.googlesource.com/456676
Reviewed-by: Robert Sesek <rsesek@chromium.org>
2017-03-16 13:36:38 -04:00
|
|
|
|
} while (read_result > 0);
|
2015-04-03 18:57:01 -04:00
|
|
|
|
|
|
|
|
|
call_error_writing_crash_report.Disarm();
|
|
|
|
|
|
|
|
|
|
UUID uuid;
|
|
|
|
|
status = database->FinishedWritingCrashReport(new_report, &uuid);
|
|
|
|
|
if (status != CrashReportDatabase::kNoError) {
|
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const char* prefix = (show_operations > 1) ? "New report ID: " : "";
|
|
|
|
|
printf("%s%s\n", prefix, uuid.ToString().c_str());
|
|
|
|
|
}
|
|
|
|
|
|
2015-03-19 18:41:01 -04:00
|
|
|
|
return EXIT_SUCCESS;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
} // namespace crashpad
|
|
|
|
|
|
2015-05-06 10:28:07 -07:00
|
|
|
|
#if defined(OS_POSIX)
|
2015-03-19 18:41:01 -04:00
|
|
|
|
int main(int argc, char* argv[]) {
|
|
|
|
|
return crashpad::DatabaseUtilMain(argc, argv);
|
|
|
|
|
}
|
2015-05-06 10:28:07 -07:00
|
|
|
|
#elif defined(OS_WIN)
|
|
|
|
|
int wmain(int argc, wchar_t* argv[]) {
|
2015-05-06 11:09:31 -07:00
|
|
|
|
return crashpad::ToolSupport::Wmain(argc, argv, crashpad::DatabaseUtilMain);
|
2015-05-06 10:28:07 -07:00
|
|
|
|
}
|
|
|
|
|
#endif // OS_POSIX
|