// 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/memory/ref_counted.h" #include "base/path_service.h" #include "base/strings/stringprintf.h" #include "chrome/browser/extensions/api/dns/host_resolver_wrapper.h" #include "chrome/browser/extensions/api/dns/mock_host_resolver_creator.h" #include "chrome/browser/extensions/api/socket/socket_api.h" #include "chrome/browser/extensions/extension_apitest.h" #include "chrome/browser/extensions/extension_function_test_utils.h" #include "chrome/browser/extensions/extension_service.h" #include "chrome/browser/extensions/extension_test_message_listener.h" #include "chrome/browser/ui/browser.h" #include "chrome/browser/ui/extensions/application_launch.h" #include "chrome/common/chrome_paths.h" #include "chrome/test/base/in_process_browser_test.h" #include "chrome/test/base/ui_test_utils.h" #include "net/dns/mock_host_resolver.h" #include "net/test/spawned_test_server/spawned_test_server.h" using extensions::Extension; namespace utils = extension_function_test_utils; namespace { const std::string kHostname = "127.0.0.1"; const int kPort = 8888; class SocketApiTest : public ExtensionApiTest { public: SocketApiTest() : resolver_event_(true, false), resolver_creator_( new extensions::MockHostResolverCreator()) { } virtual void SetUpOnMainThread() OVERRIDE { extensions::HostResolverWrapper::GetInstance()->SetHostResolverForTesting( resolver_creator_->CreateMockHostResolver()); } virtual void CleanUpOnMainThread() OVERRIDE { extensions::HostResolverWrapper::GetInstance()-> SetHostResolverForTesting(NULL); resolver_creator_->DeleteMockHostResolver(); } private: base::WaitableEvent resolver_event_; // The MockHostResolver asserts that it's used on the same thread on which // it's created, which is actually a stronger rule than its real counterpart. // But that's fine; it's good practice. scoped_refptr resolver_creator_; }; #if !defined(DISABLE_NACL) // TODO(yzshen): Build testing framework for all extensions APIs in Pepper. And // move these Pepper API tests there. class SocketPpapiTest : public SocketApiTest { public: SocketPpapiTest() { } virtual ~SocketPpapiTest() { } virtual void SetUpCommandLine(CommandLine* command_line) OVERRIDE { SocketApiTest::SetUpCommandLine(command_line); // TODO(yzshen): It is better to use switches::kEnablePepperTesting. // However, that requires adding a new DEPS entry. Considering that we are // going to move the Pepper API tests to a new place, use a string literal // for now. command_line->AppendSwitch("enable-pepper-testing"); PathService::Get(chrome::DIR_GEN_TEST_DATA, &app_dir_); app_dir_ = app_dir_.AppendASCII("ppapi/tests/extensions/socket/newlib"); } protected: void LaunchTestingApp() { const Extension* extension = LoadExtension(app_dir_); ASSERT_TRUE(extension); AppLaunchParams params(browser()->profile(), extension, extensions::LAUNCH_CONTAINER_NONE, NEW_WINDOW); params.command_line = CommandLine::ForCurrentProcess(); OpenApplication(params); } private: base::FilePath app_dir_; }; #endif } // namespace IN_PROC_BROWSER_TEST_F(SocketApiTest, SocketUDPCreateGood) { scoped_refptr socket_create_function( new extensions::SocketCreateFunction()); scoped_refptr empty_extension(utils::CreateEmptyExtension()); socket_create_function->set_extension(empty_extension.get()); socket_create_function->set_has_callback(true); scoped_ptr result(utils::RunFunctionAndReturnSingleResult( socket_create_function.get(), "[\"udp\"]", browser(), utils::NONE)); ASSERT_EQ(base::Value::TYPE_DICTIONARY, result->GetType()); base::DictionaryValue *value = static_cast(result.get()); int socketId = -1; EXPECT_TRUE(value->GetInteger("socketId", &socketId)); EXPECT_TRUE(socketId > 0); } IN_PROC_BROWSER_TEST_F(SocketApiTest, SocketTCPCreateGood) { scoped_refptr socket_create_function( new extensions::SocketCreateFunction()); scoped_refptr empty_extension(utils::CreateEmptyExtension()); socket_create_function->set_extension(empty_extension.get()); socket_create_function->set_has_callback(true); scoped_ptr result(utils::RunFunctionAndReturnSingleResult( socket_create_function.get(), "[\"tcp\"]", browser(), utils::NONE)); ASSERT_EQ(base::Value::TYPE_DICTIONARY, result->GetType()); base::DictionaryValue *value = static_cast(result.get()); int socketId = -1; EXPECT_TRUE(value->GetInteger("socketId", &socketId)); ASSERT_TRUE(socketId > 0); } IN_PROC_BROWSER_TEST_F(SocketApiTest, GetNetworkList) { scoped_refptr socket_function( new extensions::SocketGetNetworkListFunction()); scoped_refptr empty_extension(utils::CreateEmptyExtension()); socket_function->set_extension(empty_extension.get()); socket_function->set_has_callback(true); scoped_ptr result(utils::RunFunctionAndReturnSingleResult( socket_function.get(), "[]", browser(), utils::NONE)); ASSERT_EQ(base::Value::TYPE_LIST, result->GetType()); // If we're invoking socket tests, all we can confirm is that we have at // least one address, but not what it is. base::ListValue *value = static_cast(result.get()); ASSERT_TRUE(value->GetSize() > 0); } IN_PROC_BROWSER_TEST_F(SocketApiTest, SocketUDPExtension) { scoped_ptr test_server( new net::SpawnedTestServer( net::SpawnedTestServer::TYPE_UDP_ECHO, net::SpawnedTestServer::kLocalhost, base::FilePath(FILE_PATH_LITERAL("net/data")))); EXPECT_TRUE(test_server->Start()); net::HostPortPair host_port_pair = test_server->host_port_pair(); int port = host_port_pair.port(); ASSERT_TRUE(port > 0); // Test that sendTo() is properly resolving hostnames. host_port_pair.set_host("LOCALhost"); ResultCatcher catcher; catcher.RestrictToProfile(browser()->profile()); ExtensionTestMessageListener listener("info_please", true); ASSERT_TRUE(LoadExtension(test_data_dir_.AppendASCII("socket/api"))); EXPECT_TRUE(listener.WaitUntilSatisfied()); listener.Reply( base::StringPrintf("udp:%s:%d", host_port_pair.host().c_str(), port)); EXPECT_TRUE(catcher.GetNextResult()) << catcher.message(); } IN_PROC_BROWSER_TEST_F(SocketApiTest, SocketTCPExtension) { scoped_ptr test_server( new net::SpawnedTestServer( net::SpawnedTestServer::TYPE_TCP_ECHO, net::SpawnedTestServer::kLocalhost, base::FilePath(FILE_PATH_LITERAL("net/data")))); EXPECT_TRUE(test_server->Start()); net::HostPortPair host_port_pair = test_server->host_port_pair(); int port = host_port_pair.port(); ASSERT_TRUE(port > 0); // Test that connect() is properly resolving hostnames. host_port_pair.set_host("lOcAlHoSt"); ResultCatcher catcher; catcher.RestrictToProfile(browser()->profile()); ExtensionTestMessageListener listener("info_please", true); ASSERT_TRUE(LoadExtension(test_data_dir_.AppendASCII("socket/api"))); EXPECT_TRUE(listener.WaitUntilSatisfied()); listener.Reply( base::StringPrintf("tcp:%s:%d", host_port_pair.host().c_str(), port)); EXPECT_TRUE(catcher.GetNextResult()) << catcher.message(); } IN_PROC_BROWSER_TEST_F(SocketApiTest, SocketTCPServerExtension) { ResultCatcher catcher; catcher.RestrictToProfile(browser()->profile()); ExtensionTestMessageListener listener("info_please", true); ASSERT_TRUE(LoadExtension(test_data_dir_.AppendASCII("socket/api"))); EXPECT_TRUE(listener.WaitUntilSatisfied()); listener.Reply( base::StringPrintf("tcp_server:%s:%d", kHostname.c_str(), kPort)); EXPECT_TRUE(catcher.GetNextResult()) << catcher.message(); } IN_PROC_BROWSER_TEST_F(SocketApiTest, SocketTCPServerUnbindOnUnload) { ResultCatcher catcher; const Extension* extension = LoadExtension(test_data_dir_.AppendASCII("socket/unload")); ASSERT_TRUE(extension); EXPECT_TRUE(catcher.GetNextResult()) << catcher.message(); UnloadExtension(extension->id()); ASSERT_TRUE(LoadExtension(test_data_dir_.AppendASCII("socket/unload"))) << message_; EXPECT_TRUE(catcher.GetNextResult()) << catcher.message(); } IN_PROC_BROWSER_TEST_F(SocketApiTest, SocketMulticast) { ResultCatcher catcher; catcher.RestrictToProfile(browser()->profile()); ExtensionTestMessageListener listener("info_please", true); ASSERT_TRUE(LoadExtension(test_data_dir_.AppendASCII("socket/api"))); EXPECT_TRUE(listener.WaitUntilSatisfied()); listener.Reply( base::StringPrintf("multicast:%s:%d", kHostname.c_str(), kPort)); EXPECT_TRUE(catcher.GetNextResult()) << catcher.message(); } #if !defined(DISABLE_NACL) // TODO(jschuh): Hanging plugin tests. crbug.com/244653 #if defined(OS_WIN) && defined(ARCH_CPU_X86_64) #define MAYBE_UDP DISABLED_UDP #else #define MAYBE_UDP UDP #endif IN_PROC_BROWSER_TEST_F(SocketPpapiTest, MAYBE_UDP) { scoped_ptr test_server( new net::SpawnedTestServer( net::SpawnedTestServer::TYPE_UDP_ECHO, net::SpawnedTestServer::kLocalhost, base::FilePath(FILE_PATH_LITERAL("net/data")))); EXPECT_TRUE(test_server->Start()); net::HostPortPair host_port_pair = test_server->host_port_pair(); int port = host_port_pair.port(); ASSERT_TRUE(port > 0); // Test that sendTo() is properly resolving hostnames. host_port_pair.set_host("LOCALhost"); ResultCatcher catcher; catcher.RestrictToProfile(browser()->profile()); ExtensionTestMessageListener listener("info_please", true); LaunchTestingApp(); EXPECT_TRUE(listener.WaitUntilSatisfied()); listener.Reply( base::StringPrintf("udp:%s:%d", host_port_pair.host().c_str(), port)); EXPECT_TRUE(catcher.GetNextResult()) << catcher.message(); } // TODO(jschuh): Hanging plugin tests. crbug.com/244653 #if defined(OS_WIN) && defined(ARCH_CPU_X86_64) #define MAYBE_TCP DISABLED_TCP #else #define MAYBE_TCP TCP #endif IN_PROC_BROWSER_TEST_F(SocketPpapiTest, MAYBE_TCP) { scoped_ptr test_server( new net::SpawnedTestServer( net::SpawnedTestServer::TYPE_TCP_ECHO, net::SpawnedTestServer::kLocalhost, base::FilePath(FILE_PATH_LITERAL("net/data")))); EXPECT_TRUE(test_server->Start()); net::HostPortPair host_port_pair = test_server->host_port_pair(); int port = host_port_pair.port(); ASSERT_TRUE(port > 0); // Test that connect() is properly resolving hostnames. host_port_pair.set_host("lOcAlHoSt"); ResultCatcher catcher; catcher.RestrictToProfile(browser()->profile()); ExtensionTestMessageListener listener("info_please", true); LaunchTestingApp(); EXPECT_TRUE(listener.WaitUntilSatisfied()); listener.Reply( base::StringPrintf("tcp:%s:%d", host_port_pair.host().c_str(), port)); EXPECT_TRUE(catcher.GetNextResult()) << catcher.message(); } // TODO(jschuh): Hanging plugin tests. crbug.com/244653 // Also fails on official Mac builds. See http://crbug.com/312916 #if (defined(OS_WIN) && defined(ARCH_CPU_X86_64)) || \ (defined(OS_MACOSX) && defined(GOOGLE_CHROME_BUILD)) #define MAYBE_TCPServer DISABLED_TCPServer #else #define MAYBE_TCPServer TCPServer #endif IN_PROC_BROWSER_TEST_F(SocketPpapiTest, MAYBE_TCPServer) { ResultCatcher catcher; catcher.RestrictToProfile(browser()->profile()); ExtensionTestMessageListener listener("info_please", true); LaunchTestingApp(); EXPECT_TRUE(listener.WaitUntilSatisfied()); listener.Reply( base::StringPrintf("tcp_server:%s:%d", kHostname.c_str(), kPort)); EXPECT_TRUE(catcher.GetNextResult()) << catcher.message(); } // Disabled due to flakiness: http://crbug.com/314899 IN_PROC_BROWSER_TEST_F(SocketPpapiTest, DISABLED_Multicast) { ResultCatcher catcher; catcher.RestrictToProfile(browser()->profile()); ExtensionTestMessageListener listener("info_please", true); LaunchTestingApp(); EXPECT_TRUE(listener.WaitUntilSatisfied()); listener.Reply( base::StringPrintf("multicast:%s:%d", kHostname.c_str(), kPort)); EXPECT_TRUE(catcher.GetNextResult()) << catcher.message(); } #endif