mirror of
				https://github.com/caddyserver/caddy.git
				synced 2025-10-25 15:52:45 -04: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-9c017ff30dThis 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
					
				| @ -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…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user