mirror of
https://github.com/gabime/spdlog.git
synced 2025-01-26 15:39:03 +08:00
Enhance to_hex like the unix command hexdump.
This commit is contained in:
parent
8284865f9a
commit
f5313f92f1
@ -5,6 +5,8 @@
|
|||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
|
#include <cctype>
|
||||||
|
|
||||||
//
|
//
|
||||||
// Support for logging binary data as hex
|
// Support for logging binary data as hex
|
||||||
// format flags:
|
// format flags:
|
||||||
@ -12,6 +14,7 @@
|
|||||||
// {:s} - don't separate each byte with space.
|
// {:s} - don't separate each byte with space.
|
||||||
// {:p} - don't print the position on each line start.
|
// {:p} - don't print the position on each line start.
|
||||||
// {:n} - don't split the output to lines.
|
// {:n} - don't split the output to lines.
|
||||||
|
// {:a} - show ASCII if :n is not set
|
||||||
|
|
||||||
//
|
//
|
||||||
// Examples:
|
// Examples:
|
||||||
@ -20,17 +23,19 @@
|
|||||||
// logger->info("Some buffer {}", spdlog::to_hex(v));
|
// logger->info("Some buffer {}", spdlog::to_hex(v));
|
||||||
// char buf[128];
|
// char buf[128];
|
||||||
// logger->info("Some buffer {:X}", spdlog::to_hex(std::begin(buf), std::end(buf)));
|
// logger->info("Some buffer {:X}", spdlog::to_hex(std::begin(buf), std::end(buf)));
|
||||||
|
// logger->info("Some buffer {:X}", spdlog::to_hex(std::begin(buf), std::end(buf), 16));
|
||||||
|
|
||||||
namespace spdlog {
|
namespace spdlog {
|
||||||
namespace details {
|
namespace details {
|
||||||
|
|
||||||
template<typename It>
|
template<typename It>
|
||||||
class bytes_range
|
class dump_info
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
bytes_range(It range_begin, It range_end)
|
dump_info(It range_begin, It range_end, size_t size_per_line)
|
||||||
: begin_(range_begin)
|
: begin_(range_begin)
|
||||||
, end_(range_end)
|
, end_(range_end)
|
||||||
|
, size_per_line_(size_per_line)
|
||||||
{}
|
{}
|
||||||
|
|
||||||
It begin() const
|
It begin() const
|
||||||
@ -41,26 +46,31 @@ public:
|
|||||||
{
|
{
|
||||||
return end_;
|
return end_;
|
||||||
}
|
}
|
||||||
|
size_t size_per_line() const
|
||||||
|
{
|
||||||
|
return size_per_line_;
|
||||||
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
It begin_, end_;
|
It begin_, end_;
|
||||||
|
size_t size_per_line_;
|
||||||
};
|
};
|
||||||
} // namespace details
|
} // namespace details
|
||||||
|
|
||||||
// create a bytes_range that wraps the given container
|
// create a dump_info that wraps the given container
|
||||||
template<typename Container>
|
template<typename Container>
|
||||||
inline details::bytes_range<typename Container::const_iterator> to_hex(const Container &container)
|
inline details::dump_info<typename Container::const_iterator> to_hex(const Container &container, size_t size_per_line = 32)
|
||||||
{
|
{
|
||||||
static_assert(sizeof(typename Container::value_type) == 1, "sizeof(Container::value_type) != 1");
|
static_assert(sizeof(typename Container::value_type) == 1, "sizeof(Container::value_type) != 1");
|
||||||
using Iter = typename Container::const_iterator;
|
using Iter = typename Container::const_iterator;
|
||||||
return details::bytes_range<Iter>(std::begin(container), std::end(container));
|
return details::dump_info<Iter>(std::begin(container), std::end(container), size_per_line);
|
||||||
}
|
}
|
||||||
|
|
||||||
// create bytes_range from ranges
|
// create dump_info from ranges
|
||||||
template<typename It>
|
template<typename It>
|
||||||
inline details::bytes_range<It> to_hex(const It range_begin, const It range_end)
|
inline details::dump_info<It> to_hex(const It range_begin, const It range_end, size_t size_per_line = 32)
|
||||||
{
|
{
|
||||||
return details::bytes_range<It>(range_begin, range_end);
|
return details::dump_info<It>(range_begin, range_end, size_per_line);
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace spdlog
|
} // namespace spdlog
|
||||||
@ -68,7 +78,7 @@ inline details::bytes_range<It> to_hex(const It range_begin, const It range_end)
|
|||||||
namespace fmt {
|
namespace fmt {
|
||||||
|
|
||||||
template<typename T>
|
template<typename T>
|
||||||
struct formatter<spdlog::details::bytes_range<T>>
|
struct formatter<spdlog::details::dump_info<T>>
|
||||||
{
|
{
|
||||||
const std::size_t line_size = 100;
|
const std::size_t line_size = 100;
|
||||||
const char delimiter = ' ';
|
const char delimiter = ' ';
|
||||||
@ -77,6 +87,7 @@ struct formatter<spdlog::details::bytes_range<T>>
|
|||||||
bool put_delimiters = true;
|
bool put_delimiters = true;
|
||||||
bool use_uppercase = false;
|
bool use_uppercase = false;
|
||||||
bool put_positions = true; // position on start of each line
|
bool put_positions = true; // position on start of each line
|
||||||
|
bool show_ascii = false;
|
||||||
|
|
||||||
// parse the format string flags
|
// parse the format string flags
|
||||||
template<typename ParseContext>
|
template<typename ParseContext>
|
||||||
@ -98,6 +109,13 @@ struct formatter<spdlog::details::bytes_range<T>>
|
|||||||
break;
|
break;
|
||||||
case 'n':
|
case 'n':
|
||||||
put_newlines = false;
|
put_newlines = false;
|
||||||
|
show_ascii = false;
|
||||||
|
break;
|
||||||
|
case 'a':
|
||||||
|
if (put_newlines)
|
||||||
|
{
|
||||||
|
show_ascii = true;
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -108,53 +126,81 @@ struct formatter<spdlog::details::bytes_range<T>>
|
|||||||
|
|
||||||
// format the given bytes range as hex
|
// format the given bytes range as hex
|
||||||
template<typename FormatContext, typename Container>
|
template<typename FormatContext, typename Container>
|
||||||
auto format(const spdlog::details::bytes_range<Container> &the_range, FormatContext &ctx) -> decltype(ctx.out())
|
auto format(const spdlog::details::dump_info<Container> &the_range, FormatContext &ctx) -> decltype(ctx.out())
|
||||||
{
|
{
|
||||||
SPDLOG_CONSTEXPR const char *hex_upper = "0123456789ABCDEF";
|
SPDLOG_CONSTEXPR const char *hex_upper = "0123456789ABCDEF";
|
||||||
SPDLOG_CONSTEXPR const char *hex_lower = "0123456789abcdef";
|
SPDLOG_CONSTEXPR const char *hex_lower = "0123456789abcdef";
|
||||||
const char *hex_chars = use_uppercase ? hex_upper : hex_lower;
|
const char *hex_chars = use_uppercase ? hex_upper : hex_lower;
|
||||||
|
|
||||||
std::size_t pos = 0;
|
|
||||||
std::size_t column = line_size;
|
|
||||||
#if FMT_VERSION < 60000
|
#if FMT_VERSION < 60000
|
||||||
auto inserter = ctx.begin();
|
auto inserter = ctx.begin();
|
||||||
#else
|
#else
|
||||||
auto inserter = ctx.out();
|
auto inserter = ctx.out();
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
for (auto &item : the_range)
|
int size_per_line = static_cast<int>(the_range.size_per_line());
|
||||||
|
auto start_of_line = the_range.begin();
|
||||||
|
for (auto i = the_range.begin(); i != the_range.end(); i++)
|
||||||
{
|
{
|
||||||
auto ch = static_cast<unsigned char>(item);
|
auto ch = static_cast<unsigned char>(*i);
|
||||||
pos++;
|
|
||||||
|
|
||||||
if (put_newlines && column >= line_size)
|
if (put_newlines && (i == the_range.begin() || i - start_of_line >= size_per_line))
|
||||||
{
|
{
|
||||||
column = put_newline(inserter, pos);
|
if (show_ascii && i != the_range.begin())
|
||||||
|
{
|
||||||
|
*inserter++ = ' ';
|
||||||
|
*inserter++ = ' ';
|
||||||
|
for (auto j = start_of_line; j < i; j++)
|
||||||
|
{
|
||||||
|
auto pc = static_cast<unsigned char>(*j);
|
||||||
|
*inserter++ = std::isprint(pc) ? static_cast<char>(*j) : '.';
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
put_newline(inserter, i - the_range.begin());
|
||||||
|
|
||||||
// put first byte without delimiter in front of it
|
// put first byte without delimiter in front of it
|
||||||
*inserter++ = hex_chars[(ch >> 4) & 0x0f];
|
*inserter++ = hex_chars[(ch >> 4) & 0x0f];
|
||||||
*inserter++ = hex_chars[ch & 0x0f];
|
*inserter++ = hex_chars[ch & 0x0f];
|
||||||
column += 2;
|
start_of_line = i;
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (put_delimiters)
|
if (put_delimiters)
|
||||||
{
|
{
|
||||||
*inserter++ = delimiter;
|
*inserter++ = delimiter;
|
||||||
++column;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
*inserter++ = hex_chars[(ch >> 4) & 0x0f];
|
*inserter++ = hex_chars[(ch >> 4) & 0x0f];
|
||||||
*inserter++ = hex_chars[ch & 0x0f];
|
*inserter++ = hex_chars[ch & 0x0f];
|
||||||
column += 2;
|
}
|
||||||
|
if (show_ascii)
|
||||||
|
{
|
||||||
|
auto blank_num = size_per_line - (the_range.end() - start_of_line);
|
||||||
|
while (blank_num-- > 0)
|
||||||
|
{
|
||||||
|
*inserter++ = ' ';
|
||||||
|
*inserter++ = ' ';
|
||||||
|
if (put_delimiters)
|
||||||
|
{
|
||||||
|
*inserter++ = ' ';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
*inserter++ = ' ';
|
||||||
|
*inserter++ = ' ';
|
||||||
|
for (auto j = start_of_line; j != the_range.end(); j++)
|
||||||
|
{
|
||||||
|
auto pc = static_cast<unsigned char>(*j);
|
||||||
|
*inserter++ = std::isprint(pc) ? static_cast<char>(*j) : '.';
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return inserter;
|
return inserter;
|
||||||
}
|
}
|
||||||
|
|
||||||
// put newline(and position header)
|
// put newline(and position header)
|
||||||
// return the next column
|
|
||||||
template<typename It>
|
template<typename It>
|
||||||
std::size_t put_newline(It inserter, std::size_t pos)
|
void put_newline(It inserter, std::size_t pos)
|
||||||
{
|
{
|
||||||
#ifdef _WIN32
|
#ifdef _WIN32
|
||||||
*inserter++ = '\r';
|
*inserter++ = '\r';
|
||||||
@ -163,12 +209,7 @@ struct formatter<spdlog::details::bytes_range<T>>
|
|||||||
|
|
||||||
if (put_positions)
|
if (put_positions)
|
||||||
{
|
{
|
||||||
fmt::format_to(inserter, "{:<04X}: ", pos - 1);
|
fmt::format_to(inserter, "{:<04X}: ", pos);
|
||||||
return 7;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
return 1;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
@ -181,6 +181,59 @@ TEST_CASE("to_hex_no_delimiter", "[to_hex]")
|
|||||||
REQUIRE(ends_with(output, "0000: 090A0B0CFFFF" + std::string(spdlog::details::os::default_eol)));
|
REQUIRE(ends_with(output, "0000: 090A0B0CFFFF" + std::string(spdlog::details::os::default_eol)));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
TEST_CASE("to_hex_show_ascii", "[to_hex]")
|
||||||
|
{
|
||||||
|
std::ostringstream oss;
|
||||||
|
auto oss_sink = std::make_shared<spdlog::sinks::ostream_sink_mt>(oss);
|
||||||
|
spdlog::logger oss_logger("oss", oss_sink);
|
||||||
|
|
||||||
|
std::vector<unsigned char> v{9, 0xa, 0xb, 0x41, 0xc, 0x4b, 0xff, 0xff};
|
||||||
|
oss_logger.info("{:Xsa}", spdlog::to_hex(v, 8));
|
||||||
|
|
||||||
|
REQUIRE(ends_with(oss.str(), "0000: 090A0B410C4BFFFF ...A.K.." + std::string(spdlog::details::os::default_eol)));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_CASE("to_hex_different_size_per_line", "[to_hex]")
|
||||||
|
{
|
||||||
|
std::ostringstream oss;
|
||||||
|
auto oss_sink = std::make_shared<spdlog::sinks::ostream_sink_mt>(oss);
|
||||||
|
spdlog::logger oss_logger("oss", oss_sink);
|
||||||
|
|
||||||
|
std::vector<unsigned char> v{9, 0xa, 0xb, 0x41, 0xc, 0x4b, 0xff, 0xff};
|
||||||
|
|
||||||
|
oss_logger.info("{:Xsa}", spdlog::to_hex(v, 10));
|
||||||
|
REQUIRE(ends_with(oss.str(), "0000: 090A0B410C4BFFFF ...A.K.." + std::string(spdlog::details::os::default_eol)));
|
||||||
|
|
||||||
|
oss_logger.info("{:Xs}", spdlog::to_hex(v, 10));
|
||||||
|
REQUIRE(ends_with(oss.str(), "0000: 090A0B410C4BFFFF" + std::string(spdlog::details::os::default_eol)));
|
||||||
|
|
||||||
|
oss_logger.info("{:Xsa}", spdlog::to_hex(v, 6));
|
||||||
|
REQUIRE(ends_with(oss.str(),
|
||||||
|
"0000: 090A0B410C4B ...A.K" + std::string(spdlog::details::os::default_eol)
|
||||||
|
+ "0006: FFFF .." + std::string(spdlog::details::os::default_eol)));
|
||||||
|
|
||||||
|
oss_logger.info("{:Xs}", spdlog::to_hex(v, 6));
|
||||||
|
REQUIRE(ends_with(oss.str(),
|
||||||
|
"0000: 090A0B410C4B" + std::string(spdlog::details::os::default_eol)
|
||||||
|
+ "0006: FFFF" + std::string(spdlog::details::os::default_eol)));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_CASE("to_hex_no_ascii", "[to_hex]")
|
||||||
|
{
|
||||||
|
std::ostringstream oss;
|
||||||
|
auto oss_sink = std::make_shared<spdlog::sinks::ostream_sink_mt>(oss);
|
||||||
|
spdlog::logger oss_logger("oss", oss_sink);
|
||||||
|
|
||||||
|
std::vector<unsigned char> v{9, 0xa, 0xb, 0x41, 0xc, 0x4b, 0xff, 0xff};
|
||||||
|
oss_logger.info("{:Xs}", spdlog::to_hex(v, 8));
|
||||||
|
|
||||||
|
REQUIRE(ends_with(oss.str(), "0000: 090A0B410C4BFFFF" + std::string(spdlog::details::os::default_eol)));
|
||||||
|
|
||||||
|
oss_logger.info("{:Xsna}", spdlog::to_hex(v, 8));
|
||||||
|
|
||||||
|
REQUIRE(ends_with(oss.str(), "090A0B410C4BFFFF" + std::string(spdlog::details::os::default_eol)));
|
||||||
|
}
|
||||||
|
|
||||||
TEST_CASE("default logger API", "[default logger]")
|
TEST_CASE("default logger API", "[default logger]")
|
||||||
{
|
{
|
||||||
std::ostringstream oss;
|
std::ostringstream oss;
|
||||||
|
Loading…
Reference in New Issue
Block a user