Chromium Code Reviews| 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..02ec5be8aa464e344124739ee5c64b62078ffbd4 |
| --- /dev/null |
| +++ b/chrome/browser/media/router/discovery/mdns/cast_media_sink_service_unittest.cc |
| @@ -0,0 +1,301 @@ |
| +// 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 { |
| + |
| +const char SERVICE_TYPE[] = "_googlecast._tcp.local"; |
|
mark a. foltz
2017/06/28 22:43:13
This duplicates the definition in cast_media_sink_
zhaobin
2017/07/05 18:01:37
Done.
|
| + |
| +net::IPEndPoint CreateIPEndPoint(int num) { |
| + net::IPAddress ip_address; |
| + DCHECK(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(SERVICE_TYPE); |
|
mark a. foltz
2017/06/28 22:43:13
I thing std::string is optional here.
zhaobin
2017/07/05 18:01:37
Compile error if removed...cannot add 'const char*
|
| + 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()); |
| + EXPECT_TRUE(cast_channel::CastDeviceCapability::AUDIO_OUT & |
| + cast_sink.cast_data().capabilities); |
| + EXPECT_EQ(!audio_only, cast_channel::CastDeviceCapability::VIDEO_OUT & |
| + 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 { |
| + media_sink_service_->SetDnsSdRegistryForTest(&test_dns_sd_registry_); |
| + mock_timer_ = |
| + new base::MockTimer(true /*retain_user_task*/, false /*is_repeating*/); |
| + media_sink_service_->SetTimerForTest(base::WrapUnique(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_; |
|
mark a. foltz
2017/06/28 22:43:13
Consider declaring this as std::unique_ptr and usi
zhaobin
2017/07/05 18:01:37
Ownership of |mock_timer_| is passed into MediaSin
|
| + |
| + 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_->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_->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_->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(SERVICE_TYPE, 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(SERVICE_TYPE, 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(SERVICE_TYPE, 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(SERVICE_TYPE, 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 |