SyntaxTests: extend syntax tests and isoltest to support parser errors and compiler exceptions.

This commit is contained in:
Daniel Kirchner 2018-04-03 12:05:26 +02:00
parent 104a9736b3
commit 6f9644add1
5 changed files with 104 additions and 125 deletions

View File

@ -56,12 +56,23 @@ AnalysisFramework::parseAnalyseAndReturnError(
m_compiler.analyze(); m_compiler.analyze();
ErrorList errors = filterErrors(m_compiler.errors(), _reportWarnings);
if (errors.size() > 1 && !_allowMultipleErrors)
BOOST_FAIL("Multiple errors found: " + formatErrors());
return make_pair(&m_compiler.ast(""), std::move(errors));
}
ErrorList AnalysisFramework::filterErrors(ErrorList const& _errorList, bool _includeWarnings) const
{
ErrorList errors; ErrorList errors;
for (auto const& currentError: m_compiler.errors()) for (auto const& currentError: _errorList)
{ {
solAssert(currentError->comment(), ""); solAssert(currentError->comment(), "");
if (currentError->type() == Error::Type::Warning) if (currentError->type() == Error::Type::Warning)
{ {
if (!_includeWarnings)
continue;
bool ignoreWarning = false; bool ignoreWarning = false;
for (auto const& filter: m_warningsToFilter) for (auto const& filter: m_warningsToFilter)
if (currentError->comment()->find(filter) == 0) if (currentError->comment()->find(filter) == 0)
@ -73,17 +84,10 @@ AnalysisFramework::parseAnalyseAndReturnError(
continue; continue;
} }
if (_reportWarnings || (currentError->type() != Error::Type::Warning)) errors.emplace_back(currentError);
{
if (!_allowMultipleErrors && !errors.empty())
{
BOOST_FAIL("Multiple errors found: " + formatErrors());
}
errors.emplace_back(std::move(currentError));
}
} }
return make_pair(&m_compiler.ast(""), errors); return errors;
} }
SourceUnit const* AnalysisFramework::parseAndAnalyse(string const& _source) SourceUnit const* AnalysisFramework::parseAndAnalyse(string const& _source)
@ -110,7 +114,7 @@ ErrorList AnalysisFramework::expectError(std::string const& _source, bool _warni
return sourceAndErrors.second; return sourceAndErrors.second;
} }
string AnalysisFramework::formatErrors() string AnalysisFramework::formatErrors() const
{ {
string message; string message;
for (auto const& error: m_compiler.errors()) for (auto const& error: m_compiler.errors())
@ -118,7 +122,7 @@ string AnalysisFramework::formatErrors()
return message; return message;
} }
string AnalysisFramework::formatError(Error const& _error) string AnalysisFramework::formatError(Error const& _error) const
{ {
return SourceReferenceFormatter::formatExceptionInformation( return SourceReferenceFormatter::formatExceptionInformation(
_error, _error,

View File

@ -57,8 +57,8 @@ protected:
bool success(std::string const& _source); bool success(std::string const& _source);
ErrorList expectError(std::string const& _source, bool _warning = false, bool _allowMultiple = false); ErrorList expectError(std::string const& _source, bool _warning = false, bool _allowMultiple = false);
std::string formatErrors(); std::string formatErrors() const;
std::string formatError(Error const& _error); std::string formatError(Error const& _error) const;
static ContractDefinition const* retrieveContractByName(SourceUnit const& _source, std::string const& _name); static ContractDefinition const* retrieveContractByName(SourceUnit const& _source, std::string const& _name);
static FunctionTypePointer retrieveFunctionBySignature( static FunctionTypePointer retrieveFunctionBySignature(
@ -66,6 +66,9 @@ protected:
std::string const& _signature std::string const& _signature
); );
// filter out the warnings in m_warningsToFilter or all warnings if _includeWarnings is false
ErrorList filterErrors(ErrorList const& _errorList, bool _includeWarnings) const;
std::vector<std::string> m_warningsToFilter = {"This is a pre-release compiler version"}; std::vector<std::string> m_warningsToFilter = {"This is a pre-release compiler version"};
dev::solidity::CompilerStack m_compiler; dev::solidity::CompilerStack m_compiler;
}; };

View File

@ -16,6 +16,7 @@
*/ */
#include <test/libsolidity/SyntaxTest.h> #include <test/libsolidity/SyntaxTest.h>
#include <test/Options.h>
#include <boost/algorithm/string.hpp> #include <boost/algorithm/string.hpp>
#include <boost/algorithm/string/predicate.hpp> #include <boost/algorithm/string/predicate.hpp>
#include <boost/throw_exception.hpp> #include <boost/throw_exception.hpp>
@ -59,93 +60,52 @@ SyntaxTest::SyntaxTest(string const& _filename)
bool SyntaxTest::run(ostream& _stream, string const& _linePrefix, bool const _formatted) bool SyntaxTest::run(ostream& _stream, string const& _linePrefix, bool const _formatted)
{ {
m_errorList = parseAnalyseAndReturnError(m_source, true, true, true).second; m_compiler.reset();
if (!matchesExpectations(m_errorList)) m_compiler.addSource("", "pragma solidity >=0.0;\n" + m_source);
m_compiler.setEVMVersion(dev::test::Options::get().evmVersion());
if (m_compiler.parse())
m_compiler.analyze();
for (auto const& currentError: filterErrors(m_compiler.errors(), true))
m_errorList.emplace_back(SyntaxTestError{currentError->typeName(), errorMessage(*currentError)});
if (m_expectations != m_errorList)
{ {
std::string nextIndentLevel = _linePrefix + " "; string nextIndentLevel = _linePrefix + " ";
FormattedScope(_stream, _formatted, {BOLD, CYAN}) << _linePrefix << "Expected result:" << endl; FormattedScope(_stream, _formatted, {BOLD, CYAN}) << _linePrefix << "Expected result:" << endl;
printExpected(_stream, nextIndentLevel, _formatted); printErrorList(_stream, m_expectations, nextIndentLevel, _formatted);
FormattedScope(_stream, _formatted, {BOLD, CYAN}) << _linePrefix << "Obtained result:\n"; FormattedScope(_stream, _formatted, {BOLD, CYAN}) << _linePrefix << "Obtained result:\n";
printErrorList(_stream, m_errorList, nextIndentLevel, false, false, _formatted); printErrorList(_stream, m_errorList, nextIndentLevel, _formatted);
return false; return false;
} }
return true; return true;
} }
void SyntaxTest::printExpected(ostream& _stream, string const& _linePrefix, bool const _formatted) const
{
if (m_expectations.empty())
FormattedScope(_stream, _formatted, {BOLD, GREEN}) << _linePrefix << "Success" << endl;
else
for (auto const& expectation: m_expectations)
{
FormattedScope(_stream, _formatted, {BOLD, expectation.type == "Warning" ? YELLOW : RED}) <<
_linePrefix << expectation.type << ": ";
_stream << expectation.message << endl;
}
}
void SyntaxTest::printErrorList( void SyntaxTest::printErrorList(
ostream& _stream, ostream& _stream,
ErrorList const& _errorList, vector<SyntaxTestError> const& _errorList,
string const& _linePrefix, string const& _linePrefix,
bool const _ignoreWarnings,
bool const _lineNumbers,
bool const _formatted bool const _formatted
) const )
{ {
if (_errorList.empty()) if (_errorList.empty())
FormattedScope(_stream, _formatted, {BOLD, GREEN}) << _linePrefix << "Success" << endl; FormattedScope(_stream, _formatted, {BOLD, GREEN}) << _linePrefix << "Success" << endl;
else else
for (auto const& error: _errorList) for (auto const& error: _errorList)
{ {
bool isWarning = (error->type() == Error::Type::Warning);
if (isWarning && _ignoreWarnings) continue;
{ {
FormattedScope scope(_stream, _formatted, {BOLD, isWarning ? YELLOW : RED}); FormattedScope scope(_stream, _formatted, {BOLD, (error.type == "Warning") ? YELLOW : RED});
_stream << _linePrefix; _stream << _linePrefix;
if (_lineNumbers) _stream << error.type << ": ";
{
int line = offsetToLineNumber(
boost::get_error_info<errinfo_sourceLocation>(*error)->start
);
if (line >= 0)
_stream << "(" << line << "): ";
}
_stream << error->typeName() << ": ";
} }
_stream << errorMessage(*error) << endl; _stream << error.message << endl;
} }
} }
int SyntaxTest::offsetToLineNumber(int _location) const string SyntaxTest::errorMessage(Exception const& _e)
{ {
// parseAnalyseAndReturnError(...) prepends a version pragma if (_e.comment() && !_e.comment()->empty())
_location -= strlen("pragma solidity >=0.0;\n");
if (_location < 0 || static_cast<size_t>(_location) >= m_source.size())
return -1;
else
return 1 + std::count(m_source.begin(), m_source.begin() + _location, '\n');
}
bool SyntaxTest::matchesExpectations(ErrorList const& _errorList) const
{
if (_errorList.size() != m_expectations.size())
return false;
else
for (size_t i = 0; i < _errorList.size(); i++)
if (
(_errorList[i]->typeName() != m_expectations[i].type) ||
(errorMessage(*_errorList[i]) != m_expectations[i].message)
)
return false;
return true;
}
string SyntaxTest::errorMessage(Error const& _e)
{
if (_e.comment())
return boost::replace_all_copy(*_e.comment(), "\n", "\\n"); return boost::replace_all_copy(*_e.comment(), "\n", "\\n");
else else
return "NONE"; return "NONE";
@ -164,9 +124,9 @@ string SyntaxTest::parseSource(istream& _stream)
return source; return source;
} }
vector<SyntaxTestExpectation> SyntaxTest::parseExpectations(istream& _stream) vector<SyntaxTestError> SyntaxTest::parseExpectations(istream& _stream)
{ {
vector<SyntaxTestExpectation> expectations; vector<SyntaxTestError> expectations;
string line; string line;
while (getline(_stream, line)) while (getline(_stream, line))
{ {
@ -188,7 +148,7 @@ vector<SyntaxTestExpectation> SyntaxTest::parseExpectations(istream& _stream)
skipWhitespace(it, line.end()); skipWhitespace(it, line.end());
string errorMessage(it, line.end()); string errorMessage(it, line.end());
expectations.emplace_back(SyntaxTestExpectation{move(errorType), move(errorMessage)}); expectations.emplace_back(SyntaxTestError{move(errorType), move(errorMessage)});
} }
return expectations; return expectations;
} }
@ -239,9 +199,11 @@ int SyntaxTest::registerTests(
_suite.add(make_test_case( _suite.add(make_test_case(
[fullpath] [fullpath]
{ {
std::stringstream errorStream; BOOST_REQUIRE_NO_THROW({
if (!SyntaxTest(fullpath.string()).run(errorStream)) stringstream errorStream;
BOOST_ERROR("Test expectation mismatch.\n" + errorStream.str()); if (!SyntaxTest(fullpath.string()).run(errorStream))
BOOST_ERROR("Test expectation mismatch.\n" + errorStream.str());
});
}, },
_path.stem().string(), _path.stem().string(),
*filenames.back(), *filenames.back(),

View File

@ -36,10 +36,14 @@ namespace solidity
namespace test namespace test
{ {
struct SyntaxTestExpectation struct SyntaxTestError
{ {
std::string type; std::string type;
std::string message; std::string message;
bool operator==(SyntaxTestError const& _rhs) const
{
return type == _rhs.type && message == _rhs.message;
}
}; };
@ -50,21 +54,16 @@ public:
bool run(std::ostream& _stream, std::string const& _linePrefix = "", bool const _formatted = false); bool run(std::ostream& _stream, std::string const& _linePrefix = "", bool const _formatted = false);
std::vector<SyntaxTestExpectation> const& expectations() const { return m_expectations; } std::vector<SyntaxTestError> const& expectations() const { return m_expectations; }
std::string const& source() const { return m_source; } std::string const& source() const { return m_source; }
ErrorList const& errorList() const { return m_errorList; } std::vector<SyntaxTestError> const& errorList() const { return m_errorList; }
ErrorList const& compilerErrors() const { return m_compiler.errors(); }
void printExpected(std::ostream& _stream, std::string const& _linePrefix, bool const _formatted = false) const; static void printErrorList(
void printErrorList(
std::ostream& _stream, std::ostream& _stream,
ErrorList const& _errors, std::vector<SyntaxTestError> const& _errors,
std::string const& _linePrefix, std::string const& _linePrefix,
bool const _ignoreWarnings,
bool const _lineNumbers,
bool const _formatted = false bool const _formatted = false
) const; );
static int registerTests( static int registerTests(
boost::unit_test::test_suite& _suite, boost::unit_test::test_suite& _suite,
@ -72,16 +71,14 @@ public:
boost::filesystem::path const& _path boost::filesystem::path const& _path
); );
static bool isTestFilename(boost::filesystem::path const& _filename); static bool isTestFilename(boost::filesystem::path const& _filename);
static std::string errorMessage(Exception const& _e);
private: private:
bool matchesExpectations(ErrorList const& _errors) const;
static std::string errorMessage(Error const& _e);
static std::string parseSource(std::istream& _stream); static std::string parseSource(std::istream& _stream);
static std::vector<SyntaxTestExpectation> parseExpectations(std::istream& _stream); static std::vector<SyntaxTestError> parseExpectations(std::istream& _stream);
int offsetToLineNumber(int _location) const;
std::string m_source; std::string m_source;
std::vector<SyntaxTestExpectation> m_expectations; std::vector<SyntaxTestError> m_expectations;
ErrorList m_errorList; std::vector<SyntaxTestError> m_errorList;
}; };
} }

View File

@ -55,8 +55,8 @@ public:
{ {
Success, Success,
Failure, Failure,
ParserError, InputOutputError,
InputOutputError Exception
}; };
Result process(); Result process();
@ -76,7 +76,7 @@ private:
Quit Quit
}; };
Request handleResponse(bool const _parserError); Request handleResponse(bool const _exception);
void printContract() const; void printContract() const;
@ -100,7 +100,6 @@ void SyntaxTestTool::printContract() const
SyntaxTestTool::Result SyntaxTestTool::process() SyntaxTestTool::Result SyntaxTestTool::process()
{ {
bool success; bool success;
bool parserError = false;
std::stringstream outputMessages; std::stringstream outputMessages;
(FormattedScope(cout, m_formatted, {BOLD}) << m_name << ": ").flush(); (FormattedScope(cout, m_formatted, {BOLD}) << m_name << ": ").flush();
@ -119,10 +118,35 @@ SyntaxTestTool::Result SyntaxTestTool::process()
{ {
success = m_test->run(outputMessages, " ", m_formatted); success = m_test->run(outputMessages, " ", m_formatted);
} }
catch (...) catch(CompilerError const& _e)
{ {
success = false; FormattedScope(cout, m_formatted, {BOLD, RED}) <<
parserError = true; "Exception: " << SyntaxTest::errorMessage(_e) << endl;
return Result::Exception;
}
catch(InternalCompilerError const& _e)
{
FormattedScope(cout, m_formatted, {BOLD, RED}) <<
"InternalCompilerError: " << SyntaxTest::errorMessage(_e) << endl;
return Result::Exception;
}
catch(FatalError const& _e)
{
FormattedScope(cout, m_formatted, {BOLD, RED}) <<
"FatalError: " << SyntaxTest::errorMessage(_e) << endl;
return Result::Exception;
}
catch(UnimplementedFeatureError const& _e)
{
FormattedScope(cout, m_formatted, {BOLD, RED}) <<
"UnimplementedFeatureError: " << SyntaxTest::errorMessage(_e) << endl;
return Result::Exception;
}
catch(...)
{
FormattedScope(cout, m_formatted, {BOLD, RED}) <<
"Unknown Exception" << endl;
return Result::Exception;
} }
if (success) if (success)
@ -137,25 +161,14 @@ SyntaxTestTool::Result SyntaxTestTool::process()
FormattedScope(cout, m_formatted, {BOLD, CYAN}) << " Contract:" << endl; FormattedScope(cout, m_formatted, {BOLD, CYAN}) << " Contract:" << endl;
printContract(); printContract();
if (parserError) cout << outputMessages.str() << endl;
{ return Result::Failure;
cout << " ";
FormattedScope(cout, m_formatted, {INVERSE, RED}) << "Parsing failed:" << endl;
m_test->printErrorList(cout, m_test->compilerErrors(), " ", true, true, m_formatted);
cout << endl;
return Result::ParserError;
}
else
{
cout << outputMessages.str() << endl;
return Result::Failure;
}
} }
} }
SyntaxTestTool::Request SyntaxTestTool::handleResponse(bool const _parserError) SyntaxTestTool::Request SyntaxTestTool::handleResponse(bool const _exception)
{ {
if (_parserError) if (_exception)
cout << "(e)dit/(s)kip/(q)uit? "; cout << "(e)dit/(s)kip/(q)uit? ";
else else
cout << "(e)dit/(u)pdate expectations/(s)kip/(q)uit? "; cout << "(e)dit/(u)pdate expectations/(s)kip/(q)uit? ";
@ -169,7 +182,7 @@ SyntaxTestTool::Request SyntaxTestTool::handleResponse(bool const _parserError)
cout << endl; cout << endl;
return Request::Skip; return Request::Skip;
case 'u': case 'u':
if (_parserError) if (_exception)
break; break;
else else
{ {
@ -178,7 +191,7 @@ SyntaxTestTool::Request SyntaxTestTool::handleResponse(bool const _parserError)
file << m_test->source(); file << m_test->source();
file << "// ----" << endl; file << "// ----" << endl;
if (!m_test->errorList().empty()) if (!m_test->errorList().empty())
m_test->printErrorList(file, m_test->errorList(), "// ", false, false, false); m_test->printErrorList(file, m_test->errorList(), "// ", false);
return Request::Rerun; return Request::Rerun;
} }
case 'e': case 'e':
@ -231,8 +244,8 @@ SyntaxTestStats SyntaxTestTool::processPath(
switch(result) switch(result)
{ {
case Result::Failure: case Result::Failure:
case Result::ParserError: case Result::Exception:
switch(testTool.handleResponse(result == Result::ParserError)) switch(testTool.handleResponse(result == Result::Exception))
{ {
case Request::Quit: case Request::Quit:
return { successCount, runCount }; return { successCount, runCount };