2010-12-03 05:34:57 +01:00
// Copyright 2009 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 (
2013-11-06 20:49:01 +01:00
"io"
2010-12-03 05:34:57 +01:00
"reflect"
2013-07-16 08:54:42 +02:00
"strings"
2010-12-03 05:34:57 +01:00
"testing"
2012-03-02 17:38:43 +01:00
"time"
2010-12-03 05:34:57 +01:00
)
// Stripped down Atom feed data structures.
func TestUnmarshalFeed ( t * testing . T ) {
var f Feed
2012-02-01 20:26:59 +01:00
if err := Unmarshal ( [ ] byte ( atomFeedString ) , & f ) ; err != nil {
2010-12-03 05:34:57 +01:00
t . Fatalf ( "Unmarshal: %s" , err )
}
2011-03-17 00:05:44 +01:00
if ! reflect . DeepEqual ( f , atomFeed ) {
t . Fatalf ( "have %#v\nwant %#v" , f , atomFeed )
2010-12-03 05:34:57 +01:00
}
}
// hget http://codereview.appspot.com/rss/mine/rsc
2011-03-17 00:05:44 +01:00
const atomFeedString = `
2010-12-03 05:34:57 +01:00
< ? xml version = "1.0" encoding = "utf-8" ? >
2012-03-02 17:38:43 +01:00
< feed xmlns = "http://www.w3.org/2005/Atom" xml : lang = "en-us" updated = "2009-10-04T01:35:58+00:00" > < title > Code Review - My issues < / title > < link href = "http://codereview.appspot.com/" rel = "alternate" > < / link > < link href = "http://codereview.appspot.com/rss/mine/rsc" rel = "self" > < / link > < id > http : //codereview.appspot.com/</id><author><name>rietveld<></name></author><entry><title>rietveld: an attempt at pubsubhubbub
2012-01-25 21:56:26 +01:00
< / title > < link href = "http://codereview.appspot.com/126085" rel = "alternate" > < / link > < updated > 2009 - 10 - 04 T01 : 35 : 58 + 00 : 00 < / updated > < author > < name > email - address - removed < / name > < / author > < id > urn : md5 : 134 d9179c41f806be79b3a5f7877d19a < / id > < summary type = "html" >
2010-12-03 05:34:57 +01:00
An attempt at adding pubsubhubbub support to Rietveld .
http : //code.google.com/p/pubsubhubbub
http : //code.google.com/p/rietveld/issues/detail?id=155
The server side of the protocol is trivial :
1. add a & amp ; lt ; link rel = & amp ; quot ; hub & amp ; quot ; href = & amp ; quot ; hub - server & amp ; quot ; & amp ; gt ; tag to all
feeds that will be pubsubhubbubbed .
2. every time one of those feeds changes , tell the hub
with a simple POST request .
I have tested this by adding debug prints to a local hub
server and checking that the server got the right publish
requests .
I can & amp ; # 39 ; t quite get the server to work , but I think the bug
is not in my code . I think that the server expects to be
able to grab the feed and see the feed & amp ; # 39 ; s actual URL in
the link rel = & amp ; quot ; self & amp ; quot ; , but the default value for that drops
the : port from the URL , and I cannot for the life of me
figure out how to get the Atom generator deep inside
django not to do that , or even where it is doing that ,
or even what code is running to generate the Atom feed .
( I thought I knew but I added some assert False statements
and it kept running ! )
Ignoring that particular problem , I would appreciate
feedback on the right way to get the two values at
the top of feeds . py marked NOTE ( rsc ) .
< / summary > < / entry > < entry > < title > rietveld : correct tab handling
< / title > < link href = "http://codereview.appspot.com/124106" rel = "alternate" > < / link > < updated > 2009 - 10 - 03 T23 : 02 : 17 + 00 : 00 < / updated > < author > < name > email - address - removed < / name > < / author > < id > urn : md5 : 0 a2a4f19bb815101f0ba2904aed7c35a < / id > < summary type = "html" >
This fixes the buggy tab rendering that can be seen at
http : //codereview.appspot.com/116075/diff/1/2
The fundamental problem was that the tab code was
not being told what column the text began in , so it
didn & amp ; # 39 ; t know where to put the tab stops . Another problem
was that some of the code assumed that string byte
offsets were the same as column offsets , which is only
true if there are no tabs .
In the process of fixing this , I cleaned up the arguments
to Fold and ExpandTabs and renamed them Break and
_ExpandTabs so that I could be sure that I found all the
call sites . I also wanted to verify that ExpandTabs was
not being used from outside intra_region_diff . py .
< / summary > < / entry > < / feed > `
type Feed struct {
2012-03-02 17:38:43 +01:00
XMLName Name ` xml:"http://www.w3.org/2005/Atom feed" `
Title string ` xml:"title" `
Id string ` xml:"id" `
Link [ ] Link ` xml:"link" `
Updated time . Time ` xml:"updated,attr" `
Author Person ` xml:"author" `
Entry [ ] Entry ` xml:"entry" `
2010-12-03 05:34:57 +01:00
}
type Entry struct {
2012-03-02 17:38:43 +01:00
Title string ` xml:"title" `
Id string ` xml:"id" `
Link [ ] Link ` xml:"link" `
Updated time . Time ` xml:"updated" `
Author Person ` xml:"author" `
Summary Text ` xml:"summary" `
2010-12-03 05:34:57 +01:00
}
type Link struct {
2012-03-02 17:38:43 +01:00
Rel string ` xml:"rel,attr,omitempty" `
2012-01-25 21:56:26 +01:00
Href string ` xml:"href,attr" `
2010-12-03 05:34:57 +01:00
}
type Person struct {
2012-01-25 21:56:26 +01:00
Name string ` xml:"name" `
URI string ` xml:"uri" `
Email string ` xml:"email" `
InnerXML string ` xml:",innerxml" `
2010-12-03 05:34:57 +01:00
}
type Text struct {
2012-03-02 17:38:43 +01:00
Type string ` xml:"type,attr,omitempty" `
2012-01-25 21:56:26 +01:00
Body string ` xml:",chardata" `
2010-12-03 05:34:57 +01:00
}
2011-03-17 00:05:44 +01:00
var atomFeed = Feed {
2010-12-03 05:34:57 +01:00
XMLName : Name { "http://www.w3.org/2005/Atom" , "feed" } ,
Title : "Code Review - My issues" ,
Link : [ ] Link {
{ Rel : "alternate" , Href : "http://codereview.appspot.com/" } ,
{ Rel : "self" , Href : "http://codereview.appspot.com/rss/mine/rsc" } ,
} ,
Id : "http://codereview.appspot.com/" ,
2012-03-02 17:38:43 +01:00
Updated : ParseTime ( "2009-10-04T01:35:58+00:00" ) ,
2010-12-03 05:34:57 +01:00
Author : Person {
Name : "rietveld<>" ,
InnerXML : "<name>rietveld<></name>" ,
} ,
Entry : [ ] Entry {
{
Title : "rietveld: an attempt at pubsubhubbub\n" ,
Link : [ ] Link {
{ Rel : "alternate" , Href : "http://codereview.appspot.com/126085" } ,
} ,
2012-03-02 17:38:43 +01:00
Updated : ParseTime ( "2009-10-04T01:35:58+00:00" ) ,
2010-12-03 05:34:57 +01:00
Author : Person {
Name : "email-address-removed" ,
InnerXML : "<name>email-address-removed</name>" ,
} ,
Id : "urn:md5:134d9179c41f806be79b3a5f7877d19a" ,
Summary : Text {
Type : "html" ,
Body : `
An attempt at adding pubsubhubbub support to Rietveld .
http : //code.google.com/p/pubsubhubbub
http : //code.google.com/p/rietveld/issues/detail?id=155
The server side of the protocol is trivial :
1. add a & lt ; link rel = & quot ; hub & quot ; href = & quot ; hub - server & quot ; & gt ; tag to all
feeds that will be pubsubhubbubbed .
2. every time one of those feeds changes , tell the hub
with a simple POST request .
I have tested this by adding debug prints to a local hub
server and checking that the server got the right publish
requests .
I can & # 39 ; t quite get the server to work , but I think the bug
is not in my code . I think that the server expects to be
able to grab the feed and see the feed & # 39 ; s actual URL in
the link rel = & quot ; self & quot ; , but the default value for that drops
the : port from the URL , and I cannot for the life of me
figure out how to get the Atom generator deep inside
django not to do that , or even where it is doing that ,
or even what code is running to generate the Atom feed .
( I thought I knew but I added some assert False statements
and it kept running ! )
Ignoring that particular problem , I would appreciate
feedback on the right way to get the two values at
the top of feeds . py marked NOTE ( rsc ) .
` ,
} ,
} ,
{
Title : "rietveld: correct tab handling\n" ,
Link : [ ] Link {
{ Rel : "alternate" , Href : "http://codereview.appspot.com/124106" } ,
} ,
2012-03-02 17:38:43 +01:00
Updated : ParseTime ( "2009-10-03T23:02:17+00:00" ) ,
2010-12-03 05:34:57 +01:00
Author : Person {
Name : "email-address-removed" ,
InnerXML : "<name>email-address-removed</name>" ,
} ,
Id : "urn:md5:0a2a4f19bb815101f0ba2904aed7c35a" ,
Summary : Text {
Type : "html" ,
Body : `
This fixes the buggy tab rendering that can be seen at
http : //codereview.appspot.com/116075/diff/1/2
The fundamental problem was that the tab code was
not being told what column the text began in , so it
didn & # 39 ; t know where to put the tab stops . Another problem
was that some of the code assumed that string byte
offsets were the same as column offsets , which is only
true if there are no tabs .
In the process of fixing this , I cleaned up the arguments
to Fold and ExpandTabs and renamed them Break and
_ExpandTabs so that I could be sure that I found all the
call sites . I also wanted to verify that ExpandTabs was
not being used from outside intra_region_diff . py .
` ,
} ,
} ,
} ,
}
2011-01-21 19:19:03 +01:00
const pathTestString = `
2012-01-25 21:56:26 +01:00
< Result >
< Before > 1 < / Before >
< Items >
< Item1 >
< Value > A < / Value >
< / Item1 >
< Item2 >
< Value > B < / Value >
< / Item2 >
2011-01-21 19:19:03 +01:00
< Item1 >
< Value > C < / Value >
< Value > D < / Value >
< / Item1 >
2011-12-07 02:11:29 +01:00
< _ >
2012-01-25 21:56:26 +01:00
< Value > E < / Value >
2011-12-07 02:11:29 +01:00
< / _ >
2012-01-25 21:56:26 +01:00
< / Items >
< After > 2 < / After >
< / Result >
2011-01-21 19:19:03 +01:00
`
type PathTestItem struct {
Value string
}
type PathTestA struct {
2012-01-25 21:56:26 +01:00
Items [ ] PathTestItem ` xml:">Item1" `
2011-01-21 19:19:03 +01:00
Before , After string
}
type PathTestB struct {
2012-01-25 21:56:26 +01:00
Other [ ] PathTestItem ` xml:"Items>Item1" `
2011-01-21 19:19:03 +01:00
Before , After string
}
type PathTestC struct {
2012-01-25 21:56:26 +01:00
Values1 [ ] string ` xml:"Items>Item1>Value" `
Values2 [ ] string ` xml:"Items>Item2>Value" `
2011-01-21 19:19:03 +01:00
Before , After string
}
type PathTestSet struct {
Item1 [ ] PathTestItem
}
type PathTestD struct {
2012-01-25 21:56:26 +01:00
Other PathTestSet ` xml:"Items" `
2011-01-21 19:19:03 +01:00
Before , After string
}
2011-12-07 02:11:29 +01:00
type PathTestE struct {
2012-01-25 21:56:26 +01:00
Underline string ` xml:"Items>_>Value" `
2011-12-07 02:11:29 +01:00
Before , After string
}
2011-01-21 19:19:03 +01:00
var pathTests = [ ] interface { } {
& PathTestA { Items : [ ] PathTestItem { { "A" } , { "D" } } , Before : "1" , After : "2" } ,
& PathTestB { Other : [ ] PathTestItem { { "A" } , { "D" } } , Before : "1" , After : "2" } ,
& PathTestC { Values1 : [ ] string { "A" , "C" , "D" } , Values2 : [ ] string { "B" } , Before : "1" , After : "2" } ,
& PathTestD { Other : PathTestSet { Item1 : [ ] PathTestItem { { "A" } , { "D" } } } , Before : "1" , After : "2" } ,
2011-12-07 02:11:29 +01:00
& PathTestE { Underline : "E" , Before : "1" , After : "2" } ,
2011-01-21 19:19:03 +01:00
}
func TestUnmarshalPaths ( t * testing . T ) {
for _ , pt := range pathTests {
2011-05-20 02:18:15 +02:00
v := reflect . New ( reflect . TypeOf ( pt ) . Elem ( ) ) . Interface ( )
2012-02-01 20:26:59 +01:00
if err := Unmarshal ( [ ] byte ( pathTestString ) , v ) ; err != nil {
2011-01-21 19:19:03 +01:00
t . Fatalf ( "Unmarshal: %s" , err )
}
if ! reflect . DeepEqual ( v , pt ) {
t . Fatalf ( "have %#v\nwant %#v" , v , pt )
}
}
}
type BadPathTestA struct {
2011-09-16 17:47:21 +02:00
First string ` xml:"items>item1" `
Other string ` xml:"items>item2" `
2012-01-25 21:56:26 +01:00
Second string ` xml:"items" `
2011-01-21 19:19:03 +01:00
}
type BadPathTestB struct {
2011-09-16 17:47:21 +02:00
Other string ` xml:"items>item2>value" `
First string ` xml:"items>item1" `
Second string ` xml:"items>item1>value" `
2011-01-21 19:19:03 +01:00
}
2012-01-25 21:56:26 +01:00
type BadPathTestC struct {
First string
Second string ` xml:"First" `
}
type BadPathTestD struct {
BadPathEmbeddedA
BadPathEmbeddedB
}
type BadPathEmbeddedA struct {
First string
}
type BadPathEmbeddedB struct {
Second string ` xml:"First" `
}
2011-01-21 19:19:03 +01:00
var badPathTests = [ ] struct {
v , e interface { }
} {
2012-01-25 21:56:26 +01:00
{ & BadPathTestA { } , & TagPathError { reflect . TypeOf ( BadPathTestA { } ) , "First" , "items>item1" , "Second" , "items" } } ,
2011-05-20 02:18:15 +02:00
{ & BadPathTestB { } , & TagPathError { reflect . TypeOf ( BadPathTestB { } ) , "First" , "items>item1" , "Second" , "items>item1>value" } } ,
2012-01-25 21:56:26 +01:00
{ & BadPathTestC { } , & TagPathError { reflect . TypeOf ( BadPathTestC { } ) , "First" , "" , "Second" , "First" } } ,
{ & BadPathTestD { } , & TagPathError { reflect . TypeOf ( BadPathTestD { } ) , "First" , "" , "Second" , "First" } } ,
2011-01-21 19:19:03 +01:00
}
func TestUnmarshalBadPaths ( t * testing . T ) {
for _ , tt := range badPathTests {
2012-02-01 20:26:59 +01:00
err := Unmarshal ( [ ] byte ( pathTestString ) , tt . v )
2011-01-21 19:19:03 +01:00
if ! reflect . DeepEqual ( err , tt . e ) {
2012-01-25 21:56:26 +01:00
t . Fatalf ( "Unmarshal with %#v didn't fail properly:\nhave %#v,\nwant %#v" , tt . v , err , tt . e )
2011-01-21 19:19:03 +01:00
}
}
}
2011-09-16 17:47:21 +02:00
2011-10-27 01:57:58 +02:00
const OK = "OK"
const withoutNameTypeData = `
< ? xml version = "1.0" charset = "utf-8" ? >
2012-01-25 21:56:26 +01:00
< Test3 Attr = "OK" / > `
2011-10-27 01:57:58 +02:00
type TestThree struct {
2012-01-25 21:56:26 +01:00
XMLName Name ` xml:"Test3" `
Attr string ` xml:",attr" `
2011-10-27 01:57:58 +02:00
}
func TestUnmarshalWithoutNameType ( t * testing . T ) {
var x TestThree
2012-02-01 20:26:59 +01:00
if err := Unmarshal ( [ ] byte ( withoutNameTypeData ) , & x ) ; err != nil {
2011-10-27 01:57:58 +02:00
t . Fatalf ( "Unmarshal: %s" , err )
}
if x . Attr != OK {
t . Fatalf ( "have %v\nwant %v" , x . Attr , OK )
}
}
2013-01-29 21:52:43 +01:00
func TestUnmarshalAttr ( t * testing . T ) {
type ParamVal struct {
Int int ` xml:"int,attr" `
}
type ParamPtr struct {
Int * int ` xml:"int,attr" `
}
type ParamStringPtr struct {
Int * string ` xml:"int,attr" `
}
x := [ ] byte ( ` <Param int="1" /> ` )
p1 := & ParamPtr { }
if err := Unmarshal ( x , p1 ) ; err != nil {
t . Fatalf ( "Unmarshal: %s" , err )
}
if p1 . Int == nil {
t . Fatalf ( "Unmarshal failed in to *int field" )
} else if * p1 . Int != 1 {
t . Fatalf ( "Unmarshal with %s failed:\nhave %#v,\n want %#v" , x , p1 . Int , 1 )
}
p2 := & ParamVal { }
if err := Unmarshal ( x , p2 ) ; err != nil {
t . Fatalf ( "Unmarshal: %s" , err )
}
if p2 . Int != 1 {
t . Fatalf ( "Unmarshal with %s failed:\nhave %#v,\n want %#v" , x , p2 . Int , 1 )
}
p3 := & ParamStringPtr { }
if err := Unmarshal ( x , p3 ) ; err != nil {
t . Fatalf ( "Unmarshal: %s" , err )
}
if p3 . Int == nil {
t . Fatalf ( "Unmarshal failed in to *string field" )
} else if * p3 . Int != "1" {
t . Fatalf ( "Unmarshal with %s failed:\nhave %#v,\n want %#v" , x , p3 . Int , 1 )
}
}
2013-07-16 08:54:42 +02:00
type Tables struct {
HTable string ` xml:"http://www.w3.org/TR/html4/ table" `
FTable string ` xml:"http://www.w3schools.com/furniture table" `
}
var tables = [ ] struct {
xml string
tab Tables
ns string
} {
{
xml : ` <Tables> ` +
` <table xmlns="http://www.w3.org/TR/html4/">hello</table> ` +
` <table xmlns="http://www.w3schools.com/furniture">world</table> ` +
` </Tables> ` ,
tab : Tables { "hello" , "world" } ,
} ,
{
xml : ` <Tables> ` +
` <table xmlns="http://www.w3schools.com/furniture">world</table> ` +
` <table xmlns="http://www.w3.org/TR/html4/">hello</table> ` +
` </Tables> ` ,
tab : Tables { "hello" , "world" } ,
} ,
{
xml : ` <Tables xmlns:f="http://www.w3schools.com/furniture" xmlns:h="http://www.w3.org/TR/html4/"> ` +
` <f:table>world</f:table> ` +
` <h:table>hello</h:table> ` +
` </Tables> ` ,
tab : Tables { "hello" , "world" } ,
} ,
{
xml : ` <Tables> ` +
` <table>bogus</table> ` +
` </Tables> ` ,
tab : Tables { } ,
} ,
{
xml : ` <Tables> ` +
` <table>only</table> ` +
` </Tables> ` ,
tab : Tables { HTable : "only" } ,
ns : "http://www.w3.org/TR/html4/" ,
} ,
{
xml : ` <Tables> ` +
` <table>only</table> ` +
` </Tables> ` ,
tab : Tables { FTable : "only" } ,
ns : "http://www.w3schools.com/furniture" ,
} ,
{
xml : ` <Tables> ` +
` <table>only</table> ` +
` </Tables> ` ,
tab : Tables { } ,
ns : "something else entirely" ,
} ,
}
func TestUnmarshalNS ( t * testing . T ) {
for i , tt := range tables {
var dst Tables
var err error
if tt . ns != "" {
d := NewDecoder ( strings . NewReader ( tt . xml ) )
d . DefaultSpace = tt . ns
err = d . Decode ( & dst )
} else {
err = Unmarshal ( [ ] byte ( tt . xml ) , & dst )
}
if err != nil {
t . Errorf ( "#%d: Unmarshal: %v" , i , err )
continue
}
want := tt . tab
if dst != want {
t . Errorf ( "#%d: dst=%+v, want %+v" , i , dst , want )
}
}
}
func TestMarshalNS ( t * testing . T ) {
dst := Tables { "hello" , "world" }
data , err := Marshal ( & dst )
if err != nil {
t . Fatalf ( "Marshal: %v" , err )
}
want := ` <Tables><table xmlns="http://www.w3.org/TR/html4/">hello</table><table xmlns="http://www.w3schools.com/furniture">world</table></Tables> `
str := string ( data )
if str != want {
t . Errorf ( "have: %q\nwant: %q\n" , str , want )
}
}
type TableAttrs struct {
TAttr TAttr
}
type TAttr struct {
HTable string ` xml:"http://www.w3.org/TR/html4/ table,attr" `
FTable string ` xml:"http://www.w3schools.com/furniture table,attr" `
Lang string ` xml:"http://www.w3.org/XML/1998/namespace lang,attr,omitempty" `
Other1 string ` xml:"http://golang.org/xml/ other,attr,omitempty" `
Other2 string ` xml:"http://golang.org/xmlfoo/ other,attr,omitempty" `
Other3 string ` xml:"http://golang.org/json/ other,attr,omitempty" `
Other4 string ` xml:"http://golang.org/2/json/ other,attr,omitempty" `
}
var tableAttrs = [ ] struct {
xml string
tab TableAttrs
ns string
} {
{
xml : ` <TableAttrs xmlns:f="http://www.w3schools.com/furniture" xmlns:h="http://www.w3.org/TR/html4/"><TAttr ` +
` h:table="hello" f:table="world" ` +
` /></TableAttrs> ` ,
tab : TableAttrs { TAttr { HTable : "hello" , FTable : "world" } } ,
} ,
{
xml : ` <TableAttrs><TAttr xmlns:f="http://www.w3schools.com/furniture" xmlns:h="http://www.w3.org/TR/html4/" ` +
` h:table="hello" f:table="world" ` +
` /></TableAttrs> ` ,
tab : TableAttrs { TAttr { HTable : "hello" , FTable : "world" } } ,
} ,
{
xml : ` <TableAttrs><TAttr ` +
` h:table="hello" f:table="world" xmlns:f="http://www.w3schools.com/furniture" xmlns:h="http://www.w3.org/TR/html4/" ` +
` /></TableAttrs> ` ,
tab : TableAttrs { TAttr { HTable : "hello" , FTable : "world" } } ,
} ,
{
// Default space does not apply to attribute names.
xml : ` <TableAttrs xmlns="http://www.w3schools.com/furniture" xmlns:h="http://www.w3.org/TR/html4/"><TAttr ` +
` h:table="hello" table="world" ` +
` /></TableAttrs> ` ,
tab : TableAttrs { TAttr { HTable : "hello" , FTable : "" } } ,
} ,
{
// Default space does not apply to attribute names.
xml : ` <TableAttrs xmlns:f="http://www.w3schools.com/furniture"><TAttr xmlns="http://www.w3.org/TR/html4/" ` +
` table="hello" f:table="world" ` +
` /></TableAttrs> ` ,
tab : TableAttrs { TAttr { HTable : "" , FTable : "world" } } ,
} ,
{
xml : ` <TableAttrs><TAttr ` +
` table="bogus" ` +
` /></TableAttrs> ` ,
tab : TableAttrs { } ,
} ,
{
// Default space does not apply to attribute names.
xml : ` <TableAttrs xmlns:h="http://www.w3.org/TR/html4/"><TAttr ` +
` h:table="hello" table="world" ` +
` /></TableAttrs> ` ,
tab : TableAttrs { TAttr { HTable : "hello" , FTable : "" } } ,
ns : "http://www.w3schools.com/furniture" ,
} ,
{
// Default space does not apply to attribute names.
xml : ` <TableAttrs xmlns:f="http://www.w3schools.com/furniture"><TAttr ` +
` table="hello" f:table="world" ` +
` /></TableAttrs> ` ,
tab : TableAttrs { TAttr { HTable : "" , FTable : "world" } } ,
ns : "http://www.w3.org/TR/html4/" ,
} ,
{
xml : ` <TableAttrs><TAttr ` +
` table="bogus" ` +
` /></TableAttrs> ` ,
tab : TableAttrs { } ,
ns : "something else entirely" ,
} ,
}
func TestUnmarshalNSAttr ( t * testing . T ) {
for i , tt := range tableAttrs {
var dst TableAttrs
var err error
if tt . ns != "" {
d := NewDecoder ( strings . NewReader ( tt . xml ) )
d . DefaultSpace = tt . ns
err = d . Decode ( & dst )
} else {
err = Unmarshal ( [ ] byte ( tt . xml ) , & dst )
}
if err != nil {
t . Errorf ( "#%d: Unmarshal: %v" , i , err )
continue
}
want := tt . tab
if dst != want {
t . Errorf ( "#%d: dst=%+v, want %+v" , i , dst , want )
}
}
}
func TestMarshalNSAttr ( t * testing . T ) {
src := TableAttrs { TAttr { "hello" , "world" , "en_US" , "other1" , "other2" , "other3" , "other4" } }
data , err := Marshal ( & src )
if err != nil {
t . Fatalf ( "Marshal: %v" , err )
}
want := ` <TableAttrs><TAttr xmlns:html4="http://www.w3.org/TR/html4/" html4:table="hello" xmlns:furniture="http://www.w3schools.com/furniture" furniture:table="world" xml:lang="en_US" xmlns:_xml="http://golang.org/xml/" _xml:other="other1" xmlns:_xmlfoo="http://golang.org/xmlfoo/" _xmlfoo:other="other2" xmlns:json="http://golang.org/json/" json:other="other3" xmlns:json_1="http://golang.org/2/json/" json_1:other="other4"></TAttr></TableAttrs> `
str := string ( data )
if str != want {
t . Errorf ( "Marshal:\nhave: %#q\nwant: %#q\n" , str , want )
}
var dst TableAttrs
if err := Unmarshal ( data , & dst ) ; err != nil {
t . Errorf ( "Unmarshal: %v" , err )
}
if dst != src {
t . Errorf ( "Unmarshal = %q, want %q" , dst , src )
}
}
2013-11-06 20:49:01 +01:00
type MyCharData struct {
body string
}
func ( m * MyCharData ) UnmarshalXML ( d * Decoder , start StartElement ) error {
for {
t , err := d . Token ( )
if err == io . EOF { // found end of element
break
}
if err != nil {
return err
}
if char , ok := t . ( CharData ) ; ok {
m . body += string ( char )
}
}
return nil
}
var _ Unmarshaler = ( * MyCharData ) ( nil )
func ( m * MyCharData ) UnmarshalXMLAttr ( attr Attr ) error {
panic ( "must not call" )
}
type MyAttr struct {
attr string
}
func ( m * MyAttr ) UnmarshalXMLAttr ( attr Attr ) error {
m . attr = attr . Value
return nil
}
var _ UnmarshalerAttr = ( * MyAttr ) ( nil )
type MyStruct struct {
Data * MyCharData
Attr * MyAttr ` xml:",attr" `
Data2 MyCharData
Attr2 MyAttr ` xml:",attr" `
}
func TestUnmarshaler ( t * testing . T ) {
xml := ` < ? xml version = "1.0" encoding = "utf-8" ? >
< MyStruct Attr = "attr1" Attr2 = "attr2" >
< Data > hello < ! -- comment -- > world < / Data >
< Data2 > howdy < ! -- comment -- > world < / Data2 >
< / MyStruct >
`
var m MyStruct
if err := Unmarshal ( [ ] byte ( xml ) , & m ) ; err != nil {
t . Fatal ( err )
}
if m . Data == nil || m . Attr == nil || m . Data . body != "hello world" || m . Attr . attr != "attr1" || m . Data2 . body != "howdy world" || m . Attr2 . attr != "attr2" {
t . Errorf ( "m=%#+v\n" , m )
}
}
2014-07-19 10:53:52 +02:00
type Pea struct {
Cotelydon string
}
type Pod struct {
Pea interface { } ` xml:"Pea" `
}
2015-10-31 01:59:47 +01:00
// https://golang.org/issue/6836
2014-07-19 10:53:52 +02:00
func TestUnmarshalIntoInterface ( t * testing . T ) {
pod := new ( Pod )
pod . Pea = new ( Pea )
xml := ` <Pod><Pea><Cotelydon>Green stuff</Cotelydon></Pea></Pod> `
err := Unmarshal ( [ ] byte ( xml ) , pod )
if err != nil {
t . Fatalf ( "failed to unmarshal %q: %v" , xml , err )
}
pea , ok := pod . Pea . ( * Pea )
if ! ok {
t . Fatalf ( "unmarshalled into wrong type: have %T want *Pea" , pod . Pea )
}
have , want := pea . Cotelydon , "Green stuff"
if have != want {
t . Errorf ( "failed to unmarshal into interface, have %q want %q" , have , want )
}
}
2016-02-03 22:58:02 +01:00
type X struct {
D string ` xml:",comment" `
}
// Issue 11112. Unmarshal must reject invalid comments.
func TestMalformedComment ( t * testing . T ) {
testData := [ ] string {
"<X><!-- a---></X>" ,
"<X><!-- -- --></X>" ,
"<X><!-- a--b --></X>" ,
"<X><!------></X>" ,
}
for i , test := range testData {
data := [ ] byte ( test )
v := new ( X )
if err := Unmarshal ( data , v ) ; err == nil {
t . Errorf ( "%d: unmarshal should reject invalid comments" , i )
}
}
}