parse.go 2.75 KB
Newer Older
1 2 3
package cli

import (
4
	"errors"
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
5
	"fmt"
6
	"os"
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
7
	"strings"
8

9
	cmds "github.com/jbenet/go-ipfs/commands"
10 11
)

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
12 13
// Parse parses the input commandline string (cmd, flags, and args).
// returns the corresponding command Request object.
14
func Parse(input []string, roots ...*cmds.Command) (cmds.Request, *cmds.Command, error) {
15 16 17
	var root, cmd *cmds.Command
	var path, stringArgs []string
	var opts map[string]interface{}
18 19 20 21

	// use the root that matches the longest path (most accurately matches request)
	maxLength := 0
	for _, r := range roots {
22 23
		p, i, c := parsePath(input, r)
		o, s, err := parseOptions(i)
24 25 26 27
		if err != nil {
			return nil, nil, err
		}

28
		length := len(p)
29 30 31
		if length > maxLength {
			maxLength = length
			root = r
32 33 34 35
			path = p
			cmd = c
			opts = o
			stringArgs = s
36
		}
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
37
	}
38

39 40 41 42
	if maxLength == 0 {
		return nil, nil, errors.New("Not a valid subcommand")
	}

43 44 45 46 47 48
	args, err := parseArgs(stringArgs, cmd)
	if err != nil {
		return nil, nil, err
	}

	return cmds.NewRequest(path, opts, args, cmd), root, nil
49 50
}

51
// parsePath gets the command path from the command line input
52
func parsePath(input []string, root *cmds.Command) ([]string, []string, *cmds.Command) {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
53 54
	cmd := root
	i := 0
55

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
56 57 58 59
	for _, blob := range input {
		if strings.HasPrefix(blob, "-") {
			break
		}
60

61 62
		sub := cmd.Subcommand(blob)
		if sub == nil {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
63 64
			break
		}
65
		cmd = sub
66

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
67 68
		i++
	}
69

70
	return input[:i], input[i:], cmd
71 72
}

73
// parseOptions parses the raw string values of the given options
74
// returns the parsed options as strings, along with the CLI args
75
func parseOptions(input []string) (map[string]interface{}, []string, error) {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
76
	opts := make(map[string]interface{})
77
	args := []string{}
78

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
79 80
	for i := 0; i < len(input); i++ {
		blob := input[i]
81

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
82 83 84
		if strings.HasPrefix(blob, "-") {
			name := blob[1:]
			value := ""
85

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
86 87 88 89
			// support single and double dash
			if strings.HasPrefix(name, "-") {
				name = name[1:]
			}
90

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
91 92 93 94 95
			if strings.Contains(name, "=") {
				split := strings.SplitN(name, "=", 2)
				name = split[0]
				value = split[1]
			}
96

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
97 98 99
			if _, ok := opts[name]; ok {
				return nil, nil, fmt.Errorf("Duplicate values for option '%s'", name)
			}
100

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
101
			opts[name] = value
102

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
103 104 105 106
		} else {
			args = append(args, blob)
		}
	}
107

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
108
	return opts, args, nil
109
}
110 111 112 113

// Note that the argument handling here is dumb, it does not do any error-checking.
// (Arguments are further processed when the request is passed to the command to run)
func parseArgs(stringArgs []string, cmd *cmds.Command) ([]interface{}, error) {
114 115
	var argDef cmds.Argument
	args := make([]interface{}, len(stringArgs))
116

117 118 119
	for i, arg := range stringArgs {
		if i < len(cmd.Arguments) {
			argDef = cmd.Arguments[i]
120 121
		}

122 123
		if argDef.Type == cmds.ArgString {
			args[i] = arg
124 125

		} else {
126
			in, err := os.Open(arg)
127 128 129 130 131 132 133 134 135
			if err != nil {
				return nil, err
			}
			args[i] = in
		}
	}

	return args, nil
}