mirror of
				https://github.com/fmtlib/fmt.git
				synced 2025-11-03 23:51:41 +01:00 
			
		
		
		
	Signed-off-by: Vladislav Shchapov <vladislav@shchapov.ru> Signed-off-by: Vladislav Shchapov <vladislav@shchapov.ru>
		
			
				
	
	
		
			150 lines
		
	
	
		
			4.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			150 lines
		
	
	
		
			4.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
// Formatting library for C++ - tests of formatters for standard library types
 | 
						|
//
 | 
						|
// Copyright (c) 2012 - present, Victor Zverovich
 | 
						|
// All rights reserved.
 | 
						|
//
 | 
						|
// For the license information refer to format.h.
 | 
						|
 | 
						|
#include "fmt/std.h"
 | 
						|
 | 
						|
#include <string>
 | 
						|
#include <vector>
 | 
						|
 | 
						|
#include "fmt/os.h"  // fmt::system_category
 | 
						|
#include "fmt/ranges.h"
 | 
						|
#include "gtest-extra.h"  // StartsWith
 | 
						|
 | 
						|
using testing::StartsWith;
 | 
						|
 | 
						|
TEST(std_test, path) {
 | 
						|
#ifdef __cpp_lib_filesystem
 | 
						|
  EXPECT_EQ(fmt::format("{:8}", std::filesystem::path("foo")), "\"foo\"   ");
 | 
						|
  EXPECT_EQ(fmt::format("{}", std::filesystem::path("foo\"bar.txt")),
 | 
						|
            "\"foo\\\"bar.txt\"");
 | 
						|
 | 
						|
#  ifdef _WIN32
 | 
						|
  // File.txt in Russian.
 | 
						|
  const wchar_t unicode_path[] = {0x424, 0x430, 0x439, 0x43b, 0x2e,
 | 
						|
                                  0x74,  0x78,  0x74,  0};
 | 
						|
  const char unicode_u8path[] = {'"',        char(0xd0), char(0xa4), char(0xd0),
 | 
						|
                                 char(0xb0), char(0xd0), char(0xb9), char(0xd0),
 | 
						|
                                 char(0xbb), '.',        't',        'x',
 | 
						|
                                 't',        '"',        '\0'};
 | 
						|
  EXPECT_EQ(fmt::format("{}", std::filesystem::path(unicode_path)),
 | 
						|
            unicode_u8path);
 | 
						|
#  endif
 | 
						|
#endif
 | 
						|
}
 | 
						|
 | 
						|
TEST(ranges_std_test, format_vector_path) {
 | 
						|
// Test ambiguity problem described in #2954.
 | 
						|
#ifdef __cpp_lib_filesystem
 | 
						|
  auto p = std::filesystem::path("foo/bar.txt");
 | 
						|
  auto c = std::vector<std::string>{"abc", "def"};
 | 
						|
  EXPECT_EQ(fmt::format("path={}, range={}", p, c),
 | 
						|
            "path=\"foo/bar.txt\", range=[\"abc\", \"def\"]");
 | 
						|
#endif
 | 
						|
}
 | 
						|
 | 
						|
TEST(std_test, thread_id) {
 | 
						|
  EXPECT_FALSE(fmt::format("{}", std::this_thread::get_id()).empty());
 | 
						|
}
 | 
						|
 | 
						|
