blob: b170fea8de0df36f8431a96d9ed8ba2f1e39be83 [file] [log] [blame]
// Copyright 2018 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 "components/mirroring/service/session.h"
#include "base/bind.h"
#include "base/callback.h"
#include "base/json/json_reader.h"
#include "base/macros.h"
#include "base/run_loop.h"
#include "base/test/scoped_task_environment.h"
#include "base/time/time.h"
#include "base/values.h"
#include "components/mirroring/service/fake_network_service.h"
#include "components/mirroring/service/fake_video_capture_host.h"
#include "components/mirroring/service/mirror_settings.h"
#include "components/mirroring/service/receiver_response.h"
#include "components/mirroring/service/value_util.h"
#include "media/cast/test/utility/default_config.h"
#include "media/cast/test/utility/net_utility.h"
#include "mojo/public/cpp/bindings/binding.h"
#include "net/base/ip_address.h"
#include "services/viz/public/cpp/gpu/gpu.h"
#include "testing/gmock/include/gmock/gmock.h"
#include "testing/gtest/include/gtest/gtest.h"
using ::testing::InvokeWithoutArgs;
using ::testing::_;
using ::testing::AtLeast;
using ::testing::Mock;
using media::cast::FrameSenderConfig;
using media::cast::Packet;
using media::mojom::RemotingStopReason;
using media::mojom::RemotingStartFailReason;
using media::mojom::RemotingSinkMetadata;
using media::mojom::RemotingSinkMetadataPtr;
using mirroring::mojom::SessionType;
using mirroring::mojom::SessionError;
namespace mirroring {
namespace {
class MockRemotingSource final : public media::mojom::RemotingSource {
public:
MockRemotingSource() : binding_(this) {}
~MockRemotingSource() override {}
void Bind(media::mojom::RemotingSourceRequest request) {
binding_.Bind(std::move(request));
}
MOCK_METHOD0(OnSinkGone, void());
MOCK_METHOD0(OnStarted, void());
MOCK_METHOD1(OnStartFailed, void(RemotingStartFailReason));
MOCK_METHOD1(OnMessageFromSink, void(const std::vector<uint8_t>&));
MOCK_METHOD1(OnStopped, void(RemotingStopReason));
MOCK_METHOD1(OnSinkAvailable, void(const RemotingSinkMetadata&));
void OnSinkAvailable(RemotingSinkMetadataPtr metadata) override {
OnSinkAvailable(*metadata);
}
private:
mojo::Binding<media::mojom::RemotingSource> binding_;
};
} // namespace
class SessionTest : public mojom::ResourceProvider,
public mojom::SessionObserver,
public mojom::CastMessageChannel,
public ::testing::Test {
public:
SessionTest()
: receiver_endpoint_(media::cast::test::GetFreeLocalPort()),
resource_provider_binding_(this),
session_observer_binding_(this),
outbound_channel_binding_(this) {}
~SessionTest() override { scoped_task_environment_.RunUntilIdle(); }
protected:
// mojom::SessionObserver implemenation.
MOCK_METHOD1(OnError, void(SessionError));
MOCK_METHOD0(DidStart, void());
MOCK_METHOD0(DidStop, void());
MOCK_METHOD0(OnGetVideoCaptureHost, void());
MOCK_METHOD0(OnGetNetworkContext, void());
MOCK_METHOD0(OnCreateAudioStream, void());
MOCK_METHOD0(OnConnectToRemotingSource, void());
// Called when sends an outbound message.
MOCK_METHOD1(OnOutboundMessage, void(const std::string& message_type));
// mojom::CastMessageHandler implementation. For outbound messages.
void Send(mojom::CastMessagePtr message) override {
EXPECT_TRUE(message->message_namespace == mojom::kWebRtcNamespace ||
message->message_namespace == mojom::kRemotingNamespace);
std::unique_ptr<base::Value> value =
base::JSONReader::ReadDeprecated(message->json_format_data);
ASSERT_TRUE(value);
std::string message_type;
EXPECT_TRUE(GetString(*value, "type", &message_type));
if (message_type == "OFFER") {
EXPECT_TRUE(GetInt(*value, "seqNum", &offer_sequence_number_));
} else if (message_type == "GET_CAPABILITIES") {
EXPECT_TRUE(GetInt(*value, "seqNum", &capability_sequence_number_));
}
OnOutboundMessage(message_type);
}
// mojom::ResourceProvider implemenation.
void GetVideoCaptureHost(
media::mojom::VideoCaptureHostRequest request) override {
video_host_ = std::make_unique<FakeVideoCaptureHost>(std::move(request));
OnGetVideoCaptureHost();
}
void GetNetworkContext(
network::mojom::NetworkContextRequest request) override {
network_context_ = std::make_unique<MockNetworkContext>(std::move(request));
OnGetNetworkContext();
}
void CreateAudioStream(mojom::AudioStreamCreatorClientPtr client,
const media::AudioParameters& params,
uint32_t total_segments) override {
OnCreateAudioStream();
}
void ConnectToRemotingSource(
media::mojom::RemoterPtr remoter,
media::mojom::RemotingSourceRequest request) override {
remoter_ = std::move(remoter);
remoting_source_.Bind(std::move(request));
OnConnectToRemotingSource();
}
void SendAnswer() {
ASSERT_TRUE(session_);
std::vector<FrameSenderConfig> audio_configs;
std::vector<FrameSenderConfig> video_configs;
if (session_type_ != SessionType::VIDEO_ONLY) {
if (cast_mode_ == "remoting") {
audio_configs.emplace_back(MirrorSettings::GetDefaultAudioConfig(
media::cast::RtpPayloadType::REMOTE_AUDIO,
media::cast::Codec::CODEC_AUDIO_REMOTE));
} else {
EXPECT_EQ("mirroring", cast_mode_);
audio_configs.emplace_back(MirrorSettings::GetDefaultAudioConfig(
media::cast::RtpPayloadType::AUDIO_OPUS,
media::cast::Codec::CODEC_AUDIO_OPUS));
}
}
if (session_type_ != SessionType::AUDIO_ONLY) {
if (cast_mode_ == "remoting") {
video_configs.emplace_back(MirrorSettings::GetDefaultVideoConfig(
media::cast::RtpPayloadType::REMOTE_VIDEO,
media::cast::Codec::CODEC_VIDEO_REMOTE));
} else {
EXPECT_EQ("mirroring", cast_mode_);
video_configs.emplace_back(MirrorSettings::GetDefaultVideoConfig(
media::cast::RtpPayloadType::VIDEO_VP8,
media::cast::Codec::CODEC_VIDEO_VP8));
}
}
auto answer = std::make_unique<Answer>();
answer->udp_port = receiver_endpoint_.port();
answer->cast_mode = cast_mode_;
answer->supports_get_status = true;
const int number_of_configs = audio_configs.size() + video_configs.size();
for (int i = 0; i < number_of_configs; ++i) {
answer->send_indexes.push_back(i);
answer->ssrcs.push_back(31 + i); // Arbitrary receiver SSRCs.
}
ReceiverResponse response;
response.result = "ok";
response.type = ResponseType::ANSWER;
response.sequence_number = offer_sequence_number_;
response.answer = std::move(answer);
session_->OnAnswer(audio_configs, video_configs, response);
scoped_task_environment_.RunUntilIdle();
}
// Create a mirroring session. Expect to send OFFER message.
void CreateSession(SessionType session_type) {
session_type_ = session_type;
mojom::SessionParametersPtr session_params =
mojom::SessionParameters::New();
session_params->receiver_address = receiver_endpoint_.address();
session_params->type = session_type_;
session_params->receiver_model_name = "Chromecast";
cast_mode_ = "mirroring";
mojom::ResourceProviderPtr resource_provider_ptr;
mojom::SessionObserverPtr session_observer_ptr;
mojom::CastMessageChannelPtr outbound_channel_ptr;
resource_provider_binding_.Bind(mojo::MakeRequest(&resource_provider_ptr));
session_observer_binding_.Bind(mojo::MakeRequest(&session_observer_ptr));
outbound_channel_binding_.Bind(mojo::MakeRequest(&outbound_channel_ptr));
// Expect to send OFFER message when session is created.
EXPECT_CALL(*this, OnGetNetworkContext()).Times(1);
EXPECT_CALL(*this, OnError(_)).Times(0);
EXPECT_CALL(*this, OnOutboundMessage("OFFER")).Times(1);
session_ = std::make_unique<Session>(
std::move(session_params), gfx::Size(1920, 1080),
std::move(session_observer_ptr), std::move(resource_provider_ptr),
std::move(outbound_channel_ptr), mojo::MakeRequest(&inbound_channel_),
nullptr);
scoped_task_environment_.RunUntilIdle();
Mock::VerifyAndClear(this);
}
// Starts the mirroring session.
void StartSession() {
ASSERT_TRUE(cast_mode_ == "mirroring");
// Except mirroing session starts after receiving ANSWER message.
const int num_to_get_video_host =
session_type_ == SessionType::AUDIO_ONLY ? 0 : 1;
const int num_to_create_audio_stream =
session_type_ == SessionType::VIDEO_ONLY ? 0 : 1;
EXPECT_CALL(*this, OnGetVideoCaptureHost()).Times(num_to_get_video_host);
EXPECT_CALL(*this, OnCreateAudioStream()).Times(num_to_create_audio_stream);
EXPECT_CALL(*this, OnError(_)).Times(0);
EXPECT_CALL(*this, OnOutboundMessage("GET_STATUS")).Times(AtLeast(1));
EXPECT_CALL(*this, OnOutboundMessage("GET_CAPABILITIES")).Times(1);
EXPECT_CALL(*this, DidStart()).Times(1);
SendAnswer();
scoped_task_environment_.RunUntilIdle();
Mock::VerifyAndClear(this);
}
void StopSession() {
if (video_host_)
EXPECT_CALL(*video_host_, OnStopped()).Times(1);
EXPECT_CALL(*this, DidStop()).Times(1);
session_.reset();
scoped_task_environment_.RunUntilIdle();
Mock::VerifyAndClear(this);
}
void CaptureOneVideoFrame() {
ASSERT_TRUE(cast_mode_ == "mirroring");
ASSERT_TRUE(video_host_);
// Expect to send out some UDP packets.
EXPECT_CALL(*network_context_->udp_socket(), OnSend()).Times(AtLeast(1));
EXPECT_CALL(*video_host_, ReleaseBuffer(_, _, _)).Times(1);
// Send one video frame to the consumer.
video_host_->SendOneFrame(gfx::Size(64, 32), base::TimeTicks::Now());
scoped_task_environment_.RunUntilIdle();
Mock::VerifyAndClear(network_context_.get());
Mock::VerifyAndClear(video_host_.get());
}
void SignalAnswerTimeout() {
if (cast_mode_ == "mirroring") {
EXPECT_CALL(*this, DidStop()).Times(1);
EXPECT_CALL(*this, OnError(SessionError::ANSWER_TIME_OUT)).Times(1);
} else {
EXPECT_CALL(*this, DidStop()).Times(0);
EXPECT_CALL(*this, OnError(SessionError::ANSWER_TIME_OUT)).Times(0);
// Expect to send OFFER message to fallback on mirroring.
EXPECT_CALL(*this, OnOutboundMessage("OFFER")).Times(1);
// The start of remoting is expected to fail.
EXPECT_CALL(remoting_source_,
OnStartFailed(RemotingStartFailReason::SERVICE_NOT_CONNECTED))
.Times(1);
EXPECT_CALL(remoting_source_, OnSinkGone()).Times(AtLeast(1));
}
session_->OnAnswer(std::vector<FrameSenderConfig>(),
std::vector<FrameSenderConfig>(), ReceiverResponse());
scoped_task_environment_.RunUntilIdle();
cast_mode_ = "mirroring";
Mock::VerifyAndClear(this);
Mock::VerifyAndClear(&remoting_source_);
}
void SendRemotingCapabilities() {
EXPECT_CALL(*this, OnConnectToRemotingSource()).Times(1);
EXPECT_CALL(remoting_source_, OnSinkAvailable(_)).Times(1);
ReceiverResponse response;
response.result = "ok";
response.type = ResponseType::CAPABILITIES_RESPONSE;
response.sequence_number = capability_sequence_number_;
response.capabilities = std::make_unique<ReceiverCapability>();
response.capabilities->media_caps =
std::vector<std::string>({"video", "audio", "vp8", "opus"});
session_->OnCapabilitiesResponse(response);
scoped_task_environment_.RunUntilIdle();
Mock::VerifyAndClear(this);
Mock::VerifyAndClear(&remoting_source_);
}
void StartRemoting() {
base::RunLoop run_loop;
ASSERT_TRUE(remoter_.is_bound());
// GET_CAPABILITIES is only sent once at the start of mirroring.
EXPECT_CALL(*this, OnOutboundMessage("GET_CAPABILITIES")).Times(0);
EXPECT_CALL(*this, OnOutboundMessage("OFFER"))
.WillOnce(InvokeWithoutArgs(&run_loop, &base::RunLoop::Quit));
remoter_->Start();
run_loop.Run();
scoped_task_environment_.RunUntilIdle();
cast_mode_ = "remoting";
Mock::VerifyAndClear(this);
}
void RemotingStarted() {
ASSERT_TRUE(cast_mode_ == "remoting");
EXPECT_CALL(remoting_source_, OnStarted()).Times(1);
SendAnswer();
scoped_task_environment_.RunUntilIdle();
Mock::VerifyAndClear(this);
Mock::VerifyAndClear(&remoting_source_);
}
void StopRemoting() {
ASSERT_TRUE(cast_mode_ == "remoting");
const RemotingStopReason reason = RemotingStopReason::LOCAL_PLAYBACK;
// Expect to send OFFER message to fallback on mirroring.
EXPECT_CALL(*this, OnOutboundMessage("OFFER")).Times(1);
EXPECT_CALL(remoting_source_, OnStopped(reason)).Times(1);
remoter_->Stop(reason);
scoped_task_environment_.RunUntilIdle();
cast_mode_ = "mirroring";
Mock::VerifyAndClear(this);
Mock::VerifyAndClear(&remoting_source_);
}
private:
base::test::ScopedTaskEnvironment scoped_task_environment_;
const net::IPEndPoint receiver_endpoint_;
mojo::Binding<mojom::ResourceProvider> resource_provider_binding_;
mojo::Binding<mojom::SessionObserver> session_observer_binding_;
mojo::Binding<mojom::CastMessageChannel> outbound_channel_binding_;
mojom::CastMessageChannelPtr inbound_channel_;
SessionType session_type_ = SessionType::AUDIO_AND_VIDEO;
media::mojom::RemoterPtr remoter_;
MockRemotingSource remoting_source_;
std::string cast_mode_;
int32_t offer_sequence_number_ = -1;
int32_t capability_sequence_number_ = -1;
std::unique_ptr<Session> session_;
std::unique_ptr<FakeVideoCaptureHost> video_host_;
std::unique_ptr<MockNetworkContext> network_context_;
DISALLOW_COPY_AND_ASSIGN(SessionTest);
};
TEST_F(SessionTest, AudioOnlyMirroring) {
CreateSession(SessionType::AUDIO_ONLY);
StartSession();
StopSession();
}
TEST_F(SessionTest, VideoOnlyMirroring) {
CreateSession(SessionType::VIDEO_ONLY);
StartSession();
CaptureOneVideoFrame();
StopSession();
}
TEST_F(SessionTest, AudioAndVideoMirroring) {
CreateSession(SessionType::AUDIO_AND_VIDEO);
StartSession();
StopSession();
}
TEST_F(SessionTest, AnswerTimeout) {
CreateSession(SessionType::AUDIO_AND_VIDEO);
SignalAnswerTimeout();
}
TEST_F(SessionTest, SwitchToAndFromRemoting) {
CreateSession(SessionType::AUDIO_AND_VIDEO);
StartSession();
SendRemotingCapabilities();
StartRemoting();
RemotingStarted();
StopRemoting();
StopSession();
}
TEST_F(SessionTest, StopSessionWhileRemoting) {
CreateSession(SessionType::AUDIO_AND_VIDEO);
StartSession();
SendRemotingCapabilities();
StartRemoting();
RemotingStarted();
StopSession();
}
TEST_F(SessionTest, StartRemotingFailed) {
CreateSession(SessionType::AUDIO_AND_VIDEO);
StartSession();
SendRemotingCapabilities();
StartRemoting();
SignalAnswerTimeout();
// Resume mirroring.
SendAnswer();
CaptureOneVideoFrame();
StopSession();
}
} // namespace mirroring