blob: 353366eb4507751e24656629c465232043d4a3d7 [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 "base/memory/ref_counted.h"
#include "base/memory/scoped_refptr.h"
#include "base/path_service.h"
#include "base/strings/stringprintf.h"
#include "build/build_config.h"
#include "chrome/browser/devtools/devtools_window.h"
#include "chrome/browser/devtools/devtools_window_testing.h"
#include "chrome/browser/extensions/api/developer_private/developer_private_api.h"
#include "chrome/browser/extensions/extension_apitest.h"
#include "chrome/browser/extensions/extension_tab_util.h"
#include "chrome/browser/profiles/profile.h"
#include "chrome/common/chrome_paths.h"
#include "content/public/browser/render_frame_host.h"
#include "content/public/browser/render_process_host.h"
#include "content/public/test/browser_test.h"
#include "content/public/test/service_worker_test_helpers.h"
#include "extensions/browser/api_test_utils.h"
#include "extensions/browser/app_window/app_window.h"
#include "extensions/browser/app_window/app_window_registry.h"
#include "extensions/browser/browsertest_util.h"
#include "extensions/browser/extension_host_test_helper.h"
#include "extensions/browser/offscreen_document_host.h"
#include "extensions/browser/process_manager.h"
#include "extensions/common/manifest_handlers/background_info.h"
#include "extensions/common/mojom/view_type.mojom.h"
#include "extensions/test/result_catcher.h"
#include "extensions/test/test_extension_dir.h"
#include "testing/gtest/include/gtest/gtest.h"
namespace extensions {
class DeveloperPrivateApiTest : public ExtensionApiTest {
protected:
std::unique_ptr<api::developer_private::ExtensionInfo> GetExtensionInfo(
const Extension& extension) {
auto get_info_function =
base::MakeRefCounted<api::DeveloperPrivateGetExtensionInfoFunction>();
absl::optional<base::Value> result =
api_test_utils::RunFunctionAndReturnSingleResult(
get_info_function.get(),
content::JsReplace(R"([$1])", extension.id()), profile());
if (!result) {
ADD_FAILURE() << "No result back when getting extension info";
return nullptr;
}
std::unique_ptr<api::developer_private::ExtensionInfo> info =
api::developer_private::ExtensionInfo::FromValueDeprecated(*result);
if (!info)
ADD_FAILURE() << "Problem creating ExtensionInfo from result data";
return info;
}
};
IN_PROC_BROWSER_TEST_F(DeveloperPrivateApiTest, Basics) {
// Load up some extensions so that we can query their info and adjust their
// setings in the API test.
base::FilePath base_dir = test_data_dir_.AppendASCII("developer");
EXPECT_TRUE(LoadExtension(base_dir.AppendASCII("hosted_app")));
EXPECT_TRUE(InstallExtension(base_dir.AppendASCII("packaged_app"), 1,
mojom::ManifestLocation::kInternal));
LoadExtension(base_dir.AppendASCII("simple_extension"));
ASSERT_TRUE(RunExtensionTest("developer/test",
{.launch_as_platform_app = true},
{.load_as_component = true}));
}
// Tests opening the developer tools for an app window.
IN_PROC_BROWSER_TEST_F(DeveloperPrivateApiTest, InspectAppWindowView) {
base::FilePath dir;
base::PathService::Get(chrome::DIR_TEST_DATA, &dir);
dir = dir.AppendASCII("extensions")
.AppendASCII("platform_apps")
.AppendASCII("minimal");
// Load and launch a platform app.
const Extension* app = LoadAndLaunchApp(dir);
// Get the info about the app, including the inspectable views.
auto info = GetExtensionInfo(*app);
// There should be two inspectable views - the background page and the app
// window. Find the app window.
ASSERT_EQ(2u, info->views.size());
const api::developer_private::ExtensionView* window_view = nullptr;
for (const auto& view : info->views) {
if (view.type == api::developer_private::VIEW_TYPE_APP_WINDOW) {
window_view = &view;
break;
}
}
ASSERT_TRUE(window_view);
// Inspect the app window.
auto function =
base::MakeRefCounted<api::DeveloperPrivateOpenDevToolsFunction>();
api_test_utils::RunFunction(
function.get(),
base::StringPrintf("[{\"renderViewId\": %d, \"renderProcessId\": %d}]",
window_view->render_view_id,
window_view->render_process_id),
profile());
// Verify that dev tools opened.
std::list<AppWindow*> app_windows =
AppWindowRegistry::Get(profile())->GetAppWindowsForApp(app->id());
ASSERT_EQ(1u, app_windows.size());
EXPECT_TRUE(DevToolsWindow::GetInstanceForInspectedWebContents(
(*app_windows.begin())->web_contents()));
}
IN_PROC_BROWSER_TEST_F(DeveloperPrivateApiTest, InspectEmbeddedOptionsPage) {
base::FilePath dir;
base::PathService::Get(chrome::DIR_TEST_DATA, &dir);
// Load an extension that only has an embedded options_ui page.
const Extension* extension = LoadExtension(dir.AppendASCII("extensions")
.AppendASCII("delayed_install")
.AppendASCII("v1"));
ASSERT_TRUE(extension);
// Open the embedded options page.
ASSERT_TRUE(ExtensionTabUtil::OpenOptionsPage(extension, browser()));
WaitForExtensionNotIdle(extension->id());
// Get the info about the extension, including the inspectable views.
auto info = GetExtensionInfo(*extension);
// The embedded options page should show up.
ASSERT_EQ(1u, info->views.size());
const api::developer_private::ExtensionView& view = info->views[0];
ASSERT_EQ(api::developer_private::VIEW_TYPE_EXTENSION_GUEST, view.type);
// Inspect the embedded options page.
auto function =
base::MakeRefCounted<api::DeveloperPrivateOpenDevToolsFunction>();
api_test_utils::RunFunction(
function.get(),
base::StringPrintf("[{\"renderViewId\": %d, \"renderProcessId\": %d}]",
view.render_view_id, view.render_process_id),
profile());
// Verify that dev tools opened.
content::RenderFrameHost* rfh = content::RenderFrameHost::FromID(
view.render_process_id, view.render_view_id);
ASSERT_TRUE(rfh);
content::WebContents* wc = content::WebContents::FromRenderFrameHost(rfh);
ASSERT_TRUE(wc);
EXPECT_TRUE(DevToolsWindow::GetInstanceForInspectedWebContents(wc));
}
IN_PROC_BROWSER_TEST_F(DeveloperPrivateApiTest,
InspectInactiveServiceWorkerBackground) {
ResultCatcher result_catcher;
// Load an extension that is service worker-based.
const Extension* extension =
LoadExtension(test_data_dir_.AppendASCII("service_worker")
.AppendASCII("worker_based_background")
.AppendASCII("inspect"),
// Wait for the registration to be stored since we'll stop
// the worker.
{.wait_for_registration_stored = true});
ASSERT_TRUE(extension);
ASSERT_TRUE(result_catcher.GetNextResult());
// Stop the service worker.
browsertest_util::StopServiceWorkerForExtensionGlobalScope(profile(),
extension->id());
// Get the info about the extension, including the inspectable views.
auto info = GetExtensionInfo(*extension);
// There should be a worker based background for the extension.
ASSERT_EQ(1u, info->views.size());
const api::developer_private::ExtensionView& view = info->views[0];
EXPECT_EQ(
api::developer_private::VIEW_TYPE_EXTENSION_SERVICE_WORKER_BACKGROUND,
view.type);
// The service worker should be inactive (indicated by -1 for
// the process id).
EXPECT_EQ(-1, view.render_process_id);
// Inspect the inactive service worker background.
DevToolsWindowCreationObserver devtools_window_created_observer;
auto dev_tools_function =
base::MakeRefCounted<api::DeveloperPrivateOpenDevToolsFunction>();
api_test_utils::RunFunction(dev_tools_function.get(),
base::StringPrintf(
R"([{"renderViewId": -1,
"renderProcessId": -1,
"isServiceWorker": true,
"extensionId": "%s"
}])",
extension->id().c_str()),
profile());
devtools_window_created_observer.WaitForLoad();
// Verify that dev tool window opened.
scoped_refptr<content::DevToolsAgentHost> service_worker_host;
content::DevToolsAgentHost::List targets =
content::DevToolsAgentHost::GetOrCreateAll();
for (const scoped_refptr<content::DevToolsAgentHost>& host : targets) {
if (host->GetType() == content::DevToolsAgentHost::kTypeServiceWorker &&
host->GetURL() ==
extension->GetResourceURL(
BackgroundInfo::GetBackgroundServiceWorkerScript(extension))) {
EXPECT_FALSE(service_worker_host);
service_worker_host = host;
}
}
ASSERT_TRUE(service_worker_host);
EXPECT_TRUE(DevToolsWindow::FindDevToolsWindow(service_worker_host.get()));
}
IN_PROC_BROWSER_TEST_F(DeveloperPrivateApiTest,
InspectActiveServiceWorkerBackground) {
ResultCatcher result_catcher;
// Load an extension that is service worker based.
const Extension* extension =
LoadExtension(test_data_dir_.AppendASCII("service_worker")
.AppendASCII("worker_based_background")
.AppendASCII("inspect"));
ASSERT_TRUE(extension);
ASSERT_TRUE(result_catcher.GetNextResult());
// Get the info about the extension, including the inspectable views.
auto info = GetExtensionInfo(*extension);
// There should be a worker based background for the extension.
ASSERT_EQ(1u, info->views.size());
const api::developer_private::ExtensionView& view = info->views[0];
EXPECT_EQ(
api::developer_private::VIEW_TYPE_EXTENSION_SERVICE_WORKER_BACKGROUND,
view.type);
EXPECT_NE(-1, view.render_process_id);
// Inspect the service worker page.
auto dev_tools_function =
base::MakeRefCounted<api::DeveloperPrivateOpenDevToolsFunction>();
api_test_utils::RunFunction(
dev_tools_function.get(),
base::StringPrintf(
R"([{"renderViewId": -1,
"renderProcessId": %d,
"isServiceWorker": true,
"extensionId": "%s"
}])",
info->views[0].render_process_id, extension->id().c_str()),
profile());
// Find the service worker background host.
content::DevToolsAgentHost::List targets =
content::DevToolsAgentHost::GetOrCreateAll();
scoped_refptr<content::DevToolsAgentHost> service_worker_host;
for (const scoped_refptr<content::DevToolsAgentHost>& host : targets) {
if (host->GetType() == content::DevToolsAgentHost::kTypeServiceWorker &&
host->GetURL() ==
extension->GetResourceURL(
BackgroundInfo::GetBackgroundServiceWorkerScript(extension))) {
EXPECT_FALSE(service_worker_host);
service_worker_host = host;
}
}
// Verify that dev tools opened.
ASSERT_TRUE(service_worker_host);
EXPECT_TRUE(DevToolsWindow::FindDevToolsWindow(service_worker_host.get()));
}
// TODO(crbug.com/1395713): The test is flaky on MSAN and Linux. Re-enable it.
#if defined(MEMORY_SANITIZER) || BUILDFLAG(IS_LINUX)
#define MAYBE_InspectSplitModeServiceWorkerBackgrounds \
DISABLED_InspectSplitModeServiceWorkerBackgrounds
#else
#define MAYBE_InspectSplitModeServiceWorkerBackgrounds \
InspectSplitModeServiceWorkerBackgrounds
#endif
IN_PROC_BROWSER_TEST_F(DeveloperPrivateApiTest,
MAYBE_InspectSplitModeServiceWorkerBackgrounds) {
ResultCatcher result_catcher;
// Load an extension that is service worker based, split mode and enabled in
// incognito.
static constexpr char kManifest[] =
R"({
"name": "Split mode worker test",
"manifest_version": 3,
"version": "0.1",
"background": {"service_worker": "worker.js"},
"incognito": "split"
})";
TestExtensionDir test_dir;
test_dir.WriteManifest(kManifest);
test_dir.WriteFile(FILE_PATH_LITERAL("worker.js"),
"chrome.test.notifyPass();");
const Extension* extension =
LoadExtension(test_dir.UnpackedPath(), {.allow_in_incognito = true});
ASSERT_TRUE(extension);
ASSERT_TRUE(result_catcher.GetNextResult());
// Get the info about the extension, including the inspectable views.
auto info = GetExtensionInfo(*extension);
// With no incognito window open, there should be a single worker based
// background page for the extension.
ASSERT_EQ(1u, info->views.size());
int main_render_process_id = -1;
{
const api::developer_private::ExtensionView& view = info->views[0];
EXPECT_EQ(
api::developer_private::VIEW_TYPE_EXTENSION_SERVICE_WORKER_BACKGROUND,
view.type);
EXPECT_NE(-1, view.render_process_id);
main_render_process_id = view.render_process_id;
EXPECT_FALSE(view.incognito);
}
// Now open up an incognito browser window page and check the inspectable
// views again. Waiting for the result catcher will wait for the incognito
// service worker to have become active.
Browser* inconito_browser = CreateIncognitoBrowser(browser()->profile());
ASSERT_TRUE(inconito_browser);
ASSERT_TRUE(result_catcher.GetNextResult());
info = GetExtensionInfo(*extension);
// The views should now have 2 entries, one for the main worker which will be
// the same as before and a new one for the incognito worker.
ASSERT_EQ(2u, info->views.size());
EXPECT_NE(info->views[0].incognito, info->views[1].incognito);
int incognito_render_process_id = -1;
for (auto& view : info->views) {
EXPECT_EQ(
api::developer_private::VIEW_TYPE_EXTENSION_SERVICE_WORKER_BACKGROUND,
view.type);
EXPECT_NE(-1, view.render_process_id);
if (view.incognito) {
EXPECT_NE(main_render_process_id, view.render_process_id);
incognito_render_process_id = view.render_process_id;
} else {
EXPECT_EQ(main_render_process_id, view.render_process_id);
}
}
// Open a devtools window for both the primary and incognito worker.
std::string kOpenDevToolsParams =
R"([{"renderViewId": -1,
"renderProcessId": $1,
"isServiceWorker": true,
"extensionId": $2,
"incognito": $3
}])";
DevToolsWindow* main_devtools_window = nullptr;
{
DevToolsWindowCreationObserver devtools_window_created_observer;
auto dev_tools_function =
base::MakeRefCounted<api::DeveloperPrivateOpenDevToolsFunction>();
api_test_utils::RunFunction(
dev_tools_function.get(),
content::JsReplace(kOpenDevToolsParams, main_render_process_id,
extension->id().c_str(), /*incognito:*/ false),
profile());
devtools_window_created_observer.WaitForLoad();
main_devtools_window = devtools_window_created_observer.devtools_window();
}
DevToolsWindow* incognito_devtools_window = nullptr;
{
DevToolsWindowCreationObserver devtools_window_created_observer;
auto dev_tools_function =
base::MakeRefCounted<api::DeveloperPrivateOpenDevToolsFunction>();
api_test_utils::RunFunction(
dev_tools_function.get(),
content::JsReplace(kOpenDevToolsParams, incognito_render_process_id,
extension->id().c_str(), /*incognito:*/ true),
profile());
devtools_window_created_observer.WaitForLoad();
incognito_devtools_window =
devtools_window_created_observer.devtools_window();
}
// Both windows should have opened and should not point to the same window.
ASSERT_TRUE(main_devtools_window);
ASSERT_TRUE(incognito_devtools_window);
ASSERT_NE(main_devtools_window, incognito_devtools_window);
}
// Test that offscreen documents show up in the list of inspectable views and
// can be inspected.
IN_PROC_BROWSER_TEST_F(DeveloperPrivateApiTest, InspectOffscreenDocument) {
static constexpr char kManifest[] =
R"({
"name": "Offscreen Document Test",
"manifest_version": 3,
"version": "0.1"
})";
TestExtensionDir test_dir;
test_dir.WriteManifest(kManifest);
test_dir.WriteFile(FILE_PATH_LITERAL("offscreen.html"),
"<html>offscreen</html>");
const Extension* extension = LoadExtension(test_dir.UnpackedPath());
// Create an offscreen document and wait for it to load.
std::unique_ptr<OffscreenDocumentHost> offscreen_document;
GURL offscreen_url = extension->GetResourceURL("offscreen.html");
{
ExtensionHostTestHelper offscreen_waiter(profile(), extension->id());
offscreen_waiter.RestrictToType(mojom::ViewType::kOffscreenDocument);
offscreen_document = std::make_unique<OffscreenDocumentHost>(
*extension,
ProcessManager::Get(profile())
->GetSiteInstanceForURL(offscreen_url)
.get(),
offscreen_url);
offscreen_document->CreateRendererSoon();
offscreen_waiter.WaitForHostCompletedFirstLoad();
}
// Get the list of inspectable views for the extension.
auto info = GetExtensionInfo(*extension);
// The only inspectable view should be the offscreen document. Validate the
// metadata.
ASSERT_EQ(1u, info->views.size());
const api::developer_private::ExtensionView& view = info->views[0];
EXPECT_EQ(api::developer_private::VIEW_TYPE_OFFSCREEN_DOCUMENT, view.type);
content::WebContents* offscreen_contents =
offscreen_document->host_contents();
EXPECT_EQ(offscreen_url.spec(), view.url);
EXPECT_EQ(offscreen_document->render_process_host()->GetID(),
view.render_process_id);
EXPECT_EQ(offscreen_contents->GetPrimaryMainFrame()->GetRoutingID(),
view.render_view_id);
EXPECT_FALSE(view.incognito);
EXPECT_FALSE(view.is_iframe);
// The document shouldn't currently be under inspection.
EXPECT_FALSE(
DevToolsWindow::GetInstanceForInspectedWebContents(offscreen_contents));
// Call the API function to inspect the offscreen document.
auto dev_tools_function =
base::MakeRefCounted<api::DeveloperPrivateOpenDevToolsFunction>();
api_test_utils::RunFunction(
dev_tools_function.get(),
content::JsReplace(
R"([{"renderViewId": $1,
"renderProcessId": $2,
"extensionId": $3
}])",
view.render_view_id, view.render_process_id, extension->id()),
profile());
// Validate that the devtools window is now shown.
DevToolsWindow* dev_tools_window =
DevToolsWindow::GetInstanceForInspectedWebContents(offscreen_contents);
ASSERT_TRUE(dev_tools_window);
// Tidy up.
DevToolsWindowTesting::CloseDevToolsWindowSync(dev_tools_window);
}
} // namespace extensions