2018-10-16 10:41:42 +02:00
|
|
|
/*
|
|
|
|
|
* Copyright 2018 The WebRTC Project Authors. All rights reserved.
|
|
|
|
|
*
|
|
|
|
|
* Use of this source code is governed by a BSD-style license
|
|
|
|
|
* that can be found in the LICENSE file in the root of the source
|
|
|
|
|
* tree. An additional intellectual property rights grant can be found
|
|
|
|
|
* in the file PATENTS. All contributing project authors may
|
|
|
|
|
* be found in the AUTHORS file in the root of the source tree.
|
|
|
|
|
*/
|
|
|
|
|
|
2018-11-01 16:42:44 -07:00
|
|
|
#include <memory>
|
2018-10-16 10:41:42 +02:00
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
|
|
#include "api/test/loopback_media_transport.h"
|
|
|
|
|
#include "test/gmock.h"
|
|
|
|
|
|
|
|
|
|
namespace webrtc {
|
|
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
|
|
class MockMediaTransportAudioSinkInterface
|
|
|
|
|
: public MediaTransportAudioSinkInterface {
|
|
|
|
|
public:
|
|
|
|
|
MOCK_METHOD2(OnData, void(uint64_t, MediaTransportEncodedAudioFrame));
|
|
|
|
|
};
|
|
|
|
|
|
2018-11-01 16:42:44 -07:00
|
|
|
class MockDataChannelSink : public DataChannelSink {
|
|
|
|
|
public:
|
|
|
|
|
MOCK_METHOD3(OnDataReceived,
|
|
|
|
|
void(int, DataMessageType, const rtc::CopyOnWriteBuffer&));
|
|
|
|
|
MOCK_METHOD1(OnChannelClosing, void(int));
|
|
|
|
|
MOCK_METHOD1(OnChannelClosed, void(int));
|
|
|
|
|
};
|
|
|
|
|
|
2018-11-08 11:23:22 -08:00
|
|
|
class MockStateCallback : public MediaTransportStateCallback {
|
|
|
|
|
public:
|
|
|
|
|
MOCK_METHOD1(OnStateChanged, void(MediaTransportState));
|
|
|
|
|
};
|
|
|
|
|
|
2018-10-16 10:41:42 +02:00
|
|
|
// Test only uses the sequence number.
|
|
|
|
|
MediaTransportEncodedAudioFrame CreateAudioFrame(int sequence_number) {
|
|
|
|
|
static constexpr int kSamplingRateHz = 48000;
|
|
|
|
|
static constexpr int kStartingSampleIndex = 0;
|
|
|
|
|
static constexpr int kSamplesPerChannel = 480;
|
|
|
|
|
static constexpr uint8_t kPayloadType = 17;
|
|
|
|
|
|
|
|
|
|
return MediaTransportEncodedAudioFrame(
|
|
|
|
|
kSamplingRateHz, kStartingSampleIndex, kSamplesPerChannel,
|
|
|
|
|
sequence_number, MediaTransportEncodedAudioFrame::FrameType::kSpeech,
|
|
|
|
|
kPayloadType, std::vector<uint8_t>(kSamplesPerChannel));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
|
|
TEST(LoopbackMediaTransport, AudioWithNoSinkSilentlyIgnored) {
|
2018-11-01 16:42:44 -07:00
|
|
|
std::unique_ptr<rtc::Thread> thread = rtc::Thread::Create();
|
|
|
|
|
thread->Start();
|
|
|
|
|
MediaTransportPair transport_pair(thread.get());
|
2018-10-16 10:41:42 +02:00
|
|
|
transport_pair.first()->SendAudioFrame(1, CreateAudioFrame(0));
|
|
|
|
|
transport_pair.second()->SendAudioFrame(2, CreateAudioFrame(0));
|
2018-11-01 16:42:44 -07:00
|
|
|
transport_pair.FlushAsyncInvokes();
|
2018-10-16 10:41:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST(LoopbackMediaTransport, AudioDeliveredToSink) {
|
2018-11-01 16:42:44 -07:00
|
|
|
std::unique_ptr<rtc::Thread> thread = rtc::Thread::Create();
|
|
|
|
|
thread->Start();
|
|
|
|
|
MediaTransportPair transport_pair(thread.get());
|
2018-10-16 10:41:42 +02:00
|
|
|
testing::StrictMock<MockMediaTransportAudioSinkInterface> sink;
|
|
|
|
|
EXPECT_CALL(sink,
|
|
|
|
|
OnData(1, testing::Property(
|
|
|
|
|
&MediaTransportEncodedAudioFrame::sequence_number,
|
|
|
|
|
testing::Eq(10))));
|
|
|
|
|
transport_pair.second()->SetReceiveAudioSink(&sink);
|
|
|
|
|
transport_pair.first()->SendAudioFrame(1, CreateAudioFrame(10));
|
|
|
|
|
|
2018-11-01 16:42:44 -07:00
|
|
|
transport_pair.FlushAsyncInvokes();
|
2018-10-16 10:41:42 +02:00
|
|
|
transport_pair.second()->SetReceiveAudioSink(nullptr);
|
|
|
|
|
}
|
|
|
|
|
|
2018-11-01 16:42:44 -07:00
|
|
|
TEST(LoopbackMediaTransport, DataDeliveredToSink) {
|
|
|
|
|
std::unique_ptr<rtc::Thread> thread = rtc::Thread::Create();
|
|
|
|
|
thread->Start();
|
|
|
|
|
MediaTransportPair transport_pair(thread.get());
|
|
|
|
|
|
|
|
|
|
MockDataChannelSink sink;
|
|
|
|
|
transport_pair.first()->SetDataSink(&sink);
|
|
|
|
|
|
|
|
|
|
const int channel_id = 1;
|
|
|
|
|
EXPECT_CALL(sink,
|
|
|
|
|
OnDataReceived(
|
|
|
|
|
channel_id, DataMessageType::kText,
|
|
|
|
|
testing::Property<rtc::CopyOnWriteBuffer, const char*>(
|
|
|
|
|
&rtc::CopyOnWriteBuffer::cdata, testing::StrEq("foo"))));
|
|
|
|
|
|
|
|
|
|
SendDataParams params;
|
|
|
|
|
params.type = DataMessageType::kText;
|
|
|
|
|
rtc::CopyOnWriteBuffer buffer("foo");
|
|
|
|
|
transport_pair.second()->SendData(channel_id, params, buffer);
|
|
|
|
|
|
|
|
|
|
transport_pair.FlushAsyncInvokes();
|
|
|
|
|
transport_pair.first()->SetDataSink(nullptr);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST(LoopbackMediaTransport, CloseDeliveredToSink) {
|
|
|
|
|
std::unique_ptr<rtc::Thread> thread = rtc::Thread::Create();
|
|
|
|
|
thread->Start();
|
|
|
|
|
MediaTransportPair transport_pair(thread.get());
|
|
|
|
|
|
|
|
|
|
MockDataChannelSink first_sink;
|
|
|
|
|
transport_pair.first()->SetDataSink(&first_sink);
|
|
|
|
|
|
|
|
|
|
MockDataChannelSink second_sink;
|
|
|
|
|
transport_pair.second()->SetDataSink(&second_sink);
|
|
|
|
|
|
|
|
|
|
const int channel_id = 1;
|
|
|
|
|
{
|
|
|
|
|
testing::InSequence s;
|
|
|
|
|
EXPECT_CALL(second_sink, OnChannelClosing(channel_id));
|
|
|
|
|
EXPECT_CALL(second_sink, OnChannelClosed(channel_id));
|
|
|
|
|
EXPECT_CALL(first_sink, OnChannelClosed(channel_id));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
transport_pair.first()->CloseChannel(channel_id);
|
|
|
|
|
|
|
|
|
|
transport_pair.FlushAsyncInvokes();
|
|
|
|
|
transport_pair.first()->SetDataSink(nullptr);
|
|
|
|
|
transport_pair.second()->SetDataSink(nullptr);
|
|
|
|
|
}
|
|
|
|
|
|
2018-11-08 11:23:22 -08:00
|
|
|
TEST(LoopbackMediaTransport, InitialStateDeliveredWhenCallbackSet) {
|
|
|
|
|
std::unique_ptr<rtc::Thread> thread = rtc::Thread::Create();
|
|
|
|
|
thread->Start();
|
|
|
|
|
MediaTransportPair transport_pair(thread.get());
|
|
|
|
|
|
|
|
|
|
MockStateCallback state_callback;
|
|
|
|
|
|
|
|
|
|
EXPECT_CALL(state_callback, OnStateChanged(MediaTransportState::kPending));
|
|
|
|
|
transport_pair.first()->SetMediaTransportStateCallback(&state_callback);
|
|
|
|
|
transport_pair.FlushAsyncInvokes();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST(LoopbackMediaTransport, ChangedStateDeliveredWhenCallbackSet) {
|
|
|
|
|
std::unique_ptr<rtc::Thread> thread = rtc::Thread::Create();
|
|
|
|
|
thread->Start();
|
|
|
|
|
MediaTransportPair transport_pair(thread.get());
|
|
|
|
|
|
|
|
|
|
transport_pair.SetState(MediaTransportState::kWritable);
|
|
|
|
|
transport_pair.FlushAsyncInvokes();
|
|
|
|
|
|
|
|
|
|
MockStateCallback state_callback;
|
|
|
|
|
|
|
|
|
|
EXPECT_CALL(state_callback, OnStateChanged(MediaTransportState::kWritable));
|
|
|
|
|
transport_pair.first()->SetMediaTransportStateCallback(&state_callback);
|
|
|
|
|
transport_pair.FlushAsyncInvokes();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TEST(LoopbackMediaTransport, StateChangeDeliveredToCallback) {
|
|
|
|
|
std::unique_ptr<rtc::Thread> thread = rtc::Thread::Create();
|
|
|
|
|
thread->Start();
|
|
|
|
|
MediaTransportPair transport_pair(thread.get());
|
|
|
|
|
|
|
|
|
|
MockStateCallback state_callback;
|
|
|
|
|
|
|
|
|
|
EXPECT_CALL(state_callback, OnStateChanged(MediaTransportState::kPending));
|
|
|
|
|
EXPECT_CALL(state_callback, OnStateChanged(MediaTransportState::kWritable));
|
|
|
|
|
transport_pair.first()->SetMediaTransportStateCallback(&state_callback);
|
|
|
|
|
transport_pair.SetState(MediaTransportState::kWritable);
|
|
|
|
|
transport_pair.FlushAsyncInvokes();
|
|
|
|
|
}
|
|
|
|
|
|
2018-10-16 10:41:42 +02:00
|
|
|
} // namespace webrtc
|