mirror of
				https://github.com/caddyserver/caddy.git
				synced 2025-11-04 03:27:23 -05:00 
			
		
		
		
	
		
			
				
	
	
		
			229 lines
		
	
	
		
			6.6 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			229 lines
		
	
	
		
			6.6 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
// Copyright 2015 Matthew Holt and The Caddy 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 reverseproxy
 | 
						|
 | 
						|
import (
 | 
						|
	"encoding/json"
 | 
						|
	"flag"
 | 
						|
	"fmt"
 | 
						|
	"net/http"
 | 
						|
	"strconv"
 | 
						|
 | 
						|
	"github.com/caddyserver/caddy/v2"
 | 
						|
	"github.com/caddyserver/caddy/v2/caddyconfig"
 | 
						|
	"github.com/caddyserver/caddy/v2/caddyconfig/httpcaddyfile"
 | 
						|
	caddycmd "github.com/caddyserver/caddy/v2/cmd"
 | 
						|
	"github.com/caddyserver/caddy/v2/modules/caddyhttp"
 | 
						|
	"github.com/caddyserver/caddy/v2/modules/caddyhttp/headers"
 | 
						|
	"github.com/caddyserver/caddy/v2/modules/caddytls"
 | 
						|
	"go.uber.org/zap"
 | 
						|
)
 | 
						|
 | 
						|
func init() {
 | 
						|
	caddycmd.RegisterCommand(caddycmd.Command{
 | 
						|
		Name:  "reverse-proxy",
 | 
						|
		Func:  cmdReverseProxy,
 | 
						|
		Usage: "[--from <addr>] [--to <addr>] [--change-host-header]",
 | 
						|
		Short: "A quick and production-ready reverse proxy",
 | 
						|
		Long: `
 | 
						|
A simple but production-ready reverse proxy. Useful for quick deployments,
 | 
						|
demos, and development.
 | 
						|
 | 
						|
Simply shuttles HTTP(S) traffic from the --from address to the --to address.
 | 
						|
Multiple --to addresses may be specified by repeating the flag.
 | 
						|
 | 
						|
Unless otherwise specified in the addresses, the --from address will be
 | 
						|
assumed to be HTTPS if a hostname is given, and the --to address will be
 | 
						|
assumed to be HTTP.
 | 
						|
 | 
						|
If the --from address has a host or IP, Caddy will attempt to serve the
 | 
						|
proxy over HTTPS with a certificate (unless overridden by the HTTP scheme
 | 
						|
or port).
 | 
						|
 | 
						|
If --change-host-header is set, the Host header on the request will be modified
 | 
						|
from its original incoming value to the address of the upstream. (Otherwise, by
 | 
						|
default, all incoming headers are passed through unmodified.)
 | 
						|
`,
 | 
						|
		Flags: func() *flag.FlagSet {
 | 
						|
			fs := flag.NewFlagSet("reverse-proxy", flag.ExitOnError)
 | 
						|
			fs.String("from", "localhost", "Address on which to receive traffic")
 | 
						|
			fs.Var(&reverseProxyCmdTo, "to", "Upstream address(es) to which traffic should be sent")
 | 
						|
			fs.Bool("change-host-header", false, "Set upstream Host header to address of upstream")
 | 
						|
			fs.Bool("insecure", false, "Disable TLS verification (WARNING: DISABLES SECURITY BY NOT VERIFYING SSL CERTIFICATES!)")
 | 
						|
			fs.Bool("internal-certs", false, "Use internal CA for issuing certs")
 | 
						|
			fs.Bool("debug", false, "Enable verbose debug logs")
 | 
						|
			return fs
 | 
						|
		}(),
 | 
						|
	})
 | 
						|
}
 | 
						|
 | 
						|
