Added new kind of test: check for specific properties of AST

This commit is contained in:
Matheus Aguiar 2023-05-29 05:58:48 -03:00
parent ee21b03e6c
commit 969aea6d33
13 changed files with 371 additions and 1 deletions

View File

@ -144,4 +144,19 @@ bool jsonParseStrict(std::string const& _input, Json::Value& _json, std::string*
return parse(readerBuilder, _input, _json, _errs); return parse(readerBuilder, _input, _json, _errs);
} }
std::optional<Json::Value> jsonValueByPath(Json::Value const& _node, std::string_view _jsonPath)
{
if (!_node.isObject() || _jsonPath.empty())
return {};
std::string memberName = std::string(_jsonPath.substr(0, _jsonPath.find_first_of('.')));
if (!_node.isMember(memberName))
return {};
if (memberName == _jsonPath)
return _node[memberName];
return jsonValueByPath(_node[memberName], _jsonPath.substr(memberName.size() + 1));
}
} // namespace solidity::util } // namespace solidity::util

View File

@ -26,6 +26,8 @@
#include <json/json.h> #include <json/json.h>
#include <string> #include <string>
#include <string_view>
#include <optional>
namespace solidity::util namespace solidity::util
{ {
@ -67,6 +69,13 @@ std::string jsonPrint(Json::Value const& _input, JsonFormat const& _format);
/// \return \c true if the document was successfully parsed, \c false if an error occurred. /// \return \c true if the document was successfully parsed, \c false if an error occurred.
bool jsonParseStrict(std::string const& _input, Json::Value& _json, std::string* _errs = nullptr); bool jsonParseStrict(std::string const& _input, Json::Value& _json, std::string* _errs = nullptr);
/// Retrieves the value specified by @p _jsonPath by from a series of nested JSON dictionaries.
/// @param _jsonPath A dot-separated series of dictionary keys.
/// @param _node The node representing the start of the path.
/// @returns The value of the last key on the path. @a nullptr if any node on the path descends
/// into something that is not a dictionary or the key is not present.
std::optional<Json::Value> jsonValueByPath(Json::Value const& _node, std::string_view _jsonPath);
namespace detail namespace detail
{ {

View File

@ -105,6 +105,8 @@ set(libsolidity_sources
libsolidity/ViewPureChecker.cpp libsolidity/ViewPureChecker.cpp
libsolidity/analysis/FunctionCallGraph.cpp libsolidity/analysis/FunctionCallGraph.cpp
libsolidity/interface/FileReader.cpp libsolidity/interface/FileReader.cpp
libsolidity/ASTPropertyTest.h
libsolidity/ASTPropertyTest.cpp
) )
detect_stray_source_files("${libsolidity_sources}" "libsolidity/") detect_stray_source_files("${libsolidity_sources}" "libsolidity/")

View File

@ -21,6 +21,7 @@
#include <test/TestCase.h> #include <test/TestCase.h>
#include <test/libsolidity/ABIJsonTest.h> #include <test/libsolidity/ABIJsonTest.h>
#include <test/libsolidity/ASTJSONTest.h> #include <test/libsolidity/ASTJSONTest.h>
#include <test/libsolidity/ASTPropertyTest.h>
#include <test/libsolidity/GasTest.h> #include <test/libsolidity/GasTest.h>
#include <test/libsolidity/MemoryGuardTest.h> #include <test/libsolidity/MemoryGuardTest.h>
#include <test/libsolidity/SyntaxTest.h> #include <test/libsolidity/SyntaxTest.h>
@ -76,7 +77,8 @@ Testsuite const g_interactiveTestsuites[] = {
{"JSON ABI", "libsolidity", "ABIJson", false, false, &ABIJsonTest::create}, {"JSON ABI", "libsolidity", "ABIJson", false, false, &ABIJsonTest::create},
{"SMT Checker", "libsolidity", "smtCheckerTests", true, false, &SMTCheckerTest::create}, {"SMT Checker", "libsolidity", "smtCheckerTests", true, false, &SMTCheckerTest::create},
{"Gas Estimates", "libsolidity", "gasTests", false, false, &GasTest::create}, {"Gas Estimates", "libsolidity", "gasTests", false, false, &GasTest::create},
{"Memory Guard Tests", "libsolidity", "memoryGuardTests", false, false, &MemoryGuardTest::create}, {"Memory Guard", "libsolidity", "memoryGuardTests", false, false, &MemoryGuardTest::create},
{"AST Properties", "libsolidity", "astPropertyTests", false, false, &ASTPropertyTest::create},
}; };
} }

View File

