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>
|
2015-10-20 22:21:52 +00:00
|
|
|
#include <libsolidity/parsing/Scanner.h>
|
|
|
|
#include <libsolidity/parsing/Parser.h>
|
2017-05-11 13:26:35 +00:00
|
|
|
#include <libsolidity/interface/ErrorReporter.h>
|
2018-03-14 11:04:04 +00:00
|
|
|
#include <test/Options.h>
|
|
|
|
#include <test/libsolidity/ErrorCheck.h>
|
2014-10-07 16:25:04 +00:00
|
|
|
|
2014-12-03 06:46:55 +00:00
|
|
|
using namespace std;
|
|
|
|
|
2014-10-16 12:08:54 +00:00
|
|
|
namespace dev
|
|
|
|
{
|
|
|
|
namespace solidity
|
|
|
|
{
|
|
|
|
namespace test
|
|
|
|
{
|
2014-10-07 16:25:04 +00:00
|
|
|
|
2014-10-16 12:08:54 +00:00
|
|
|
namespace
|
|
|
|
{
|
2015-10-14 18:37:41 +00:00
|
|
|
ASTPointer<ContractDefinition> parseText(std::string const& _source, ErrorList& _errors)
|
2014-10-16 12:08:54 +00:00
|
|
|
{
|
2017-05-11 13:26:35 +00:00
|
|
|
ErrorReporter errorReporter(_errors);
|
|
|
|
ASTPointer<SourceUnit> sourceUnit = Parser(errorReporter).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;
|
|
|
|
}
|
|
|
|
|
2016-12-06 09:35:56 +00:00
|
|
|
Error getError(std::string const& _source)
|
|
|
|
{
|
|
|
|
ErrorList errors;
|
|
|
|
try
|
|
|
|
{
|
|
|
|
parseText(_source, errors);
|
|
|
|
}
|
|
|
|
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)
|
|
|
|
|
2014-10-08 18:53:50 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(empty_function)
|
|
|
|
{
|
2016-12-01 03:39:30 +00:00
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
uint256 stateVar;
|
|
|
|
function functionName(bytes20 arg1, address addr) constant
|
|
|
|
returns (int id)
|
|
|
|
{ }
|
|
|
|
}
|
|
|
|
)";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2014-10-08 18:53:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(no_function_params)
|
|
|
|
{
|
2016-12-01 03:39:30 +00:00
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
uint256 stateVar;
|
|
|
|
function functionName() {}
|
|
|
|
}
|
|
|
|
)";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2014-10-08 18:53:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(single_function_param)
|
|
|
|
{
|
2016-12-01 03:39:30 +00:00
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
uint256 stateVar;
|
|
|
|
function functionName(bytes32 input) returns (bytes32 out) {}
|
|
|
|
}
|
|
|
|
)";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2014-10-08 18:53:50 +00:00
|
|
|
}
|
|
|
|
|
2017-09-21 21:42:34 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(single_function_param_trailing_comma)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
function(uint a,) {}
|
|
|
|
}
|
|
|
|
)";
|
|
|
|
CHECK_PARSE_ERROR(text, "Unexpected trailing comma in parameter list.");
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(single_return_param_trailing_comma)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
function() returns (uint a,) {}
|
|
|
|
}
|
|
|
|
)";
|
|
|
|
CHECK_PARSE_ERROR(text, "Unexpected trailing comma in parameter list.");
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(single_modifier_arg_trailing_comma)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
modifier modTest(uint a,) { _; }
|
|
|
|
function(uint a) {}
|
|
|
|
}
|
|
|
|
)";
|
|
|
|
CHECK_PARSE_ERROR(text, "Unexpected trailing comma in parameter list.");
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(single_event_arg_trailing_comma)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
event Test(uint a,);
|
|
|
|
function(uint a) {}
|
|
|
|
}
|
|
|
|
)";
|
|
|
|
CHECK_PARSE_ERROR(text, "Unexpected trailing comma in parameter list.");
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(multiple_function_param_trailing_comma)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
function(uint a, uint b,) {}
|
|
|
|
}
|
|
|
|
)";
|
|
|
|
CHECK_PARSE_ERROR(text, "Unexpected trailing comma in parameter list.");
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(multiple_return_param_trailing_comma)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
function() returns (uint a, uint b,) {}
|
|
|
|
}
|
|
|
|
)";
|
|
|
|
CHECK_PARSE_ERROR(text, "Unexpected trailing comma in parameter list.");
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(multiple_modifier_arg_trailing_comma)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
modifier modTest(uint a, uint b,) { _; }
|
|
|
|
function(uint a) {}
|
|
|
|
}
|
|
|
|
)";
|
|
|
|
CHECK_PARSE_ERROR(text, "Unexpected trailing comma in parameter list.");
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(multiple_event_arg_trailing_comma)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
event Test(uint a, uint b,);
|
|
|
|
function(uint a) {}
|
|
|
|
}
|
|
|
|
)";
|
|
|
|
CHECK_PARSE_ERROR(text, "Unexpected trailing comma in parameter list.");
|
|
|
|
}
|
|
|
|
|
2015-03-20 16:50:26 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(function_no_body)
|
|
|
|
{
|
2016-12-01 03:39:30 +00:00
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
function functionName(bytes32 input) returns (bytes32 out);
|
|
|
|
}
|
|
|
|
)";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2015-03-20 16:50:26 +00:00
|
|
|
}
|
|
|
|
|
2015-02-04 16:06:54 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(missing_parameter_name_in_named_args)
|
|
|
|
{
|
2016-12-01 03:39:30 +00:00
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
function a(uint a, uint b, uint c) returns (uint r) { r = a * 100 + b * 10 + c * 1; }
|
|
|
|
function b() returns (uint r) { r = a({: 1, : 2, : 3}); }
|
|
|
|
}
|
|
|
|
)";
|
2016-12-06 09:35:56 +00:00
|
|
|
CHECK_PARSE_ERROR(text, "Expected identifier");
|
2015-02-04 16:06:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(missing_argument_in_named_args)
|
|
|
|
{
|
2016-12-01 03:39:30 +00:00
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
function a(uint a, uint b, uint c) returns (uint r) { r = a * 100 + b * 10 + c * 1; }
|
|
|
|
function b() returns (uint r) { r = a({a: , b: , c: }); }
|
|
|
|
}
|
|
|
|
)";
|
2016-12-06 09:35:56 +00:00
|
|
|
CHECK_PARSE_ERROR(text, "Expected primary expression");
|
2015-02-04 16:06:54 +00:00
|
|
|
}
|
|
|
|
|
2017-06-14 03:09:11 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(trailing_comma_in_named_args)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
function a(uint a, uint b, uint c) returns (uint r) { r = a * 100 + b * 10 + c * 1; }
|
|
|
|
function b() returns (uint r) { r = a({a: 1, b: 2, c: 3, }); }
|
|
|
|
}
|
|
|
|
)";
|
|
|
|
CHECK_PARSE_ERROR(text, "Unexpected trailing comma");
|
|
|
|
}
|
|
|
|
|
2015-02-28 08:10:19 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(two_exact_functions)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
function fun(uint a) returns(uint r) { return a; }
|
|
|
|
function fun(uint a) returns(uint r) { return a; }
|
|
|
|
}
|
|
|
|
)";
|
|
|
|
// with support of overloaded functions, during parsing,
|
|
|
|
// we can't determine whether they match exactly, however
|
|
|
|
// it will throw DeclarationError in following stage.
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2015-02-28 08:10:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(overloaded_functions)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
function fun(uint a) returns(uint r) { return a; }
|
|
|
|
function fun(uint a, uint b) returns(uint r) { return a + b; }
|
|
|
|
}
|
|
|
|
)";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2015-02-28 08:10:19 +00:00
|
|
|
}
|
|
|
|
|
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
|
|
|
|
function functionName(bytes32 input) returns (bytes32 out) {}
|
|
|
|
}
|
|
|
|
)";
|
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
|
|
|
|
function functionName(bytes32 input) returns (bytes32 out) {}
|
|
|
|
}
|
|
|
|
)";
|
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
|
|
|
|
function functionName1(bytes32 input) returns (bytes32 out) {}
|
|
|
|
/// This is test function 2
|
|
|
|
function functionName2(bytes32 input) returns (bytes32 out) {}
|
|
|
|
// nothing to see here
|
|
|
|
function functionName3(bytes32 input) returns (bytes32 out) {}
|
|
|
|
/// This is test function 4
|
|
|
|
function functionName4(bytes32 input) returns (bytes32 out) {}
|
|
|
|
}
|
|
|
|
)";
|
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
|
|
|
|
function functionName1(bytes32 input) returns (bytes32 out) {}
|
|
|
|
}
|
|
|
|
)";
|
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
|
|
|
|
function fun(bytes32 input) returns (bytes32 out) {}
|
|
|
|
}
|
|
|
|
)";
|
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-08 18:53:50 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(struct_definition)
|
|
|
|
{
|
2016-12-01 03:39:30 +00:00
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
uint256 stateVar;
|
|
|
|
struct MyStructName {
|
|
|
|
address addr;
|
|
|
|
uint256 count;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2014-10-08 18:53:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(mapping)
|
|
|
|
{
|
2016-12-01 03:39:30 +00:00
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
mapping(address => bytes32) names;
|
|
|
|
}
|
|
|
|
)";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2014-10-08 18:53:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(mapping_in_struct)
|
|
|
|
{
|
2016-12-01 03:39:30 +00:00
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
struct test_struct {
|
|
|
|
address addr;
|
|
|
|
uint256 count;
|
|
|
|
mapping(bytes32 => test_struct) self_reference;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2014-10-08 18:53:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(mapping_to_mapping_in_struct)
|
|
|
|
{
|
2016-12-01 03:39:30 +00:00
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
struct test_struct {
|
|
|
|
address addr;
|
|
|
|
mapping (uint64 => mapping (bytes32 => uint)) complex_mapping;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2014-10-07 16:25:04 +00:00
|
|
|
}
|
|
|
|
|
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
|
|
|
}
|
|
|
|
|
2015-03-01 03:43:42 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(variable_definition_in_mapping)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
function fun() {
|
2015-03-12 11:25:07 +00:00
|
|
|
mapping(var=>bytes32) d;
|
2015-03-01 03:43:42 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
2016-12-06 09:35:56 +00:00
|
|
|
CHECK_PARSE_ERROR(text, "Expected elementary type name for mapping key type");
|
2015-03-01 03:43:42 +00:00
|
|
|
}
|
|
|
|
|
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
|
|
|
}
|
|
|
|
|
2015-02-08 11:23:17 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(exp_expression)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
function fun(uint256 a) {
|
|
|
|
uint256 x = 3 ** a;
|
|
|
|
}
|
2016-12-01 03:39:30 +00:00
|
|
|
}
|
|
|
|
)";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2015-02-08 11:23:17 +00:00
|
|
|
}
|
|
|
|
|
2014-10-09 13:57:49 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(while_loop)
|
|
|
|
{
|
2016-12-01 03:39:30 +00:00
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
function fun(uint256 a) {
|
|
|
|
while (true) { uint256 x = 1; break; continue; } x = 9;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2014-10-09 13:57:49 +00:00
|
|
|
}
|
|
|
|
|
2014-12-15 16:54:29 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(for_loop_vardef_initexpr)
|
2014-12-15 16:45:18 +00:00
|
|
|
{
|
2016-12-01 03:39:30 +00:00
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
function fun(uint256 a) {
|
|
|
|
for (uint256 i = 0; i < 10; i++) {
|
|
|
|
uint256 x = i; break; continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2014-12-15 16:54:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(for_loop_simple_initexpr)
|
|
|
|
{
|
2016-12-01 03:39:30 +00:00
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
function fun(uint256 a) {
|
|
|
|
uint256 i =0;
|
|
|
|
for (i = 0; i < 10; i++) {
|
|
|
|
uint256 x = i; break; continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2014-12-15 16:45:18 +00:00
|
|
|
}
|
|
|
|
|
2014-12-16 08:51:51 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(for_loop_simple_noexpr)
|
|
|
|
{
|
2016-12-01 03:39:30 +00:00
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
function fun(uint256 a) {
|
|
|
|
uint256 i =0;
|
|
|
|
for (;;) {
|
|
|
|
uint256 x = i; break; continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2014-12-16 08:51:51 +00:00
|
|
|
}
|
|
|
|
|
2014-12-16 13:46:17 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(for_loop_single_stmt_body)
|
|
|
|
{
|
2016-12-01 03:39:30 +00:00
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
function fun(uint256 a) {
|
|
|
|
uint256 i = 0;
|
|
|
|
for (i = 0; i < 10; i++)
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2014-12-16 13:46:17 +00:00
|
|
|
}
|
|
|
|
|
2014-10-09 13:57:49 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(if_statement)
|
|
|
|
{
|
2016-12-01 03:39:30 +00:00
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
function fun(uint256 a) {
|
|
|
|
if (a >= 8) { return 2; } else { var b = 7; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2014-10-09 13:57:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(else_if_statement)
|
|
|
|
{
|
2016-12-01 03:39:30 +00:00
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
function fun(uint256 a) returns (address b) {
|
|
|
|
if (a < 0) b = 0x67; else if (a == 0) b = 0x12; else b = 0x78;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
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-01-21 10:16:18 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(placeholder_in_function_context)
|
|
|
|
{
|
2016-12-01 03:39:30 +00:00
|
|
|
char const* text = R"(
|
|
|
|
contract c {
|
|
|
|
function fun() returns (uint r) {
|
|
|
|
var _ = 8;
|
|
|
|
return _ + 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2015-01-21 10:16:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(modifier)
|
|
|
|
{
|
2016-12-01 03:39:30 +00:00
|
|
|
char const* text = R"(
|
|
|
|
contract c {
|
|
|
|
modifier mod { if (msg.sender == 0) _; }
|
|
|
|
}
|
|
|
|
)";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2015-01-21 10:16:18 +00:00
|
|
|
}
|
|
|
|
|
2016-09-05 12:54:50 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(modifier_without_semicolon)
|
|
|
|
{
|
2016-12-01 03:39:30 +00:00
|
|
|
char const* text = R"(
|
|
|
|
contract c {
|
|
|
|
modifier mod { if (msg.sender == 0) _ }
|
|
|
|
}
|
|
|
|
)";
|
2016-12-06 09:35:56 +00:00
|
|
|
CHECK_PARSE_ERROR(text, "Expected token Semicolon got");
|
2016-09-05 12:54:50 +00:00
|
|
|
}
|
|
|
|
|
2015-01-21 10:16:18 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(modifier_arguments)
|
|
|
|
{
|
2016-12-01 03:39:30 +00:00
|
|
|
char const* text = R"(
|
|
|
|
contract c {
|
2016-12-06 09:35:56 +00:00
|
|
|
modifier mod(address a) { if (msg.sender == a) _; }
|
2016-12-01 03:39:30 +00:00
|
|
|
}
|
|
|
|
)";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2015-01-21 10:16:18 +00:00
|
|
|
}
|
|
|
|
|
2015-01-22 00:02:38 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(modifier_invocation)
|
|
|
|
{
|
2016-12-01 03:39:30 +00:00
|
|
|
char const* text = R"(
|
|
|
|
contract c {
|
|
|
|
modifier mod1(uint a) { if (msg.sender == a) _; }
|
|
|
|
modifier mod2 { if (msg.sender == 2) _; }
|
|
|
|
function f() mod1(7) mod2 { }
|
|
|
|
}
|
|
|
|
)";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2015-01-22 00:02:38 +00:00
|
|
|
}
|
|
|
|
|
2015-01-29 21:50:20 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(fallback_function)
|
|
|
|
{
|
2016-12-01 03:39:30 +00:00
|
|
|
char const* text = R"(
|
|
|
|
contract c {
|
|
|
|
function() { }
|
|
|
|
}
|
|
|
|
)";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2015-01-29 21:50:20 +00:00
|
|
|
}
|
|
|
|
|
2015-01-29 13:35:28 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(event)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract c {
|
|
|
|
event e();
|
|
|
|
})";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2015-01-29 13:35:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(event_arguments)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract c {
|
2015-03-12 12:39:12 +00:00
|
|
|
event e(uint a, bytes32 s);
|
2015-01-29 13:35:28 +00:00
|
|
|
})";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2015-01-29 13:35:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(event_arguments_indexed)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract c {
|
2015-03-12 12:39:12 +00:00
|
|
|
event e(uint a, bytes32 indexed s, bool indexed b);
|
2015-01-29 13:35:28 +00:00
|
|
|
})";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2015-01-29 13:35:28 +00:00
|
|
|
}
|
|
|
|
|
2017-11-09 03:02:39 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(event_with_no_argument_list_fails)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract c {
|
|
|
|
event e;
|
|
|
|
}
|
|
|
|
)";
|
|
|
|
CHECK_PARSE_ERROR(text, "Expected token LParen got 'Semicolon'");
|
|
|
|
}
|
|
|
|
|
2015-02-02 16:24:09 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(visibility_specifiers)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract c {
|
|
|
|
uint private a;
|
2015-02-22 18:37:54 +00:00
|
|
|
uint internal b;
|
2015-02-02 16:24:09 +00:00
|
|
|
uint public c;
|
|
|
|
uint d;
|
|
|
|
function f() {}
|
|
|
|
function f_priv() private {}
|
|
|
|
function f_public() public {}
|
2015-02-22 18:37:54 +00:00
|
|
|
function f_internal() internal {}
|
2015-02-02 16:24:09 +00:00
|
|
|
})";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
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;
|
2015-02-02 16:24:09 +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 {}
|
|
|
|
})";
|
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
|
|
|
}
|
|
|
|
|
2017-08-09 19:58:57 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(multiple_statemutability_specifiers)
|
2017-06-17 15:21:46 +00:00
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract c {
|
|
|
|
function f() payable payable {}
|
|
|
|
})";
|
2017-08-09 19:58:57 +00:00
|
|
|
CHECK_PARSE_ERROR(text, "State mutability already specified as \"payable\".");
|
|
|
|
text = R"(
|
2017-06-17 15:21:46 +00:00
|
|
|
contract c {
|
|
|
|
function f() constant constant {}
|
|
|
|
})";
|
2017-08-09 19:58:57 +00:00
|
|
|
CHECK_PARSE_ERROR(text, "State mutability already specified as \"view\".");
|
2017-08-21 22:21:27 +00:00
|
|
|
text = R"(
|
|
|
|
contract c {
|
|
|
|
function f() constant view {}
|
|
|
|
})";
|
|
|
|
CHECK_PARSE_ERROR(text, "State mutability already specified as \"view\".");
|
2017-08-09 19:58:57 +00:00
|
|
|
text = R"(
|
|
|
|
contract c {
|
|
|
|
function f() payable constant {}
|
|
|
|
})";
|
|
|
|
CHECK_PARSE_ERROR(text, "State mutability already specified as \"payable\".");
|
2017-08-15 01:26:24 +00:00
|
|
|
text = R"(
|
|
|
|
contract c {
|
|
|
|
function f() pure payable {}
|
|
|
|
})";
|
|
|
|
CHECK_PARSE_ERROR(text, "State mutability already specified as \"pure\".");
|
|
|
|
text = R"(
|
|
|
|
contract c {
|
|
|
|
function f() pure constant {}
|
|
|
|
})";
|
|
|
|
CHECK_PARSE_ERROR(text, "State mutability already specified as \"pure\".");
|
2017-06-17 15:21:46 +00:00
|
|
|
}
|
|
|
|
|
2015-02-04 21:02:35 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(literal_constants_with_ether_subdenominations)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract c {
|
|
|
|
function c ()
|
|
|
|
{
|
|
|
|
a = 1 wei;
|
|
|
|
b = 2 szabo;
|
|
|
|
c = 3 finney;
|
|
|
|
b = 4 ether;
|
|
|
|
}
|
|
|
|
uint256 a;
|
|
|
|
uint256 b;
|
|
|
|
uint256 c;
|
|
|
|
uint256 d;
|
|
|
|
})";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2015-02-04 21:02:35 +00:00
|
|
|
}
|
|
|
|
|
2015-02-06 12:38:10 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(literal_constants_with_ether_subdenominations_in_expressions)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract c {
|
|
|
|
function c ()
|
|
|
|
{
|
|
|
|
a = 1 wei * 100 wei + 7 szabo - 3;
|
|
|
|
}
|
|
|
|
uint256 a;
|
|
|
|
})";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2015-02-06 12:38:10 +00:00
|
|
|
}
|
|
|
|
|
2015-02-13 21:52:04 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(enum_valid_declaration)
|
2015-02-10 13:51:40 +00:00
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract c {
|
2015-02-13 22:47:55 +00:00
|
|
|
enum validEnum { Value1, Value2, Value3, Value4 }
|
2015-02-10 13:51:40 +00:00
|
|
|
function c ()
|
|
|
|
{
|
2015-02-13 21:52:04 +00:00
|
|
|
a = foo.Value3;
|
2015-02-10 13:51:40 +00:00
|
|
|
}
|
|
|
|
uint256 a;
|
|
|
|
})";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2015-02-10 13:51:40 +00:00
|
|
|
}
|
|
|
|
|
2015-02-11 15:37:46 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(empty_enum_declaration)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract c {
|
2015-02-13 22:47:55 +00:00
|
|
|
enum foo { }
|
2015-02-11 15:37:46 +00:00
|
|
|
})";
|
2016-12-06 09:35:56 +00:00
|
|
|
CHECK_PARSE_ERROR(text, "enum with no members is not allowed");
|
2015-02-11 15:37:46 +00:00
|
|
|
}
|
|
|
|
|
2015-02-11 16:23:13 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(malformed_enum_declaration)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract c {
|
2015-02-13 22:47:55 +00:00
|
|
|
enum foo { WARNING,}
|
2015-02-11 16:23:13 +00:00
|
|
|
})";
|
2016-12-06 09:35:56 +00:00
|
|
|
CHECK_PARSE_ERROR(text, "Expected Identifier after");
|
2015-02-11 16:23:13 +00:00
|
|
|
}
|
|
|
|
|
2015-02-13 23:43:02 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(external_function)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract c {
|
|
|
|
function x() external {}
|
|
|
|
})";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2015-02-13 23:43:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(external_variable)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract c {
|
|
|
|
uint external x;
|
|
|
|
})";
|
2016-12-06 09:35:56 +00:00
|
|
|
CHECK_PARSE_ERROR(text, "Expected identifier");
|
2015-02-13 23:43:02 +00:00
|
|
|
}
|
|
|
|
|
2015-10-15 14:27:26 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(arrays_in_storage)
|
2015-02-20 14:52:30 +00:00
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract c {
|
|
|
|
uint[10] a;
|
|
|
|
uint[] a2;
|
|
|
|
struct x { uint[2**20] b; y[0] c; }
|
|
|
|
struct y { uint d; mapping(uint=>x)[] e; }
|
|
|
|
})";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2015-02-20 14:52:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(arrays_in_events)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract c {
|
2015-03-12 12:39:12 +00:00
|
|
|
event e(uint[10] a, bytes7[8] indexed b, c[3] x);
|
2015-02-20 14:52:30 +00:00
|
|
|
})";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2015-02-20 14:52:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(arrays_in_expressions)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract c {
|
|
|
|
function f() { c[10] a = 7; uint8[10 * 2] x; }
|
|
|
|
})";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2015-02-20 14:52:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(multi_arrays)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract c {
|
|
|
|
mapping(uint => mapping(uint => int8)[8][][9])[] x;
|
|
|
|
})";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2015-02-20 14:52:30 +00:00
|
|
|
}
|
|
|
|
|
2015-03-13 17:16:04 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(constant_is_keyword)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract Foo {
|
|
|
|
uint constant = 4;
|
|
|
|
})";
|
2016-12-06 09:35:56 +00:00
|
|
|
CHECK_PARSE_ERROR(text, "Expected identifier");
|
2015-03-13 17:16:04 +00:00
|
|
|
}
|
|
|
|
|
2018-03-02 09:36:04 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(keyword_is_reserved)
|
|
|
|
{
|
|
|
|
auto keywords = {
|
|
|
|
"abstract",
|
|
|
|
"after",
|
|
|
|
"case",
|
|
|
|
"catch",
|
|
|
|
"default",
|
|
|
|
"final",
|
|
|
|
"in",
|
|
|
|
"inline",
|
|
|
|
"let",
|
|
|
|
"match",
|
|
|
|
"null",
|
|
|
|
"of",
|
|
|
|
"relocatable",
|
|
|
|
"static",
|
|
|
|
"switch",
|
|
|
|
"try",
|
|
|
|
"type",
|
|
|
|
"typeof"
|
|
|
|
};
|
|
|
|
|
|
|
|
for (const auto& keyword: keywords)
|
|
|
|
{
|
|
|
|
auto text = std::string("contract ") + keyword + " {}";
|
|
|
|
CHECK_PARSE_ERROR(text.c_str(), "Expected identifier");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-04-15 22:06:57 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(var_array)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract Foo {
|
|
|
|
function f() { var[] a; }
|
|
|
|
})";
|
2016-12-06 09:35:56 +00:00
|
|
|
CHECK_PARSE_ERROR(text, "Expected identifier");
|
2015-04-15 22:06:57 +00:00
|
|
|
}
|
|
|
|
|
2015-06-05 09:07:50 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(location_specifiers_for_params)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract Foo {
|
|
|
|
function f(uint[] storage constant x, uint[] memory y) { }
|
|
|
|
}
|
|
|
|
)";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2015-06-05 09:07:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(location_specifiers_for_locals)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract Foo {
|
|
|
|
function f() {
|
|
|
|
uint[] storage x;
|
|
|
|
uint[] memory y;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2015-06-05 09:07:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(location_specifiers_for_state)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract Foo {
|
|
|
|
uint[] memory x;
|
|
|
|
})";
|
2016-12-06 09:35:56 +00:00
|
|
|
CHECK_PARSE_ERROR(text, "Expected identifier");
|
2015-06-05 09:07:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(location_specifiers_with_var)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract Foo {
|
|
|
|
function f() { var memory x; }
|
|
|
|
})";
|
2016-12-06 09:35:56 +00:00
|
|
|
CHECK_PARSE_ERROR(text, "Location specifier needs explicit type name");
|
2015-06-05 09:07:50 +00:00
|
|
|
}
|
|
|
|
|
2015-09-10 12:26:34 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(empty_comment)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
//
|
|
|
|
contract test
|
|
|
|
{}
|
|
|
|
)";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2015-09-10 12:26:34 +00:00
|
|
|
}
|
|
|
|
|
2016-11-30 16:28:07 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(comment_end_with_double_star)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract C1 {
|
|
|
|
/**
|
|
|
|
**/
|
|
|
|
}
|
|
|
|
contract C2 {}
|
|
|
|
)";
|
|
|
|
BOOST_CHECK(successParse(text));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-09-08 14:48:33 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(library_simple)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
library Lib {
|
|
|
|
function f() { }
|
|
|
|
}
|
|
|
|
)";
|
2015-10-14 18:37:41 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(local_const_variable)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract Foo {
|
|
|
|
function localConst() returns (uint ret)
|
|
|
|
{
|
|
|
|
uint constant local = 4;
|
|
|
|
return local;
|
|
|
|
}
|
|
|
|
})";
|
2016-12-06 09:35:56 +00:00
|
|
|
CHECK_PARSE_ERROR(text, "Expected token Semicolon");
|
2015-09-08 14:48:33 +00:00
|
|
|
}
|
|
|
|
|
2015-10-09 14:26:27 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(multi_variable_declaration)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
2015-10-12 21:02:35 +00:00
|
|
|
contract C {
|
2015-10-09 14:26:27 +00:00
|
|
|
function f() {
|
|
|
|
var (a,b,c) = g();
|
|
|
|
var (d) = 2;
|
|
|
|
var (,e) = 3;
|
|
|
|
var (f,) = 4;
|
|
|
|
var (x,,) = g();
|
|
|
|
var (,y,) = g();
|
2015-10-13 10:22:57 +00:00
|
|
|
var () = g();
|
|
|
|
var (,,) = g();
|
2015-10-09 14:26:27 +00:00
|
|
|
}
|
|
|
|
function g() returns (uint, uint, uint) {}
|
|
|
|
}
|
|
|
|
)";
|
2015-10-15 09:50:25 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2015-10-09 14:26:27 +00:00
|
|
|
}
|
|
|
|
|
2015-10-12 21:02:35 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(tuples)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract C {
|
|
|
|
function f() {
|
|
|
|
uint a = (1);
|
|
|
|
var (b,) = (1,);
|
|
|
|
var (c,d) = (1, 2 + a);
|
|
|
|
var (e,) = (1, 2, b);
|
2016-01-04 08:11:13 +00:00
|
|
|
(a) = 3;
|
2015-10-12 21:02:35 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
2015-10-15 16:14:14 +00:00
|
|
|
BOOST_CHECK(successParse(text));
|
2015-10-12 21:02:35 +00:00
|
|
|
}
|
|
|
|
|
2017-08-08 21:58:06 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(tuples_without_commas)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract C {
|
|
|
|
function f() {
|
|
|
|
var a = (2 2);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
|
|
|
CHECK_PARSE_ERROR(text, "Expected token Comma");
|
|
|
|
}
|
|
|
|
|
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));
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(from_is_not_keyword)
|
|
|
|
{
|
|
|
|
// "from" is not a keyword although it is used as a keyword in import directives.
|
|
|
|
char const* text = R"(
|
|
|
|
contract from {
|
|
|
|
}
|
|
|
|
)";
|
|
|
|
BOOST_CHECK(successParse(text));
|
|
|
|
}
|
|
|
|
|
2015-12-16 22:47:37 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(inline_array_declaration)
|
2015-12-15 16:57:57 +00:00
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract c {
|
2015-12-16 18:55:52 +00:00
|
|
|
uint[] a;
|
2015-12-16 22:47:37 +00:00
|
|
|
function f() returns (uint, uint) {
|
2015-12-16 18:55:52 +00:00
|
|
|
a = [1,2,3];
|
2015-12-16 22:47:37 +00:00
|
|
|
return (a[3], [2,3,4][0]);
|
2015-12-16 20:50:40 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
|
|
|
BOOST_CHECK(successParse(text));
|
|
|
|
}
|
|
|
|
|
2015-12-15 16:57:57 +00:00
|
|
|
|
2015-12-16 22:47:37 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(inline_array_empty_cells_check_lvalue)
|
2015-12-16 18:55:52 +00:00
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract c {
|
|
|
|
uint[] a;
|
2015-12-16 22:47:37 +00:00
|
|
|
function f() returns (uint) {
|
2015-12-16 18:55:52 +00:00
|
|
|
a = [,2,3];
|
2015-12-16 22:47:37 +00:00
|
|
|
return (a[0]);
|
2015-12-16 20:50:40 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
2016-12-06 09:35:56 +00:00
|
|
|
CHECK_PARSE_ERROR(text, "Expected expression");
|
2015-12-16 20:50:40 +00:00
|
|
|
}
|
|
|
|
|
2015-12-16 22:47:37 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(inline_array_empty_cells_check_without_lvalue)
|
2015-12-16 20:50:40 +00:00
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract c {
|
|
|
|
uint[] a;
|
|
|
|
function f() returns (uint, uint) {
|
2015-12-16 22:47:37 +00:00
|
|
|
return ([3, ,4][0]);
|
2015-12-16 18:55:52 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
2016-12-06 09:35:56 +00:00
|
|
|
CHECK_PARSE_ERROR(text, "Expected expression");
|
2015-12-16 18:55:52 +00:00
|
|
|
}
|
|
|
|
|
2015-12-22 16:49:09 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(conditional_true_false_literal)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract A {
|
|
|
|
function f() {
|
|
|
|
uint x = true ? 1 : 0;
|
|
|
|
uint y = false ? 0 : 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
|
|
|
BOOST_CHECK(successParse(text));
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(conditional_with_constants)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract A {
|
|
|
|
function f() {
|
|
|
|
uint x = 3 > 0 ? 3 : 0;
|
|
|
|
uint y = (3 > 0) ? 3 : 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
|
|
|
BOOST_CHECK(successParse(text));
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(conditional_with_variables)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract A {
|
|
|
|
function f() {
|
|
|
|
uint x = 3;
|
|
|
|
uint y = 1;
|
|
|
|
uint z = (x > y) ? x : y;
|
|
|
|
uint w = x > y ? x : y;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
|
|
|
BOOST_CHECK(successParse(text));
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(conditional_multiple)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract A {
|
|
|
|
function f() {
|
|
|
|
uint x = 3 < 0 ? 2 > 1 ? 2 : 1 : 7 > 2 ? 7 : 6;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
|
|
|
BOOST_CHECK(successParse(text));
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(conditional_with_assignment)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract A {
|
|
|
|
function f() {
|
|
|
|
uint y = 1;
|
|
|
|
uint x = 3 < 0 ? x = 3 : 6;
|
2015-12-23 12:16:50 +00:00
|
|
|
true ? x = 3 : 4;
|
2015-12-22 16:49:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
|
|
|
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");
|
|
|
|
}
|
|
|
|
|
2016-02-18 22:39:11 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(declaring_fixed_and_ufixed_variables)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract A {
|
|
|
|
fixed40x40 storeMe;
|
|
|
|
function f(ufixed x, fixed32x32 y) {
|
|
|
|
ufixed8x8 a;
|
|
|
|
fixed b;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
|
|
|
BOOST_CHECK(successParse(text));
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(declaring_fixed_literal_variables)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract A {
|
|
|
|
fixed40x40 pi = 3.14;
|
|
|
|
}
|
|
|
|
)";
|
|
|
|
BOOST_CHECK(successParse(text));
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(no_double_radix_in_fixed_literal)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract A {
|
|
|
|
fixed40x40 pi = 3.14.15;
|
|
|
|
}
|
|
|
|
)";
|
2016-12-06 09:35:56 +00:00
|
|
|
CHECK_PARSE_ERROR(text, "Expected token Semicolon");
|
2016-02-18 22:39:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(invalid_fixed_conversion_leading_zeroes_check)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
function f() {
|
|
|
|
fixed a = 1.0x2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
2016-12-06 09:35:56 +00:00
|
|
|
CHECK_PARSE_ERROR(text, "Expected primary expression");
|
2016-02-18 22:39:11 +00:00
|
|
|
}
|
|
|
|
|
2016-08-26 18:37:10 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(payable_accessor)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
uint payable x;
|
|
|
|
}
|
|
|
|
)";
|
2016-12-06 09:35:56 +00:00
|
|
|
CHECK_PARSE_ERROR(text, "Expected identifier");
|
2016-08-26 18:37:10 +00:00
|
|
|
}
|
|
|
|
|
2016-09-27 19:37:32 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(function_type_in_expression)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
function f(uint x, uint y) returns (uint a) {}
|
|
|
|
function g() {
|
|
|
|
function (uint, uint) internal returns (uint) f1 = f;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
|
|
|
BOOST_CHECK(successParse(text));
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(function_type_as_storage_variable)
|
|
|
|
{
|
2016-10-10 21:06:44 +00:00
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
function (uint, uint) internal returns (uint) f1;
|
|
|
|
}
|
|
|
|
)";
|
|
|
|
BOOST_CHECK(successParse(text));
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(function_type_as_storage_variable_with_modifiers)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
function (uint, uint) modifier1() returns (uint) f1;
|
|
|
|
}
|
|
|
|
)";
|
2016-12-06 09:35:56 +00:00
|
|
|
CHECK_PARSE_ERROR(text, "Expected token LBrace");
|
2016-10-10 21:06:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(function_type_as_storage_variable_with_assignment)
|
|
|
|
{
|
2016-09-27 19:37:32 +00:00
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
function f(uint x, uint y) returns (uint a) {}
|
|
|
|
function (uint, uint) internal returns (uint) f1 = f;
|
|
|
|
}
|
|
|
|
)";
|
|
|
|
BOOST_CHECK(successParse(text));
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(function_type_in_struct)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
struct S {
|
|
|
|
function (uint x, uint y) internal returns (uint a) f;
|
|
|
|
function (uint, uint) external returns (uint) g;
|
|
|
|
uint d;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
|
|
|
BOOST_CHECK(successParse(text));
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOST_AUTO_TEST_CASE(function_type_as_parameter)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
function f(function(uint) external returns (uint) g) internal returns (uint a) {
|
|
|
|
return g(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
|
|
|
BOOST_CHECK(successParse(text));
|
|
|
|
}
|
|
|
|
|
2016-10-14 15:32:13 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(calling_function)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
function f() {
|
|
|
|
function() returns(function() returns(function() returns(function() returns(uint)))) x;
|
|
|
|
uint y;
|
|
|
|
y = x()()()();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
|
|
|
BOOST_CHECK(successParse(text));
|
|
|
|
}
|
|
|
|
|
2016-10-19 16:43:42 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(mapping_and_array_of_functions)
|
2016-10-14 15:46:48 +00:00
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
2016-10-19 16:43:42 +00:00
|
|
|
mapping (address => function() internal returns (uint)) a;
|
|
|
|
mapping (address => function() external) b;
|
|
|
|
mapping (address => function() external[]) c;
|
|
|
|
function() external[] d;
|
2016-10-14 15:46:48 +00:00
|
|
|
}
|
|
|
|
)";
|
|
|
|
BOOST_CHECK(successParse(text));
|
|
|
|
}
|
|
|
|
|
2016-11-11 14:23:50 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(function_type_state_variable)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
function() x;
|
|
|
|
function() y = x;
|
|
|
|
}
|
|
|
|
)";
|
|
|
|
BOOST_CHECK(successParse(text));
|
|
|
|
}
|
|
|
|
|
2017-02-14 14:58:04 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(scientific_notation)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract test {
|
|
|
|
uint256 a = 2e10;
|
|
|
|
uint256 b = 2E10;
|
|
|
|
uint256 c = 200e-2;
|
|
|
|
uint256 d = 2E10 wei;
|
2017-03-15 12:21:58 +00:00
|
|
|
uint256 e = 2.5e10;
|
2017-02-14 14:58:04 +00:00
|
|
|
}
|
|
|
|
)";
|
|
|
|
BOOST_CHECK(successParse(text));
|
|
|
|
}
|
|
|
|
|
2017-02-11 21:17:24 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(interface)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
interface Interface {
|
|
|
|
function f();
|
|
|
|
}
|
|
|
|
)";
|
|
|
|
BOOST_CHECK(successParse(text));
|
|
|
|
}
|
2017-02-14 14:58:04 +00:00
|
|
|
|
2017-08-28 12:50:18 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(newInvalidTypeName)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract C {
|
|
|
|
function f() {
|
|
|
|
new var;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
|
|
|
CHECK_PARSE_ERROR(text, "Expected explicit type name");
|
|
|
|
}
|
|
|
|
|
2018-02-22 14:06:14 +00:00
|
|
|
BOOST_AUTO_TEST_CASE(emitWithoutEvent)
|
|
|
|
{
|
|
|
|
char const* text = R"(
|
|
|
|
contract C {
|
|
|
|
event A();
|
|
|
|
function f() {
|
|
|
|
emit A;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
|
|
|
CHECK_PARSE_ERROR(text, "Expected token LParen got 'Semicolon'");
|
|
|
|
}
|
|
|
|
|
2014-10-07 16:25:04 +00:00
|
|
|
BOOST_AUTO_TEST_SUITE_END()
|
|
|
|
|
2014-10-16 12:08:54 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
} // end namespaces
|