94 lines
2.7 KiB
Go
94 lines
2.7 KiB
Go
package application
|
|
|
|
import (
|
|
"context"
|
|
"crypto/tls"
|
|
"net/http"
|
|
"net/http/httputil"
|
|
"net/url"
|
|
"time"
|
|
|
|
"github.com/getsentry/sentry-go"
|
|
"github.com/prometheus/client_golang/prometheus"
|
|
log "github.com/sirupsen/logrus"
|
|
"goauthentik.io/internal/outpost/proxyv2/metrics"
|
|
"goauthentik.io/internal/utils/web"
|
|
)
|
|
|
|
func (a *Application) getUpstreamTransport() http.RoundTripper {
|
|
return &http.Transport{
|
|
TLSClientConfig: &tls.Config{InsecureSkipVerify: !*a.proxyConfig.InternalHostSslValidation},
|
|
}
|
|
}
|
|
|
|
func (a *Application) configureProxy() error {
|
|
// Reverse proxy to the application server
|
|
u, err := url.Parse(*a.proxyConfig.InternalHost)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
rsp := sentry.StartSpan(context.TODO(), "authentik.outposts.proxy.application_transport")
|
|
rp := &httputil.ReverseProxy{
|
|
Director: a.proxyModifyRequest(u),
|
|
Transport: web.NewTracingTransport(rsp.Context(), a.getUpstreamTransport()),
|
|
ErrorHandler: a.newProxyErrorHandler(),
|
|
ModifyResponse: a.proxyModifyResponse,
|
|
FlushInterval: -1,
|
|
}
|
|
a.mux.PathPrefix("/").HandlerFunc(func(rw http.ResponseWriter, r *http.Request) {
|
|
defer func() {
|
|
err := recover()
|
|
if err == nil || err == http.ErrAbortHandler {
|
|
return
|
|
}
|
|
log.WithError(err.(error)).Error("recover in reverse proxy")
|
|
}()
|
|
claims, err := a.checkAuth(rw, r)
|
|
if claims == nil && a.IsAllowlisted(r.URL) {
|
|
a.log.Trace("path can be accessed without authentication")
|
|
} else if claims == nil && err != nil {
|
|
a.log.WithError(err).Trace("no claims")
|
|
a.redirectToStart(rw, r)
|
|
return
|
|
} else {
|
|
a.addHeaders(r.Header, claims)
|
|
}
|
|
before := time.Now()
|
|
rp.ServeHTTP(rw, r)
|
|
after := time.Since(before)
|
|
|
|
metrics.UpstreamTiming.With(prometheus.Labels{
|
|
"outpost_name": a.outpostName,
|
|
"upstream_host": r.URL.Host,
|
|
"method": r.Method,
|
|
"scheme": r.URL.Scheme,
|
|
"host": web.GetHost(r),
|
|
}).Observe(float64(after))
|
|
})
|
|
return nil
|
|
}
|
|
|
|
func (a *Application) proxyModifyRequest(ou *url.URL) func(req *http.Request) {
|
|
return func(r *http.Request) {
|
|
r.Header.Set("X-Forwarded-Host", r.Host)
|
|
r.URL.Scheme = ou.Scheme
|
|
r.URL.Host = ou.Host
|
|
claims := a.getClaimsFromSession(r)
|
|
if claims != nil && claims.Proxy != nil && claims.Proxy.BackendOverride != "" {
|
|
u, err := url.Parse(claims.Proxy.BackendOverride)
|
|
if err != nil {
|
|
a.log.WithField("backend_override", claims.Proxy.BackendOverride).WithError(err).Warning("failed parse user backend override")
|
|
} else {
|
|
r.URL.Scheme = u.Scheme
|
|
r.URL.Host = u.Host
|
|
}
|
|
}
|
|
a.log.WithField("upstream_url", r.URL.String()).Trace("final upstream url")
|
|
}
|
|
}
|
|
|
|
func (a *Application) proxyModifyResponse(res *http.Response) error {
|
|
res.Header.Set("X-Powered-By", "goauthentik.io")
|
|
return nil
|
|
}
|