@ -0,0 +1,204 @@
/*
This file is part of solidity.
solidity is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
solidity is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with solidity. If not, see <http://www.gnu.org/licenses/>.
*/
// SPDX-License-Identifier: GPL-3.0
#include <test/libsolidity/ASTPropertyTest.h>
#include <test/Common.h>
#include <libsolidity/ast/ASTJsonExporter.h>
#include <libsolidity/interface/CompilerStack.h>
#include <liblangutil/Common.h>
#include <liblangutil/SourceReferenceFormatter.h>
#include <libsolutil/JSON.h>
#include <boost/algorithm/string.hpp>
#include <boost/throw_exception.hpp>
#include <range/v3/algorithm/find_if.hpp>
#include <range/v3/range/conversion.hpp>
#include <range/v3/view/split.hpp>
#include <queue>
using namespace solidity::util;
using namespace solidity::langutil;
using namespace solidity::frontend;
using namespace solidity::frontend::test;
using namespace solidity;
using namespace std;
ASTPropertyTest::ASTPropertyTest(string const& _filename):
TestCase(_filename)
{
if (!boost::algorithm::ends_with(_filename, ".sol"))
BOOST_THROW_EXCEPTION(runtime_error("Not a Solidity file: \"" + _filename + "\"."));
m_source = m_reader.source();
readExpectations();
soltestAssert(m_tests.size() > 0, "No tests specified in " + _filename);
}
string ASTPropertyTest::formatExpectations(bool _obtainedResult)
{
string expectations;
for (string const& testId: m_testOrder)
{
soltestAssert(m_tests.count(testId) > 0);
expectations +=
testId +
": " +
(_obtainedResult ? m_tests[testId].obtainedValue : m_tests[testId].expectedValue)
+ "\n";
}
return expectations;
}
vector<StringPair> ASTPropertyTest::readKeyValuePairs(string const& _input)
{
vector<StringPair> result;
for (string line: _input | ranges::views::split('\n') | ranges::to<vector<string>>)
{
boost::trim(line);
if (line.empty())
continue;
soltestAssert(
ranges::all_of(line, [](char c) { return isprint(c); }),
"Non-printable character(s) found in property test: " + line
);
auto colonPosition = line.find_first_of(':');
soltestAssert(colonPosition != string::npos, "Property test is missing a colon: " + line);
StringPair pair{
boost::trim_copy(line.substr(0, colonPosition)),
boost::trim_copy(line.substr(colonPosition + 1))
};
soltestAssert(!get<0>(pair).empty() != false, "Empty key in property test: " + line);
soltestAssert(!get<1>(pair).empty() != false, "Empty value in property test: " + line);
result.push_back(pair);
}
return result;
}
void ASTPropertyTest::readExpectations()
{
for (auto const& [testId, testExpectation]: readKeyValuePairs(m_reader.simpleExpectations()))
{
soltestAssert(m_tests.count(testId) == 0, "More than one expectation for test \"" + testId + "\"");
m_tests.emplace(testId, Test{"", testExpectation, ""});
m_testOrder.push_back(testId);
}
m_expectation = formatExpectations(false /* _obtainedResult */);
}
void ASTPropertyTest::extractTestsFromAST(Json::Value const& _astJson)
{
queue<Json::Value> nodesToVisit;
nodesToVisit.push(_astJson);
while (!nodesToVisit.empty())
{
Json::Value& node = nodesToVisit.front();
if (node.isArray())
for (auto&& member: node)
nodesToVisit.push(member);
else if (node.isObject())
for (string const& memberName: node.getMemberNames())
{
if (memberName != "documentation")
{
nodesToVisit.push(node[memberName]);
continue;
}
string nodeDocstring = node["documentation"].isObject() ?
node["documentation"]["text"].asString() :
node["documentation"].asString();
soltestAssert(!nodeDocstring.empty());
vector<StringPair> pairs = readKeyValuePairs(nodeDocstring);
if (pairs.empty())
continue;
for (auto const& [testId, testedProperty]: pairs)
{
soltestAssert(
m_tests.count(testId) > 0,
"Test \"" + testId + "\" does not have a corresponding expected value."
);
soltestAssert(
m_tests[testId].property.empty(),
"Test \"" + testId + "\" was already defined before."
);
m_tests[testId].property = testedProperty;
soltestAssert(node.isMember("nodeType"));
optional<Json::Value> propertyNode = jsonValueByPath(node, testedProperty);
soltestAssert(
propertyNode.has_value(),
node["nodeType"].asString() + " node does not have a property named \""s + testedProperty + "\""
);
soltestAssert(
!propertyNode->isObject() && !propertyNode->isArray(),
"Property \"" + testedProperty + "\" is an object or an array."
);
m_tests[testId].obtainedValue = propertyNode->asString();
}
}
nodesToVisit.pop();
}
auto firstTestWithoutProperty = ranges::find_if(
m_tests,
[&](auto const& _testCase) { return _testCase.second.property.empty(); }
);
soltestAssert(
firstTestWithoutProperty == ranges::end(m_tests),
"AST property not defined for test \"" + firstTestWithoutProperty->first + "\""
);
m_obtainedResult = formatExpectations(true /* _obtainedResult */);
}
TestCase::TestResult ASTPropertyTest::run(ostream& _stream, string const& _linePrefix, bool const _formatted)
{
CompilerStack compiler;
compiler.setSources({{
"A",
"pragma solidity >=0.0;\n// SPDX-License-Identifier: GPL-3.0\n" + m_source
}});
compiler.setEVMVersion(solidity::test::CommonOptions::get().evmVersion());
compiler.setOptimiserSettings(solidity::test::CommonOptions::get().optimize);
if (!compiler.parseAndAnalyze())
BOOST_THROW_EXCEPTION(runtime_error(
"Parsing contract failed" +
SourceReferenceFormatter::formatErrorInformation(compiler.errors(), compiler, _formatted)
));
Json::Value astJson = ASTJsonExporter(compiler.state()).toJson(compiler.ast("A"));
soltestAssert(astJson);
extractTestsFromAST(astJson);
return checkResult(_stream, _linePrefix, _formatted);
}

