123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342 |
- /*
- Copyright 2014 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 portforward
- import (
- "fmt"
- "net/http"
- "net/url"
- "os"
- "os/signal"
- "strconv"
- "strings"
- "time"
- "github.com/spf13/cobra"
- corev1 "k8s.io/api/core/v1"
- metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
- "k8s.io/cli-runtime/pkg/genericclioptions"
- "k8s.io/client-go/kubernetes/scheme"
- corev1client "k8s.io/client-go/kubernetes/typed/core/v1"
- restclient "k8s.io/client-go/rest"
- "k8s.io/client-go/tools/portforward"
- "k8s.io/client-go/transport/spdy"
- cmdutil "k8s.io/kubernetes/pkg/kubectl/cmd/util"
- "k8s.io/kubernetes/pkg/kubectl/polymorphichelpers"
- "k8s.io/kubernetes/pkg/kubectl/util"
- "k8s.io/kubernetes/pkg/kubectl/util/i18n"
- "k8s.io/kubernetes/pkg/kubectl/util/templates"
- )
- // PortForwardOptions contains all the options for running the port-forward cli command.
- type PortForwardOptions struct {
- Namespace string
- PodName string
- RESTClient *restclient.RESTClient
- Config *restclient.Config
- PodClient corev1client.PodsGetter
- Address []string
- Ports []string
- PortForwarder portForwarder
- StopChannel chan struct{}
- ReadyChannel chan struct{}
- }
- var (
- portforwardLong = templates.LongDesc(i18n.T(`
- Forward one or more local ports to a pod. This command requires the node to have 'socat' installed.
- Use resource type/name such as deployment/mydeployment to select a pod. Resource type defaults to 'pod' if omitted.
- If there are multiple pods matching the criteria, a pod will be selected automatically. The
- forwarding session ends when the selected pod terminates, and rerun of the command is needed
- to resume forwarding.`))
- portforwardExample = templates.Examples(i18n.T(`
- # Listen on ports 5000 and 6000 locally, forwarding data to/from ports 5000 and 6000 in the pod
- kubectl port-forward pod/mypod 5000 6000
- # Listen on ports 5000 and 6000 locally, forwarding data to/from ports 5000 and 6000 in a pod selected by the deployment
- kubectl port-forward deployment/mydeployment 5000 6000
- # Listen on ports 5000 and 6000 locally, forwarding data to/from ports 5000 and 6000 in a pod selected by the service
- kubectl port-forward service/myservice 5000 6000
- # Listen on port 8888 locally, forwarding to 5000 in the pod
- kubectl port-forward pod/mypod 8888:5000
- # Listen on port 8888 on all addresses, forwarding to 5000 in the pod
- kubectl port-forward --address 0.0.0.0 pod/mypod 8888:5000
- # Listen on port 8888 on localhost and selected IP, forwarding to 5000 in the pod
- kubectl port-forward --address localhost,10.19.21.23 pod/mypod 8888:5000
- # Listen on a random port locally, forwarding to 5000 in the pod
- kubectl port-forward pod/mypod :5000`))
- )
- const (
- // Amount of time to wait until at least one pod is running
- defaultPodPortForwardWaitTimeout = 60 * time.Second
- )
- func NewCmdPortForward(f cmdutil.Factory, streams genericclioptions.IOStreams) *cobra.Command {
- opts := &PortForwardOptions{
- PortForwarder: &defaultPortForwarder{
- IOStreams: streams,
- },
- }
- cmd := &cobra.Command{
- Use: "port-forward TYPE/NAME [options] [LOCAL_PORT:]REMOTE_PORT [...[LOCAL_PORT_N:]REMOTE_PORT_N]",
- DisableFlagsInUseLine: true,
- Short: i18n.T("Forward one or more local ports to a pod"),
- Long: portforwardLong,
- Example: portforwardExample,
- Run: func(cmd *cobra.Command, args []string) {
- if err := opts.Complete(f, cmd, args); err != nil {
- cmdutil.CheckErr(err)
- }
- if err := opts.Validate(); err != nil {
- cmdutil.CheckErr(cmdutil.UsageErrorf(cmd, "%v", err.Error()))
- }
- if err := opts.RunPortForward(); err != nil {
- cmdutil.CheckErr(err)
- }
- },
- }
- cmdutil.AddPodRunningTimeoutFlag(cmd, defaultPodPortForwardWaitTimeout)
- cmd.Flags().StringSliceVar(&opts.Address, "address", []string{"localhost"}, "Addresses to listen on (comma separated). Only accepts IP addresses or localhost as a value. When localhost is supplied, kubectl will try to bind on both 127.0.0.1 and ::1 and will fail if neither of these addresses are available to bind.")
- // TODO support UID
- return cmd
- }
- type portForwarder interface {
- ForwardPorts(method string, url *url.URL, opts PortForwardOptions) error
- }
- type defaultPortForwarder struct {
- genericclioptions.IOStreams
- }
- func (f *defaultPortForwarder) ForwardPorts(method string, url *url.URL, opts PortForwardOptions) error {
- transport, upgrader, err := spdy.RoundTripperFor(opts.Config)
- if err != nil {
- return err
- }
- dialer := spdy.NewDialer(upgrader, &http.Client{Transport: transport}, method, url)
- fw, err := portforward.NewOnAddresses(dialer, opts.Address, opts.Ports, opts.StopChannel, opts.ReadyChannel, f.Out, f.ErrOut)
- if err != nil {
- return err
- }
- return fw.ForwardPorts()
- }
- // splitPort splits port string which is in form of [LOCAL PORT]:REMOTE PORT
- // and returns local and remote ports separately
- func splitPort(port string) (local, remote string) {
- parts := strings.Split(port, ":")
- if len(parts) == 2 {
- return parts[0], parts[1]
- }
- return parts[0], parts[0]
- }
- // Translates service port to target port
- // It rewrites ports as needed if the Service port declares targetPort.
- // It returns an error when a named targetPort can't find a match in the pod, or the Service did not declare
- // the port.
- func translateServicePortToTargetPort(ports []string, svc corev1.Service, pod corev1.Pod) ([]string, error) {
- var translated []string
- for _, port := range ports {
- localPort, remotePort := splitPort(port)
- portnum, err := strconv.Atoi(remotePort)
- if err != nil {
- svcPort, err := util.LookupServicePortNumberByName(svc, remotePort)
- if err != nil {
- return nil, err
- }
- portnum = int(svcPort)
- if localPort == remotePort {
- localPort = strconv.Itoa(portnum)
- }
- }
- containerPort, err := util.LookupContainerPortNumberByServicePort(svc, pod, int32(portnum))
- if err != nil {
- // can't resolve a named port, or Service did not declare this port, return an error
- return nil, err
- }
- if int32(portnum) != containerPort {
- translated = append(translated, fmt.Sprintf("%s:%d", localPort, containerPort))
- } else {
- translated = append(translated, port)
- }
- }
- return translated, nil
- }
- // convertPodNamedPortToNumber converts named ports into port numbers
- // It returns an error when a named port can't be found in the pod containers
- func convertPodNamedPortToNumber(ports []string, pod corev1.Pod) ([]string, error) {
- var converted []string
- for _, port := range ports {
- localPort, remotePort := splitPort(port)
- containerPortStr := remotePort
- _, err := strconv.Atoi(remotePort)
- if err != nil {
- containerPort, err := util.LookupContainerPortNumberByName(pod, remotePort)
- if err != nil {
- return nil, err
- }
- containerPortStr = strconv.Itoa(int(containerPort))
- }
- if localPort != remotePort {
- converted = append(converted, fmt.Sprintf("%s:%s", localPort, containerPortStr))
- } else {
- converted = append(converted, containerPortStr)
- }
- }
- return converted, nil
- }
- // Complete completes all the required options for port-forward cmd.
- func (o *PortForwardOptions) Complete(f cmdutil.Factory, cmd *cobra.Command, args []string) error {
- var err error
- if len(args) < 2 {
- return cmdutil.UsageErrorf(cmd, "TYPE/NAME and list of ports are required for port-forward")
- }
- o.Namespace, _, err = f.ToRawKubeConfigLoader().Namespace()
- if err != nil {
- return err
- }
- builder := f.NewBuilder().
- WithScheme(scheme.Scheme, scheme.Scheme.PrioritizedVersionsAllGroups()...).
- ContinueOnError().
- NamespaceParam(o.Namespace).DefaultNamespace()
- getPodTimeout, err := cmdutil.GetPodRunningTimeoutFlag(cmd)
- if err != nil {
- return cmdutil.UsageErrorf(cmd, err.Error())
- }
- resourceName := args[0]
- builder.ResourceNames("pods", resourceName)
- obj, err := builder.Do().Object()
- if err != nil {
- return err
- }
- forwardablePod, err := polymorphichelpers.AttachablePodForObjectFn(f, obj, getPodTimeout)
- if err != nil {
- return err
- }
- o.PodName = forwardablePod.Name
- // handle service port mapping to target port if needed
- switch t := obj.(type) {
- case *corev1.Service:
- o.Ports, err = translateServicePortToTargetPort(args[1:], *t, *forwardablePod)
- if err != nil {
- return err
- }
- default:
- o.Ports, err = convertPodNamedPortToNumber(args[1:], *forwardablePod)
- if err != nil {
- return err
- }
- }
- clientset, err := f.KubernetesClientSet()
- if err != nil {
- return err
- }
- o.PodClient = clientset.CoreV1()
- o.Config, err = f.ToRESTConfig()
- if err != nil {
- return err
- }
- o.RESTClient, err = f.RESTClient()
- if err != nil {
- return err
- }
- o.StopChannel = make(chan struct{}, 1)
- o.ReadyChannel = make(chan struct{})
- return nil
- }
- // Validate validates all the required options for port-forward cmd.
- func (o PortForwardOptions) Validate() error {
- if len(o.PodName) == 0 {
- return fmt.Errorf("pod name or resource type/name must be specified")
- }
- if len(o.Ports) < 1 {
- return fmt.Errorf("at least 1 PORT is required for port-forward")
- }
- if o.PortForwarder == nil || o.PodClient == nil || o.RESTClient == nil || o.Config == nil {
- return fmt.Errorf("client, client config, restClient, and portforwarder must be provided")
- }
- return nil
- }
- // RunPortForward implements all the necessary functionality for port-forward cmd.
- func (o PortForwardOptions) RunPortForward() error {
- pod, err := o.PodClient.Pods(o.Namespace).Get(o.PodName, metav1.GetOptions{})
- if err != nil {
- return err
- }
- if pod.Status.Phase != corev1.PodRunning {
- return fmt.Errorf("unable to forward port because pod is not running. Current status=%v", pod.Status.Phase)
- }
- signals := make(chan os.Signal, 1)
- signal.Notify(signals, os.Interrupt)
- defer signal.Stop(signals)
- go func() {
- <-signals
- if o.StopChannel != nil {
- close(o.StopChannel)
- }
- }()
- req := o.RESTClient.Post().
- Resource("pods").
- Namespace(o.Namespace).
- Name(pod.Name).
- SubResource("portforward")
- return o.PortForwarder.ForwardPorts("POST", req.URL(), o)
- }
|