CommandLineOptions.cs 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436
  1. using System;
  2. using System.Collections.Generic;
  3. using System.IO;
  4. using System.Xml.Xsl;
  5. using google.protobuf;
  6. using System.Xml;
  7. using System.Text;
  8. using System.Xml.Serialization;
  9. using System.Runtime.CompilerServices;
  10. using System.ComponentModel;
  11. using System.Diagnostics;
  12. namespace ProtoBuf.CodeGenerator
  13. {
  14. public sealed class CommandLineOptions
  15. {
  16. private TextWriter errorWriter = Console.Error;
  17. private string workingDirectory = Environment.CurrentDirectory;
  18. /// <summary>
  19. /// Root directory for the session
  20. /// </summary>
  21. public string WorkingDirectory
  22. {
  23. get { return workingDirectory; }
  24. set { workingDirectory = value; }
  25. }
  26. /// <summary>
  27. /// Nominates a writer for error messages (else stderr is used)
  28. /// </summary>
  29. public TextWriter ErrorWriter
  30. {
  31. get { return errorWriter; }
  32. set { errorWriter = value; }
  33. }
  34. [MethodImpl(MethodImplOptions.NoInlining)]
  35. public static int Main(params string[] args)
  36. {
  37. CommandLineOptions opt = null;
  38. try
  39. {
  40. opt = Parse(Console.Out, args);
  41. opt.Execute();
  42. return opt.ShowHelp ? 1 : 0; // count help as a non-success (we didn't generate code)
  43. }
  44. catch (Exception ex)
  45. {
  46. Console.Error.Write(ex.Message);
  47. return 1;
  48. }
  49. }
  50. private string template = TemplateCSharp, outPath = "", defaultNamespace;
  51. private bool showLogo = true, showHelp, writeErrorsToFile;
  52. private readonly List<string> inPaths = new List<string>();
  53. private readonly List<string> args = new List<string>();
  54. private int messageCount;
  55. public int MessageCount { get { return messageCount; } }
  56. public bool WriteErrorsToFile { get { return writeErrorsToFile; } set { writeErrorsToFile = value; } }
  57. public string Template { get { return template; } set { template = value; } }
  58. public string DefaultNamespace { get { return defaultNamespace; } set { defaultNamespace = value; } }
  59. public bool ShowLogo { get { return showLogo; } set { showLogo = value; } }
  60. public string OutPath { get { return outPath; } set { outPath = value; } }
  61. public bool ShowHelp { get { return showHelp; } set { showHelp = value; } }
  62. private readonly XsltArgumentList xsltOptions = new XsltArgumentList();
  63. public XsltArgumentList XsltOptions { get { return xsltOptions; } }
  64. public List<string> InPaths { get { return inPaths; } }
  65. public List<string> Arguments { get { return args; } }
  66. private readonly TextWriter messageOutput;
  67. public static CommandLineOptions Parse(TextWriter messageOutput, params string[] args)
  68. {
  69. CommandLineOptions options = new CommandLineOptions(messageOutput);
  70. string key, value;
  71. for (int i = 0; i < args.Length; i++)
  72. {
  73. string arg = args[i].Trim();
  74. if (arg.StartsWith("-o:"))
  75. {
  76. if (!string.IsNullOrEmpty(options.OutPath)) options.ShowHelp = true;
  77. options.OutPath = arg.Substring(3).Trim();
  78. }
  79. else if (arg.StartsWith("-p:"))
  80. {
  81. Split(arg.Substring(3), out key, out value);
  82. options.XsltOptions.AddParam(key, "", value ?? "true");
  83. }
  84. else if (arg.StartsWith("-t:"))
  85. {
  86. options.Template = arg.Substring(3).Trim();
  87. }
  88. else if (arg.StartsWith("-ns:"))
  89. {
  90. options.DefaultNamespace = arg.Substring(4).Trim();
  91. }
  92. else if (arg == "/?" || arg == "-h")
  93. {
  94. options.ShowHelp = true;
  95. }
  96. else if (arg == "-q") // quiet
  97. {
  98. options.ShowLogo = false;
  99. }
  100. else if (arg == "-d")
  101. {
  102. options.Arguments.Add("--include_imports");
  103. }
  104. else if (arg.StartsWith("-i:"))
  105. {
  106. options.InPaths.Add(arg.Substring(3).Trim());
  107. }
  108. else if (arg == "-writeErrors")
  109. {
  110. options.WriteErrorsToFile = true;
  111. }
  112. else if (arg.StartsWith("-w:"))
  113. {
  114. options.WorkingDirectory = arg.Substring(3).Trim();
  115. }
  116. else
  117. {
  118. options.ShowHelp = true;
  119. }
  120. }
  121. if (options.InPaths.Count == 0)
  122. {
  123. options.ShowHelp = (string)options.XsltOptions.GetParam("help", "") != "true";
  124. }
  125. return options;
  126. }
  127. static readonly char[] SplitTokens = { '=' };
  128. private static void Split(string arg, out string key, out string value)
  129. {
  130. string[] parts = arg.Trim().Split(SplitTokens, 2);
  131. key = parts[0].Trim();
  132. value = parts.Length > 1 ? parts[1].Trim() : null;
  133. }
  134. public CommandLineOptions(TextWriter messageOutput)
  135. {
  136. if (messageOutput == null) throw new ArgumentNullException("messageOutput");
  137. this.messageOutput = messageOutput;
  138. // handling this (even trivially) suppresses the default write;
  139. // we'll also use it to track any messages that are generated
  140. XsltOptions.XsltMessageEncountered += delegate { messageCount++; };
  141. }
  142. public const string TemplateCSharp = "csharp";
  143. private string code;
  144. public string Code { get { return code; } private set { code = value; } }
  145. public void Execute()
  146. {
  147. StringBuilder errors = new StringBuilder();
  148. string oldDir = Environment.CurrentDirectory;
  149. Environment.CurrentDirectory = WorkingDirectory;
  150. try
  151. {
  152. if (string.IsNullOrEmpty(OutPath))
  153. {
  154. WriteErrorsToFile = false; // can't be
  155. }
  156. else if (WriteErrorsToFile)
  157. {
  158. ErrorWriter = new StringWriter(errors);
  159. }
  160. try
  161. {
  162. if (ShowLogo)
  163. {
  164. StringBuilder sb = new StringBuilder();
  165. for(int i = 0; i < InPaths.Count; i++)
  166. {
  167. sb.Append(InPaths[i]);
  168. if (i < InPaths.Count - 1)
  169. {
  170. sb.Append(";");
  171. }
  172. }
  173. messageOutput.WriteLine(Properties.Resources.LogoText + " : " + sb);
  174. }
  175. if (ShowHelp)
  176. {
  177. messageOutput.WriteLine(Properties.Resources.Usage);
  178. return;
  179. }
  180. string xml = LoadFilesAsXml(this);
  181. Code = ApplyTransform(this, xml);
  182. if (this.OutPath == "-") { }
  183. else if (!string.IsNullOrEmpty(this.OutPath))
  184. {
  185. File.WriteAllText(this.OutPath, Code);
  186. }
  187. else if (string.IsNullOrEmpty(this.OutPath))
  188. {
  189. messageOutput.Write(Code);
  190. }
  191. }
  192. catch (Exception ex)
  193. {
  194. if (WriteErrorsToFile)
  195. {
  196. // if we had a parse fail and were able to capture something
  197. // sensible, then just write that; otherwise use the exception
  198. // as well
  199. string body = (ex is ProtoParseException && errors.Length > 0) ?
  200. errors.ToString() : (ex.Message + Environment.NewLine + errors);
  201. File.WriteAllText(this.OutPath, body);
  202. }
  203. throw;
  204. }
  205. }
  206. finally
  207. {
  208. try { Environment.CurrentDirectory = oldDir; }
  209. catch (Exception ex) { Trace.WriteLine(ex); }
  210. }
  211. }
  212. private static string LoadFilesAsXml(CommandLineOptions options)
  213. {
  214. FileDescriptorSet set = new FileDescriptorSet();
  215. foreach (string inPath in options.InPaths)
  216. {
  217. InputFileLoader.Merge(set, inPath, options.ErrorWriter, options.Arguments.ToArray());
  218. }
  219. set = ApplyComment(set);
  220. XmlSerializer xser = new XmlSerializer(typeof(FileDescriptorSet));
  221. XmlWriterSettings settings = new XmlWriterSettings();
  222. settings.Indent = true;
  223. settings.IndentChars = " ";
  224. settings.NewLineHandling = NewLineHandling.Entitize;
  225. StringBuilder sb = new StringBuilder();
  226. using (XmlWriter writer = XmlWriter.Create(sb, settings))
  227. {
  228. xser.Serialize(writer, set);
  229. }
  230. return sb.ToString();
  231. }
  232. private static string ApplyTransform(CommandLineOptions options, string xml)
  233. {
  234. XmlWriterSettings settings = new XmlWriterSettings();
  235. settings.ConformanceLevel = ConformanceLevel.Auto;
  236. settings.CheckCharacters = false;
  237. StringBuilder sb = new StringBuilder();
  238. using (XmlReader reader = XmlReader.Create(new StringReader(xml)))
  239. using (TextWriter writer = new StringWriter(sb))
  240. {
  241. XslCompiledTransform xslt = new XslCompiledTransform();
  242. string xsltTemplate = Path.ChangeExtension(options.Template, "xslt");
  243. if (!File.Exists(xsltTemplate))
  244. {
  245. string localXslt = InputFileLoader.CombinePathFromAppRoot(xsltTemplate);
  246. if (File.Exists(localXslt))
  247. xsltTemplate = localXslt;
  248. }
  249. try
  250. {
  251. xslt.Load(xsltTemplate);
  252. }
  253. catch (Exception ex)
  254. {
  255. throw new InvalidOperationException("Unable to load tranform: " + options.Template, ex);
  256. }
  257. options.XsltOptions.RemoveParam("defaultNamespace", "");
  258. if (options.DefaultNamespace != null)
  259. {
  260. options.XsltOptions.AddParam("defaultNamespace", "", options.DefaultNamespace);
  261. }
  262. xslt.Transform(reader, options.XsltOptions, writer);
  263. }
  264. return sb.ToString();
  265. }
  266. private static void ApplyComment(FileDescriptorProto file, SourceCodeInfo.Location location)
  267. {
  268. if (!string.IsNullOrEmpty(location.leading_comments) ||
  269. !string.IsNullOrEmpty(location.trailing_comments))
  270. {
  271. switch (location.path.Count)
  272. {
  273. case 1:
  274. if (location.path[0] == 2)
  275. {
  276. file.comments = GenComments(location);
  277. }
  278. break;
  279. case 2:
  280. if (location.path[0] == 4)
  281. {
  282. // message define
  283. file.message_type[location.path[1]].comments = GenComments(location);
  284. }
  285. else if (location.path[0] == 5)
  286. {
  287. // enum define
  288. file.enum_type[location.path[1]].comments = GenComments(location);
  289. }
  290. else if (location.path[0] == 6)
  291. {
  292. // service define
  293. file.service[location.path[1]].comments = GenComments(location);
  294. }
  295. break;
  296. case 4:
  297. if (location.path[0] == 4 && location.path[2] == 2)
  298. {
  299. // message fields
  300. file.message_type[location.path[1]].field[location.path[3]].comments = GenComments(location);
  301. }
  302. else if (location.path[0] == 4 && location.path[2] == 4)
  303. {
  304. // message enums
  305. file.message_type[location.path[1]].enum_type[location.path[3]].comments = GenComments(location);
  306. }
  307. else if (location.path[0] == 5 && location.path[2] == 2)
  308. {
  309. // enum values
  310. file.enum_type[location.path[1]].value[location.path[3]].comments = GenComments(location);
  311. }
  312. else if (location.path[0] == 6 && location.path[2] == 2)
  313. {
  314. // service methods
  315. file.service[location.path[1]].method[location.path[3]].comments = GenComments(location);
  316. }
  317. break;
  318. case 6:
  319. if (location.path[0] == 4 && location.path[2] == 4 && location.path[4] == 2)
  320. {
  321. // message inner enums values
  322. file.message_type[location.path[1]].enum_type[location.path[3]].value[location.path[5]].comments = GenComments(location);
  323. }
  324. break;
  325. }
  326. }
  327. }
  328. private static List<string> GenComments(SourceCodeInfo.Location location)
  329. {
  330. var ret = new List<string>();
  331. if (location.path.Count > 1)
  332. {
  333. if (!string.IsNullOrEmpty(location.leading_comments))
  334. {
  335. ret.Add(GenCommentLine(location.leading_comments));
  336. }
  337. if (!string.IsNullOrEmpty(location.trailing_comments))
  338. {
  339. ret.Add(GenCommentLine(location.trailing_comments));
  340. }
  341. }
  342. else
  343. {
  344. StringBuilder sb = new StringBuilder();
  345. if (!string.IsNullOrEmpty(location.leading_comments))
  346. {
  347. sb.Append(location.leading_comments);
  348. }
  349. if (!string.IsNullOrEmpty(location.trailing_comments))
  350. {
  351. sb.AppendLine();
  352. sb.Append(location.trailing_comments);
  353. }
  354. var lines = sb.ToString().Split('\n');
  355. ret.AddRange(lines);
  356. }
  357. return ret;
  358. }
  359. public static string GenCommentLine(string comment)
  360. {
  361. var lines = comment.Trim().Split('\n');
  362. StringBuilder sb = new StringBuilder();
  363. for (int i = 0; i < lines.Length; i++)
  364. {
  365. sb.Append(lines[i]);
  366. if (i < lines.Length - 1)
  367. {
  368. sb.Append(CommentSplitChar);
  369. }
  370. }
  371. return sb.ToString();
  372. }
  373. public static string GenIndentPrefix(int indent)
  374. {
  375. if (indent == 0) return "";
  376. StringBuilder sb = new StringBuilder();
  377. for (int i = 0; i < indent; i++)
  378. {
  379. sb.Append(CommentIndentChar);
  380. }
  381. return sb.ToString().Trim();
  382. }
  383. private static FileDescriptorSet ApplyComment(FileDescriptorSet set)
  384. {
  385. try
  386. {
  387. foreach (var file in set.file)
  388. {
  389. foreach (var location in file.source_code_info.location)
  390. {
  391. if (location.path != null)
  392. {
  393. ApplyComment(file, location);
  394. }
  395. }
  396. }
  397. }
  398. catch (Exception err)
  399. {
  400. Trace.WriteLine(err);
  401. }
  402. return set;
  403. }
  404. public static string CommentIndentChar = " ";
  405. public static string CommentSplitChar = " ";
  406. }
  407. }