/* * Copyright (c) 2017 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 "api/rtp_parameters.h" #include #include #include #include #include "rtc_base/checks.h" #include "test/gtest.h" namespace webrtc { namespace { RtpParameters CreateRtpParametersWithCodecs( const std::vector& active, const std::vector>& codecs) { RTC_DCHECK_EQ(active.size(), codecs.size()); RtpParameters parameters; for (size_t i = 0; i < codecs.size(); ++i) { RtpEncodingParameters encoding; encoding.active = active[i]; encoding.codec = codecs[i]; parameters.encodings.push_back(encoding); } return parameters; } } // namespace static const char kExtensionUri1[] = "extension-uri1"; static const char kExtensionUri2[] = "extension-uri2"; static const RtpExtension kExtension1(kExtensionUri1, 1); static const RtpExtension kExtension1Encrypted(kExtensionUri1, 10, true); static const RtpExtension kExtension2(kExtensionUri2, 2); TEST(RtpExtensionTest, DeduplicateHeaderExtensions) { std::vector extensions; std::vector filtered; extensions.clear(); extensions.push_back(kExtension1); extensions.push_back(kExtension1Encrypted); filtered = RtpExtension::DeduplicateHeaderExtensions( extensions, RtpExtension::Filter::kDiscardEncryptedExtension); EXPECT_EQ(1u, filtered.size()); EXPECT_EQ(std::vector{kExtension1}, filtered); extensions.clear(); extensions.push_back(kExtension1); extensions.push_back(kExtension1Encrypted); filtered = RtpExtension::DeduplicateHeaderExtensions( extensions, RtpExtension::Filter::kPreferEncryptedExtension); EXPECT_EQ(1u, filtered.size()); EXPECT_EQ(std::vector{kExtension1Encrypted}, filtered); extensions.clear(); extensions.push_back(kExtension1); extensions.push_back(kExtension1Encrypted); filtered = RtpExtension::DeduplicateHeaderExtensions( extensions, RtpExtension::Filter::kRequireEncryptedExtension); EXPECT_EQ(1u, filtered.size()); EXPECT_EQ(std::vector{kExtension1Encrypted}, filtered); extensions.clear(); extensions.push_back(kExtension1Encrypted); extensions.push_back(kExtension1); filtered = RtpExtension::DeduplicateHeaderExtensions( extensions, RtpExtension::Filter::kDiscardEncryptedExtension); EXPECT_EQ(1u, filtered.size()); EXPECT_EQ(std::vector{kExtension1}, filtered); extensions.clear(); extensions.push_back(kExtension1Encrypted); extensions.push_back(kExtension1); filtered = RtpExtension::DeduplicateHeaderExtensions( extensions, RtpExtension::Filter::kPreferEncryptedExtension); EXPECT_EQ(1u, filtered.size()); EXPECT_EQ(std::vector{kExtension1Encrypted}, filtered); extensions.clear(); extensions.push_back(kExtension1Encrypted); extensions.push_back(kExtension1); filtered = RtpExtension::DeduplicateHeaderExtensions( extensions, RtpExtension::Filter::kRequireEncryptedExtension); EXPECT_EQ(1u, filtered.size()); EXPECT_EQ(std::vector{kExtension1Encrypted}, filtered); extensions.clear(); extensions.push_back(kExtension1); extensions.push_back(kExtension2); filtered = RtpExtension::DeduplicateHeaderExtensions( extensions, RtpExtension::Filter::kDiscardEncryptedExtension); EXPECT_EQ(2u, filtered.size()); EXPECT_EQ(extensions, filtered); filtered = RtpExtension::DeduplicateHeaderExtensions( extensions, RtpExtension::Filter::kPreferEncryptedExtension); EXPECT_EQ(2u, filtered.size()); EXPECT_EQ(extensions, filtered); filtered = RtpExtension::DeduplicateHeaderExtensions( extensions, RtpExtension::Filter::kRequireEncryptedExtension); EXPECT_EQ(0u, filtered.size()); extensions.clear(); extensions.push_back(kExtension1); extensions.push_back(kExtension2); extensions.push_back(kExtension1Encrypted); filtered = RtpExtension::DeduplicateHeaderExtensions( extensions, RtpExtension::Filter::kDiscardEncryptedExtension); EXPECT_EQ(2u, filtered.size()); EXPECT_EQ((std::vector{kExtension1, kExtension2}), filtered); filtered = RtpExtension::DeduplicateHeaderExtensions( extensions, RtpExtension::Filter::kPreferEncryptedExtension); EXPECT_EQ(2u, filtered.size()); EXPECT_EQ((std::vector{kExtension1Encrypted, kExtension2}), filtered); filtered = RtpExtension::DeduplicateHeaderExtensions( extensions, RtpExtension::Filter::kRequireEncryptedExtension); EXPECT_EQ(1u, filtered.size()); EXPECT_EQ((std::vector{kExtension1Encrypted}), filtered); } // Test that the filtered vector is sorted so that for a given unsorted array of // extensions, the filtered vector will always be laied out the same (for easy // comparison). TEST(RtpExtensionTest, DeduplicateHeaderExtensionsSorted) { const std::vector extensions = { RtpExtension("cde1", 11, false), RtpExtension("cde2", 12, true), RtpExtension("abc1", 3, false), RtpExtension("abc2", 4, true), RtpExtension("cde3", 9, true), RtpExtension("cde4", 10, false), RtpExtension("abc3", 1, true), RtpExtension("abc4", 2, false), RtpExtension("bcd3", 7, false), RtpExtension("bcd1", 8, true), RtpExtension("bcd2", 5, true), RtpExtension("bcd4", 6, false), }; auto encrypted = RtpExtension::DeduplicateHeaderExtensions( extensions, RtpExtension::Filter::kRequireEncryptedExtension); const std::vector expected_sorted_encrypted = { RtpExtension("abc2", 4, true), RtpExtension("abc3", 1, true), RtpExtension("bcd1", 8, true), RtpExtension("bcd2", 5, true), RtpExtension("cde2", 12, true), RtpExtension("cde3", 9, true)}; EXPECT_EQ(expected_sorted_encrypted, encrypted); auto unencypted = RtpExtension::DeduplicateHeaderExtensions( extensions, RtpExtension::Filter::kDiscardEncryptedExtension); const std::vector expected_sorted_unencrypted = { RtpExtension("abc1", 3, false), RtpExtension("abc4", 2, false), RtpExtension("bcd3", 7, false), RtpExtension("bcd4", 6, false), RtpExtension("cde1", 11, false), RtpExtension("cde4", 10, false)}; EXPECT_EQ(expected_sorted_unencrypted, unencypted); } TEST(RtpExtensionTest, FindHeaderExtensionByUriAndEncryption) { std::vector extensions; extensions.clear(); EXPECT_EQ(nullptr, RtpExtension::FindHeaderExtensionByUriAndEncryption( extensions, kExtensionUri1, false)); extensions.clear(); extensions.push_back(kExtension1); EXPECT_EQ(kExtension1, *RtpExtension::FindHeaderExtensionByUriAndEncryption( extensions, kExtensionUri1, false)); EXPECT_EQ(nullptr, RtpExtension::FindHeaderExtensionByUriAndEncryption( extensions, kExtensionUri1, true)); EXPECT_EQ(nullptr, RtpExtension::FindHeaderExtensionByUriAndEncryption( extensions, kExtensionUri2, false)); extensions.clear(); extensions.push_back(kExtension1); extensions.push_back(kExtension2); extensions.push_back(kExtension1Encrypted); EXPECT_EQ(kExtension1, *RtpExtension::FindHeaderExtensionByUriAndEncryption( extensions, kExtensionUri1, false)); EXPECT_EQ(kExtension2, *RtpExtension::FindHeaderExtensionByUriAndEncryption( extensions, kExtensionUri2, false)); EXPECT_EQ(kExtension1Encrypted, *RtpExtension::FindHeaderExtensionByUriAndEncryption( extensions, kExtensionUri1, true)); EXPECT_EQ(nullptr, RtpExtension::FindHeaderExtensionByUriAndEncryption( extensions, kExtensionUri2, true)); } TEST(RtpExtensionTest, FindHeaderExtensionByUri) { std::vector extensions; extensions.clear(); EXPECT_EQ(nullptr, RtpExtension::FindHeaderExtensionByUri( extensions, kExtensionUri1, RtpExtension::Filter::kDiscardEncryptedExtension)); EXPECT_EQ(nullptr, RtpExtension::FindHeaderExtensionByUri( extensions, kExtensionUri1, RtpExtension::Filter::kPreferEncryptedExtension)); EXPECT_EQ(nullptr, RtpExtension::FindHeaderExtensionByUri( extensions, kExtensionUri1, RtpExtension::Filter::kRequireEncryptedExtension)); extensions.clear(); extensions.push_back(kExtension1); EXPECT_EQ(kExtension1, *RtpExtension::FindHeaderExtensionByUri( extensions, kExtensionUri1, RtpExtension::Filter::kDiscardEncryptedExtension)); EXPECT_EQ(kExtension1, *RtpExtension::FindHeaderExtensionByUri( extensions, kExtensionUri1, RtpExtension::Filter::kPreferEncryptedExtension)); EXPECT_EQ(nullptr, RtpExtension::FindHeaderExtensionByUri( extensions, kExtensionUri1, RtpExtension::Filter::kRequireEncryptedExtension)); EXPECT_EQ(nullptr, RtpExtension::FindHeaderExtensionByUri( extensions, kExtensionUri2, RtpExtension::Filter::kDiscardEncryptedExtension)); EXPECT_EQ(nullptr, RtpExtension::FindHeaderExtensionByUri( extensions, kExtensionUri2, RtpExtension::Filter::kPreferEncryptedExtension)); EXPECT_EQ(nullptr, RtpExtension::FindHeaderExtensionByUri( extensions, kExtensionUri2, RtpExtension::Filter::kRequireEncryptedExtension)); extensions.clear(); extensions.push_back(kExtension1); extensions.push_back(kExtension1Encrypted); EXPECT_EQ(kExtension1, *RtpExtension::FindHeaderExtensionByUri( extensions, kExtensionUri1, RtpExtension::Filter::kDiscardEncryptedExtension)); extensions.clear(); extensions.push_back(kExtension1); extensions.push_back(kExtension1Encrypted); EXPECT_EQ(kExtension1Encrypted, *RtpExtension::FindHeaderExtensionByUri( extensions, kExtensionUri1, RtpExtension::Filter::kPreferEncryptedExtension)); extensions.clear(); extensions.push_back(kExtension1); extensions.push_back(kExtension1Encrypted); EXPECT_EQ(kExtension1Encrypted, *RtpExtension::FindHeaderExtensionByUri( extensions, kExtensionUri1, RtpExtension::Filter::kRequireEncryptedExtension)); extensions.clear(); extensions.push_back(kExtension1Encrypted); extensions.push_back(kExtension1); EXPECT_EQ(kExtension1, *RtpExtension::FindHeaderExtensionByUri( extensions, kExtensionUri1, RtpExtension::Filter::kDiscardEncryptedExtension)); extensions.clear(); extensions.push_back(kExtension1Encrypted); extensions.push_back(kExtension1); EXPECT_EQ(kExtension1Encrypted, *RtpExtension::FindHeaderExtensionByUri( extensions, kExtensionUri1, RtpExtension::Filter::kPreferEncryptedExtension)); extensions.clear(); extensions.push_back(kExtension1Encrypted); extensions.push_back(kExtension1); EXPECT_EQ(kExtension1Encrypted, *RtpExtension::FindHeaderExtensionByUri( extensions, kExtensionUri1, RtpExtension::Filter::kRequireEncryptedExtension)); extensions.clear(); extensions.push_back(kExtension1); extensions.push_back(kExtension2); EXPECT_EQ(kExtension1, *RtpExtension::FindHeaderExtensionByUri( extensions, kExtensionUri1, RtpExtension::Filter::kDiscardEncryptedExtension)); EXPECT_EQ(kExtension1, *RtpExtension::FindHeaderExtensionByUri( extensions, kExtensionUri1, RtpExtension::Filter::kPreferEncryptedExtension)); EXPECT_EQ(nullptr, RtpExtension::FindHeaderExtensionByUri( extensions, kExtensionUri1, RtpExtension::Filter::kRequireEncryptedExtension)); EXPECT_EQ(kExtension2, *RtpExtension::FindHeaderExtensionByUri( extensions, kExtensionUri2, RtpExtension::Filter::kDiscardEncryptedExtension)); EXPECT_EQ(kExtension2, *RtpExtension::FindHeaderExtensionByUri( extensions, kExtensionUri2, RtpExtension::Filter::kPreferEncryptedExtension)); EXPECT_EQ(nullptr, RtpExtension::FindHeaderExtensionByUri( extensions, kExtensionUri2, RtpExtension::Filter::kRequireEncryptedExtension)); extensions.clear(); extensions.push_back(kExtension1); extensions.push_back(kExtension2); extensions.push_back(kExtension1Encrypted); EXPECT_EQ(kExtension1, *RtpExtension::FindHeaderExtensionByUri( extensions, kExtensionUri1, RtpExtension::Filter::kDiscardEncryptedExtension)); EXPECT_EQ(kExtension1Encrypted, *RtpExtension::FindHeaderExtensionByUri( extensions, kExtensionUri1, RtpExtension::Filter::kPreferEncryptedExtension)); EXPECT_EQ(kExtension1Encrypted, *RtpExtension::FindHeaderExtensionByUri( extensions, kExtensionUri1, RtpExtension::Filter::kRequireEncryptedExtension)); EXPECT_EQ(kExtension2, *RtpExtension::FindHeaderExtensionByUri( extensions, kExtensionUri2, RtpExtension::Filter::kDiscardEncryptedExtension)); EXPECT_EQ(kExtension2, *RtpExtension::FindHeaderExtensionByUri( extensions, kExtensionUri2, RtpExtension::Filter::kPreferEncryptedExtension)); EXPECT_EQ(nullptr, RtpExtension::FindHeaderExtensionByUri( extensions, kExtensionUri2, RtpExtension::Filter::kRequireEncryptedExtension)); } TEST(CodecParameterMap, ParsesKeyValueFmtpParameterSet) { std::string params = "key1=value1;key2=value2"; CodecParameterMap codec_params; ASSERT_TRUE(ParseFmtpParameterSet(params, codec_params).ok()); EXPECT_EQ(2U, codec_params.size()); EXPECT_EQ(codec_params["key1"], "value1"); EXPECT_EQ(codec_params["key2"], "value2"); } TEST(CodecParameterMap, ParsesNonKeyValueFmtpParameterSet) { std::string params = "not-in-key-value-format"; CodecParameterMap codec_params; ASSERT_TRUE(ParseFmtpParameterSet(params, codec_params).ok()); EXPECT_EQ(1U, codec_params.size()); EXPECT_EQ(codec_params[""], "not-in-key-value-format"); } TEST(RtpParametersTest, IsMixedCodec) { RtpParameters parameters; RtpCodec codec1, codec2; codec1.name = "codec1"; codec2.name = "codec2"; parameters = CreateRtpParametersWithCodecs({}, {}); EXPECT_FALSE(parameters.IsMixedCodec()); parameters = CreateRtpParametersWithCodecs({true}, {codec1}); EXPECT_FALSE(parameters.IsMixedCodec()); parameters = CreateRtpParametersWithCodecs({true}, {std::nullopt}); EXPECT_FALSE(parameters.IsMixedCodec()); parameters = CreateRtpParametersWithCodecs({true, true}, {codec1, codec2}); EXPECT_TRUE(parameters.IsMixedCodec()); // Inactive encoding parameters are ignored. parameters = CreateRtpParametersWithCodecs({false, true}, {codec1, codec2}); EXPECT_FALSE(parameters.IsMixedCodec()); // Even if some codecs are nullopt, differing codec presence/values // among active encodings is considered mixed. parameters = CreateRtpParametersWithCodecs({true, true}, {std::nullopt, codec2}); EXPECT_TRUE(parameters.IsMixedCodec()); parameters = CreateRtpParametersWithCodecs({true, true}, {std::nullopt, std::nullopt}); EXPECT_FALSE(parameters.IsMixedCodec()); parameters = CreateRtpParametersWithCodecs({true, true, true}, {std::nullopt, codec1, codec2}); EXPECT_TRUE(parameters.IsMixedCodec()); } } // namespace webrtc