pulumi/sdk/go/common/encoding/marshal.go

178 lines
4.5 KiB
Go

// Copyright 2016-2022, Pulumi Corporation.
//
// 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 encoding
import (
"bytes"
"compress/gzip"
"encoding/json"
"fmt"
"io"
"path/filepath"
"github.com/pulumi/pulumi/sdk/v3/go/common/util/yamlutil"
yaml "gopkg.in/yaml.v3"
)
var (
JSONExt = ".json"
YAMLExt = ".yaml"
GZIPExt = ".gz"
)
// Exts contains a list of all the valid marshalable extension types.
var Exts = []string{
JSONExt,
YAMLExt,
// Although ".yml" is not a sanctioned YAML extension, it is used quite broadly; so we will support it.
".yml",
}
// Detect auto-detects a marshaler for the given path.
func Detect(path string) (Marshaler, string) {
ext := filepath.Ext(path)
if ext == "" {
ext = DefaultExt() // default to the first (preferred) marshaler.
}
return Marshalers[ext], ext
}
// Marshalers is a map of extension to a Marshaler object for that extension.
var Marshalers map[string]Marshaler
// Default returns the default marshaler object.
func Default() Marshaler {
return Marshalers[DefaultExt()]
}
// DefaultExt returns the default extension to use.
func DefaultExt() string {
return Exts[0]
}
// Marshaler is a type that knows how to marshal and unmarshal data in one format.
type Marshaler interface {
Marshal(v interface{}) ([]byte, error)
Unmarshal(data []byte, v interface{}) error
}
// JSON is a Marshaler that marshals and unmarshals JSON with indented printing.
var JSON Marshaler = &jsonMarshaler{}
type jsonMarshaler struct{}
func (m *jsonMarshaler) Marshal(v interface{}) ([]byte, error) {
var buf bytes.Buffer
enc := json.NewEncoder(&buf)
enc.SetEscapeHTML(false)
enc.SetIndent("", " ")
err := enc.Encode(v)
if err != nil {
return nil, err
}
return buf.Bytes(), nil
}
func (m *jsonMarshaler) Unmarshal(data []byte, v interface{}) error {
// IDEA: use a "strict" marshaler, so that we can warn on unrecognized keys (avoiding silly mistakes). We should
// set aside an officially sanctioned area in the metadata for extensibility by 3rd parties.
return json.Unmarshal(data, v)
}
var YAML Marshaler = &yamlMarshaler{}
type yamlMarshaler struct{}
func (m *yamlMarshaler) Marshal(v interface{}) ([]byte, error) {
if r, ok := v.(yamlutil.HasRawValue); ok {
if len(r.RawValue()) > 0 {
// Attempt a comment preserving edit:
return yamlutil.Edit(r.RawValue(), v)
}
}
return yamlutil.YamlEncode(v)
}
func (m *yamlMarshaler) Unmarshal(data []byte, v interface{}) error {
// IDEA: use a "strict" marshaler, so that we can warn on unrecognized keys (avoiding silly mistakes). We should
// set aside an officially sanctioned area in the metadata for extensibility by 3rd parties.
err := yaml.Unmarshal(data, v)
if err != nil {
// Return type errors directly
if _, ok := err.(*yaml.TypeError); ok {
return err
}
// Other errors will be parse errors due to invalid syntax
return fmt.Errorf("invalid YAML file: %w", err)
}
return nil
}
type gzipMarshaller struct {
inner Marshaler
}
func (m *gzipMarshaller) Marshal(v interface{}) ([]byte, error) {
b, err := m.inner.Marshal(v)
if err != nil {
return nil, err
}
var buf bytes.Buffer
writer := gzip.NewWriter(&buf)
defer writer.Close()
_, err = writer.Write(b)
if err != nil {
return nil, err
}
if err := writer.Close(); err != nil {
return nil, err
}
return buf.Bytes(), nil
}
func (m *gzipMarshaller) Unmarshal(data []byte, v interface{}) error {
buf := bytes.NewBuffer(data)
reader, err := gzip.NewReader(buf)
if err != nil {
return err
}
defer reader.Close()
inflated, err := io.ReadAll(reader)
if err != nil {
return err
}
if err := reader.Close(); err != nil {
return err
}
return m.inner.Unmarshal(inflated, v)
}
// IsCompressed returns if data is zip compressed.
func IsCompressed(buf []byte) bool {
// Taken from compress/gzip/gunzip.go
return len(buf) >= 3 && buf[0] == 31 && buf[1] == 139 && buf[2] == 8
}
func Gzip(m Marshaler) Marshaler {
_, alreadyGZIP := m.(*gzipMarshaller)
if alreadyGZIP {
return m
}
return &gzipMarshaller{m}
}