blob: 39eb4f40b73b504a38064687c04722014d93716f [file] [log] [blame]
Avi Drissmane4622aa2022-09-08 20:36:061// Copyright 2012 The Chromium Authors
license.botbf09a502008-08-24 00:55:552// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
initial.commit586acc5fe2008-07-26 22:42:524
danakj51d26a42024-04-25 14:23:565#ifdef UNSAFE_BUFFERS_BUILD
6// TODO(crbug.com/40284755): Remove this and spanify to fix the errors.
7#pragma allow_unsafe_buffers
8#endif
9
[email protected]978df342009-11-24 06:21:5310#include "base/base64.h"
[email protected]24f111a2012-09-23 04:51:0411
Helmut Januschka0fc785b2024-04-17 21:13:3612#include <string_view>
13
David Benjamin48808e22022-09-21 19:39:1214#include "base/numerics/checked_math.h"
Charlie Harrison7b633d92022-11-29 05:23:5015#include "base/strings/escape.h"
David Benjamin48808e22022-09-21 19:39:1216#include "base/test/gtest_util.h"
David Benjamin47bb5ec2022-02-01 23:12:2817#include "testing/gmock/include/gmock/gmock.h"
initial.commit586acc5fe2008-07-26 22:42:5218#include "testing/gtest/include/gtest/gtest.h"
David Benjamin48808e22022-09-21 19:39:1219#include "third_party/modp_b64/modp_b64.h"
initial.commit586acc5fe2008-07-26 22:42:5220
[email protected]24f111a2012-09-23 04:51:0421namespace base {
initial.commit586acc5fe2008-07-26 22:42:5222
23TEST(Base64Test, Basic) {
24 const std::string kText = "hello world";
25 const std::string kBase64Text = "aGVsbG8gd29ybGQ=";
26
[email protected]24f111a2012-09-23 04:51:0427 std::string decoded;
initial.commit586acc5fe2008-07-26 22:42:5228 bool ok;
29
wd l4a70b3ff2023-09-26 20:13:3530 std::string encoded = Base64Encode(kText);
initial.commit586acc5fe2008-07-26 22:42:5231 EXPECT_EQ(kBase64Text, encoded);
32
[email protected]24f111a2012-09-23 04:51:0433 ok = Base64Decode(encoded, &decoded);
initial.commit586acc5fe2008-07-26 22:42:5234 EXPECT_TRUE(ok);
35 EXPECT_EQ(kText, decoded);
36}
[email protected]24f111a2012-09-23 04:51:0437
Nidhi Jaju95d13c22024-11-30 01:58:5538TEST(Base64Test, ForgivingAndStrictDecode) {
Charlie Harrison7b633d92022-11-29 05:23:5039 struct {
40 const char* in;
41
42 // nullptr indicates a decode failure.
Nidhi Jaju95d13c22024-11-30 01:58:5543 const char* expected_out_forgiving;
44 const char* expected_out_strict;
Charlie Harrison7b633d92022-11-29 05:23:5045 } kTestCases[] = {
46 // Failures that should apply in all decoding modes:
47 //
48 // - Characters not in the base64 alphabet
Nidhi Jaju95d13c22024-11-30 01:58:5549 {"abc&", nullptr, nullptr},
50 {"ab-d", nullptr, nullptr},
Charlie Harrison7b633d92022-11-29 05:23:5051 // - input len % 4 == 1
Nidhi Jaju95d13c22024-11-30 01:58:5552 {"abcde", nullptr, nullptr},
53 {"a", nullptr, nullptr},
Charlie Harrison7b633d92022-11-29 05:23:5054
55 // Invalid padding causes failure if kForgiving is set.
Nidhi Jaju95d13c22024-11-30 01:58:5556 {"abcd=", nullptr, nullptr},
57 {"abcd==", nullptr, nullptr},
58 {"abcd===", nullptr, nullptr},
59 {"abcd====", nullptr, nullptr},
60 {"abcd==============", nullptr, nullptr},
61 {"abcde===", nullptr, nullptr},
62 {"=", nullptr, nullptr},
63 {"====", nullptr, nullptr},
Charlie Harrison7b633d92022-11-29 05:23:5064
65 // Otherwise, inputs that are multiples of 4 always succeed, this matches
66 // kStrict mode.
Nidhi Jaju95d13c22024-11-30 01:58:5567 {"abcd", "i\xB7\x1D", "i\xB7\x1D"},
68 {"abc=", "i\xB7", "i\xB7"},
69 {"abcdefgh", "i\xB7\x1Dy\xF8!", "i\xB7\x1Dy\xF8!"},
Charlie Harrison7b633d92022-11-29 05:23:5070
71 // kForgiving mode allows for omitting padding (to a multiple of 4) if
72 // len % 4 != 1.
Nidhi Jaju95d13c22024-11-30 01:58:5573 {"abcdef", "i\xB7\x1Dy", nullptr},
74 {"abc", "i\xB7", nullptr},
75 {"ab", "i", nullptr},
Charlie Harrison7b633d92022-11-29 05:23:5076
77 // Whitespace should be allowed if kForgiving is set, matching
78 // https://infra.spec.whatwg.org/#ascii-whitespace:
79 // ASCII whitespace is U+0009 TAB '\t', U+000A LF '\n', U+000C FF '\f',
80 // U+000D CR '\r', or U+0020 SPACE ' '.
Nidhi Jaju95d13c22024-11-30 01:58:5581 {" a bcd", "i\xB7\x1D", nullptr},
82 {"ab\t\tc=", "i\xB7", nullptr},
83 {"ab c\ndefgh", "i\xB7\x1Dy\xF8!", nullptr},
84 {"a\tb\nc\f d\r", "i\xB7\x1D", nullptr},
85 {"this should fail", "\xB6\x18\xAC\xB2\x1A.\x95\xD7\xDA\x8A", nullptr},
Charlie Harrison7b633d92022-11-29 05:23:5086
87 // U+000B VT '\v' is _not_ valid whitespace to be stripped.
Nidhi Jaju95d13c22024-11-30 01:58:5588 {"ab\vcd", nullptr, nullptr},
Charlie Harrison7b633d92022-11-29 05:23:5089
90 // Empty string should yield an empty result.
Nidhi Jaju95d13c22024-11-30 01:58:5591 {"", "", ""},
Charlie Harrison7b633d92022-11-29 05:23:5092 };
93 for (const auto& test_case : kTestCases) {
94 SCOPED_TRACE(::testing::Message()
Nidhi Jaju95d13c22024-11-30 01:58:5595 << "Forgiving: " << EscapeAllExceptUnreserved(test_case.in));
Charlie Harrison7b633d92022-11-29 05:23:5096 std::string output;
97 bool success =
98 Base64Decode(test_case.in, &output, Base64DecodePolicy::kForgiving);
Nidhi Jaju95d13c22024-11-30 01:58:5599 bool expected_success = test_case.expected_out_forgiving != nullptr;
Charlie Harrison7b633d92022-11-29 05:23:50100 EXPECT_EQ(success, expected_success);
101 if (expected_success) {
Nidhi Jaju95d13c22024-11-30 01:58:55102 EXPECT_EQ(output, test_case.expected_out_forgiving);
103 }
104 }
105 for (const auto& test_case : kTestCases) {
106 SCOPED_TRACE(::testing::Message()
107 << "Strict: " << EscapeAllExceptUnreserved(test_case.in));
108 std::string output;
109 bool success =
110 Base64Decode(test_case.in, &output, Base64DecodePolicy::kStrict);
111 bool expected_success = test_case.expected_out_strict != nullptr;
112 EXPECT_EQ(success, expected_success);
113 if (expected_success) {
114 EXPECT_EQ(output, test_case.expected_out_strict);
Charlie Harrison7b633d92022-11-29 05:23:50115 }
116 }
117}
118
Collin Bakere21f723d2019-09-05 20:05:41119TEST(Base64Test, Binary) {
120 const uint8_t kData[] = {0x00, 0x01, 0xFE, 0xFF};
121
David Benjamin48808e22022-09-21 19:39:12122 std::string binary_encoded = Base64Encode(kData);
Collin Bakere21f723d2019-09-05 20:05:41123
Helmut Januschka0fc785b2024-04-17 21:13:36124 // Check that encoding the same data through the std::string_view interface
125 // gives the same results.
Tom Sepez2c860fd2024-02-01 21:31:26126 std::string string_piece_encoded = Base64Encode(
Helmut Januschka0fc785b2024-04-17 21:13:36127 std::string_view(reinterpret_cast<const char*>(kData), sizeof(kData)));
Collin Bakere21f723d2019-09-05 20:05:41128
129 EXPECT_EQ(binary_encoded, string_piece_encoded);
David Benjamin47bb5ec2022-02-01 23:12:28130
131 EXPECT_THAT(Base64Decode(binary_encoded),
132 testing::Optional(testing::ElementsAreArray(kData)));
133 EXPECT_FALSE(Base64Decode("invalid base64!"));
David Benjamin48808e22022-09-21 19:39:12134
135 std::string encoded_with_prefix = "PREFIX";
136 Base64EncodeAppend(kData, &encoded_with_prefix);
137 EXPECT_EQ(encoded_with_prefix, "PREFIX" + binary_encoded);
Collin Bakere21f723d2019-09-05 20:05:41138}
139
georgesak85e05a732014-11-11 17:19:43140TEST(Base64Test, InPlace) {
141 const std::string kText = "hello world";
142 const std::string kBase64Text = "aGVsbG8gd29ybGQ=";
georgesak85e05a732014-11-11 17:19:43143
wd l4a70b3ff2023-09-26 20:13:35144 std::string text = Base64Encode(kText);
georgesak85e05a732014-11-11 17:19:43145 EXPECT_EQ(kBase64Text, text);
146
147 bool ok = Base64Decode(text, &text);
148 EXPECT_TRUE(ok);
149 EXPECT_EQ(text, kText);
150}
151
David Benjamin48808e22022-09-21 19:39:12152TEST(Base64Test, Overflow) {
153 // `Base64Encode` makes the input larger, which means inputs whose base64
154 // output overflows `size_t`. Actually allocating a span of this size will
155 // likely fail, but we test it with a fake span and assume a correct
156 // implementation will check for overflow before touching the input.
157 //
158 // Note that, with or without an overflow check, the function will still
159 // crash. This test is only meaningful because `EXPECT_CHECK_DEATH` looks for
160 // a `CHECK`-based failure.
161 uint8_t b;
Peter Kastingae7e314f2024-11-27 18:04:07162 auto large_span = span(&b, MODP_B64_MAX_INPUT_LEN + 1);
David Benjamin48808e22022-09-21 19:39:12163 EXPECT_CHECK_DEATH(Base64Encode(large_span));
164
165 std::string output = "PREFIX";
166 EXPECT_CHECK_DEATH(Base64EncodeAppend(large_span, &output));
167
Charlie Harrison39d533222022-11-22 23:49:39168 // `modp_b64_encode_data_len` is a macro, so check `MODP_B64_MAX_INPUT_LEN` is
David Benjamin48808e22022-09-21 19:39:12169 // correct be verifying the computation doesn't overflow.
170 base::CheckedNumeric<size_t> max_len = MODP_B64_MAX_INPUT_LEN;
Charlie Harrison39d533222022-11-22 23:49:39171 EXPECT_TRUE(modp_b64_encode_data_len(max_len).IsValid());
David Benjamin48808e22022-09-21 19:39:12172}
173
[email protected]24f111a2012-09-23 04:51:04174} // namespace base