mirror of
				https://github.com/superseriousbusiness/gotosocial.git
				synced 2025-11-03 21:52:25 -06:00 
			
		
		
		
	- github.com/KimMachineGun/automemlimit v0.7.2 => v0.7.3
- github.com/gin-contrib/cors v1.7.5 => v1.7.6
- github.com/minio/minio-go/v7 v7.0.92 => v7.0.94
- github.com/spf13/cast v1.8.0 => v1.9.2
- github.com/uptrace/bun{,/*} v1.2.11 => v1.2.14
- golang.org/x/image v0.27.0 => v0.28.0
- golang.org/x/net v0.40.0 => v0.41.0
- code.superseriousbusiness.org/go-swagger v0.31.0-gts-go1.23-fix => v0.32.3-gts-go1.23-fix
Reviewed-on: https://codeberg.org/superseriousbusiness/gotosocial/pulls/4304
Co-authored-by: kim <grufwub@gmail.com>
Co-committed-by: kim <grufwub@gmail.com>
		
	
			
		
			
				
	
	
		
			1225 lines
		
	
	
	
		
			28 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			1225 lines
		
	
	
	
		
			28 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
//go:build notmono || codec.notmono
 | 
						|
 | 
						|
// Copyright (c) 2012-2020 Ugorji Nwoke. All rights reserved.
 | 
						|
// Use of this source code is governed by a MIT license found in the LICENSE file.
 | 
						|
 | 
						|
package codec
 | 
						|
 | 
						|
// By default, this json support uses base64 encoding for bytes, because you cannot
 | 
						|
// store and read any arbitrary string in json (only unicode).
 | 
						|
// However, the user can configre how to encode/decode bytes.
 | 
						|
//
 | 
						|
// This library specifically supports UTF-8 for encoding and decoding only.
 | 
						|
//
 | 
						|
// Note that the library will happily encode/decode things which are not valid
 | 
						|
// json e.g. a map[int64]string. We do it for consistency. With valid json,
 | 
						|
// we will encode and decode appropriately.
 | 
						|
// Users can specify their map type if necessary to force it.
 | 
						|
//
 | 
						|
// We cannot use strconv.(Q|Unq)uote because json quotes/unquotes differently.
 | 
						|
 | 
						|
import (
 | 
						|
	"encoding/base64"
 | 
						|
	"io"
 | 
						|
	"math"
 | 
						|
	"reflect"
 | 
						|
	"strconv"
 | 
						|
	"time"
 | 
						|
	"unicode"
 | 
						|
	"unicode/utf16"
 | 
						|
	"unicode/utf8"
 | 
						|
)
 | 
						|
 | 
						|
type jsonEncDriver[T encWriter] struct {
 | 
						|
	noBuiltInTypes
 | 
						|
	h *JsonHandle
 | 
						|
	e *encoderBase
 | 
						|
	s *bitset256 // safe set for characters (taking h.HTMLAsIs into consideration)
 | 
						|
 | 
						|
	w T
 | 
						|
	// se interfaceExtWrapper
 | 
						|
 | 
						|
	enc encoderI
 | 
						|
 | 
						|
	timeFmtLayout string
 | 
						|
	byteFmter     jsonBytesFmter
 | 
						|
	// ---- cpu cache line boundary???
 | 
						|
 | 
						|
	// bytes2Arr bool
 | 
						|
	// time2Num  bool
 | 
						|
	timeFmt  jsonTimeFmt
 | 
						|
	bytesFmt jsonBytesFmt
 | 
						|
 | 
						|
	di int8   // indent per: if negative, use tabs
 | 
						|
	d  bool   // indenting?
 | 
						|
	dl uint16 // indent level
 | 
						|
 | 
						|
	ks bool // map key as string
 | 
						|
	is byte // integer as string
 | 
						|
 | 
						|
	typical bool
 | 
						|
 | 
						|
	rawext bool // rawext configured on the handle
 | 
						|
 | 
						|
	// buf *[]byte // used mostly for encoding []byte
 | 
						|
 | 
						|
	// scratch buffer for: encode time, numbers, etc
 | 
						|
	//
 | 
						|
	// RFC3339Nano uses 35 chars: 2006-01-02T15:04:05.999999999Z07:00
 | 
						|
	// MaxUint64 uses 20 chars: 18446744073709551615
 | 
						|
	// floats are encoded using: f/e fmt, and -1 precision, or 1 if no fractions.
 | 
						|
	// This means we are limited by the number of characters for the
 | 
						|
	// mantissa (up to 17), exponent (up to 3), signs (up to 3), dot (up to 1), E (up to 1)
 | 
						|
	// for a total of 24 characters.
 | 
						|
	//    -xxx.yyyyyyyyyyyye-zzz
 | 
						|
	// Consequently, 35 characters should be sufficient for encoding time, integers or floats.
 | 
						|
	// We use up all the remaining bytes to make this use full cache lines.
 | 
						|
	b [48]byte
 | 
						|
}
 | 
						|
 | 
						|
func (e *jsonEncDriver[T]) writeIndent() {
 | 
						|
	e.w.writen1('\n')
 | 
						|
	x := int(e.di) * int(e.dl)
 | 
						|
	if e.di < 0 {
 | 
						|
		x = -x
 | 
						|
		for x > len(jsonTabs) {
 | 
						|
			e.w.writeb(jsonTabs[:])
 | 
						|
			x -= len(jsonTabs)
 | 
						|
		}
 | 
						|
		e.w.writeb(jsonTabs[:x])
 | 
						|
	} else {
 | 
						|
		for x > len(jsonSpaces) {
 | 
						|
			e.w.writeb(jsonSpaces[:])
 | 
						|
			x -= len(jsonSpaces)
 | 
						|
		}
 | 
						|
		e.w.writeb(jsonSpaces[:x])
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
func (e *jsonEncDriver[T]) WriteArrayElem(firstTime bool) {
 | 
						|
	if !firstTime {
 | 
						|
		e.w.writen1(',')
 | 
						|
	}
 | 
						|
	if e.d {
 | 
						|
		e.writeIndent()
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
func (e *jsonEncDriver[T]) WriteMapElemKey(firstTime bool) {
 | 
						|
	if !firstTime {
 | 
						|
		e.w.writen1(',')
 | 
						|
	}
 | 
						|
	if e.d {
 | 
						|
		e.writeIndent()
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
func (e *jsonEncDriver[T]) WriteMapElemValue() {
 | 
						|
	if e.d {
 | 
						|
		e.w.writen2(':', ' ')
 | 
						|
	} else {
 | 
						|
		e.w.writen1(':')
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
func (e *jsonEncDriver[T]) EncodeNil() {
 | 
						|
	// We always encode nil as just null (never in quotes)
 | 
						|
	// so we can easily decode if a nil in the json stream ie if initial token is n.
 | 
						|
 | 
						|
	// e.w.writestr(jsonLits[jsonLitN : jsonLitN+4])
 | 
						|
	e.w.writeb(jsonNull)
 | 
						|
}
 | 
						|
 | 
						|
func (e *jsonEncDriver[T]) encodeIntAsUint(v int64, quotes bool) {
 | 
						|
	neg := v < 0
 | 
						|
	if neg {
 | 
						|
		v = -v
 | 
						|
	}
 | 
						|
	e.encodeUint(neg, quotes, uint64(v))
 | 
						|
}
 | 
						|
 | 
						|
func (e *jsonEncDriver[T]) EncodeTime(t time.Time) {
 | 
						|
	// Do NOT use MarshalJSON, as it allocates internally.
 | 
						|
	// instead, we call AppendFormat directly, using our scratch buffer (e.b)
 | 
						|
 | 
						|
	if t.IsZero() {
 | 
						|
		e.EncodeNil()
 | 
						|
		return
 | 
						|
	}
 | 
						|
	switch e.timeFmt {
 | 
						|
	case jsonTimeFmtStringLayout:
 | 
						|
		e.b[0] = '"'
 | 
						|
		b := t.AppendFormat(e.b[1:1], e.timeFmtLayout)
 | 
						|
		e.b[len(b)+1] = '"'
 | 
						|
		e.w.writeb(e.b[:len(b)+2])
 | 
						|
	case jsonTimeFmtUnix:
 | 
						|
		e.encodeIntAsUint(t.Unix(), false)
 | 
						|
	case jsonTimeFmtUnixMilli:
 | 
						|
		e.encodeIntAsUint(t.UnixMilli(), false)
 | 
						|
	case jsonTimeFmtUnixMicro:
 | 
						|
		e.encodeIntAsUint(t.UnixMicro(), false)
 | 
						|
	case jsonTimeFmtUnixNano:
 | 
						|
		e.encodeIntAsUint(t.UnixNano(), false)
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
func (e *jsonEncDriver[T]) EncodeExt(rv interface{}, basetype reflect.Type, xtag uint64, ext Ext) {
 | 
						|
	if ext == SelfExt {
 | 
						|
		e.enc.encodeAs(rv, basetype, false)
 | 
						|
	} else if v := ext.ConvertExt(rv); v == nil {
 | 
						|
		e.writeNilBytes()
 | 
						|
	} else {
 | 
						|
		e.enc.encodeI(v)
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
func (e *jsonEncDriver[T]) EncodeRawExt(re *RawExt) {
 | 
						|
	if re.Data != nil {
 | 
						|
		e.w.writeb(re.Data)
 | 
						|
	} else if re.Value != nil {
 | 
						|
		e.enc.encodeI(re.Value)
 | 
						|
	} else {
 | 
						|
		e.EncodeNil()
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
func (e *jsonEncDriver[T]) EncodeBool(b bool) {
 | 
						|
	e.w.writestr(jsonEncBoolStrs[bool2int(e.ks && e.e.c == containerMapKey)%2][bool2int(b)%2])
 | 
						|
}
 | 
						|
 | 
						|
func (e *jsonEncDriver[T]) encodeFloat(f float64, bitsize, fmt byte, prec int8) {
 | 
						|
	var blen uint
 | 
						|
	if e.ks && e.e.c == containerMapKey {
 | 
						|
		blen = 2 + uint(len(strconv.AppendFloat(e.b[1:1], f, fmt, int(prec), int(bitsize))))
 | 
						|
		// _ = e.b[:blen]
 | 
						|
		e.b[0] = '"'
 | 
						|
		e.b[blen-1] = '"'
 | 
						|
		e.w.writeb(e.b[:blen])
 | 
						|
	} else {
 | 
						|
		e.w.writeb(strconv.AppendFloat(e.b[:0], f, fmt, int(prec), int(bitsize)))
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
func (e *jsonEncDriver[T]) EncodeFloat64(f float64) {
 | 
						|
	if math.IsNaN(f) || math.IsInf(f, 0) {
 | 
						|
		e.EncodeNil()
 | 
						|
		return
 | 
						|
	}
 | 
						|
	fmt, prec := jsonFloatStrconvFmtPrec64(f)
 | 
						|
	e.encodeFloat(f, 64, fmt, prec)
 | 
						|
}
 | 
						|
 | 
						|
func (e *jsonEncDriver[T]) EncodeFloat32(f float32) {
 | 
						|
	if math.IsNaN(float64(f)) || math.IsInf(float64(f), 0) {
 | 
						|
		e.EncodeNil()
 | 
						|
		return
 | 
						|
	}
 | 
						|
	fmt, prec := jsonFloatStrconvFmtPrec32(f)
 | 
						|
	e.encodeFloat(float64(f), 32, fmt, prec)
 | 
						|
}
 | 
						|
 | 
						|
func (e *jsonEncDriver[T]) encodeUint(neg bool, quotes bool, u uint64) {
 | 
						|
	e.w.writeb(jsonEncodeUint(neg, quotes, u, &e.b))
 | 
						|
}
 | 
						|
 | 
						|
func (e *jsonEncDriver[T]) EncodeInt(v int64) {
 | 
						|
	quotes := e.is == 'A' || e.is == 'L' && (v > 1<<53 || v < -(1<<53)) ||
 | 
						|
		(e.ks && e.e.c == containerMapKey)
 | 
						|
 | 
						|
	if cpu32Bit {
 | 
						|
		if quotes {
 | 
						|
			blen := 2 + len(strconv.AppendInt(e.b[1:1], v, 10))
 | 
						|
			e.b[0] = '"'
 | 
						|
			e.b[blen-1] = '"'
 | 
						|
			e.w.writeb(e.b[:blen])
 | 
						|
		} else {
 | 
						|
			e.w.writeb(strconv.AppendInt(e.b[:0], v, 10))
 | 
						|
		}
 | 
						|
		return
 | 
						|
	}
 | 
						|
 | 
						|
	if v < 0 {
 | 
						|
		e.encodeUint(true, quotes, uint64(-v))
 | 
						|
	} else {
 | 
						|
		e.encodeUint(false, quotes, uint64(v))
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
func (e *jsonEncDriver[T]) EncodeUint(v uint64) {
 | 
						|
	quotes := e.is == 'A' || e.is == 'L' && v > 1<<53 ||
 | 
						|
		(e.ks && e.e.c == containerMapKey)
 | 
						|
 | 
						|
	if cpu32Bit {
 | 
						|
		// use strconv directly, as optimized encodeUint only works on 64-bit alone
 | 
						|
		if quotes {
 | 
						|
			blen := 2 + len(strconv.AppendUint(e.b[1:1], v, 10))
 | 
						|
			e.b[0] = '"'
 | 
						|
			e.b[blen-1] = '"'
 | 
						|
			e.w.writeb(e.b[:blen])
 | 
						|
		} else {
 | 
						|
			e.w.writeb(strconv.AppendUint(e.b[:0], v, 10))
 | 
						|
		}
 | 
						|
		return
 | 
						|
	}
 | 
						|
 | 
						|
	e.encodeUint(false, quotes, v)
 | 
						|
}
 | 
						|
 | 
						|
func (e *jsonEncDriver[T]) EncodeString(v string) {
 | 
						|
	if e.h.StringToRaw {
 | 
						|
		e.EncodeStringBytesRaw(bytesView(v))
 | 
						|
		return
 | 
						|
	}
 | 
						|
	e.quoteStr(v)
 | 
						|
}
 | 
						|
 | 
						|
func (e *jsonEncDriver[T]) EncodeStringNoEscape4Json(v string) { e.w.writeqstr(v) }
 | 
						|
 | 
						|
func (e *jsonEncDriver[T]) EncodeStringBytesRaw(v []byte) {
 | 
						|
	if e.rawext {
 | 
						|
		// explicitly convert v to interface{} so that v doesn't escape to heap
 | 
						|
		iv := e.h.RawBytesExt.ConvertExt(any(v))
 | 
						|
		if iv == nil {
 | 
						|
			e.EncodeNil()
 | 
						|
		} else {
 | 
						|
			e.enc.encodeI(iv)
 | 
						|
		}
 | 
						|
		return
 | 
						|
	}
 | 
						|
 | 
						|
	if e.bytesFmt == jsonBytesFmtArray {
 | 
						|
		e.WriteArrayStart(len(v))
 | 
						|
		for j := range v {
 | 
						|
			e.WriteArrayElem(j == 0)
 | 
						|
			e.encodeUint(false, false, uint64(v[j]))
 | 
						|
		}
 | 
						|
		e.WriteArrayEnd()
 | 
						|
		return
 | 
						|
	}
 | 
						|
 | 
						|
	// hardcode base64, so we call direct (not via interface) and hopefully inline
 | 
						|
	var slen int
 | 
						|
	if e.bytesFmt == jsonBytesFmtBase64 {
 | 
						|
		slen = base64.StdEncoding.EncodedLen(len(v))
 | 
						|
	} else {
 | 
						|
		slen = e.byteFmter.EncodedLen(len(v))
 | 
						|
	}
 | 
						|
	slen += 2
 | 
						|
 | 
						|
	// bs := e.e.blist.check(*e.buf, n)[:slen]
 | 
						|
	// *e.buf = bs
 | 
						|
 | 
						|
	bs := e.e.blist.peek(slen, false)[:slen]
 | 
						|
 | 
						|
	if e.bytesFmt == jsonBytesFmtBase64 {
 | 
						|
		base64.StdEncoding.Encode(bs[1:], v)
 | 
						|
	} else {
 | 
						|
		e.byteFmter.Encode(bs[1:], v)
 | 
						|
	}
 | 
						|
 | 
						|
	bs[len(bs)-1] = '"'
 | 
						|
	bs[0] = '"'
 | 
						|
	e.w.writeb(bs)
 | 
						|
}
 | 
						|
 | 
						|
func (e *jsonEncDriver[T]) EncodeBytes(v []byte) {
 | 
						|
	if v == nil {
 | 
						|
		e.writeNilBytes()
 | 
						|
		return
 | 
						|
	}
 | 
						|
	e.EncodeStringBytesRaw(v)
 | 
						|
}
 | 
						|
 | 
						|
func (e *jsonEncDriver[T]) writeNilOr(v []byte) {
 | 
						|
	if !e.h.NilCollectionToZeroLength {
 | 
						|
		v = jsonNull
 | 
						|
	}
 | 
						|
	e.w.writeb(v)
 | 
						|
}
 | 
						|
 | 
						|
func (e *jsonEncDriver[T]) writeNilBytes() {
 | 
						|
	e.writeNilOr(jsonArrayEmpty)
 | 
						|
}
 | 
						|
 | 
						|
func (e *jsonEncDriver[T]) writeNilArray() {
 | 
						|
	e.writeNilOr(jsonArrayEmpty)
 | 
						|
}
 | 
						|
 | 
						|
func (e *jsonEncDriver[T]) writeNilMap() {
 | 
						|
	e.writeNilOr(jsonMapEmpty)
 | 
						|
}
 | 
						|
 | 
						|
// indent is done as below:
 | 
						|
//   - newline and indent are added before each mapKey or arrayElem
 | 
						|
//   - newline and indent are added before each ending,
 | 
						|
//     except there was no entry (so we can have {} or [])
 | 
						|
 | 
						|
func (e *jsonEncDriver[T]) WriteArrayEmpty() {
 | 
						|
	e.w.writen2('[', ']')
 | 
						|
}
 | 
						|
 | 
						|
func (e *jsonEncDriver[T]) WriteMapEmpty() {
 | 
						|
	e.w.writen2('{', '}')
 | 
						|
}
 | 
						|
 | 
						|
func (e *jsonEncDriver[T]) WriteArrayStart(length int) {
 | 
						|
	if e.d {
 | 
						|
		e.dl++
 | 
						|
	}
 | 
						|
	e.w.writen1('[')
 | 
						|
}
 | 
						|
 | 
						|
func (e *jsonEncDriver[T]) WriteArrayEnd() {
 | 
						|
	if e.d {
 | 
						|
		e.dl--
 | 
						|
		// No need as encoder handles zero-len already
 | 
						|
		// if e.e.c != containerArrayStart {
 | 
						|
		e.writeIndent()
 | 
						|
	}
 | 
						|
	e.w.writen1(']')
 | 
						|
}
 | 
						|
 | 
						|
func (e *jsonEncDriver[T]) WriteMapStart(length int) {
 | 
						|
	if e.d {
 | 
						|
		e.dl++
 | 
						|
	}
 | 
						|
	e.w.writen1('{')
 | 
						|
}
 | 
						|
 | 
						|
func (e *jsonEncDriver[T]) WriteMapEnd() {
 | 
						|
	if e.d {
 | 
						|
		e.dl--
 | 
						|
		// No need as encoder handles zero-len already
 | 
						|
		// if e.e.c != containerMapStart {
 | 
						|
		e.writeIndent()
 | 
						|
	}
 | 
						|
	e.w.writen1('}')
 | 
						|
}
 | 
						|
 | 
						|
func (e *jsonEncDriver[T]) quoteStr(s string) {
 | 
						|
	// adapted from std pkg encoding/json
 | 
						|
	const hex = "0123456789abcdef"
 | 
						|
	e.w.writen1('"')
 | 
						|
	var i, start uint
 | 
						|
	for i < uint(len(s)) {
 | 
						|
		// encode all bytes < 0x20 (except \r, \n).
 | 
						|
		// also encode < > & to prevent security holes when served to some browsers.
 | 
						|
 | 
						|
		// We optimize for ascii, by assuming that most characters are in the BMP
 | 
						|
		// and natively consumed by json without much computation.
 | 
						|
 | 
						|
		// if 0x20 <= b && b != '\\' && b != '"' && b != '<' && b != '>' && b != '&' {
 | 
						|
		// if (htmlasis && jsonCharSafeSet.isset(b)) || jsonCharHtmlSafeSet.isset(b) {
 | 
						|
		b := s[i]
 | 
						|
		if e.s.isset(b) {
 | 
						|
			i++
 | 
						|
			continue
 | 
						|
		}
 | 
						|
		if b < utf8.RuneSelf {
 | 
						|
			if start < i {
 | 
						|
				e.w.writestr(s[start:i])
 | 
						|
			}
 | 
						|
			switch b {
 | 
						|
			case '\\':
 | 
						|
				e.w.writen2('\\', '\\')
 | 
						|
			case '"':
 | 
						|
				e.w.writen2('\\', '"')
 | 
						|
			case '\n':
 | 
						|
				e.w.writen2('\\', 'n')
 | 
						|
			case '\t':
 | 
						|
				e.w.writen2('\\', 't')
 | 
						|
			case '\r':
 | 
						|
				e.w.writen2('\\', 'r')
 | 
						|
			case '\b':
 | 
						|
				e.w.writen2('\\', 'b')
 | 
						|
			case '\f':
 | 
						|
				e.w.writen2('\\', 'f')
 | 
						|
			default:
 | 
						|
				e.w.writestr(`\u00`)
 | 
						|
				e.w.writen2(hex[b>>4], hex[b&0xF])
 | 
						|
			}
 | 
						|
			i++
 | 
						|
			start = i
 | 
						|
			continue
 | 
						|
		}
 | 
						|
		c, size := utf8.DecodeRuneInString(s[i:])
 | 
						|
		if c == utf8.RuneError && size == 1 { // meaning invalid encoding (so output as-is)
 | 
						|
			if start < i {
 | 
						|
				e.w.writestr(s[start:i])
 | 
						|
			}
 | 
						|
			e.w.writestr(`\uFFFD`)
 | 
						|
			i++
 | 
						|
			start = i
 | 
						|
			continue
 | 
						|
		}
 | 
						|
		// U+2028 is LINE SEPARATOR. U+2029 is PARAGRAPH SEPARATOR.
 | 
						|
		// Both technically valid JSON, but bomb on JSONP, so fix here *unconditionally*.
 | 
						|
		if jsonEscapeMultiByteUnicodeSep && (c == '\u2028' || c == '\u2029') {
 | 
						|
			if start < i {
 | 
						|
				e.w.writestr(s[start:i])
 | 
						|
			}
 | 
						|
			e.w.writestr(`\u202`)
 | 
						|
			e.w.writen1(hex[c&0xF])
 | 
						|
			i += uint(size)
 | 
						|
			start = i
 | 
						|
			continue
 | 
						|
		}
 | 
						|
		i += uint(size)
 | 
						|
	}
 | 
						|
	if start < uint(len(s)) {
 | 
						|
		e.w.writestr(s[start:])
 | 
						|
	}
 | 
						|
	e.w.writen1('"')
 | 
						|
}
 | 
						|
 | 
						|
func (e *jsonEncDriver[T]) atEndOfEncode() {
 | 
						|
	if e.h.TermWhitespace {
 | 
						|
		var c byte = ' ' // default is that scalar is written, so output space
 | 
						|
		if e.e.c != 0 {
 | 
						|
			c = '\n' // for containers (map/list), output a newline
 | 
						|
		}
 | 
						|
		e.w.writen1(c)
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
// ----------
 | 
						|
 | 
						|
type jsonDecDriver[T decReader] struct {
 | 
						|
	noBuiltInTypes
 | 
						|
	decDriverNoopNumberHelper
 | 
						|
	h *JsonHandle
 | 
						|
	d *decoderBase
 | 
						|
 | 
						|
	r T
 | 
						|
 | 
						|
	// scratch buffer used for base64 decoding (DecodeBytes in reuseBuf mode),
 | 
						|
	// or reading doubleQuoted string (DecodeStringAsBytes, DecodeNaked)
 | 
						|
	buf []byte
 | 
						|
 | 
						|
	tok  uint8   // used to store the token read right after skipWhiteSpace
 | 
						|
	_    bool    // found null
 | 
						|
	_    byte    // padding
 | 
						|
	bstr [4]byte // scratch used for string \UXXX parsing
 | 
						|
 | 
						|
	jsonHandleOpts
 | 
						|
 | 
						|
	// se  interfaceExtWrapper
 | 
						|
 | 
						|
	// ---- cpu cache line boundary?
 | 
						|
 | 
						|
	// bytes bool
 | 
						|
 | 
						|
	dec decoderI
 | 
						|
}
 | 
						|
 | 
						|
func (d *jsonDecDriver[T]) ReadMapStart() int {
 | 
						|
	d.advance()
 | 
						|
	if d.tok == 'n' {
 | 
						|
		d.checkLit3([3]byte{'u', 'l', 'l'}, d.r.readn3())
 | 
						|
		return containerLenNil
 | 
						|
	}
 | 
						|
	if d.tok != '{' {
 | 
						|
		halt.errorByte("read map - expect char '{' but got char: ", d.tok)
 | 
						|
	}
 | 
						|
	d.tok = 0
 | 
						|
	return containerLenUnknown
 | 
						|
}
 | 
						|
 | 
						|
func (d *jsonDecDriver[T]) ReadArrayStart() int {
 | 
						|
	d.advance()
 | 
						|
	if d.tok == 'n' {
 | 
						|
		d.checkLit3([3]byte{'u', 'l', 'l'}, d.r.readn3())
 | 
						|
		return containerLenNil
 | 
						|
	}
 | 
						|
	if d.tok != '[' {
 | 
						|
		halt.errorByte("read array - expect char '[' but got char ", d.tok)
 | 
						|
	}
 | 
						|
	d.tok = 0
 | 
						|
	return containerLenUnknown
 | 
						|
}
 | 
						|
 | 
						|
// MARKER:
 | 
						|
// We attempted making sure CheckBreak can be inlined, by moving the skipWhitespace
 | 
						|
// call to an explicit (noinline) function call.
 | 
						|
// However, this forces CheckBreak to always incur a function call if there was whitespace,
 | 
						|
// with no clear benefit.
 | 
						|
 | 
						|
func (d *jsonDecDriver[T]) CheckBreak() bool {
 | 
						|
	d.advance()
 | 
						|
	return d.tok == '}' || d.tok == ']'
 | 
						|
}
 | 
						|
 | 
						|
func (d *jsonDecDriver[T]) checkSep(xc byte) {
 | 
						|
	d.advance()
 | 
						|
	if d.tok != xc {
 | 
						|
		d.readDelimError(xc)
 | 
						|
	}
 | 
						|
	d.tok = 0
 | 
						|
}
 | 
						|
 | 
						|
func (d *jsonDecDriver[T]) ReadArrayElem(firstTime bool) {
 | 
						|
	if !firstTime {
 | 
						|
		d.checkSep(',')
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
func (d *jsonDecDriver[T]) ReadArrayEnd() {
 | 
						|
	d.checkSep(']')
 | 
						|
}
 | 
						|
 | 
						|
func (d *jsonDecDriver[T]) ReadMapElemKey(firstTime bool) {
 | 
						|
	d.ReadArrayElem(firstTime)
 | 
						|
}
 | 
						|
 | 
						|
func (d *jsonDecDriver[T]) ReadMapElemValue() {
 | 
						|
	d.checkSep(':')
 | 
						|
}
 | 
						|
 | 
						|
func (d *jsonDecDriver[T]) ReadMapEnd() {
 | 
						|
	d.checkSep('}')
 | 
						|
}
 | 
						|
 | 
						|
//go:inline
 | 
						|
func (d *jsonDecDriver[T]) readDelimError(xc uint8) {
 | 
						|
	halt.errorf("read json delimiter - expect char '%c' but got char '%c'", xc, d.tok)
 | 
						|
}
 | 
						|
 | 
						|
// MARKER: checkLit takes the readn(3|4) result as a parameter so they can be inlined.
 | 
						|
// We pass the array directly to errorf, as passing slice pushes past inlining threshold,
 | 
						|
// and passing slice also might cause allocation of the bs array on the heap.
 | 
						|
 | 
						|
func (d *jsonDecDriver[T]) checkLit3(got, expect [3]byte) {
 | 
						|
	if jsonValidateSymbols && got != expect {
 | 
						|
		jsonCheckLitErr3(got, expect)
 | 
						|
	}
 | 
						|
	d.tok = 0
 | 
						|
}
 | 
						|
 | 
						|
func (d *jsonDecDriver[T]) checkLit4(got, expect [4]byte) {
 | 
						|
	if jsonValidateSymbols && got != expect {
 | 
						|
		jsonCheckLitErr4(got, expect)
 | 
						|
	}
 | 
						|
	d.tok = 0
 | 
						|
}
 | 
						|
 | 
						|
func (d *jsonDecDriver[T]) skipWhitespace() {
 | 
						|
	d.tok = d.r.skipWhitespace()
 | 
						|
}
 | 
						|
 | 
						|
func (d *jsonDecDriver[T]) advance() {
 | 
						|
	// handles jsonReadNum returning possibly non-printable value as tok
 | 
						|
	if d.tok < 33 { // d.tok == 0 {
 | 
						|
		d.skipWhitespace()
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
func (d *jsonDecDriver[T]) nextValueBytes() []byte {
 | 
						|
	consumeString := func() {
 | 
						|
	TOP:
 | 
						|
		_, c := d.r.jsonReadAsisChars()
 | 
						|
		if c == '\\' { // consume next one and try again
 | 
						|
			d.r.readn1()
 | 
						|
			goto TOP
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	d.advance() // ignore leading whitespace
 | 
						|
	d.r.startRecording()
 | 
						|
 | 
						|
	// cursor = d.d.rb.c - 1 // cursor starts just before non-whitespace token
 | 
						|
	switch d.tok {
 | 
						|
	default:
 | 
						|
		_, d.tok = d.r.jsonReadNum()
 | 
						|
	case 'n':
 | 
						|
		d.checkLit3([3]byte{'u', 'l', 'l'}, d.r.readn3())
 | 
						|
	case 'f':
 | 
						|
		d.checkLit4([4]byte{'a', 'l', 's', 'e'}, d.r.readn4())
 | 
						|
	case 't':
 | 
						|
		d.checkLit3([3]byte{'r', 'u', 'e'}, d.r.readn3())
 | 
						|
	case '"':
 | 
						|
		consumeString()
 | 
						|
		d.tok = 0
 | 
						|
	case '{', '[':
 | 
						|
		var elem struct{}
 | 
						|
		var stack []struct{}
 | 
						|
 | 
						|
		stack = append(stack, elem)
 | 
						|
 | 
						|
		for len(stack) != 0 {
 | 
						|
			c := d.r.readn1()
 | 
						|
			switch c {
 | 
						|
			case '"':
 | 
						|
				consumeString()
 | 
						|
			case '{', '[':
 | 
						|
				stack = append(stack, elem)
 | 
						|
			case '}', ']':
 | 
						|
				stack = stack[:len(stack)-1]
 | 
						|
			}
 | 
						|
		}
 | 
						|
		d.tok = 0
 | 
						|
	}
 | 
						|
	return d.r.stopRecording()
 | 
						|
}
 | 
						|
 | 
						|
func (d *jsonDecDriver[T]) TryNil() bool {
 | 
						|
	d.advance()
 | 
						|
	// we don't try to see if quoted "null" was here.
 | 
						|
	// only the plain string: null denotes a nil (ie not quotes)
 | 
						|
	if d.tok == 'n' {
 | 
						|
		d.checkLit3([3]byte{'u', 'l', 'l'}, d.r.readn3())
 | 
						|
		return true
 | 
						|
	}
 | 
						|
	return false
 | 
						|
}
 | 
						|
 | 
						|
func (d *jsonDecDriver[T]) DecodeBool() (v bool) {
 | 
						|
	d.advance()
 | 
						|
	// bool can be in quotes if and only if it's a map key
 | 
						|
	fquot := d.d.c == containerMapKey && d.tok == '"'
 | 
						|
	if fquot {
 | 
						|
		d.tok = d.r.readn1()
 | 
						|
	}
 | 
						|
	switch d.tok {
 | 
						|
	case 'f':
 | 
						|
		d.checkLit4([4]byte{'a', 'l', 's', 'e'}, d.r.readn4())
 | 
						|
		// v = false
 | 
						|
	case 't':
 | 
						|
		d.checkLit3([3]byte{'r', 'u', 'e'}, d.r.readn3())
 | 
						|
		v = true
 | 
						|
	case 'n':
 | 
						|
		d.checkLit3([3]byte{'u', 'l', 'l'}, d.r.readn3())
 | 
						|
		// v = false
 | 
						|
	default:
 | 
						|
		halt.errorByte("decode bool: got first char: ", d.tok)
 | 
						|
		// v = false // "unreachable"
 | 
						|
	}
 | 
						|
	if fquot {
 | 
						|
		d.r.readn1()
 | 
						|
	}
 | 
						|
	return
 | 
						|
}
 | 
						|
 | 
						|
func (d *jsonDecDriver[T]) DecodeTime() (t time.Time) {
 | 
						|
	// read string, and pass the string into json.unmarshal
 | 
						|
	d.advance()
 | 
						|
	if d.tok == 'n' {
 | 
						|
		d.checkLit3([3]byte{'u', 'l', 'l'}, d.r.readn3())
 | 
						|
		return
 | 
						|
	}
 | 
						|
	var bs []byte
 | 
						|
	// if a number, use the timeFmtNum
 | 
						|
	if d.tok != '"' {
 | 
						|
		bs, d.tok = d.r.jsonReadNum()
 | 
						|
		i := d.parseInt64(bs)
 | 
						|
		switch d.timeFmtNum {
 | 
						|
		case jsonTimeFmtUnix:
 | 
						|
			t = time.Unix(i, 0)
 | 
						|
		case jsonTimeFmtUnixMilli:
 | 
						|
			t = time.UnixMilli(i)
 | 
						|
		case jsonTimeFmtUnixMicro:
 | 
						|
			t = time.UnixMicro(i)
 | 
						|
		case jsonTimeFmtUnixNano:
 | 
						|
			t = time.Unix(0, i)
 | 
						|
		default:
 | 
						|
			halt.errorStr("invalid timeFmtNum")
 | 
						|
		}
 | 
						|
		return
 | 
						|
	}
 | 
						|
 | 
						|
	// d.tok is now '"'
 | 
						|
	// d.ensureReadingString()
 | 
						|
	bs = d.readUnescapedString()
 | 
						|
	var err error
 | 
						|
	for _, v := range d.timeFmtLayouts {
 | 
						|
		t, err = time.Parse(v, stringView(bs))
 | 
						|
		if err == nil {
 | 
						|
			return
 | 
						|
		}
 | 
						|
	}
 | 
						|
	halt.errorStr("error decoding time")
 | 
						|
	return
 | 
						|
}
 | 
						|
 | 
						|
func (d *jsonDecDriver[T]) ContainerType() (vt valueType) {
 | 
						|
	// check container type by checking the first char
 | 
						|
	d.advance()
 | 
						|
 | 
						|
	// optimize this, so we don't do 4 checks but do one computation.
 | 
						|
	// return jsonContainerSet[d.tok]
 | 
						|
 | 
						|
	// ContainerType is mostly called for Map and Array,
 | 
						|
	// so this conditional is good enough (max 2 checks typically)
 | 
						|
	if d.tok == '{' {
 | 
						|
		return valueTypeMap
 | 
						|
	} else if d.tok == '[' {
 | 
						|
		return valueTypeArray
 | 
						|
	} else if d.tok == 'n' {
 | 
						|
		d.checkLit3([3]byte{'u', 'l', 'l'}, d.r.readn3())
 | 
						|
		return valueTypeNil
 | 
						|
	} else if d.tok == '"' {
 | 
						|
		return valueTypeString
 | 
						|
	}
 | 
						|
	return valueTypeUnset
 | 
						|
}
 | 
						|
 | 
						|
func (d *jsonDecDriver[T]) decNumBytes() (bs []byte) {
 | 
						|
	d.advance()
 | 
						|
	if d.tok == '"' {
 | 
						|
		bs = d.r.jsonReadUntilDblQuote()
 | 
						|
		d.tok = 0
 | 
						|
	} else if d.tok == 'n' {
 | 
						|
		d.checkLit3([3]byte{'u', 'l', 'l'}, d.r.readn3())
 | 
						|
	} else {
 | 
						|
		bs, d.tok = d.r.jsonReadNum()
 | 
						|
	}
 | 
						|
	return
 | 
						|
}
 | 
						|
 | 
						|
func (d *jsonDecDriver[T]) DecodeUint64() (u uint64) {
 | 
						|
	b := d.decNumBytes()
 | 
						|
	u, neg, ok := parseInteger_bytes(b)
 | 
						|
	if neg {
 | 
						|
		halt.errorf("negative number cannot be decoded as uint64: %s", any(b))
 | 
						|
	}
 | 
						|
	if !ok {
 | 
						|
		halt.onerror(strconvParseErr(b, "ParseUint"))
 | 
						|
	}
 | 
						|
	return
 | 
						|
}
 | 
						|
 | 
						|
func (d *jsonDecDriver[T]) DecodeInt64() (v int64) {
 | 
						|
	return d.parseInt64(d.decNumBytes())
 | 
						|
}
 | 
						|
 | 
						|
func (d *jsonDecDriver[T]) parseInt64(b []byte) (v int64) {
 | 
						|
	u, neg, ok := parseInteger_bytes(b)
 | 
						|
	if !ok {
 | 
						|
		halt.onerror(strconvParseErr(b, "ParseInt"))
 | 
						|
	}
 | 
						|
	if chkOvf.Uint2Int(u, neg) {
 | 
						|
		halt.errorBytes("overflow decoding number from ", b)
 | 
						|
	}
 | 
						|
	if neg {
 | 
						|
		v = -int64(u)
 | 
						|
	} else {
 | 
						|
		v = int64(u)
 | 
						|
	}
 | 
						|
	return
 | 
						|
}
 | 
						|
 | 
						|
func (d *jsonDecDriver[T]) DecodeFloat64() (f float64) {
 | 
						|
	var err error
 | 
						|
	bs := d.decNumBytes()
 | 
						|
	if len(bs) == 0 {
 | 
						|
		return
 | 
						|
	}
 | 
						|
	f, err = parseFloat64(bs)
 | 
						|
	halt.onerror(err)
 | 
						|
	return
 | 
						|
}
 | 
						|
 | 
						|
func (d *jsonDecDriver[T]) DecodeFloat32() (f float32) {
 | 
						|
	var err error
 | 
						|
	bs := d.decNumBytes()
 | 
						|
	if len(bs) == 0 {
 | 
						|
		return
 | 
						|
	}
 | 
						|
	f, err = parseFloat32(bs)
 | 
						|
	halt.onerror(err)
 | 
						|
	return
 | 
						|
}
 | 
						|
 | 
						|
func (d *jsonDecDriver[T]) advanceNil() (ok bool) {
 | 
						|
	d.advance()
 | 
						|
	if d.tok == 'n' {
 | 
						|
		d.checkLit3([3]byte{'u', 'l', 'l'}, d.r.readn3())
 | 
						|
		return true
 | 
						|
	}
 | 
						|
	return false
 | 
						|
}
 | 
						|
 | 
						|
func (d *jsonDecDriver[T]) DecodeExt(rv interface{}, basetype reflect.Type, xtag uint64, ext Ext) {
 | 
						|
	if d.advanceNil() {
 | 
						|
		return
 | 
						|
	}
 | 
						|
	if ext == SelfExt {
 | 
						|
		d.dec.decodeAs(rv, basetype, false)
 | 
						|
	} else {
 | 
						|
		d.dec.interfaceExtConvertAndDecode(rv, ext)
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
func (d *jsonDecDriver[T]) DecodeRawExt(re *RawExt) {
 | 
						|
	if d.advanceNil() {
 | 
						|
		return
 | 
						|
	}
 | 
						|
	d.dec.decode(&re.Value)
 | 
						|
}
 | 
						|
 | 
						|
func (d *jsonDecDriver[T]) decBytesFromArray(bs []byte) []byte {
 | 
						|
	d.advance()
 | 
						|
	if d.tok != ']' {
 | 
						|
		bs = append(bs, uint8(d.DecodeUint64()))
 | 
						|
		d.advance()
 | 
						|
	}
 | 
						|
	for d.tok != ']' {
 | 
						|
		if d.tok != ',' {
 | 
						|
			halt.errorByte("read array element - expect char ',' but got char: ", d.tok)
 | 
						|
		}
 | 
						|
		d.tok = 0
 | 
						|
		bs = append(bs, uint8(chkOvf.UintV(d.DecodeUint64(), 8)))
 | 
						|
		d.advance()
 | 
						|
	}
 | 
						|
	d.tok = 0
 | 
						|
	return bs
 | 
						|
}
 | 
						|
 | 
						|
func (d *jsonDecDriver[T]) DecodeBytes() (bs []byte, state dBytesAttachState) {
 | 
						|
	d.advance()
 | 
						|
	state = dBytesDetach
 | 
						|
	if d.tok == 'n' {
 | 
						|
		d.checkLit3([3]byte{'u', 'l', 'l'}, d.r.readn3())
 | 
						|
		return
 | 
						|
	}
 | 
						|
	state = dBytesAttachBuffer
 | 
						|
	// if decoding into raw bytes, and the RawBytesExt is configured, use it to decode.
 | 
						|
	if d.rawext {
 | 
						|
		d.buf = d.buf[:0]
 | 
						|
		d.dec.interfaceExtConvertAndDecode(&d.buf, d.h.RawBytesExt)
 | 
						|
		bs = d.buf
 | 
						|
		return
 | 
						|
	}
 | 
						|
	// check if an "array" of uint8's (see ContainerType for how to infer if an array)
 | 
						|
	if d.tok == '[' {
 | 
						|
		d.tok = 0
 | 
						|
		// bsOut, _ = fastpathTV.DecSliceUint8V(bs, true, d.d)
 | 
						|
		bs = d.decBytesFromArray(d.buf[:0])
 | 
						|
		d.buf = bs
 | 
						|
		return
 | 
						|
	}
 | 
						|
 | 
						|
	// base64 encodes []byte{} as "", and we encode nil []byte as null.
 | 
						|
	// Consequently, base64 should decode null as a nil []byte, and "" as an empty []byte{}.
 | 
						|
 | 
						|
	d.ensureReadingString()
 | 
						|
	bs1 := d.readUnescapedString()
 | 
						|
	// base64 is most compact of supported formats; it's decodedlen is sufficient for all
 | 
						|
	slen := base64.StdEncoding.DecodedLen(len(bs1))
 | 
						|
	if slen == 0 {
 | 
						|
		bs = zeroByteSlice
 | 
						|
		state = dBytesDetach
 | 
						|
	} else if slen <= cap(d.buf) {
 | 
						|
		bs = d.buf[:slen]
 | 
						|
	} else {
 | 
						|
		d.buf = d.d.blist.putGet(d.buf, slen)[:slen]
 | 
						|
		bs = d.buf
 | 
						|
	}
 | 
						|
	var err error
 | 
						|
	for _, v := range d.byteFmters {
 | 
						|
		// slen := v.DecodedLen(len(bs1))
 | 
						|
		slen, err = v.Decode(bs, bs1)
 | 
						|
		if err == nil {
 | 
						|
			bs = bs[:slen]
 | 
						|
			return
 | 
						|
		}
 | 
						|
	}
 | 
						|
	halt.errorf("error decoding byte string '%s': %v", any(bs1), err)
 | 
						|
	return
 | 
						|
}
 | 
						|
 | 
						|
func (d *jsonDecDriver[T]) DecodeStringAsBytes() (bs []byte, state dBytesAttachState) {
 | 
						|
	d.advance()
 | 
						|
 | 
						|
	var cond bool
 | 
						|
	// common case - hoist outside the switch statement
 | 
						|
	if d.tok == '"' {
 | 
						|
		d.tok = 0
 | 
						|
		bs, cond = d.dblQuoteStringAsBytes()
 | 
						|
		state = d.d.attachState(cond)
 | 
						|
		return
 | 
						|
	}
 | 
						|
 | 
						|
	state = dBytesDetach
 | 
						|
	// handle non-string scalar: null, true, false or a number
 | 
						|
	switch d.tok {
 | 
						|
	case 'n':
 | 
						|
		d.checkLit3([3]byte{'u', 'l', 'l'}, d.r.readn3())
 | 
						|
		// out = nil // []byte{}
 | 
						|
	case 'f':
 | 
						|
		d.checkLit4([4]byte{'a', 'l', 's', 'e'}, d.r.readn4())
 | 
						|
		bs = jsonLitb[jsonLitF : jsonLitF+5]
 | 
						|
	case 't':
 | 
						|
		d.checkLit3([3]byte{'r', 'u', 'e'}, d.r.readn3())
 | 
						|
		bs = jsonLitb[jsonLitT : jsonLitT+4]
 | 
						|
	default:
 | 
						|
		// try to parse a valid number
 | 
						|
		bs, d.tok = d.r.jsonReadNum()
 | 
						|
		state = d.d.attachState(!d.d.bytes)
 | 
						|
	}
 | 
						|
	return
 | 
						|
}
 | 
						|
 | 
						|
func (d *jsonDecDriver[T]) ensureReadingString() {
 | 
						|
	if d.tok != '"' {
 | 
						|
		halt.errorByte("expecting string starting with '\"'; got ", d.tok)
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
func (d *jsonDecDriver[T]) readUnescapedString() (bs []byte) {
 | 
						|
	// d.ensureReadingString()
 | 
						|
	bs = d.r.jsonReadUntilDblQuote()
 | 
						|
	d.tok = 0
 | 
						|
	return
 | 
						|
}
 | 
						|
 | 
						|
func (d *jsonDecDriver[T]) dblQuoteStringAsBytes() (buf []byte, usingBuf bool) {
 | 
						|
	bs, c := d.r.jsonReadAsisChars()
 | 
						|
	if c == '"' {
 | 
						|
		return bs, !d.d.bytes
 | 
						|
	}
 | 
						|
	buf = append(d.buf[:0], bs...)
 | 
						|
 | 
						|
	checkUtf8 := d.h.ValidateUnicode
 | 
						|
	usingBuf = true
 | 
						|
 | 
						|
	for {
 | 
						|
		// c is now '\'
 | 
						|
		c = d.r.readn1()
 | 
						|
 | 
						|
		switch c {
 | 
						|
		case '"', '\\', '/', '\'':
 | 
						|
			buf = append(buf, c)
 | 
						|
		case 'b':
 | 
						|
			buf = append(buf, '\b')
 | 
						|
		case 'f':
 | 
						|
			buf = append(buf, '\f')
 | 
						|
		case 'n':
 | 
						|
			buf = append(buf, '\n')
 | 
						|
		case 'r':
 | 
						|
			buf = append(buf, '\r')
 | 
						|
		case 't':
 | 
						|
			buf = append(buf, '\t')
 | 
						|
		case 'u':
 | 
						|
			rr := d.appendStringAsBytesSlashU()
 | 
						|
			if checkUtf8 && rr == unicode.ReplacementChar {
 | 
						|
				d.buf = buf
 | 
						|
				halt.errorBytes("invalid UTF-8 character found after: ", buf)
 | 
						|
			}
 | 
						|
			buf = append(buf, d.bstr[:utf8.EncodeRune(d.bstr[:], rr)]...)
 | 
						|
		default:
 | 
						|
			d.buf = buf
 | 
						|
			halt.errorByte("unsupported escaped value: ", c)
 | 
						|
		}
 | 
						|
 | 
						|
		bs, c = d.r.jsonReadAsisChars()
 | 
						|
		buf = append(buf, bs...)
 | 
						|
		if c == '"' {
 | 
						|
			break
 | 
						|
		}
 | 
						|
	}
 | 
						|
	d.buf = buf
 | 
						|
	return
 | 
						|
}
 | 
						|
 | 
						|
func (d *jsonDecDriver[T]) appendStringAsBytesSlashU() (r rune) {
 | 
						|
	var rr uint32
 | 
						|
	cs := d.r.readn4()
 | 
						|
	if rr = jsonSlashURune(cs); rr == unicode.ReplacementChar {
 | 
						|
		return unicode.ReplacementChar
 | 
						|
	}
 | 
						|
	r = rune(rr)
 | 
						|
	if utf16.IsSurrogate(r) {
 | 
						|
		csu := d.r.readn2()
 | 
						|
		cs = d.r.readn4()
 | 
						|
		if csu[0] == '\\' && csu[1] == 'u' {
 | 
						|
			if rr = jsonSlashURune(cs); rr == unicode.ReplacementChar {
 | 
						|
				return unicode.ReplacementChar
 | 
						|
			}
 | 
						|
			return utf16.DecodeRune(r, rune(rr))
 | 
						|
		}
 | 
						|
		return unicode.ReplacementChar
 | 
						|
	}
 | 
						|
	return
 | 
						|
}
 | 
						|
 | 
						|
func (d *jsonDecDriver[T]) DecodeNaked() {
 | 
						|
	z := d.d.naked()
 | 
						|
 | 
						|
	d.advance()
 | 
						|
	var bs []byte
 | 
						|
	var err error
 | 
						|
	switch d.tok {
 | 
						|
	case 'n':
 | 
						|
		d.checkLit3([3]byte{'u', 'l', 'l'}, d.r.readn3())
 | 
						|
		z.v = valueTypeNil
 | 
						|
	case 'f':
 | 
						|
		d.checkLit4([4]byte{'a', 'l', 's', 'e'}, d.r.readn4())
 | 
						|
		z.v = valueTypeBool
 | 
						|
		z.b = false
 | 
						|
	case 't':
 | 
						|
		d.checkLit3([3]byte{'r', 'u', 'e'}, d.r.readn3())
 | 
						|
		z.v = valueTypeBool
 | 
						|
		z.b = true
 | 
						|
	case '{':
 | 
						|
		z.v = valueTypeMap // don't consume. kInterfaceNaked will call ReadMapStart
 | 
						|
	case '[':
 | 
						|
		z.v = valueTypeArray // don't consume. kInterfaceNaked will call ReadArrayStart
 | 
						|
	case '"':
 | 
						|
		// if a string, and MapKeyAsString, then try to decode it as a bool or number first
 | 
						|
		d.tok = 0
 | 
						|
		bs, z.b = d.dblQuoteStringAsBytes()
 | 
						|
		att := d.d.attachState(z.b)
 | 
						|
		if jsonNakedBoolNumInQuotedStr &&
 | 
						|
			d.h.MapKeyAsString && len(bs) > 0 && d.d.c == containerMapKey {
 | 
						|
			switch string(bs) {
 | 
						|
			// case "null": // nil is never quoted
 | 
						|
			// 	z.v = valueTypeNil
 | 
						|
			case "true":
 | 
						|
				z.v = valueTypeBool
 | 
						|
				z.b = true
 | 
						|
			case "false":
 | 
						|
				z.v = valueTypeBool
 | 
						|
				z.b = false
 | 
						|
			default: // check if a number: float, int or uint
 | 
						|
				if err = jsonNakedNum(z, bs, d.h.PreferFloat, d.h.SignedInteger); err != nil {
 | 
						|
					z.v = valueTypeString
 | 
						|
					z.s = d.d.detach2Str(bs, att)
 | 
						|
				}
 | 
						|
			}
 | 
						|
		} else {
 | 
						|
			z.v = valueTypeString
 | 
						|
			z.s = d.d.detach2Str(bs, att)
 | 
						|
		}
 | 
						|
	default: // number
 | 
						|
		bs, d.tok = d.r.jsonReadNum()
 | 
						|
		if len(bs) == 0 {
 | 
						|
			halt.errorStr("decode number from empty string")
 | 
						|
		}
 | 
						|
		if err = jsonNakedNum(z, bs, d.h.PreferFloat, d.h.SignedInteger); err != nil {
 | 
						|
			halt.errorf("decode number from %s: %v", any(bs), err)
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
func (e *jsonEncDriver[T]) reset() {
 | 
						|
	e.dl = 0
 | 
						|
	// e.resetState()
 | 
						|
	// (htmlasis && jsonCharSafeSet.isset(b)) || jsonCharHtmlSafeSet.isset(b)
 | 
						|
	// cache values from the handle
 | 
						|
	e.typical = e.h.typical()
 | 
						|
	if e.h.HTMLCharsAsIs {
 | 
						|
		e.s = &jsonCharSafeBitset
 | 
						|
	} else {
 | 
						|
		e.s = &jsonCharHtmlSafeBitset
 | 
						|
	}
 | 
						|
	e.di = int8(e.h.Indent)
 | 
						|
	e.d = e.h.Indent != 0
 | 
						|
	e.ks = e.h.MapKeyAsString
 | 
						|
	e.is = e.h.IntegerAsString
 | 
						|
 | 
						|
	var ho jsonHandleOpts
 | 
						|
	ho.reset(e.h)
 | 
						|
	e.timeFmt = ho.timeFmt
 | 
						|
	e.bytesFmt = ho.bytesFmt
 | 
						|
	e.timeFmtLayout = ""
 | 
						|
	e.byteFmter = nil
 | 
						|
	if len(ho.timeFmtLayouts) > 0 {
 | 
						|
		e.timeFmtLayout = ho.timeFmtLayouts[0]
 | 
						|
	}
 | 
						|
	if len(ho.byteFmters) > 0 {
 | 
						|
		e.byteFmter = ho.byteFmters[0]
 | 
						|
	}
 | 
						|
	e.rawext = ho.rawext
 | 
						|
}
 | 
						|
 | 
						|
func (d *jsonDecDriver[T]) reset() {
 | 
						|
	d.buf = d.d.blist.check(d.buf, 256)
 | 
						|
	d.tok = 0
 | 
						|
	// d.resetState()
 | 
						|
	d.jsonHandleOpts.reset(d.h)
 | 
						|
}
 | 
						|
 | 
						|
// ----
 | 
						|
//
 | 
						|
// The following below are similar across all format files (except for the format name).
 | 
						|
//
 | 
						|
// We keep them together here, so that we can easily copy and compare.
 | 
						|
 | 
						|
// ----
 | 
						|
 | 
						|
func (d *jsonEncDriver[T]) init(hh Handle, shared *encoderBase, enc encoderI) (fp interface{}) {
 | 
						|
	callMake(&d.w)
 | 
						|
	d.h = hh.(*JsonHandle)
 | 
						|
	d.e = shared
 | 
						|
	if shared.bytes {
 | 
						|
		fp = jsonFpEncBytes
 | 
						|
	} else {
 | 
						|
		fp = jsonFpEncIO
 | 
						|
	}
 | 
						|
	// d.w.init()
 | 
						|
	d.init2(enc)
 | 
						|
	return
 | 
						|
}
 | 
						|
 | 
						|
func (e *jsonEncDriver[T]) writeBytesAsis(b []byte) { e.w.writeb(b) }
 | 
						|
 | 
						|
// func (e *jsonEncDriver[T]) writeStringAsisDblQuoted(v string) { e.w.writeqstr(v) }
 | 
						|
func (e *jsonEncDriver[T]) writerEnd() { e.w.end() }
 | 
						|
 | 
						|
func (e *jsonEncDriver[T]) resetOutBytes(out *[]byte) {
 | 
						|
	e.w.resetBytes(*out, out)
 | 
						|
}
 | 
						|
 | 
						|
func (e *jsonEncDriver[T]) resetOutIO(out io.Writer) {
 | 
						|
	e.w.resetIO(out, e.h.WriterBufferSize, &e.e.blist)
 | 
						|
}
 | 
						|
 | 
						|
// ----
 | 
						|
 | 
						|
func (d *jsonDecDriver[T]) init(hh Handle, shared *decoderBase, dec decoderI) (fp interface{}) {
 | 
						|
	callMake(&d.r)
 | 
						|
	d.h = hh.(*JsonHandle)
 | 
						|
	d.d = shared
 | 
						|
	if shared.bytes {
 | 
						|
		fp = jsonFpDecBytes
 | 
						|
	} else {
 | 
						|
		fp = jsonFpDecIO
 | 
						|
	}
 | 
						|
	// d.r.init()
 | 
						|
	d.init2(dec)
 | 
						|
	return
 | 
						|
}
 | 
						|
 | 
						|
func (d *jsonDecDriver[T]) NumBytesRead() int {
 | 
						|
	return int(d.r.numread())
 | 
						|
}
 | 
						|
 | 
						|
func (d *jsonDecDriver[T]) resetInBytes(in []byte) {
 | 
						|
	d.r.resetBytes(in)
 | 
						|
}
 | 
						|
 | 
						|
func (d *jsonDecDriver[T]) resetInIO(r io.Reader) {
 | 
						|
	d.r.resetIO(r, d.h.ReaderBufferSize, d.h.MaxInitLen, &d.d.blist)
 | 
						|
}
 | 
						|
 | 
						|
// ---- (custom stanza)
 | 
						|
 | 
						|
func (d *jsonDecDriver[T]) descBd() (s string) {
 | 
						|
	halt.onerror(errJsonNoBd)
 | 
						|
	return
 | 
						|
}
 | 
						|
 | 
						|
func (d *jsonEncDriver[T]) init2(enc encoderI) {
 | 
						|
	d.enc = enc
 | 
						|
	// d.e.js = true
 | 
						|
}
 | 
						|
 | 
						|
func (d *jsonDecDriver[T]) init2(dec decoderI) {
 | 
						|
	d.dec = dec
 | 
						|
	// var x []byte
 | 
						|
	// d.buf = &x
 | 
						|
	// d.buf = new([]byte)
 | 
						|
	d.buf = d.buf[:0]
 | 
						|
	// d.d.js = true
 | 
						|
	d.d.jsms = d.h.MapKeyAsString
 | 
						|
}
 |