@@ -21,6 +21,7 @@ import (
21
21
mathrand "math/rand"
22
22
"net"
23
23
"net/http"
24
+ "net/textproto"
24
25
"sort"
25
26
"strconv"
26
27
"strings"
@@ -212,9 +213,10 @@ type clientStream struct {
212
213
done chan struct {} // closed when stream remove from cc.streams map; close calls guarded by cc.mu
213
214
214
215
// owned by clientConnReadLoop:
215
- firstByte bool // got the first response byte
216
- pastHeaders bool // got first MetaHeadersFrame (actual headers)
217
- pastTrailers bool // got optional second MetaHeadersFrame (trailers)
216
+ firstByte bool // got the first response byte
217
+ pastHeaders bool // got first MetaHeadersFrame (actual headers)
218
+ pastTrailers bool // got optional second MetaHeadersFrame (trailers)
219
+ num1xx uint8 // number of 1xx responses seen
218
220
219
221
trailer http.Header // accumulated trailers
220
222
resTrailer * http.Header // client's Response.Trailer
@@ -238,6 +240,17 @@ func awaitRequestCancel(req *http.Request, done <-chan struct{}) error {
238
240
}
239
241
}
240
242
243
+ var got1xxFuncForTests func (int , textproto.MIMEHeader ) error
244
+
245
+ // get1xxTraceFunc returns the value of request's httptrace.ClientTrace.Got1xxResponse func,
246
+ // if any. It returns nil if not set or if the Go version is too old.
247
+ func (cs * clientStream ) get1xxTraceFunc () func (int , textproto.MIMEHeader ) error {
248
+ if fn := got1xxFuncForTests ; fn != nil {
249
+ return fn
250
+ }
251
+ return traceGot1xxResponseFunc (cs .trace )
252
+ }
253
+
241
254
// awaitRequestCancel waits for the user to cancel a request, its context to
242
255
// expire, or for the request to be done (any way it might be removed from the
243
256
// cc.streams map: peer reset, successful completion, TCP connection breakage,
@@ -1734,8 +1747,7 @@ func (rl *clientConnReadLoop) processHeaders(f *MetaHeadersFrame) error {
1734
1747
// is the detail.
1735
1748
//
1736
1749
// As a special case, handleResponse may return (nil, nil) to skip the
1737
- // frame (currently only used for 100 expect continue). This special
1738
- // case is going away after Issue 13851 is fixed.
1750
+ // frame (currently only used for 1xx responses).
1739
1751
func (rl * clientConnReadLoop ) handleResponse (cs * clientStream , f * MetaHeadersFrame ) (* http.Response , error ) {
1740
1752
if f .Truncated {
1741
1753
return nil , errResponseHeaderListSize
@@ -1750,37 +1762,54 @@ func (rl *clientConnReadLoop) handleResponse(cs *clientStream, f *MetaHeadersFra
1750
1762
return nil , errors .New ("malformed response from server: malformed non-numeric status pseudo header" )
1751
1763
}
1752
1764
1753
- if statusCode == 100 {
1754
- traceGot100Continue (cs .trace )
1755
- if cs .on100 != nil {
1756
- cs .on100 () // forces any write delay timer to fire
1765
+ header := make (http.Header )
1766
+ var trailerValue string
1767
+ for _ , hf := range f .RegularFields () {
1768
+ key := http .CanonicalHeaderKey (hf .Name )
1769
+ if key == "Trailer" {
1770
+ trailerValue = hf .Value
1771
+ } else {
1772
+ header [key ] = append (header [key ], hf .Value )
1773
+ }
1774
+ }
1775
+
1776
+ if statusCode >= 100 && statusCode <= 199 {
1777
+ cs .num1xx ++
1778
+ const max1xxResponses = 5 // arbitrary bound on number of informational responses, same as net/http
1779
+ if cs .num1xx > max1xxResponses {
1780
+ return nil , errors .New ("http2: too many 1xx informational responses" )
1781
+ }
1782
+ if fn := cs .get1xxTraceFunc (); fn != nil {
1783
+ if err := fn (statusCode , textproto .MIMEHeader (header )); err != nil {
1784
+ return nil , err
1785
+ }
1786
+ }
1787
+ if statusCode == 100 {
1788
+ traceGot100Continue (cs .trace )
1789
+ if cs .on100 != nil {
1790
+ cs .on100 () // forces any write delay timer to fire
1791
+ }
1757
1792
}
1758
1793
cs .pastHeaders = false // do it all again
1759
1794
return nil , nil
1760
1795
}
1761
1796
1762
- header := make (http.Header )
1763
1797
res := & http.Response {
1764
1798
Proto : "HTTP/2.0" ,
1765
1799
ProtoMajor : 2 ,
1766
1800
Header : header ,
1767
1801
StatusCode : statusCode ,
1768
1802
Status : status + " " + http .StatusText (statusCode ),
1769
1803
}
1770
- for _ , hf := range f .RegularFields () {
1771
- key := http .CanonicalHeaderKey (hf .Name )
1772
- if key == "Trailer" {
1773
- t := res .Trailer
1774
- if t == nil {
1775
- t = make (http.Header )
1776
- res .Trailer = t
1777
- }
1778
- foreachHeaderElement (hf .Value , func (v string ) {
1779
- t [http .CanonicalHeaderKey (v )] = nil
1780
- })
1781
- } else {
1782
- header [key ] = append (header [key ], hf .Value )
1804
+ if trailerValue != "" {
1805
+ t := res .Trailer
1806
+ if t == nil {
1807
+ t = make (http.Header )
1808
+ res .Trailer = t
1783
1809
}
1810
+ foreachHeaderElement (trailerValue , func (v string ) {
1811
+ t [http .CanonicalHeaderKey (v )] = nil
1812
+ })
1784
1813
}
1785
1814
1786
1815
streamEnded := f .StreamEnded ()
0 commit comments