package yqlib import ( "encoding/xml" "fmt" "io" "strings" yaml "gopkg.in/yaml.v3" ) type xmlEncoder struct { xmlEncoder *xml.Encoder attributePrefix string contentName string } func NewXmlEncoder(writer io.Writer, indent int, attributePrefix string, contentName string) Encoder { encoder := xml.NewEncoder(writer) var indentString = "" for index := 0; index < indent; index++ { indentString = indentString + " " } encoder.Indent("", indentString) return &xmlEncoder{encoder, attributePrefix, contentName} } func (e *xmlEncoder) Encode(node *yaml.Node) error { switch node.Kind { case yaml.MappingNode: err := e.encodeTopLevelMap(node) if err != nil { return err } case yaml.DocumentNode: err := e.encodeComment(headAndLineComment(node)) if err != nil { return err } err = e.Encode(unwrapDoc(node)) if err != nil { return err } err = e.encodeComment(footComment(node)) if err != nil { return err } case yaml.ScalarNode: var charData xml.CharData = []byte(node.Value) err := e.xmlEncoder.EncodeToken(charData) if err != nil { return err } return e.xmlEncoder.Flush() default: return fmt.Errorf("unsupported type %v", node.Tag) } var charData xml.CharData = []byte("\n") return e.xmlEncoder.EncodeToken(charData) } 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.encodeComment(headAndLineComment(key)) if err != nil { return err } err = e.doEncode(value, start) if err != nil { return err } err = e.encodeComment(footComment(key)) if err != nil { return err } } return nil } func (e *xmlEncoder) encodeStart(node *yaml.Node, start xml.StartElement) error { err := e.xmlEncoder.EncodeToken(start) if err != nil { return err } return e.encodeComment(headAndLineComment(node)) } func (e *xmlEncoder) encodeEnd(node *yaml.Node, start xml.StartElement) error { err := e.xmlEncoder.EncodeToken(start.End()) if err != nil { return err } return e.encodeComment(footComment(node)) } func (e *xmlEncoder) doEncode(node *yaml.Node, start xml.StartElement) error { switch node.Kind { case yaml.MappingNode: return e.encodeMap(node, start) case yaml.SequenceNode: return e.encodeArray(node, start) case yaml.ScalarNode: err := e.encodeStart(node, start) if err != nil { return err } var charData xml.CharData = []byte(node.Value) err = e.xmlEncoder.EncodeToken(charData) if err != nil { return err } return e.encodeEnd(node, start) } return fmt.Errorf("unsupported type %v", node.Tag) } func (e *xmlEncoder) encodeComment(commentStr string) error { if commentStr != "" { var comment xml.Comment = []byte(commentStr) err := e.xmlEncoder.EncodeToken(comment) if err != nil { return err } } return nil } func (e *xmlEncoder) encodeArray(node *yaml.Node, start xml.StartElement) error { for i := 0; i < len(node.Content); i++ { value := node.Content[i] err := e.doEncode(value, start.Copy()) if err != nil { return err } } return nil } 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 { key := node.Content[i] value := node.Content[i+1] if strings.HasPrefix(key.Value, e.attributePrefix) && key.Value != e.contentName { 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.encodeStart(node, 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] err := e.encodeComment(headAndLineComment(key)) if err != nil { return err } if !strings.HasPrefix(key.Value, e.attributePrefix) && key.Value != e.contentName { start := xml.StartElement{Name: xml.Name{Local: key.Value}} err := e.doEncode(value, start) if err != nil { return err } } else if key.Value == e.contentName { // directly encode the contents var charData xml.CharData = []byte(value.Value) err = e.xmlEncoder.EncodeToken(charData) if err != nil { return err } } err = e.encodeComment(footComment(key)) if err != nil { return err } } return e.encodeEnd(node, start) }