MadProps.AppArgs
Something I find myself doing a bit, particlarly in business applications, is parsing command-line arguments. For example, if you have one application that could be run against different servers or databases, you can set up a shortcut like this:
myapp.exe /server=sqlserver1 /database=mydb
The application can then configure its database connection at runtime. This is a contrived example but you get the idea.
Typically, command-line parsing code is quite messy. It's lots of foreach
loops and string matching. And then you have to worry about keeping your help text (displayed if the command line is missing a required argument, for example) up to date.
So I've had a crack at a simplified way to parse command line arguments, and I've called it MadProps.AppArgs.
It's pretty easy to use. First, you define a simple POCO class to represent all the arguments your application might take. Like this:
public class MyArgs
{
public string Server { get; set; }
public string Database { get; set; }
}
Now in your application's entry point, you can add a using MadProps.AppArgs
statement, and you'll have an extension method you can use like this:
class Program
{
static void Main(string[] args)
{
var myArgs = args.As<MyArgs>();
}
}
That's all you need to do! Now run the application using the command line above, and you'll have an instance of MyArgs
with its Server
property set to "sqlserver1" and its Database
property set to "mydb"!
Short Names
Let's assume you don't want the argument named "/database" but instead want it named "/db". You could rename the property on your class, or you could use the System.ComponentModel.DataAnnotations.Display
attribute to give it a short name:
public class MyArgs
{
public string Server { get; set; }
[Display(ShortName = "db")]
public string Database { get; set; }
}
Now you can run the app like this for the same result:
myapp.exe /server=sqlserver1 /db=mydb
Required Arguments
Now, you might decide that the Database
argument is required, so you can add the System.ComponentModel.DataAnnotations.Required
attribute:
public class MyArgs
{
public string Server { get; set; }
[Required]
[Display(ShortName = "db")]
public string Database { get; set; }
}
If we leave the database argument off the command line, our call to As<MyArgs>()
will throw an ArgumentException
. We might want to catch that and show the user how to run the app. Let's do that.
class Program
{
static void Main(string[] args)
{
try
{
var myArgs = args.As<MyArgs>();
}
catch (ArgumentException ex)
{
Console.WriteLine(ex.Message);
Console.WriteLine();
Console.WriteLine(AppArgs.HelpFor<MyArgs>());
}
}
}
Now if we run the app and omit the /db argument, we see this output:
Argument missing: 'db' ConsoleApplication10 /db=value [/server=value] /db /server
Argument Descriptions
The output above kind of helpful in that it has told us which arguments are required and which are optional, but it'd be nicer if the arguments had descriptions. Let's do that with the System.ComponentModel.Description
attribute, and/or with the Description
property of the Display
attribute:
public class MyArgs
{
[Description("The database server to connect to")]
public string Server { get; set; }
[Required]
[Display(ShortName = "db", Description="The database to use")]
public string Database { get; set; }
}
Let's run it again without the "/db" argument:
Argument missing: 'db' ConsoleApplication10 /db=value [/server=value] /db The database to use /server The database server to connect to
That's better!
Switches
What if we want a parameter that tells the application to run as an admin, but doesn't have a value? These are sometimes called "command-line switches". So we can run our app like this:
myapp.exe /server=sqlserver1 /db=mydb /admin
That's as easy as adding a boolean property to our class, like this:
public class MyArgs
{
[Description("The database server to connect to")]
public string Server { get; set; }
[Required]
[Display(ShortName = "db", Description = "The database to use")]
public string Database { get; set; }
[Description("Runs the program as an administrator")]
public bool Admin { get; set; }
}
Let's see what happens to our output now if I omit the "/db" parameter again:
Argument missing: 'db' ConsoleApplication10 /db=value [/admin] [/server=value] /admin Runs the program as an administrator /db The database to use /server The database server to connect to
So the new "/admin" parameter has been added to the documentation, and you can see in the example command line that it does not require a value.
Other Argument Types
Let's add another property with a different type. Let's say we want to pass a "timeout" parameter of type int. We'll add the property:
public class MyArgs
{
[Description("The database server to connect to")]
public string Server { get; set; }
[Required]
[Display(ShortName = "db", Description = "The database to use")]
public string Database { get; set; }
[Description("Runs the program as an administrator")]
public bool Admin { get; set; }
[Description("Timeout, in seconds, for commands")]
public int Timeout { get; set; }
}
Now we can pass "/timeout=100" on the command line and the property will receive the value. If we pass "/timeout=foo", which is not a valid integer, we'll get a type conversion exception that we'll also need to catch.
Shortcuts
MadProps.AppArgs will do a PowerShell-like trick of only requiring enough characters in an argument to uniquely identify it, so we could also run our application like this:
myapp.exe /s=sqlserver1 /d=mydb /a /t=100
Where do I get it?
You can check out MadProps.AppArgs on Bitbucket. I don't have a binary download, but feel free to pull down the source and have a play. There are features I haven't listed here, and maybe more to come!
Trackbacks
- MadProps.AppArgs « Mas-Tool's Favorites | http://mas-tool.com/?p=2760
- rimonabantexcellence site title | http://www.rimonabantexcellence.com/t.php?ahr0cdovl21hdhroyw1pbhrvbi5vcmcvbwfkchjvchmtyxbwyxjncz9yzxzpc2lvbj0x
No new comments are allowed on this post.
Comments
Scott Bussinger
Sweet! But you've got to get this up in Nuget. It'll make so much easier for us to play with!
As Rob Connery put it:
Rick Ratayczak
This is really clever! I also vote for putting this on Nuget.
Peter
Really nice! I was going to suggest adding array support, but after checking out the code it looks like you read my mind ...
One more idea: how about adding support for default values (using System.ComponentModel.DefaultValueAttribute)?
Finally, here's another vote for adding it to NuGet. Maybe you could add it as a loose source file - so that when I install it I just get a copy of AppArgs.cs in the root of my project.
Jon Simpson
Very cool - nicely done! I will pile on for the NuGet package.
Scott Bussinger
Peter's idea of using NuGet to deliver just the loose source file is not a bad idea given how small it is (though perhaps creating a MadProps folder to put it in would be better then dumping it in the root).
Daniel Cazzulino discusses how to do it if you've not done it before (it's easy).
Matt Hamilton
Lots of good feedback so far! Thanks guys!
So people don't mind the idea of a loose file Nuget package? I had considered Nuget (this would be my first package) but hadn't thought about the loose file idea. I suppose that would work for .NET 3.5SP1 as well as .NET 4 projects, which is a bonus.
@Peter re: DefaultValueAttribute. I did consider that, but then I thought, "hey, this is just a POCO! If I want one of my properties to have a default value, I'll just add a constructor and set it there!" What do you think?
Peter
Hi Matt -
I had the same thought right after posting my comment. Assigning default values in the constructor makes sense.
Great work!