Movatterモバイル変換


[0]ホーム

URL:


Google Git
Sign in
chromium /chromium /src /refs/heads/main /. /base /base64_unittest.cc
blob: 39eb4f40b73b504a38064687c04722014d93716f [file] [log] [blame] [edit]
// 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<string_view>
#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{
constchar* in;
// nullptr indicates a decode failure.
constchar* expected_out_forgiving;
constchar* 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(constauto& 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(constauto& 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){
constuint8_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<constchar*>(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<size_t> max_len= MODP_B64_MAX_INPUT_LEN;
EXPECT_TRUE(modp_b64_encode_data_len(max_len).IsValid());
}
}// namespace base

[8]ページ先頭

©2009-2025 Movatter.jp