2019-03-14 09:30:04 +01:00
|
|
|
package service
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2020-11-06 09:26:03 +01:00
|
|
|
"errors"
|
2019-03-14 09:30:04 +01:00
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"net"
|
|
|
|
"net/http"
|
|
|
|
"net/http/httputil"
|
|
|
|
"net/url"
|
2022-06-27 15:16:08 +02:00
|
|
|
"strings"
|
2019-03-14 09:30:04 +01:00
|
|
|
"time"
|
|
|
|
|
2020-08-17 18:04:03 +02:00
|
|
|
ptypes "github.com/traefik/paerser/types"
|
2020-09-16 15:46:04 +02:00
|
|
|
"github.com/traefik/traefik/v2/pkg/config/dynamic"
|
|
|
|
"github.com/traefik/traefik/v2/pkg/log"
|
2022-09-22 10:00:09 +02:00
|
|
|
"golang.org/x/net/http/httpguts"
|
2019-03-14 09:30:04 +01:00
|
|
|
)
|
|
|
|
|
2020-05-11 12:06:07 +02:00
|
|
|
// StatusClientClosedRequest non-standard HTTP status code for client disconnection.
|
2019-03-14 09:30:04 +01:00
|
|
|
const StatusClientClosedRequest = 499
|
|
|
|
|
2020-05-11 12:06:07 +02:00
|
|
|
// StatusClientClosedRequestText non-standard HTTP status for client disconnection.
|
2019-03-14 09:30:04 +01:00
|
|
|
const StatusClientClosedRequestText = "Client Closed Request"
|
|
|
|
|
2020-09-11 15:40:03 +02:00
|
|
|
func buildProxy(passHostHeader *bool, responseForwarding *dynamic.ResponseForwarding, roundTripper http.RoundTripper, bufferPool httputil.BufferPool) (http.Handler, error) {
|
2020-08-17 18:04:03 +02:00
|
|
|
var flushInterval ptypes.Duration
|
2019-03-14 09:30:04 +01:00
|
|
|
if responseForwarding != nil {
|
|
|
|
err := flushInterval.Set(responseForwarding.FlushInterval)
|
|
|
|
if err != nil {
|
2020-05-11 12:06:07 +02:00
|
|
|
return nil, fmt.Errorf("error creating flush interval: %w", err)
|
2019-03-14 09:30:04 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if flushInterval == 0 {
|
2020-08-17 18:04:03 +02:00
|
|
|
flushInterval = ptypes.Duration(100 * time.Millisecond)
|
2019-03-14 09:30:04 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
proxy := &httputil.ReverseProxy{
|
|
|
|
Director: func(outReq *http.Request) {
|
|
|
|
u := outReq.URL
|
|
|
|
if outReq.RequestURI != "" {
|
|
|
|
parsedURL, err := url.ParseRequestURI(outReq.RequestURI)
|
|
|
|
if err == nil {
|
|
|
|
u = parsedURL
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
outReq.URL.Path = u.Path
|
|
|
|
outReq.URL.RawPath = u.RawPath
|
2022-06-27 15:16:08 +02:00
|
|
|
outReq.URL.RawQuery = strings.ReplaceAll(u.RawQuery, ";", "&")
|
2019-03-14 09:30:04 +01:00
|
|
|
outReq.RequestURI = "" // Outgoing request should not have RequestURI
|
|
|
|
|
|
|
|
outReq.Proto = "HTTP/1.1"
|
|
|
|
outReq.ProtoMajor = 1
|
|
|
|
outReq.ProtoMinor = 1
|
|
|
|
|
|
|
|
// Do not pass client Host header unless optsetter PassHostHeader is set.
|
2019-09-30 18:12:04 +02:00
|
|
|
if passHostHeader != nil && !*passHostHeader {
|
2019-03-14 09:30:04 +01:00
|
|
|
outReq.Host = outReq.URL.Host
|
|
|
|
}
|
|
|
|
|
2019-09-17 16:12:04 +02:00
|
|
|
// Even if the websocket RFC says that headers should be case-insensitive,
|
2020-04-27 18:12:04 +02:00
|
|
|
// some servers need Sec-WebSocket-Key, Sec-WebSocket-Extensions, Sec-WebSocket-Accept,
|
|
|
|
// Sec-WebSocket-Protocol and Sec-WebSocket-Version to be case-sensitive.
|
2019-09-17 16:12:04 +02:00
|
|
|
// https://tools.ietf.org/html/rfc6455#page-20
|
2022-09-22 10:00:09 +02:00
|
|
|
if isWebSocketUpgrade(outReq) {
|
|
|
|
outReq.Header["Sec-WebSocket-Key"] = outReq.Header["Sec-Websocket-Key"]
|
|
|
|
outReq.Header["Sec-WebSocket-Extensions"] = outReq.Header["Sec-Websocket-Extensions"]
|
|
|
|
outReq.Header["Sec-WebSocket-Accept"] = outReq.Header["Sec-Websocket-Accept"]
|
|
|
|
outReq.Header["Sec-WebSocket-Protocol"] = outReq.Header["Sec-Websocket-Protocol"]
|
|
|
|
outReq.Header["Sec-WebSocket-Version"] = outReq.Header["Sec-Websocket-Version"]
|
|
|
|
delete(outReq.Header, "Sec-Websocket-Key")
|
|
|
|
delete(outReq.Header, "Sec-Websocket-Extensions")
|
|
|
|
delete(outReq.Header, "Sec-Websocket-Accept")
|
|
|
|
delete(outReq.Header, "Sec-Websocket-Protocol")
|
|
|
|
delete(outReq.Header, "Sec-Websocket-Version")
|
|
|
|
}
|
2019-03-14 09:30:04 +01:00
|
|
|
},
|
2020-09-11 15:40:03 +02:00
|
|
|
Transport: roundTripper,
|
2020-09-01 18:16:04 +02:00
|
|
|
FlushInterval: time.Duration(flushInterval),
|
|
|
|
BufferPool: bufferPool,
|
2019-03-14 09:30:04 +01:00
|
|
|
ErrorHandler: func(w http.ResponseWriter, request *http.Request, err error) {
|
|
|
|
statusCode := http.StatusInternalServerError
|
|
|
|
|
|
|
|
switch {
|
2020-11-06 09:26:03 +01:00
|
|
|
case errors.Is(err, io.EOF):
|
2019-03-14 09:30:04 +01:00
|
|
|
statusCode = http.StatusBadGateway
|
2020-11-06 09:26:03 +01:00
|
|
|
case errors.Is(err, context.Canceled):
|
2019-03-14 09:30:04 +01:00
|
|
|
statusCode = StatusClientClosedRequest
|
|
|
|
default:
|
2020-11-06 09:26:03 +01:00
|
|
|
var netErr net.Error
|
|
|
|
if errors.As(err, &netErr) {
|
|
|
|
if netErr.Timeout() {
|
2019-03-14 09:30:04 +01:00
|
|
|
statusCode = http.StatusGatewayTimeout
|
|
|
|
} else {
|
|
|
|
statusCode = http.StatusBadGateway
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
log.Debugf("'%d %s' caused by: %v", statusCode, statusText(statusCode), err)
|
|
|
|
w.WriteHeader(statusCode)
|
|
|
|
_, werr := w.Write([]byte(statusText(statusCode)))
|
|
|
|
if werr != nil {
|
|
|
|
log.Debugf("Error while writing status code", werr)
|
|
|
|
}
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
return proxy, nil
|
|
|
|
}
|
|
|
|
|
2022-09-22 10:00:09 +02:00
|
|
|
func isWebSocketUpgrade(req *http.Request) bool {
|
|
|
|
if !httpguts.HeaderValuesContainsToken(req.Header["Connection"], "Upgrade") {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
return strings.EqualFold(req.Header.Get("Upgrade"), "websocket")
|
|
|
|
}
|
|
|
|
|
2019-03-14 09:30:04 +01:00
|
|
|
func statusText(statusCode int) string {
|
|
|
|
if statusCode == StatusClientClosedRequest {
|
|
|
|
return StatusClientClosedRequestText
|
|
|
|
}
|
|
|
|
return http.StatusText(statusCode)
|
|
|
|
}
|