mirror of
				https://gitea.com/Lydanne/buildx.git
				synced 2025-10-30 23:53:48 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			345 lines
		
	
	
		
			7.4 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			345 lines
		
	
	
		
			7.4 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package bake
 | |
| 
 | |
| import (
 | |
| 	"context"
 | |
| 	"io/ioutil"
 | |
| 	"strings"
 | |
| 
 | |
| 	"github.com/moby/buildkit/session/auth/authprovider"
 | |
| 	"github.com/pkg/errors"
 | |
| 	"github.com/tonistiigi/buildx/build"
 | |
| )
 | |
| 
 | |
| func ReadTargets(ctx context.Context, files, targets, overrides []string) (map[string]Target, error) {
 | |
| 	var c Config
 | |
| 	for _, f := range files {
 | |
| 		cfg, err := ParseFile(f)
 | |
| 		if err != nil {
 | |
| 			return nil, err
 | |
| 		}
 | |
| 		c = mergeConfig(c, *cfg)
 | |
| 	}
 | |
| 	if err := c.setOverrides(overrides); err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	m := map[string]Target{}
 | |
| 	for _, n := range targets {
 | |
| 		for _, n := range c.ResolveGroup(n) {
 | |
| 			t, err := c.ResolveTarget(n)
 | |
| 			if err != nil {
 | |
| 				return nil, err
 | |
| 			}
 | |
| 			if t != nil {
 | |
| 				m[n] = *t
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	return m, nil
 | |
| }
 | |
| 
 | |
| func ParseFile(fn string) (*Config, error) {
 | |
| 	dt, err := ioutil.ReadFile(fn)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	fnl := strings.ToLower(fn)
 | |
| 	if strings.HasSuffix(fnl, ".yml") || strings.HasSuffix(fnl, ".yaml") {
 | |
| 		return ParseCompose(dt)
 | |
| 	}
 | |
| 
 | |
| 	if strings.HasSuffix(fnl, ".json") || strings.HasSuffix(fnl, ".hcl") {
 | |
| 		return ParseHCL(dt)
 | |
| 	}
 | |
| 
 | |
| 	cfg, err := ParseCompose(dt)
 | |
| 	if err != nil {
 | |
| 		cfg, err2 := ParseHCL(dt)
 | |
| 		if err2 != nil {
 | |
| 			return nil, errors.Errorf("failed to parse %s: parsing yaml: %s, parsing hcl: %s", fn, err.Error(), err2.Error())
 | |
| 		}
 | |
| 		return cfg, nil
 | |
| 	}
 | |
| 	return cfg, nil
 | |
| }
 | |
| 
 | |
| type Config struct {
 | |
| 	Group  map[string]Group
 | |
| 	Target map[string]Target
 | |
| }
 | |
| 
 | |
| func mergeConfig(c1, c2 Config) Config {
 | |
| 	for k, g := range c2.Group {
 | |
| 		if c1.Group == nil {
 | |
| 			c1.Group = map[string]Group{}
 | |
| 		}
 | |
| 		c1.Group[k] = g
 | |
| 	}
 | |
| 
 | |
| 	for k, t := range c2.Target {
 | |
| 		if c1.Target == nil {
 | |
| 			c1.Target = map[string]Target{}
 | |
| 		}
 | |
| 		if base, ok := c1.Target[k]; ok {
 | |
| 			t = merge(base, t)
 | |
| 		}
 | |
| 		c1.Target[k] = t
 | |
| 	}
 | |
| 	return c1
 | |
| }
 | |
| 
 | |
| func (c Config) setOverrides(v []string) error {
 | |
| 	for _, v := range v {
 | |
| 		parts := strings.SplitN(v, "=", 2)
 | |
| 		if len(parts) != 2 {
 | |
| 			return errors.Errorf("invalid override %s, expected target.name=value", v)
 | |
| 		}
 | |
| 		keys := strings.SplitN(parts[0], ".", 3)
 | |
| 		if len(keys) < 2 {
 | |
| 			return errors.Errorf("invalid override key %s, expected target.name", parts[0])
 | |
| 		}
 | |
| 
 | |
| 		name := keys[0]
 | |
| 
 | |
| 		t, ok := c.Target[name]
 | |
| 		if !ok {
 | |
| 			return errors.Errorf("unknown target %s", name)
 | |
| 		}
 | |
| 
 | |
| 		switch keys[1] {
 | |
| 		case "context":
 | |
| 			t.Context = parts[1]
 | |
| 		case "dockerfile":
 | |
| 			t.Dockerfile = parts[1]
 | |
| 		case "args":
 | |
| 			if len(keys) != 3 {
 | |
| 				return errors.Errorf("invalid key %s, args requires name", parts[0])
 | |
| 			}
 | |
| 			if t.Args == nil {
 | |
| 				t.Args = map[string]string{}
 | |
| 			}
 | |
| 			t.Args[keys[2]] = parts[1]
 | |
| 		case "labels":
 | |
| 			if len(keys) != 3 {
 | |
| 				return errors.Errorf("invalid key %s, lanels requires name", parts[0])
 | |
| 			}
 | |
| 			if t.Labels == nil {
 | |
| 				t.Labels = map[string]string{}
 | |
| 			}
 | |
| 			t.Labels[keys[2]] = parts[1]
 | |
| 		case "tags":
 | |
| 			t.Tags = append(t.Tags, parts[1])
 | |
| 		case "cache-from":
 | |
| 			t.CacheFrom = append(t.CacheFrom, parts[1])
 | |
| 		case "target":
 | |
| 			s := parts[1]
 | |
| 			t.Target = &s
 | |
| 		case "secrets":
 | |
| 			t.Secrets = append(t.Secrets, parts[1])
 | |
| 		case "ssh":
 | |
| 			t.SSH = append(t.SSH, parts[1])
 | |
| 		case "platform":
 | |
| 			t.Platforms = append(t.Platforms, parts[1])
 | |
| 		default:
 | |
| 			return errors.Errorf("unknown key: %s", keys[1])
 | |
| 		}
 | |
| 		c.Target[name] = t
 | |
| 	}
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func (c Config) ResolveGroup(name string) []string {
 | |
| 	return c.group(name, map[string]struct{}{})
 | |
| }
 | |
| 
 | |
| func (c Config) group(name string, visited map[string]struct{}) []string {
 | |
| 	if _, ok := visited[name]; ok {
 | |
| 		return nil
 | |
| 	}
 | |
| 	g, ok := c.Group[name]
 | |
| 	if !ok {
 | |
| 		return []string{name}
 | |
| 	}
 | |
| 	visited[name] = struct{}{}
 | |
| 	targets := make([]string, 0, len(g.Targets))
 | |
| 	for _, t := range g.Targets {
 | |
| 		targets = append(targets, c.group(t, visited)...)
 | |
| 	}
 | |
| 	return targets
 | |
| }
 | |
| 
 | |
| func (c Config) ResolveTarget(name string) (*Target, error) {
 | |
| 	return c.target(name, map[string]struct{}{})
 | |
| }
 | |
| 
 | |
| func (c Config) target(name string, visited map[string]struct{}) (*Target, error) {
 | |
| 	if _, ok := visited[name]; ok {
 | |
| 		return nil, nil
 | |
| 	}
 | |
| 	visited[name] = struct{}{}
 | |
| 	t, ok := c.Target[name]
 | |
| 	if !ok {
 | |
| 		return nil, errors.Errorf("failed to find target %s", name)
 | |
| 	}
 | |
| 	var tt Target
 | |
| 	for _, name := range t.Inherits {
 | |
| 		t, err := c.target(name, visited)
 | |
| 		if err != nil {
 | |
| 			return nil, err
 | |
| 		}
 | |
| 		if t != nil {
 | |
| 			tt = merge(tt, *t)
 | |
| 		}
 | |
| 	}
 | |
| 	t.Inherits = nil
 | |
| 	tt = merge(merge(defaultTarget(), t), tt)
 | |
| 	tt.normalize()
 | |
| 	return &tt, nil
 | |
| }
 | |
| 
 | |
| type Group struct {
 | |
| 	Targets []string
 | |
| 	// Target // TODO?
 | |
| }
 | |
| 
 | |
| type Target struct {
 | |
| 	Inherits   []string          `json:"inherits,omitempty"`
 | |
| 	Context    string            `json:"context,omitempty"`
 | |
| 	Dockerfile string            `json:"dockerfile,omitempty"`
 | |
| 	Args       map[string]string `json:"args,omitempty"`
 | |
| 	Labels     map[string]string `json:"labels,omitempty"`
 | |
| 	Tags       []string          `json:"tags,omitempty"`
 | |
| 	CacheFrom  []string          `json:"cache-from,omitempty"`
 | |
| 	Target     *string           `json:"target,omitempty"`
 | |
| 	Secrets    []string          `json:"secret,omitempty"`
 | |
| 	SSH        []string          `json:"ssh,omitempty"`
 | |
| 	Platforms  []string          `json:"platform,omitempty"`
 | |
| }
 | |
| 
 | |
| func (t *Target) normalize() {
 | |
| 	t.Tags = removeDupes(t.Tags)
 | |
| 	t.Secrets = removeDupes(t.Secrets)
 | |
| 	t.SSH = removeDupes(t.SSH)
 | |
| 	t.Platforms = removeDupes(t.Platforms)
 | |
| }
 | |
| 
 | |
| func TargetsToBuildOpt(m map[string]Target) (map[string]build.Options, error) {
 | |
| 	m2 := make(map[string]build.Options, len(m))
 | |
| 	for k, v := range m {
 | |
| 		bo, err := toBuildOpt(v)
 | |
| 		if err != nil {
 | |
| 			return nil, err
 | |
| 		}
 | |
| 		m2[k] = *bo
 | |
| 	}
 | |
| 	return m2, nil
 | |
| }
 | |
| 
 | |
| func toBuildOpt(t Target) (*build.Options, error) {
 | |
| 	if t.Context == "-" {
 | |
| 		return nil, errors.Errorf("context from stdin not allowed in bake")
 | |
| 	}
 | |
| 	if t.Dockerfile == "-" {
 | |
| 		return nil, errors.Errorf("dockerfile from stdin not allowed in bake")
 | |
| 	}
 | |
| 
 | |
| 	bo := &build.Options{
 | |
| 		Inputs: build.Inputs{
 | |
| 			ContextPath:    t.Context,
 | |
| 			DockerfilePath: t.Dockerfile,
 | |
| 		},
 | |
| 		Tags:      t.Tags,
 | |
| 		BuildArgs: t.Args,
 | |
| 		Labels:    t.Labels,
 | |
| 		// CacheFrom: t.CacheFrom,
 | |
| 	}
 | |
| 
 | |
| 	platforms, err := build.ParsePlatformSpecs(t.Platforms)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	bo.Platforms = platforms
 | |
| 
 | |
| 	bo.Session = append(bo.Session, authprovider.NewDockerAuthProvider())
 | |
| 
 | |
| 	secrets, err := build.ParseSecretSpecs(t.Secrets)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	bo.Session = append(bo.Session, secrets)
 | |
| 
 | |
| 	ssh, err := build.ParseSSHSpecs(t.SSH)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	bo.Session = append(bo.Session, ssh)
 | |
| 
 | |
| 	if t.Target != nil {
 | |
| 		bo.Target = *t.Target
 | |
| 	}
 | |
| 
 | |
| 	return bo, nil
 | |
| }
 | |
| 
 | |
| func defaultTarget() Target {
 | |
| 	return Target{
 | |
| 		Context:    ".",
 | |
| 		Dockerfile: "Dockerfile",
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func merge(t1, t2 Target) Target {
 | |
| 	if t2.Context != "" {
 | |
| 		t1.Context = t2.Context
 | |
| 	}
 | |
| 	if t2.Dockerfile != "" {
 | |
| 		t1.Dockerfile = t2.Dockerfile
 | |
| 	}
 | |
| 	for k, v := range t2.Args {
 | |
| 		if t1.Args == nil {
 | |
| 			t1.Args = map[string]string{}
 | |
| 		}
 | |
| 		t1.Args[k] = v
 | |
| 	}
 | |
| 	for k, v := range t2.Labels {
 | |
| 		if t1.Labels == nil {
 | |
| 			t1.Labels = map[string]string{}
 | |
| 		}
 | |
| 		t1.Labels[k] = v
 | |
| 	}
 | |
| 	if t2.Tags != nil { // no merge
 | |
| 		t1.Tags = t2.Tags
 | |
| 	}
 | |
| 	if t2.CacheFrom != nil {
 | |
| 		t1.CacheFrom = t2.CacheFrom
 | |
| 	}
 | |
| 	if t2.Target != nil {
 | |
| 		t1.Target = t2.Target
 | |
| 	}
 | |
| 	if t2.Secrets != nil { // merge
 | |
| 		t1.Secrets = append(t1.Secrets, t2.Secrets...)
 | |
| 	}
 | |
| 	if t2.SSH != nil { // merge
 | |
| 		t1.SSH = append(t1.SSH, t2.SSH...)
 | |
| 	}
 | |
| 	if t2.Platforms != nil { // no merge
 | |
| 		t1.Platforms = t2.Platforms
 | |
| 	}
 | |
| 	t1.Inherits = append(t1.Inherits, t2.Inherits...)
 | |
| 	return t1
 | |
| }
 | |
| 
 | |
| func removeDupes(s []string) []string {
 | |
| 	i := 0
 | |
| 	seen := make(map[string]struct{}, len(s))
 | |
| 	for _, v := range s {
 | |
| 		if _, ok := seen[v]; ok {
 | |
| 			continue
 | |
| 		}
 | |
| 		seen[v] = struct{}{}
 | |
| 		s[i] = v
 | |
| 		i++
 | |
| 	}
 | |
| 	return s[:i]
 | |
| }
 | 
