open-vault/sdk/helper/jsonutil/json.go

104 lines
3 KiB
Go
Raw Normal View History

// Copyright (c) HashiCorp, Inc.
// SPDX-License-Identifier: MPL-2.0
2016-07-07 14:51:49 +00:00
package jsonutil
import (
"bytes"
"compress/gzip"
2016-07-07 14:51:49 +00:00
"encoding/json"
"fmt"
"io"
"github.com/hashicorp/errwrap"
"github.com/hashicorp/vault/sdk/helper/compressutil"
2016-08-04 21:20:37 +00:00
)
2016-07-07 15:29:38 +00:00
// Encodes/Marshals the given object into JSON
2016-07-07 14:51:49 +00:00
func EncodeJSON(in interface{}) ([]byte, error) {
2016-08-04 21:20:37 +00:00
if in == nil {
return nil, fmt.Errorf("input for encoding is nil")
}
2016-07-07 14:51:49 +00:00
var buf bytes.Buffer
enc := json.NewEncoder(&buf)
if err := enc.Encode(in); err != nil {
return nil, err
}
return buf.Bytes(), nil
}
2016-08-09 07:43:03 +00:00
// EncodeJSONAndCompress encodes the given input into JSON and compresses the
2016-08-09 15:01:59 +00:00
// encoded value (using Gzip format BestCompression level, by default). A
// canary byte is placed at the beginning of the returned bytes for the logic
// in decompression method to identify compressed input.
2016-08-09 14:33:41 +00:00
func EncodeJSONAndCompress(in interface{}, config *compressutil.CompressionConfig) ([]byte, error) {
2016-08-09 07:43:03 +00:00
if in == nil {
return nil, fmt.Errorf("input for encoding is nil")
}
// First JSON encode the given input
encodedBytes, err := EncodeJSON(in)
if err != nil {
return nil, err
}
2016-08-09 14:33:41 +00:00
if config == nil {
config = &compressutil.CompressionConfig{
Type: compressutil.CompressionTypeGzip,
GzipCompressionLevel: gzip.BestCompression,
}
}
return compressutil.Compress(encodedBytes, config)
2016-08-09 07:43:03 +00:00
}
2016-08-09 15:01:59 +00:00
// DecodeJSON tries to decompress the given data. The call to decompress, fails
// if the content was not compressed in the first place, which is identified by
// a canary byte before the compressed data. If the data is not compressed, it
// is JSON decoded directly. Otherwise the decompressed data will be JSON
// decoded.
2016-07-07 14:51:49 +00:00
func DecodeJSON(data []byte, out interface{}) error {
2016-08-09 14:33:41 +00:00
if data == nil || len(data) == 0 {
2016-07-07 15:29:38 +00:00
return fmt.Errorf("'data' being decoded is nil")
}
if out == nil {
return fmt.Errorf("output parameter 'out' is nil")
2016-07-07 14:51:49 +00:00
}
2016-08-09 15:01:59 +00:00
// Decompress the data if it was compressed in the first place
2016-08-09 14:33:41 +00:00
decompressedBytes, uncompressed, err := compressutil.Decompress(data)
if err != nil {
return errwrap.Wrapf("failed to decompress JSON: {{err}}", err)
2016-08-09 14:33:41 +00:00
}
if !uncompressed && (decompressedBytes == nil || len(decompressedBytes) == 0) {
return fmt.Errorf("decompressed data being decoded is invalid")
}
2016-08-09 15:01:59 +00:00
2016-08-09 14:33:41 +00:00
// If the input supplied failed to contain the compression canary, it
// will be notified by the compression utility. Decode the decompressed
// input.
if !uncompressed {
data = decompressedBytes
}
2016-07-07 14:51:49 +00:00
return DecodeJSONFromReader(bytes.NewReader(data), out)
}
2016-07-07 15:29:38 +00:00
// Decodes/Unmarshals the given io.Reader pointing to a JSON, into a desired object
2016-07-07 14:51:49 +00:00
func DecodeJSONFromReader(r io.Reader, out interface{}) error {
2016-07-07 15:29:38 +00:00
if r == nil {
return fmt.Errorf("'io.Reader' being decoded is nil")
}
if out == nil {
return fmt.Errorf("output parameter 'out' is nil")
}
2016-07-07 14:51:49 +00:00
dec := json.NewDecoder(r)
2018-03-20 18:54:10 +00:00
// While decoding JSON values, interpret the integer values as `json.Number`s instead of `float64`.
2016-07-07 14:51:49 +00:00
dec.UseNumber()
// Since 'out' is an interface representing a pointer, pass it to the decoder without an '&'
return dec.Decode(out)
}