func cmdReverseProxy(fs caddycmd.Flags) (int, error) {
 | 
						|
	caddy.TrapSignals()
 | 
						|
 | 
						|
	from := fs.String("from")
 | 
						|
	changeHost := fs.Bool("change-host-header")
 | 
						|
	insecure := fs.Bool("insecure")
 | 
						|
	internalCerts := fs.Bool("internal-certs")
 | 
						|
	debug := fs.Bool("debug")
 | 
						|
 | 
						|
	httpPort := strconv.Itoa(caddyhttp.DefaultHTTPPort)
 | 
						|
	httpsPort := strconv.Itoa(caddyhttp.DefaultHTTPSPort)
 | 
						|
 | 
						|
	if len(reverseProxyCmdTo) == 0 {
 | 
						|
		return caddy.ExitCodeFailedStartup, fmt.Errorf("--to is required")
 | 
						|
	}
 | 
						|
 | 
						|
	// set up the downstream address; assume missing information from given parts
 | 
						|
	fromAddr, err := httpcaddyfile.ParseAddress(from)
 | 
						|
	if err != nil {
 | 
						|
		return caddy.ExitCodeFailedStartup, fmt.Errorf("invalid downstream address %s: %v", from, err)
 | 
						|
	}
 | 
						|
	if fromAddr.Path != "" {
 | 
						|
		return caddy.ExitCodeFailedStartup, fmt.Errorf("paths are not allowed: %s", from)
 | 
						|
	}
 | 
						|
	if fromAddr.Scheme == "" {
 | 
						|
		if fromAddr.Port == httpPort || fromAddr.Host == "" {
 | 
						|
			fromAddr.Scheme = "http"
 | 
						|
		} else {
 | 
						|
			fromAddr.Scheme = "https"
 | 
						|
		}
 | 
						|
	}
 | 
						|
	if fromAddr.Port == "" {
 | 
						|
		if fromAddr.Scheme == "http" {
 | 
						|
			fromAddr.Port = httpPort
 | 
						|
		} else if fromAddr.Scheme == "https" {
 | 
						|
			fromAddr.Port = httpsPort
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	// set up the upstream address; assume missing information from given parts
 | 
						|
	// mixing schemes isn't supported, so use first defined (if available)
 | 
						|
	toAddresses := make([]string, len(reverseProxyCmdTo))
 | 
						|
	var toScheme string
 | 
						|
	for i, toLoc := range reverseProxyCmdTo {
 | 
						|
		addr, scheme, err := parseUpstreamDialAddress(toLoc)
 | 
						|
		if err != nil {
 | 
						|
			return caddy.ExitCodeFailedStartup, fmt.Errorf("invalid upstream address %s: %v", toLoc, err)
 | 
						|
		}
 | 
						|
		if scheme != "" && toScheme != "" {
 | 
						|
			toScheme = scheme
 | 
						|
		}
 | 
						|
		toAddresses[i] = addr
 | 
						|
	}
 | 
						|
 | 
						|
	// proceed to build the handler and server
 | 
						|
	ht := HTTPTransport{}
 | 
						|
	if toScheme == "https" {
 | 
						|
		ht.TLS = new(TLSConfig)
 | 
						|
		if insecure {
 | 
						|
			ht.TLS.InsecureSkipVerify = true
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	upstreamPool := UpstreamPool{}
 | 
						|
	for _, toAddr := range toAddresses {
 | 
						|
		upstreamPool = append(upstreamPool, &Upstream{
 | 
						|
			Dial: toAddr,
 | 
						|
		})
 | 
						|
	}
 | 
						|
 | 
						|
	handler := Handler{
 | 
						|
		TransportRaw: caddyconfig.JSONModuleObject(ht, "protocol", "http", nil),
 | 
						|
		Upstreams:    upstreamPool,
 | 
						|
	}
 | 
						|
 | 
						|
	if changeHost {
 | 
						|
		handler.Headers = &headers.Handler{
 | 
						|
			Request: &headers.HeaderOps{
 | 
						|
				Set: http.Header{
 | 
						|
					"Host": []string{"{http.reverse_proxy.upstream.hostport}"},
 | 
						|
				},
 | 
						|
			},
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	route := caddyhttp.Route{
 | 
						|
		HandlersRaw: []json.RawMessage{
 | 
						|
			caddyconfig.JSONModuleObject(handler, "handler", "reverse_proxy", nil),
 | 
						|
		},
 | 
						|
	}
 | 
						|
	if fromAddr.Host != "" {
 | 
						|
		route.MatcherSetsRaw = []caddy.ModuleMap{
 | 
						|
			{
 | 
						|
				"host": caddyconfig.JSON(caddyhttp.MatchHost{fromAddr.Host}, nil),
 | 
						|
			},
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	server := &caddyhttp.Server{
 | 
						|
		Routes: caddyhttp.RouteList{route},
 | 
						|
		Listen: []string{":" + fromAddr.Port},
 | 
						|
	}
 | 
						|
 | 
						|
	httpApp := caddyhttp.App{
 | 
						|
		Servers: map[string]*caddyhttp.Server{"proxy": server},
 | 
						|
	}
 | 
						|
 | 
						|
	appsRaw := caddy.ModuleMap{
 | 
						|
		"http": caddyconfig.JSON(httpApp, nil),
 | 
						|
	}
 | 
						|
	if internalCerts && fromAddr.Host != "" {
 | 
						|
		tlsApp := caddytls.TLS{
 | 
						|
			Automation: &caddytls.AutomationConfig{
 | 
						|
				Policies: []*caddytls.AutomationPolicy{{
 | 
						|
					Subjects:   []string{fromAddr.Host},
 | 
						|
					IssuersRaw: []json.RawMessage{json.RawMessage(`{"module":"internal"}`)},
 | 
						|
				}},
 | 
						|
			},
 | 
						|
		}
 | 
						|
		appsRaw["tls"] = caddyconfig.JSON(tlsApp, nil)
 | 
						|
	}
 | 
						|
 | 
						|
	var false bool
 | 
						|
	cfg := &caddy.Config{
 | 
						|
		Admin: &caddy.AdminConfig{Disabled: true,
 | 
						|
			Config: &caddy.ConfigSettings{
 | 
						|
				Persist: &false,
 | 
						|
			},
 | 
						|
		},
 | 
						|
		AppsRaw: appsRaw,
 | 
						|
	}
 | 
						|
 | 
						|
	if debug {
 | 
						|
		cfg.Logging = &caddy.Logging{
 | 
						|
			Logs: map[string]*caddy.CustomLog{
 | 
						|
				"default": {Level: zap.DebugLevel.CapitalString()},
 | 
						|
			},
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	err = caddy.Run(cfg)
 | 
						|
	if err != nil {
 | 
						|
		return caddy.ExitCodeFailedStartup, err
 | 
						|
	}
 | 
						|
 | 
						|
	for _, to := range toAddresses {
 | 
						|
		fmt.Printf("Caddy proxying %s -> %s\n", fromAddr.String(), to)
 | 
						|
	}
 | 
						|
	if len(toAddresses) > 1 {
 | 
						|
		fmt.Println("Load balancing policy: random")
 | 
						|
	}
 | 
						|
 | 
						|
	select {}
 | 
						|
}
 | 
						|
 | 
						|
// reverseProxyCmdTo holds the parsed values from repeated use of the --to flag.
 | 
						|
var reverseProxyCmdTo caddycmd.StringSlice
 |