|
|
|
package driver
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"net"
|
|
|
|
"os"
|
|
|
|
"sort"
|
|
|
|
"strings"
|
|
|
|
"sync"
|
|
|
|
|
|
|
|
dockerclient "github.com/docker/docker/client"
|
|
|
|
"github.com/moby/buildkit/client"
|
|
|
|
specs "github.com/opencontainers/image-spec/specs-go/v1"
|
|
|
|
"github.com/pkg/errors"
|
|
|
|
"k8s.io/client-go/rest"
|
|
|
|
)
|
|
|
|
|
|
|
|
type Factory interface {
|
|
|
|
Name() string
|
|
|
|
Usage() string
|
|
|
|
Priority(ctx context.Context, endpoint string, api dockerclient.APIClient) int
|
|
|
|
New(ctx context.Context, cfg InitConfig) (Driver, error)
|
|
|
|
AllowsInstances() bool
|
|
|
|
}
|
|
|
|
|
|
|
|
type BuildkitConfig struct {
|
|
|
|
// Entitlements []string
|
|
|
|
// Rootless bool
|
|
|
|
}
|
|
|
|
|
|
|
|
type KubeClientConfig interface {
|
|
|
|
ClientConfig() (*rest.Config, error)
|
|
|
|
Namespace() (string, bool, error)
|
|
|
|
}
|
|
|
|
|
|
|
|
type KubeClientConfigInCluster struct{}
|
|
|
|
|
|
|
|
func (k KubeClientConfigInCluster) ClientConfig() (*rest.Config, error) {
|
|
|
|
return rest.InClusterConfig()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (k KubeClientConfigInCluster) Namespace() (string, bool, error) {
|
|
|
|
namespace, err := os.ReadFile("/var/run/secrets/kubernetes.io/serviceaccount/namespace")
|
|
|
|
if err != nil {
|
|
|
|
return "", false, err
|
|
|
|
}
|
|
|
|
return strings.TrimSpace(string(namespace)), true, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
type InitConfig struct {
|
|
|
|
// This object needs updates to be generic for different drivers
|
|
|
|
Name string
|
|
|
|
EndpointAddr string
|
|
|
|
DockerAPI dockerclient.APIClient
|
|
|
|
KubeClientConfig KubeClientConfig
|
|
|
|
BuildkitFlags []string
|
|
|
|
Files map[string][]byte
|
|
|
|
DriverOpts map[string]string
|
|
|
|
SecurityOpts map[string]string
|
|
|
|
Auth Auth
|
|
|
|
Platforms []specs.Platform
|
|
|
|
// ContextPathHash can be used for determining pods in the driver instance
|
|
|
|
ContextPathHash string
|
|
|
|
}
|
|
|
|
|
|
|
|
var drivers map[string]Factory
|
|
|
|
|
|
|
|
func Register(f Factory) {
|
|
|
|
if drivers == nil {
|
|
|
|
drivers = map[string]Factory{}
|
|
|
|
}
|
|
|
|
drivers[f.Name()] = f
|
|
|
|
}
|
|
|
|
|
|
|
|
func GetDefaultFactory(ctx context.Context, ep string, c dockerclient.APIClient, instanceRequired bool) (Factory, error) {
|
|
|
|
if len(drivers) == 0 {
|
|
|
|
return nil, errors.Errorf("no drivers available")
|
|
|
|
}
|
|
|
|
type p struct {
|
|
|
|
f Factory
|
|
|
|
priority int
|
|
|
|
}
|
|
|
|
dd := make([]p, 0, len(drivers))
|
|
|
|
for _, f := range drivers {
|
|
|
|
if instanceRequired && !f.AllowsInstances() {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
dd = append(dd, p{f: f, priority: f.Priority(ctx, ep, c)})
|
|
|
|
}
|
|
|
|
sort.Slice(dd, func(i, j int) bool {
|
|
|
|
return dd[i].priority < dd[j].priority
|
|
|
|
})
|
|
|
|
return dd[0].f, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func GetFactory(name string, instanceRequired bool) (Factory, error) {
|
|
|
|
for _, f := range drivers {
|
|
|
|
if f.Name() == name {
|
|
|
|
if instanceRequired && !f.AllowsInstances() {
|
|
|
|
return nil, errors.Errorf("additional instances of driver %q cannot be created", name)
|
|
|
|
}
|
|
|
|
return f, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil, errors.Errorf("failed to find driver %q", name)
|
|
|
|
}
|
|
|
|
|
|
|
|
func GetDriver(ctx context.Context, name string, f Factory, endpointAddr string, api dockerclient.APIClient, auth Auth, kcc KubeClientConfig, flags []string, files map[string][]byte, do map[string]string, so map[string]string, platforms []specs.Platform, contextPathHash string) (*DriverHandle, error) {
|
|
|
|
ic := InitConfig{
|
|
|
|
EndpointAddr: endpointAddr,
|
|
|
|
DockerAPI: api,
|
|
|
|
KubeClientConfig: kcc,
|
|
|
|
Name: name,
|
|
|
|
BuildkitFlags: flags,
|
|
|
|
DriverOpts: do,
|
|
|
|
SecurityOpts: so,
|
|
|
|
Auth: auth,
|
|
|
|
Platforms: platforms,
|
|
|
|
ContextPathHash: contextPathHash,
|
|
|
|
Files: files,
|
|
|
|
}
|
|
|
|
if f == nil {
|
|
|
|
var err error
|
|
|
|
f, err = GetDefaultFactory(ctx, endpointAddr, api, false)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
d, err := f.New(ctx, ic)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return &DriverHandle{Driver: d}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func GetFactories(instanceRequired bool) []Factory {
|
|
|
|
ds := make([]Factory, 0, len(drivers))
|
|
|
|
for _, d := range drivers {
|
|
|
|
if instanceRequired && !d.AllowsInstances() {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
ds = append(ds, d)
|
|
|
|
}
|
|
|
|
sort.Slice(ds, func(i, j int) bool {
|
|
|
|
return ds[i].Name() < ds[j].Name()
|
|
|
|
})
|
|
|
|
return ds
|
|
|
|
}
|
|
|
|
|
|
|
|
type DriverHandle struct {
|
|
|
|
Driver
|
|
|
|
client *client.Client
|
|
|
|
err error
|
|
|
|
once sync.Once
|
|
|
|
featuresOnce sync.Once
|
|
|
|
features map[Feature]bool
|
|
|
|
historyAPISupportedOnce sync.Once
|
|
|
|
historyAPISupported bool
|
|
|
|
hostGatewayIPOnce sync.Once
|
|
|
|
hostGatewayIP net.IP
|
|
|
|
hostGatewayIPErr error
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *DriverHandle) Client(ctx context.Context) (*client.Client, error) {
|
|
|
|
d.once.Do(func() {
|
|
|
|
d.client, d.err = d.Driver.Client(ctx)
|
|
|
|
})
|
|
|
|
return d.client, d.err
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *DriverHandle) Features(ctx context.Context) map[Feature]bool {
|
|
|
|
d.featuresOnce.Do(func() {
|
|
|
|
d.features = d.Driver.Features(ctx)
|
|
|
|
})
|
|
|
|
return d.features
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *DriverHandle) HistoryAPISupported(ctx context.Context) bool {
|
|
|
|
d.historyAPISupportedOnce.Do(func() {
|
|
|
|
if c, err := d.Client(ctx); err == nil {
|
|
|
|
d.historyAPISupported = historyAPISupported(ctx, c)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
return d.historyAPISupported
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *DriverHandle) HostGatewayIP(ctx context.Context) (net.IP, error) {
|
|
|
|
d.hostGatewayIPOnce.Do(func() {
|
|
|
|
if !d.Driver.IsMobyDriver() {
|
|
|
|
d.hostGatewayIPErr = errors.New("host-gateway is only supported with the docker driver")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
c, err := d.Client(ctx)
|
|
|
|
if err != nil {
|
|
|
|
d.hostGatewayIPErr = err
|
|
|
|
return
|
|
|
|
}
|
|
|
|
workers, err := c.ListWorkers(ctx)
|
|
|
|
if err != nil {
|
|
|
|
d.hostGatewayIPErr = errors.Wrap(err, "listing workers")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
for _, w := range workers {
|
|
|
|
// should match github.com/docker/docker/builder/builder-next/worker/label.HostGatewayIP const
|
|
|
|
if v, ok := w.Labels["org.mobyproject.buildkit.worker.moby.host-gateway-ip"]; ok && v != "" {
|
|
|
|
ip := net.ParseIP(v)
|
|
|
|
if ip == nil {
|
|
|
|
d.hostGatewayIPErr = errors.Errorf("failed to parse host-gateway IP: %s", v)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
d.hostGatewayIP = ip
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
d.hostGatewayIPErr = errors.New("host-gateway IP not found")
|
|
|
|
})
|
|
|
|
return d.hostGatewayIP, d.hostGatewayIPErr
|
|
|
|
}
|