aboutsummaryrefslogtreecommitdiff
path: root/command.go
blob: 71f7878320c332f15e9ad1439c8e0777549e356d (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
// Handler to make management of subcommands easier.
package main

import (
  "fmt"
  "flag"
  "os"
)

type (
  Command struct {
    Use   string // command name (used for matching)
    Short string // a short description  to display
    Long  string // a long help text
    Example string // an example string
    Run   func(*Command, []string) // the command to run

    flagSet *flag.FlagSet // internal flagset with all flags
    commands []*Command   // the list of subcommands
  }
)

// This function adds a new sub command.
func (c *Command) AddCommand(cmds... *Command) {
  res := c.commands
  for _, cmd := range cmds {
    res = append(res, cmd)
  }
  c.commands = res
}

// Evaluate the arguments and call either the subcommand or parse it as flags.
func (c *Command) eval(args []string) error {
  var name string = ""
  var rest []string = []string{}

  if len(args) > 0 {
    name = args[0]
  }
  if len(args) > 1 {
    rest = args[1:]
  }

  if name == "help" {
    c.Help(rest)
    return nil
  }

  for _, cmd := range c.commands {
    if cmd.Use == name {
      return cmd.eval(rest)
    }
  }
  if err := c.Flags().Parse(rest); err != nil { return err }
  if c.Run != nil {
    c.Run(c, rest)
  } else {
    c.Help(rest)
  }
  return nil
}

// Execute the command. It will fetch os.Args[1:] itself.
func (c *Command) Execute() error {
  return c.eval(os.Args[1:])
}

// Return the flagset currently in use.
func (c *Command) Flags() *flag.FlagSet {
  if c.flagSet == nil {
    c.flagSet = flag.NewFlagSet(c.Use, flag.ContinueOnError)
  }
  return c.flagSet
}

// Print the help for the current command or a subcommand.
func (c *Command) Help(args []string) {
  if len(args) > 0 {
    for _, cmd := range c.commands {
      if args[0] == cmd.Use {
        cmd.Help([]string{})
        return
      }
    }
  }
  if c.Long != "" { fmt.Println(c.Long, "\n") }
  c.Usage()
}

// Print the usage information.
func (c *Command) Usage() {
  usage := ""
  if c.Use != "" {
    usage = usage + " " + c.Use
  }
  if len(c.commands) > 0 {
    usage = usage + " command"
  }
  if c.flagSet != nil {
    usage = usage + " [flags]"
  }
  fmt.Printf("Usage: %s%s\n", os.Args[0], usage)

  if len(c.commands) > 0 {
    fmt.Printf("\nwhere command is one of:\n")
    for _, cmd := range c.commands {
      fmt.Printf("\t%s\t\t%s\n", cmd.Use, cmd.Short)
    }
  }
  if c.flagSet != nil {
    fmt.Printf("\nwhere flags is any of:\n")
    c.Flags().SetOutput(os.Stdout)
    c.Flags().PrintDefaults()
  }
  if c.Example != "" {
    fmt.Println("\nexample:")
    fmt.Printf("\t%s\n", c.Example)
  }
}