123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357 |
- //go:generate go run bytesconv_table_gen.go
- package fasthttp
- import (
- "bufio"
- "bytes"
- "errors"
- "fmt"
- "io"
- "math"
- "net"
- "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 len(sub) > 0 {
- 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 {
- dst = make([]byte, net.IPv4len)
- }
- copy(dst, net.IPv4zero)
- dst = dst.To4()
- if dst == nil {
- // developer sanity-check
- panic("BUG: dst must not be nil")
- }
- 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")
- }
- var b [20]byte
- buf := b[:]
- i := len(buf)
- var q int
- for n >= 10 {
- i--
- q = n / 10
- buf[i] = '0' + byte(n-q*10)
- n = q
- }
- i--
- buf[i] = '0' + byte(n)
- dst = append(dst, buf[i:]...)
- return dst
- }
- // 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
- }
- var (
- errEmptyFloat = errors.New("empty float number")
- errDuplicateFloatPoint = errors.New("duplicate point found in float number")
- errUnexpectedFloatEnd = errors.New("unexpected end of float number")
- errInvalidFloatExponent = errors.New("invalid float number exponent")
- errUnexpectedFloatChar = errors.New("unexpected char found in float number")
- )
- // ParseUfloat parses unsigned float from buf.
- func ParseUfloat(buf []byte) (float64, error) {
- if len(buf) == 0 {
- return -1, errEmptyFloat
- }
- b := buf
- var v uint64
- offset := 1.0
- var pointFound bool
- for i, c := range b {
- if c < '0' || c > '9' {
- if c == '.' {
- if pointFound {
- return -1, errDuplicateFloatPoint
- }
- pointFound = true
- continue
- }
- if c == 'e' || c == 'E' {
- if i+1 >= len(b) {
- return -1, errUnexpectedFloatEnd
- }
- b = b[i+1:]
- minus := -1
- switch b[0] {
- case '+':
- b = b[1:]
- minus = 1
- case '-':
- b = b[1:]
- default:
- minus = 1
- }
- vv, err := ParseUint(b)
- if err != nil {
- return -1, errInvalidFloatExponent
- }
- return float64(v) * offset * math.Pow10(minus*vv), nil
- }
- return -1, errUnexpectedFloatChar
- }
- v = 10*v + uint64(c-'0')
- if pointFound {
- offset /= 10
- }
- }
- return float64(v) * offset, nil
- }
- 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
- }
|