blob: 17aed4fb8870ebe8d27c4d018671acc9700b266d [file] [log] [blame]
// Copyright 2012 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "chrome/browser/ui/webui/crashes_ui.h"
#include <stddef.h>
#include <memory>
#include <string>
#include <vector>
#include "base/functional/bind.h"
#include "base/functional/callback_helpers.h"
#include "base/memory/ref_counted_memory.h"
#include "base/strings/utf_string_conversions.h"
#include "base/system/sys_info.h"
#include "base/values.h"
#include "build/build_config.h"
#include "build/chromeos_buildflags.h"
#include "chrome/browser/crash_upload_list/crash_upload_list.h"
#include "chrome/browser/metrics/chrome_metrics_service_accessor.h"
#include "chrome/browser/metrics/metrics_reporting_state.h"
#include "chrome/browser/profiles/profile.h"
#include "chrome/browser/signin/identity_manager_factory.h"
#include "chrome/common/url_constants.h"
#include "chrome/grit/branded_strings.h"
#include "components/crash/core/browser/crashes_ui_util.h"
#include "components/grit/components_scaled_resources.h"
#include "components/grit/dev_ui_components_resources.h"
#include "components/signin/public/identity_manager/identity_manager.h"
#include "components/version_info/version_info.h"
#include "content/public/browser/web_contents.h"
#include "content/public/browser/web_ui.h"
#include "content/public/browser/web_ui_data_source.h"
#include "content/public/browser/web_ui_message_handler.h"
#include "google_apis/gaia/gaia_auth_util.h"
#include "ui/base/resource/resource_bundle.h"
#if BUILDFLAG(IS_CHROMEOS_ASH)
#include "chromeos/ash/components/dbus/debug_daemon/debug_daemon_client.h"
#endif
#if BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS)
#include "components/crash/core/app/crashpad.h"
#endif
using content::WebContents;
using content::WebUIMessageHandler;
namespace {
void CreateAndAddCrashesUIHTMLSource(Profile* profile) {
content::WebUIDataSource* source = content::WebUIDataSource::CreateAndAdd(
profile, chrome::kChromeUICrashesHost);
for (size_t i = 0; i < crash_reporter::kCrashesUILocalizedStringsCount; ++i) {
source->AddLocalizedString(
crash_reporter::kCrashesUILocalizedStrings[i].name,
crash_reporter::kCrashesUILocalizedStrings[i].resource_id);
}
source->AddLocalizedString(crash_reporter::kCrashesUIShortProductName,
IDS_SHORT_PRODUCT_NAME);
source->UseStringsJs();
source->AddResourcePath(crash_reporter::kCrashesUICrashesJS,
IDR_CRASH_CRASHES_JS);
source->AddResourcePath(crash_reporter::kCrashesUICrashesCSS,
IDR_CRASH_CRASHES_CSS);
source->AddResourcePath(crash_reporter::kCrashesUISadTabSVG,
IDR_CRASH_SADTAB_SVG);
source->SetDefaultResource(IDR_CRASH_CRASHES_HTML);
}
////////////////////////////////////////////////////////////////////////////////
//
// CrashesDOMHandler
//
////////////////////////////////////////////////////////////////////////////////
// The handler for Javascript messages for the chrome://crashes/ page.
class CrashesDOMHandler : public WebUIMessageHandler {
public:
CrashesDOMHandler();
CrashesDOMHandler(const CrashesDOMHandler&) = delete;
CrashesDOMHandler& operator=(const CrashesDOMHandler&) = delete;
~CrashesDOMHandler() override;
// WebUIMessageHandler implementation.
void RegisterMessages() override;
private:
void OnUploadListAvailable();
// Asynchronously fetches the list of crashes. Called from JS.
void HandleRequestCrashes(const base::Value::List& args);
#if BUILDFLAG(IS_CHROMEOS_ASH)
// Asynchronously triggers crash uploading. Called from JS.
void HandleRequestUploads(const base::Value::List& args);
#endif
// Sends the recent crashes list JS.
void UpdateUI();
// Asynchronously requests a user triggered upload. Called from JS.
void HandleRequestSingleCrashUpload(const base::Value::List& args);
scoped_refptr<UploadList> upload_list_;
bool list_available_;
bool first_load_;
};
CrashesDOMHandler::CrashesDOMHandler()
: list_available_(false), first_load_(true) {
upload_list_ = CreateCrashUploadList();
}
CrashesDOMHandler::~CrashesDOMHandler() {
upload_list_->CancelLoadCallback();
}
void CrashesDOMHandler::RegisterMessages() {
upload_list_->Load(base::BindOnce(&CrashesDOMHandler::OnUploadListAvailable,
base::Unretained(this)));
web_ui()->RegisterMessageCallback(
crash_reporter::kCrashesUIRequestCrashList,
base::BindRepeating(&CrashesDOMHandler::HandleRequestCrashes,
base::Unretained(this)));
#if BUILDFLAG(IS_CHROMEOS_ASH)
web_ui()->RegisterMessageCallback(
crash_reporter::kCrashesUIRequestCrashUpload,
base::BindRepeating(&CrashesDOMHandler::HandleRequestUploads,
base::Unretained(this)));
#endif
web_ui()->RegisterMessageCallback(
crash_reporter::kCrashesUIRequestSingleCrashUpload,
base::BindRepeating(&CrashesDOMHandler::HandleRequestSingleCrashUpload,
base::Unretained(this)));
}
void CrashesDOMHandler::HandleRequestCrashes(const base::Value::List& args) {
AllowJavascript();
if (first_load_) {
first_load_ = false;
if (list_available_)
UpdateUI();
} else {
list_available_ = false;
upload_list_->Load(base::BindOnce(&CrashesDOMHandler::OnUploadListAvailable,
base::Unretained(this)));
}
}
#if BUILDFLAG(IS_CHROMEOS_ASH)
void CrashesDOMHandler::HandleRequestUploads(const base::Value::List& args) {
ash::DebugDaemonClient* debugd_client = ash::DebugDaemonClient::Get();
DCHECK(debugd_client);
debugd_client->UploadCrashes(base::BindOnce([](bool success) {
if (!success) {
LOG(WARNING) << "crash_sender failed or timed out";
}
}));
}
#endif
void CrashesDOMHandler::OnUploadListAvailable() {
list_available_ = true;
if (!first_load_)
UpdateUI();
}
void CrashesDOMHandler::UpdateUI() {
bool crash_reporting_enabled =
ChromeMetricsServiceAccessor::IsMetricsAndCrashReportingEnabled();
bool system_crash_reporter = false;
#if BUILDFLAG(IS_CHROMEOS)
// Chrome OS has a system crash reporter.
system_crash_reporter = true;
#endif
bool is_internal = false;
auto* identity_manager =
IdentityManagerFactory::GetForProfile(Profile::FromWebUI(web_ui()));
if (identity_manager) {
is_internal = gaia::IsGoogleInternalAccountEmail(
identity_manager->GetPrimaryAccountInfo(signin::ConsentLevel::kSignin)
.email);
}
bool manual_uploads_supported = false;
#if BUILDFLAG(IS_WIN) || BUILDFLAG(IS_MAC) || BUILDFLAG(IS_LINUX) || \
BUILDFLAG(IS_ANDROID)
manual_uploads_supported = true;
#endif
bool allow_manual_uploads =
manual_uploads_supported &&
(crash_reporting_enabled || !IsMetricsReportingPolicyManaged());
// Show crash reports regardless of |crash_reporting_enabled| when it is
// possible to manually upload reports.
bool upload_list = manual_uploads_supported || crash_reporting_enabled;
base::Value::List crash_list;
if (upload_list)
crash_reporter::UploadListToValue(upload_list_.get(), &crash_list);
base::Value::Dict result;
result.Set("enabled", crash_reporting_enabled);
result.Set("dynamicBackend", system_crash_reporter);
result.Set("manualUploads", allow_manual_uploads);
result.Set("crashes", std::move(crash_list));
result.Set("version", version_info::GetVersionNumber());
result.Set("os", base::SysInfo::OperatingSystemName() + " " +
base::SysInfo::OperatingSystemVersion());
result.Set("isGoogleAccount", is_internal);
FireWebUIListener(crash_reporter::kCrashesUIUpdateCrashList, result);
}
void CrashesDOMHandler::HandleRequestSingleCrashUpload(
const base::Value::List& args) {
// Only allow manual uploads if crash uploads aren’t disabled by policy.
if (!ChromeMetricsServiceAccessor::IsMetricsAndCrashReportingEnabled() &&
IsMetricsReportingPolicyManaged()) {
return;
}
std::string local_id = args[0].GetString();
upload_list_->RequestSingleUploadAsync(local_id);
}
} // namespace
///////////////////////////////////////////////////////////////////////////////
//
// CrashesUI
//
///////////////////////////////////////////////////////////////////////////////
CrashesUI::CrashesUI(content::WebUI* web_ui) : WebUIController(web_ui) {
web_ui->AddMessageHandler(std::make_unique<CrashesDOMHandler>());
// Set up the chrome://crashes/ source.
CreateAndAddCrashesUIHTMLSource(Profile::FromWebUI(web_ui));
}
// static
base::RefCountedMemory* CrashesUI::GetFaviconResourceBytes(
ui::ResourceScaleFactor scale_factor) {
return ui::ResourceBundle::GetSharedInstance().LoadDataResourceBytesForScale(
IDR_CRASH_SAD_FAVICON, scale_factor);
}