| /* |
| * libjingle |
| * Copyright 2004--2011, 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 <stdio.h> |
| |
| #include <list> |
| |
| #include "talk/app/webrtcv1/unittest_utilities.h" |
| #include "talk/app/webrtcv1/webrtcsession.h" |
| #include "talk/base/fakenetwork.h" |
| #include "talk/base/gunit.h" |
| #include "talk/base/helpers.h" |
| #include "talk/base/scoped_ptr.h" |
| #include "talk/base/thread.h" |
| #include "talk/p2p/base/fakesession.h" |
| #include "talk/p2p/base/portallocator.h" |
| #include "talk/p2p/base/sessiondescription.h" |
| #include "talk/p2p/client/fakeportallocator.h" |
| #include "talk/session/phone/dummydevicemanager.h" |
| #include "talk/session/phone/fakewebrtcvcmfactory.h" |
| #include "talk/session/phone/fakewebrtcvideocapturemodule.h" |
| #include "talk/session/phone/mediasessionclient.h" |
| #include "talk/session/phone/webrtcmediaengine.h" |
| #include "talk/session/phone/webrtcvideocapturer.h" |
| |
| class WebRtcSessionTest |
| : public sigslot::has_slots<>, |
| public testing::Test { |
| public: |
| enum CallbackId { |
| kNone, |
| kOnAddStream, |
| kOnRemoveStream, |
| kOnLocalDescription, |
| kOnFailedCall, |
| }; |
| |
| WebRtcSessionTest() |
| : last_was_video_(false), |
| last_description_ptr_(NULL), |
| session_(NULL), |
| receiving_(false), |
| allocator_(NULL), |
| channel_manager_(NULL), |
| video_capturer_(NULL), |
| worker_thread_(NULL), |
| signaling_thread_(NULL) { |
| } |
| |
| ~WebRtcSessionTest() { |
| session_.reset(); |
| // Ensure the VideoCapturer be unregistered before destroyed. |
| channel_manager_->SetVideoCapturer(NULL, 0); |
| } |
| |
| void OnAddStream(const std::string& stream_id, bool video) { |
| callback_ids_.push_back(kOnAddStream); |
| last_stream_id_ = stream_id; |
| last_was_video_ = video; |
| } |
| void OnRemoveStream(const std::string& stream_id, bool video) { |
| callback_ids_.push_back(kOnRemoveStream); |
| last_stream_id_ = stream_id; |
| last_was_video_ = video; |
| } |
| void OnLocalDescription( |
| const cricket::SessionDescription* desc, |
| const std::vector<cricket::Candidate>& candidates) { |
| callback_ids_.push_back(kOnLocalDescription); |
| last_description_ptr_.reset(CopySessionDescription(desc)); |
| CopyCandidates(candidates, &last_candidates_); |
| } |
| cricket::SessionDescription* GetLocalDescription( |
| std::vector<cricket::Candidate>* candidates) { |
| if (last_candidates_.empty()) { |
| return NULL; |
| } |
| if (!last_description_ptr_.get()) { |
| return NULL; |
| } |
| CopyCandidates(last_candidates_, candidates); |
| return CopySessionDescription(last_description_ptr_.get()); |
| } |
| |
| void OnFailedCall() { |
| callback_ids_.push_back(kOnFailedCall); |
| } |
| |
| CallbackId PopOldestCallback() { |
| if (callback_ids_.empty()) { |
| return kNone; |
| } |
| const CallbackId return_value = callback_ids_.front(); |
| callback_ids_.pop_front(); |
| return return_value; |
| } |
| |
| CallbackId PeekOldestCallback() { |
| if (callback_ids_.empty()) { |
| return kNone; |
| } |
| const CallbackId return_value = callback_ids_.front(); |
| return return_value; |
| } |
| |
| void Reset() { |
| callback_ids_.clear(); |
| last_stream_id_ = ""; |
| last_was_video_ = false; |
| last_description_ptr_.reset(); |
| last_candidates_.clear(); |
| } |
| |
| bool WaitForCallback(CallbackId id, int timeout_ms) { |
| bool success = false; |
| for (int ms = 0; ms < timeout_ms; ms++) { |
| const CallbackId peek_id = PeekOldestCallback(); |
| if (peek_id == id) { |
| PopOldestCallback(); |
| success = true; |
| break; |
| } else if (peek_id != kNone) { |
| success = false; |
| break; |
| } |
| talk_base::Thread::Current()->ProcessMessages(1); |
| } |
| return success; |
| } |
| |
| bool Init(bool receiving) { |
| if (signaling_thread_ != NULL) |
| return false; |
| signaling_thread_ = talk_base::Thread::Current(); |
| receiving_ = receiving; |
| |
| if (worker_thread_!= NULL) |
| return false; |
| worker_thread_ = talk_base::Thread::Current(); |
| |
| cricket::FakePortAllocator* fake_port_allocator = |
| new cricket::FakePortAllocator(worker_thread_, NULL); |
| |
| allocator_.reset(static_cast<cricket::PortAllocator*>(fake_port_allocator)); |
| |
| cricket::DummyDeviceManager* device_manager( |
| new cricket::DummyDeviceManager()); |
| cricket::WebRtcMediaEngine* webrtc_media_engine( |
| new cricket::WebRtcMediaEngine(NULL, NULL, NULL)); |
| channel_manager_.reset(new cricket::ChannelManager(webrtc_media_engine, |
| device_manager, |
| worker_thread_)); |
| if (!channel_manager_->Init()) |
| return false; |
| |
| FakeWebRtcVideoCaptureModule* vcm = |
| new FakeWebRtcVideoCaptureModule(NULL, 123); |
| video_capturer_.reset(new cricket::WebRtcVideoCapturer); |
| if (!video_capturer_->Init(vcm)) { |
| return false; |
| } |
| // The SetVideoCapturer call doesn't transfer ownership. |
| if (!channel_manager_->SetVideoCapturer(video_capturer_.get(), 0)) |
| return false; |
| |
| talk_base::CreateRandomString(8, &id_); |
| |
| session_.reset(new webrtc::WebRtcSession( |
| id_, receiving_ , allocator_.get(), |
| channel_manager_.get(), |
| signaling_thread_)); |
| session_->SignalAddStream.connect(this, &WebRtcSessionTest::OnAddStream); |
| session_->SignalRemoveStream.connect(this, |
| &WebRtcSessionTest::OnRemoveStream); |
| session_->SignalLocalDescription.connect(this, |
| &WebRtcSessionTest::OnLocalDescription); |
| session_->SignalFailedCall.connect(this, &WebRtcSessionTest::OnFailedCall); |
| return true; |
| } |
| |
| // All session APIs must be called from the signaling thread. |
| bool CallInitiate() { |
| return session_->Initiate(); |
| } |
| |
| bool CallConnect() { |
| if (!session_->Connect()) |
| return false; |
| // This callback does not happen with FakeTransport! |
| if (!WaitForCallback(kOnLocalDescription, 1000)) { |
| return false; |
| } |
| return true; |
| } |
| |
| bool CallOnRemoteDescription( |
| cricket::SessionDescription* description, |
| std::vector<cricket::Candidate> candidates) { |
| if (!session_->OnRemoteDescription(description, candidates)) { |
| return false; |
| } |
| if (!WaitForCallback(kOnAddStream, 1000)) { |
| return false; |
| } |
| return true; |
| } |
| |
| bool CallOnInitiateMessage( |
| cricket::SessionDescription* description, |
| const std::vector<cricket::Candidate>& candidates) { |
| if (!session_->OnInitiateMessage(description, candidates)) { |
| return false; |
| } |
| if (!WaitForCallback(kOnAddStream, 1000)) { |
| return false; |
| } |
| return true; |
| } |
| |
| bool CallCreateVoiceChannel(const std::string& stream_id) { |
| if (!session_->CreateVoiceChannel(stream_id)) { |
| return false; |
| } |
| return true; |
| } |
| |
| bool CallCreateVideoChannel(const std::string& stream_id) { |
| if (!session_->CreateVideoChannel(stream_id)) { |
| return false; |
| } |
| return true; |
| } |
| |
| bool CallRemoveStream(const std::string& stream_id) { |
| return session_->RemoveStream(stream_id); |
| } |
| |
| void CallRemoveAllStreams() { |
| session_->RemoveAllStreams(); |
| } |
| |
| bool CallHasChannel(const std::string& label) { |
| return session_->HasStream(label); |
| } |
| |
| bool CallHasChannel(bool video) { |
| return session_->HasChannel(video); |
| } |
| |
| bool CallHasAudioChannel() { |
| return session_->HasAudioChannel(); |
| } |
| |
| bool CallHasVideoChannel() { |
| return session_->HasVideoChannel(); |
| } |
| |
| bool CallSetVideoRenderer(const std::string& stream_id, |
| cricket::VideoRenderer* renderer) { |
| return session_->SetVideoRenderer(stream_id, renderer); |
| } |
| |
| const std::vector<cricket::Candidate>& CallLocalCandidates() { |
| return session_->local_candidates(); |
| } |
| |
| private: |
| std::list<CallbackId> callback_ids_; |
| |
| std::string last_stream_id_; |
| bool last_was_video_; |
| talk_base::scoped_ptr<cricket::SessionDescription> last_description_ptr_; |
| std::vector<cricket::Candidate> last_candidates_; |
| |
| talk_base::scoped_ptr<webrtc::WebRtcSession> session_; |
| std::string id_; |
| bool receiving_; |
| |
| talk_base::scoped_ptr<cricket::PortAllocator> allocator_; |
| |
| talk_base::scoped_ptr<cricket::ChannelManager> channel_manager_; |
| talk_base::scoped_ptr<cricket::WebRtcVideoCapturer> video_capturer_; |
| |
| talk_base::Thread* worker_thread_; |
| talk_base::Thread* signaling_thread_; |
| }; |
| |
| bool CallbackReceived(WebRtcSessionTest* session, int timeout) { |
| EXPECT_EQ_WAIT(WebRtcSessionTest::kNone, session->PeekOldestCallback(), |
| timeout); |
| const WebRtcSessionTest::CallbackId peek_id = |
| session->PeekOldestCallback(); |
| return peek_id != WebRtcSessionTest::kNone; |
| } |
| |
| TEST_F(WebRtcSessionTest, InitializationReceiveSanity) { |
| const bool kReceiving = true; |
| ASSERT_TRUE(Init(kReceiving)); |
| ASSERT_TRUE(CallInitiate()); |
| |
| // Should return false because no stream has been set up yet. |
| EXPECT_FALSE(CallConnect()); |
| const bool kVideo = true; |
| EXPECT_FALSE(CallHasChannel(kVideo)); |
| EXPECT_FALSE(CallHasChannel(!kVideo)); |
| |
| EXPECT_EQ(kNone, PopOldestCallback()); |
| } |
| |
| TEST_F(WebRtcSessionTest, AudioSendCallSetUp) { |
| const bool kReceiving = false; |
| ASSERT_TRUE(Init(kReceiving)); |
| |
| ASSERT_TRUE(CallInitiate()); |
| |
| ASSERT_TRUE(CallCreateVoiceChannel("Audio")); |
| ASSERT_TRUE(CallConnect()); |
| |
| std::vector<cricket::Candidate> candidates; |
| cricket::SessionDescription* local_session = GetLocalDescription( |
| &candidates); |
| ASSERT_FALSE(candidates.empty()); |
| ASSERT_FALSE(local_session == NULL); |
| if (!CallOnRemoteDescription(local_session, candidates)) { |
| delete local_session; |
| FAIL(); |
| } |
| |
| // All callbacks should be caught. Assert it. |
| ASSERT_FALSE(CallbackReceived(this, 1000)); |
| ASSERT_TRUE(CallHasAudioChannel() && |
| !CallHasVideoChannel()); |
| } |
| |
| TEST_F(WebRtcSessionTest, VideoSendCallSetUp) { |
| const bool kReceiving = false; |
| ASSERT_TRUE(Init(kReceiving)); |
| |
| ASSERT_TRUE(CallInitiate()); |
| |
| ASSERT_TRUE(CallCreateVideoChannel("Video")); |
| ASSERT_TRUE(CallConnect()); |
| |
| std::vector<cricket::Candidate> candidates; |
| cricket::SessionDescription* local_session = GetLocalDescription( |
| &candidates); |
| ASSERT_FALSE(candidates.empty()); |
| ASSERT_FALSE(local_session == NULL); |
| |
| if (!CallOnRemoteDescription(local_session, candidates)) { |
| delete local_session; |
| FAIL(); |
| } |
| |
| // All callbacks should be caught. Assert it. |
| ASSERT_FALSE(CallbackReceived(this, 1000)); |
| ASSERT_TRUE(!CallHasAudioChannel() && |
| CallHasVideoChannel()); |
| } |
| |
| TEST_F(WebRtcSessionTest, AudioReceiveCallSetUp) { |
| const bool kReceiving = true; |
| const bool video = false; |
| |
| ASSERT_TRUE(Init(kReceiving)); |
| |
| std::vector<cricket::Candidate> candidates; |
| cricket::SessionDescription* local_session = |
| GenerateFakeSession(video, &candidates); |
| ASSERT_FALSE(candidates.empty()); |
| ASSERT_FALSE(local_session == NULL); |
| ASSERT_TRUE(CallInitiate()); |
| if (!CallOnInitiateMessage(local_session, candidates)) { |
| delete local_session; |
| FAIL(); |
| } |
| ASSERT_TRUE(CallConnect()); |
| ASSERT_FALSE(CallbackReceived(this, 1000)); |
| |
| ASSERT_TRUE(CallHasAudioChannel() && |
| !CallHasVideoChannel()); |
| } |
| |
| TEST_F(WebRtcSessionTest, VideoReceiveCallSetUp) { |
| const bool kReceiving = true; |
| const bool video = true; |
| |
| ASSERT_TRUE(Init(kReceiving)); |
| |
| std::vector<cricket::Candidate> candidates; |
| cricket::SessionDescription* local_session = |
| GenerateFakeSession(video, &candidates); |
| ASSERT_FALSE(candidates.empty()); |
| ASSERT_FALSE(local_session == NULL); |
| ASSERT_TRUE(CallInitiate()); |
| if (!CallOnInitiateMessage(local_session, candidates)) { |
| delete local_session; |
| FAIL(); |
| } |
| ASSERT_TRUE(CallConnect()); |
| ASSERT_FALSE(CallbackReceived(this, 1000)); |
| ASSERT_TRUE(!CallHasAudioChannel() && |
| CallHasVideoChannel()); |
| } |