blob: bf7d2c8686557a88dea527d5999321a8d1677c15 [file]
// Copyright 2023 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/api/user_scripts/user_scripts_apitest.h"
#include "base/feature_list.h"
#include "base/one_shot_event.h"
#include "base/test/bind.h"
#include "base/test/scoped_feature_list.h"
#include "base/test/scoped_run_loop_timeout.h"
#include "build/build_config.h"
#include "chrome/browser/extensions/extension_apitest.h"
#include "chrome/browser/extensions/extension_util.h"
#include "chrome/browser/extensions/user_scripts_test_util.h"
#include "chrome/browser/profiles/profile.h"
#include "content/public/browser/web_contents.h"
#include "content/public/test/browser_test.h"
#include "content/public/test/browser_test_utils.h"
#include "content/public/test/test_navigation_observer.h"
#include "extensions/browser/background_script_executor.h"
#include "extensions/browser/extension_prefs.h"
#include "extensions/browser/extension_registry.h"
#include "extensions/browser/extension_util.h"
#include "extensions/browser/renderer_startup_helper.h"
#include "extensions/browser/user_script_manager.h"
#include "extensions/buildflags/buildflags.h"
#include "extensions/common/extension_features.h"
#include "extensions/common/features/feature_developer_mode_only.h"
#include "extensions/common/user_scripts_allowed_state.h"
#include "extensions/common/utils/content_script_utils.h"
#include "extensions/test/extension_test_message_listener.h"
#include "extensions/test/result_catcher.h"
#include "extensions/test/test_extension_dir.h"
#include "net/dns/mock_host_resolver.h"
#include "testing/gtest/include/gtest/gtest.h"
static_assert(BUILDFLAG(ENABLE_EXTENSIONS_CORE));
namespace extensions {
void UserScriptsAPITest::SetUpOnMainThread() {
ExtensionApiTest::SetUpOnMainThread();
host_resolver()->AddRule("*", "127.0.0.1");
ASSERT_TRUE(StartEmbeddedTestServer());
}
void UserScriptsAPITest::OpenInCurrentTab(const GURL& url) {
content::WebContents* web_contents = GetActiveWebContents();
ASSERT_TRUE(web_contents);
// NavigateToURL() waits for the load to stop and verifies the navigation
// succeeded.
ASSERT_TRUE(NavigateToURL(web_contents, url));
EXPECT_EQ(url, web_contents->GetLastCommittedURL());
}
content::RenderFrameHost* UserScriptsAPITest::OpenInNewTab(const GURL& url) {
content::TestNavigationObserver nav_observer(url);
nav_observer.StartWatchingNewWebContents();
NavigateToURLInNewTab(url);
auto* web_contents = GetActiveWebContents();
EXPECT_TRUE(content::WaitForLoadStop(web_contents));
nav_observer.Wait();
EXPECT_TRUE(nav_observer.last_navigation_succeeded());
EXPECT_EQ(url, web_contents->GetLastCommittedURL());
return web_contents->GetPrimaryMainFrame();
}
content::EvalJsResult UserScriptsAPITest::GetInjectedElements(
content::RenderFrameHost* host) {
static constexpr char kGetInjectedScripts[] =
R"(const divs = document.body.getElementsByTagName('div');
JSON.stringify(Array.from(divs).map(div => div.id).sort());)";
return content::EvalJs(host, kGetInjectedScripts);
}
// Loads the extension and pauses in-between loading and running the tests to
// enable the userScripts API.
testing::AssertionResult UserScriptsAPITest::RunUserScriptsExtensionTestImpl(
const base::FilePath& extension_path,
bool allow_api) {
// Load the extension.
ExtensionTestMessageListener test_ready_listener(
"ready",
allow_api ? ReplyBehavior::kWillReply : ReplyBehavior::kWontReply);
ResultCatcher catcher;
const Extension* extension = LoadExtension(extension_path);
if (!extension) {
return testing::AssertionFailure() << "Failed to load extension";
}
if (allow_api) {
// Wait until extension tests are ready to run, then allow the
// userScripts API, then continue on with the API testing.
bool extension_ready = test_ready_listener.WaitUntilSatisfied();
if (!extension_ready) {
testing::AssertionFailure()
<< "extension did not signal that it was ready after loading";
}
user_scripts_test_util::SetUserScriptsAPIAllowed(profile(), extension->id(),
/*allowed=*/true);
test_ready_listener.Reply("");
}
// Observe each test result.
{
base::test::ScopedRunLoopTimeout timeout(
FROM_HERE, std::nullopt,
base::BindRepeating(
[](const base::FilePath& extension_path) {
return "GetNextResult timeout while "
"RunUserScriptsExtensionTest: " +
extension_path.MaybeAsASCII();
},
extension_path));
if (!catcher.GetNextResult()) {
return testing::AssertionFailure() << catcher.message();
}
}
return testing::AssertionSuccess();
;
}
testing::AssertionResult UserScriptsAPITest::RunUserScriptsExtensionTest(
const char* extension_sub_path) {
const base::FilePath& root_path = test_data_dir_;
base::FilePath extension_path = root_path.AppendASCII(extension_sub_path);
return RunUserScriptsExtensionTestImpl(extension_path, /*allow_api=*/true);
}
testing::AssertionResult
UserScriptsAPITest::RunUserScriptsExtensionTestNotAllowed(
const base::FilePath& extension_path) {
return RunUserScriptsExtensionTestImpl(extension_path, /*allow_api=*/false);
}
UserScriptsAPITest::UserScriptsAPITest() {
scoped_feature_list_.InitWithFeatures(
/*enabled_features=*/
{extensions_features::kApiUserScriptsMultipleWorlds,
extensions_features::kApiUserScriptsExecute},
/*disabled_features=*/{});
}
UserScriptsAPITest::~UserScriptsAPITest() = default;
IN_PROC_BROWSER_TEST_F(UserScriptsAPITest, GetUserScripts) {
ASSERT_TRUE(RunUserScriptsExtensionTest("user_scripts/get_scripts"))
<< message_;
}
// TODO(crbug.com/40935741, crbug.com/335421977): Flaky on Linux debug and on
// "Linux ChromiumOS MSan Tests".
#if (BUILDFLAG(IS_LINUX) && !defined(NDEBUG)) || \
(BUILDFLAG(IS_CHROMEOS) && defined(MEMORY_SANITIZER))
#define MAYBE_RegisterUserScripts DISABLED_RegisterUserScripts
#else
#define MAYBE_RegisterUserScripts RegisterUserScripts
#endif
IN_PROC_BROWSER_TEST_F(UserScriptsAPITest, MAYBE_RegisterUserScripts) {
ASSERT_TRUE(RunUserScriptsExtensionTest("user_scripts/register")) << message_;
}
IN_PROC_BROWSER_TEST_F(UserScriptsAPITest, UnregisterUserScripts) {
ASSERT_TRUE(RunUserScriptsExtensionTest("user_scripts/unregister"))
<< message_;
}
IN_PROC_BROWSER_TEST_F(UserScriptsAPITest, UpdateUserScripts) {
ASSERT_TRUE(RunUserScriptsExtensionTest("user_scripts/update")) << message_;
}
IN_PROC_BROWSER_TEST_F(UserScriptsAPITest, ExecuteUserScripts) {
ASSERT_TRUE(RunUserScriptsExtensionTest("user_scripts/execute")) << message_;
}
IN_PROC_BROWSER_TEST_F(UserScriptsAPITest, ExecuteUserScripts_Subframes) {
// Open up two tabs, each with cross-site iframes, one at a.com and one at
// d.com. In both cases, the cross-site iframes point to b.com and c.com.
OpenInCurrentTab(
embedded_test_server()->GetURL("a.com", "/iframe_cross_site.html"));
OpenInNewTab(
embedded_test_server()->GetURL("d.com", "/iframe_cross_site.html"));
ASSERT_TRUE(
RunUserScriptsExtensionTest("user_scripts/execute_with_subframes"))
<< message_;
}
IN_PROC_BROWSER_TEST_F(UserScriptsAPITest, ExecuteUserScripts_SizeLimit) {
auto single_scripts_limit_reset =
script_parsing::CreateScopedMaxScriptLengthForTesting(700u);
ASSERT_TRUE(RunUserScriptsExtensionTest("user_scripts/execute_size_limit"))
<< message_;
}
// TODO(crbug.com/335421977): Flaky on "Linux ChromiumOS MSan Tests".
#if BUILDFLAG(IS_CHROMEOS) && defined(MEMORY_SANITIZER)
#define MAYBE_ConfigureWorld DISABLED_ConfigureWorld
#else
#define MAYBE_ConfigureWorld ConfigureWorld
#endif
IN_PROC_BROWSER_TEST_F(UserScriptsAPITest, MAYBE_ConfigureWorld) {
ASSERT_TRUE(RunUserScriptsExtensionTest("user_scripts/configure_world"))
<< message_;
}
IN_PROC_BROWSER_TEST_F(UserScriptsAPITest,
UserScriptInjectionOrderIsAlphabetical) {
ASSERT_TRUE(RunUserScriptsExtensionTest("user_scripts/injection_order"))
<< message_;
}
IN_PROC_BROWSER_TEST_F(UserScriptsAPITest, GetAndRemoveWorlds) {
ASSERT_TRUE(RunUserScriptsExtensionTest("user_scripts/get_and_remove_worlds"))
<< message_;
}
// Tests that registered user scripts are disabled when the userScripts API is
// not allowed and are re-enabled if when the API is allowed again.
IN_PROC_BROWSER_TEST_F(UserScriptsAPITest,
UserScriptsAreDisabledWhenAPIIsNotAllowed) {
const Extension* extension =
LoadExtension(test_data_dir_.AppendASCII("user_scripts/allowed_tests"));
ASSERT_TRUE(extension);
user_scripts_test_util::SetUserScriptsAPIAllowed(profile(), extension->id(),
/*allowed=*/true);
// Register a user script and a content script.
EXPECT_EQ("success",
BackgroundScriptExecutor::ExecuteScript(
profile(), extension->id(), "registerUserScripts();",
BackgroundScriptExecutor::ResultCapture::kSendScriptResult));
EXPECT_EQ("success",
BackgroundScriptExecutor::ExecuteScript(
profile(), extension->id(), "registerContentScript();",
BackgroundScriptExecutor::ResultCapture::kSendScriptResult));
const GURL url =
embedded_test_server()->GetURL("example.com", "/simple.html");
// Open a new tab.
content::RenderFrameHost* new_tab = OpenInNewTab(url);
// Since the userScript API is available (as part of this test suite's setup),
// both the user script and the content script should inject.
EXPECT_EQ(R"(["content-script","user-script-code","user-script-file"])",
GetInjectedElements(new_tab));
user_scripts_test_util::SetUserScriptsAPIAllowed(profile(), extension->id(),
/*allowed=*/false);
// Open a new tab. Now, user scripts should be disabled. However, content
// scripts should still inject.
new_tab = OpenInNewTab(url);
EXPECT_EQ(R"(["content-script"])", GetInjectedElements(new_tab));
user_scripts_test_util::SetUserScriptsAPIAllowed(profile(), extension->id(),
/*allowed=*/true);
// Open a new tab. The user script should inject again.
new_tab = OpenInNewTab(url);
EXPECT_EQ(R"(["content-script","user-script-code","user-script-file"])",
GetInjectedElements(new_tab));
}
// Tests that unregisterContentScripts unregisters only content scripts and
// not user scripts.
IN_PROC_BROWSER_TEST_F(UserScriptsAPITest,
ScriptingAPIDoesNotAffectUserScripts) {
ASSERT_TRUE(RunUserScriptsExtensionTest("scripting/dynamic_user_scripts"))
<< message_;
}
#if BUILDFLAG(ENABLE_EXTENSIONS)
// Base test fixture for tests spanning multiple sessions where a custom arg
// is set before the test is run.
// TODO(crbug.com/40200835): PRE_ tests are not supported on Android and all
// these tests require a PRE_ step.
class PersistentUserScriptsAPITest : public UserScriptsAPITest {
public:
PersistentUserScriptsAPITest() = default;
// UserScriptsAPITest override.
void SetUp() override {
// Initialize the listener object here before calling SetUp. This avoids a
// race condition where the extension loads (as part of browser startup) and
// sends a message before a message listener in C++ has been initialized.
listener_ = std::make_unique<ExtensionTestMessageListener>(
"ready", ReplyBehavior::kWillReply);
UserScriptsAPITest::SetUp();
}
// Reset listener before the browser gets torn down.
void TearDownOnMainThread() override {
listener_.reset();
UserScriptsAPITest::TearDownOnMainThread();
}
protected:
// Used to wait for results from extension tests. This is initialized before
// the test is run which avoids a race condition where the extension is loaded
// (as part of startup) and finishes its tests before the ResultCatcher is
// created.
ResultCatcher result_catcher_;
// Used to wait for the extension to load and send a ready message so the test
// can reply which the extension waits for to start its testing functions.
// This ensures that the testing functions will run after the browser has
// finished initializing.
std::unique_ptr<ExtensionTestMessageListener> listener_;
};
// Tests that registered user scripts persist across sessions. The test is run
// across three sessions.
// TODO(crbug.com/40200835): PRE_ tests are not supported on Android.
IN_PROC_BROWSER_TEST_F(PersistentUserScriptsAPITest,
PRE_PRE_PersistentScripts) {
const Extension* extension = LoadExtension(
test_data_dir_.AppendASCII("user_scripts/persistent_scripts"));
ASSERT_TRUE(extension);
ASSERT_TRUE(listener_->WaitUntilSatisfied());
user_scripts_test_util::SetUserScriptsAPIAllowed(profile(), extension->id(),
/*allowed=*/true);
listener_->Reply(
testing::UnitTest::GetInstance()->current_test_info()->name());
EXPECT_TRUE(result_catcher_.GetNextResult()) << result_catcher_.message();
}
IN_PROC_BROWSER_TEST_F(PersistentUserScriptsAPITest, PRE_PersistentScripts) {
ASSERT_TRUE(listener_->WaitUntilSatisfied());
listener_->Reply(
testing::UnitTest::GetInstance()->current_test_info()->name());
EXPECT_TRUE(result_catcher_.GetNextResult()) << result_catcher_.message();
}
IN_PROC_BROWSER_TEST_F(PersistentUserScriptsAPITest, PersistentScripts) {
ASSERT_TRUE(listener_->WaitUntilSatisfied());
listener_->Reply(
testing::UnitTest::GetInstance()->current_test_info()->name());
EXPECT_TRUE(result_catcher_.GetNextResult()) << result_catcher_.message();
}
// Tests that the world configuration of a registered user script is persisted
// across sessions. The test is run across three sessions.
// TODO(crbug.com/40200835): PRE_ tests are not supported on Android.
IN_PROC_BROWSER_TEST_F(PersistentUserScriptsAPITest,
PRE_PRE_PersistentWorldConfiguration) {
const Extension* extension = LoadExtension(
test_data_dir_.AppendASCII("user_scripts/persistent_configure_world"));
ASSERT_TRUE(extension);
ASSERT_TRUE(listener_->WaitUntilSatisfied());
user_scripts_test_util::SetUserScriptsAPIAllowed(profile(), extension->id(),
/*allowed=*/true);
listener_->Reply(
testing::UnitTest::GetInstance()->current_test_info()->name());
EXPECT_TRUE(result_catcher_.GetNextResult()) << result_catcher_.message();
}
IN_PROC_BROWSER_TEST_F(PersistentUserScriptsAPITest,
PRE_PersistentWorldConfiguration) {
ASSERT_TRUE(listener_->WaitUntilSatisfied());
listener_->Reply(
testing::UnitTest::GetInstance()->current_test_info()->name());
EXPECT_TRUE(result_catcher_.GetNextResult()) << result_catcher_.message();
}
IN_PROC_BROWSER_TEST_F(PersistentUserScriptsAPITest,
PersistentWorldConfiguration) {
ASSERT_TRUE(listener_->WaitUntilSatisfied());
listener_->Reply(
testing::UnitTest::GetInstance()->current_test_info()->name());
EXPECT_TRUE(result_catcher_.GetNextResult()) << result_catcher_.message();
}
#endif // BUILDFLAG(ENABLE_EXTENSIONS)
class UserScriptsAPITestWithoutAPIAllowed : public UserScriptsAPITest {
public:
UserScriptsAPITestWithoutAPIAllowed() = default;
// UserScriptsAPITest override.
void SetUp() override {
// Initialize the listener object here before calling SetUp. This avoids a
// race condition where the extension loads (as part of browser startup) and
// sends a message before a message listener in C++ has been initialized.
background_started_listener_ =
std::make_unique<ExtensionTestMessageListener>("started");
UserScriptsAPITest::SetUp();
}
// Reset listener before the browser gets torn down.
void TearDown() override {
background_started_listener_.reset();
UserScriptsAPITest::TearDown();
}
protected:
std::unique_ptr<ExtensionTestMessageListener> background_started_listener_;
};
// Verifies that the `chrome.userScripts` API is undefined if the API is not
// allowed yet.
IN_PROC_BROWSER_TEST_F(UserScriptsAPITestWithoutAPIAllowed,
UserScriptsAPIIsUndefinedWithoutAPIAllowed) {
static constexpr char kManifest[] =
R"({
"name": "user scripts",
"manifest_version": 3,
"version": "0.1",
"background": {"service_worker": "background.js"},
"permissions": ["userScripts"]
})";
static constexpr char kBackgroundJs[] =
R"(chrome.test.runTests([
function userScriptsIsUndefined() {
chrome.test.assertTrue(chrome.userScripts === undefined);
chrome.test.succeed();
},
]);)";
TestExtensionDir test_dir;
test_dir.WriteManifest(kManifest);
test_dir.WriteFile(FILE_PATH_LITERAL("background.js"), kBackgroundJs);
ASSERT_TRUE(RunUserScriptsExtensionTestNotAllowed(test_dir.UnpackedPath()))
<< message_;
}
// Tests that registered user scripts are properly ignored when loading
// stored dynamic scripts if the API is not allowed.
// TODO(crbug.com/441364550): Flaky on desktop Android.
#if BUILDFLAG(IS_ANDROID)
#define MAYBE_PRE_UserScriptsDisabledOnStartupIfAPINotAllowed \
DISABLED_PRE_UserScriptsDisabledOnStartupIfAPINotAllowed
#else
#define MAYBE_PRE_UserScriptsDisabledOnStartupIfAPINotAllowed \
PRE_UserScriptsDisabledOnStartupIfAPINotAllowed
#endif
IN_PROC_BROWSER_TEST_F(UserScriptsAPITestWithoutAPIAllowed,
MAYBE_PRE_UserScriptsDisabledOnStartupIfAPINotAllowed) {
// Load an extension and register user scripts and a dynamic content script.
const Extension* extension =
LoadExtension(test_data_dir_.AppendASCII("user_scripts/allowed_tests"));
ASSERT_TRUE(extension);
ASSERT_TRUE(background_started_listener_->WaitUntilSatisfied());
user_scripts_test_util::SetUserScriptsAPIAllowed(profile(), extension->id(),
/*allowed=*/true);
EXPECT_EQ("success",
BackgroundScriptExecutor::ExecuteScript(
profile(), extension->id(), "registerUserScripts();",
BackgroundScriptExecutor::ResultCapture::kSendScriptResult));
EXPECT_EQ("success",
BackgroundScriptExecutor::ExecuteScript(
profile(), extension->id(), "registerContentScript();",
BackgroundScriptExecutor::ResultCapture::kSendScriptResult));
const GURL url =
embedded_test_server()->GetURL("example.com", "/simple.html");
// To start, all scripts should inject.
content::RenderFrameHost* new_tab = OpenInNewTab(url);
EXPECT_EQ(R"(["content-script","user-script-code","user-script-file"])",
GetInjectedElements(new_tab));
// Disallow userScript API, and then re-open the browser...
user_scripts_test_util::SetUserScriptsAPIAllowed(profile(), extension->id(),
/*allowed=*/false);
}
// TODO(crbug.com/441364550): Flaky on desktop Android.
#if BUILDFLAG(IS_ANDROID)
#define MAYBE_UserScriptsDisabledOnStartupIfAPINotAllowed \
DISABLED_UserScriptsDisabledOnStartupIfAPINotAllowed
#else
#define MAYBE_UserScriptsDisabledOnStartupIfAPINotAllowed \
UserScriptsDisabledOnStartupIfAPINotAllowed
#endif
IN_PROC_BROWSER_TEST_F(UserScriptsAPITestWithoutAPIAllowed,
MAYBE_UserScriptsDisabledOnStartupIfAPINotAllowed) {
// Wait until the extension loads so we can get it's ID.
ASSERT_TRUE(background_started_listener_->WaitUntilSatisfied());
// Find the extension's ID so we can make some assertions.
ExtensionId extension_id;
for (const auto& extension :
ExtensionRegistry::Get(profile())->enabled_extensions()) {
if (extension->name() == "Test") {
extension_id = extension->id();
}
}
ASSERT_TRUE(!extension_id.empty());
// userScripts should remain disallowed after browser restart.
EXPECT_FALSE(GetCurrentUserScriptAllowedState(
util::GetBrowserContextId(profile()), extension_id)
.value_or(false));
const GURL url =
embedded_test_server()->GetURL("example.com", "/simple.html");
// And, to start, only the content script should inject.
content::RenderFrameHost* new_tab = OpenInNewTab(url);
EXPECT_EQ(R"(["content-script"])", GetInjectedElements(new_tab));
user_scripts_test_util::SetUserScriptsAPIAllowed(profile(), extension_id,
/*allowed=*/true);
// All scripts should once again inject.
new_tab = OpenInNewTab(url);
EXPECT_EQ(R"(["content-script","user-script-code","user-script-file"])",
GetInjectedElements(new_tab));
}
} // namespace extensions