Enhance to_hex like the unix command hexdump.

This commit is contained in:
ngugcx 2020-03-15 12:01:31 +08:00
parent 8284865f9a
commit f5313f92f1
2 changed files with 122 additions and 28 deletions

View File

@ -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;
} }
} }
}; };

View File

@ -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;