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

import (
4
	"bytes"
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
5
	"fmt"
6
	"os"
7
	"path"
8
	"path/filepath"
9
	"runtime"
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
10
	"strings"
11

12 13 14
	cmds "github.com/ipfs/go-ipfs/commands"
	files "github.com/ipfs/go-ipfs/commands/files"
	u "github.com/ipfs/go-ipfs/util"
15 16
)

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
17 18
// Parse parses the input commandline string (cmd, flags, and args).
// returns the corresponding command Request object.
19
func Parse(input []string, stdin *os.File, root *cmds.Command) (cmds.Request, *cmds.Command, []string, error) {
Etienne Laurin's avatar
Etienne Laurin committed
20
	path, opts, stringVals, cmd, err := parseOpts(input, root)
21
	if err != nil {
Etienne Laurin's avatar
Etienne Laurin committed
22
		return nil, nil, path, err
23 24
	}

25 26
	optDefs, err := root.GetOptions(path)
	if err != nil {
27
		return nil, cmd, path, err
28 29
	}

30 31 32 33
	req, err := cmds.NewRequest(path, opts, nil, nil, cmd, optDefs)
	if err != nil {
		return nil, cmd, path, err
	}
34

35 36 37 38 39 40 41
	// if -r is provided, and it is associated with the package builtin
	// recursive path option, allow recursive file paths
	recursiveOpt := req.Option(cmds.RecShort)
	recursive := false
	if recursiveOpt != nil && recursiveOpt.Definition() == cmds.OptionRecursivePath {
		recursive, _, err = recursiveOpt.Bool()
		if err != nil {
42
			return req, nil, nil, u.ErrCast()
43
		}
44
	}
45

46
	stringArgs, fileArgs, err := parseArgs(stringVals, stdin, cmd.Arguments, recursive, root)
47
	if err != nil {
48
		return req, cmd, path, err
49
	}
50 51
	req.SetArguments(stringArgs)

52 53 54 55
	if len(fileArgs) > 0 {
		file := files.NewSliceFile("", "", fileArgs)
		req.SetFiles(file)
	}
56 57 58

	err = cmd.CheckArguments(req)
	if err != nil {
59
		return req, cmd, path, err
60 61
	}

62
	return req, cmd, path, nil
63 64
}

Etienne Laurin's avatar
Etienne Laurin committed
65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84
// Parse a command line made up of sub-commands, short arguments, long arguments and positional arguments
func parseOpts(args []string, root *cmds.Command) (
	path []string,
	opts map[string]interface{},
	stringVals []string,
	cmd *cmds.Command,
	err error,
) {
	path = make([]string, 0, len(args))
	stringVals = make([]string, 0, len(args))
	optDefs := map[string]cmds.Option{}
	opts = map[string]interface{}{}
	cmd = root

	// parseFlag checks that a flag is valid and saves it into opts
	// Returns true if the optional second argument is used
	parseFlag := func(name string, arg *string, mustUse bool) (bool, error) {
		if _, ok := opts[name]; ok {
			return false, fmt.Errorf("Duplicate values for option '%s'", name)
		}
85

Etienne Laurin's avatar
Etienne Laurin committed
86 87 88 89
		optDef, found := optDefs[name]
		if !found {
			err = fmt.Errorf("Unrecognized option '%s'", name)
			return false, err
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
90
		}
91 92 93 94 95 96
		// mustUse implies that you must use the argument given after the '='
		// eg. -r=true means you must take true into consideration
		//		mustUse == true in the above case
		// eg. ipfs -r <file> means disregard <file> since there is no '='
		//		mustUse == false in the above situation
		//arg == nil implies the flag was specified without an argument
Etienne Laurin's avatar
Etienne Laurin committed
97
		if optDef.Type() == cmds.Bool {
98 99 100 101 102 103 104 105 106 107 108 109 110 111
			if arg == nil || !mustUse {
				opts[name] = true
				return false, nil
			}
			argVal := strings.ToLower(*arg)
			switch argVal {
			case "true":
				opts[name] = true
				return true, nil
			case "false":
				opts[name] = false
				return true, nil
			default:
				return true, fmt.Errorf("Option '%s' takes true/false arguments, but was passed '%s'", name, argVal)
Etienne Laurin's avatar
Etienne Laurin committed
112 113 114 115 116 117 118
			}
		} else {
			if arg == nil {
				return true, fmt.Errorf("Missing argument for option '%s'", name)
			}
			opts[name] = *arg
			return true, nil
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
119 120
		}
	}
121

Etienne Laurin's avatar
Etienne Laurin committed
122 123 124 125
	optDefs, err = root.GetOptions(path)
	if err != nil {
		return
	}
126

Etienne Laurin's avatar
Etienne Laurin committed
127 128 129 130 131 132 133
	consumed := false
	for i, arg := range args {
		switch {
		case consumed:
			// arg was already consumed by the preceding flag
			consumed = false
			continue
134

Etienne Laurin's avatar
Etienne Laurin committed
135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156
		case arg == "--":
			// treat all remaining arguments as positional arguments
			stringVals = append(stringVals, args[i+1:]...)
			return

		case strings.HasPrefix(arg, "--"):
			// arg is a long flag, with an optional argument specified
			// using `=' or in args[i+1]
			var slurped bool
			var next *string
			split := strings.SplitN(arg, "=", 2)
			if len(split) == 2 {
				slurped = false
				arg = split[0]
				next = &split[1]
			} else {
				slurped = true
				if i+1 < len(args) {
					next = &args[i+1]
				} else {
					next = nil
				}
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
157
			}
Etienne Laurin's avatar
Etienne Laurin committed
158 159 160
			consumed, err = parseFlag(arg[2:], next, len(split) == 2)
			if err != nil {
				return
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
161
			}
Etienne Laurin's avatar
Etienne Laurin committed
162 163
			if !slurped {
				consumed = false
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
164
			}
165

Etienne Laurin's avatar
Etienne Laurin committed
166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189
		case strings.HasPrefix(arg, "-") && arg != "-":
			// args is one or more flags in short form, followed by an optional argument
			// all flags except the last one have type bool
			for arg = arg[1:]; len(arg) != 0; arg = arg[1:] {
				var rest *string
				var slurped bool
				mustUse := false
				if len(arg) > 1 {
					slurped = false
					str := arg[1:]
					if len(str) > 0 && str[0] == '=' {
						str = str[1:]
						mustUse = true
					}
					rest = &str
				} else {
					slurped = true
					if i+1 < len(args) {
						rest = &args[i+1]
					} else {
						rest = nil
					}
				}
				var end bool
rht's avatar
rht committed
190
				end, err = parseFlag(arg[:1], rest, mustUse)
Etienne Laurin's avatar
Etienne Laurin committed
191 192 193 194 195 196 197 198
				if err != nil {
					return
				}
				if end {
					consumed = slurped
					break
				}
			}
199

Etienne Laurin's avatar
Etienne Laurin committed
200 201 202 203 204 205 206 207 208 209
		default:
			// arg is a sub-command or a positional argument
			sub := cmd.Subcommand(arg)
			if sub != nil {
				cmd = sub
				path = append(path, arg)
				optDefs, err = root.GetOptions(path)
				if err != nil {
					return
				}
210 211 212 213 214 215 216

				// If we've come across an external binary call, pass all the remaining
				// arguments on to it
				if cmd.External {
					stringVals = append(stringVals, args[i+1:]...)
					return
				}
Etienne Laurin's avatar
Etienne Laurin committed
217 218 219
			} else {
				stringVals = append(stringVals, arg)
			}
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
220 221
		}
	}
Etienne Laurin's avatar
Etienne Laurin committed
222
	return
223
}
224

