diff options
Diffstat (limited to 'src/pkg/xml/marshal_test.go')
| -rw-r--r-- | src/pkg/xml/marshal_test.go | 305 | 
1 files changed, 305 insertions, 0 deletions
| diff --git a/src/pkg/xml/marshal_test.go b/src/pkg/xml/marshal_test.go new file mode 100644 index 000000000..5b972fafe --- /dev/null +++ b/src/pkg/xml/marshal_test.go @@ -0,0 +1,305 @@ +// 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 ( +	"reflect" +	"testing" + +	"os" +	"bytes" +	"strings" +	"strconv" +) + +type DriveType int + +const ( +	HyperDrive DriveType = iota +	ImprobabilityDrive +) + +type Passenger struct { +	Name   []string `xml:"name"` +	Weight float32  `xml:"weight"` +} + +type Ship struct { +	XMLName Name `xml:"spaceship"` + +	Name      string       `xml:"attr"` +	Pilot     string       `xml:"attr"` +	Drive     DriveType    `xml:"drive"` +	Age       uint         `xml:"age"` +	Passenger []*Passenger `xml:"passenger"` +	secret    string +} + +type RawXML string + +func (rx RawXML) MarshalXML() ([]byte, os.Error) { +	return []byte(rx), nil +} + +type NamedType string + +type Port struct { +	XMLName Name   `xml:"port"` +	Type    string `xml:"attr"` +	Number  string `xml:"chardata"` +} + +type Domain struct { +	XMLName Name   `xml:"domain"` +	Country string `xml:"attr"` +	Name    []byte `xml:"chardata"` +} + +type Book struct { +	XMLName Name   `xml:"book"` +	Title   string `xml:"chardata"` +} + +type SecretAgent struct { +	XMLName   Name   `xml:"agent"` +	Handle    string `xml:"attr"` +	Identity  string +	Obfuscate string `xml:"innerxml"` +} + +var nilStruct *Ship + +var marshalTests = []struct { +	Value     interface{} +	ExpectXML string +}{ +	// Test nil marshals to nothing +	{Value: nil, ExpectXML: ``}, +	{Value: nilStruct, ExpectXML: ``}, + +	// Test value types (no tag name, so ???) +	{Value: true, ExpectXML: `<???>true</???>`}, +	{Value: int(42), ExpectXML: `<???>42</???>`}, +	{Value: int8(42), ExpectXML: `<???>42</???>`}, +	{Value: int16(42), ExpectXML: `<???>42</???>`}, +	{Value: int32(42), ExpectXML: `<???>42</???>`}, +	{Value: uint(42), ExpectXML: `<???>42</???>`}, +	{Value: uint8(42), ExpectXML: `<???>42</???>`}, +	{Value: uint16(42), ExpectXML: `<???>42</???>`}, +	{Value: uint32(42), ExpectXML: `<???>42</???>`}, +	{Value: float32(1.25), ExpectXML: `<???>1.25</???>`}, +	{Value: float64(1.25), ExpectXML: `<???>1.25</???>`}, +	{Value: uintptr(0xFFDD), ExpectXML: `<???>65501</???>`}, +	{Value: "gopher", ExpectXML: `<???>gopher</???>`}, +	{Value: []byte("gopher"), ExpectXML: `<???>gopher</???>`}, +	{Value: "</>", ExpectXML: `<???></></???>`}, +	{Value: []byte("</>"), ExpectXML: `<???></></???>`}, +	{Value: [3]byte{'<', '/', '>'}, ExpectXML: `<???></></???>`}, +	{Value: NamedType("potato"), ExpectXML: `<???>potato</???>`}, +	{Value: []int{1, 2, 3}, ExpectXML: `<???>1</???><???>2</???><???>3</???>`}, +	{Value: [3]int{1, 2, 3}, ExpectXML: `<???>1</???><???>2</???><???>3</???>`}, + +	// Test innerxml +	{Value: RawXML("</>"), ExpectXML: `</>`}, +	{ +		Value: &SecretAgent{ +			Handle:    "007", +			Identity:  "James Bond", +			Obfuscate: "<redacted/>", +		}, +		//ExpectXML: `<agent handle="007"><redacted/></agent>`, +		ExpectXML: `<agent handle="007"><Identity>James Bond</Identity><redacted/></agent>`, +	}, + +	// 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: &Domain{Name: []byte("google.com&friends")}, ExpectXML: `<domain>google.com&friends</domain>`}, +	{Value: &Book{Title: "Pride & Prejudice"}, ExpectXML: `<book>Pride & Prejudice</book>`}, +	{Value: atomValue, ExpectXML: atomXml}, +	{ +		Value: &Ship{ +			Name:  "Heart of Gold", +			Pilot: "Computer", +			Age:   1, +			Drive: ImprobabilityDrive, +			Passenger: []*Passenger{ +				&Passenger{ +					Name:   []string{"Zaphod", "Beeblebrox"}, +					Weight: 7.25, +				}, +				&Passenger{ +					Name:   []string{"Trisha", "McMillen"}, +					Weight: 5.5, +				}, +				&Passenger{ +					Name:   []string{"Ford", "Prefect"}, +					Weight: 7, +				}, +				&Passenger{ +					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>`, +	}, +} + +func TestMarshal(t *testing.T) { +	for idx, test := range marshalTests { +		buf := bytes.NewBuffer(nil) +		err := Marshal(buf, test.Value) +		if err != nil { +			t.Errorf("#%d: Error: %s", idx, err) +			continue +		} +		if got, want := buf.String(), test.ExpectXML; got != want { +			if strings.Contains(want, "\n") { +				t.Errorf("#%d: marshal(%#v) - GOT:\n%s\nWANT:\n%s", idx, test.Value, got, want) +			} else { +				t.Errorf("#%d: marshal(%#v) = %#q want %#q", idx, test.Value, got, want) +			} +		} +	} +} + +var marshalErrorTests = []struct { +	Value      interface{} +	ExpectErr  string +	ExpectKind reflect.Kind +}{ +	{ +		Value:      make(chan bool), +		ExpectErr:  "xml: unsupported type: chan bool", +		ExpectKind: reflect.Chan, +	}, +	{ +		Value: map[string]string{ +			"question": "What do you get when you multiply six by nine?", +			"answer":   "42", +		}, +		ExpectErr:  "xml: unsupported type: map[string] string", +		ExpectKind: reflect.Map, +	}, +	{ +		Value:      map[*Ship]bool{nil: false}, +		ExpectErr:  "xml: unsupported type: map[*xml.Ship] bool", +		ExpectKind: reflect.Map, +	}, +} + +func TestMarshalErrors(t *testing.T) { +	for idx, test := range marshalErrorTests { +		buf := bytes.NewBuffer(nil) +		err := Marshal(buf, test.Value) +		if got, want := err, test.ExpectErr; got == nil { +			t.Errorf("#%d: want error %s", idx, want) +			continue +		} else if got.String() != want { +			t.Errorf("#%d: marshal(%#v) = [error] %q, want %q", idx, test.Value, got, want) +		} +		if got, want := err.(*UnsupportedTypeError).Type.Kind(), test.ExpectKind; got != want { +			t.Errorf("#%d: marshal(%#v) = [error kind] %s, want %s", idx, test.Value, got, want) +		} +	} +} + +// Do invertibility testing on the various structures that we test +func TestUnmarshal(t *testing.T) { +	for i, test := range marshalTests { +		// Skip the nil pointers +		if i <= 1 { +			continue +		} + +		var dest interface{} + +		switch test.Value.(type) { +		case *Ship, Ship: +			dest = &Ship{} +		case *Port, Port: +			dest = &Port{} +		case *Domain, Domain: +			dest = &Domain{} +		case *Feed, Feed: +			dest = &Feed{} +		default: +			continue +		} + +		buffer := bytes.NewBufferString(test.ExpectXML) +		err := Unmarshal(buffer, dest) + +		// Don't compare XMLNames +		switch fix := dest.(type) { +		case *Ship: +			fix.XMLName = Name{} +		case *Port: +			fix.XMLName = Name{} +		case *Domain: +			fix.XMLName = Name{} +		case *Feed: +			fix.XMLName = Name{} +			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(%#s) = %#v, want %#v", i, test.ExpectXML, got, want) +		} +	} +} + +func BenchmarkMarshal(b *testing.B) { +	idx := len(marshalTests) - 1 +	test := marshalTests[idx] + +	buf := bytes.NewBuffer(nil) +	for i := 0; i < b.N; i++ { +		Marshal(buf, test.Value) +		buf.Truncate(0) +	} +} + +func BenchmarkUnmarshal(b *testing.B) { +	idx := len(marshalTests) - 1 +	test := marshalTests[idx] +	sm := &Ship{} +	xml := []byte(test.ExpectXML) + +	for i := 0; i < b.N; i++ { +		buffer := bytes.NewBuffer(xml) +		Unmarshal(buffer, sm) +	} +} | 
