Copyright 2016 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.

package httpguts

import (
	
	
	

	
)

var isTokenTable = [127]bool{
	'!':  true,
	'#':  true,
	'$':  true,
	'%':  true,
	'&':  true,
	'\'': true,
	'*':  true,
	'+':  true,
	'-':  true,
	'.':  true,
	'0':  true,
	'1':  true,
	'2':  true,
	'3':  true,
	'4':  true,
	'5':  true,
	'6':  true,
	'7':  true,
	'8':  true,
	'9':  true,
	'A':  true,
	'B':  true,
	'C':  true,
	'D':  true,
	'E':  true,
	'F':  true,
	'G':  true,
	'H':  true,
	'I':  true,
	'J':  true,
	'K':  true,
	'L':  true,
	'M':  true,
	'N':  true,
	'O':  true,
	'P':  true,
	'Q':  true,
	'R':  true,
	'S':  true,
	'T':  true,
	'U':  true,
	'W':  true,
	'V':  true,
	'X':  true,
	'Y':  true,
	'Z':  true,
	'^':  true,
	'_':  true,
	'`':  true,
	'a':  true,
	'b':  true,
	'c':  true,
	'd':  true,
	'e':  true,
	'f':  true,
	'g':  true,
	'h':  true,
	'i':  true,
	'j':  true,
	'k':  true,
	'l':  true,
	'm':  true,
	'n':  true,
	'o':  true,
	'p':  true,
	'q':  true,
	'r':  true,
	's':  true,
	't':  true,
	'u':  true,
	'v':  true,
	'w':  true,
	'x':  true,
	'y':  true,
	'z':  true,
	'|':  true,
	'~':  true,
}

func ( rune) bool {
	 := int()
	return  < len(isTokenTable) && isTokenTable[]
}

