172 lines
4.9 KiB
Go
172 lines
4.9 KiB
Go
/*
|
|
Copyright 2016 The Kubernetes Authors.
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
you may not use this file except in compliance with the License.
|
|
You may obtain a copy of the License at
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
See the License for the specific language governing permissions and
|
|
limitations under the License.
|
|
*/
|
|
|
|
package cmd
|
|
|
|
import (
|
|
"fmt"
|
|
"io"
|
|
"io/ioutil"
|
|
"path"
|
|
|
|
"github.com/renstrom/dedent"
|
|
"github.com/spf13/cobra"
|
|
|
|
kubeadmapi "k8s.io/kubernetes/cmd/kubeadm/app/apis/kubeadm"
|
|
kubeadmapiext "k8s.io/kubernetes/cmd/kubeadm/app/apis/kubeadm/v1alpha1"
|
|
"k8s.io/kubernetes/cmd/kubeadm/app/apis/kubeadm/validation"
|
|
"k8s.io/kubernetes/cmd/kubeadm/app/discovery"
|
|
kubenode "k8s.io/kubernetes/cmd/kubeadm/app/node"
|
|
kubeconfigphase "k8s.io/kubernetes/cmd/kubeadm/app/phases/kubeconfig"
|
|
"k8s.io/kubernetes/cmd/kubeadm/app/preflight"
|
|
kubeadmutil "k8s.io/kubernetes/cmd/kubeadm/app/util"
|
|
"k8s.io/kubernetes/pkg/api"
|
|
clientcmdapi "k8s.io/kubernetes/pkg/client/unversioned/clientcmd/api"
|
|
"k8s.io/kubernetes/pkg/runtime"
|
|
)
|
|
|
|
var (
|
|
joinDoneMsgf = dedent.Dedent(`
|
|
Node join complete:
|
|
* Certificate signing request sent to master and response
|
|
received.
|
|
* Kubelet informed of new secure connection details.
|
|
|
|
Run 'kubectl get nodes' on the master to see this machine join.
|
|
`)
|
|
)
|
|
|
|
// NewCmdJoin returns "kubeadm join" command.
|
|
func NewCmdJoin(out io.Writer) *cobra.Command {
|
|
versioned := &kubeadmapiext.NodeConfiguration{}
|
|
api.Scheme.Default(versioned)
|
|
cfg := kubeadmapi.NodeConfiguration{}
|
|
api.Scheme.Convert(versioned, &cfg, nil)
|
|
|
|
var skipPreFlight bool
|
|
var cfgPath string
|
|
|
|
cmd := &cobra.Command{
|
|
Use: "join <master address>",
|
|
Short: "Run this on any machine you wish to join an existing cluster",
|
|
Run: func(cmd *cobra.Command, args []string) {
|
|
j, err := NewJoin(cfgPath, args, &cfg, skipPreFlight)
|
|
kubeadmutil.CheckErr(err)
|
|
kubeadmutil.CheckErr(j.Validate())
|
|
kubeadmutil.CheckErr(j.Run(out))
|
|
},
|
|
}
|
|
|
|
cmd.PersistentFlags().StringVar(&cfgPath, "config", cfgPath, "Path to kubeadm config file")
|
|
|
|
cmd.PersistentFlags().BoolVar(
|
|
&skipPreFlight, "skip-preflight-checks", false,
|
|
"skip preflight checks normally run before modifying the system",
|
|
)
|
|
|
|
cmd.PersistentFlags().Var(
|
|
discovery.NewDiscoveryValue(&cfg.Discovery), "discovery",
|
|
"The discovery method kubeadm will use for connecting nodes to the master",
|
|
)
|
|
|
|
return cmd
|
|
}
|
|
|
|
type Join struct {
|
|
cfg *kubeadmapi.NodeConfiguration
|
|
}
|
|
|
|
func NewJoin(cfgPath string, args []string, cfg *kubeadmapi.NodeConfiguration, skipPreFlight bool) (*Join, error) {
|
|
|
|
fmt.Println("[kubeadm] WARNING: kubeadm is in alpha, please do not use it for production clusters.")
|
|
|
|
if cfgPath != "" {
|
|
b, err := ioutil.ReadFile(cfgPath)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("unable to read config from %q [%v]", cfgPath, err)
|
|
}
|
|
if err := runtime.DecodeInto(api.Codecs.UniversalDecoder(), b, cfg); err != nil {
|
|
return nil, fmt.Errorf("unable to decode config from %q [%v]", cfgPath, err)
|
|
}
|
|
}
|
|
|
|
if !skipPreFlight {
|
|
fmt.Println("[preflight] Running pre-flight checks")
|
|
|
|
// First, check if we're root separately from the other preflight checks and fail fast
|
|
if err := preflight.RunRootCheckOnly(); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// Then continue with the others...
|
|
if err := preflight.RunJoinNodeChecks(cfg); err != nil {
|
|
return nil, err
|
|
}
|
|
} else {
|
|
fmt.Println("[preflight] Skipping pre-flight checks")
|
|
}
|
|
|
|
// Try to start the kubelet service in case it's inactive
|
|
preflight.TryStartKubelet()
|
|
|
|
return &Join{cfg: cfg}, nil
|
|
}
|
|
|
|
func (j *Join) Validate() error {
|
|
return validation.ValidateNodeConfiguration(j.cfg).ToAggregate()
|
|
}
|
|
|
|
// Run executes worked node provisioning and tries to join an existing cluster.
|
|
func (j *Join) Run(out io.Writer) error {
|
|
clusterInfo, err := kubenode.RetrieveTrustedClusterInfo(j.cfg.Discovery.Token)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
var cfg *clientcmdapi.Config
|
|
// TODO: delete this first block when we move Token to the discovery interface
|
|
if j.cfg.Discovery.Token != nil {
|
|
connectionDetails, err := kubenode.EstablishMasterConnection(j.cfg.Discovery.Token, clusterInfo)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
err = kubenode.CheckForNodeNameDuplicates(connectionDetails)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
cfg, err = kubenode.PerformTLSBootstrapDeprecated(connectionDetails)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
} else {
|
|
cfg, err = discovery.For(j.cfg.Discovery)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if err := kubenode.PerformTLSBootstrap(cfg); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
err = kubeconfigphase.WriteKubeconfigToDisk(path.Join(kubeadmapi.GlobalEnvParams.KubernetesDir, kubeconfigphase.KubeletKubeConfigFileName), cfg)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
fmt.Fprintf(out, joinDoneMsgf)
|
|
return nil
|
|
}
|