// Copyright 2015 The Crashpad Authors // // 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 "handler/crash_report_upload_thread.h" #include <errno.h> #include <time.h> #include <algorithm> #include <map> #include <memory> #include <vector> #include "base/logging.h" #include "base/notreached.h" #include "base/strings/stringprintf.h" #include "base/strings/utf_string_conversions.h" #include "build/build_config.h" #include "client/settings.h" #include "handler/minidump_to_upload_parameters.h" #include "snapshot/minidump/process_snapshot_minidump.h" #include "snapshot/module_snapshot.h" #include "util/file/file_reader.h" #include "util/misc/metrics.h" #include "util/misc/uuid.h" #include "util/net/http_body.h" #include "util/net/http_multipart_builder.h" #include "util/net/http_transport.h" #include "util/net/url.h" #include "util/stdlib/map_insert.h" #if BUILDFLAG(IS_APPLE) #include "handler/mac/file_limit_annotation.h" #endif // BUILDFLAG(IS_APPLE) #if BUILDFLAG(IS_IOS) #include "util/ios/scoped_background_task.h" #endif // BUILDFLAG(IS_IOS) namespace crashpad { namespace { // The number of seconds to wait between checking for pending reports. const int kRetryWorkIntervalSeconds = 15 * 60; #if BUILDFLAG(IS_IOS) // The number of times to attempt to upload a pending report, repeated on // failure. Attempts will happen once per launch, once per call to // ReportPending(), and, if Options.watch_pending_reports is true, once every // kRetryWorkIntervalSeconds. Currently iOS only. const int kRetryAttempts = 5; #endif // Wraps a reference to a no-args function (which can be empty). When this // object goes out of scope, invokes the function if it is non-empty. // // The lifetime of the function must outlive the lifetime of this object. class ScopedFunctionInvoker final { public: ScopedFunctionInvoker(const std::function<void()>& function) : function_(function) {} ScopedFunctionInvoker(const ScopedFunctionInvoker&) = delete; ScopedFunctionInvoker& operator=(const ScopedFunctionInvoker&) = delete; ~ScopedFunctionInvoker() { if (function_) { function_(); } } private: const std::function<void()>& function_; }; } // namespace CrashReportUploadThread::CrashReportUploadThread( CrashReportDatabase* database, const std::string& url, const Options& options, ProcessPendingReportsObservationCallback callback) : options_(options), callback_(callback), url_(url), // When watching for pending reports, check every 15 minutes, even in the // absence of a signal from the handler thread. This allows for failed // uploads to be retried periodically, and for pending reports written by // other processes to be recognized. thread_(options.watch_pending_reports ? kRetryWorkIntervalSeconds : WorkerThread::kIndefiniteWait, this), known_pending_report_uuids_(), database_(database) { DCHECK(!url_.empty()); } CrashReportUploadThread::~CrashReportUploadThread() { } void CrashReportUploadThread::ReportPending(const UUID& report_uuid) { known_pending_report_uuids_.PushBack(report_uuid); if (thread_.is_running()) thread_.DoWorkNow(); } void CrashReportUploadThread::Start() { thread_.Start( options_.watch_pending_reports ? 0.0 : WorkerThread::kIndefiniteWait); } void CrashReportUploadThread::Stop() { thread_.Stop(); } void CrashReportUploadThread::ProcessPendingReports() { #if BUILDFLAG(IS_IOS) internal::ScopedBackgroundTask scoper("CrashReportUploadThread"); #endif // BUILDFLAG(IS_IOS) // If callback_ is non-empty, invoke it when this function returns after // uploads complete (regardless of whether or not that succeeded). ScopedFunctionInvoker scoped_function_invoker(callback_); std::vector<UUID> known_report_uuids = known_pending_report_uuids_.Drain(); for (const UUID& report_uuid : known_report_uuids) { CrashReportDatabase::Report report; if (database_->LookUpCrashReport(report_uuid, &report) != CrashReportDatabase::kNoError) { continue; } ProcessPendingReport(report); // Respect Stop() being called after at least one attempt to process a // report. if (!thread_.is_running()) { return; } } // Known pending reports are always processed (above). The rest of this // function is concerned with scanning for pending reports not already known // to this thread. if (!options_.watch_pending_reports) { return; } std::vector<CrashReportDatabase::Report> reports; if (database_->GetPendingReports(&reports) != CrashReportDatabase::kNoError) { // The database is sick. It might be prudent to stop trying to poke it from // this thread by abandoning the thread altogether. On the other hand, if // the problem is transient, it might be possible to talk to it again on the // next pass. For now, take the latter approach. return; } for (const CrashReportDatabase::Report& report : reports) { if (std::find(known_report_uuids.begin(), known_report_uuids.end(), report.uuid) != known_report_uuids.end()) { // An attempt to process the report already occurred above. The report is // still pending, so upload must have failed. Don’t retry it immediately, // it can wait until at least the next pass through this method. continue; } ProcessPendingReport(report); // Respect Stop() being called after at least one attempt to process a // report. if (!thread_.is_running()) { return; } } } void CrashReportUploadThread::ProcessPendingReport( const CrashReportDatabase::Report& report) { #if BUILDFLAG(IS_APPLE) RecordFileLimitAnnotation(); #endif // BUILDFLAG(IS_APPLE) Settings* const settings = database_->GetSettings(); bool uploads_enabled; if (!report.upload_explicitly_requested && (!settings->GetUploadsEnabled(&uploads_enabled) || !uploads_enabled)) { // Don’t attempt an upload if there’s no URL to upload to. Allow upload if // it has been explicitly requested by the user, otherwise, respect the // upload-enabled state stored in the database’s settings. database_->SkipReportUpload(report.uuid, Metrics::CrashSkippedReason::kUploadsDisabled); return; } if (ShouldRateLimitUpload(report)) return; #if BUILDFLAG(IS_IOS) if (ShouldRateLimitRetry(report)) return; #endif // BUILDFLAG(IS_IOS) std::unique_ptr<const CrashReportDatabase::UploadReport> upload_report; CrashReportDatabase::OperationStatus status = database_->GetReportForUploading(report.uuid, &upload_report); switch (status) { case CrashReportDatabase::kNoError: break; case CrashReportDatabase::kBusyError: case CrashReportDatabase::kReportNotFound: // Someone else may have gotten to it first. If they’re working on it now, // this will be kBusyError. If they’ve already finished with it, it’ll be // kReportNotFound. return; case CrashReportDatabase::kFileSystemError: case CrashReportDatabase::kDatabaseError: // In these cases, SkipReportUpload() might not work either, but it’s best // to at least try to get the report out of the way. database_->SkipReportUpload(report.uuid, Metrics::CrashSkippedReason::kDatabaseError); return; case CrashReportDatabase::kCannotRequestUpload: NOTREACHED(); return; } std::string response_body; UploadResult upload_result = UploadReport(upload_report.get(), &response_body); switch (upload_result) { case UploadResult::kSuccess: database_->RecordUploadComplete(std::move(upload_report), response_body); break; case UploadResult::kPermanentFailure: upload_report.reset(); database_->SkipReportUpload( report.uuid, Metrics::CrashSkippedReason::kPrepareForUploadFailed); break; case UploadResult::kRetry: #if BUILDFLAG(IS_IOS) if (upload_report->upload_attempts > kRetryAttempts) { upload_report.reset(); database_->SkipReportUpload(report.uuid, Metrics::CrashSkippedReason::kUploadFailed); } else { Metrics::CrashUploadSkipped( Metrics::CrashSkippedReason::kUploadFailedButCanRetry); retry_uuid_time_map_[report.uuid] = time(nullptr) + (1 << upload_report->upload_attempts) * kRetryWorkIntervalSeconds; } #else upload_report.reset(); // TODO(mark): Deal with retries properly: don’t call SkipReportUplaod() // if the result was kRetry and the report hasn’t already been retried // too many times. database_->SkipReportUpload(report.uuid, Metrics::CrashSkippedReason::kUploadFailed); #endif break; } } CrashReportUploadThread::UploadResult CrashReportUploadThread::UploadReport( const CrashReportDatabase::UploadReport* report, std::string* response_body) { std::map<std::string, std::string> parameters; FileReader* reader = report->Reader(); FileOffset start_offset = reader->SeekGet(); if (start_offset < 0) { return UploadResult::kPermanentFailure; } // Ignore any errors that might occur when attempting to interpret the // minidump file. This may result in its being uploaded with few or no // parameters, but as long as there’s a dump file, the server can decide what // to do with it. ProcessSnapshotMinidump minidump_process_snapshot; if (minidump_process_snapshot.Initialize(reader)) { parameters = BreakpadHTTPFormParametersFromMinidump(&minidump_process_snapshot); } if (!reader->SeekSet(start_offset)) { return UploadResult::kPermanentFailure; } HTTPMultipartBuilder http_multipart_builder; http_multipart_builder.SetGzipEnabled(options_.upload_gzip); static constexpr char kMinidumpKey[] = "upload_file_minidump"; for (const auto& kv : parameters) { if (kv.first == kMinidumpKey) { LOG(WARNING) << "reserved key " << kv.first << ", discarding value " << kv.second; } else { http_multipart_builder.SetFormData(kv.first, kv.second); } } for (const auto& it : report->GetAttachments()) { http_multipart_builder.SetFileAttachment( it.first, it.first, it.second, "application/octet-stream"); } http_multipart_builder.SetFileAttachment(kMinidumpKey, report->uuid.ToString() + ".dmp", reader, "application/octet-stream"); std::unique_ptr<HTTPTransport> http_transport(HTTPTransport::Create()); if (!http_transport) { return UploadResult::kPermanentFailure; } HTTPHeaders content_headers; http_multipart_builder.PopulateContentHeaders(&content_headers); for (const auto& content_header : content_headers) { http_transport->SetHeader(content_header.first, content_header.second); } http_transport->SetBodyStream(http_multipart_builder.GetBodyStream()); // TODO(mark): The timeout should be configurable by the client. http_transport->SetTimeout(internal::kUploadReportTimeoutSeconds); std::string url = url_; if (options_.identify_client_via_url) { // Add parameters to the URL which identify the client to the server. static constexpr struct { const char* key; const char* url_field_name; } kURLParameterMappings[] = { {"prod", "product"}, {"ver", "version"}, {"guid", "guid"}, }; for (const auto& parameter_mapping : kURLParameterMappings) { const auto it = parameters.find(parameter_mapping.key); if (it != parameters.end()) { url.append( base::StringPrintf("%c%s=%s", url.find('?') == std::string::npos ? '?' : '&', parameter_mapping.url_field_name, URLEncode(it->second).c_str())); } } } http_transport->SetURL(url); if (!http_transport->ExecuteSynchronously(response_body)) { return UploadResult::kRetry; } return UploadResult::kSuccess; } void CrashReportUploadThread::DoWork(const WorkerThread* thread) { ProcessPendingReports(); } bool CrashReportUploadThread::ShouldRateLimitUpload( const CrashReportDatabase::Report& report) { if (report.upload_explicitly_requested || !options_.rate_limit) return false; Settings* const settings = database_->GetSettings(); time_t last_upload_attempt_time; if (settings->GetLastUploadAttemptTime(&last_upload_attempt_time)) { time_t now = time(nullptr); if (now >= last_upload_attempt_time) { // If the most recent upload attempt occurred within the past hour, // don’t attempt to upload the new report. If it happened longer ago, // attempt to upload the report. constexpr int kUploadAttemptIntervalSeconds = 60 * 60; // 1 hour if (now - last_upload_attempt_time < kUploadAttemptIntervalSeconds) { database_->SkipReportUpload( report.uuid, Metrics::CrashSkippedReason::kUploadThrottled); return true; } } else { // The most recent upload attempt purportedly occurred in the future. If // it “happened” at least one day in the future, assume that the last // upload attempt time is bogus, and attempt to upload the report. If // the most recent upload time is in the future but within one day, // accept it and don’t attempt to upload the report. constexpr int kBackwardsClockTolerance = 60 * 60 * 24; // 1 day if (last_upload_attempt_time - now < kBackwardsClockTolerance) { database_->SkipReportUpload( report.uuid, Metrics::CrashSkippedReason::kUnexpectedTime); return true; } } } return false; } #if BUILDFLAG(IS_IOS) bool CrashReportUploadThread::ShouldRateLimitRetry( const CrashReportDatabase::Report& report) { if (retry_uuid_time_map_.find(report.uuid) != retry_uuid_time_map_.end()) { time_t now = time(nullptr); if (now < retry_uuid_time_map_[report.uuid]) { return true; } else { retry_uuid_time_map_.erase(report.uuid); } } return false; } #endif } // namespace crashpad