mirror of
				https://github.com/k3s-io/kubernetes.git
				synced 2025-10-30 21:30:16 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			159 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			159 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| /*
 | |
| 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 cmd
 | |
| 
 | |
| import (
 | |
| 	"errors"
 | |
| 	"fmt"
 | |
| 	"io"
 | |
| 	"net"
 | |
| 	"os"
 | |
| 	"strings"
 | |
| 
 | |
| 	"github.com/golang/glog"
 | |
| 	"github.com/spf13/cobra"
 | |
| 	"k8s.io/kubernetes/pkg/kubectl"
 | |
| 	"k8s.io/kubernetes/pkg/kubectl/cmd/templates"
 | |
| 	cmdutil "k8s.io/kubernetes/pkg/kubectl/cmd/util"
 | |
| 	"k8s.io/kubernetes/pkg/util/i18n"
 | |
| )
 | |
| 
 | |
| var (
 | |
| 	defaultPort = 8001
 | |
| 	proxyLong   = templates.LongDesc(i18n.T(`
 | |
| 		Creates a proxy server or application-level gateway between localhost and 
 | |
| 		the Kubernetes API Server. It also allows serving static content over specified 
 | |
| 		HTTP path. All incoming data enters through one port and gets forwarded to 
 | |
| 		the remote kubernetes API Server port, except for the path matching the static content path.`))
 | |
| 
 | |
| 	proxyExample = templates.Examples(i18n.T(`
 | |
| 		# To proxy all of the kubernetes api and nothing else, use:
 | |
| 
 | |
| 		    $ kubectl proxy --api-prefix=/
 | |
| 
 | |
| 		# To proxy only part of the kubernetes api and also some static files:
 | |
| 
 | |
| 		    $ kubectl proxy --www=/my/files --www-prefix=/static/ --api-prefix=/api/
 | |
| 
 | |
| 		# The above lets you 'curl localhost:8001/api/v1/pods'.
 | |
| 
 | |
| 		# To proxy the entire kubernetes api at a different root, use:
 | |
| 
 | |
| 		    $ kubectl proxy --api-prefix=/custom/
 | |
| 
 | |
| 		# The above lets you 'curl localhost:8001/custom/api/v1/pods'
 | |
| 
 | |
| 		# Run a proxy to kubernetes apiserver on port 8011, serving static content from ./local/www/
 | |
| 		kubectl proxy --port=8011 --www=./local/www/
 | |
| 
 | |
| 		# Run a proxy to kubernetes apiserver on an arbitrary local port.
 | |
| 		# The chosen port for the server will be output to stdout.
 | |
| 		kubectl proxy --port=0
 | |
| 
 | |
| 		# Run a proxy to kubernetes apiserver, changing the api prefix to k8s-api
 | |
| 		# This makes e.g. the pods api available at localhost:8001/k8s-api/v1/pods/
 | |
| 		kubectl proxy --api-prefix=/k8s-api`))
 | |
| )
 | |
| 
 | |
| func NewCmdProxy(f cmdutil.Factory, out io.Writer) *cobra.Command {
 | |
| 	cmd := &cobra.Command{
 | |
| 		Use:     "proxy [--port=PORT] [--www=static-dir] [--www-prefix=prefix] [--api-prefix=prefix]",
 | |
| 		Short:   i18n.T("Run a proxy to the Kubernetes API server"),
 | |
| 		Long:    proxyLong,
 | |
| 		Example: proxyExample,
 | |
| 		Run: func(cmd *cobra.Command, args []string) {
 | |
| 			err := RunProxy(f, out, cmd)
 | |
| 			cmdutil.CheckErr(err)
 | |
| 		},
 | |
| 	}
 | |
| 	cmd.Flags().StringP("www", "w", "", "Also serve static files from the given directory under the specified prefix.")
 | |
| 	cmd.Flags().StringP("www-prefix", "P", "/static/", "Prefix to serve static files under, if static file directory is specified.")
 | |
| 	cmd.Flags().StringP("api-prefix", "", "/", "Prefix to serve the proxied API under.")
 | |
| 	cmd.Flags().String("accept-paths", kubectl.DefaultPathAcceptRE, "Regular expression for paths that the proxy should accept.")
 | |
| 	cmd.Flags().String("reject-paths", kubectl.DefaultPathRejectRE, "Regular expression for paths that the proxy should reject. Paths specified here will be rejected even accepted by --accept-paths.")
 | |
| 	cmd.Flags().String("accept-hosts", kubectl.DefaultHostAcceptRE, "Regular expression for hosts that the proxy should accept.")
 | |
| 	cmd.Flags().String("reject-methods", kubectl.DefaultMethodRejectRE, "Regular expression for HTTP methods that the proxy should reject (example --reject-methods='POST,PUT,PATCH'). ")
 | |
| 	cmd.Flags().IntP("port", "p", defaultPort, "The port on which to run the proxy. Set to 0 to pick a random port.")
 | |
| 	cmd.Flags().StringP("address", "", "127.0.0.1", "The IP address on which to serve on.")
 | |
| 	cmd.Flags().Bool("disable-filter", false, "If true, disable request filtering in the proxy. This is dangerous, and can leave you vulnerable to XSRF attacks, when used with an accessible port.")
 | |
| 	cmd.Flags().StringP("unix-socket", "u", "", "Unix socket on which to run the proxy.")
 | |
| 	return cmd
 | |
| }
 | |
| 
 | |
| func RunProxy(f cmdutil.Factory, out io.Writer, cmd *cobra.Command) error {
 | |
| 	path := cmdutil.GetFlagString(cmd, "unix-socket")
 | |
| 	port := cmdutil.GetFlagInt(cmd, "port")
 | |
| 	address := cmdutil.GetFlagString(cmd, "address")
 | |
| 
 | |
| 	if port != defaultPort && path != "" {
 | |
| 		return errors.New("Don't specify both --unix-socket and --port")
 | |
| 	}
 | |
| 
 | |
| 	clientConfig, err := f.ClientConfig()
 | |
| 	if err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 
 | |
| 	staticPrefix := cmdutil.GetFlagString(cmd, "www-prefix")
 | |
| 	if !strings.HasSuffix(staticPrefix, "/") {
 | |
| 		staticPrefix += "/"
 | |
| 	}
 | |
| 	staticDir := cmdutil.GetFlagString(cmd, "www")
 | |
| 	if staticDir != "" {
 | |
| 		fileInfo, err := os.Stat(staticDir)
 | |
| 		if err != nil {
 | |
| 			glog.Warning("Failed to stat static file directory "+staticDir+": ", err)
 | |
| 		} else if !fileInfo.IsDir() {
 | |
| 			glog.Warning("Static file directory " + staticDir + " is not a directory")
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	apiProxyPrefix := cmdutil.GetFlagString(cmd, "api-prefix")
 | |
| 	if !strings.HasSuffix(apiProxyPrefix, "/") {
 | |
| 		apiProxyPrefix += "/"
 | |
| 	}
 | |
| 	filter := &kubectl.FilterServer{
 | |
| 		AcceptPaths:   kubectl.MakeRegexpArrayOrDie(cmdutil.GetFlagString(cmd, "accept-paths")),
 | |
| 		RejectPaths:   kubectl.MakeRegexpArrayOrDie(cmdutil.GetFlagString(cmd, "reject-paths")),
 | |
| 		AcceptHosts:   kubectl.MakeRegexpArrayOrDie(cmdutil.GetFlagString(cmd, "accept-hosts")),
 | |
| 		RejectMethods: kubectl.MakeRegexpArrayOrDie(cmdutil.GetFlagString(cmd, "reject-methods")),
 | |
| 	}
 | |
| 	if cmdutil.GetFlagBool(cmd, "disable-filter") {
 | |
| 		if path == "" {
 | |
| 			glog.Warning("Request filter disabled, your proxy is vulnerable to XSRF attacks, please be cautious")
 | |
| 		}
 | |
| 		filter = nil
 | |
| 	}
 | |
| 
 | |
| 	server, err := kubectl.NewProxyServer(staticDir, apiProxyPrefix, staticPrefix, filter, clientConfig)
 | |
| 
 | |
| 	// Separate listening from serving so we can report the bound port
 | |
| 	// when it is chosen by os (eg: port == 0)
 | |
| 	var l net.Listener
 | |
| 	if path == "" {
 | |
| 		l, err = server.Listen(address, port)
 | |
| 	} else {
 | |
| 		l, err = server.ListenUnix(path)
 | |
| 	}
 | |
| 	if err != nil {
 | |
| 		glog.Fatal(err)
 | |
| 	}
 | |
| 	fmt.Fprintf(out, "Starting to serve on %s", l.Addr().String())
 | |
| 	glog.Fatal(server.ServeOnListener(l))
 | |
| 	return nil
 | |
| }
 |