mirror of
https://gitee.com/energye/energy.git
synced 2024-12-05 21:28:17 +08:00
705 lines
15 KiB
Go
705 lines
15 KiB
Go
//----------------------------------------
|
||
//
|
||
// Copyright © yanghy. All Rights Reserved.
|
||
//
|
||
// Licensed under Apache License Version 2.0, January 2004
|
||
//
|
||
// https://www.apache.org/licenses/LICENSE-2.0
|
||
//
|
||
//----------------------------------------
|
||
|
||
package json
|
||
|
||
import (
|
||
"encoding/json"
|
||
"github.com/energye/energy/common"
|
||
. "github.com/energye/energy/consts"
|
||
jsoniter "github.com/json-iterator/go"
|
||
"reflect"
|
||
"strconv"
|
||
"strings"
|
||
)
|
||
|
||
type BaseJSON interface {
|
||
Size() int
|
||
Type() GO_VALUE_TYPE
|
||
Data() any
|
||
String() string
|
||
Int() int
|
||
UInt() uint
|
||
Bytes() []byte
|
||
Float() float64
|
||
Bool() bool
|
||
JSONObject() JSONObject
|
||
JSONArray() JSONArray
|
||
ToJSONString() string
|
||
IsString() bool
|
||
IsInt() bool
|
||
IsUInt() bool
|
||
IsBytes() bool
|
||
IsFloat() bool
|
||
IsBool() bool
|
||
IsObject() bool
|
||
IsArray() bool
|
||
Clear()
|
||
}
|
||
|
||
type JSONArray interface {
|
||
BaseJSON
|
||
Add(value ...any)
|
||
SetByIndex(index int, value any)
|
||
RemoveByIndex(index int)
|
||
GetStringByIndex(index int) string
|
||
GetIntByIndex(index int) int
|
||
GetUIntByIndex(index int) uint
|
||
GetBytesByIndex(index int) []byte
|
||
GetFloatByIndex(index int) float64
|
||
GetBoolByIndex(index int) bool
|
||
GetArrayByIndex(index int) JSONArray
|
||
GetObjectByIndex(index int) JSONObject
|
||
GetByIndex(index int) JSON
|
||
}
|
||
|
||
type JSONObject interface {
|
||
BaseJSON
|
||
Set(key string, value any)
|
||
RemoveByKey(key string)
|
||
GetStringByKey(key string) string
|
||
GetIntByKey(key string) int
|
||
GetUIntByKey(key string) uint
|
||
GetBytesByKey(key string) []byte
|
||
GetFloatByKey(key string) float64
|
||
GetBoolByKey(key string) bool
|
||
GetArrayByKey(key string) JSONArray
|
||
GetObjectByKey(key string) JSONObject
|
||
GetByKey(key string) JSON
|
||
Keys() []string
|
||
}
|
||
|
||
// JSON Object
|
||
type JSON interface {
|
||
JSONArray
|
||
JSONObject
|
||
}
|
||
|
||
type jsonData struct {
|
||
T GO_VALUE_TYPE // type
|
||
S int // size
|
||
V any // value
|
||
}
|
||
|
||
// NewJSON 返回 jsonData 对象,JSONArray or JSONObject
|
||
func NewJSON(data []byte) JSON {
|
||
if data == nil {
|
||
return nil
|
||
}
|
||
var v any
|
||
if err := jsoniter.Unmarshal(data, &v); err == nil {
|
||
rv := reflect.ValueOf(v)
|
||
switch rv.Kind() {
|
||
case reflect.Slice:
|
||
if v, ok := v.([]any); ok {
|
||
return &jsonData{T: GO_VALUE_SLICE, V: v, S: len(v)}
|
||
}
|
||
case reflect.Map:
|
||
if v, ok := v.(map[string]any); ok {
|
||
return &jsonData{T: GO_VALUE_MAP, V: v, S: len(v)}
|
||
}
|
||
}
|
||
}
|
||
return nil
|
||
}
|
||
|
||
// NewJSONArray 字节JSONArray / 数组 / 切片 转换
|
||
func NewJSONArray(value any) JSONArray {
|
||
if value != nil {
|
||
// 如果 []byte 就必须是 字节JSONArray
|
||
switch value.(type) {
|
||
case []byte:
|
||
if v := NewJSON(value.([]byte)); v != nil {
|
||
return v.JSONArray()
|
||
} else {
|
||
return nil
|
||
}
|
||
}
|
||
rv := reflect.ValueOf(value)
|
||
if rv.Kind() != reflect.Slice && rv.Kind() != reflect.Array {
|
||
return nil
|
||
}
|
||
//目的是为了转为any类型
|
||
if byt, err := jsoniter.Marshal(value); err == nil {
|
||
var v []any
|
||
if err = jsoniter.Unmarshal(byt, &v); err == nil {
|
||
return &jsonData{T: GO_VALUE_SLICE, V: v, S: len(v)}
|
||
}
|
||
}
|
||
}
|
||
return &jsonData{T: GO_VALUE_SLICE, V: make([]any, 0), S: 0}
|
||
}
|
||
|
||
// NewJSONObject 字节JSONObject / 结构 / JSONObject 转换
|
||
func NewJSONObject(value any) JSONObject {
|
||
if value != nil {
|
||
// 如果 []byte 就必须是 字节JSONObject
|
||
switch value.(type) {
|
||
case []byte:
|
||
if v := NewJSON(value.([]byte)); v != nil {
|
||
return v.JSONObject()
|
||
} else {
|
||
return nil
|
||
}
|
||
}
|
||
rv := reflect.ValueOf(value)
|
||
kind := rv.Kind()
|
||
if kind != reflect.Map && kind != reflect.Struct {
|
||
return nil
|
||
}
|
||
//目的是为了转为any类型
|
||
if byt, err := jsoniter.Marshal(value); err == nil {
|
||
var v map[string]any
|
||
if err = jsoniter.Unmarshal(byt, &v); err == nil {
|
||
return &jsonData{T: GO_VALUE_MAP, V: v, S: len(v)}
|
||
}
|
||
}
|
||
}
|
||
return &jsonData{T: GO_VALUE_MAP, V: make(map[string]any), S: 0}
|
||
}
|
||
|
||
func (m *jsonData) Size() int {
|
||
return m.S
|
||
}
|
||
|
||
func (m *jsonData) Type() GO_VALUE_TYPE {
|
||
return m.T
|
||
}
|
||
|
||
func (m *jsonData) Data() any {
|
||
return m.V
|
||
}
|
||
|
||
func (m *jsonData) Add(value ...any) {
|
||
if m.IsArray() {
|
||
tmp := make([]any, len(value))
|
||
for i, v := range value {
|
||
switch v.(type) {
|
||
case JSON:
|
||
tmp[i] = v.(JSON).Data()
|
||
case JSONObject:
|
||
tmp[i] = v.(JSONObject).Data()
|
||
case JSONArray:
|
||
tmp[i] = v.(JSONArray).Data()
|
||
default:
|
||
tmp[i] = v
|
||
}
|
||
}
|
||
m.V = append(m.V.([]any), tmp...)
|
||
m.S += len(value)
|
||
}
|
||
}
|
||
|
||
func (m *jsonData) SetByIndex(index int, value any) {
|
||
if m.IsArray() && index < m.S {
|
||
switch value.(type) {
|
||
case JSON:
|
||
m.V.([]any)[index] = value.(JSON).Data()
|
||
case JSONObject:
|
||
m.V.([]any)[index] = value.(JSONObject).Data()
|
||
case JSONArray:
|
||
m.V.([]any)[index] = value.(JSONArray).Data()
|
||
default:
|
||
m.V.([]any)[index] = value
|
||
}
|
||
}
|
||
}
|
||
|
||
func (m *jsonData) RemoveByIndex(index int) {
|
||
if m.IsArray() && index >= 0 && index < m.S {
|
||
v := m.V.([]any)
|
||
m.V = append(v[:index], v[index+1:]...)
|
||
m.S--
|
||
}
|
||
}
|
||
|
||
func (m *jsonData) GetStringByIndex(index int) string {
|
||
if m.IsArray() && index < m.S {
|
||
if r, ok := m.V.([]any)[index].(string); ok {
|
||
return r
|
||
}
|
||
}
|
||
return ""
|
||
}
|
||
|
||
func (m *jsonData) GetIntByIndex(index int) int {
|
||
if m.IsArray() && index < m.S {
|
||
return m.toInt(m.V.([]any)[index])
|
||
}
|
||
return 0
|
||
}
|
||
|
||
func (m *jsonData) GetUIntByIndex(index int) uint {
|
||
if m.IsArray() && index < m.S {
|
||
return m.toUInt(m.V.([]any)[index])
|
||
}
|
||
return 0
|
||
}
|
||
|
||
func (m *jsonData) GetBytesByIndex(index int) []byte {
|
||
if m.IsArray() && index < m.S {
|
||
return m.toBytes(m.V.([]any)[index])
|
||
}
|
||
return nil
|
||
}
|
||
|
||
func (m *jsonData) GetFloatByIndex(index int) float64 {
|
||
if m.IsArray() && index < m.S {
|
||
return m.toFloat64(m.V.([]any)[index])
|
||
}
|
||
return 0
|
||
}
|
||
|
||
func (m *jsonData) GetBoolByIndex(index int) bool {
|
||
if m.IsArray() && index < m.S {
|
||
s := m.V.([]any)[index]
|
||
switch s.(type) {
|
||
case bool:
|
||
return s.(bool)
|
||
}
|
||
}
|
||
return false
|
||
}
|
||
|
||
func (m *jsonData) GetArrayByIndex(index int) JSONArray {
|
||
if m.IsArray() && index < m.S {
|
||
if v, ok := m.V.([]any)[index].([]any); ok {
|
||
return &jsonData{T: GO_VALUE_SLICE, V: v, S: len(v)}
|
||
}
|
||
}
|
||
return nil
|
||
}
|
||
|
||
func (m *jsonData) GetObjectByIndex(index int) JSONObject {
|
||
if m.IsArray() && index < m.S {
|
||
if v, ok := m.V.([]any)[index].(map[string]any); ok {
|
||
return &jsonData{T: GO_VALUE_MAP, V: v, S: len(v)}
|
||
}
|
||
}
|
||
return nil
|
||
}
|
||
|
||
func (m *jsonData) GetByIndex(index int) JSON {
|
||
if m.IsArray() && index < m.S {
|
||
s := m.V.([]any)[index]
|
||
switch s.(type) {
|
||
case json.Number:
|
||
if v, err := s.(json.Number).Int64(); err == nil {
|
||
return &jsonData{T: GO_VALUE_INT, V: v, S: strconv.IntSize}
|
||
}
|
||
case string:
|
||
if r, ok := s.(string); ok {
|
||
return &jsonData{T: GO_VALUE_STRING, V: r, S: len(r)}
|
||
}
|
||
case int, int8, int16, int32, int64:
|
||
if s := m.GetIntByIndex(index); s != 0 {
|
||
return &jsonData{T: GO_VALUE_INT, V: s, S: strconv.IntSize}
|
||
}
|
||
case uint, uint8, uint16, uint32, uint64:
|
||
if s := m.GetUIntByIndex(index); s != 0 {
|
||
return &jsonData{T: GO_VALUE_UINT, V: s, S: strconv.IntSize}
|
||
}
|
||
case []byte:
|
||
if s := m.GetBytesByIndex(index); s != nil {
|
||
return &jsonData{T: GO_VALUE_SLICE_BYTE, V: s, S: len(s)}
|
||
}
|
||
case float32, float64:
|
||
//不带有 . 转为 int 类型
|
||
sv := m.toFloat64(s)
|
||
if strings.Index(strconv.FormatFloat(sv, 'G', -1, 64), ".") != -1 {
|
||
return &jsonData{T: GO_VALUE_FLOAT64, V: sv, S: 8}
|
||
} else {
|
||
return &jsonData{T: GO_VALUE_INT, V: sv, S: strconv.IntSize}
|
||
}
|
||
case bool:
|
||
return &jsonData{T: GO_VALUE_BOOL, V: m.GetBoolByIndex(index), S: 1}
|
||
case []any:
|
||
if index < m.S {
|
||
if v, ok := m.V.([]any)[index].([]any); ok {
|
||
return &jsonData{T: GO_VALUE_SLICE, V: v, S: len(v)}
|
||
}
|
||
}
|
||
case map[string]any:
|
||
if v, ok := s.(map[string]any); ok {
|
||
return &jsonData{T: GO_VALUE_MAP, V: v, S: len(v)}
|
||
}
|
||
}
|
||
}
|
||
return nil
|
||
}
|
||
|
||
func (m *jsonData) Set(key string, value any) {
|
||
if m.IsObject() {
|
||
switch value.(type) {
|
||
case JSON:
|
||
value = value.(JSON).Data()
|
||
case JSONObject:
|
||
value = value.(JSONObject).Data()
|
||
case JSONArray:
|
||
value = value.(JSONArray).Data()
|
||
}
|
||
m.V.(map[string]any)[key] = value
|
||
m.S++
|
||
}
|
||
}
|
||
|
||
func (m *jsonData) RemoveByKey(key string) {
|
||
if m.IsObject() {
|
||
if _, ok := m.V.(map[string]any)[key]; ok {
|
||
delete(m.V.(map[string]any), key)
|
||
m.S--
|
||
}
|
||
}
|
||
}
|
||
|
||
func (m *jsonData) GetStringByKey(key string) string {
|
||
if m.IsObject() {
|
||
if r, ok := m.V.(map[string]any)[key].(string); ok {
|
||
return r
|
||
}
|
||
}
|
||
return ""
|
||
}
|
||
|
||
func (m *jsonData) GetIntByKey(key string) int {
|
||
if m.IsObject() {
|
||
return m.toInt(m.V.(map[string]any)[key])
|
||
}
|
||
return 0
|
||
}
|
||
|
||
func (m *jsonData) GetUIntByKey(key string) uint {
|
||
if m.IsObject() {
|
||
return m.toUInt(m.V.(map[string]any)[key])
|
||
}
|
||
return 0
|
||
}
|
||
|
||
func (m *jsonData) GetBytesByKey(key string) []byte {
|
||
if m.IsObject() {
|
||
return m.toBytes(m.V.(map[string]any)[key])
|
||
}
|
||
return nil
|
||
}
|
||
|
||
func (m *jsonData) GetFloatByKey(key string) float64 {
|
||
if m.IsObject() {
|
||
return m.toFloat64(m.V.(map[string]any)[key])
|
||
}
|
||
return 0
|
||
}
|
||
|
||
func (m *jsonData) GetBoolByKey(key string) bool {
|
||
if m.IsObject() {
|
||
if s, ok := m.V.(map[string]any)[key].(bool); ok {
|
||
return s
|
||
}
|
||
}
|
||
return false
|
||
}
|
||
|
||
func (m *jsonData) GetArrayByKey(key string) JSONArray {
|
||
if m.IsObject() {
|
||
if v, ok := m.V.(map[string]any)[key].([]any); ok {
|
||
return &jsonData{T: GO_VALUE_SLICE, V: v, S: len(v)}
|
||
}
|
||
}
|
||
return nil
|
||
}
|
||
|
||
func (m *jsonData) GetObjectByKey(key string) JSONObject {
|
||
if m.IsObject() {
|
||
if v, ok := m.V.(map[string]any)[key].(map[string]any); ok {
|
||
return &jsonData{T: GO_VALUE_MAP, V: v, S: len(v)}
|
||
}
|
||
}
|
||
return nil
|
||
}
|
||
|
||
func (m *jsonData) GetByKey(key string) JSON {
|
||
if m.IsObject() {
|
||
s := m.V.(map[string]any)[key]
|
||
switch s.(type) {
|
||
case json.Number:
|
||
if v, err := s.(json.Number).Int64(); err == nil {
|
||
return &jsonData{T: GO_VALUE_INT, V: v, S: strconv.IntSize}
|
||
}
|
||
case string:
|
||
if r, ok := s.(string); ok {
|
||
return &jsonData{T: GO_VALUE_STRING, V: r, S: len(r)}
|
||
}
|
||
case int, int8, int16, int32, int64:
|
||
if s := m.GetIntByKey(key); s != 0 {
|
||
return &jsonData{T: GO_VALUE_INT, V: s, S: strconv.IntSize}
|
||
}
|
||
case uint, uint8, uint16, uint32, uint64:
|
||
if s := m.GetUIntByKey(key); s != 0 {
|
||
return &jsonData{T: GO_VALUE_UINT, V: s, S: strconv.IntSize}
|
||
}
|
||
case []byte:
|
||
if s := m.GetBytesByKey(key); s != nil {
|
||
return &jsonData{T: GO_VALUE_SLICE_BYTE, V: s, S: len(s)}
|
||
}
|
||
case float32, float64:
|
||
//不带有 . 转为 int 类型
|
||
sv := m.toFloat64(s)
|
||
if strings.Index(strconv.FormatFloat(sv, 'G', -1, 64), ".") != -1 {
|
||
return &jsonData{T: GO_VALUE_FLOAT64, V: sv, S: 8}
|
||
} else {
|
||
return &jsonData{T: GO_VALUE_INT, V: int(sv), S: strconv.IntSize}
|
||
}
|
||
case bool:
|
||
return &jsonData{T: GO_VALUE_BOOL, V: m.GetBytesByKey(key), S: 1}
|
||
case []any:
|
||
if v, ok := m.V.(map[string]any)[key].([]any); ok {
|
||
return &jsonData{T: GO_VALUE_SLICE, V: v, S: len(v)}
|
||
}
|
||
case map[string]any:
|
||
if v, ok := s.(map[string]any); ok {
|
||
return &jsonData{T: GO_VALUE_MAP, V: v, S: len(v)}
|
||
}
|
||
}
|
||
}
|
||
return nil
|
||
}
|
||
|
||
func (m *jsonData) String() string {
|
||
if m.IsString() {
|
||
return m.V.(string)
|
||
}
|
||
return ""
|
||
}
|
||
|
||
func (m *jsonData) Int() int {
|
||
return m.toInt(m.V)
|
||
}
|
||
|
||
func (m *jsonData) UInt() uint {
|
||
return m.toUInt(m.V)
|
||
}
|
||
|
||
func (m *jsonData) Bytes() []byte {
|
||
return m.toBytes(m.V)
|
||
}
|
||
|
||
func (m *jsonData) Float() float64 {
|
||
return m.toFloat64(m.V)
|
||
}
|
||
|
||
func (m *jsonData) Bool() bool {
|
||
if m.IsBool() {
|
||
return m.V.(bool)
|
||
}
|
||
return false
|
||
}
|
||
|
||
func (m *jsonData) JSONObject() JSONObject {
|
||
if m.IsObject() {
|
||
return m
|
||
}
|
||
return nil
|
||
}
|
||
|
||
func (m *jsonData) JSONArray() JSONArray {
|
||
if m.IsArray() {
|
||
return m
|
||
}
|
||
return nil
|
||
}
|
||
|
||
func (m *jsonData) toBytes(s any) []byte {
|
||
switch s.(type) {
|
||
case []byte:
|
||
return s.([]byte)
|
||
case string:
|
||
return []byte(s.(string))
|
||
case bool:
|
||
return []byte{common.BoolToByte(s.(bool))}
|
||
case float32:
|
||
return common.Float32ToBytes(s.(float32))
|
||
case float64:
|
||
return common.Float64ToBytes(s.(float64))
|
||
case int:
|
||
return common.IntToBytes(s.(int))
|
||
case int8:
|
||
return common.Int8ToBytes(s.(int8))
|
||
case int16:
|
||
return common.Int16ToBytes(s.(int16))
|
||
case int32:
|
||
return common.Int32ToBytes(s.(int32))
|
||
case int64:
|
||
return common.Int64ToBytes(s.(int64))
|
||
case uint:
|
||
return common.UIntToBytes(s.(uint))
|
||
case uint8:
|
||
return common.UInt8ToBytes(s.(uint8))
|
||
case uint16:
|
||
return common.UInt16ToBytes(s.(uint16))
|
||
case uint32:
|
||
return common.UInt32ToBytes(s.(uint32))
|
||
case uint64:
|
||
return common.UInt64ToBytes(s.(uint64))
|
||
default: // slice or map or other
|
||
if r, err := jsoniter.Marshal(s); err == nil {
|
||
return r
|
||
}
|
||
}
|
||
return nil
|
||
}
|
||
|
||
func (m *jsonData) Keys() []string {
|
||
if m.IsObject() {
|
||
var result []string
|
||
for key, _ := range m.V.(map[string]any) {
|
||
result = append(result, key)
|
||
}
|
||
return result
|
||
}
|
||
return nil
|
||
}
|
||
|
||
func (m *jsonData) ToJSONString() string {
|
||
return string(m.Bytes())
|
||
}
|
||
|
||
func (m *jsonData) IsString() bool {
|
||
return m.T == GO_VALUE_STRING
|
||
}
|
||
|
||
func (m *jsonData) IsInt() bool {
|
||
return m.T == GO_VALUE_INT
|
||
}
|
||
|
||
func (m *jsonData) IsUInt() bool {
|
||
return m.T == GO_VALUE_UINT
|
||
}
|
||
|
||
func (m *jsonData) IsBytes() bool {
|
||
return m.T == GO_VALUE_SLICE_BYTE
|
||
}
|
||
|
||
func (m *jsonData) IsFloat() bool {
|
||
return m.T == GO_VALUE_FLOAT64
|
||
}
|
||
|
||
func (m *jsonData) IsBool() bool {
|
||
return m.T == GO_VALUE_BOOL
|
||
}
|
||
|
||
func (m *jsonData) IsObject() bool {
|
||
return m.T == GO_VALUE_MAP
|
||
}
|
||
|
||
func (m *jsonData) IsArray() bool {
|
||
return m.T == GO_VALUE_SLICE
|
||
}
|
||
|
||
func (m *jsonData) Clear() {
|
||
if m.IsObject() {
|
||
m.V = make(map[string]any)
|
||
} else if m.IsArray() {
|
||
m.V = make([]any, 0)
|
||
} else {
|
||
m.V = nil
|
||
}
|
||
m.S = 0
|
||
m.T = GO_VALUE_INVALID
|
||
}
|
||
|
||
func (m *jsonData) toFloat64(s any) float64 {
|
||
switch s.(type) {
|
||
case float32:
|
||
return float64(s.(float32))
|
||
case float64:
|
||
return s.(float64)
|
||
case int:
|
||
return float64(s.(int))
|
||
case int8:
|
||
return float64(s.(int8))
|
||
case int16:
|
||
return float64(s.(int16))
|
||
case int32:
|
||
return float64(s.(int32))
|
||
case int64:
|
||
return float64(s.(int64))
|
||
case uint:
|
||
return float64(s.(uint))
|
||
case uint8:
|
||
return float64(s.(uint8))
|
||
case uint16:
|
||
return float64(s.(uint16))
|
||
case uint32:
|
||
return float64(s.(uint32))
|
||
case uint64:
|
||
return float64(s.(uint64))
|
||
}
|
||
return 0
|
||
}
|
||
|
||
func (m *jsonData) toInt(s any) int {
|
||
switch s.(type) {
|
||
case float32:
|
||
return int(s.(float32))
|
||
case float64:
|
||
return int(s.(float64))
|
||
case int:
|
||
return s.(int)
|
||
case int8:
|
||
return int(s.(int8))
|
||
case int16:
|
||
return int(s.(int16))
|
||
case int32:
|
||
return int(s.(int32))
|
||
case int64:
|
||
return int(s.(int64))
|
||
case uint:
|
||
return int(s.(uint))
|
||
case uint8:
|
||
return int(s.(uint8))
|
||
case uint16:
|
||
return int(s.(uint16))
|
||
case uint32:
|
||
return int(s.(uint32))
|
||
case uint64:
|
||
return int(s.(uint64))
|
||
}
|
||
return 0
|
||
}
|
||
|
||
func (m *jsonData) toUInt(s any) uint {
|
||
switch s.(type) {
|
||
case float32:
|
||
return uint(s.(float32))
|
||
case float64:
|
||
return uint(s.(float64))
|
||
case int:
|
||
return uint(s.(int))
|
||
case int8:
|
||
return uint(s.(int8))
|
||
case int16:
|
||
return uint(s.(int16))
|
||
case int32:
|
||
return uint(s.(int32))
|
||
case int64:
|
||
return uint(s.(int64))
|
||
case uint:
|
||
return s.(uint)
|
||
case uint8:
|
||
return uint(s.(uint8))
|
||
case uint16:
|
||
return uint(s.(uint16))
|
||
case uint32:
|
||
return uint(s.(uint32))
|
||
case uint64:
|
||
return uint(s.(uint64))
|
||
}
|
||
return 0
|
||
}
|