You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
397 lines
8.9 KiB
Go
397 lines
8.9 KiB
Go
package bake
|
|
|
|
import (
|
|
"context"
|
|
"io/ioutil"
|
|
"os"
|
|
"strings"
|
|
|
|
"github.com/docker/buildx/build"
|
|
"github.com/docker/buildx/util/platformutil"
|
|
"github.com/moby/buildkit/session/auth/authprovider"
|
|
"github.com/pkg/errors"
|
|
)
|
|
|
|
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 "cache-to":
|
|
t.CacheTo = append(t.CacheTo, 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])
|
|
case "output":
|
|
t.Outputs = append(t.Outputs, 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) {
|
|
t, err := c.target(name, map[string]struct{}{})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if t.Context == nil {
|
|
s := "."
|
|
t.Context = &s
|
|
}
|
|
if t.Dockerfile == nil {
|
|
s := "Dockerfile"
|
|
t.Dockerfile = &s
|
|
}
|
|
return t, nil
|
|
}
|
|
|
|
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" hcl:"inherits,omitempty"`
|
|
Context *string `json:"context,omitempty" hcl:"context,omitempty"`
|
|
Dockerfile *string `json:"dockerfile,omitempty" hcl:"dockerfile,omitempty"`
|
|
Args map[string]string `json:"args,omitempty" hcl:"args,omitempty"`
|
|
Labels map[string]string `json:"labels,omitempty" hcl:"labels,omitempty"`
|
|
Tags []string `json:"tags,omitempty" hcl:"tags,omitempty"`
|
|
CacheFrom []string `json:"cache-from,omitempty" hcl:"cache-from,omitempty"`
|
|
CacheTo []string `json:"cache-to,omitempty" hcl:"cache-to,omitempty"`
|
|
Target *string `json:"target,omitempty" hcl:"target,omitempty"`
|
|
Secrets []string `json:"secret,omitempty" hcl:"secret,omitempty"`
|
|
SSH []string `json:"ssh,omitempty" hcl:"ssh,omitempty"`
|
|
Platforms []string `json:"platforms,omitempty" hcl:"platforms,omitempty"`
|
|
Outputs []string `json:"output,omitempty" hcl:"output,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)
|
|
t.CacheFrom = removeDupes(t.CacheFrom)
|
|
t.CacheTo = removeDupes(t.CacheTo)
|
|
t.Outputs = removeDupes(t.Outputs)
|
|
}
|
|
|
|
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 v := t.Context; v != nil && *v == "-" {
|
|
return nil, errors.Errorf("context from stdin not allowed in bake")
|
|
}
|
|
if v := t.Dockerfile; v != nil && *v == "-" {
|
|
return nil, errors.Errorf("dockerfile from stdin not allowed in bake")
|
|
}
|
|
|
|
contextPath := "."
|
|
if t.Context != nil {
|
|
contextPath = *t.Context
|
|
}
|
|
dockerfilePath := "Dockerfile"
|
|
if t.Dockerfile != nil {
|
|
dockerfilePath = *t.Dockerfile
|
|
}
|
|
|
|
bo := &build.Options{
|
|
Inputs: build.Inputs{
|
|
ContextPath: contextPath,
|
|
DockerfilePath: dockerfilePath,
|
|
},
|
|
Tags: t.Tags,
|
|
BuildArgs: t.Args,
|
|
Labels: t.Labels,
|
|
}
|
|
|
|
platforms, err := platformutil.Parse(t.Platforms)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
bo.Platforms = platforms
|
|
|
|
bo.Session = append(bo.Session, authprovider.NewDockerAuthProvider(os.Stderr))
|
|
|
|
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
|
|
}
|
|
|
|
cacheImports, err := build.ParseCacheEntry(t.CacheFrom)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
bo.CacheFrom = cacheImports
|
|
|
|
cacheExports, err := build.ParseCacheEntry(t.CacheTo)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
bo.CacheTo = cacheExports
|
|
|
|
outputs, err := build.ParseOutputs(t.Outputs)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
bo.Exports = outputs
|
|
|
|
return bo, nil
|
|
}
|
|
|
|
func defaultTarget() Target {
|
|
return Target{}
|
|
}
|
|
|
|
func merge(t1, t2 Target) Target {
|
|
if t2.Context != nil {
|
|
t1.Context = t2.Context
|
|
}
|
|
if t2.Dockerfile != nil {
|
|
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.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
|
|
}
|
|
if t2.CacheFrom != nil { // no merge
|
|
t1.CacheFrom = append(t1.CacheFrom, t2.CacheFrom...)
|
|
}
|
|
if t2.CacheTo != nil { // no merge
|
|
t1.CacheTo = t2.CacheTo
|
|
}
|
|
if t2.Outputs != nil { // no merge
|
|
t1.Outputs = t2.Outputs
|
|
}
|
|
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]
|
|
}
|