2014-08-05 16:17:40 -04:00
|
|
|
package parser
|
|
|
|
|
2014-08-07 01:56:44 -04:00
|
|
|
// line parsers are dispatch calls that parse a single unit of text into a
|
|
|
|
// Node object which contains the whole statement. Dockerfiles have varied
|
|
|
|
// (but not usually unique, see ONBUILD for a unique example) parsing rules
|
|
|
|
// per-command, and these unify the processing in a way that makes it
|
|
|
|
// manageable.
|
|
|
|
|
2014-08-05 16:17:40 -04:00
|
|
|
import (
|
|
|
|
"encoding/json"
|
2014-08-07 03:42:10 -04:00
|
|
|
"errors"
|
2014-08-31 08:39:36 -04:00
|
|
|
"fmt"
|
2014-08-05 16:17:40 -04:00
|
|
|
"strings"
|
2014-09-25 22:28:24 -04:00
|
|
|
"unicode"
|
2014-08-05 16:17:40 -04:00
|
|
|
)
|
|
|
|
|
2014-08-07 03:42:10 -04:00
|
|
|
var (
|
2015-01-03 00:40:43 -05:00
|
|
|
errDockerfileNotStringArray = errors.New("When using JSON array syntax, arrays must be comprised of strings only.")
|
2014-08-07 03:42:10 -04:00
|
|
|
)
|
|
|
|
|
2014-08-05 16:17:40 -04:00
|
|
|
// ignore the current argument. This will still leave a command parsed, but
|
|
|
|
// will not incorporate the arguments into the ast.
|
2014-08-13 06:07:41 -04:00
|
|
|
func parseIgnore(rest string) (*Node, map[string]bool, error) {
|
|
|
|
return &Node{}, nil, nil
|
2014-08-05 16:17:40 -04:00
|
|
|
}
|
|
|
|
|
2014-08-07 01:56:44 -04:00
|
|
|
// used for onbuild. Could potentially be used for anything that represents a
|
|
|
|
// statement with sub-statements.
|
|
|
|
//
|
|
|
|
// ONBUILD RUN foo bar -> (onbuild (run foo bar))
|
|
|
|
//
|
2014-08-13 06:07:41 -04:00
|
|
|
func parseSubCommand(rest string) (*Node, map[string]bool, error) {
|
2014-08-05 16:17:40 -04:00
|
|
|
_, child, err := parseLine(rest)
|
|
|
|
if err != nil {
|
2014-08-13 06:07:41 -04:00
|
|
|
return nil, nil, err
|
2014-08-05 16:17:40 -04:00
|
|
|
}
|
|
|
|
|
2014-08-13 06:07:41 -04:00
|
|
|
return &Node{Children: []*Node{child}}, nil, nil
|
2014-08-05 16:17:40 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// parse environment like statements. Note that this does *not* handle
|
|
|
|
// variable interpolation, which will be handled in the evaluator.
|
2014-08-13 06:07:41 -04:00
|
|
|
func parseEnv(rest string) (*Node, map[string]bool, error) {
|
2014-09-25 22:28:24 -04:00
|
|
|
// This is kind of tricky because we need to support the old
|
|
|
|
// variant: ENV name value
|
|
|
|
// as well as the new one: ENV name=value ...
|
|
|
|
// The trigger to know which one is being used will be whether we hit
|
|
|
|
// a space or = first. space ==> old, "=" ==> new
|
|
|
|
|
|
|
|
const (
|
|
|
|
inSpaces = iota // looking for start of a word
|
|
|
|
inWord
|
|
|
|
inQuote
|
|
|
|
)
|
|
|
|
|
|
|
|
words := []string{}
|
|
|
|
phase := inSpaces
|
|
|
|
word := ""
|
|
|
|
quote := '\000'
|
|
|
|
blankOK := false
|
|
|
|
var ch rune
|
|
|
|
|
|
|
|
for pos := 0; pos <= len(rest); pos++ {
|
|
|
|
if pos != len(rest) {
|
|
|
|
ch = rune(rest[pos])
|
|
|
|
}
|
|
|
|
|
|
|
|
if phase == inSpaces { // Looking for start of word
|
|
|
|
if pos == len(rest) { // end of input
|
|
|
|
break
|
|
|
|
}
|
|
|
|
if unicode.IsSpace(ch) { // skip spaces
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
phase = inWord // found it, fall thru
|
|
|
|
}
|
|
|
|
if (phase == inWord || phase == inQuote) && (pos == len(rest)) {
|
|
|
|
if blankOK || len(word) > 0 {
|
|
|
|
words = append(words, word)
|
|
|
|
}
|
|
|
|
break
|
|
|
|
}
|
|
|
|
if phase == inWord {
|
|
|
|
if unicode.IsSpace(ch) {
|
|
|
|
phase = inSpaces
|
|
|
|
if blankOK || len(word) > 0 {
|
|
|
|
words = append(words, word)
|
|
|
|
|
|
|
|
// Look for = and if no there assume
|
|
|
|
// we're doing the old stuff and
|
|
|
|
// just read the rest of the line
|
|
|
|
if !strings.Contains(word, "=") {
|
|
|
|
word = strings.TrimSpace(rest[pos:])
|
|
|
|
words = append(words, word)
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
word = ""
|
|
|
|
blankOK = false
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if ch == '\'' || ch == '"' {
|
|
|
|
quote = ch
|
|
|
|
blankOK = true
|
|
|
|
phase = inQuote
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if ch == '\\' {
|
|
|
|
if pos+1 == len(rest) {
|
|
|
|
continue // just skip \ at end
|
|
|
|
}
|
|
|
|
pos++
|
|
|
|
ch = rune(rest[pos])
|
|
|
|
}
|
|
|
|
word += string(ch)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if phase == inQuote {
|
|
|
|
if ch == quote {
|
|
|
|
phase = inWord
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if ch == '\\' {
|
|
|
|
if pos+1 == len(rest) {
|
|
|
|
phase = inWord
|
|
|
|
continue // just skip \ at end
|
|
|
|
}
|
|
|
|
pos++
|
|
|
|
ch = rune(rest[pos])
|
|
|
|
}
|
|
|
|
word += string(ch)
|
|
|
|
}
|
|
|
|
}
|
2014-08-31 08:39:36 -04:00
|
|
|
|
2014-09-25 22:28:24 -04:00
|
|
|
if len(words) == 0 {
|
|
|
|
return nil, nil, fmt.Errorf("ENV must have some arguments")
|
2014-08-31 08:39:36 -04:00
|
|
|
}
|
|
|
|
|
2014-09-25 22:28:24 -04:00
|
|
|
// Old format (ENV name value)
|
|
|
|
var rootnode *Node
|
|
|
|
|
|
|
|
if !strings.Contains(words[0], "=") {
|
|
|
|
node := &Node{}
|
|
|
|
rootnode = node
|
|
|
|
strs := TOKEN_WHITESPACE.Split(rest, 2)
|
|
|
|
|
|
|
|
if len(strs) < 2 {
|
|
|
|
return nil, nil, fmt.Errorf("ENV must have two arguments")
|
|
|
|
}
|
|
|
|
|
|
|
|
node.Value = strs[0]
|
|
|
|
node.Next = &Node{}
|
|
|
|
node.Next.Value = strs[1]
|
|
|
|
} else {
|
|
|
|
var prevNode *Node
|
|
|
|
for i, word := range words {
|
|
|
|
if !strings.Contains(word, "=") {
|
|
|
|
return nil, nil, fmt.Errorf("Syntax error - can't find = in %q. Must be of the form: name=value", word)
|
|
|
|
}
|
|
|
|
parts := strings.SplitN(word, "=", 2)
|
|
|
|
|
|
|
|
name := &Node{}
|
|
|
|
value := &Node{}
|
|
|
|
|
|
|
|
name.Next = value
|
|
|
|
name.Value = parts[0]
|
|
|
|
value.Value = parts[1]
|
|
|
|
|
|
|
|
if i == 0 {
|
|
|
|
rootnode = name
|
|
|
|
} else {
|
|
|
|
prevNode.Next = name
|
|
|
|
}
|
|
|
|
prevNode = value
|
|
|
|
}
|
|
|
|
}
|
2014-08-05 16:17:40 -04:00
|
|
|
|
2014-08-13 06:07:41 -04:00
|
|
|
return rootnode, nil, nil
|
2014-08-05 16:17:40 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// parses a whitespace-delimited set of arguments. The result is effectively a
|
|
|
|
// linked list of string arguments.
|
2014-08-13 06:07:41 -04:00
|
|
|
func parseStringsWhitespaceDelimited(rest string) (*Node, map[string]bool, error) {
|
2014-08-10 07:01:10 -04:00
|
|
|
node := &Node{}
|
2014-08-05 16:17:40 -04:00
|
|
|
rootnode := node
|
2014-08-05 18:41:09 -04:00
|
|
|
prevnode := node
|
2014-08-05 16:17:40 -04:00
|
|
|
for _, str := range TOKEN_WHITESPACE.Split(rest, -1) { // use regexp
|
2014-08-05 18:41:09 -04:00
|
|
|
prevnode = node
|
|
|
|
node.Value = str
|
2014-08-10 07:01:10 -04:00
|
|
|
node.Next = &Node{}
|
2014-08-05 16:17:40 -04:00
|
|
|
node = node.Next
|
|
|
|
}
|
|
|
|
|
2014-08-05 18:41:09 -04:00
|
|
|
// XXX to get around regexp.Split *always* providing an empty string at the
|
|
|
|
// end due to how our loop is constructed, nil out the last node in the
|
|
|
|
// chain.
|
|
|
|
prevnode.Next = nil
|
|
|
|
|
2014-08-13 06:07:41 -04:00
|
|
|
return rootnode, nil, nil
|
2014-08-05 16:17:40 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// parsestring just wraps the string in quotes and returns a working node.
|
2014-08-13 06:07:41 -04:00
|
|
|
func parseString(rest string) (*Node, map[string]bool, error) {
|
|
|
|
n := &Node{}
|
|
|
|
n.Value = rest
|
|
|
|
return n, nil, nil
|
2014-08-05 16:17:40 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// parseJSON converts JSON arrays to an AST.
|
2014-08-13 06:07:41 -04:00
|
|
|
func parseJSON(rest string) (*Node, map[string]bool, error) {
|
2015-01-03 00:40:43 -05:00
|
|
|
var myJson []interface{}
|
2014-08-05 16:17:40 -04:00
|
|
|
if err := json.Unmarshal([]byte(rest), &myJson); err != nil {
|
2014-08-13 06:07:41 -04:00
|
|
|
return nil, nil, err
|
2014-08-05 16:17:40 -04:00
|
|
|
}
|
|
|
|
|
2015-01-03 00:40:43 -05:00
|
|
|
var top, prev *Node
|
2014-08-05 16:17:40 -04:00
|
|
|
for _, str := range myJson {
|
2015-01-03 00:40:43 -05:00
|
|
|
if s, ok := str.(string); !ok {
|
|
|
|
return nil, nil, errDockerfileNotStringArray
|
|
|
|
} else {
|
|
|
|
node := &Node{Value: s}
|
|
|
|
if prev == nil {
|
|
|
|
top = node
|
|
|
|
} else {
|
|
|
|
prev.Next = node
|
|
|
|
}
|
|
|
|
prev = node
|
2014-08-05 16:17:40 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-03 00:40:43 -05:00
|
|
|
return top, map[string]bool{"json": true}, nil
|
2014-08-05 16:17:40 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// parseMaybeJSON determines if the argument appears to be a JSON array. If
|
|
|
|
// so, passes to parseJSON; if not, quotes the result and returns a single
|
|
|
|
// node.
|
2014-08-13 06:07:41 -04:00
|
|
|
func parseMaybeJSON(rest string) (*Node, map[string]bool, error) {
|
2014-08-05 16:17:40 -04:00
|
|
|
rest = strings.TrimSpace(rest)
|
|
|
|
|
2014-08-19 07:14:21 -04:00
|
|
|
node, attrs, err := parseJSON(rest)
|
2014-08-08 16:44:57 -04:00
|
|
|
|
2014-08-19 07:14:21 -04:00
|
|
|
if err == nil {
|
|
|
|
return node, attrs, nil
|
|
|
|
}
|
2015-01-03 00:40:43 -05:00
|
|
|
if err == errDockerfileNotStringArray {
|
2014-08-19 07:14:21 -04:00
|
|
|
return nil, nil, err
|
2014-08-05 16:17:40 -04:00
|
|
|
}
|
|
|
|
|
2014-08-19 07:14:21 -04:00
|
|
|
node = &Node{}
|
2014-08-05 18:41:09 -04:00
|
|
|
node.Value = rest
|
2014-08-13 06:07:41 -04:00
|
|
|
return node, nil, nil
|
2014-08-05 16:17:40 -04:00
|
|
|
}
|
2014-09-11 09:27:51 -04:00
|
|
|
|
|
|
|
// parseMaybeJSONToList determines if the argument appears to be a JSON array. If
|
|
|
|
// so, passes to parseJSON; if not, attmpts to parse it as a whitespace
|
|
|
|
// delimited string.
|
|
|
|
func parseMaybeJSONToList(rest string) (*Node, map[string]bool, error) {
|
|
|
|
rest = strings.TrimSpace(rest)
|
|
|
|
|
|
|
|
node, attrs, err := parseJSON(rest)
|
|
|
|
|
|
|
|
if err == nil {
|
|
|
|
return node, attrs, nil
|
|
|
|
}
|
2015-01-03 00:40:43 -05:00
|
|
|
if err == errDockerfileNotStringArray {
|
2014-09-11 09:27:51 -04:00
|
|
|
return nil, nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return parseStringsWhitespaceDelimited(rest)
|
|
|
|
}
|