123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287 |
- // Copyright GoFrame Author(https://goframe.org). All Rights Reserved.
- //
- // This Source Code Form is subject to the terms of the MIT License.
- // If a copy of the MIT was not distributed with this file,
- // You can obtain one at https://github.com/gogf/gf.
- package gbinary
- import (
- "bytes"
- "context"
- "encoding/binary"
- "fmt"
- "math"
- "github.com/gogf/gf/v2/errors/gerror"
- "github.com/gogf/gf/v2/internal/intlog"
- )
- // BeEncode encodes one or multiple `values` into bytes using BigEndian.
- // It uses type asserting checking the type of each value of `values` and internally
- // calls corresponding converting function do the bytes converting.
- //
- // It supports common variable type asserting, and finally it uses fmt.Sprintf converting
- // value to string and then to bytes.
- func BeEncode(values ...interface{}) []byte {
- buf := new(bytes.Buffer)
- for i := 0; i < len(values); i++ {
- if values[i] == nil {
- return buf.Bytes()
- }
- switch value := values[i].(type) {
- case int:
- buf.Write(BeEncodeInt(value))
- case int8:
- buf.Write(BeEncodeInt8(value))
- case int16:
- buf.Write(BeEncodeInt16(value))
- case int32:
- buf.Write(BeEncodeInt32(value))
- case int64:
- buf.Write(BeEncodeInt64(value))
- case uint:
- buf.Write(BeEncodeUint(value))
- case uint8:
- buf.Write(BeEncodeUint8(value))
- case uint16:
- buf.Write(BeEncodeUint16(value))
- case uint32:
- buf.Write(BeEncodeUint32(value))
- case uint64:
- buf.Write(BeEncodeUint64(value))
- case bool:
- buf.Write(BeEncodeBool(value))
- case string:
- buf.Write(BeEncodeString(value))
- case []byte:
- buf.Write(value)
- case float32:
- buf.Write(BeEncodeFloat32(value))
- case float64:
- buf.Write(BeEncodeFloat64(value))
- default:
- if err := binary.Write(buf, binary.BigEndian, value); err != nil {
- intlog.Errorf(context.TODO(), `%+v`, err)
- buf.Write(BeEncodeString(fmt.Sprintf("%v", value)))
- }
- }
- }
- return buf.Bytes()
- }
- func BeEncodeByLength(length int, values ...interface{}) []byte {
- b := BeEncode(values...)
- if len(b) < length {
- b = append(b, make([]byte, length-len(b))...)
- } else if len(b) > length {
- b = b[0:length]
- }
- return b
- }
- func BeDecode(b []byte, values ...interface{}) error {
- var (
- err error
- buf = bytes.NewBuffer(b)
- )
- for i := 0; i < len(values); i++ {
- if err = binary.Read(buf, binary.BigEndian, values[i]); err != nil {
- err = gerror.Wrap(err, `binary.Read failed`)
- return err
- }
- }
- return nil
- }
- func BeEncodeString(s string) []byte {
- return []byte(s)
- }
- func BeDecodeToString(b []byte) string {
- return string(b)
- }
- func BeEncodeBool(b bool) []byte {
- if b {
- return []byte{1}
- } else {
- return []byte{0}
- }
- }
- func BeEncodeInt(i int) []byte {
- if i <= math.MaxInt8 {
- return BeEncodeInt8(int8(i))
- } else if i <= math.MaxInt16 {
- return BeEncodeInt16(int16(i))
- } else if i <= math.MaxInt32 {
- return BeEncodeInt32(int32(i))
- } else {
- return BeEncodeInt64(int64(i))
- }
- }
- func BeEncodeUint(i uint) []byte {
- if i <= math.MaxUint8 {
- return BeEncodeUint8(uint8(i))
- } else if i <= math.MaxUint16 {
- return BeEncodeUint16(uint16(i))
- } else if i <= math.MaxUint32 {
- return BeEncodeUint32(uint32(i))
- } else {
- return BeEncodeUint64(uint64(i))
- }
- }
- func BeEncodeInt8(i int8) []byte {
- return []byte{byte(i)}
- }
- func BeEncodeUint8(i uint8) []byte {
- return []byte{i}
- }
- func BeEncodeInt16(i int16) []byte {
- b := make([]byte, 2)
- binary.BigEndian.PutUint16(b, uint16(i))
- return b
- }
- func BeEncodeUint16(i uint16) []byte {
- b := make([]byte, 2)
- binary.BigEndian.PutUint16(b, i)
- return b
- }
- func BeEncodeInt32(i int32) []byte {
- b := make([]byte, 4)
- binary.BigEndian.PutUint32(b, uint32(i))
- return b
- }
- func BeEncodeUint32(i uint32) []byte {
- b := make([]byte, 4)
- binary.BigEndian.PutUint32(b, i)
- return b
- }
- func BeEncodeInt64(i int64) []byte {
- b := make([]byte, 8)
- binary.BigEndian.PutUint64(b, uint64(i))
- return b
- }
- func BeEncodeUint64(i uint64) []byte {
- b := make([]byte, 8)
- binary.BigEndian.PutUint64(b, i)
- return b
- }
- func BeEncodeFloat32(f float32) []byte {
- bits := math.Float32bits(f)
- b := make([]byte, 4)
- binary.BigEndian.PutUint32(b, bits)
- return b
- }
- func BeEncodeFloat64(f float64) []byte {
- bits := math.Float64bits(f)
- b := make([]byte, 8)
- binary.BigEndian.PutUint64(b, bits)
- return b
- }
- func BeDecodeToInt(b []byte) int {
- if len(b) < 2 {
- return int(BeDecodeToUint8(b))
- } else if len(b) < 3 {
- return int(BeDecodeToUint16(b))
- } else if len(b) < 5 {
- return int(BeDecodeToUint32(b))
- } else {
- return int(BeDecodeToUint64(b))
- }
- }
- func BeDecodeToUint(b []byte) uint {
- if len(b) < 2 {
- return uint(BeDecodeToUint8(b))
- } else if len(b) < 3 {
- return uint(BeDecodeToUint16(b))
- } else if len(b) < 5 {
- return uint(BeDecodeToUint32(b))
- } else {
- return uint(BeDecodeToUint64(b))
- }
- }
- func BeDecodeToBool(b []byte) bool {
- if len(b) == 0 {
- return false
- }
- if bytes.Equal(b, make([]byte, len(b))) {
- return false
- }
- return true
- }
- func BeDecodeToInt8(b []byte) int8 {
- if len(b) == 0 {
- panic(`empty slice given`)
- }
- return int8(b[0])
- }
- func BeDecodeToUint8(b []byte) uint8 {
- if len(b) == 0 {
- panic(`empty slice given`)
- }
- return b[0]
- }
- func BeDecodeToInt16(b []byte) int16 {
- return int16(binary.BigEndian.Uint16(BeFillUpSize(b, 2)))
- }
- func BeDecodeToUint16(b []byte) uint16 {
- return binary.BigEndian.Uint16(BeFillUpSize(b, 2))
- }
- func BeDecodeToInt32(b []byte) int32 {
- return int32(binary.BigEndian.Uint32(BeFillUpSize(b, 4)))
- }
- func BeDecodeToUint32(b []byte) uint32 {
- return binary.BigEndian.Uint32(BeFillUpSize(b, 4))
- }
- func BeDecodeToInt64(b []byte) int64 {
- return int64(binary.BigEndian.Uint64(BeFillUpSize(b, 8)))
- }
- func BeDecodeToUint64(b []byte) uint64 {
- return binary.BigEndian.Uint64(BeFillUpSize(b, 8))
- }
- func BeDecodeToFloat32(b []byte) float32 {
- return math.Float32frombits(binary.BigEndian.Uint32(BeFillUpSize(b, 4)))
- }
- func BeDecodeToFloat64(b []byte) float64 {
- return math.Float64frombits(binary.BigEndian.Uint64(BeFillUpSize(b, 8)))
- }
- // BeFillUpSize fills up the bytes `b` to given length `l` using big BigEndian.
- //
- // Note that it creates a new bytes slice by copying the original one to avoid changing
- // the original parameter bytes.
- func BeFillUpSize(b []byte, l int) []byte {
- if len(b) >= l {
- return b[:l]
- }
- c := make([]byte, l)
- copy(c[l-len(b):], b)
- return c
- }
|