mirror of
https://github.com/ethereum/solidity
synced 2023-10-03 13:03:40 +00:00
Generalize existing functions for indenting and prefixing
This commit is contained in:
parent
c6b04df78c
commit
996e9cacd0
@ -23,6 +23,10 @@
|
||||
*/
|
||||
|
||||
#include <libsolutil/StringUtils.h>
|
||||
|
||||
#include <boost/algorithm/string/trim.hpp>
|
||||
|
||||
#include <sstream>
|
||||
#include <string>
|
||||
#include <vector>
|
||||
|
||||
@ -190,3 +194,34 @@ std::string solidity::util::formatNumberReadable(bigint const& _value, bool _use
|
||||
return sign + str;
|
||||
}
|
||||
|
||||
std::string solidity::util::prefixLines(
|
||||
std::string const& _input,
|
||||
std::string const& _prefix,
|
||||
bool _trimPrefix
|
||||
)
|
||||
{
|
||||
std::ostringstream output;
|
||||
printPrefixed(output, _input, _prefix, _trimPrefix, false /* _ensureFinalNewline */);
|
||||
return output.str();
|
||||
}
|
||||
|
||||
void solidity::util::printPrefixed(
|
||||
std::ostream& _output,
|
||||
std::string const& _input,
|
||||
std::string const& _prefix,
|
||||
bool _trimPrefix,
|
||||
bool _ensureFinalNewline
|
||||
)
|
||||
{
|
||||
std::istringstream input(_input);
|
||||
std::string line;
|
||||
while (std::getline(input, line))
|
||||
{
|
||||
if (line.empty() && _trimPrefix)
|
||||
_output << boost::trim_right_copy(_prefix);
|
||||
else
|
||||
_output << _prefix << line;
|
||||
if (!input.eof() || _ensureFinalNewline)
|
||||
_output << '\n';
|
||||
}
|
||||
}
|
||||
|
@ -168,7 +168,7 @@ inline bool isDigit(char _c)
|
||||
return isdigit(_c, std::locale::classic());
|
||||
}
|
||||
|
||||
// Checks if character is printable using classic "C" locale
|
||||
/// Checks if character is printable using classic "C" locale
|
||||
/// @param _c character to be checked
|
||||
/// @return true if _c is a printable character, false otherwise.
|
||||
inline bool isPrint(char _c)
|
||||
@ -176,4 +176,36 @@ inline bool isPrint(char _c)
|
||||
return isprint(_c, std::locale::classic());
|
||||
}
|
||||
|
||||
/// Adds a prefix to every line in the input.
|
||||
/// @see printPrefixed()
|
||||
std::string prefixLines(
|
||||
std::string const& _input,
|
||||
std::string const& _prefix,
|
||||
bool _trimPrefix = true
|
||||
);
|
||||
|
||||
/// Prints to a stream, adding a prefix to every line in the input.
|
||||
/// Assumes \n as the line separator.
|
||||
/// @param _trimPrefix If true, the function avoids introducing trailing whitespace on empty lines.
|
||||
/// This is achieved by removing trailing spaces from the prefix on such lines.
|
||||
/// Note that tabs and newlines are not removed, only spaces are.
|
||||
/// @param _finalNewline If true, an extra \n will be printed at the end of @a _input if it does
|
||||
/// not already end with one.
|
||||
void printPrefixed(
|
||||
std::ostream& _output,
|
||||
std::string const& _input,
|
||||
std::string const& _prefix,
|
||||
bool _trimPrefix = true,
|
||||
bool _ensureFinalNewline = true
|
||||
);
|
||||
|
||||
/// Adds a standard indent of 4 spaces to every line in the input.
|
||||
/// Assumes \n as the line separator.
|
||||
/// @param _indentEmptyLines If true, the indent will be applied to empty lines as well, resulting
|
||||
/// such lines containing trailing whitespace.
|
||||
inline std::string indent(std::string const& _input, bool _indentEmptyLines = false)
|
||||
{
|
||||
return prefixLines(_input, " ", !_indentEmptyLines);
|
||||
}
|
||||
|
||||
}
|
||||
|
@ -30,8 +30,6 @@
|
||||
#include <libsolutil/StringUtils.h>
|
||||
|
||||
#include <boost/algorithm/string.hpp>
|
||||
#include <boost/algorithm/string/split.hpp>
|
||||
#include <boost/algorithm/string/replace.hpp>
|
||||
|
||||
#include <range/v3/view/transform.hpp>
|
||||
|
||||
@ -40,18 +38,6 @@ using namespace solidity::langutil;
|
||||
using namespace solidity::util;
|
||||
using namespace solidity::yul;
|
||||
|
||||
namespace
|
||||
{
|
||||
|
||||
std::string indent(std::string const& _input)
|
||||
{
|
||||
if (_input.empty())
|
||||
return _input;
|
||||
return boost::replace_all_copy(" " + _input, "\n", "\n ");
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
std::string Data::toString(Dialect const*, DebugInfoSelection const&, CharStreamProvider const*) const
|
||||
{
|
||||
return "data \"" + name.str() + "\" hex\"" + util::toHex(data) + "\"";
|
||||
@ -86,7 +72,7 @@ std::string Object::toString(
|
||||
for (auto const& obj: subObjects)
|
||||
inner += "\n" + obj->toString(_dialect, _debugInfoSelection, _soliditySourceProvider);
|
||||
|
||||
return useSrcComment + "object \"" + name.str() + "\" {\n" + indent(inner) + "\n}";
|
||||
return useSrcComment + "object \"" + name.str() + "\" {\n" + indent(inner, true /* _indentEmptyLines */) + "\n}";
|
||||
}
|
||||
|
||||
Json::Value Data::toJson() const
|
||||
|
@ -20,6 +20,7 @@
|
||||
#include <test/Common.h>
|
||||
#include <test/TestCase.h>
|
||||
#include <libsolutil/CommonIO.h>
|
||||
#include <libsolutil/StringUtils.h>
|
||||
#include <boost/algorithm/string.hpp>
|
||||
#include <boost/algorithm/string/predicate.hpp>
|
||||
#include <boost/test/unit_test.hpp>
|
||||
@ -144,11 +145,8 @@ void CommonSyntaxTest::printSource(ostream& _stream, string const& _linePrefix,
|
||||
else
|
||||
{
|
||||
if (outputSourceNames)
|
||||
_stream << _linePrefix << "==== Source: " + name << " ====" << endl;
|
||||
stringstream stream(source);
|
||||
string line;
|
||||
while (getline(stream, line))
|
||||
_stream << _linePrefix << line << endl;
|
||||
printPrefixed(_stream, "==== Source: " + name + " ====", _linePrefix);
|
||||
printPrefixed(_stream, source, _linePrefix);
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -20,8 +20,8 @@
|
||||
#include <test/TestCase.h>
|
||||
|
||||
#include <libsolutil/AnsiColorized.h>
|
||||
#include <libsolutil/StringUtils.h>
|
||||
|
||||
#include <boost/algorithm/string.hpp>
|
||||
#include <boost/algorithm/string/predicate.hpp>
|
||||
|
||||
#include <iostream>
|
||||
@ -31,6 +31,7 @@ using namespace std;
|
||||
using namespace solidity;
|
||||
using namespace solidity::frontend;
|
||||
using namespace solidity::frontend::test;
|
||||
using namespace solidity::util;
|
||||
|
||||
void TestCase::printSettings(ostream& _stream, const string& _linePrefix, const bool)
|
||||
{
|
||||
@ -69,26 +70,14 @@ void TestCase::expect(string::iterator& _it, string::iterator _end, string::valu
|
||||
++_it;
|
||||
}
|
||||
|
||||
void TestCase::printIndented(ostream& _stream, string const& _output, string const& _linePrefix) const
|
||||
{
|
||||
stringstream output(_output);
|
||||
string line;
|
||||
while (getline(output, line))
|
||||
if (line.empty())
|
||||
// Avoid trailing spaces.
|
||||
_stream << boost::trim_right_copy(_linePrefix) << endl;
|
||||
else
|
||||
_stream << _linePrefix << line << endl;
|
||||
}
|
||||
|
||||
void TestCase::printSource(ostream& _stream, string const& _linePrefix, bool const) const
|
||||
{
|
||||
printIndented(_stream, m_source, _linePrefix);
|
||||
printPrefixed(_stream, m_source, _linePrefix);
|
||||
}
|
||||
|
||||
void TestCase::printUpdatedExpectations(ostream& _stream, string const& _linePrefix) const
|
||||
{
|
||||
printIndented(_stream, m_obtainedResult, _linePrefix);
|
||||
printPrefixed(_stream, m_obtainedResult, _linePrefix);
|
||||
}
|
||||
|
||||
TestCase::TestResult TestCase::checkResult(std::ostream& _stream, const std::string& _linePrefix, bool const _formatted)
|
||||
@ -99,10 +88,10 @@ TestCase::TestResult TestCase::checkResult(std::ostream& _stream, const std::str
|
||||
util::AnsiColorized(_stream, _formatted, {util::formatting::BOLD, util::formatting::CYAN})
|
||||
<< _linePrefix << "Expected result:" << endl;
|
||||
// TODO could compute a simple diff with highlighted lines
|
||||
printIndented(_stream, m_expectation, nextIndentLevel);
|
||||
printPrefixed(_stream, m_expectation, nextIndentLevel);
|
||||
util::AnsiColorized(_stream, _formatted, {util::formatting::BOLD, util::formatting::CYAN})
|
||||
<< _linePrefix << "Obtained result:" << endl;
|
||||
printIndented(_stream, m_obtainedResult, nextIndentLevel);
|
||||
printPrefixed(_stream, m_obtainedResult, nextIndentLevel);
|
||||
return TestResult::Failure;
|
||||
}
|
||||
return TestResult::Success;
|
||||
|
@ -100,7 +100,6 @@ protected:
|
||||
++_it;
|
||||
}
|
||||
|
||||
void printIndented(std::ostream& _stream, std::string const& _output, std::string const& _linePrefix = "") const;
|
||||
TestCase::TestResult checkResult(std::ostream& _stream, const std::string& _linePrefix, bool const _formatted);
|
||||
|
||||
std::string m_source;
|
||||
|
@ -270,12 +270,7 @@ bool ASTJSONTest::runTest(
|
||||
"Expected result" <<
|
||||
(!_variant.name().empty() ? " (" + _variant.name() + "):" : ":") <<
|
||||
std::endl;
|
||||
{
|
||||
std::istringstream stream(_variant.expectation);
|
||||
std::string line;
|
||||
while (getline(stream, line))
|
||||
_stream << nextIndentLevel << line << std::endl;
|
||||
}
|
||||
printPrefixed(_stream, _variant.expectation, nextIndentLevel);
|
||||
_stream << std::endl;
|
||||
|
||||
AnsiColorized(_stream, _formatted, {BOLD, CYAN}) <<
|
||||
@ -283,12 +278,7 @@ bool ASTJSONTest::runTest(
|
||||
"Obtained result" <<
|
||||
(!_variant.name().empty() ? " (" + _variant.name() + "):" : ":") <<
|
||||
std::endl;
|
||||
{
|
||||
std::istringstream stream(_variant.result);
|
||||
std::string line;
|
||||
while (getline(stream, line))
|
||||
_stream << nextIndentLevel << line << std::endl;
|
||||
}
|
||||
printPrefixed(_stream, _variant.result, nextIndentLevel);
|
||||
_stream << std::endl;
|
||||
return false;
|
||||
}
|
||||
@ -301,11 +291,8 @@ void ASTJSONTest::printSource(std::ostream& _stream, std::string const& _linePre
|
||||
for (auto const& source: m_sources)
|
||||
{
|
||||
if (m_sources.size() > 1 || source.first != "a")
|
||||
_stream << _linePrefix << sourceDelimiter << source.first << " ====" << std::endl << std::endl;
|
||||
std::stringstream stream(source.second);
|
||||
std::string line;
|
||||
while (getline(stream, line))
|
||||
_stream << _linePrefix << line << std::endl;
|
||||
printPrefixed(_stream, sourceDelimiter + source.first + " ====\n", _linePrefix);
|
||||
printPrefixed(_stream, source.second, _linePrefix);
|
||||
_stream << std::endl;
|
||||
}
|
||||
}
|
||||
|
@ -158,11 +158,3 @@ TestCase::TestResult GasTest::run(std::ostream& _stream, std::string const& _lin
|
||||
return TestResult::Failure;
|
||||
}
|
||||
}
|
||||
|
||||
void GasTest::printSource(std::ostream& _stream, std::string const& _linePrefix, bool) const
|
||||
{
|
||||
std::string line;
|
||||
std::istringstream input(m_source);
|
||||
while (getline(input, line))
|
||||
_stream << _linePrefix << line << std::endl;
|
||||
}
|
||||
|
@ -41,7 +41,6 @@ public:
|
||||
|
||||
TestResult run(std::ostream& _stream, std::string const& _linePrefix = "", bool _formatted = false) override;
|
||||
|
||||
void printSource(std::ostream &_stream, std::string const &_linePrefix = "", bool _formatted = false) const override;
|
||||
void printUpdatedExpectations(std::ostream& _stream, std::string const& _linePrefix) const override;
|
||||
|
||||
protected:
|
||||
|
@ -100,8 +100,9 @@ void NatspecJSONTest::printExpectedResult(ostream& _stream, string const& _lineP
|
||||
if (!m_expectedNatspecJSON.empty())
|
||||
{
|
||||
_stream << _linePrefix << "----" << endl;
|
||||
printIndented(_stream, formatNatspecExpectations(m_expectedNatspecJSON), _linePrefix);
|
||||
printPrefixed(_stream, formatNatspecExpectations(m_expectedNatspecJSON), _linePrefix);
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
void NatspecJSONTest::printObtainedResult(ostream& _stream, string const& _linePrefix, bool _formatted) const
|
||||
@ -114,7 +115,7 @@ void NatspecJSONTest::printObtainedResult(ostream& _stream, string const& _lineP
|
||||
_stream << _linePrefix << "----" << endl;
|
||||
// TODO: Diff both versions and highlight differences.
|
||||
// We should have a helper for doing that in newly defined test cases without much effort.
|
||||
printIndented(_stream, formatNatspecExpectations(natspecJSON), _linePrefix);
|
||||
printPrefixed(_stream, formatNatspecExpectations(natspecJSON), _linePrefix);
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -647,11 +647,8 @@ void SemanticTest::printSource(std::ostream& _stream, std::string const& _linePr
|
||||
else
|
||||
{
|
||||
if (outputNames)
|
||||
_stream << _linePrefix << "==== Source: " + name << " ====" << std::endl;
|
||||
std::stringstream stream(source);
|
||||
std::string line;
|
||||
while (getline(stream, line))
|
||||
_stream << _linePrefix << line << std::endl;
|
||||
printPrefixed(_stream, "==== Source: " + name + " ====", _linePrefix);
|
||||
printPrefixed(_stream, source, _linePrefix);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -31,6 +31,7 @@
|
||||
#include <liblangutil/Scanner.h>
|
||||
|
||||
#include <libsolutil/AnsiColorized.h>
|
||||
#include <libsolutil/StringUtils.h>
|
||||
|
||||
#include <fstream>
|
||||
|
||||
@ -86,7 +87,7 @@ TestCase::TestResult YulOptimizerTest::run(ostream& _stream, string const& _line
|
||||
{
|
||||
util::AnsiColorized(_stream, _formatted, {util::formatting::BOLD, util::formatting::CYAN})
|
||||
<< _linePrefix << "Result after the optimiser:" << endl;
|
||||
printIndented(_stream, printed, _linePrefix + " ");
|
||||
printPrefixed(_stream, printed, _linePrefix + " ");
|
||||
return TestResult::FatalError;
|
||||
}
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user