Source file src/net/http/httputil/dump.go

     1  // Copyright 2009 The Go Authors. All rights reserved.
     2  // Use of this source code is governed by a BSD-style
     3  // license that can be found in the LICENSE file.
     4  
     5  package httputil
     6  
     7  import (
     8  	"bufio"
     9  	"bytes"
    10  	"errors"
    11  	"fmt"
    12  	"io"
    13  	"net"
    14  	"net/http"
    15  	"net/url"
    16  	"strings"
    17  	"time"
    18  )
    19  
    20  // drainBody reads all of b to memory and then returns two equivalent
    21  // ReadClosers yielding the same bytes.
    22  //
    23  // It returns an error if the initial slurp of all bytes fails. It does not attempt
    24  // to make the returned ReadClosers have identical error-matching behavior.
    25  func drainBody(b io.ReadCloser) (r1, r2 io.ReadCloser, err error) {
    26  	if b == nil || b == http.NoBody {
    27  		// No copying needed. Preserve the magic sentinel meaning of NoBody.
    28  		return http.NoBody, http.NoBody, nil
    29  	}
    30  	var buf bytes.Buffer
    31  	if _, err = buf.ReadFrom(b); err != nil {
    32  		return nil, b, err
    33  	}
    34  	if err = b.Close(); err != nil {
    35  		return nil, b, err
    36  	}
    37  	return io.NopCloser(&buf), io.NopCloser(bytes.NewReader(buf.Bytes())), nil
    38  }
    39  
    40  // dumpConn is a net.Conn which writes to Writer and reads from Reader
    41  type dumpConn struct {
    42  	io.Writer
    43  	io.Reader
    44  }
    45  
    46  func (c *dumpConn) Close() error                       { return nil }
    47  func (c *dumpConn) LocalAddr() net.Addr                { return nil }
    48  func (c *dumpConn) RemoteAddr() net.Addr               { return nil }
    49  func (c *dumpConn) SetDeadline(t time.Time) error      { return nil }
    50  func (c *dumpConn) SetReadDeadline(t time.Time) error  { return nil }
    51  func (c *dumpConn) SetWriteDeadline(t time.Time) error { return nil }
    52  
    53  type neverEnding byte
    54  
    55  func (b neverEnding) Read(p []byte) (n int, err error) {
    56  	for i := range p {
    57  		p[i] = byte(b)
    58  	}
    59  	return len(p), nil
    60  }
    61  
    62  // outgoingLength is a copy of the unexported
    63  // (*http.Request).outgoingLength method.
    64  func outgoingLength(req *http.Request) int64 {
    65  	if req.Body == nil || req.Body == http.NoBody {
    66  		return 0
    67  	}
    68  	if req.ContentLength != 0 {
    69  		return req.ContentLength
    70  	}
    71  	return -1
    72  }
    73  
    74  // DumpRequestOut is like [DumpRequest] but for outgoing client requests. It
    75  // includes any headers that the standard [http.Transport] adds, such as
    76  // User-Agent.
    77  func DumpRequestOut(req *http.Request, body bool) ([]byte, error) {
    78  	save := req.Body
    79  	dummyBody := false
    80  	if !body {
    81  		contentLength := outgoingLength(req)
    82  		if contentLength != 0 {
    83  			req.Body = io.NopCloser(io.LimitReader(neverEnding('x'), contentLength))
    84  			dummyBody = true
    85  		}
    86  	} else {
    87  		var err error
    88  		save, req.Body, err = drainBody(req.Body)
    89  		if err != nil {
    90  			return nil, err
    91  		}
    92  	}
    93  
    94  	// Since we're using the actual Transport code to write the request,
    95  	// switch to http so the Transport doesn't try to do an SSL
    96  	// negotiation with our dumpConn and its bytes.Buffer & pipe.
    97  	// The wire format for https and http are the same, anyway.
    98  	reqSend := req
    99  	if req.URL.Scheme == "https" {
   100  		reqSend = new(http.Request)
   101  		*reqSend = *req
   102  		reqSend.URL = new(url.URL)
   103  		*reqSend.URL = *req.URL
   104  		reqSend.URL.Scheme = "http"
   105  	}
   106  
   107  	// Use the actual Transport code to record what we would send
   108  	// on the wire, but not using TCP.  Use a Transport with a
   109  	// custom dialer that returns a fake net.Conn that waits
   110  	// for the full input (and recording it), and then responds
   111  	// with a dummy response.
   112  	var buf bytes.Buffer // records the output
   113  	pr, pw := io.Pipe()
   114  	defer pr.Close()
   115  	defer pw.Close()
   116  	dr := &delegateReader{c: make(chan io.Reader)}
   117  
   118  	t := &http.Transport{
   119  		Dial: func(net, addr string) (net.Conn, error) {
   120  			return &dumpConn{io.MultiWriter(&buf, pw), dr}, nil
   121  		},
   122  	}
   123  	defer t.CloseIdleConnections()
   124  
   125  	// We need this channel to ensure that the reader
   126  	// goroutine exits if t.RoundTrip returns an error.
   127  	// See golang.org/issue/32571.
   128  	quitReadCh := make(chan struct{})
   129  	// Wait for the request before replying with a dummy response:
   130  	go func() {
   131  		req, err := http.ReadRequest(bufio.NewReader(pr))
   132  		if err == nil {
   133  			// Ensure all the body is read; otherwise
   134  			// we'll get a partial dump.
   135  			io.Copy(io.Discard, req.Body)
   136  			req.Body.Close()
   137  		}
   138  		select {
   139  		case dr.c <- strings.NewReader("HTTP/1.1 204 No Content\r\nConnection: close\r\n\r\n"):
   140  		case <-quitReadCh:
   141  			// Ensure delegateReader.Read doesn't block forever if we get an error.
   142  			close(dr.c)
   143  		}
   144  	}()
   145  
   146  	_, err := t.RoundTrip(reqSend)
   147  
   148  	req.Body = save
   149  	if err != nil {
   150  		dr.err = err
   151  		close(quitReadCh)
   152  		return nil, err
   153  	}
   154  	dump := buf.Bytes()
   155  
   156  	// If we used a dummy body above, remove it now.
   157  	// TODO: if the req.ContentLength is large, we allocate memory
   158  	// unnecessarily just to slice it off here. But this is just
   159  	// a debug function, so this is acceptable for now. We could
   160  	// discard the body earlier if this matters.
   161  	if dummyBody {
   162  		if i := bytes.Index(dump, []byte("\r\n\r\n")); i >= 0 {
   163  			dump = dump[:i+4]
   164  		}
   165  	}
   166  	return dump, nil
   167  }
   168  
   169  // delegateReader is a reader that delegates to another reader,
   170  // once it arrives on a channel.
   171  type delegateReader struct {
   172  	c   chan io.Reader
   173  	err error     // only used if r is nil and c is closed.
   174  	r   io.Reader // nil until received from c
   175  }
   176  
   177  func (r *delegateReader) Read(p []byte) (int, error) {
   178  	if r.r == nil {
   179  		var ok bool
   180  		if r.r, ok = <-r.c; !ok {
   181  			return 0, r.err
   182  		}
   183  	}
   184  	return r.r.Read(p)
   185  }
   186  
   187  // Return value if nonempty, def otherwise.
   188  func valueOrDefault(value, def string) string {
   189  	if value != "" {
   190  		return value
   191  	}
   192  	return def
   193  }
   194  
   195  var reqWriteExcludeHeaderDump = map[string]bool{
   196  	"Host":              true, // not in Header map anyway
   197  	"Transfer-Encoding": true,
   198  	"Trailer":           true,
   199  }
   200  
   201  // DumpRequest returns the given request in its HTTP/1.x wire
   202  // representation. It should only be used by servers to debug client
   203  // requests. The returned representation is an approximation only;
   204  // some details of the initial request are lost while parsing it into
   205  // an [http.Request]. In particular, the order and case of header field
   206  // names are lost. The order of values in multi-valued headers is kept
   207  // intact. HTTP/2 requests are dumped in HTTP/1.x form, not in their
   208  // original binary representations.
   209  //
   210  // If body is true, DumpRequest also returns the body. To do so, it
   211  // consumes req.Body and then replaces it with a new [io.ReadCloser]
   212  // that yields the same bytes. If DumpRequest returns an error,
   213  // the state of req is undefined.
   214  //
   215  // The documentation for [http.Request.Write] details which fields
   216  // of req are included in the dump.
   217  func DumpRequest(req *http.Request, body bool) ([]byte, error) {
   218  	var err error
   219  	save := req.Body
   220  	if !body || req.Body == nil {
   221  		req.Body = nil
   222  	} else {
   223  		save, req.Body, err = drainBody(req.Body)
   224  		if err != nil {
   225  			return nil, err
   226  		}
   227  	}
   228  
   229  	var b bytes.Buffer
   230  
   231  	// By default, print out the unmodified req.RequestURI, which
   232  	// is always set for incoming server requests. But because we
   233  	// previously used req.URL.RequestURI and the docs weren't
   234  	// always so clear about when to use DumpRequest vs
   235  	// DumpRequestOut, fall back to the old way if the caller
   236  	// provides a non-server Request.
   237  	reqURI := req.RequestURI
   238  	if reqURI == "" {
   239  		reqURI = req.URL.RequestURI()
   240  	}
   241  
   242  	fmt.Fprintf(&b, "%s %s HTTP/%d.%d\r\n", valueOrDefault(req.Method, "GET"),
   243  		reqURI, req.ProtoMajor, req.ProtoMinor)
   244  
   245  	absRequestURI := strings.HasPrefix(req.RequestURI, "http://") || strings.HasPrefix(req.RequestURI, "https://")
   246  	if !absRequestURI {
   247  		host := req.Host
   248  		if host == "" && req.URL != nil {
   249  			host = req.URL.Host
   250  		}
   251  		if host != "" {
   252  			fmt.Fprintf(&b, "Host: %s\r\n", host)
   253  		}
   254  	}
   255  
   256  	chunked := len(req.TransferEncoding) > 0 && req.TransferEncoding[0] == "chunked"
   257  	if len(req.TransferEncoding) > 0 {
   258  		fmt.Fprintf(&b, "Transfer-Encoding: %s\r\n", strings.Join(req.TransferEncoding, ","))
   259  	}
   260  
   261  	err = req.Header.WriteSubset(&b, reqWriteExcludeHeaderDump)
   262  	if err != nil {
   263  		return nil, err
   264  	}
   265  
   266  	io.WriteString(&b, "\r\n")
   267  
   268  	if req.Body != nil {
   269  		var dest io.Writer = &b
   270  		if chunked {
   271  			dest = NewChunkedWriter(dest)
   272  		}
   273  		_, err = io.Copy(dest, req.Body)
   274  		if chunked {
   275  			dest.(io.Closer).Close()
   276  			io.WriteString(&b, "\r\n")
   277  		}
   278  	}
   279  
   280  	req.Body = save
   281  	if err != nil {
   282  		return nil, err
   283  	}
   284  	return b.Bytes(), nil
   285  }
   286  
   287  // errNoBody is a sentinel error value used by failureToReadBody so we
   288  // can detect that the lack of body was intentional.
   289  var errNoBody = errors.New("sentinel error value")
   290  
   291  // failureToReadBody is an io.ReadCloser that just returns errNoBody on
   292  // Read. It's swapped in when we don't actually want to consume
   293  // the body, but need a non-nil one, and want to distinguish the
   294  // error from reading the dummy body.
   295  type failureToReadBody struct{}
   296  
   297  func (failureToReadBody) Read([]byte) (int, error) { return 0, errNoBody }
   298  func (failureToReadBody) Close() error             { return nil }
   299  
   300  // emptyBody is an instance of empty reader.
   301  var emptyBody = io.NopCloser(strings.NewReader(""))
   302  
   303  // DumpResponse is like DumpRequest but dumps a response.
   304  func DumpResponse(resp *http.Response, body bool) ([]byte, error) {
   305  	var b bytes.Buffer
   306  	var err error
   307  	save := resp.Body
   308  	savecl := resp.ContentLength
   309  
   310  	if !body {
   311  		// For content length of zero. Make sure the body is an empty
   312  		// reader, instead of returning error through failureToReadBody{}.
   313  		if resp.ContentLength == 0 {
   314  			resp.Body = emptyBody
   315  		} else {
   316  			resp.Body = failureToReadBody{}
   317  		}
   318  	} else if resp.Body == nil {
   319  		resp.Body = emptyBody
   320  	} else {
   321  		save, resp.Body, err = drainBody(resp.Body)
   322  		if err != nil {
   323  			return nil, err
   324  		}
   325  	}
   326  	err = resp.Write(&b)
   327  	if err == errNoBody {
   328  		err = nil
   329  	}
   330  	resp.Body = save
   331  	resp.ContentLength = savecl
   332  	if err != nil {
   333  		return nil, err
   334  	}
   335  	return b.Bytes(), nil
   336  }
   337  

View as plain text