using System.Runtime.CompilerServices; using System.Text.RegularExpressions; [assembly: InternalsVisibleTo("CapnpC.CSharp.Generator.Tests")] namespace CapnpC.CSharp.Generator { /// /// Represents a capnp.exe output message /// 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(@":(?\d+):(?\d+)(-(?\d+))?:", RegexOptions.Compiled | RegexOptions.RightToLeft); /// /// Constructs an instance from given message /// /// output message (one line) 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(); } } } /// /// The original message /// public string FullMessage { get; } /// /// Whether the message could be decompsed into [filename]:[line]:[column]: [category]: [text] /// public bool IsParseSuccess { get; } /// /// Parsed file name (null iff not IsParseSuccess) /// public string FileName { get; } /// /// Parsed line (0 if not IsParseSuccess) /// public int Line { get; } /// /// Parsed column (0 if not IsParseSuccess) /// public int Column { get; } /// /// Parsed end column (0 if there is none) /// public int EndColumn { get; } /// /// Parsed category (e.g. "error", null iff not IsParseSuccess) /// public string Category { get; } /// /// Parsed message body text (0 if not IsParseSuccess) /// public string MessageText { get; } } }