k3s/pkg/agent/flannel/setup.go

175 lines
4.6 KiB
Go
Raw Normal View History

2019-01-01 08:23:01 +00:00
package flannel
import (
"context"
"errors"
2019-09-03 23:41:54 +00:00
"fmt"
2019-09-06 00:39:18 +00:00
"os"
2019-01-09 16:54:15 +00:00
"path/filepath"
2019-01-01 08:23:01 +00:00
"strings"
"time"
2019-01-09 16:54:15 +00:00
"github.com/rancher/k3s/pkg/agent/util"
"github.com/rancher/k3s/pkg/daemons/config"
"github.com/rancher/k3s/pkg/version"
2019-01-01 08:23:01 +00:00
"github.com/sirupsen/logrus"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
v1 "k8s.io/client-go/kubernetes/typed/core/v1"
2019-01-01 08:23:01 +00:00
)
const (
cniConf = `{
"name":"cbr0",
"cniVersion":"0.3.1",
"plugins":[
{
"type":"flannel",
"delegate":{
2019-08-11 05:50:21 +00:00
"hairpinMode":true,
2019-01-01 08:23:01 +00:00
"forceAddress":true,
"isDefaultGateway":true
}
},
{
"type":"portmap",
"capabilities":{
"portMappings":true
}
}
]
}
`
2019-09-03 23:41:54 +00:00
flannelConf = `{
"Network": "%CIDR%",
"Backend": %backend%
2019-01-01 08:23:01 +00:00
}
`
2019-09-03 23:41:54 +00:00
vxlanBackend = `{
"Type": "vxlan"
}`
2019-11-27 17:25:55 +00:00
hostGWBackend = `{
"Type": "host-gw"
}`
2019-09-03 23:41:54 +00:00
ipsecBackend = `{
"Type": "ipsec",
"UDPEncap": true,
"PSK": "%psk%"
}`
wireguardBackend = `{
"Type": "extension",
"PreStartupCommand": "wg genkey | tee %flannelConfDir%/privatekey | wg pubkey",
"PostStartupCommand": "export SUBNET_IP=$(echo $SUBNET | cut -d'/' -f 1); ip link del flannel.1 2>/dev/null; echo $PATH >&2; wg-add.sh flannel.1 && wg set flannel.1 listen-port 51820 private-key %flannelConfDir%/privatekey && ip addr add $SUBNET_IP/32 dev flannel.1 && ip link set flannel.1 up && ip route add $NETWORK dev flannel.1",
2019-09-03 23:41:54 +00:00
"ShutdownCommand": "ip link del flannel.1",
"SubnetAddCommand": "read PUBLICKEY; wg set flannel.1 peer $PUBLICKEY endpoint $PUBLIC_IP:51820 allowed-ips $SUBNET persistent-keepalive 25",
2019-09-03 23:41:54 +00:00
"SubnetRemoveCommand": "read PUBLICKEY; wg set flannel.1 peer $PUBLICKEY remove"
}`
2019-01-01 08:23:01 +00:00
)
2019-09-03 23:41:54 +00:00
func Prepare(ctx context.Context, nodeConfig *config.Node) error {
if err := createCNIConf(nodeConfig.AgentConfig.CNIConfDir); err != nil {
2019-01-09 16:54:15 +00:00
return err
}
2019-09-03 23:41:54 +00:00
return createFlannelConf(nodeConfig)
2019-01-09 16:54:15 +00:00
}
func Run(ctx context.Context, nodeConfig *config.Node, nodes v1.NodeInterface) error {
2019-09-03 23:41:54 +00:00
nodeName := nodeConfig.AgentConfig.NodeName
2019-01-01 08:23:01 +00:00
for {
2020-03-26 21:08:47 +00:00
node, err := nodes.Get(ctx, nodeName, metav1.GetOptions{})
2019-01-01 08:23:01 +00:00
if err == nil && node.Spec.PodCIDR != "" {
break
}
if err == nil {
logrus.Info("Waiting for node " + nodeName + " CIDR not assigned yet")
2019-01-01 08:23:01 +00:00
} else {
logrus.Infof("Waiting for node %s: %v", nodeName, err)
2019-01-01 08:23:01 +00:00
}
time.Sleep(2 * time.Second)
}
logrus.Info("Node CIDR assigned for: " + nodeName)
2019-01-01 08:23:01 +00:00
2019-01-09 16:54:15 +00:00
go func() {
err := flannel(ctx, nodeConfig.FlannelIface, nodeConfig.FlannelConf, nodeConfig.AgentConfig.KubeConfigKubelet)
if err != nil && !errors.Is(err, context.Canceled) {
logrus.Fatalf("flannel exited: %v", err)
}
2019-01-09 16:54:15 +00:00
}()
2019-01-01 08:23:01 +00:00
return nil
2019-01-01 08:23:01 +00:00
}
2019-01-09 16:54:15 +00:00
func createCNIConf(dir string) error {
if dir == "" {
return nil
}
p := filepath.Join(dir, "10-flannel.conflist")
return util.WriteFile(p, cniConf)
2019-01-01 08:23:01 +00:00
}
2019-09-03 23:41:54 +00:00
func createFlannelConf(nodeConfig *config.Node) error {
if nodeConfig.FlannelConf == "" {
2019-01-09 16:54:15 +00:00
return nil
}
2019-09-03 23:41:54 +00:00
if nodeConfig.FlannelConfOverride {
logrus.Infof("Using custom flannel conf defined at %s", nodeConfig.FlannelConf)
2019-08-08 05:56:09 +00:00
return nil
}
confJSON := strings.ReplaceAll(flannelConf, "%CIDR%", nodeConfig.AgentConfig.ClusterCIDR.String())
2019-09-03 23:41:54 +00:00
var backendConf string
switch nodeConfig.FlannelBackend {
case config.FlannelBackendVXLAN:
backendConf = vxlanBackend
2019-11-27 17:25:55 +00:00
case config.FlannelBackendHostGW:
backendConf = hostGWBackend
2019-09-03 23:41:54 +00:00
case config.FlannelBackendIPSEC:
backendConf = strings.ReplaceAll(ipsecBackend, "%psk%", nodeConfig.AgentConfig.IPSECPSK)
2019-09-06 00:39:18 +00:00
if err := setupStrongSwan(nodeConfig); err != nil {
return err
}
2019-09-03 23:41:54 +00:00
case config.FlannelBackendWireguard:
backendConf = strings.ReplaceAll(wireguardBackend, "%flannelConfDir%", filepath.Dir(nodeConfig.FlannelConf))
2019-09-03 23:41:54 +00:00
default:
return fmt.Errorf("Cannot configure unknown flannel backend '%s'", nodeConfig.FlannelBackend)
}
confJSON = strings.ReplaceAll(confJSON, "%backend%", backendConf)
2019-09-03 23:41:54 +00:00
return util.WriteFile(nodeConfig.FlannelConf, confJSON)
2019-01-01 08:23:01 +00:00
}
2019-09-06 00:39:18 +00:00
func setupStrongSwan(nodeConfig *config.Node) error {
// if data dir env is not set point to root
dataDir := os.Getenv(version.ProgramUpper + "_DATA_DIR")
2019-09-06 00:39:18 +00:00
if dataDir == "" {
dataDir = "/"
2019-09-06 00:39:18 +00:00
}
2020-04-25 07:22:48 +00:00
dataDir = filepath.Join(dataDir, "etc", "strongswan")
2019-09-06 00:39:18 +00:00
info, err := os.Lstat(nodeConfig.AgentConfig.StrongSwanDir)
// something exists but is not a symlink, return
if err == nil && info.Mode()&os.ModeSymlink == 0 {
return nil
}
if err == nil {
target, err := os.Readlink(nodeConfig.AgentConfig.StrongSwanDir)
// current link is the same, return
if err == nil && target == dataDir {
return nil
}
}
2019-09-06 00:39:18 +00:00
// clean up strongswan old link
os.Remove(nodeConfig.AgentConfig.StrongSwanDir)
// make new strongswan link
return os.Symlink(dataDir, nodeConfig.AgentConfig.StrongSwanDir)
2019-11-26 20:30:12 +00:00
}