build: multi-node build support

Signed-off-by: Tonis Tiigi <tonistiigi@gmail.com>
pull/37/head
Tonis Tiigi 6 years ago
parent 5ddd3d2459
commit 0398fa337b

@ -11,17 +11,21 @@ import (
"strings" "strings"
"sync" "sync"
"github.com/containerd/containerd/images"
"github.com/containerd/containerd/platforms" "github.com/containerd/containerd/platforms"
clitypes "github.com/docker/cli/cli/config/types"
"github.com/docker/distribution/reference" "github.com/docker/distribution/reference"
dockerclient "github.com/docker/docker/client" dockerclient "github.com/docker/docker/client"
"github.com/docker/docker/pkg/urlutil" "github.com/docker/docker/pkg/urlutil"
"github.com/moby/buildkit/client" "github.com/moby/buildkit/client"
"github.com/moby/buildkit/session" "github.com/moby/buildkit/session"
"github.com/moby/buildkit/session/upload/uploadprovider" "github.com/moby/buildkit/session/upload/uploadprovider"
"github.com/opencontainers/go-digest"
specs "github.com/opencontainers/image-spec/specs-go/v1" specs "github.com/opencontainers/image-spec/specs-go/v1"
"github.com/pkg/errors" "github.com/pkg/errors"
"github.com/sirupsen/logrus" "github.com/sirupsen/logrus"
"github.com/tonistiigi/buildx/driver" "github.com/tonistiigi/buildx/driver"
"github.com/tonistiigi/buildx/util/imagetools"
"github.com/tonistiigi/buildx/util/progress" "github.com/tonistiigi/buildx/util/progress"
"golang.org/x/sync/errgroup" "golang.org/x/sync/errgroup"
) )
@ -66,76 +70,243 @@ type DriverInfo struct {
Err error Err error
} }
type Auth interface {
GetAuthConfig(registryHostname string) (clitypes.AuthConfig, error)
}
type DockerAPI interface { type DockerAPI interface {
DockerAPI(name string) (dockerclient.APIClient, error) DockerAPI(name string) (dockerclient.APIClient, error)
} }
func getFirstDriver(drivers []DriverInfo) (driver.Driver, error) { func filterAvailableDrivers(drivers []DriverInfo) ([]DriverInfo, error) {
out := make([]DriverInfo, 0, len(drivers))
err := errors.Errorf("no drivers found") err := errors.Errorf("no drivers found")
for _, di := range drivers { for _, di := range drivers {
if di.Driver != nil { if di.Err == nil && di.Driver != nil {
return di.Driver, nil out = append(out, di)
} }
if di.Err != nil { if di.Err != nil {
err = di.Err err = di.Err
} }
} }
if len(out) > 0 {
return out, nil
}
return nil, err return nil, err
} }
func Build(ctx context.Context, drivers []DriverInfo, opt map[string]Options, docker DockerAPI, pw progress.Writer) (map[string]*client.SolveResponse, error) { type driverPair struct {
if len(drivers) == 0 { driverIndex int
return nil, errors.Errorf("driver required for build") platforms []specs.Platform
so *client.SolveOpt
}
func driverIndexes(m map[string][]driverPair) []int {
out := make([]int, 0, len(m))
visited := map[int]struct{}{}
for _, dp := range m {
for _, d := range dp {
if _, ok := visited[d.driverIndex]; ok {
continue
}
visited[d.driverIndex] = struct{}{}
out = append(out, d.driverIndex)
}
}
return out
} }
if len(drivers) > 1 { func allIndexes(l int) []int {
return nil, errors.Errorf("multiple drivers currently not supported") out := make([]int, 0, l)
for i := 0; i < l; i++ {
out = append(out, i)
}
return out
} }
d, err := getFirstDriver(drivers) func ensureBooted(ctx context.Context, drivers []DriverInfo, idxs []int, pw progress.Writer) ([]*client.Client, error) {
clients := make([]*client.Client, len(drivers))
eg, ctx := errgroup.WithContext(ctx)
for _, i := range idxs {
func(i int) {
eg.Go(func() error {
c, err := driver.Boot(ctx, drivers[i].Driver, pw)
if err != nil { if err != nil {
return nil, err return err
} }
_, isDefaultMobyDriver := d.(interface { clients[i] = c
IsDefaultMobyDriver() return nil
}) })
}(i)
}
if err := eg.Wait(); err != nil {
return nil, err
}
return clients, nil
}
func splitToDriverPairs(availablePlatforms map[string]int, opt map[string]Options) map[string][]driverPair {
m := map[string][]driverPair{}
for k, opt := range opt {
mm := map[int][]specs.Platform{}
for _, p := range opt.Platforms {
k := platforms.Format(p)
idx := availablePlatforms[k] // default 0
pp := mm[idx]
pp = append(pp, p)
mm[idx] = pp
}
dps := make([]driverPair, 0, 2)
for idx, pp := range mm {
dps = append(dps, driverPair{driverIndex: idx, platforms: pp})
}
m[k] = dps
}
return m
}
func resolveDrivers(ctx context.Context, drivers []DriverInfo, opt map[string]Options, pw progress.Writer) (map[string][]driverPair, []*client.Client, error) {
availablePlatforms := map[string]int{}
for i, d := range drivers {
for _, p := range d.Platform {
availablePlatforms[platforms.Format(p)] = i
}
}
undetectedPlatform := false
allPlatforms := map[string]int{}
for _, opt := range opt { for _, opt := range opt {
if !isDefaultMobyDriver && len(opt.Exports) == 0 { for _, p := range opt.Platforms {
logrus.Warnf("No output specified for %s driver. Build result will only remain in the build cache. To push result image into registry use --push or to load image into docker use --load", d.Factory().Name()) k := platforms.Format(p)
break allPlatforms[k] = -1
if _, ok := availablePlatforms[k]; !ok {
undetectedPlatform = true
}
} }
} }
c, err := driver.Boot(ctx, d, pw) // fast path
if len(drivers) == 1 || len(allPlatforms) == 0 {
m := map[string][]driverPair{}
for k, opt := range opt {
m[k] = []driverPair{{driverIndex: 0, platforms: opt.Platforms}}
}
clients, err := ensureBooted(ctx, drivers, driverIndexes(m), pw)
if err != nil { if err != nil {
close(pw.Status()) return nil, nil, err
<-pw.Done() }
return nil, err return m, clients, nil
} }
withPrefix := len(opt) > 1 // map based on existing platforms
if !undetectedPlatform {
m := splitToDriverPairs(availablePlatforms, opt)
clients, err := ensureBooted(ctx, drivers, driverIndexes(m), pw)
if err != nil {
return nil, nil, err
}
return m, clients, nil
}
mw := progress.NewMultiWriter(pw) // boot all drivers in k
clients, err := ensureBooted(ctx, drivers, allIndexes(len(drivers)), pw)
if err != nil {
return nil, nil, err
}
eg, ctx := errgroup.WithContext(ctx) eg, ctx := errgroup.WithContext(ctx)
workers := make([][]*client.WorkerInfo, len(clients))
resp := map[string]*client.SolveResponse{} for i, c := range clients {
var mu sync.Mutex if c == nil {
continue
}
func(i int) {
eg.Go(func() error {
ww, err := clients[i].ListWorkers(ctx)
if err != nil {
return errors.Wrap(err, "listing workers")
}
workers[i] = ww
return nil
})
}(i)
}
for k, opt := range opt { if err := eg.Wait(); err != nil {
pw := mw.WithPrefix(k, withPrefix) return nil, nil, err
}
for i, ww := range workers {
for _, w := range ww {
for _, p := range w.Platforms {
p = platforms.Normalize(p)
ps := platforms.Format(p)
if _, ok := availablePlatforms[ps]; !ok {
availablePlatforms[ps] = i
}
}
}
}
return splitToDriverPairs(availablePlatforms, opt), clients, nil
}
func toRepoOnly(in string) (string, error) {
m := map[string]struct{}{}
p := strings.Split(in, ",")
for _, pp := range p {
n, err := reference.ParseNormalizedNamed(pp)
if err != nil {
return "", err
}
m[n.Name()] = struct{}{}
}
out := make([]string, 0, len(m))
for k := range m {
out = append(out, k)
}
return strings.Join(out, ","), nil
}
func isDefaultMobyDriver(d driver.Driver) bool {
_, ok := d.(interface {
IsDefaultMobyDriver()
})
return ok
}
func toSolveOpt(d driver.Driver, multiDriver bool, opt Options, dl dockerLoadCallback) (solveOpt *client.SolveOpt, release func(), err error) {
defers := make([]func(), 0, 2)
release = func() {
for _, f := range defers {
f()
}
}
defer func() {
if err != nil {
release()
}
}()
if opt.ImageIDFile != "" { if opt.ImageIDFile != "" {
if len(opt.Platforms) != 0 { if multiDriver || len(opt.Platforms) != 0 {
return nil, errors.Errorf("image ID file cannot be specified when building for multiple platforms") return nil, nil, errors.Errorf("image ID file cannot be specified when building for multiple platforms")
} }
// Avoid leaving a stale file if we eventually fail // Avoid leaving a stale file if we eventually fail
if err := os.Remove(opt.ImageIDFile); err != nil && !os.IsNotExist(err) { if err := os.Remove(opt.ImageIDFile); err != nil && !os.IsNotExist(err) {
return nil, errors.Wrap(err, "removing image ID file") return nil, nil, errors.Wrap(err, "removing image ID file")
} }
} }
// inline cache from build arg
if v, ok := opt.BuildArgs["BUILDKIT_INLINE_CACHE"]; ok { if v, ok := opt.BuildArgs["BUILDKIT_INLINE_CACHE"]; ok {
if v, _ := strconv.ParseBool(v); v { if v, _ := strconv.ParseBool(v); v {
opt.CacheTo = append(opt.CacheTo, client.CacheOptionsEntry{ opt.CacheTo = append(opt.CacheTo, client.CacheOptionsEntry{
@ -147,7 +318,7 @@ func Build(ctx context.Context, drivers []DriverInfo, opt map[string]Options, do
for _, e := range opt.CacheTo { for _, e := range opt.CacheTo {
if e.Type != "inline" && !d.Features()[driver.CacheExport] { if e.Type != "inline" && !d.Features()[driver.CacheExport] {
return nil, notSupported(d, driver.CacheExport) return nil, nil, notSupported(d, driver.CacheExport)
} }
} }
@ -159,6 +330,15 @@ func Build(ctx context.Context, drivers []DriverInfo, opt map[string]Options, do
CacheImports: opt.CacheFrom, CacheImports: opt.CacheFrom,
} }
if multiDriver {
// force creation of manifest list
so.FrontendAttrs["multi-platform"] = "true"
}
_, isDefaultMobyDriver := d.(interface {
IsDefaultMobyDriver()
})
switch len(opt.Exports) { switch len(opt.Exports) {
case 1: case 1:
// valid // valid
@ -169,15 +349,16 @@ func Build(ctx context.Context, drivers []DriverInfo, opt map[string]Options, do
opt.Exports = []client.ExportEntry{{Type: "image", Attrs: map[string]string{}}} opt.Exports = []client.ExportEntry{{Type: "image", Attrs: map[string]string{}}}
} }
default: default:
return nil, errors.Errorf("multiple outputs currently unsupported") return nil, nil, errors.Errorf("multiple outputs currently unsupported")
} }
// fill in image exporter names from tags
if len(opt.Tags) > 0 { if len(opt.Tags) > 0 {
tags := make([]string, len(opt.Tags)) tags := make([]string, len(opt.Tags))
for i, tag := range opt.Tags { for i, tag := range opt.Tags {
ref, err := reference.Parse(tag) ref, err := reference.Parse(tag)
if err != nil { if err != nil {
return nil, errors.Wrapf(err, "invalid tag %q", tag) return nil, nil, errors.Wrapf(err, "invalid tag %q", tag)
} }
tags[i] = ref.String() tags[i] = ref.String()
} }
@ -191,55 +372,54 @@ func Build(ctx context.Context, drivers []DriverInfo, opt map[string]Options, do
for _, e := range opt.Exports { for _, e := range opt.Exports {
if e.Type == "image" && e.Attrs["name"] == "" && e.Attrs["push"] != "" { if e.Type == "image" && e.Attrs["name"] == "" && e.Attrs["push"] != "" {
if ok, _ := strconv.ParseBool(e.Attrs["push"]); ok { if ok, _ := strconv.ParseBool(e.Attrs["push"]); ok {
return nil, errors.Errorf("tag is needed when pushing to registry") return nil, nil, errors.Errorf("tag is needed when pushing to registry")
} }
} }
} }
} }
// set up exporters
for i, e := range opt.Exports { for i, e := range opt.Exports {
if (e.Type == "local" || e.Type == "tar") && opt.ImageIDFile != "" { if (e.Type == "local" || e.Type == "tar") && opt.ImageIDFile != "" {
return nil, errors.Errorf("local and tar exporters are incompatible with image ID file") return nil, nil, errors.Errorf("local and tar exporters are incompatible with image ID file")
} }
if e.Type == "oci" && !d.Features()[driver.OCIExporter] { if e.Type == "oci" && !d.Features()[driver.OCIExporter] {
return nil, notSupported(d, driver.OCIExporter) return nil, nil, notSupported(d, driver.OCIExporter)
} }
if e.Type == "docker" { if e.Type == "docker" {
if e.Output == nil { if e.Output == nil {
if isDefaultMobyDriver { if isDefaultMobyDriver {
e.Type = "image" e.Type = "image"
} else { } else {
w, cancel, err := newDockerLoader(ctx, docker, e.Attrs["context"], mw) w, cancel, err := dl(e.Attrs["context"])
if err != nil { if err != nil {
return nil, err return nil, nil, err
} }
defer cancel() defers = append(defers, cancel)
opt.Exports[i].Output = w opt.Exports[i].Output = w
} }
} else if !d.Features()[driver.DockerExporter] { } else if !d.Features()[driver.DockerExporter] {
return nil, notSupported(d, driver.DockerExporter) return nil, nil, notSupported(d, driver.DockerExporter)
} }
} }
if e.Type == "image" && isDefaultMobyDriver { if e.Type == "image" && isDefaultMobyDriver {
opt.Exports[i].Type = "moby" opt.Exports[i].Type = "moby"
if e.Attrs["push"] != "" { if e.Attrs["push"] != "" {
if ok, _ := strconv.ParseBool(e.Attrs["push"]); ok { if ok, _ := strconv.ParseBool(e.Attrs["push"]); ok {
return nil, errors.Errorf("auto-push is currently not implemented for docker driver") return nil, nil, errors.Errorf("auto-push is currently not implemented for docker driver")
} }
} }
} }
} }
// TODO: handle loading to docker daemon
so.Exports = opt.Exports so.Exports = opt.Exports
so.Session = opt.Session so.Session = opt.Session
release, err := LoadInputs(opt.Inputs, &so) releaseLoad, err := LoadInputs(opt.Inputs, &so)
if err != nil { if err != nil {
return nil, err return nil, nil, err
} }
defer release() defers = append(defers, releaseLoad)
if opt.Pull { if opt.Pull {
so.FrontendAttrs["image-resolve-mode"] = "pull" so.FrontendAttrs["image-resolve-mode"] = "pull"
@ -257,31 +437,223 @@ func Build(ctx context.Context, drivers []DriverInfo, opt map[string]Options, do
so.FrontendAttrs["label:"+k] = v so.FrontendAttrs["label:"+k] = v
} }
// set platforms
if len(opt.Platforms) != 0 { if len(opt.Platforms) != 0 {
pp := make([]string, len(opt.Platforms)) pp := make([]string, len(opt.Platforms))
for i, p := range opt.Platforms { for i, p := range opt.Platforms {
pp[i] = platforms.Format(p) pp[i] = platforms.Format(p)
} }
if len(pp) > 1 && !d.Features()[driver.MultiPlatform] { if len(pp) > 1 && !d.Features()[driver.MultiPlatform] {
return nil, notSupported(d, driver.MultiPlatform) return nil, nil, notSupported(d, driver.MultiPlatform)
} }
so.FrontendAttrs["platform"] = strings.Join(pp, ",") so.FrontendAttrs["platform"] = strings.Join(pp, ",")
} }
// setup networkmode
switch opt.NetworkMode { switch opt.NetworkMode {
case "host", "none": case "host", "none":
so.FrontendAttrs["force-network-mode"] = opt.NetworkMode so.FrontendAttrs["force-network-mode"] = opt.NetworkMode
case "", "default": case "", "default":
default: default:
return nil, errors.Errorf("network mode %q not supported by buildkit", opt.NetworkMode) return nil, nil, errors.Errorf("network mode %q not supported by buildkit", opt.NetworkMode)
} }
// setup extrahosts
extraHosts, err := toBuildkitExtraHosts(opt.ExtraHosts) extraHosts, err := toBuildkitExtraHosts(opt.ExtraHosts)
if err != nil { if err != nil {
return nil, err return nil, nil, err
} }
so.FrontendAttrs["add-hosts"] = extraHosts so.FrontendAttrs["add-hosts"] = extraHosts
return &so, release, nil
}
func Build(ctx context.Context, drivers []DriverInfo, opt map[string]Options, docker DockerAPI, auth Auth, pw progress.Writer) (resp map[string]*client.SolveResponse, err error) {
if len(drivers) == 0 {
return nil, errors.Errorf("driver required for build")
}
drivers, err = filterAvailableDrivers(drivers)
if err != nil {
return nil, errors.Wrapf(err, "no valid drivers found")
}
var noMobyDriver driver.Driver
for _, d := range drivers {
if !isDefaultMobyDriver(d.Driver) {
noMobyDriver = d.Driver
break
}
}
if noMobyDriver != nil {
for _, opt := range opt {
if len(opt.Exports) == 0 {
logrus.Warnf("No output specified for %s driver. Build result will only remain in the build cache. To push result image into registry use --push or to load image into docker use --load", noMobyDriver.Factory().Name())
break
}
}
}
m, clients, err := resolveDrivers(ctx, drivers, opt, pw)
if err != nil {
close(pw.Status())
<-pw.Done()
return nil, err
}
defers := make([]func(), 0, 2)
defer func() {
if err != nil {
for _, f := range defers {
f()
}
}
}()
mw := progress.NewMultiWriter(pw)
eg, ctx := errgroup.WithContext(ctx)
for k, opt := range opt {
multiDriver := len(m[k]) > 1
for i, dp := range m[k] {
d := drivers[dp.driverIndex].Driver
opt.Platforms = dp.platforms
so, release, err := toSolveOpt(d, multiDriver, opt, func(name string) (io.WriteCloser, func(), error) {
return newDockerLoader(ctx, docker, name, mw)
})
if err != nil {
return nil, err
}
defers = append(defers, release)
m[k][i].so = so
}
}
resp = map[string]*client.SolveResponse{}
var respMu sync.Mutex
multiTarget := len(opt) > 1
for k, opt := range opt {
err := func() error {
opt := opt
dps := m[k]
multiDriver := len(m[k]) > 1
res := make([]*client.SolveResponse, len(dps))
wg := &sync.WaitGroup{}
wg.Add(len(dps))
var pushNames string
if multiDriver {
for i, e := range opt.Exports {
switch e.Type {
case "oci", "tar":
return errors.Errorf("%s for multi-node builds currently not supported", e.Type)
case "image":
if e.Attrs["push"] != "" {
if ok, _ := strconv.ParseBool(e.Attrs["push"]); ok {
pushNames = e.Attrs["name"]
if pushNames == "" {
return errors.Errorf("tag is needed when pushing to registry")
}
names, err := toRepoOnly(e.Attrs["name"])
if err != nil {
return err
}
e.Attrs["name"] = names
e.Attrs["push-by-digest"] = "true"
opt.Exports[i].Attrs = e.Attrs
}
}
}
}
}
eg.Go(func() error {
pw := mw.WithPrefix("default", false)
defer close(pw.Status())
wg.Wait()
select {
case <-ctx.Done():
return ctx.Err()
default:
}
respMu.Lock()
resp[k] = res[0]
respMu.Unlock()
if len(res) == 1 {
if opt.ImageIDFile != "" {
return ioutil.WriteFile(opt.ImageIDFile, []byte(res[0].ExporterResponse["containerimage.digest"]), 0644)
}
return nil
}
if pushNames != "" {
progress.Write(pw, "merging manifest list", func() error {
descs := make([]specs.Descriptor, 0, len(res))
for _, r := range res {
s, ok := r.ExporterResponse["containerimage.digest"]
if ok {
descs = append(descs, specs.Descriptor{
Digest: digest.Digest(s),
MediaType: images.MediaTypeDockerSchema2ManifestList,
Size: -1,
})
}
}
if len(descs) > 0 {
itpull := imagetools.New(imagetools.Opt{
Auth: auth,
})
names := strings.Split(pushNames, ",")
dt, desc, err := itpull.Combine(ctx, names[0], descs)
if err != nil {
return err
}
if opt.ImageIDFile != "" {
return ioutil.WriteFile(opt.ImageIDFile, []byte(desc.Digest), 0644)
}
itpush := imagetools.New(imagetools.Opt{
Auth: auth,
})
for _, n := range names {
nn, err := reference.ParseNormalizedNamed(n)
if err != nil {
return err
}
if err := itpush.Push(ctx, nn, desc, dt); err != nil {
return err
}
}
respMu.Lock()
resp[k] = &client.SolveResponse{
ExporterResponse: map[string]string{
"containerimage.digest": desc.Digest.String(),
},
}
respMu.Unlock()
}
return nil
})
}
return nil
})
for i, dp := range dps {
func(i int, dp driverPair) {
pw := mw.WithPrefix(k, multiTarget)
c := clients[dp.driverIndex]
var statusCh chan *client.SolveStatus var statusCh chan *client.SolveStatus
if pw != nil { if pw != nil {
pw = progress.ResetTime(pw) pw = progress.ResetTime(pw)
@ -293,18 +665,23 @@ func Build(ctx context.Context, drivers []DriverInfo, opt map[string]Options, do
} }
eg.Go(func() error { eg.Go(func() error {
rr, err := c.Solve(ctx, nil, so, statusCh) defer wg.Done()
rr, err := c.Solve(ctx, nil, *dp.so, statusCh)
if err != nil { if err != nil {
return err return err
} }
mu.Lock() res[i] = rr
resp[k] = rr
mu.Unlock()
if opt.ImageIDFile != "" {
return ioutil.WriteFile(opt.ImageIDFile, []byte(rr.ExporterResponse["containerimage.digest"]), 0644)
}
return nil return nil
}) })
}(i, dp)
}
return nil
}()
if err != nil {
return nil, err
}
} }
if err := eg.Wait(); err != nil { if err := eg.Wait(); err != nil {
@ -423,6 +800,8 @@ func notSupported(d driver.Driver, f driver.Feature) error {
return errors.Errorf("%s feature is currently not supported for %s driver. Please switch to a different driver (eg. \"docker buildx create --use\")", f, d.Factory().Name()) return errors.Errorf("%s feature is currently not supported for %s driver. Please switch to a different driver (eg. \"docker buildx create --use\")", f, d.Factory().Name())
} }
type dockerLoadCallback func(name string) (io.WriteCloser, func(), error)
func newDockerLoader(ctx context.Context, d DockerAPI, name string, mw *progress.MultiWriter) (io.WriteCloser, func(), error) { func newDockerLoader(ctx context.Context, d DockerAPI, name string, mw *progress.MultiWriter) (io.WriteCloser, func(), error) {
c, err := d.DockerAPI(name) c, err := d.DockerAPI(name)
if err != nil { if err != nil {

@ -180,7 +180,7 @@ func buildTargets(ctx context.Context, dockerCli command.Cli, opts map[string]bu
defer cancel() defer cancel()
pw := progress.NewPrinter(ctx2, os.Stderr, progressMode) pw := progress.NewPrinter(ctx2, os.Stderr, progressMode)
_, err = build.Build(ctx, dis, opts, dockerAPI(dockerCli), pw) _, err = build.Build(ctx, dis, opts, dockerAPI(dockerCli), dockerCli.ConfigFile(), pw)
return err return err
} }

@ -6,6 +6,7 @@ import (
"encoding/json" "encoding/json"
"github.com/containerd/containerd/content" "github.com/containerd/containerd/content"
"github.com/containerd/containerd/errdefs"
"github.com/containerd/containerd/images" "github.com/containerd/containerd/images"
"github.com/docker/distribution/reference" "github.com/docker/distribution/reference"
"github.com/opencontainers/go-digest" "github.com/opencontainers/go-digest"
@ -152,10 +153,17 @@ func (r *Resolver) Push(ctx context.Context, ref reference.Named, desc ocispec.D
} }
cw, err := p.Push(ctx, desc) cw, err := p.Push(ctx, desc)
if err != nil { if err != nil {
if errdefs.IsAlreadyExists(err) {
return nil
}
return err return err
} }
return content.Copy(ctx, cw, bytes.NewReader(dt), desc.Size, desc.Digest) err = content.Copy(ctx, cw, bytes.NewReader(dt), desc.Size, desc.Digest)
if errdefs.IsAlreadyExists(err) {
return nil
}
return err
} }
func (r *Resolver) loadConfig(ctx context.Context, in string, dt []byte) (*ocispec.Image, error) { func (r *Resolver) loadConfig(ctx context.Context, in string, dt []byte) (*ocispec.Image, error) {

@ -3,6 +3,7 @@ package progress
import ( import (
"context" "context"
"strings" "strings"
"sync"
"github.com/moby/buildkit/client" "github.com/moby/buildkit/client"
"golang.org/x/sync/errgroup" "golang.org/x/sync/errgroup"
@ -11,6 +12,8 @@ import (
type MultiWriter struct { type MultiWriter struct {
w Writer w Writer
eg *errgroup.Group eg *errgroup.Group
once sync.Once
ready chan struct{}
} }
func (mw *MultiWriter) WithPrefix(pfx string, force bool) Writer { func (mw *MultiWriter) WithPrefix(pfx string, force bool) Writer {
@ -21,6 +24,9 @@ func (mw *MultiWriter) WithPrefix(pfx string, force bool) Writer {
in: in, in: in,
} }
mw.eg.Go(func() error { mw.eg.Go(func() error {
mw.once.Do(func() {
close(mw.ready)
})
for { for {
select { select {
case v, ok := <-in: case v, ok := <-in:
@ -77,7 +83,10 @@ func NewMultiWriter(pw Writer) *MultiWriter {
} }
eg, _ := errgroup.WithContext(context.TODO()) eg, _ := errgroup.WithContext(context.TODO())
ready := make(chan struct{})
go func() { go func() {
<-ready
eg.Wait() eg.Wait()
close(pw.Status()) close(pw.Status())
}() }()
@ -85,6 +94,7 @@ func NewMultiWriter(pw Writer) *MultiWriter {
return &MultiWriter{ return &MultiWriter{
w: pw, w: pw,
eg: eg, eg: eg,
ready: ready,
} }
} }

@ -1,7 +1,11 @@
package progress package progress
import ( import (
"time"
"github.com/moby/buildkit/client" "github.com/moby/buildkit/client"
"github.com/moby/buildkit/identity"
"github.com/opencontainers/go-digest"
) )
type Writer interface { type Writer interface {
@ -9,3 +13,31 @@ type Writer interface {
Err() error Err() error
Status() chan *client.SolveStatus Status() chan *client.SolveStatus
} }
func Write(w Writer, name string, f func() error) {
status := w.Status()
dgst := digest.FromBytes([]byte(identity.NewID()))
tm := time.Now()
vtx := client.Vertex{
Digest: dgst,
Name: name,
Started: &tm,
}
status <- &client.SolveStatus{
Vertexes: []*client.Vertex{&vtx},
}
err := f()
tm2 := time.Now()
vtx2 := vtx
vtx2.Completed = &tm2
if err != nil {
vtx2.Error = err.Error()
}
status <- &client.SolveStatus{
Vertexes: []*client.Vertex{&vtx2},
}
}

@ -27,12 +27,12 @@ github.com/beorn7/perks/quantile
# github.com/containerd/console v0.0.0-20181022165439-0650fd9eeb50 # github.com/containerd/console v0.0.0-20181022165439-0650fd9eeb50
github.com/containerd/console github.com/containerd/console
# github.com/containerd/containerd v1.3.0-0.20190321141026-ceba56893a76 # github.com/containerd/containerd v1.3.0-0.20190321141026-ceba56893a76
github.com/containerd/containerd/platforms
github.com/containerd/containerd/images github.com/containerd/containerd/images
github.com/containerd/containerd/platforms
github.com/containerd/containerd/content github.com/containerd/containerd/content
github.com/containerd/containerd/errdefs
github.com/containerd/containerd/remotes github.com/containerd/containerd/remotes
github.com/containerd/containerd/remotes/docker github.com/containerd/containerd/remotes/docker
github.com/containerd/containerd/errdefs
github.com/containerd/containerd/log github.com/containerd/containerd/log
github.com/containerd/containerd/content/local github.com/containerd/containerd/content/local
github.com/containerd/containerd/labels github.com/containerd/containerd/labels
@ -98,6 +98,7 @@ github.com/davecgh/go-spew/spew
# github.com/docker/cli v0.0.0-20190321234815-f40f9c240ab0 => github.com/tiborvass/cli v0.0.0-20190419012645-1ed02c40fe68 # github.com/docker/cli v0.0.0-20190321234815-f40f9c240ab0 => github.com/tiborvass/cli v0.0.0-20190419012645-1ed02c40fe68
github.com/docker/cli/cli/compose/loader github.com/docker/cli/cli/compose/loader
github.com/docker/cli/cli/compose/types github.com/docker/cli/cli/compose/types
github.com/docker/cli/cli/config/types
github.com/docker/cli/cli-plugins/manager github.com/docker/cli/cli-plugins/manager
github.com/docker/cli/cli-plugins/plugin github.com/docker/cli/cli-plugins/plugin
github.com/docker/cli/cli/command github.com/docker/cli/cli/command
@ -106,7 +107,6 @@ github.com/docker/cli/cli
github.com/docker/cli/cli/config github.com/docker/cli/cli/config
github.com/docker/cli/cli/context/docker github.com/docker/cli/cli/context/docker
github.com/docker/cli/opts github.com/docker/cli/opts
github.com/docker/cli/cli/config/types
github.com/docker/cli/cli/compose/interpolation github.com/docker/cli/cli/compose/interpolation
github.com/docker/cli/cli/compose/schema github.com/docker/cli/cli/compose/schema
github.com/docker/cli/cli/compose/template github.com/docker/cli/cli/compose/template

Loading…
Cancel
Save