mirror of
https://github.com/FabInfra/capnproto-dotnetcore_Runtime.git
synced 2025-03-12 14:51:41 +01:00
117 lines
4.4 KiB
C#
117 lines
4.4 KiB
C#
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; }
|
|
}
|
|
}
|