Set command-line arguments free!

This is a little manifesto for setting command-line arguments free. They are hostages of rigorist minds for 3 decades, and now it has to stop.

When writing command-line tools, there seems to be a cosmic convergence towards this model:

$ command subcommand(?) -abc -d --long-opt foo bar baz

This is OK, and it certainly saves some brain power when it comes to manipulating hundreds of different commands every day. Older tools, such as find, even look suspicious for not comforming to that model. Better, most programming languages have tools to handle that easily, more or less.

What's wrong with that?

I certainly won't argue that conventions are not a good thing, they mostly are, and I could elaborate on this another day. By the way if you want to learn some basics about UNIX conventions and velociraptors, I strongly recommend watching this excellent talk that George Brocklehurst gave at dotRB 2013, he exposes some fundamentals really well, and it's funny.

But this form of passing arguments is certainly not for all kind of programs, and at least sometimes, I'd say that a much more free-form of arguments is liberating. Beyond this philosophical and weak aergument, this form of passing arguments in day-to-day operations has a number of practical drawbacks, where a bit more intelligence built inside your tools may boost your productivity:

A tale of 2 beautiful programs

I experimented with that recently for two programs.

The first is logtailer, a multi-hosts "tail -f" available here. Server names and absolute paths are easy to detect reliably, so the tool accepts arguments in any order, and in any number, as long as there's at least one remote server and one file among them. I don't have to remember any weird option, and I can use expansions or globbing directly (though I need to quote globs if I want them to be interpreted on the remote server):

$ logtailer host1.foo.com host2.foo.com /var/log/syslog
$ logtailer /var/log/syslog host1.foo.com /var/log/foo.log
$ logtailer host{1,2,3,4}.foo.com /var/log/syslog "/var/log/**/*.log"

The second example is a private script called ec2list that saves me dozens of clicks a day in the AWS EC2 console. This script lists all our EC2 instances (== virtual machines) and displays a number fo basic properties like their public IP, region, and some tag values that follow our own conventions here at Botify.

# (all instances)
$ ec2list
name=i-0d7b56a1 type=m3.medium region=eu-central-1 az=eu-central-1a ip=47.51.157.251 Env=staging Role=web
name=i-7086fec2 type=m3.xlarge region=us-west-2 az=us-west-2a ip=54.246.17.235 Env=production Role=crawler
...

Parameters can be used to filter things after retrieval (like a grep -F on values):

# (only crawler instances, in production)
$ ec2list production crawler
name=i-7086fec2 type=m3.xlarge region=us-west-2 az=us-west-2a ip=54.246.17.235 Env=production Role=crawler

Parameters ending with a "=" are treated as "extractors". If 0 extractor, displays everything. If 1 extractor, displays only the values for the given keys (great for further shell integrations!). If 2+ extractors, it retains the original format, but displays only the requested columns:

# (all crawler instances names, in production)
$ ec2list name= production crawler
i-7086fec2
...
$ ec2list name= type= production crawler
name=i-7086fec2 type=m3.xlarge
...

OK I could have done this with some cut, grep, sed, ... But I'm discussing UX here: both for writing, and maybe even some day for other people who will read a script or a documentation that uses this. Also I think I'll stop here with ec2list, if I need more complex operations (typically some "AND" or "OR" in filters), better use the standard tooling after a pipe.

Coupling the two makes tailing logs for instance an order of magnitude easier and more expressive than before:

$ logtailer $(ec2list ip= staging crawler) /var/log/syslog "/var/log/botify/*.log"

Compared with something like:

$ badlogtailer \
--hosts $(echo $(badec2list --fields ip --filters staging,crawler|sed "s/^ip=//") |sed "s/ /,/") \
--files "/var/log/syslog,/var/log/botify/*.log"

(the latest example is exagerated, it could be simpler with some positional arguments, but you get the idea)

Conclusion

This is not a revolution, and other tools certainly already do that. But it's something anyway.

Overall like many things in engineering, it's a matter of tradeoffs: standardizing generic commands used once in a while is a very good thing for the UNIX/Linux users community as a whole. But little tools you use 20+ times a day don't have to follow those rules, they have to be flexible and follow your needs so you maximize both productivity and happiness.

Now go set some command-line arguments free!