/* * Copyright 2021 ByteDance Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package ast import ( `encoding/json` `fmt` `strconv` `unsafe` `github.com/bytedance/sonic/internal/native/types` `github.com/bytedance/sonic/internal/rt` ) const ( _V_NONE types.ValueType = 0 _V_NODE_BASE types.ValueType = 1 << 5 _V_LAZY types.ValueType = 1 << 7 _V_RAW types.ValueType = 1 << 8 _V_NUMBER = _V_NODE_BASE + 1 _V_ANY = _V_NODE_BASE + 2 _V_ARRAY_LAZY = _V_LAZY | types.V_ARRAY _V_OBJECT_LAZY = _V_LAZY | types.V_OBJECT _MASK_LAZY = _V_LAZY - 1 _MASK_RAW = _V_RAW - 1 ) const ( V_NONE = 0 V_ERROR = 1 V_NULL = int(types.V_NULL) V_TRUE = int(types.V_TRUE) V_FALSE = int(types.V_FALSE) V_ARRAY = int(types.V_ARRAY) V_OBJECT = int(types.V_OBJECT) V_STRING = int(types.V_STRING) V_NUMBER = int(_V_NUMBER) V_ANY = int(_V_ANY) ) type Node struct { t types.ValueType l uint p unsafe.Pointer } // UnmarshalJSON is just an adapter to json.Unmarshaler. // If you want better performance, use Searcher.GetByPath() directly func (self *Node) UnmarshalJSON(data []byte) (err error) { *self = NewRaw(string(data)) return self.Check() } /** Node Type Accessor **/ // Type returns json type represented by the node // It will be one of belows: // V_NONE = 0 (empty node, key not exists) // V_ERROR = 1 (error node) // V_NULL = 2 (json value `null`, key exists) // V_TRUE = 3 (json value `true`) // V_FALSE = 4 (json value `false`) // V_ARRAY = 5 (json value array) // V_OBJECT = 6 (json value object) // V_STRING = 7 (json value string) // V_NUMBER = 33 (json value number ) // V_ANY = 34 (golang interface{}) func (self Node) Type() int { return int(self.t & _MASK_LAZY & _MASK_RAW) } func (self Node) itype() types.ValueType { return self.t & _MASK_LAZY & _MASK_RAW } // Exists returns false only if the self is nil or empty node V_NONE func (self *Node) Exists() bool { return self.Valid() && self.t != _V_NONE } // Valid reports if self is NOT V_ERROR or nil func (self *Node) Valid() bool { if self == nil { return false } return self.t != V_ERROR } // Check checks if the node itself is valid, and return: // - ErrNotExist If the node is nil // - Its underlying error If the node is V_ERROR func (self *Node) Check() error { if self == nil { return ErrNotExist } else if self.t != V_ERROR { return nil } else { return self } } // IsRaw returns true if node's underlying value is raw json func (self Node) IsRaw() bool { return self.t&_V_RAW != 0 } func (self *Node) isLazy() bool { return self != nil && self.t&_V_LAZY != 0 } func (self *Node) isAny() bool { return self != nil && self.t == _V_ANY } /** Simple Value Methods **/ // Raw returns json representation of the node, func (self *Node) Raw() (string, error) { if self == nil { return "", ErrNotExist } if !self.IsRaw() { buf, err := self.MarshalJSON() return rt.Mem2Str(buf), err } return self.toString(), nil } func (self *Node) checkRaw() error { if err := self.Check(); err != nil { return err } if self.IsRaw() { self.parseRaw(false) } return self.Check() } // Bool returns bool value represented by this node, // including types.V_TRUE|V_FALSE|V_NUMBER|V_STRING|V_ANY|V_NULL, // V_NONE will return error func (self *Node) Bool() (bool, error) { if err := self.checkRaw(); err != nil { return false, err } switch self.t { case types.V_TRUE : return true , nil case types.V_FALSE : return false, nil case types.V_NULL : return false, nil case _V_NUMBER : if i, err := self.toInt64(); err == nil { return i != 0, nil } else if f, err := self.toFloat64(); err == nil { return f != 0, nil } else { return false, err } case types.V_STRING: return strconv.ParseBool(self.toString()) case _V_ANY : any := self.packAny() switch v := any.(type) { case bool : return v, nil case int : return v != 0, nil case int8 : return v != 0, nil case int16 : return v != 0, nil case int32 : return v != 0, nil case int64 : return v != 0, nil case uint : return v != 0, nil case uint8 : return v != 0, nil case uint16 : return v != 0, nil case uint32 : return v != 0, nil case uint64 : return v != 0, nil case float32: return v != 0, nil case float64: return v != 0, nil case string : return strconv.ParseBool(v) case json.Number: if i, err := v.Int64(); err == nil { return i != 0, nil } else if f, err := v.Float64(); err == nil { return f != 0, nil } else { return false, err } default: return false, ErrUnsupportType } default : return false, ErrUnsupportType } } // Int64 casts the node to int64 value, // including V_NUMBER|V_TRUE|V_FALSE|V_ANY|V_STRING // V_NONE it will return error func (self *Node) Int64() (int64, error) { if err := self.checkRaw(); err != nil { return 0, err } switch self.t { case _V_NUMBER, types.V_STRING : if i, err := self.toInt64(); err == nil { return i, nil } else if f, err := self.toFloat64(); err == nil { return int64(f), nil } else { return 0, err } case types.V_TRUE : return 1, nil case types.V_FALSE : return 0, nil case types.V_NULL : return 0, nil case _V_ANY : any := self.packAny() switch v := any.(type) { case bool : if v { return 1, nil } else { return 0, nil } case int : return int64(v), nil case int8 : return int64(v), nil case int16 : return int64(v), nil case int32 : return int64(v), nil case int64 : return int64(v), nil case uint : return int64(v), nil case uint8 : return int64(v), nil case uint16 : return int64(v), nil case uint32 : return int64(v), nil case uint64 : return int64(v), nil case float32: return int64(v), nil case float64: return int64(v), nil case string : if i, err := strconv.ParseInt(v, 10, 64); err == nil { return i, nil } else if f, err := strconv.ParseFloat(v, 64); err == nil { return int64(f), nil } else { return 0, err } case json.Number: if i, err := v.Int64(); err == nil { return i, nil } else if f, err := v.Float64(); err == nil { return int64(f), nil } else { return 0, err } default: return 0, ErrUnsupportType } default : return 0, ErrUnsupportType } } // StrictInt64 exports underlying int64 value, including V_NUMBER, V_ANY func (self *Node) StrictInt64() (int64, error) { if err := self.checkRaw(); err != nil { return 0, err } switch self.t { case _V_NUMBER : return self.toInt64() case _V_ANY : any := self.packAny() switch v := any.(type) { case int : return int64(v), nil case int8 : return int64(v), nil case int16 : return int64(v), nil case int32 : return int64(v), nil case int64 : return int64(v), nil case uint : return int64(v), nil case uint8 : return int64(v), nil case uint16: return int64(v), nil case uint32: return int64(v), nil case uint64: return int64(v), nil case json.Number: if i, err := v.Int64(); err == nil { return i, nil } else { return 0, err } default: return 0, ErrUnsupportType } default : return 0, ErrUnsupportType } } func castNumber(v bool) json.Number { if v { return json.Number("1") } else { return json.Number("0") } } // Number casts node to float64, // including V_NUMBER|V_TRUE|V_FALSE|V_ANY|V_STRING|V_NULL, // V_NONE it will return error func (self *Node) Number() (json.Number, error) { if err := self.checkRaw(); err != nil { return json.Number(""), err } switch self.t { case _V_NUMBER : return self.toNumber(), nil case types.V_STRING : if _, err := self.toInt64(); err == nil { return self.toNumber(), nil } else if _, err := self.toFloat64(); err == nil { return self.toNumber(), nil } else { return json.Number(""), err } case types.V_TRUE : return json.Number("1"), nil case types.V_FALSE : return json.Number("0"), nil case types.V_NULL : return json.Number("0"), nil case _V_ANY : any := self.packAny() switch v := any.(type) { case bool : return castNumber(v), nil case int : return castNumber(v != 0), nil case int8 : return castNumber(v != 0), nil case int16 : return castNumber(v != 0), nil case int32 : return castNumber(v != 0), nil case int64 : return castNumber(v != 0), nil case uint : return castNumber(v != 0), nil case uint8 : return castNumber(v != 0), nil case uint16 : return castNumber(v != 0), nil case uint32 : return castNumber(v != 0), nil case uint64 : return castNumber(v != 0), nil case float32: return castNumber(v != 0), nil case float64: return castNumber(v != 0), nil case string : if _, err := strconv.ParseFloat(v, 64); err == nil { return json.Number(v), nil } else { return json.Number(""), err } case json.Number: return v, nil default: return json.Number(""), ErrUnsupportType } default : return json.Number(""), ErrUnsupportType } } // Number exports underlying float64 value, including V_NUMBER, V_ANY of json.Number func (self *Node) StrictNumber() (json.Number, error) { if err := self.checkRaw(); err != nil { return json.Number(""), err } switch self.t { case _V_NUMBER : return self.toNumber() , nil case _V_ANY : if v, ok := self.packAny().(json.Number); ok { return v, nil } else { return json.Number(""), ErrUnsupportType } default : return json.Number(""), ErrUnsupportType } } // String cast node to string, // including V_NUMBER|V_TRUE|V_FALSE|V_ANY|V_STRING|V_NULL, // V_NONE it will return error func (self *Node) String() (string, error) { if err := self.checkRaw(); err != nil { return "", err } switch self.t { case types.V_NULL : return "" , nil case types.V_TRUE : return "true" , nil case types.V_FALSE : return "false", nil case types.V_STRING, _V_NUMBER : return self.toString(), nil case _V_ANY : any := self.packAny() switch v := any.(type) { case bool : return strconv.FormatBool(v), nil case int : return strconv.Itoa(v), nil case int8 : return strconv.Itoa(int(v)), nil case int16 : return strconv.Itoa(int(v)), nil case int32 : return strconv.Itoa(int(v)), nil case int64 : return strconv.Itoa(int(v)), nil case uint : return strconv.Itoa(int(v)), nil case uint8 : return strconv.Itoa(int(v)), nil case uint16 : return strconv.Itoa(int(v)), nil case uint32 : return strconv.Itoa(int(v)), nil case uint64 : return strconv.Itoa(int(v)), nil case float32: return strconv.FormatFloat(float64(v), 'g', -1, 64), nil case float64: return strconv.FormatFloat(float64(v), 'g', -1, 64), nil case string : return v, nil case json.Number: return v.String(), nil default: return "", ErrUnsupportType } default : return "" , ErrUnsupportType } } // StrictString returns string value (unescaped), includeing V_STRING, V_ANY of string. // In other cases, it will return empty string. func (self *Node) StrictString() (string, error) { if err := self.checkRaw(); err != nil { return "", err } switch self.t { case types.V_STRING : return self.toString(), nil case _V_ANY : if v, ok := self.packAny().(string); ok { return v, nil } else { return "", ErrUnsupportType } default : return "", ErrUnsupportType } } // Float64 cast node to float64, // including V_NUMBER|V_TRUE|V_FALSE|V_ANY|V_STRING|V_NULL, // V_NONE it will return error func (self *Node) Float64() (float64, error) { if err := self.checkRaw(); err != nil { return 0.0, err } switch self.t { case _V_NUMBER, types.V_STRING : return self.toFloat64() case types.V_TRUE : return 1.0, nil case types.V_FALSE : return 0.0, nil case types.V_NULL : return 0.0, nil case _V_ANY : any := self.packAny() switch v := any.(type) { case bool : if v { return 1.0, nil } else { return 0.0, nil } case int : return float64(v), nil case int8 : return float64(v), nil case int16 : return float64(v), nil case int32 : return float64(v), nil case int64 : return float64(v), nil case uint : return float64(v), nil case uint8 : return float64(v), nil case uint16 : return float64(v), nil case uint32 : return float64(v), nil case uint64 : return float64(v), nil case float32: return float64(v), nil case float64: return float64(v), nil case string : if f, err := strconv.ParseFloat(v, 64); err == nil { return float64(f), nil } else { return 0, err } case json.Number: if f, err := v.Float64(); err == nil { return float64(f), nil } else { return 0, err } default : return 0, ErrUnsupportType } default : return 0.0, ErrUnsupportType } } // Float64 exports underlying float64 value, includeing V_NUMBER, V_ANY func (self *Node) StrictFloat64() (float64, error) { if err := self.checkRaw(); err != nil { return 0.0, err } switch self.t { case _V_NUMBER : return self.toFloat64() case _V_ANY : any := self.packAny() switch v := any.(type) { case float32 : return float64(v), nil case float64 : return float64(v), nil default : return 0, ErrUnsupportType } default : return 0.0, ErrUnsupportType } } /** Sequencial Value Methods **/ // Len returns children count of a array|object|string node // WARN: For partially loaded node, it also works but only counts the parsed children func (self *Node) Len() (int, error) { if err := self.checkRaw(); err != nil { return 0, err } if self.t == types.V_ARRAY || self.t == types.V_OBJECT || self.t == _V_ARRAY_LAZY || self.t == _V_OBJECT_LAZY || self.t == types.V_STRING { return int(self.l), nil } else if self.t == _V_NONE || self.t == types.V_NULL { return 0, nil } else { return 0, ErrUnsupportType } } func (self Node) len() int { return int(self.l) } // Cap returns malloc capacity of a array|object node for children func (self *Node) Cap() (int, error) { if err := self.checkRaw(); err != nil { return 0, err } switch self.t { case types.V_ARRAY: return (*linkedNodes)(self.p).Cap(), nil case types.V_OBJECT: return (*linkedPairs)(self.p).Cap(), nil case _V_ARRAY_LAZY: return (*parseArrayStack)(self.p).v.Cap(), nil case _V_OBJECT_LAZY: return (*parseObjectStack)(self.p).v.Cap(), nil case _V_NONE, types.V_NULL: return 0, nil default: return 0, ErrUnsupportType } } // Set sets the node of given key under self, and reports if the key has existed. // // If self is V_NONE or V_NULL, it becomes V_OBJECT and sets the node at the key. func (self *Node) Set(key string, node Node) (bool, error) { if err := self.Check(); err != nil { return false, err } if err := node.Check(); err != nil { return false, err } if self.t == _V_NONE || self.t == types.V_NULL { *self = NewObject([]Pair{{key, node}}) return false, nil } else if self.itype() != types.V_OBJECT { return false, ErrUnsupportType } p := self.Get(key) if !p.Exists() { // self must be fully-loaded here if self.len() == 0 { *self = newObject(new(linkedPairs)) } s := (*linkedPairs)(self.p) s.Push(Pair{key, node}) self.l++ return false, nil } else if err := p.Check(); err != nil { return false, err } *p = node return true, nil } // SetAny wraps val with V_ANY node, and Set() the node. func (self *Node) SetAny(key string, val interface{}) (bool, error) { return self.Set(key, NewAny(val)) } // Unset REMOVE (soft) the node of given key under object parent, and reports if the key has existed. func (self *Node) Unset(key string) (bool, error) { if err := self.should(types.V_OBJECT, "an object"); err != nil { return false, err } // NOTICE: must get acurate length before deduct if err := self.skipAllKey(); err != nil { return false, err } p, i := self.skipKey(key) if !p.Exists() { return false, nil } else if err := p.Check(); err != nil { return false, err } self.removePair(i) return true, nil } // SetByIndex sets the node of given index, and reports if the key has existed. // // The index must be within self's children. func (self *Node) SetByIndex(index int, node Node) (bool, error) { if err := self.Check(); err != nil { return false, err } if err := node.Check(); err != nil { return false, err } if index == 0 && (self.t == _V_NONE || self.t == types.V_NULL) { *self = NewArray([]Node{node}) return false, nil } p := self.Index(index) if !p.Exists() { return false, ErrNotExist } else if err := p.Check(); err != nil { return false, err } *p = node return true, nil } // SetAny wraps val with V_ANY node, and SetByIndex() the node. func (self *Node) SetAnyByIndex(index int, val interface{}) (bool, error) { return self.SetByIndex(index, NewAny(val)) } // UnsetByIndex REOMVE (softly) the node of given index. // // WARN: this will change address of elements, which is a dangerous action. // Use Unset() for object or Pop() for array instead. func (self *Node) UnsetByIndex(index int) (bool, error) { if err := self.checkRaw(); err != nil { return false, err } var p *Node it := self.itype() if it == types.V_ARRAY { if err := self.skipAllIndex(); err != nil { return false, err } p = self.nodeAt(index) } else if it == types.V_OBJECT { if err := self.skipAllKey(); err != nil { return false, err } pr := self.pairAt(index) if pr == nil { return false, ErrNotExist } p = &pr.Value } else { return false, ErrUnsupportType } if !p.Exists() { return false, ErrNotExist } // last elem if index == self.len() - 1 { return true, self.Pop() } // not last elem, self.len() change but linked-chunk not change if it == types.V_ARRAY { self.removeNode(index) }else if it == types.V_OBJECT { self.removePair(index) } return true, nil } // Add appends the given node under self. // // If self is V_NONE or V_NULL, it becomes V_ARRAY and sets the node at index 0. func (self *Node) Add(node Node) error { if err := self.Check(); err != nil { return err } if self != nil && (self.t == _V_NONE || self.t == types.V_NULL) { *self = NewArray([]Node{node}) return nil } if err := self.should(types.V_ARRAY, "an array"); err != nil { return err } s, err := self.unsafeArray() if err != nil { return err } // Notice: array won't have unset node in tail s.Push(node) self.l++ return nil } // Pop remove the last child of the V_Array or V_Object node. func (self *Node) Pop() error { if err := self.checkRaw(); err != nil { return err } if it := self.itype(); it == types.V_ARRAY { s, err := self.unsafeArray() if err != nil { return err } // remove tail unset nodes for i := s.Len()-1; i >= 0; i-- { if s.At(i).Exists() { s.Pop() self.l-- break } s.Pop() } } else if it == types.V_OBJECT { s, err := self.unsafeMap() if err != nil { return err } // remove tail unset nodes for i := s.Len()-1; i >= 0; i-- { if p := s.At(i); p != nil && p.Value.Exists() { s.Pop() self.l-- break } s.Pop() } } else { return ErrUnsupportType } return nil } // Move moves the child at src index to dst index, // meanwhile slides sliblings from src+1 to dst. // // WARN: this will change address of elements, which is a dangerous action. func (self *Node) Move(dst, src int) error { if err := self.should(types.V_ARRAY, "an array"); err != nil { return err } s, err := self.unsafeArray() if err != nil { return err } // check if any unset node exists if l := s.Len(); self.len() != l { di, si := dst, src // find real pos of src and dst for i := 0; i < l; i++ { if s.At(i).Exists() { di-- si-- } if di == -1 { dst = i di-- } if si == -1 { src = i si-- } if di == -2 && si == -2 { break } } } s.MoveOne(src, dst) return nil } // SetAny wraps val with V_ANY node, and Add() the node. func (self *Node) AddAny(val interface{}) error { return self.Add(NewAny(val)) } // GetByPath load given path on demands, // which only ensure nodes before this path got parsed. // // Note, the api expects the json is well-formed at least, // otherwise it may return unexpected result. func (self *Node) GetByPath(path ...interface{}) *Node { if !self.Valid() { return self } var s = self for _, p := range path { switch p := p.(type) { case int: s = s.Index(p) if !s.Valid() { return s } case string: s = s.Get(p) if !s.Valid() { return s } default: panic("path must be either int or string") } } return s } // Get loads given key of an object node on demands func (self *Node) Get(key string) *Node { if err := self.should(types.V_OBJECT, "an object"); err != nil { return unwrapError(err) } n, _ := self.skipKey(key) return n } // Index indexies node at given idx, // node type CAN be either V_OBJECT or V_ARRAY func (self *Node) Index(idx int) *Node { if err := self.checkRaw(); err != nil { return unwrapError(err) } it := self.itype() if it == types.V_ARRAY { return self.skipIndex(idx) }else if it == types.V_OBJECT { pr := self.skipIndexPair(idx) if pr == nil { return newError(_ERR_NOT_FOUND, "value not exists") } return &pr.Value } else { return newError(_ERR_UNSUPPORT_TYPE, fmt.Sprintf("unsupported type: %v", self.itype())) } } // IndexPair indexies pair at given idx, // node type MUST be either V_OBJECT func (self *Node) IndexPair(idx int) *Pair { if err := self.should(types.V_OBJECT, "an object"); err != nil { return nil } return self.skipIndexPair(idx) } // IndexOrGet firstly use idx to index a value and check if its key matches // If not, then use the key to search value func (self *Node) IndexOrGet(idx int, key string) *Node { if err := self.should(types.V_OBJECT, "an object"); err != nil { return unwrapError(err) } pr := self.skipIndexPair(idx) if pr != nil && pr.Key == key { return &pr.Value } n, _ := self.skipKey(key) return n } /** Generic Value Converters **/ // Map loads all keys of an object node func (self *Node) Map() (map[string]interface{}, error) { if self.isAny() { any := self.packAny() if v, ok := any.(map[string]interface{}); ok { return v, nil } else { return nil, ErrUnsupportType } } if err := self.should(types.V_OBJECT, "an object"); err != nil { return nil, err } if err := self.loadAllKey(); err != nil { return nil, err } return self.toGenericObject() } // MapUseNumber loads all keys of an object node, with numeric nodes casted to json.Number func (self *Node) MapUseNumber() (map[string]interface{}, error) { if self.isAny() { any := self.packAny() if v, ok := any.(map[string]interface{}); ok { return v, nil } else { return nil, ErrUnsupportType } } if err := self.should(types.V_OBJECT, "an object"); err != nil { return nil, err } if err := self.loadAllKey(); err != nil { return nil, err } return self.toGenericObjectUseNumber() } // MapUseNode scans both parsed and non-parsed chidren nodes, // and map them by their keys func (self *Node) MapUseNode() (map[string]Node, error) { if self.isAny() { any := self.packAny() if v, ok := any.(map[string]Node); ok { return v, nil } else { return nil, ErrUnsupportType } } if err := self.should(types.V_OBJECT, "an object"); err != nil { return nil, err } if err := self.skipAllKey(); err != nil { return nil, err } return self.toGenericObjectUseNode() } // MapUnsafe exports the underlying pointer to its children map // WARN: don't use it unless you know what you are doing // // Deprecated: this API now returns copied nodes instead of directly reference, // func (self *Node) UnsafeMap() ([]Pair, error) { // if err := self.should(types.V_OBJECT, "an object"); err != nil { // return nil, err // } // if err := self.skipAllKey(); err != nil { // return nil, err // } // return self.toGenericObjectUsePair() // } //go:nocheckptr func (self *Node) unsafeMap() (*linkedPairs, error) { if err := self.skipAllKey(); err != nil { return nil, err } if self.p == nil { *self = newObject(new(linkedPairs)) } return (*linkedPairs)(self.p), nil } // SortKeys sorts children of a V_OBJECT node in ascending key-order. // If recurse is true, it recursively sorts children's children as long as a V_OBJECT node is found. func (self *Node) SortKeys(recurse bool) error { // check raw node first if err := self.checkRaw(); err != nil { return err } if self.itype() == types.V_OBJECT { return self.sortKeys(recurse) } else { var err error err2 := self.ForEach(func(path Sequence, node *Node) bool { it := node.itype() if it == types.V_ARRAY || it == types.V_OBJECT { err = node.SortKeys(recurse) if err != nil { return false } } return true }) if err != nil { return err } return err2 } } func (self *Node) sortKeys(recurse bool) (err error) { ps, err := self.unsafeMap() if err != nil { return err } ps.Sort() if recurse { var sc Scanner sc = func(path Sequence, node *Node) bool { if node.itype() == types.V_OBJECT { if err := node.sortKeys(recurse); err != nil { return false } } if node.itype() == types.V_ARRAY { if err := node.ForEach(sc); err != nil { return false } } return true } if err := self.ForEach(sc); err != nil { return err } } return nil } // Array loads all indexes of an array node func (self *Node) Array() ([]interface{}, error) { if self.isAny() { any := self.packAny() if v, ok := any.([]interface{}); ok { return v, nil } else { return nil, ErrUnsupportType } } if err := self.should(types.V_ARRAY, "an array"); err != nil { return nil, err } if err := self.loadAllIndex(); err != nil { return nil, err } return self.toGenericArray() } // ArrayUseNumber loads all indexes of an array node, with numeric nodes casted to json.Number func (self *Node) ArrayUseNumber() ([]interface{}, error) { if self.isAny() { any := self.packAny() if v, ok := any.([]interface{}); ok { return v, nil } else { return nil, ErrUnsupportType } } if err := self.should(types.V_ARRAY, "an array"); err != nil { return nil, err } if err := self.loadAllIndex(); err != nil { return nil, err } return self.toGenericArrayUseNumber() } // ArrayUseNode copys both parsed and non-parsed chidren nodes, // and indexes them by original order func (self *Node) ArrayUseNode() ([]Node, error) { if self.isAny() { any := self.packAny() if v, ok := any.([]Node); ok { return v, nil } else { return nil, ErrUnsupportType } } if err := self.should(types.V_ARRAY, "an array"); err != nil { return nil, err } if err := self.skipAllIndex(); err != nil { return nil, err } return self.toGenericArrayUseNode() } // ArrayUnsafe exports the underlying pointer to its children array // WARN: don't use it unless you know what you are doing // // Deprecated: this API now returns copied nodes instead of directly reference, // which has no difference with ArrayUseNode // func (self *Node) UnsafeArray() ([]Node, error) { // if err := self.should(types.V_ARRAY, "an array"); err != nil { // return nil, err // } // if err := self.skipAllIndex(); err != nil { // return nil, err // } // return self.toGenericArrayUseNode() // } func (self *Node) unsafeArray() (*linkedNodes, error) { if err := self.skipAllIndex(); err != nil { return nil, err } if self.p == nil { *self = newArray(new(linkedNodes)) } return (*linkedNodes)(self.p), nil } // Interface loads all children under all pathes from this node, // and converts itself as generic type. // WARN: all numberic nodes are casted to float64 func (self *Node) Interface() (interface{}, error) { if err := self.checkRaw(); err != nil { return nil, err } switch self.t { case V_ERROR : return nil, self.Check() case types.V_NULL : return nil, nil case types.V_TRUE : return true, nil case types.V_FALSE : return false, nil case types.V_ARRAY : return self.toGenericArray() case types.V_OBJECT : return self.toGenericObject() case types.V_STRING : return self.toString(), nil case _V_NUMBER : v, err := self.toFloat64() if err != nil { return nil, err } return v, nil case _V_ARRAY_LAZY : if err := self.loadAllIndex(); err != nil { return nil, err } return self.toGenericArray() case _V_OBJECT_LAZY : if err := self.loadAllKey(); err != nil { return nil, err } return self.toGenericObject() case _V_ANY: switch v := self.packAny().(type) { case Node : return v.Interface() case *Node: return v.Interface() default : return v, nil } default : return nil, ErrUnsupportType } } func (self *Node) packAny() interface{} { return *(*interface{})(self.p) } // InterfaceUseNumber works same with Interface() // except numberic nodes are casted to json.Number func (self *Node) InterfaceUseNumber() (interface{}, error) { if err := self.checkRaw(); err != nil { return nil, err } switch self.t { case V_ERROR : return nil, self.Check() case types.V_NULL : return nil, nil case types.V_TRUE : return true, nil case types.V_FALSE : return false, nil case types.V_ARRAY : return self.toGenericArrayUseNumber() case types.V_OBJECT : return self.toGenericObjectUseNumber() case types.V_STRING : return self.toString(), nil case _V_NUMBER : return self.toNumber(), nil case _V_ARRAY_LAZY : if err := self.loadAllIndex(); err != nil { return nil, err } return self.toGenericArrayUseNumber() case _V_OBJECT_LAZY : if err := self.loadAllKey(); err != nil { return nil, err } return self.toGenericObjectUseNumber() case _V_ANY : return self.packAny(), nil default : return nil, ErrUnsupportType } } // InterfaceUseNode clone itself as a new node, // or its children as map[string]Node (or []Node) func (self *Node) InterfaceUseNode() (interface{}, error) { if err := self.checkRaw(); err != nil { return nil, err } switch self.t { case types.V_ARRAY : return self.toGenericArrayUseNode() case types.V_OBJECT : return self.toGenericObjectUseNode() case _V_ARRAY_LAZY : if err := self.skipAllIndex(); err != nil { return nil, err } return self.toGenericArrayUseNode() case _V_OBJECT_LAZY : if err := self.skipAllKey(); err != nil { return nil, err } return self.toGenericObjectUseNode() default : return *self, self.Check() } } // LoadAll loads all the node's children and children's children as parsed. // After calling it, the node can be safely used on concurrency func (self *Node) LoadAll() error { if self.IsRaw() { self.parseRaw(true) return self.Check() } switch self.itype() { case types.V_ARRAY: e := self.len() if err := self.loadAllIndex(); err != nil { return err } for i := 0; i < e; i++ { n := self.nodeAt(i) if n.IsRaw() { n.parseRaw(true) } if err := n.Check(); err != nil { return err } } return nil case types.V_OBJECT: e := self.len() if err := self.loadAllKey(); err != nil { return err } for i := 0; i < e; i++ { n := self.pairAt(i) if n.Value.IsRaw() { n.Value.parseRaw(true) } if err := n.Value.Check(); err != nil { return err } } return nil default: return self.Check() } } // Load loads the node's children as parsed. // After calling it, only the node itself can be used on concurrency (not include its children) func (self *Node) Load() error { if err := self.checkRaw(); err != nil { return err } switch self.t { case _V_ARRAY_LAZY: return self.skipAllIndex() case _V_OBJECT_LAZY: return self.skipAllKey() default: return self.Check() } } /**---------------------------------- Internal Helper Methods ----------------------------------**/ func (self *Node) should(t types.ValueType, s string) error { if err := self.checkRaw(); err != nil { return err } if self.itype() != t { return ErrUnsupportType } return nil } func (self *Node) nodeAt(i int) *Node { var p *linkedNodes if self.isLazy() { _, stack := self.getParserAndArrayStack() p = &stack.v } else { p = (*linkedNodes)(self.p) if l := p.Len(); l != self.len() { // some nodes got unset, iterate to skip them for j:=0; j 0 { /* linear search */ var p *Pair var i int if lazy { s := (*parseObjectStack)(self.p) p, i = s.v.Get(key) } else { p, i = (*linkedPairs)(self.p).Get(key) } if p != nil { return &p.Value, i } } /* not found */ if !lazy { return nil, -1 } // lazy load for last, i := self.skipNextPair(), nb; last != nil; last, i = self.skipNextPair(), i+1 { if last.Value.Check() != nil { return &last.Value, -1 } if last.Key == key { return &last.Value, i } } return nil, -1 } func (self *Node) skipIndex(index int) *Node { nb := self.len() if nb > index { v := self.nodeAt(index) return v } if !self.isLazy() { return nil } // lazy load for last := self.skipNextNode(); last != nil; last = self.skipNextNode(){ if last.Check() != nil { return last } if self.len() > index { return last } } return nil } func (self *Node) skipIndexPair(index int) *Pair { nb := self.len() if nb > index { return self.pairAt(index) } if !self.isLazy() { return nil } // lazy load for last := self.skipNextPair(); last != nil; last = self.skipNextPair(){ if last.Value.Check() != nil { return last } if self.len() > index { return last } } return nil } func (self *Node) loadAllIndex() error { if !self.isLazy() { return nil } var err types.ParsingError parser, stack := self.getParserAndArrayStack() parser.noLazy = true *self, err = parser.decodeArray(&stack.v) if err != 0 { return parser.ExportError(err) } return nil } func (self *Node) loadAllKey() error { if !self.isLazy() { return nil } var err types.ParsingError parser, stack := self.getParserAndObjectStack() parser.noLazy = true *self, err = parser.decodeObject(&stack.v) if err != 0 { return parser.ExportError(err) } return nil } func (self *Node) removeNode(i int) { node := self.nodeAt(i) if node == nil { return } *node = Node{} // NOTICE: not be consistent with linkedNode.Len() self.l-- } func (self *Node) removePair(i int) { last := self.pairAt(i) if last == nil { return } *last = Pair{} // NOTICE: should be consistent with linkedPair.Len() self.l-- } func (self *Node) toGenericArray() ([]interface{}, error) { nb := self.len() if nb == 0 { return []interface{}{}, nil } ret := make([]interface{}, 0, nb) /* convert each item */ it := self.values() for v := it.next(); v != nil; v = it.next() { vv, err := v.Interface() if err != nil { return nil, err } ret = append(ret, vv) } /* all done */ return ret, nil } func (self *Node) toGenericArrayUseNumber() ([]interface{}, error) { nb := self.len() if nb == 0 { return []interface{}{}, nil } ret := make([]interface{}, 0, nb) /* convert each item */ it := self.values() for v := it.next(); v != nil; v = it.next() { vv, err := v.InterfaceUseNumber() if err != nil { return nil, err } ret = append(ret, vv) } /* all done */ return ret, nil } func (self *Node) toGenericArrayUseNode() ([]Node, error) { var nb = self.len() if nb == 0 { return []Node{}, nil } var s = (*linkedNodes)(self.p) var out = make([]Node, nb) s.ToSlice(out) return out, nil } func (self *Node) toGenericObject() (map[string]interface{}, error) { nb := self.len() if nb == 0 { return map[string]interface{}{}, nil } ret := make(map[string]interface{}, nb) /* convert each item */ it := self.properties() for v := it.next(); v != nil; v = it.next() { vv, err := v.Value.Interface() if err != nil { return nil, err } ret[v.Key] = vv } /* all done */ return ret, nil } func (self *Node) toGenericObjectUseNumber() (map[string]interface{}, error) { nb := self.len() if nb == 0 { return map[string]interface{}{}, nil } ret := make(map[string]interface{}, nb) /* convert each item */ it := self.properties() for v := it.next(); v != nil; v = it.next() { vv, err := v.Value.InterfaceUseNumber() if err != nil { return nil, err } ret[v.Key] = vv } /* all done */ return ret, nil } func (self *Node) toGenericObjectUseNode() (map[string]Node, error) { var nb = self.len() if nb == 0 { return map[string]Node{}, nil } var s = (*linkedPairs)(self.p) var out = make(map[string]Node, nb) s.ToMap(out) /* all done */ return out, nil } /**------------------------------------ Factory Methods ------------------------------------**/ var ( nullNode = Node{t: types.V_NULL} trueNode = Node{t: types.V_TRUE} falseNode = Node{t: types.V_FALSE} ) // NewRaw creates a node of raw json. // If the input json is invalid, NewRaw returns a error Node. func NewRaw(json string) Node { parser := NewParserObj(json) start, err := parser.skip() if err != 0 { return *newError(err, err.Message()) } it := switchRawType(parser.s[start]) if it == _V_NONE { return Node{} } return newRawNode(parser.s[start:parser.p], it) } // NewAny creates a node of type V_ANY if any's type isn't Node or *Node, // which stores interface{} and can be only used for `.Interface()`\`.MarshalJSON()`. func NewAny(any interface{}) Node { switch n := any.(type) { case Node: return n case *Node: return *n default: return Node{ t: _V_ANY, p: unsafe.Pointer(&any), } } } // NewBytes encodes given src with Base64 (RFC 4648), and creates a node of type V_STRING. func NewBytes(src []byte) Node { if len(src) == 0 { panic("empty src bytes") } out := encodeBase64(src) return NewString(out) } // NewNull creates a node of type V_NULL func NewNull() Node { return Node{ p: nil, t: types.V_NULL, } } // NewBool creates a node of type bool: // If v is true, returns V_TRUE node // If v is false, returns V_FALSE node func NewBool(v bool) Node { var t = types.V_FALSE if v { t = types.V_TRUE } return Node{ p: nil, t: t, } } // NewNumber creates a json.Number node // v must be a decimal string complying with RFC8259 func NewNumber(v string) Node { return Node{ l: uint(len(v)), p: rt.StrPtr(v), t: _V_NUMBER, } } func (node Node) toNumber() json.Number { return json.Number(rt.StrFrom(node.p, int64(node.l))) } func (self Node) toString() string { return rt.StrFrom(self.p, int64(self.l)) } func (node Node) toFloat64() (float64, error) { ret, err := node.toNumber().Float64() if err != nil { return 0, err } return ret, nil } func (node Node) toInt64() (int64, error) { ret,err := node.toNumber().Int64() if err != nil { return 0, err } return ret, nil } func newBytes(v []byte) Node { return Node{ t: types.V_STRING, p: mem2ptr(v), l: uint(len(v)), } } // NewString creates a node of type V_STRING. // v is considered to be a valid UTF-8 string, // which means it won't be validated and unescaped. // when the node is encoded to json, v will be escaped. func NewString(v string) Node { return Node{ t: types.V_STRING, p: rt.StrPtr(v), l: uint(len(v)), } } // NewArray creates a node of type V_ARRAY, // using v as its underlying children func NewArray(v []Node) Node { s := new(linkedNodes) s.FromSlice(v) return newArray(s) } func newArray(v *linkedNodes) Node { return Node{ t: types.V_ARRAY, l: uint(v.Len()), p: unsafe.Pointer(v), } } func (self *Node) setArray(v *linkedNodes) { self.t = types.V_ARRAY self.l = uint(v.Len()) self.p = unsafe.Pointer(v) } // NewObject creates a node of type V_OBJECT, // using v as its underlying children func NewObject(v []Pair) Node { s := new(linkedPairs) s.FromSlice(v) return newObject(s) } func newObject(v *linkedPairs) Node { return Node{ t: types.V_OBJECT, l: uint(v.Len()), p: unsafe.Pointer(v), } } func (self *Node) setObject(v *linkedPairs) { self.t = types.V_OBJECT self.l = uint(v.Len()) self.p = unsafe.Pointer(v) } func newRawNode(str string, typ types.ValueType) Node { return Node{ t: _V_RAW | typ, p: rt.StrPtr(str), l: uint(len(str)), } } func (self *Node) parseRaw(full bool) { raw := self.toString() parser := NewParserObj(raw) if full { parser.noLazy = true parser.skipValue = false } var e types.ParsingError *self, e = parser.Parse() if e != 0 { *self = *newSyntaxError(parser.syntaxError(e)) } } var typeJumpTable = [256]types.ValueType{ '"' : types.V_STRING, '-' : _V_NUMBER, '0' : _V_NUMBER, '1' : _V_NUMBER, '2' : _V_NUMBER, '3' : _V_NUMBER, '4' : _V_NUMBER, '5' : _V_NUMBER, '6' : _V_NUMBER, '7' : _V_NUMBER, '8' : _V_NUMBER, '9' : _V_NUMBER, '[' : types.V_ARRAY, 'f' : types.V_FALSE, 'n' : types.V_NULL, 't' : types.V_TRUE, '{' : types.V_OBJECT, } func switchRawType(c byte) types.ValueType { return typeJumpTable[c] }