mirror of
https://github.com/caddyserver/caddy.git
synced 2024-12-30 22:34:15 -05:00
reverseproxy: Incorporate latest proxy changes from stdlib (#4266)
I went through the commits that touched stdlib's `reverseproxy.go` file, and copied over all the changes that are to code that was copied into Caddy. The commits I pulled changes from: -2cc347382f
-a5cea062b3
-ecdbffd4ec
-21898524f6
-ca3c0df1f8
-9c017ff30d
This may also fix https://github.com/caddyserver/caddy/issues/4247 because of the change to `copyResponse` to set `mlw.flushPending = true` right away.
This commit is contained in:
parent
68c5c71659
commit
e6c29ce081
2 changed files with 37 additions and 15 deletions
|
@ -23,6 +23,7 @@ import (
|
||||||
"io"
|
"io"
|
||||||
"net"
|
"net"
|
||||||
"net/http"
|
"net/http"
|
||||||
|
"net/textproto"
|
||||||
"net/url"
|
"net/url"
|
||||||
"regexp"
|
"regexp"
|
||||||
"strconv"
|
"strconv"
|
||||||
|
@ -80,10 +81,13 @@ type Handler struct {
|
||||||
// Upstreams is the list of backends to proxy to.
|
// Upstreams is the list of backends to proxy to.
|
||||||
Upstreams UpstreamPool `json:"upstreams,omitempty"`
|
Upstreams UpstreamPool `json:"upstreams,omitempty"`
|
||||||
|
|
||||||
// Adjusts how often to flush the response buffer. A
|
// Adjusts how often to flush the response buffer. By default,
|
||||||
// negative value disables response buffering.
|
// no periodic flushing is done. A negative value disables
|
||||||
// TODO: figure out good defaults and write docs for this
|
// response buffering, and flushes immediately after each
|
||||||
// (see https://github.com/caddyserver/caddy/issues/1460)
|
// write to the client. This option is ignored when the upstream's
|
||||||
|
// response is recognized as a streaming response, or if its
|
||||||
|
// content length is -1; for such responses, writes are flushed
|
||||||
|
// to the client immediately.
|
||||||
FlushInterval caddy.Duration `json:"flush_interval,omitempty"`
|
FlushInterval caddy.Duration `json:"flush_interval,omitempty"`
|
||||||
|
|
||||||
// Headers manipulates headers between Caddy and the backend.
|
// Headers manipulates headers between Caddy and the backend.
|
||||||
|
@ -528,13 +532,19 @@ func (h Handler) prepareRequest(req *http.Request) error {
|
||||||
// If we aren't the first proxy retain prior
|
// If we aren't the first proxy retain prior
|
||||||
// X-Forwarded-For information as a comma+space
|
// X-Forwarded-For information as a comma+space
|
||||||
// separated list and fold multiple headers into one.
|
// separated list and fold multiple headers into one.
|
||||||
if prior, ok := req.Header["X-Forwarded-For"]; ok {
|
prior, ok := req.Header["X-Forwarded-For"]
|
||||||
|
omit := ok && prior == nil // Issue 38079: nil now means don't populate the header
|
||||||
|
if len(prior) > 0 {
|
||||||
clientIP = strings.Join(prior, ", ") + ", " + clientIP
|
clientIP = strings.Join(prior, ", ") + ", " + clientIP
|
||||||
}
|
}
|
||||||
|
if !omit {
|
||||||
req.Header.Set("X-Forwarded-For", clientIP)
|
req.Header.Set("X-Forwarded-For", clientIP)
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if req.Header.Get("X-Forwarded-Proto") == "" {
|
prior, ok := req.Header["X-Forwarded-Proto"]
|
||||||
|
omit := ok && prior == nil
|
||||||
|
if len(prior) == 0 && !omit {
|
||||||
// set X-Forwarded-Proto; many backend apps expect this too
|
// set X-Forwarded-Proto; many backend apps expect this too
|
||||||
proto := "https"
|
proto := "https"
|
||||||
if req.TLS == nil {
|
if req.TLS == nil {
|
||||||
|
@ -827,10 +837,10 @@ func upgradeType(h http.Header) string {
|
||||||
// removeConnectionHeaders removes hop-by-hop headers listed in the "Connection" header of h.
|
// removeConnectionHeaders removes hop-by-hop headers listed in the "Connection" header of h.
|
||||||
// See RFC 7230, section 6.1
|
// See RFC 7230, section 6.1
|
||||||
func removeConnectionHeaders(h http.Header) {
|
func removeConnectionHeaders(h http.Header) {
|
||||||
if c := h.Get("Connection"); c != "" {
|
for _, f := range h["Connection"] {
|
||||||
for _, f := range strings.Split(c, ",") {
|
for _, sf := range strings.Split(f, ",") {
|
||||||
if f = strings.TrimSpace(f); f != "" {
|
if sf = textproto.TrimString(sf); sf != "" {
|
||||||
h.Del(f)
|
h.Del(sf)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -32,6 +32,9 @@ import (
|
||||||
func (h Handler) handleUpgradeResponse(logger *zap.Logger, rw http.ResponseWriter, req *http.Request, res *http.Response) {
|
func (h Handler) handleUpgradeResponse(logger *zap.Logger, rw http.ResponseWriter, req *http.Request, res *http.Response) {
|
||||||
reqUpType := upgradeType(req.Header)
|
reqUpType := upgradeType(req.Header)
|
||||||
resUpType := upgradeType(res.Header)
|
resUpType := upgradeType(res.Header)
|
||||||
|
// TODO: Update to use "net/http/internal/ascii" once we bumped
|
||||||
|
// the minimum Go version to 1.17.
|
||||||
|
// See https://github.com/golang/go/commit/5c489514bc5e61ad9b5b07bd7d8ec65d66a0512a
|
||||||
if reqUpType != resUpType {
|
if reqUpType != resUpType {
|
||||||
h.logger.Debug("backend tried to switch to unexpected protocol via Upgrade header",
|
h.logger.Debug("backend tried to switch to unexpected protocol via Upgrade header",
|
||||||
zap.String("backend_upgrade", resUpType),
|
zap.String("backend_upgrade", resUpType),
|
||||||
|
@ -39,8 +42,6 @@ func (h Handler) handleUpgradeResponse(logger *zap.Logger, rw http.ResponseWrite
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
copyHeader(res.Header, rw.Header())
|
|
||||||
|
|
||||||
hj, ok := rw.(http.Hijacker)
|
hj, ok := rw.(http.Hijacker)
|
||||||
if !ok {
|
if !ok {
|
||||||
h.logger.Sugar().Errorf("can't switch protocols using non-Hijacker ResponseWriter type %T", rw)
|
h.logger.Sugar().Errorf("can't switch protocols using non-Hijacker ResponseWriter type %T", rw)
|
||||||
|
@ -78,6 +79,9 @@ func (h Handler) handleUpgradeResponse(logger *zap.Logger, rw http.ResponseWrite
|
||||||
logger.Debug("connection closed", zap.Duration("duration", time.Since(start)))
|
logger.Debug("connection closed", zap.Duration("duration", time.Since(start)))
|
||||||
}()
|
}()
|
||||||
|
|
||||||
|
copyHeader(rw.Header(), res.Header)
|
||||||
|
|
||||||
|
res.Header = rw.Header()
|
||||||
res.Body = nil // so res.Write only writes the headers; we have res.Body in backConn above
|
res.Body = nil // so res.Write only writes the headers; we have res.Body in backConn above
|
||||||
if err := res.Write(brw); err != nil {
|
if err := res.Write(brw); err != nil {
|
||||||
h.logger.Debug("response write", zap.Error(err))
|
h.logger.Debug("response write", zap.Error(err))
|
||||||
|
@ -107,13 +111,16 @@ func (h Handler) flushInterval(req *http.Request, res *http.Response) time.Durat
|
||||||
return -1 // negative means immediately
|
return -1 // negative means immediately
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// We might have the case of streaming for which Content-Length might be unset.
|
||||||
|
if res.ContentLength == -1 {
|
||||||
|
return -1
|
||||||
|
}
|
||||||
|
|
||||||
// for h2 and h2c upstream streaming data to client (issues #3556 and #3606)
|
// for h2 and h2c upstream streaming data to client (issues #3556 and #3606)
|
||||||
if h.isBidirectionalStream(req, res) {
|
if h.isBidirectionalStream(req, res) {
|
||||||
return -1
|
return -1
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO: more specific cases? e.g. res.ContentLength == -1? (this TODO is from the std lib, but
|
|
||||||
// strangely similar to our isBidirectionalStream function that we implemented ourselves)
|
|
||||||
return time.Duration(h.FlushInterval)
|
return time.Duration(h.FlushInterval)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -142,6 +149,11 @@ func (h Handler) copyResponse(dst io.Writer, src io.Reader, flushInterval time.D
|
||||||
latency: flushInterval,
|
latency: flushInterval,
|
||||||
}
|
}
|
||||||
defer mlw.stop()
|
defer mlw.stop()
|
||||||
|
|
||||||
|
// set up initial timer so headers get flushed even if body writes are delayed
|
||||||
|
mlw.flushPending = true
|
||||||
|
mlw.t = time.AfterFunc(flushInterval, mlw.delayedFlush)
|
||||||
|
|
||||||
dst = mlw
|
dst = mlw
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue