Index: chrome/browser/media/router/discovery/mdns/cast_media_sink_service_unittest.cc |
diff --git a/chrome/browser/media/router/discovery/mdns/cast_media_sink_service_unittest.cc b/chrome/browser/media/router/discovery/mdns/cast_media_sink_service_unittest.cc |
new file mode 100644 |
index 0000000000000000000000000000000000000000..7e12a97596500edeca3b1df9530c3a218431c756 |
--- /dev/null |
+++ b/chrome/browser/media/router/discovery/mdns/cast_media_sink_service_unittest.cc |
@@ -0,0 +1,309 @@ |
+// Copyright 2017 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 "chrome/browser/media/router/discovery/mdns/cast_media_sink_service.h" |
+#include "base/strings/stringprintf.h" |
+#include "base/test/mock_callback.h" |
+#include "base/timer/mock_timer.h" |
+#include "chrome/browser/media/router/discovery/mdns/mock_dns_sd_registry.h" |
+#include "chrome/browser/media/router/test_helper.h" |
+#include "chrome/test/base/testing_profile.h" |
+#include "components/cast_channel/cast_socket.h" |
+#include "components/cast_channel/cast_socket_service.h" |
+#include "components/cast_channel/cast_test_util.h" |
+#include "content/public/test/test_browser_thread_bundle.h" |
+#include "testing/gmock/include/gmock/gmock.h" |
+#include "testing/gtest/include/gtest/gtest.h" |
+ |
+using ::testing::_; |
+using ::testing::Return; |
+using ::testing::SaveArg; |
+ |
+namespace { |
+ |
+net::IPEndPoint CreateIPEndPoint(int num) { |
+ net::IPAddress ip_address; |
+ CHECK(ip_address.AssignFromIPLiteral( |
+ base::StringPrintf("192.168.0.10%d", num))); |
+ return net::IPEndPoint(ip_address, 8009 + num); |
+} |
+ |
+media_router::DnsSdService CreateDnsService(int num) { |
+ net::IPEndPoint ip_endpoint = CreateIPEndPoint(num); |
+ media_router::DnsSdService service; |
+ service.service_name = |
+ "_myDevice." + |
+ std::string(media_router::CastMediaSinkService::kCastServiceType); |
+ service.ip_address = ip_endpoint.address().ToString(); |
+ service.service_host_port = ip_endpoint.ToString(); |
+ service.service_data.push_back(base::StringPrintf("id=service %d", num)); |
+ service.service_data.push_back( |
+ base::StringPrintf("fn=friendly name %d", num)); |
+ service.service_data.push_back(base::StringPrintf("md=model name %d", num)); |
+ |
+ return service; |
+} |
+ |
+void VerifyMediaSinkInternal(const media_router::MediaSinkInternal& cast_sink, |
+ const media_router::DnsSdService& service, |
+ int channel_id, |
+ bool audio_only) { |
+ std::string id = base::StringPrintf("service %d", channel_id); |
+ std::string name = base::StringPrintf("friendly name %d", channel_id); |
+ std::string model_name = base::StringPrintf("model name %d", channel_id); |
+ EXPECT_EQ(id, cast_sink.sink().id()); |
+ EXPECT_EQ(name, cast_sink.sink().name()); |
+ EXPECT_EQ(model_name, cast_sink.cast_data().model_name); |
+ EXPECT_EQ(service.ip_address, cast_sink.cast_data().ip_address.ToString()); |
+ |
+ int capabilities = cast_channel::CastDeviceCapability::AUDIO_OUT; |
+ if (!audio_only) |
+ capabilities |= cast_channel::CastDeviceCapability::VIDEO_OUT; |
+ EXPECT_EQ(capabilities, cast_sink.cast_data().capabilities); |
+ EXPECT_EQ(channel_id, cast_sink.cast_data().cast_channel_id); |
+} |
+ |
+} // namespace |
+ |
+namespace media_router { |
+ |
+class MockCastSocketService : public cast_channel::CastSocketService { |
+ public: |
+ MOCK_METHOD4(OpenSocket, |
+ int(const net::IPEndPoint& ip_endpoint, |
+ net::NetLog* net_log, |
+ const cast_channel::CastSocket::OnOpenCallback& open_cb, |
+ cast_channel::CastSocket::Observer* observer)); |
+ MOCK_CONST_METHOD1(GetSocket, cast_channel::CastSocket*(int channel_id)); |
+ |
+ private: |
+ ~MockCastSocketService() {} |
+}; |
+ |
+class CastMediaSinkServiceTest : public ::testing::Test { |
+ public: |
+ CastMediaSinkServiceTest() |
+ : mock_cast_socket_service_(new MockCastSocketService()), |
+ media_sink_service_( |
+ new CastMediaSinkService(mock_sink_discovered_cb_.Get(), |
+ mock_cast_socket_service_.get())), |
+ test_dns_sd_registry_(media_sink_service_.get()) {} |
+ |
+ void SetUp() override { |
+ auto mock_timer = base::MakeUnique<base::MockTimer>( |
+ true /*retain_user_task*/, false /*is_repeating*/); |
+ mock_timer_ = mock_timer.get(); |
+ media_sink_service_->SetTimerForTest(std::move(mock_timer)); |
+ } |
+ |
+ protected: |
+ const content::TestBrowserThreadBundle thread_bundle_; |
+ TestingProfile profile_; |
+ |
+ base::MockCallback<MediaSinkService::OnSinksDiscoveredCallback> |
+ mock_sink_discovered_cb_; |
+ scoped_refptr<MockCastSocketService> mock_cast_socket_service_; |
+ scoped_refptr<CastMediaSinkService> media_sink_service_; |
+ MockDnsSdRegistry test_dns_sd_registry_; |
+ base::MockTimer* mock_timer_; |
+ |
+ DISALLOW_COPY_AND_ASSIGN(CastMediaSinkServiceTest); |
+}; |
+ |
+TEST_F(CastMediaSinkServiceTest, TestReStartAfterStop) { |
+ EXPECT_CALL(test_dns_sd_registry_, AddObserver(media_sink_service_.get())) |
+ .Times(2); |
+ EXPECT_CALL(test_dns_sd_registry_, RegisterDnsSdListener(_)).Times(2); |
+ EXPECT_FALSE(mock_timer_->IsRunning()); |
+ media_sink_service_->SetDnsSdRegistryForTest(&test_dns_sd_registry_); |
+ media_sink_service_->Start(); |
+ EXPECT_TRUE(mock_timer_->IsRunning()); |
+ |
+ EXPECT_CALL(test_dns_sd_registry_, RemoveObserver(media_sink_service_.get())); |
+ EXPECT_CALL(test_dns_sd_registry_, UnregisterDnsSdListener(_)); |
+ media_sink_service_->Stop(); |
+ |
+ mock_timer_ = |
+ new base::MockTimer(true /*retain_user_task*/, false /*is_repeating*/); |
+ media_sink_service_->SetTimerForTest(base::WrapUnique(mock_timer_)); |
+ media_sink_service_->SetDnsSdRegistryForTest(&test_dns_sd_registry_); |
+ media_sink_service_->Start(); |
+ EXPECT_TRUE(mock_timer_->IsRunning()); |
+} |
+ |
+TEST_F(CastMediaSinkServiceTest, TestMultipleStartAndStop) { |
+ EXPECT_CALL(test_dns_sd_registry_, AddObserver(media_sink_service_.get())); |
+ EXPECT_CALL(test_dns_sd_registry_, RegisterDnsSdListener(_)); |
+ media_sink_service_->SetDnsSdRegistryForTest(&test_dns_sd_registry_); |
+ media_sink_service_->Start(); |
+ media_sink_service_->Start(); |
+ EXPECT_TRUE(mock_timer_->IsRunning()); |
+ |
+ EXPECT_CALL(test_dns_sd_registry_, RemoveObserver(media_sink_service_.get())); |
+ EXPECT_CALL(test_dns_sd_registry_, UnregisterDnsSdListener(_)); |
+ media_sink_service_->Stop(); |
+ media_sink_service_->Stop(); |
+} |
+ |
+TEST_F(CastMediaSinkServiceTest, TestOnChannelOpenedOnIOThread) { |
+ DnsSdService service = CreateDnsService(1); |
+ cast_channel::MockCastSocket socket; |
+ EXPECT_CALL(*mock_cast_socket_service_, GetSocket(1)) |
+ .WillOnce(Return(&socket)); |
+ |
+ media_sink_service_->current_services_.push_back(service); |
+ media_sink_service_->OnChannelOpenedOnIOThread( |
+ service, 1, cast_channel::ChannelError::NONE); |
+ // Invoke CastMediaSinkService::OnChannelOpenedOnUIThread on the UI thread. |
+ base::RunLoop().RunUntilIdle(); |
+ |
+ // Verify sink content |
+ EXPECT_EQ(size_t(1), media_sink_service_->current_sinks_.size()); |
+ for (const auto& sink_it : media_sink_service_->current_sinks_) |
+ VerifyMediaSinkInternal(sink_it, service, 1, false); |
+} |
+ |
+TEST_F(CastMediaSinkServiceTest, TestMultipleOnChannelOpenedOnIOThread) { |
+ DnsSdService service1 = CreateDnsService(1); |
+ DnsSdService service2 = CreateDnsService(2); |
+ DnsSdService service3 = CreateDnsService(3); |
+ |
+ cast_channel::MockCastSocket socket2; |
+ cast_channel::MockCastSocket socket3; |
+ // Fail to open channel 1. |
+ EXPECT_CALL(*mock_cast_socket_service_, GetSocket(1)) |
+ .WillOnce(Return(nullptr)); |
+ EXPECT_CALL(*mock_cast_socket_service_, GetSocket(2)) |
+ .WillOnce(Return(&socket2)); |
+ EXPECT_CALL(*mock_cast_socket_service_, GetSocket(3)) |
+ .WillOnce(Return(&socket2)); |
+ |
+ // Current round of Dns discovery finds service1 and service 2. |
+ media_sink_service_->current_services_.push_back(service1); |
+ media_sink_service_->current_services_.push_back(service2); |
+ media_sink_service_->OnChannelOpenedOnIOThread( |
+ service1, 1, cast_channel::ChannelError::NONE); |
+ media_sink_service_->OnChannelOpenedOnIOThread( |
+ service2, 2, cast_channel::ChannelError::NONE); |
+ media_sink_service_->OnChannelOpenedOnIOThread( |
+ service3, 3, cast_channel::ChannelError::NONE); |
+ // Invoke CastMediaSinkService::OnChannelOpenedOnUIThread on the UI thread. |
+ base::RunLoop().RunUntilIdle(); |
+ |
+ // Verify sink content |
+ EXPECT_EQ(size_t(1), media_sink_service_->current_sinks_.size()); |
+ for (const auto& sink_it : media_sink_service_->current_sinks_) |
+ VerifyMediaSinkInternal(sink_it, service2, 2, false); |
+} |
+ |
+TEST_F(CastMediaSinkServiceTest, TestOnDnsSdEvent) { |
+ DnsSdService service1 = CreateDnsService(1); |
+ DnsSdService service2 = CreateDnsService(2); |
+ net::IPEndPoint ip_endpoint1 = CreateIPEndPoint(1); |
+ net::IPEndPoint ip_endpoint2 = CreateIPEndPoint(2); |
+ |
+ // Add dns services. |
+ DnsSdRegistry::DnsSdServiceList service_list{service1, service2}; |
+ |
+ cast_channel::CastSocket::OnOpenCallback callback1; |
+ cast_channel::CastSocket::OnOpenCallback callback2; |
+ EXPECT_CALL(*mock_cast_socket_service_, OpenSocket(ip_endpoint1, _, _, _)) |
+ .WillOnce(DoAll(SaveArg<2>(&callback1), Return(1))); |
+ EXPECT_CALL(*mock_cast_socket_service_, OpenSocket(ip_endpoint2, _, _, _)) |
+ .WillOnce(DoAll(SaveArg<2>(&callback2), Return(2))); |
+ |
+ // Invoke CastSocketService::OpenSocket on the IO thread. |
+ media_sink_service_->OnDnsSdEvent(CastMediaSinkService::kCastServiceType, |
+ service_list); |
+ base::RunLoop().RunUntilIdle(); |
+ |
+ cast_channel::MockCastSocket socket1; |
+ cast_channel::MockCastSocket socket2; |
+ |
+ EXPECT_CALL(*mock_cast_socket_service_, GetSocket(1)) |
+ .WillOnce(Return(&socket1)); |
+ EXPECT_CALL(*mock_cast_socket_service_, GetSocket(2)) |
+ .WillOnce(Return(&socket2)); |
+ |
+ callback1.Run(1, cast_channel::ChannelError::NONE); |
+ callback2.Run(2, cast_channel::ChannelError::NONE); |
+ |
+ // Invoke CastMediaSinkService::OnChannelOpenedOnUIThread on the UI thread. |
+ base::RunLoop().RunUntilIdle(); |
+ // Verify sink content |
+ EXPECT_EQ(size_t(2), media_sink_service_->current_sinks_.size()); |
+} |
+ |
+TEST_F(CastMediaSinkServiceTest, TestMultipleOnDnsSdEvent) { |
+ DnsSdService service1 = CreateDnsService(1); |
+ DnsSdService service2 = CreateDnsService(2); |
+ DnsSdService service3 = CreateDnsService(3); |
+ net::IPEndPoint ip_endpoint1 = CreateIPEndPoint(1); |
+ net::IPEndPoint ip_endpoint2 = CreateIPEndPoint(2); |
+ net::IPEndPoint ip_endpoint3 = CreateIPEndPoint(3); |
+ |
+ // 1st round finds service 1 & 2. |
+ DnsSdRegistry::DnsSdServiceList service_list1{service1, service2}; |
+ media_sink_service_->OnDnsSdEvent(CastMediaSinkService::kCastServiceType, |
+ service_list1); |
+ |
+ EXPECT_CALL(*mock_cast_socket_service_, OpenSocket(ip_endpoint1, _, _, _)); |
+ EXPECT_CALL(*mock_cast_socket_service_, OpenSocket(ip_endpoint2, _, _, _)); |
+ base::RunLoop().RunUntilIdle(); |
+ |
+ // Channel 2 opened. |
+ media_sink_service_->OnChannelOpenedOnUIThread(service2, 2, false); |
+ |
+ // 2nd round finds service 2 & 3. |
+ DnsSdRegistry::DnsSdServiceList service_list2{service2, service3}; |
+ media_sink_service_->OnDnsSdEvent(CastMediaSinkService::kCastServiceType, |
+ service_list2); |
+ |
+ EXPECT_CALL(*mock_cast_socket_service_, OpenSocket(ip_endpoint2, _, _, _)); |
+ EXPECT_CALL(*mock_cast_socket_service_, OpenSocket(ip_endpoint3, _, _, _)); |
+ base::RunLoop().RunUntilIdle(); |
+ |
+ // Channel 1 and 3 opened. |
+ media_sink_service_->OnChannelOpenedOnUIThread(service1, 1, false); |
+ media_sink_service_->OnChannelOpenedOnUIThread(service3, 3, false); |
+ |
+ EXPECT_EQ(size_t(1), media_sink_service_->current_sinks_.size()); |
+ for (const auto& sink_it : media_sink_service_->current_sinks_) |
+ VerifyMediaSinkInternal(sink_it, service3, 3, false); |
+} |
+ |
+TEST_F(CastMediaSinkServiceTest, TestTimer) { |
+ DnsSdService service1 = CreateDnsService(1); |
+ DnsSdService service2 = CreateDnsService(2); |
+ net::IPEndPoint ip_endpoint1 = CreateIPEndPoint(1); |
+ net::IPEndPoint ip_endpoint2 = CreateIPEndPoint(2); |
+ |
+ EXPECT_FALSE(mock_timer_->IsRunning()); |
+ // finds service 1 & 2. |
+ DnsSdRegistry::DnsSdServiceList service_list1{service1, service2}; |
+ media_sink_service_->OnDnsSdEvent(CastMediaSinkService::kCastServiceType, |
+ service_list1); |
+ |
+ EXPECT_CALL(*mock_cast_socket_service_, OpenSocket(ip_endpoint1, _, _, _)); |
+ EXPECT_CALL(*mock_cast_socket_service_, OpenSocket(ip_endpoint2, _, _, _)); |
+ base::RunLoop().RunUntilIdle(); |
+ |
+ // Channel 2 is opened. |
+ media_sink_service_->OnChannelOpenedOnUIThread(service2, 2, false); |
+ |
+ std::vector<MediaSinkInternal> sinks; |
+ EXPECT_CALL(mock_sink_discovered_cb_, Run(_)).WillOnce(SaveArg<0>(&sinks)); |
+ |
+ // Fire timer. |
+ mock_timer_->Fire(); |
+ EXPECT_EQ(size_t(1), sinks.size()); |
+ VerifyMediaSinkInternal(sinks[0], service2, 2, false); |
+ |
+ EXPECT_FALSE(mock_timer_->IsRunning()); |
+ // Channel 1 is opened and timer is restarted. |
+ media_sink_service_->OnChannelOpenedOnUIThread(service1, 1, false); |
+ EXPECT_TRUE(mock_timer_->IsRunning()); |
+} |
+ |
+} // namespace media_router |