2018-06-21 21:55:12 +02:00
|
|
|
//
|
|
|
|
// Copyright 2017 The Abseil Authors.
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
2019-03-08 16:27:53 +01:00
|
|
|
// https://www.apache.org/licenses/LICENSE-2.0
|
2018-06-21 21:55:12 +02:00
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
//
|
|
|
|
|
|
|
|
#include "absl/strings/internal/str_format/extension.h"
|
|
|
|
|
|
|
|
#include <random>
|
|
|
|
#include <string>
|
2019-03-19 19:14:01 +01:00
|
|
|
|
2020-04-17 17:13:06 +02:00
|
|
|
#include "absl/strings/cord.h"
|
|
|
|
#include "gtest/gtest.h"
|
2018-06-21 21:55:12 +02:00
|
|
|
#include "absl/strings/str_format.h"
|
2020-04-17 17:13:06 +02:00
|
|
|
#include "absl/strings/string_view.h"
|
2018-06-21 21:55:12 +02:00
|
|
|
|
2020-04-17 17:13:06 +02:00
|
|
|
namespace my_namespace {
|
|
|
|
class UserDefinedType {
|
|
|
|
public:
|
|
|
|
UserDefinedType() = default;
|
|
|
|
|
|
|
|
void Append(absl::string_view str) { value_.append(str.data(), str.size()); }
|
|
|
|
const std::string& Value() const { return value_; }
|
|
|
|
|
|
|
|
friend void AbslFormatFlush(UserDefinedType* x, absl::string_view str) {
|
|
|
|
x->Append(str);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
std::string value_;
|
|
|
|
};
|
|
|
|
} // namespace my_namespace
|
2018-06-21 21:55:12 +02:00
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
std::string MakeRandomString(size_t len) {
|
|
|
|
std::random_device rd;
|
|
|
|
std::mt19937 gen(rd());
|
|
|
|
std::uniform_int_distribution<> dis('a', 'z');
|
|
|
|
std::string s(len, '0');
|
|
|
|
for (char& c : s) {
|
|
|
|
c = dis(gen);
|
|
|
|
}
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(FormatExtensionTest, SinkAppendSubstring) {
|
|
|
|
for (size_t chunk_size : {1, 10, 100, 1000, 10000}) {
|
|
|
|
std::string expected, actual;
|
|
|
|
absl::str_format_internal::FormatSinkImpl sink(&actual);
|
|
|
|
for (size_t chunks = 0; chunks < 10; ++chunks) {
|
|
|
|
std::string rand = MakeRandomString(chunk_size);
|
|
|
|
expected += rand;
|
|
|
|
sink.Append(rand);
|
|
|
|
}
|
|
|
|
sink.Flush();
|
|
|
|
EXPECT_EQ(actual, expected);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(FormatExtensionTest, SinkAppendChars) {
|
|
|
|
for (size_t chunk_size : {1, 10, 100, 1000, 10000}) {
|
|
|
|
std::string expected, actual;
|
|
|
|
absl::str_format_internal::FormatSinkImpl sink(&actual);
|
|
|
|
for (size_t chunks = 0; chunks < 10; ++chunks) {
|
|
|
|
std::string rand = MakeRandomString(1);
|
|
|
|
expected.append(chunk_size, rand[0]);
|
|
|
|
sink.Append(chunk_size, rand[0]);
|
|
|
|
}
|
|
|
|
sink.Flush();
|
|
|
|
EXPECT_EQ(actual, expected);
|
|
|
|
}
|
|
|
|
}
|
2020-04-17 17:13:06 +02:00
|
|
|
|
|
|
|
TEST(FormatExtensionTest, CordSink) {
|
|
|
|
absl::Cord c;
|
|
|
|
absl::Format(&c, "There were %04d little %s.", 3, "pigs");
|
|
|
|
EXPECT_EQ(c, "There were 0003 little pigs.");
|
|
|
|
absl::Format(&c, "And %-3llx bad wolf!", 1);
|
|
|
|
EXPECT_EQ(c, "There were 0003 little pigs.And 1 bad wolf!");
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(FormatExtensionTest, CustomSink) {
|
|
|
|
my_namespace::UserDefinedType sink;
|
|
|
|
absl::Format(&sink, "There were %04d little %s.", 3, "pigs");
|
|
|
|
EXPECT_EQ("There were 0003 little pigs.", sink.Value());
|
|
|
|
absl::Format(&sink, "And %-3llx bad wolf!", 1);
|
|
|
|
EXPECT_EQ("There were 0003 little pigs.And 1 bad wolf!", sink.Value());
|
|
|
|
}
|
|
|
|
|
2018-06-21 21:55:12 +02:00
|
|
|
} // namespace
|