Provide locations for docstring parsing errors.

This commit is contained in:
chriseth 2021-03-01 16:54:21 +01:00
parent 44493ad428
commit f4050e81c6
9 changed files with 34 additions and 37 deletions

View File

@ -6,6 +6,7 @@ Language Features:
Compiler Features: Compiler Features:
* AST: Export NatSpec comments above each statement as their documentation. * AST: Export NatSpec comments above each statement as their documentation.
* Inline Assembly: Do not warn anymore about variables or functions being shadowed by EVM opcodes. * Inline Assembly: Do not warn anymore about variables or functions being shadowed by EVM opcodes.
* NatSpec: Provide source locations for parsing errors.
* Optimizer: Simple inlining when jumping to small blocks that jump again after a few side-effect free opcodes. * Optimizer: Simple inlining when jumping to small blocks that jump again after a few side-effect free opcodes.

View File

@ -234,16 +234,6 @@ void ErrorReporter::fatalTypeError(ErrorId _error, SourceLocation const& _locati
); );
} }
void ErrorReporter::docstringParsingError(ErrorId _error, string const& _description)
{
error(
_error,
Error::Type::DocstringParsingError,
SourceLocation(),
_description
);
}
void ErrorReporter::docstringParsingError(ErrorId _error, SourceLocation const& _location, string const& _description) void ErrorReporter::docstringParsingError(ErrorId _error, SourceLocation const& _location, string const& _description)
{ {
error( error(

View File

@ -112,7 +112,6 @@ public:
void fatalTypeError(ErrorId _error, SourceLocation const& _location, std::string const& _description); void fatalTypeError(ErrorId _error, SourceLocation const& _location, std::string const& _description);
void fatalTypeError(ErrorId _error, SourceLocation const& _location, SecondarySourceLocation const& _secondLocation, std::string const& _description); void fatalTypeError(ErrorId _error, SourceLocation const& _location, SecondarySourceLocation const& _secondLocation, std::string const& _description);
void docstringParsingError(ErrorId _error, std::string const& _description);
void docstringParsingError(ErrorId _error, SourceLocation const& _location, std::string const& _description); void docstringParsingError(ErrorId _error, SourceLocation const& _location, std::string const& _description);
ErrorList const& errors() const; ErrorList const& errors() const;

View File

@ -25,7 +25,6 @@
#include <libsolidity/analysis/DocStringAnalyser.h> #include <libsolidity/analysis/DocStringAnalyser.h>
#include <libsolidity/ast/AST.h> #include <libsolidity/ast/AST.h>
#include <libsolidity/parsing/DocStringParser.h>
#include <liblangutil/ErrorReporter.h> #include <liblangutil/ErrorReporter.h>
using namespace std; using namespace std;

View File

@ -147,12 +147,10 @@ void DocStringTagParser::parseDocStrings(
string const& _nodeName string const& _nodeName
) )
{ {
DocStringParser parser; if (!_node.documentation())
if (_node.documentation() && !_node.documentation()->text()->empty()) return;
{
parser.parse(*_node.documentation()->text(), m_errorReporter); _annotation.docTags = DocStringParser{*_node.documentation(), m_errorReporter}.parse();
_annotation.docTags = parser.tags();
}
size_t returnTagsVisited = 0; size_t returnTagsVisited = 0;
for (auto const& docTag: _annotation.docTags) for (auto const& docTag: _annotation.docTags)

View File

@ -18,6 +18,8 @@
#include <libsolidity/parsing/DocStringParser.h> #include <libsolidity/parsing/DocStringParser.h>
#include <libsolidity/ast/AST.h>
#include <liblangutil/Common.h> #include <liblangutil/Common.h>
#include <liblangutil/ErrorReporter.h> #include <liblangutil/ErrorReporter.h>
#include <liblangutil/Exceptions.h> #include <liblangutil/Exceptions.h>
@ -78,25 +80,26 @@ string::const_iterator skipWhitespace(
} }
void DocStringParser::parse(string const& _docString, ErrorReporter& _errorReporter) multimap<string, DocTag> DocStringParser::parse()
{ {
m_errorReporter = &_errorReporter;
m_lastTag = nullptr; m_lastTag = nullptr;
m_docTags = {};
auto currPos = _docString.begin(); solAssert(m_node.text(), "");
auto end = _docString.end(); iter currPos = m_node.text()->begin();
iter end = m_node.text()->end();
while (currPos != end) while (currPos != end)
{ {
auto tagPos = find(currPos, end, '@'); iter tagPos = find(currPos, end, '@');
auto nlPos = find(currPos, end, '\n'); iter nlPos = find(currPos, end, '\n');
if (tagPos != end && tagPos < nlPos) if (tagPos != end && tagPos < nlPos)
{ {
// we found a tag // we found a tag
auto tagNameEndPos = firstWhitespaceOrNewline(tagPos, end); iter tagNameEndPos = firstWhitespaceOrNewline(tagPos, end);
auto tagName = string(tagPos + 1, tagNameEndPos); string tagName{tagPos + 1, tagNameEndPos};
auto tagDataPos = (tagNameEndPos != end) ? tagNameEndPos + 1 : tagNameEndPos; iter tagDataPos = (tagNameEndPos != end) ? tagNameEndPos + 1 : tagNameEndPos;
currPos = parseDocTag(tagDataPos, end, tagName); currPos = parseDocTag(tagDataPos, end, tagName);
} }
else if (!!m_lastTag) // continuation of the previous tag else if (!!m_lastTag) // continuation of the previous tag
@ -104,7 +107,7 @@ void DocStringParser::parse(string const& _docString, ErrorReporter& _errorRepor
else if (currPos != end) else if (currPos != end)
{ {
// if it begins without a tag then consider it as @notice // if it begins without a tag then consider it as @notice
if (currPos == _docString.begin()) if (currPos == m_node.text()->begin())
{ {
currPos = parseDocTag(currPos, end, "notice"); currPos = parseDocTag(currPos, end, "notice");
continue; continue;
@ -115,6 +118,7 @@ void DocStringParser::parse(string const& _docString, ErrorReporter& _errorRepor
currPos = nlPos + 1; currPos = nlPos + 1;
} }
} }
return move(m_docTags);
} }
DocStringParser::iter DocStringParser::parseDocTagLine(iter _pos, iter _end, bool _appending) DocStringParser::iter DocStringParser::parseDocTagLine(iter _pos, iter _end, bool _appending)
@ -135,7 +139,7 @@ DocStringParser::iter DocStringParser::parseDocTagParam(iter _pos, iter _end)
auto nameStartPos = skipWhitespace(_pos, _end); auto nameStartPos = skipWhitespace(_pos, _end);
if (nameStartPos == _end) if (nameStartPos == _end)
{ {
m_errorReporter->docstringParsingError(3335_error, "No param name given"); m_errorReporter.docstringParsingError(3335_error, m_node.location(), "No param name given");
return _end; return _end;
} }
auto nameEndPos = firstNonIdentifier(nameStartPos, _end); auto nameEndPos = firstNonIdentifier(nameStartPos, _end);
@ -146,7 +150,7 @@ DocStringParser::iter DocStringParser::parseDocTagParam(iter _pos, iter _end)
if (descStartPos == nlPos) if (descStartPos == nlPos)
{ {
m_errorReporter->docstringParsingError(9942_error, "No description given for param " + paramName); m_errorReporter.docstringParsingError(9942_error, m_node.location(), "No description given for param " + paramName);
return _end; return _end;
} }

View File

@ -24,6 +24,7 @@
#pragma once #pragma once
#include <libsolidity/ast/ASTAnnotations.h> #include <libsolidity/ast/ASTAnnotations.h>
#include <liblangutil/SourceLocation.h>
#include <string> #include <string>
namespace solidity::langutil namespace solidity::langutil
@ -34,13 +35,17 @@ class ErrorReporter;
namespace solidity::frontend namespace solidity::frontend
{ {
class StructurallyDocumented;
class DocStringParser class DocStringParser
{ {
public: public:
/// Parse the given @a _docString and stores the parsed components internally. /// @param _documentedNode the node whose documentation is parsed.
void parse(std::string const& _docString, langutil::ErrorReporter& _errorReporter); DocStringParser(StructuredDocumentation const& _documentedNode, langutil::ErrorReporter& _errorReporter):
m_node(_documentedNode),
std::multimap<std::string, DocTag> const& tags() const { return m_docTags; } m_errorReporter(_errorReporter)
{}
std::multimap<std::string, DocTag> parse();
private: private:
using iter = std::string::const_iterator; using iter = std::string::const_iterator;
@ -58,10 +63,11 @@ private:
/// Creates and inserts a new tag and adjusts m_lastTag. /// Creates and inserts a new tag and adjusts m_lastTag.
void newTag(std::string const& _tagName); void newTag(std::string const& _tagName);
StructuredDocumentation const& m_node;
langutil::ErrorReporter& m_errorReporter;
/// Mapping tag name -> content. /// Mapping tag name -> content.
std::multimap<std::string, DocTag> m_docTags; std::multimap<std::string, DocTag> m_docTags;
DocTag* m_lastTag = nullptr; DocTag* m_lastTag = nullptr;
langutil::ErrorReporter* m_errorReporter = nullptr;
}; };
} }

View File

@ -3,4 +3,4 @@ contract C {
function vote(uint id) public {} function vote(uint id) public {}
} }
// ---- // ----
// DocstringParsingError 9942: No description given for param id // DocstringParsingError 9942: (17-30): No description given for param id

View File

@ -3,4 +3,4 @@ abstract contract C {
function vote(uint id) public {} function vote(uint id) public {}
} }
// ---- // ----
// DocstringParsingError 3335: No param name given // DocstringParsingError 3335: (26-36): No param name given