mirror of
https://github.com/openziti/zrok.git
synced 2024-11-29 11:34:07 +01:00
1940 lines
48 KiB
Go
1940 lines
48 KiB
Go
|
// Copyright 2011 The Go Authors. All rights reserved.
|
||
|
// Use of this source code is governed by a BSD-style
|
||
|
// license that can be found in the LICENSE file.
|
||
|
|
||
|
package xml
|
||
|
|
||
|
import (
|
||
|
"bytes"
|
||
|
"errors"
|
||
|
"fmt"
|
||
|
"io"
|
||
|
"reflect"
|
||
|
"strconv"
|
||
|
"strings"
|
||
|
"sync"
|
||
|
"testing"
|
||
|
"time"
|
||
|
)
|
||
|
|
||
|
type DriveType int
|
||
|
|
||
|
const (
|
||
|
HyperDrive DriveType = iota
|
||
|
ImprobabilityDrive
|
||
|
)
|
||
|
|
||
|
type Passenger struct {
|
||
|
Name []string `xml:"name"`
|
||
|
Weight float32 `xml:"weight"`
|
||
|
}
|
||
|
|
||
|
type Ship struct {
|
||
|
XMLName struct{} `xml:"spaceship"`
|
||
|
|
||
|
Name string `xml:"name,attr"`
|
||
|
Pilot string `xml:"pilot,attr"`
|
||
|
Drive DriveType `xml:"drive"`
|
||
|
Age uint `xml:"age"`
|
||
|
Passenger []*Passenger `xml:"passenger"`
|
||
|
secret string
|
||
|
}
|
||
|
|
||
|
type NamedType string
|
||
|
|
||
|
type Port struct {
|
||
|
XMLName struct{} `xml:"port"`
|
||
|
Type string `xml:"type,attr,omitempty"`
|
||
|
Comment string `xml:",comment"`
|
||
|
Number string `xml:",chardata"`
|
||
|
}
|
||
|
|
||
|
type Domain struct {
|
||
|
XMLName struct{} `xml:"domain"`
|
||
|
Country string `xml:",attr,omitempty"`
|
||
|
Name []byte `xml:",chardata"`
|
||
|
Comment []byte `xml:",comment"`
|
||
|
}
|
||
|
|
||
|
type Book struct {
|
||
|
XMLName struct{} `xml:"book"`
|
||
|
Title string `xml:",chardata"`
|
||
|
}
|
||
|
|
||
|
type Event struct {
|
||
|
XMLName struct{} `xml:"event"`
|
||
|
Year int `xml:",chardata"`
|
||
|
}
|
||
|
|
||
|
type Movie struct {
|
||
|
XMLName struct{} `xml:"movie"`
|
||
|
Length uint `xml:",chardata"`
|
||
|
}
|
||
|
|
||
|
type Pi struct {
|
||
|
XMLName struct{} `xml:"pi"`
|
||
|
Approximation float32 `xml:",chardata"`
|
||
|
}
|
||
|
|
||
|
type Universe struct {
|
||
|
XMLName struct{} `xml:"universe"`
|
||
|
Visible float64 `xml:",chardata"`
|
||
|
}
|
||
|
|
||
|
type Particle struct {
|
||
|
XMLName struct{} `xml:"particle"`
|
||
|
HasMass bool `xml:",chardata"`
|
||
|
}
|
||
|
|
||
|
type Departure struct {
|
||
|
XMLName struct{} `xml:"departure"`
|
||
|
When time.Time `xml:",chardata"`
|
||
|
}
|
||
|
|
||
|
type SecretAgent struct {
|
||
|
XMLName struct{} `xml:"agent"`
|
||
|
Handle string `xml:"handle,attr"`
|
||
|
Identity string
|
||
|
Obfuscate string `xml:",innerxml"`
|
||
|
}
|
||
|
|
||
|
type NestedItems struct {
|
||
|
XMLName struct{} `xml:"result"`
|
||
|
Items []string `xml:">item"`
|
||
|
Item1 []string `xml:"Items>item1"`
|
||
|
}
|
||
|
|
||
|
type NestedOrder struct {
|
||
|
XMLName struct{} `xml:"result"`
|
||
|
Field1 string `xml:"parent>c"`
|
||
|
Field2 string `xml:"parent>b"`
|
||
|
Field3 string `xml:"parent>a"`
|
||
|
}
|
||
|
|
||
|
type MixedNested struct {
|
||
|
XMLName struct{} `xml:"result"`
|
||
|
A string `xml:"parent1>a"`
|
||
|
B string `xml:"b"`
|
||
|
C string `xml:"parent1>parent2>c"`
|
||
|
D string `xml:"parent1>d"`
|
||
|
}
|
||
|
|
||
|
type NilTest struct {
|
||
|
A interface{} `xml:"parent1>parent2>a"`
|
||
|
B interface{} `xml:"parent1>b"`
|
||
|
C interface{} `xml:"parent1>parent2>c"`
|
||
|
}
|
||
|
|
||
|
type Service struct {
|
||
|
XMLName struct{} `xml:"service"`
|
||
|
Domain *Domain `xml:"host>domain"`
|
||
|
Port *Port `xml:"host>port"`
|
||
|
Extra1 interface{}
|
||
|
Extra2 interface{} `xml:"host>extra2"`
|
||
|
}
|
||
|
|
||
|
var nilStruct *Ship
|
||
|
|
||
|
type EmbedA struct {
|
||
|
EmbedC
|
||
|
EmbedB EmbedB
|
||
|
FieldA string
|
||
|
}
|
||
|
|
||
|
type EmbedB struct {
|
||
|
FieldB string
|
||
|
*EmbedC
|
||
|
}
|
||
|
|
||
|
type EmbedC struct {
|
||
|
FieldA1 string `xml:"FieldA>A1"`
|
||
|
FieldA2 string `xml:"FieldA>A2"`
|
||
|
FieldB string
|
||
|
FieldC string
|
||
|
}
|
||
|
|
||
|
type NameCasing struct {
|
||
|
XMLName struct{} `xml:"casing"`
|
||
|
Xy string
|
||
|
XY string
|
||
|
XyA string `xml:"Xy,attr"`
|
||
|
XYA string `xml:"XY,attr"`
|
||
|
}
|
||
|
|
||
|
type NamePrecedence struct {
|
||
|
XMLName Name `xml:"Parent"`
|
||
|
FromTag XMLNameWithoutTag `xml:"InTag"`
|
||
|
FromNameVal XMLNameWithoutTag
|
||
|
FromNameTag XMLNameWithTag
|
||
|
InFieldName string
|
||
|
}
|
||
|
|
||
|
type XMLNameWithTag struct {
|
||
|
XMLName Name `xml:"InXMLNameTag"`
|
||
|
Value string `xml:",chardata"`
|
||
|
}
|
||
|
|
||
|
type XMLNameWithNSTag struct {
|
||
|
XMLName Name `xml:"ns InXMLNameWithNSTag"`
|
||
|
Value string `xml:",chardata"`
|
||
|
}
|
||
|
|
||
|
type XMLNameWithoutTag struct {
|
||
|
XMLName Name
|
||
|
Value string `xml:",chardata"`
|
||
|
}
|
||
|
|
||
|
type NameInField struct {
|
||
|
Foo Name `xml:"ns foo"`
|
||
|
}
|
||
|
|
||
|
type AttrTest struct {
|
||
|
Int int `xml:",attr"`
|
||
|
Named int `xml:"int,attr"`
|
||
|
Float float64 `xml:",attr"`
|
||
|
Uint8 uint8 `xml:",attr"`
|
||
|
Bool bool `xml:",attr"`
|
||
|
Str string `xml:",attr"`
|
||
|
Bytes []byte `xml:",attr"`
|
||
|
}
|
||
|
|
||
|
type OmitAttrTest struct {
|
||
|
Int int `xml:",attr,omitempty"`
|
||
|
Named int `xml:"int,attr,omitempty"`
|
||
|
Float float64 `xml:",attr,omitempty"`
|
||
|
Uint8 uint8 `xml:",attr,omitempty"`
|
||
|
Bool bool `xml:",attr,omitempty"`
|
||
|
Str string `xml:",attr,omitempty"`
|
||
|
Bytes []byte `xml:",attr,omitempty"`
|
||
|
}
|
||
|
|
||
|
type OmitFieldTest struct {
|
||
|
Int int `xml:",omitempty"`
|
||
|
Named int `xml:"int,omitempty"`
|
||
|
Float float64 `xml:",omitempty"`
|
||
|
Uint8 uint8 `xml:",omitempty"`
|
||
|
Bool bool `xml:",omitempty"`
|
||
|
Str string `xml:",omitempty"`
|
||
|
Bytes []byte `xml:",omitempty"`
|
||
|
Ptr *PresenceTest `xml:",omitempty"`
|
||
|
}
|
||
|
|
||
|
type AnyTest struct {
|
||
|
XMLName struct{} `xml:"a"`
|
||
|
Nested string `xml:"nested>value"`
|
||
|
AnyField AnyHolder `xml:",any"`
|
||
|
}
|
||
|
|
||
|
type AnyOmitTest struct {
|
||
|
XMLName struct{} `xml:"a"`
|
||
|
Nested string `xml:"nested>value"`
|
||
|
AnyField *AnyHolder `xml:",any,omitempty"`
|
||
|
}
|
||
|
|
||
|
type AnySliceTest struct {
|
||
|
XMLName struct{} `xml:"a"`
|
||
|
Nested string `xml:"nested>value"`
|
||
|
AnyField []AnyHolder `xml:",any"`
|
||
|
}
|
||
|
|
||
|
type AnyHolder struct {
|
||
|
XMLName Name
|
||
|
XML string `xml:",innerxml"`
|
||
|
}
|
||
|
|
||
|
type RecurseA struct {
|
||
|
A string
|
||
|
B *RecurseB
|
||
|
}
|
||
|
|
||
|
type RecurseB struct {
|
||
|
A *RecurseA
|
||
|
B string
|
||
|
}
|
||
|
|
||
|
type PresenceTest struct {
|
||
|
Exists *struct{}
|
||
|
}
|
||
|
|
||
|
type IgnoreTest struct {
|
||
|
PublicSecret string `xml:"-"`
|
||
|
}
|
||
|
|
||
|
type MyBytes []byte
|
||
|
|
||
|
type Data struct {
|
||
|
Bytes []byte
|
||
|
Attr []byte `xml:",attr"`
|
||
|
Custom MyBytes
|
||
|
}
|
||
|
|
||
|
type Plain struct {
|
||
|
V interface{}
|
||
|
}
|
||
|
|
||
|
type MyInt int
|
||
|
|
||
|
type EmbedInt struct {
|
||
|
MyInt
|
||
|
}
|
||
|
|
||
|
type Strings struct {
|
||
|
X []string `xml:"A>B,omitempty"`
|
||
|
}
|
||
|
|
||
|
type PointerFieldsTest struct {
|
||
|
XMLName Name `xml:"dummy"`
|
||
|
Name *string `xml:"name,attr"`
|
||
|
Age *uint `xml:"age,attr"`
|
||
|
Empty *string `xml:"empty,attr"`
|
||
|
Contents *string `xml:",chardata"`
|
||
|
}
|
||
|
|
||
|
type ChardataEmptyTest struct {
|
||
|
XMLName Name `xml:"test"`
|
||
|
Contents *string `xml:",chardata"`
|
||
|
}
|
||
|
|
||
|
type MyMarshalerTest struct {
|
||
|
}
|
||
|
|
||
|
var _ Marshaler = (*MyMarshalerTest)(nil)
|
||
|
|
||
|
func (m *MyMarshalerTest) MarshalXML(e *Encoder, start StartElement) error {
|
||
|
e.EncodeToken(start)
|
||
|
e.EncodeToken(CharData([]byte("hello world")))
|
||
|
e.EncodeToken(EndElement{start.Name})
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
type MyMarshalerAttrTest struct{}
|
||
|
|
||
|
var _ MarshalerAttr = (*MyMarshalerAttrTest)(nil)
|
||
|
|
||
|
func (m *MyMarshalerAttrTest) MarshalXMLAttr(name Name) (Attr, error) {
|
||
|
return Attr{name, "hello world"}, nil
|
||
|
}
|
||
|
|
||
|
type MyMarshalerValueAttrTest struct{}
|
||
|
|
||
|
var _ MarshalerAttr = MyMarshalerValueAttrTest{}
|
||
|
|
||
|
func (m MyMarshalerValueAttrTest) MarshalXMLAttr(name Name) (Attr, error) {
|
||
|
return Attr{name, "hello world"}, nil
|
||
|
}
|
||
|
|
||
|
type MarshalerStruct struct {
|
||
|
Foo MyMarshalerAttrTest `xml:",attr"`
|
||
|
}
|
||
|
|
||
|
type MarshalerValueStruct struct {
|
||
|
Foo MyMarshalerValueAttrTest `xml:",attr"`
|
||
|
}
|
||
|
|
||
|
type InnerStruct struct {
|
||
|
XMLName Name `xml:"testns outer"`
|
||
|
}
|
||
|
|
||
|
type OuterStruct struct {
|
||
|
InnerStruct
|
||
|
IntAttr int `xml:"int,attr"`
|
||
|
}
|
||
|
|
||
|
type OuterNamedStruct struct {
|
||
|
InnerStruct
|
||
|
XMLName Name `xml:"outerns test"`
|
||
|
IntAttr int `xml:"int,attr"`
|
||
|
}
|
||
|
|
||
|
type OuterNamedOrderedStruct struct {
|
||
|
XMLName Name `xml:"outerns test"`
|
||
|
InnerStruct
|
||
|
IntAttr int `xml:"int,attr"`
|
||
|
}
|
||
|
|
||
|
type OuterOuterStruct struct {
|
||
|
OuterStruct
|
||
|
}
|
||
|
|
||
|
type NestedAndChardata struct {
|
||
|
AB []string `xml:"A>B"`
|
||
|
Chardata string `xml:",chardata"`
|
||
|
}
|
||
|
|
||
|
type NestedAndComment struct {
|
||
|
AB []string `xml:"A>B"`
|
||
|
Comment string `xml:",comment"`
|
||
|
}
|
||
|
|
||
|
type XMLNSFieldStruct struct {
|
||
|
Ns string `xml:"xmlns,attr"`
|
||
|
Body string
|
||
|
}
|
||
|
|
||
|
type NamedXMLNSFieldStruct struct {
|
||
|
XMLName struct{} `xml:"testns test"`
|
||
|
Ns string `xml:"xmlns,attr"`
|
||
|
Body string
|
||
|
}
|
||
|
|
||
|
type XMLNSFieldStructWithOmitEmpty struct {
|
||
|
Ns string `xml:"xmlns,attr,omitempty"`
|
||
|
Body string
|
||
|
}
|
||
|
|
||
|
type NamedXMLNSFieldStructWithEmptyNamespace struct {
|
||
|
XMLName struct{} `xml:"test"`
|
||
|
Ns string `xml:"xmlns,attr"`
|
||
|
Body string
|
||
|
}
|
||
|
|
||
|
type RecursiveXMLNSFieldStruct struct {
|
||
|
Ns string `xml:"xmlns,attr"`
|
||
|
Body *RecursiveXMLNSFieldStruct `xml:",omitempty"`
|
||
|
Text string `xml:",omitempty"`
|
||
|
}
|
||
|
|
||
|
func ifaceptr(x interface{}) interface{} {
|
||
|
return &x
|
||
|
}
|
||
|
|
||
|
var (
|
||
|
nameAttr = "Sarah"
|
||
|
ageAttr = uint(12)
|
||
|
contentsAttr = "lorem ipsum"
|
||
|
)
|
||
|
|
||
|
// Unless explicitly stated as such (or *Plain), all of the
|
||
|
// tests below are two-way tests. When introducing new tests,
|
||
|
// please try to make them two-way as well to ensure that
|
||
|
// marshalling and unmarshalling are as symmetrical as feasible.
|
||
|
var marshalTests = []struct {
|
||
|
Value interface{}
|
||
|
ExpectXML string
|
||
|
MarshalOnly bool
|
||
|
UnmarshalOnly bool
|
||
|
}{
|
||
|
// Test nil marshals to nothing
|
||
|
{Value: nil, ExpectXML: ``, MarshalOnly: true},
|
||
|
{Value: nilStruct, ExpectXML: ``, MarshalOnly: true},
|
||
|
|
||
|
// Test value types
|
||
|
{Value: &Plain{true}, ExpectXML: `<Plain><V>true</V></Plain>`},
|
||
|
{Value: &Plain{false}, ExpectXML: `<Plain><V>false</V></Plain>`},
|
||
|
{Value: &Plain{int(42)}, ExpectXML: `<Plain><V>42</V></Plain>`},
|
||
|
{Value: &Plain{int8(42)}, ExpectXML: `<Plain><V>42</V></Plain>`},
|
||
|
{Value: &Plain{int16(42)}, ExpectXML: `<Plain><V>42</V></Plain>`},
|
||
|
{Value: &Plain{int32(42)}, ExpectXML: `<Plain><V>42</V></Plain>`},
|
||
|
{Value: &Plain{uint(42)}, ExpectXML: `<Plain><V>42</V></Plain>`},
|
||
|
{Value: &Plain{uint8(42)}, ExpectXML: `<Plain><V>42</V></Plain>`},
|
||
|
{Value: &Plain{uint16(42)}, ExpectXML: `<Plain><V>42</V></Plain>`},
|
||
|
{Value: &Plain{uint32(42)}, ExpectXML: `<Plain><V>42</V></Plain>`},
|
||
|
{Value: &Plain{float32(1.25)}, ExpectXML: `<Plain><V>1.25</V></Plain>`},
|
||
|
{Value: &Plain{float64(1.25)}, ExpectXML: `<Plain><V>1.25</V></Plain>`},
|
||
|
{Value: &Plain{uintptr(0xFFDD)}, ExpectXML: `<Plain><V>65501</V></Plain>`},
|
||
|
{Value: &Plain{"gopher"}, ExpectXML: `<Plain><V>gopher</V></Plain>`},
|
||
|
{Value: &Plain{[]byte("gopher")}, ExpectXML: `<Plain><V>gopher</V></Plain>`},
|
||
|
{Value: &Plain{"</>"}, ExpectXML: `<Plain><V></></V></Plain>`},
|
||
|
{Value: &Plain{[]byte("</>")}, ExpectXML: `<Plain><V></></V></Plain>`},
|
||
|
{Value: &Plain{[3]byte{'<', '/', '>'}}, ExpectXML: `<Plain><V></></V></Plain>`},
|
||
|
{Value: &Plain{NamedType("potato")}, ExpectXML: `<Plain><V>potato</V></Plain>`},
|
||
|
{Value: &Plain{[]int{1, 2, 3}}, ExpectXML: `<Plain><V>1</V><V>2</V><V>3</V></Plain>`},
|
||
|
{Value: &Plain{[3]int{1, 2, 3}}, ExpectXML: `<Plain><V>1</V><V>2</V><V>3</V></Plain>`},
|
||
|
{Value: ifaceptr(true), MarshalOnly: true, ExpectXML: `<bool>true</bool>`},
|
||
|
|
||
|
// Test time.
|
||
|
{
|
||
|
Value: &Plain{time.Unix(1e9, 123456789).UTC()},
|
||
|
ExpectXML: `<Plain><V>2001-09-09T01:46:40.123456789Z</V></Plain>`,
|
||
|
},
|
||
|
|
||
|
// A pointer to struct{} may be used to test for an element's presence.
|
||
|
{
|
||
|
Value: &PresenceTest{new(struct{})},
|
||
|
ExpectXML: `<PresenceTest><Exists></Exists></PresenceTest>`,
|
||
|
},
|
||
|
{
|
||
|
Value: &PresenceTest{},
|
||
|
ExpectXML: `<PresenceTest></PresenceTest>`,
|
||
|
},
|
||
|
|
||
|
// A pointer to struct{} may be used to test for an element's presence.
|
||
|
{
|
||
|
Value: &PresenceTest{new(struct{})},
|
||
|
ExpectXML: `<PresenceTest><Exists></Exists></PresenceTest>`,
|
||
|
},
|
||
|
{
|
||
|
Value: &PresenceTest{},
|
||
|
ExpectXML: `<PresenceTest></PresenceTest>`,
|
||
|
},
|
||
|
|
||
|
// A []byte field is only nil if the element was not found.
|
||
|
{
|
||
|
Value: &Data{},
|
||
|
ExpectXML: `<Data></Data>`,
|
||
|
UnmarshalOnly: true,
|
||
|
},
|
||
|
{
|
||
|
Value: &Data{Bytes: []byte{}, Custom: MyBytes{}, Attr: []byte{}},
|
||
|
ExpectXML: `<Data Attr=""><Bytes></Bytes><Custom></Custom></Data>`,
|
||
|
UnmarshalOnly: true,
|
||
|
},
|
||
|
|
||
|
// Check that []byte works, including named []byte types.
|
||
|
{
|
||
|
Value: &Data{Bytes: []byte("ab"), Custom: MyBytes("cd"), Attr: []byte{'v'}},
|
||
|
ExpectXML: `<Data Attr="v"><Bytes>ab</Bytes><Custom>cd</Custom></Data>`,
|
||
|
},
|
||
|
|
||
|
// Test innerxml
|
||
|
{
|
||
|
Value: &SecretAgent{
|
||
|
Handle: "007",
|
||
|
Identity: "James Bond",
|
||
|
Obfuscate: "<redacted/>",
|
||
|
},
|
||
|
ExpectXML: `<agent handle="007"><Identity>James Bond</Identity><redacted/></agent>`,
|
||
|
MarshalOnly: true,
|
||
|
},
|
||
|
{
|
||
|
Value: &SecretAgent{
|
||
|
Handle: "007",
|
||
|
Identity: "James Bond",
|
||
|
Obfuscate: "<Identity>James Bond</Identity><redacted/>",
|
||
|
},
|
||
|
ExpectXML: `<agent handle="007"><Identity>James Bond</Identity><redacted/></agent>`,
|
||
|
UnmarshalOnly: true,
|
||
|
},
|
||
|
|
||
|
// Test structs
|
||
|
{Value: &Port{Type: "ssl", Number: "443"}, ExpectXML: `<port type="ssl">443</port>`},
|
||
|
{Value: &Port{Number: "443"}, ExpectXML: `<port>443</port>`},
|
||
|
{Value: &Port{Type: "<unix>"}, ExpectXML: `<port type="<unix>"></port>`},
|
||
|
{Value: &Port{Number: "443", Comment: "https"}, ExpectXML: `<port><!--https-->443</port>`},
|
||
|
{Value: &Port{Number: "443", Comment: "add space-"}, ExpectXML: `<port><!--add space- -->443</port>`, MarshalOnly: true},
|
||
|
{Value: &Domain{Name: []byte("google.com&friends")}, ExpectXML: `<domain>google.com&friends</domain>`},
|
||
|
{Value: &Domain{Name: []byte("google.com"), Comment: []byte(" &friends ")}, ExpectXML: `<domain>google.com<!-- &friends --></domain>`},
|
||
|
{Value: &Book{Title: "Pride & Prejudice"}, ExpectXML: `<book>Pride & Prejudice</book>`},
|
||
|
{Value: &Event{Year: -3114}, ExpectXML: `<event>-3114</event>`},
|
||
|
{Value: &Movie{Length: 13440}, ExpectXML: `<movie>13440</movie>`},
|
||
|
{Value: &Pi{Approximation: 3.14159265}, ExpectXML: `<pi>3.1415927</pi>`},
|
||
|
{Value: &Universe{Visible: 9.3e13}, ExpectXML: `<universe>9.3e+13</universe>`},
|
||
|
{Value: &Particle{HasMass: true}, ExpectXML: `<particle>true</particle>`},
|
||
|
{Value: &Departure{When: ParseTime("2013-01-09T00:15:00-09:00")}, ExpectXML: `<departure>2013-01-09T00:15:00-09:00</departure>`},
|
||
|
{Value: atomValue, ExpectXML: atomXml},
|
||
|
{
|
||
|
Value: &Ship{
|
||
|
Name: "Heart of Gold",
|
||
|
Pilot: "Computer",
|
||
|
Age: 1,
|
||
|
Drive: ImprobabilityDrive,
|
||
|
Passenger: []*Passenger{
|
||
|
{
|
||
|
Name: []string{"Zaphod", "Beeblebrox"},
|
||
|
Weight: 7.25,
|
||
|
},
|
||
|
{
|
||
|
Name: []string{"Trisha", "McMillen"},
|
||
|
Weight: 5.5,
|
||
|
},
|
||
|
{
|
||
|
Name: []string{"Ford", "Prefect"},
|
||
|
Weight: 7,
|
||
|
},
|
||
|
{
|
||
|
Name: []string{"Arthur", "Dent"},
|
||
|
Weight: 6.75,
|
||
|
},
|
||
|
},
|
||
|
},
|
||
|
ExpectXML: `<spaceship name="Heart of Gold" pilot="Computer">` +
|
||
|
`<drive>` + strconv.Itoa(int(ImprobabilityDrive)) + `</drive>` +
|
||
|
`<age>1</age>` +
|
||
|
`<passenger>` +
|
||
|
`<name>Zaphod</name>` +
|
||
|
`<name>Beeblebrox</name>` +
|
||
|
`<weight>7.25</weight>` +
|
||
|
`</passenger>` +
|
||
|
`<passenger>` +
|
||
|
`<name>Trisha</name>` +
|
||
|
`<name>McMillen</name>` +
|
||
|
`<weight>5.5</weight>` +
|
||
|
`</passenger>` +
|
||
|
`<passenger>` +
|
||
|
`<name>Ford</name>` +
|
||
|
`<name>Prefect</name>` +
|
||
|
`<weight>7</weight>` +
|
||
|
`</passenger>` +
|
||
|
`<passenger>` +
|
||
|
`<name>Arthur</name>` +
|
||
|
`<name>Dent</name>` +
|
||
|
`<weight>6.75</weight>` +
|
||
|
`</passenger>` +
|
||
|
`</spaceship>`,
|
||
|
},
|
||
|
|
||
|
// Test a>b
|
||
|
{
|
||
|
Value: &NestedItems{Items: nil, Item1: nil},
|
||
|
ExpectXML: `<result>` +
|
||
|
`<Items>` +
|
||
|
`</Items>` +
|
||
|
`</result>`,
|
||
|
},
|
||
|
{
|
||
|
Value: &NestedItems{Items: []string{}, Item1: []string{}},
|
||
|
ExpectXML: `<result>` +
|
||
|
`<Items>` +
|
||
|
`</Items>` +
|
||
|
`</result>`,
|
||
|
MarshalOnly: true,
|
||
|
},
|
||
|
{
|
||
|
Value: &NestedItems{Items: nil, Item1: []string{"A"}},
|
||
|
ExpectXML: `<result>` +
|
||
|
`<Items>` +
|
||
|
`<item1>A</item1>` +
|
||
|
`</Items>` +
|
||
|
`</result>`,
|
||
|
},
|
||
|
{
|
||
|
Value: &NestedItems{Items: []string{"A", "B"}, Item1: nil},
|
||
|
ExpectXML: `<result>` +
|
||
|
`<Items>` +
|
||
|
`<item>A</item>` +
|
||
|
`<item>B</item>` +
|
||
|
`</Items>` +
|
||
|
`</result>`,
|
||
|
},
|
||
|
{
|
||
|
Value: &NestedItems{Items: []string{"A", "B"}, Item1: []string{"C"}},
|
||
|
ExpectXML: `<result>` +
|
||
|
`<Items>` +
|
||
|
`<item>A</item>` +
|
||
|
`<item>B</item>` +
|
||
|
`<item1>C</item1>` +
|
||
|
`</Items>` +
|
||
|
`</result>`,
|
||
|
},
|
||
|
{
|
||
|
Value: &NestedOrder{Field1: "C", Field2: "B", Field3: "A"},
|
||
|
ExpectXML: `<result>` +
|
||
|
`<parent>` +
|
||
|
`<c>C</c>` +
|
||
|
`<b>B</b>` +
|
||
|
`<a>A</a>` +
|
||
|
`</parent>` +
|
||
|
`</result>`,
|
||
|
},
|
||
|
{
|
||
|
Value: &NilTest{A: "A", B: nil, C: "C"},
|
||
|
ExpectXML: `<NilTest>` +
|
||
|
`<parent1>` +
|
||
|
`<parent2><a>A</a></parent2>` +
|
||
|
`<parent2><c>C</c></parent2>` +
|
||
|
`</parent1>` +
|
||
|
`</NilTest>`,
|
||
|
MarshalOnly: true, // Uses interface{}
|
||
|
},
|
||
|
{
|
||
|
Value: &MixedNested{A: "A", B: "B", C: "C", D: "D"},
|
||
|
ExpectXML: `<result>` +
|
||
|
`<parent1><a>A</a></parent1>` +
|
||
|
`<b>B</b>` +
|
||
|
`<parent1>` +
|
||
|
`<parent2><c>C</c></parent2>` +
|
||
|
`<d>D</d>` +
|
||
|
`</parent1>` +
|
||
|
`</result>`,
|
||
|
},
|
||
|
{
|
||
|
Value: &Service{Port: &Port{Number: "80"}},
|
||
|
ExpectXML: `<service><host><port>80</port></host></service>`,
|
||
|
},
|
||
|
{
|
||
|
Value: &Service{},
|
||
|
ExpectXML: `<service></service>`,
|
||
|
},
|
||
|
{
|
||
|
Value: &Service{Port: &Port{Number: "80"}, Extra1: "A", Extra2: "B"},
|
||
|
ExpectXML: `<service>` +
|
||
|
`<host><port>80</port></host>` +
|
||
|
`<Extra1>A</Extra1>` +
|
||
|
`<host><extra2>B</extra2></host>` +
|
||
|
`</service>`,
|
||
|
MarshalOnly: true,
|
||
|
},
|
||
|
{
|
||
|
Value: &Service{Port: &Port{Number: "80"}, Extra2: "example"},
|
||
|
ExpectXML: `<service>` +
|
||
|
`<host><port>80</port></host>` +
|
||
|
`<host><extra2>example</extra2></host>` +
|
||
|
`</service>`,
|
||
|
MarshalOnly: true,
|
||
|
},
|
||
|
{
|
||
|
Value: &struct {
|
||
|
XMLName struct{} `xml:"space top"`
|
||
|
A string `xml:"x>a"`
|
||
|
B string `xml:"x>b"`
|
||
|
C string `xml:"space x>c"`
|
||
|
C1 string `xml:"space1 x>c"`
|
||
|
D1 string `xml:"space1 x>d"`
|
||
|
E1 string `xml:"x>e"`
|
||
|
}{
|
||
|
A: "a",
|
||
|
B: "b",
|
||
|
C: "c",
|
||
|
C1: "c1",
|
||
|
D1: "d1",
|
||
|
E1: "e1",
|
||
|
},
|
||
|
ExpectXML: `<top xmlns="space">` +
|
||
|
`<x><a>a</a><b>b</b><c>c</c></x>` +
|
||
|
`<x xmlns="space1">` +
|
||
|
`<c>c1</c>` +
|
||
|
`<d>d1</d>` +
|
||
|
`</x>` +
|
||
|
`<x>` +
|
||
|
`<e>e1</e>` +
|
||
|
`</x>` +
|
||
|
`</top>`,
|
||
|
},
|
||
|
{
|
||
|
Value: &struct {
|
||
|
XMLName Name
|
||
|
A string `xml:"x>a"`
|
||
|
B string `xml:"x>b"`
|
||
|
C string `xml:"space x>c"`
|
||
|
C1 string `xml:"space1 x>c"`
|
||
|
D1 string `xml:"space1 x>d"`
|
||
|
}{
|
||
|
XMLName: Name{
|
||
|
Space: "space0",
|
||
|
Local: "top",
|
||
|
},
|
||
|
A: "a",
|
||
|
B: "b",
|
||
|
C: "c",
|
||
|
C1: "c1",
|
||
|
D1: "d1",
|
||
|
},
|
||
|
ExpectXML: `<top xmlns="space0">` +
|
||
|
`<x><a>a</a><b>b</b></x>` +
|
||
|
`<x xmlns="space"><c>c</c></x>` +
|
||
|
`<x xmlns="space1">` +
|
||
|
`<c>c1</c>` +
|
||
|
`<d>d1</d>` +
|
||
|
`</x>` +
|
||
|
`</top>`,
|
||
|
},
|
||
|
{
|
||
|
Value: &struct {
|
||
|
XMLName struct{} `xml:"top"`
|
||
|
B string `xml:"space x>b"`
|
||
|
B1 string `xml:"space1 x>b"`
|
||
|
}{
|
||
|
B: "b",
|
||
|
B1: "b1",
|
||
|
},
|
||
|
ExpectXML: `<top>` +
|
||
|
`<x xmlns="space"><b>b</b></x>` +
|
||
|
`<x xmlns="space1"><b>b1</b></x>` +
|
||
|
`</top>`,
|
||
|
},
|
||
|
|
||
|
// Test struct embedding
|
||
|
{
|
||
|
Value: &EmbedA{
|
||
|
EmbedC: EmbedC{
|
||
|
FieldA1: "", // Shadowed by A.A
|
||
|
FieldA2: "", // Shadowed by A.A
|
||
|
FieldB: "A.C.B",
|
||
|
FieldC: "A.C.C",
|
||
|
},
|
||
|
EmbedB: EmbedB{
|
||
|
FieldB: "A.B.B",
|
||
|
EmbedC: &EmbedC{
|
||
|
FieldA1: "A.B.C.A1",
|
||
|
FieldA2: "A.B.C.A2",
|
||
|
FieldB: "", // Shadowed by A.B.B
|
||
|
FieldC: "A.B.C.C",
|
||
|
},
|
||
|
},
|
||
|
FieldA: "A.A",
|
||
|
},
|
||
|
ExpectXML: `<EmbedA>` +
|
||
|
`<FieldB>A.C.B</FieldB>` +
|
||
|
`<FieldC>A.C.C</FieldC>` +
|
||
|
`<EmbedB>` +
|
||
|
`<FieldB>A.B.B</FieldB>` +
|
||
|
`<FieldA>` +
|
||
|
`<A1>A.B.C.A1</A1>` +
|
||
|
`<A2>A.B.C.A2</A2>` +
|
||
|
`</FieldA>` +
|
||
|
`<FieldC>A.B.C.C</FieldC>` +
|
||
|
`</EmbedB>` +
|
||
|
`<FieldA>A.A</FieldA>` +
|
||
|
`</EmbedA>`,
|
||
|
},
|
||
|
|
||
|
// Test that name casing matters
|
||
|
{
|
||
|
Value: &NameCasing{Xy: "mixed", XY: "upper", XyA: "mixedA", XYA: "upperA"},
|
||
|
ExpectXML: `<casing Xy="mixedA" XY="upperA"><Xy>mixed</Xy><XY>upper</XY></casing>`,
|
||
|
},
|
||
|
|
||
|
// Test the order in which the XML element name is chosen
|
||
|
{
|
||
|
Value: &NamePrecedence{
|
||
|
FromTag: XMLNameWithoutTag{Value: "A"},
|
||
|
FromNameVal: XMLNameWithoutTag{XMLName: Name{Local: "InXMLName"}, Value: "B"},
|
||
|
FromNameTag: XMLNameWithTag{Value: "C"},
|
||
|
InFieldName: "D",
|
||
|
},
|
||
|
ExpectXML: `<Parent>` +
|
||
|
`<InTag>A</InTag>` +
|
||
|
`<InXMLName>B</InXMLName>` +
|
||
|
`<InXMLNameTag>C</InXMLNameTag>` +
|
||
|
`<InFieldName>D</InFieldName>` +
|
||
|
`</Parent>`,
|
||
|
MarshalOnly: true,
|
||
|
},
|
||
|
{
|
||
|
Value: &NamePrecedence{
|
||
|
XMLName: Name{Local: "Parent"},
|
||
|
FromTag: XMLNameWithoutTag{XMLName: Name{Local: "InTag"}, Value: "A"},
|
||
|
FromNameVal: XMLNameWithoutTag{XMLName: Name{Local: "FromNameVal"}, Value: "B"},
|
||
|
FromNameTag: XMLNameWithTag{XMLName: Name{Local: "InXMLNameTag"}, Value: "C"},
|
||
|
InFieldName: "D",
|
||
|
},
|
||
|
ExpectXML: `<Parent>` +
|
||
|
`<InTag>A</InTag>` +
|
||
|
`<FromNameVal>B</FromNameVal>` +
|
||
|
`<InXMLNameTag>C</InXMLNameTag>` +
|
||
|
`<InFieldName>D</InFieldName>` +
|
||
|
`</Parent>`,
|
||
|
UnmarshalOnly: true,
|
||
|
},
|
||
|
|
||
|
// xml.Name works in a plain field as well.
|
||
|
{
|
||
|
Value: &NameInField{Name{Space: "ns", Local: "foo"}},
|
||
|
ExpectXML: `<NameInField><foo xmlns="ns"></foo></NameInField>`,
|
||
|
},
|
||
|
{
|
||
|
Value: &NameInField{Name{Space: "ns", Local: "foo"}},
|
||
|
ExpectXML: `<NameInField><foo xmlns="ns"><ignore></ignore></foo></NameInField>`,
|
||
|
UnmarshalOnly: true,
|
||
|
},
|
||
|
|
||
|
// Marshaling zero xml.Name uses the tag or field name.
|
||
|
{
|
||
|
Value: &NameInField{},
|
||
|
ExpectXML: `<NameInField><foo xmlns="ns"></foo></NameInField>`,
|
||
|
MarshalOnly: true,
|
||
|
},
|
||
|
|
||
|
// Test attributes
|
||
|
{
|
||
|
Value: &AttrTest{
|
||
|
Int: 8,
|
||
|
Named: 9,
|
||
|
Float: 23.5,
|
||
|
Uint8: 255,
|
||
|
Bool: true,
|
||
|
Str: "str",
|
||
|
Bytes: []byte("byt"),
|
||
|
},
|
||
|
ExpectXML: `<AttrTest Int="8" int="9" Float="23.5" Uint8="255"` +
|
||
|
` Bool="true" Str="str" Bytes="byt"></AttrTest>`,
|
||
|
},
|
||
|
{
|
||
|
Value: &AttrTest{Bytes: []byte{}},
|
||
|
ExpectXML: `<AttrTest Int="0" int="0" Float="0" Uint8="0"` +
|
||
|
` Bool="false" Str="" Bytes=""></AttrTest>`,
|
||
|
},
|
||
|
{
|
||
|
Value: &OmitAttrTest{
|
||
|
Int: 8,
|
||
|
Named: 9,
|
||
|
Float: 23.5,
|
||
|
Uint8: 255,
|
||
|
Bool: true,
|
||
|
Str: "str",
|
||
|
Bytes: []byte("byt"),
|
||
|
},
|
||
|
ExpectXML: `<OmitAttrTest Int="8" int="9" Float="23.5" Uint8="255"` +
|
||
|
` Bool="true" Str="str" Bytes="byt"></OmitAttrTest>`,
|
||
|
},
|
||
|
{
|
||
|
Value: &OmitAttrTest{},
|
||
|
ExpectXML: `<OmitAttrTest></OmitAttrTest>`,
|
||
|
},
|
||
|
|
||
|
// pointer fields
|
||
|
{
|
||
|
Value: &PointerFieldsTest{Name: &nameAttr, Age: &ageAttr, Contents: &contentsAttr},
|
||
|
ExpectXML: `<dummy name="Sarah" age="12">lorem ipsum</dummy>`,
|
||
|
MarshalOnly: true,
|
||
|
},
|
||
|
|
||
|
// empty chardata pointer field
|
||
|
{
|
||
|
Value: &ChardataEmptyTest{},
|
||
|
ExpectXML: `<test></test>`,
|
||
|
MarshalOnly: true,
|
||
|
},
|
||
|
|
||
|
// omitempty on fields
|
||
|
{
|
||
|
Value: &OmitFieldTest{
|
||
|
Int: 8,
|
||
|
Named: 9,
|
||
|
Float: 23.5,
|
||
|
Uint8: 255,
|
||
|
Bool: true,
|
||
|
Str: "str",
|
||
|
Bytes: []byte("byt"),
|
||
|
Ptr: &PresenceTest{},
|
||
|
},
|
||
|
ExpectXML: `<OmitFieldTest>` +
|
||
|
`<Int>8</Int>` +
|
||
|
`<int>9</int>` +
|
||
|
`<Float>23.5</Float>` +
|
||
|
`<Uint8>255</Uint8>` +
|
||
|
`<Bool>true</Bool>` +
|
||
|
`<Str>str</Str>` +
|
||
|
`<Bytes>byt</Bytes>` +
|
||
|
`<Ptr></Ptr>` +
|
||
|
`</OmitFieldTest>`,
|
||
|
},
|
||
|
{
|
||
|
Value: &OmitFieldTest{},
|
||
|
ExpectXML: `<OmitFieldTest></OmitFieldTest>`,
|
||
|
},
|
||
|
|
||
|
// Test ",any"
|
||
|
{
|
||
|
ExpectXML: `<a><nested><value>known</value></nested><other><sub>unknown</sub></other></a>`,
|
||
|
Value: &AnyTest{
|
||
|
Nested: "known",
|
||
|
AnyField: AnyHolder{
|
||
|
XMLName: Name{Local: "other"},
|
||
|
XML: "<sub>unknown</sub>",
|
||
|
},
|
||
|
},
|
||
|
},
|
||
|
{
|
||
|
Value: &AnyTest{Nested: "known",
|
||
|
AnyField: AnyHolder{
|
||
|
XML: "<unknown/>",
|
||
|
XMLName: Name{Local: "AnyField"},
|
||
|
},
|
||
|
},
|
||
|
ExpectXML: `<a><nested><value>known</value></nested><AnyField><unknown/></AnyField></a>`,
|
||
|
},
|
||
|
{
|
||
|
ExpectXML: `<a><nested><value>b</value></nested></a>`,
|
||
|
Value: &AnyOmitTest{
|
||
|
Nested: "b",
|
||
|
},
|
||
|
},
|
||
|
{
|
||
|
ExpectXML: `<a><nested><value>b</value></nested><c><d>e</d></c><g xmlns="f"><h>i</h></g></a>`,
|
||
|
Value: &AnySliceTest{
|
||
|
Nested: "b",
|
||
|
AnyField: []AnyHolder{
|
||
|
{
|
||
|
XMLName: Name{Local: "c"},
|
||
|
XML: "<d>e</d>",
|
||
|
},
|
||
|
{
|
||
|
XMLName: Name{Space: "f", Local: "g"},
|
||
|
XML: "<h>i</h>",
|
||
|
},
|
||
|
},
|
||
|
},
|
||
|
},
|
||
|
{
|
||
|
ExpectXML: `<a><nested><value>b</value></nested></a>`,
|
||
|
Value: &AnySliceTest{
|
||
|
Nested: "b",
|
||
|
},
|
||
|
},
|
||
|
|
||
|
// Test recursive types.
|
||
|
{
|
||
|
Value: &RecurseA{
|
||
|
A: "a1",
|
||
|
B: &RecurseB{
|
||
|
A: &RecurseA{"a2", nil},
|
||
|
B: "b1",
|
||
|
},
|
||
|
},
|
||
|
ExpectXML: `<RecurseA><A>a1</A><B><A><A>a2</A></A><B>b1</B></B></RecurseA>`,
|
||
|
},
|
||
|
|
||
|
// Test ignoring fields via "-" tag
|
||
|
{
|
||
|
ExpectXML: `<IgnoreTest></IgnoreTest>`,
|
||
|
Value: &IgnoreTest{},
|
||
|
},
|
||
|
{
|
||
|
ExpectXML: `<IgnoreTest></IgnoreTest>`,
|
||
|
Value: &IgnoreTest{PublicSecret: "can't tell"},
|
||
|
MarshalOnly: true,
|
||
|
},
|
||
|
{
|
||
|
ExpectXML: `<IgnoreTest><PublicSecret>ignore me</PublicSecret></IgnoreTest>`,
|
||
|
Value: &IgnoreTest{},
|
||
|
UnmarshalOnly: true,
|
||
|
},
|
||
|
|
||
|
// Test escaping.
|
||
|
{
|
||
|
ExpectXML: `<a><nested><value>dquote: "; squote: '; ampersand: &; less: <; greater: >;</value></nested><empty></empty></a>`,
|
||
|
Value: &AnyTest{
|
||
|
Nested: `dquote: "; squote: '; ampersand: &; less: <; greater: >;`,
|
||
|
AnyField: AnyHolder{XMLName: Name{Local: "empty"}},
|
||
|
},
|
||
|
},
|
||
|
{
|
||
|
ExpectXML: `<a><nested><value>newline: 
; cr: 
; tab: 	;</value></nested><AnyField></AnyField></a>`,
|
||
|
Value: &AnyTest{
|
||
|
Nested: "newline: \n; cr: \r; tab: \t;",
|
||
|
AnyField: AnyHolder{XMLName: Name{Local: "AnyField"}},
|
||
|
},
|
||
|
},
|
||
|
{
|
||
|
ExpectXML: "<a><nested><value>1\r2\r\n3\n\r4\n5</value></nested></a>",
|
||
|
Value: &AnyTest{
|
||
|
Nested: "1\n2\n3\n\n4\n5",
|
||
|
},
|
||
|
UnmarshalOnly: true,
|
||
|
},
|
||
|
{
|
||
|
ExpectXML: `<EmbedInt><MyInt>42</MyInt></EmbedInt>`,
|
||
|
Value: &EmbedInt{
|
||
|
MyInt: 42,
|
||
|
},
|
||
|
},
|
||
|
// Test omitempty with parent chain; see golang.org/issue/4168.
|
||
|
{
|
||
|
ExpectXML: `<Strings><A></A></Strings>`,
|
||
|
Value: &Strings{},
|
||
|
},
|
||
|
// Custom marshalers.
|
||
|
{
|
||
|
ExpectXML: `<MyMarshalerTest>hello world</MyMarshalerTest>`,
|
||
|
Value: &MyMarshalerTest{},
|
||
|
},
|
||
|
{
|
||
|
ExpectXML: `<MarshalerStruct Foo="hello world"></MarshalerStruct>`,
|
||
|
Value: &MarshalerStruct{},
|
||
|
},
|
||
|
{
|
||
|
ExpectXML: `<MarshalerValueStruct Foo="hello world"></MarshalerValueStruct>`,
|
||
|
Value: &MarshalerValueStruct{},
|
||
|
},
|
||
|
{
|
||
|
ExpectXML: `<outer xmlns="testns" int="10"></outer>`,
|
||
|
Value: &OuterStruct{IntAttr: 10},
|
||
|
},
|
||
|
{
|
||
|
ExpectXML: `<test xmlns="outerns" int="10"></test>`,
|
||
|
Value: &OuterNamedStruct{XMLName: Name{Space: "outerns", Local: "test"}, IntAttr: 10},
|
||
|
},
|
||
|
{
|
||
|
ExpectXML: `<test xmlns="outerns" int="10"></test>`,
|
||
|
Value: &OuterNamedOrderedStruct{XMLName: Name{Space: "outerns", Local: "test"}, IntAttr: 10},
|
||
|
},
|
||
|
{
|
||
|
ExpectXML: `<outer xmlns="testns" int="10"></outer>`,
|
||
|
Value: &OuterOuterStruct{OuterStruct{IntAttr: 10}},
|
||
|
},
|
||
|
{
|
||
|
ExpectXML: `<NestedAndChardata><A><B></B><B></B></A>test</NestedAndChardata>`,
|
||
|
Value: &NestedAndChardata{AB: make([]string, 2), Chardata: "test"},
|
||
|
},
|
||
|
{
|
||
|
ExpectXML: `<NestedAndComment><A><B></B><B></B></A><!--test--></NestedAndComment>`,
|
||
|
Value: &NestedAndComment{AB: make([]string, 2), Comment: "test"},
|
||
|
},
|
||
|
{
|
||
|
ExpectXML: `<XMLNSFieldStruct xmlns="http://example.com/ns"><Body>hello world</Body></XMLNSFieldStruct>`,
|
||
|
Value: &XMLNSFieldStruct{Ns: "http://example.com/ns", Body: "hello world"},
|
||
|
},
|
||
|
{
|
||
|
ExpectXML: `<testns:test xmlns:testns="testns" xmlns="http://example.com/ns"><Body>hello world</Body></testns:test>`,
|
||
|
Value: &NamedXMLNSFieldStruct{Ns: "http://example.com/ns", Body: "hello world"},
|
||
|
},
|
||
|
{
|
||
|
ExpectXML: `<testns:test xmlns:testns="testns"><Body>hello world</Body></testns:test>`,
|
||
|
Value: &NamedXMLNSFieldStruct{Ns: "", Body: "hello world"},
|
||
|
},
|
||
|
{
|
||
|
ExpectXML: `<XMLNSFieldStructWithOmitEmpty><Body>hello world</Body></XMLNSFieldStructWithOmitEmpty>`,
|
||
|
Value: &XMLNSFieldStructWithOmitEmpty{Body: "hello world"},
|
||
|
},
|
||
|
{
|
||
|
// The xmlns attribute must be ignored because the <test>
|
||
|
// element is in the empty namespace, so it's not possible
|
||
|
// to set the default namespace to something non-empty.
|
||
|
ExpectXML: `<test><Body>hello world</Body></test>`,
|
||
|
Value: &NamedXMLNSFieldStructWithEmptyNamespace{Ns: "foo", Body: "hello world"},
|
||
|
MarshalOnly: true,
|
||
|
},
|
||
|
{
|
||
|
ExpectXML: `<RecursiveXMLNSFieldStruct xmlns="foo"><Body xmlns=""><Text>hello world</Text></Body></RecursiveXMLNSFieldStruct>`,
|
||
|
Value: &RecursiveXMLNSFieldStruct{
|
||
|
Ns: "foo",
|
||
|
Body: &RecursiveXMLNSFieldStruct{
|
||
|
Text: "hello world",
|
||
|
},
|
||
|
},
|
||
|
},
|
||
|
}
|
||
|
|
||
|
func TestMarshal(t *testing.T) {
|
||
|
for idx, test := range marshalTests {
|
||
|
if test.UnmarshalOnly {
|
||
|
continue
|
||
|
}
|
||
|
data, err := Marshal(test.Value)
|
||
|
if err != nil {
|
||
|
t.Errorf("#%d: marshal(%#v): %s", idx, test.Value, err)
|
||
|
continue
|
||
|
}
|
||
|
if got, want := string(data), test.ExpectXML; got != want {
|
||
|
if strings.Contains(want, "\n") {
|
||
|
t.Errorf("#%d: marshal(%#v):\nHAVE:\n%s\nWANT:\n%s", idx, test.Value, got, want)
|
||
|
} else {
|
||
|
t.Errorf("#%d: marshal(%#v):\nhave %#q\nwant %#q", idx, test.Value, got, want)
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
type AttrParent struct {
|
||
|
X string `xml:"X>Y,attr"`
|
||
|
}
|
||
|
|
||
|
type BadAttr struct {
|
||
|
Name []string `xml:"name,attr"`
|
||
|
}
|
||
|
|
||
|
var marshalErrorTests = []struct {
|
||
|
Value interface{}
|
||
|
Err string
|
||
|
Kind reflect.Kind
|
||
|
}{
|
||
|
{
|
||
|
Value: make(chan bool),
|
||
|
Err: "xml: unsupported type: chan bool",
|
||
|
Kind: reflect.Chan,
|
||
|
},
|
||
|
{
|
||
|
Value: map[string]string{
|
||
|
"question": "What do you get when you multiply six by nine?",
|
||
|
"answer": "42",
|
||
|
},
|
||
|
Err: "xml: unsupported type: map[string]string",
|
||
|
Kind: reflect.Map,
|
||
|
},
|
||
|
{
|
||
|
Value: map[*Ship]bool{nil: false},
|
||
|
Err: "xml: unsupported type: map[*xml.Ship]bool",
|
||
|
Kind: reflect.Map,
|
||
|
},
|
||
|
{
|
||
|
Value: &Domain{Comment: []byte("f--bar")},
|
||
|
Err: `xml: comments must not contain "--"`,
|
||
|
},
|
||
|
// Reject parent chain with attr, never worked; see golang.org/issue/5033.
|
||
|
{
|
||
|
Value: &AttrParent{},
|
||
|
Err: `xml: X>Y chain not valid with attr flag`,
|
||
|
},
|
||
|
{
|
||
|
Value: BadAttr{[]string{"X", "Y"}},
|
||
|
Err: `xml: unsupported type: []string`,
|
||
|
},
|
||
|
}
|
||
|
|
||
|
var marshalIndentTests = []struct {
|
||
|
Value interface{}
|
||
|
Prefix string
|
||
|
Indent string
|
||
|
ExpectXML string
|
||
|
}{
|
||
|
{
|
||
|
Value: &SecretAgent{
|
||
|
Handle: "007",
|
||
|
Identity: "James Bond",
|
||
|
Obfuscate: "<redacted/>",
|
||
|
},
|
||
|
Prefix: "",
|
||
|
Indent: "\t",
|
||
|
ExpectXML: fmt.Sprintf("<agent handle=\"007\">\n\t<Identity>James Bond</Identity><redacted/>\n</agent>"),
|
||
|
},
|
||
|
}
|
||
|
|
||
|
func TestMarshalErrors(t *testing.T) {
|
||
|
for idx, test := range marshalErrorTests {
|
||
|
data, err := Marshal(test.Value)
|
||
|
if err == nil {
|
||
|
t.Errorf("#%d: marshal(%#v) = [success] %q, want error %v", idx, test.Value, data, test.Err)
|
||
|
continue
|
||
|
}
|
||
|
if err.Error() != test.Err {
|
||
|
t.Errorf("#%d: marshal(%#v) = [error] %v, want %v", idx, test.Value, err, test.Err)
|
||
|
}
|
||
|
if test.Kind != reflect.Invalid {
|
||
|
if kind := err.(*UnsupportedTypeError).Type.Kind(); kind != test.Kind {
|
||
|
t.Errorf("#%d: marshal(%#v) = [error kind] %s, want %s", idx, test.Value, kind, test.Kind)
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Do invertibility testing on the various structures that we test
|
||
|
func TestUnmarshal(t *testing.T) {
|
||
|
for i, test := range marshalTests {
|
||
|
if test.MarshalOnly {
|
||
|
continue
|
||
|
}
|
||
|
if _, ok := test.Value.(*Plain); ok {
|
||
|
continue
|
||
|
}
|
||
|
vt := reflect.TypeOf(test.Value)
|
||
|
dest := reflect.New(vt.Elem()).Interface()
|
||
|
err := Unmarshal([]byte(test.ExpectXML), dest)
|
||
|
|
||
|
switch fix := dest.(type) {
|
||
|
case *Feed:
|
||
|
fix.Author.InnerXML = ""
|
||
|
for i := range fix.Entry {
|
||
|
fix.Entry[i].Author.InnerXML = ""
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if err != nil {
|
||
|
t.Errorf("#%d: unexpected error: %#v", i, err)
|
||
|
} else if got, want := dest, test.Value; !reflect.DeepEqual(got, want) {
|
||
|
t.Errorf("#%d: unmarshal(%q):\nhave %#v\nwant %#v", i, test.ExpectXML, got, want)
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func TestMarshalIndent(t *testing.T) {
|
||
|
for i, test := range marshalIndentTests {
|
||
|
data, err := MarshalIndent(test.Value, test.Prefix, test.Indent)
|
||
|
if err != nil {
|
||
|
t.Errorf("#%d: Error: %s", i, err)
|
||
|
continue
|
||
|
}
|
||
|
if got, want := string(data), test.ExpectXML; got != want {
|
||
|
t.Errorf("#%d: MarshalIndent:\nGot:%s\nWant:\n%s", i, got, want)
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
type limitedBytesWriter struct {
|
||
|
w io.Writer
|
||
|
remain int // until writes fail
|
||
|
}
|
||
|
|
||
|
func (lw *limitedBytesWriter) Write(p []byte) (n int, err error) {
|
||
|
if lw.remain <= 0 {
|
||
|
println("error")
|
||
|
return 0, errors.New("write limit hit")
|
||
|
}
|
||
|
if len(p) > lw.remain {
|
||
|
p = p[:lw.remain]
|
||
|
n, _ = lw.w.Write(p)
|
||
|
lw.remain = 0
|
||
|
return n, errors.New("write limit hit")
|
||
|
}
|
||
|
n, err = lw.w.Write(p)
|
||
|
lw.remain -= n
|
||
|
return n, err
|
||
|
}
|
||
|
|
||
|
func TestMarshalWriteErrors(t *testing.T) {
|
||
|
var buf bytes.Buffer
|
||
|
const writeCap = 1024
|
||
|
w := &limitedBytesWriter{&buf, writeCap}
|
||
|
enc := NewEncoder(w)
|
||
|
var err error
|
||
|
var i int
|
||
|
const n = 4000
|
||
|
for i = 1; i <= n; i++ {
|
||
|
err = enc.Encode(&Passenger{
|
||
|
Name: []string{"Alice", "Bob"},
|
||
|
Weight: 5,
|
||
|
})
|
||
|
if err != nil {
|
||
|
break
|
||
|
}
|
||
|
}
|
||
|
if err == nil {
|
||
|
t.Error("expected an error")
|
||
|
}
|
||
|
if i == n {
|
||
|
t.Errorf("expected to fail before the end")
|
||
|
}
|
||
|
if buf.Len() != writeCap {
|
||
|
t.Errorf("buf.Len() = %d; want %d", buf.Len(), writeCap)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func TestMarshalWriteIOErrors(t *testing.T) {
|
||
|
enc := NewEncoder(errWriter{})
|
||
|
|
||
|
expectErr := "unwritable"
|
||
|
err := enc.Encode(&Passenger{})
|
||
|
if err == nil || err.Error() != expectErr {
|
||
|
t.Errorf("EscapeTest = [error] %v, want %v", err, expectErr)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func TestMarshalFlush(t *testing.T) {
|
||
|
var buf bytes.Buffer
|
||
|
enc := NewEncoder(&buf)
|
||
|
if err := enc.EncodeToken(CharData("hello world")); err != nil {
|
||
|
t.Fatalf("enc.EncodeToken: %v", err)
|
||
|
}
|
||
|
if buf.Len() > 0 {
|
||
|
t.Fatalf("enc.EncodeToken caused actual write: %q", buf.Bytes())
|
||
|
}
|
||
|
if err := enc.Flush(); err != nil {
|
||
|
t.Fatalf("enc.Flush: %v", err)
|
||
|
}
|
||
|
if buf.String() != "hello world" {
|
||
|
t.Fatalf("after enc.Flush, buf.String() = %q, want %q", buf.String(), "hello world")
|
||
|
}
|
||
|
}
|
||
|
|
||
|
var encodeElementTests = []struct {
|
||
|
desc string
|
||
|
value interface{}
|
||
|
start StartElement
|
||
|
expectXML string
|
||
|
}{{
|
||
|
desc: "simple string",
|
||
|
value: "hello",
|
||
|
start: StartElement{
|
||
|
Name: Name{Local: "a"},
|
||
|
},
|
||
|
expectXML: `<a>hello</a>`,
|
||
|
}, {
|
||
|
desc: "string with added attributes",
|
||
|
value: "hello",
|
||
|
start: StartElement{
|
||
|
Name: Name{Local: "a"},
|
||
|
Attr: []Attr{{
|
||
|
Name: Name{Local: "x"},
|
||
|
Value: "y",
|
||
|
}, {
|
||
|
Name: Name{Local: "foo"},
|
||
|
Value: "bar",
|
||
|
}},
|
||
|
},
|
||
|
expectXML: `<a x="y" foo="bar">hello</a>`,
|
||
|
}, {
|
||
|
desc: "start element with default name space",
|
||
|
value: struct {
|
||
|
Foo XMLNameWithNSTag
|
||
|
}{
|
||
|
Foo: XMLNameWithNSTag{
|
||
|
Value: "hello",
|
||
|
},
|
||
|
},
|
||
|
start: StartElement{
|
||
|
Name: Name{Space: "ns", Local: "a"},
|
||
|
Attr: []Attr{{
|
||
|
Name: Name{Local: "xmlns"},
|
||
|
// "ns" is the name space defined in XMLNameWithNSTag
|
||
|
Value: "ns",
|
||
|
}},
|
||
|
},
|
||
|
expectXML: `<a xmlns="ns"><InXMLNameWithNSTag>hello</InXMLNameWithNSTag></a>`,
|
||
|
}, {
|
||
|
desc: "start element in name space with different default name space",
|
||
|
value: struct {
|
||
|
Foo XMLNameWithNSTag
|
||
|
}{
|
||
|
Foo: XMLNameWithNSTag{
|
||
|
Value: "hello",
|
||
|
},
|
||
|
},
|
||
|
start: StartElement{
|
||
|
Name: Name{Space: "ns2", Local: "a"},
|
||
|
Attr: []Attr{{
|
||
|
Name: Name{Local: "xmlns"},
|
||
|
// "ns" is the name space defined in XMLNameWithNSTag
|
||
|
Value: "ns",
|
||
|
}},
|
||
|
},
|
||
|
expectXML: `<ns2:a xmlns:ns2="ns2" xmlns="ns"><InXMLNameWithNSTag>hello</InXMLNameWithNSTag></ns2:a>`,
|
||
|
}, {
|
||
|
desc: "XMLMarshaler with start element with default name space",
|
||
|
value: &MyMarshalerTest{},
|
||
|
start: StartElement{
|
||
|
Name: Name{Space: "ns2", Local: "a"},
|
||
|
Attr: []Attr{{
|
||
|
Name: Name{Local: "xmlns"},
|
||
|
// "ns" is the name space defined in XMLNameWithNSTag
|
||
|
Value: "ns",
|
||
|
}},
|
||
|
},
|
||
|
expectXML: `<ns2:a xmlns:ns2="ns2" xmlns="ns">hello world</ns2:a>`,
|
||
|
}}
|
||
|
|
||
|
func TestEncodeElement(t *testing.T) {
|
||
|
for idx, test := range encodeElementTests {
|
||
|
var buf bytes.Buffer
|
||
|
enc := NewEncoder(&buf)
|
||
|
err := enc.EncodeElement(test.value, test.start)
|
||
|
if err != nil {
|
||
|
t.Fatalf("enc.EncodeElement: %v", err)
|
||
|
}
|
||
|
err = enc.Flush()
|
||
|
if err != nil {
|
||
|
t.Fatalf("enc.Flush: %v", err)
|
||
|
}
|
||
|
if got, want := buf.String(), test.expectXML; got != want {
|
||
|
t.Errorf("#%d(%s): EncodeElement(%#v, %#v):\nhave %#q\nwant %#q", idx, test.desc, test.value, test.start, got, want)
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func BenchmarkMarshal(b *testing.B) {
|
||
|
b.ReportAllocs()
|
||
|
for i := 0; i < b.N; i++ {
|
||
|
Marshal(atomValue)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func BenchmarkUnmarshal(b *testing.B) {
|
||
|
b.ReportAllocs()
|
||
|
xml := []byte(atomXml)
|
||
|
for i := 0; i < b.N; i++ {
|
||
|
Unmarshal(xml, &Feed{})
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// golang.org/issue/6556
|
||
|
func TestStructPointerMarshal(t *testing.T) {
|
||
|
type A struct {
|
||
|
XMLName string `xml:"a"`
|
||
|
B []interface{}
|
||
|
}
|
||
|
type C struct {
|
||
|
XMLName Name
|
||
|
Value string `xml:"value"`
|
||
|
}
|
||
|
|
||
|
a := new(A)
|
||
|
a.B = append(a.B, &C{
|
||
|
XMLName: Name{Local: "c"},
|
||
|
Value: "x",
|
||
|
})
|
||
|
|
||
|
b, err := Marshal(a)
|
||
|
if err != nil {
|
||
|
t.Fatal(err)
|
||
|
}
|
||
|
if x := string(b); x != "<a><c><value>x</value></c></a>" {
|
||
|
t.Fatal(x)
|
||
|
}
|
||
|
var v A
|
||
|
err = Unmarshal(b, &v)
|
||
|
if err != nil {
|
||
|
t.Fatal(err)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
var encodeTokenTests = []struct {
|
||
|
desc string
|
||
|
toks []Token
|
||
|
want string
|
||
|
err string
|
||
|
}{{
|
||
|
desc: "start element with name space",
|
||
|
toks: []Token{
|
||
|
StartElement{Name{"space", "local"}, nil},
|
||
|
},
|
||
|
want: `<space:local xmlns:space="space">`,
|
||
|
}, {
|
||
|
desc: "start element with no name",
|
||
|
toks: []Token{
|
||
|
StartElement{Name{"space", ""}, nil},
|
||
|
},
|
||
|
err: "xml: start tag with no name",
|
||
|
}, {
|
||
|
desc: "end element with no name",
|
||
|
toks: []Token{
|
||
|
EndElement{Name{"space", ""}},
|
||
|
},
|
||
|
err: "xml: end tag with no name",
|
||
|
}, {
|
||
|
desc: "char data",
|
||
|
toks: []Token{
|
||
|
CharData("foo"),
|
||
|
},
|
||
|
want: `foo`,
|
||
|
}, {
|
||
|
desc: "char data with escaped chars",
|
||
|
toks: []Token{
|
||
|
CharData(" \t\n"),
|
||
|
},
|
||
|
want: " 	\n",
|
||
|
}, {
|
||
|
desc: "comment",
|
||
|
toks: []Token{
|
||
|
Comment("foo"),
|
||
|
},
|
||
|
want: `<!--foo-->`,
|
||
|
}, {
|
||
|
desc: "comment with invalid content",
|
||
|
toks: []Token{
|
||
|
Comment("foo-->"),
|
||
|
},
|
||
|
err: "xml: EncodeToken of Comment containing --> marker",
|
||
|
}, {
|
||
|
desc: "proc instruction",
|
||
|
toks: []Token{
|
||
|
ProcInst{"Target", []byte("Instruction")},
|
||
|
},
|
||
|
want: `<?Target Instruction?>`,
|
||
|
}, {
|
||
|
desc: "proc instruction with empty target",
|
||
|
toks: []Token{
|
||
|
ProcInst{"", []byte("Instruction")},
|
||
|
},
|
||
|
err: "xml: EncodeToken of ProcInst with invalid Target",
|
||
|
}, {
|
||
|
desc: "proc instruction with bad content",
|
||
|
toks: []Token{
|
||
|
ProcInst{"", []byte("Instruction?>")},
|
||
|
},
|
||
|
err: "xml: EncodeToken of ProcInst with invalid Target",
|
||
|
}, {
|
||
|
desc: "directive",
|
||
|
toks: []Token{
|
||
|
Directive("foo"),
|
||
|
},
|
||
|
want: `<!foo>`,
|
||
|
}, {
|
||
|
desc: "more complex directive",
|
||
|
toks: []Token{
|
||
|
Directive("DOCTYPE doc [ <!ELEMENT doc '>'> <!-- com>ment --> ]"),
|
||
|
},
|
||
|
want: `<!DOCTYPE doc [ <!ELEMENT doc '>'> <!-- com>ment --> ]>`,
|
||
|
}, {
|
||
|
desc: "directive instruction with bad name",
|
||
|
toks: []Token{
|
||
|
Directive("foo>"),
|
||
|
},
|
||
|
err: "xml: EncodeToken of Directive containing wrong < or > markers",
|
||
|
}, {
|
||
|
desc: "end tag without start tag",
|
||
|
toks: []Token{
|
||
|
EndElement{Name{"foo", "bar"}},
|
||
|
},
|
||
|
err: "xml: end tag </bar> without start tag",
|
||
|
}, {
|
||
|
desc: "mismatching end tag local name",
|
||
|
toks: []Token{
|
||
|
StartElement{Name{"", "foo"}, nil},
|
||
|
EndElement{Name{"", "bar"}},
|
||
|
},
|
||
|
err: "xml: end tag </bar> does not match start tag <foo>",
|
||
|
want: `<foo>`,
|
||
|
}, {
|
||
|
desc: "mismatching end tag namespace",
|
||
|
toks: []Token{
|
||
|
StartElement{Name{"space", "foo"}, nil},
|
||
|
EndElement{Name{"another", "foo"}},
|
||
|
},
|
||
|
err: "xml: end tag </foo> in namespace another does not match start tag <foo> in namespace space",
|
||
|
want: `<space:foo xmlns:space="space">`,
|
||
|
}, {
|
||
|
desc: "start element with explicit namespace",
|
||
|
toks: []Token{
|
||
|
StartElement{Name{"space", "local"}, []Attr{
|
||
|
{Name{"xmlns", "x"}, "space"},
|
||
|
{Name{"space", "foo"}, "value"},
|
||
|
}},
|
||
|
},
|
||
|
want: `<x:local xmlns:x="space" x:foo="value">`,
|
||
|
}, {
|
||
|
desc: "start element with explicit namespace and colliding prefix",
|
||
|
toks: []Token{
|
||
|
StartElement{Name{"space", "local"}, []Attr{
|
||
|
{Name{"xmlns", "x"}, "space"},
|
||
|
{Name{"space", "foo"}, "value"},
|
||
|
{Name{"x", "bar"}, "other"},
|
||
|
}},
|
||
|
},
|
||
|
want: `<x:local xmlns:x_1="x" xmlns:x="space" x:foo="value" x_1:bar="other">`,
|
||
|
}, {
|
||
|
desc: "start element using previously defined namespace",
|
||
|
toks: []Token{
|
||
|
StartElement{Name{"", "local"}, []Attr{
|
||
|
{Name{"xmlns", "x"}, "space"},
|
||
|
}},
|
||
|
StartElement{Name{"space", "foo"}, []Attr{
|
||
|
{Name{"space", "x"}, "y"},
|
||
|
}},
|
||
|
},
|
||
|
want: `<local xmlns:x="space"><x:foo x:x="y">`,
|
||
|
}, {
|
||
|
desc: "nested name space with same prefix",
|
||
|
toks: []Token{
|
||
|
StartElement{Name{"", "foo"}, []Attr{
|
||
|
{Name{"xmlns", "x"}, "space1"},
|
||
|
}},
|
||
|
StartElement{Name{"", "foo"}, []Attr{
|
||
|
{Name{"xmlns", "x"}, "space2"},
|
||
|
}},
|
||
|
StartElement{Name{"", "foo"}, []Attr{
|
||
|
{Name{"space1", "a"}, "space1 value"},
|
||
|
{Name{"space2", "b"}, "space2 value"},
|
||
|
}},
|
||
|
EndElement{Name{"", "foo"}},
|
||
|
EndElement{Name{"", "foo"}},
|
||
|
StartElement{Name{"", "foo"}, []Attr{
|
||
|
{Name{"space1", "a"}, "space1 value"},
|
||
|
{Name{"space2", "b"}, "space2 value"},
|
||
|
}},
|
||
|
},
|
||
|
want: `<foo xmlns:x="space1"><foo xmlns:x="space2"><foo xmlns:space1="space1" space1:a="space1 value" x:b="space2 value"></foo></foo><foo xmlns:space2="space2" x:a="space1 value" space2:b="space2 value">`,
|
||
|
}, {
|
||
|
desc: "start element defining several prefixes for the same name space",
|
||
|
toks: []Token{
|
||
|
StartElement{Name{"space", "foo"}, []Attr{
|
||
|
{Name{"xmlns", "a"}, "space"},
|
||
|
{Name{"xmlns", "b"}, "space"},
|
||
|
{Name{"space", "x"}, "value"},
|
||
|
}},
|
||
|
},
|
||
|
want: `<a:foo xmlns:a="space" a:x="value">`,
|
||
|
}, {
|
||
|
desc: "nested element redefines name space",
|
||
|
toks: []Token{
|
||
|
StartElement{Name{"", "foo"}, []Attr{
|
||
|
{Name{"xmlns", "x"}, "space"},
|
||
|
}},
|
||
|
StartElement{Name{"space", "foo"}, []Attr{
|
||
|
{Name{"xmlns", "y"}, "space"},
|
||
|
{Name{"space", "a"}, "value"},
|
||
|
}},
|
||
|
},
|
||
|
want: `<foo xmlns:x="space"><x:foo x:a="value">`,
|
||
|
}, {
|
||
|
desc: "nested element creates alias for default name space",
|
||
|
toks: []Token{
|
||
|
StartElement{Name{"space", "foo"}, []Attr{
|
||
|
{Name{"", "xmlns"}, "space"},
|
||
|
}},
|
||
|
StartElement{Name{"space", "foo"}, []Attr{
|
||
|
{Name{"xmlns", "y"}, "space"},
|
||
|
{Name{"space", "a"}, "value"},
|
||
|
}},
|
||
|
},
|
||
|
want: `<foo xmlns="space"><foo xmlns:y="space" y:a="value">`,
|
||
|
}, {
|
||
|
desc: "nested element defines default name space with existing prefix",
|
||
|
toks: []Token{
|
||
|
StartElement{Name{"", "foo"}, []Attr{
|
||
|
{Name{"xmlns", "x"}, "space"},
|
||
|
}},
|
||
|
StartElement{Name{"space", "foo"}, []Attr{
|
||
|
{Name{"", "xmlns"}, "space"},
|
||
|
{Name{"space", "a"}, "value"},
|
||
|
}},
|
||
|
},
|
||
|
want: `<foo xmlns:x="space"><foo xmlns="space" x:a="value">`,
|
||
|
}, {
|
||
|
desc: "nested element uses empty attribute name space when default ns defined",
|
||
|
toks: []Token{
|
||
|
StartElement{Name{"space", "foo"}, []Attr{
|
||
|
{Name{"", "xmlns"}, "space"},
|
||
|
}},
|
||
|
StartElement{Name{"space", "foo"}, []Attr{
|
||
|
{Name{"", "attr"}, "value"},
|
||
|
}},
|
||
|
},
|
||
|
want: `<foo xmlns="space"><foo attr="value">`,
|
||
|
}, {
|
||
|
desc: "redefine xmlns",
|
||
|
toks: []Token{
|
||
|
StartElement{Name{"", "foo"}, []Attr{
|
||
|
{Name{"foo", "xmlns"}, "space"},
|
||
|
}},
|
||
|
},
|
||
|
err: `xml: cannot redefine xmlns attribute prefix`,
|
||
|
}, {
|
||
|
desc: "xmlns with explicit name space #1",
|
||
|
toks: []Token{
|
||
|
StartElement{Name{"space", "foo"}, []Attr{
|
||
|
{Name{"xml", "xmlns"}, "space"},
|
||
|
}},
|
||
|
},
|
||
|
want: `<foo xmlns="space">`,
|
||
|
}, {
|
||
|
desc: "xmlns with explicit name space #2",
|
||
|
toks: []Token{
|
||
|
StartElement{Name{"space", "foo"}, []Attr{
|
||
|
{Name{xmlURL, "xmlns"}, "space"},
|
||
|
}},
|
||
|
},
|
||
|
want: `<foo xmlns="space">`,
|
||
|
}, {
|
||
|
desc: "empty name space declaration is ignored",
|
||
|
toks: []Token{
|
||
|
StartElement{Name{"", "foo"}, []Attr{
|
||
|
{Name{"xmlns", "foo"}, ""},
|
||
|
}},
|
||
|
},
|
||
|
want: `<foo>`,
|
||
|
}, {
|
||
|
desc: "attribute with no name is ignored",
|
||
|
toks: []Token{
|
||
|
StartElement{Name{"", "foo"}, []Attr{
|
||
|
{Name{"", ""}, "value"},
|
||
|
}},
|
||
|
},
|
||
|
want: `<foo>`,
|
||
|
}, {
|
||
|
desc: "namespace URL with non-valid name",
|
||
|
toks: []Token{
|
||
|
StartElement{Name{"/34", "foo"}, []Attr{
|
||
|
{Name{"/34", "x"}, "value"},
|
||
|
}},
|
||
|
},
|
||
|
want: `<_:foo xmlns:_="/34" _:x="value">`,
|
||
|
}, {
|
||
|
desc: "nested element resets default namespace to empty",
|
||
|
toks: []Token{
|
||
|
StartElement{Name{"space", "foo"}, []Attr{
|
||
|
{Name{"", "xmlns"}, "space"},
|
||
|
}},
|
||
|
StartElement{Name{"", "foo"}, []Attr{
|
||
|
{Name{"", "xmlns"}, ""},
|
||
|
{Name{"", "x"}, "value"},
|
||
|
{Name{"space", "x"}, "value"},
|
||
|
}},
|
||
|
},
|
||
|
want: `<foo xmlns="space"><foo xmlns:space="space" xmlns="" x="value" space:x="value">`,
|
||
|
}, {
|
||
|
desc: "nested element requires empty default name space",
|
||
|
toks: []Token{
|
||
|
StartElement{Name{"space", "foo"}, []Attr{
|
||
|
{Name{"", "xmlns"}, "space"},
|
||
|
}},
|
||
|
StartElement{Name{"", "foo"}, nil},
|
||
|
},
|
||
|
want: `<foo xmlns="space"><foo xmlns="">`,
|
||
|
}, {
|
||
|
desc: "attribute uses name space from xmlns",
|
||
|
toks: []Token{
|
||
|
StartElement{Name{"some/space", "foo"}, []Attr{
|
||
|
{Name{"", "attr"}, "value"},
|
||
|
{Name{"some/space", "other"}, "other value"},
|
||
|
}},
|
||
|
},
|
||
|
want: `<space:foo xmlns:space="some/space" attr="value" space:other="other value">`,
|
||
|
}, {
|
||
|
desc: "default name space should not be used by attributes",
|
||
|
toks: []Token{
|
||
|
StartElement{Name{"space", "foo"}, []Attr{
|
||
|
{Name{"", "xmlns"}, "space"},
|
||
|
{Name{"xmlns", "bar"}, "space"},
|
||
|
{Name{"space", "baz"}, "foo"},
|
||
|
}},
|
||
|
StartElement{Name{"space", "baz"}, nil},
|
||
|
EndElement{Name{"space", "baz"}},
|
||
|
EndElement{Name{"space", "foo"}},
|
||
|
},
|
||
|
want: `<foo xmlns:bar="space" xmlns="space" bar:baz="foo"><baz></baz></foo>`,
|
||
|
}, {
|
||
|
desc: "default name space not used by attributes, not explicitly defined",
|
||
|
toks: []Token{
|
||
|
StartElement{Name{"space", "foo"}, []Attr{
|
||
|
{Name{"", "xmlns"}, "space"},
|
||
|
{Name{"space", "baz"}, "foo"},
|
||
|
}},
|
||
|
StartElement{Name{"space", "baz"}, nil},
|
||
|
EndElement{Name{"space", "baz"}},
|
||
|
EndElement{Name{"space", "foo"}},
|
||
|
},
|
||
|
want: `<foo xmlns:space="space" xmlns="space" space:baz="foo"><baz></baz></foo>`,
|
||
|
}, {
|
||
|
desc: "impossible xmlns declaration",
|
||
|
toks: []Token{
|
||
|
StartElement{Name{"", "foo"}, []Attr{
|
||
|
{Name{"", "xmlns"}, "space"},
|
||
|
}},
|
||
|
StartElement{Name{"space", "bar"}, []Attr{
|
||
|
{Name{"space", "attr"}, "value"},
|
||
|
}},
|
||
|
},
|
||
|
want: `<foo><space:bar xmlns:space="space" space:attr="value">`,
|
||
|
}}
|
||
|
|
||
|
func TestEncodeToken(t *testing.T) {
|
||
|
loop:
|
||
|
for i, tt := range encodeTokenTests {
|
||
|
var buf bytes.Buffer
|
||
|
enc := NewEncoder(&buf)
|
||
|
var err error
|
||
|
for j, tok := range tt.toks {
|
||
|
err = enc.EncodeToken(tok)
|
||
|
if err != nil && j < len(tt.toks)-1 {
|
||
|
t.Errorf("#%d %s token #%d: %v", i, tt.desc, j, err)
|
||
|
continue loop
|
||
|
}
|
||
|
}
|
||
|
errorf := func(f string, a ...interface{}) {
|
||
|
t.Errorf("#%d %s token #%d:%s", i, tt.desc, len(tt.toks)-1, fmt.Sprintf(f, a...))
|
||
|
}
|
||
|
switch {
|
||
|
case tt.err != "" && err == nil:
|
||
|
errorf(" expected error; got none")
|
||
|
continue
|
||
|
case tt.err == "" && err != nil:
|
||
|
errorf(" got error: %v", err)
|
||
|
continue
|
||
|
case tt.err != "" && err != nil && tt.err != err.Error():
|
||
|
errorf(" error mismatch; got %v, want %v", err, tt.err)
|
||
|
continue
|
||
|
}
|
||
|
if err := enc.Flush(); err != nil {
|
||
|
errorf(" %v", err)
|
||
|
continue
|
||
|
}
|
||
|
if got := buf.String(); got != tt.want {
|
||
|
errorf("\ngot %v\nwant %v", got, tt.want)
|
||
|
continue
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func TestProcInstEncodeToken(t *testing.T) {
|
||
|
var buf bytes.Buffer
|
||
|
enc := NewEncoder(&buf)
|
||
|
|
||
|
if err := enc.EncodeToken(ProcInst{"xml", []byte("Instruction")}); err != nil {
|
||
|
t.Fatalf("enc.EncodeToken: expected to be able to encode xml target ProcInst as first token, %s", err)
|
||
|
}
|
||
|
|
||
|
if err := enc.EncodeToken(ProcInst{"Target", []byte("Instruction")}); err != nil {
|
||
|
t.Fatalf("enc.EncodeToken: expected to be able to add non-xml target ProcInst")
|
||
|
}
|
||
|
|
||
|
if err := enc.EncodeToken(ProcInst{"xml", []byte("Instruction")}); err == nil {
|
||
|
t.Fatalf("enc.EncodeToken: expected to not be allowed to encode xml target ProcInst when not first token")
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func TestDecodeEncode(t *testing.T) {
|
||
|
var in, out bytes.Buffer
|
||
|
in.WriteString(`<?xml version="1.0" encoding="UTF-8"?>
|
||
|
<?Target Instruction?>
|
||
|
<root>
|
||
|
</root>
|
||
|
`)
|
||
|
dec := NewDecoder(&in)
|
||
|
enc := NewEncoder(&out)
|
||
|
for tok, err := dec.Token(); err == nil; tok, err = dec.Token() {
|
||
|
err = enc.EncodeToken(tok)
|
||
|
if err != nil {
|
||
|
t.Fatalf("enc.EncodeToken: Unable to encode token (%#v), %v", tok, err)
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Issue 9796. Used to fail with GORACE="halt_on_error=1" -race.
|
||
|
func TestRace9796(t *testing.T) {
|
||
|
type A struct{}
|
||
|
type B struct {
|
||
|
C []A `xml:"X>Y"`
|
||
|
}
|
||
|
var wg sync.WaitGroup
|
||
|
for i := 0; i < 2; i++ {
|
||
|
wg.Add(1)
|
||
|
go func() {
|
||
|
Marshal(B{[]A{{}}})
|
||
|
wg.Done()
|
||
|
}()
|
||
|
}
|
||
|
wg.Wait()
|
||
|
}
|
||
|
|
||
|
func TestIsValidDirective(t *testing.T) {
|
||
|
testOK := []string{
|
||
|
"<>",
|
||
|
"< < > >",
|
||
|
"<!DOCTYPE '<' '>' '>' <!--nothing-->>",
|
||
|
"<!DOCTYPE doc [ <!ELEMENT doc ANY> <!ELEMENT doc ANY> ]>",
|
||
|
"<!DOCTYPE doc [ <!ELEMENT doc \"ANY> '<' <!E\" LEMENT '>' doc ANY> ]>",
|
||
|
"<!DOCTYPE doc <!-- just>>>> a < comment --> [ <!ITEM anything> ] >",
|
||
|
}
|
||
|
testKO := []string{
|
||
|
"<",
|
||
|
">",
|
||
|
"<!--",
|
||
|
"-->",
|
||
|
"< > > < < >",
|
||
|
"<!dummy <!-- > -->",
|
||
|
"<!DOCTYPE doc '>",
|
||
|
"<!DOCTYPE doc '>'",
|
||
|
"<!DOCTYPE doc <!--comment>",
|
||
|
}
|
||
|
for _, s := range testOK {
|
||
|
if !isValidDirective(Directive(s)) {
|
||
|
t.Errorf("Directive %q is expected to be valid", s)
|
||
|
}
|
||
|
}
|
||
|
for _, s := range testKO {
|
||
|
if isValidDirective(Directive(s)) {
|
||
|
t.Errorf("Directive %q is expected to be invalid", s)
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Issue 11719. EncodeToken used to silently eat tokens with an invalid type.
|
||
|
func TestSimpleUseOfEncodeToken(t *testing.T) {
|
||
|
var buf bytes.Buffer
|
||
|
enc := NewEncoder(&buf)
|
||
|
if err := enc.EncodeToken(&StartElement{Name: Name{"", "object1"}}); err == nil {
|
||
|
t.Errorf("enc.EncodeToken: pointer type should be rejected")
|
||
|
}
|
||
|
if err := enc.EncodeToken(&EndElement{Name: Name{"", "object1"}}); err == nil {
|
||
|
t.Errorf("enc.EncodeToken: pointer type should be rejected")
|
||
|
}
|
||
|
if err := enc.EncodeToken(StartElement{Name: Name{"", "object2"}}); err != nil {
|
||
|
t.Errorf("enc.EncodeToken: StartElement %s", err)
|
||
|
}
|
||
|
if err := enc.EncodeToken(EndElement{Name: Name{"", "object2"}}); err != nil {
|
||
|
t.Errorf("enc.EncodeToken: EndElement %s", err)
|
||
|
}
|
||
|
if err := enc.EncodeToken(Universe{}); err == nil {
|
||
|
t.Errorf("enc.EncodeToken: invalid type not caught")
|
||
|
}
|
||
|
if err := enc.Flush(); err != nil {
|
||
|
t.Errorf("enc.Flush: %s", err)
|
||
|
}
|
||
|
if buf.Len() == 0 {
|
||
|
t.Errorf("enc.EncodeToken: empty buffer")
|
||
|
}
|
||
|
want := "<object2></object2>"
|
||
|
if buf.String() != want {
|
||
|
t.Errorf("enc.EncodeToken: expected %q; got %q", want, buf.String())
|
||
|
}
|
||
|
}
|