// Copyright (c) 2012 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "base/files/file_path.h" #include "base/path_service.h" #include "base/strings/string_number_conversions.h" #include "base/strings/stringprintf.h" #include "base/values.h" #include "chrome/browser/extensions/event_router.h" #include "chrome/browser/extensions/extension_apitest.h" #include "chrome/browser/extensions/extension_system.h" #include "chrome/browser/profiles/profile.h" #include "chrome/browser/ui/browser.h" #include "chrome/browser/ui/tabs/tab_strip_model.h" #include "chrome/common/chrome_notification_types.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/notification_registrar.h" #include "content/public/browser/notification_service.h" #include "content/public/test/browser_test_utils.h" #include "googleurl/src/gurl.h" #include "net/dns/mock_host_resolver.h" #include "net/test/embedded_test_server/embedded_test_server.h" namespace { class MessageSender : public content::NotificationObserver { public: MessageSender() { registrar_.Add(this, chrome::NOTIFICATION_EXTENSION_HOST_DID_STOP_LOADING, content::NotificationService::AllSources()); } private: static scoped_ptr BuildEventArguments(const bool last_message, const std::string& data) { DictionaryValue* event = new DictionaryValue(); event->SetBoolean("lastMessage", last_message); event->SetString("data", data); scoped_ptr arguments(new ListValue()); arguments->Append(event); return arguments.Pass(); } static scoped_ptr BuildEvent( scoped_ptr event_args, Profile* profile, GURL event_url) { scoped_ptr event(new extensions::Event( "test.onMessage", event_args.Pass())); event->restrict_to_profile = profile; event->event_url = event_url; return event.Pass(); } virtual void Observe(int type, const content::NotificationSource& source, const content::NotificationDetails& details) OVERRIDE { extensions::EventRouter* event_router = extensions::ExtensionSystem::Get( content::Source(source).ptr())->event_router(); // Sends four messages to the extension. All but the third message sent // from the origin http://b.com/ are supposed to arrive. event_router->BroadcastEvent(BuildEvent( BuildEventArguments(false, "no restriction"), content::Source(source).ptr(), GURL())); event_router->BroadcastEvent(BuildEvent( BuildEventArguments(false, "http://a.com/"), content::Source(source).ptr(), GURL("http://a.com/"))); event_router->BroadcastEvent(BuildEvent( BuildEventArguments(false, "http://b.com/"), content::Source(source).ptr(), GURL("http://b.com/"))); event_router->BroadcastEvent(BuildEvent( BuildEventArguments(true, "last message"), content::Source(source).ptr(), GURL())); } content::NotificationRegistrar registrar_; }; } // namespace // Tests that message passing between extensions and content scripts works. IN_PROC_BROWSER_TEST_F(ExtensionApiTest, Messaging) { ASSERT_TRUE(StartTestServer()); ASSERT_TRUE(RunExtensionTest("messaging/connect")) << message_; } // Tests that message passing from one extension to another works. IN_PROC_BROWSER_TEST_F(ExtensionApiTest, MessagingExternal) { ASSERT_TRUE(LoadExtension( test_data_dir_.AppendASCII("..").AppendASCII("good") .AppendASCII("Extensions") .AppendASCII("bjafgdebaacbbbecmhlhpofkepfkgcpa") .AppendASCII("1.0"))); ASSERT_TRUE(RunExtensionTest("messaging/connect_external")) << message_; } // Tests that messages with event_urls are only passed to extensions with // appropriate permissions. IN_PROC_BROWSER_TEST_F(ExtensionApiTest, MessagingEventURL) { MessageSender sender; ASSERT_TRUE(RunExtensionTest("messaging/event_url")) << message_; } // Tests connecting from a panel to its extension. class PanelMessagingTest : public ExtensionApiTest { virtual void SetUpCommandLine(CommandLine* command_line) OVERRIDE { ExtensionApiTest::SetUpCommandLine(command_line); command_line->AppendSwitch(switches::kEnablePanels); } }; IN_PROC_BROWSER_TEST_F(PanelMessagingTest, MessagingPanel) { ASSERT_TRUE(RunExtensionTest("messaging/connect_panel")) << message_; } // Tests externally_connectable between a web page and an extension. // // TODO(kalman): Test between extensions. This is already tested in this file, // but not with externally_connectable set in the manifest. // // TODO(kalman): Test with host permissions. class ExternallyConnectableMessagingTest : public ExtensionApiTest { protected: // Result codes from the test. These must match up with |results| in // c/t/d/extensions/api_test/externally_connectable/sites/assertions.json. enum Result { OK = 0, NAMESPACE_NOT_DEFINED = 1, FUNCTION_NOT_DEFINED = 2, COULD_NOT_ESTABLISH_CONNECTION_ERROR = 3, OTHER_ERROR = 4, INCORRECT_RESPONSE_SENDER = 5, INCORRECT_RESPONSE_MESSAGE = 6, }; Result CanConnectAndSendMessages(const std::string& extension_id) { int result; CHECK(content::ExecuteScriptAndExtractInt( browser()->tab_strip_model()->GetActiveWebContents(), "assertions.canConnectAndSendMessages('" + extension_id + "')", &result)); return static_cast(result); } testing::AssertionResult AreAnyNonWebApisDefined() { // All runtime API methods are non-web except for sendRequest and connect. const std::string non_messaging_apis[] = { "getBackgroundPage", "getManifest", "getURL", "reload", "requestUpdateCheck", "connectNative", "sendNativeMessage", "onStartup", "onInstalled", "onSuspend", "onSuspendCanceled", "onUpdateAvailable", "onBrowserUpdateAvailable", "onConnect", "onConnectExternal", "onMessage", "onMessageExternal", "id", }; return AreAnyRuntimePropertiesDefined(std::vector( non_messaging_apis, non_messaging_apis + arraysize(non_messaging_apis))); } GURL GetURLForPath(const std::string& host, const std::string& path) { std::string port = base::IntToString(embedded_test_server()->port()); GURL::Replacements replacements; replacements.SetHostStr(host); replacements.SetPortStr(port); return embedded_test_server()->GetURL(path).ReplaceComponents(replacements); } private: testing::AssertionResult AreAnyRuntimePropertiesDefined( const std::vector& names) { for (size_t i = 0; i < names.size(); ++i) { if (IsRuntimePropertyDefined(names[i]) == OK) return testing::AssertionSuccess() << names[i] << " is defined"; } return testing::AssertionFailure() << "none of " << names.size() << " properties are defined"; } Result IsRuntimePropertyDefined(const std::string& name) { int result_int; CHECK(content::ExecuteScriptAndExtractInt( browser()->tab_strip_model()->GetActiveWebContents(), "assertions.isDefined('" + name + "')", &result_int)); return static_cast(result_int); } }; // Flaky on Windows. http://crbug.com/248413 #if defined(OS_WIN) #define MAYBE_ExternallyConnectableMessaging \ DISABLED_ExternallyConnectableMessaging #else #define MAYBE_ExternallyConnectableMessaging ExternallyConnectableMessaging #endif IN_PROC_BROWSER_TEST_F(ExternallyConnectableMessagingTest, MAYBE_ExternallyConnectableMessaging) { const char kExtensionDir[] = "messaging/externally_connectable"; // The extension allows connections from chromium.org but not google.com. const char kChromiumOrg[] = "www.chromium.org"; const char kGoogleCom[] = "www.google.com"; base::FilePath test_data; ASSERT_TRUE(PathService::Get(chrome::DIR_TEST_DATA, &test_data)); embedded_test_server()->ServeFilesFromDirectory( test_data.AppendASCII("extensions/api_test").AppendASCII(kExtensionDir)); ASSERT_TRUE(embedded_test_server()->InitializeAndWaitUntilReady()); host_resolver()->AddRule("*", embedded_test_server()->base_url().host()); const GURL kChromiumOrgUrl = GetURLForPath(kChromiumOrg, "/sites/chromium.org.html"); const GURL kGoogleComUrl = GetURLForPath(kGoogleCom, "/sites/google.com.html"); // When an extension isn't installed all attempts to connect to it should // fail. const std::string kFakeId = "aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa"; ui_test_utils::NavigateToURL(browser(), kChromiumOrgUrl); EXPECT_EQ(COULD_NOT_ESTABLISH_CONNECTION_ERROR, CanConnectAndSendMessages(kFakeId)); EXPECT_FALSE(AreAnyNonWebApisDefined()); ui_test_utils::NavigateToURL(browser(), kGoogleComUrl); EXPECT_EQ(COULD_NOT_ESTABLISH_CONNECTION_ERROR, CanConnectAndSendMessages(kFakeId)); EXPECT_FALSE(AreAnyNonWebApisDefined()); // Install the web connectable extension. chromium.org can connect to it, // google.com can't. const extensions::Extension* web_connectable = LoadExtension( test_data_dir_.AppendASCII(kExtensionDir).AppendASCII("web_connectable")); ui_test_utils::NavigateToURL(browser(), kChromiumOrgUrl); EXPECT_EQ(OK, CanConnectAndSendMessages(web_connectable->id())); EXPECT_FALSE(AreAnyNonWebApisDefined()); ui_test_utils::NavigateToURL(browser(), kGoogleComUrl); EXPECT_EQ(COULD_NOT_ESTABLISH_CONNECTION_ERROR, CanConnectAndSendMessages(web_connectable->id())); EXPECT_FALSE(AreAnyNonWebApisDefined()); // Install the non-connectable extension. Nothing can connect to it. const extensions::Extension* not_connectable = LoadExtension( test_data_dir_.AppendASCII(kExtensionDir).AppendASCII("not_connectable")); ui_test_utils::NavigateToURL(browser(), kChromiumOrgUrl); EXPECT_EQ(COULD_NOT_ESTABLISH_CONNECTION_ERROR, CanConnectAndSendMessages(not_connectable->id())); EXPECT_FALSE(AreAnyNonWebApisDefined()); ui_test_utils::NavigateToURL(browser(), kGoogleComUrl); EXPECT_EQ(COULD_NOT_ESTABLISH_CONNECTION_ERROR, CanConnectAndSendMessages(not_connectable->id())); EXPECT_FALSE(AreAnyNonWebApisDefined()); }