#include "base/base64.h"
#include "base/numerics/checked_math.h"
#include "base/strings/escape.h"
#include "base/test/gtest_util.h"
#include "testing/gmock/include/gmock/gmock.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "third_party/modp_b64/modp_b64.h"
namespace base {
TEST(Base64Test, Basic) {
const std::string kText = "hello world";
const std::string kBase64Text = "aGVsbG8gd29ybGQ=";
std::string encoded;
std::string decoded;
bool ok;
Base64Encode(kText, &encoded);
EXPECT_EQ(kBase64Text, encoded);
ok = Base64Decode(encoded, &decoded);
EXPECT_TRUE(ok);
EXPECT_EQ(kText, decoded);
}
TEST(Base64Test, Forgiving) {
struct {
const char* in;
const char* expected_out;
} kTestCases[] = {
{"abc&", nullptr},
{"ab-d", nullptr},
{"abcde", nullptr},
{"a", nullptr},
{"abcd=", nullptr},
{"abcd==", nullptr},
{"abcd===", nullptr},
{"abcd====", nullptr},
{"abcd==============", nullptr},
{"=", nullptr},
{"====", nullptr},
{"abcd", "i\xB7\x1D"},
{"abc=", "i\xB7"},
{"abcdefgh", "i\xB7\x1Dy\xF8!"},
{"abcdef", "i\xB7\x1Dy"},
{"abc", "i\xB7"},
{"ab", "i"},
{" a bcd", "i\xB7\x1D"},
{"ab\t\tc=", "i\xB7"},
{"ab c\ndefgh", "i\xB7\x1Dy\xF8!"},
{"a\tb\nc\f d\r", "i\xB7\x1D"},
{"ab\vcd", nullptr},
{"", ""},
};
for (const auto& test_case : kTestCases) {
SCOPED_TRACE(::testing::Message()
<< EscapeAllExceptUnreserved(test_case.in));
std::string output;
bool success =
Base64Decode(test_case.in, &output, Base64DecodePolicy::kForgiving);
bool expected_success = test_case.expected_out != nullptr;
EXPECT_EQ(success, expected_success);
if (expected_success) {
EXPECT_EQ(output, test_case.expected_out);
}
}
}
TEST(Base64Test, Binary) {
const uint8_t kData[] = {0x00, 0x01, 0xFE, 0xFF};
std::string binary_encoded = Base64Encode(kData);
std::string string_piece_encoded;
Base64Encode(StringPiece(reinterpret_cast<const char*>(kData), sizeof(kData)),
&string_piece_encoded);
EXPECT_EQ(binary_encoded, string_piece_encoded);
EXPECT_THAT(Base64Decode(binary_encoded),
testing::Optional(testing::ElementsAreArray(kData)));
EXPECT_FALSE(Base64Decode("invalid base64!"));
std::string encoded_with_prefix = "PREFIX";
Base64EncodeAppend(kData, &encoded_with_prefix);
EXPECT_EQ(encoded_with_prefix, "PREFIX" + binary_encoded);
}
TEST(Base64Test, InPlace) {
const std::string kText = "hello world";
const std::string kBase64Text = "aGVsbG8gd29ybGQ=";
std::string text(kText);
Base64Encode(text, &text);
EXPECT_EQ(kBase64Text, text);
bool ok = Base64Decode(text, &text);
EXPECT_TRUE(ok);
EXPECT_EQ(text, kText);
}
TEST(Base64Test, Overflow) {
uint8_t b;
auto large_span = base::make_span(&b, MODP_B64_MAX_INPUT_LEN + 1);
EXPECT_CHECK_DEATH(Base64Encode(large_span));
std::string output = "PREFIX";
EXPECT_CHECK_DEATH(Base64EncodeAppend(large_span, &output));
base::CheckedNumeric<size_t> max_len = MODP_B64_MAX_INPUT_LEN;
EXPECT_TRUE(modp_b64_encode_data_len(max_len).IsValid());
}
}