/* Copyright © 2020 The k3d Author(s) Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ package docker import ( "context" "fmt" "net" "strings" "github.com/docker/docker/api/types" "github.com/docker/docker/api/types/filters" "github.com/docker/docker/api/types/network" "github.com/docker/docker/client" runtimeErr "github.com/rancher/k3d/v4/pkg/runtimes/errors" k3d "github.com/rancher/k3d/v4/pkg/types" log "github.com/sirupsen/logrus" ) // CreateNetworkIfNotPresent creates a new docker network // @return: network name, exists, error func (d Docker) CreateNetworkIfNotPresent(ctx context.Context, name string) (string, bool, error) { // (0) create new docker client docker, err := client.NewClientWithOpts(client.FromEnv, client.WithAPIVersionNegotiation()) if err != nil { log.Errorln("Failed to create docker client") return "", false, err } defer docker.Close() // (1) configure list filters filter := filters.NewArgs() filter.Add("name", fmt.Sprintf("^/?%s$", name)) // regex filtering for exact name match // (2) get filtered list of networks networkList, err := docker.NetworkList(ctx, types.NetworkListOptions{ Filters: filter, }) if err != nil { log.Errorln("Failed to list docker networks") return "", false, err } // (2.1) If possible, return an existing network if len(networkList) > 1 { log.Warnf("Found %d networks instead of only one. Choosing the first one: '%s'.", len(networkList), networkList[0].ID) } if len(networkList) > 0 { log.Infof("Network with name '%s' already exists with ID '%s'", name, networkList[0].ID) return networkList[0].ID, true, nil } // (3) Create a new network network, err := docker.NetworkCreate(ctx, name, types.NetworkCreate{ Labels: k3d.DefaultObjectLabels, }) if err != nil { log.Errorln("Failed to create network") return "", false, err } log.Infof("Created network '%s'", name) return network.ID, false, nil } // DeleteNetwork deletes a network func (d Docker) DeleteNetwork(ctx context.Context, ID string) error { // (0) create new docker client docker, err := client.NewClientWithOpts(client.FromEnv, client.WithAPIVersionNegotiation()) if err != nil { log.Errorln("Failed to create docker client") return err } defer docker.Close() // (3) delete network if err := docker.NetworkRemove(ctx, ID); err != nil { if strings.HasSuffix(err.Error(), "active endpoints") { return runtimeErr.ErrRuntimeNetworkNotEmpty } return err } return nil } // GetNetwork gets information about a network by its ID func GetNetwork(ctx context.Context, ID string) (types.NetworkResource, error) { docker, err := client.NewClientWithOpts(client.FromEnv, client.WithAPIVersionNegotiation()) if err != nil { log.Errorln("Failed to create docker client") return types.NetworkResource{}, err } defer docker.Close() return docker.NetworkInspect(ctx, ID, types.NetworkInspectOptions{}) } // GetGatewayIP returns the IP of the network gateway func GetGatewayIP(ctx context.Context, network string) (net.IP, error) { bridgeNetwork, err := GetNetwork(ctx, network) if err != nil { log.Errorf("Failed to get bridge network with name '%s'", network) return nil, err } gatewayIP := net.ParseIP(bridgeNetwork.IPAM.Config[0].Gateway) return gatewayIP, nil } // ConnectNodeToNetwork connects a node to a network func (d Docker) ConnectNodeToNetwork(ctx context.Context, node *k3d.Node, networkName string) error { // get container container, err := getNodeContainer(ctx, node) if err != nil { return err } // get docker client docker, err := client.NewClientWithOpts(client.FromEnv, client.WithAPIVersionNegotiation()) if err != nil { log.Errorln("Failed to create docker client") return err } defer docker.Close() // get network networkResource, err := GetNetwork(ctx, networkName) if err != nil { log.Errorf("Failed to get network '%s'", networkName) return err } // connect container to network return docker.NetworkConnect(ctx, networkResource.ID, container.ID, &network.EndpointSettings{}) } // DisconnectNodeFromNetwork disconnects a node from a network (u don't say :O) func (d Docker) DisconnectNodeFromNetwork(ctx context.Context, node *k3d.Node, networkName string) error { // get container container, err := getNodeContainer(ctx, node) if err != nil { return err } // get docker client docker, err := client.NewClientWithOpts(client.FromEnv, client.WithAPIVersionNegotiation()) if err != nil { log.Errorln("Failed to create docker client") return err } defer docker.Close() // get network networkResource, err := GetNetwork(ctx, networkName) if err != nil { log.Errorf("Failed to get network '%s'", networkName) return err } return docker.NetworkDisconnect(ctx, networkResource.ID, container.ID, true) }