225
func parseArgs(inputs []string, stdin *os.File, argDefs []cmds.Argument, recursive bool, root *cmds.Command) ([]string, []files.File, error) {
226 227 228 229 230
	// ignore stdin on Windows
	if runtime.GOOS == "windows" {
		stdin = nil
	}

231 232
	// check if stdin is coming from terminal or is being piped in
	if stdin != nil {
233
		if term, err := isTerminal(stdin); err != nil {
234
			return nil, nil, err
235 236
		} else if term {
			stdin = nil // set to nil so we ignore it
237 238 239
		}
	}

240
	// count required argument definitions
241
	numRequired := 0
242
	for _, argDef := range argDefs {
243
		if argDef.Required {
244
			numRequired++
245
		}
246
	}
247

248 249 250
	// count number of values provided by user.
	// if there is at least one ArgDef, we can safely trigger the inputs loop
	// below to parse stdin.
251
	numInputs := len(inputs)
252
	if len(argDefs) > 0 && argDefs[len(argDefs)-1].SupportsStdin && stdin != nil {
253
		numInputs += 1
254 255
	}

256 257 258
	// if we have more arg values provided than argument definitions,
	// and the last arg definition is not variadic (or there are no definitions), return an error
	notVariadic := len(argDefs) == 0 || !argDefs[len(argDefs)-1].Variadic
Christian Couder's avatar
Christian Couder committed
259
	if notVariadic && len(inputs) > len(argDefs) {
260 261 262 263 264 265 266 267 268
		suggestions := suggestUnknownCmd(inputs, root)

		if len(suggestions) > 1 {
			return nil, nil, fmt.Errorf("Unknown Command \"%s\"\n\nDid you mean any of these?\n\n\t%s", inputs[0], strings.Join(suggestions, "\n\t"))
		} else if len(suggestions) > 0 {
			return nil, nil, fmt.Errorf("Unknown Command \"%s\"\n\nDid you mean this?\n\n\t%s", inputs[0], suggestions[0])
		} else {
			return nil, nil, fmt.Errorf("Unknown Command \"%s\"\n", inputs[0])
		}
269 270
	}

271
	stringArgs := make([]string, 0, numInputs)
272
	fileArgs := make([]files.File, 0, numInputs)
273 274

	argDefIndex := 0 // the index of the current argument definition
275
	for i := 0; i < numInputs; i++ {
276
		argDef := getArgDef(argDefIndex, argDefs)
277

278
		// skip optional argument definitions if there aren't sufficient remaining inputs
279 280 281 282 283
		for numInputs-i <= numRequired && !argDef.Required {
			argDefIndex++
			argDef = getArgDef(argDefIndex, argDefs)
		}
		if argDef.Required {
284
			numRequired--
285
		}
286

287
		var err error
288
		if argDef.Type == cmds.ArgString {
289
			if stdin == nil || !argDef.SupportsStdin {
290
				// add string values
291
				stringArgs, inputs = appendString(stringArgs, inputs)
292

293
			} else {
294 295 296 297 298 299 300 301 302
				if len(inputs) > 0 {
					// don't use stdin if we have inputs
					stdin = nil
				} else {
					// if we have a stdin, read it in and use the data as a string value
					stringArgs, stdin, err = appendStdinAsString(stringArgs, stdin)
					if err != nil {
						return nil, nil, err
					}
303 304
				}
			}
305
		} else if argDef.Type == cmds.ArgFile {
306
			if stdin == nil || !argDef.SupportsStdin {
307
				// treat stringArg values as file paths
308
				fileArgs, inputs, err = appendFile(fileArgs, inputs, argDef, recursive)
309 310 311 312
				if err != nil {
					return nil, nil, err
				}

313
			} else {
314 315 316 317 318 319 320
				if len(inputs) > 0 {
					// don't use stdin if we have inputs
					stdin = nil
				} else {
					// if we have a stdin, create a file from it
					fileArgs, stdin = appendStdinAsFile(fileArgs, stdin)
				}
321 322
			}
		}
323 324

		argDefIndex++
325 326
	}

327
	// check to make sure we didn't miss any required arguments
328 329 330 331 332
	if len(argDefs) > argDefIndex {
		for _, argDef := range argDefs[argDefIndex:] {
			if argDef.Required {
				return nil, nil, fmt.Errorf("Argument '%s' is required", argDef.Name)
			}
333 334 335
		}
	}

336
	return stringArgs, fileArgs, nil
337
}
338

