diff options
Diffstat (limited to 'third_party/libwebrtc/pc/sctp_transport_unittest.cc')
-rw-r--r-- | third_party/libwebrtc/pc/sctp_transport_unittest.cc | 221 |
1 files changed, 221 insertions, 0 deletions
diff --git a/third_party/libwebrtc/pc/sctp_transport_unittest.cc b/third_party/libwebrtc/pc/sctp_transport_unittest.cc new file mode 100644 index 0000000000..1e12a0a9d2 --- /dev/null +++ b/third_party/libwebrtc/pc/sctp_transport_unittest.cc @@ -0,0 +1,221 @@ +/* + * Copyright 2019 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. + */ + +#include "pc/sctp_transport.h" + +#include <utility> +#include <vector> + +#include "absl/memory/memory.h" +#include "absl/types/optional.h" +#include "api/dtls_transport_interface.h" +#include "api/transport/data_channel_transport_interface.h" +#include "media/base/media_channel.h" +#include "p2p/base/fake_dtls_transport.h" +#include "p2p/base/p2p_constants.h" +#include "p2p/base/packet_transport_internal.h" +#include "pc/dtls_transport.h" +#include "rtc_base/copy_on_write_buffer.h" +#include "rtc_base/gunit.h" +#include "test/gmock.h" +#include "test/gtest.h" + +constexpr int kDefaultTimeout = 1000; // milliseconds +constexpr int kTestMaxSctpStreams = 1234; + +using cricket::FakeDtlsTransport; +using ::testing::ElementsAre; + +namespace webrtc { + +namespace { + +class FakeCricketSctpTransport : public cricket::SctpTransportInternal { + public: + void SetDtlsTransport(rtc::PacketTransportInternal* transport) override {} + bool Start(int local_port, int remote_port, int max_message_size) override { + return true; + } + bool OpenStream(int sid) override { return true; } + bool ResetStream(int sid) override { return true; } + bool SendData(int sid, + const SendDataParams& params, + const rtc::CopyOnWriteBuffer& payload, + cricket::SendDataResult* result = nullptr) override { + return true; + } + bool ReadyToSendData() override { return true; } + void set_debug_name_for_testing(const char* debug_name) override {} + int max_message_size() const override { return 0; } + absl::optional<int> max_outbound_streams() const override { + return max_outbound_streams_; + } + absl::optional<int> max_inbound_streams() const override { + return max_inbound_streams_; + } + // Methods exposed for testing + void SendSignalReadyToSendData() { SignalReadyToSendData(); } + + void SendSignalAssociationChangeCommunicationUp() { + SignalAssociationChangeCommunicationUp(); + } + + void SendSignalClosingProcedureStartedRemotely() { + SignalClosingProcedureStartedRemotely(1); + } + + void SendSignalClosingProcedureComplete() { + SignalClosingProcedureComplete(1); + } + void set_max_outbound_streams(int streams) { + max_outbound_streams_ = streams; + } + void set_max_inbound_streams(int streams) { max_inbound_streams_ = streams; } + + private: + absl::optional<int> max_outbound_streams_; + absl::optional<int> max_inbound_streams_; +}; + +} // namespace + +class TestSctpTransportObserver : public SctpTransportObserverInterface { + public: + TestSctpTransportObserver() : info_(SctpTransportState::kNew) {} + + void OnStateChange(SctpTransportInformation info) override { + info_ = info; + states_.push_back(info.state()); + } + + SctpTransportState State() { + if (states_.size() > 0) { + return states_[states_.size() - 1]; + } else { + return SctpTransportState::kNew; + } + } + + const std::vector<SctpTransportState>& States() { return states_; } + + const SctpTransportInformation LastReceivedInformation() { return info_; } + + private: + std::vector<SctpTransportState> states_; + SctpTransportInformation info_; +}; + +class SctpTransportTest : public ::testing::Test { + public: + SctpTransport* transport() { return transport_.get(); } + SctpTransportObserverInterface* observer() { return &observer_; } + + void CreateTransport() { + auto cricket_sctp_transport = + absl::WrapUnique(new FakeCricketSctpTransport()); + transport_ = + rtc::make_ref_counted<SctpTransport>(std::move(cricket_sctp_transport)); + } + + void AddDtlsTransport() { + std::unique_ptr<cricket::DtlsTransportInternal> cricket_transport = + std::make_unique<FakeDtlsTransport>( + "audio", cricket::ICE_CANDIDATE_COMPONENT_RTP); + dtls_transport_ = + rtc::make_ref_counted<DtlsTransport>(std::move(cricket_transport)); + transport_->SetDtlsTransport(dtls_transport_); + } + + void CompleteSctpHandshake() { + CricketSctpTransport()->SendSignalReadyToSendData(); + // The computed MaxChannels shall be the minimum of the outgoing + // and incoming # of streams. + CricketSctpTransport()->set_max_outbound_streams(kTestMaxSctpStreams); + CricketSctpTransport()->set_max_inbound_streams(kTestMaxSctpStreams + 1); + CricketSctpTransport()->SendSignalAssociationChangeCommunicationUp(); + } + + FakeCricketSctpTransport* CricketSctpTransport() { + return static_cast<FakeCricketSctpTransport*>(transport_->internal()); + } + + rtc::AutoThread main_thread_; + rtc::scoped_refptr<SctpTransport> transport_; + rtc::scoped_refptr<DtlsTransport> dtls_transport_; + TestSctpTransportObserver observer_; +}; + +TEST(SctpTransportSimpleTest, CreateClearDelete) { + rtc::AutoThread main_thread; + std::unique_ptr<cricket::SctpTransportInternal> fake_cricket_sctp_transport = + absl::WrapUnique(new FakeCricketSctpTransport()); + rtc::scoped_refptr<SctpTransport> sctp_transport = + rtc::make_ref_counted<SctpTransport>( + std::move(fake_cricket_sctp_transport)); + ASSERT_TRUE(sctp_transport->internal()); + ASSERT_EQ(SctpTransportState::kNew, sctp_transport->Information().state()); + sctp_transport->Clear(); + ASSERT_FALSE(sctp_transport->internal()); + ASSERT_EQ(SctpTransportState::kClosed, sctp_transport->Information().state()); +} + +TEST_F(SctpTransportTest, EventsObservedWhenConnecting) { + CreateTransport(); + transport()->RegisterObserver(observer()); + AddDtlsTransport(); + CompleteSctpHandshake(); + ASSERT_EQ_WAIT(SctpTransportState::kConnected, observer_.State(), + kDefaultTimeout); + EXPECT_THAT(observer_.States(), ElementsAre(SctpTransportState::kConnecting, + SctpTransportState::kConnected)); +} + +TEST_F(SctpTransportTest, CloseWhenClearing) { + CreateTransport(); + transport()->RegisterObserver(observer()); + AddDtlsTransport(); + CompleteSctpHandshake(); + ASSERT_EQ_WAIT(SctpTransportState::kConnected, observer_.State(), + kDefaultTimeout); + transport()->Clear(); + ASSERT_EQ_WAIT(SctpTransportState::kClosed, observer_.State(), + kDefaultTimeout); +} + +TEST_F(SctpTransportTest, MaxChannelsSignalled) { + CreateTransport(); + transport()->RegisterObserver(observer()); + AddDtlsTransport(); + EXPECT_FALSE(transport()->Information().MaxChannels()); + EXPECT_FALSE(observer_.LastReceivedInformation().MaxChannels()); + CompleteSctpHandshake(); + ASSERT_EQ_WAIT(SctpTransportState::kConnected, observer_.State(), + kDefaultTimeout); + EXPECT_TRUE(transport()->Information().MaxChannels()); + EXPECT_EQ(kTestMaxSctpStreams, *(transport()->Information().MaxChannels())); + EXPECT_TRUE(observer_.LastReceivedInformation().MaxChannels()); + EXPECT_EQ(kTestMaxSctpStreams, + *(observer_.LastReceivedInformation().MaxChannels())); +} + +TEST_F(SctpTransportTest, CloseWhenTransportCloses) { + CreateTransport(); + transport()->RegisterObserver(observer()); + AddDtlsTransport(); + CompleteSctpHandshake(); + ASSERT_EQ_WAIT(SctpTransportState::kConnected, observer_.State(), + kDefaultTimeout); + static_cast<cricket::FakeDtlsTransport*>(dtls_transport_->internal()) + ->SetDtlsState(DtlsTransportState::kClosed); + ASSERT_EQ_WAIT(SctpTransportState::kClosed, observer_.State(), + kDefaultTimeout); +} + +} // namespace webrtc |