2024-08-24 15:54:19 +03:00
|
|
|
package network
|
2024-06-15 14:08:27 +03:00
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
|
|
|
"net/http"
|
|
|
|
"net/url"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"go.uber.org/zap"
|
|
|
|
|
|
|
|
"github.com/caddyserver/caddy/v2"
|
2024-08-24 15:54:19 +03:00
|
|
|
"github.com/caddyserver/caddy/v2/caddyconfig/caddyfile"
|
2024-06-15 14:08:27 +03:00
|
|
|
)
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
caddy.RegisterModule(ProxyFromURL{})
|
2024-08-24 15:54:19 +03:00
|
|
|
caddy.RegisterModule(ProxyFromNone{})
|
2024-06-15 14:08:27 +03:00
|
|
|
}
|
|
|
|
|
2024-08-26 20:46:27 +03:00
|
|
|
// The "url" proxy source uses the defined URL as the proxy
|
2024-06-15 14:08:27 +03:00
|
|
|
type ProxyFromURL struct {
|
|
|
|
URL string `json:"url"`
|
|
|
|
|
|
|
|
ctx caddy.Context
|
|
|
|
logger *zap.Logger
|
|
|
|
}
|
|
|
|
|
|
|
|
// CaddyModule implements Module.
|
|
|
|
func (p ProxyFromURL) CaddyModule() caddy.ModuleInfo {
|
|
|
|
return caddy.ModuleInfo{
|
|
|
|
ID: "caddy.network_proxy.source.url",
|
|
|
|
New: func() caddy.Module {
|
|
|
|
return &ProxyFromURL{}
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *ProxyFromURL) Provision(ctx caddy.Context) error {
|
|
|
|
p.ctx = ctx
|
|
|
|
p.logger = ctx.Logger()
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Validate implements Validator.
|
|
|
|
func (p ProxyFromURL) Validate() error {
|
|
|
|
if _, err := url.Parse(p.URL); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// ProxyFunc implements ProxyFuncProducer.
|
|
|
|
func (p ProxyFromURL) ProxyFunc() func(*http.Request) (*url.URL, error) {
|
|
|
|
if strings.Contains(p.URL, "{") && strings.Contains(p.URL, "}") {
|
|
|
|
// courtesy of @ImpostorKeanu: https://github.com/caddyserver/caddy/pull/6397
|
|
|
|
return func(r *http.Request) (*url.URL, error) {
|
|
|
|
// retrieve the replacer from context.
|
|
|
|
repl, ok := r.Context().Value(caddy.ReplacerCtxKey).(*caddy.Replacer)
|
|
|
|
if !ok {
|
|
|
|
err := errors.New("failed to obtain replacer from request")
|
|
|
|
p.logger.Error(err.Error())
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// apply placeholders to the value
|
|
|
|
// note: h.ForwardProxyURL should never be empty at this point
|
|
|
|
s := repl.ReplaceAll(p.URL, "")
|
|
|
|
if s == "" {
|
2024-08-26 20:46:27 +03:00
|
|
|
p.logger.Error("network_proxy URL was empty after applying placeholders",
|
2024-06-15 14:08:27 +03:00
|
|
|
zap.String("initial_value", p.URL),
|
|
|
|
zap.String("final_value", s),
|
|
|
|
zap.String("hint", "check for invalid placeholders"))
|
2024-08-26 20:46:27 +03:00
|
|
|
return nil, errors.New("empty value for network_proxy URL")
|
2024-06-15 14:08:27 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// parse the url
|
|
|
|
pUrl, err := url.Parse(s)
|
|
|
|
if err != nil {
|
2024-08-26 20:46:27 +03:00
|
|
|
p.logger.Warn("failed to derive transport proxy from network_proxy URL")
|
2024-06-15 14:08:27 +03:00
|
|
|
pUrl = nil
|
|
|
|
} else if pUrl.Host == "" || strings.Split("", pUrl.Host)[0] == ":" {
|
|
|
|
// url.Parse does not return an error on these values:
|
|
|
|
//
|
|
|
|
// - http://:80
|
|
|
|
// - pUrl.Host == ":80"
|
|
|
|
// - /some/path
|
|
|
|
// - pUrl.Host == ""
|
|
|
|
//
|
|
|
|
// Super edge cases, but humans are human.
|
2024-08-26 20:46:27 +03:00
|
|
|
err = errors.New("supplied network_proxy URL is missing a host value")
|
2024-06-15 14:08:27 +03:00
|
|
|
pUrl = nil
|
|
|
|
} else {
|
|
|
|
p.logger.Debug("setting transport proxy url", zap.String("url", s))
|
|
|
|
}
|
|
|
|
|
|
|
|
return pUrl, err
|
|
|
|
}
|
|
|
|
}
|
2024-08-24 15:54:19 +03:00
|
|
|
return func(r *http.Request) (*url.URL, error) {
|
2024-06-15 14:08:27 +03:00
|
|
|
return url.Parse(p.URL)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-08-24 15:54:19 +03:00
|
|
|
// UnmarshalCaddyfile implements caddyfile.Unmarshaler.
|
|
|
|
func (p *ProxyFromURL) UnmarshalCaddyfile(d *caddyfile.Dispenser) error {
|
|
|
|
d.Next()
|
|
|
|
d.Next()
|
|
|
|
p.URL = d.Val()
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2024-08-26 20:46:27 +03:00
|
|
|
// The "none" proxy source module disables the use of network proxy.
|
2024-08-24 15:54:19 +03:00
|
|
|
type ProxyFromNone struct{}
|
|
|
|
|
|
|
|
func (p ProxyFromNone) CaddyModule() caddy.ModuleInfo {
|
|
|
|
return caddy.ModuleInfo{
|
|
|
|
ID: "caddy.network_proxy.source.none",
|
|
|
|
New: func() caddy.Module {
|
|
|
|
return &ProxyFromNone{}
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// UnmarshalCaddyfile implements caddyfile.Unmarshaler.
|
|
|
|
func (p ProxyFromNone) UnmarshalCaddyfile(d *caddyfile.Dispenser) error {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// ProxyFunc implements ProxyFuncProducer.
|
|
|
|
func (p ProxyFromNone) ProxyFunc() func(*http.Request) (*url.URL, error) {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2024-06-15 14:08:27 +03:00
|
|
|
var (
|
|
|
|
_ caddy.Module = ProxyFromURL{}
|
|
|
|
_ caddy.Provisioner = &ProxyFromURL{}
|
|
|
|
_ caddy.Validator = ProxyFromURL{}
|
|
|
|
_ caddy.ProxyFuncProducer = ProxyFromURL{}
|
2024-08-24 15:54:19 +03:00
|
|
|
_ caddyfile.Unmarshaler = &ProxyFromURL{}
|
|
|
|
|
|
|
|
_ caddy.Module = ProxyFromNone{}
|
|
|
|
_ caddy.ProxyFuncProducer = ProxyFromNone{}
|
|
|
|
_ caddyfile.Unmarshaler = ProxyFromNone{}
|
2024-06-15 14:08:27 +03:00
|
|
|
)
|