mirror of
https://github.com/yusing/godoxy.git
synced 2026-03-17 23:03:49 +01:00
Add a DisplayNameKey struct to pass display names from routes through the task parent hierarchy to the health monitor. This allows the health monitor to use more descriptive names for logging instead of internal task names. BREAKING CHANGE: The monitor.DisplayNameKey struct is now part of the public API
242 lines
5.8 KiB
Go
Executable File
242 lines
5.8 KiB
Go
Executable File
package route
|
|
|
|
import (
|
|
"errors"
|
|
"net/http"
|
|
"sync"
|
|
|
|
"github.com/rs/zerolog/log"
|
|
"github.com/yusing/godoxy/agent/pkg/agent"
|
|
"github.com/yusing/godoxy/agent/pkg/agentproxy"
|
|
entrypoint "github.com/yusing/godoxy/internal/entrypoint/types"
|
|
"github.com/yusing/godoxy/internal/health/monitor"
|
|
"github.com/yusing/godoxy/internal/idlewatcher"
|
|
"github.com/yusing/godoxy/internal/logging/accesslog"
|
|
gphttp "github.com/yusing/godoxy/internal/net/gphttp"
|
|
"github.com/yusing/godoxy/internal/net/gphttp/loadbalancer"
|
|
"github.com/yusing/godoxy/internal/net/gphttp/middleware"
|
|
nettypes "github.com/yusing/godoxy/internal/net/types"
|
|
route "github.com/yusing/godoxy/internal/route/types"
|
|
"github.com/yusing/godoxy/internal/types"
|
|
"github.com/yusing/goutils/http/reverseproxy"
|
|
"github.com/yusing/goutils/task"
|
|
"github.com/yusing/goutils/version"
|
|
)
|
|
|
|
type ReveseProxyRoute struct {
|
|
*Route
|
|
|
|
loadBalancer *loadbalancer.LoadBalancer
|
|
handler http.Handler
|
|
rp *reverseproxy.ReverseProxy
|
|
}
|
|
|
|
var _ types.ReverseProxyRoute = (*ReveseProxyRoute)(nil)
|
|
|
|
// var globalMux = http.NewServeMux() // TODO: support regex subdomain matching.
|
|
|
|
func NewReverseProxyRoute(base *Route) (*ReveseProxyRoute, error) {
|
|
httpConfig := base.HTTPConfig
|
|
proxyURL := base.ProxyURL
|
|
|
|
var trans *http.Transport
|
|
a := base.GetAgent()
|
|
if a != nil {
|
|
trans = a.Transport()
|
|
proxyURL = nettypes.NewURL(agent.HTTPProxyURL)
|
|
} else {
|
|
tlsConfig, err := httpConfig.BuildTLSConfig(&base.ProxyURL.URL)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
trans = gphttp.NewTransportWithTLSConfig(tlsConfig)
|
|
if httpConfig.ResponseHeaderTimeout > 0 {
|
|
trans.ResponseHeaderTimeout = httpConfig.ResponseHeaderTimeout
|
|
}
|
|
if httpConfig.DisableCompression {
|
|
trans.DisableCompression = true
|
|
}
|
|
}
|
|
|
|
service := base.Name()
|
|
rp := reverseproxy.NewReverseProxy(service, &proxyURL.URL, trans)
|
|
|
|
scheme := base.Scheme
|
|
retried := false
|
|
retryLock := sync.Mutex{}
|
|
if scheme == route.SchemeHTTP || scheme == route.SchemeHTTPS {
|
|
rp.OnSchemeMisMatch = func() (retry bool) { // switch scheme and retry
|
|
retryLock.Lock()
|
|
defer retryLock.Unlock()
|
|
|
|
if retried {
|
|
return false
|
|
}
|
|
|
|
retried = true
|
|
|
|
if scheme == route.SchemeHTTP {
|
|
rp.TargetURL.Scheme = "https"
|
|
} else {
|
|
rp.TargetURL.Scheme = "http"
|
|
}
|
|
rp.Info().Msgf("scheme mismatch detected, retrying with %s", rp.TargetURL.Scheme)
|
|
return true
|
|
}
|
|
}
|
|
|
|
if len(base.Middlewares) > 0 {
|
|
err := middleware.PatchReverseProxy(rp, base.Middlewares)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
|
|
if a != nil {
|
|
cfg := agentproxy.Config{
|
|
Scheme: base.ProxyURL.Scheme,
|
|
Host: base.ProxyURL.Host,
|
|
HTTPConfig: httpConfig,
|
|
}
|
|
setHeaderFunc := cfg.SetAgentProxyConfigHeaders
|
|
if !a.Version.IsOlderThan(version.New(0, 18, 6)) {
|
|
setHeaderFunc = cfg.SetAgentProxyConfigHeadersLegacy
|
|
}
|
|
|
|
ori := rp.HandlerFunc
|
|
rp.HandlerFunc = func(w http.ResponseWriter, r *http.Request) {
|
|
setHeaderFunc(r.Header)
|
|
ori(w, r)
|
|
}
|
|
}
|
|
|
|
r := &ReveseProxyRoute{
|
|
Route: base,
|
|
rp: rp,
|
|
}
|
|
return r, nil
|
|
}
|
|
|
|
// ReverseProxy implements routes.ReverseProxyRoute.
|
|
func (r *ReveseProxyRoute) ReverseProxy() *reverseproxy.ReverseProxy {
|
|
return r.rp
|
|
}
|
|
|
|
// Start implements task.TaskStarter.
|
|
func (r *ReveseProxyRoute) Start(parent task.Parent) error {
|
|
r.task = parent.Subtask("http."+r.Name(), false)
|
|
r.task.SetValue(monitor.DisplayNameKey{}, r.DisplayName())
|
|
|
|
switch {
|
|
case r.UseIdleWatcher():
|
|
waker, err := idlewatcher.NewWatcher(parent, r, r.IdlewatcherConfig())
|
|
if err != nil {
|
|
r.task.Finish(err)
|
|
return err
|
|
}
|
|
r.handler = waker
|
|
r.HealthMon = waker
|
|
case r.UseHealthCheck():
|
|
r.HealthMon = monitor.NewMonitor(r)
|
|
}
|
|
|
|
if r.handler == nil {
|
|
r.handler = r.rp
|
|
}
|
|
|
|
if r.UseAccessLog() {
|
|
var err error
|
|
r.rp.AccessLogger, err = accesslog.NewAccessLogger(r.task, r.AccessLog)
|
|
if err != nil {
|
|
r.task.Finish(err)
|
|
return err
|
|
}
|
|
}
|
|
|
|
if len(r.Rules) > 0 {
|
|
r.handler = r.Rules.BuildHandler(r.handler.ServeHTTP)
|
|
}
|
|
|
|
if r.HealthMon != nil {
|
|
if err := r.HealthMon.Start(r.task); err != nil {
|
|
log.Warn().Err(err).Msg("health monitor error")
|
|
r.HealthMon = nil
|
|
}
|
|
}
|
|
|
|
ep := entrypoint.FromCtx(parent.Context())
|
|
if ep == nil {
|
|
err := errors.New("entrypoint not initialized")
|
|
r.task.Finish(err)
|
|
return err
|
|
}
|
|
|
|
if r.UseLoadBalance() {
|
|
if err := r.addToLoadBalancer(parent, ep); err != nil {
|
|
r.task.Finish(err)
|
|
return err
|
|
}
|
|
} else {
|
|
if err := ep.StartAddRoute(r); err != nil {
|
|
r.task.Finish(err)
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (r *ReveseProxyRoute) ServeHTTP(w http.ResponseWriter, req *http.Request) {
|
|
// req.Header.Set("Accept-Encoding", "identity")
|
|
r.handler.ServeHTTP(w, req)
|
|
}
|
|
|
|
var lbLock sync.Mutex
|
|
|
|
func (r *ReveseProxyRoute) addToLoadBalancer(parent task.Parent, ep entrypoint.Entrypoint) error {
|
|
var lb *loadbalancer.LoadBalancer
|
|
cfg := r.LoadBalance
|
|
lbLock.Lock()
|
|
defer lbLock.Unlock()
|
|
|
|
l, ok := ep.HTTPRoutes().Get(cfg.Link)
|
|
var linked *ReveseProxyRoute
|
|
if ok {
|
|
linked = l.(*ReveseProxyRoute) // it must be a reverse proxy route
|
|
lb = linked.loadBalancer
|
|
lb.UpdateConfigIfNeeded(cfg)
|
|
if linked.Homepage.Name == "" {
|
|
linked.Homepage = r.Homepage
|
|
}
|
|
} else {
|
|
lb = loadbalancer.New(cfg)
|
|
_ = lb.Start(parent) // always return nil
|
|
linked = &ReveseProxyRoute{
|
|
Route: &Route{
|
|
Alias: cfg.Link,
|
|
Homepage: r.Homepage,
|
|
Bind: r.Bind,
|
|
Metadata: Metadata{
|
|
LisURL: r.ListenURL(),
|
|
task: lb.Task(),
|
|
},
|
|
},
|
|
loadBalancer: lb,
|
|
handler: lb,
|
|
}
|
|
linked.SetHealthMonitor(lb)
|
|
if err := ep.StartAddRoute(linked); err != nil {
|
|
lb.Finish(err)
|
|
return err
|
|
}
|
|
}
|
|
r.loadBalancer = lb
|
|
|
|
server := loadbalancer.NewServer(r.task.Name(), r.ProxyURL, r.LoadBalance.Weight, r.handler, r.HealthMon)
|
|
lb.AddServer(server)
|
|
r.task.OnCancel("lb_remove_server", func() {
|
|
lb.RemoveServer(server)
|
|
})
|
|
return nil
|
|
}
|