Require modifiers to contain "_".

This commit is contained in:
chriseth 2016-08-06 15:08:06 +02:00
parent e3e4d84f33
commit 6df6728165
3 changed files with 39 additions and 1 deletions

View File

@ -40,13 +40,26 @@ void SyntaxChecker::syntaxError(SourceLocation const& _location, std::string con
m_errors.push_back(err); m_errors.push_back(err);
} }
bool SyntaxChecker::visit(ModifierDefinition const&)
{
m_placeholderFound = false;
return true;
}
void SyntaxChecker::endVisit(ModifierDefinition const& _modifier)
{
if (!m_placeholderFound)
syntaxError(_modifier.body().location(), "Modifier body does not contain '_'.");
m_placeholderFound = false;
}
bool SyntaxChecker::visit(WhileStatement const&) bool SyntaxChecker::visit(WhileStatement const&)
{ {
m_inLoopDepth++; m_inLoopDepth++;
return true; return true;
} }
void SyntaxChecker::endVisit(WhileStatement const&) void SyntaxChecker::endVisit(WhileStatement const& )
{ {
m_inLoopDepth--; m_inLoopDepth--;
} }
@ -78,3 +91,9 @@ bool SyntaxChecker::visit(Break const& _breakStatement)
return true; return true;
} }
bool SyntaxChecker::visit(const PlaceholderStatement&)
{
m_placeholderFound = true;
return true;
}

View File

@ -31,6 +31,7 @@ namespace solidity
/** /**
* The module that performs syntax analysis on the AST: * The module that performs syntax analysis on the AST:
* - whether continue/break is in a for/while loop. * - whether continue/break is in a for/while loop.
* - whether a modifier contains at least one '_'
*/ */
class SyntaxChecker: private ASTConstVisitor class SyntaxChecker: private ASTConstVisitor
{ {
@ -44,6 +45,9 @@ private:
/// Adds a new error to the list of errors. /// Adds a new error to the list of errors.
void syntaxError(SourceLocation const& _location, std::string const& _description); void syntaxError(SourceLocation const& _location, std::string const& _description);
virtual bool visit(ModifierDefinition const& _modifier) override;
virtual void endVisit(ModifierDefinition const& _modifier) override;
virtual bool visit(WhileStatement const& _whileStatement) override; virtual bool visit(WhileStatement const& _whileStatement) override;
virtual void endVisit(WhileStatement const& _whileStatement) override; virtual void endVisit(WhileStatement const& _whileStatement) override;
virtual bool visit(ForStatement const& _forStatement) override; virtual bool visit(ForStatement const& _forStatement) override;
@ -52,8 +56,13 @@ private:
virtual bool visit(Continue const& _continueStatement) override; virtual bool visit(Continue const& _continueStatement) override;
virtual bool visit(Break const& _breakStatement) override; virtual bool visit(Break const& _breakStatement) override;
virtual bool visit(PlaceholderStatement const& _placeholderStatement) override;
ErrorList& m_errors; ErrorList& m_errors;
/// Flag that indicates whether a function modifier actually contains '_'.
bool m_placeholderFound = false;
int m_inLoopDepth = 0; int m_inLoopDepth = 0;
}; };

View File

@ -3823,6 +3823,16 @@ BOOST_AUTO_TEST_CASE(unused_return_value_delegatecall)
BOOST_CHECK(expectError(text, true) == Error::Type::Warning); BOOST_CHECK(expectError(text, true) == Error::Type::Warning);
} }
BOOST_AUTO_TEST_CASE(modifier_without_underscore)
{
char const* text = R"(
contract test {
modifier m() {}
}
)";
BOOST_CHECK(expectError(text, true) == Error::Type::SyntaxError);
}
BOOST_AUTO_TEST_SUITE_END() BOOST_AUTO_TEST_SUITE_END()
} }