Source File
request.go
Belonging Package
net/http
// Copyright 2009 The Go Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.
// HTTP Request reading and parsing.
package http
import (
urlpkg
_ // for linkname
)
const (
defaultMaxMemory = 32 << 20 // 32 MB
)
// ErrMissingFile is returned by FormFile when the provided file field name
// is either not present in the request or not a file field.
var ErrMissingFile = errors.New("http: no such file")
// ProtocolError represents an HTTP protocol error.
//
// Deprecated: Not all errors in the http package related to protocol errors
// are of type ProtocolError.
type ProtocolError struct {
ErrorString string
}
func ( *ProtocolError) () string { return .ErrorString }
// Is lets http.ErrNotSupported match errors.ErrUnsupported.
func ( *ProtocolError) ( error) bool {
return == ErrNotSupported && == errors.ErrUnsupported
}
var (
// ErrNotSupported indicates that a feature is not supported.
//
// It is returned by ResponseController methods to indicate that
// the handler does not support the method, and by the Push method
// of Pusher implementations to indicate that HTTP/2 Push support
// is not available.
ErrNotSupported = &ProtocolError{"feature not supported"}
// Deprecated: ErrUnexpectedTrailer is no longer returned by
// anything in the net/http package. Callers should not
// compare errors against this variable.
ErrUnexpectedTrailer = &ProtocolError{"trailer header without chunked transfer encoding"}
// ErrMissingBoundary is returned by Request.MultipartReader when the
// request's Content-Type does not include a "boundary" parameter.
ErrMissingBoundary = &ProtocolError{"no multipart boundary param in Content-Type"}
// ErrNotMultipart is returned by Request.MultipartReader when the
// request's Content-Type is not multipart/form-data.
ErrNotMultipart = &ProtocolError{"request Content-Type isn't multipart/form-data"}
// Deprecated: ErrHeaderTooLong is no longer returned by
// anything in the net/http package. Callers should not
// compare errors against this variable.
ErrHeaderTooLong = &ProtocolError{"header too long"}
// Deprecated: ErrShortBody is no longer returned by
// anything in the net/http package. Callers should not
// compare errors against this variable.
ErrShortBody = &ProtocolError{"entity body too short"}
// Deprecated: ErrMissingContentLength is no longer returned by
// anything in the net/http package. Callers should not
// compare errors against this variable.
ErrMissingContentLength = &ProtocolError{"missing ContentLength in HEAD response"}
)
func badStringError(, string) error { return fmt.Errorf("%s %q", , ) }
// Headers that Request.Write handles itself and should be skipped.
var reqWriteExcludeHeader = map[string]bool{
"Host": true, // not in Header map anyway
"User-Agent": true,
"Content-Length": true,
"Transfer-Encoding": true,
"Trailer": true,
}
// A Request represents an HTTP request received by a server
// or to be sent by a client.
//
// The field semantics differ slightly between client and server
// usage. In addition to the notes on the fields below, see the
// documentation for [Request.Write] and [RoundTripper].
type Request struct {
// Method specifies the HTTP method (GET, POST, PUT, etc.).
// For client requests, an empty string means GET.
Method string
// URL specifies either the URI being requested (for server
// requests) or the URL to access (for client requests).
//
// For server requests, the URL is parsed from the URI
// supplied on the Request-Line as stored in RequestURI. For
// most requests, fields other than Path and RawQuery will be
// empty. (See RFC 7230, Section 5.3)
//
// For client requests, the URL's Host specifies the server to
// connect to, while the Request's Host field optionally
// specifies the Host header value to send in the HTTP
// request.
URL *url.URL
// The protocol version for incoming server requests.
//
// For client requests, these fields are ignored. The HTTP
// client code always uses either HTTP/1.1 or HTTP/2.
// See the docs on Transport for details.
Proto string // "HTTP/1.0"
ProtoMajor int // 1
ProtoMinor int // 0
// Header contains the request header fields either received
// by the server or to be sent by the client.
//
// If a server received a request with header lines,
//
// Host: example.com
// accept-encoding: gzip, deflate
// Accept-Language: en-us
// fOO: Bar
// foo: two
//
// then
//
// Header = map[string][]string{
// "Accept-Encoding": {"gzip, deflate"},
// "Accept-Language": {"en-us"},
// "Foo": {"Bar", "two"},
// }
//
// For incoming requests, the Host header is promoted to the
// Request.Host field and removed from the Header map.
//
// HTTP defines that header names are case-insensitive. The
// request parser implements this by using CanonicalHeaderKey,
// making the first character and any characters following a
// hyphen uppercase and the rest lowercase.
//
// For client requests, certain headers such as Content-Length
// and Connection are automatically written when needed and
// values in Header may be ignored. See the documentation
// for the Request.Write method.
Header Header
// Body is the request's body.
//
// For client requests, a nil body means the request has no
// body, such as a GET request. The HTTP Client's Transport
// is responsible for calling the Close method.
//
// For server requests, the Request Body is always non-nil
// but will return EOF immediately when no body is present.
// The Server will close the request body. The ServeHTTP
// Handler does not need to.
//
// Body must allow Read to be called concurrently with Close.
// In particular, calling Close should unblock a Read waiting
// for input.
Body io.ReadCloser
// GetBody defines an optional func to return a new copy of
// Body. It is used for client requests when a redirect requires
// reading the body more than once. Use of GetBody still
// requires setting Body.
//
// For server requests, it is unused.
GetBody func() (io.ReadCloser, error)
// ContentLength records the length of the associated content.
// The value -1 indicates that the length is unknown.
// Values >= 0 indicate that the given number of bytes may
// be read from Body.
//
// For client requests, a value of 0 with a non-nil Body is
// also treated as unknown.
ContentLength int64
// TransferEncoding lists the transfer encodings from outermost to
// innermost. An empty list denotes the "identity" encoding.
// TransferEncoding can usually be ignored; chunked encoding is
// automatically added and removed as necessary when sending and
// receiving requests.
TransferEncoding []string
// Close indicates whether to close the connection after
// replying to this request (for servers) or after sending this
// request and reading its response (for clients).
//
// For server requests, the HTTP server handles this automatically
// and this field is not needed by Handlers.
//
// For client requests, setting this field prevents re-use of
// TCP connections between requests to the same hosts, as if
// Transport.DisableKeepAlives were set.
Close bool
// For server requests, Host specifies the host on which the
// URL is sought. For HTTP/1 (per RFC 7230, section 5.4), this
// is either the value of the "Host" header or the host name
// given in the URL itself. For HTTP/2, it is the value of the
// ":authority" pseudo-header field.
// It may be of the form "host:port". For international domain
// names, Host may be in Punycode or Unicode form. Use
// golang.org/x/net/idna to convert it to either format if
// needed.
// To prevent DNS rebinding attacks, server Handlers should
// validate that the Host header has a value for which the
// Handler considers itself authoritative. The included
// ServeMux supports patterns registered to particular host
// names and thus protects its registered Handlers.
//
// For client requests, Host optionally overrides the Host
// header to send. If empty, the Request.Write method uses
// the value of URL.Host. Host may contain an international
// domain name.
Host string
// Form contains the parsed form data, including both the URL
// field's query parameters and the PATCH, POST, or PUT form data.
// This field is only available after ParseForm is called.
// The HTTP client ignores Form and uses Body instead.
Form url.Values
// PostForm contains the parsed form data from PATCH, POST
// or PUT body parameters.
//
// This field is only available after ParseForm is called.
// The HTTP client ignores PostForm and uses Body instead.
PostForm url.Values
// MultipartForm is the parsed multipart form, including file uploads.
// This field is only available after ParseMultipartForm is called.
// The HTTP client ignores MultipartForm and uses Body instead.
MultipartForm *multipart.Form
// Trailer specifies additional headers that are sent after the request
// body.
//
// For server requests, the Trailer map initially contains only the
// trailer keys, with nil values. (The client declares which trailers it
// will later send.) While the handler is reading from Body, it must
// not reference Trailer. After reading from Body returns EOF, Trailer
// can be read again and will contain non-nil values, if they were sent
// by the client.
//
// For client requests, Trailer must be initialized to a map containing
// the trailer keys to later send. The values may be nil or their final
// values. The ContentLength must be 0 or -1, to send a chunked request.
// After the HTTP request is sent the map values can be updated while
// the request body is read. Once the body returns EOF, the caller must
// not mutate Trailer.
//
// Few HTTP clients, servers, or proxies support HTTP trailers.
Trailer Header
// RemoteAddr allows HTTP servers and other software to record
// the network address that sent the request, usually for
// logging. This field is not filled in by ReadRequest and
// has no defined format. The HTTP server in this package
// sets RemoteAddr to an "IP:port" address before invoking a
// handler.
// This field is ignored by the HTTP client.
RemoteAddr string
// RequestURI is the unmodified request-target of the
// Request-Line (RFC 7230, Section 3.1.1) as sent by the client
// to a server. Usually the URL field should be used instead.
// It is an error to set this field in an HTTP client request.
RequestURI string
// TLS allows HTTP servers and other software to record
// information about the TLS connection on which the request
// was received. This field is not filled in by ReadRequest.
// The HTTP server in this package sets the field for
// TLS-enabled connections before invoking a handler;
// otherwise it leaves the field nil.
// This field is ignored by the HTTP client.
TLS *tls.ConnectionState
// Cancel is an optional channel whose closure indicates that the client
// request should be regarded as canceled. Not all implementations of
// RoundTripper may support Cancel.
//
// For server requests, this field is not applicable.
//
// Deprecated: Set the Request's context with NewRequestWithContext
// instead. If a Request's Cancel field and context are both
// set, it is undefined whether Cancel is respected.
Cancel <-chan struct{}
// Response is the redirect response which caused this request
// to be created. This field is only populated during client
// redirects.
Response *Response
// Pattern is the [ServeMux] pattern that matched the request.
// It is empty if the request was not matched against a pattern.
Pattern string
// ctx is either the client or server context. It should only
// be modified via copying the whole Request using Clone or WithContext.
// It is unexported to prevent people from using Context wrong
// and mutating the contexts held by callers of the same request.
ctx context.Context
// The following fields are for requests matched by ServeMux.
pat *pattern // the pattern that matched
matches []string // values for the matching wildcards in pat
otherValues map[string]string // for calls to SetPathValue that don't match a wildcard
}
// Context returns the request's context. To change the context, use
// [Request.Clone] or [Request.WithContext].
//
// The returned context is always non-nil; it defaults to the
// background context.
//
// For outgoing client requests, the context controls cancellation.
//
// For incoming server requests, the context is canceled when the
// client's connection closes, the request is canceled (with HTTP/2),
// or when the ServeHTTP method returns.
func ( *Request) () context.Context {
if .ctx != nil {
return .ctx
}
return context.Background()
}
// WithContext returns a shallow copy of r with its context changed
// to ctx. The provided ctx must be non-nil.
//
// For outgoing client request, the context controls the entire
// lifetime of a request and its response: obtaining a connection,
// sending the request, and reading the response headers and body.
//
// To create a new request with a context, use [NewRequestWithContext].
// To make a deep copy of a request with a new context, use [Request.Clone].
func ( *Request) ( context.Context) *Request {
if == nil {
panic("nil context")
}
:= new(Request)
* = *
.ctx =
return
}
// Clone returns a deep copy of r with its context changed to ctx.
// The provided ctx must be non-nil.
//
// Clone only makes a shallow copy of the Body field.
//
// For an outgoing client request, the context controls the entire
// lifetime of a request and its response: obtaining a connection,
// sending the request, and reading the response headers and body.
func ( *Request) ( context.Context) *Request {
if == nil {
panic("nil context")
}
:= new(Request)
* = *
.ctx =
.URL = cloneURL(.URL)
.Header = .Header.Clone()
.Trailer = .Trailer.Clone()
if := .TransferEncoding; != nil {
:= make([]string, len())
copy(, )
.TransferEncoding =
}
.Form = cloneURLValues(.Form)
.PostForm = cloneURLValues(.PostForm)
.MultipartForm = cloneMultipartForm(.MultipartForm)
// Copy matches and otherValues. See issue 61410.
if := .matches; != nil {
:= make([]string, len())
copy(, )
.matches =
}
.otherValues = maps.Clone(.otherValues)
return
}
// ProtoAtLeast reports whether the HTTP protocol used
// in the request is at least major.minor.
func ( *Request) (, int) bool {
return .ProtoMajor > ||
.ProtoMajor == && .ProtoMinor >=
}
// UserAgent returns the client's User-Agent, if sent in the request.
func ( *Request) () string {
return .Header.Get("User-Agent")
}
// Cookies parses and returns the HTTP cookies sent with the request.
func ( *Request) () []*Cookie {
return readCookies(.Header, "")
}
// CookiesNamed parses and returns the named HTTP cookies sent with the request
// or an empty slice if none matched.
func ( *Request) ( string) []*Cookie {
if == "" {
return []*Cookie{}
}
return readCookies(.Header, )
}
// ErrNoCookie is returned by Request's Cookie method when a cookie is not found.
var ErrNoCookie = errors.New("http: named cookie not present")
// Cookie returns the named cookie provided in the request or
// [ErrNoCookie] if not found.
// If multiple cookies match the given name, only one cookie will
// be returned.
func ( *Request) ( string) (*Cookie, error) {
if == "" {
return nil, ErrNoCookie
}
for , := range readCookies(.Header, ) {
return , nil
}
return nil, ErrNoCookie
}
// AddCookie adds a cookie to the request. Per RFC 6265 section 5.4,
// AddCookie does not attach more than one [Cookie] header field. That
// means all cookies, if any, are written into the same line,
// separated by semicolon.
// AddCookie only sanitizes c's name and value, and does not sanitize
// a Cookie header already present in the request.
func ( *Request) ( *Cookie) {
:= fmt.Sprintf("%s=%s", sanitizeCookieName(.Name), sanitizeCookieValue(.Value, .Quoted))
if := .Header.Get("Cookie"); != "" {
.Header.Set("Cookie", +"; "+)
} else {
.Header.Set("Cookie", )
}
}
// Referer returns the referring URL, if sent in the request.
//
// Referer is misspelled as in the request itself, a mistake from the
// earliest days of HTTP. This value can also be fetched from the
// [Header] map as Header["Referer"]; the benefit of making it available
// as a method is that the compiler can diagnose programs that use the
// alternate (correct English) spelling req.Referrer() but cannot
// diagnose programs that use Header["Referrer"].
func ( *Request) () string {
return .Header.Get("Referer")
}
// multipartByReader is a sentinel value.
// Its presence in Request.MultipartForm indicates that parsing of the request
// body has been handed off to a MultipartReader instead of ParseMultipartForm.
var multipartByReader = &multipart.Form{
Value: make(map[string][]string),
File: make(map[string][]*multipart.FileHeader),
}
// MultipartReader returns a MIME multipart reader if this is a
// multipart/form-data or a multipart/mixed POST request, else returns nil and an error.
// Use this function instead of [Request.ParseMultipartForm] to
// process the request body as a stream.
func ( *Request) () (*multipart.Reader, error) {
if .MultipartForm == multipartByReader {
return nil, errors.New("http: MultipartReader called twice")
}
if .MultipartForm != nil {
return nil, errors.New("http: multipart handled by ParseMultipartForm")
}
.MultipartForm = multipartByReader
return .multipartReader(true)
}
func ( *Request) ( bool) (*multipart.Reader, error) {
:= .Header.Get("Content-Type")
if == "" {
return nil, ErrNotMultipart
}
if .Body == nil {
return nil, errors.New("missing form body")
}
, , := mime.ParseMediaType()
if != nil || !( == "multipart/form-data" || && == "multipart/mixed") {
return nil, ErrNotMultipart
}
, := ["boundary"]
if ! {
return nil, ErrMissingBoundary
}
return multipart.NewReader(.Body, ), nil
}
// isH2Upgrade reports whether r represents the http2 "client preface"
// magic string.
func ( *Request) () bool {
return .Method == "PRI" && len(.Header) == 0 && .URL.Path == "*" && .Proto == "HTTP/2.0"
}
// Return value if nonempty, def otherwise.
func valueOrDefault(, string) string {
if != "" {
return
}
return
}
// NOTE: This is not intended to reflect the actual Go version being used.
// It was changed at the time of Go 1.1 release because the former User-Agent
// had ended up blocked by some intrusion detection systems.
// See https://codereview.appspot.com/7532043.
const defaultUserAgent = "Go-http-client/1.1"
// Write writes an HTTP/1.1 request, which is the header and body, in wire format.
// This method consults the following fields of the request:
//
// Host
// URL
// Method (defaults to "GET")
// Header
// ContentLength
// TransferEncoding
// Body
//
// If Body is present, Content-Length is <= 0 and [Request.TransferEncoding]
// hasn't been set to "identity", Write adds "Transfer-Encoding:
// chunked" to the header. Body is closed after it is sent.
func ( *Request) ( io.Writer) error {
return .write(, false, nil, nil)
}
// WriteProxy is like [Request.Write] but writes the request in the form
// expected by an HTTP proxy. In particular, [Request.WriteProxy] writes the
// initial Request-URI line of the request with an absolute URI, per
// section 5.3 of RFC 7230, including the scheme and host.
// In either case, WriteProxy also writes a Host header, using
// either r.Host or r.URL.Host.
func ( *Request) ( io.Writer) error {
return .write(, true, nil, nil)
}
// errMissingHost is returned by Write when there is no Host or URL present in
// the Request.
var errMissingHost = errors.New("http: Request.Write on Request with no Host or URL set")
// extraHeaders may be nil
// waitForContinue may be nil
// always closes body
func ( *Request) ( io.Writer, bool, Header, func() bool) ( error) {
:= httptrace.ContextClientTrace(.Context())
if != nil && .WroteRequest != nil {
defer func() {
.WroteRequest(httptrace.WroteRequestInfo{
Err: ,
})
}()
}
:= false
defer func() {
if {
return
}
if := .closeBody(); != nil && == nil {
=
}
}()
// Find the target host. Prefer the Host: header, but if that
// is not given, use the host from the request URL.
//
// Clean the host, in case it arrives with unexpected stuff in it.
:= .Host
if == "" {
if .URL == nil {
return errMissingHost
}
= .URL.Host
}
, = httpguts.PunycodeHostPort()
if != nil {
return
}
// Validate that the Host header is a valid header in general,
// but don't validate the host itself. This is sufficient to avoid
// header or request smuggling via the Host field.
// The server can (and will, if it's a net/http server) reject
// the request if it doesn't consider the host valid.
if !httpguts.ValidHostHeader() {
// Historically, we would truncate the Host header after '/' or ' '.
// Some users have relied on this truncation to convert a network
// address such as Unix domain socket path into a valid, ignored
// Host header (see https://go.dev/issue/61431).
//
// We don't preserve the truncation, because sending an altered
// header field opens a smuggling vector. Instead, zero out the
// Host header entirely if it isn't valid. (An empty Host is valid;
// see RFC 9112 Section 3.2.)
//
// Return an error if we're sending to a proxy, since the proxy
// probably can't do anything useful with an empty Host header.
if ! {
= ""
} else {
return errors.New("http: invalid Host header")
}
}
// According to RFC 6874, an HTTP client, proxy, or other
// intermediary must remove any IPv6 zone identifier attached
// to an outgoing URI.
= removeZone()
:= .URL.RequestURI()
if && .URL.Scheme != "" && .URL.Opaque == "" {
= .URL.Scheme + "://" + +
} else if .Method == "CONNECT" && .URL.Path == "" {
// CONNECT requests normally give just the host and port, not a full URL.
=
if .URL.Opaque != "" {
= .URL.Opaque
}
}
if stringContainsCTLByte() {
return errors.New("net/http: can't write control character in Request.URL")
}
// TODO: validate r.Method too? At least it's less likely to
// come from an attacker (more likely to be a constant in
// code).
// Wrap the writer in a bufio Writer if it's not already buffered.
// Don't always call NewWriter, as that forces a bytes.Buffer
// and other small bufio Writers to have a minimum 4k buffer
// size.
var *bufio.Writer
if , := .(io.ByteWriter); ! {
= bufio.NewWriter()
=
}
_, = fmt.Fprintf(, "%s %s HTTP/1.1\r\n", valueOrDefault(.Method, "GET"), )
if != nil {
return
}
// Header lines
_, = fmt.Fprintf(, "Host: %s\r\n", )
if != nil {
return
}
if != nil && .WroteHeaderField != nil {
.WroteHeaderField("Host", []string{})
}
// Use the defaultUserAgent unless the Header contains one, which
// may be blank to not send the header.
:= defaultUserAgent
if .Header.has("User-Agent") {
= .Header.Get("User-Agent")
}
if != "" {
= headerNewlineToSpace.Replace()
= textproto.TrimString()
_, = fmt.Fprintf(, "User-Agent: %s\r\n", )
if != nil {
return
}
if != nil && .WroteHeaderField != nil {
.WroteHeaderField("User-Agent", []string{})
}
}
// Process Body,ContentLength,Close,Trailer
, := newTransferWriter()
if != nil {
return
}
= .writeHeader(, )
if != nil {
return
}
= .Header.writeSubset(, reqWriteExcludeHeader, )
if != nil {
return
}
if != nil {
= .write(, )
if != nil {
return
}
}
_, = io.WriteString(, "\r\n")
if != nil {
return
}
if != nil && .WroteHeaders != nil {
.WroteHeaders()
}
// Flush and wait for 100-continue if expected.
if != nil {
if , := .(*bufio.Writer); {
= .Flush()
if != nil {
return
}
}
if != nil && .Wait100Continue != nil {
.Wait100Continue()
}
if !() {
= true
.closeBody()
return nil
}
}
if , := .(*bufio.Writer); && .FlushHeaders {
if := .Flush(); != nil {
return
}
}
// Write body and trailer
= true
= .writeBody()
if != nil {
if .bodyReadError == {
= requestBodyReadError{}
}
return
}
if != nil {
return .Flush()
}
return nil
}
// requestBodyReadError wraps an error from (*Request).write to indicate
// that the error came from a Read call on the Request.Body.
// This error type should not escape the net/http package to users.
type requestBodyReadError struct{ error }
func idnaASCII( string) (string, error) {
// TODO: Consider removing this check after verifying performance is okay.
// Right now punycode verification, length checks, context checks, and the
// permissible character tests are all omitted. It also prevents the ToASCII
// call from salvaging an invalid IDN, when possible. As a result it may be
// possible to have two IDNs that appear identical to the user where the
// ASCII-only version causes an error downstream whereas the non-ASCII
// version does not.
// Note that for correct ASCII IDNs ToASCII will only do considerably more
// work, but it will not cause an allocation.
if ascii.Is() {
return , nil
}
return idna.Lookup.ToASCII()
}
// removeZone removes IPv6 zone identifier from host.
// E.g., "[fe80::1%en0]:8080" to "[fe80::1]:8080"
func removeZone( string) string {
if !strings.HasPrefix(, "[") {
return
}
:= strings.LastIndex(, "]")
if < 0 {
return
}
:= strings.LastIndex([:], "%")
if < 0 {
return
}
return [:] + [:]
}
// ParseHTTPVersion parses an HTTP version string according to RFC 7230, section 2.6.
// "HTTP/1.0" returns (1, 0, true). Note that strings without
// a minor version, such as "HTTP/2", are not valid.
func ( string) (, int, bool) {
switch {
case "HTTP/1.1":
return 1, 1, true
case "HTTP/1.0":
return 1, 0, true
}
if !strings.HasPrefix(, "HTTP/") {
return 0, 0, false
}
if len() != len("HTTP/X.Y") {
return 0, 0, false
}
if [6] != '.' {
return 0, 0, false
}
, := strconv.ParseUint([5:6], 10, 0)
if != nil {
return 0, 0, false
}
, := strconv.ParseUint([7:8], 10, 0)
if != nil {
return 0, 0, false
}
return int(), int(), true
}
func validMethod( string) bool {
/*
Method = "OPTIONS" ; Section 9.2
| "GET" ; Section 9.3
| "HEAD" ; Section 9.4
| "POST" ; Section 9.5
| "PUT" ; Section 9.6
| "DELETE" ; Section 9.7
| "TRACE" ; Section 9.8
| "CONNECT" ; Section 9.9
| extension-method
extension-method = token
token = 1*<any CHAR except CTLs or separators>
*/
return len() > 0 && strings.IndexFunc(, isNotToken) == -1
}
// NewRequest wraps [NewRequestWithContext] using [context.Background].
func (, string, io.Reader) (*Request, error) {
return NewRequestWithContext(context.Background(), , , )
}
// NewRequestWithContext returns a new [Request] given a method, URL, and
// optional body.
//
// If the provided body is also an [io.Closer], the returned
// [Request.Body] is set to body and will be closed (possibly
// asynchronously) by the Client methods Do, Post, and PostForm,
// and [Transport.RoundTrip].
//
// NewRequestWithContext returns a Request suitable for use with
// [Client.Do] or [Transport.RoundTrip]. To create a request for use with
// testing a Server Handler, either use the [NewRequest] function in the
// net/http/httptest package, use [ReadRequest], or manually update the
// Request fields. For an outgoing client request, the context
// controls the entire lifetime of a request and its response:
// obtaining a connection, sending the request, and reading the
// response headers and body. See the Request type's documentation for
// the difference between inbound and outbound request fields.
//
// If body is of type [*bytes.Buffer], [*bytes.Reader], or
// [*strings.Reader], the returned request's ContentLength is set to its
// exact value (instead of -1), GetBody is populated (so 307 and 308
// redirects can replay the body), and Body is set to [NoBody] if the
// ContentLength is 0.
func ( context.Context, , string, io.Reader) (*Request, error) {
if == "" {
// We document that "" means "GET" for Request.Method, and people have
// relied on that from NewRequest, so keep that working.
// We still enforce validMethod for non-empty methods.
= "GET"
}
if !validMethod() {
return nil, fmt.Errorf("net/http: invalid method %q", )
}
if == nil {
return nil, errors.New("net/http: nil Context")
}
, := urlpkg.Parse()
if != nil {
return nil,
}
, := .(io.ReadCloser)
if ! && != nil {
= io.NopCloser()
}
// The host's colon:port should be normalized. See Issue 14836.
.Host = removeEmptyPort(.Host)
:= &Request{
ctx: ,
Method: ,
URL: ,
Proto: "HTTP/1.1",
ProtoMajor: 1,
ProtoMinor: 1,
Header: make(Header),
Body: ,
Host: .Host,
}
if != nil {
switch v := .(type) {
case *bytes.Buffer:
.ContentLength = int64(.Len())
:= .Bytes()
.GetBody = func() (io.ReadCloser, error) {
:= bytes.NewReader()
return io.NopCloser(), nil
}
case *bytes.Reader:
.ContentLength = int64(.Len())
:= *
.GetBody = func() (io.ReadCloser, error) {
:=
return io.NopCloser(&), nil
}
case *strings.Reader:
.ContentLength = int64(.Len())
:= *
.GetBody = func() (io.ReadCloser, error) {
:=
return io.NopCloser(&), nil
}
default:
// This is where we'd set it to -1 (at least
// if body != NoBody) to mean unknown, but
// that broke people during the Go 1.8 testing
// period. People depend on it being 0 I
// guess. Maybe retry later. See Issue 18117.
}
// For client requests, Request.ContentLength of 0
// means either actually 0, or unknown. The only way
// to explicitly say that the ContentLength is zero is
// to set the Body to nil. But turns out too much code
// depends on NewRequest returning a non-nil Body,
// so we use a well-known ReadCloser variable instead
// and have the http package also treat that sentinel
// variable to mean explicitly zero.
if .GetBody != nil && .ContentLength == 0 {
.Body = NoBody
.GetBody = func() (io.ReadCloser, error) { return NoBody, nil }
}
}
return , nil
}
// BasicAuth returns the username and password provided in the request's
// Authorization header, if the request uses HTTP Basic Authentication.
// See RFC 2617, Section 2.
func ( *Request) () (, string, bool) {
:= .Header.Get("Authorization")
if == "" {
return "", "", false
}
return parseBasicAuth()
}
// parseBasicAuth parses an HTTP Basic Authentication string.
// "Basic QWxhZGRpbjpvcGVuIHNlc2FtZQ==" returns ("Aladdin", "open sesame", true).
//
// parseBasicAuth should be an internal detail,
// but widely used packages access it using linkname.
// Notable members of the hall of shame include:
// - github.com/sagernet/sing
//
// Do not remove or change the type signature.
// See go.dev/issue/67401.
//
//go:linkname parseBasicAuth
func parseBasicAuth( string) (, string, bool) {
const = "Basic "
// Case insensitive prefix match. See Issue 22736.
if len() < len() || !ascii.EqualFold([:len()], ) {
return "", "", false
}
, := base64.StdEncoding.DecodeString([len():])
if != nil {
return "", "", false
}
:= string()
, , = strings.Cut(, ":")
if ! {
return "", "", false
}
return , , true
}
// SetBasicAuth sets the request's Authorization header to use HTTP
// Basic Authentication with the provided username and password.
//
// With HTTP Basic Authentication the provided username and password
// are not encrypted. It should generally only be used in an HTTPS
// request.
//
// The username may not contain a colon. Some protocols may impose
// additional requirements on pre-escaping the username and
// password. For instance, when used with OAuth2, both arguments must
// be URL encoded first with [url.QueryEscape].
func ( *Request) (, string) {
.Header.Set("Authorization", "Basic "+basicAuth(, ))
}
// parseRequestLine parses "GET /foo HTTP/1.1" into its three parts.
func parseRequestLine( string) (, , string, bool) {
, , := strings.Cut(, " ")
, , := strings.Cut(, " ")
if ! || ! {
return "", "", "", false
}
return , , , true
}
var textprotoReaderPool sync.Pool
func newTextprotoReader( *bufio.Reader) *textproto.Reader {
if := textprotoReaderPool.Get(); != nil {
:= .(*textproto.Reader)
.R =
return
}
return textproto.NewReader()
}
func putTextprotoReader( *textproto.Reader) {
.R = nil
textprotoReaderPool.Put()
}
// ReadRequest reads and parses an incoming request from b.
//
// ReadRequest is a low-level function and should only be used for
// specialized applications; most code should use the [Server] to read
// requests and handle them via the [Handler] interface. ReadRequest
// only supports HTTP/1.x requests. For HTTP/2, use golang.org/x/net/http2.
func ( *bufio.Reader) (*Request, error) {
, := readRequest()
if != nil {
return nil,
}
delete(.Header, "Host")
return ,
}
// readRequest should be an internal detail,
// but widely used packages access it using linkname.
// Notable members of the hall of shame include:
// - github.com/sagernet/sing
// - github.com/v2fly/v2ray-core/v4
// - github.com/v2fly/v2ray-core/v5
//
// Do not remove or change the type signature.
// See go.dev/issue/67401.
//
//go:linkname readRequest
func readRequest( *bufio.Reader) ( *Request, error) {
:= newTextprotoReader()
defer putTextprotoReader()
= new(Request)
// First line: GET /index.html HTTP/1.0
var string
if , = .ReadLine(); != nil {
return nil,
}
defer func() {
if == io.EOF {
= io.ErrUnexpectedEOF
}
}()
var bool
.Method, .RequestURI, .Proto, = parseRequestLine()
if ! {
return nil, badStringError("malformed HTTP request", )
}
if !validMethod(.Method) {
return nil, badStringError("invalid method", .Method)
}
:= .RequestURI
if .ProtoMajor, .ProtoMinor, = ParseHTTPVersion(.Proto); ! {
return nil, badStringError("malformed HTTP version", .Proto)
}
// CONNECT requests are used two different ways, and neither uses a full URL:
// The standard use is to tunnel HTTPS through an HTTP proxy.
// It looks like "CONNECT www.google.com:443 HTTP/1.1", and the parameter is
// just the authority section of a URL. This information should go in req.URL.Host.
//
// The net/rpc package also uses CONNECT, but there the parameter is a path
// that starts with a slash. It can be parsed with the regular URL parser,
// and the path will end up in req.URL.Path, where it needs to be in order for
// RPC to work.
:= .Method == "CONNECT" && !strings.HasPrefix(, "/")
if {
= "http://" +
}
if .URL, = url.ParseRequestURI(); != nil {
return nil,
}
if {
// Strip the bogus "http://" back off.
.URL.Scheme = ""
}
// Subsequent lines: Key: value.
, := .ReadMIMEHeader()
if != nil {
return nil,
}
.Header = Header()
if len(.Header["Host"]) > 1 {
return nil, fmt.Errorf("too many Host headers")
}
// RFC 7230, section 5.3: Must treat
// GET /index.html HTTP/1.1
// Host: www.google.com
// and
// GET http://www.google.com/index.html HTTP/1.1
// Host: doesntmatter
// the same. In the second case, any Host line is ignored.
.Host = .URL.Host
if .Host == "" {
.Host = .Header.get("Host")
}
fixPragmaCacheControl(.Header)
.Close = shouldClose(.ProtoMajor, .ProtoMinor, .Header, false)
= readTransfer(, )
if != nil {
return nil,
}
if .isH2Upgrade() {
// Because it's neither chunked, nor declared:
.ContentLength = -1
// We want to give handlers a chance to hijack the
// connection, but we need to prevent the Server from
// dealing with the connection further if it's not
// hijacked. Set Close to ensure that:
.Close = true
}
return , nil
}
// MaxBytesReader is similar to [io.LimitReader] but is intended for
// limiting the size of incoming request bodies. In contrast to
// io.LimitReader, MaxBytesReader's result is a ReadCloser, returns a
// non-nil error of type [*MaxBytesError] for a Read beyond the limit,
// and closes the underlying reader when its Close method is called.
//
// MaxBytesReader prevents clients from accidentally or maliciously
// sending a large request and wasting server resources. If possible,
// it tells the [ResponseWriter] to close the connection after the limit
// has been reached.
func ( ResponseWriter, io.ReadCloser, int64) io.ReadCloser {
if < 0 { // Treat negative limits as equivalent to 0.
= 0
}
return &maxBytesReader{w: , r: , i: , n: }
}
// MaxBytesError is returned by [MaxBytesReader] when its read limit is exceeded.
type MaxBytesError struct {
Limit int64
}
func ( *MaxBytesError) () string {
// Due to Hyrum's law, this text cannot be changed.
return "http: request body too large"
}
type maxBytesReader struct {
w ResponseWriter
r io.ReadCloser // underlying reader
i int64 // max bytes initially, for MaxBytesError
n int64 // max bytes remaining
err error // sticky error
}
func ( *maxBytesReader) ( []byte) ( int, error) {
if .err != nil {
return 0, .err
}
if len() == 0 {
return 0, nil
}
// If they asked for a 32KB byte read but only 5 bytes are
// remaining, no need to read 32KB. 6 bytes will answer the
// question of the whether we hit the limit or go past it.
// 0 < len(p) < 2^63
if int64(len())-1 > .n {
= [:.n+1]
}
, = .r.Read()
if int64() <= .n {
.n -= int64()
.err =
return ,
}
= int(.n)
.n = 0
// The server code and client code both use
// maxBytesReader. This "requestTooLarge" check is
// only used by the server code. To prevent binaries
// which only using the HTTP Client code (such as
// cmd/go) from also linking in the HTTP server, don't
// use a static type assertion to the server
// "*response" type. Check this interface instead:
type interface {
()
}
if , := .w.(); {
.()
}
.err = &MaxBytesError{.i}
return , .err
}
func ( *maxBytesReader) () error {
return .r.Close()
}
func copyValues(, url.Values) {
for , := range {
[] = append([], ...)
}
}
func parsePostForm( *Request) ( url.Values, error) {
if .Body == nil {
= errors.New("missing form body")
return
}
:= .Header.Get("Content-Type")
// RFC 7231, section 3.1.1.5 - empty type
// MAY be treated as application/octet-stream
if == "" {
= "application/octet-stream"
}
, _, = mime.ParseMediaType()
switch {
case == "application/x-www-form-urlencoded":
var io.Reader = .Body
:= int64(1<<63 - 1)
if , := .Body.(*maxBytesReader); ! {
= int64(10 << 20) // 10 MB is a lot of text.
= io.LimitReader(.Body, +1)
}
, := io.ReadAll()
if != nil {
if == nil {
=
}
break
}
if int64(len()) > {
= errors.New("http: POST too large")
return
}
, = url.ParseQuery(string())
if == nil {
=
}
case == "multipart/form-data":
// handled by ParseMultipartForm (which is calling us, or should be)
// TODO(bradfitz): there are too many possible
// orders to call too many functions here.
// Clean this up and write more tests.
// request_test.go contains the start of this,
// in TestParseMultipartFormOrder and others.
}
return
}
// ParseForm populates r.Form and r.PostForm.
//
// For all requests, ParseForm parses the raw query from the URL and updates
// r.Form.
//
// For POST, PUT, and PATCH requests, it also reads the request body, parses it
// as a form and puts the results into both r.PostForm and r.Form. Request body
// parameters take precedence over URL query string values in r.Form.
//
// If the request Body's size has not already been limited by [MaxBytesReader],
// the size is capped at 10MB.
//
// For other HTTP methods, or when the Content-Type is not
// application/x-www-form-urlencoded, the request Body is not read, and
// r.PostForm is initialized to a non-nil, empty value.
//
// [Request.ParseMultipartForm] calls ParseForm automatically.
// ParseForm is idempotent.
func ( *Request) () error {
var error
if .PostForm == nil {
if .Method == "POST" || .Method == "PUT" || .Method == "PATCH" {
.PostForm, = parsePostForm()
}
if .PostForm == nil {
.PostForm = make(url.Values)
}
}
if .Form == nil {
if len(.PostForm) > 0 {
.Form = make(url.Values)
copyValues(.Form, .PostForm)
}
var url.Values
if .URL != nil {
var error
, = url.ParseQuery(.URL.RawQuery)
if == nil {
=
}
}
if == nil {
= make(url.Values)
}
if .Form == nil {
.Form =
} else {
copyValues(.Form, )
}
}
return
}
// ParseMultipartForm parses a request body as multipart/form-data.
// The whole request body is parsed and up to a total of maxMemory bytes of
// its file parts are stored in memory, with the remainder stored on
// disk in temporary files.
// ParseMultipartForm calls [Request.ParseForm] if necessary.
// If ParseForm returns an error, ParseMultipartForm returns it but also
// continues parsing the request body.
// After one call to ParseMultipartForm, subsequent calls have no effect.
func ( *Request) ( int64) error {
if .MultipartForm == multipartByReader {
return errors.New("http: multipart handled by MultipartReader")
}
var error
if .Form == nil {
// Let errors in ParseForm fall through, and just
// return it at the end.
= .ParseForm()
}
if .MultipartForm != nil {
return nil
}
, := .multipartReader(false)
if != nil {
return
}
, := .ReadForm()
if != nil {
return
}
if .PostForm == nil {
.PostForm = make(url.Values)
}
for , := range .Value {
.Form[] = append(.Form[], ...)
// r.PostForm should also be populated. See Issue 9305.
.PostForm[] = append(.PostForm[], ...)
}
.MultipartForm =
return
}
// FormValue returns the first value for the named component of the query.
// The precedence order:
// 1. application/x-www-form-urlencoded form body (POST, PUT, PATCH only)
// 2. query parameters (always)
// 3. multipart/form-data form body (always)
//
// FormValue calls [Request.ParseMultipartForm] and [Request.ParseForm]
// if necessary and ignores any errors returned by these functions.
// If key is not present, FormValue returns the empty string.
// To access multiple values of the same key, call ParseForm and
// then inspect [Request.Form] directly.
func ( *Request) ( string) string {
if .Form == nil {
.ParseMultipartForm(defaultMaxMemory)
}
if := .Form[]; len() > 0 {
return [0]
}
return ""
}
// PostFormValue returns the first value for the named component of the POST,
// PUT, or PATCH request body. URL query parameters are ignored.
// PostFormValue calls [Request.ParseMultipartForm] and [Request.ParseForm] if necessary and ignores
// any errors returned by these functions.
// If key is not present, PostFormValue returns the empty string.
func ( *Request) ( string) string {
if .PostForm == nil {
.ParseMultipartForm(defaultMaxMemory)
}
if := .PostForm[]; len() > 0 {
return [0]
}
return ""
}
// FormFile returns the first file for the provided form key.
// FormFile calls [Request.ParseMultipartForm] and [Request.ParseForm] if necessary.
func ( *Request) ( string) (multipart.File, *multipart.FileHeader, error) {
if .MultipartForm == multipartByReader {
return nil, nil, errors.New("http: multipart handled by MultipartReader")
}
if .MultipartForm == nil {
:= .ParseMultipartForm(defaultMaxMemory)
if != nil {
return nil, nil,
}
}
if .MultipartForm != nil && .MultipartForm.File != nil {
if := .MultipartForm.File[]; len() > 0 {
, := [0].Open()
return , [0],
}
}
return nil, nil, ErrMissingFile
}
// PathValue returns the value for the named path wildcard in the [ServeMux] pattern
// that matched the request.
// It returns the empty string if the request was not matched against a pattern
// or there is no such wildcard in the pattern.
func ( *Request) ( string) string {
if := .patIndex(); >= 0 {
return .matches[]
}
return .otherValues[]
}
// SetPathValue sets name to value, so that subsequent calls to r.PathValue(name)
// return value.
func ( *Request) (, string) {
if := .patIndex(); >= 0 {
.matches[] =
} else {
if .otherValues == nil {
.otherValues = map[string]string{}
}
.otherValues[] =
}
}
// patIndex returns the index of name in the list of named wildcards of the
// request's pattern, or -1 if there is no such name.
func ( *Request) ( string) int {
// The linear search seems expensive compared to a map, but just creating the map
// takes a lot of time, and most patterns will just have a couple of wildcards.
if .pat == nil {
return -1
}
:= 0
for , := range .pat.segments {
if .wild && .s != "" {
if == .s {
return
}
++
}
}
return -1
}
func ( *Request) () bool {
return hasToken(.Header.get("Expect"), "100-continue")
}
func ( *Request) () bool {
if .ProtoMajor != 1 || .ProtoMinor != 0 {
return false
}
return hasToken(.Header.get("Connection"), "keep-alive")
}
func ( *Request) () bool {
if .Close {
return true
}
return hasToken(.Header.get("Connection"), "close")
}
func ( *Request) () error {
if .Body == nil {
return nil
}
return .Body.Close()
}
func ( *Request) () bool {
if .Body == nil || .Body == NoBody || .GetBody != nil {
switch valueOrDefault(.Method, "GET") {
case "GET", "HEAD", "OPTIONS", "TRACE":
return true
}
// The Idempotency-Key, while non-standard, is widely used to
// mean a POST or other request is idempotent. See
// https://golang.org/issue/19943#issuecomment-421092421
if .Header.has("Idempotency-Key") || .Header.has("X-Idempotency-Key") {
return true
}
}
return false
}
// outgoingLength reports the Content-Length of this outgoing (Client) request.
// It maps 0 into -1 (unknown) when the Body is non-nil.
func ( *Request) () int64 {
if .Body == nil || .Body == NoBody {
return 0
}
if .ContentLength != 0 {
return .ContentLength
}
return -1
}
// requestMethodUsuallyLacksBody reports whether the given request
// method is one that typically does not involve a request body.
// This is used by the Transport (via
// transferWriter.shouldSendChunkedRequestBody) to determine whether
// we try to test-read a byte from a non-nil Request.Body when
// Request.outgoingLength() returns -1. See the comments in
// shouldSendChunkedRequestBody.
func requestMethodUsuallyLacksBody( string) bool {
switch {
case "GET", "HEAD", "DELETE", "OPTIONS", "PROPFIND", "SEARCH":
return true
}
return false
}
// requiresHTTP1 reports whether this request requires being sent on
// an HTTP/1 connection.
func ( *Request) () bool {
return hasToken(.Header.Get("Connection"), "upgrade") &&
ascii.EqualFold(.Header.Get("Upgrade"), "websocket")
}
The pages are generated with Golds v0.7.3. (GOOS=linux GOARCH=amd64) Golds is a Go 101 project developed by Tapir Liu. PR and bug reports are welcome and can be submitted to the issue list. Please follow @zigo_101 (reachable from the left QR code) to get the latest news of Golds. |