Skip to content


Subversion checkout URL

You can clone with
Download ZIP
A chain-of-command framework written in Go
Failed to load latest commit information.
cli Adding support for CLI Subcommands.
convert More documentation fixes.
database Add sync to statement cache.
doc ServeTLS should also not use mux.
example Added a Codegansta-like API.
fmt changed param contents to content to match web.Flush.
io When muliwriter fails logging, send messages to stderr.
log Fixed broken send() function.
safely Added safely.GoDo.
web Fixed code format error in comment.
.gitignore Ignore .test files.
.travis.yml Now require 1.3 for graceful shutdowns. Updated CHANGELOG for 1.2 release.
LICENSE.txt Updated documentation. Added a Codegansta-like API.
commands.go Fixes to Godocs.
commands_test.go Cleaning up the comments and adding tests to commands.go
context.go Fixes to Godocs.
context_test.go Completed the code test coverate for context.go and cleaned up some c…
cookoo.go Added exampels in the test files.
cookoo_test.go Added exampels in the test files. Stubs added.
getter.go Fixed tests and added docs.
getter_test.go Fixed tests and added docs.
glide.yaml Added glide.yaml.
mapper.go Support CmdSpec in addition to commands.
mapper_test.go Tests are good.
params.go Fixes to Godocs.
params_test.go Passed the codebase through go fmt.
registry.go Added DoesCmdDef to registry.
registry_test.go Support CmdSpec in addition to commands.
router.go Fixed documentation formatting.
router_test.go Added tests for CmdDef.
sync_context.go Initial add of SyncContext.


A chain-of-command framework written in Go

Build Status GoDoc


$ cd $GOPATH
$ go get

Use it as follows (from example/example.go):

package main

import (
    // This is the path to Cookoo

func main() {

    // Build a new Cookoo app.
    registry, router, context := cookoo.Cookoo()

    // Fill the registry.
            Name: "TEST",
            Help: "A test route",
            Does: cookoo.Tasks{
                    Name: "hi",
                    Fn:   HelloWorld,

    // Execute the route.
    router.HandleRequest("TEST", context, false)

func HelloWorld(cxt cookoo.Context, params *cookoo.Params) (interface{}, cookoo.Interrupt) {
    fmt.Println("Hello World")
    return true, nil


A Real Example

For a real example of Cookoo, take a look at Skunk.

Here's what Skunk's registry looks like:

    Route("scaffold", "Scaffold a new app.").
        Does(LoadSettings, "settings").
            Using("file").WithDefault(homedir + "/settings.json").From("cxt:SettingsFile").
        Does(MakeDirectories, "dirs").
        Does(RenderTemplates, "template").
    Route("help", "Print help").
        Does(Usage, "Testing")

This has two routes:

  • scaffold
  • help

The help route just runs the command Usage, which looks like this:

func Usage(cxt cookoo.Context, params *cookoo.Params) interface{} {
    fmt.Println("Usage: skunk PROJECTNAME")
    return true

That is a good example of a basic command.

The scaffold route is more complex. It performs the following commands (in order):

  • LoadSettings: Load a settings.json file into the context.
  • MakeDirectories: Make a bunch of directories.
  • RenderTemplates: Perform template conversions on some files.

The MakeDirectories command is an example of a more complex command. It takes two parameters (declared with Using().From()):

  1. basedir: The base directory where the new subdirectories will be created. This comes from the cxt:basedir source, which means Cookoo looks in the Context object for a value named basedir.
  2. directoies: An array of directory names that this command will create. These come from cxt:directories, which means that the Context object is queried for the value of directories. In this case, that value is actually loaded from the settings.json file into the context by the LoadSettings command.`

With that in mind, let's look at the command:

// The MakeDirectories command.
// All commands take a Context and a Params object, and return an
// interface{}
func MakeDirectories(cxt cookoo.Context, params *cookoo.Params) interface{} {

    // This is how we get something out of the Params object. This is the
    // value that was passed in by `Using('basedir').From('cxt:basedir')
    basedir := params.Get("basedir", ".").(string)

    // This is another way to get a parameter value. This form allows us
    // to conveniently check that the parameter exists.
    d, ok := params.Has("directories")
    if !ok {
        // Did nothing. But we don't want to raise an error.
        return false

    // We do have to do an explicit type conversion.
    directories := d.([]interface{})

    // Here we do the work of creating directories.
    for _, dir := range directories {
        dname := path.Join(basedir, dir.(string))
        os.MkdirAll(dname, 0755)

    // We don't really have anything special to return, so we just
    // indicate that the command was successful.
    return true

This is a basic example of working with Cookoo. But far more sophisticated workflows can be built inexpensively and quickly, and in a style that encourages building small and re-usable chunks of code.

Something went wrong with that request. Please try again.