package commands import ( "context" "net/url" "os" "strings" "github.com/docker/buildx/build" "github.com/docker/buildx/driver" ctxkube "github.com/docker/buildx/driver/kubernetes/context" remoteutil "github.com/docker/buildx/driver/remote/util" "github.com/docker/buildx/store" "github.com/docker/buildx/store/storeutil" "github.com/docker/buildx/util/platformutil" "github.com/docker/buildx/util/progress" "github.com/docker/cli/cli/command" "github.com/docker/cli/cli/context/docker" ctxstore "github.com/docker/cli/cli/context/store" dopts "github.com/docker/cli/opts" dockerclient "github.com/docker/docker/client" "github.com/moby/buildkit/util/grpcerrors" specs "github.com/opencontainers/image-spec/specs-go/v1" "github.com/pkg/errors" "github.com/sirupsen/logrus" "golang.org/x/sync/errgroup" "google.golang.org/grpc/codes" "k8s.io/client-go/tools/clientcmd" ) // validateEndpoint validates that endpoint is either a context or a docker host func validateEndpoint(dockerCli command.Cli, ep string) (string, error) { de, err := storeutil.GetDockerEndpoint(dockerCli, ep) if err == nil && de != "" { if ep == "default" { return de, nil } return ep, nil } h, err := dopts.ParseHost(true, ep) if err != nil { return "", errors.Wrapf(err, "failed to parse endpoint %s", ep) } return h, nil } // validateBuildkitEndpoint validates that endpoint is a valid buildkit host func validateBuildkitEndpoint(ep string) (string, error) { if err := remoteutil.IsValidEndpoint(ep); err != nil { return "", err } return ep, nil } // driversForNodeGroup returns drivers for a nodegroup instance func driversForNodeGroup(ctx context.Context, dockerCli command.Cli, ng *store.NodeGroup, contextPathHash string) ([]build.DriverInfo, error) { eg, _ := errgroup.WithContext(ctx) dis := make([]build.DriverInfo, len(ng.Nodes)) var f driver.Factory if ng.Driver != "" { var err error f, err = driver.GetFactory(ng.Driver, true) if err != nil { return nil, err } } else { // empty driver means nodegroup was implicitly created as a default // driver for a docker context and allows falling back to a // docker-container driver for older daemon that doesn't support // buildkit (< 18.06). ep := ng.Nodes[0].Endpoint dockerapi, err := clientForEndpoint(dockerCli, ep) if err != nil { return nil, err } // check if endpoint is healthy is needed to determine the driver type. // if this fails then can't continue with driver selection. if _, err = dockerapi.Ping(ctx); err != nil { return nil, err } f, err = driver.GetDefaultFactory(ctx, ep, dockerapi, false) if err != nil { return nil, err } ng.Driver = f.Name() } imageopt, err := storeutil.GetImageConfig(dockerCli, ng) if err != nil { return nil, err } for i, n := range ng.Nodes { func(i int, n store.Node) { eg.Go(func() error { di := build.DriverInfo{ Name: n.Name, Platform: n.Platforms, ProxyConfig: storeutil.GetProxyConfig(dockerCli), } defer func() { dis[i] = di }() dockerapi, err := clientForEndpoint(dockerCli, n.Endpoint) if err != nil { di.Err = err return nil } // TODO: replace the following line with dockerclient.WithAPIVersionNegotiation option in clientForEndpoint dockerapi.NegotiateAPIVersion(ctx) contextStore := dockerCli.ContextStore() var kcc driver.KubeClientConfig kcc, err = configFromContext(n.Endpoint, contextStore) if err != nil { // err is returned if n.Endpoint is non-context name like "unix:///var/run/docker.sock". // try again with name="default". // FIXME: n should retain real context name. kcc, err = configFromContext("default", contextStore) if err != nil { logrus.Error(err) } } tryToUseKubeConfigInCluster := false if kcc == nil { tryToUseKubeConfigInCluster = true } else { if _, err := kcc.ClientConfig(); err != nil { tryToUseKubeConfigInCluster = true } } if tryToUseKubeConfigInCluster { kccInCluster := driver.KubeClientConfigInCluster{} if _, err := kccInCluster.ClientConfig(); err == nil { logrus.Debug("using kube config in cluster") kcc = kccInCluster } } d, err := driver.GetDriver(ctx, "buildx_buildkit_"+n.Name, f, n.Endpoint, dockerapi, imageopt.Auth, kcc, n.Flags, n.Files, n.DriverOpts, n.Platforms, contextPathHash) if err != nil { di.Err = err return nil } di.Driver = d di.ImageOpt = imageopt return nil }) }(i, n) } if err := eg.Wait(); err != nil { return nil, err } return dis, nil } func configFromContext(endpointName string, s ctxstore.Reader) (clientcmd.ClientConfig, error) { if strings.HasPrefix(endpointName, "kubernetes://") { u, _ := url.Parse(endpointName) if kubeconfig := u.Query().Get("kubeconfig"); kubeconfig != "" { _ = os.Setenv(clientcmd.RecommendedConfigPathEnvVar, kubeconfig) } rules := clientcmd.NewDefaultClientConfigLoadingRules() apiConfig, err := rules.Load() if err != nil { return nil, err } return clientcmd.NewDefaultClientConfig(*apiConfig, &clientcmd.ConfigOverrides{}), nil } return ctxkube.ConfigFromContext(endpointName, s) } // clientForEndpoint returns a docker client for an endpoint func clientForEndpoint(dockerCli command.Cli, name string) (dockerclient.APIClient, error) { list, err := dockerCli.ContextStore().List() if err != nil { return nil, err } for _, l := range list { if l.Name == name { dep, ok := l.Endpoints["docker"] if !ok { return nil, errors.Errorf("context %q does not have a Docker endpoint", name) } epm, ok := dep.(docker.EndpointMeta) if !ok { return nil, errors.Errorf("endpoint %q is not of type EndpointMeta, %T", dep, dep) } ep, err := docker.WithTLSData(dockerCli.ContextStore(), name, epm) if err != nil { return nil, err } clientOpts, err := ep.ClientOpts() if err != nil { return nil, err } return dockerclient.NewClientWithOpts(clientOpts...) } } ep := docker.Endpoint{ EndpointMeta: docker.EndpointMeta{ Host: name, }, } clientOpts, err := ep.ClientOpts() if err != nil { return nil, err } return dockerclient.NewClientWithOpts(clientOpts...) } func getInstanceOrDefault(ctx context.Context, dockerCli command.Cli, instance, contextPathHash string) ([]build.DriverInfo, error) { var defaultOnly bool if instance == "default" && instance != dockerCli.CurrentContext() { return nil, errors.Errorf("use `docker --context=default buildx` to switch to default context") } if instance == "default" || instance == dockerCli.CurrentContext() { instance = "" defaultOnly = true } list, err := dockerCli.ContextStore().List() if err != nil { return nil, err } for _, l := range list { if l.Name == instance { return nil, errors.Errorf("use `docker --context=%s buildx` to switch to context %s", instance, instance) } } if instance != "" { return getInstanceByName(ctx, dockerCli, instance, contextPathHash) } return getDefaultDrivers(ctx, dockerCli, defaultOnly, contextPathHash) } func getInstanceByName(ctx context.Context, dockerCli command.Cli, instance, contextPathHash string) ([]build.DriverInfo, error) { txn, release, err := storeutil.GetStore(dockerCli) if err != nil { return nil, err } defer release() ng, err := txn.NodeGroupByName(instance) if err != nil { return nil, err } return driversForNodeGroup(ctx, dockerCli, ng, contextPathHash) } // getDefaultDrivers returns drivers based on current cli config func getDefaultDrivers(ctx context.Context, dockerCli command.Cli, defaultOnly bool, contextPathHash string) ([]build.DriverInfo, error) { txn, release, err := storeutil.GetStore(dockerCli) if err != nil { return nil, err } defer release() if !defaultOnly { ng, err := storeutil.GetCurrentInstance(txn, dockerCli) if err != nil { return nil, err } if ng != nil { return driversForNodeGroup(ctx, dockerCli, ng, contextPathHash) } } imageopt, err := storeutil.GetImageConfig(dockerCli, nil) if err != nil { return nil, err } d, err := driver.GetDriver(ctx, "buildx_buildkit_default", nil, "", dockerCli.Client(), imageopt.Auth, nil, nil, nil, nil, nil, contextPathHash) if err != nil { return nil, err } return []build.DriverInfo{ { Name: "default", Driver: d, ImageOpt: imageopt, ProxyConfig: storeutil.GetProxyConfig(dockerCli), }, }, nil } func loadInfoData(ctx context.Context, d *dinfo) error { if d.di.Driver == nil { return nil } info, err := d.di.Driver.Info(ctx) if err != nil { return err } d.info = info if info.Status == driver.Running { c, err := d.di.Driver.Client(ctx) if err != nil { return err } workers, err := c.ListWorkers(ctx) if err != nil { return errors.Wrap(err, "listing workers") } for _, w := range workers { d.platforms = append(d.platforms, w.Platforms...) } d.platforms = platformutil.Dedupe(d.platforms) inf, err := c.Info(ctx) if err != nil { if st, ok := grpcerrors.AsGRPCStatus(err); ok && st.Code() == codes.Unimplemented { d.version, err = d.di.Driver.Version(ctx) if err != nil { return errors.Wrap(err, "getting version") } } } else { d.version = inf.BuildkitVersion.Version } } return nil } func loadNodeGroupData(ctx context.Context, dockerCli command.Cli, ngi *nginfo) error { eg, _ := errgroup.WithContext(ctx) dis, err := driversForNodeGroup(ctx, dockerCli, ngi.ng, "") if err != nil { return err } ngi.drivers = make([]dinfo, len(dis)) for i, di := range dis { d := di ngi.drivers[i].di = &d func(d *dinfo) { eg.Go(func() error { if err := loadInfoData(ctx, d); err != nil { d.err = err } return nil }) }(&ngi.drivers[i]) } if eg.Wait(); err != nil { return err } kubernetesDriverCount := 0 for _, di := range ngi.drivers { if di.info != nil && len(di.info.DynamicNodes) > 0 { kubernetesDriverCount++ } } isAllKubernetesDrivers := len(ngi.drivers) == kubernetesDriverCount if isAllKubernetesDrivers { var drivers []dinfo var dynamicNodes []store.Node for _, di := range ngi.drivers { // dynamic nodes are used in Kubernetes driver. // Kubernetes pods are dynamically mapped to BuildKit Nodes. if di.info != nil && len(di.info.DynamicNodes) > 0 { for i := 0; i < len(di.info.DynamicNodes); i++ { // all []dinfo share *build.DriverInfo and *driver.Info diClone := di if pl := di.info.DynamicNodes[i].Platforms; len(pl) > 0 { diClone.platforms = pl } drivers = append(drivers, di) } dynamicNodes = append(dynamicNodes, di.info.DynamicNodes...) } } // not append (remove the static nodes in the store) ngi.ng.Nodes = dynamicNodes ngi.drivers = drivers ngi.ng.Dynamic = true } return nil } func hasNodeGroup(list []*nginfo, ngi *nginfo) bool { for _, l := range list { if ngi.ng.Name == l.ng.Name { return true } } return false } func dockerAPI(dockerCli command.Cli) *api { return &api{dockerCli: dockerCli} } type api struct { dockerCli command.Cli } func (a *api) DockerAPI(name string) (dockerclient.APIClient, error) { if name == "" { name = a.dockerCli.CurrentContext() } return clientForEndpoint(a.dockerCli, name) } type dinfo struct { di *build.DriverInfo info *driver.Info platforms []specs.Platform version string err error } type nginfo struct { ng *store.NodeGroup drivers []dinfo err error } // inactive checks if all nodes are inactive for this builder func (n *nginfo) inactive() bool { for idx := range n.ng.Nodes { d := n.drivers[idx] if d.info != nil && d.info.Status == driver.Running { return false } } return true } func boot(ctx context.Context, ngi *nginfo) (bool, error) { toBoot := make([]int, 0, len(ngi.drivers)) for i, d := range ngi.drivers { if d.err != nil || d.di.Err != nil || d.di.Driver == nil || d.info == nil { continue } if d.info.Status != driver.Running { toBoot = append(toBoot, i) } } if len(toBoot) == 0 { return false, nil } printer, err := progress.NewPrinter(context.TODO(), os.Stderr, os.Stderr, progress.PrinterModeAuto) if err != nil { return false, err } baseCtx := ctx eg, _ := errgroup.WithContext(ctx) for _, idx := range toBoot { func(idx int) { eg.Go(func() error { pw := progress.WithPrefix(printer, ngi.ng.Nodes[idx].Name, len(toBoot) > 1) _, err := driver.Boot(ctx, baseCtx, ngi.drivers[idx].di.Driver, pw) if err != nil { ngi.drivers[idx].err = err } return nil }) }(idx) } err = eg.Wait() err1 := printer.Wait() if err == nil { err = err1 } return true, err }