func ( rune) bool {
	return !IsTokenRune()
}
HeaderValuesContainsToken reports whether any string in values contains the provided token, ASCII case-insensitively.
func ( []string,  string) bool {
	for ,  := range  {
		if headerValueContainsToken(, ) {
			return true
		}
	}
	return false
}
isOWS reports whether b is an optional whitespace byte, as defined by RFC 7230 section 3.2.3.
func ( byte) bool { return  == ' ' ||  == '\t' }
trimOWS returns x with all optional whitespace removes from the beginning and end.
TODO: consider using strings.Trim(x, " \t") instead, if and when it's fast enough. See issue 10292. But this ASCII-only code will probably always beat UTF-8 aware code.
	for len() > 0 && isOWS([0]) {
		 = [1:]
	}
	for len() > 0 && isOWS([len()-1]) {
		 = [:len()-1]
	}
	return 
}
headerValueContainsToken reports whether v (assumed to be a 0#element, in the ABNF extension described in RFC 7230 section 7) contains token amongst its comma-separated tokens, ASCII case-insensitively.
func ( string,  string) bool {
	 = trimOWS()
	if  := strings.IndexByte(, ',');  != -1 {
		return tokenEqual(trimOWS([:]), ) || ([+1:], )
	}
	return tokenEqual(, )
}
lowerASCII returns the ASCII lowercase version of b.
func ( byte) byte {
	if 'A' <=  &&  <= 'Z' {
		return  + ('a' - 'A')
	}
	return 
}
tokenEqual reports whether t1 and t2 are equal, ASCII case-insensitively.
func (,  string) bool {
	if len() != len() {
		return false
	}
	for ,  := range  {
No UTF-8 or non-ASCII allowed in tokens.
			return false
		}
		if lowerASCII(byte()) != lowerASCII([]) {
			return false
		}
	}
	return true
}
isLWS reports whether b is linear white space, according to http://www.w3.org/Protocols/rfc2616/rfc2616-sec2.html#sec2.2 LWS = [CRLF] 1*( SP | HT )
func ( byte) bool { return  == ' ' ||  == '\t' }
isCTL reports whether b is a control byte, according to http://www.w3.org/Protocols/rfc2616/rfc2616-sec2.html#sec2.2 CTL = <any US-ASCII control character (octets 0 - 31) and DEL (127)>
func ( byte) bool {
	const  = 0x7f // a CTL
	return  < ' ' ||  == 
}
ValidHeaderFieldName reports whether v is a valid HTTP/1.x header name. HTTP/2 imposes the additional restriction that uppercase ASCII letters are not allowed. RFC 7230 says: header-field = field-name ":" OWS field-value OWS field-name = token token = 1*tchar tchar = "!" / "#" / "$" / "%" / "&" / "'" / "*" / "+" / "-" / "." / "^" / "_" / "`" / "|" / "~" / DIGIT / ALPHA
func ( string) bool {
	if len() == 0 {
		return false
	}
	for ,  := range  {
		if !IsTokenRune() {
			return false
		}
	}
	return true
}
ValidHostHeader reports whether h is a valid host header.
The latest spec is actually this: http://tools.ietf.org/html/rfc7230#section-5.4 Host = uri-host [ ":" port ] Where uri-host is: http://tools.ietf.org/html/rfc3986#section-3.2.2 But we're going to be much more lenient for now and just search for any byte that's not a valid byte in any of those expressions.
	for  := 0;  < len(); ++ {
		if !validHostByte[[]] {
			return false
		}
	}
	return true
}
See the validHostHeader comment.
var validHostByte = [256]bool{
	'0': true, '1': true, '2': true, '3': true, '4': true, '5': true, '6': true, '7': true,
	'8': true, '9': true,

	'a': true, 'b': true, 'c': true, 'd': true, 'e': true, 'f': true, 'g': true, 'h': true,
	'i': true, 'j': true, 'k': true, 'l': true, 'm': true, 'n': true, 'o': true, 'p': true,
	'q': true, 'r': true, 's': true, 't': true, 'u': true, 'v': true, 'w': true, 'x': true,
	'y': true, 'z': true,

	'A': true, 'B': true, 'C': true, 'D': true, 'E': true, 'F': true, 'G': true, 'H': true,
	'I': true, 'J': true, 'K': true, 'L': true, 'M': true, 'N': true, 'O': true, 'P': true,
	'Q': true, 'R': true, 'S': true, 'T': true, 'U': true, 'V': true, 'W': true, 'X': true,
	'Y': true, 'Z': true,

	'!':  true, // sub-delims
	'$':  true, // sub-delims
	'%':  true, // pct-encoded (and used in IPv6 zones)
	'&':  true, // sub-delims
	'(':  true, // sub-delims
	')':  true, // sub-delims
	'*':  true, // sub-delims
	'+':  true, // sub-delims
	',':  true, // sub-delims
	'-':  true, // unreserved
	'.':  true, // unreserved
	':':  true, // IPv6address + Host expression's optional port
	';':  true, // sub-delims
	'=':  true, // sub-delims
	'[':  true,
	'\'': true, // sub-delims
	']':  true,
	'_':  true, // unreserved
	'~':  true, // unreserved
}
ValidHeaderFieldValue reports whether v is a valid "field-value" according to http://www.w3.org/Protocols/rfc2616/rfc2616-sec4.html#sec4.2 : message-header = field-name ":" [ field-value ] field-value = *( field-content | LWS ) field-content = <the OCTETs making up the field-value and consisting of either *TEXT or combinations of token, separators, and quoted-string> http://www.w3.org/Protocols/rfc2616/rfc2616-sec2.html#sec2.2 : TEXT = <any OCTET except CTLs, but including LWS> LWS = [CRLF] 1*( SP | HT ) CTL = <any US-ASCII control character (octets 0 - 31) and DEL (127)> RFC 7230 says: field-value = *( field-content / obs-fold ) obj-fold = N/A to http2, and deprecated field-content = field-vchar [ 1*( SP / HTAB ) field-vchar ] field-vchar = VCHAR / obs-text obs-text = %x80-FF VCHAR = "any visible [USASCII] character" http2 further says: "Similarly, HTTP/2 allows header field values that are not valid. While most of the values that can be encoded will not alter header field parsing, carriage return (CR, ASCII 0xd), line feed (LF, ASCII 0xa), and the zero character (NUL, ASCII 0x0) might be exploited by an attacker if they are translated verbatim. Any request or response that contains a character not permitted in a header field value MUST be treated as malformed (Section 8.1.2.6). Valid characters are defined by the field-content ABNF rule in Section 3.2 of [RFC7230]." This function does not (yet?) properly handle the rejection of strings that begin or end with SP or HTAB.
func ( string) bool {
	for  := 0;  < len(); ++ {
		 := []
		if isCTL() && !isLWS() {
			return false
		}
	}
	return true
}

func ( string) bool {
	for  := 0;  < len(); ++ {
		if [] >= utf8.RuneSelf {
			return false
		}
	}
	return true
}
PunycodeHostPort returns the IDNA Punycode version of the provided "host" or "host:port" string.
func ( string) (string, error) {
	if isASCII() {
		return , nil
	}

	, ,  := net.SplitHostPort()
The input 'v' argument was just a "host" argument, without a port. This error should not be returned to the caller.
		 = 
		 = ""
	}
	,  = idna.ToASCII()
Non-UTF-8? Not representable in Punycode, in any case.
		return "", 
	}
	if  == "" {
		return , nil
	}
	return net.JoinHostPort(, ), nil