* api: allow API requests as managed outpost's account when using secret_key Signed-off-by: Jens Langhammer <jens.langhammer@beryju.org> * root: load secret key from env Signed-off-by: Jens Langhammer <jens.langhammer@beryju.org> * outposts: make listener IP configurable Signed-off-by: Jens Langhammer <jens.langhammer@beryju.org> * outpost/proxy: run outpost in background and pass requests conditionally Signed-off-by: Jens Langhammer <jens.langhammer@beryju.org> * outpost: unify branding to embedded Signed-off-by: Jens Langhammer <jens.langhammer@beryju.org> * web/admin: fix embedded outpost not being editable Signed-off-by: Jens Langhammer <jens.langhammer@beryju.org> * web: fix mismatched host detection Signed-off-by: Jens Langhammer <jens.langhammer@beryju.org> * tests/e2e: fix LDAP test not including user for embedded outpost Signed-off-by: Jens Langhammer <jens.langhammer@beryju.org> * tests/e2e: fix user matching Signed-off-by: Jens Langhammer <jens.langhammer@beryju.org> * api: add tests for secret_key auth Signed-off-by: Jens Langhammer <jens.langhammer@beryju.org> * root: load environment variables using github.com/Netflix/go-env Signed-off-by: Jens Langhammer <jens.langhammer@beryju.org>
		
			
				
	
	
		
			122 lines
		
	
	
		
			3.1 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			122 lines
		
	
	
		
			3.1 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
package proxy
 | 
						|
 | 
						|
import (
 | 
						|
	"bufio"
 | 
						|
	"errors"
 | 
						|
	"fmt"
 | 
						|
	"net"
 | 
						|
	"net/http"
 | 
						|
	"time"
 | 
						|
 | 
						|
	log "github.com/sirupsen/logrus"
 | 
						|
	"goauthentik.io/internal/utils/web"
 | 
						|
)
 | 
						|
 | 
						|
// responseLogger is wrapper of http.ResponseWriter that keeps track of its HTTP status
 | 
						|
// code and body size
 | 
						|
type responseLogger struct {
 | 
						|
	w        http.ResponseWriter
 | 
						|
	status   int
 | 
						|
	size     int
 | 
						|
	upstream string
 | 
						|
	authInfo string
 | 
						|
}
 | 
						|
 | 
						|
// Header returns the ResponseWriter's Header
 | 
						|
func (l *responseLogger) Header() http.Header {
 | 
						|
	return l.w.Header()
 | 
						|
}
 | 
						|
 | 
						|
// Support Websocket
 | 
						|
func (l *responseLogger) Hijack() (rwc net.Conn, buf *bufio.ReadWriter, err error) {
 | 
						|
	if hj, ok := l.w.(http.Hijacker); ok {
 | 
						|
		return hj.Hijack()
 | 
						|
	}
 | 
						|
	return nil, nil, errors.New("http.Hijacker is not available on writer")
 | 
						|
}
 | 
						|
 | 
						|
// ExtractGAPMetadata extracts and removes GAP headers from the ResponseWriter's
 | 
						|
// Header
 | 
						|
func (l *responseLogger) ExtractGAPMetadata() {
 | 
						|
	upstream := l.w.Header().Get("GAP-Upstream-Address")
 | 
						|
	if upstream != "" {
 | 
						|
		l.upstream = upstream
 | 
						|
		l.w.Header().Del("GAP-Upstream-Address")
 | 
						|
	}
 | 
						|
	authInfo := l.w.Header().Get("GAP-Auth")
 | 
						|
	if authInfo != "" {
 | 
						|
		l.authInfo = authInfo
 | 
						|
		l.w.Header().Del("GAP-Auth")
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
// Write writes the response using the ResponseWriter
 | 
						|
func (l *responseLogger) Write(b []byte) (int, error) {
 | 
						|
	if l.status == 0 {
 | 
						|
		// The status will be StatusOK if WriteHeader has not been called yet
 | 
						|
		l.status = http.StatusOK
 | 
						|
	}
 | 
						|
	l.ExtractGAPMetadata()
 | 
						|
	size, err := l.w.Write(b)
 | 
						|
	l.size += size
 | 
						|
	return size, err
 | 
						|
}
 | 
						|
 | 
						|
// WriteHeader writes the status code for the Response
 | 
						|
func (l *responseLogger) WriteHeader(s int) {
 | 
						|
	l.ExtractGAPMetadata()
 | 
						|
	l.w.WriteHeader(s)
 | 
						|
	l.status = s
 | 
						|
}
 | 
						|
 | 
						|
// Status returns the response status code
 | 
						|
func (l *responseLogger) Status() int {
 | 
						|
	return l.status
 | 
						|
}
 | 
						|
 | 
						|
// Size returns the response size
 | 
						|
func (l *responseLogger) Size() int {
 | 
						|
	return l.size
 | 
						|
}
 | 
						|
 | 
						|
// Flush sends any buffered data to the client
 | 
						|
func (l *responseLogger) Flush() {
 | 
						|
	if flusher, ok := l.w.(http.Flusher); ok {
 | 
						|
		flusher.Flush()
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
// loggingHandler is the http.Handler implementation for LoggingHandler
 | 
						|
type loggingHandler struct {
 | 
						|
	handler http.Handler
 | 
						|
	logger  *log.Entry
 | 
						|
}
 | 
						|
 | 
						|
// LoggingHandler provides an http.Handler which logs requests to the HTTP server
 | 
						|
func LoggingHandler(h http.Handler) http.Handler {
 | 
						|
	return loggingHandler{
 | 
						|
		handler: h,
 | 
						|
		logger:  log.WithField("logger", "authentik.outpost.proxy-http-server"),
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
func (h loggingHandler) ServeHTTP(w http.ResponseWriter, req *http.Request) {
 | 
						|
	t := time.Now()
 | 
						|
	url := *req.URL
 | 
						|
	responseLogger := &responseLogger{w: w}
 | 
						|
	h.handler.ServeHTTP(responseLogger, req)
 | 
						|
	duration := float64(time.Since(t)) / float64(time.Millisecond)
 | 
						|
	h.logger.WithFields(log.Fields{
 | 
						|
		"host":              req.RemoteAddr,
 | 
						|
		"vhost":             web.GetHost(req),
 | 
						|
		"request_protocol":  req.Proto,
 | 
						|
		"runtime":           fmt.Sprintf("%0.3f", duration),
 | 
						|
		"method":            req.Method,
 | 
						|
		"size":              responseLogger.Size(),
 | 
						|
		"status":            responseLogger.Status(),
 | 
						|
		"upstream":          responseLogger.upstream,
 | 
						|
		"request_useragent": req.UserAgent(),
 | 
						|
		"request_username":  responseLogger.authInfo,
 | 
						|
	}).Info(url.RequestURI())
 | 
						|
}
 |