forked from subchen/go-xmldom
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathdom.go
131 lines (114 loc) · 2.36 KB
/
dom.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
// XML DOM processing for Golang, supports xpath query
package xmldom
import (
"bytes"
"encoding/xml"
"fmt"
"io"
"os"
"strings"
)
func Must(doc *Document, err error) *Document {
if err != nil {
panic(err)
}
return doc
}
func ParseObject(v interface{}) (*Document, error) {
data, err := xml.Marshal(v)
if err != nil {
return nil, err
}
return Parse(bytes.NewReader(data))
}
func ParseXML(s string) (*Document, error) {
return Parse(strings.NewReader(s))
}
func ParseFile(filename string) (*Document, error) {
file, err := os.Open(filename)
if err != nil {
return nil, err
}
defer file.Close()
return Parse(file)
}
func Parse(r io.Reader) (*Document, error) {
doc := NewDocument("empty")
err := doc.Parse(r)
if err != nil {
return nil, err
}
return doc, nil
}
func (doc *Document) ParseObject(v interface{}) error {
data, err := xml.Marshal(v)
if err != nil {
return fmt.Errorf("marshal object error: %v", err)
}
return doc.Parse(bytes.NewReader(data))
}
func (doc *Document) ParseXML(s string) error {
return doc.Parse(strings.NewReader(s))
}
func (doc *Document) Parse(r io.Reader) error {
p := xml.NewDecoder(r)
t, err := p.Token()
if err != nil {
return err
}
doc.Root = nil
var e *Node
for t != nil {
switch token := t.(type) {
case xml.StartElement:
// a new node
el := new(Node)
el.Document = doc
el.Parent = e
el.Name = token.Name
for _, attr := range token.Attr {
attribute := attr
el.Attributes = append(el.Attributes, &attribute)
}
if e != nil {
e.Children = append(e.Children, el)
}
e = el
if doc.Root == nil {
doc.Root = e
}
case xml.EndElement:
e = e.Parent
case xml.CharData:
// text node
if e != nil {
// a new node
el := new(Node)
el.Document = doc
el.Parent = e
if strings.TrimSpace(string(token)) != "" {
if doc.TextSafeMode {
el.Text = string(bytes.TrimSpace(token))
} else {
el.Text = string(token)
}
}
if el.Text != "" {
e.Children = append(e.Children, el)
}
}
case xml.ProcInst:
doc.ProcInst = stringifyProcInst(&token)
case xml.Directive:
doc.Directives = append(doc.Directives, stringifyDirective(&token))
}
// get the next token
t, err = p.Token()
}
// Make sure that reading stopped on EOF
if err != io.EOF {
return err
}
// All is good, return the document
return nil
}