also add byte indices to diagnostic ranges
For tools wanting to read from the source file this makes it much easier to look up the code.
This commit is contained in:
parent
bad253bad5
commit
b115a6333a
|
@ -75,7 +75,8 @@ public:
|
||||||
// highlight on the whitespace between attribute and function name
|
// highlight on the whitespace between attribute and function name
|
||||||
auto tok = autoTokens[$ - 1];
|
auto tok = autoTokens[$ - 1];
|
||||||
auto whitespace = tok.column + (tok.text.length ? tok.text.length : str(tok.type).length);
|
auto whitespace = tok.column + (tok.text.length ? tok.text.length : str(tok.type).length);
|
||||||
addErrorMessage(tok.line, whitespace, whitespace + 1, KEY, MESSAGE_INSERT);
|
auto whitespaceIndex = tok.index + (tok.text.length ? tok.text.length : str(tok.type).length);
|
||||||
|
addErrorMessage([whitespaceIndex, whitespaceIndex + 1], tok.line, [whitespace, whitespace + 1], KEY, MESSAGE_INSERT);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
addErrorMessage(autoTokens, KEY, MESSAGE);
|
addErrorMessage(autoTokens, KEY, MESSAGE);
|
||||||
|
|
|
@ -13,6 +13,8 @@ struct Message
|
||||||
{
|
{
|
||||||
/// Name of the file where the warning was triggered.
|
/// Name of the file where the warning was triggered.
|
||||||
string fileName;
|
string fileName;
|
||||||
|
/// Byte index from start of file the warning was triggered.
|
||||||
|
size_t startIndex, endIndex;
|
||||||
/// Line number where the warning was triggered, 1-based.
|
/// Line number where the warning was triggered, 1-based.
|
||||||
size_t startLine, endLine;
|
size_t startLine, endLine;
|
||||||
/// Column number where the warning was triggered. (in bytes)
|
/// Column number where the warning was triggered. (in bytes)
|
||||||
|
@ -33,7 +35,11 @@ struct Message
|
||||||
static Diagnostic from(string fileName, const Token token, string message)
|
static Diagnostic from(string fileName, const Token token, string message)
|
||||||
{
|
{
|
||||||
auto text = token.text.length ? token.text : str(token.type);
|
auto text = token.text.length ? token.text : str(token.type);
|
||||||
return from(fileName, token.line, token.column, token.column + text.length, message);
|
return from(fileName,
|
||||||
|
[token.index, token.index + text.length],
|
||||||
|
token.line,
|
||||||
|
[token.column, token.column + text.length],
|
||||||
|
message);
|
||||||
}
|
}
|
||||||
|
|
||||||
static Diagnostic from(string fileName, const Token[] tokens, string message)
|
static Diagnostic from(string fileName, const Token[] tokens, string message)
|
||||||
|
@ -41,17 +47,21 @@ struct Message
|
||||||
auto start = tokens.length ? tokens[0] : Token.init;
|
auto start = tokens.length ? tokens[0] : Token.init;
|
||||||
auto end = tokens.length ? tokens[$ - 1] : Token.init;
|
auto end = tokens.length ? tokens[$ - 1] : Token.init;
|
||||||
auto endText = end.text.length ? end.text : str(end.type);
|
auto endText = end.text.length ? end.text : str(end.type);
|
||||||
return from(fileName, start.line, end.line, start.column, end.column + endText.length, message);
|
return from(fileName,
|
||||||
|
[start.index, end.index + endText.length],
|
||||||
|
[start.line, end.line],
|
||||||
|
[start.column, end.column + endText.length],
|
||||||
|
message);
|
||||||
}
|
}
|
||||||
|
|
||||||
static Diagnostic from(string fileName, size_t line, size_t startColumn, size_t endColumn, string message)
|
static Diagnostic from(string fileName, size_t[2] index, size_t line, size_t[2] columns, string message)
|
||||||
{
|
{
|
||||||
return Message.Diagnostic(fileName, line, line, startColumn, endColumn, message);
|
return Message.Diagnostic(fileName, index[0], index[1], line, line, columns[0], columns[1], message);
|
||||||
}
|
}
|
||||||
|
|
||||||
static Diagnostic from(string fileName, size_t startLine, size_t endLine, size_t startColumn, size_t endColumn, string message)
|
static Diagnostic from(string fileName, size_t[2] index, size_t[2] lines, size_t[2] columns, string message)
|
||||||
{
|
{
|
||||||
return Message.Diagnostic(fileName, startLine, endLine, startColumn, endColumn, message);
|
return Message.Diagnostic(fileName, index[0], index[1], lines[0], lines[1], columns[0], columns[1], message);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -177,14 +187,14 @@ protected:
|
||||||
addErrorMessage(Message.Diagnostic.from(fileName, tokens, message), key);
|
addErrorMessage(Message.Diagnostic.from(fileName, tokens, message), key);
|
||||||
}
|
}
|
||||||
|
|
||||||
void addErrorMessage(size_t line, size_t startColumn, size_t endColumn, string key, string message)
|
void addErrorMessage(size_t[2] index, size_t line, size_t[2] columns, string key, string message)
|
||||||
{
|
{
|
||||||
addErrorMessage(line, line, startColumn, endColumn, key, message);
|
addErrorMessage(index, [line, line], columns, key, message);
|
||||||
}
|
}
|
||||||
|
|
||||||
void addErrorMessage(size_t startLine, size_t endLine, size_t startColumn, size_t endColumn, string key, string message)
|
void addErrorMessage(size_t[2] index, size_t[2] lines, size_t[2] columns, string key, string message)
|
||||||
{
|
{
|
||||||
auto d = Message.Diagnostic(fileName, startLine, endLine, startColumn, endColumn, message);
|
auto d = Message.Diagnostic.from(fileName, index, lines, columns, message);
|
||||||
_messages.insert(Message(d, key, getName()));
|
_messages.insert(Message(d, key, getName()));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -33,12 +33,12 @@ final class IfConstraintsIndentCheck : BaseAnalyzer
|
||||||
// t.line (unsigned) may be 0 if the token is uninitialized/broken, so don't subtract from it
|
// t.line (unsigned) may be 0 if the token is uninitialized/broken, so don't subtract from it
|
||||||
// equivalent to: firstSymbolAtLine.length < t.line - 1
|
// equivalent to: firstSymbolAtLine.length < t.line - 1
|
||||||
while (firstSymbolAtLine.length + 1 < t.line)
|
while (firstSymbolAtLine.length + 1 < t.line)
|
||||||
firstSymbolAtLine ~= Pos(1);
|
firstSymbolAtLine ~= Pos(1, t.index);
|
||||||
|
|
||||||
// insert a new line with positions if new line is reached
|
// insert a new line with positions if new line is reached
|
||||||
// (previous while pads skipped lines)
|
// (previous while pads skipped lines)
|
||||||
if (firstSymbolAtLine.length < t.line)
|
if (firstSymbolAtLine.length < t.line)
|
||||||
firstSymbolAtLine ~= Pos(t.column, t.type == tok!"if");
|
firstSymbolAtLine ~= Pos(t.column, t.index, t.type == tok!"if");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -96,6 +96,7 @@ private:
|
||||||
static struct Pos
|
static struct Pos
|
||||||
{
|
{
|
||||||
size_t column;
|
size_t column;
|
||||||
|
size_t index;
|
||||||
bool isIf;
|
bool isIf;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -123,7 +124,7 @@ private:
|
||||||
if (r.empty)
|
if (r.empty)
|
||||||
addErrorMessage(if_, KEY, MESSAGE);
|
addErrorMessage(if_, KEY, MESSAGE);
|
||||||
else if (pDecl.column != r.front.column)
|
else if (pDecl.column != r.front.column)
|
||||||
addErrorMessage(if_.line, min(if_.column, pDecl.column), if_.column + 2, KEY, MESSAGE);
|
addErrorMessage([min(if_.index, pDecl.index), if_.index + 2], if_.line, [min(if_.column, pDecl.column), if_.column + 2], KEY, MESSAGE);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -62,7 +62,7 @@ private:
|
||||||
|
|
||||||
if (tok.line != lastErrorLine)
|
if (tok.line != lastErrorLine)
|
||||||
{
|
{
|
||||||
addErrorMessage(tok.line, maxLineLength, max(maxLineLength + 1, tok.column + 1), KEY, message);
|
addErrorMessage([0, 0], tok.line, [maxLineLength, max(maxLineLength + 1, tok.column + 1)], KEY, message);
|
||||||
lastErrorLine = tok.line;
|
lastErrorLine = tok.line;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -139,7 +139,7 @@ void messageFunction(Message message, bool isError)
|
||||||
|
|
||||||
void messageFunctionJSON(string fileName, size_t line, size_t column, string message, bool)
|
void messageFunctionJSON(string fileName, size_t line, size_t column, string message, bool)
|
||||||
{
|
{
|
||||||
writeJSON(Message(Message.Diagnostic.from(fileName, line, column, column, message), "dscanner.syntax"));
|
writeJSON(Message(Message.Diagnostic.from(fileName, [0, 0], line, [column, column], message), "dscanner.syntax"));
|
||||||
}
|
}
|
||||||
|
|
||||||
void writeJSON(Message message)
|
void writeJSON(Message message)
|
||||||
|
@ -199,8 +199,9 @@ void generateReport(string[] fileNames, const StaticAnalysisConfig config,
|
||||||
auto reporter = new DScannerJsonReporter();
|
auto reporter = new DScannerJsonReporter();
|
||||||
|
|
||||||
auto writeMessages = delegate void(string fileName, size_t line, size_t column, string message, bool isError){
|
auto writeMessages = delegate void(string fileName, size_t line, size_t column, string message, bool isError){
|
||||||
|
// TODO: proper index and column ranges
|
||||||
reporter.addMessage(
|
reporter.addMessage(
|
||||||
Message(Message.Diagnostic.from(fileName, line, column, column, message), "dscanner.syntax"),
|
Message(Message.Diagnostic.from(fileName, [0, 0], line, [column, column], message), "dscanner.syntax"),
|
||||||
isError);
|
isError);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -239,8 +240,9 @@ void generateSonarQubeGenericIssueDataReport(string[] fileNames, const StaticAna
|
||||||
auto reporter = new SonarQubeGenericIssueDataReporter();
|
auto reporter = new SonarQubeGenericIssueDataReporter();
|
||||||
|
|
||||||
auto writeMessages = delegate void(string fileName, size_t line, size_t column, string message, bool isError){
|
auto writeMessages = delegate void(string fileName, size_t line, size_t column, string message, bool isError){
|
||||||
|
// TODO: proper index and column ranges
|
||||||
reporter.addMessage(
|
reporter.addMessage(
|
||||||
Message(Message.Diagnostic.from(fileName, line, column, column, message), "dscanner.syntax"),
|
Message(Message.Diagnostic.from(fileName, [0, 0], line, [column, column], message), "dscanner.syntax"),
|
||||||
isError);
|
isError);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -327,8 +329,9 @@ const(Module) parseModule(string fileName, ubyte[] code, RollbackAllocator* p,
|
||||||
ulong* linesOfCode = null, uint* errorCount = null, uint* warningCount = null)
|
ulong* linesOfCode = null, uint* errorCount = null, uint* warningCount = null)
|
||||||
{
|
{
|
||||||
auto writeMessages = delegate(string fileName, size_t line, size_t column, string message, bool isError){
|
auto writeMessages = delegate(string fileName, size_t line, size_t column, string message, bool isError){
|
||||||
|
// TODO: proper index and column ranges
|
||||||
return messageFunctionFormat(errorFormat,
|
return messageFunctionFormat(errorFormat,
|
||||||
Message(Message.Diagnostic.from(fileName, line, column, column, message), "dscanner.syntax"),
|
Message(Message.Diagnostic.from(fileName, [0, 0], line, [column, column], message), "dscanner.syntax"),
|
||||||
isError);
|
isError);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue