| // Copyright 2018 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 <lib/fidl/cpp/binding.h> |
| #include <lib/zx/channel.h> |
| |
| #include "base/fuchsia/file_utils.h" |
| #include "base/fuchsia/fuchsia_logging.h" |
| #include "base/fuchsia/scoped_service_binding.h" |
| #include "base/fuchsia/service_directory.h" |
| #include "base/fuchsia/service_directory_client.h" |
| #include "base/message_loop/message_loop.h" |
| #include "base/strings/stringprintf.h" |
| #include "base/test/bind_test_util.h" |
| #include "base/test/test_timeouts.h" |
| #include "fuchsia/base/agent_impl.h" |
| #include "fuchsia/base/fake_component_context.h" |
| #include "fuchsia/base/fit_adapter.h" |
| #include "fuchsia/base/mem_buffer_util.h" |
| #include "fuchsia/base/result_receiver.h" |
| #include "fuchsia/runners/cast/cast_runner.h" |
| #include "fuchsia/runners/cast/fake_application_config_manager.h" |
| #include "fuchsia/runners/common/web_component.h" |
| #include "fuchsia/runners/common/web_content_runner.h" |
| #include "net/test/embedded_test_server/default_handlers.h" |
| #include "testing/gtest/include/gtest/gtest.h" |
| |
| namespace castrunner { |
| |
| namespace { |
| |
| const char kTestServerRoot[] = |
| FILE_PATH_LITERAL("fuchsia/runners/cast/testdata"); |
| |
| void ComponentErrorHandler(zx_status_t status) { |
| ZX_LOG(ERROR, status) << "Component launch failed"; |
| ADD_FAILURE(); |
| } |
| |
| class FakeCastChannel : public chromium::cast::CastChannel { |
| public: |
| explicit FakeCastChannel(base::fuchsia::ServiceDirectory* directory) |
| : binding_(directory, this) {} |
| |
| // Returns null if the Cast channel is not open. |
| const chromium::web::MessagePortPtr& port() const { return port_; } |
| |
| void set_on_opened(base::OnceClosure on_opened) { |
| on_opened_ = std::move(on_opened); |
| } |
| |
| protected: |
| // chromium::cast::CastChannel implementation. |
| void OnOpened(fidl::InterfaceHandle<chromium::web::MessagePort> channel, |
| OnOpenedCallback callback_ignored) override { |
| port_ = channel.Bind(); |
| |
| if (on_opened_) |
| std::move(on_opened_).Run(); |
| |
| callback_ignored(); |
| } |
| |
| const base::fuchsia::ScopedServiceBinding<chromium::cast::CastChannel> |
| binding_; |
| |
| // Null until the Cast app connects to the Cast channel. |
| chromium::web::MessagePortPtr port_; |
| |
| // Invoked when the contect opens a new Cast channel, if set. |
| base::OnceClosure on_opened_; |
| |
| DISALLOW_COPY_AND_ASSIGN(FakeCastChannel); |
| }; |
| |
| class FakeComponentState : public cr_fuchsia::AgentImpl::ComponentStateBase { |
| public: |
| FakeComponentState( |
| base::StringPiece component_url, |
| chromium::cast::ApplicationConfigManager* app_config_manager) |
| : ComponentStateBase(component_url), |
| app_config_binding_(service_directory(), app_config_manager), |
| cast_channel_(std::make_unique<FakeCastChannel>(service_directory())) {} |
| ~FakeComponentState() override { |
| if (on_delete_) |
| std::move(on_delete_).Run(); |
| } |
| |
| void set_on_delete(base::OnceClosure on_delete) { |
| on_delete_ = std::move(on_delete); |
| } |
| |
| FakeCastChannel* cast_channel() { return cast_channel_.get(); } |
| void ClearCastChannel() { cast_channel_.reset(); } |
| |
| protected: |
| const base::fuchsia::ScopedServiceBinding< |
| chromium::cast::ApplicationConfigManager> |
| app_config_binding_; |
| std::unique_ptr<FakeCastChannel> cast_channel_; |
| base::OnceClosure on_delete_; |
| |
| DISALLOW_COPY_AND_ASSIGN(FakeComponentState); |
| }; |
| |
| } // namespace |
| |
| class CastRunnerIntegrationTest : public testing::Test { |
| public: |
| CastRunnerIntegrationTest() |
| : run_timeout_( |
| TestTimeouts::action_timeout(), |
| base::MakeExpectedNotRunClosure(FROM_HERE, "Run() timed out.")) { |
| // Create a new test ServiceDirectory, and ServiceDirectoryClient connected |
| // to it, for tests to use to drive the CastRunner. |
| fidl::InterfaceHandle<fuchsia::io::Directory> directory; |
| public_services_ = std::make_unique<base::fuchsia::ServiceDirectory>( |
| directory.NewRequest()); |
| |
| // Create the CastRunner, published into |test_services_|. |
| cast_runner_ = std::make_unique<CastRunner>( |
| public_services_.get(), WebContentRunner::CreateIncognitoWebContext()); |
| |
| // Connect to the CastRunner's fuchsia.sys.Runner interface. |
| base::fuchsia::ServiceDirectoryClient public_directory_client( |
| std::move(directory)); |
| cast_runner_ptr_ = |
| public_directory_client.ConnectToService<fuchsia::sys::Runner>(); |
| cast_runner_ptr_.set_error_handler([](zx_status_t status) { |
| ZX_LOG(ERROR, status) << "CastRunner closed channel."; |
| ADD_FAILURE(); |
| }); |
| } |
| |
| void SetUp() override { |
| test_server_.ServeFilesFromSourceDirectory(kTestServerRoot); |
| net::test_server::RegisterDefaultHandlers(&test_server_); |
| ASSERT_TRUE(test_server_.Start()); |
| } |
| |
| void TearDown() override { |
| // Disconnect the CastRunner & let things tear-down. |
| cast_runner_ptr_.Unbind(); |
| base::RunLoop().RunUntilIdle(); |
| } |
| |
| void WaitUntilCastChannelOpened() { |
| if (component_state_->cast_channel()->port()) |
| return; |
| |
| base::RunLoop run_loop; |
| component_state_->cast_channel()->set_on_opened(run_loop.QuitClosure()); |
| run_loop.Run(); |
| |
| ASSERT_TRUE(component_state_->cast_channel()->port()); |
| } |
| |
| fuchsia::sys::ComponentControllerPtr StartCastComponent( |
| base::StringPiece component_url) { |
| DCHECK(!component_state_); |
| |
| // Create a ServiceDirectory and publish the ComponentContext into it. |
| fidl::InterfaceHandle<fuchsia::io::Directory> directory; |
| component_services_ = std::make_unique<base::fuchsia::ServiceDirectory>( |
| directory.NewRequest()); |
| component_context_ = std::make_unique<cr_fuchsia::FakeComponentContext>( |
| base::BindRepeating(&CastRunnerIntegrationTest::OnComponentConnect, |
| base::Unretained(this)), |
| component_services_.get(), component_url); |
| |
| // Configure the Runner, including a service directory channel to publish |
| // services to. |
| fuchsia::sys::StartupInfo startup_info; |
| startup_info.launch_info.url = component_url.as_string(); |
| |
| // Place the ServiceDirectory in the |flat_namespace|. |
| startup_info.flat_namespace.paths.emplace_back( |
| base::fuchsia::kServiceDirectoryPath); |
| startup_info.flat_namespace.directories.emplace_back( |
| directory.TakeChannel()); |
| |
| fuchsia::sys::Package package; |
| package.resolved_url = component_url.as_string(); |
| |
| fuchsia::sys::ComponentControllerPtr component_controller; |
| cast_runner_ptr_->StartComponent(std::move(package), |
| std::move(startup_info), |
| component_controller.NewRequest()); |
| |
| EXPECT_TRUE(component_controller.is_bound()); |
| return component_controller; |
| } |
| |
| protected: |
| std::unique_ptr<cr_fuchsia::AgentImpl::ComponentStateBase> OnComponentConnect( |
| base::StringPiece component_url) { |
| auto component_state = std::make_unique<FakeComponentState>( |
| component_url, &app_config_manager_); |
| component_state_ = component_state.get(); |
| return component_state; |
| } |
| |
| const base::RunLoop::ScopedRunTimeoutForTest run_timeout_; |
| base::MessageLoopForIO message_loop_; |
| net::EmbeddedTestServer test_server_; |
| |
| // Returns fake Cast application information to the CastRunner. |
| FakeApplicationConfigManager app_config_manager_; |
| |
| // Incoming service directory, ComponentContext and per-component state. |
| std::unique_ptr<base::fuchsia::ServiceDirectory> component_services_; |
| std::unique_ptr<cr_fuchsia::FakeComponentContext> component_context_; |
| FakeComponentState* component_state_ = nullptr; |
| |
| // ServiceDirectory into which the CastRunner will publish itself. |
| std::unique_ptr<base::fuchsia::ServiceDirectory> public_services_; |
| |
| std::unique_ptr<CastRunner> cast_runner_; |
| fuchsia::sys::RunnerPtr cast_runner_ptr_; |
| |
| DISALLOW_COPY_AND_ASSIGN(CastRunnerIntegrationTest); |
| }; |
| |
| // A basic integration test ensuring a basic cast request launches the right |
| // URL in the Chromium service. |
| TEST_F(CastRunnerIntegrationTest, BasicRequest) { |
| const char kBlankAppId[] = "00000000"; |
| const char kBlankAppPath[] = "/defaultresponse"; |
| app_config_manager_.AddAppMapping(kBlankAppId, |
| test_server_.GetURL(kBlankAppPath)); |
| |
| // Launch the test-app component. |
| fuchsia::sys::ComponentControllerPtr component_controller = |
| StartCastComponent(base::StringPrintf("cast:%s", kBlankAppId)); |
| component_controller.set_error_handler(&ComponentErrorHandler); |
| |
| // Access the NavigationController from the WebComponent. The test will hang |
| // here if no WebComponent was created. |
| fuchsia::web::NavigationControllerPtr nav_controller; |
| { |
| base::RunLoop run_loop; |
| cr_fuchsia::ResultReceiver<WebComponent*> web_component( |
| run_loop.QuitClosure()); |
| cast_runner_->GetWebComponentForTest(web_component.GetReceiveCallback()); |
| run_loop.Run(); |
| ASSERT_NE(*web_component, nullptr); |
| (*web_component) |
| ->frame() |
| ->GetNavigationController(nav_controller.NewRequest()); |
| } |
| |
| // Ensure the NavigationState has the expected URL. |
| { |
| base::RunLoop run_loop; |
| cr_fuchsia::ResultReceiver<fuchsia::web::NavigationState> nav_entry( |
| run_loop.QuitClosure()); |
| nav_controller->GetVisibleEntry( |
| cr_fuchsia::CallbackToFitFunction(nav_entry.GetReceiveCallback())); |
| run_loop.Run(); |
| ASSERT_TRUE(nav_entry->has_url()); |
| EXPECT_EQ(nav_entry->url(), test_server_.GetURL(kBlankAppPath).spec()); |
| } |
| |
| // Verify that the component is torn down when |component_controller| is |
| // unbound. |
| base::RunLoop run_loop; |
| component_state_->set_on_delete(run_loop.QuitClosure()); |
| component_controller.Unbind(); |
| run_loop.Run(); |
| } |
| |
| TEST_F(CastRunnerIntegrationTest, IncorrectCastAppId) { |
| // Launch the a component with an invalid Cast app Id. |
| fuchsia::sys::ComponentControllerPtr component_controller = |
| StartCastComponent("cast:99999999"); |
| component_controller.set_error_handler(&ComponentErrorHandler); |
| |
| // Run the loop until the ComponentController is dropped, or a WebComponent is |
| // created. |
| base::RunLoop run_loop; |
| component_controller.set_error_handler([&run_loop](zx_status_t status) { |
| EXPECT_EQ(status, ZX_ERR_PEER_CLOSED); |
| run_loop.Quit(); |
| }); |
| cr_fuchsia::ResultReceiver<WebComponent*> web_component( |
| run_loop.QuitClosure()); |
| cast_runner_->GetWebComponentForTest(web_component.GetReceiveCallback()); |
| run_loop.Run(); |
| EXPECT_FALSE(web_component.has_value()); |
| } |
| |
| TEST_F(CastRunnerIntegrationTest, CastChannel) { |
| const char kCastChannelAppId[] = "00000001"; |
| const char kCastChannelAppPath[] = "/cast_channel.html"; |
| app_config_manager_.AddAppMapping(kCastChannelAppId, |
| test_server_.GetURL(kCastChannelAppPath)); |
| |
| // Launch the test-app component. |
| fuchsia::sys::ComponentControllerPtr component_controller = |
| StartCastComponent(base::StringPrintf("cast:%s", kCastChannelAppId)); |
| component_controller.set_error_handler(&ComponentErrorHandler); |
| |
| // Access the NavigationController from the WebComponent. The test will hang |
| // here if no WebComponent was created. |
| fuchsia::web::NavigationControllerPtr nav_controller; |
| { |
| base::RunLoop run_loop; |
| cr_fuchsia::ResultReceiver<WebComponent*> web_component( |
| run_loop.QuitClosure()); |
| cast_runner_->GetWebComponentForTest(web_component.GetReceiveCallback()); |
| run_loop.Run(); |
| ASSERT_NE(*web_component, nullptr); |
| (*web_component) |
| ->frame() |
| ->GetNavigationController(nav_controller.NewRequest()); |
| } |
| |
| // Ensure the NavigationState has the expected URL. |
| { |
| base::RunLoop run_loop; |
| cr_fuchsia::ResultReceiver<fuchsia::web::NavigationState> nav_entry( |
| run_loop.QuitClosure()); |
| nav_controller->GetVisibleEntry( |
| cr_fuchsia::CallbackToFitFunction(nav_entry.GetReceiveCallback())); |
| run_loop.Run(); |
| ASSERT_TRUE(nav_entry->has_url()); |
| EXPECT_EQ(nav_entry->url(), |
| test_server_.GetURL(kCastChannelAppPath).spec()); |
| } |
| |
| WaitUntilCastChannelOpened(); |
| |
| auto expected_list = {"this", "is", "a", "test"}; |
| for (const std::string& expected : expected_list) { |
| base::RunLoop run_loop; |
| cr_fuchsia::ResultReceiver<chromium::web::WebMessage> message( |
| run_loop.QuitClosure()); |
| component_state_->cast_channel()->port()->ReceiveMessage( |
| cr_fuchsia::CallbackToFitFunction(message.GetReceiveCallback())); |
| run_loop.Run(); |
| |
| std::string data; |
| ASSERT_TRUE(cr_fuchsia::StringFromMemBuffer(message->data, &data)); |
| EXPECT_EQ(data, expected); |
| } |
| |
| // Shutdown the component and wait for the teardown of its state. |
| base::RunLoop run_loop; |
| component_state_->set_on_delete(run_loop.QuitClosure()); |
| component_controller.Unbind(); |
| run_loop.Run(); |
| } |
| |
| TEST_F(CastRunnerIntegrationTest, CastChannelConsumerDropped) { |
| const char kCastChannelAppId[] = "00000001"; |
| const char kCastChannelAppPath[] = "/cast_channel.html"; |
| app_config_manager_.AddAppMapping(kCastChannelAppId, |
| test_server_.GetURL(kCastChannelAppPath)); |
| |
| // Launch the test-app component. |
| fuchsia::sys::ComponentControllerPtr component_controller = |
| StartCastComponent(base::StringPrintf("cast:%s", kCastChannelAppId)); |
| |
| // Spin the message loop to handle creation of the component state. |
| base::RunLoop().RunUntilIdle(); |
| ASSERT_TRUE(component_state_); |
| |
| // Disconnecting the CastChannel should trigger the component to teardown, |
| // resulting in our ComponentControllerPtr being dropped, and the component |
| // state held by the agent being deleted. |
| component_state_->set_on_delete(base::MakeExpectedRunClosure(FROM_HERE)); |
| component_state_->ClearCastChannel(); |
| base::RunLoop run_loop; |
| component_controller.set_error_handler([&run_loop](zx_status_t status) { |
| EXPECT_EQ(status, ZX_ERR_PEER_CLOSED); |
| run_loop.Quit(); |
| }); |
| run_loop.Run(); |
| EXPECT_FALSE(component_controller.is_bound()); |
| |
| // Give the fake agent an opportunity to perform teardown cleanup. |
| base::RunLoop().RunUntilIdle(); |
| } |
| |
| TEST_F(CastRunnerIntegrationTest, CastChannelComponentControllerDropped) { |
| const char kCastChannelAppId[] = "00000001"; |
| const char kCastChannelAppPath[] = "/cast_channel.html"; |
| app_config_manager_.AddAppMapping(kCastChannelAppId, |
| test_server_.GetURL(kCastChannelAppPath)); |
| |
| // Launch the test-app component. |
| fuchsia::sys::ComponentControllerPtr component_controller = |
| StartCastComponent(base::StringPrintf("cast:%s", kCastChannelAppId)); |
| |
| // Spin the message loop to handle creation of the component state. |
| base::RunLoop().RunUntilIdle(); |
| ASSERT_TRUE(component_state_); |
| |
| // Expect that disconnecting the ComponentController will destroy the Cast |
| // component. |
| base::RunLoop run_loop; |
| component_state_->set_on_delete(run_loop.QuitClosure()); |
| component_controller.Unbind(); |
| run_loop.Run(); |
| } |
| |
| } // namespace castrunner |