2019-01-01 08:23:01 +00:00
|
|
|
package control
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"math/rand"
|
|
|
|
"net"
|
2022-03-23 08:37:11 +00:00
|
|
|
"net/http"
|
2019-01-01 08:23:01 +00:00
|
|
|
"os"
|
|
|
|
"path/filepath"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
"time"
|
|
|
|
|
2022-03-28 20:45:39 +00:00
|
|
|
"github.com/k3s-io/k3s/pkg/authenticator"
|
2022-03-02 23:47:27 +00:00
|
|
|
"github.com/k3s-io/k3s/pkg/cluster"
|
|
|
|
"github.com/k3s-io/k3s/pkg/daemons/config"
|
|
|
|
"github.com/k3s-io/k3s/pkg/daemons/control/deps"
|
|
|
|
"github.com/k3s-io/k3s/pkg/daemons/executor"
|
|
|
|
"github.com/k3s-io/k3s/pkg/util"
|
|
|
|
"github.com/k3s-io/k3s/pkg/version"
|
2019-12-16 18:44:13 +00:00
|
|
|
"github.com/pkg/errors"
|
2019-01-01 08:23:01 +00:00
|
|
|
"github.com/sirupsen/logrus"
|
2021-09-13 22:24:00 +00:00
|
|
|
authorizationv1 "k8s.io/api/authorization/v1"
|
2019-10-15 21:17:26 +00:00
|
|
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
2021-03-18 22:40:29 +00:00
|
|
|
"k8s.io/apimachinery/pkg/util/wait"
|
2019-11-09 06:07:12 +00:00
|
|
|
"k8s.io/client-go/kubernetes"
|
2021-09-13 22:24:00 +00:00
|
|
|
authorizationv1client "k8s.io/client-go/kubernetes/typed/authorization/v1"
|
2019-10-15 21:17:26 +00:00
|
|
|
"k8s.io/client-go/tools/clientcmd"
|
2019-01-01 08:23:01 +00:00
|
|
|
"k8s.io/kubernetes/pkg/kubeapiserver/authorizer/modes"
|
2021-09-10 21:02:51 +00:00
|
|
|
proxyutil "k8s.io/kubernetes/pkg/proxy/util"
|
2020-04-27 17:09:58 +00:00
|
|
|
|
|
|
|
// for client metric registration
|
|
|
|
_ "k8s.io/component-base/metrics/prometheus/restclient"
|
2019-01-01 08:23:01 +00:00
|
|
|
)
|
|
|
|
|
2022-03-29 07:35:54 +00:00
|
|
|
func getLocalhostIP(serviceCIDR []*net.IPNet) net.IP {
|
|
|
|
IPv6OnlyService, _ := util.IsIPv6OnlyCIDRs(serviceCIDR)
|
|
|
|
if IPv6OnlyService {
|
|
|
|
return net.ParseIP("::1")
|
|
|
|
}
|
|
|
|
return net.ParseIP("127.0.0.1")
|
|
|
|
}
|
2019-09-03 23:41:54 +00:00
|
|
|
|
2022-03-23 08:37:11 +00:00
|
|
|
type roundTripFunc func(req *http.Request) (*http.Response, error)
|
|
|
|
|
|
|
|
func (w roundTripFunc) RoundTrip(req *http.Request) (*http.Response, error) {
|
|
|
|
return w(req)
|
|
|
|
}
|
|
|
|
|
2019-01-01 08:23:01 +00:00
|
|
|
func Server(ctx context.Context, cfg *config.Control) error {
|
|
|
|
rand.Seed(time.Now().UTC().UnixNano())
|
|
|
|
|
2022-02-24 19:01:14 +00:00
|
|
|
if err := prepare(ctx, cfg); err != nil {
|
2019-12-16 18:44:13 +00:00
|
|
|
return errors.Wrap(err, "preparing server")
|
2019-01-01 08:23:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
cfg.Runtime.Tunnel = setupTunnel()
|
2021-09-10 21:02:51 +00:00
|
|
|
proxyutil.DisableProxyHostnameCheck = true
|
2019-01-01 08:23:01 +00:00
|
|
|
|
2022-03-28 20:45:39 +00:00
|
|
|
authArgs := []string{
|
|
|
|
"--basic-auth-file=" + cfg.Runtime.PasswdFile,
|
|
|
|
"--client-ca-file=" + cfg.Runtime.ClientCA,
|
|
|
|
}
|
|
|
|
auth, err := authenticator.FromArgs(authArgs)
|
2021-10-12 06:13:10 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2022-03-28 20:45:39 +00:00
|
|
|
cfg.Runtime.Authenticator = auth
|
2019-01-01 08:23:01 +00:00
|
|
|
|
2021-02-12 15:35:57 +00:00
|
|
|
if !cfg.DisableAPIServer {
|
2022-02-24 19:01:14 +00:00
|
|
|
go waitForAPIServerHandlers(ctx, cfg.Runtime)
|
2021-10-12 06:13:10 +00:00
|
|
|
|
2022-02-24 19:01:14 +00:00
|
|
|
if err := apiServer(ctx, cfg); err != nil {
|
2021-02-12 15:35:57 +00:00
|
|
|
return err
|
|
|
|
}
|
2022-02-16 22:19:58 +00:00
|
|
|
}
|
2019-11-04 11:08:47 +00:00
|
|
|
|
2022-02-16 22:19:58 +00:00
|
|
|
// Wait for an apiserver to become available before starting additional controllers,
|
|
|
|
// even if we're not running an apiserver locally.
|
|
|
|
if err := waitForAPIServerInBackground(ctx, cfg.Runtime); err != nil {
|
|
|
|
return err
|
2021-02-12 15:35:57 +00:00
|
|
|
}
|
2019-01-01 08:23:01 +00:00
|
|
|
|
2021-02-12 15:35:57 +00:00
|
|
|
if !cfg.DisableScheduler {
|
2022-02-24 19:01:14 +00:00
|
|
|
if err := scheduler(ctx, cfg); err != nil {
|
2020-04-27 17:09:58 +00:00
|
|
|
return err
|
|
|
|
}
|
2019-01-01 08:23:01 +00:00
|
|
|
}
|
2021-02-12 15:35:57 +00:00
|
|
|
if !cfg.DisableControllerManager {
|
2022-02-24 19:01:14 +00:00
|
|
|
if err := controllerManager(ctx, cfg); err != nil {
|
2021-02-12 15:35:57 +00:00
|
|
|
return err
|
|
|
|
}
|
2020-04-27 17:09:58 +00:00
|
|
|
}
|
2019-01-01 08:23:01 +00:00
|
|
|
|
2019-10-15 21:17:26 +00:00
|
|
|
if !cfg.DisableCCM {
|
2022-02-24 19:01:14 +00:00
|
|
|
if err := cloudControllerManager(ctx, cfg); err != nil {
|
2021-06-29 14:28:38 +00:00
|
|
|
return err
|
|
|
|
}
|
2019-10-15 21:17:26 +00:00
|
|
|
}
|
|
|
|
|
2019-01-01 08:23:01 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2022-02-24 19:01:14 +00:00
|
|
|
func controllerManager(ctx context.Context, cfg *config.Control) error {
|
|
|
|
runtime := cfg.Runtime
|
2019-04-05 00:43:00 +00:00
|
|
|
argsMap := map[string]string{
|
2021-09-16 21:28:23 +00:00
|
|
|
"feature-gates": "JobTrackingWithFinalizers=true",
|
2019-05-29 18:53:51 +00:00
|
|
|
"kubeconfig": runtime.KubeConfigController,
|
2021-09-13 22:23:00 +00:00
|
|
|
"authorization-kubeconfig": runtime.KubeConfigController,
|
|
|
|
"authentication-kubeconfig": runtime.KubeConfigController,
|
2019-04-05 00:43:00 +00:00
|
|
|
"service-account-private-key-file": runtime.ServiceKey,
|
|
|
|
"allocate-node-cidrs": "true",
|
2021-09-10 21:02:51 +00:00
|
|
|
"cluster-cidr": util.JoinIPNets(cfg.ClusterIPRanges),
|
2019-05-29 18:53:51 +00:00
|
|
|
"root-ca-file": runtime.ServerCA,
|
2020-07-10 20:08:04 +00:00
|
|
|
"profiling": "false",
|
2022-03-29 07:35:54 +00:00
|
|
|
"bind-address": getLocalhostIP(cfg.ServiceIPRanges).String(),
|
2021-07-02 23:42:09 +00:00
|
|
|
"secure-port": "10257",
|
2019-05-29 18:53:51 +00:00
|
|
|
"use-service-account-credentials": "true",
|
2021-05-01 12:58:06 +00:00
|
|
|
"cluster-signing-kube-apiserver-client-cert-file": runtime.ClientCA,
|
|
|
|
"cluster-signing-kube-apiserver-client-key-file": runtime.ClientCAKey,
|
|
|
|
"cluster-signing-kubelet-client-cert-file": runtime.ClientCA,
|
|
|
|
"cluster-signing-kubelet-client-key-file": runtime.ClientCAKey,
|
|
|
|
"cluster-signing-kubelet-serving-cert-file": runtime.ServerCA,
|
|
|
|
"cluster-signing-kubelet-serving-key-file": runtime.ServerCAKey,
|
2022-02-02 17:15:41 +00:00
|
|
|
"cluster-signing-legacy-unknown-cert-file": runtime.ServerCA,
|
|
|
|
"cluster-signing-legacy-unknown-key-file": runtime.ServerCAKey,
|
2019-01-01 08:23:01 +00:00
|
|
|
}
|
2019-01-22 21:14:58 +00:00
|
|
|
if cfg.NoLeaderElect {
|
2019-04-05 00:43:00 +00:00
|
|
|
argsMap["leader-elect"] = "false"
|
2019-01-22 21:14:58 +00:00
|
|
|
}
|
2020-11-03 17:27:56 +00:00
|
|
|
if !cfg.DisableCCM {
|
|
|
|
argsMap["configure-cloud-routes"] = "false"
|
2020-11-09 20:09:39 +00:00
|
|
|
argsMap["controllers"] = "*,-service,-route,-cloud-node-lifecycle"
|
2020-11-03 17:27:56 +00:00
|
|
|
}
|
2019-04-05 00:43:00 +00:00
|
|
|
|
2021-07-28 20:04:19 +00:00
|
|
|
args := config.GetArgs(argsMap, cfg.ExtraControllerArgs)
|
2020-04-27 17:09:58 +00:00
|
|
|
logrus.Infof("Running kube-controller-manager %s", config.ArgString(args))
|
2019-04-05 00:43:00 +00:00
|
|
|
|
2022-02-24 19:01:14 +00:00
|
|
|
return executor.ControllerManager(ctx, cfg.Runtime.APIServerReady, args)
|
2019-01-01 08:23:01 +00:00
|
|
|
}
|
|
|
|
|
2022-02-24 19:01:14 +00:00
|
|
|
func scheduler(ctx context.Context, cfg *config.Control) error {
|
|
|
|
runtime := cfg.Runtime
|
2019-04-05 00:43:00 +00:00
|
|
|
argsMap := map[string]string{
|
2021-09-13 22:23:00 +00:00
|
|
|
"kubeconfig": runtime.KubeConfigScheduler,
|
|
|
|
"authorization-kubeconfig": runtime.KubeConfigScheduler,
|
|
|
|
"authentication-kubeconfig": runtime.KubeConfigScheduler,
|
2022-03-29 07:35:54 +00:00
|
|
|
"bind-address": getLocalhostIP(cfg.ServiceIPRanges).String(),
|
2021-09-13 22:23:00 +00:00
|
|
|
"secure-port": "10259",
|
|
|
|
"profiling": "false",
|
2019-01-01 08:23:01 +00:00
|
|
|
}
|
2019-01-22 21:14:58 +00:00
|
|
|
if cfg.NoLeaderElect {
|
2019-04-05 00:43:00 +00:00
|
|
|
argsMap["leader-elect"] = "false"
|
2019-01-22 21:14:58 +00:00
|
|
|
}
|
2021-07-28 20:04:19 +00:00
|
|
|
args := config.GetArgs(argsMap, cfg.ExtraSchedulerAPIArgs)
|
2019-04-05 00:43:00 +00:00
|
|
|
|
2020-04-27 17:09:58 +00:00
|
|
|
logrus.Infof("Running kube-scheduler %s", config.ArgString(args))
|
2022-02-24 19:01:14 +00:00
|
|
|
return executor.Scheduler(ctx, cfg.Runtime.APIServerReady, args)
|
2019-01-01 08:23:01 +00:00
|
|
|
}
|
|
|
|
|
2022-02-24 19:01:14 +00:00
|
|
|
func apiServer(ctx context.Context, cfg *config.Control) error {
|
|
|
|
runtime := cfg.Runtime
|
2021-09-16 21:28:23 +00:00
|
|
|
argsMap := map[string]string{
|
|
|
|
"feature-gates": "JobTrackingWithFinalizers=true",
|
|
|
|
}
|
2019-06-11 22:48:47 +00:00
|
|
|
|
|
|
|
setupStorageBackend(argsMap, cfg)
|
2019-01-01 08:23:01 +00:00
|
|
|
|
2020-04-25 07:22:48 +00:00
|
|
|
certDir := filepath.Join(cfg.DataDir, "tls", "temporary-certs")
|
2019-02-08 04:14:20 +00:00
|
|
|
os.MkdirAll(certDir, 0700)
|
|
|
|
|
2019-04-05 00:43:00 +00:00
|
|
|
argsMap["cert-dir"] = certDir
|
|
|
|
argsMap["allow-privileged"] = "true"
|
|
|
|
argsMap["authorization-mode"] = strings.Join([]string{modes.ModeNode, modes.ModeRBAC}, ",")
|
|
|
|
argsMap["service-account-signing-key-file"] = runtime.ServiceKey
|
2021-09-10 21:02:51 +00:00
|
|
|
argsMap["service-cluster-ip-range"] = util.JoinIPNets(cfg.ServiceIPRanges)
|
2021-02-01 19:11:17 +00:00
|
|
|
argsMap["service-node-port-range"] = cfg.ServiceNodePortRange.String()
|
2019-04-05 00:43:00 +00:00
|
|
|
argsMap["advertise-port"] = strconv.Itoa(cfg.AdvertisePort)
|
2019-05-29 18:53:51 +00:00
|
|
|
if cfg.AdvertiseIP != "" {
|
|
|
|
argsMap["advertise-address"] = cfg.AdvertiseIP
|
|
|
|
}
|
2019-04-05 00:43:00 +00:00
|
|
|
argsMap["insecure-port"] = "0"
|
2020-04-28 22:00:30 +00:00
|
|
|
argsMap["secure-port"] = strconv.Itoa(cfg.APIServerPort)
|
|
|
|
if cfg.APIServerBindAddress == "" {
|
2022-03-29 07:35:54 +00:00
|
|
|
argsMap["bind-address"] = getLocalhostIP(cfg.ServiceIPRanges).String()
|
2020-04-28 22:00:30 +00:00
|
|
|
} else {
|
|
|
|
argsMap["bind-address"] = cfg.APIServerBindAddress
|
|
|
|
}
|
2019-05-29 18:53:51 +00:00
|
|
|
argsMap["tls-cert-file"] = runtime.ServingKubeAPICert
|
|
|
|
argsMap["tls-private-key-file"] = runtime.ServingKubeAPIKey
|
2019-04-05 00:43:00 +00:00
|
|
|
argsMap["service-account-key-file"] = runtime.ServiceKey
|
2021-04-28 05:45:33 +00:00
|
|
|
argsMap["service-account-issuer"] = "https://kubernetes.default.svc." + cfg.ClusterDomain
|
|
|
|
argsMap["api-audiences"] = "https://kubernetes.default.svc." + cfg.ClusterDomain + "," + version.Program
|
2019-09-02 17:03:17 +00:00
|
|
|
argsMap["kubelet-certificate-authority"] = runtime.ServerCA
|
2019-05-29 18:53:51 +00:00
|
|
|
argsMap["kubelet-client-certificate"] = runtime.ClientKubeAPICert
|
|
|
|
argsMap["kubelet-client-key"] = runtime.ClientKubeAPIKey
|
2019-04-10 18:09:38 +00:00
|
|
|
argsMap["requestheader-client-ca-file"] = runtime.RequestHeaderCA
|
2021-03-03 18:14:12 +00:00
|
|
|
argsMap["requestheader-allowed-names"] = deps.RequestHeaderCN
|
2019-04-10 18:09:38 +00:00
|
|
|
argsMap["proxy-client-cert-file"] = runtime.ClientAuthProxyCert
|
|
|
|
argsMap["proxy-client-key-file"] = runtime.ClientAuthProxyKey
|
|
|
|
argsMap["requestheader-extra-headers-prefix"] = "X-Remote-Extra-"
|
|
|
|
argsMap["requestheader-group-headers"] = "X-Remote-Group"
|
|
|
|
argsMap["requestheader-username-headers"] = "X-Remote-User"
|
2019-05-29 18:53:51 +00:00
|
|
|
argsMap["client-ca-file"] = runtime.ClientCA
|
2019-09-05 06:34:47 +00:00
|
|
|
argsMap["enable-admission-plugins"] = "NodeRestriction"
|
2019-08-27 04:36:56 +00:00
|
|
|
argsMap["anonymous-auth"] = "false"
|
2020-06-12 19:09:41 +00:00
|
|
|
argsMap["profiling"] = "false"
|
2019-12-12 22:41:10 +00:00
|
|
|
if cfg.EncryptSecrets {
|
|
|
|
argsMap["encryption-provider-config"] = runtime.EncryptionConfig
|
|
|
|
}
|
2021-07-28 20:04:19 +00:00
|
|
|
args := config.GetArgs(argsMap, cfg.ExtraAPIArgs)
|
2019-04-05 00:43:00 +00:00
|
|
|
|
2020-04-27 17:09:58 +00:00
|
|
|
logrus.Infof("Running kube-apiserver %s", config.ArgString(args))
|
2021-03-03 18:14:12 +00:00
|
|
|
|
2020-05-05 22:02:16 +00:00
|
|
|
return executor.APIServer(ctx, runtime.ETCDReady, args)
|
2019-01-01 08:23:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func defaults(config *config.Control) {
|
|
|
|
if config.ClusterIPRange == nil {
|
|
|
|
_, clusterIPNet, _ := net.ParseCIDR("10.42.0.0/16")
|
|
|
|
config.ClusterIPRange = clusterIPNet
|
|
|
|
}
|
|
|
|
|
|
|
|
if config.ServiceIPRange == nil {
|
|
|
|
_, serviceIPNet, _ := net.ParseCIDR("10.43.0.0/16")
|
|
|
|
config.ServiceIPRange = serviceIPNet
|
|
|
|
}
|
|
|
|
|
2019-01-09 16:54:15 +00:00
|
|
|
if len(config.ClusterDNS) == 0 {
|
|
|
|
config.ClusterDNS = net.ParseIP("10.43.0.10")
|
|
|
|
}
|
|
|
|
|
2019-01-01 08:23:01 +00:00
|
|
|
if config.AdvertisePort == 0 {
|
2019-05-29 18:53:51 +00:00
|
|
|
config.AdvertisePort = config.HTTPSPort
|
2019-01-01 08:23:01 +00:00
|
|
|
}
|
|
|
|
|
2020-04-28 22:00:30 +00:00
|
|
|
if config.APIServerPort == 0 {
|
2019-08-27 04:37:47 +00:00
|
|
|
if config.HTTPSPort != 0 {
|
2020-04-28 22:00:30 +00:00
|
|
|
config.APIServerPort = config.HTTPSPort + 1
|
2019-08-27 04:37:47 +00:00
|
|
|
} else {
|
2020-04-28 22:00:30 +00:00
|
|
|
config.APIServerPort = 6444
|
2019-08-27 04:37:47 +00:00
|
|
|
}
|
2019-01-01 08:23:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if config.DataDir == "" {
|
|
|
|
config.DataDir = "./management-state"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-02-24 19:01:14 +00:00
|
|
|
func prepare(ctx context.Context, config *config.Control) error {
|
2019-01-01 08:23:01 +00:00
|
|
|
var err error
|
|
|
|
|
|
|
|
defaults(config)
|
|
|
|
|
2019-07-26 21:54:44 +00:00
|
|
|
if err := os.MkdirAll(config.DataDir, 0700); err != nil {
|
2019-01-01 08:23:01 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
config.DataDir, err = filepath.Abs(config.DataDir)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2020-04-25 07:22:48 +00:00
|
|
|
os.MkdirAll(filepath.Join(config.DataDir, "tls"), 0700)
|
|
|
|
os.MkdirAll(filepath.Join(config.DataDir, "cred"), 0700)
|
|
|
|
|
2022-02-24 19:01:14 +00:00
|
|
|
deps.CreateRuntimeCertFiles(config)
|
2019-12-12 22:41:10 +00:00
|
|
|
|
2019-11-11 22:18:26 +00:00
|
|
|
cluster := cluster.New(config)
|
|
|
|
|
2021-11-10 12:33:42 +00:00
|
|
|
if err := cluster.Bootstrap(ctx, false); err != nil {
|
2019-06-27 19:00:43 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2022-02-24 19:01:14 +00:00
|
|
|
if err := deps.GenServerDeps(config); err != nil {
|
2019-10-27 05:53:25 +00:00
|
|
|
return err
|
2019-07-26 21:54:44 +00:00
|
|
|
}
|
|
|
|
|
2020-05-05 21:59:15 +00:00
|
|
|
ready, err := cluster.Start(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2022-02-24 19:01:14 +00:00
|
|
|
config.Runtime.ETCDReady = ready
|
2020-05-05 21:59:15 +00:00
|
|
|
return nil
|
2019-05-29 18:53:51 +00:00
|
|
|
}
|
2019-01-01 08:23:01 +00:00
|
|
|
|
2019-06-11 22:48:47 +00:00
|
|
|
func setupStorageBackend(argsMap map[string]string, cfg *config.Control) {
|
2019-07-26 21:54:44 +00:00
|
|
|
argsMap["storage-backend"] = "etcd3"
|
2019-06-11 22:48:47 +00:00
|
|
|
// specify the endpoints
|
2019-11-16 00:12:27 +00:00
|
|
|
if len(cfg.Datastore.Endpoint) > 0 {
|
|
|
|
argsMap["etcd-servers"] = cfg.Datastore.Endpoint
|
2019-06-11 22:48:47 +00:00
|
|
|
}
|
|
|
|
// storage backend tls configuration
|
2021-08-30 20:43:25 +00:00
|
|
|
if len(cfg.Datastore.BackendTLSConfig.CAFile) > 0 {
|
|
|
|
argsMap["etcd-cafile"] = cfg.Datastore.BackendTLSConfig.CAFile
|
2019-06-11 22:48:47 +00:00
|
|
|
}
|
2021-08-30 20:43:25 +00:00
|
|
|
if len(cfg.Datastore.BackendTLSConfig.CertFile) > 0 {
|
|
|
|
argsMap["etcd-certfile"] = cfg.Datastore.BackendTLSConfig.CertFile
|
2019-06-11 22:48:47 +00:00
|
|
|
}
|
2021-08-30 20:43:25 +00:00
|
|
|
if len(cfg.Datastore.BackendTLSConfig.KeyFile) > 0 {
|
|
|
|
argsMap["etcd-keyfile"] = cfg.Datastore.BackendTLSConfig.KeyFile
|
2019-06-11 22:48:47 +00:00
|
|
|
}
|
|
|
|
}
|
2019-09-26 22:38:04 +00:00
|
|
|
|
2022-02-24 19:01:14 +00:00
|
|
|
func cloudControllerManager(ctx context.Context, cfg *config.Control) error {
|
|
|
|
runtime := cfg.Runtime
|
2019-10-15 21:17:26 +00:00
|
|
|
argsMap := map[string]string{
|
2021-06-29 14:28:38 +00:00
|
|
|
"profiling": "false",
|
2021-09-10 21:02:51 +00:00
|
|
|
"allocate-node-cidrs": "true",
|
|
|
|
"cloud-provider": version.Program,
|
|
|
|
"cluster-cidr": util.JoinIPNets(cfg.ClusterIPRanges),
|
|
|
|
"configure-cloud-routes": "false",
|
|
|
|
"kubeconfig": runtime.KubeConfigCloudController,
|
2021-09-13 22:23:00 +00:00
|
|
|
"authorization-kubeconfig": runtime.KubeConfigCloudController,
|
|
|
|
"authentication-kubeconfig": runtime.KubeConfigCloudController,
|
2021-09-10 21:02:51 +00:00
|
|
|
"node-status-update-frequency": "1m0s",
|
2022-03-29 07:35:54 +00:00
|
|
|
"bind-address": getLocalhostIP(cfg.ServiceIPRanges).String(),
|
2021-09-10 21:02:51 +00:00
|
|
|
"port": "0",
|
2019-10-15 21:17:26 +00:00
|
|
|
}
|
|
|
|
if cfg.NoLeaderElect {
|
2021-06-29 14:28:38 +00:00
|
|
|
argsMap["leader-elect"] = "false"
|
2019-10-15 21:17:26 +00:00
|
|
|
}
|
2021-07-28 20:04:19 +00:00
|
|
|
args := config.GetArgs(argsMap, cfg.ExtraCloudControllerArgs)
|
2019-10-15 21:17:26 +00:00
|
|
|
|
2021-06-29 14:28:38 +00:00
|
|
|
logrus.Infof("Running cloud-controller-manager %s", config.ArgString(args))
|
2020-12-01 01:06:26 +00:00
|
|
|
|
2021-06-29 14:28:38 +00:00
|
|
|
ccmRBACReady := make(chan struct{})
|
2020-12-01 01:06:26 +00:00
|
|
|
|
2021-06-29 14:28:38 +00:00
|
|
|
go func() {
|
|
|
|
defer close(ccmRBACReady)
|
2019-10-15 21:17:26 +00:00
|
|
|
|
2021-06-29 14:28:38 +00:00
|
|
|
apiReadyLoop:
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case <-ctx.Done():
|
|
|
|
return
|
2022-02-24 19:01:14 +00:00
|
|
|
case <-cfg.Runtime.APIServerReady:
|
2021-06-29 14:28:38 +00:00
|
|
|
break apiReadyLoop
|
|
|
|
case <-time.After(30 * time.Second):
|
|
|
|
logrus.Infof("Waiting for API server to become available")
|
|
|
|
}
|
|
|
|
}
|
2019-10-15 21:17:26 +00:00
|
|
|
|
2021-06-29 14:28:38 +00:00
|
|
|
logrus.Infof("Waiting for cloud-controller-manager privileges to become available")
|
2019-10-15 21:17:26 +00:00
|
|
|
for {
|
2021-06-29 14:28:38 +00:00
|
|
|
select {
|
|
|
|
case <-ctx.Done():
|
|
|
|
return
|
2022-02-24 19:01:14 +00:00
|
|
|
case err := <-promise(func() error { return checkForCloudControllerPrivileges(ctx, cfg.Runtime, 5*time.Second) }):
|
2021-06-29 14:28:38 +00:00
|
|
|
if err != nil {
|
2021-09-13 23:20:53 +00:00
|
|
|
logrus.Infof("Waiting for cloud-controller-manager privileges to become available: %v", err)
|
2019-11-05 11:44:35 +00:00
|
|
|
continue
|
|
|
|
}
|
2021-06-29 14:28:38 +00:00
|
|
|
return
|
2019-10-15 21:17:26 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}()
|
2021-06-29 14:28:38 +00:00
|
|
|
|
2021-09-13 22:20:03 +00:00
|
|
|
return executor.CloudControllerManager(ctx, ccmRBACReady, args)
|
2019-10-15 21:17:26 +00:00
|
|
|
}
|
|
|
|
|
2021-09-13 22:20:03 +00:00
|
|
|
// checkForCloudControllerPrivileges makes a SubjectAccessReview request to the apiserver
|
|
|
|
// to validate that the embedded cloud controller manager has the required privileges,
|
|
|
|
// and does not return until the requested access is granted.
|
|
|
|
// If the CCM RBAC changes, the ResourceAttributes checked for by this function should
|
|
|
|
// be modified to check for the most recently added privilege.
|
|
|
|
func checkForCloudControllerPrivileges(ctx context.Context, runtime *config.ControlRuntime, timeout time.Duration) error {
|
2019-10-15 21:17:26 +00:00
|
|
|
restConfig, err := clientcmd.BuildConfigFromFlags("", runtime.KubeConfigAdmin)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2021-10-29 18:39:27 +00:00
|
|
|
authClient, err := authorizationv1client.NewForConfig(restConfig)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2021-09-13 22:24:00 +00:00
|
|
|
sar := &authorizationv1.SubjectAccessReview{
|
|
|
|
Spec: authorizationv1.SubjectAccessReviewSpec{
|
|
|
|
User: version.Program + "-cloud-controller-manager",
|
|
|
|
ResourceAttributes: &authorizationv1.ResourceAttributes{
|
|
|
|
Namespace: metav1.NamespaceSystem,
|
|
|
|
Verb: "get",
|
|
|
|
Resource: "configmaps",
|
|
|
|
Name: "extension-apiserver-authentication",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
2021-06-29 14:28:38 +00:00
|
|
|
err = wait.PollImmediate(time.Second, timeout, func() (bool, error) {
|
2021-09-13 22:24:00 +00:00
|
|
|
r, err := authClient.SubjectAccessReviews().Create(ctx, sar, metav1.CreateOptions{})
|
2021-06-29 14:28:38 +00:00
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
2021-09-13 22:24:00 +00:00
|
|
|
if r.Status.Allowed {
|
|
|
|
return true, nil
|
|
|
|
}
|
|
|
|
return false, nil
|
2021-06-29 14:28:38 +00:00
|
|
|
})
|
2021-09-13 23:20:53 +00:00
|
|
|
return err
|
2019-10-15 21:17:26 +00:00
|
|
|
}
|
2019-11-04 11:08:47 +00:00
|
|
|
|
2021-10-12 06:13:10 +00:00
|
|
|
func waitForAPIServerHandlers(ctx context.Context, runtime *config.ControlRuntime) {
|
|
|
|
auth, handler, err := executor.APIServerHandlers(ctx)
|
|
|
|
if err != nil {
|
|
|
|
logrus.Fatalf("Failed to get request handlers from apiserver: %v", err)
|
|
|
|
}
|
2022-03-28 20:45:39 +00:00
|
|
|
runtime.Authenticator = authenticator.Combine(runtime.Authenticator, auth)
|
2021-10-12 06:13:10 +00:00
|
|
|
runtime.APIServer = handler
|
|
|
|
}
|
|
|
|
|
2020-04-28 22:44:05 +00:00
|
|
|
func waitForAPIServerInBackground(ctx context.Context, runtime *config.ControlRuntime) error {
|
2019-11-04 11:08:47 +00:00
|
|
|
restConfig, err := clientcmd.BuildConfigFromFlags("", runtime.KubeConfigAdmin)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2022-03-23 08:37:11 +00:00
|
|
|
// By default, idle connections to the apiserver are returned to a global pool
|
|
|
|
// between requests. Explicitly flag this client's request for closure so that
|
|
|
|
// we re-dial through the loadbalancer in case the endpoints have changed.
|
|
|
|
restConfig.Wrap(func(rt http.RoundTripper) http.RoundTripper {
|
|
|
|
return roundTripFunc(func(req *http.Request) (*http.Response, error) {
|
|
|
|
req.Close = true
|
|
|
|
return rt.RoundTrip(req)
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2019-11-09 06:07:12 +00:00
|
|
|
k8sClient, err := kubernetes.NewForConfig(restConfig)
|
2019-11-04 11:08:47 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2020-04-28 22:44:05 +00:00
|
|
|
done := make(chan struct{})
|
|
|
|
runtime.APIServerReady = done
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
defer close(done)
|
2020-05-05 22:02:16 +00:00
|
|
|
|
|
|
|
etcdLoop:
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case <-ctx.Done():
|
|
|
|
return
|
|
|
|
case <-runtime.ETCDReady:
|
|
|
|
break etcdLoop
|
|
|
|
case <-time.After(30 * time.Second):
|
|
|
|
logrus.Infof("Waiting for etcd server to become available")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-28 22:44:05 +00:00
|
|
|
logrus.Infof("Waiting for API server to become available")
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case <-ctx.Done():
|
|
|
|
return
|
2021-11-11 21:01:49 +00:00
|
|
|
case err := <-promise(func() error { return util.WaitForAPIServerReady(ctx, k8sClient, 30*time.Second) }):
|
2020-04-28 22:44:05 +00:00
|
|
|
if err != nil {
|
|
|
|
logrus.Infof("Waiting for API server to become available")
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
return nil
|
2019-11-09 06:07:12 +00:00
|
|
|
}
|
2019-11-04 11:08:47 +00:00
|
|
|
|
2019-11-09 06:07:12 +00:00
|
|
|
func promise(f func() error) <-chan error {
|
|
|
|
c := make(chan error, 1)
|
|
|
|
go func() {
|
|
|
|
c <- f()
|
|
|
|
close(c)
|
|
|
|
}()
|
|
|
|
return c
|
2019-11-04 11:08:47 +00:00
|
|
|
}
|