mirror of
https://github.com/fmtlib/fmt.git
synced 2025-02-04 06:39:59 +00:00
Add args-test
This commit is contained in:
parent
1147782c79
commit
a6408a3b09
@ -87,6 +87,7 @@ function(add_fmt_test name)
|
|||||||
add_test(NAME ${name} COMMAND ${name})
|
add_test(NAME ${name} COMMAND ${name})
|
||||||
endfunction()
|
endfunction()
|
||||||
|
|
||||||
|
add_fmt_test(args-test)
|
||||||
add_fmt_test(assert-test)
|
add_fmt_test(assert-test)
|
||||||
add_fmt_test(chrono-test)
|
add_fmt_test(chrono-test)
|
||||||
add_fmt_test(color-test)
|
add_fmt_test(color-test)
|
||||||
|
149
test/args-test.cc
Normal file
149
test/args-test.cc
Normal file
@ -0,0 +1,149 @@
|
|||||||
|
// Formatting library for C++ - dynamic argument store tests
|
||||||
|
//
|
||||||
|
// Copyright (c) 2012 - present, Victor Zverovich
|
||||||
|
// All rights reserved.
|
||||||
|
//
|
||||||
|
// For the license information refer to format.h.
|
||||||
|
|
||||||
|
#include "fmt/args.h"
|
||||||
|
|
||||||
|
#include "gmock.h"
|
||||||
|
|
||||||
|
TEST(ArgsTest, Basic) {
|
||||||
|
fmt::dynamic_format_arg_store<fmt::format_context> store;
|
||||||
|
store.push_back(42);
|
||||||
|
store.push_back("abc1");
|
||||||
|
store.push_back(1.5f);
|
||||||
|
EXPECT_EQ("42 and abc1 and 1.5", fmt::vformat("{} and {} and {}", store));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(ArgsTest, StringsAndRefs) {
|
||||||
|
// Unfortunately the tests are compiled with old ABI so strings use COW.
|
||||||
|
fmt::dynamic_format_arg_store<fmt::format_context> store;
|
||||||
|
char str[] = "1234567890";
|
||||||
|
store.push_back(str);
|
||||||
|
store.push_back(std::cref(str));
|
||||||
|
store.push_back(fmt::string_view{str});
|
||||||
|
str[0] = 'X';
|
||||||
|
|
||||||
|
std::string result = fmt::vformat("{} and {} and {}", store);
|
||||||
|
EXPECT_EQ("1234567890 and X234567890 and X234567890", result);
|
||||||
|
}
|
||||||
|
|
||||||
|
struct custom_type {
|
||||||
|
int i = 0;
|
||||||
|
};
|
||||||
|
|
||||||
|
FMT_BEGIN_NAMESPACE
|
||||||
|
template <> struct formatter<custom_type> {
|
||||||
|
auto parse(format_parse_context& ctx) const -> decltype(ctx.begin()) {
|
||||||
|
return ctx.begin();
|
||||||
|
}
|
||||||
|
|
||||||
|
template <typename FormatContext>
|
||||||
|
auto format(const custom_type& p, FormatContext& ctx) -> decltype(ctx.out()) {
|
||||||
|
return format_to(ctx.out(), "cust={}", p.i);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
FMT_END_NAMESPACE
|
||||||
|
|
||||||
|
TEST(ArgsTest, CustomFormat) {
|
||||||
|
fmt::dynamic_format_arg_store<fmt::format_context> store;
|
||||||
|
custom_type c{};
|
||||||
|
store.push_back(c);
|
||||||
|
++c.i;
|
||||||
|
store.push_back(c);
|
||||||
|
++c.i;
|
||||||
|
store.push_back(std::cref(c));
|
||||||
|
++c.i;
|
||||||
|
std::string result = fmt::vformat("{} and {} and {}", store);
|
||||||
|
EXPECT_EQ("cust=0 and cust=1 and cust=3", result);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(ArgsTest, NamedInt) {
|
||||||
|
fmt::dynamic_format_arg_store<fmt::format_context> store;
|
||||||
|
store.push_back(fmt::arg("a1", 42));
|
||||||
|
EXPECT_EQ("42", fmt::vformat("{a1}", store));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(ArgsTest, NamedStrings) {
|
||||||
|
fmt::dynamic_format_arg_store<fmt::format_context> store;
|
||||||
|
char str[]{"1234567890"};
|
||||||
|
store.push_back(fmt::arg("a1", str));
|
||||||
|
store.push_back(fmt::arg("a2", std::cref(str)));
|
||||||
|
str[0] = 'X';
|
||||||
|
EXPECT_EQ("1234567890 and X234567890", fmt::vformat("{a1} and {a2}", store));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(ArgsTest, NamedArgByRef) {
|
||||||
|
fmt::dynamic_format_arg_store<fmt::format_context> store;
|
||||||
|
char band[] = "Rolling Stones";
|
||||||
|
store.push_back(fmt::arg("band", std::cref(band)));
|
||||||
|
band[9] = 'c'; // Changing band affects the output.
|
||||||
|
EXPECT_EQ(fmt::vformat("{band}", store), "Rolling Scones");
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(ArgsTest, NamedCustomFormat) {
|
||||||
|
fmt::dynamic_format_arg_store<fmt::format_context> store;
|
||||||
|
custom_type c{};
|
||||||
|
store.push_back(fmt::arg("c1", c));
|
||||||
|
++c.i;
|
||||||
|
store.push_back(fmt::arg("c2", c));
|
||||||
|
++c.i;
|
||||||
|
store.push_back(fmt::arg("c_ref", std::cref(c)));
|
||||||
|
++c.i;
|
||||||
|
std::string result = fmt::vformat("{c1} and {c2} and {c_ref}", store);
|
||||||
|
EXPECT_EQ("cust=0 and cust=1 and cust=3", result);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(ArgsTest, Clear) {
|
||||||
|
fmt::dynamic_format_arg_store<fmt::format_context> store;
|
||||||
|
store.push_back(42);
|
||||||
|
|
||||||
|
std::string result = fmt::vformat("{}", store);
|
||||||
|
EXPECT_EQ("42", result);
|
||||||
|
|
||||||
|
store.push_back(43);
|
||||||
|
result = fmt::vformat("{} and {}", store);
|
||||||
|
EXPECT_EQ("42 and 43", result);
|
||||||
|
|
||||||
|
store.clear();
|
||||||
|
store.push_back(44);
|
||||||
|
result = fmt::vformat("{}", store);
|
||||||
|
EXPECT_EQ("44", result);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(ArgsTest, Reserve) {
|
||||||
|
fmt::dynamic_format_arg_store<fmt::format_context> store;
|
||||||
|
store.reserve(2, 1);
|
||||||
|
store.push_back(1.5f);
|
||||||
|
store.push_back(fmt::arg("a1", 42));
|
||||||
|
std::string result = fmt::vformat("{a1} and {}", store);
|
||||||
|
EXPECT_EQ("42 and 1.5", result);
|
||||||
|
}
|
||||||
|
|
||||||
|
struct copy_throwable {
|
||||||
|
copy_throwable() {}
|
||||||
|
copy_throwable(const copy_throwable&) { throw "deal with it"; }
|
||||||
|
};
|
||||||
|
|
||||||
|
FMT_BEGIN_NAMESPACE
|
||||||
|
template <> struct formatter<copy_throwable> {
|
||||||
|
auto parse(format_parse_context& ctx) const -> decltype(ctx.begin()) {
|
||||||
|
return ctx.begin();
|
||||||
|
}
|
||||||
|
auto format(copy_throwable, format_context& ctx) -> decltype(ctx.out()) {
|
||||||
|
return ctx.out();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
FMT_END_NAMESPACE
|
||||||
|
|
||||||
|
TEST(ArgsTest, ThrowOnCopy) {
|
||||||
|
fmt::dynamic_format_arg_store<fmt::format_context> store;
|
||||||
|
store.push_back(std::string("foo"));
|
||||||
|
try {
|
||||||
|
store.push_back(copy_throwable());
|
||||||
|
} catch (...) {
|
||||||
|
}
|
||||||
|
EXPECT_EQ(fmt::vformat("{}", store), "foo");
|
||||||
|
}
|
@ -23,7 +23,7 @@
|
|||||||
# include <windows.h>
|
# include <windows.h>
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#include "fmt/args.h"
|
#include "fmt/core.h"
|
||||||
#if defined(FMT_COMPILE_TIME_CHECKS) && FMT_COMPILE_TIME_CHECKS
|
#if defined(FMT_COMPILE_TIME_CHECKS) && FMT_COMPILE_TIME_CHECKS
|
||||||
# include "fmt/format.h"
|
# include "fmt/format.h"
|
||||||
#endif
|
#endif
|
||||||
@ -401,145 +401,6 @@ TEST(ArgTest, VisitInvalidArg) {
|
|||||||
fmt::visit_format_arg(visitor, arg);
|
fmt::visit_format_arg(visitor, arg);
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST(FormatDynArgsTest, Basic) {
|
|
||||||
fmt::dynamic_format_arg_store<fmt::format_context> store;
|
|
||||||
store.push_back(42);
|
|
||||||
store.push_back("abc1");
|
|
||||||
store.push_back(1.5f);
|
|
||||||
EXPECT_EQ("42 and abc1 and 1.5", fmt::vformat("{} and {} and {}", store));
|
|
||||||
}
|
|
||||||
|
|
||||||
TEST(FormatDynArgsTest, StringsAndRefs) {
|
|
||||||
// Unfortunately the tests are compiled with old ABI so strings use COW.
|
|
||||||
fmt::dynamic_format_arg_store<fmt::format_context> store;
|
|
||||||
char str[] = "1234567890";
|
|
||||||
store.push_back(str);
|
|
||||||
store.push_back(std::cref(str));
|
|
||||||
store.push_back(fmt::string_view{str});
|
|
||||||
str[0] = 'X';
|
|
||||||
|
|
||||||
std::string result = fmt::vformat("{} and {} and {}", store);
|
|
||||||
EXPECT_EQ("1234567890 and X234567890 and X234567890", result);
|
|
||||||
}
|
|
||||||
|
|
||||||
struct custom_type {
|
|
||||||
int i = 0;
|
|
||||||
};
|
|
||||||
|
|
||||||
FMT_BEGIN_NAMESPACE
|
|
||||||
template <> struct formatter<custom_type> {
|
|
||||||
auto parse(format_parse_context& ctx) const -> decltype(ctx.begin()) {
|
|
||||||
return ctx.begin();
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename FormatContext>
|
|
||||||
auto format(const custom_type& p, FormatContext& ctx) -> decltype(ctx.out()) {
|
|
||||||
return format_to(ctx.out(), "cust={}", p.i);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
FMT_END_NAMESPACE
|
|
||||||
|
|
||||||
TEST(FormatDynArgsTest, CustomFormat) {
|
|
||||||
fmt::dynamic_format_arg_store<fmt::format_context> store;
|
|
||||||
custom_type c{};
|
|
||||||
store.push_back(c);
|
|
||||||
++c.i;
|
|
||||||
store.push_back(c);
|
|
||||||
++c.i;
|
|
||||||
store.push_back(std::cref(c));
|
|
||||||
++c.i;
|
|
||||||
std::string result = fmt::vformat("{} and {} and {}", store);
|
|
||||||
EXPECT_EQ("cust=0 and cust=1 and cust=3", result);
|
|
||||||
}
|
|
||||||
|
|
||||||
TEST(FormatDynArgsTest, NamedInt) {
|
|
||||||
fmt::dynamic_format_arg_store<fmt::format_context> store;
|
|
||||||
store.push_back(fmt::arg("a1", 42));
|
|
||||||
EXPECT_EQ("42", fmt::vformat("{a1}", store));
|
|
||||||
}
|
|
||||||
|
|
||||||
TEST(FormatDynArgsTest, NamedStrings) {
|
|
||||||
fmt::dynamic_format_arg_store<fmt::format_context> store;
|
|
||||||
char str[]{"1234567890"};
|
|
||||||
store.push_back(fmt::arg("a1", str));
|
|
||||||
store.push_back(fmt::arg("a2", std::cref(str)));
|
|
||||||
str[0] = 'X';
|
|
||||||
EXPECT_EQ("1234567890 and X234567890", fmt::vformat("{a1} and {a2}", store));
|
|
||||||
}
|
|
||||||
|
|
||||||
TEST(FormatDynArgsTest, NamedArgByRef) {
|
|
||||||
fmt::dynamic_format_arg_store<fmt::format_context> store;
|
|
||||||
char band[] = "Rolling Stones";
|
|
||||||
store.push_back(fmt::arg("band", std::cref(band)));
|
|
||||||
band[9] = 'c'; // Changing band affects the output.
|
|
||||||
EXPECT_EQ(fmt::vformat("{band}", store), "Rolling Scones");
|
|
||||||
}
|
|
||||||
|
|
||||||
TEST(FormatDynArgsTest, NamedCustomFormat) {
|
|
||||||
fmt::dynamic_format_arg_store<fmt::format_context> store;
|
|
||||||
custom_type c{};
|
|
||||||
store.push_back(fmt::arg("c1", c));
|
|
||||||
++c.i;
|
|
||||||
store.push_back(fmt::arg("c2", c));
|
|
||||||
++c.i;
|
|
||||||
store.push_back(fmt::arg("c_ref", std::cref(c)));
|
|
||||||
++c.i;
|
|
||||||
std::string result = fmt::vformat("{c1} and {c2} and {c_ref}", store);
|
|
||||||
EXPECT_EQ("cust=0 and cust=1 and cust=3", result);
|
|
||||||
}
|
|
||||||
|
|
||||||
TEST(FormatDynArgsTest, Clear) {
|
|
||||||
fmt::dynamic_format_arg_store<fmt::format_context> store;
|
|
||||||
store.push_back(42);
|
|
||||||
|
|
||||||
std::string result = fmt::vformat("{}", store);
|
|
||||||
EXPECT_EQ("42", result);
|
|
||||||
|
|
||||||
store.push_back(43);
|
|
||||||
result = fmt::vformat("{} and {}", store);
|
|
||||||
EXPECT_EQ("42 and 43", result);
|
|
||||||
|
|
||||||
store.clear();
|
|
||||||
store.push_back(44);
|
|
||||||
result = fmt::vformat("{}", store);
|
|
||||||
EXPECT_EQ("44", result);
|
|
||||||
}
|
|
||||||
|
|
||||||
TEST(FormatDynArgsTest, Reserve) {
|
|
||||||
fmt::dynamic_format_arg_store<fmt::format_context> store;
|
|
||||||
store.reserve(2, 1);
|
|
||||||
store.push_back(1.5f);
|
|
||||||
store.push_back(fmt::arg("a1", 42));
|
|
||||||
std::string result = fmt::vformat("{a1} and {}", store);
|
|
||||||
EXPECT_EQ("42 and 1.5", result);
|
|
||||||
}
|
|
||||||
|
|
||||||
struct copy_throwable {
|
|
||||||
copy_throwable() {}
|
|
||||||
copy_throwable(const copy_throwable&) { throw "deal with it"; }
|
|
||||||
};
|
|
||||||
|
|
||||||
FMT_BEGIN_NAMESPACE
|
|
||||||
template <> struct formatter<copy_throwable> {
|
|
||||||
auto parse(format_parse_context& ctx) const -> decltype(ctx.begin()) {
|
|
||||||
return ctx.begin();
|
|
||||||
}
|
|
||||||
auto format(copy_throwable, format_context& ctx) -> decltype(ctx.out()) {
|
|
||||||
return ctx.out();
|
|
||||||
}
|
|
||||||
};
|
|
||||||
FMT_END_NAMESPACE
|
|
||||||
|
|
||||||
TEST(FormatDynArgsTest, ThrowOnCopy) {
|
|
||||||
fmt::dynamic_format_arg_store<fmt::format_context> store;
|
|
||||||
store.push_back(std::string("foo"));
|
|
||||||
try {
|
|
||||||
store.push_back(copy_throwable());
|
|
||||||
} catch (...) {
|
|
||||||
}
|
|
||||||
EXPECT_EQ(fmt::vformat("{}", store), "foo");
|
|
||||||
}
|
|
||||||
|
|
||||||
TEST(StringViewTest, ValueType) {
|
TEST(StringViewTest, ValueType) {
|
||||||
static_assert(std::is_same<string_view::value_type, char>::value, "");
|
static_assert(std::is_same<string_view::value_type, char>::value, "");
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user