117 lines
4.4 KiB
C#
Raw Permalink Normal View History

2019-09-10 22:48:25 +02:00
using System.Runtime.CompilerServices;
using System.Text.RegularExpressions;
[assembly: InternalsVisibleTo("CapnpC.CSharp.Generator.Tests")]
namespace CapnpC.CSharp.Generator
{
/// <summary>
/// Represents a capnp.exe output message
/// </summary>
public class CapnpMessage
{
// capnp outputs look like this:
// empty.capnp:1:1: error: File does not declare an ID. I've generated one for you. Add this line to your file: @0xc82955a0c779197d;
// f:\code\invalid.capnp:9:7-8: error: Ordinal @0 originally used here.
// Parsing them is harder than it seems because the colon may be part of the file name (as in the example above).
// And it becomes even worse! NTFS has a rarely used feature called "alternate data streams", identified by a colon:
// f:\code\somefile:stream.capnp:9:7-8: error: Ordinal @0 originally used here.
// What about a name which looks like a line number? (Hint: the 10 denotes the alternate data stream)
// f:\code\somefile:10:9:7-8: error: Ordinal @0 originally used here.
// Watching for the *last* colon as message separator does not work either, either. See first example.
// Strategy: Watch out for the *last* occurence of pattern :[line]:[column]
static readonly Regex LineColumnRegex = new Regex(@":(?<Line>\d+):(?<Column>\d+)(-(?<EndColumn>\d+))?:", RegexOptions.Compiled | RegexOptions.RightToLeft);
/// <summary>
/// Constructs an instance from given message
/// </summary>
/// <param name="fullMessage">output message (one line)</param>
public CapnpMessage(string fullMessage)
{
FullMessage = fullMessage;
var match = LineColumnRegex.Match(fullMessage);
if (match.Success)
{
IsParseSuccess = true;
FileName = fullMessage.Substring(0, match.Index);
var lineMatch = match.Groups["Line"];
if (lineMatch.Success)
{
int.TryParse(lineMatch.Value, out int value);
Line = value;
}
var columnMatch = match.Groups["Column"];
if (columnMatch.Success)
{
int.TryParse(columnMatch.Value, out int value);
Column = value;
}
var endColumnMatch = match.Groups["EndColumn"];
if (endColumnMatch.Success)
{
int.TryParse(endColumnMatch.Value, out int value);
EndColumn = value;
}
int restIndex = match.Index + match.Length;
int bodyIndex = fullMessage.IndexOf(':', restIndex);
if (bodyIndex >= 0)
{
Category = fullMessage.Substring(restIndex, bodyIndex - restIndex).Trim();
MessageText = fullMessage.Substring(bodyIndex + 1).Trim();
}
else
{
// Never observed "in the wild", just in case...
Category = string.Empty;
MessageText = fullMessage.Substring(restIndex).Trim();
}
}
}
/// <summary>
/// The original message
/// </summary>
public string FullMessage { get; }
/// <summary>
/// Whether the message could be decompsed into [filename]:[line]:[column]: [category]: [text]
/// </summary>
public bool IsParseSuccess { get; }
/// <summary>
/// Parsed file name (null iff not IsParseSuccess)
/// </summary>
public string FileName { get; }
/// <summary>
/// Parsed line (0 if not IsParseSuccess)
/// </summary>
public int Line { get; }
/// <summary>
/// Parsed column (0 if not IsParseSuccess)
/// </summary>
public int Column { get; }
/// <summary>
/// Parsed end column (0 if there is none)
/// </summary>
public int EndColumn { get; }
/// <summary>
/// Parsed category (e.g. "error", null iff not IsParseSuccess)
/// </summary>
public string Category { get; }
/// <summary>
/// Parsed message body text (0 if not IsParseSuccess)
/// </summary>
public string MessageText { get; }
}
}