TEST(std_test, variant) {
 | 
						|
#ifdef __cpp_lib_variant
 | 
						|
  EXPECT_EQ(fmt::format("{}", std::monostate{}), "monostate");
 | 
						|
  using V0 = std::variant<int, float, std::string, char>;
 | 
						|
  V0 v0(42);
 | 
						|
  V0 v1(1.5f);
 | 
						|
  V0 v2("hello");
 | 
						|
  V0 v3('i');
 | 
						|
  EXPECT_EQ(fmt::format("{}", v0), "variant(42)");
 | 
						|
  EXPECT_EQ(fmt::format("{}", v1), "variant(1.5)");
 | 
						|
  EXPECT_EQ(fmt::format("{}", v2), "variant(\"hello\")");
 | 
						|
  EXPECT_EQ(fmt::format("{}", v3), "variant('i')");
 | 
						|
 | 
						|
  struct unformattable {};
 | 
						|
  EXPECT_FALSE((fmt::is_formattable<unformattable>::value));
 | 
						|
  EXPECT_FALSE((fmt::is_formattable<std::variant<unformattable>>::value));
 | 
						|
  EXPECT_FALSE((fmt::is_formattable<std::variant<unformattable, int>>::value));
 | 
						|
  EXPECT_FALSE((fmt::is_formattable<std::variant<int, unformattable>>::value));
 | 
						|
  EXPECT_FALSE(
 | 
						|
      (fmt::is_formattable<std::variant<unformattable, unformattable>>::value));
 | 
						|
  EXPECT_TRUE((fmt::is_formattable<std::variant<int, float>>::value));
 | 
						|
 | 
						|
  using V1 = std::variant<std::monostate, std::string, std::string>;
 | 
						|
  V1 v4{};
 | 
						|
  V1 v5{std::in_place_index<1>, "yes, this is variant"};
 | 
						|
 | 
						|
  EXPECT_EQ(fmt::format("{}", v4), "variant(monostate)");
 | 
						|
  EXPECT_EQ(fmt::format("{}", v5), "variant(\"yes, this is variant\")");
 | 
						|
 | 
						|
  volatile int i = 42;  // Test compile error before GCC 11 described in #3068.
 | 
						|
  EXPECT_EQ(fmt::format("{}", i), "42");
 | 
						|
#endif
 | 
						|
}
 | 
						|
 | 
						|
TEST(std_test, error_code) {
 | 
						|
  EXPECT_EQ("generic:42",
 | 
						|
            fmt::format(FMT_STRING("{0}"),
 | 
						|
                        std::error_code(42, std::generic_category())));
 | 
						|
  EXPECT_EQ("system:42",
 | 
						|
            fmt::format(FMT_STRING("{0}"),
 | 
						|
                        std::error_code(42, fmt::system_category())));
 | 
						|
  EXPECT_EQ("system:-42",
 | 
						|
            fmt::format(FMT_STRING("{0}"),
 | 
						|
                        std::error_code(-42, fmt::system_category())));
 | 
						|
}
 | 
						|
 | 
						|
template <typename Catch> void exception_test() {
 | 
						|
  try {
 | 
						|
    throw std::runtime_error("Test Exception");
 | 
						|
  } catch (const Catch& ex) {
 | 
						|
    EXPECT_EQ("Test Exception", fmt::format("{}", ex));
 | 
						|
    EXPECT_EQ("std::runtime_error: Test Exception", fmt::format("{:t}", ex));
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
namespace my_ns1 {
 | 
						|
namespace my_ns2 {
 | 
						|
struct my_exception : public std::exception {
 | 
						|
 private:
 | 
						|
  std::string msg;
 | 
						|
 | 
						|
 public:
 | 
						|
  my_exception(const std::string& s) : msg(s) {}
 | 
						|
  const char* what() const noexcept override;
 | 
						|
};
 | 
						|
const char* my_exception::what() const noexcept { return msg.c_str(); }
 | 
						|
}  // namespace my_ns2
 | 
						|
}  // namespace my_ns1
 | 
						|
 | 
						|
TEST(std_test, exception) {
 | 
						|
  exception_test<std::exception>();
 | 
						|
  exception_test<std::runtime_error>();
 | 
						|
 | 
						|
  try {
 | 
						|
    using namespace my_ns1::my_ns2;
 | 
						|
    throw my_exception("My Exception");
 | 
						|
  } catch (const std::exception& ex) {
 | 
						|
    EXPECT_EQ("my_ns1::my_ns2::my_exception: My Exception",
 | 
						|
              fmt::format("{:t}", ex));
 | 
						|
    EXPECT_EQ("My Exception", fmt::format("{:}", ex));
 | 
						|
  }
 | 
						|
 | 
						|
  try {
 | 
						|
    throw std::system_error(std::error_code(), "message");
 | 
						|
  } catch (const std::system_error& ex) {
 | 
						|
    EXPECT_THAT(fmt::format("{:t}", ex), StartsWith("std::system_error: "));
 | 
						|
  }
 | 
						|
 | 
						|
#ifdef __cpp_lib_filesystem
 | 
						|
  try {
 | 
						|
    throw std::filesystem::filesystem_error("message", std::error_code());
 | 
						|
  } catch (const std::filesystem::filesystem_error& ex) {
 | 
						|
    EXPECT_THAT(fmt::format("{:t}", ex),
 | 
						|
                StartsWith("std::filesystem::filesystem_error: "));
 | 
						|
  }
 | 
						|
#endif
 | 
						|
}
 |