339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357
func getArgDef(i int, argDefs []cmds.Argument) *cmds.Argument {
	if i < len(argDefs) {
		// get the argument definition (usually just argDefs[i])
		return &argDefs[i]

	} else if len(argDefs) > 0 {
		// but if i > len(argDefs) we use the last argument definition)
		return &argDefs[len(argDefs)-1]
	}

	// only happens if there aren't any definitions
	return nil
}

func appendString(args, inputs []string) ([]string, []string) {
	return append(args, inputs[0]), inputs[1:]
}

func appendStdinAsString(args []string, stdin *os.File) ([]string, *os.File, error) {
358
	buf := new(bytes.Buffer)
359 360 361 362 363 364

	_, err := buf.ReadFrom(stdin)
	if err != nil {
		return nil, nil, err
	}

365 366
	input := strings.TrimSpace(buf.String())
	return append(args, strings.Split(input, "\n")...), nil, nil
367 368
}

369
func appendFile(args []files.File, inputs []string, argDef *cmds.Argument, recursive bool) ([]files.File, []string, error) {
370
	fpath := filepath.ToSlash(filepath.Clean(inputs[0]))
371

372 373 374 375 376 377 378
	if fpath == "." {
		cwd, err := os.Getwd()
		if err != nil {
			return nil, nil, err
		}
		fpath = cwd
	}
Jeromy's avatar
Jeromy committed
379
	stat, err := os.Lstat(fpath)
380 381 382 383 384 385 386
	if err != nil {
		return nil, nil, err
	}

	if stat.IsDir() {
		if !argDef.Recursive {
			err = fmt.Errorf("Invalid path '%s', argument '%s' does not support directories",
387
				fpath, argDef.Name)
388 389 390 391
			return nil, nil, err
		}
		if !recursive {
			err = fmt.Errorf("'%s' is a directory, use the '-%s' flag to specify directories",
392
				fpath, cmds.RecShort)
393 394 395 396
			return nil, nil, err
		}
	}

397
	arg, err := files.NewSerialFile(path.Base(fpath), fpath, stat)
398 399 400 401 402 403
	if err != nil {
		return nil, nil, err
	}
	return append(args, arg), inputs[1:], nil
}

404
func appendStdinAsFile(args []files.File, stdin *os.File) ([]files.File, *os.File) {
405
	arg := files.NewReaderFile("", "", stdin, nil)
406 407 408 409 410 411 412 413 414 415 416 417 418 419 420
	return append(args, arg), nil
}

// isTerminal returns true if stdin is a Stdin pipe (e.g. `cat file | ipfs`),
// and false otherwise (e.g. nothing is being piped in, so stdin is
// coming from the terminal)
func isTerminal(stdin *os.File) (bool, error) {
	stat, err := stdin.Stat()
	if err != nil {
		return false, err
	}

	// if stdin is a CharDevice, return true
	return ((stat.Mode() & os.ModeCharDevice) != 0), nil
}