2017-02-07 22:33:23 +01:00
|
|
|
package shellwords
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
|
|
|
"os"
|
|
|
|
"regexp"
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
ParseEnv bool = false
|
|
|
|
ParseBacktick bool = false
|
|
|
|
)
|
|
|
|
|
|
|
|
var envRe = regexp.MustCompile(`\$({[a-zA-Z0-9_]+}|[a-zA-Z0-9_]+)`)
|
|
|
|
|
|
|
|
func isSpace(r rune) bool {
|
|
|
|
switch r {
|
|
|
|
case ' ', '\t', '\r', '\n':
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
func replaceEnv(s string) string {
|
|
|
|
return envRe.ReplaceAllStringFunc(s, func(s string) string {
|
|
|
|
s = s[1:]
|
|
|
|
if s[0] == '{' {
|
|
|
|
s = s[1 : len(s)-1]
|
|
|
|
}
|
|
|
|
return os.Getenv(s)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
type Parser struct {
|
|
|
|
ParseEnv bool
|
|
|
|
ParseBacktick bool
|
2017-04-11 17:10:46 +02:00
|
|
|
Position int
|
2017-02-07 22:33:23 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func NewParser() *Parser {
|
2017-04-11 17:10:46 +02:00
|
|
|
return &Parser{ParseEnv, ParseBacktick, 0}
|
2017-02-07 22:33:23 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func (p *Parser) Parse(line string) ([]string, error) {
|
|
|
|
args := []string{}
|
|
|
|
buf := ""
|
|
|
|
var escaped, doubleQuoted, singleQuoted, backQuote bool
|
|
|
|
backtick := ""
|
|
|
|
|
2017-04-11 17:10:46 +02:00
|
|
|
pos := -1
|
|
|
|
got := false
|
|
|
|
|
|
|
|
loop:
|
|
|
|
for i, r := range line {
|
2017-02-07 22:33:23 +01:00
|
|
|
if escaped {
|
|
|
|
buf += string(r)
|
|
|
|
escaped = false
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
if r == '\\' {
|
|
|
|
if singleQuoted {
|
|
|
|
buf += string(r)
|
|
|
|
} else {
|
|
|
|
escaped = true
|
|
|
|
}
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
if isSpace(r) {
|
|
|
|
if singleQuoted || doubleQuoted || backQuote {
|
|
|
|
buf += string(r)
|
|
|
|
backtick += string(r)
|
2017-04-11 17:10:46 +02:00
|
|
|
} else if got {
|
2017-02-07 22:33:23 +01:00
|
|
|
if p.ParseEnv {
|
|
|
|
buf = replaceEnv(buf)
|
|
|
|
}
|
|
|
|
args = append(args, buf)
|
|
|
|
buf = ""
|
2017-04-11 17:10:46 +02:00
|
|
|
got = false
|
2017-02-07 22:33:23 +01:00
|
|
|
}
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
switch r {
|
|
|
|
case '`':
|
|
|
|
if !singleQuoted && !doubleQuoted {
|
|
|
|
if p.ParseBacktick {
|
|
|
|
if backQuote {
|
|
|
|
out, err := shellRun(backtick)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
buf = out
|
|
|
|
}
|
|
|
|
backtick = ""
|
|
|
|
backQuote = !backQuote
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
backtick = ""
|
|
|
|
backQuote = !backQuote
|
|
|
|
}
|
|
|
|
case '"':
|
|
|
|
if !singleQuoted {
|
|
|
|
doubleQuoted = !doubleQuoted
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
case '\'':
|
|
|
|
if !doubleQuoted {
|
|
|
|
singleQuoted = !singleQuoted
|
|
|
|
continue
|
|
|
|
}
|
2017-04-11 17:10:46 +02:00
|
|
|
case ';', '&', '|', '<', '>':
|
|
|
|
if !(escaped || singleQuoted || doubleQuoted || backQuote) {
|
|
|
|
pos = i
|
|
|
|
break loop
|
|
|
|
}
|
2017-02-07 22:33:23 +01:00
|
|
|
}
|
|
|
|
|
2017-04-11 17:10:46 +02:00
|
|
|
got = true
|
2017-02-07 22:33:23 +01:00
|
|
|
buf += string(r)
|
|
|
|
if backQuote {
|
|
|
|
backtick += string(r)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-04-11 17:10:46 +02:00
|
|
|
if got {
|
2017-02-07 22:33:23 +01:00
|
|
|
if p.ParseEnv {
|
|
|
|
buf = replaceEnv(buf)
|
|
|
|
}
|
|
|
|
args = append(args, buf)
|
|
|
|
}
|
|
|
|
|
|
|
|
if escaped || singleQuoted || doubleQuoted || backQuote {
|
|
|
|
return nil, errors.New("invalid command line string")
|
|
|
|
}
|
|
|
|
|
2017-04-11 17:10:46 +02:00
|
|
|
p.Position = pos
|
|
|
|
|
2017-02-07 22:33:23 +01:00
|
|
|
return args, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func Parse(line string) ([]string, error) {
|
|
|
|
return NewParser().Parse(line)
|
|
|
|
}
|