| // Copyright 2020 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 "third_party/blink/renderer/modules/mediastream/pushable_media_stream_video_source.h" |
| |
| #include "base/run_loop.h" |
| #include "media/base/bind_to_current_loop.h" |
| #include "testing/gtest/include/gtest/gtest.h" |
| #include "third_party/blink/public/mojom/mediastream/media_stream.mojom-blink.h" |
| #include "third_party/blink/public/web/modules/mediastream/media_stream_video_sink.h" |
| #include "third_party/blink/public/web/web_heap.h" |
| #include "third_party/blink/renderer/modules/mediastream/media_stream_video_track.h" |
| #include "third_party/blink/renderer/modules/mediastream/mock_media_stream_video_source.h" |
| #include "third_party/blink/renderer/modules/mediastream/video_track_adapter_settings.h" |
| #include "third_party/blink/renderer/platform/mediastream/media_stream_source.h" |
| #include "third_party/blink/renderer/platform/testing/io_task_runner_testing_platform_support.h" |
| #include "third_party/blink/renderer/platform/wtf/cross_thread_functional.h" |
| |
| namespace blink { |
| |
| namespace { |
| |
| class FakeMediaStreamVideoSink : public MediaStreamVideoSink { |
| public: |
| FakeMediaStreamVideoSink(base::TimeTicks* capture_time, |
| media::VideoFrameMetadata* metadata, |
| gfx::Size* natural_size, |
| base::OnceClosure got_frame_cb) |
| : capture_time_(capture_time), |
| metadata_(metadata), |
| natural_size_(natural_size), |
| got_frame_cb_(std::move(got_frame_cb)) {} |
| |
| void ConnectToTrack(const WebMediaStreamTrack& track) { |
| MediaStreamVideoSink::ConnectToTrack( |
| track, |
| ConvertToBaseRepeatingCallback( |
| CrossThreadBindRepeating(&FakeMediaStreamVideoSink::OnVideoFrame, |
| WTF::CrossThreadUnretained(this))), |
| true); |
| } |
| |
| void DisconnectFromTrack() { MediaStreamVideoSink::DisconnectFromTrack(); } |
| |
| void OnVideoFrame(scoped_refptr<media::VideoFrame> frame, |
| std::vector<scoped_refptr<media::VideoFrame>> scaled_frames, |
| base::TimeTicks capture_time) { |
| *capture_time_ = capture_time; |
| *metadata_ = frame->metadata(); |
| *natural_size_ = frame->natural_size(); |
| std::move(got_frame_cb_).Run(); |
| } |
| |
| private: |
| base::TimeTicks* const capture_time_; |
| media::VideoFrameMetadata* const metadata_; |
| gfx::Size* const natural_size_; |
| base::OnceClosure got_frame_cb_; |
| }; |
| |
| MediaStreamSource* CreateConnectedMediaStreamSource( |
| MediaStreamVideoSource* video_source) { |
| MediaStreamSource* media_stream_source = |
| MakeGarbageCollected<MediaStreamSource>( |
| "dummy_source_id", MediaStreamSource::kTypeVideo, "dummy_source_name", |
| false /* remote */); |
| media_stream_source->SetPlatformSource(base::WrapUnique(video_source)); |
| return media_stream_source; |
| } |
| |
| WebMediaStreamTrack StartVideoSource(MediaStreamVideoSource* video_source) { |
| return MediaStreamVideoTrack::CreateVideoTrack( |
| video_source, MediaStreamVideoSource::ConstraintsOnceCallback(), |
| /*enabled=*/true); |
| } |
| |
| MediaStreamSource* CreateAndStartMediaStreamSource( |
| MediaStreamVideoSource* video_source) { |
| MediaStreamSource* source = CreateConnectedMediaStreamSource(video_source); |
| StartVideoSource(video_source); |
| return source; |
| } |
| |
| } // namespace |
| |
| class PushableMediaStreamVideoSourceTest : public testing::Test { |
| public: |
| PushableMediaStreamVideoSourceTest() { |
| pushable_video_source_ = new PushableMediaStreamVideoSource(); |
| stream_source_ = CreateConnectedMediaStreamSource(pushable_video_source_); |
| } |
| |
| void TearDown() override { |
| stream_source_ = nullptr; |
| WebHeap::CollectAllGarbageForTesting(); |
| } |
| |
| WebMediaStreamTrack StartSource() { |
| return StartVideoSource(pushable_video_source_); |
| } |
| |
| protected: |
| ScopedTestingPlatformSupport<IOTaskRunnerTestingPlatformSupport> platform_; |
| |
| Persistent<MediaStreamSource> stream_source_; |
| PushableMediaStreamVideoSource* pushable_video_source_; |
| }; |
| |
| TEST_F(PushableMediaStreamVideoSourceTest, StartAndStop) { |
| EXPECT_EQ(MediaStreamSource::kReadyStateLive, |
| stream_source_->GetReadyState()); |
| EXPECT_FALSE(pushable_video_source_->running()); |
| |
| WebMediaStreamTrack track = StartSource(); |
| EXPECT_EQ(MediaStreamSource::kReadyStateLive, |
| stream_source_->GetReadyState()); |
| EXPECT_TRUE(pushable_video_source_->running()); |
| |
| // If the pushable source stops, the MediaStreamSource should stop. |
| pushable_video_source_->StopSource(); |
| EXPECT_EQ(MediaStreamSource::kReadyStateEnded, |
| stream_source_->GetReadyState()); |
| EXPECT_FALSE(pushable_video_source_->running()); |
| } |
| |
| TEST_F(PushableMediaStreamVideoSourceTest, FramesPropagateToSink) { |
| WebMediaStreamTrack track = StartSource(); |
| base::RunLoop run_loop; |
| base::TimeTicks reference_capture_time = base::TimeTicks::Now(); |
| base::TimeTicks capture_time; |
| media::VideoFrameMetadata metadata; |
| gfx::Size natural_size; |
| FakeMediaStreamVideoSink fake_sink( |
| &capture_time, &metadata, &natural_size, |
| media::BindToCurrentLoop(run_loop.QuitClosure())); |
| fake_sink.ConnectToTrack(track); |
| const scoped_refptr<media::VideoFrame> frame = |
| media::VideoFrame::CreateBlackFrame(gfx::Size(100, 50)); |
| frame->metadata().frame_rate = 30.0; |
| |
| pushable_video_source_->PushFrame(frame, reference_capture_time); |
| run_loop.Run(); |
| |
| fake_sink.DisconnectFromTrack(); |
| EXPECT_EQ(reference_capture_time, capture_time); |
| EXPECT_EQ(30.0, *metadata.frame_rate); |
| EXPECT_EQ(natural_size.width(), 100); |
| EXPECT_EQ(natural_size.height(), 50); |
| } |
| |
| TEST_F(PushableMediaStreamVideoSourceTest, ForwardToUpstream) { |
| MockMediaStreamVideoSource* mock_source = new MockMediaStreamVideoSource(); |
| PushableMediaStreamVideoSource* pushable_video_source = |
| new PushableMediaStreamVideoSource(mock_source->GetWeakPtr()); |
| CreateAndStartMediaStreamSource(mock_source); |
| CreateAndStartMediaStreamSource(pushable_video_source); |
| |
| EXPECT_CALL(*mock_source, OnRequestRefreshFrame()); |
| pushable_video_source->RequestRefreshFrame(); |
| |
| EXPECT_CALL(*mock_source, |
| OnFrameDropped(media::VideoCaptureFrameDropReason:: |
| kResolutionAdapterFrameIsNotValid)); |
| pushable_video_source->OnFrameDropped( |
| media::VideoCaptureFrameDropReason::kResolutionAdapterFrameIsNotValid); |
| |
| EXPECT_CALL(*mock_source, OnFrameFeedback(media::VideoFrameFeedback())); |
| pushable_video_source->GetFeedbackCallback().Run(media::VideoFrameFeedback()); |
| } |
| |
| } // namespace blink |