View File

@ -0,0 +1,66 @@
/*
This file is part of solidity.
solidity is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
solidity is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with solidity. If not, see <http://www.gnu.org/licenses/>.
*/
// SPDX-License-Identifier: GPL-3.0
#pragma once
#include <test/TestCase.h>
#include <libsolutil/JSON.h>
#include <string>
#include <vector>
namespace solidity::frontend
{
class CompilerStack;
}
namespace solidity::frontend::test
{
using StringPair = std::pair<std::string, std::string>;
class ASTPropertyTest: public TestCase
{
public:
static std::unique_ptr<TestCase> create(Config const& _config)
{
return std::make_unique<ASTPropertyTest>(_config.filename);
}
ASTPropertyTest(std::string const& _filename);
TestResult run(std::ostream& _stream, std::string const& _linePrefix = "", bool const _formatted = false) override;
private:
struct Test
{
std::string property;
std::string expectedValue;
std::string obtainedValue;
};
void readExpectations();
std::vector<StringPair> readKeyValuePairs(std::string const& _input);
void extractTestsFromAST(Json::Value const& _astJson);
std::string formatExpectations(bool _obtainedResult = true);
std::vector<std::string> m_testOrder;
std::map<std::string, Test> m_tests;
};
}

View File

@ -0,0 +1,9 @@
/// TestCase1: name
contract C {
///
///
function f() public pure { }
}
// ----
// TestCase1: C
//

View File

@ -0,0 +1,13 @@
contract C {
function f() public pure {
/// TestCase1: condition.operator
/// TestCase2: initializationExpression.initialValue.value
/// TestCase3: loopExpression.expression.subExpression.name
for(uint i = 1; i < 42; i++) {
}
}
}
// ----
// TestCase1: <
// TestCase2: 1
// TestCase3: i

View File

@ -0,0 +1,11 @@
contract C {
/// TestCase1: name
/// TestCase2: functionSelector
/// TestCase3: visibility
function singleFunction() public pure {
}
}
// ----
// TestCase1: singleFunction
// TestCase2: 3d33252c
// TestCase3: public

View File

@ -0,0 +1,17 @@
contract C {
function f() public pure {
/// TestCase1: condition.operator
for(uint i = 0; i < 42; ++i) {
}
/// TestCase2: initializationExpression.initialValue.value
for(uint i = 1; i < 42; i = i * 2) {
}
/// TestCase3: loopExpression.expression.subExpression.name
for(uint i = 0; i < 42; i++) {
}
}
}
// ----
// TestCase1: <
// TestCase2: 1
// TestCase3: i

View File

@ -0,0 +1,9 @@
/// TestCase1: nameLocation
/// TestCase2: src
contract C {
function f() public pure {
}
}
// ----
// TestCase1: 115:1:-1
// TestCase2: 106:51:-1

View File

@ -0,0 +1,12 @@
/// TestContractC: name
contract C {
/// TestStateVarX: stateVariable
uint x;
/// TestFunctionF: visibility
function f() public pure {
}
}
// ----
// TestContractC: C
// TestStateVarX: true
// TestFunctionF: public

View File

@ -32,6 +32,7 @@ add_executable(isoltest
../ExecutionFramework.cpp ../ExecutionFramework.cpp
../libsolidity/ABIJsonTest.cpp ../libsolidity/ABIJsonTest.cpp
../libsolidity/ASTJSONTest.cpp ../libsolidity/ASTJSONTest.cpp
../libsolidity/ASTPropertyTest.cpp
../libsolidity/SMTCheckerTest.cpp ../libsolidity/SMTCheckerTest.cpp
../libyul/Common.cpp ../libyul/Common.cpp
../libyul/ControlFlowGraphTest.cpp ../libyul/ControlFlowGraphTest.cpp