1
0
mirror of https://github.com/ipfs/kubo.git synced 2025-08-06 03:19:47 +08:00
Files
kubo/commands/command.go
2014-10-20 07:04:52 -07:00

121 lines
2.5 KiB
Go

package commands
import (
"fmt"
"reflect"
"strings"
)
type Command struct {
Help string
Options []Option
f func(*Request, *Response)
subcommands map[string]*Command
}
// Register adds a subcommand
func (c *Command) Register(id string, sub *Command) error {
if c.subcommands == nil {
c.subcommands = make(map[string]*Command)
}
// check for duplicate option names (only checks downwards)
names := make(map[string]bool)
globalCommand.checkOptions(names)
c.checkOptions(names)
err := sub.checkOptions(names)
if err != nil {
return err
}
if _, ok := c.subcommands[id]; ok {
return fmt.Errorf("There is already a subcommand registered with id '%s'", id)
}
c.subcommands[id] = sub
return nil
}
// Call invokes the command at the given subcommand path
func (c *Command) Call(path []string, req *Request) *Response {
options := make([]Option, len(c.Options))
copy(options, c.Options)
options = append(options, globalOptions...)
cmd := c
res := &Response{req: req}
if path != nil {
for i, id := range path {
cmd = c.Sub(id)
if cmd == nil {
pathS := strings.Join(path[0:i], "/")
res.SetError(fmt.Errorf("Undefined command: '%s'", pathS), Client)
return res
}
options = append(options, cmd.Options...)
}
}
optionsMap := make(map[string]Option)
for _, opt := range options {
for _, name := range opt.Names {
optionsMap[name] = opt
}
}
for k, v := range req.options {
opt, ok := optionsMap[k]
if !ok {
res.SetError(fmt.Errorf("Unrecognized command option: '%s'", k), Client)
return res
}
for _, name := range opt.Names {
if _, ok = req.options[name]; name != k && ok {
res.SetError(fmt.Errorf("Duplicate command options were provided ('%s' and '%s')",
k, name), Client)
return res
}
}
kind := reflect.TypeOf(v).Kind()
if kind != opt.Type {
res.SetError(fmt.Errorf("Option '%s' should be type '%s', but got type '%s'",
k, opt.Type.String(), kind.String()), Client)
return res
}
}
cmd.f(req, res)
return res
}
// Sub returns the subcommand with the given id
func (c *Command) Sub(id string) *Command {
return c.subcommands[id]
}
func (c *Command) checkOptions(names map[string]bool) error {
for _, opt := range c.Options {
for _, name := range opt.Names {
if _, ok := names[name]; ok {
return fmt.Errorf("Multiple options are using the same name ('%s')", name)
}
names[name] = true
}
}
for _, cmd := range c.subcommands {
err := cmd.checkOptions(names)
if err != nil {
return err
}
}
return nil
}