| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297 |
- //go:generate go run bytesconv_table_gen.go
- package fasthttp
- import (
- "bufio"
- "bytes"
- "errors"
- "fmt"
- "io"
- "net"
- "strconv"
- "sync"
- "time"
- )
- // AppendHTMLEscape appends html-escaped s to dst and returns the extended dst.
- func AppendHTMLEscape(dst []byte, s string) []byte {
- var (
- prev int
- sub string
- )
- for i, n := 0, len(s); i < n; i++ {
- sub = ""
- switch s[i] {
- case '&':
- sub = "&"
- case '<':
- sub = "<"
- case '>':
- sub = ">"
- case '"':
- sub = """ // """ is shorter than """.
- case '\'':
- sub = "'" // "'" is shorter than "'" and apos was not in HTML until HTML5.
- }
- if sub != "" {
- dst = append(dst, s[prev:i]...)
- dst = append(dst, sub...)
- prev = i + 1
- }
- }
- return append(dst, s[prev:]...)
- }
- // AppendHTMLEscapeBytes appends html-escaped s to dst and returns
- // the extended dst.
- func AppendHTMLEscapeBytes(dst, s []byte) []byte {
- return AppendHTMLEscape(dst, b2s(s))
- }
- // AppendIPv4 appends string representation of the given ip v4 to dst
- // and returns the extended dst.
- func AppendIPv4(dst []byte, ip net.IP) []byte {
- ip = ip.To4()
- if ip == nil {
- return append(dst, "non-v4 ip passed to AppendIPv4"...)
- }
- dst = AppendUint(dst, int(ip[0]))
- for i := 1; i < 4; i++ {
- dst = append(dst, '.')
- dst = AppendUint(dst, int(ip[i]))
- }
- return dst
- }
- var errEmptyIPStr = errors.New("empty ip address string")
- // ParseIPv4 parses ip address from ipStr into dst and returns the extended dst.
- func ParseIPv4(dst net.IP, ipStr []byte) (net.IP, error) {
- if len(ipStr) == 0 {
- return dst, errEmptyIPStr
- }
- if len(dst) < net.IPv4len || len(dst) > net.IPv4len {
- dst = make([]byte, net.IPv4len)
- }
- copy(dst, net.IPv4zero)
- dst = dst.To4() // dst is always non-nil here
- b := ipStr
- for i := 0; i < 3; i++ {
- n := bytes.IndexByte(b, '.')
- if n < 0 {
- return dst, fmt.Errorf("cannot find dot in ipStr %q", ipStr)
- }
- v, err := ParseUint(b[:n])
- if err != nil {
- return dst, fmt.Errorf("cannot parse ipStr %q: %w", ipStr, err)
- }
- if v > 255 {
- return dst, fmt.Errorf("cannot parse ipStr %q: ip part cannot exceed 255: parsed %d", ipStr, v)
- }
- dst[i] = byte(v)
- b = b[n+1:]
- }
- v, err := ParseUint(b)
- if err != nil {
- return dst, fmt.Errorf("cannot parse ipStr %q: %w", ipStr, err)
- }
- if v > 255 {
- return dst, fmt.Errorf("cannot parse ipStr %q: ip part cannot exceed 255: parsed %d", ipStr, v)
- }
- dst[3] = byte(v)
- return dst, nil
- }
- // AppendHTTPDate appends HTTP-compliant (RFC1123) representation of date
- // to dst and returns the extended dst.
- func AppendHTTPDate(dst []byte, date time.Time) []byte {
- dst = date.In(time.UTC).AppendFormat(dst, time.RFC1123)
- copy(dst[len(dst)-3:], strGMT)
- return dst
- }
- // ParseHTTPDate parses HTTP-compliant (RFC1123) date.
- func ParseHTTPDate(date []byte) (time.Time, error) {
- return time.Parse(time.RFC1123, b2s(date))
- }
- // AppendUint appends n to dst and returns the extended dst.
- func AppendUint(dst []byte, n int) []byte {
- if n < 0 {
- // developer sanity-check
- panic("BUG: int must be positive")
- }
- return strconv.AppendUint(dst, uint64(n), 10)
- }
- // ParseUint parses uint from buf.
- func ParseUint(buf []byte) (int, error) {
- v, n, err := parseUintBuf(buf)
- if n != len(buf) {
- return -1, errUnexpectedTrailingChar
- }
- return v, err
- }
- var (
- errEmptyInt = errors.New("empty integer")
- errUnexpectedFirstChar = errors.New("unexpected first char found. Expecting 0-9")
- errUnexpectedTrailingChar = errors.New("unexpected trailing char found. Expecting 0-9")
- errTooLongInt = errors.New("too long int")
- )
- func parseUintBuf(b []byte) (int, int, error) {
- n := len(b)
- if n == 0 {
- return -1, 0, errEmptyInt
- }
- v := 0
- for i := 0; i < n; i++ {
- c := b[i]
- k := c - '0'
- if k > 9 {
- if i == 0 {
- return -1, i, errUnexpectedFirstChar
- }
- return v, i, nil
- }
- vNew := 10*v + int(k)
- // Test for overflow.
- if vNew < v {
- return -1, i, errTooLongInt
- }
- v = vNew
- }
- return v, n, nil
- }
- // ParseUfloat parses unsigned float from buf.
- func ParseUfloat(buf []byte) (float64, error) {
- // The implementation of parsing a float string is not easy.
- // We believe that the conservative approach is to call strconv.ParseFloat.
- // https://github.com/valyala/fasthttp/pull/1865
- res, err := strconv.ParseFloat(b2s(buf), 64)
- if res < 0 {
- return -1, errors.New("negative input is invalid")
- }
- if err != nil {
- return -1, err
- }
- return res, err
- }
- var (
- errEmptyHexNum = errors.New("empty hex number")
- errTooLargeHexNum = errors.New("too large hex number")
- )
- func readHexInt(r *bufio.Reader) (int, error) {
- var k, i, n int
- for {
- c, err := r.ReadByte()
- if err != nil {
- if err == io.EOF && i > 0 {
- return n, nil
- }
- return -1, err
- }
- k = int(hex2intTable[c])
- if k == 16 {
- if i == 0 {
- return -1, errEmptyHexNum
- }
- if err := r.UnreadByte(); err != nil {
- return -1, err
- }
- return n, nil
- }
- if i >= maxHexIntChars {
- return -1, errTooLargeHexNum
- }
- n = (n << 4) | k
- i++
- }
- }
- var hexIntBufPool sync.Pool
- func writeHexInt(w *bufio.Writer, n int) error {
- if n < 0 {
- // developer sanity-check
- panic("BUG: int must be positive")
- }
- v := hexIntBufPool.Get()
- if v == nil {
- v = make([]byte, maxHexIntChars+1)
- }
- buf := v.([]byte)
- i := len(buf) - 1
- for {
- buf[i] = lowerhex[n&0xf]
- n >>= 4
- if n == 0 {
- break
- }
- i--
- }
- _, err := w.Write(buf[i:])
- hexIntBufPool.Put(v)
- return err
- }
- const (
- upperhex = "0123456789ABCDEF"
- lowerhex = "0123456789abcdef"
- )
- func lowercaseBytes(b []byte) {
- for i := 0; i < len(b); i++ {
- p := &b[i]
- *p = toLowerTable[*p]
- }
- }
- // AppendUnquotedArg appends url-decoded src to dst and returns appended dst.
- //
- // dst may point to src. In this case src will be overwritten.
- func AppendUnquotedArg(dst, src []byte) []byte {
- return decodeArgAppend(dst, src)
- }
- // AppendQuotedArg appends url-encoded src to dst and returns appended dst.
- func AppendQuotedArg(dst, src []byte) []byte {
- for _, c := range src {
- switch {
- case c == ' ':
- dst = append(dst, '+')
- case quotedArgShouldEscapeTable[int(c)] != 0:
- dst = append(dst, '%', upperhex[c>>4], upperhex[c&0xf])
- default:
- dst = append(dst, c)
- }
- }
- return dst
- }
- func appendQuotedPath(dst, src []byte) []byte {
- // Fix issue in https://github.com/golang/go/issues/11202
- if len(src) == 1 && src[0] == '*' {
- return append(dst, '*')
- }
- for _, c := range src {
- if quotedPathShouldEscapeTable[int(c)] != 0 {
- dst = append(dst, '%', upperhex[c>>4], upperhex[c&0xf])
- } else {
- dst = append(dst, c)
- }
- }
- return dst
- }
|