package docker import ( "archive/tar" "bytes" "context" "io" "io/ioutil" "net" "os" "time" "github.com/docker/buildx/driver" "github.com/docker/buildx/util/progress" "github.com/docker/docker/api/types" dockertypes "github.com/docker/docker/api/types" "github.com/docker/docker/api/types/container" "github.com/docker/docker/api/types/network" dockerclient "github.com/docker/docker/client" "github.com/docker/docker/pkg/stdcopy" "github.com/moby/buildkit/client" "github.com/pkg/errors" ) var defaultBuildkitImage = "moby/buildkit:buildx-stable-1" // TODO: make this verified type Driver struct { driver.InitConfig factory driver.Factory netMode string image string } func (d *Driver) Bootstrap(ctx context.Context, l progress.Logger) error { return progress.Wrap("[internal] booting buildkit", l, func(sub progress.SubLogger) error { _, err := d.DockerAPI.ContainerInspect(ctx, d.Name) if err != nil { if dockerclient.IsErrNotFound(err) { return d.create(ctx, sub) } return err } return sub.Wrap("starting container "+d.Name, func() error { if err := d.start(ctx, sub); err != nil { return err } if err := d.wait(ctx); err != nil { return err } return nil }) }) } func (d *Driver) create(ctx context.Context, l progress.SubLogger) error { imageName := defaultBuildkitImage if d.image != "" { imageName = d.image } if err := l.Wrap("pulling image "+imageName, func() error { rc, err := d.DockerAPI.ImageCreate(ctx, imageName, types.ImageCreateOptions{}) if err != nil { return err } _, err = io.Copy(ioutil.Discard, rc) return err }); err != nil { return err } cfg := &container.Config{ Image: imageName, } if d.InitConfig.BuildkitFlags != nil { cfg.Cmd = d.InitConfig.BuildkitFlags } if err := l.Wrap("creating container "+d.Name, func() error { hc := &container.HostConfig{ Privileged: true, } if d.netMode != "" { hc.NetworkMode = container.NetworkMode(d.netMode) } _, err := d.DockerAPI.ContainerCreate(ctx, cfg, hc, &network.NetworkingConfig{}, d.Name) if err != nil { return err } if f := d.InitConfig.ConfigFile; f != "" { buf, err := readFileToTar(f) if err != nil { return err } if err := d.DockerAPI.CopyToContainer(ctx, d.Name, "/", buf, dockertypes.CopyToContainerOptions{}); err != nil { return err } } if err := d.start(ctx, l); err != nil { return err } if err := d.wait(ctx); err != nil { return err } return nil }); err != nil { return err } return nil } func (d *Driver) wait(ctx context.Context) error { try := 0 for { if err := d.run(ctx, []string{"buildctl", "debug", "workers"}); err != nil { if try > 10 { return err } select { case <-ctx.Done(): return ctx.Err() case <-time.After(time.Duration(100+try*20) * time.Millisecond): try++ continue } } return nil } } func (d *Driver) exec(ctx context.Context, cmd []string) (string, net.Conn, error) { execConfig := types.ExecConfig{ Cmd: cmd, AttachStdin: true, AttachStdout: true, AttachStderr: true, } response, err := d.DockerAPI.ContainerExecCreate(ctx, d.Name, execConfig) if err != nil { return "", nil, err } execID := response.ID if execID == "" { return "", nil, errors.New("exec ID empty") } resp, err := d.DockerAPI.ContainerExecAttach(ctx, execID, types.ExecStartCheck{}) if err != nil { return "", nil, err } return execID, resp.Conn, nil } func (d *Driver) run(ctx context.Context, cmd []string) error { id, conn, err := d.exec(ctx, cmd) if err != nil { return err } if _, err := io.Copy(ioutil.Discard, conn); err != nil { return err } conn.Close() resp, err := d.DockerAPI.ContainerExecInspect(ctx, id) if err != nil { return err } if resp.ExitCode != 0 { return errors.Errorf("exit code %d", resp.ExitCode) } return nil } func (d *Driver) start(ctx context.Context, l progress.SubLogger) error { return d.DockerAPI.ContainerStart(ctx, d.Name, types.ContainerStartOptions{}) } func (d *Driver) Info(ctx context.Context) (*driver.Info, error) { container, err := d.DockerAPI.ContainerInspect(ctx, d.Name) if err != nil { if dockerclient.IsErrNotFound(err) { return &driver.Info{ Status: driver.Inactive, }, nil } return nil, err } if container.State.Running { return &driver.Info{ Status: driver.Running, }, nil } return &driver.Info{ Status: driver.Stopped, }, nil } func (d *Driver) Stop(ctx context.Context, force bool) error { info, err := d.Info(ctx) if err != nil { return err } if info.Status == driver.Running { return d.DockerAPI.ContainerStop(ctx, d.Name, nil) } return nil } func (d *Driver) Rm(ctx context.Context, force bool) error { info, err := d.Info(ctx) if err != nil { return err } if info.Status != driver.Inactive { return d.DockerAPI.ContainerRemove(ctx, d.Name, dockertypes.ContainerRemoveOptions{ RemoveVolumes: true, Force: true, }) } return nil } func (d *Driver) Client(ctx context.Context) (*client.Client, error) { _, conn, err := d.exec(ctx, []string{"buildctl", "dial-stdio"}) if err != nil { return nil, err } conn = demuxConn(conn) return client.New(ctx, "", client.WithDialer(func(string, time.Duration) (net.Conn, error) { return conn, nil })) } func (d *Driver) Factory() driver.Factory { return d.factory } func (d *Driver) Features() map[driver.Feature]bool { return map[driver.Feature]bool{ driver.OCIExporter: true, driver.DockerExporter: true, driver.CacheExport: true, driver.MultiPlatform: true, } } func demuxConn(c net.Conn) net.Conn { pr, pw := io.Pipe() // TODO: rewrite parser with Reader() to avoid goroutine switch go stdcopy.StdCopy(pw, os.Stdout, c) return &demux{ Conn: c, Reader: pr, } } type demux struct { net.Conn io.Reader } func (d *demux) Read(dt []byte) (int, error) { return d.Reader.Read(dt) } func readFileToTar(fn string) (*bytes.Buffer, error) { buf := bytes.NewBuffer(nil) tw := tar.NewWriter(buf) dt, err := ioutil.ReadFile(fn) if err != nil { return nil, err } if err := tw.WriteHeader(&tar.Header{ Name: "/etc/buildkit/buildkitd.toml", Size: int64(len(dt)), Mode: 0644, }); err != nil { return nil, err } if _, err := tw.Write(dt); err != nil { return nil, err } if err := tw.Close(); err != nil { return nil, err } return buf, nil }