2019-10-04 01:00:41 +03:00
|
|
|
// 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"
|
2019-10-31 20:34:54 +03:00
|
|
|
"fmt"
|
2019-11-04 22:18:42 +03:00
|
|
|
"net"
|
2019-10-04 01:00:41 +03:00
|
|
|
"net/http"
|
|
|
|
"net/url"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"github.com/caddyserver/caddy/v2"
|
|
|
|
"github.com/caddyserver/caddy/v2/caddyconfig"
|
|
|
|
caddycmd "github.com/caddyserver/caddy/v2/cmd"
|
|
|
|
"github.com/caddyserver/caddy/v2/modules/caddyhttp"
|
|
|
|
"github.com/caddyserver/caddy/v2/modules/caddyhttp/headers"
|
|
|
|
)
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
caddycmd.RegisterCommand(caddycmd.Command{
|
|
|
|
Name: "reverse-proxy",
|
|
|
|
Func: cmdReverseProxy,
|
2020-02-28 06:52:41 +03:00
|
|
|
Usage: "[--from <addr>] [--to <addr>] [--change-host-header]",
|
2019-10-04 01:00:41 +03:00
|
|
|
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 traffic from the --from address to the --to address.
|
|
|
|
|
|
|
|
If the --from address has a domain name, Caddy will attempt to serve the
|
|
|
|
proxy over HTTPS with a certificate.
|
2020-02-28 06:52:41 +03:00
|
|
|
|
|
|
|
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.)
|
2019-10-04 01:00:41 +03:00
|
|
|
`,
|
|
|
|
Flags: func() *flag.FlagSet {
|
|
|
|
fs := flag.NewFlagSet("file-server", flag.ExitOnError)
|
2020-02-28 06:52:41 +03:00
|
|
|
fs.String("from", "", "Address on which to receive traffic")
|
|
|
|
fs.String("to", "", "Upstream address to which to to proxy traffic")
|
|
|
|
fs.Bool("change-host-header", false, "Set upstream Host header to address of upstream")
|
2019-10-04 01:00:41 +03:00
|
|
|
return fs
|
|
|
|
}(),
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func cmdReverseProxy(fs caddycmd.Flags) (int, error) {
|
|
|
|
from := fs.String("from")
|
|
|
|
to := fs.String("to")
|
2020-02-28 06:52:41 +03:00
|
|
|
changeHost := fs.Bool("change-host-header")
|
2019-10-04 01:00:41 +03:00
|
|
|
|
|
|
|
if from == "" {
|
2020-03-13 20:06:08 +03:00
|
|
|
from = "localhost:443"
|
2019-10-04 01:00:41 +03:00
|
|
|
}
|
|
|
|
|
2019-10-31 20:34:54 +03:00
|
|
|
// URLs need a scheme in order to parse successfully
|
2019-10-04 01:00:41 +03:00
|
|
|
if !strings.Contains(from, "://") {
|
|
|
|
from = "http://" + from
|
|
|
|
}
|
2019-10-31 20:34:54 +03:00
|
|
|
if !strings.Contains(to, "://") {
|
|
|
|
to = "http://" + to
|
|
|
|
}
|
2019-10-04 01:00:41 +03:00
|
|
|
|
|
|
|
fromURL, err := url.Parse(from)
|
|
|
|
if err != nil {
|
2019-10-31 20:34:54 +03:00
|
|
|
return caddy.ExitCodeFailedStartup, fmt.Errorf("parsing 'from' URL: %v", err)
|
2019-10-04 01:00:41 +03:00
|
|
|
}
|
|
|
|
toURL, err := url.Parse(to)
|
|
|
|
if err != nil {
|
2019-10-31 20:34:54 +03:00
|
|
|
return caddy.ExitCodeFailedStartup, fmt.Errorf("parsing 'to' URL: %v", err)
|
2019-10-04 01:00:41 +03:00
|
|
|
}
|
|
|
|
|
2019-11-04 22:18:42 +03:00
|
|
|
if toURL.Port() == "" {
|
|
|
|
toPort := "80"
|
|
|
|
if toURL.Scheme == "https" {
|
|
|
|
toPort = "443"
|
|
|
|
}
|
|
|
|
toURL.Host = net.JoinHostPort(toURL.Host, toPort)
|
|
|
|
}
|
|
|
|
|
2019-10-04 01:00:41 +03:00
|
|
|
ht := HTTPTransport{}
|
|
|
|
if toURL.Scheme == "https" {
|
|
|
|
ht.TLS = new(TLSConfig)
|
|
|
|
}
|
|
|
|
|
|
|
|
handler := Handler{
|
|
|
|
TransportRaw: caddyconfig.JSONModuleObject(ht, "protocol", "http", nil),
|
|
|
|
Upstreams: UpstreamPool{{Dial: toURL.Host}},
|
2020-02-28 06:52:41 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
if changeHost {
|
|
|
|
handler.Headers = &headers.Handler{
|
2019-10-04 01:00:41 +03:00
|
|
|
Request: &headers.HeaderOps{
|
|
|
|
Set: http.Header{
|
2020-02-28 06:52:41 +03:00
|
|
|
"Host": []string{"{http.reverse_proxy.upstream.hostport}"},
|
2019-10-04 01:00:41 +03:00
|
|
|
},
|
|
|
|
},
|
2020-02-28 06:52:41 +03:00
|
|
|
}
|
2019-10-04 01:00:41 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
route := caddyhttp.Route{
|
|
|
|
HandlersRaw: []json.RawMessage{
|
|
|
|
caddyconfig.JSONModuleObject(handler, "handler", "reverse_proxy", nil),
|
|
|
|
},
|
|
|
|
}
|
2019-10-31 20:34:54 +03:00
|
|
|
urlHost := fromURL.Hostname()
|
|
|
|
if urlHost != "" {
|
2019-12-10 23:36:46 +03:00
|
|
|
route.MatcherSetsRaw = []caddy.ModuleMap{
|
|
|
|
caddy.ModuleMap{
|
2019-10-31 20:34:54 +03:00
|
|
|
"host": caddyconfig.JSON(caddyhttp.MatchHost{urlHost}, nil),
|
2019-10-04 01:00:41 +03:00
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-13 20:06:08 +03:00
|
|
|
listen := ":443"
|
2019-10-31 20:34:54 +03:00
|
|
|
if urlPort := fromURL.Port(); urlPort != "" {
|
|
|
|
listen = ":" + urlPort
|
2019-10-04 01:00:41 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
server := &caddyhttp.Server{
|
|
|
|
Routes: caddyhttp.RouteList{route},
|
|
|
|
Listen: []string{listen},
|
|
|
|
}
|
|
|
|
|
|
|
|
httpApp := caddyhttp.App{
|
|
|
|
Servers: map[string]*caddyhttp.Server{"proxy": server},
|
|
|
|
}
|
|
|
|
|
|
|
|
cfg := &caddy.Config{
|
2019-11-16 01:52:19 +03:00
|
|
|
Admin: &caddy.AdminConfig{Disabled: true},
|
2019-12-10 23:36:46 +03:00
|
|
|
AppsRaw: caddy.ModuleMap{
|
2019-10-04 01:00:41 +03:00
|
|
|
"http": caddyconfig.JSON(httpApp, nil),
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
err = caddy.Run(cfg)
|
|
|
|
if err != nil {
|
|
|
|
return caddy.ExitCodeFailedStartup, err
|
|
|
|
}
|
|
|
|
|
2019-10-31 20:34:54 +03:00
|
|
|
fmt.Printf("Caddy 2 proxying from %s to %s\n", fromURL, toURL)
|
2019-10-04 01:00:41 +03:00
|
|
|
|
|
|
|
select {}
|
|
|
|
}
|