// 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/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/common/chrome_notification_types.h" #include "chrome/common/chrome_switches.h" #include "content/public/browser/notification_registrar.h" #include "content/public/browser/notification_service.h" #include "googleurl/src/gurl.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_; }