mirror of
https://github.com/catchorg/Catch2.git
synced 2025-10-17 01:05:32 +02:00
The old code was exceedingly simple, as it went char-by-char and decided whether to write it to the output stream as-is, or escaped. This caused a _lot_ of stream writes of individual characters. The new code instead looks for characters that need escaping, and bulk-writes the non-escaped characters in between them. This leads to about the same performance for strings that comprise of only escaped characters, and 3-10x improvement for strings without any escaping needed. In practice, we should expect the former rather than the latter, but this is still nice improvement.
166 lines
5.4 KiB
C++
166 lines
5.4 KiB
C++
|
|
// Copyright Catch2 Authors
|
|
// Distributed under the Boost Software License, Version 1.0.
|
|
// (See accompanying file LICENSE.txt or copy at
|
|
// https://www.boost.org/LICENSE_1_0.txt)
|
|
|
|
// SPDX-License-Identifier: BSL-1.0
|
|
#include <catch2/internal/catch_enforce.hpp>
|
|
#include <catch2/internal/catch_jsonwriter.hpp>
|
|
#include <catch2/internal/catch_unreachable.hpp>
|
|
|
|
namespace Catch {
|
|
|
|
namespace {
|
|
static bool needsEscape( char c ) {
|
|
return c == '"' || c == '\\' || c == '\b' || c == '\f' ||
|
|
c == '\n' || c == '\r' || c == '\t';
|
|
}
|
|
|
|
static Catch::StringRef makeEscapeStringRef( char c ) {
|
|
if ( c == '"' ) {
|
|
return "\\\""_sr;
|
|
} else if ( c == '\\' ) {
|
|
return "\\\\"_sr;
|
|
} else if ( c == '\b' ) {
|
|
return "\\b"_sr;
|
|
} else if ( c == '\f' ) {
|
|
return "\\f"_sr;
|
|
} else if ( c == '\n' ) {
|
|
return "\\n"_sr;
|
|
} else if ( c == '\r' ) {
|
|
return "\\r"_sr;
|
|
} else if ( c == '\t' ) {
|
|
return "\\t"_sr;
|
|
}
|
|
Catch::Detail::Unreachable();
|
|
}
|
|
} // namespace
|
|
|
|
void JsonUtils::indent( std::ostream& os, std::uint64_t level ) {
|
|
for ( std::uint64_t i = 0; i < level; ++i ) {
|
|
os << " ";
|
|
}
|
|
}
|
|
void JsonUtils::appendCommaNewline( std::ostream& os,
|
|
bool& should_comma,
|
|
std::uint64_t level ) {
|
|
if ( should_comma ) { os << ','; }
|
|
should_comma = true;
|
|
os << '\n';
|
|
indent( os, level );
|
|
}
|
|
|
|
JsonObjectWriter::JsonObjectWriter( std::ostream& os ):
|
|
JsonObjectWriter{ os, 0 } {}
|
|
|
|
JsonObjectWriter::JsonObjectWriter( std::ostream& os,
|
|
std::uint64_t indent_level ):
|
|
m_os{ os }, m_indent_level{ indent_level } {
|
|
m_os << '{';
|
|
}
|
|
JsonObjectWriter::JsonObjectWriter( JsonObjectWriter&& source ) noexcept:
|
|
m_os{ source.m_os },
|
|
m_indent_level{ source.m_indent_level },
|
|
m_should_comma{ source.m_should_comma },
|
|
m_active{ source.m_active } {
|
|
source.m_active = false;
|
|
}
|
|
|
|
JsonObjectWriter::~JsonObjectWriter() {
|
|
if ( !m_active ) { return; }
|
|
|
|
m_os << '\n';
|
|
JsonUtils::indent( m_os, m_indent_level );
|
|
m_os << '}';
|
|
}
|
|
|
|
JsonValueWriter JsonObjectWriter::write( StringRef key ) {
|
|
JsonUtils::appendCommaNewline(
|
|
m_os, m_should_comma, m_indent_level + 1 );
|
|
|
|
m_os << '"' << key << "\": ";
|
|
return JsonValueWriter{ m_os, m_indent_level + 1 };
|
|
}
|
|
|
|
JsonArrayWriter::JsonArrayWriter( std::ostream& os ):
|
|
JsonArrayWriter{ os, 0 } {}
|
|
JsonArrayWriter::JsonArrayWriter( std::ostream& os,
|
|
std::uint64_t indent_level ):
|
|
m_os{ os }, m_indent_level{ indent_level } {
|
|
m_os << '[';
|
|
}
|
|
JsonArrayWriter::JsonArrayWriter( JsonArrayWriter&& source ) noexcept:
|
|
m_os{ source.m_os },
|
|
m_indent_level{ source.m_indent_level },
|
|
m_should_comma{ source.m_should_comma },
|
|
m_active{ source.m_active } {
|
|
source.m_active = false;
|
|
}
|
|
JsonArrayWriter::~JsonArrayWriter() {
|
|
if ( !m_active ) { return; }
|
|
|
|
m_os << '\n';
|
|
JsonUtils::indent( m_os, m_indent_level );
|
|
m_os << ']';
|
|
}
|
|
|
|
JsonObjectWriter JsonArrayWriter::writeObject() {
|
|
JsonUtils::appendCommaNewline(
|
|
m_os, m_should_comma, m_indent_level + 1 );
|
|
return JsonObjectWriter{ m_os, m_indent_level + 1 };
|
|
}
|
|
|
|
JsonArrayWriter JsonArrayWriter::writeArray() {
|
|
JsonUtils::appendCommaNewline(
|
|
m_os, m_should_comma, m_indent_level + 1 );
|
|
return JsonArrayWriter{ m_os, m_indent_level + 1 };
|
|
}
|
|
|
|
JsonArrayWriter& JsonArrayWriter::write( bool value ) {
|
|
return writeImpl( value );
|
|
}
|
|
|
|
JsonValueWriter::JsonValueWriter( std::ostream& os ):
|
|
JsonValueWriter{ os, 0 } {}
|
|
|
|
JsonValueWriter::JsonValueWriter( std::ostream& os,
|
|
std::uint64_t indent_level ):
|
|
m_os{ os }, m_indent_level{ indent_level } {}
|
|
|
|
JsonObjectWriter JsonValueWriter::writeObject() && {
|
|
return JsonObjectWriter{ m_os, m_indent_level };
|
|
}
|
|
|
|
JsonArrayWriter JsonValueWriter::writeArray() && {
|
|
return JsonArrayWriter{ m_os, m_indent_level };
|
|
}
|
|
|
|
void JsonValueWriter::write( Catch::StringRef value ) && {
|
|
writeImpl( value, true );
|
|
}
|
|
|
|
void JsonValueWriter::write( bool value ) && {
|
|
writeImpl( value ? "true"_sr : "false"_sr, false );
|
|
}
|
|
|
|
void JsonValueWriter::writeImpl( Catch::StringRef value, bool quote ) {
|
|
if ( quote ) { m_os << '"'; }
|
|
size_t current_start = 0;
|
|
for ( size_t i = 0; i < value.size(); ++i ) {
|
|
if ( needsEscape( value[i] ) ) {
|
|
if ( current_start < i ) {
|
|
m_os << value.substr( current_start, i - current_start );
|
|
}
|
|
m_os << makeEscapeStringRef( value[i] );
|
|
current_start = i + 1;
|
|
}
|
|
}
|
|
if ( current_start < value.size() ) {
|
|
m_os << value.substr( current_start, value.size() - current_start );
|
|
}
|
|
if ( quote ) { m_os << '"'; }
|
|
}
|
|
|
|
} // namespace Catch
|