mirror of https://github.com/containers/podman.git
				
				
				
			
		
			
				
	
	
		
			160 lines
		
	
	
		
			4.1 KiB
		
	
	
	
		
			Go
		
	
	
	
			
		
		
	
	
			160 lines
		
	
	
		
			4.1 KiB
		
	
	
	
		
			Go
		
	
	
	
| package imagebuilder
 | |
| 
 | |
| import (
 | |
| 	"fmt"
 | |
| 	"io"
 | |
| 	"strings"
 | |
| 
 | |
| 	"github.com/openshift/imagebuilder/dockerfile/command"
 | |
| 	"github.com/openshift/imagebuilder/dockerfile/parser"
 | |
| )
 | |
| 
 | |
| // ParseDockerfile parses the provided stream as a canonical Dockerfile
 | |
| func ParseDockerfile(r io.Reader) (*parser.Node, error) {
 | |
| 	result, err := parser.Parse(r)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	return result.AST, nil
 | |
| }
 | |
| 
 | |
| // Environment variable interpolation will happen on these statements only.
 | |
| var replaceEnvAllowed = map[string]bool{
 | |
| 	command.Env:        true,
 | |
| 	command.Label:      true,
 | |
| 	command.Add:        true,
 | |
| 	command.Copy:       true,
 | |
| 	command.Workdir:    true,
 | |
| 	command.Expose:     true,
 | |
| 	command.Volume:     true,
 | |
| 	command.User:       true,
 | |
| 	command.StopSignal: true,
 | |
| 	command.Arg:        true,
 | |
| }
 | |
| 
 | |
| // Certain commands are allowed to have their args split into more
 | |
| // words after env var replacements. Meaning:
 | |
| //   ENV foo="123 456"
 | |
| //   EXPOSE $foo
 | |
| // should result in the same thing as:
 | |
| //   EXPOSE 123 456
 | |
| // and not treat "123 456" as a single word.
 | |
| // Note that: EXPOSE "$foo" and EXPOSE $foo are not the same thing.
 | |
| // Quotes will cause it to still be treated as single word.
 | |
| var allowWordExpansion = map[string]bool{
 | |
| 	command.Expose: true,
 | |
| }
 | |
| 
 | |
| // Step represents the input Env and the output command after all
 | |
| // post processing of the command arguments is done.
 | |
| type Step struct {
 | |
| 	Env []string
 | |
| 
 | |
| 	Command  string
 | |
| 	Args     []string
 | |
| 	Flags    []string
 | |
| 	Attrs    map[string]bool
 | |
| 	Message  string
 | |
| 	Original string
 | |
| }
 | |
| 
 | |
| // Resolve transforms a parsed Dockerfile line into a command to execute,
 | |
| // resolving any arguments.
 | |
| //
 | |
| // Almost all nodes will have this structure:
 | |
| // Child[Node, Node, Node] where Child is from parser.Node.Children and each
 | |
| // node comes from parser.Node.Next. This forms a "line" with a statement and
 | |
| // arguments and we process them in this normalized form by hitting
 | |
| // evaluateTable with the leaf nodes of the command and the Builder object.
 | |
| //
 | |
| // ONBUILD is a special case; in this case the parser will emit:
 | |
| // Child[Node, Child[Node, Node...]] where the first node is the literal
 | |
| // "onbuild" and the child entrypoint is the command of the ONBUILD statement,
 | |
| // such as `RUN` in ONBUILD RUN foo. There is special case logic in here to
 | |
| // deal with that, at least until it becomes more of a general concern with new
 | |
| // features.
 | |
| func (b *Step) Resolve(ast *parser.Node) error {
 | |
| 	cmd := ast.Value
 | |
| 	upperCasedCmd := strings.ToUpper(cmd)
 | |
| 
 | |
| 	// To ensure the user is given a decent error message if the platform
 | |
| 	// on which the daemon is running does not support a builder command.
 | |
| 	if err := platformSupports(strings.ToLower(cmd)); err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 
 | |
| 	attrs := ast.Attributes
 | |
| 	original := ast.Original
 | |
| 	flags := ast.Flags
 | |
| 	strList := []string{}
 | |
| 	msg := upperCasedCmd
 | |
| 
 | |
| 	if len(ast.Flags) > 0 {
 | |
| 		msg += " " + strings.Join(ast.Flags, " ")
 | |
| 	}
 | |
| 
 | |
| 	if cmd == "onbuild" {
 | |
| 		if ast.Next == nil {
 | |
| 			return fmt.Errorf("ONBUILD requires at least one argument")
 | |
| 		}
 | |
| 		ast = ast.Next.Children[0]
 | |
| 		strList = append(strList, ast.Value)
 | |
| 		msg += " " + ast.Value
 | |
| 
 | |
| 		if len(ast.Flags) > 0 {
 | |
| 			msg += " " + strings.Join(ast.Flags, " ")
 | |
| 		}
 | |
| 
 | |
| 	}
 | |
| 
 | |
| 	// count the number of nodes that we are going to traverse first
 | |
| 	// so we can pre-create the argument and message array. This speeds up the
 | |
| 	// allocation of those list a lot when they have a lot of arguments
 | |
| 	cursor := ast
 | |
| 	var n int
 | |
| 	for cursor.Next != nil {
 | |
| 		cursor = cursor.Next
 | |
| 		n++
 | |
| 	}
 | |
| 	msgList := make([]string, n)
 | |
| 
 | |
| 	var i int
 | |
| 	envs := b.Env
 | |
| 	for ast.Next != nil {
 | |
| 		ast = ast.Next
 | |
| 		str := ast.Value
 | |
| 		if replaceEnvAllowed[cmd] {
 | |
| 			var err error
 | |
| 			var words []string
 | |
| 
 | |
| 			if allowWordExpansion[cmd] {
 | |
| 				words, err = ProcessWords(str, envs)
 | |
| 				if err != nil {
 | |
| 					return err
 | |
| 				}
 | |
| 				strList = append(strList, words...)
 | |
| 			} else {
 | |
| 				str, err = ProcessWord(str, envs)
 | |
| 				if err != nil {
 | |
| 					return err
 | |
| 				}
 | |
| 				strList = append(strList, str)
 | |
| 			}
 | |
| 		} else {
 | |
| 			strList = append(strList, str)
 | |
| 		}
 | |
| 		msgList[i] = ast.Value
 | |
| 		i++
 | |
| 	}
 | |
| 
 | |
| 	msg += " " + strings.Join(msgList, " ")
 | |
| 
 | |
| 	b.Message = msg
 | |
| 	b.Command = cmd
 | |
| 	b.Args = strList
 | |
| 	b.Original = original
 | |
| 	b.Attrs = attrs
 | |
| 	b.Flags = flags
 | |
| 	return nil
 | |
| }
 |