// Copyright 2014 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 #include #include "base/bind.h" #include "base/macros.h" #include "base/memory/ptr_util.h" #include "base/message_loop/message_loop.h" #include "base/run_loop.h" #include "base/single_thread_task_runner.h" #include "base/threading/thread_task_runner_handle.h" #include "device/serial/serial.mojom.h" #include "device/serial/serial_service_impl.h" #include "device/serial/test_serial_io_handler.h" #include "mojo/public/cpp/bindings/interface_ptr.h" #include "mojo/public/cpp/bindings/interface_request.h" #include "mojo/public/cpp/bindings/strong_binding.h" #include "testing/gtest/include/gtest/gtest.h" namespace device { namespace { class FakeSerialDeviceEnumerator : public SerialDeviceEnumerator { std::vector GetDevices() override { std::vector devices(1); devices[0] = serial::DeviceInfo::New(); devices[0]->path = "device"; return devices; } }; class FailToOpenIoHandler : public TestSerialIoHandler { public: void Open(const std::string& port, const serial::ConnectionOptions& options, const OpenCompleteCallback& callback) override { callback.Run(false); } protected: ~FailToOpenIoHandler() override {} }; } // namespace class SerialServiceTest : public testing::Test { public: SerialServiceTest() : connected_(false), expecting_error_(false) {} void StoreDevices(std::vector devices) { devices_ = std::move(devices); StopMessageLoop(); } void OnConnectionError() { StopMessageLoop(); EXPECT_TRUE(expecting_error_); } void RunMessageLoop() { run_loop_.reset(new base::RunLoop); run_loop_->Run(); } void StopMessageLoop() { ASSERT_TRUE(run_loop_); message_loop_.task_runner()->PostTask(FROM_HERE, run_loop_->QuitClosure()); } void OnGotInfo(serial::ConnectionInfoPtr options) { connected_ = true; StopMessageLoop(); } scoped_refptr ReturnIoHandler() { return io_handler_; } void RunConnectTest(const std::string& path, bool expecting_success) { if (!io_handler_.get()) io_handler_ = new TestSerialIoHandler; mojo::InterfacePtr service; mojo::MakeStrongBinding( base::MakeUnique( new SerialConnectionFactory( base::Bind(&SerialServiceTest::ReturnIoHandler, base::Unretained(this)), base::ThreadTaskRunnerHandle::Get()), base::MakeUnique()), mojo::MakeRequest(&service)); mojo::InterfacePtr connection; mojo::InterfacePtr sink; mojo::InterfacePtr source; mojo::InterfacePtr source_client; mojo::MakeRequest(&source_client); service->Connect(path, serial::ConnectionOptions::New(), mojo::MakeRequest(&connection), mojo::MakeRequest(&sink), mojo::MakeRequest(&source), std::move(source_client)); connection.set_connection_error_handler(base::Bind( &SerialServiceTest::OnConnectionError, base::Unretained(this))); expecting_error_ = !expecting_success; connection->GetInfo( base::Bind(&SerialServiceTest::OnGotInfo, base::Unretained(this))); RunMessageLoop(); EXPECT_EQ(!expecting_success, connection.encountered_error()); EXPECT_EQ(expecting_success, connected_); connection.reset(); } base::MessageLoop message_loop_; std::unique_ptr run_loop_; std::vector devices_; scoped_refptr io_handler_; bool connected_; bool expecting_error_; serial::ConnectionInfoPtr info_; private: DISALLOW_COPY_AND_ASSIGN(SerialServiceTest); }; TEST_F(SerialServiceTest, GetDevices) { mojo::InterfacePtr service; SerialServiceImpl::Create(NULL, NULL, mojo::MakeRequest(&service)); service.set_connection_error_handler(base::Bind( &SerialServiceTest::OnConnectionError, base::Unretained(this))); std::vector result; service->GetDevices( base::Bind(&SerialServiceTest::StoreDevices, base::Unretained(this))); RunMessageLoop(); // Because we're running on unknown hardware, only check that we received a // non-null result. EXPECT_TRUE(devices_); } TEST_F(SerialServiceTest, Connect) { RunConnectTest("device", true); } TEST_F(SerialServiceTest, ConnectInvalidPath) { RunConnectTest("invalid_path", false); } TEST_F(SerialServiceTest, ConnectOpenFailed) { io_handler_ = new FailToOpenIoHandler; RunConnectTest("device", false); } } // namespace device