mirror of
				https://github.com/superseriousbusiness/gotosocial.git
				synced 2025-10-31 14:52:30 -05:00 
			
		
		
		
	
		
			
				
	
	
		
			225 lines
		
	
	
	
		
			4.2 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			225 lines
		
	
	
	
		
			4.2 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package msgpack
 | |
| 
 | |
| import (
 | |
| 	"math"
 | |
| 	"reflect"
 | |
| 	"sort"
 | |
| 
 | |
| 	"github.com/vmihailenco/msgpack/v5/msgpcode"
 | |
| )
 | |
| 
 | |
| func encodeMapValue(e *Encoder, v reflect.Value) error {
 | |
| 	if v.IsNil() {
 | |
| 		return e.EncodeNil()
 | |
| 	}
 | |
| 
 | |
| 	if err := e.EncodeMapLen(v.Len()); err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 
 | |
| 	iter := v.MapRange()
 | |
| 	for iter.Next() {
 | |
| 		if err := e.EncodeValue(iter.Key()); err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 		if err := e.EncodeValue(iter.Value()); err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func encodeMapStringBoolValue(e *Encoder, v reflect.Value) error {
 | |
| 	if v.IsNil() {
 | |
| 		return e.EncodeNil()
 | |
| 	}
 | |
| 
 | |
| 	if err := e.EncodeMapLen(v.Len()); err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 
 | |
| 	m := v.Convert(mapStringBoolType).Interface().(map[string]bool)
 | |
| 	if e.flags&sortMapKeysFlag != 0 {
 | |
| 		return e.encodeSortedMapStringBool(m)
 | |
| 	}
 | |
| 
 | |
| 	for mk, mv := range m {
 | |
| 		if err := e.EncodeString(mk); err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 		if err := e.EncodeBool(mv); err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func encodeMapStringStringValue(e *Encoder, v reflect.Value) error {
 | |
| 	if v.IsNil() {
 | |
| 		return e.EncodeNil()
 | |
| 	}
 | |
| 
 | |
| 	if err := e.EncodeMapLen(v.Len()); err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 
 | |
| 	m := v.Convert(mapStringStringType).Interface().(map[string]string)
 | |
| 	if e.flags&sortMapKeysFlag != 0 {
 | |
| 		return e.encodeSortedMapStringString(m)
 | |
| 	}
 | |
| 
 | |
| 	for mk, mv := range m {
 | |
| 		if err := e.EncodeString(mk); err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 		if err := e.EncodeString(mv); err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func encodeMapStringInterfaceValue(e *Encoder, v reflect.Value) error {
 | |
| 	if v.IsNil() {
 | |
| 		return e.EncodeNil()
 | |
| 	}
 | |
| 	m := v.Convert(mapStringInterfaceType).Interface().(map[string]interface{})
 | |
| 	if e.flags&sortMapKeysFlag != 0 {
 | |
| 		return e.EncodeMapSorted(m)
 | |
| 	}
 | |
| 	return e.EncodeMap(m)
 | |
| }
 | |
| 
 | |
| func (e *Encoder) EncodeMap(m map[string]interface{}) error {
 | |
| 	if m == nil {
 | |
| 		return e.EncodeNil()
 | |
| 	}
 | |
| 	if err := e.EncodeMapLen(len(m)); err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 	for mk, mv := range m {
 | |
| 		if err := e.EncodeString(mk); err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 		if err := e.Encode(mv); err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 	}
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func (e *Encoder) EncodeMapSorted(m map[string]interface{}) error {
 | |
| 	if m == nil {
 | |
| 		return e.EncodeNil()
 | |
| 	}
 | |
| 	if err := e.EncodeMapLen(len(m)); err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 
 | |
| 	keys := make([]string, 0, len(m))
 | |
| 
 | |
| 	for k := range m {
 | |
| 		keys = append(keys, k)
 | |
| 	}
 | |
| 
 | |
| 	sort.Strings(keys)
 | |
| 
 | |
| 	for _, k := range keys {
 | |
| 		if err := e.EncodeString(k); err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 		if err := e.Encode(m[k]); err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func (e *Encoder) encodeSortedMapStringBool(m map[string]bool) error {
 | |
| 	keys := make([]string, 0, len(m))
 | |
| 	for k := range m {
 | |
| 		keys = append(keys, k)
 | |
| 	}
 | |
| 	sort.Strings(keys)
 | |
| 
 | |
| 	for _, k := range keys {
 | |
| 		err := e.EncodeString(k)
 | |
| 		if err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 		if err = e.EncodeBool(m[k]); err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func (e *Encoder) encodeSortedMapStringString(m map[string]string) error {
 | |
| 	keys := make([]string, 0, len(m))
 | |
| 	for k := range m {
 | |
| 		keys = append(keys, k)
 | |
| 	}
 | |
| 	sort.Strings(keys)
 | |
| 
 | |
| 	for _, k := range keys {
 | |
| 		err := e.EncodeString(k)
 | |
| 		if err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 		if err = e.EncodeString(m[k]); err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func (e *Encoder) EncodeMapLen(l int) error {
 | |
| 	if l < 16 {
 | |
| 		return e.writeCode(msgpcode.FixedMapLow | byte(l))
 | |
| 	}
 | |
| 	if l <= math.MaxUint16 {
 | |
| 		return e.write2(msgpcode.Map16, uint16(l))
 | |
| 	}
 | |
| 	return e.write4(msgpcode.Map32, uint32(l))
 | |
| }
 | |
| 
 | |
| func encodeStructValue(e *Encoder, strct reflect.Value) error {
 | |
| 	structFields := structs.Fields(strct.Type(), e.structTag)
 | |
| 	if e.flags&arrayEncodedStructsFlag != 0 || structFields.AsArray {
 | |
| 		return encodeStructValueAsArray(e, strct, structFields.List)
 | |
| 	}
 | |
| 	fields := structFields.OmitEmpty(e, strct)
 | |
| 
 | |
| 	if err := e.EncodeMapLen(len(fields)); err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 
 | |
| 	for _, f := range fields {
 | |
| 		if err := e.EncodeString(f.name); err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 		if err := f.EncodeValue(e, strct); err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func encodeStructValueAsArray(e *Encoder, strct reflect.Value, fields []*field) error {
 | |
| 	if err := e.EncodeArrayLen(len(fields)); err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 	for _, f := range fields {
 | |
| 		if err := f.EncodeValue(e, strct); err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 	}
 | |
| 	return nil
 | |
| }
 |