123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436 |
- using System;
- using System.Collections.Generic;
- using System.IO;
- using System.Xml.Xsl;
- using google.protobuf;
- using System.Xml;
- using System.Text;
- using System.Xml.Serialization;
- using System.Runtime.CompilerServices;
- using System.ComponentModel;
- using System.Diagnostics;
- namespace ProtoBuf.CodeGenerator
- {
- public sealed class CommandLineOptions
- {
- private TextWriter errorWriter = Console.Error;
- private string workingDirectory = Environment.CurrentDirectory;
- /// <summary>
- /// Root directory for the session
- /// </summary>
- public string WorkingDirectory
- {
- get { return workingDirectory; }
- set { workingDirectory = value; }
- }
- /// <summary>
- /// Nominates a writer for error messages (else stderr is used)
- /// </summary>
- public TextWriter ErrorWriter
- {
- get { return errorWriter; }
- set { errorWriter = value; }
- }
- [MethodImpl(MethodImplOptions.NoInlining)]
- public static int Main(params string[] args)
- {
- CommandLineOptions opt = null;
- try
- {
- opt = Parse(Console.Out, args);
- opt.Execute();
- return opt.ShowHelp ? 1 : 0; // count help as a non-success (we didn't generate code)
- }
- catch (Exception ex)
- {
- Console.Error.Write(ex.Message);
- return 1;
- }
- }
- private string template = TemplateCSharp, outPath = "", defaultNamespace;
- private bool showLogo = true, showHelp, writeErrorsToFile;
- private readonly List<string> inPaths = new List<string>();
- private readonly List<string> args = new List<string>();
- private int messageCount;
- public int MessageCount { get { return messageCount; } }
- public bool WriteErrorsToFile { get { return writeErrorsToFile; } set { writeErrorsToFile = value; } }
- public string Template { get { return template; } set { template = value; } }
- public string DefaultNamespace { get { return defaultNamespace; } set { defaultNamespace = value; } }
- public bool ShowLogo { get { return showLogo; } set { showLogo = value; } }
- public string OutPath { get { return outPath; } set { outPath = value; } }
- public bool ShowHelp { get { return showHelp; } set { showHelp = value; } }
- private readonly XsltArgumentList xsltOptions = new XsltArgumentList();
- public XsltArgumentList XsltOptions { get { return xsltOptions; } }
- public List<string> InPaths { get { return inPaths; } }
- public List<string> Arguments { get { return args; } }
- private readonly TextWriter messageOutput;
- public static CommandLineOptions Parse(TextWriter messageOutput, params string[] args)
- {
- CommandLineOptions options = new CommandLineOptions(messageOutput);
- string key, value;
- for (int i = 0; i < args.Length; i++)
- {
- string arg = args[i].Trim();
- if (arg.StartsWith("-o:"))
- {
- if (!string.IsNullOrEmpty(options.OutPath)) options.ShowHelp = true;
- options.OutPath = arg.Substring(3).Trim();
- }
- else if (arg.StartsWith("-p:"))
- {
- Split(arg.Substring(3), out key, out value);
- options.XsltOptions.AddParam(key, "", value ?? "true");
- }
- else if (arg.StartsWith("-t:"))
- {
- options.Template = arg.Substring(3).Trim();
- }
- else if (arg.StartsWith("-ns:"))
- {
- options.DefaultNamespace = arg.Substring(4).Trim();
- }
- else if (arg == "/?" || arg == "-h")
- {
- options.ShowHelp = true;
- }
- else if (arg == "-q") // quiet
- {
- options.ShowLogo = false;
- }
- else if (arg == "-d")
- {
- options.Arguments.Add("--include_imports");
- }
- else if (arg.StartsWith("-i:"))
- {
- options.InPaths.Add(arg.Substring(3).Trim());
- }
- else if (arg == "-writeErrors")
- {
- options.WriteErrorsToFile = true;
- }
- else if (arg.StartsWith("-w:"))
- {
- options.WorkingDirectory = arg.Substring(3).Trim();
- }
- else
- {
- options.ShowHelp = true;
- }
- }
- if (options.InPaths.Count == 0)
- {
- options.ShowHelp = (string)options.XsltOptions.GetParam("help", "") != "true";
- }
- return options;
- }
- static readonly char[] SplitTokens = { '=' };
- private static void Split(string arg, out string key, out string value)
- {
- string[] parts = arg.Trim().Split(SplitTokens, 2);
- key = parts[0].Trim();
- value = parts.Length > 1 ? parts[1].Trim() : null;
- }
- public CommandLineOptions(TextWriter messageOutput)
- {
- if (messageOutput == null) throw new ArgumentNullException("messageOutput");
- this.messageOutput = messageOutput;
- // handling this (even trivially) suppresses the default write;
- // we'll also use it to track any messages that are generated
- XsltOptions.XsltMessageEncountered += delegate { messageCount++; };
- }
- public const string TemplateCSharp = "csharp";
- private string code;
- public string Code { get { return code; } private set { code = value; } }
- public void Execute()
- {
- StringBuilder errors = new StringBuilder();
- string oldDir = Environment.CurrentDirectory;
- Environment.CurrentDirectory = WorkingDirectory;
- try
- {
- if (string.IsNullOrEmpty(OutPath))
- {
- WriteErrorsToFile = false; // can't be
- }
- else if (WriteErrorsToFile)
- {
- ErrorWriter = new StringWriter(errors);
- }
- try
- {
- if (ShowLogo)
- {
- StringBuilder sb = new StringBuilder();
- for(int i = 0; i < InPaths.Count; i++)
- {
- sb.Append(InPaths[i]);
- if (i < InPaths.Count - 1)
- {
- sb.Append(";");
- }
- }
- messageOutput.WriteLine(Properties.Resources.LogoText + " : " + sb);
- }
- if (ShowHelp)
- {
- messageOutput.WriteLine(Properties.Resources.Usage);
- return;
- }
- string xml = LoadFilesAsXml(this);
- Code = ApplyTransform(this, xml);
- if (this.OutPath == "-") { }
- else if (!string.IsNullOrEmpty(this.OutPath))
- {
- File.WriteAllText(this.OutPath, Code);
- }
- else if (string.IsNullOrEmpty(this.OutPath))
- {
- messageOutput.Write(Code);
- }
- }
- catch (Exception ex)
- {
- if (WriteErrorsToFile)
- {
- // if we had a parse fail and were able to capture something
- // sensible, then just write that; otherwise use the exception
- // as well
- string body = (ex is ProtoParseException && errors.Length > 0) ?
- errors.ToString() : (ex.Message + Environment.NewLine + errors);
- File.WriteAllText(this.OutPath, body);
- }
- throw;
- }
- }
- finally
- {
- try { Environment.CurrentDirectory = oldDir; }
- catch (Exception ex) { Trace.WriteLine(ex); }
- }
- }
- private static string LoadFilesAsXml(CommandLineOptions options)
- {
- FileDescriptorSet set = new FileDescriptorSet();
- foreach (string inPath in options.InPaths)
- {
- InputFileLoader.Merge(set, inPath, options.ErrorWriter, options.Arguments.ToArray());
- }
- set = ApplyComment(set);
- XmlSerializer xser = new XmlSerializer(typeof(FileDescriptorSet));
- XmlWriterSettings settings = new XmlWriterSettings();
- settings.Indent = true;
- settings.IndentChars = " ";
- settings.NewLineHandling = NewLineHandling.Entitize;
- StringBuilder sb = new StringBuilder();
- using (XmlWriter writer = XmlWriter.Create(sb, settings))
- {
- xser.Serialize(writer, set);
- }
- return sb.ToString();
- }
- private static string ApplyTransform(CommandLineOptions options, string xml)
- {
- XmlWriterSettings settings = new XmlWriterSettings();
- settings.ConformanceLevel = ConformanceLevel.Auto;
- settings.CheckCharacters = false;
- StringBuilder sb = new StringBuilder();
- using (XmlReader reader = XmlReader.Create(new StringReader(xml)))
- using (TextWriter writer = new StringWriter(sb))
- {
- XslCompiledTransform xslt = new XslCompiledTransform();
- string xsltTemplate = Path.ChangeExtension(options.Template, "xslt");
- if (!File.Exists(xsltTemplate))
- {
- string localXslt = InputFileLoader.CombinePathFromAppRoot(xsltTemplate);
- if (File.Exists(localXslt))
- xsltTemplate = localXslt;
- }
- try
- {
- xslt.Load(xsltTemplate);
- }
- catch (Exception ex)
- {
- throw new InvalidOperationException("Unable to load tranform: " + options.Template, ex);
- }
- options.XsltOptions.RemoveParam("defaultNamespace", "");
- if (options.DefaultNamespace != null)
- {
- options.XsltOptions.AddParam("defaultNamespace", "", options.DefaultNamespace);
- }
- xslt.Transform(reader, options.XsltOptions, writer);
- }
- return sb.ToString();
- }
- private static void ApplyComment(FileDescriptorProto file, SourceCodeInfo.Location location)
- {
- if (!string.IsNullOrEmpty(location.leading_comments) ||
- !string.IsNullOrEmpty(location.trailing_comments))
- {
- switch (location.path.Count)
- {
- case 1:
- if (location.path[0] == 2)
- {
- file.comments = GenComments(location);
- }
- break;
- case 2:
- if (location.path[0] == 4)
- {
- // message define
- file.message_type[location.path[1]].comments = GenComments(location);
- }
- else if (location.path[0] == 5)
- {
- // enum define
- file.enum_type[location.path[1]].comments = GenComments(location);
- }
- else if (location.path[0] == 6)
- {
- // service define
- file.service[location.path[1]].comments = GenComments(location);
- }
- break;
- case 4:
- if (location.path[0] == 4 && location.path[2] == 2)
- {
- // message fields
- file.message_type[location.path[1]].field[location.path[3]].comments = GenComments(location);
- }
- else if (location.path[0] == 4 && location.path[2] == 4)
- {
- // message enums
- file.message_type[location.path[1]].enum_type[location.path[3]].comments = GenComments(location);
- }
- else if (location.path[0] == 5 && location.path[2] == 2)
- {
- // enum values
- file.enum_type[location.path[1]].value[location.path[3]].comments = GenComments(location);
- }
- else if (location.path[0] == 6 && location.path[2] == 2)
- {
- // service methods
- file.service[location.path[1]].method[location.path[3]].comments = GenComments(location);
- }
- break;
- case 6:
- if (location.path[0] == 4 && location.path[2] == 4 && location.path[4] == 2)
- {
- // message inner enums values
- file.message_type[location.path[1]].enum_type[location.path[3]].value[location.path[5]].comments = GenComments(location);
- }
- break;
- }
- }
- }
- private static List<string> GenComments(SourceCodeInfo.Location location)
- {
- var ret = new List<string>();
- if (location.path.Count > 1)
- {
- if (!string.IsNullOrEmpty(location.leading_comments))
- {
- ret.Add(GenCommentLine(location.leading_comments));
- }
- if (!string.IsNullOrEmpty(location.trailing_comments))
- {
- ret.Add(GenCommentLine(location.trailing_comments));
- }
- }
- else
- {
- StringBuilder sb = new StringBuilder();
- if (!string.IsNullOrEmpty(location.leading_comments))
- {
- sb.Append(location.leading_comments);
- }
- if (!string.IsNullOrEmpty(location.trailing_comments))
- {
- sb.AppendLine();
- sb.Append(location.trailing_comments);
- }
- var lines = sb.ToString().Split('\n');
- ret.AddRange(lines);
- }
- return ret;
- }
- public static string GenCommentLine(string comment)
- {
- var lines = comment.Trim().Split('\n');
- StringBuilder sb = new StringBuilder();
- for (int i = 0; i < lines.Length; i++)
- {
- sb.Append(lines[i]);
- if (i < lines.Length - 1)
- {
- sb.Append(CommentSplitChar);
- }
- }
- return sb.ToString();
- }
- public static string GenIndentPrefix(int indent)
- {
- if (indent == 0) return "";
- StringBuilder sb = new StringBuilder();
- for (int i = 0; i < indent; i++)
- {
- sb.Append(CommentIndentChar);
- }
- return sb.ToString().Trim();
- }
- private static FileDescriptorSet ApplyComment(FileDescriptorSet set)
- {
- try
- {
- foreach (var file in set.file)
- {
- foreach (var location in file.source_code_info.location)
- {
- if (location.path != null)
- {
- ApplyComment(file, location);
- }
- }
- }
- }
- catch (Exception err)
- {
- Trace.WriteLine(err);
- }
- return set;
- }
- public static string CommentIndentChar = " ";
- public static string CommentSplitChar = " ";
- }
- }
|