blob: d4d321c5bf2cbcdbce82f2dd0c13a2d5ece95f8f [file] [log] [blame]
// Copyright 2024 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "base/features.h"
#include "base/files/file.h"
#include "base/path_service.h"
#include "chrome/browser/browser_features.h"
#include "chrome/browser/devtools/devtools_window.h"
#include "chrome/browser/devtools/protocol/devtools_protocol_test_support.h"
#include "chrome/browser/extensions/scoped_test_mv2_enabler.h"
#include "chrome/browser/ui/browser.h"
#include "chrome/common/chrome_paths.h"
#include "chrome/common/chrome_switches.h"
#include "chrome/test/base/ui_test_utils.h"
#include "content/public/browser/browser_context.h"
#include "content/public/browser/devtools_agent_host.h"
#include "content/public/test/browser_test.h"
#include "extensions/browser/api/storage/storage_area_namespace.h"
#include "extensions/browser/api/storage/storage_frontend.h"
#include "extensions/browser/extension_registrar.h"
#include "extensions/browser/extension_registry.h"
#include "extensions/common/extension.h"
#include "extensions/common/extension_builder.h"
#include "extensions/common/extension_features.h"
#include "extensions/common/extension_id.h"
#include "extensions/common/extension_set.h"
#include "extensions/common/mojom/manifest.mojom-shared.h"
#include "extensions/test/extension_background_page_waiter.h"
#include "extensions/test/extension_test_message_listener.h"
namespace {
class DevToolsExtensionsProtocolTest : public DevToolsProtocolTestBase {
public:
void SetUpOnMainThread() override {
DevToolsProtocolTestBase::SetUpOnMainThread();
AttachToBrowserTarget();
}
void SetUpCommandLine(base::CommandLine* command_line) override {
DevToolsProtocolTestBase::SetUpCommandLine(command_line);
command_line->RemoveSwitch(::switches::kEnableUnsafeExtensionDebugging);
}
const base::Value::Dict* SendLoadUnpackedCommand(const std::string& path) {
base::FilePath extension_path =
base::PathService::CheckedGet(chrome::DIR_TEST_DATA)
.AppendASCII("devtools")
.AppendASCII("extensions")
.AppendASCII(path);
base::Value::Dict params;
params.Set("path", extension_path.AsUTF8Unsafe());
return SendCommandSync("Extensions.loadUnpacked", std::move(params));
}
const base::Value::Dict* SendStorageCommand(
const std::string& command,
const extensions::Extension* extension,
base::Value::Dict extra_params) {
base::Value::Dict storage_params;
storage_params.Set("id", extension->id());
storage_params.Set("storageArea", "local");
storage_params.Merge(std::move(extra_params));
const base::Value::Dict* get_result =
SendCommandSync(command, std::move(storage_params));
return get_result;
}
private:
// TODO(https://crbug.com/40804030): Remove this when updated to use MV3.
extensions::ScopedTestMV2Enabler mv2_enabler_;
};
class DevToolsExtensionsProtocolWithUnsafeDebuggingTest
: public DevToolsExtensionsProtocolTest {
public:
DevToolsExtensionsProtocolWithUnsafeDebuggingTest() {
scoped_feature_list_.InitAndEnableFeature(
extensions_features::kExtensionDisableUnsupportedDeveloper);
}
void SetUpCommandLine(base::CommandLine* command_line) override {
DevToolsExtensionsProtocolTest::SetUpCommandLine(command_line);
command_line->AppendSwitch(::switches::kEnableUnsafeExtensionDebugging);
}
private:
base::test::ScopedFeatureList scoped_feature_list_;
};
IN_PROC_BROWSER_TEST_F(DevToolsExtensionsProtocolTest, CannotInstallExtension) {
ASSERT_FALSE(SendLoadUnpackedCommand("simple_background_page"));
}
IN_PROC_BROWSER_TEST_F(DevToolsExtensionsProtocolTest,
CannotUninstallExtension) {
auto extension =
extensions::ExtensionBuilder("unpacked")
.SetLocation(extensions::mojom::ManifestLocation::kUnpacked)
.Build();
extensions::ExtensionRegistrar::Get(browser()->profile())
->AddExtension(extension.get());
std::string id = extension.get()->id();
extensions::ExtensionRegistry* registry =
extensions::ExtensionRegistry::Get(browser()->profile());
const extensions::Extension* extension_before =
registry->GetInstalledExtension(id);
ASSERT_TRUE(extension_before);
base::Value::Dict params;
params.Set("id", id);
const base::Value::Dict* uninstall_result =
SendCommandSync("Extensions.uninstall", std::move(params));
ASSERT_FALSE(uninstall_result);
const extensions::Extension* extension_after =
registry->GetInstalledExtension(id);
ASSERT_TRUE(extension_after);
}
IN_PROC_BROWSER_TEST_F(DevToolsExtensionsProtocolWithUnsafeDebuggingTest,
CanInstallExtension) {
const base::Value::Dict* result =
SendLoadUnpackedCommand("simple_background_page");
ASSERT_TRUE(result);
ASSERT_TRUE(result->FindString("id"));
extensions::ExtensionRegistry* registry =
extensions::ExtensionRegistry::Get(browser()->profile());
const extensions::Extension* extension = registry->GetExtensionById(
*result->FindString("id"), extensions::ExtensionRegistry::ENABLED);
ASSERT_TRUE(extension);
ASSERT_EQ(extension->id(), *result->FindString("id"));
ASSERT_EQ(extension->location(),
extensions::mojom::ManifestLocation::kUnpacked);
}
IN_PROC_BROWSER_TEST_F(DevToolsExtensionsProtocolWithUnsafeDebuggingTest,
ThrowsOnWrongPath) {
const base::Value::Dict* result = SendLoadUnpackedCommand("non-existent");
ASSERT_FALSE(result);
}
IN_PROC_BROWSER_TEST_F(DevToolsExtensionsProtocolWithUnsafeDebuggingTest,
CanUninstallExtension) {
const base::Value::Dict* install_result =
SendLoadUnpackedCommand("simple_background_page");
std::string id = *install_result->FindString("id");
extensions::ExtensionRegistry* registry =
extensions::ExtensionRegistry::Get(browser()->profile());
const extensions::Extension* extension_before =
registry->GetInstalledExtension(id);
ASSERT_TRUE(extension_before);
base::Value::Dict params;
params.Set("id", id);
const base::Value::Dict* uninstall_result =
SendCommandSync("Extensions.uninstall", std::move(params));
ASSERT_TRUE(uninstall_result);
const extensions::Extension* extension_after =
registry->GetInstalledExtension(id);
ASSERT_FALSE(extension_after);
}
IN_PROC_BROWSER_TEST_F(DevToolsExtensionsProtocolWithUnsafeDebuggingTest,
CannotUninstallNonUnpackedExtension) {
auto extension =
extensions::ExtensionBuilder("unpacked")
.SetLocation(extensions::mojom::ManifestLocation::kComponent)
.Build();
extensions::ExtensionRegistrar::Get(browser()->profile())
->AddExtension(extension.get());
std::string id = extension.get()->id();
extensions::ExtensionRegistry* registry =
extensions::ExtensionRegistry::Get(browser()->profile());
const extensions::Extension* extension_before =
registry->GetInstalledExtension(id);
ASSERT_TRUE(extension_before);
base::Value::Dict params;
params.Set("id", id);
const base::Value::Dict* uninstall_result =
SendCommandSync("Extensions.uninstall", std::move(params));
ASSERT_FALSE(uninstall_result);
const extensions::Extension* extension_after =
registry->GetInstalledExtension(id);
ASSERT_TRUE(extension_after);
}
IN_PROC_BROWSER_TEST_F(DevToolsExtensionsProtocolWithUnsafeDebuggingTest,
FailsToUninstallNonexistentExtension) {
extensions::ExtensionRegistry* registry =
extensions::ExtensionRegistry::Get(browser()->profile());
std::string id = "non-existent-id";
const extensions::Extension* extension = registry->GetInstalledExtension(id);
ASSERT_FALSE(extension);
base::Value::Dict params;
params.Set("id", id);
const base::Value::Dict* uninstallResult =
SendCommandSync("Extensions.uninstall", std::move(params));
ASSERT_FALSE(uninstallResult);
const extensions::Extension* extensionAfter =
registry->GetInstalledExtension(id);
ASSERT_FALSE(extensionAfter);
}
// Returns the `DevToolsAgentHost` associated with an extension's service
// worker if available.
scoped_refptr<content::DevToolsAgentHost> FindExtensionHost(
const std::string& id) {
for (auto& host : content::DevToolsAgentHost::GetOrCreateAll()) {
if (host->GetType() == content::DevToolsAgentHost::kTypeServiceWorker &&
host->GetURL().host() == id) {
return host;
}
}
return nullptr;
}
// Returns the `DevToolsAgentHost` associated with an extension page if
// available.
scoped_refptr<content::DevToolsAgentHost> FindBackgroundPageHost(
const std::string& path) {
for (auto& host : content::DevToolsAgentHost::GetOrCreateAll()) {
if (host->GetType() == "background_page" && host->GetURL().path() == path) {
return host;
}
}
return nullptr;
}
// Returns the `DevToolsAgentHost` associated with an extension page if
// available.
scoped_refptr<content::DevToolsAgentHost> FindPageHost(
const std::string& path) {
for (auto& host : content::DevToolsAgentHost::GetOrCreateAll()) {
if (host->GetType() == content::DevToolsAgentHost::kTypePage &&
host->GetURL().path() == path) {
return host;
}
}
return nullptr;
}
IN_PROC_BROWSER_TEST_F(DevToolsExtensionsProtocolWithUnsafeDebuggingTest,
CanGetStorageValues) {
ExtensionTestMessageListener activated_listener("WORKER_ACTIVATED");
const base::Value::Dict* load_result =
SendLoadUnpackedCommand("service_worker");
ASSERT_TRUE(load_result);
extensions::ExtensionRegistry* registry =
extensions::ExtensionRegistry::Get(browser()->profile());
const extensions::Extension* extension = registry->GetExtensionById(
*load_result->FindString("id"), extensions::ExtensionRegistry::ENABLED);
ASSERT_TRUE(extension);
// Ensure service worker has had time to initialize.
EXPECT_TRUE(activated_listener.WaitUntilSatisfied());
// Access to storage commands is only allowed from a target associated with
// the extension. Attach to the extension service worker to be able to test
// the method.
DetachProtocolClient();
agent_host_ = FindExtensionHost(extension->id());
agent_host_->AttachClient(this);
// Set some dummy values in storage.
ASSERT_TRUE(SendStorageCommand(
"Extensions.setStorageItems", extension,
base::Value::Dict().Set("values", base::Value::Dict()
.Set("foo", "bar")
.Set("other", "value")
.Set("remove-on-clear", "value"))));
// Check only the requested keys are returned.
const base::Value::Dict* get_result = SendStorageCommand(
"Extensions.getStorageItems", extension,
base::Value::Dict().Set("keys", base::Value::List().Append("foo")));
ASSERT_TRUE(get_result);
ASSERT_EQ(*get_result->FindDict("data")->FindString("foo"), "bar");
ASSERT_FALSE(get_result->FindDict("data")->contains("other"));
// Remove the `foo` key.
ASSERT_TRUE(SendStorageCommand(
"Extensions.removeStorageItems", extension,
base::Value::Dict().Set("keys", base::Value::List().Append("foo"))));
// Check the `foo` key no longer exists.
const base::Value::Dict* get_result_2 = SendStorageCommand(
"Extensions.getStorageItems", extension,
base::Value::Dict().Set("keys", base::Value::List().Append("foo")));
ASSERT_TRUE(get_result_2);
ASSERT_FALSE(get_result_2->FindDict("data")->contains("foo"));
// Clear the storage area.
ASSERT_TRUE(SendStorageCommand("Extensions.clearStorageItems", extension,
base::Value::Dict()));
// Check the `remove-on-clear` key no longer exists.
const base::Value::Dict* get_result_3 = SendStorageCommand(
"Extensions.getStorageItems", extension,
base::Value::Dict().Set("keys",
base::Value::List().Append("remove-on-clear")));
ASSERT_TRUE(get_result_3);
ASSERT_FALSE(get_result_3->FindDict("data")->contains("remove-on-clear"));
}
IN_PROC_BROWSER_TEST_F(DevToolsExtensionsProtocolWithUnsafeDebuggingTest,
CanGetStorageValuesBackgroundPage) {
const base::Value::Dict* load_result =
SendLoadUnpackedCommand("background_page_storage_access");
ASSERT_TRUE(load_result);
extensions::ExtensionRegistry* registry =
extensions::ExtensionRegistry::Get(browser()->profile());
const extensions::Extension* extension = registry->GetExtensionById(
*load_result->FindString("id"), extensions::ExtensionRegistry::ENABLED);
ASSERT_TRUE(extension);
DetachProtocolClient();
extensions::ExtensionBackgroundPageWaiter(browser()->profile(), *extension)
.WaitForBackgroundOpen();
agent_host_ = FindBackgroundPageHost("/_generated_background_page.html");
agent_host_->AttachClient(this);
ASSERT_TRUE(SendStorageCommand("Extensions.getStorageItems", extension,
base::Value::Dict()));
}
IN_PROC_BROWSER_TEST_F(DevToolsExtensionsProtocolWithUnsafeDebuggingTest,
CanGetStorageValuesContentScript) {
const base::Value::Dict* load_result =
SendLoadUnpackedCommand("simple_content_script");
ASSERT_TRUE(load_result);
extensions::ExtensionRegistry* registry =
extensions::ExtensionRegistry::Get(browser()->profile());
const extensions::Extension* extension = registry->GetExtensionById(
*load_result->FindString("id"), extensions::ExtensionRegistry::ENABLED);
ASSERT_TRUE(extension);
ASSERT_TRUE(embedded_test_server()->Start());
GURL url =
embedded_test_server()->GetURL("/devtools/page_with_content_script.html");
ASSERT_TRUE(ui_test_utils::NavigateToURL(browser(), url));
DetachProtocolClient();
agent_host_ = FindPageHost("/devtools/page_with_content_script.html");
agent_host_->AttachClient(this);
ASSERT_TRUE(SendStorageCommand("Extensions.getStorageItems", extension,
base::Value::Dict()));
}
IN_PROC_BROWSER_TEST_F(DevToolsExtensionsProtocolWithUnsafeDebuggingTest,
CannotGetStorageValuesWithoutContentScript) {
// Load an extension with no associated content scripts.
const base::Value::Dict* load_result =
SendLoadUnpackedCommand("service_worker");
ASSERT_TRUE(load_result);
extensions::ExtensionRegistry* registry =
extensions::ExtensionRegistry::Get(browser()->profile());
const extensions::Extension* extension = registry->GetExtensionById(
*load_result->FindString("id"), extensions::ExtensionRegistry::ENABLED);
ASSERT_TRUE(extension);
ASSERT_TRUE(embedded_test_server()->Start());
GURL url =
embedded_test_server()->GetURL("/devtools/page_with_content_script.html");
ASSERT_TRUE(ui_test_utils::NavigateToURL(browser(), url));
DetachProtocolClient();
agent_host_ = FindPageHost("/devtools/page_with_content_script.html");
agent_host_->AttachClient(this);
const base::Value::Dict* get_result = SendStorageCommand(
"Extensions.getStorageItems", extension, base::Value::Dict());
// Command should fail as extension has not injected content script.
EXPECT_FALSE(get_result);
ASSERT_EQ(*error()->FindString("message"), "Extension not found.");
}
// Test to ensure that the target associated with an extension service worker
// cannot access data from the storage associated with another extension.
IN_PROC_BROWSER_TEST_F(DevToolsExtensionsProtocolWithUnsafeDebuggingTest,
CannotGetStorageValuesUnrelatedTarget) {
ExtensionTestMessageListener activated_listener("WORKER_ACTIVATED");
const base::Value::Dict* load_result =
SendLoadUnpackedCommand("service_worker");
ASSERT_TRUE(load_result);
const std::string first_extension_id = *load_result->FindString("id");
// Ensure service worker has had time to initialize.
EXPECT_TRUE(activated_listener.WaitUntilSatisfied());
// Load a second extension.
load_result = SendLoadUnpackedCommand("simple_background_page");
ASSERT_TRUE(load_result);
const std::string second_extension_id = *load_result->FindString("id");
// Attach to first extension.
DetachProtocolClient();
agent_host_ = FindExtensionHost(first_extension_id);
agent_host_->AttachClient(this);
// Try to load data from the second extension from a context associated with
// the first extension. This should be blocked.
base::Value::Dict storage_params;
storage_params.Set("id", second_extension_id);
storage_params.Set("storageArea", "local");
const base::Value::Dict* get_result =
SendCommandSync("Extensions.getStorageItems", std::move(storage_params));
// Command should fail as target does not have access.
EXPECT_FALSE(get_result);
ASSERT_EQ(*error()->FindString("message"), "Extension not found.");
}
} // namespace