request.go 4.11 KB
Newer Older
Matt Bell's avatar
Matt Bell committed
1 2
package commands

3
import (
4
	"errors"
5 6 7
	"fmt"
	"reflect"
	"strconv"
8 9 10

	"github.com/jbenet/go-ipfs/config"
	"github.com/jbenet/go-ipfs/core"
11 12
)

13 14
type optMap map[string]interface{}

15 16 17 18 19 20
type Context struct {
	ConfigRoot string
	Config     *config.Config
	Node       *core.IpfsNode
}

Matt Bell's avatar
Matt Bell committed
21
// Request represents a call to a command from a consumer
22 23 24
type Request interface {
	Path() []string
	Option(name string) (interface{}, bool)
25
	Options() map[string]interface{}
26
	SetOption(name string, val interface{})
27
	Arguments() []interface{} // TODO: make argument value type instead of using interface{}
28
	Context() *Context
29
	SetContext(Context)
30
	Command() *Command
31 32 33 34 35

	ConvertOptions(options map[string]Option) error
}

type request struct {
36 37 38 39 40 41
	path       []string
	options    optMap
	arguments  []interface{}
	cmd        *Command
	ctx        Context
	optionDefs map[string]Option
Matt Bell's avatar
Matt Bell committed
42 43
}

44 45
// Path returns the command path of this request
func (r *request) Path() []string {
46 47 48
	return r.path
}

49 50
// Option returns the value of the option for given name.
func (r *request) Option(name string) (interface{}, bool) {
51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70
	val, found := r.options[name]
	if found {
		return val, found
	}

	// if a value isn't defined for that name, we will try to look it up by its aliases

	// find the option with the specified name
	option, found := r.optionDefs[name]
	if found {
		// try all the possible names, break if we find a value
		for _, n := range option.Names {
			val, found := r.options[n]
			if found {
				return val, found
			}
		}
	}

	return nil, false
Matt Bell's avatar
Matt Bell committed
71 72
}

73 74 75 76 77 78 79 80 81
// Options returns a copy of the option map
func (r *request) Options() map[string]interface{} {
	output := make(optMap)
	for k, v := range r.options {
		output[k] = v
	}
	return output
}

82 83 84
// SetOption sets the value of the option for given name.
func (r *request) SetOption(name string, val interface{}) {
	r.options[name] = val
85 86
}

87
// Arguments returns the arguments slice
88
func (r *request) Arguments() []interface{} {
Matt Bell's avatar
Matt Bell committed
89
	return r.arguments
90
}
Matt Bell's avatar
Matt Bell committed
91

92 93 94 95
func (r *request) Context() *Context {
	return &r.ctx
}

96 97 98 99
func (r *request) SetContext(ctx Context) {
	r.ctx = ctx
}

100 101 102 103
func (r *request) Command() *Command {
	return r.cmd
}

Matt Bell's avatar
Matt Bell committed
104 105
type converter func(string) (interface{}, error)

106
var converters = map[reflect.Kind]converter{
Matt Bell's avatar
Matt Bell committed
107
	Bool: func(v string) (interface{}, error) {
108 109 110 111 112
		if v == "" {
			return true, nil
		}
		return strconv.ParseBool(v)
	},
Matt Bell's avatar
Matt Bell committed
113
	Int: func(v string) (interface{}, error) {
114 115
		return strconv.ParseInt(v, 0, 32)
	},
Matt Bell's avatar
Matt Bell committed
116
	Uint: func(v string) (interface{}, error) {
117 118
		return strconv.ParseInt(v, 0, 32)
	},
Matt Bell's avatar
Matt Bell committed
119
	Float: func(v string) (interface{}, error) {
120 121 122 123
		return strconv.ParseFloat(v, 64)
	},
}

124
func (r *request) ConvertOptions(options map[string]Option) error {
125 126 127 128 129
	converted := make(map[string]interface{})

	for k, v := range r.options {
		opt, ok := options[k]
		if !ok {
130
			continue
131 132 133 134 135 136 137 138
		}

		kind := reflect.TypeOf(v).Kind()
		var value interface{}

		if kind != opt.Type {
			if kind == String {
				convert := converters[opt.Type]
139 140 141 142 143
				str, ok := v.(string)
				if !ok {
					return errors.New("cast error")
				}
				val, err := convert(str)
144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163
				if err != nil {
					return fmt.Errorf("Could not convert string value '%s' to type '%s'",
						v, opt.Type.String())
				}
				value = val

			} else {
				return fmt.Errorf("Option '%s' should be type '%s', but got type '%s'",
					k, opt.Type.String(), kind.String())
			}
		} else {
			value = v
		}

		for _, name := range opt.Names {
			if _, ok := r.options[name]; name != k && ok {
				return fmt.Errorf("Duplicate command options were provided ('%s' and '%s')",
					k, name)
			}

Matt Bell's avatar
Matt Bell committed
164
			converted[name] = value
165 166 167 168 169 170 171
		}
	}

	r.options = converted
	return nil
}

172 173
// NewEmptyRequest initializes an empty request
func NewEmptyRequest() Request {
174
	return NewRequest(nil, nil, nil, nil, nil)
Matt Bell's avatar
Matt Bell committed
175 176
}

177
// NewRequest returns a request initialized with given arguments
178
func NewRequest(path []string, opts optMap, args []interface{}, cmd *Command, optDefs map[string]Option) Request {
179
	if path == nil {
Matt Bell's avatar
Matt Bell committed
180
		path = make([]string, 0)
Matt Bell's avatar
Matt Bell committed
181
	}
182 183 184 185
	if opts == nil {
		opts = make(map[string]interface{})
	}
	if args == nil {
186
		args = make([]interface{}, 0)
187
	}
188 189 190 191
	if optDefs == nil {
		optDefs = make(map[string]Option)
	}
	return &request{path, opts, args, cmd, Context{}, optDefs}
Matt Bell's avatar
Matt Bell committed
192
}