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
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
)
|
|
|
|
|
2014-08-07 03:42:10 -04:00
|
|
|
var (
|
2014-08-13 06:07:41 -04:00
|
|
|
errDockerfileJSONNesting = errors.New("You may not nest arrays in Dockerfile statements.")
|
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-08-10 07:01:10 -04:00
|
|
|
node := &Node{}
|
2014-08-05 16:17:40 -04:00
|
|
|
rootnode := node
|
|
|
|
strs := TOKEN_WHITESPACE.Split(rest, 2)
|
2014-08-31 08:39:36 -04:00
|
|
|
|
|
|
|
if len(strs) < 2 {
|
|
|
|
return nil, nil, fmt.Errorf("ENV must have two arguments")
|
|
|
|
}
|
|
|
|
|
2014-08-05 18:41:09 -04:00
|
|
|
node.Value = strs[0]
|
2014-08-10 07:01:10 -04:00
|
|
|
node.Next = &Node{}
|
2014-08-05 18:41:09 -04:00
|
|
|
node.Next.Value = strs[1]
|
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) {
|
2014-08-05 16:17:40 -04:00
|
|
|
var (
|
|
|
|
myJson []interface{}
|
2014-08-10 07:01:10 -04:00
|
|
|
next = &Node{}
|
2014-08-05 16:17:40 -04:00
|
|
|
orignext = next
|
2014-08-05 18:41:09 -04:00
|
|
|
prevnode = next
|
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
|
|
|
}
|
|
|
|
|
|
|
|
for _, str := range myJson {
|
|
|
|
switch str.(type) {
|
2014-08-08 16:44:57 -04:00
|
|
|
case string:
|
2014-08-05 16:17:40 -04:00
|
|
|
case float64:
|
|
|
|
str = strconv.FormatFloat(str.(float64), 'G', -1, 64)
|
2014-08-08 16:44:57 -04:00
|
|
|
default:
|
2014-08-13 06:07:41 -04:00
|
|
|
return nil, nil, errDockerfileJSONNesting
|
2014-08-05 16:17:40 -04:00
|
|
|
}
|
2014-08-05 18:41:09 -04:00
|
|
|
next.Value = str.(string)
|
2014-08-10 07:01:10 -04:00
|
|
|
next.Next = &Node{}
|
2014-08-05 18:41:09 -04:00
|
|
|
prevnode = next
|
2014-08-05 16:17:40 -04:00
|
|
|
next = next.Next
|
|
|
|
}
|
|
|
|
|
2014-08-05 18:41:09 -04:00
|
|
|
prevnode.Next = nil
|
|
|
|
|
2014-08-13 06:07:41 -04:00
|
|
|
return orignext, 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
|
|
|
|
}
|
|
|
|
if err == errDockerfileJSONNesting {
|
|
|
|
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
|
|
|
|
}
|
|
|
|
if err == errDockerfileJSONNesting {
|
|
|
|
return nil, nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return parseStringsWhitespaceDelimited(rest)
|
|
|
|
}
|