blob: 63e8efac0739329974690baf56bf8c4395e70505 [file] [log] [blame]
// Copyright 2014 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/extensions/extension_error_controller.h"
#include "base/memory/raw_ptr.h"
#include "base/memory/ref_counted.h"
#include "base/values.h"
#include "chrome/browser/extensions/extension_error_ui.h"
#include "chrome/browser/extensions/extension_service.h"
#include "chrome/browser/extensions/extension_service_test_base.h"
#include "chrome/browser/profiles/profile.h"
#include "chrome/test/base/testing_profile.h"
#include "components/sync_preferences/testing_pref_service_syncable.h"
#include "extensions/browser/blocklist_extension_prefs.h"
#include "extensions/browser/extension_prefs.h"
#include "extensions/browser/extension_registry.h"
#include "extensions/browser/pref_names.h"
#include "extensions/common/extension.h"
#include "extensions/common/extension_builder.h"
namespace extensions {
namespace {
// Create a mock for the UI component of the error alert that is shown for
// blocklisted extensions. This allows us to test which extensions the alert
// is showing, and also eliminates the UI component (since this is a unit
// test).
class MockExtensionErrorUI : public ExtensionErrorUI {
public:
explicit MockExtensionErrorUI(ExtensionErrorUI::Delegate* delegate);
~MockExtensionErrorUI() override;
// Wrappers around the similar methods in ExtensionErrorUI.
void CloseUI();
void Accept();
void Details();
ExtensionErrorUI::Delegate* delegate() { return delegate_; }
private:
// ExtensionErrorUI implementation.
bool ShowErrorInBubbleView() override;
void ShowExtensions() override;
void Close() override;
// Keep a copy of the delegate around for ourselves.
raw_ptr<ExtensionErrorUI::Delegate> delegate_;
};
// We use this as a slight hack to get the created Error UI, if any. We should
// only ever have one (since this is a single-profile test), and this avoids
// the need for any kind of accessor to the ErrorController from
// ExtensionService.
MockExtensionErrorUI* g_error_ui = nullptr;
MockExtensionErrorUI::MockExtensionErrorUI(ExtensionErrorUI::Delegate* delegate)
: delegate_(delegate) {
// We should never make more than one of these in a test.
DCHECK(!g_error_ui);
g_error_ui = this;
}
MockExtensionErrorUI::~MockExtensionErrorUI() {
g_error_ui = nullptr;
}
void MockExtensionErrorUI::CloseUI() {
delegate_->OnAlertClosed();
}
void MockExtensionErrorUI::Accept() {
delegate_->OnAlertAccept();
}
void MockExtensionErrorUI::Details() {
delegate_->OnAlertDetails();
}
bool MockExtensionErrorUI::ShowErrorInBubbleView() {
return true;
}
void MockExtensionErrorUI::ShowExtensions() {}
void MockExtensionErrorUI::Close() {
CloseUI();
}
ExtensionErrorUI* CreateMockUI(ExtensionErrorUI::Delegate* delegate) {
return new MockExtensionErrorUI(delegate);
}
// Builds and returns a simple extension.
scoped_refptr<const Extension> BuildExtension() {
return ExtensionBuilder()
.SetManifest(base::Value::Dict()
.Set("name", "My Wonderful Extension")
.Set("version", "0.1.1.0")
.Set("manifest_version", 2))
.Build();
}
} // namespace
class ExtensionErrorControllerUnitTest : public ExtensionServiceTestBase {
protected:
void SetUp() override;
// Add an extension to chrome, and mark it as blocklisted in the prefs.
testing::AssertionResult AddBlocklistedExtension(const Extension* extension);
// Set enterprise policy to block `extension`. Use nullptr to not block any
// extension.
void SetBlockExtensionPolicy(const Extension* extension);
// Return the ExtensionPrefs associated with the test.
ExtensionPrefs* GetPrefs();
};
void ExtensionErrorControllerUnitTest::SetUp() {
ExtensionServiceTestBase::SetUp();
// Make sure we use the mock UI instead of the real UI.
ExtensionErrorController::SetUICreateMethodForTesting(CreateMockUI);
// We don't want a first-run ExtensionService, since we ignore warnings
// for new profiles.
ExtensionServiceInitParams params;
params.is_first_run = false;
InitializeExtensionService(params);
}
testing::AssertionResult
ExtensionErrorControllerUnitTest::AddBlocklistedExtension(
const Extension* extension) {
blocklist_prefs::SetSafeBrowsingExtensionBlocklistState(
extension->id(), BitMapBlocklistState::BLOCKLISTED_MALWARE, GetPrefs());
service_->AddExtension(extension);
// Make sure the extension is added to the blocklisted set.
if (!ExtensionRegistry::Get(profile())->blocklisted_extensions().Contains(
extension->id())) {
return testing::AssertionFailure()
<< "Failed to add blocklisted extension.";
}
return testing::AssertionSuccess();
}
void ExtensionErrorControllerUnitTest::SetBlockExtensionPolicy(
const Extension* extension) {
base::Value::List block_list;
if (extension) {
block_list.Append(extension->id());
}
testing_pref_service()->SetManagedPref(pref_names::kInstallDenyList,
std::move(block_list));
}
ExtensionPrefs* ExtensionErrorControllerUnitTest::GetPrefs() {
return ExtensionPrefs::Get(profile());
}
// Test that closing the extension alert for blocklisted extensions counts
// as acknowledging them in the prefs.
TEST_F(ExtensionErrorControllerUnitTest, ClosingAcknowledgesBlocklisted) {
// Add a blocklisted extension.
scoped_refptr<const Extension> extension = BuildExtension();
ASSERT_TRUE(AddBlocklistedExtension(extension.get()));
service_->Init();
// Make sure that we created an error "ui" to warn about the blocklisted
// extension.
ASSERT_TRUE(g_error_ui);
ExtensionErrorUI::Delegate* delegate = g_error_ui->delegate();
ASSERT_TRUE(delegate);
// Make sure that the blocklisted extension is reported (and that no other
// extensions are).
const ExtensionSet& delegate_blocklisted_extensions =
delegate->GetBlocklistedExtensions();
EXPECT_EQ(1u, delegate_blocklisted_extensions.size());
EXPECT_TRUE(delegate_blocklisted_extensions.Contains(extension->id()));
// Close, and verify that the extension ids now acknowledged.
g_error_ui->CloseUI();
EXPECT_TRUE(GetPrefs()->IsBlocklistedExtensionAcknowledged(extension->id()));
// Verify we cleaned up after ourselves.
EXPECT_FALSE(g_error_ui);
}
// Test that clicking "accept" on the extension alert counts as acknowledging
// blocklisted extensions.
TEST_F(ExtensionErrorControllerUnitTest, AcceptingAcknowledgesBlocklisted) {
// Add a blocklisted extension.
scoped_refptr<const Extension> extension = BuildExtension();
ASSERT_TRUE(AddBlocklistedExtension(extension.get()));
service_->Init();
// Make sure that we created an error "ui" to warn about the blocklisted
// extension.
ASSERT_TRUE(g_error_ui);
// Accept, and verify that the extension ids now acknowledged.
g_error_ui->Accept();
EXPECT_TRUE(GetPrefs()->IsBlocklistedExtensionAcknowledged(extension->id()));
// Verify we cleaned up after ourselves.
EXPECT_FALSE(g_error_ui);
}
// Test that we don't warn for extensions which are blocklisted, but have
// already been acknowledged.
TEST_F(ExtensionErrorControllerUnitTest, DontWarnForAcknowledgedBlocklisted) {
scoped_refptr<const Extension> extension = BuildExtension();
ASSERT_TRUE(AddBlocklistedExtension(extension.get()));
GetPrefs()->AcknowledgeBlocklistedExtension(extension->id());
service_->Init();
// We should never have made an alert, because the extension should already
// be acknowledged.
ASSERT_FALSE(g_error_ui);
}
// Test there is no error ui if no extension is blocked by policy.
TEST_F(ExtensionErrorControllerUnitTest,
ExtensionIsNotBlockedByEnterprisePolicy) {
scoped_refptr<const Extension> extension = BuildExtension();
service_->Init();
service_->AddExtension(extension.get());
EXPECT_FALSE(g_error_ui);
}
// Test error ui is presented and acknowledged whe an extension is blocked by
// policy.
TEST_F(ExtensionErrorControllerUnitTest, ExtensionIsBlockedByEnterprisePolicy) {
scoped_refptr<const Extension> extension = BuildExtension();
service_->Init();
service_->AddExtension(extension.get());
SetBlockExtensionPolicy(extension.get());
ASSERT_TRUE(g_error_ui);
g_error_ui->Accept();
EXPECT_TRUE(GetPrefs()->IsBlocklistedExtensionAcknowledged(extension->id()));
EXPECT_FALSE(g_error_ui);
}
// Test the case that the error UI is accepted when we no longer need to show
// error for a blocked extension. It includes the case that the policy is
// updated or the extension is moved to the disabled list.
TEST_F(ExtensionErrorControllerUnitTest, ExtensionIsUnblockedBeforeUIAccepted) {
scoped_refptr<const Extension> extension = BuildExtension();
service_->Init();
service_->AddExtension(extension.get());
SetBlockExtensionPolicy(extension.get());
ASSERT_TRUE(g_error_ui);
// Reset extension policy
SetBlockExtensionPolicy(nullptr);
g_error_ui->Accept();
EXPECT_TRUE(GetPrefs()->IsBlocklistedExtensionAcknowledged(extension->id()));
EXPECT_FALSE(g_error_ui);
}
} // namespace extensions