| /* | 
 |  *  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 <cstddef> | 
 | #include <optional> | 
 | #include <string> | 
 | #include <vector> | 
 |  | 
 | #include "rtc_base/checks.h" | 
 | #include "test/gtest.h" | 
 |  | 
 | namespace webrtc { | 
 |  | 
 | namespace { | 
 | RtpParameters CreateRtpParametersWithCodecs( | 
 |     const std::vector<bool>& active, | 
 |     const std::vector<std::optional<RtpCodec>>& 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<RtpExtension> extensions; | 
 |   std::vector<RtpExtension> 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<RtpExtension>{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<RtpExtension>{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<RtpExtension>{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<RtpExtension>{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<RtpExtension>{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<RtpExtension>{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<RtpExtension>{kExtension1, kExtension2}), filtered); | 
 |   filtered = RtpExtension::DeduplicateHeaderExtensions( | 
 |       extensions, RtpExtension::Filter::kPreferEncryptedExtension); | 
 |   EXPECT_EQ(2u, filtered.size()); | 
 |   EXPECT_EQ((std::vector<RtpExtension>{kExtension1Encrypted, kExtension2}), | 
 |             filtered); | 
 |   filtered = RtpExtension::DeduplicateHeaderExtensions( | 
 |       extensions, RtpExtension::Filter::kRequireEncryptedExtension); | 
 |   EXPECT_EQ(1u, filtered.size()); | 
 |   EXPECT_EQ((std::vector<RtpExtension>{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<RtpExtension> 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<RtpExtension> 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<RtpExtension> 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<RtpExtension> 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<RtpExtension> 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 |