|
|
|
package driver
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
|
|
|
|
"github.com/docker/buildx/store"
|
|
|
|
"github.com/docker/buildx/util/progress"
|
|
|
|
clitypes "github.com/docker/cli/cli/config/types"
|
|
|
|
controlapi "github.com/moby/buildkit/api/services/control"
|
|
|
|
"github.com/moby/buildkit/client"
|
|
|
|
"github.com/moby/buildkit/util/grpcerrors"
|
|
|
|
"github.com/pkg/errors"
|
|
|
|
"google.golang.org/grpc/codes"
|
|
|
|
)
|
|
|
|
|
|
|
|
var ErrNotRunning = errors.Errorf("driver not running")
|
|
|
|
var ErrNotConnecting = errors.Errorf("driver not connecting")
|
|
|
|
|
|
|
|
type Status int
|
|
|
|
|
|
|
|
const (
|
|
|
|
Inactive Status = iota
|
|
|
|
Starting
|
|
|
|
Running
|
|
|
|
Stopping
|
|
|
|
Stopped
|
|
|
|
)
|
|
|
|
|
|
|
|
func (s Status) String() string {
|
|
|
|
switch s {
|
|
|
|
case Inactive:
|
|
|
|
return "inactive"
|
|
|
|
case Starting:
|
|
|
|
return "starting"
|
|
|
|
case Running:
|
|
|
|
return "running"
|
|
|
|
case Stopping:
|
|
|
|
return "stopping"
|
|
|
|
case Stopped:
|
|
|
|
return "stopped"
|
|
|
|
}
|
|
|
|
return "unknown"
|
|
|
|
}
|
|
|
|
|
|
|
|
type Info struct {
|
|
|
|
Status Status
|
|
|
|
// DynamicNodes must be empty if the actual nodes are statically listed in the store
|
|
|
|
DynamicNodes []store.Node
|
|
|
|
}
|
|
|
|
|
|
|
|
type Auth interface {
|
|
|
|
GetAuthConfig(registryHostname string) (clitypes.AuthConfig, error)
|
|
|
|
}
|
|
|
|
|
|
|
|
type Driver interface {
|
|
|
|
Factory() Factory
|
|
|
|
Bootstrap(context.Context, progress.Logger) error
|
|
|
|
Info(context.Context) (*Info, error)
|
|
|
|
Version(context.Context) (string, error)
|
|
|
|
Stop(ctx context.Context, force bool) error
|
|
|
|
Rm(ctx context.Context, force, rmVolume, rmDaemon bool) error
|
|
|
|
Client(ctx context.Context) (*client.Client, error)
|
|
|
|
Features(ctx context.Context) map[Feature]bool
|
|
|
|
IsMobyDriver() bool
|
|
|
|
Config() InitConfig
|
|
|
|
}
|
|
|
|
|
|
|
|
func Boot(ctx, clientContext context.Context, d Driver, pw progress.Writer) (*client.Client, error) {
|
|
|
|
try := 0
|
|
|
|
for {
|
|
|
|
info, err := d.Info(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
try++
|
|
|
|
if info.Status != Running {
|
|
|
|
if try > 2 {
|
|
|
|
return nil, errors.Errorf("failed to bootstrap %T driver in attempts", d)
|
|
|
|
}
|
|
|
|
if err := d.Bootstrap(ctx, pw.Write); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
c, err := d.Client(clientContext)
|
|
|
|
if err != nil {
|
|
|
|
if errors.Cause(err) == ErrNotRunning && try <= 2 {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return c, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func HistoryAPISupported(ctx context.Context, c *client.Client) (res bool) {
|
|
|
|
res = true
|
|
|
|
checkErrF := func(err error) {
|
|
|
|
if s, ok := grpcerrors.AsGRPCStatus(err); ok {
|
|
|
|
if s.Code() == codes.Unimplemented {
|
|
|
|
res = false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
cl, err := c.ControlClient().ListenBuildHistory(ctx, &controlapi.BuildHistoryRequest{
|
|
|
|
ActiveOnly: true,
|
|
|
|
Ref: "buildx-dummy-ref", // dummy ref to check if the server supports the API
|
|
|
|
EarlyExit: true,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
checkErrF(err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
_, err = cl.Recv()
|
|
|
|
checkErrF(err)
|
|
|
|
return
|
|
|
|
}
|