2014-10-07 16:25:04 +00:00
|
|
|
/*
|
2016-11-18 23:13:20 +00:00
|
|
|
This file is part of solidity.
|
2014-10-07 16:25:04 +00:00
|
|
|
|
2016-11-18 23:13:20 +00:00
|
|
|
solidity is free software: you can redistribute it and/or modify
|
2014-10-07 16:25:04 +00:00
|
|
|
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.
|
|
|
|
|
2016-11-18 23:13:20 +00:00
|
|
|
solidity is distributed in the hope that it will be useful,
|
2014-10-07 16:25:04 +00:00
|
|
|
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
|
2016-11-18 23:13:20 +00:00
|
|
|
along with solidity. If not, see <http://www.gnu.org/licenses/>.
|
2014-10-07 16:25:04 +00:00
|
|
|
*/
|
|
|
|
/**
|
|
|
|
* @author Christian <c@ethdev.com>
|
|
|
|
* @date 2014
|
|
|
|
* Unit tests for the solidity parser.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <string>
|
2014-12-03 06:46:55 +00:00
|
|
|
#include <memory>
|
2018-11-14 13:59:30 +00:00
|
|
|
#include <liblangutil/Scanner.h>
|
2015-10-20 22:21:52 +00:00
|
|
|
#include <libsolidity/parsing/Parser.h>
|
2018-11-14 13:59:30 +00:00
|
|
|
#include <liblangutil/ErrorReporter.h>
|
2020-01-14 16:48:17 +00:00
|
|
|
#include <test/Common.h>
|
2018-03-14 11:04:04 +00:00
|
|
|
#include <test/libsolidity/ErrorCheck.h>
|
2019-04-15 12:50:00 +00:00
|
|
|
#include <libsolidity/ast/ASTVisitor.h>
|
2014-10-07 16:25:04 +00:00
|
|
|
|
2020-01-14 14:55:31 +00:00
|
|
|
#include <boost/test/unit_test.hpp>
|
|
|
|
|
2014-12-03 06:46:55 +00:00
|
|
|
using namespace std;
|
2019-12-23 15:50:30 +00:00
|
|
|
using namespace solidity::langutil;
|
2014-12-03 06:46:55 +00:00
|
|
|
|
2019-12-23 15:50:30 +00:00
|
|
|
namespace solidity::frontend::test
|
2014-10-16 12:08:54 +00:00
|
|
|
{
|
2014-10-07 16:25:04 +00:00
|
|
|
|
2014-10-16 12:08:54 +00:00
|
|
|
namespace
|
|
|
|
{
|
2019-07-17 19:29:56 +00:00
|
|
|
ASTPointer<ContractDefinition> parseText(std::string const& _source, ErrorList& _errors, bool errorRecovery = false)
|
2014-10-16 12:08:54 +00:00
|
|
|
{
|
2017-05-11 13:26:35 +00:00
|
|
|
ErrorReporter errorReporter(_errors);
|
2019-05-22 11:57:48 +00:00
|
|
|
ASTPointer<SourceUnit> sourceUnit = Parser(
|
|
|
|
errorReporter,
|
2020-01-14 16:48:17 +00:00
|
|
|
solidity::test::CommonOptions::get().evmVersion(),
|
2019-07-17 19:29:56 +00:00
|
|
|
errorRecovery
|
2019-05-22 11:57:48 +00:00
|
|
|
).parse(std::make_shared<Scanner>(CharStream(_source, "")));
|
2015-10-15 14:27:26 +00:00
|
|
|
if (!sourceUnit)
|
2015-10-14 18:37:41 +00:00
|
|
|
return ASTPointer<ContractDefinition>();
|
2015-08-31 16:44:29 +00:00
|
|
|
for (ASTPointer<ASTNode> const& node: sourceUnit->nodes())
|
2014-12-03 06:46:55 +00:00
|
|
|
if (ASTPointer<ContractDefinition> contract = dynamic_pointer_cast<ContractDefinition>(node))
|
|
|
|
return contract;
|
|
|
|
BOOST_FAIL("No contract found in source.");
|
|
|
|
return ASTPointer<ContractDefinition>();
|
2014-10-16 12:08:54 +00:00
|
|
|
}
|
2014-12-15 16:45:18 +00:00
|
|
|
|
2015-10-14 18:37:41 +00:00
|
|
|
bool successParse(std::string const& _source)
|
|
|
|
{
|
|
|
|
ErrorList errors;
|
|
|
|
try
|
|
|
|
{
|
|
|
|
auto sourceUnit = parseText(_source, errors);
|
2015-10-15 14:27:26 +00:00
|
|
|
if (!sourceUnit)
|
2015-10-14 18:37:41 +00:00
|
|
|
return false;
|
|
|
|
}
|
2016-03-18 08:22:15 +00:00
|
|
|
catch (FatalError const& /*_exception*/)
|
2015-10-14 18:37:41 +00:00
|
|
|
{
|
|
|
|
if (Error::containsErrorOfType(errors, Error::Type::ParserError))
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (Error::containsErrorOfType(errors, Error::Type::ParserError))
|
|
|
|
return false;
|
|
|
|
|
2015-10-15 14:27:26 +00:00
|
|
|
BOOST_CHECK(Error::containsOnlyWarnings(errors));
|
2015-10-14 18:37:41 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2019-07-17 19:29:56 +00:00
|
|
|
Error getError(std::string const& _source, bool errorRecovery = false)
|
2016-12-06 09:35:56 +00:00
|
|
|
{
|
|
|
|
ErrorList errors;
|
|
|
|
try
|
|
|
|
{
|
2019-07-17 19:29:56 +00:00
|
|
|
parseText(_source, errors, errorRecovery);
|
2016-12-06 09:35:56 +00:00
|
|
|
}
|
|
|
|
catch (FatalError const& /*_exception*/)
|
|
|
|
{
|
|
|
|
// no-op
|
|
|
|
}
|
|
|
|
Error const* error = Error::containsErrorOfType(errors, Error::Type::ParserError);
|
|
|
|
BOOST_REQUIRE(error);
|
|
|
|
return *error;
|
|
|
|
}
|
|
|
|
|
2015-10-14 18:37:41 +00:00
|
|
|
void checkFunctionNatspec(
|
2015-11-23 22:57:17 +00:00
|
|
|
FunctionDefinition const* _function,
|
2015-10-14 18:37:41 +00:00
|
|
|
std::string const& _expectedDoc
|
|
|
|
)
|
2015-01-21 18:07:03 +00:00
|
|
|
{
|
2015-08-31 16:44:29 +00:00
|
|
|
auto doc = _function->documentation();
|
2015-01-21 18:07:03 +00:00
|
|
|
BOOST_CHECK_MESSAGE(doc != nullptr, "Function does not have Natspec Doc as expected");
|
|
|
|
BOOST_CHECK_EQUAL(*doc, _expectedDoc);
|
|
|
|
}
|
|
|
|
|
2014-10-08 18:53:50 +00:00
|
|
|
}
|
|
|
|
|
2016-12-06 09:35:56 +00:00
|
|
|
#define CHECK_PARSE_ERROR(source, substring) \
|
|
|
|
do \
|
|
|
|
{\
|
|
|
|
Error err = getError((source)); \
|
|
|
|
BOOST_CHECK(searchErrorMessage(err, (substring))); \
|
|
|
|
}\
|
|
|
|
while(0)
|
|
|
|
|
2014-11-27 14:21:22 +00:00
|
|
|
|
2014-10-07 16:25:04 +00:00
|
|
|
BOOST_AUTO_TEST_SUITE(SolidityParser)
|
|
|
|
|
2019-10-30 13:34:37 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(reserved_keywords)
|
|
|
|
{
|
|
|
|
BOOST_CHECK(!TokenTraits::isReservedKeyword(Token::Identifier));
|
2019-11-11 16:09:31 +00:00
|
|
|
BOOST_CHECK(TokenTraits::isReservedKeyword(Token::After));
|
2019-10-30 13:34:37 +00:00
|
|
|
BOOST_CHECK(TokenTraits::isReservedKeyword(Token::Unchecked));
|
|
|
|
BOOST_CHECK(!TokenTraits::isReservedKeyword(Token::Illegal));
|
|
|
|
}
|
|
|
|
|
2018-12-18 11:17:52 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(unsatisfied_version)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
pragma solidity ^99.99.0;
|
|
|
|
)";
|
|
|
|
CHECK_PARSE_ERROR(text, "Source file requires different compiler version");
|
|
|
|
}
|
|
|
|
|
2018-12-12 15:45:17 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(unsatisfied_version_followed_by_invalid_syntax)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
pragma solidity ^99.99.0;
|
|
|
|
this is surely invalid
|
|
|
|
)";
|
|
|
|
CHECK_PARSE_ERROR(text, "Source file requires different compiler version");
|
|
|
|
}
|
|
|
|
|
2019-07-17 19:29:56 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(unsatisfied_version_with_recovery)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
pragma solidity ^99.99.0;
|
|
|
|
contract test {
|
|
|
|
uint ;
|
|
|
|
}
|
|
|
|
)";
|
|
|
|
Error err = getError(text, true);
|
|
|
|
BOOST_CHECK(searchErrorMessage(err, "Expected identifier but got ';'"));
|
|
|
|
}
|
|
|
|
|
2014-11-27 14:21:22 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(function_natspec_documentation)
|
|
|
|
{
|
2016-12-01 03:39:30 +00:00
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
uint256 stateVar;
|
|
|
|
/// This is a test function
|
2018-06-29 14:52:41 +00:00
|
|
|
function functionName(bytes32 input) public returns (bytes32 out) {}
|
2016-12-01 03:39:30 +00:00
|
|
|
}
|
|
|
|
)";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2015-11-23 23:20:37 +00:00
|
|
|
ErrorList errors;
|
|
|
|
ASTPointer<ContractDefinition> contract = parseText(text, errors);
|
2015-11-23 22:57:17 +00:00
|
|
|
FunctionDefinition const* function = nullptr;
|
2015-11-23 23:20:37 +00:00
|
|
|
auto functions = contract->definedFunctions();
|
2015-10-14 18:37:41 +00:00
|
|
|
|
2017-09-20 09:52:41 +00:00
|
|
|
BOOST_REQUIRE_MESSAGE(function = functions.at(0), "Failed to retrieve function");
|
2015-01-21 18:07:03 +00:00
|
|
|
checkFunctionNatspec(function, "This is a test function");
|
2014-11-27 14:21:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(function_normal_comments)
|
|
|
|
{
|
2015-11-23 22:57:17 +00:00
|
|
|
FunctionDefinition const* function = nullptr;
|
2016-12-01 03:39:30 +00:00
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
uint256 stateVar;
|
|
|
|
// We won't see this comment
|
2018-06-29 14:52:41 +00:00
|
|
|
function functionName(bytes32 input) public returns (bytes32 out) {}
|
2016-12-01 03:39:30 +00:00
|
|
|
}
|
|
|
|
)";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
|
|
|
ErrorList errors;
|
2015-11-23 23:20:37 +00:00
|
|
|
ASTPointer<ContractDefinition> contract = parseText(text, errors);
|
|
|
|
auto functions = contract->definedFunctions();
|
2017-09-20 09:52:41 +00:00
|
|
|
BOOST_REQUIRE_MESSAGE(function = functions.at(0), "Failed to retrieve function");
|
2015-08-31 16:44:29 +00:00
|
|
|
BOOST_CHECK_MESSAGE(function->documentation() == nullptr,
|
2015-01-21 18:07:03 +00:00
|
|
|
"Should not have gotten a Natspecc comment for this function");
|
2014-11-27 14:21:22 +00:00
|
|
|
}
|
|
|
|
|
2014-11-27 17:57:50 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(multiple_functions_natspec_documentation)
|
|
|
|
{
|
2015-11-23 22:57:17 +00:00
|
|
|
FunctionDefinition const* function = nullptr;
|
2016-12-01 03:39:30 +00:00
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
uint256 stateVar;
|
|
|
|
/// This is test function 1
|
2018-06-29 14:52:41 +00:00
|
|
|
function functionName1(bytes32 input) public returns (bytes32 out) {}
|
2016-12-01 03:39:30 +00:00
|
|
|
/// This is test function 2
|
2018-06-29 14:52:41 +00:00
|
|
|
function functionName2(bytes32 input) public returns (bytes32 out) {}
|
2016-12-01 03:39:30 +00:00
|
|
|
// nothing to see here
|
2018-06-29 14:52:41 +00:00
|
|
|
function functionName3(bytes32 input) public returns (bytes32 out) {}
|
2016-12-01 03:39:30 +00:00
|
|
|
/// This is test function 4
|
2018-06-29 14:52:41 +00:00
|
|
|
function functionName4(bytes32 input) public returns (bytes32 out) {}
|
2016-12-01 03:39:30 +00:00
|
|
|
}
|
|
|
|
)";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
|
|
|
ErrorList errors;
|
2015-11-23 23:20:37 +00:00
|
|
|
ASTPointer<ContractDefinition> contract = parseText(text, errors);
|
|
|
|
auto functions = contract->definedFunctions();
|
2014-11-27 17:57:50 +00:00
|
|
|
|
2017-09-20 09:52:41 +00:00
|
|
|
BOOST_REQUIRE_MESSAGE(function = functions.at(0), "Failed to retrieve function");
|
2015-01-21 18:07:03 +00:00
|
|
|
checkFunctionNatspec(function, "This is test function 1");
|
2014-11-27 17:57:50 +00:00
|
|
|
|
2017-09-20 09:52:41 +00:00
|
|
|
BOOST_REQUIRE_MESSAGE(function = functions.at(1), "Failed to retrieve function");
|
2015-01-21 18:07:03 +00:00
|
|
|
checkFunctionNatspec(function, "This is test function 2");
|
2014-11-27 17:57:50 +00:00
|
|
|
|
2017-09-20 09:52:41 +00:00
|
|
|
BOOST_REQUIRE_MESSAGE(function = functions.at(2), "Failed to retrieve function");
|
2015-08-31 16:44:29 +00:00
|
|
|
BOOST_CHECK_MESSAGE(function->documentation() == nullptr,
|
2014-11-28 00:26:37 +00:00
|
|
|
"Should not have gotten natspec comment for functionName3()");
|
2014-11-27 17:57:50 +00:00
|
|
|
|
2017-09-20 09:52:41 +00:00
|
|
|
BOOST_REQUIRE_MESSAGE(function = functions.at(3), "Failed to retrieve function");
|
2015-01-21 18:07:03 +00:00
|
|
|
checkFunctionNatspec(function, "This is test function 4");
|
2014-11-27 17:57:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(multiline_function_documentation)
|
|
|
|
{
|
2015-11-23 22:57:17 +00:00
|
|
|
FunctionDefinition const* function = nullptr;
|
2016-12-01 03:39:30 +00:00
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
uint256 stateVar;
|
|
|
|
/// This is a test function
|
|
|
|
/// and it has 2 lines
|
2018-06-29 14:52:41 +00:00
|
|
|
function functionName1(bytes32 input) public returns (bytes32 out) {}
|
2016-12-01 03:39:30 +00:00
|
|
|
}
|
|
|
|
)";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
|
|
|
ErrorList errors;
|
2015-11-23 23:20:37 +00:00
|
|
|
ASTPointer<ContractDefinition> contract = parseText(text, errors);
|
|
|
|
auto functions = contract->definedFunctions();
|
2017-09-20 09:52:41 +00:00
|
|
|
BOOST_REQUIRE_MESSAGE(function = functions.at(0), "Failed to retrieve function");
|
2015-01-21 18:07:03 +00:00
|
|
|
checkFunctionNatspec(function, "This is a test function\n"
|
|
|
|
" and it has 2 lines");
|
2014-11-27 17:57:50 +00:00
|
|
|
}
|
|
|
|
|
2014-11-27 23:40:00 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(natspec_comment_in_function_body)
|
|
|
|
{
|
2015-11-23 22:57:17 +00:00
|
|
|
FunctionDefinition const* function = nullptr;
|
2016-12-01 03:39:30 +00:00
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
/// fun1 description
|
|
|
|
function fun1(uint256 a) {
|
|
|
|
var b;
|
|
|
|
/// I should not interfere with actual natspec comments
|
|
|
|
uint256 c;
|
|
|
|
mapping(address=>bytes32) d;
|
|
|
|
bytes7 name = "Solidity";
|
|
|
|
}
|
|
|
|
/// This is a test function
|
|
|
|
/// and it has 2 lines
|
2018-06-29 14:52:41 +00:00
|
|
|
function fun(bytes32 input) public returns (bytes32 out) {}
|
2016-12-01 03:39:30 +00:00
|
|
|
}
|
|
|
|
)";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
|
|
|
ErrorList errors;
|
2015-11-23 23:20:37 +00:00
|
|
|
ASTPointer<ContractDefinition> contract = parseText(text, errors);
|
|
|
|
auto functions = contract->definedFunctions();
|
2014-11-27 23:40:00 +00:00
|
|
|
|
2017-09-20 09:52:41 +00:00
|
|
|
BOOST_REQUIRE_MESSAGE(function = functions.at(0), "Failed to retrieve function");
|
2015-01-21 18:07:03 +00:00
|
|
|
checkFunctionNatspec(function, "fun1 description");
|
2014-11-27 23:40:00 +00:00
|
|
|
|
2017-09-20 09:52:41 +00:00
|
|
|
BOOST_REQUIRE_MESSAGE(function = functions.at(1), "Failed to retrieve function");
|
2015-01-21 18:07:03 +00:00
|
|
|
checkFunctionNatspec(function, "This is a test function\n"
|
|
|
|
" and it has 2 lines");
|
2014-11-27 23:40:00 +00:00
|
|
|
}
|
|
|
|
|
2014-11-30 22:33:04 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(natspec_docstring_between_keyword_and_signature)
|
|
|
|
{
|
2015-11-23 22:57:17 +00:00
|
|
|
FunctionDefinition const* function = nullptr;
|
2016-12-01 03:39:30 +00:00
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
uint256 stateVar;
|
|
|
|
function ///I am in the wrong place
|
|
|
|
fun1(uint256 a) {
|
|
|
|
var b;
|
|
|
|
/// I should not interfere with actual natspec comments
|
|
|
|
uint256 c;
|
|
|
|
mapping(address=>bytes32) d;
|
|
|
|
bytes7 name = "Solidity";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
|
|
|
ErrorList errors;
|
2015-11-23 23:20:37 +00:00
|
|
|
ASTPointer<ContractDefinition> contract = parseText(text, errors);
|
|
|
|
auto functions = contract->definedFunctions();
|
2014-11-30 22:33:04 +00:00
|
|
|
|
2017-09-20 09:52:41 +00:00
|
|
|
BOOST_REQUIRE_MESSAGE(function = functions.at(0), "Failed to retrieve function");
|
2015-08-31 16:44:29 +00:00
|
|
|
BOOST_CHECK_MESSAGE(!function->documentation(),
|
2014-11-30 22:33:04 +00:00
|
|
|
"Shouldn't get natspec docstring for this function");
|
|
|
|
}
|
|
|
|
|
2014-11-28 10:17:18 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(natspec_docstring_after_signature)
|
|
|
|
{
|
2015-11-23 22:57:17 +00:00
|
|
|
FunctionDefinition const* function = nullptr;
|
2016-12-01 03:39:30 +00:00
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
uint256 stateVar;
|
|
|
|
function fun1(uint256 a) {
|
|
|
|
/// I should have been above the function signature
|
|
|
|
var b;
|
|
|
|
/// I should not interfere with actual natspec comments
|
|
|
|
uint256 c;
|
|
|
|
mapping(address=>bytes32) d;
|
|
|
|
bytes7 name = "Solidity";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
|
|
|
ErrorList errors;
|
2015-11-23 23:20:37 +00:00
|
|
|
ASTPointer<ContractDefinition> contract = parseText(text, errors);
|
|
|
|
auto functions = contract->definedFunctions();
|
2014-11-28 10:17:18 +00:00
|
|
|
|
2017-09-20 09:52:41 +00:00
|
|
|
BOOST_REQUIRE_MESSAGE(function = functions.at(0), "Failed to retrieve function");
|
2015-08-31 16:44:29 +00:00
|
|
|
BOOST_CHECK_MESSAGE(!function->documentation(),
|
2014-11-28 10:17:18 +00:00
|
|
|
"Shouldn't get natspec docstring for this function");
|
|
|
|
}
|
|
|
|
|
2014-10-09 13:57:49 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(variable_definition)
|
|
|
|
{
|
2016-12-01 03:39:30 +00:00
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
function fun(uint256 a) {
|
|
|
|
var b;
|
|
|
|
uint256 c;
|
|
|
|
mapping(address=>bytes32) d;
|
|
|
|
customtype varname;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2014-10-09 13:57:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(variable_definition_with_initialization)
|
|
|
|
{
|
2016-12-01 03:39:30 +00:00
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
function fun(uint256 a) {
|
|
|
|
var b = 2;
|
|
|
|
uint256 c = 0x87;
|
|
|
|
mapping(address=>bytes32) d;
|
|
|
|
bytes7 name = "Solidity";
|
|
|
|
customtype varname;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2014-10-09 13:57:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(operator_expression)
|
|
|
|
{
|
2016-12-01 03:39:30 +00:00
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
function fun(uint256 a) {
|
|
|
|
uint256 x = (1 + 4) || false && (1 - 12) + -9;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2014-10-09 13:57:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(complex_expression)
|
|
|
|
{
|
2016-12-01 03:39:30 +00:00
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
function fun(uint256 a) {
|
|
|
|
uint256 x = (1 + 4).member(++67)[a/=9] || true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2014-10-09 13:57:49 +00:00
|
|
|
}
|
|
|
|
|
2014-11-04 14:14:31 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(statement_starting_with_type_conversion)
|
|
|
|
{
|
2016-12-01 03:39:30 +00:00
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
function fun() {
|
|
|
|
uint64(2);
|
|
|
|
uint64[7](3);
|
|
|
|
uint64[](3);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2015-02-21 17:25:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(type_conversion_to_dynamic_array)
|
|
|
|
{
|
2016-12-01 03:39:30 +00:00
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
function fun() {
|
|
|
|
var x = uint64[](3);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2014-11-04 14:14:31 +00:00
|
|
|
}
|
2014-10-07 16:25:04 +00:00
|
|
|
|
2014-12-03 06:46:55 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(import_directive)
|
|
|
|
{
|
2016-12-01 03:39:30 +00:00
|
|
|
char const* text = R"(
|
|
|
|
import "abc";
|
|
|
|
contract test {
|
|
|
|
function fun() {
|
|
|
|
uint64(2);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2014-12-03 06:46:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(multiple_contracts)
|
|
|
|
{
|
2016-12-01 03:39:30 +00:00
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
function fun() {
|
|
|
|
uint64(2);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
contract test2 {
|
|
|
|
function fun() {
|
|
|
|
uint64(2);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2014-12-03 06:46:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(multiple_contracts_and_imports)
|
|
|
|
{
|
2016-12-01 03:39:30 +00:00
|
|
|
char const* text = R"(
|
|
|
|
import "abc";
|
|
|
|
contract test {
|
|
|
|
function fun() {
|
|
|
|
uint64(2);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
import "def";
|
|
|
|
contract test2 {
|
|
|
|
function fun() {
|
|
|
|
uint64(2);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
import "ghi";
|
|
|
|
)";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2014-12-03 06:46:55 +00:00
|
|
|
}
|
|
|
|
|
2015-01-15 15:15:01 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(contract_inheritance)
|
|
|
|
{
|
2016-12-01 03:39:30 +00:00
|
|
|
char const* text = R"(
|
|
|
|
contract base {
|
|
|
|
function fun() {
|
|
|
|
uint64(2);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
contract derived is base {
|
|
|
|
function fun() {
|
|
|
|
uint64(2);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2015-01-15 15:15:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(contract_multiple_inheritance)
|
|
|
|
{
|
2016-12-01 03:39:30 +00:00
|
|
|
char const* text = R"(
|
|
|
|
contract base {
|
|
|
|
function fun() {
|
|
|
|
uint64(2);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
contract derived is base, nonExisting {
|
|
|
|
function fun() {
|
|
|
|
uint64(2);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2015-01-15 15:15:01 +00:00
|
|
|
}
|
|
|
|
|
2015-01-19 20:05:47 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(contract_multiple_inheritance_with_arguments)
|
|
|
|
{
|
2016-12-01 03:39:30 +00:00
|
|
|
char const* text = R"(
|
|
|
|
contract base {
|
|
|
|
function fun() {
|
|
|
|
uint64(2);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
contract derived is base(2), nonExisting("abc", "def", base.fun()) {
|
|
|
|
function fun() {
|
|
|
|
uint64(2);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2015-01-19 20:05:47 +00:00
|
|
|
}
|
|
|
|
|
2015-02-02 16:24:09 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(multiple_visibility_specifiers)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract c {
|
2015-02-22 18:37:54 +00:00
|
|
|
uint private internal a;
|
2018-05-16 13:52:24 +00:00
|
|
|
}
|
|
|
|
)";
|
2017-08-09 12:35:38 +00:00
|
|
|
CHECK_PARSE_ERROR(text, "Visibility already specified as \"private\".");
|
2017-08-09 12:22:19 +00:00
|
|
|
text = R"(
|
|
|
|
contract c {
|
|
|
|
function f() private external {}
|
2018-05-16 13:52:24 +00:00
|
|
|
}
|
|
|
|
)";
|
2017-08-09 12:35:38 +00:00
|
|
|
CHECK_PARSE_ERROR(text, "Visibility already specified as \"private\".");
|
2015-02-02 16:24:09 +00:00
|
|
|
}
|
|
|
|
|
2018-03-02 09:36:04 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(keyword_is_reserved)
|
|
|
|
{
|
|
|
|
auto keywords = {
|
|
|
|
"after",
|
2018-03-01 11:18:44 +00:00
|
|
|
"alias",
|
|
|
|
"apply",
|
|
|
|
"auto",
|
2018-03-02 09:36:04 +00:00
|
|
|
"case",
|
|
|
|
"catch",
|
2018-03-01 11:18:44 +00:00
|
|
|
"copyof",
|
2018-03-02 09:36:04 +00:00
|
|
|
"default",
|
2018-03-01 11:18:44 +00:00
|
|
|
"define",
|
2018-03-02 09:36:04 +00:00
|
|
|
"final",
|
2018-03-01 11:18:44 +00:00
|
|
|
"immutable",
|
|
|
|
"implements",
|
2018-03-02 09:36:04 +00:00
|
|
|
"in",
|
|
|
|
"inline",
|
|
|
|
"let",
|
2018-03-01 11:18:44 +00:00
|
|
|
"macro",
|
2018-03-02 09:36:04 +00:00
|
|
|
"match",
|
2018-03-01 11:18:44 +00:00
|
|
|
"mutable",
|
2018-03-02 09:36:04 +00:00
|
|
|
"null",
|
|
|
|
"of",
|
2018-03-01 11:18:44 +00:00
|
|
|
"partial",
|
|
|
|
"promise",
|
|
|
|
"reference",
|
2018-03-02 09:36:04 +00:00
|
|
|
"relocatable",
|
2018-03-01 11:18:44 +00:00
|
|
|
"sealed",
|
|
|
|
"sizeof",
|
2018-03-02 09:36:04 +00:00
|
|
|
"static",
|
2018-03-01 11:18:44 +00:00
|
|
|
"supports",
|
2018-03-02 09:36:04 +00:00
|
|
|
"switch",
|
|
|
|
"try",
|
2018-03-01 11:18:44 +00:00
|
|
|
"typedef",
|
|
|
|
"typeof",
|
|
|
|
"unchecked"
|
2018-03-02 09:36:04 +00:00
|
|
|
};
|
|
|
|
|
2019-11-11 16:09:31 +00:00
|
|
|
BOOST_CHECK_EQUAL(std::size(keywords), static_cast<int>(Token::Unchecked) - static_cast<int>(Token::After) + 1);
|
|
|
|
|
2019-02-13 15:22:42 +00:00
|
|
|
for (auto const& keyword: keywords)
|
2018-03-02 09:36:04 +00:00
|
|
|
{
|
|
|
|
auto text = std::string("contract ") + keyword + " {}";
|
2018-03-01 11:18:44 +00:00
|
|
|
CHECK_PARSE_ERROR(text.c_str(), string("Expected identifier but got reserved keyword '") + keyword + "'");
|
2018-03-02 09:36:04 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-10-16 14:12:14 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(member_access_parser_ambiguity)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract C {
|
|
|
|
struct S { uint a; uint b; uint[][][] c; }
|
|
|
|
function f() {
|
|
|
|
C.S x;
|
|
|
|
C.S memory y;
|
|
|
|
C.S[10] memory z;
|
|
|
|
C.S[10](x);
|
|
|
|
x.a = 2;
|
|
|
|
x.c[1][2][3] = 9;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
|
|
|
BOOST_CHECK(successParse(text));
|
|
|
|
}
|
|
|
|
|
2015-11-22 19:39:10 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(using_for)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract C {
|
|
|
|
struct s { uint a; }
|
|
|
|
using LibraryName for uint;
|
|
|
|
using Library2 for *;
|
|
|
|
using Lib for s;
|
|
|
|
function f() {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
|
|
|
BOOST_CHECK(successParse(text));
|
|
|
|
}
|
|
|
|
|
2015-12-14 17:01:40 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(complex_import)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
import "abc" as x;
|
|
|
|
import * as x from "abc";
|
|
|
|
import {a as b, c as d, f} from "def";
|
|
|
|
contract x {}
|
|
|
|
)";
|
|
|
|
BOOST_CHECK(successParse(text));
|
|
|
|
}
|
|
|
|
|
2017-08-14 16:58:56 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(recursion_depth1)
|
|
|
|
{
|
|
|
|
string text("contract C { bytes");
|
|
|
|
for (size_t i = 0; i < 30000; i++)
|
|
|
|
text += "[";
|
|
|
|
CHECK_PARSE_ERROR(text.c_str(), "Maximum recursion depth reached during parsing");
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(recursion_depth2)
|
|
|
|
{
|
|
|
|
string text("contract C { function f() {");
|
|
|
|
for (size_t i = 0; i < 30000; i++)
|
|
|
|
text += "{";
|
|
|
|
CHECK_PARSE_ERROR(text, "Maximum recursion depth reached during parsing");
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(recursion_depth3)
|
|
|
|
{
|
|
|
|
string text("contract C { function f() { uint x = f(");
|
|
|
|
for (size_t i = 0; i < 30000; i++)
|
|
|
|
text += "(";
|
|
|
|
CHECK_PARSE_ERROR(text, "Maximum recursion depth reached during parsing");
|
|
|
|
}
|
|
|
|
|
2017-08-14 16:59:17 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(recursion_depth4)
|
|
|
|
{
|
|
|
|
string text("contract C { function f() { uint a;");
|
|
|
|
for (size_t i = 0; i < 30000; i++)
|
|
|
|
text += "(";
|
|
|
|
text += "a";
|
|
|
|
for (size_t i = 0; i < 30000; i++)
|
|
|
|
text += "++)";
|
|
|
|
text += "}}";
|
|
|
|
CHECK_PARSE_ERROR(text, "Maximum recursion depth reached during parsing");
|
|
|
|
}
|
|
|
|
|
2019-04-15 12:50:00 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(inline_asm_end_location)
|
|
|
|
{
|
|
|
|
auto sourceCode = std::string(R"(
|
|
|
|
contract C {
|
|
|
|
function f() public pure returns (uint y) {
|
|
|
|
uint a;
|
|
|
|
assembly { a := 0x12345678 }
|
|
|
|
uint z = a;
|
|
|
|
y = z;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)");
|
|
|
|
ErrorList errors;
|
|
|
|
auto contract = parseText(sourceCode, errors);
|
|
|
|
|
|
|
|
class CheckInlineAsmLocation: public ASTConstVisitor
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
bool visited = false;
|
|
|
|
virtual bool visit(InlineAssembly const& _inlineAsm)
|
|
|
|
{
|
|
|
|
auto loc = _inlineAsm.location();
|
|
|
|
auto asmStr = loc.source->source().substr(loc.start, loc.end - loc.start);
|
|
|
|
BOOST_CHECK_EQUAL(asmStr, "assembly { a := 0x12345678 }");
|
|
|
|
visited = true;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
CheckInlineAsmLocation visitor;
|
|
|
|
contract->accept(visitor);
|
|
|
|
|
|
|
|
BOOST_CHECK_MESSAGE(visitor.visited, "No inline asm block found?!");
|
|
|
|
}
|
|
|
|
|
2014-10-07 16:25:04 +00:00
|
|
|
BOOST_AUTO_TEST_SUITE_END()
|
|
|
|
|
2014-10-16 12:08:54 +00:00
|
|
|
} // end namespaces
|