mirror of
https://github.com/mikefarah/yq.git
synced 2025-01-12 11:05:37 +00:00
wip
This commit is contained in:
parent
851a43b9b6
commit
ae8df5ea87
@ -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>
|
||||||
|
```
|
||||||
|
|
||||||
|
@ -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())
|
||||||
}
|
}
|
||||||
|
@ -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
|
||||||
|
Loading…
Reference in New Issue
Block a user