mirror of https://github.com/k3d-io/k3d
Merge pull request #670 from rancher/feature/cluster-edit-ports
commit
4380675446
@ -0,0 +1,124 @@ |
||||
/* |
||||
Copyright © 2020-2021 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 cluster |
||||
|
||||
import ( |
||||
"github.com/rancher/k3d/v4/cmd/util" |
||||
cliutil "github.com/rancher/k3d/v4/cmd/util" |
||||
"github.com/rancher/k3d/v4/pkg/client" |
||||
conf "github.com/rancher/k3d/v4/pkg/config/v1alpha3" |
||||
"github.com/rancher/k3d/v4/pkg/runtimes" |
||||
k3d "github.com/rancher/k3d/v4/pkg/types" |
||||
log "github.com/sirupsen/logrus" |
||||
"github.com/spf13/cobra" |
||||
) |
||||
|
||||
// NewCmdClusterEdit returns a new cobra command
|
||||
func NewCmdClusterEdit() *cobra.Command { |
||||
|
||||
// create new cobra command
|
||||
cmd := &cobra.Command{ |
||||
Use: "edit NAME", |
||||
Short: "[EXPERIMENTAL] Edit cluster(s).", |
||||
Long: `[EXPERIMENTAL] Edit cluster(s).`, |
||||
Args: cobra.ExactArgs(1), |
||||
Aliases: []string{"update"}, |
||||
ValidArgsFunction: util.ValidArgsAvailableNodes, |
||||
Run: func(cmd *cobra.Command, args []string) { |
||||
|
||||
existingCluster, changeset := parseEditClusterCmd(cmd, args) |
||||
|
||||
log.Debugf("===== Current =====\n%+v\n===== Changeset =====\n%+v\n", existingCluster, changeset) |
||||
|
||||
if err := client.ClusterEditChangesetSimple(cmd.Context(), runtimes.SelectedRuntime, existingCluster, changeset); err != nil { |
||||
log.Fatalf("Failed to update the cluster: %v", err) |
||||
} |
||||
|
||||
log.Infof("Successfully updated %s", existingCluster.Name) |
||||
|
||||
}, |
||||
} |
||||
|
||||
// add subcommands
|
||||
|
||||
// add flags
|
||||
cmd.Flags().StringArray("port-add", nil, "[EXPERIMENTAL] Map ports from the node containers (via the serverlb) to the host (Format: `[HOST:][HOSTPORT:]CONTAINERPORT[/PROTOCOL][@NODEFILTER]`)\n - Example: `k3d node edit k3d-mycluster-serverlb --port-add 8080:80`") |
||||
|
||||
// done
|
||||
return cmd |
||||
} |
||||
|
||||
// parseEditClusterCmd parses the command input into variables required to delete nodes
|
||||
func parseEditClusterCmd(cmd *cobra.Command, args []string) (*k3d.Cluster, *conf.SimpleConfig) { |
||||
|
||||
existingCluster, err := client.ClusterGet(cmd.Context(), runtimes.SelectedRuntime, &k3d.Cluster{Name: args[0]}) |
||||
if err != nil { |
||||
log.Fatalln(err) |
||||
} |
||||
|
||||
if existingCluster == nil { |
||||
log.Infof("Cluster %s not found", args[0]) |
||||
return nil, nil |
||||
} |
||||
|
||||
changeset := conf.SimpleConfig{} |
||||
|
||||
/* |
||||
* --port-add |
||||
*/ |
||||
portFlags, err := cmd.Flags().GetStringArray("port-add") |
||||
if err != nil { |
||||
log.Errorln(err) |
||||
return nil, nil |
||||
} |
||||
|
||||
// init portmap
|
||||
changeset.Ports = []conf.PortWithNodeFilters{} |
||||
|
||||
portFilterMap := make(map[string][]string, 1) |
||||
for _, portFlag := range portFlags { |
||||
|
||||
// split node filter from the specified volume
|
||||
portmap, filters, err := cliutil.SplitFiltersFromFlag(portFlag) |
||||
if err != nil { |
||||
log.Fatalln(err) |
||||
} |
||||
|
||||
// create new entry or append filter to existing entry
|
||||
if _, exists := portFilterMap[portmap]; exists { |
||||
log.Fatalln("Same Portmapping can not be used for multiple nodes") |
||||
} else { |
||||
portFilterMap[portmap] = filters |
||||
} |
||||
} |
||||
|
||||
for port, nodeFilters := range portFilterMap { |
||||
changeset.Ports = append(changeset.Ports, conf.PortWithNodeFilters{ |
||||
Port: port, |
||||
NodeFilters: nodeFilters, |
||||
}) |
||||
} |
||||
|
||||
log.Tracef("PortFilterMap: %+v", portFilterMap) |
||||
|
||||
return existingCluster, &changeset |
||||
} |
@ -0,0 +1,127 @@ |
||||
/* |
||||
Copyright © 2020-2021 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 client |
||||
|
||||
import ( |
||||
"context" |
||||
"errors" |
||||
"fmt" |
||||
"strings" |
||||
|
||||
"github.com/docker/go-connections/nat" |
||||
"github.com/rancher/k3d/v4/pkg/config/types" |
||||
config "github.com/rancher/k3d/v4/pkg/config/v1alpha3" |
||||
"github.com/rancher/k3d/v4/pkg/runtimes" |
||||
k3d "github.com/rancher/k3d/v4/pkg/types" |
||||
"github.com/rancher/k3d/v4/pkg/util" |
||||
log "github.com/sirupsen/logrus" |
||||
"gopkg.in/yaml.v2" |
||||
) |
||||
|
||||
var ( |
||||
ErrNodeAddPortsExists error = errors.New("port exists on target") |
||||
) |
||||
|
||||
func TransformPorts(ctx context.Context, runtime runtimes.Runtime, cluster *k3d.Cluster, portsWithNodeFilters []config.PortWithNodeFilters) error { |
||||
nodeCount := len(cluster.Nodes) |
||||
nodeList := cluster.Nodes |
||||
|
||||
for _, portWithNodeFilters := range portsWithNodeFilters { |
||||
log.Tracef("inspecting port mapping for %s with nodefilters %s", portWithNodeFilters.Port, portWithNodeFilters.NodeFilters) |
||||
if len(portWithNodeFilters.NodeFilters) == 0 && nodeCount > 1 { |
||||
log.Infof("portmapping '%s' lacks a nodefilter, but there's more than one node: defaulting to %s", portWithNodeFilters.Port, types.DefaultTargetsNodefiltersPortMappings) |
||||
portWithNodeFilters.NodeFilters = types.DefaultTargetsNodefiltersPortMappings |
||||
} |
||||
|
||||
for _, f := range portWithNodeFilters.NodeFilters { |
||||
if strings.HasPrefix(f, "loadbalancer") { |
||||
log.Infof("portmapping '%s' targets the loadbalancer: defaulting to %s", portWithNodeFilters.Port, types.DefaultTargetsNodefiltersPortMappings) |
||||
portWithNodeFilters.NodeFilters = types.DefaultTargetsNodefiltersPortMappings |
||||
break |
||||
} |
||||
} |
||||
|
||||
filteredNodes, err := util.FilterNodesWithSuffix(nodeList, portWithNodeFilters.NodeFilters) |
||||
if err != nil { |
||||
return err |
||||
} |
||||
|
||||
for suffix, nodes := range filteredNodes { |
||||
portmappings, err := nat.ParsePortSpec(portWithNodeFilters.Port) |
||||
if err != nil { |
||||
return fmt.Errorf("error parsing port spec '%s': %+v", portWithNodeFilters.Port, err) |
||||
} |
||||
|
||||
if suffix == "proxy" || suffix == util.NodeFilterSuffixNone { // proxy is the default suffix for port mappings
|
||||
if cluster.ServerLoadBalancer == nil { |
||||
return fmt.Errorf("port-mapping of type 'proxy' specified, but loadbalancer is disabled") |
||||
} |
||||
if err := addPortMappings(cluster.ServerLoadBalancer.Node, portmappings); err != nil { |
||||
return err |
||||
} |
||||
for _, pm := range portmappings { |
||||
if err := loadbalancerAddPortConfigs(cluster.ServerLoadBalancer, pm, nodes); err != nil { |
||||
return err |
||||
} |
||||
} |
||||
} else if suffix == "direct" { |
||||
if len(nodes) > 1 { |
||||
return fmt.Errorf("error: cannot apply a direct port-mapping (%s) to more than one node", portmappings) |
||||
} |
||||
for _, node := range nodes { |
||||
if err := addPortMappings(node, portmappings); err != nil { |
||||
return err |
||||
} |
||||
} |
||||
} else if suffix != util.NodeFilterMapKeyAll { |
||||
return fmt.Errorf("error adding port mappings: unknown suffix %s", suffix) |
||||
} |
||||
} |
||||
|
||||
} |
||||
|
||||
// print generated loadbalancer config
|
||||
if log.GetLevel() >= log.DebugLevel { |
||||
yamlized, err := yaml.Marshal(cluster.ServerLoadBalancer.Config) |
||||
if err != nil { |
||||
log.Errorf("error printing loadbalancer config: %v", err) |
||||
} else { |
||||
log.Debugf("generated loadbalancer config:\n%s", string(yamlized)) |
||||
} |
||||
} |
||||
return nil |
||||
} |
||||
|
||||
func addPortMappings(node *k3d.Node, portmappings []nat.PortMapping) error { |
||||
|
||||
if node.Ports == nil { |
||||
node.Ports = nat.PortMap{} |
||||
} |
||||
for _, pm := range portmappings { |
||||
if _, exists := node.Ports[pm.Port]; exists { |
||||
node.Ports[pm.Port] = append(node.Ports[pm.Port], pm.Binding) |
||||
} else { |
||||
node.Ports[pm.Port] = []nat.PortBinding{pm.Binding} |
||||
} |
||||
} |
||||
return nil |
||||
} |
Loading…
Reference in new issue