[go: nahoru, domu]

blob: 6edc4498e40790c1c8733570c7628f8aea3f0142 [file] [log] [blame]
// Copyright 2013 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "extensions/browser/api/feedback_private/feedback_service.h"
#include <memory>
#include <string>
#include <utility>
#include "base/barrier_closure.h"
#include "base/bind.h"
#include "base/files/file_util.h"
#include "base/logging.h"
#include "base/memory/ref_counted.h"
#include "base/metrics/histogram_functions.h"
#include "base/metrics/histogram_macros.h"
#include "base/metrics/statistics_recorder.h"
#include "base/strings/strcat.h"
#include "base/task/thread_pool.h"
#include "base/time/time.h"
#include "build/chromeos_buildflags.h"
#include "components/feedback/feedback_data.h"
#include "components/feedback/feedback_report.h"
#include "components/feedback/system_logs/system_logs_fetcher.h"
#include "components/feedback/system_logs/system_logs_source.h"
#include "content/public/browser/browser_context.h"
#include "extensions/browser/api/extensions_api_client.h"
#include "extensions/browser/api/feedback_private/feedback_private_delegate.h"
#include "extensions/browser/blob_reader.h"
#include "net/base/network_change_notifier.h"
#if BUILDFLAG(IS_CHROMEOS_ASH)
#include "ash/public/cpp/assistant/controller/assistant_controller.h"
#include "chromeos/ash/services/assistant/public/cpp/assistant_service.h"
#endif // BUILDFLAG(IS_CHROMEOS_ASH)
namespace extensions {
using system_logs::SysLogsFetcherCallback;
using system_logs::SystemLogsFetcher;
using system_logs::SystemLogsResponse;
namespace {
#if BUILDFLAG(IS_CHROMEOS_ASH)
constexpr base::FilePath::CharType kBluetoothLogsFilePath[] =
FILE_PATH_LITERAL("/var/log/bluetooth/log.bz2");
constexpr base::FilePath::CharType kBluetoothLogsFilePathOld[] =
FILE_PATH_LITERAL("/var/log/bluetooth/log.bz2.old");
constexpr base::FilePath::CharType kBluetoothQualityReportFilePath[] =
FILE_PATH_LITERAL("/var/log/bluetooth/bluetooth_quality_report");
constexpr char kBluetoothLogsAttachmentName[] = "bluetooth_logs.bz2";
constexpr char kBluetoothLogsAttachmentNameOld[] = "bluetooth_logs.old.bz2";
constexpr char kBluetoothQualityReportAttachmentName[] =
"bluetooth_quality_report";
constexpr char kLacrosHistogramsFilename[] = "lacros_histograms.zip";
void LoadBluetoothLogs(scoped_refptr<feedback::FeedbackData> feedback_data) {
std::string bluetooth_logs;
if (base::ReadFileToString(base::FilePath(kBluetoothLogsFilePath),
&bluetooth_logs)) {
feedback_data->AddFile(kBluetoothLogsAttachmentName,
std::move(bluetooth_logs));
}
if (base::ReadFileToString(base::FilePath(kBluetoothLogsFilePathOld),
&bluetooth_logs)) {
feedback_data->AddFile(kBluetoothLogsAttachmentNameOld,
std::move(bluetooth_logs));
}
if (base::ReadFileToString(base::FilePath(kBluetoothQualityReportFilePath),
&bluetooth_logs)) {
feedback_data->AddFile(kBluetoothQualityReportAttachmentName,
std::move(bluetooth_logs));
}
}
#endif
constexpr char kLacrosLogEntryPrefix[] = "Lacros ";
} // namespace
FeedbackService::FeedbackService(content::BrowserContext* browser_context)
: FeedbackService(
browser_context,
ExtensionsAPIClient::Get()->GetFeedbackPrivateDelegate()) {}
FeedbackService::FeedbackService(content::BrowserContext* browser_context,
FeedbackPrivateDelegate* delegate)
: browser_context_(browser_context), delegate_(delegate) {}
FeedbackService::~FeedbackService() = default;
// After the attached file and screenshot if available are fetched, the callback
// will be invoked. Other further processing will be done in background. The
// report will be sent out once all data are in place.
void FeedbackService::SendFeedback(
const FeedbackParams& params,
scoped_refptr<feedback::FeedbackData> feedback_data,
SendFeedbackCallback callback) {
auto* browser_client = ExtensionsBrowserClient::Get();
feedback_data->set_locale(browser_client->GetApplicationLocale());
feedback_data->set_user_agent(browser_client->GetUserAgent());
FetchAttachedFileAndScreenshot(
feedback_data,
base::BindOnce(&FeedbackService::OnAttachedFileAndScreenshotFetched, this,
params, feedback_data, std::move(callback)));
}
void FeedbackService::FetchAttachedFileAndScreenshot(
scoped_refptr<feedback::FeedbackData> feedback_data,
base::OnceClosure callback) {
const bool must_attach_file = !feedback_data->attached_file_uuid().empty();
const bool must_attach_screenshot = !feedback_data->screenshot_uuid().empty();
auto barrier_closure = base::BarrierClosure(
(must_attach_file ? 1 : 0) + (must_attach_screenshot ? 1 : 0),
std::move(callback));
if (must_attach_file) {
auto populate_attached_file = base::BindOnce(
[](scoped_refptr<feedback::FeedbackData> feedback_data,
std::unique_ptr<std::string> data, int64_t length) {
feedback_data->set_attached_file_uuid(std::string());
if (data)
feedback_data->AttachAndCompressFileData(std::move(*data));
},
feedback_data);
BlobReader::Read(browser_context_, feedback_data->attached_file_uuid(),
std::move(populate_attached_file).Then(barrier_closure));
}
if (must_attach_screenshot) {
auto populate_screenshot = base::BindOnce(
[](scoped_refptr<feedback::FeedbackData> feedback_data,
std::unique_ptr<std::string> data, int64_t length) {
feedback_data->set_screenshot_uuid(std::string());
if (data)
feedback_data->set_image(std::move(*data));
},
feedback_data);
BlobReader::Read(browser_context_, feedback_data->screenshot_uuid(),
std::move(populate_screenshot).Then(barrier_closure));
}
}
void FeedbackService::OnAttachedFileAndScreenshotFetched(
const FeedbackParams& params,
scoped_refptr<feedback::FeedbackData> feedback_data,
SendFeedbackCallback callback) {
if (params.load_system_info) {
// The user has chosen to send system logs. They (and on ash more logs)
// will be loaded in the background without blocking the client.
FetchSystemInformation(params, feedback_data);
} else {
#if BUILDFLAG(IS_CHROMEOS_ASH)
if (feedback_data->sys_info()->size() > 0) {
// The user has chosen to send system logs which has been loaded from the
// client side. On ash, extra logs need to be fetched.
FetchExtraLogs(params, feedback_data);
} else {
// The user has chosen not to send system logs.
OnAllLogsFetched(params, feedback_data);
}
#else
OnAllLogsFetched(params, feedback_data);
#endif // BUILDFLAG(IS_CHROMEOS_ASH)
}
base::UmaHistogramMediumTimes(
"Feedback.Duration.FormSubmitToConfirmation",
base::TimeTicks::Now() - params.form_submit_time);
// True means report will be sent shortly.
// False means report will be sent once the device is online.
const bool status = !net::NetworkChangeNotifier::IsOffline();
UMA_HISTOGRAM_BOOLEAN("Feedback.ReportSending.Online", status);
// Notify client that data submitted has been received successfully. The
// report will be sent out once further processing is done.
std::move(callback).Run(status);
}
void FeedbackService::FetchSystemInformation(
const FeedbackParams& params,
scoped_refptr<feedback::FeedbackData> feedback_data) {
base::TimeTicks fetch_start_time = base::TimeTicks::Now();
delegate_->FetchSystemInformation(
browser_context_,
base::BindOnce(&FeedbackService::OnSystemInformationFetched, this,
fetch_start_time, params, feedback_data));
}
void FeedbackService::OnSystemInformationFetched(
base::TimeTicks fetch_start_time,
const FeedbackParams& params,
scoped_refptr<feedback::FeedbackData> feedback_data,
std::unique_ptr<system_logs::SystemLogsResponse> sys_info) {
// Fetching is currently slow and could take up to 2 minutes on Chrome OS.
base::UmaHistogramMediumTimes("Feedback.Duration.FetchSystemInformation",
base::TimeTicks::Now() - fetch_start_time);
if (sys_info) {
for (auto& itr : *sys_info) {
if (FeedbackCommon::IncludeInSystemLogs(itr.first,
params.is_internal_email))
feedback_data->AddLog(std::move(itr.first), std::move(itr.second));
}
}
#if BUILDFLAG(IS_CHROMEOS_ASH)
FetchExtraLogs(params, feedback_data);
#else
OnAllLogsFetched(params, feedback_data);
#endif // BUILDFLAG(IS_CHROMEOS_ASH)
}
#if BUILDFLAG(IS_CHROMEOS_ASH)
void FeedbackService::FetchExtraLogs(
const FeedbackParams& params,
scoped_refptr<feedback::FeedbackData> feedback_data) {
delegate_->FetchExtraLogs(
feedback_data,
base::BindOnce(&FeedbackService::OnExtraLogsFetched, this, params));
}
void FeedbackService::OnExtraLogsFetched(
const FeedbackParams& params,
scoped_refptr<feedback::FeedbackData> feedback_data) {
delegate_->GetLacrosHistograms(
base::BindOnce(&FeedbackService::OnLacrosHistogramsFetched, this, params,
feedback_data));
}
void FeedbackService::OnLacrosHistogramsFetched(
const FeedbackParams& params,
scoped_refptr<feedback::FeedbackData> feedback_data,
const std::string& compressed_histograms) {
if (!compressed_histograms.empty()) {
feedback_data->AddFile(kLacrosHistogramsFilename,
std::move(compressed_histograms));
}
if (params.send_bluetooth_logs) {
base::ThreadPool::PostTaskAndReply(
FROM_HERE, {base::MayBlock()},
base::BindOnce(&LoadBluetoothLogs, feedback_data),
base::BindOnce(&FeedbackService::OnAllLogsFetched, this, params,
feedback_data));
} else {
OnAllLogsFetched(params, feedback_data);
}
}
#endif // BUILDFLAG(IS_CHROMEOS_ASH)
void FeedbackService::OnAllLogsFetched(
const FeedbackParams& params,
scoped_refptr<feedback::FeedbackData> feedback_data) {
if (!params.send_tab_titles) {
feedback_data->RemoveLog(
feedback::FeedbackReport::kMemUsageWithTabTitlesKey);
// On Lacros, the key has a prefix "Lacros ".
feedback_data->RemoveLog(
base::StrCat({kLacrosLogEntryPrefix,
feedback::FeedbackReport::kMemUsageWithTabTitlesKey}));
}
feedback_data->CompressSystemInfo();
if (params.send_histograms) {
std::string histograms =
base::StatisticsRecorder::ToJSON(base::JSON_VERBOSITY_LEVEL_FULL);
feedback_data->SetAndCompressHistograms(std::move(histograms));
}
DCHECK(feedback_data->attached_file_uuid().empty());
DCHECK(feedback_data->screenshot_uuid().empty());
#if BUILDFLAG(IS_CHROMEOS_ASH)
// Send feedback to Assistant server if triggered from Google Assistant.
if (feedback_data->from_assistant()) {
ash::AssistantController::Get()->SendAssistantFeedback(
feedback_data->assistant_debug_info_allowed(),
feedback_data->description(), feedback_data->image());
}
#endif
// Signal the feedback object that the data from the feedback page has been
// filled - the object will manage sending of the actual report.
feedback_data->OnFeedbackPageDataComplete();
base::UmaHistogramTimes("Feedback.Duration.FormSubmitToSendQueue",
base::TimeTicks::Now() - params.form_submit_time);
}
} // namespace extensions