/DBSourceTools_root/DBSourceTools/DBSourceToolsLib/ItdmCommandLine.cs

# · C# · 373 lines · 225 code · 44 blank · 104 comment · 20 complexity · 6e88dcce24182daf1562fe289c357cea MD5 · raw file

  1. using System;
  2. using System.Collections.Generic;
  3. using System.Collections.Specialized;
  4. using System.Text;
  5. namespace ITDM
  6. {
  7. /// <summary>
  8. /// Represents an error occuring during command line parsing.
  9. /// </summary>
  10. public class CmdLineException : Exception
  11. {
  12. public CmdLineException(string parameter, string message)
  13. :
  14. base(String.Format("Syntax error of parameter -{0}: {1}", parameter, message))
  15. {
  16. }
  17. public CmdLineException(string message)
  18. :
  19. base(message)
  20. {
  21. }
  22. }
  23. /// <summary>
  24. /// Represents a command line parameter.
  25. /// Parameters are words in the command line beginning with a hyphen (-).
  26. /// The value of the parameter is the next word in
  27. /// </summary>
  28. public class CmdLineParameter
  29. {
  30. private string _name;
  31. private string _value = "";
  32. private bool _required = false;
  33. private string _helpMessage = "";
  34. private bool _exists = false;
  35. /// <summary>
  36. /// Creates a new instance of this class.
  37. /// </summary>
  38. /// <param name="name">Name of parameter.</param>
  39. /// <param name="required">Require that the parameter is present in the command line.</param>
  40. /// <param name="helpMessage">The explanation of the parameter to add to the help screen.</param>
  41. public CmdLineParameter(string name, bool required, string helpMessage)
  42. {
  43. _name = name;
  44. _required = required;
  45. _helpMessage = helpMessage;
  46. }
  47. /// <summary>
  48. /// Sets the value of the parameter.
  49. /// </summary>
  50. /// <param name="value">A string containing a integer expression.</param>
  51. public virtual void SetValue(string value)
  52. {
  53. _value = value;
  54. _exists = true;
  55. }
  56. /// <summary>
  57. /// Returns the value of the parameter.
  58. /// </summary>
  59. public string Value
  60. {
  61. get { return _value; }
  62. }
  63. /// <summary>
  64. /// Returns the help message associated with the parameter.
  65. /// </summary>
  66. public string Help
  67. {
  68. get { return _helpMessage; }
  69. }
  70. /// <summary>
  71. /// Returns true if the parameter was found in the command line.
  72. /// </summary>
  73. public bool Exists
  74. {
  75. get { return _exists; }
  76. }
  77. /// <summary>
  78. /// Returns true if the parameter is required in the command line.
  79. /// </summary>
  80. public bool Required
  81. {
  82. get { return _required; }
  83. }
  84. /// <summary>
  85. /// Returns the name of the parameter.
  86. /// </summary>
  87. public string Name
  88. {
  89. get { return _name; }
  90. }
  91. }
  92. /// <summary>
  93. /// Represents an integer command line parameter.
  94. /// </summary>
  95. public class CmdLineInt : CmdLineParameter
  96. {
  97. private int _min = int.MinValue;
  98. private int _max = int.MaxValue;
  99. private int _value;
  100. /// <summary>
  101. /// Creates a new instance of this class.
  102. /// </summary>
  103. /// <param name="name">Name of parameter.</param>
  104. /// <param name="required">Require that the parameter is present in the command line.</param>
  105. /// <param name="helpMessage">The explanation of the parameter to add to the help screen.</param>
  106. public CmdLineInt(string name, bool required, string helpMessage)
  107. : base(name, required, helpMessage)
  108. {
  109. }
  110. /// <summary>
  111. /// Creates a new instance of this class.
  112. /// </summary>
  113. /// <param name="name">Name of parameter.</param>
  114. /// <param name="required">Require that the parameter is present in the command line.</param>
  115. /// <param name="helpMessage">The explanation of the parameter to add to the help screen.</param>
  116. /// <param name="min">The minimum value of the parameter.</param>
  117. /// <param name="max">The maximum valie of the parameter.</param>
  118. public CmdLineInt(string name, bool required, string helpMessage, int min, int max)
  119. : base(name, required, helpMessage)
  120. {
  121. _min = min;
  122. _max = max;
  123. }
  124. /// <summary>
  125. /// Sets the value of the parameter.
  126. /// </summary>
  127. /// <param name="value">A string containing a integer expression.</param>
  128. public override void SetValue(string value)
  129. {
  130. base.SetValue(value);
  131. int i = 0;
  132. try
  133. {
  134. i = Convert.ToInt32(value);
  135. }
  136. catch (Exception)
  137. {
  138. throw new CmdLineException(base.Name, "Value is not an integer.");
  139. }
  140. if (i < _min) throw new CmdLineException(base.Name, String.Format("Value must be greather or equal to {0}.", _min));
  141. if (i > _max) throw new CmdLineException(base.Name, String.Format("Value must be less or equal to {0}.", _max));
  142. _value = i;
  143. }
  144. /// <summary>
  145. /// Returns the current value of the parameter.
  146. /// </summary>
  147. new public int Value
  148. {
  149. get { return _value; }
  150. }
  151. /// <summary>
  152. /// A implicit converion to a int data type.
  153. /// </summary>
  154. /// <param name="s"></param>
  155. /// <returns></returns>
  156. public static implicit operator int(CmdLineInt s)
  157. {
  158. return s.Value;
  159. }
  160. }
  161. /// <summary>
  162. /// Represents an string command line parameter.
  163. /// </summary>
  164. public class CmdLineString : CmdLineParameter
  165. {
  166. public CmdLineString(string name, bool required, string helpMessage)
  167. : base(name, required, helpMessage)
  168. {
  169. }
  170. public static implicit operator string(CmdLineString s)
  171. {
  172. return s.Value;
  173. }
  174. }
  175. /// <summary>
  176. /// Provides a simple strongly typed interface to work with command line parameters.
  177. /// </summary>
  178. public class CmdLine
  179. {
  180. // A private dictonary containing the parameters.
  181. private Dictionary<string, CmdLineParameter> parameters = new Dictionary<string, CmdLineParameter>();
  182. /// <summary>
  183. /// Creats a new empty command line object.
  184. /// </summary>
  185. public CmdLine()
  186. {
  187. }
  188. /// <summary>
  189. /// Returns a command line parameter by the name.
  190. /// </summary>
  191. /// <param name="name">The name of the parameter (the word after the initial hyphen (-).</param>
  192. /// <returns>A reference to the named comman line object.</returns>
  193. public CmdLineParameter this[string name]
  194. {
  195. get
  196. {
  197. if (!parameters.ContainsKey(name))
  198. throw new CmdLineException(name, "Not a registered parameter.");
  199. return parameters[name];
  200. }
  201. }
  202. /// <summary>
  203. /// Registers a parameter to be used and adds it to the help screen.
  204. /// </summary>
  205. /// <param name="p">The parameter to add.</param>
  206. public void RegisterParameter(CmdLineParameter parameter)
  207. {
  208. if (parameters.ContainsKey(parameter.Name))
  209. throw new CmdLineException(parameter.Name, "Parameter is already registered.");
  210. parameters.Add(parameter.Name, parameter);
  211. }
  212. /// <summary>
  213. /// Registers parameters to be used and adds hem to the help screen.
  214. /// </summary>
  215. /// <param name="p">The parameter to add.</param>
  216. public void RegisterParameter(CmdLineParameter[] parameters)
  217. {
  218. foreach (CmdLineParameter p in parameters)
  219. RegisterParameter(p);
  220. }
  221. /// <summary>
  222. /// Parses the command line and sets the value of each registered parmaters.
  223. /// </summary>
  224. /// <param name="args">The arguments array sent to main()</param>
  225. /// <returns>Any reminding strings after arguments has been processed.</returns>
  226. public string[] Parse(string[] args)
  227. {
  228. int i = 0;
  229. List<string> new_args = new List<string>();
  230. while (i < args.Length)
  231. {
  232. if (args[i].Length > 1 && args[i][0] == '-')
  233. {
  234. // The current string is a parameter name
  235. string key = args[i].Substring(1, args[i].Length - 1).ToLower();
  236. string value = "";
  237. i++;
  238. if (i < args.Length)
  239. {
  240. if (args[i].Length > 0 && args[i][0] == '-')
  241. {
  242. // The next string is a new parameter, do not nothing
  243. }
  244. else
  245. {
  246. // The next string is a value, read the value and move forward
  247. value = args[i];
  248. i++;
  249. }
  250. }
  251. if (!parameters.ContainsKey(key))
  252. throw new CmdLineException(key, "Parameter is not allowed.");
  253. if (parameters[key].Exists)
  254. throw new CmdLineException(key, "Parameter is specified more than once.");
  255. parameters[key].SetValue(value);
  256. }
  257. else
  258. {
  259. new_args.Add(args[i]);
  260. i++;
  261. }
  262. }
  263. // Check that required parameters are present in the command line.
  264. foreach (string key in parameters.Keys)
  265. if (parameters[key].Required && !parameters[key].Exists)
  266. throw new CmdLineException(key, "Required parameter is not found.");
  267. return new_args.ToArray();
  268. }
  269. /// <summary>
  270. /// Generates the help screen.
  271. /// </summary>
  272. public string HelpScreen()
  273. {
  274. int len = 0;
  275. foreach (string key in parameters.Keys)
  276. len = Math.Max(len, key.Length);
  277. string help = "\nParameters:\n\n";
  278. foreach (string key in parameters.Keys)
  279. {
  280. string s = "-" + parameters[key].Name;
  281. while (s.Length < len + 3)
  282. s += " ";
  283. s += parameters[key].Help + "\n";
  284. help += s;
  285. }
  286. return help;
  287. }
  288. }
  289. /// <summary>
  290. /// Represents a CmdLine object to use with console applications.
  291. /// The -help parameter will be registered automatically.
  292. /// Any errors will be written to the console instead of generating exceptions.
  293. /// </summary>
  294. public class ConsoleCmdLine : CmdLine
  295. {
  296. public ConsoleCmdLine()
  297. {
  298. base.RegisterParameter(new CmdLineString("help", false, "Prints the help screen."));
  299. }
  300. public new string[] Parse(string[] args)
  301. {
  302. string[] ret = null;
  303. string error = "";
  304. try
  305. {
  306. ret = base.Parse(args);
  307. }
  308. catch (CmdLineException ex)
  309. {
  310. error = ex.Message;
  311. }
  312. if (this["help"].Exists)
  313. {
  314. //foreach(string s in base.HelpScreen().Split('\n'))
  315. // Console.WriteLine(s);
  316. Console.WriteLine(base.HelpScreen());
  317. System.Environment.Exit(0);
  318. }
  319. if (error != "")
  320. {
  321. Console.WriteLine(error);
  322. Console.WriteLine("Use -help for more information.");
  323. System.Windows.Forms.SendKeys.SendWait("{ENTER}");
  324. System.Environment.Exit(1);
  325. }
  326. return ret;
  327. }
  328. }
  329. }