This commit is contained in:
Mike Farah 2021-12-21 16:08:37 +11:00
parent 851a43b9b6
commit ae8df5ea87
3 changed files with 87 additions and 20 deletions

View File

@ -120,3 +120,24 @@ will output
</pets> </pets>
``` ```
## Encode xml: attributes
Fields with the matching xml-attribute-prefix are assumed to be attributes.
Given a sample.yml file of:
```yaml
cat:
+name: tiger
meows: true
```
then
```bash
yq e -o=xml '.' sample.yml
```
will output
```xml
<cat name="tiger">
<meows>true</meows>
</cat>
```

View File

@ -4,15 +4,17 @@ import (
"encoding/xml" "encoding/xml"
"fmt" "fmt"
"io" "io"
"strings"
yaml "gopkg.in/yaml.v3" yaml "gopkg.in/yaml.v3"
) )
type xmlEncoder struct { type xmlEncoder struct {
xmlEncoder *xml.Encoder xmlEncoder *xml.Encoder
attributePrefix string
} }
func NewXmlEncoder(writer io.Writer, indent int) Encoder { func NewXmlEncoder(writer io.Writer, indent int, attributePrefix string) Encoder {
encoder := xml.NewEncoder(writer) encoder := xml.NewEncoder(writer)
var indentString = "" var indentString = ""
@ -20,12 +22,12 @@ func NewXmlEncoder(writer io.Writer, indent int) Encoder {
indentString = indentString + " " indentString = indentString + " "
} }
encoder.Indent("", indentString) encoder.Indent("", indentString)
return &xmlEncoder{encoder} return &xmlEncoder{encoder, attributePrefix}
} }
func (e *xmlEncoder) Encode(node *yaml.Node) error { func (e *xmlEncoder) Encode(node *yaml.Node) error {
switch node.Kind { switch node.Kind {
case yaml.MappingNode: case yaml.MappingNode:
return e.encodeMap(node) return e.encodeTopLevelMap(node)
case yaml.DocumentNode: case yaml.DocumentNode:
return e.Encode(unwrapDoc(node)) return e.Encode(unwrapDoc(node))
case yaml.ScalarNode: case yaml.ScalarNode:
@ -35,18 +37,24 @@ func (e *xmlEncoder) Encode(node *yaml.Node) error {
return fmt.Errorf("unsupported type %v", node.Tag) return fmt.Errorf("unsupported type %v", node.Tag)
} }
func (e *xmlEncoder) encodeTopLevelMap(node *yaml.Node) error {
for i := 0; i < len(node.Content); i += 2 {
key := node.Content[i]
value := node.Content[i+1]
start := xml.StartElement{Name: xml.Name{Local: key.Value}}
err := e.doEncode(value, start)
if err != nil {
return err
}
}
return nil
}
func (e *xmlEncoder) doEncode(node *yaml.Node, start xml.StartElement) error { func (e *xmlEncoder) doEncode(node *yaml.Node, start xml.StartElement) error {
switch node.Kind { switch node.Kind {
case yaml.MappingNode: case yaml.MappingNode:
err := e.xmlEncoder.EncodeToken(start) return e.encodeMap(node, start)
if err != nil {
return err
}
err = e.encodeMap(node)
if err != nil {
return err
}
return e.xmlEncoder.EncodeToken(start.End())
case yaml.SequenceNode: case yaml.SequenceNode:
return e.encodeArray(node, start) return e.encodeArray(node, start)
case yaml.ScalarNode: case yaml.ScalarNode:
@ -77,16 +85,41 @@ func (e *xmlEncoder) encodeArray(node *yaml.Node, start xml.StartElement) error
return nil return nil
} }
func (e *xmlEncoder) encodeMap(node *yaml.Node) error { func (e *xmlEncoder) encodeMap(node *yaml.Node, start xml.StartElement) error {
//first find all the attributes and put them on the start token
for i := 0; i < len(node.Content); i += 2 { for i := 0; i < len(node.Content); i += 2 {
key := node.Content[i] key := node.Content[i]
value := node.Content[i+1] value := node.Content[i+1]
if strings.HasPrefix(key.Value, e.attributePrefix) {
if value.Kind == yaml.ScalarNode {
attributeName := strings.Replace(key.Value, e.attributePrefix, "", 1)
start.Attr = append(start.Attr, xml.Attr{Name: xml.Name{Local: attributeName}, Value: value.Value})
} else {
return fmt.Errorf("cannot use %v as attribute, only scalars are supported", value.Tag)
}
}
}
err := e.xmlEncoder.EncodeToken(start)
if err != nil {
return err
}
//now we encode non attribute tokens
for i := 0; i < len(node.Content); i += 2 {
key := node.Content[i]
value := node.Content[i+1]
if !strings.HasPrefix(key.Value, e.attributePrefix) {
start := xml.StartElement{Name: xml.Name{Local: key.Value}} start := xml.StartElement{Name: xml.Name{Local: key.Value}}
err := e.doEncode(value, start) err := e.doEncode(value, start)
if err != nil { if err != nil {
return err return err
} }
} }
return nil }
return e.xmlEncoder.EncodeToken(start.End())
} }

View File

@ -28,7 +28,7 @@ func yamlToXml(sampleYaml string, indent int) string {
var output bytes.Buffer var output bytes.Buffer
writer := bufio.NewWriter(&output) writer := bufio.NewWriter(&output)
var encoder = NewXmlEncoder(writer, indent) var encoder = NewXmlEncoder(writer, indent, "+")
inputs, err := readDocuments(strings.NewReader(sampleYaml), "sample.yml", 0, NewYamlDecoder()) inputs, err := readDocuments(strings.NewReader(sampleYaml), "sample.yml", 0, NewYamlDecoder())
if err != nil { if err != nil {
panic(err) panic(err)
@ -88,6 +88,19 @@ var xmlScenarios = []xmlScenario{
expected: "<pets>\n <cat>purrs</cat>\n <cat>meows</cat>\n</pets>", expected: "<pets>\n <cat>purrs</cat>\n <cat>meows</cat>\n</pets>",
encodeScenario: true, encodeScenario: true,
}, },
{
description: "Encode xml: attributes",
subdescription: "Fields with the matching xml-attribute-prefix are assumed to be attributes.",
input: "cat:\n +name: tiger\n meows: true\n",
expected: "<cat name=\"tiger\">\n <meows>true</meows>\n</cat>",
encodeScenario: true,
},
{
skipDoc: true,
input: "cat:\n ++name: tiger\n meows: true\n",
expected: "<cat +name=\"tiger\">\n <meows>true</meows>\n</cat>",
encodeScenario: true,
},
} }
//encode //encode