|
|
|
package store
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
|
|
|
|
"github.com/containerd/containerd/platforms"
|
|
|
|
"github.com/docker/buildx/util/confutil"
|
|
|
|
"github.com/docker/buildx/util/platformutil"
|
|
|
|
specs "github.com/opencontainers/image-spec/specs-go/v1"
|
|
|
|
"github.com/pkg/errors"
|
|
|
|
"github.com/sirupsen/logrus"
|
|
|
|
)
|
|
|
|
|
|
|
|
type NodeGroup struct {
|
|
|
|
Name string
|
|
|
|
Driver string
|
|
|
|
Nodes []Node
|
|
|
|
Dynamic bool
|
|
|
|
|
|
|
|
// skip the following fields from being saved in the store
|
|
|
|
DockerContext bool `json:"-"`
|
|
|
|
}
|
|
|
|
|
|
|
|
type Node struct {
|
|
|
|
Name string
|
|
|
|
Endpoint string
|
|
|
|
Platforms []specs.Platform
|
|
|
|
Flags []string
|
|
|
|
DriverOpts map[string]string
|
|
|
|
|
|
|
|
Files map[string][]byte
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ng *NodeGroup) Leave(name string) error {
|
|
|
|
if ng.Dynamic {
|
|
|
|
return errors.New("dynamic node group does not support Leave")
|
|
|
|
}
|
|
|
|
i := ng.findNode(name)
|
|
|
|
if i == -1 {
|
|
|
|
return errors.Errorf("node %q not found for %s", name, ng.Name)
|
|
|
|
}
|
|
|
|
if len(ng.Nodes) == 1 {
|
|
|
|
return errors.Errorf("can not leave last node, do you want to rm instance instead?")
|
|
|
|
}
|
|
|
|
ng.Nodes = append(ng.Nodes[:i], ng.Nodes[i+1:]...)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ng *NodeGroup) Update(name, endpoint string, platforms []string, endpointsSet bool, actionAppend bool, flags []string, configFile string, do map[string]string) error {
|
|
|
|
if ng.Dynamic {
|
|
|
|
return errors.New("dynamic node group does not support Update")
|
|
|
|
}
|
|
|
|
i := ng.findNode(name)
|
|
|
|
if i == -1 && !actionAppend {
|
|
|
|
if len(ng.Nodes) > 0 {
|
|
|
|
return errors.Errorf("node %s not found, did you mean to append?", name)
|
|
|
|
}
|
|
|
|
ng.Nodes = nil
|
|
|
|
}
|
|
|
|
|
|
|
|
pp, err := platformutil.Parse(platforms)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
var files map[string][]byte
|
|
|
|
if configFile != "" {
|
|
|
|
files, err = confutil.LoadConfigFiles(configFile)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if i != -1 {
|
|
|
|
n := ng.Nodes[i]
|
|
|
|
needsRestart := false
|
|
|
|
if endpointsSet {
|
|
|
|
n.Endpoint = endpoint
|
|
|
|
needsRestart = true
|
|
|
|
}
|
|
|
|
if len(platforms) > 0 {
|
|
|
|
n.Platforms = pp
|
|
|
|
}
|
|
|
|
if flags != nil {
|
|
|
|
n.Flags = flags
|
|
|
|
needsRestart = true
|
|
|
|
}
|
|
|
|
if do != nil {
|
|
|
|
n.DriverOpts = do
|
|
|
|
needsRestart = true
|
|
|
|
}
|
|
|
|
if configFile != "" {
|
|
|
|
for k, v := range files {
|
|
|
|
n.Files[k] = v
|
|
|
|
}
|
|
|
|
needsRestart = true
|
|
|
|
}
|
|
|
|
if needsRestart {
|
|
|
|
logrus.Warn("new settings may not be used until builder is restarted")
|
|
|
|
}
|
|
|
|
|
|
|
|
ng.Nodes[i] = n
|
|
|
|
if err := ng.validateDuplicates(endpoint, i); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
if name == "" {
|
|
|
|
name = ng.nextNodeName()
|
|
|
|
}
|
|
|
|
|
|
|
|
name, err = ValidateName(name)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
n := Node{
|
|
|
|
Name: name,
|
|
|
|
Endpoint: endpoint,
|
|
|
|
Platforms: pp,
|
|
|
|
Flags: flags,
|
|
|
|
DriverOpts: do,
|
|
|
|
Files: files,
|
|
|
|
}
|
|
|
|
|
|
|
|
ng.Nodes = append(ng.Nodes, n)
|
|
|
|
|
|
|
|
if err := ng.validateDuplicates(endpoint, len(ng.Nodes)-1); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ng *NodeGroup) Copy() *NodeGroup {
|
|
|
|
nodes := make([]Node, len(ng.Nodes))
|
|
|
|
for i, node := range ng.Nodes {
|
|
|
|
nodes[i] = *node.Copy()
|
|
|
|
}
|
|
|
|
return &NodeGroup{
|
|
|
|
Name: ng.Name,
|
|
|
|
Driver: ng.Driver,
|
|
|
|
Nodes: nodes,
|
|
|
|
Dynamic: ng.Dynamic,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (n *Node) Copy() *Node {
|
|
|
|
platforms := []specs.Platform{}
|
|
|
|
copy(platforms, n.Platforms)
|
|
|
|
flags := []string{}
|
|
|
|
copy(flags, n.Flags)
|
|
|
|
driverOpts := map[string]string{}
|
|
|
|
for k, v := range n.DriverOpts {
|
|
|
|
driverOpts[k] = v
|
|
|
|
}
|
|
|
|
files := map[string][]byte{}
|
|
|
|
for k, v := range n.Files {
|
|
|
|
vv := []byte{}
|
|
|
|
copy(vv, v)
|
|
|
|
files[k] = vv
|
|
|
|
}
|
|
|
|
return &Node{
|
|
|
|
Name: n.Name,
|
|
|
|
Endpoint: n.Endpoint,
|
|
|
|
Platforms: platforms,
|
|
|
|
Flags: flags,
|
|
|
|
DriverOpts: driverOpts,
|
|
|
|
Files: files,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ng *NodeGroup) validateDuplicates(ep string, idx int) error {
|
|
|
|
i := 0
|
|
|
|
for _, n := range ng.Nodes {
|
|
|
|
if n.Endpoint == ep {
|
|
|
|
i++
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if i > 1 {
|
|
|
|
return errors.Errorf("invalid duplicate endpoint %s", ep)
|
|
|
|
}
|
|
|
|
|
|
|
|
m := map[string]struct{}{}
|
|
|
|
for _, p := range ng.Nodes[idx].Platforms {
|
|
|
|
m[platforms.Format(p)] = struct{}{}
|
|
|
|
}
|
|
|
|
|
|
|
|
for i := range ng.Nodes {
|
|
|
|
if i == idx {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
ng.Nodes[i].Platforms = filterPlatforms(ng.Nodes[i].Platforms, m)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ng *NodeGroup) findNode(name string) int {
|
|
|
|
for i, n := range ng.Nodes {
|
|
|
|
if n.Name == name {
|
|
|
|
return i
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return -1
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ng *NodeGroup) nextNodeName() string {
|
|
|
|
i := 0
|
|
|
|
for {
|
|
|
|
name := fmt.Sprintf("%s%d", ng.Name, i)
|
|
|
|
if ii := ng.findNode(name); ii != -1 {
|
|
|
|
i++
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
return name
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func filterPlatforms(in []specs.Platform, m map[string]struct{}) []specs.Platform {
|
|
|
|
out := make([]specs.Platform, 0, len(in))
|
|
|
|
for _, p := range in {
|
|
|
|
if _, ok := m[platforms.Format(p)]; !ok {
|
|
|
|
out = append(out, p)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return out
|
|
|
|
}
|