mirror of
https://github.com/mikefarah/yq.git
synced 2025-01-24 23:35:40 +00:00
1be3b31bbc
json.Encoder and json.Marshal implicitly use HTMLEscape to convert >, <, &, with \u003c, \u003e, \u0026. This behavior carries over to yq, where chars will be escaped when outputting json but not when outputting yaml. This changeset disables this behavior via encoder.SetEscapeHTML(false). Unfortunately there is no equivalent option for json.Marshal, so its single usage has been replaced with an encoder (with escaping disabled).
254 lines
5.3 KiB
Go
254 lines
5.3 KiB
Go
package yqlib
|
|
|
|
import (
|
|
"bytes"
|
|
"encoding/json"
|
|
"fmt"
|
|
"io"
|
|
|
|
yaml "gopkg.in/yaml.v3"
|
|
)
|
|
|
|
type Encoder interface {
|
|
Encode(node *yaml.Node) error
|
|
}
|
|
|
|
type yamlEncoder struct {
|
|
destination io.Writer
|
|
indent int
|
|
colorise bool
|
|
firstDoc bool
|
|
}
|
|
|
|
func NewYamlEncoder(destination io.Writer, indent int, colorise bool) Encoder {
|
|
if indent < 0 {
|
|
indent = 0
|
|
}
|
|
return &yamlEncoder{destination, indent, colorise, true}
|
|
}
|
|
|
|
func (ye *yamlEncoder) Encode(node *yaml.Node) error {
|
|
|
|
destination := ye.destination
|
|
tempBuffer := bytes.NewBuffer(nil)
|
|
if ye.colorise {
|
|
destination = tempBuffer
|
|
}
|
|
|
|
var encoder = yaml.NewEncoder(destination)
|
|
|
|
encoder.SetIndent(ye.indent)
|
|
// TODO: work out if the first doc had a separator or not.
|
|
if ye.firstDoc {
|
|
ye.firstDoc = false
|
|
} else if _, err := destination.Write([]byte("---\n")); err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := encoder.Encode(node); err != nil {
|
|
return err
|
|
}
|
|
|
|
if ye.colorise {
|
|
return colorizeAndPrint(tempBuffer.Bytes(), ye.destination)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
type jsonEncoder struct {
|
|
encoder *json.Encoder
|
|
}
|
|
|
|
func mapKeysToStrings(node *yaml.Node) {
|
|
|
|
if node.Kind == yaml.MappingNode {
|
|
for index, child := range node.Content {
|
|
if index%2 == 0 { // its a map key
|
|
child.Tag = "!!str"
|
|
}
|
|
}
|
|
}
|
|
|
|
for _, child := range node.Content {
|
|
mapKeysToStrings(child)
|
|
}
|
|
}
|
|
|
|
func NewJsonEncoder(destination io.Writer, indent int) Encoder {
|
|
var encoder = json.NewEncoder(destination)
|
|
encoder.SetEscapeHTML(false) // do not escape html chars e.g. &, <, >
|
|
|
|
var indentString = ""
|
|
|
|
for index := 0; index < indent; index++ {
|
|
indentString = indentString + " "
|
|
}
|
|
encoder.SetIndent("", indentString)
|
|
return &jsonEncoder{encoder}
|
|
}
|
|
|
|
func (je *jsonEncoder) Encode(node *yaml.Node) error {
|
|
var dataBucket orderedMap
|
|
// firstly, convert all map keys to strings
|
|
mapKeysToStrings(node)
|
|
errorDecoding := node.Decode(&dataBucket)
|
|
if errorDecoding != nil {
|
|
return errorDecoding
|
|
}
|
|
return je.encoder.Encode(dataBucket)
|
|
}
|
|
|
|
// orderedMap allows to marshal and unmarshal JSON and YAML values keeping the
|
|
// order of keys and values in a map or an object.
|
|
type orderedMap struct {
|
|
// if this is an object, kv != nil. If this is not an object, kv == nil.
|
|
kv []orderedMapKV
|
|
altVal interface{}
|
|
}
|
|
|
|
type orderedMapKV struct {
|
|
K string
|
|
V orderedMap
|
|
}
|
|
|
|
func (o *orderedMap) UnmarshalJSON(data []byte) error {
|
|
switch data[0] {
|
|
case '{':
|
|
// initialise so that even if the object is empty it is not nil
|
|
o.kv = []orderedMapKV{}
|
|
|
|
// create decoder
|
|
dec := json.NewDecoder(bytes.NewReader(data))
|
|
_, err := dec.Token() // open object
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// cycle through k/v
|
|
var tok json.Token
|
|
for tok, err = dec.Token(); err != io.EOF; tok, err = dec.Token() {
|
|
// we can expect two types: string or Delim. Delim automatically means
|
|
// that it is the closing bracket of the object, whereas string means
|
|
// that there is another key.
|
|
if _, ok := tok.(json.Delim); ok {
|
|
break
|
|
}
|
|
kv := orderedMapKV{
|
|
K: tok.(string),
|
|
}
|
|
if err := dec.Decode(&kv.V); err != nil {
|
|
return err
|
|
}
|
|
o.kv = append(o.kv, kv)
|
|
}
|
|
// unexpected error
|
|
if err != nil && err != io.EOF {
|
|
return err
|
|
}
|
|
return nil
|
|
case '[':
|
|
var arr []orderedMap
|
|
return json.Unmarshal(data, &arr)
|
|
}
|
|
|
|
return json.Unmarshal(data, &o.altVal)
|
|
}
|
|
|
|
func (o orderedMap) MarshalJSON() ([]byte, error) {
|
|
buf := new(bytes.Buffer)
|
|
enc := json.NewEncoder(buf)
|
|
enc.SetEscapeHTML(false) // do not escape html chars e.g. &, <, >
|
|
if o.kv == nil {
|
|
if err := enc.Encode(o.altVal); err != nil {
|
|
return nil, err
|
|
}
|
|
return buf.Bytes(), nil
|
|
}
|
|
buf.WriteByte('{')
|
|
for idx, el := range o.kv {
|
|
if err := enc.Encode(el.K); err != nil {
|
|
return nil, err
|
|
}
|
|
buf.WriteByte(':')
|
|
if err := enc.Encode(el.V); err != nil {
|
|
return nil, err
|
|
}
|
|
if idx != len(o.kv)-1 {
|
|
buf.WriteByte(',')
|
|
}
|
|
}
|
|
buf.WriteByte('}')
|
|
return buf.Bytes(), nil
|
|
}
|
|
|
|
func (o *orderedMap) UnmarshalYAML(node *yaml.Node) error {
|
|
switch node.Kind {
|
|
case yaml.DocumentNode:
|
|
if len(node.Content) == 0 {
|
|
return nil
|
|
}
|
|
return o.UnmarshalYAML(node.Content[0])
|
|
case yaml.AliasNode:
|
|
return o.UnmarshalYAML(node.Alias)
|
|
case yaml.ScalarNode:
|
|
return node.Decode(&o.altVal)
|
|
case yaml.MappingNode:
|
|
// set kv to non-nil
|
|
o.kv = []orderedMapKV{}
|
|
for i := 0; i < len(node.Content); i += 2 {
|
|
var key string
|
|
var val orderedMap
|
|
if err := node.Content[i].Decode(&key); err != nil {
|
|
return err
|
|
}
|
|
if err := node.Content[i+1].Decode(&val); err != nil {
|
|
return err
|
|
}
|
|
o.kv = append(o.kv, orderedMapKV{
|
|
K: key,
|
|
V: val,
|
|
})
|
|
}
|
|
return nil
|
|
case yaml.SequenceNode:
|
|
var res []orderedMap
|
|
if err := node.Decode(&res); err != nil {
|
|
return err
|
|
}
|
|
o.altVal = res
|
|
o.kv = nil
|
|
return nil
|
|
case 0:
|
|
// null
|
|
o.kv = nil
|
|
o.altVal = nil
|
|
return nil
|
|
default:
|
|
return fmt.Errorf("orderedMap: invalid yaml node")
|
|
}
|
|
}
|
|
|
|
func (o *orderedMap) MarshalYAML() (interface{}, error) {
|
|
// fast path: kv is nil, use altVal
|
|
if o.kv == nil {
|
|
return o.altVal, nil
|
|
}
|
|
content := make([]*yaml.Node, 0, len(o.kv)*2)
|
|
for _, val := range o.kv {
|
|
n := new(yaml.Node)
|
|
if err := n.Encode(val.V); err != nil {
|
|
return nil, err
|
|
}
|
|
content = append(content, &yaml.Node{
|
|
Kind: yaml.ScalarNode,
|
|
Tag: "!!str",
|
|
Value: val.K,
|
|
}, n)
|
|
}
|
|
return &yaml.Node{
|
|
Kind: yaml.MappingNode,
|
|
Tag: "!!map",
|
|
Content: content,
|
|
}, nil
|
|
}
|