PNG  IHDRX cHRMz&u0`:pQ<bKGD pHYsodtIME MeqIDATxw]Wug^Qd˶ 6`!N:!@xI~)%7%@Bh&`lnjVF29gΨ4E$|>cɚ{gk= %,a KX%,a KX%,a KX%,a KX%,a KX%,a KX%, b` ǟzeאfp]<!SJmɤY޲ڿ,%c ~ع9VH.!Ͳz&QynֺTkRR.BLHi٪:l;@(!MԴ=žI,:o&N'Kù\vRmJ雵֫AWic H@" !: Cé||]k-Ha oݜ:y F())u]aG7*JV@J415p=sZH!=!DRʯvɱh~V\}v/GKY$n]"X"}t@ xS76^[bw4dsce)2dU0 CkMa-U5tvLƀ~mlMwfGE/-]7XAƟ`׮g ewxwC4\[~7@O-Q( a*XGƒ{ ՟}$_y3tĐƤatgvێi|K=uVyrŲlLӪuܿzwk$m87k( `múcE)"@rK( z4$D; 2kW=Xb$V[Ru819קR~qloѱDyįݎ*mxw]y5e4K@ЃI0A D@"BDk_)N\8͜9dz"fK0zɿvM /.:2O{ Nb=M=7>??Zuo32 DLD@D| &+֎C #B8ַ`bOb $D#ͮҪtx]%`ES`Ru[=¾!@Od37LJ0!OIR4m]GZRJu$‡c=%~s@6SKy?CeIh:[vR@Lh | (BhAMy=݃  G"'wzn޺~8ԽSh ~T*A:xR[ܹ?X[uKL_=fDȊ؂p0}7=D$Ekq!/t.*2ʼnDbŞ}DijYaȲ(""6HA;:LzxQ‘(SQQ}*PL*fc\s `/d'QXW, e`#kPGZuŞuO{{wm[&NBTiiI0bukcA9<4@SӊH*؎4U/'2U5.(9JuDfrޱtycU%j(:RUbArLֺN)udA':uGQN"-"Is.*+k@ `Ojs@yU/ H:l;@yyTn}_yw!VkRJ4P)~y#)r,D =ě"Q]ci'%HI4ZL0"MJy 8A{ aN<8D"1#IJi >XjX֔#@>-{vN!8tRݻ^)N_╗FJEk]CT՟ YP:_|H1@ CBk]yKYp|og?*dGvzنzӴzjֺNkC~AbZƷ`.H)=!QͷVTT(| u78y֮}|[8-Vjp%2JPk[}ԉaH8Wpqhwr:vWª<}l77_~{s۴V+RCģ%WRZ\AqHifɤL36: #F:p]Bq/z{0CU6ݳEv_^k7'>sq*+kH%a`0ԣisqにtү04gVgW΂iJiS'3w.w}l6MC2uԯ|>JF5`fV5m`Y**Db1FKNttu]4ccsQNnex/87+}xaUW9y>ͯ骵G{䩓Գ3+vU}~jJ.NFRD7<aJDB1#ҳgSb,+CS?/ VG J?|?,2#M9}B)MiE+G`-wo߫V`fio(}S^4e~V4bHOYb"b#E)dda:'?}׮4繏`{7Z"uny-?ǹ;0MKx{:_pÚmFמ:F " .LFQLG)Q8qN q¯¯3wOvxDb\. BKD9_NN &L:4D{mm o^tֽ:q!ƥ}K+<"m78N< ywsard5+вz~mnG)=}lYݧNj'QJS{S :UYS-952?&O-:W}(!6Mk4+>A>j+i|<<|;ر^߉=HE|V#F)Emm#}/"y GII웻Jі94+v뾧xu~5C95~ūH>c@덉pʃ1/4-A2G%7>m;–Y,cyyaln" ?ƻ!ʪ<{~h~i y.zZB̃/,雋SiC/JFMmBH&&FAbϓO^tubbb_hZ{_QZ-sύodFgO(6]TJA˯#`۶ɟ( %$&+V'~hiYy>922 Wp74Zkq+Ovn錄c>8~GqܲcWꂎz@"1A.}T)uiW4="jJ2W7mU/N0gcqܗOO}?9/wìXžΏ0 >֩(V^Rh32!Hj5`;O28؇2#ݕf3 ?sJd8NJ@7O0 b־?lldщ̡&|9C.8RTWwxWy46ah嘦mh٤&l zCy!PY?: CJyв]dm4ǜҐR޻RլhX{FƯanшQI@x' ao(kUUuxW_Ñ줮[w8 FRJ(8˼)_mQ _!RJhm=!cVmm ?sFOnll6Qk}alY}; "baӌ~M0w,Ggw2W:G/k2%R,_=u`WU R.9T"v,<\Ik޽/2110Ӿxc0gyC&Ny޽JҢrV6N ``یeA16"J³+Rj*;BϜkZPJaÍ<Jyw:NP8/D$ 011z֊Ⱳ3ι֘k1V_"h!JPIΣ'ɜ* aEAd:ݺ>y<}Lp&PlRfTb1]o .2EW\ͮ]38؋rTJsǏP@芎sF\> P^+dYJLbJ C-xϐn> ι$nj,;Ǖa FU *择|h ~izť3ᤓ`K'-f tL7JK+vf2)V'-sFuB4i+m+@My=O҈0"|Yxoj,3]:cо3 $#uŘ%Y"y죯LebqtҢVzq¼X)~>4L׶m~[1_k?kxֺQ`\ |ٛY4Ѯr!)N9{56(iNq}O()Em]=F&u?$HypWUeB\k]JɩSع9 Zqg4ZĊo oMcjZBU]B\TUd34ݝ~:7ڶSUsB0Z3srx 7`:5xcx !qZA!;%͚7&P H<WL!džOb5kF)xor^aujƍ7 Ǡ8/p^(L>ὴ-B,{ۇWzֺ^k]3\EE@7>lYBȝR.oHnXO/}sB|.i@ɥDB4tcm,@ӣgdtJ!lH$_vN166L__'Z)y&kH;:,Y7=J 9cG) V\hjiE;gya~%ks_nC~Er er)muuMg2;֫R)Md) ,¶ 2-wr#F7<-BBn~_(o=KO㭇[Xv eN_SMgSҐ BS헃D%g_N:/pe -wkG*9yYSZS.9cREL !k}<4_Xs#FmҶ:7R$i,fi!~' # !6/S6y@kZkZcX)%5V4P]VGYq%H1!;e1MV<!ϐHO021Dp= HMs~~a)ަu7G^];git!Frl]H/L$=AeUvZE4P\.,xi {-~p?2b#amXAHq)MWǾI_r`S Hz&|{ +ʖ_= (YS(_g0a03M`I&'9vl?MM+m~}*xT۲(fY*V4x@29s{DaY"toGNTO+xCAO~4Ϳ;p`Ѫ:>Ҵ7K 3}+0 387x\)a"/E>qpWB=1 ¨"MP(\xp߫́A3+J] n[ʼnӼaTbZUWb={~2ooKױӰp(CS\S筐R*JغV&&"FA}J>G֐p1ٸbk7 ŘH$JoN <8s^yk_[;gy-;߉DV{c B yce% aJhDȶ 2IdйIB/^n0tNtџdcKj4϶v~- CBcgqx9= PJ) dMsjpYB] GD4RDWX +h{y`,3ꊕ$`zj*N^TP4L:Iz9~6s) Ga:?y*J~?OrMwP\](21sZUD ?ܟQ5Q%ggW6QdO+\@ ̪X'GxN @'4=ˋ+*VwN ne_|(/BDfj5(Dq<*tNt1х!MV.C0 32b#?n0pzj#!38}޴o1KovCJ`8ŗ_"]] rDUy޲@ Ȗ-;xџ'^Y`zEd?0„ DAL18IS]VGq\4o !swV7ˣι%4FѮ~}6)OgS[~Q vcYbL!wG3 7띸*E Pql8=jT\꘿I(z<[6OrR8ºC~ډ]=rNl[g|v TMTղb-o}OrP^Q]<98S¤!k)G(Vkwyqyr޽Nv`N/e p/~NAOk \I:G6]4+K;j$R:Mi #*[AȚT,ʰ,;N{HZTGMoּy) ]%dHء9Պ䠬|<45,\=[bƟ8QXeB3- &dҩ^{>/86bXmZ]]yޚN[(WAHL$YAgDKp=5GHjU&99v簪C0vygln*P)9^͞}lMuiH!̍#DoRBn9l@ xA/_v=ȺT{7Yt2N"4!YN`ae >Q<XMydEB`VU}u]嫇.%e^ánE87Mu\t`cP=AD/G)sI"@MP;)]%fH9'FNsj1pVhY&9=0pfuJ&gޤx+k:!r˭wkl03׼Ku C &ѓYt{.O.zҏ z}/tf_wEp2gvX)GN#I ݭ߽v/ .& и(ZF{e"=V!{zW`, ]+LGz"(UJp|j( #V4, 8B 0 9OkRrlɱl94)'VH9=9W|>PS['G(*I1==C<5"Pg+x'K5EMd؞Af8lG ?D FtoB[je?{k3zQ vZ;%Ɠ,]E>KZ+T/ EJxOZ1i #T<@ I}q9/t'zi(EMqw`mYkU6;[t4DPeckeM;H}_g pMww}k6#H㶏+b8雡Sxp)&C $@'b,fPߑt$RbJ'vznuS ~8='72_`{q纶|Q)Xk}cPz9p7O:'|G~8wx(a 0QCko|0ASD>Ip=4Q, d|F8RcU"/KM opKle M3#i0c%<7׿p&pZq[TR"BpqauIp$ 8~Ĩ!8Սx\ւdT>>Z40ks7 z2IQ}ItԀ<-%S⍤};zIb$I 5K}Q͙D8UguWE$Jh )cu4N tZl+[]M4k8֦Zeq֮M7uIqG 1==tLtR,ƜSrHYt&QP윯Lg' I,3@P'}'R˪e/%-Auv·ñ\> vDJzlӾNv5:|K/Jb6KI9)Zh*ZAi`?S {aiVDԲuy5W7pWeQJk֤#5&V<̺@/GH?^τZL|IJNvI:'P=Ϛt"¨=cud S Q.Ki0 !cJy;LJR;G{BJy޺[^8fK6)=yʊ+(k|&xQ2`L?Ȓ2@Mf 0C`6-%pKpm')c$׻K5[J*U[/#hH!6acB JA _|uMvDyk y)6OPYjœ50VT K}cǻP[ $:]4MEA.y)|B)cf-A?(e|lɉ#P9V)[9t.EiQPDѠ3ϴ;E:+Օ t ȥ~|_N2,ZJLt4! %ա]u {+=p.GhNcŞQI?Nd'yeh n7zi1DB)1S | S#ًZs2|Ɛy$F SxeX{7Vl.Src3E℃Q>b6G ўYCmtկ~=K0f(=LrAS GN'ɹ9<\!a`)֕y[uՍ[09` 9 +57ts6}b4{oqd+J5fa/,97J#6yν99mRWxJyѡyu_TJc`~W>l^q#Ts#2"nD1%fS)FU w{ܯ R{ ˎ󅃏џDsZSQS;LV;7 Od1&1n$ N /.q3~eNɪ]E#oM~}v֯FڦwyZ=<<>Xo稯lfMFV6p02|*=tV!c~]fa5Y^Q_WN|Vs 0ҘދU97OI'N2'8N֭fgg-}V%y]U4 峧p*91#9U kCac_AFңĪy뚇Y_AiuYyTTYЗ-(!JFLt›17uTozc. S;7A&&<ԋ5y;Ro+:' *eYJkWR[@F %SHWP 72k4 qLd'J "zB6{AC0ƁA6U.'F3:Ȅ(9ΜL;D]m8ڥ9}dU "v!;*13Rg^fJyShyy5auA?ɩGHRjo^]׽S)Fm\toy 4WQS@mE#%5ʈfFYDX ~D5Ϡ9tE9So_aU4?Ѽm%&c{n>.KW1Tlb}:j uGi(JgcYj0qn+>) %\!4{LaJso d||u//P_y7iRJ߬nHOy) l+@$($VFIQ9%EeKʈU. ia&FY̒mZ=)+qqoQn >L!qCiDB;Y<%} OgBxB!ØuG)WG9y(Ą{_yesuZmZZey'Wg#C~1Cev@0D $a@˲(.._GimA:uyw֬%;@!JkQVM_Ow:P.s\)ot- ˹"`B,e CRtaEUP<0'}r3[>?G8xU~Nqu;Wm8\RIkբ^5@k+5(By'L&'gBJ3ݶ!/㮻w҅ yqPWUg<e"Qy*167΃sJ\oz]T*UQ<\FԎ`HaNmڜ6DysCask8wP8y9``GJ9lF\G g's Nn͵MLN֪u$| /|7=]O)6s !ĴAKh]q_ap $HH'\1jB^s\|- W1:=6lJBqjY^LsPk""`]w)󭃈,(HC ?䔨Y$Sʣ{4Z+0NvQkhol6C.婧/u]FwiVjZka&%6\F*Ny#8O,22+|Db~d ~Çwc N:FuuCe&oZ(l;@ee-+Wn`44AMK➝2BRՈt7g*1gph9N) *"TF*R(#'88pm=}X]u[i7bEc|\~EMn}P瘊J)K.0i1M6=7'_\kaZ(Th{K*GJyytw"IO-PWJk)..axӝ47"89Cc7ĐBiZx 7m!fy|ϿF9CbȩV 9V-՛^pV̌ɄS#Bv4-@]Vxt-Z, &ֺ*diؠ2^VXbs֔Ìl.jQ]Y[47gj=幽ex)A0ip׳ W2[ᎇhuE^~q흙L} #-b۸oFJ_QP3r6jr+"nfzRJTUqoaۍ /$d8Mx'ݓ= OՃ| )$2mcM*cЙj}f };n YG w0Ia!1Q.oYfr]DyISaP}"dIӗթO67jqR ҊƐƈaɤGG|h;t]䗖oSv|iZqX)oalv;۩meEJ\!8=$4QU4Xo&VEĊ YS^E#d,yX_> ۘ-e\ "Wa6uLĜZi`aD9.% w~mB(02G[6y.773a7 /=o7D)$Z 66 $bY^\CuP. (x'"J60׿Y:Oi;F{w佩b+\Yi`TDWa~|VH)8q/=9!g߆2Y)?ND)%?Ǐ`k/sn:;O299yB=a[Ng 3˲N}vLNy;*?x?~L&=xyӴ~}q{qE*IQ^^ͧvü{Huu=R|>JyUlZV, B~/YF!Y\u_ݼF{_C)LD]m {H 0ihhadd nUkf3oٺCvE\)QJi+֥@tDJkB$1!Đr0XQ|q?d2) Ӣ_}qv-< FŊ߫%roppVBwü~JidY4:}L6M7f٬F "?71<2#?Jyy4뷢<_a7_=Q E=S1И/9{+93֮E{ǂw{))?maÆm(uLE#lïZ  ~d];+]h j?!|$F}*"4(v'8s<ŏUkm7^7no1w2ؗ}TrͿEk>p'8OB7d7R(A 9.*Mi^ͳ; eeUwS+C)uO@ =Sy]` }l8^ZzRXj[^iUɺ$tj))<sbDJfg=Pk_{xaKo1:-uyG0M ԃ\0Lvuy'ȱc2Ji AdyVgVh!{]/&}}ċJ#%d !+87<;qN޼Nفl|1N:8ya  8}k¾+-$4FiZYÔXk*I&'@iI99)HSh4+2G:tGhS^繿 Kتm0 вDk}֚+QT4;sC}rՅE,8CX-e~>G&'9xpW,%Fh,Ry56Y–hW-(v_,? ; qrBk4-V7HQ;ˇ^Gv1JVV%,ik;D_W!))+BoS4QsTM;gt+ndS-~:11Sgv!0qRVh!"Ȋ(̦Yl.]PQWgٳE'`%W1{ndΗBk|Ž7ʒR~,lnoa&:ü$ 3<a[CBݮwt"o\ePJ=Hz"_c^Z.#ˆ*x z̝grY]tdkP*:97YľXyBkD4N.C_[;F9`8& !AMO c `@BA& Ost\-\NX+Xp < !bj3C&QL+*&kAQ=04}cC!9~820G'PC9xa!w&bo_1 Sw"ܱ V )Yl3+ס2KoXOx]"`^WOy :3GO0g;%Yv㐫(R/r (s } u B &FeYZh0y> =2<Ϟc/ -u= c&׭,.0"g"7 6T!vl#sc>{u/Oh Bᾈ)۴74]x7 gMӒ"d]U)}" v4co[ ɡs 5Gg=XR14?5A}D "b{0$L .\4y{_fe:kVS\\O]c^W52LSBDM! C3Dhr̦RtArx4&agaN3Cf<Ԉp4~ B'"1@.b_/xQ} _߃҉/gٓ2Qkqp0շpZ2fԫYz< 4L.Cyυι1t@鎫Fe sYfsF}^ V}N<_`p)alٶ "(XEAVZ<)2},:Ir*#m_YӼ R%a||EƼIJ,,+f"96r/}0jE/)s)cjW#w'Sʯ5<66lj$a~3Kʛy 2:cZ:Yh))+a߭K::N,Q F'qB]={.]h85C9cr=}*rk?vwV렵ٸW Rs%}rNAkDv|uFLBkWY YkX מ|)1!$#3%y?pF<@<Rr0}: }\J [5FRxY<9"SQdE(Q*Qʻ)q1E0B_O24[U'],lOb ]~WjHޏTQ5Syu wq)xnw8~)c 쫬gٲߠ H% k5dƝk> kEj,0% b"vi2Wس_CuK)K{n|>t{P1򨾜j>'kEkƗBg*H%'_aY6Bn!TL&ɌOb{c`'d^{t\i^[uɐ[}q0lM˕G:‚4kb祔c^:?bpg… +37stH:0}en6x˟%/<]BL&* 5&fK9Mq)/iyqtA%kUe[ڛKN]Ě^,"`/ s[EQQm?|XJ߅92m]G.E΃ח U*Cn.j_)Tѧj̿30ڇ!A0=͜ar I3$C^-9#|pk!)?7.x9 @OO;WƝZBFU keZ75F6Tc6"ZȚs2y/1 ʵ:u4xa`C>6Rb/Yм)^=+~uRd`/|_8xbB0?Ft||Z\##|K 0>>zxv8۴吅q 8ĥ)"6>~\8:qM}#͚'ĉ#p\׶ l#bA?)|g g9|8jP(cr,BwV (WliVxxᡁ@0Okn;ɥh$_ckCgriv}>=wGzβ KkBɛ[˪ !J)h&k2%07δt}!d<9;I&0wV/ v 0<H}L&8ob%Hi|޶o&h1L|u֦y~󛱢8fٲUsւ)0oiFx2}X[zVYr_;N(w]_4B@OanC?gĦx>мgx>ΛToZoOMp>40>V Oy V9iq!4 LN,ˢu{jsz]|"R޻&'ƚ{53ўFu(<٪9:΋]B;)B>1::8;~)Yt|0(pw2N%&X,URBK)3\zz&}ax4;ǟ(tLNg{N|Ǽ\G#C9g$^\}p?556]/RP.90 k,U8/u776s ʪ_01چ|\N 0VV*3H鴃J7iI!wG_^ypl}r*jɤSR 5QN@ iZ#1ٰy;_\3\BQQ x:WJv츟ٯ$"@6 S#qe딇(/P( Dy~TOϻ<4:-+F`0||;Xl-"uw$Цi󼕝mKʩorz"mϺ$F:~E'ҐvD\y?Rr8_He@ e~O,T.(ފR*cY^m|cVR[8 JҡSm!ΆԨb)RHG{?MpqrmN>߶Y)\p,d#xۆWY*,l6]v0h15M˙MS8+EdI='LBJIH7_9{Caз*Lq,dt >+~ّeʏ?xԕ4bBAŚjﵫ!'\Ը$WNvKO}ӽmSşذqsOy?\[,d@'73'j%kOe`1.g2"e =YIzS2|zŐƄa\U,dP;jhhhaxǶ?КZ՚.q SE+XrbOu%\GتX(H,N^~]JyEZQKceTQ]VGYqnah;y$cQahT&QPZ*iZ8UQQM.qo/T\7X"u?Mttl2Xq(IoW{R^ ux*SYJ! 4S.Jy~ BROS[V|žKNɛP(L6V^|cR7i7nZW1Fd@ Ara{詑|(T*dN]Ko?s=@ |_EvF]׍kR)eBJc" MUUbY6`~V޴dJKß&~'d3i WWWWWW
Current Directory: /opt/golang/1.22.0/src/encoding/xml
Viewing File: /opt/golang/1.22.0/src/encoding/xml/marshal_test.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 any `xml:"parent1>parent2>a"` B any `xml:"parent1>b"` C any `xml:"parent1>parent2>c"` } type Service struct { XMLName struct{} `xml:"service"` Domain *Domain `xml:"host>domain"` Port *Port `xml:"host>port"` Extra1 any Extra2 any `xml:"host>extra2"` } var nilStruct *Ship type EmbedA struct { EmbedC EmbedB EmbedB FieldA string embedD } type EmbedB struct { FieldB string *EmbedC } type EmbedC struct { FieldA1 string `xml:"FieldA>A1"` FieldA2 string `xml:"FieldA>A2"` FieldB string FieldC string } type embedD struct { fieldD string FieldE string // Promoted and visible when embedD is embedded. } 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 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 AttrsTest struct { Attrs []Attr `xml:",any,attr"` 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"` PStr *string `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"` PStr *string `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 any } 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 PointerAnonFields struct { *MyInt *NamedType } 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 } func (m *MyMarshalerAttrTest) UnmarshalXMLAttr(attr Attr) error { return nil } type MarshalerStruct struct { Foo MyMarshalerAttrTest `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 CDataTest struct { Chardata string `xml:",cdata"` } type NestedAndCData struct { AB []string `xml:"A>B"` CDATA string `xml:",cdata"` } func ifaceptr(x any) any { return &x } func stringptr(x string) *string { return &x } type T1 struct{} type T2 struct{} type IndirComment struct { T1 T1 Comment *string `xml:",comment"` T2 T2 } type DirectComment struct { T1 T1 Comment string `xml:",comment"` T2 T2 } type IfaceComment struct { T1 T1 Comment any `xml:",comment"` T2 T2 } type IndirChardata struct { T1 T1 Chardata *string `xml:",chardata"` T2 T2 } type DirectChardata struct { T1 T1 Chardata string `xml:",chardata"` T2 T2 } type IfaceChardata struct { T1 T1 Chardata any `xml:",chardata"` T2 T2 } type IndirCDATA struct { T1 T1 CDATA *string `xml:",cdata"` T2 T2 } type DirectCDATA struct { T1 T1 CDATA string `xml:",cdata"` T2 T2 } type IfaceCDATA struct { T1 T1 CDATA any `xml:",cdata"` T2 T2 } type IndirInnerXML struct { T1 T1 InnerXML *string `xml:",innerxml"` T2 T2 } type DirectInnerXML struct { T1 T1 InnerXML string `xml:",innerxml"` T2 T2 } type IfaceInnerXML struct { T1 T1 InnerXML any `xml:",innerxml"` T2 T2 } type IndirElement struct { T1 T1 Element *string T2 T2 } type DirectElement struct { T1 T1 Element string T2 T2 } type IfaceElement struct { T1 T1 Element any T2 T2 } type IndirOmitEmpty struct { T1 T1 OmitEmpty *string `xml:",omitempty"` T2 T2 } type DirectOmitEmpty struct { T1 T1 OmitEmpty string `xml:",omitempty"` T2 T2 } type IfaceOmitEmpty struct { T1 T1 OmitEmpty any `xml:",omitempty"` T2 T2 } type IndirAny struct { T1 T1 Any *string `xml:",any"` T2 T2 } type DirectAny struct { T1 T1 Any string `xml:",any"` T2 T2 } type IfaceAny struct { T1 T1 Any any `xml:",any"` T2 T2 } type Generic[T any] struct { X T } var ( nameAttr = "Sarah" ageAttr = uint(12) contentsAttr = "lorem ipsum" empty = "" ) // 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 // marshaling and unmarshaling are as symmetrical as feasible. var marshalTests = []struct { Value any ExpectXML string MarshalOnly bool MarshalError string UnmarshalOnly bool UnmarshalError string }{ // 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>&lt;/&gt;</V></Plain>`}, {Value: &Plain{[]byte("</>")}, ExpectXML: `<Plain><V>&lt;/&gt;</V></Plain>`}, {Value: &Plain{[3]byte{'<', '/', '>'}}, ExpectXML: `<Plain><V>&lt;/&gt;</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 []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="&lt;unix&gt;"></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&amp;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 &amp; 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: &Generic[int]{1}, ExpectXML: `<Generic><X>1</X></Generic>`}, { 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"` }{ A: "a", B: "b", C: "c", C1: "c1", D1: "d1", }, ExpectXML: `<top xmlns="space">` + `<x><a>a</a><b>b</b><c xmlns="space">c</c>` + `<c xmlns="space1">c1</c>` + `<d xmlns="space1">d1</d>` + `</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>` + `<c xmlns="space">c</c>` + `<c xmlns="space1">c1</c>` + `<d xmlns="space1">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><b xmlns="space">b</b>` + `<b xmlns="space1">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", embedD: embedD{ FieldE: "A.D.E", }, }, 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>` + `<FieldE>A.D.E</FieldE>` + `</EmbedA>`, }, // Anonymous struct pointer field which is nil { Value: &EmbedB{}, ExpectXML: `<EmbedB><FieldB></FieldB></EmbedB>`, }, // Other kinds of nil anonymous fields { Value: &PointerAnonFields{}, ExpectXML: `<PointerAnonFields></PointerAnonFields>`, }, // 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: &AttrsTest{ Attrs: []Attr{ {Name: Name{Local: "Answer"}, Value: "42"}, {Name: Name{Local: "Int"}, Value: "8"}, {Name: Name{Local: "int"}, Value: "9"}, {Name: Name{Local: "Float"}, Value: "23.5"}, {Name: Name{Local: "Uint8"}, Value: "255"}, {Name: Name{Local: "Bool"}, Value: "true"}, {Name: Name{Local: "Str"}, Value: "str"}, {Name: Name{Local: "Bytes"}, Value: "byt"}, }, }, ExpectXML: `<AttrsTest Answer="42" Int="8" int="9" Float="23.5" Uint8="255" Bool="true" Str="str" Bytes="byt" Int="0" int="0" Float="0" Uint8="0" Bool="false" Str="" Bytes=""></AttrsTest>`, MarshalOnly: true, }, { Value: &AttrsTest{ Attrs: []Attr{ {Name: Name{Local: "Answer"}, Value: "42"}, }, Int: 8, Named: 9, Float: 23.5, Uint8: 255, Bool: true, Str: "str", Bytes: []byte("byt"), }, ExpectXML: `<AttrsTest Answer="42" Int="8" int="9" Float="23.5" Uint8="255" Bool="true" Str="str" Bytes="byt"></AttrsTest>`, }, { Value: &AttrsTest{ Attrs: []Attr{ {Name: Name{Local: "Int"}, Value: "0"}, {Name: Name{Local: "int"}, Value: "0"}, {Name: Name{Local: "Float"}, Value: "0"}, {Name: Name{Local: "Uint8"}, Value: "0"}, {Name: Name{Local: "Bool"}, Value: "false"}, {Name: Name{Local: "Str"}}, {Name: Name{Local: "Bytes"}}, }, Bytes: []byte{}, }, ExpectXML: `<AttrsTest Int="0" int="0" Float="0" Uint8="0" Bool="false" Str="" Bytes="" Int="0" int="0" Float="0" Uint8="0" Bool="false" Str="" Bytes=""></AttrsTest>`, MarshalOnly: true, }, { Value: &OmitAttrTest{ Int: 8, Named: 9, Float: 23.5, Uint8: 255, Bool: true, Str: "str", Bytes: []byte("byt"), PStr: &empty, }, ExpectXML: `<OmitAttrTest Int="8" int="9" Float="23.5" Uint8="255"` + ` Bool="true" Str="str" Bytes="byt" PStr=""></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"), PStr: &empty, 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>` + `<PStr></PStr>` + `<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: &#34;; squote: &#39;; ampersand: &amp;; less: &lt;; greater: &gt;;</value></nested><empty></empty></a>`, Value: &AnyTest{ Nested: `dquote: "; squote: '; ampersand: &; less: <; greater: >;`, AnyField: AnyHolder{XMLName: Name{Local: "empty"}}, }, }, { ExpectXML: `<a><nested><value>newline: &#xA;; cr: &#xD;; tab: &#x9;;</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 outputting CDATA-wrapped text. { ExpectXML: `<CDataTest></CDataTest>`, Value: &CDataTest{}, }, { ExpectXML: `<CDataTest><![CDATA[http://example.com/tests/1?foo=1&bar=baz]]></CDataTest>`, Value: &CDataTest{ Chardata: "http://example.com/tests/1?foo=1&bar=baz", }, }, { ExpectXML: `<CDataTest><![CDATA[Literal <![CDATA[Nested]]]]><![CDATA[>!]]></CDataTest>`, Value: &CDataTest{ Chardata: "Literal <![CDATA[Nested]]>!", }, }, { ExpectXML: `<CDataTest><![CDATA[<![CDATA[Nested]]]]><![CDATA[> Literal!]]></CDataTest>`, Value: &CDataTest{ Chardata: "<![CDATA[Nested]]> Literal!", }, }, { ExpectXML: `<CDataTest><![CDATA[<![CDATA[Nested]]]]><![CDATA[> Literal! <![CDATA[Nested]]]]><![CDATA[> Literal!]]></CDataTest>`, Value: &CDataTest{ Chardata: "<![CDATA[Nested]]> Literal! <![CDATA[Nested]]> Literal!", }, }, { ExpectXML: `<CDataTest><![CDATA[<![CDATA[<![CDATA[Nested]]]]><![CDATA[>]]]]><![CDATA[>]]></CDataTest>`, Value: &CDataTest{ Chardata: "<![CDATA[<![CDATA[Nested]]>]]>", }, }, // 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: `<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: `<NestedAndCData><A><B></B><B></B></A><![CDATA[test]]></NestedAndCData>`, Value: &NestedAndCData{AB: make([]string, 2), CDATA: "test"}, }, // Test pointer indirection in various kinds of fields. // https://golang.org/issue/19063 { ExpectXML: `<IndirComment><T1></T1><!--hi--><T2></T2></IndirComment>`, Value: &IndirComment{Comment: stringptr("hi")}, MarshalOnly: true, }, { ExpectXML: `<IndirComment><T1></T1><T2></T2></IndirComment>`, Value: &IndirComment{Comment: stringptr("")}, MarshalOnly: true, }, { ExpectXML: `<IndirComment><T1></T1><T2></T2></IndirComment>`, Value: &IndirComment{Comment: nil}, MarshalError: "xml: bad type for comment field of xml.IndirComment", }, { ExpectXML: `<IndirComment><T1></T1><!--hi--><T2></T2></IndirComment>`, Value: &IndirComment{Comment: nil}, UnmarshalOnly: true, }, { ExpectXML: `<IfaceComment><T1></T1><!--hi--><T2></T2></IfaceComment>`, Value: &IfaceComment{Comment: "hi"}, MarshalOnly: true, }, { ExpectXML: `<IfaceComment><T1></T1><!--hi--><T2></T2></IfaceComment>`, Value: &IfaceComment{Comment: nil}, UnmarshalOnly: true, }, { ExpectXML: `<IfaceComment><T1></T1><T2></T2></IfaceComment>`, Value: &IfaceComment{Comment: nil}, MarshalError: "xml: bad type for comment field of xml.IfaceComment", }, { ExpectXML: `<IfaceComment><T1></T1><T2></T2></IfaceComment>`, Value: &IfaceComment{Comment: nil}, UnmarshalOnly: true, }, { ExpectXML: `<DirectComment><T1></T1><!--hi--><T2></T2></DirectComment>`, Value: &DirectComment{Comment: string("hi")}, }, { ExpectXML: `<DirectComment><T1></T1><T2></T2></DirectComment>`, Value: &DirectComment{Comment: string("")}, }, { ExpectXML: `<IndirChardata><T1></T1>hi<T2></T2></IndirChardata>`, Value: &IndirChardata{Chardata: stringptr("hi")}, }, { ExpectXML: `<IndirChardata><T1></T1><![CDATA[hi]]><T2></T2></IndirChardata>`, Value: &IndirChardata{Chardata: stringptr("hi")}, UnmarshalOnly: true, // marshals without CDATA }, { ExpectXML: `<IndirChardata><T1></T1><T2></T2></IndirChardata>`, Value: &IndirChardata{Chardata: stringptr("")}, }, { ExpectXML: `<IndirChardata><T1></T1><T2></T2></IndirChardata>`, Value: &IndirChardata{Chardata: nil}, MarshalOnly: true, // unmarshal leaves Chardata=stringptr("") }, { ExpectXML: `<IfaceChardata><T1></T1>hi<T2></T2></IfaceChardata>`, Value: &IfaceChardata{Chardata: string("hi")}, UnmarshalError: "cannot unmarshal into interface {}", }, { ExpectXML: `<IfaceChardata><T1></T1><![CDATA[hi]]><T2></T2></IfaceChardata>`, Value: &IfaceChardata{Chardata: string("hi")}, UnmarshalOnly: true, // marshals without CDATA UnmarshalError: "cannot unmarshal into interface {}", }, { ExpectXML: `<IfaceChardata><T1></T1><T2></T2></IfaceChardata>`, Value: &IfaceChardata{Chardata: string("")}, UnmarshalError: "cannot unmarshal into interface {}", }, { ExpectXML: `<IfaceChardata><T1></T1><T2></T2></IfaceChardata>`, Value: &IfaceChardata{Chardata: nil}, UnmarshalError: "cannot unmarshal into interface {}", }, { ExpectXML: `<DirectChardata><T1></T1>hi<T2></T2></DirectChardata>`, Value: &DirectChardata{Chardata: string("hi")}, }, { ExpectXML: `<DirectChardata><T1></T1><![CDATA[hi]]><T2></T2></DirectChardata>`, Value: &DirectChardata{Chardata: string("hi")}, UnmarshalOnly: true, // marshals without CDATA }, { ExpectXML: `<DirectChardata><T1></T1><T2></T2></DirectChardata>`, Value: &DirectChardata{Chardata: string("")}, }, { ExpectXML: `<IndirCDATA><T1></T1><![CDATA[hi]]><T2></T2></IndirCDATA>`, Value: &IndirCDATA{CDATA: stringptr("hi")}, }, { ExpectXML: `<IndirCDATA><T1></T1>hi<T2></T2></IndirCDATA>`, Value: &IndirCDATA{CDATA: stringptr("hi")}, UnmarshalOnly: true, // marshals with CDATA }, { ExpectXML: `<IndirCDATA><T1></T1><T2></T2></IndirCDATA>`, Value: &IndirCDATA{CDATA: stringptr("")}, }, { ExpectXML: `<IndirCDATA><T1></T1><T2></T2></IndirCDATA>`, Value: &IndirCDATA{CDATA: nil}, MarshalOnly: true, // unmarshal leaves CDATA=stringptr("") }, { ExpectXML: `<IfaceCDATA><T1></T1><![CDATA[hi]]><T2></T2></IfaceCDATA>`, Value: &IfaceCDATA{CDATA: string("hi")}, UnmarshalError: "cannot unmarshal into interface {}", }, { ExpectXML: `<IfaceCDATA><T1></T1>hi<T2></T2></IfaceCDATA>`, Value: &IfaceCDATA{CDATA: string("hi")}, UnmarshalOnly: true, // marshals with CDATA UnmarshalError: "cannot unmarshal into interface {}", }, { ExpectXML: `<IfaceCDATA><T1></T1><T2></T2></IfaceCDATA>`, Value: &IfaceCDATA{CDATA: string("")}, UnmarshalError: "cannot unmarshal into interface {}", }, { ExpectXML: `<IfaceCDATA><T1></T1><T2></T2></IfaceCDATA>`, Value: &IfaceCDATA{CDATA: nil}, UnmarshalError: "cannot unmarshal into interface {}", }, { ExpectXML: `<DirectCDATA><T1></T1><![CDATA[hi]]><T2></T2></DirectCDATA>`, Value: &DirectCDATA{CDATA: string("hi")}, }, { ExpectXML: `<DirectCDATA><T1></T1>hi<T2></T2></DirectCDATA>`, Value: &DirectCDATA{CDATA: string("hi")}, UnmarshalOnly: true, // marshals with CDATA }, { ExpectXML: `<DirectCDATA><T1></T1><T2></T2></DirectCDATA>`, Value: &DirectCDATA{CDATA: string("")}, }, { ExpectXML: `<IndirInnerXML><T1></T1><hi/><T2></T2></IndirInnerXML>`, Value: &IndirInnerXML{InnerXML: stringptr("<hi/>")}, MarshalOnly: true, }, { ExpectXML: `<IndirInnerXML><T1></T1><T2></T2></IndirInnerXML>`, Value: &IndirInnerXML{InnerXML: stringptr("")}, MarshalOnly: true, }, { ExpectXML: `<IndirInnerXML><T1></T1><T2></T2></IndirInnerXML>`, Value: &IndirInnerXML{InnerXML: nil}, }, { ExpectXML: `<IndirInnerXML><T1></T1><hi/><T2></T2></IndirInnerXML>`, Value: &IndirInnerXML{InnerXML: nil}, UnmarshalOnly: true, }, { ExpectXML: `<IfaceInnerXML><T1></T1><hi/><T2></T2></IfaceInnerXML>`, Value: &IfaceInnerXML{InnerXML: "<hi/>"}, MarshalOnly: true, }, { ExpectXML: `<IfaceInnerXML><T1></T1><hi/><T2></T2></IfaceInnerXML>`, Value: &IfaceInnerXML{InnerXML: nil}, UnmarshalOnly: true, }, { ExpectXML: `<IfaceInnerXML><T1></T1><T2></T2></IfaceInnerXML>`, Value: &IfaceInnerXML{InnerXML: nil}, }, { ExpectXML: `<IfaceInnerXML><T1></T1><T2></T2></IfaceInnerXML>`, Value: &IfaceInnerXML{InnerXML: nil}, UnmarshalOnly: true, }, { ExpectXML: `<DirectInnerXML><T1></T1><hi/><T2></T2></DirectInnerXML>`, Value: &DirectInnerXML{InnerXML: string("<hi/>")}, MarshalOnly: true, }, { ExpectXML: `<DirectInnerXML><T1></T1><hi/><T2></T2></DirectInnerXML>`, Value: &DirectInnerXML{InnerXML: string("<T1></T1><hi/><T2></T2>")}, UnmarshalOnly: true, }, { ExpectXML: `<DirectInnerXML><T1></T1><T2></T2></DirectInnerXML>`, Value: &DirectInnerXML{InnerXML: string("")}, MarshalOnly: true, }, { ExpectXML: `<DirectInnerXML><T1></T1><T2></T2></DirectInnerXML>`, Value: &DirectInnerXML{InnerXML: string("<T1></T1><T2></T2>")}, UnmarshalOnly: true, }, { ExpectXML: `<IndirElement><T1></T1><Element>hi</Element><T2></T2></IndirElement>`, Value: &IndirElement{Element: stringptr("hi")}, }, { ExpectXML: `<IndirElement><T1></T1><Element></Element><T2></T2></IndirElement>`, Value: &IndirElement{Element: stringptr("")}, }, { ExpectXML: `<IndirElement><T1></T1><T2></T2></IndirElement>`, Value: &IndirElement{Element: nil}, }, { ExpectXML: `<IfaceElement><T1></T1><Element>hi</Element><T2></T2></IfaceElement>`, Value: &IfaceElement{Element: "hi"}, MarshalOnly: true, }, { ExpectXML: `<IfaceElement><T1></T1><Element>hi</Element><T2></T2></IfaceElement>`, Value: &IfaceElement{Element: nil}, UnmarshalOnly: true, }, { ExpectXML: `<IfaceElement><T1></T1><T2></T2></IfaceElement>`, Value: &IfaceElement{Element: nil}, }, { ExpectXML: `<IfaceElement><T1></T1><T2></T2></IfaceElement>`, Value: &IfaceElement{Element: nil}, UnmarshalOnly: true, }, { ExpectXML: `<DirectElement><T1></T1><Element>hi</Element><T2></T2></DirectElement>`, Value: &DirectElement{Element: string("hi")}, }, { ExpectXML: `<DirectElement><T1></T1><Element></Element><T2></T2></DirectElement>`, Value: &DirectElement{Element: string("")}, }, { ExpectXML: `<IndirOmitEmpty><T1></T1><OmitEmpty>hi</OmitEmpty><T2></T2></IndirOmitEmpty>`, Value: &IndirOmitEmpty{OmitEmpty: stringptr("hi")}, }, { // Note: Changed in Go 1.8 to include <OmitEmpty> element (because x.OmitEmpty != nil). ExpectXML: `<IndirOmitEmpty><T1></T1><OmitEmpty></OmitEmpty><T2></T2></IndirOmitEmpty>`, Value: &IndirOmitEmpty{OmitEmpty: stringptr("")}, MarshalOnly: true, }, { ExpectXML: `<IndirOmitEmpty><T1></T1><OmitEmpty></OmitEmpty><T2></T2></IndirOmitEmpty>`, Value: &IndirOmitEmpty{OmitEmpty: stringptr("")}, UnmarshalOnly: true, }, { ExpectXML: `<IndirOmitEmpty><T1></T1><T2></T2></IndirOmitEmpty>`, Value: &IndirOmitEmpty{OmitEmpty: nil}, }, { ExpectXML: `<IfaceOmitEmpty><T1></T1><OmitEmpty>hi</OmitEmpty><T2></T2></IfaceOmitEmpty>`, Value: &IfaceOmitEmpty{OmitEmpty: "hi"}, MarshalOnly: true, }, { ExpectXML: `<IfaceOmitEmpty><T1></T1><OmitEmpty>hi</OmitEmpty><T2></T2></IfaceOmitEmpty>`, Value: &IfaceOmitEmpty{OmitEmpty: nil}, UnmarshalOnly: true, }, { ExpectXML: `<IfaceOmitEmpty><T1></T1><T2></T2></IfaceOmitEmpty>`, Value: &IfaceOmitEmpty{OmitEmpty: nil}, }, { ExpectXML: `<IfaceOmitEmpty><T1></T1><T2></T2></IfaceOmitEmpty>`, Value: &IfaceOmitEmpty{OmitEmpty: nil}, UnmarshalOnly: true, }, { ExpectXML: `<DirectOmitEmpty><T1></T1><OmitEmpty>hi</OmitEmpty><T2></T2></DirectOmitEmpty>`, Value: &DirectOmitEmpty{OmitEmpty: string("hi")}, }, { ExpectXML: `<DirectOmitEmpty><T1></T1><T2></T2></DirectOmitEmpty>`, Value: &DirectOmitEmpty{OmitEmpty: string("")}, }, { ExpectXML: `<IndirAny><T1></T1><Any>hi</Any><T2></T2></IndirAny>`, Value: &IndirAny{Any: stringptr("hi")}, }, { ExpectXML: `<IndirAny><T1></T1><Any></Any><T2></T2></IndirAny>`, Value: &IndirAny{Any: stringptr("")}, }, { ExpectXML: `<IndirAny><T1></T1><T2></T2></IndirAny>`, Value: &IndirAny{Any: nil}, }, { ExpectXML: `<IfaceAny><T1></T1><Any>hi</Any><T2></T2></IfaceAny>`, Value: &IfaceAny{Any: "hi"}, MarshalOnly: true, }, { ExpectXML: `<IfaceAny><T1></T1><Any>hi</Any><T2></T2></IfaceAny>`, Value: &IfaceAny{Any: nil}, UnmarshalOnly: true, }, { ExpectXML: `<IfaceAny><T1></T1><T2></T2></IfaceAny>`, Value: &IfaceAny{Any: nil}, }, { ExpectXML: `<IfaceAny><T1></T1><T2></T2></IfaceAny>`, Value: &IfaceAny{Any: nil}, UnmarshalOnly: true, }, { ExpectXML: `<DirectAny><T1></T1><Any>hi</Any><T2></T2></DirectAny>`, Value: &DirectAny{Any: string("hi")}, }, { ExpectXML: `<DirectAny><T1></T1><Any></Any><T2></T2></DirectAny>`, Value: &DirectAny{Any: string("")}, }, { ExpectXML: `<IndirFoo><T1></T1><Foo>hi</Foo><T2></T2></IndirFoo>`, Value: &IndirAny{Any: stringptr("hi")}, UnmarshalOnly: true, }, { ExpectXML: `<IndirFoo><T1></T1><Foo></Foo><T2></T2></IndirFoo>`, Value: &IndirAny{Any: stringptr("")}, UnmarshalOnly: true, }, { ExpectXML: `<IndirFoo><T1></T1><T2></T2></IndirFoo>`, Value: &IndirAny{Any: nil}, UnmarshalOnly: true, }, { ExpectXML: `<IfaceFoo><T1></T1><Foo>hi</Foo><T2></T2></IfaceFoo>`, Value: &IfaceAny{Any: nil}, UnmarshalOnly: true, }, { ExpectXML: `<IfaceFoo><T1></T1><T2></T2></IfaceFoo>`, Value: &IfaceAny{Any: nil}, UnmarshalOnly: true, }, { ExpectXML: `<IfaceFoo><T1></T1><T2></T2></IfaceFoo>`, Value: &IfaceAny{Any: nil}, UnmarshalOnly: true, }, { ExpectXML: `<DirectFoo><T1></T1><Foo>hi</Foo><T2></T2></DirectFoo>`, Value: &DirectAny{Any: string("hi")}, UnmarshalOnly: true, }, { ExpectXML: `<DirectFoo><T1></T1><Foo></Foo><T2></T2></DirectFoo>`, Value: &DirectAny{Any: string("")}, UnmarshalOnly: true, }, } func TestMarshal(t *testing.T) { for idx, test := range marshalTests { if test.UnmarshalOnly { continue } t.Run(fmt.Sprintf("%d", idx), func(t *testing.T) { data, err := Marshal(test.Value) if err != nil { if test.MarshalError == "" { t.Errorf("marshal(%#v): %s", test.Value, err) return } if !strings.Contains(err.Error(), test.MarshalError) { t.Errorf("marshal(%#v): %s, want %q", test.Value, err, test.MarshalError) } return } if test.MarshalError != "" { t.Errorf("Marshal succeeded, want error %q", test.MarshalError) return } if got, want := string(data), test.ExpectXML; got != want { if strings.Contains(want, "\n") { t.Errorf("marshal(%#v):\nHAVE:\n%s\nWANT:\n%s", test.Value, got, want) } else { t.Errorf("marshal(%#v):\nhave %#q\nwant %#q", test.Value, got, want) } } }) } } type AttrParent struct { X string `xml:"X>Y,attr"` } type BadAttr struct { Name map[string]string `xml:"name,attr"` } var marshalErrorTests = []struct { Value any 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{map[string]string{"X": "Y"}}, Err: `xml: unsupported type: map[string]string`, }, } var marshalIndentTests = []struct { Value any 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 } if test.ExpectXML == `<top>`+ `<x><b xmlns="space">b</b>`+ `<b xmlns="space1">b1</b></x>`+ `</top>` { // TODO(rogpeppe): re-enable this test in // https://go-review.googlesource.com/#/c/5910/ continue } vt := reflect.TypeOf(test.Value) dest := reflect.New(vt.Elem()).Interface() err := Unmarshal([]byte(test.ExpectXML), dest) t.Run(fmt.Sprintf("%d", i), func(t *testing.T) { switch fix := dest.(type) { case *Feed: fix.Author.InnerXML = "" for i := range fix.Entry { fix.Entry[i].Author.InnerXML = "" } } if err != nil { if test.UnmarshalError == "" { t.Errorf("unmarshal(%#v): %s", test.ExpectXML, err) return } if !strings.Contains(err.Error(), test.UnmarshalError) { t.Errorf("unmarshal(%#v): %s, want %q", test.ExpectXML, err, test.UnmarshalError) } return } if got, want := dest, test.Value; !reflect.DeepEqual(got, want) { t.Errorf("unmarshal(%q):\nhave %#v\nwant %#v", 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 strings.Builder 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.String()) } 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") } } func BenchmarkMarshal(b *testing.B) { b.ReportAllocs() b.RunParallel(func(pb *testing.PB) { for pb.Next() { Marshal(atomValue) } }) } func BenchmarkUnmarshal(b *testing.B) { b.ReportAllocs() xml := []byte(atomXML) b.RunParallel(func(pb *testing.PB) { for pb.Next() { Unmarshal(xml, &Feed{}) } }) } // golang.org/issue/6556 func TestStructPointerMarshal(t *testing.T) { type A struct { XMLName string `xml:"a"` B []any } 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: `<local xmlns="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: " &#x9;\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: `<foo xmlns="space">`, }, { desc: "start element with explicit namespace", toks: []Token{ StartElement{Name{"space", "local"}, []Attr{ {Name{"xmlns", "x"}, "space"}, {Name{"space", "foo"}, "value"}, }}, }, want: `<local xmlns="space" xmlns:_xmlns="xmlns" _xmlns:x="space" xmlns:space="space" space: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: `<local xmlns="space" xmlns:_xmlns="xmlns" _xmlns:x="space" xmlns:space="space" space:foo="value" xmlns:x="x" x: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:_xmlns="xmlns" _xmlns:x="space"><foo xmlns="space" xmlns:space="space" space: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:_xmlns="xmlns" _xmlns:x="space1"><foo _xmlns:x="space2"><foo xmlns:space1="space1" space1:a="space1 value" xmlns:space2="space2" space2:b="space2 value"></foo></foo><foo xmlns:space1="space1" space1:a="space1 value" xmlns:space2="space2" 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: `<foo xmlns="space" xmlns:_xmlns="xmlns" _xmlns:a="space" _xmlns:b="space" xmlns:space="space" space: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:_xmlns="xmlns" _xmlns:x="space"><foo xmlns="space" _xmlns:y="space" xmlns:space="space" space: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" xmlns="space"><foo xmlns="space" xmlns:_xmlns="xmlns" _xmlns:y="space" xmlns:space="space" space: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:_xmlns="xmlns" _xmlns:x="space"><foo xmlns="space" xmlns="space" xmlns:space="space" space: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" xmlns="space"><foo xmlns="space" attr="value">`, }, { desc: "redefine xmlns", toks: []Token{ StartElement{Name{"", "foo"}, []Attr{ {Name{"foo", "xmlns"}, "space"}, }}, }, want: `<foo xmlns:foo="foo" foo:xmlns="space">`, }, { desc: "xmlns with explicit name space #1", toks: []Token{ StartElement{Name{"space", "foo"}, []Attr{ {Name{"xml", "xmlns"}, "space"}, }}, }, want: `<foo xmlns="space" xmlns:_xml="xml" _xml:xmlns="space">`, }, { desc: "xmlns with explicit name space #2", toks: []Token{ StartElement{Name{"space", "foo"}, []Attr{ {Name{xmlURL, "xmlns"}, "space"}, }}, }, want: `<foo xmlns="space" xml:xmlns="space">`, }, { desc: "empty name space declaration is ignored", toks: []Token{ StartElement{Name{"", "foo"}, []Attr{ {Name{"xmlns", "foo"}, ""}, }}, }, want: `<foo xmlns:_xmlns="xmlns" _xmlns: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" 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" xmlns="space"><foo xmlns="" x="value" xmlns:space="space" 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" xmlns="space"><foo>`, }, { 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: `<foo xmlns="some/space" attr="value" xmlns:space="some/space" 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="space" xmlns="space" xmlns:_xmlns="xmlns" _xmlns:bar="space" xmlns:space="space" space:baz="foo"><baz xmlns="space"></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" xmlns="space" xmlns:space="space" space:baz="foo"><baz xmlns="space"></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 xmlns="space"><bar xmlns="space" xmlns:space="space" space:attr="value">`, }, { desc: "reserved namespace prefix -- all lower case", toks: []Token{ StartElement{Name{"", "foo"}, []Attr{ {Name{"http://www.w3.org/2001/xmlSchema-instance", "nil"}, "true"}, }}, }, want: `<foo xmlns:_xmlSchema-instance="http://www.w3.org/2001/xmlSchema-instance" _xmlSchema-instance:nil="true">`, }, { desc: "reserved namespace prefix -- all upper case", toks: []Token{ StartElement{Name{"", "foo"}, []Attr{ {Name{"http://www.w3.org/2001/XMLSchema-instance", "nil"}, "true"}, }}, }, want: `<foo xmlns:_XMLSchema-instance="http://www.w3.org/2001/XMLSchema-instance" _XMLSchema-instance:nil="true">`, }, { desc: "reserved namespace prefix -- all mixed case", toks: []Token{ StartElement{Name{"", "foo"}, []Attr{ {Name{"http://www.w3.org/2001/XmLSchema-instance", "nil"}, "true"}, }}, }, want: `<foo xmlns:_XmLSchema-instance="http://www.w3.org/2001/XmLSchema-instance" _XmLSchema-instance:nil="true">`, }} func TestEncodeToken(t *testing.T) { loop: for i, tt := range encodeTokenTests { var buf strings.Builder 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 ...any) { 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 strings.Builder 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()) } } // Issue 16158. Decoder.unmarshalAttr ignores the return value of copyValue. func TestIssue16158(t *testing.T) { const data = `<foo b="HELLOWORLD"></foo>` err := Unmarshal([]byte(data), &struct { B byte `xml:"b,attr,omitempty"` }{}) if err == nil { t.Errorf("Unmarshal: expected error, got nil") } } // Issue 20953. Crash on invalid XMLName attribute. type InvalidXMLName struct { XMLName Name `xml:"error"` Type struct { XMLName Name `xml:"type,attr"` } } func TestInvalidXMLName(t *testing.T) { var buf bytes.Buffer enc := NewEncoder(&buf) if err := enc.Encode(InvalidXMLName{}); err == nil { t.Error("unexpected success") } else if want := "invalid tag"; !strings.Contains(err.Error(), want) { t.Errorf("error %q does not contain %q", err, want) } } // Issue 50164. Crash on zero value XML attribute. type LayerOne struct { XMLName Name `xml:"l1"` Value *float64 `xml:"value,omitempty"` *LayerTwo `xml:",omitempty"` } type LayerTwo struct { ValueTwo *int `xml:"value_two,attr,omitempty"` } func TestMarshalZeroValue(t *testing.T) { proofXml := `<l1><value>1.2345</value></l1>` var l1 LayerOne err := Unmarshal([]byte(proofXml), &l1) if err != nil { t.Fatalf("unmarshal XML error: %v", err) } want := float64(1.2345) got := *l1.Value if got != want { t.Fatalf("unexpected unmarshal result, want %f but got %f", want, got) } // Marshal again (or Encode again) // In issue 50164, here `Marshal(l1)` will panic because of the zero value of xml attribute ValueTwo `value_two`. anotherXML, err := Marshal(l1) if err != nil { t.Fatalf("marshal XML error: %v", err) } if string(anotherXML) != proofXml { t.Fatalf("unexpected unmarshal result, want %q but got %q", proofXml, anotherXML) } } var closeTests = []struct { desc string toks []Token want string err string }{{ desc: "unclosed start element", toks: []Token{ StartElement{Name{"", "foo"}, nil}, }, want: `<foo>`, err: "unclosed tag <foo>", }, { desc: "closed element", toks: []Token{ StartElement{Name{"", "foo"}, nil}, EndElement{Name{"", "foo"}}, }, want: `<foo></foo>`, }, { desc: "directive", toks: []Token{ Directive("foo"), }, want: `<!foo>`, }} func TestClose(t *testing.T) { for _, tt := range closeTests { tt := tt t.Run(tt.desc, func(t *testing.T) { var out strings.Builder enc := NewEncoder(&out) for j, tok := range tt.toks { if err := enc.EncodeToken(tok); err != nil { t.Fatalf("token #%d: %v", j, err) } } err := enc.Close() switch { case tt.err != "" && err == nil: t.Error(" expected error; got none") case tt.err == "" && err != nil: t.Errorf(" got error: %v", err) case tt.err != "" && err != nil && tt.err != err.Error(): t.Errorf(" error mismatch; got %v, want %v", err, tt.err) } if got := out.String(); got != tt.want { t.Errorf("\ngot %v\nwant %v", got, tt.want) } t.Log(enc.p.closed) if err := enc.EncodeToken(Directive("foo")); err == nil { t.Errorf("unexpected success when encoding after Close") } }) } }