mirror of
https://github.com/ethereum/solidity
synced 2023-10-03 13:03:40 +00:00
Added new kind of test: check for specific properties of AST
This commit is contained in:
parent
ee21b03e6c
commit
969aea6d33
@ -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
|
||||||
|
@ -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
|
||||||
{
|
{
|
||||||
|
|
||||||
|
@ -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/")
|
||||||
|
|
||||||
|
@ -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},
|
||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
|
204
test/libsolidity/ASTPropertyTest.cpp
Normal file
204
test/libsolidity/ASTPropertyTest.cpp
Normal 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);
|
||||||
|
}
|
66
test/libsolidity/ASTPropertyTest.h
Normal file
66
test/libsolidity/ASTPropertyTest.h
Normal 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;
|
||||||
|
};
|
||||||
|
|
||||||
|
}
|
9
test/libsolidity/astPropertyTests/blank_test_case.sol
Normal file
9
test/libsolidity/astPropertyTests/blank_test_case.sol
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
/// TestCase1: name
|
||||||
|
contract C {
|
||||||
|
///
|
||||||
|
///
|
||||||
|
function f() public pure { }
|
||||||
|
}
|
||||||
|
// ----
|
||||||
|
// TestCase1: C
|
||||||
|
//
|
@ -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
|
@ -0,0 +1,11 @@
|
|||||||
|
contract C {
|
||||||
|
/// TestCase1: name
|
||||||
|
/// TestCase2: functionSelector
|
||||||
|
/// TestCase3: visibility
|
||||||
|
function singleFunction() public pure {
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// ----
|
||||||
|
// TestCase1: singleFunction
|
||||||
|
// TestCase2: 3d33252c
|
||||||
|
// TestCase3: public
|
17
test/libsolidity/astPropertyTests/nested_properties.sol
Normal file
17
test/libsolidity/astPropertyTests/nested_properties.sol
Normal 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
|
@ -0,0 +1,9 @@
|
|||||||
|
/// TestCase1: nameLocation
|
||||||
|
/// TestCase2: src
|
||||||
|
contract C {
|
||||||
|
function f() public pure {
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// ----
|
||||||
|
// TestCase1: 115:1:-1
|
||||||
|
// TestCase2: 106:51:-1
|
12
test/libsolidity/astPropertyTests/simple_properties.sol
Normal file
12
test/libsolidity/astPropertyTests/simple_properties.sol
Normal 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
|
@ -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
|
||||||
|
Loading…
Reference in New Issue
Block a user