/* * libjingle * Copyright 2012 Google Inc. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * * 1. Redistributions of source code must retain the above copyright notice, * this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright notice, * this list of conditions and the following disclaimer in the documentation * and/or other materials provided with the distribution. * 3. The name of the author may not be used to endorse or promote products * derived from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO * EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include #include "talk/app/webrtc/fakeportallocatorfactory.h" #include "talk/app/webrtc/mediastreaminterface.h" #include "talk/app/webrtc/peerconnectionfactory.h" #include "talk/app/webrtc/test/fakedtlsidentitystore.h" #include "talk/app/webrtc/test/fakevideotrackrenderer.h" #include "talk/app/webrtc/videosourceinterface.h" #include "talk/media/base/fakevideocapturer.h" #include "talk/media/webrtc/webrtccommon.h" #include "talk/media/webrtc/webrtcvoe.h" #include "webrtc/base/gunit.h" #include "webrtc/base/scoped_ptr.h" #include "webrtc/base/thread.h" using webrtc::DataChannelInterface; using webrtc::DtlsIdentityStoreInterface; using webrtc::FakeVideoTrackRenderer; using webrtc::MediaStreamInterface; using webrtc::PeerConnectionFactoryInterface; using webrtc::PeerConnectionInterface; using webrtc::PeerConnectionObserver; using webrtc::PortAllocatorFactoryInterface; using webrtc::VideoSourceInterface; using webrtc::VideoTrackInterface; namespace { typedef std::vector StunConfigurations; typedef std::vector TurnConfigurations; static const char kStunIceServer[] = "stun:stun.l.google.com:19302"; static const char kTurnIceServer[] = "turn:test%40hello.com@test.com:1234"; static const char kTurnIceServerWithTransport[] = "turn:test@hello.com?transport=tcp"; static const char kSecureTurnIceServer[] = "turns:test@hello.com?transport=tcp"; static const char kSecureTurnIceServerWithoutTransportParam[] = "turns:test_no_transport@hello.com:443"; static const char kSecureTurnIceServerWithoutTransportAndPortParam[] = "turns:test_no_transport@hello.com"; static const char kTurnIceServerWithNoUsernameInUri[] = "turn:test.com:1234"; static const char kTurnPassword[] = "turnpassword"; static const int kDefaultStunPort = 3478; static const int kDefaultStunTlsPort = 5349; static const char kTurnUsername[] = "test"; static const char kStunIceServerWithIPv4Address[] = "stun:1.2.3.4:1234"; static const char kStunIceServerWithIPv4AddressWithoutPort[] = "stun:1.2.3.4"; static const char kStunIceServerWithIPv6Address[] = "stun:[2401:fa00:4::]:1234"; static const char kStunIceServerWithIPv6AddressWithoutPort[] = "stun:[2401:fa00:4::]"; static const char kTurnIceServerWithIPv6Address[] = "turn:test@[2401:fa00:4::]:1234"; class NullPeerConnectionObserver : public PeerConnectionObserver { public: virtual void OnMessage(const std::string& msg) {} virtual void OnSignalingMessage(const std::string& msg) {} virtual void OnSignalingChange( PeerConnectionInterface::SignalingState new_state) {} virtual void OnAddStream(MediaStreamInterface* stream) {} virtual void OnRemoveStream(MediaStreamInterface* stream) {} virtual void OnDataChannel(DataChannelInterface* data_channel) {} virtual void OnRenegotiationNeeded() {} virtual void OnIceConnectionChange( PeerConnectionInterface::IceConnectionState new_state) {} virtual void OnIceGatheringChange( PeerConnectionInterface::IceGatheringState new_state) {} virtual void OnIceCandidate(const webrtc::IceCandidateInterface* candidate) {} }; } // namespace class PeerConnectionFactoryTest : public testing::Test { void SetUp() { factory_ = webrtc::CreatePeerConnectionFactory(rtc::Thread::Current(), rtc::Thread::Current(), NULL, NULL, NULL); ASSERT_TRUE(factory_.get() != NULL); allocator_factory_ = webrtc::FakePortAllocatorFactory::Create(); } protected: void VerifyStunConfigurations(StunConfigurations stun_config) { webrtc::FakePortAllocatorFactory* allocator = static_cast( allocator_factory_.get()); ASSERT_TRUE(allocator != NULL); EXPECT_EQ(stun_config.size(), allocator->stun_configs().size()); for (size_t i = 0; i < stun_config.size(); ++i) { EXPECT_EQ(stun_config[i].server.ToString(), allocator->stun_configs()[i].server.ToString()); } } void VerifyTurnConfigurations(TurnConfigurations turn_config) { webrtc::FakePortAllocatorFactory* allocator = static_cast( allocator_factory_.get()); ASSERT_TRUE(allocator != NULL); EXPECT_EQ(turn_config.size(), allocator->turn_configs().size()); for (size_t i = 0; i < turn_config.size(); ++i) { EXPECT_EQ(turn_config[i].server.ToString(), allocator->turn_configs()[i].server.ToString()); EXPECT_EQ(turn_config[i].username, allocator->turn_configs()[i].username); EXPECT_EQ(turn_config[i].password, allocator->turn_configs()[i].password); EXPECT_EQ(turn_config[i].transport_type, allocator->turn_configs()[i].transport_type); } } rtc::scoped_refptr factory_; NullPeerConnectionObserver observer_; rtc::scoped_refptr allocator_factory_; }; // Verify creation of PeerConnection using internal ADM, video factory and // internal libjingle threads. TEST(PeerConnectionFactoryTestInternal, CreatePCUsingInternalModules) { rtc::scoped_refptr factory( webrtc::CreatePeerConnectionFactory()); NullPeerConnectionObserver observer; webrtc::PeerConnectionInterface::IceServers servers; rtc::scoped_ptr dtls_identity_store( new FakeDtlsIdentityStore()); rtc::scoped_refptr pc( factory->CreatePeerConnection( servers, nullptr, nullptr, dtls_identity_store.Pass(), &observer)); EXPECT_TRUE(pc.get() != nullptr); } // This test verifies creation of PeerConnection with valid STUN and TURN // configuration. Also verifies the URL's parsed correctly as expected. TEST_F(PeerConnectionFactoryTest, CreatePCUsingIceServers) { PeerConnectionInterface::RTCConfiguration config; webrtc::PeerConnectionInterface::IceServer ice_server; ice_server.uri = kStunIceServer; config.servers.push_back(ice_server); ice_server.uri = kTurnIceServer; ice_server.password = kTurnPassword; config.servers.push_back(ice_server); ice_server.uri = kTurnIceServerWithTransport; ice_server.password = kTurnPassword; config.servers.push_back(ice_server); rtc::scoped_ptr dtls_identity_store( new FakeDtlsIdentityStore()); rtc::scoped_refptr pc( factory_->CreatePeerConnection(config, nullptr, allocator_factory_.get(), dtls_identity_store.Pass(), &observer_)); EXPECT_TRUE(pc.get() != NULL); StunConfigurations stun_configs; webrtc::PortAllocatorFactoryInterface::StunConfiguration stun1( "stun.l.google.com", 19302); stun_configs.push_back(stun1); VerifyStunConfigurations(stun_configs); TurnConfigurations turn_configs; webrtc::PortAllocatorFactoryInterface::TurnConfiguration turn1( "test.com", 1234, "test@hello.com", kTurnPassword, "udp", false); turn_configs.push_back(turn1); webrtc::PortAllocatorFactoryInterface::TurnConfiguration turn2( "hello.com", kDefaultStunPort, "test", kTurnPassword, "tcp", false); turn_configs.push_back(turn2); VerifyTurnConfigurations(turn_configs); } // This test verifies creation of PeerConnection with valid STUN and TURN // configuration. Also verifies the list of URL's parsed correctly as expected. TEST_F(PeerConnectionFactoryTest, CreatePCUsingIceServersUrls) { PeerConnectionInterface::RTCConfiguration config; webrtc::PeerConnectionInterface::IceServer ice_server; ice_server.urls.push_back(kStunIceServer); ice_server.urls.push_back(kTurnIceServer); ice_server.urls.push_back(kTurnIceServerWithTransport); ice_server.password = kTurnPassword; config.servers.push_back(ice_server); rtc::scoped_ptr dtls_identity_store( new FakeDtlsIdentityStore()); rtc::scoped_refptr pc( factory_->CreatePeerConnection(config, nullptr, allocator_factory_.get(), dtls_identity_store.Pass(), &observer_)); EXPECT_TRUE(pc.get() != NULL); StunConfigurations stun_configs; webrtc::PortAllocatorFactoryInterface::StunConfiguration stun1( "stun.l.google.com", 19302); stun_configs.push_back(stun1); VerifyStunConfigurations(stun_configs); TurnConfigurations turn_configs; webrtc::PortAllocatorFactoryInterface::TurnConfiguration turn1( "test.com", 1234, "test@hello.com", kTurnPassword, "udp", false); turn_configs.push_back(turn1); webrtc::PortAllocatorFactoryInterface::TurnConfiguration turn2( "hello.com", kDefaultStunPort, "test", kTurnPassword, "tcp", false); turn_configs.push_back(turn2); VerifyTurnConfigurations(turn_configs); } // This test verifies creation of PeerConnection with valid STUN and TURN // configuration. Also verifies the URL's parsed correctly as expected. // This version doesn't use RTCConfiguration. // TODO(mallinath) - Remove this method after clients start using RTCConfig. TEST_F(PeerConnectionFactoryTest, CreatePCUsingIceServersOldSignature) { webrtc::PeerConnectionInterface::IceServers ice_servers; webrtc::PeerConnectionInterface::IceServer ice_server; ice_server.uri = kStunIceServer; ice_servers.push_back(ice_server); ice_server.uri = kTurnIceServer; ice_server.password = kTurnPassword; ice_servers.push_back(ice_server); ice_server.uri = kTurnIceServerWithTransport; ice_server.password = kTurnPassword; ice_servers.push_back(ice_server); rtc::scoped_ptr dtls_identity_store( new FakeDtlsIdentityStore()); rtc::scoped_refptr pc( factory_->CreatePeerConnection(ice_servers, nullptr, allocator_factory_.get(), dtls_identity_store.Pass(), &observer_)); EXPECT_TRUE(pc.get() != NULL); StunConfigurations stun_configs; webrtc::PortAllocatorFactoryInterface::StunConfiguration stun1( "stun.l.google.com", 19302); stun_configs.push_back(stun1); VerifyStunConfigurations(stun_configs); TurnConfigurations turn_configs; webrtc::PortAllocatorFactoryInterface::TurnConfiguration turn1( "test.com", 1234, "test@hello.com", kTurnPassword, "udp", false); turn_configs.push_back(turn1); webrtc::PortAllocatorFactoryInterface::TurnConfiguration turn2( "hello.com", kDefaultStunPort, "test", kTurnPassword, "tcp", false); turn_configs.push_back(turn2); VerifyTurnConfigurations(turn_configs); } TEST_F(PeerConnectionFactoryTest, CreatePCUsingNoUsernameInUri) { PeerConnectionInterface::RTCConfiguration config; webrtc::PeerConnectionInterface::IceServer ice_server; ice_server.uri = kStunIceServer; config.servers.push_back(ice_server); ice_server.uri = kTurnIceServerWithNoUsernameInUri; ice_server.username = kTurnUsername; ice_server.password = kTurnPassword; config.servers.push_back(ice_server); rtc::scoped_ptr dtls_identity_store( new FakeDtlsIdentityStore()); rtc::scoped_refptr pc( factory_->CreatePeerConnection(config, nullptr, allocator_factory_.get(), dtls_identity_store.Pass(), &observer_)); EXPECT_TRUE(pc.get() != NULL); TurnConfigurations turn_configs; webrtc::PortAllocatorFactoryInterface::TurnConfiguration turn( "test.com", 1234, kTurnUsername, kTurnPassword, "udp", false); turn_configs.push_back(turn); VerifyTurnConfigurations(turn_configs); } // This test verifies the PeerConnection created properly with TURN url which // has transport parameter in it. TEST_F(PeerConnectionFactoryTest, CreatePCUsingTurnUrlWithTransportParam) { PeerConnectionInterface::RTCConfiguration config; webrtc::PeerConnectionInterface::IceServer ice_server; ice_server.uri = kTurnIceServerWithTransport; ice_server.password = kTurnPassword; config.servers.push_back(ice_server); rtc::scoped_ptr dtls_identity_store( new FakeDtlsIdentityStore()); rtc::scoped_refptr pc( factory_->CreatePeerConnection(config, nullptr, allocator_factory_.get(), dtls_identity_store.Pass(), &observer_)); EXPECT_TRUE(pc.get() != NULL); TurnConfigurations turn_configs; webrtc::PortAllocatorFactoryInterface::TurnConfiguration turn( "hello.com", kDefaultStunPort, "test", kTurnPassword, "tcp", false); turn_configs.push_back(turn); VerifyTurnConfigurations(turn_configs); } TEST_F(PeerConnectionFactoryTest, CreatePCUsingSecureTurnUrl) { PeerConnectionInterface::RTCConfiguration config; webrtc::PeerConnectionInterface::IceServer ice_server; ice_server.uri = kSecureTurnIceServer; ice_server.password = kTurnPassword; config.servers.push_back(ice_server); ice_server.uri = kSecureTurnIceServerWithoutTransportParam; ice_server.password = kTurnPassword; config.servers.push_back(ice_server); ice_server.uri = kSecureTurnIceServerWithoutTransportAndPortParam; ice_server.password = kTurnPassword; config.servers.push_back(ice_server); rtc::scoped_ptr dtls_identity_store( new FakeDtlsIdentityStore()); rtc::scoped_refptr pc( factory_->CreatePeerConnection(config, nullptr, allocator_factory_.get(), dtls_identity_store.Pass(), &observer_)); EXPECT_TRUE(pc.get() != NULL); TurnConfigurations turn_configs; webrtc::PortAllocatorFactoryInterface::TurnConfiguration turn1( "hello.com", kDefaultStunTlsPort, "test", kTurnPassword, "tcp", true); turn_configs.push_back(turn1); // TURNS with transport param should be default to tcp. webrtc::PortAllocatorFactoryInterface::TurnConfiguration turn2( "hello.com", 443, "test_no_transport", kTurnPassword, "tcp", true); turn_configs.push_back(turn2); webrtc::PortAllocatorFactoryInterface::TurnConfiguration turn3( "hello.com", kDefaultStunTlsPort, "test_no_transport", kTurnPassword, "tcp", true); turn_configs.push_back(turn3); VerifyTurnConfigurations(turn_configs); } TEST_F(PeerConnectionFactoryTest, CreatePCUsingIPLiteralAddress) { PeerConnectionInterface::RTCConfiguration config; webrtc::PeerConnectionInterface::IceServer ice_server; ice_server.uri = kStunIceServerWithIPv4Address; config.servers.push_back(ice_server); ice_server.uri = kStunIceServerWithIPv4AddressWithoutPort; config.servers.push_back(ice_server); ice_server.uri = kStunIceServerWithIPv6Address; config.servers.push_back(ice_server); ice_server.uri = kStunIceServerWithIPv6AddressWithoutPort; config.servers.push_back(ice_server); ice_server.uri = kTurnIceServerWithIPv6Address; ice_server.password = kTurnPassword; config.servers.push_back(ice_server); rtc::scoped_ptr dtls_identity_store( new FakeDtlsIdentityStore()); rtc::scoped_refptr pc( factory_->CreatePeerConnection(config, nullptr, allocator_factory_.get(), dtls_identity_store.Pass(), &observer_)); EXPECT_TRUE(pc.get() != NULL); StunConfigurations stun_configs; webrtc::PortAllocatorFactoryInterface::StunConfiguration stun1( "1.2.3.4", 1234); stun_configs.push_back(stun1); webrtc::PortAllocatorFactoryInterface::StunConfiguration stun2( "1.2.3.4", 3478); stun_configs.push_back(stun2); // Default port webrtc::PortAllocatorFactoryInterface::StunConfiguration stun3( "2401:fa00:4::", 1234); stun_configs.push_back(stun3); webrtc::PortAllocatorFactoryInterface::StunConfiguration stun4( "2401:fa00:4::", 3478); stun_configs.push_back(stun4); // Default port VerifyStunConfigurations(stun_configs); TurnConfigurations turn_configs; webrtc::PortAllocatorFactoryInterface::TurnConfiguration turn1( "2401:fa00:4::", 1234, "test", kTurnPassword, "udp", false); turn_configs.push_back(turn1); VerifyTurnConfigurations(turn_configs); } // This test verifies the captured stream is rendered locally using a // local video track. TEST_F(PeerConnectionFactoryTest, LocalRendering) { cricket::FakeVideoCapturer* capturer = new cricket::FakeVideoCapturer(); // The source take ownership of |capturer|. rtc::scoped_refptr source( factory_->CreateVideoSource(capturer, NULL)); ASSERT_TRUE(source.get() != NULL); rtc::scoped_refptr track( factory_->CreateVideoTrack("testlabel", source)); ASSERT_TRUE(track.get() != NULL); FakeVideoTrackRenderer local_renderer(track); EXPECT_EQ(0, local_renderer.num_rendered_frames()); EXPECT_TRUE(capturer->CaptureFrame()); EXPECT_EQ(1, local_renderer.num_rendered_frames()); track->set_enabled(false); EXPECT_TRUE(capturer->CaptureFrame()); EXPECT_EQ(1, local_renderer.num_rendered_frames()); track->set_enabled(true); EXPECT_TRUE(capturer->CaptureFrame()); EXPECT_EQ(2, local_renderer.num_rendered_frames()); }