mirror of
https://github.com/mollyim/webrtc.git
synced 2025-05-15 23:01:21 +01:00

Bug: webrtc:14801 Change-Id: I7c14597e39b312c26573f034dca444cc1d90e332 Reviewed-on: https://webrtc-review.googlesource.com/c/src/+/295480 Commit-Queue: Philip Eliasson <philipel@webrtc.org> Reviewed-by: Mirko Bonadei <mbonadei@webrtc.org> Reviewed-by: Björn Terelius <terelius@webrtc.org> Cr-Commit-Position: refs/heads/main@{#39449}
190 lines
5.9 KiB
C++
190 lines
5.9 KiB
C++
/*
|
|
* Copyright (c) 2023 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 "logging/rtc_event_log/encoder/optional_blob_encoding.h"
|
|
|
|
#include <string>
|
|
#include <vector>
|
|
|
|
#include "test/gmock.h"
|
|
#include "test/gtest.h"
|
|
|
|
using ::testing::ElementsAre;
|
|
using ::testing::IsEmpty;
|
|
|
|
namespace webrtc {
|
|
namespace {
|
|
|
|
class BitBuilder {
|
|
public:
|
|
BitBuilder& Bit(uint8_t bit) {
|
|
if (total_bits_ % 8 == 0) {
|
|
bits_.push_back(0);
|
|
}
|
|
bits_[total_bits_ / 8] |= bit << (7 - (total_bits_ % 8));
|
|
++total_bits_;
|
|
return *this;
|
|
}
|
|
|
|
BitBuilder& Bytes(const std::vector<uint8_t>& bytes) {
|
|
for (uint8_t byte : bytes) {
|
|
for (int i = 1; i <= 8; ++i) {
|
|
uint8_t bit = (byte >> (8 - i)) & 1;
|
|
Bit(bit);
|
|
}
|
|
}
|
|
return *this;
|
|
}
|
|
|
|
BitBuilder& ByteAlign() {
|
|
while (total_bits_ % 8 > 0) {
|
|
Bit(0);
|
|
}
|
|
return *this;
|
|
}
|
|
|
|
std::string AsString() { return std::string(bits_.begin(), bits_.end()); }
|
|
|
|
private:
|
|
std::vector<uint8_t> bits_;
|
|
uint64_t total_bits_ = 0;
|
|
};
|
|
|
|
TEST(OptionalBlobEncoding, AllBlobsPresent) {
|
|
std::string encoded = EncodeOptionalBlobs({"a", "b", "c"});
|
|
std::string expected = BitBuilder()
|
|
.Bit(1)
|
|
.ByteAlign()
|
|
.Bytes({0x01, 'a'})
|
|
.Bytes({0x01, 'b'})
|
|
.Bytes({0x01, 'c'})
|
|
.AsString();
|
|
EXPECT_EQ(encoded, expected);
|
|
}
|
|
|
|
TEST(OptionalBlobEncoding, SomeBlobsPresent) {
|
|
std::string encoded = EncodeOptionalBlobs({"a", absl::nullopt, "c"});
|
|
std::string expected = BitBuilder()
|
|
.Bit(0)
|
|
.Bit(1)
|
|
.Bit(0)
|
|
.Bit(1)
|
|
.ByteAlign()
|
|
.Bytes({0x01, 'a'})
|
|
.Bytes({0x01, 'c'})
|
|
.AsString();
|
|
EXPECT_EQ(encoded, expected);
|
|
}
|
|
|
|
TEST(OptionalBlobEncoding, NoBlobsPresent) {
|
|
std::string encoded =
|
|
EncodeOptionalBlobs({absl::nullopt, absl::nullopt, absl::nullopt});
|
|
std::string expected = BitBuilder().Bit(0).Bit(0).Bit(0).Bit(0).AsString();
|
|
EXPECT_EQ(encoded, expected);
|
|
}
|
|
|
|
TEST(OptionalBlobEncoding, ZeroBlobs) {
|
|
std::string encoded = EncodeOptionalBlobs({});
|
|
EXPECT_EQ(encoded, std::string());
|
|
}
|
|
|
|
TEST(OptionalBlobEncoding, LongBlobs) {
|
|
std::string medium_string(100, 'a');
|
|
std::string long_string(200, 'b');
|
|
std::string encoded = EncodeOptionalBlobs({medium_string, long_string});
|
|
std::string expected =
|
|
BitBuilder()
|
|
.Bit(1)
|
|
.ByteAlign()
|
|
.Bytes({0x64})
|
|
.Bytes({medium_string.begin(), medium_string.end()})
|
|
.Bytes({0xC8, 0x01})
|
|
.Bytes({long_string.begin(), long_string.end()})
|
|
.AsString();
|
|
EXPECT_EQ(encoded, expected);
|
|
}
|
|
|
|
TEST(OptionalBlobDecoding, AllBlobsPresent) {
|
|
std::string encoded = BitBuilder()
|
|
.Bit(1)
|
|
.ByteAlign()
|
|
.Bytes({0x01, 'a'})
|
|
.Bytes({0x01, 'b'})
|
|
.Bytes({0x01, 'c'})
|
|
.AsString();
|
|
auto decoded = DecodeOptionalBlobs(encoded, 3);
|
|
EXPECT_THAT(decoded, ElementsAre("a", "b", "c"));
|
|
}
|
|
|
|
TEST(OptionalBlobDecoding, SomeBlobsPresent) {
|
|
std::string encoded = BitBuilder()
|
|
.Bit(0)
|
|
.Bit(1)
|
|
.Bit(0)
|
|
.Bit(1)
|
|
.ByteAlign()
|
|
.Bytes({0x01, 'a'})
|
|
.Bytes({0x01, 'c'})
|
|
.AsString();
|
|
auto decoded = DecodeOptionalBlobs(encoded, 3);
|
|
EXPECT_THAT(decoded, ElementsAre("a", absl::nullopt, "c"));
|
|
}
|
|
|
|
TEST(OptionalBlobDecoding, NoBlobsPresent) {
|
|
std::string encoded =
|
|
BitBuilder().Bit(0).Bit(0).Bit(0).Bit(0).ByteAlign().AsString();
|
|
auto decoded = DecodeOptionalBlobs(encoded, 3);
|
|
EXPECT_THAT(decoded,
|
|
ElementsAre(absl::nullopt, absl::nullopt, absl::nullopt));
|
|
}
|
|
|
|
TEST(OptionalBlobDecoding, ZeroBlobs) {
|
|
std::string encoded;
|
|
auto decoded = DecodeOptionalBlobs(encoded, 0);
|
|
EXPECT_THAT(decoded, IsEmpty());
|
|
}
|
|
|
|
TEST(OptionalBlobDecoding, LongBlobs) {
|
|
std::string medium_string(100, 'a');
|
|
std::string long_string(200, 'b');
|
|
std::string encoded = BitBuilder()
|
|
.Bit(1)
|
|
.ByteAlign()
|
|
.Bytes({0x64})
|
|
.Bytes({medium_string.begin(), medium_string.end()})
|
|
.Bytes({0xC8, 0x01})
|
|
.Bytes({long_string.begin(), long_string.end()})
|
|
.AsString();
|
|
auto decoded = DecodeOptionalBlobs(encoded, 2);
|
|
EXPECT_THAT(decoded, ElementsAre(medium_string, long_string));
|
|
}
|
|
|
|
TEST(OptionalBlobDecoding, TooShortEncodedBlobLength) {
|
|
std::string encoded =
|
|
BitBuilder().Bit(1).ByteAlign().Bytes({0x01, 'a', 'b'}).AsString();
|
|
auto decoded = DecodeOptionalBlobs(encoded, 1);
|
|
EXPECT_THAT(decoded, IsEmpty());
|
|
}
|
|
|
|
TEST(OptionalBlobDecoding, TooLongEncodedBlobLength) {
|
|
std::string encoded =
|
|
BitBuilder().Bit(1).ByteAlign().Bytes({0x03, 'a', 'b'}).AsString();
|
|
auto decoded = DecodeOptionalBlobs(encoded, 1);
|
|
EXPECT_THAT(decoded, IsEmpty());
|
|
}
|
|
|
|
TEST(OptionalBlobDecoding, TooLongEncodedBufferLength) {
|
|
std::string encoded = BitBuilder().Bytes({0x00, 0x00, 0x00}).AsString();
|
|
auto decoded = DecodeOptionalBlobs(encoded, 8);
|
|
EXPECT_THAT(decoded, IsEmpty());
|
|
}
|
|
|
|
} // namespace
|
|
} // namespace webrtc
|