Skip to content

Commit

Permalink
bake: initial implementation
Browse files Browse the repository at this point in the history
Signed-off-by: Tonis Tiigi <tonistiigi@gmail.com>
  • Loading branch information
tonistiigi committed Apr 10, 2019
1 parent 9129a49 commit a932d52
Show file tree
Hide file tree
Showing 127 changed files with 23,924 additions and 86 deletions.
4 changes: 1 addition & 3 deletions Dockerfile
Original file line number Diff line number Diff line change
Expand Up @@ -53,9 +53,7 @@ COPY --from=buildx-build /usr/bin/buildx /buildx.exe
FROM binaries-$TARGETOS AS binaries

FROM alpine AS demo-env
RUN apk add --no-cache iptables tmux git
RUN apt-get update && apt-get install -y iptables tmux git curl vim file
RUN curl https://get.docker.com/ | CHANNEL=test sh
RUN apk add --no-cache iptables tmux git vim less
RUN mkdir -p /usr/local/lib/docker/cli-plugins && ln -s /usr/local/bin/buildx /usr/local/lib/docker/cli-plugins/docker-buildx
COPY ./hack/demo-env/entrypoint.sh /usr/local/bin
COPY ./hack/demo-env/tmux.conf /root/.tmux.conf
Expand Down
344 changes: 344 additions & 0 deletions bake/bake.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,344 @@
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]
}
Loading

0 comments on commit a932d52

Please sign in to comment.