// Copyright 2012 The Chromium Authors // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #ifdef UNSAFE_BUFFERS_BUILD // TODO(crbug.com/40284755): Remove this and spanify to fix the errors. #pragma allow_unsafe_buffers #endif #include "base/base64.h" #include #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 decoded; bool ok; std::string encoded = Base64Encode(kText); EXPECT_EQ(kBase64Text, encoded); ok = Base64Decode(encoded, &decoded); EXPECT_TRUE(ok); EXPECT_EQ(kText, decoded); } TEST(Base64Test, ForgivingAndStrictDecode) { struct { const char* in; // nullptr indicates a decode failure. const char* expected_out_forgiving; const char* expected_out_strict; } kTestCases[] = { // Failures that should apply in all decoding modes: // // - Characters not in the base64 alphabet {"abc&", nullptr, nullptr}, {"ab-d", nullptr, nullptr}, // - input len % 4 == 1 {"abcde", nullptr, nullptr}, {"a", nullptr, nullptr}, // Invalid padding causes failure if kForgiving is set. {"abcd=", nullptr, nullptr}, {"abcd==", nullptr, nullptr}, {"abcd===", nullptr, nullptr}, {"abcd====", nullptr, nullptr}, {"abcd==============", nullptr, nullptr}, {"abcde===", nullptr, nullptr}, {"=", nullptr, nullptr}, {"====", nullptr, nullptr}, // Otherwise, inputs that are multiples of 4 always succeed, this matches // kStrict mode. {"abcd", "i\xB7\x1D", "i\xB7\x1D"}, {"abc=", "i\xB7", "i\xB7"}, {"abcdefgh", "i\xB7\x1Dy\xF8!", "i\xB7\x1Dy\xF8!"}, // kForgiving mode allows for omitting padding (to a multiple of 4) if // len % 4 != 1. {"abcdef", "i\xB7\x1Dy", nullptr}, {"abc", "i\xB7", nullptr}, {"ab", "i", nullptr}, // Whitespace should be allowed if kForgiving is set, matching // https://infra.spec.whatwg.org/#ascii-whitespace: // ASCII whitespace is U+0009 TAB '\t', U+000A LF '\n', U+000C FF '\f', // U+000D CR '\r', or U+0020 SPACE ' '. {" a bcd", "i\xB7\x1D", nullptr}, {"ab\t\tc=", "i\xB7", nullptr}, {"ab c\ndefgh", "i\xB7\x1Dy\xF8!", nullptr}, {"a\tb\nc\f d\r", "i\xB7\x1D", nullptr}, {"this should fail", "\xB6\x18\xAC\xB2\x1A.\x95\xD7\xDA\x8A", nullptr}, // U+000B VT '\v' is _not_ valid whitespace to be stripped. {"ab\vcd", nullptr, nullptr}, // Empty string should yield an empty result. {"", "", ""}, }; for (const auto& test_case : kTestCases) { SCOPED_TRACE(::testing::Message() << "Forgiving: " << EscapeAllExceptUnreserved(test_case.in)); std::string output; bool success = Base64Decode(test_case.in, &output, Base64DecodePolicy::kForgiving); bool expected_success = test_case.expected_out_forgiving != nullptr; EXPECT_EQ(success, expected_success); if (expected_success) { EXPECT_EQ(output, test_case.expected_out_forgiving); } } for (const auto& test_case : kTestCases) { SCOPED_TRACE(::testing::Message() << "Strict: " << EscapeAllExceptUnreserved(test_case.in)); std::string output; bool success = Base64Decode(test_case.in, &output, Base64DecodePolicy::kStrict); bool expected_success = test_case.expected_out_strict != nullptr; EXPECT_EQ(success, expected_success); if (expected_success) { EXPECT_EQ(output, test_case.expected_out_strict); } } } TEST(Base64Test, Binary) { const uint8_t kData[] = {0x00, 0x01, 0xFE, 0xFF}; std::string binary_encoded = Base64Encode(kData); // Check that encoding the same data through the std::string_view interface // gives the same results. std::string string_piece_encoded = Base64Encode( std::string_view(reinterpret_cast(kData), sizeof(kData))); 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 = Base64Encode(kText); EXPECT_EQ(kBase64Text, text); bool ok = Base64Decode(text, &text); EXPECT_TRUE(ok); EXPECT_EQ(text, kText); } TEST(Base64Test, Overflow) { // `Base64Encode` makes the input larger, which means inputs whose base64 // output overflows `size_t`. Actually allocating a span of this size will // likely fail, but we test it with a fake span and assume a correct // implementation will check for overflow before touching the input. // // Note that, with or without an overflow check, the function will still // crash. This test is only meaningful because `EXPECT_CHECK_DEATH` looks for // a `CHECK`-based failure. uint8_t b; auto large_span = 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)); // `modp_b64_encode_data_len` is a macro, so check `MODP_B64_MAX_INPUT_LEN` is // correct be verifying the computation doesn't overflow. base::CheckedNumeric max_len = MODP_B64_MAX_INPUT_LEN; EXPECT_TRUE(modp_b64_encode_data_len(max_len).IsValid()); } } // namespace base