Ayende @ Rahien

My name is Oren Eini
Founder of Hibernating Rhinos LTD and RavenDB.
You can reach me by phone or email:


+972 52-548-6969

, @ Q c

Posts: 6,007 | Comments: 44,760

filter by tags archive

Code of the week

time to read 1 min | 172 words

This piece of code make me so happy:

private static IDictionary<string, Func<Stream, ICommand>> commandFactories =
	new Dictionary<string, Func<Stream, ICommand>>
			{"add", s => new AddCommand(s)},
			{"append", s => new AppendCommand(s)},
			{"cas", s => new CasCommand(s)},
			{"delete", s => new DeleteCommand(s)},
			{"prepend", s => new PrependCommand(s)},
			{"replace", s => new ReplaceCommand(s)},
			{"set", s => new SetCommand(s)},

			// retrieval
			{"get", s => new GetCommand(s)},
			{"gets", s => new GetsCommand(s)},

			{"incr", s => new IncrCommand(s)},
			{"decr", s => new DecrCommand(s)},

			{"flush_all", s => new FlushAllCommand(s)},
			{"quit", s => new QuitCommand(s)},
			{"version", s => new VersionCommand(s)},

This is something that I do quite often, and it is good to have a way to do it easily.


James L

They look like memcached commands

Alex Simkin

You should have request commands from IoC container, not from the dictionary.

Gary A

So what does it do, and why did you do it this way?

Harry M

Have I just made a DSL?

public static void Main(string[] args)


        var me = new Person("Harry");

        var you = new Person("Bob");

        var options = new Dictionary<string, Action<string>>()


                        {"I", s => DoSomething(me, s)},

                        {"You", s => DoSomething(you, s)}


        Parse("I say hello", options);

        Parse("I jump", options);

        Parse("You say goodbye", options);


    private static void Parse(string command, IDictionary<string, Action<string>> options)


        string[] parts = command.Split(new char[] {' '}, 2);

        options[parts[0]](parts.Length > 1 ? parts[1] : null);


    private static void DoSomething(Person who, string what)


        var options = new Dictionary<string, Action<string>>()


                        {"say", s => who.Say(s)},

                        {"jump", s => who.Jump()}


        Parse(what, options);


    class Person


        private readonly string name;

        public Person(string name)


            this.name = name;


        public void Say(string message)


            Console.WriteLine("{0} says: '{1}'", name, message);


        public void Jump()


            Console.WriteLine("{0} jumps", name);


Vijay Santhanam

love the dictionary lookups with the generic delegates/lambdas too.

as much as i'm interested in your code, i'm doubley interested in your design process. e.g., how much analyse/OO modelling you do before you start writing real code?

With tools like R#/svn is the up-front design pretty much dead for garage projects?

would like to read about the steps you take before you get to code. i ask because as a sophisicated practitioner, you take intermediate working steps for granted whilst someone at my level struggles to keep up sometimes.


The more I dig into C# 3.0 the more I infatuate with it.



private static IDictionary do you gain anything by using the IDictionary interface verse just using a regular Dictionary class?

Ayende Rahien

I like using the most abstract type possible

Andrey Shchekin

What about

IDictionary<string, Func<Stream, ICommand>> CreateCommandFactories(IEnumerable types) {

      return (

            from type in types

            select new {

                  Name = type.Name.Replace("Command", "").ToLowerInvariant(),

                  Factory = (Func<Stream, ICommand>)(s => Activator.CreateInstance(type, s))



            x => x.Name,

            x => x.Factory



Of course better way would be to use Regex for parsing and Expression.Compile for constructor.

Ayende Rahien

That involves reflection. I wanted this to be zero reflection effort

Dmitry Gusarov

Ok, but Andrey suggested an excellent duplication removal.

I thought of it... what is more valuable? Avoiding reflection or avoiding duplication... In any case we have a tests ;)

Ayende Rahien


In this case, what I wanted was no reflection. The price paid for it is worth it, I think.

Comment preview

Comments have been closed on this topic.


No future posts left, oh my!


  1. Speaking (3):
    23 Sep 2015 - Build Stuff 2015 (Lithuania & Ukraine), Nov 18 - 24
  2. Production postmortem (11):
    22 Sep 2015 - The case of the Unicode Poo
  3. Technical observations from my wife (2):
    15 Sep 2015 - Disk speeds
  4. Find the bug (5):
    11 Sep 2015 - The concurrent memory buster
  5. Buffer allocation strategies (3):
    09 Sep 2015 - Bad usage patterns
View all series



Main feed Feed Stats
Comments feed   Comments Feed Stats