2021-01-17 21:46:25 +08:00
|
|
|
// Copyright GoFrame Author(https://goframe.org). All Rights Reserved.
|
2018-03-27 15:09:17 +08:00
|
|
|
//
|
|
|
|
|
// 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,
|
2019-02-02 16:18:25 +08:00
|
|
|
// You can obtain one at https://github.com/gogf/gf.
|
2018-03-27 15:09:17 +08:00
|
|
|
|
|
|
|
|
package gtype
|
|
|
|
|
|
2019-09-29 15:59:09 +08:00
|
|
|
import (
|
2019-09-29 20:12:59 +08:00
|
|
|
"bytes"
|
2019-09-29 15:59:09 +08:00
|
|
|
"encoding/base64"
|
|
|
|
|
"github.com/gogf/gf/util/gconv"
|
|
|
|
|
"sync/atomic"
|
|
|
|
|
)
|
2018-03-27 15:09:17 +08:00
|
|
|
|
2019-12-20 23:23:50 +08:00
|
|
|
// Bytes is a struct for concurrent-safe operation for type []byte.
|
2018-03-27 15:09:17 +08:00
|
|
|
type Bytes struct {
|
2019-04-15 22:55:12 +08:00
|
|
|
value atomic.Value
|
2018-03-27 15:09:17 +08:00
|
|
|
}
|
|
|
|
|
|
2019-12-20 23:23:50 +08:00
|
|
|
// NewBytes creates and returns a concurrent-safe object for []byte type,
|
2019-04-15 22:55:12 +08:00
|
|
|
// with given initial value <value>.
|
2019-06-19 09:06:52 +08:00
|
|
|
func NewBytes(value ...[]byte) *Bytes {
|
|
|
|
|
t := &Bytes{}
|
|
|
|
|
if len(value) > 0 {
|
|
|
|
|
t.value.Store(value[0])
|
|
|
|
|
}
|
|
|
|
|
return t
|
2018-03-27 15:09:17 +08:00
|
|
|
}
|
|
|
|
|
|
2019-04-15 22:55:12 +08:00
|
|
|
// Clone clones and returns a new concurrent-safe object for []byte type.
|
2019-06-18 08:37:21 +08:00
|
|
|
func (v *Bytes) Clone() *Bytes {
|
2019-06-19 09:06:52 +08:00
|
|
|
return NewBytes(v.Val())
|
2018-08-28 17:06:49 +08:00
|
|
|
}
|
|
|
|
|
|
2019-05-08 17:21:18 +08:00
|
|
|
// Set atomically stores <value> into t.value and returns the previous value of t.value.
|
2019-04-15 22:55:12 +08:00
|
|
|
// Note: The parameter <value> cannot be nil.
|
2019-06-18 08:37:21 +08:00
|
|
|
func (v *Bytes) Set(value []byte) (old []byte) {
|
2019-06-19 09:06:52 +08:00
|
|
|
old = v.Val()
|
|
|
|
|
v.value.Store(value)
|
|
|
|
|
return
|
2018-03-27 15:09:17 +08:00
|
|
|
}
|
|
|
|
|
|
2019-12-20 23:23:50 +08:00
|
|
|
// Val atomically loads and returns t.value.
|
2019-06-18 08:37:21 +08:00
|
|
|
func (v *Bytes) Val() []byte {
|
2019-06-19 09:06:52 +08:00
|
|
|
if s := v.value.Load(); s != nil {
|
|
|
|
|
return s.([]byte)
|
|
|
|
|
}
|
|
|
|
|
return nil
|
2018-04-15 22:02:06 +08:00
|
|
|
}
|
2019-09-29 15:59:09 +08:00
|
|
|
|
2019-10-01 16:03:18 +08:00
|
|
|
// String implements String interface for string printing.
|
|
|
|
|
func (v *Bytes) String() string {
|
|
|
|
|
return string(v.Val())
|
|
|
|
|
}
|
|
|
|
|
|
2019-09-29 15:59:09 +08:00
|
|
|
// MarshalJSON implements the interface MarshalJSON for json.Marshal.
|
|
|
|
|
func (v *Bytes) MarshalJSON() ([]byte, error) {
|
|
|
|
|
val := v.Val()
|
|
|
|
|
dst := make([]byte, base64.StdEncoding.EncodedLen(len(val)))
|
|
|
|
|
base64.StdEncoding.Encode(dst, val)
|
|
|
|
|
return gconv.UnsafeStrToBytes(`"` + gconv.UnsafeBytesToStr(dst) + `"`), nil
|
|
|
|
|
}
|
2019-09-29 20:12:59 +08:00
|
|
|
|
|
|
|
|
// UnmarshalJSON implements the interface UnmarshalJSON for json.Unmarshal.
|
|
|
|
|
func (v *Bytes) UnmarshalJSON(b []byte) error {
|
|
|
|
|
src := make([]byte, base64.StdEncoding.DecodedLen(len(b)))
|
|
|
|
|
n, err := base64.StdEncoding.Decode(src, bytes.Trim(b, `"`))
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
v.Set(src[:n])
|
|
|
|
|
return nil
|
|
|
|
|
}
|
2020-01-20 19:56:42 +08:00
|
|
|
|
|
|
|
|
// UnmarshalValue is an interface implement which sets any type of value for <v>.
|
|
|
|
|
func (v *Bytes) UnmarshalValue(value interface{}) error {
|
|
|
|
|
v.Set(gconv.Bytes(value))
|
|
|
|
|
return nil
|
|
|
|
|
}
|