ws2_32: name is never NULL as array (Coverity).
[wine] / dlls / msxml3 / tests / domdoc.c
1 /*
2  * XML test
3  *
4  * Copyright 2005 Mike McCormack for CodeWeavers
5  * Copyright 2007-2008 Alistair Leslie-Hughes
6  * Copyright 2010 Adam Martinson for CodeWeavers
7  *
8  * This library is free software; you can redistribute it and/or
9  * modify it under the terms of the GNU Lesser General Public
10  * License as published by the Free Software Foundation; either
11  * version 2.1 of the License, or (at your option) any later version.
12  *
13  * This library is distributed in the hope that it will be useful,
14  * but WITHOUT ANY WARRANTY; without even the implied warranty of
15  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
16  * Lesser General Public License for more details.
17  *
18  * You should have received a copy of the GNU Lesser General Public
19  * License along with this library; if not, write to the Free Software
20  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
21  */
22
23
24 #define COBJMACROS
25
26 #include "windows.h"
27 #include "ole2.h"
28 #include "objsafe.h"
29 #include "msxml2.h"
30 #include "msxml2did.h"
31 #include "dispex.h"
32 #include <stdio.h>
33 #include <assert.h>
34
35 #include "wine/test.h"
36
37 #include "initguid.h"
38
39 DEFINE_GUID(IID_IObjectSafety, 0xcb5bdc81, 0x93c1, 0x11cf, 0x8f,0x20, 0x00,0x80,0x5f,0x2c,0xd0,0x64);
40
41 static int g_unexpectedcall, g_expectedcall;
42
43 typedef struct
44 {
45     const struct IDispatchVtbl *lpVtbl;
46     LONG ref;
47 } dispevent;
48
49 static inline dispevent *impl_from_IDispatch( IDispatch *iface )
50 {
51     return (dispevent *)((char*)iface - FIELD_OFFSET(dispevent, lpVtbl));
52 }
53
54 static HRESULT WINAPI dispevent_QueryInterface(IDispatch *iface, REFIID riid, void **ppvObject)
55 {
56     *ppvObject = NULL;
57
58     if ( IsEqualGUID( riid, &IID_IDispatch) ||
59          IsEqualGUID( riid, &IID_IUnknown) )
60     {
61         *ppvObject = iface;
62     }
63     else
64         return E_NOINTERFACE;
65
66     IDispatch_AddRef( iface );
67
68     return S_OK;
69 }
70
71 static ULONG WINAPI dispevent_AddRef(IDispatch *iface)
72 {
73     dispevent *This = impl_from_IDispatch( iface );
74     return InterlockedIncrement( &This->ref );
75 }
76
77 static ULONG WINAPI dispevent_Release(IDispatch *iface)
78 {
79     dispevent *This = impl_from_IDispatch( iface );
80     ULONG ref = InterlockedDecrement( &This->ref );
81
82     if (ref == 0)
83         HeapFree(GetProcessHeap(), 0, This);
84
85     return ref;
86 }
87
88 static HRESULT WINAPI dispevent_GetTypeInfoCount(IDispatch *iface, UINT *pctinfo)
89 {
90     g_unexpectedcall++;
91     *pctinfo = 0;
92     return S_OK;
93 }
94
95 static HRESULT WINAPI dispevent_GetTypeInfo(IDispatch *iface, UINT iTInfo,
96         LCID lcid, ITypeInfo **ppTInfo)
97 {
98     g_unexpectedcall++;
99     return S_OK;
100 }
101
102 static HRESULT WINAPI dispevent_GetIDsOfNames(IDispatch *iface, REFIID riid,
103         LPOLESTR *rgszNames, UINT cNames, LCID lcid, DISPID *rgDispId)
104 {
105     g_unexpectedcall++;
106     return S_OK;
107 }
108
109 static HRESULT WINAPI dispevent_Invoke(IDispatch *iface, DISPID member, REFIID riid,
110         LCID lcid, WORD flags, DISPPARAMS *params, VARIANT *result,
111         EXCEPINFO *excepInfo, UINT *argErr)
112 {
113     ok(member == 0, "expected 0 member, got %d\n", member);
114     ok(lcid == LOCALE_SYSTEM_DEFAULT, "expected LOCALE_SYSTEM_DEFAULT, got lcid %x\n", lcid);
115     ok(flags == DISPATCH_METHOD, "expected DISPATCH_METHOD, got %d\n", flags);
116
117     ok(params->cArgs == 0, "got %d\n", params->cArgs);
118     ok(params->cNamedArgs == 0, "got %d\n", params->cNamedArgs);
119     ok(params->rgvarg == NULL, "got %p\n", params->rgvarg);
120     ok(params->rgdispidNamedArgs == NULL, "got %p\n", params->rgdispidNamedArgs);
121
122     ok(result == NULL, "got %p\n", result);
123     ok(excepInfo == NULL, "got %p\n", excepInfo);
124     ok(argErr == NULL, "got %p\n", argErr);
125
126     g_expectedcall++;
127     return E_FAIL;
128 }
129
130 static const IDispatchVtbl dispeventVtbl =
131 {
132     dispevent_QueryInterface,
133     dispevent_AddRef,
134     dispevent_Release,
135     dispevent_GetTypeInfoCount,
136     dispevent_GetTypeInfo,
137     dispevent_GetIDsOfNames,
138     dispevent_Invoke
139 };
140
141 static IDispatch* create_dispevent(void)
142 {
143     dispevent *event = HeapAlloc(GetProcessHeap(), 0, sizeof(*event));
144
145     event->lpVtbl = &dispeventVtbl;
146     event->ref = 1;
147
148     return (IDispatch*)&event->lpVtbl;
149 }
150
151
152 static const WCHAR szEmpty[] = { 0 };
153 static const WCHAR szIncomplete[] = {
154     '<','?','x','m','l',' ',
155     'v','e','r','s','i','o','n','=','\'','1','.','0','\'','?','>','\n',0
156 };
157 static const WCHAR szComplete1[] = {
158     '<','?','x','m','l',' ',
159     'v','e','r','s','i','o','n','=','\'','1','.','0','\'','?','>','\n',
160     '<','o','p','e','n','>','<','/','o','p','e','n','>','\n',0
161 };
162 static const WCHAR szComplete2[] = {
163     '<','?','x','m','l',' ',
164     'v','e','r','s','i','o','n','=','\'','1','.','0','\'','?','>','\n',
165     '<','o','>','<','/','o','>','\n',0
166 };
167 static const WCHAR szComplete3[] = {
168     '<','?','x','m','l',' ',
169     'v','e','r','s','i','o','n','=','\'','1','.','0','\'','?','>','\n',
170     '<','a','>','<','/','a','>','\n',0
171 };
172 static const WCHAR szComplete4[] = {
173     '<','?','x','m','l',' ','v','e','r','s','i','o','n','=','\'','1','.','0','\'','?','>','\n',
174     '<','l','c',' ','d','l','=','\'','s','t','r','1','\'','>','\n',
175         '<','b','s',' ','v','r','=','\'','s','t','r','2','\'',' ','s','z','=','\'','1','2','3','4','\'','>',
176             'f','n','1','.','t','x','t','\n',
177         '<','/','b','s','>','\n',
178         '<','p','r',' ','i','d','=','\'','s','t','r','3','\'',' ','v','r','=','\'','1','.','2','.','3','\'',' ',
179                     'p','n','=','\'','w','i','n','e',' ','2','0','0','5','0','8','0','4','\'','>','\n',
180             'f','n','2','.','t','x','t','\n',
181         '<','/','p','r','>','\n',
182         '<','e','m','p','t','y','>','<','/','e','m','p','t','y','>','\n',
183         '<','f','o','>','\n',
184             '<','b','a','>','\n',
185                 'f','1','\n',
186             '<','/','b','a','>','\n',
187         '<','/','f','o','>','\n',
188     '<','/','l','c','>','\n',0
189 };
190 static const WCHAR szComplete5[] = {
191     '<','S',':','s','e','a','r','c','h',' ','x','m','l','n','s',':','D','=','"','D','A','V',':','"',' ',
192     'x','m','l','n','s',':','C','=','"','u','r','n',':','s','c','h','e','m','a','s','-','m','i','c','r','o','s','o','f','t','-','c','o','m',':','o','f','f','i','c','e',':','c','l','i','p','g','a','l','l','e','r','y','"',
193     ' ','x','m','l','n','s',':','S','=','"','u','r','n',':','s','c','h','e','m','a','s','-','m','i','c','r','o','s','o','f','t','-','c','o','m',':','o','f','f','i','c','e',':','c','l','i','p','g','a','l','l','e','r','y',':','s','e','a','r','c','h','"','>',
194         '<','S',':','s','c','o','p','e','>',
195             '<','S',':','d','e','e','p','>','/','<','/','S',':','d','e','e','p','>',
196         '<','/','S',':','s','c','o','p','e','>',
197         '<','S',':','c','o','n','t','e','n','t','f','r','e','e','t','e','x','t','>',
198             '<','C',':','t','e','x','t','o','r','p','r','o','p','e','r','t','y','/','>',
199             'c','o','m','p','u','t','e','r',
200         '<','/','S',':','c','o','n','t','e','n','t','f','r','e','e','t','e','x','t','>',
201     '<','/','S',':','s','e','a','r','c','h','>',0
202 };
203
204 static const WCHAR szComplete6[] = {
205     '<','?','x','m','l',' ','v','e','r','s','i','o','n','=','\'','1','.','0','\'',' ',
206     'e','n','c','o','d','i','n','g','=','\'','W','i','n','d','o','w','s','-','1','2','5','2','\'','?','>','\n',
207     '<','o','p','e','n','>','<','/','o','p','e','n','>','\n',0
208 };
209
210 static const CHAR szNonUnicodeXML[] =
211 "<?xml version='1.0' encoding='Windows-1252'?>\n"
212 "<open></open>\n";
213
214 static const CHAR szExampleXML[] =
215 "<?xml version='1.0' encoding='utf-8'?>\n"
216 "<root xmlns:foo='urn:uuid:86B2F87F-ACB6-45cd-8B77-9BDB92A01A29'>\n"
217 "    <elem>\n"
218 "        <a>A1 field</a>\n"
219 "        <b>B1 field</b>\n"
220 "        <c>C1 field</c>\n"
221 "        <description xmlns:foo='http://www.winehq.org' xmlns:bar='urn:uuid:86B2F87F-ACB6-45cd-8B77-9BDB92A01A29'>\n"
222 "            <html xmlns='http://www.w3.org/1999/xhtml'>\n"
223 "                This is <strong>a</strong> <i>description</i>. <bar:x/>\n"
224 "            </html>\n"
225 "            <html xml:space='preserve' xmlns='http://www.w3.org/1999/xhtml'>\n"
226 "                This is <strong>a</strong> <i>description</i> with preserved whitespace. <bar:x/>\n"
227 "            </html>\n"
228 "        </description>\n"
229 "    </elem>\n"
230 "\n"
231 "    <elem>\n"
232 "        <a>A2 field</a>\n"
233 "        <b>B2 field</b>\n"
234 "        <c type=\"old\">C2 field</c>\n"
235 "    </elem>\n"
236 "\n"
237 "    <elem xmlns='urn:uuid:86B2F87F-ACB6-45cd-8B77-9BDB92A01A29'>\n"
238 "        <a>A3 field</a>\n"
239 "        <b>B3 field</b>\n"
240 "        <c>C3 field</c>\n"
241 "    </elem>\n"
242 "\n"
243 "    <elem>\n"
244 "        <a>A4 field</a>\n"
245 "        <b>B4 field</b>\n"
246 "        <foo:c>C4 field</foo:c>\n"
247 "    </elem>\n"
248 "</root>\n";
249
250 static const CHAR szNodeTypesXML[] =
251 "<?xml version='1.0'?>"
252 "<!-- comment node 0 -->"
253 "<root id='0' depth='0'>"
254 "   <!-- comment node 1 -->"
255 "   text node 0"
256 "   <x id='1' depth='1'>"
257 "       <?foo value='PI for x'?>"
258 "       <!-- comment node 2 -->"
259 "       text node 1"
260 "       <a id='3' depth='2'/>"
261 "       <b id='4' depth='2'/>"
262 "       <c id='5' depth='2'/>"
263 "   </x>"
264 "   <y id='2' depth='1'>"
265 "       <?bar value='PI for y'?>"
266 "       <!-- comment node 3 -->"
267 "       text node 2"
268 "       <a id='6' depth='2'/>"
269 "       <b id='7' depth='2'/>"
270 "       <c id='8' depth='2'/>"
271 "   </y>"
272 "</root>";
273
274 static const CHAR szTransformXML[] =
275 "<?xml version=\"1.0\"?>\n"
276 "<greeting>\n"
277 "Hello World\n"
278 "</greeting>";
279
280 static  const CHAR szTransformSSXML[] =
281 "<xsl:stylesheet xmlns:xsl=\"http://www.w3.org/1999/XSL/Transform\" version=\"1.0\">\n"
282 "   <xsl:output method=\"html\"/>\n"
283 "   <xsl:template match=\"/\">\n"
284 "       <xsl:apply-templates select=\"greeting\"/>\n"
285 "   </xsl:template>\n"
286 "   <xsl:template match=\"greeting\">\n"
287 "       <html>\n"
288 "           <body>\n"
289 "               <h1>\n"
290 "                   <xsl:value-of select=\".\"/>\n"
291 "               </h1>\n"
292 "           </body>\n"
293 "       </html>\n"
294 "   </xsl:template>\n"
295 "</xsl:stylesheet>";
296
297 static  const CHAR szTransformOutput[] =
298 "<html><body><h1>"
299 "Hello World"
300 "</h1></body></html>";
301
302 static const CHAR szTypeValueXML[] =
303 "<?xml version=\"1.0\" encoding=\"utf-8\"?>\n"
304 "<root xmlns:dt=\"urn:schemas-microsoft-com:datatypes\">\n"
305 "   <string>Wine</string>\n"
306 "   <string2 dt:dt=\"string\">String</string2>\n"
307 "   <number dt:dt=\"number\">12.44</number>\n"
308 "   <number2 dt:dt=\"NUMbEr\">-3.71e3</number2>\n"
309 "   <int dt:dt=\"int\">-13</int>\n"
310 "   <fixed dt:dt=\"fixed.14.4\">7322.9371</fixed>\n"
311 "   <bool dt:dt=\"boolean\">1</bool>\n"
312 "   <datetime dt:dt=\"datetime\">2009-11-18T03:21:33.12</datetime>\n"
313 "   <datetimetz dt:dt=\"datetime.tz\">2003-07-11T11:13:57+03:00</datetimetz>\n"
314 "   <date dt:dt=\"date\">3721-11-01</date>\n"
315 "   <time dt:dt=\"time\">13:57:12.31321</time>\n"
316 "   <timetz dt:dt=\"time.tz\">23:21:01.13+03:21</timetz>\n"
317 "   <i1 dt:dt=\"i1\">-13</i1>\n"
318 "   <i2 dt:dt=\"i2\">31915</i2>\n"
319 "   <i4 dt:dt=\"i4\">-312232</i4>\n"
320 "   <ui1 dt:dt=\"ui1\">123</ui1>\n"
321 "   <ui2 dt:dt=\"ui2\">48282</ui2>\n"
322 "   <ui4 dt:dt=\"ui4\">949281</ui4>\n"
323 "   <r4 dt:dt=\"r4\">213124.0</r4>\n"
324 "   <r8 dt:dt=\"r8\">0.412</r8>\n"
325 "   <float dt:dt=\"float\">41221.421</float>\n"
326 "   <uuid dt:dt=\"uuid\">333C7BC4-460F-11D0-BC04-0080C7055a83</uuid>\n"
327 "   <binhex dt:dt=\"bin.hex\">fffca012003c</binhex>\n"
328 "   <binbase64 dt:dt=\"bin.base64\">YmFzZTY0IHRlc3Q=</binbase64>\n"
329 "</root>";
330
331 static const CHAR szBasicTransformSSXMLPart1[] =
332 "<?xml version=\"1.0\"?>"
333 "<xsl:stylesheet version=\"1.0\" xmlns:xsl=\"http://www.w3.org/1999/XSL/Transform\" >"
334 "<xsl:output method=\"html\"/>\n"
335 "<xsl:template match=\"/\">"
336 "<HTML><BODY><TABLE>"
337 "        <xsl:apply-templates select='document(\"";
338
339 static const CHAR szBasicTransformSSXMLPart2[] =
340 "\")/bottle/wine'>"
341 "           <xsl:sort select=\"cost\"/><xsl:sort select=\"name\"/>"
342 "        </xsl:apply-templates>"
343 "</TABLE></BODY></HTML>"
344 "</xsl:template>"
345 "<xsl:template match=\"bottle\">"
346 "   <TR><xsl:apply-templates select=\"name\" /><xsl:apply-templates select=\"cost\" /></TR>"
347 "</xsl:template>"
348 "<xsl:template match=\"name\">"
349 "   <TD><xsl:apply-templates /></TD>"
350 "</xsl:template>"
351 "<xsl:template match=\"cost\">"
352 "   <TD><xsl:apply-templates /></TD>"
353 "</xsl:template>"
354 "</xsl:stylesheet>";
355
356 static const CHAR szBasicTransformXML[] =
357 "<?xml version=\"1.0\"?><bottle><wine><name>Wine</name><cost>$25.00</cost></wine></bottle>";
358
359 static const CHAR szBasicTransformOutput[] =
360 "<HTML><BODY><TABLE><TD>Wine</TD><TD>$25.00</TD></TABLE></BODY></HTML>";
361
362 #define SZ_EMAIL_DTD \
363 "<!DOCTYPE email ["\
364 "   <!ELEMENT email         (recipients,from,reply-to?,subject,body,attachment*)>"\
365 "       <!ATTLIST email attachments IDREFS #REQUIRED>"\
366 "       <!ATTLIST email sent (yes|no) \"no\">"\
367 "   <!ELEMENT recipients    (to+,cc*)>"\
368 "   <!ELEMENT to            (#PCDATA)>"\
369 "       <!ATTLIST to name CDATA #IMPLIED>"\
370 "   <!ELEMENT cc            (#PCDATA)>"\
371 "       <!ATTLIST cc name CDATA #IMPLIED>"\
372 "   <!ELEMENT from          (#PCDATA)>"\
373 "       <!ATTLIST from name CDATA #IMPLIED>"\
374 "   <!ELEMENT reply-to      (#PCDATA)>"\
375 "       <!ATTLIST reply-to name CDATA #IMPLIED>"\
376 "   <!ELEMENT subject       ANY>"\
377 "   <!ELEMENT body          ANY>"\
378 "       <!ATTLIST body enc CDATA #FIXED \"UTF-8\">"\
379 "   <!ELEMENT attachment    (#PCDATA)>"\
380 "       <!ATTLIST attachment id ID #REQUIRED>"\
381 "]>"
382
383 static const CHAR szEmailXML[] =
384 "<?xml version=\"1.0\"?>"
385 SZ_EMAIL_DTD
386 "<email attachments=\"patch1\">"
387 "   <recipients>"
388 "       <to>wine-patches@winehq.org</to>"
389 "   </recipients>"
390 "   <from name=\"Anonymous\">user@localhost</from>"
391 "   <subject>msxml3/tests: DTD validation (try 87)</subject>"
392 "   <body>"
393 "       It no longer causes spontaneous combustion..."
394 "   </body>"
395 "   <attachment id=\"patch1\">0001-msxml3-tests-DTD-validation.patch</attachment>"
396 "</email>";
397
398 static const CHAR szEmailXML_0D[] =
399 "<?xml version=\"1.0\"?>"
400 SZ_EMAIL_DTD
401 "<email attachments=\"patch1\">"
402 "   <recipients>"
403 "       <to>wine-patches@winehq.org</to>"
404 "   </recipients>"
405 "   <from name=\"Anonymous\">user@localhost</from>"
406 "   <subject>msxml3/tests: DTD validation (try 88)</subject>"
407 "   <body>"
408 "       <undecl />"
409 "       XML_ELEMENT_UNDECLARED 0xC00CE00D"
410 "   </body>"
411 "   <attachment id=\"patch1\">0001-msxml3-tests-DTD-validation.patch</attachment>"
412 "</email>";
413
414 static const CHAR szEmailXML_0E[] =
415 "<?xml version=\"1.0\"?>"
416 SZ_EMAIL_DTD
417 "<email attachments=\"patch1\">"
418 "   <recipients>"
419 "       <to>wine-patches@winehq.org</to>"
420 "   </recipients>"
421 "   <from name=\"Anonymous\">user@localhost</from>"
422 "   <subject>msxml3/tests: DTD validation (try 89)</subject>"
423 "   <body>"
424 "       XML_ELEMENT_ID_NOT_FOUND 0xC00CE00E"
425 "   </body>"
426 "   <attachment id=\"patch\">0001-msxml3-tests-DTD-validation.patch</attachment>"
427 "</email>";
428
429 static const CHAR szEmailXML_11[] =
430 "<?xml version=\"1.0\"?>"
431 SZ_EMAIL_DTD
432 "<email attachments=\"patch1\">"
433 "   <recipients>"
434 "   </recipients>"
435 "   <from name=\"Anonymous\">user@localhost</from>"
436 "   <subject>msxml3/tests: DTD validation (try 90)</subject>"
437 "   <body>"
438 "       XML_EMPTY_NOT_ALLOWED 0xC00CE011"
439 "   </body>"
440 "   <attachment id=\"patch1\">0001-msxml3-tests-DTD-validation.patch</attachment>"
441 "</email>";
442
443 static const CHAR szEmailXML_13[] =
444 "<?xml version=\"1.0\"?>"
445 SZ_EMAIL_DTD
446 "<msg attachments=\"patch1\">"
447 "   <recipients>"
448 "       <to>wine-patches@winehq.org</to>"
449 "   </recipients>"
450 "   <from name=\"Anonymous\">user@localhost</from>"
451 "   <subject>msxml3/tests: DTD validation (try 91)</subject>"
452 "   <body>"
453 "       XML_ROOT_NAME_MISMATCH 0xC00CE013"
454 "   </body>"
455 "   <attachment id=\"patch1\">0001-msxml3-tests-DTD-validation.patch</attachment>"
456 "</msg>";
457
458 static const CHAR szEmailXML_14[] =
459 "<?xml version=\"1.0\"?>"
460 SZ_EMAIL_DTD
461 "<email attachments=\"patch1\">"
462 "   <to>wine-patches@winehq.org</to>"
463 "   <from name=\"Anonymous\">user@localhost</from>"
464 "   <subject>msxml3/tests: DTD validation (try 92)</subject>"
465 "   <body>"
466 "       XML_INVALID_CONTENT 0xC00CE014"
467 "   </body>"
468 "   <attachment id=\"patch1\">0001-msxml3-tests-DTD-validation.patch</attachment>"
469 "</email>";
470
471 static const CHAR szEmailXML_15[] =
472 "<?xml version=\"1.0\"?>"
473 SZ_EMAIL_DTD
474 "<email attachments=\"patch1\" ip=\"127.0.0.1\">"
475 "   <recipients>"
476 "       <to>wine-patches@winehq.org</to>"
477 "   </recipients>"
478 "   <from name=\"Anonymous\">user@localhost</from>"
479 "   <subject>msxml3/tests: DTD validation (try 93)</subject>"
480 "   <body>"
481 "       XML_ATTRIBUTE_NOT_DEFINED 0xC00CE015"
482 "   </body>"
483 "   <attachment id=\"patch1\">0001-msxml3-tests-DTD-validation.patch</attachment>"
484 "</email>";
485
486 static const CHAR szEmailXML_16[] =
487 "<?xml version=\"1.0\"?>"
488 SZ_EMAIL_DTD
489 "<email attachments=\"patch1\">"
490 "   <recipients>"
491 "       <to>wine-patches@winehq.org</to>"
492 "   </recipients>"
493 "   <from name=\"Anonymous\">user@localhost</from>"
494 "   <subject>msxml3/tests: DTD validation (try 94)</subject>"
495 "   <body enc=\"ASCII\">"
496 "       XML_ATTRIBUTE_FIXED 0xC00CE016"
497 "   </body>"
498 "   <attachment id=\"patch1\">0001-msxml3-tests-DTD-validation.patch</attachment>"
499 "</email>";
500
501 static const CHAR szEmailXML_17[] =
502 "<?xml version=\"1.0\"?>"
503 SZ_EMAIL_DTD
504 "<email attachments=\"patch1\" sent=\"true\">"
505 "   <recipients>"
506 "       <to>wine-patches@winehq.org</to>"
507 "   </recipients>"
508 "   <from name=\"Anonymous\">user@localhost</from>"
509 "   <subject>msxml3/tests: DTD validation (try 95)</subject>"
510 "   <body>"
511 "       XML_ATTRIBUTE_VALUE 0xC00CE017"
512 "   </body>"
513 "   <attachment id=\"patch1\">0001-msxml3-tests-DTD-validation.patch</attachment>"
514 "</email>";
515
516 static const CHAR szEmailXML_18[] =
517 "<?xml version=\"1.0\"?>"
518 SZ_EMAIL_DTD
519 "<email attachments=\"patch1\">"
520 "   oops"
521 "   <recipients>"
522 "       <to>wine-patches@winehq.org</to>"
523 "   </recipients>"
524 "   <from name=\"Anonymous\">user@localhost</from>"
525 "   <subject>msxml3/tests: DTD validation (try 96)</subject>"
526 "   <body>"
527 "       XML_ILLEGAL_TEXT 0xC00CE018"
528 "   </body>"
529 "   <attachment id=\"patch1\">0001-msxml3-tests-DTD-validation.patch</attachment>"
530 "</email>";
531
532 static const CHAR szEmailXML_20[] =
533 "<?xml version=\"1.0\"?>"
534 SZ_EMAIL_DTD
535 "<email>"
536 "   <recipients>"
537 "       <to>wine-patches@winehq.org</to>"
538 "   </recipients>"
539 "   <from name=\"Anonymous\">user@localhost</from>"
540 "   <subject>msxml3/tests: DTD validation (try 97)</subject>"
541 "   <body>"
542 "       XML_REQUIRED_ATTRIBUTE_MISSING 0xC00CE020"
543 "   </body>"
544 "   <attachment id=\"patch1\">0001-msxml3-tests-DTD-validation.patch</attachment>"
545 "</email>";
546
547 static const WCHAR szNonExistentFile[] = {
548     'c', ':', '\\', 'N', 'o', 'n', 'e', 'x', 'i', 's', 't', 'e', 'n', 't', '.', 'x', 'm', 'l', 0
549 };
550 static const WCHAR szNonExistentAttribute[] = {
551     'n','o','n','E','x','i','s','i','t','i','n','g','A','t','t','r','i','b','u','t','e',0
552 };
553 static const WCHAR szDocument[] = {
554     '#', 'd', 'o', 'c', 'u', 'm', 'e', 'n', 't', 0
555 };
556
557 static const WCHAR szOpen[] = { 'o','p','e','n',0 };
558 static WCHAR szdl[] = { 'd','l',0 };
559 static const WCHAR szvr[] = { 'v','r',0 };
560 static const WCHAR szlc[] = { 'l','c',0 };
561 static WCHAR szbs[] = { 'b','s',0 };
562 static const WCHAR szstr1[] = { 's','t','r','1',0 };
563 static const WCHAR szstr2[] = { 's','t','r','2',0 };
564 static const WCHAR szstar[] = { '*',0 };
565 static const WCHAR szfn1_txt[] = {'f','n','1','.','t','x','t',0};
566
567 static WCHAR szComment[] = {'A',' ','C','o','m','m','e','n','t',0 };
568 static WCHAR szCommentXML[] = {'<','!','-','-','A',' ','C','o','m','m','e','n','t','-','-','>',0 };
569 static WCHAR szCommentNodeText[] = {'#','c','o','m','m','e','n','t',0 };
570
571 static WCHAR szElement[] = {'E','l','e','T','e','s','t', 0 };
572 static WCHAR szElementXML[]  = {'<','E','l','e','T','e','s','t','/','>',0 };
573 static WCHAR szElementXML2[] = {'<','E','l','e','T','e','s','t',' ','A','t','t','r','=','"','"','/','>',0 };
574 static WCHAR szElementXML3[] = {'<','E','l','e','T','e','s','t',' ','A','t','t','r','=','"','"','>',
575                                 'T','e','s','t','i','n','g','N','o','d','e','<','/','E','l','e','T','e','s','t','>',0 };
576 static WCHAR szElementXML4[] = {'<','E','l','e','T','e','s','t',' ','A','t','t','r','=','"','"','>',
577                                 '&','a','m','p',';','x',' ',0x2103,'<','/','E','l','e','T','e','s','t','>',0 };
578
579 static WCHAR szAttribute[] = {'A','t','t','r',0 };
580 static WCHAR szAttributeXML[] = {'A','t','t','r','=','"','"',0 };
581
582 static WCHAR szCData[] = {'[','1',']','*','2','=','3',';',' ','&','g','e','e',' ','t','h','a','t','s',
583                           ' ','n','o','t',' ','r','i','g','h','t','!', 0};
584 static WCHAR szCDataXML[] = {'<','!','[','C','D','A','T','A','[','[','1',']','*','2','=','3',';',' ','&',
585                              'g','e','e',' ','t','h','a','t','s',' ','n','o','t',' ','r','i','g','h','t',
586                              '!',']',']','>',0};
587 static WCHAR szCDataNodeText[] = {'#','c','d','a','t','a','-','s','e','c','t','i','o','n',0 };
588 static WCHAR szDocFragmentText[] = {'#','d','o','c','u','m','e','n','t','-','f','r','a','g','m','e','n','t',0 };
589
590 static WCHAR szEntityRef[] = {'e','n','t','i','t','y','r','e','f',0 };
591 static WCHAR szEntityRefXML[] = {'&','e','n','t','i','t','y','r','e','f',';',0 };
592 static WCHAR szStrangeChars[] = {'&','x',' ',0x2103, 0};
593
594 #define expect_bstr_eq_and_free(bstr, expect) { \
595     BSTR bstrExp = alloc_str_from_narrow(expect); \
596     ok(lstrcmpW(bstr, bstrExp) == 0, "String differs\n"); \
597     SysFreeString(bstr); \
598     SysFreeString(bstrExp); \
599 }
600
601 #define expect_eq(expr, value, type, format) { type ret = (expr); ok((value) == ret, #expr " expected " format " got " format "\n", value, ret); }
602
603 #define ole_check(expr) { \
604     HRESULT r = expr; \
605     ok(r == S_OK, #expr " returned %x\n", r); \
606 }
607
608 #define ole_expect(expr, expect) { \
609     HRESULT r = expr; \
610     ok(r == (expect), #expr " returned %x, expected %x\n", r, expect); \
611 }
612
613 #define double_eq(x, y) ok((x)-(y)<=1e-14*(x) && (x)-(y)>=-1e-14*(x), "expected %.16g, got %.16g\n", x, y)
614
615 static void* _create_object(const GUID *clsid, const char *name, const IID *iid, int line)
616 {
617     void *obj = NULL;
618     HRESULT hr;
619
620     hr = CoCreateInstance(clsid, NULL, CLSCTX_INPROC_SERVER, iid, &obj);
621     if (hr != S_OK)
622         win_skip_(__FILE__,line)("failed to create %s instance: 0x%08x\n", name, hr);
623
624     return obj;
625 }
626
627 #define _create(cls) cls, #cls
628
629 #define create_document(iid) _create_object(&_create(CLSID_DOMDocument), iid, __LINE__)
630 #define create_document_version(v, iid) _create_object(&_create(CLSID_DOMDocument ## v), iid, __LINE__)
631 #define create_cache(iid) _create_object(&_create(CLSID_XMLSchemaCache), iid, __LINE__)
632 #define create_cache_version(v, iid) _create_object(&_create(CLSID_XMLSchemaCache ## v), iid, __LINE__)
633 #define create_xsltemplate(iid) _create_object(&_create(CLSID_XSLTemplate), iid, __LINE__)
634
635 static BSTR alloc_str_from_narrow(const char *str)
636 {
637     int len = MultiByteToWideChar(CP_ACP, 0, str, -1, NULL, 0);
638     BSTR ret = SysAllocStringLen(NULL, len - 1);  /* NUL character added automatically */
639     MultiByteToWideChar(CP_ACP, 0, str, -1, ret, len);
640     return ret;
641 }
642
643 static BSTR alloced_bstrs[256];
644 static int alloced_bstrs_count;
645
646 static BSTR _bstr_(const char *str)
647 {
648     assert(alloced_bstrs_count < sizeof(alloced_bstrs)/sizeof(alloced_bstrs[0]));
649     alloced_bstrs[alloced_bstrs_count] = alloc_str_from_narrow(str);
650     return alloced_bstrs[alloced_bstrs_count++];
651 }
652
653 static void free_bstrs(void)
654 {
655     int i;
656     for (i = 0; i < alloced_bstrs_count; i++)
657         SysFreeString(alloced_bstrs[i]);
658     alloced_bstrs_count = 0;
659 }
660
661 static VARIANT _variantbstr_(const char *str)
662 {
663     VARIANT v;
664     V_VT(&v) = VT_BSTR;
665     V_BSTR(&v) = _bstr_(str);
666     return v;
667 }
668
669 static BOOL compareIgnoreReturns(BSTR sLeft, BSTR sRight)
670 {
671     for (;;)
672     {
673         while (*sLeft == '\r' || *sLeft == '\n') sLeft++;
674         while (*sRight == '\r' || *sRight == '\n') sRight++;
675         if (*sLeft != *sRight) return FALSE;
676         if (!*sLeft) return TRUE;
677         sLeft++;
678         sRight++;
679     }
680 }
681
682 static void get_str_for_type(DOMNodeType type, char *buf)
683 {
684     switch (type)
685     {
686         case NODE_ATTRIBUTE:
687             strcpy(buf, "A");
688             break;
689         case NODE_ELEMENT:
690             strcpy(buf, "E");
691             break;
692         case NODE_DOCUMENT:
693             strcpy(buf, "D");
694             break;
695         case NODE_TEXT:
696             strcpy(buf, "T");
697             break;
698         case NODE_COMMENT:
699             strcpy(buf, "C");
700             break;
701         case NODE_PROCESSING_INSTRUCTION:
702             strcpy(buf, "P");
703             break;
704         default:
705             wsprintfA(buf, "[%d]", type);
706     }
707 }
708
709 #define test_disp(u) _test_disp(__LINE__,u)
710 static void _test_disp(unsigned line, IUnknown *unk)
711 {
712     DISPID dispid = DISPID_XMLDOM_NODELIST_RESET;
713     IDispatchEx *dispex;
714     DWORD dwProps = 0;
715     BSTR sName;
716     UINT ticnt;
717     IUnknown *pUnk;
718     HRESULT hres;
719
720     hres = IUnknown_QueryInterface(unk, &IID_IDispatchEx, (void**)&dispex);
721     ok_(__FILE__,line) (hres == S_OK, "Could not get IDispatch: %08x\n", hres);
722     if(FAILED(hres))
723         return;
724
725     ticnt = 0xdeadbeef;
726     hres = IDispatchEx_GetTypeInfoCount(dispex, &ticnt);
727     ok_(__FILE__,line) (hres == S_OK, "GetTypeInfoCount failed: %08x\n", hres);
728     ok_(__FILE__,line) (ticnt == 1, "ticnt=%u\n", ticnt);
729
730     sName = SysAllocString( szstar );
731     hres = IDispatchEx_DeleteMemberByName(dispex, sName, fdexNameCaseSensitive);
732     ok(hres == E_NOTIMPL, "expected E_NOTIMPL got %08x\n", hres);
733     SysFreeString( sName );
734
735     hres = IDispatchEx_DeleteMemberByDispID(dispex, dispid);
736     ok(hres == E_NOTIMPL, "expected E_NOTIMPL got %08x\n", hres);
737
738     hres = IDispatchEx_GetMemberProperties(dispex, dispid, grfdexPropCanAll, &dwProps);
739     ok(hres == E_NOTIMPL, "expected E_NOTIMPL got %08x\n", hres);
740     ok(dwProps == 0, "expected 0 got %d\n", dwProps);
741
742     hres = IDispatchEx_GetMemberName(dispex, dispid, &sName);
743     ok(hres == E_NOTIMPL, "expected E_NOTIMPL got %08x\n", hres);
744     if(SUCCEEDED(hres))
745         SysFreeString(sName);
746
747     hres = IDispatchEx_GetNextDispID(dispex, fdexEnumDefault, DISPID_XMLDOM_NODELIST_RESET, &dispid);
748     ok(hres == E_NOTIMPL, "expected E_NOTIMPL got %08x\n", hres);
749
750     hres = IDispatchEx_GetNameSpaceParent(dispex, &pUnk);
751     ok(hres == E_NOTIMPL, "expected E_NOTIMPL got %08x\n", hres);
752     if(hres == S_OK && pUnk)
753         IUnknown_Release(pUnk);
754
755     IDispatchEx_Release(dispex);
756 }
757
758 static int get_node_position(IXMLDOMNode *node)
759 {
760     HRESULT r;
761     int pos = 0;
762
763     IXMLDOMNode_AddRef(node);
764     do
765     {
766         IXMLDOMNode *new_node;
767
768         pos++;
769         r = IXMLDOMNode_get_previousSibling(node, &new_node);
770         ok(SUCCEEDED(r), "get_previousSibling failed\n");
771         IXMLDOMNode_Release(node);
772         node = new_node;
773     } while (r == S_OK);
774     return pos;
775 }
776
777 static void node_to_string(IXMLDOMNode *node, char *buf)
778 {
779     HRESULT r = S_OK;
780     DOMNodeType type;
781
782     if (node == NULL)
783     {
784         lstrcpyA(buf, "(null)");
785         return;
786     }
787
788     IXMLDOMNode_AddRef(node);
789     while (r == S_OK)
790     {
791         IXMLDOMNode *new_node;
792
793         ole_check(IXMLDOMNode_get_nodeType(node, &type));
794         get_str_for_type(type, buf);
795         buf+=strlen(buf);
796
797         if (type == NODE_ATTRIBUTE)
798         {
799             BSTR bstr;
800             ole_check(IXMLDOMNode_get_nodeName(node, &bstr));
801             *(buf++) = '\'';
802             wsprintfA(buf, "%ws", bstr);
803             buf += strlen(buf);
804             *(buf++) = '\'';
805             SysFreeString(bstr);
806
807             r = IXMLDOMNode_selectSingleNode(node, _bstr_(".."), &new_node);
808         }
809         else
810         {
811             r = IXMLDOMNode_get_parentNode(node, &new_node);
812             wsprintf(buf, "%d", get_node_position(node));
813             buf += strlen(buf);
814         }
815
816         ok(SUCCEEDED(r), "get_parentNode failed (%08x)\n", r);
817         IXMLDOMNode_Release(node);
818         node = new_node;
819         if (r == S_OK)
820             *(buf++) = '.';
821     }
822
823     *buf = 0;
824 }
825
826 static char *list_to_string(IXMLDOMNodeList *list)
827 {
828     static char buf[4096];
829     char *pos = buf;
830     LONG len = 0;
831     int i;
832
833     if (list == NULL)
834     {
835         lstrcpyA(buf, "(null)");
836         return buf;
837     }
838     ole_check(IXMLDOMNodeList_get_length(list, &len));
839     for (i = 0; i < len; i++)
840     {
841         IXMLDOMNode *node;
842         if (i > 0)
843             *(pos++) = ' ';
844         ole_check(IXMLDOMNodeList_nextNode(list, &node));
845         node_to_string(node, pos);
846         pos += strlen(pos);
847         IXMLDOMNode_Release(node);
848     }
849     *pos = 0;
850     return buf;
851 }
852
853 #define expect_node(node, expstr) { char str[4096]; node_to_string(node, str); ok(strcmp(str, expstr)==0, "Invalid node: %s, expected %s\n", str, expstr); }
854 #define expect_list_and_release(list, expstr) { char *str = list_to_string(list); ok(strcmp(str, expstr)==0, "Invalid node list: %s, expected %s\n", str, expstr); if (list) IXMLDOMNodeList_Release(list); }
855
856 static void test_domdoc( void )
857 {
858     HRESULT r;
859     IXMLDOMDocument *doc;
860     IXMLDOMParseError *error;
861     IXMLDOMElement *element = NULL;
862     IXMLDOMNode *node;
863     IXMLDOMText *nodetext = NULL;
864     IXMLDOMComment *node_comment = NULL;
865     IXMLDOMAttribute *node_attr = NULL;
866     IXMLDOMNode *nodeChild = NULL;
867     IXMLDOMProcessingInstruction *nodePI = NULL;
868     ISupportErrorInfo *support_error = NULL;
869     VARIANT_BOOL b;
870     VARIANT var;
871     BSTR str;
872     LONG code;
873     LONG nLength = 0;
874     WCHAR buff[100];
875
876     doc = create_document(&IID_IXMLDOMDocument);
877     if (!doc) return;
878
879     test_disp((IUnknown*)doc);
880
881 if (0)
882 {
883     /* crashes on native */
884     IXMLDOMDocument_loadXML( doc, (BSTR)0x1, NULL );
885 }
886
887     /* try some stupid things */
888     r = IXMLDOMDocument_loadXML( doc, NULL, NULL );
889     ok( r == S_FALSE, "loadXML succeeded\n");
890
891     b = VARIANT_TRUE;
892     r = IXMLDOMDocument_loadXML( doc, NULL, &b );
893     ok( r == S_FALSE, "loadXML succeeded\n");
894     ok( b == VARIANT_FALSE, "failed to load XML string\n");
895
896     /* try to load a document from a nonexistent file */
897     b = VARIANT_TRUE;
898     str = SysAllocString( szNonExistentFile );
899     VariantInit(&var);
900     V_VT(&var) = VT_BSTR;
901     V_BSTR(&var) = str;
902
903     r = IXMLDOMDocument_load( doc, var, &b);
904     ok( r == S_FALSE, "loadXML succeeded\n");
905     ok( b == VARIANT_FALSE, "succeeded in loading XML string\n");
906     SysFreeString( str );
907
908     /* try load an empty document */
909     b = VARIANT_TRUE;
910     str = SysAllocString( szEmpty );
911     r = IXMLDOMDocument_loadXML( doc, str, &b );
912     ok( r == S_FALSE, "loadXML succeeded\n");
913     ok( b == VARIANT_FALSE, "succeeded in loading XML string\n");
914     SysFreeString( str );
915
916     r = IXMLDOMDocument_get_async( doc, &b );
917     ok( r == S_OK, "get_async failed (%08x)\n", r);
918     ok( b == VARIANT_TRUE, "Wrong default value\n");
919
920     /* check that there's no document element */
921     element = NULL;
922     r = IXMLDOMDocument_get_documentElement( doc, &element );
923     ok( r == S_FALSE, "should be no document element\n");
924
925     /* try finding a node */
926     node = NULL;
927     str = SysAllocString( szstr1 );
928     r = IXMLDOMDocument_selectSingleNode( doc, str, &node );
929     ok( r == S_FALSE, "ret %08x\n", r );
930     SysFreeString( str );
931
932     b = VARIANT_TRUE;
933     str = SysAllocString( szIncomplete );
934     r = IXMLDOMDocument_loadXML( doc, str, &b );
935     ok( r == S_FALSE, "loadXML succeeded\n");
936     ok( b == VARIANT_FALSE, "succeeded in loading XML string\n");
937     SysFreeString( str );
938
939     /* check that there's no document element */
940     element = (IXMLDOMElement*)1;
941     r = IXMLDOMDocument_get_documentElement( doc, &element );
942     ok( r == S_FALSE, "should be no document element\n");
943     ok( element == NULL, "Element should be NULL\n");
944
945     /* test for BSTR handling, pass broken BSTR */
946     memcpy(&buff[2], szComplete1, sizeof(szComplete1));
947     /* just a big length */
948     *(DWORD*)buff = 0xf0f0;
949     b = VARIANT_FALSE;
950     r = IXMLDOMDocument_loadXML( doc, &buff[2], &b );
951     ok( r == S_OK, "loadXML failed\n");
952     ok( b == VARIANT_TRUE, "failed to load XML string\n");
953
954     /* loadXML ignores the encoding attribute and always expects Unicode */
955     b = VARIANT_FALSE;
956     str = SysAllocString( szComplete6 );
957     r = IXMLDOMDocument_loadXML( doc, str, &b );
958     ok( r == S_OK, "loadXML failed\n");
959     ok( b == VARIANT_TRUE, "failed to load XML string\n");
960     SysFreeString( str );
961
962     /* try a BSTR containing a Windows-1252 document */
963     b = VARIANT_TRUE;
964     str = SysAllocStringByteLen( szNonUnicodeXML, sizeof(szNonUnicodeXML) - 1 );
965     r = IXMLDOMDocument_loadXML( doc, str, &b );
966     ok( r == S_FALSE, "loadXML succeeded\n");
967     ok( b == VARIANT_FALSE, "succeeded in loading XML string\n");
968     SysFreeString( str );
969
970     /* try to load something valid */
971     b = VARIANT_FALSE;
972     str = SysAllocString( szComplete1 );
973     r = IXMLDOMDocument_loadXML( doc, str, &b );
974     ok( r == S_OK, "loadXML failed\n");
975     ok( b == VARIANT_TRUE, "failed to load XML string\n");
976     SysFreeString( str );
977
978     /* check if nodename is correct */
979     r = IXMLDOMDocument_get_nodeName( doc, NULL );
980     ok ( r == E_INVALIDARG, "get_nodeName (NULL) wrong code\n");
981
982     str = (BSTR)0xdeadbeef;
983     r = IXMLDOMDocument_get_baseName( doc, &str );
984     ok ( r == S_FALSE, "got 0x%08x\n", r);
985     ok (str == NULL, "got %p\n", str);
986
987     /* content doesn't matter here */
988     str = NULL;
989     r = IXMLDOMDocument_get_nodeName( doc, &str );
990     ok ( r == S_OK, "get_nodeName wrong code\n");
991     ok ( str != NULL, "str is null\n");
992     ok( !lstrcmpW( str, szDocument ), "incorrect nodeName\n");
993     SysFreeString( str );
994
995     /* test put_text */
996     r = IXMLDOMDocument_put_text( doc, _bstr_("Should Fail") );
997     ok( r == E_FAIL, "ret %08x\n", r );
998
999     /* check that there's a document element */
1000     element = NULL;
1001     r = IXMLDOMDocument_get_documentElement( doc, &element );
1002     ok( r == S_OK, "should be a document element\n");
1003     if( element )
1004     {
1005         IObjectIdentity *ident;
1006         BSTR tag = NULL;
1007
1008         test_disp((IUnknown*)element);
1009
1010         r = IXMLDOMElement_QueryInterface( element, &IID_IObjectIdentity, (void**)&ident );
1011         ok( r == E_NOINTERFACE, "ret %08x\n", r);
1012
1013         r = IXMLDOMElement_get_tagName( element, NULL );
1014         ok( r == E_INVALIDARG, "ret %08x\n", r);
1015
1016         /* check if the tag is correct */
1017         r = IXMLDOMElement_get_tagName( element, &tag );
1018         ok( r == S_OK, "couldn't get tag name\n");
1019         ok( tag != NULL, "tag was null\n");
1020         ok( !lstrcmpW( tag, szOpen ), "incorrect tag name\n");
1021         SysFreeString( tag );
1022
1023         /* figure out what happens if we try to reload the document */
1024         str = SysAllocString( szComplete2 );
1025         r = IXMLDOMDocument_loadXML( doc, str, &b );
1026         ok( r == S_OK, "loadXML failed\n");
1027         ok( b == VARIANT_TRUE, "failed to load XML string\n");
1028         SysFreeString( str );
1029
1030         /* check if the tag is still correct */
1031         tag = NULL;
1032         r = IXMLDOMElement_get_tagName( element, &tag );
1033         ok( r == S_OK, "couldn't get tag name\n");
1034         ok( tag != NULL, "tag was null\n");
1035         ok( !lstrcmpW( tag, szOpen ), "incorrect tag name\n");
1036         SysFreeString( tag );
1037
1038         IXMLDOMElement_Release( element );
1039         element = NULL;
1040     }
1041
1042     /* as soon as we call loadXML again, the document element will disappear */
1043     b = 2;
1044     r = IXMLDOMDocument_loadXML( doc, NULL, NULL );
1045     ok( r == S_FALSE, "loadXML failed\n");
1046     ok( b == 2, "variant modified\n");
1047     r = IXMLDOMDocument_get_documentElement( doc, &element );
1048     ok( r == S_FALSE, "should be no document element\n");
1049
1050     /* try to load something else simple and valid */
1051     b = VARIANT_FALSE;
1052     str = SysAllocString( szComplete3 );
1053     r = IXMLDOMDocument_loadXML( doc, str, &b );
1054     ok( r == S_OK, "loadXML failed\n");
1055     ok( b == VARIANT_TRUE, "failed to load XML string\n");
1056     SysFreeString( str );
1057
1058     /* try something a little more complicated */
1059     b = FALSE;
1060     str = SysAllocString( szComplete4 );
1061     r = IXMLDOMDocument_loadXML( doc, str, &b );
1062     ok( r == S_OK, "loadXML failed\n");
1063     ok( b == VARIANT_TRUE, "failed to load XML string\n");
1064     SysFreeString( str );
1065
1066     r = IXMLDOMDocument_get_parseError( doc, &error );
1067     ok( r == S_OK, "returns %08x\n", r );
1068
1069     r = IXMLDOMParseError_get_errorCode( error, &code );
1070     ok( r == S_FALSE, "returns %08x\n", r );
1071     ok( code == 0, "code %d\n", code );
1072     IXMLDOMParseError_Release( error );
1073
1074     /* test createTextNode */
1075     r = IXMLDOMDocument_createTextNode(doc, _bstr_(""), &nodetext);
1076     ok( r == S_OK, "returns %08x\n", r );
1077     IXMLDOMText_Release(nodetext);
1078
1079     str = SysAllocString( szOpen );
1080     r = IXMLDOMDocument_createTextNode(doc, str, NULL);
1081     ok( r == E_INVALIDARG, "returns %08x\n", r );
1082     r = IXMLDOMDocument_createTextNode(doc, str, &nodetext);
1083     ok( r == S_OK, "returns %08x\n", r );
1084     SysFreeString( str );
1085     if(nodetext)
1086     {
1087         IXMLDOMNamedNodeMap *pAttribs;
1088
1089         r = IXMLDOMText_QueryInterface(nodetext, &IID_IXMLDOMElement, (void**)&element);
1090         ok(r == E_NOINTERFACE, "ret %08x\n", r );
1091
1092         /* Text Last Child Checks */
1093         r = IXMLDOMText_get_lastChild(nodetext, NULL);
1094         ok(r == E_INVALIDARG, "ret %08x\n", r );
1095
1096         nodeChild = (IXMLDOMNode*)0x1;
1097         r = IXMLDOMText_get_lastChild(nodetext, &nodeChild);
1098         ok(r == S_FALSE, "ret %08x\n", r );
1099         ok(nodeChild == NULL, "nodeChild not NULL\n");
1100
1101         /* test get_attributes */
1102         r = IXMLDOMText_get_attributes( nodetext, NULL );
1103         ok( r == E_INVALIDARG, "get_attributes returned wrong code\n");
1104
1105         pAttribs = (IXMLDOMNamedNodeMap*)0x1;
1106         r = IXMLDOMText_get_attributes( nodetext, &pAttribs);
1107         ok(r == S_FALSE, "ret %08x\n", r );
1108         ok( pAttribs == NULL, "pAttribs not NULL\n");
1109
1110         /* test get_dataType */
1111         r = IXMLDOMText_get_dataType(nodetext, &var);
1112         ok(r == S_FALSE, "ret %08x\n", r );
1113         ok( V_VT(&var) == VT_NULL, "incorrect dataType type\n");
1114         VariantClear(&var);
1115
1116         /* test length property */
1117         r = IXMLDOMText_get_length(nodetext, NULL);
1118         ok(r == E_INVALIDARG, "ret %08x\n", r );
1119
1120         r = IXMLDOMText_get_length(nodetext, &nLength);
1121         ok(r == S_OK, "ret %08x\n", r );
1122         ok(nLength == 4, "expected 4 got %d\n", nLength);
1123
1124         /* test nodeTypeString */
1125         r = IXMLDOMText_get_nodeTypeString(nodetext, &str);
1126         ok(r == S_OK, "ret %08x\n", r );
1127         ok( !lstrcmpW( str, _bstr_("text") ), "incorrect nodeTypeString string\n");
1128         SysFreeString(str);
1129
1130         /* put data Tests */
1131         r = IXMLDOMText_put_data(nodetext, _bstr_("This &is a ; test <>\\"));
1132         ok(r == S_OK, "ret %08x\n", r );
1133
1134         /* get data Tests */
1135         r = IXMLDOMText_get_data(nodetext, &str);
1136         ok(r == S_OK, "ret %08x\n", r );
1137         ok( !lstrcmpW( str, _bstr_("This &is a ; test <>\\") ), "incorrect put_data string\n");
1138         SysFreeString(str);
1139
1140         /* Confirm XML text is good */
1141         r = IXMLDOMText_get_xml(nodetext, &str);
1142         ok(r == S_OK, "ret %08x\n", r );
1143         ok( !lstrcmpW( str, _bstr_("This &amp;is a ; test &lt;&gt;\\") ), "incorrect xml string\n");
1144         SysFreeString(str);
1145
1146         /* Confirm we get the put_data Text back */
1147         r = IXMLDOMText_get_text(nodetext, &str);
1148         ok(r == S_OK, "ret %08x\n", r );
1149         ok( !lstrcmpW( str, _bstr_("This &is a ; test <>\\") ), "incorrect xml string\n");
1150         SysFreeString(str);
1151
1152         /* test substringData */
1153         r = IXMLDOMText_substringData(nodetext, 0, 4, NULL);
1154         ok(r == E_INVALIDARG, "ret %08x\n", r );
1155
1156         /* test substringData - Invalid offset */
1157         str = (BSTR)&szElement;
1158         r = IXMLDOMText_substringData(nodetext, -1, 4, &str);
1159         ok(r == E_INVALIDARG, "ret %08x\n", r );
1160         ok( str == NULL, "incorrect string\n");
1161
1162         /* test substringData - Invalid offset */
1163         str = (BSTR)&szElement;
1164         r = IXMLDOMText_substringData(nodetext, 30, 0, &str);
1165         ok(r == S_FALSE, "ret %08x\n", r );
1166         ok( str == NULL, "incorrect string\n");
1167
1168         /* test substringData - Invalid size */
1169         str = (BSTR)&szElement;
1170         r = IXMLDOMText_substringData(nodetext, 0, -1, &str);
1171         ok(r == E_INVALIDARG, "ret %08x\n", r );
1172         ok( str == NULL, "incorrect string\n");
1173
1174         /* test substringData - Invalid size */
1175         str = (BSTR)&szElement;
1176         r = IXMLDOMText_substringData(nodetext, 2, 0, &str);
1177         ok(r == S_FALSE, "ret %08x\n", r );
1178         ok( str == NULL, "incorrect string\n");
1179
1180         /* test substringData - Start of string */
1181         r = IXMLDOMText_substringData(nodetext, 0, 4, &str);
1182         ok(r == S_OK, "ret %08x\n", r );
1183         ok( !lstrcmpW( str, _bstr_("This") ), "incorrect substringData string\n");
1184         SysFreeString(str);
1185
1186         /* test substringData - Middle of string */
1187         r = IXMLDOMText_substringData(nodetext, 13, 4, &str);
1188         ok(r == S_OK, "ret %08x\n", r );
1189         ok( !lstrcmpW( str, _bstr_("test") ), "incorrect substringData string\n");
1190         SysFreeString(str);
1191
1192         /* test substringData - End of string */
1193         r = IXMLDOMText_substringData(nodetext, 20, 4, &str);
1194         ok(r == S_OK, "ret %08x\n", r );
1195         ok( !lstrcmpW( str, _bstr_("\\") ), "incorrect substringData string\n");
1196         SysFreeString(str);
1197
1198         /* test appendData */
1199         r = IXMLDOMText_appendData(nodetext, NULL);
1200         ok(r == S_OK, "ret %08x\n", r );
1201
1202         r = IXMLDOMText_appendData(nodetext, _bstr_(""));
1203         ok(r == S_OK, "ret %08x\n", r );
1204
1205         r = IXMLDOMText_appendData(nodetext, _bstr_("Append"));
1206         ok(r == S_OK, "ret %08x\n", r );
1207
1208         r = IXMLDOMText_get_text(nodetext, &str);
1209         ok(r == S_OK, "ret %08x\n", r );
1210         ok( !lstrcmpW( str, _bstr_("This &is a ; test <>\\Append") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
1211         SysFreeString(str);
1212
1213         /* test insertData */
1214         str = SysAllocStringLen(NULL, 0);
1215         r = IXMLDOMText_insertData(nodetext, -1, str);
1216         ok(r == S_OK, "ret %08x\n", r );
1217
1218         r = IXMLDOMText_insertData(nodetext, -1, NULL);
1219         ok(r == S_OK, "ret %08x\n", r );
1220
1221         r = IXMLDOMText_insertData(nodetext, 1000, str);
1222         ok(r == S_OK, "ret %08x\n", r );
1223
1224         r = IXMLDOMText_insertData(nodetext, 1000, NULL);
1225         ok(r == S_OK, "ret %08x\n", r );
1226
1227         r = IXMLDOMText_insertData(nodetext, 0, NULL);
1228         ok(r == S_OK, "ret %08x\n", r );
1229
1230         r = IXMLDOMText_insertData(nodetext, 0, str);
1231         ok(r == S_OK, "ret %08x\n", r );
1232         SysFreeString(str);
1233
1234         r = IXMLDOMText_insertData(nodetext, -1, _bstr_("Inserting"));
1235         ok(r == E_INVALIDARG, "ret %08x\n", r );
1236
1237         r = IXMLDOMText_insertData(nodetext, 1000, _bstr_("Inserting"));
1238         ok(r == E_INVALIDARG, "ret %08x\n", r );
1239
1240         r = IXMLDOMText_insertData(nodetext, 0, _bstr_("Begin "));
1241         ok(r == S_OK, "ret %08x\n", r );
1242
1243         r = IXMLDOMText_insertData(nodetext, 17, _bstr_("Middle"));
1244         ok(r == S_OK, "ret %08x\n", r );
1245
1246         r = IXMLDOMText_insertData(nodetext, 39, _bstr_(" End"));
1247         ok(r == S_OK, "ret %08x\n", r );
1248
1249         r = IXMLDOMText_get_text(nodetext, &str);
1250         ok(r == S_OK, "ret %08x\n", r );
1251         ok( !lstrcmpW( str, _bstr_("Begin This &is a Middle; test <>\\Append End") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
1252         SysFreeString(str);
1253
1254         /* delete data */
1255         /* invalid arguments */
1256         r = IXMLDOMText_deleteData(nodetext, -1, 1);
1257         ok(r == E_INVALIDARG, "ret %08x\n", r );
1258
1259         r = IXMLDOMText_deleteData(nodetext, 0, 0);
1260         ok(r == S_OK, "ret %08x\n", r );
1261
1262         r = IXMLDOMText_deleteData(nodetext, 0, -1);
1263         ok(r == E_INVALIDARG, "ret %08x\n", r );
1264
1265         r = IXMLDOMText_get_length(nodetext, &nLength);
1266         ok(r == S_OK, "ret %08x\n", r );
1267         ok(nLength == 43, "expected 43 got %d\n", nLength);
1268
1269         r = IXMLDOMText_deleteData(nodetext, nLength, 1);
1270         ok(r == S_OK, "ret %08x\n", r );
1271
1272         r = IXMLDOMText_deleteData(nodetext, nLength+1, 1);
1273         ok(r == E_INVALIDARG, "ret %08x\n", r );
1274
1275         /* delete from start */
1276         r = IXMLDOMText_deleteData(nodetext, 0, 5);
1277         ok(r == S_OK, "ret %08x\n", r );
1278
1279         r = IXMLDOMText_get_length(nodetext, &nLength);
1280         ok(r == S_OK, "ret %08x\n", r );
1281         ok(nLength == 38, "expected 38 got %d\n", nLength);
1282
1283         r = IXMLDOMText_get_text(nodetext, &str);
1284         ok(r == S_OK, "ret %08x\n", r );
1285         ok( !lstrcmpW( str, _bstr_("This &is a Middle; test <>\\Append End") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
1286         SysFreeString(str);
1287
1288         /* delete from end */
1289         r = IXMLDOMText_deleteData(nodetext, 35, 3);
1290         ok(r == S_OK, "ret %08x\n", r );
1291
1292         r = IXMLDOMText_get_length(nodetext, &nLength);
1293         ok(r == S_OK, "ret %08x\n", r );
1294         ok(nLength == 35, "expected 35 got %d\n", nLength);
1295
1296         r = IXMLDOMText_get_text(nodetext, &str);
1297         ok(r == S_OK, "ret %08x\n", r );
1298         ok( !lstrcmpW( str, _bstr_("This &is a Middle; test <>\\Append") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
1299         SysFreeString(str);
1300
1301         /* delete from inside */
1302         r = IXMLDOMText_deleteData(nodetext, 1, 33);
1303         ok(r == S_OK, "ret %08x\n", r );
1304
1305         r = IXMLDOMText_get_length(nodetext, &nLength);
1306         ok(r == S_OK, "ret %08x\n", r );
1307         ok(nLength == 2, "expected 2 got %d\n", nLength);
1308
1309         r = IXMLDOMText_get_text(nodetext, &str);
1310         ok(r == S_OK, "ret %08x\n", r );
1311         ok( !lstrcmpW( str, _bstr_("") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
1312         SysFreeString(str);
1313
1314         /* delete whole data ... */
1315         r = IXMLDOMText_get_length(nodetext, &nLength);
1316         ok(r == S_OK, "ret %08x\n", r );
1317
1318         r = IXMLDOMText_deleteData(nodetext, 0, nLength);
1319         ok(r == S_OK, "ret %08x\n", r );
1320         /* ... and try again with empty string */
1321         r = IXMLDOMText_deleteData(nodetext, 0, nLength);
1322         ok(r == S_OK, "ret %08x\n", r );
1323
1324         /* test put_data */
1325         V_VT(&var) = VT_BSTR;
1326         V_BSTR(&var) = SysAllocString(szstr1);
1327         r = IXMLDOMText_put_nodeValue(nodetext, var);
1328         ok(r == S_OK, "ret %08x\n", r );
1329         VariantClear(&var);
1330
1331         r = IXMLDOMText_get_text(nodetext, &str);
1332         ok(r == S_OK, "ret %08x\n", r );
1333         ok( !lstrcmpW( str, szstr1 ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
1334         SysFreeString(str);
1335
1336         /* test put_data */
1337         V_VT(&var) = VT_I4;
1338         V_I4(&var) = 99;
1339         r = IXMLDOMText_put_nodeValue(nodetext, var);
1340         ok(r == S_OK, "ret %08x\n", r );
1341         VariantClear(&var);
1342
1343         r = IXMLDOMText_get_text(nodetext, &str);
1344         ok(r == S_OK, "ret %08x\n", r );
1345         ok( !lstrcmpW( str, _bstr_("99") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
1346         SysFreeString(str);
1347
1348         /* ::replaceData() */
1349         V_VT(&var) = VT_BSTR;
1350         V_BSTR(&var) = SysAllocString(szstr1);
1351         r = IXMLDOMText_put_nodeValue(nodetext, var);
1352         ok(r == S_OK, "ret %08x\n", r );
1353         VariantClear(&var);
1354
1355         r = IXMLDOMText_replaceData(nodetext, 6, 0, NULL);
1356         ok(r == E_INVALIDARG, "ret %08x\n", r );
1357         r = IXMLDOMText_get_text(nodetext, &str);
1358         ok(r == S_OK, "ret %08x\n", r );
1359         ok( !lstrcmpW( str, _bstr_("str1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
1360         SysFreeString(str);
1361
1362         r = IXMLDOMText_replaceData(nodetext, 0, 0, NULL);
1363         ok(r == S_OK, "ret %08x\n", r );
1364         r = IXMLDOMText_get_text(nodetext, &str);
1365         ok(r == S_OK, "ret %08x\n", r );
1366         ok( !lstrcmpW( str, _bstr_("str1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
1367         SysFreeString(str);
1368
1369         /* NULL pointer means delete */
1370         r = IXMLDOMText_replaceData(nodetext, 0, 1, NULL);
1371         ok(r == S_OK, "ret %08x\n", r );
1372         r = IXMLDOMText_get_text(nodetext, &str);
1373         ok(r == S_OK, "ret %08x\n", r );
1374         ok( !lstrcmpW( str, _bstr_("tr1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
1375         SysFreeString(str);
1376
1377         /* empty string means delete */
1378         r = IXMLDOMText_replaceData(nodetext, 0, 1, _bstr_(""));
1379         ok(r == S_OK, "ret %08x\n", r );
1380         r = IXMLDOMText_get_text(nodetext, &str);
1381         ok(r == S_OK, "ret %08x\n", r );
1382         ok( !lstrcmpW( str, _bstr_("r1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
1383         SysFreeString(str);
1384
1385         /* zero count means insert */
1386         r = IXMLDOMText_replaceData(nodetext, 0, 0, _bstr_("a"));
1387         ok(r == S_OK, "ret %08x\n", r );
1388         r = IXMLDOMText_get_text(nodetext, &str);
1389         ok(r == S_OK, "ret %08x\n", r );
1390         ok( !lstrcmpW( str, _bstr_("ar1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
1391         SysFreeString(str);
1392
1393         r = IXMLDOMText_replaceData(nodetext, 0, 2, NULL);
1394         ok(r == S_OK, "ret %08x\n", r );
1395
1396         r = IXMLDOMText_insertData(nodetext, 0, _bstr_("m"));
1397         ok(r == S_OK, "ret %08x\n", r );
1398         r = IXMLDOMText_get_text(nodetext, &str);
1399         ok(r == S_OK, "ret %08x\n", r );
1400         ok( !lstrcmpW( str, _bstr_("m1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
1401         SysFreeString(str);
1402
1403         /* nonempty string, count greater than its length */
1404         r = IXMLDOMText_replaceData(nodetext, 0, 2, _bstr_("a1.2"));
1405         ok(r == S_OK, "ret %08x\n", r );
1406         r = IXMLDOMText_get_text(nodetext, &str);
1407         ok(r == S_OK, "ret %08x\n", r );
1408         ok( !lstrcmpW( str, _bstr_("a1.2") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
1409         SysFreeString(str);
1410
1411         /* nonempty string, count less than its length */
1412         r = IXMLDOMText_replaceData(nodetext, 0, 1, _bstr_("wine"));
1413         ok(r == S_OK, "ret %08x\n", r );
1414         r = IXMLDOMText_get_text(nodetext, &str);
1415         ok(r == S_OK, "ret %08x\n", r );
1416         ok( !lstrcmpW( str, _bstr_("wine1.2") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
1417         SysFreeString(str);
1418
1419         IXMLDOMText_Release( nodetext );
1420     }
1421
1422     /* test Create Comment */
1423     r = IXMLDOMDocument_createComment(doc, NULL, NULL);
1424     ok( r == E_INVALIDARG, "returns %08x\n", r );
1425     node_comment = (IXMLDOMComment*)0x1;
1426
1427     /* empty comment */
1428     r = IXMLDOMDocument_createComment(doc, _bstr_(""), &node_comment);
1429     ok( r == S_OK, "returns %08x\n", r );
1430     str = (BSTR)0x1;
1431     r = IXMLDOMComment_get_data(node_comment, &str);
1432     ok( r == S_OK, "returns %08x\n", r );
1433     ok( str && SysStringLen(str) == 0, "expected empty string data\n");
1434     IXMLDOMComment_Release(node_comment);
1435     SysFreeString(str);
1436
1437     r = IXMLDOMDocument_createComment(doc, NULL, &node_comment);
1438     ok( r == S_OK, "returns %08x\n", r );
1439     str = (BSTR)0x1;
1440     r = IXMLDOMComment_get_data(node_comment, &str);
1441     ok( r == S_OK, "returns %08x\n", r );
1442     ok( str && (SysStringLen(str) == 0), "expected empty string data\n");
1443     IXMLDOMComment_Release(node_comment);
1444     SysFreeString(str);
1445
1446     str = SysAllocString(szComment);
1447     r = IXMLDOMDocument_createComment(doc, str, &node_comment);
1448     SysFreeString(str);
1449     ok( r == S_OK, "returns %08x\n", r );
1450     if(node_comment)
1451     {
1452         /* Last Child Checks */
1453         r = IXMLDOMComment_get_lastChild(node_comment, NULL);
1454         ok(r == E_INVALIDARG, "ret %08x\n", r );
1455
1456         nodeChild = (IXMLDOMNode*)0x1;
1457         r = IXMLDOMComment_get_lastChild(node_comment, &nodeChild);
1458         ok(r == S_FALSE, "ret %08x\n", r );
1459         ok(nodeChild == NULL, "pLastChild not NULL\n");
1460
1461         /* baseName */
1462         str = (BSTR)0xdeadbeef;
1463         IXMLDOMComment_get_baseName(node_comment, &str);
1464         ok(r == S_FALSE, "ret %08x\n", r );
1465         ok(str == NULL, "Expected NULL\n");
1466
1467         IXMLDOMComment_Release( node_comment );
1468     }
1469
1470     /* test Create Attribute */
1471     str = SysAllocString(szAttribute);
1472     r = IXMLDOMDocument_createAttribute(doc, NULL, NULL);
1473     ok( r == E_INVALIDARG, "returns %08x\n", r );
1474     r = IXMLDOMDocument_createAttribute(doc, str, &node_attr);
1475     ok( r == S_OK, "returns %08x\n", r );
1476     IXMLDOMText_Release( node_attr);
1477     SysFreeString(str);
1478
1479     /* test Processing Instruction */
1480     str = SysAllocStringLen(NULL, 0);
1481     r = IXMLDOMDocument_createProcessingInstruction(doc, str, str, NULL);
1482     ok( r == E_INVALIDARG, "returns %08x\n", r );
1483     r = IXMLDOMDocument_createProcessingInstruction(doc, NULL, str, &nodePI);
1484     ok( r == E_FAIL, "returns %08x\n", r );
1485     r = IXMLDOMDocument_createProcessingInstruction(doc, str, str, &nodePI);
1486     ok( r == E_FAIL, "returns %08x\n", r );
1487     SysFreeString(str);
1488
1489     r = IXMLDOMDocument_createProcessingInstruction(doc, _bstr_("xml"), _bstr_("version=\"1.0\""), &nodePI);
1490     ok( r == S_OK, "returns %08x\n", r );
1491     if(nodePI)
1492     {
1493         /* Last Child Checks */
1494         r = IXMLDOMProcessingInstruction_get_lastChild(nodePI, NULL);
1495         ok(r == E_INVALIDARG, "ret %08x\n", r );
1496
1497         nodeChild = (IXMLDOMNode*)0x1;
1498         r = IXMLDOMProcessingInstruction_get_lastChild(nodePI, &nodeChild);
1499         ok(r == S_FALSE, "ret %08x\n", r );
1500         ok(nodeChild == NULL, "nodeChild not NULL\n");
1501
1502         r = IXMLDOMProcessingInstruction_get_dataType(nodePI, &var);
1503         ok(r == S_FALSE, "ret %08x\n", r );
1504         ok( V_VT(&var) == VT_NULL, "incorrect dataType type\n");
1505         VariantClear(&var);
1506
1507         /* test nodeName */
1508         r = IXMLDOMProcessingInstruction_get_nodeName(nodePI, &str);
1509         ok(r == S_OK, "ret %08x\n", r );
1510         ok( !lstrcmpW( str, _bstr_("xml") ), "incorrect nodeName string\n");
1511         SysFreeString(str);
1512
1513         /* test baseName */
1514         str = (BSTR)0x1;
1515         r = IXMLDOMProcessingInstruction_get_baseName(nodePI, &str);
1516         ok(r == S_OK, "ret %08x\n", r );
1517         ok( !lstrcmpW( str, _bstr_("xml") ), "incorrect nodeName string\n");
1518         SysFreeString(str);
1519
1520         /* test Target */
1521         r = IXMLDOMProcessingInstruction_get_target(nodePI, &str);
1522         ok(r == S_OK, "ret %08x\n", r );
1523         ok( !lstrcmpW( str, _bstr_("xml") ), "incorrect target string\n");
1524         SysFreeString(str);
1525
1526         /* test nodeTypeString */
1527         r = IXMLDOMProcessingInstruction_get_nodeTypeString(nodePI, &str);
1528         ok(r == S_OK, "ret %08x\n", r );
1529         ok( !lstrcmpW( str, _bstr_("processinginstruction") ), "incorrect nodeTypeString string\n");
1530         SysFreeString(str);
1531
1532         /* test get_nodeValue */
1533         r = IXMLDOMProcessingInstruction_get_nodeValue(nodePI, &var);
1534         ok(r == S_OK, "ret %08x\n", r );
1535         ok( !lstrcmpW( V_BSTR(&var), _bstr_("version=\"1.0\"") ), "incorrect data string\n");
1536         VariantClear(&var);
1537
1538         /* test get_data */
1539         r = IXMLDOMProcessingInstruction_get_data(nodePI, &str);
1540         ok(r == S_OK, "ret %08x\n", r );
1541         ok( !lstrcmpW( str, _bstr_("version=\"1.0\"") ), "incorrect data string\n");
1542         SysFreeString(str);
1543
1544         /* test put_data */
1545         r = IXMLDOMProcessingInstruction_put_data(nodePI, _bstr_("version=\"1.0\" encoding=\"UTF-8\""));
1546         ok(r == E_FAIL, "ret %08x\n", r );
1547
1548         /* test put_data */
1549         V_VT(&var) = VT_BSTR;
1550         V_BSTR(&var) = SysAllocString(szOpen);  /* Doesn't matter what the string is, cannot set an xml node. */
1551         r = IXMLDOMProcessingInstruction_put_nodeValue(nodePI, var);
1552         ok(r == E_FAIL, "ret %08x\n", r );
1553         VariantClear(&var);
1554
1555         /* test get nodeName */
1556         r = IXMLDOMProcessingInstruction_get_nodeName(nodePI, &str);
1557         ok( !lstrcmpW( str, _bstr_("xml") ), "incorrect nodeName string\n");
1558         ok(r == S_OK, "ret %08x\n", r );
1559         SysFreeString(str);
1560
1561         IXMLDOMProcessingInstruction_Release(nodePI);
1562     }
1563
1564     r = IXMLDOMDocument_QueryInterface( doc, &IID_ISupportErrorInfo, (void**)&support_error );
1565     ok( r == S_OK, "ret %08x\n", r );
1566     if(r == S_OK)
1567     {
1568         r = ISupportErrorInfo_InterfaceSupportsErrorInfo( support_error, &IID_IXMLDOMDocument );
1569         todo_wine ok( r == S_OK, "ret %08x\n", r );
1570         ISupportErrorInfo_Release( support_error );
1571     }
1572
1573     r = IXMLDOMDocument_Release( doc );
1574     ok( r == 0, "document ref count incorrect\n");
1575
1576     free_bstrs();
1577 }
1578
1579 static void test_persiststreaminit(void)
1580 {
1581     IXMLDOMDocument *doc;
1582     IPersistStreamInit *streaminit;
1583     HRESULT hr;
1584
1585     doc = create_document(&IID_IXMLDOMDocument);
1586     if (!doc) return;
1587
1588     hr = IXMLDOMDocument_QueryInterface(doc, &IID_IPersistStreamInit, (void**)&streaminit);
1589     ok( hr == S_OK, "failed with 0x%08x\n", hr );
1590
1591     hr = IPersistStreamInit_InitNew(streaminit);
1592     ok( hr == S_OK, "failed with 0x%08x\n", hr );
1593
1594     IXMLDOMDocument_Release(doc);
1595 }
1596
1597 static void test_domnode( void )
1598 {
1599     HRESULT r;
1600     IXMLDOMDocument *doc, *owner = NULL;
1601     IXMLDOMElement *element = NULL;
1602     IXMLDOMNamedNodeMap *map = NULL;
1603     IXMLDOMNode *node = NULL, *next = NULL;
1604     IXMLDOMNodeList *list = NULL;
1605     IXMLDOMAttribute *attr = NULL;
1606     DOMNodeType type = NODE_INVALID;
1607     VARIANT_BOOL b;
1608     BSTR str;
1609     VARIANT var;
1610     LONG count;
1611
1612     doc = create_document(&IID_IXMLDOMDocument);
1613     if (!doc) return;
1614
1615     b = FALSE;
1616     str = SysAllocString( szComplete4 );
1617     r = IXMLDOMDocument_loadXML( doc, str, &b );
1618     ok( r == S_OK, "loadXML failed\n");
1619     ok( b == VARIANT_TRUE, "failed to load XML string\n");
1620     SysFreeString( str );
1621
1622     b = 1;
1623     r = IXMLDOMNode_hasChildNodes( doc, &b );
1624     ok( r == S_OK, "hasChildNoes bad return\n");
1625     ok( b == VARIANT_TRUE, "hasChildNoes wrong result\n");
1626
1627     r = IXMLDOMDocument_get_documentElement( doc, &element );
1628     ok( r == S_OK, "should be a document element\n");
1629     ok( element != NULL, "should be an element\n");
1630
1631     VariantInit(&var);
1632     ok( V_VT(&var) == VT_EMPTY, "variant init failed\n");
1633
1634     r = IXMLDOMNode_get_nodeValue( doc, NULL );
1635     ok(r == E_INVALIDARG, "get_nodeValue ret %08x\n", r );
1636
1637     r = IXMLDOMNode_get_nodeValue( doc, &var );
1638     ok( r == S_FALSE, "nextNode returned wrong code\n");
1639     ok( V_VT(&var) == VT_NULL, "variant wasn't empty\n");
1640     ok( V_BSTR(&var) == NULL, "variant value wasn't null\n");
1641
1642     if (element)
1643     {
1644         owner = NULL;
1645         r = IXMLDOMNode_get_ownerDocument( element, &owner );
1646         ok( r == S_OK, "get_ownerDocument return code\n");
1647         ok( owner != doc, "get_ownerDocument return\n");
1648         IXMLDOMDocument_Release(owner);
1649
1650         type = NODE_INVALID;
1651         r = IXMLDOMNode_get_nodeType( element, &type);
1652         ok( r == S_OK, "got %08x\n", r);
1653         ok( type == NODE_ELEMENT, "node not an element\n");
1654
1655         str = NULL;
1656         r = IXMLDOMNode_get_baseName( element, &str );
1657         ok( r == S_OK, "get_baseName returned wrong code\n");
1658         ok( lstrcmpW(str,szlc) == 0, "basename was wrong\n");
1659         SysFreeString(str);
1660
1661         /* check if nodename is correct */
1662         r = IXMLDOMElement_get_nodeName( element, NULL );
1663         ok ( r == E_INVALIDARG, "get_nodeName (NULL) wrong code\n");
1664
1665         /* content doesn't matter here */
1666         str = NULL;
1667         r = IXMLDOMElement_get_nodeName( element, &str );
1668         ok ( r == S_OK, "get_nodeName wrong code\n");
1669         ok ( str != NULL, "str is null\n");
1670         ok( !lstrcmpW( str, szlc ), "incorrect nodeName\n");
1671         SysFreeString( str );
1672
1673         str = SysAllocString( szNonExistentFile );      
1674         V_VT(&var) = VT_I4;
1675         V_I4(&var) = 0x1234;
1676         r = IXMLDOMElement_getAttribute( element, str, &var );
1677         ok( r == E_FAIL, "getAttribute ret %08x\n", r );
1678         ok( V_VT(&var) == VT_NULL || V_VT(&var) == VT_EMPTY, "vt = %x\n", V_VT(&var));
1679         VariantClear(&var);
1680
1681         r = IXMLDOMElement_getAttributeNode( element, str, NULL);
1682         ok( r == E_FAIL, "getAttributeNode ret %08x\n", r );
1683
1684         attr = (IXMLDOMAttribute*)0xdeadbeef;
1685         r = IXMLDOMElement_getAttributeNode( element, str, &attr);
1686         ok( r == E_FAIL, "getAttributeNode ret %08x\n", r );
1687         ok( attr == NULL, "getAttributeNode ret %p, expected NULL\n", attr );
1688         SysFreeString( str );
1689
1690         attr = (IXMLDOMAttribute*)0xdeadbeef;
1691         str = SysAllocString( szNonExistentAttribute );
1692         r = IXMLDOMElement_getAttributeNode( element, str, &attr);
1693         ok( r == S_FALSE, "getAttributeNode ret %08x\n", r );
1694         ok( attr == NULL, "getAttributeNode ret %p, expected NULL\n", attr );
1695         SysFreeString( str );
1696
1697         str = SysAllocString( szdl );   
1698         V_VT(&var) = VT_I4;
1699         V_I4(&var) = 0x1234;
1700         r = IXMLDOMElement_getAttribute( element, str, &var );
1701         ok( r == S_OK, "getAttribute ret %08x\n", r );
1702         ok( V_VT(&var) == VT_BSTR, "vt = %x\n", V_VT(&var));
1703         ok( !lstrcmpW(V_BSTR(&var), szstr1), "wrong attr value\n");
1704         VariantClear( &var );
1705
1706         r = IXMLDOMElement_getAttribute( element, NULL, &var );
1707         ok( r == E_INVALIDARG, "getAttribute ret %08x\n", r );
1708
1709         r = IXMLDOMElement_getAttribute( element, str, NULL );
1710         ok( r == E_INVALIDARG, "getAttribute ret %08x\n", r );
1711
1712         attr = NULL;
1713         r = IXMLDOMElement_getAttributeNode( element, str, &attr);
1714         ok( r == S_OK, "GetAttributeNode ret %08x\n", r );
1715         ok( attr != NULL, "getAttributeNode returned NULL\n" );
1716         if (attr)
1717         {
1718             r = IXMLDOMAttribute_get_parentNode( attr, NULL );
1719             ok( r == E_INVALIDARG, "Expected E_INVALIDARG, ret %08x\n", r );
1720
1721             /* attribute doesn't have a parent in msxml interpretation */
1722             node = (IXMLDOMNode*)0xdeadbeef;
1723             r = IXMLDOMAttribute_get_parentNode( attr, &node );
1724             ok( r == S_FALSE, "Expected S_FALSE, ret %08x\n", r );
1725             ok( node == NULL, "Expected NULL, got %p\n", node );
1726
1727             IXMLDOMAttribute_Release(attr);
1728         }
1729
1730         SysFreeString( str );
1731
1732         r = IXMLDOMElement_get_attributes( element, &map );
1733         ok( r == S_OK, "get_attributes returned wrong code\n");
1734         ok( map != NULL, "should be attributes\n");
1735
1736         b = 1;
1737         r = IXMLDOMNode_hasChildNodes( element, &b );
1738         ok( r == S_OK, "hasChildNoes bad return\n");
1739         ok( b == VARIANT_TRUE, "hasChildNoes wrong result\n");
1740     }
1741     else
1742         ok( FALSE, "no element\n");
1743
1744     if (map)
1745     {
1746         ISupportErrorInfo *support_error;
1747         r = IXMLDOMNamedNodeMap_QueryInterface( map, &IID_ISupportErrorInfo, (void**)&support_error );
1748         ok( r == S_OK, "ret %08x\n", r );
1749
1750         r = ISupportErrorInfo_InterfaceSupportsErrorInfo( support_error, &IID_IXMLDOMNamedNodeMap );
1751 todo_wine
1752 {
1753         ok( r == S_OK, "ret %08x\n", r );
1754 }
1755         ISupportErrorInfo_Release( support_error );
1756
1757         str = SysAllocString( szdl );
1758         r = IXMLDOMNamedNodeMap_getNamedItem( map, str, &node );
1759         ok( r == S_OK, "getNamedItem returned wrong code\n");
1760         ok( node != NULL, "should be attributes\n");
1761         IXMLDOMNode_Release(node);
1762         SysFreeString( str );
1763
1764         str = SysAllocString( szdl );
1765         r = IXMLDOMNamedNodeMap_getNamedItem( map, str, NULL );
1766         ok( r == E_INVALIDARG, "getNamedItem should return E_INVALIDARG\n");
1767         SysFreeString( str );
1768
1769         /* something that isn't in szComplete4 */
1770         str = SysAllocString( szOpen );
1771         node = (IXMLDOMNode *) 1;
1772         r = IXMLDOMNamedNodeMap_getNamedItem( map, str, &node );
1773         ok( r == S_FALSE, "getNamedItem found a node that wasn't there\n");
1774         ok( node == NULL, "getNamedItem should have returned NULL\n");
1775         SysFreeString( str );
1776
1777         /* test indexed access of attributes */
1778         r = IXMLDOMNamedNodeMap_get_length( map, NULL );
1779         ok ( r == E_INVALIDARG, "get_length should return E_INVALIDARG\n");
1780
1781         r = IXMLDOMNamedNodeMap_get_length( map, &count );
1782         ok ( r == S_OK, "get_length wrong code\n");
1783         ok ( count == 1, "get_length != 1\n");
1784
1785         node = NULL;
1786         r = IXMLDOMNamedNodeMap_get_item( map, -1, &node);
1787         ok ( r == S_FALSE, "get_item (-1) wrong code\n");
1788         ok ( node == NULL, "there is no node\n");
1789
1790         node = NULL;
1791         r = IXMLDOMNamedNodeMap_get_item( map, 1, &node);
1792         ok ( r == S_FALSE, "get_item (1) wrong code\n");
1793         ok ( node == NULL, "there is no attribute\n");
1794
1795         node = NULL;
1796         r = IXMLDOMNamedNodeMap_get_item( map, 0, &node);
1797         ok ( r == S_OK, "get_item (0) wrong code\n");
1798         ok ( node != NULL, "should be attribute\n");
1799
1800         r = IXMLDOMNode_get_nodeName( node, NULL );
1801         ok ( r == E_INVALIDARG, "get_nodeName (NULL) wrong code\n");
1802
1803         /* content doesn't matter here */
1804         str = NULL;
1805         r = IXMLDOMNode_get_nodeName( node, &str );
1806         ok ( r == S_OK, "get_nodeName wrong code\n");
1807         ok ( str != NULL, "str is null\n");
1808         ok( !lstrcmpW( str, szdl ), "incorrect node name\n");
1809         SysFreeString( str );
1810         IXMLDOMNode_Release( node );
1811
1812         /* test sequential access of attributes */
1813         node = NULL;
1814         r = IXMLDOMNamedNodeMap_nextNode( map, &node );
1815         ok ( r == S_OK, "nextNode (first time) wrong code\n");
1816         ok ( node != NULL, "nextNode, should be attribute\n");
1817         IXMLDOMNode_Release( node );
1818
1819         r = IXMLDOMNamedNodeMap_nextNode( map, &node );
1820         ok ( r != S_OK, "nextNode (second time) wrong code\n");
1821         ok ( node == NULL, "nextNode, there is no attribute\n");
1822
1823         r = IXMLDOMNamedNodeMap_reset( map );
1824         ok ( r == S_OK, "reset should return S_OK\n");
1825
1826         r = IXMLDOMNamedNodeMap_nextNode( map, &node );
1827         ok ( r == S_OK, "nextNode (third time) wrong code\n");
1828         ok ( node != NULL, "nextNode, should be attribute\n");
1829     }
1830     else
1831         ok( FALSE, "no map\n");
1832
1833     if (node)
1834     {
1835         type = NODE_INVALID;
1836         r = IXMLDOMNode_get_nodeType( node, &type);
1837         ok( r == S_OK, "getNamedItem returned wrong code\n");
1838         ok( type == NODE_ATTRIBUTE, "node not an attribute\n");
1839
1840         str = NULL;
1841         r = IXMLDOMNode_get_baseName( node, NULL );
1842         ok( r == E_INVALIDARG, "get_baseName returned wrong code\n");
1843
1844         str = NULL;
1845         r = IXMLDOMNode_get_baseName( node, &str );
1846         ok( r == S_OK, "get_baseName returned wrong code\n");
1847         ok( lstrcmpW(str,szdl) == 0, "basename was wrong\n");
1848         SysFreeString( str );
1849
1850         r = IXMLDOMNode_get_nodeValue( node, &var );
1851         ok( r == S_OK, "returns %08x\n", r );
1852         ok( V_VT(&var) == VT_BSTR, "vt %x\n", V_VT(&var));
1853         ok( !lstrcmpW(V_BSTR(&var), szstr1), "nodeValue incorrect\n");
1854         VariantClear(&var);
1855
1856         r = IXMLDOMNode_get_childNodes( node, NULL );
1857         ok( r == E_INVALIDARG, "get_childNodes returned wrong code\n");
1858
1859         r = IXMLDOMNode_get_childNodes( node, &list );
1860         ok( r == S_OK, "get_childNodes returned wrong code\n");
1861
1862         if (list)
1863         {
1864             r = IXMLDOMNodeList_nextNode( list, &next );
1865             ok( r == S_OK, "nextNode returned wrong code\n");
1866         }
1867         else
1868             ok( FALSE, "no childlist\n");
1869
1870         if (next)
1871         {
1872             b = 1;
1873             r = IXMLDOMNode_hasChildNodes( next, &b );
1874             ok( r == S_FALSE, "hasChildNoes bad return\n");
1875             ok( b == VARIANT_FALSE, "hasChildNoes wrong result\n");
1876
1877             type = NODE_INVALID;
1878             r = IXMLDOMNode_get_nodeType( next, &type);
1879             ok( r == S_OK, "getNamedItem returned wrong code\n");
1880             ok( type == NODE_TEXT, "node not text\n");
1881
1882             str = (BSTR) 1;
1883             r = IXMLDOMNode_get_baseName( next, &str );
1884             ok( r == S_FALSE, "get_baseName returned wrong code\n");
1885             ok( str == NULL, "basename was wrong\n");
1886             SysFreeString(str);
1887         }
1888         else
1889             ok( FALSE, "no next\n");
1890
1891         if (next)
1892             IXMLDOMNode_Release( next );
1893         next = NULL;
1894         if (list)
1895             IXMLDOMNodeList_Release( list );
1896         list = NULL;
1897         if (node)
1898             IXMLDOMNode_Release( node );
1899     }
1900     else
1901         ok( FALSE, "no node\n");
1902     node = NULL;
1903
1904     if (map)
1905         IXMLDOMNamedNodeMap_Release( map );
1906
1907     /* now traverse the tree from the root element */
1908     if (element)
1909     {
1910         r = IXMLDOMNode_get_childNodes( element, &list );
1911         ok( r == S_OK, "get_childNodes returned wrong code\n");
1912
1913         /* using get_item for child list doesn't advance the position */
1914         ole_check(IXMLDOMNodeList_get_item(list, 1, &node));
1915         expect_node(node, "E2.E2.D1");
1916         IXMLDOMNode_Release(node);
1917         ole_check(IXMLDOMNodeList_nextNode(list, &node));
1918         expect_node(node, "E1.E2.D1");
1919         IXMLDOMNode_Release(node);
1920         ole_check(IXMLDOMNodeList_reset(list));
1921
1922         IXMLDOMNodeList_AddRef(list);
1923         expect_list_and_release(list, "E1.E2.D1 E2.E2.D1 E3.E2.D1 E4.E2.D1");
1924         ole_check(IXMLDOMNodeList_reset(list));
1925
1926         node = (void*)0xdeadbeef;
1927         str = SysAllocString(szdl);
1928         r = IXMLDOMNode_selectSingleNode( element, str, &node );
1929         SysFreeString(str);
1930         ok( r == S_FALSE, "ret %08x\n", r );
1931         ok( node == NULL, "node %p\n", node );
1932
1933         str = SysAllocString(szbs);
1934         r = IXMLDOMNode_selectSingleNode( element, str, &node );
1935         SysFreeString(str);
1936         ok( r == S_OK, "ret %08x\n", r );
1937         r = IXMLDOMNode_Release( node );
1938         ok( r == 0, "ret %08x\n", r );
1939     }
1940     else
1941         ok( FALSE, "no element\n");
1942
1943     if (list)
1944     {
1945         r = IXMLDOMNodeList_QueryInterface(list, &IID_IDispatch, NULL);
1946         ok( r == E_INVALIDARG || r == E_POINTER, "ret %08x\n", r );
1947
1948         r = IXMLDOMNodeList_get_item(list, 0, NULL);
1949         ok(r == E_INVALIDARG, "Exected E_INVALIDARG got %08x\n", r);
1950
1951         r = IXMLDOMNodeList_get_length(list, NULL);
1952         ok(r == E_INVALIDARG, "Exected E_INVALIDARG got %08x\n", r);
1953
1954         r = IXMLDOMNodeList_get_length( list, &count );
1955         ok( r == S_OK, "get_length returns %08x\n", r );
1956         ok( count == 4, "get_length got %d\n", count );
1957
1958         r = IXMLDOMNodeList_nextNode(list, NULL);
1959         ok(r == E_INVALIDARG, "Exected E_INVALIDARG got %08x\n", r);
1960
1961         r = IXMLDOMNodeList_nextNode( list, &node );
1962         ok( r == S_OK, "nextNode returned wrong code\n");
1963     }
1964     else
1965         ok( FALSE, "no list\n");
1966
1967     if (node)
1968     {
1969         type = NODE_INVALID;
1970         r = IXMLDOMNode_get_nodeType( node, &type);
1971         ok( r == S_OK, "getNamedItem returned wrong code\n");
1972         ok( type == NODE_ELEMENT, "node not text\n");
1973
1974         VariantInit(&var);
1975         ok( V_VT(&var) == VT_EMPTY, "variant init failed\n");
1976         r = IXMLDOMNode_get_nodeValue( node, &var );
1977         ok( r == S_FALSE, "nextNode returned wrong code\n");
1978         ok( V_VT(&var) == VT_NULL, "variant wasn't empty\n");
1979         ok( V_BSTR(&var) == NULL, "variant value wasn't null\n");
1980
1981         r = IXMLDOMNode_hasChildNodes( node, NULL );
1982         ok( r == E_INVALIDARG, "hasChildNoes bad return\n");
1983
1984         b = 1;
1985         r = IXMLDOMNode_hasChildNodes( node, &b );
1986         ok( r == S_OK, "hasChildNoes bad return\n");
1987         ok( b == VARIANT_TRUE, "hasChildNoes wrong result\n");
1988
1989         str = NULL;
1990         r = IXMLDOMNode_get_baseName( node, &str );
1991         ok( r == S_OK, "get_baseName returned wrong code\n");
1992         ok( lstrcmpW(str,szbs) == 0, "basename was wrong\n");
1993         SysFreeString(str);
1994     }
1995     else
1996         ok( FALSE, "no node\n");
1997
1998     if (node)
1999         IXMLDOMNode_Release( node );
2000     if (list)
2001         IXMLDOMNodeList_Release( list );
2002     if (element)
2003         IXMLDOMElement_Release( element );
2004
2005     b = FALSE;
2006     str = SysAllocString( szComplete5 );
2007     r = IXMLDOMDocument_loadXML( doc, str, &b );
2008     ok( r == S_OK, "loadXML failed\n");
2009     ok( b == VARIANT_TRUE, "failed to load XML string\n");
2010     SysFreeString( str );
2011
2012     b = 1;
2013     r = IXMLDOMNode_hasChildNodes( doc, &b );
2014     ok( r == S_OK, "hasChildNoes bad return\n");
2015     ok( b == VARIANT_TRUE, "hasChildNoes wrong result\n");
2016
2017     r = IXMLDOMDocument_get_documentElement( doc, &element );
2018     ok( r == S_OK, "should be a document element\n");
2019     ok( element != NULL, "should be an element\n");
2020
2021     if (element)
2022     {
2023         static const WCHAR szSSearch[] = {'S',':','s','e','a','r','c','h',0};
2024         BSTR tag = NULL;
2025
2026         /* check if the tag is correct */
2027         r = IXMLDOMElement_get_tagName( element, &tag );
2028         ok( r == S_OK, "couldn't get tag name\n");
2029         ok( tag != NULL, "tag was null\n");
2030         ok( !lstrcmpW( tag, szSSearch ), "incorrect tag name\n");
2031         SysFreeString( tag );
2032     }
2033
2034     if (element)
2035         IXMLDOMElement_Release( element );
2036     ok(IXMLDOMDocument_Release( doc ) == 0, "document is not destroyed\n");
2037 }
2038
2039 static void test_refs(void)
2040 {
2041     HRESULT r;
2042     BSTR str;
2043     VARIANT_BOOL b;
2044     IXMLDOMDocument *doc;
2045     IXMLDOMElement *element = NULL;
2046     IXMLDOMNode *node = NULL, *node2;
2047     IXMLDOMNodeList *node_list = NULL;
2048     LONG ref;
2049     IUnknown *unk, *unk2;
2050
2051     doc = create_document(&IID_IXMLDOMDocument);
2052     if (!doc) return;
2053
2054     ref = IXMLDOMDocument_Release(doc);
2055     ok( ref == 0, "ref %d\n", ref);
2056
2057     doc = create_document(&IID_IXMLDOMDocument);
2058     if (!doc) return;
2059
2060     str = SysAllocString( szComplete4 );
2061     r = IXMLDOMDocument_loadXML( doc, str, &b );
2062     ok( r == S_OK, "loadXML failed\n");
2063     ok( b == VARIANT_TRUE, "failed to load XML string\n");
2064     SysFreeString( str );
2065
2066     ref = IXMLDOMDocument_AddRef( doc );
2067     ok( ref == 2, "ref %d\n", ref );
2068     ref = IXMLDOMDocument_AddRef( doc );
2069     ok( ref == 3, "ref %d\n", ref );
2070     IXMLDOMDocument_Release( doc );
2071     IXMLDOMDocument_Release( doc );
2072
2073     r = IXMLDOMDocument_get_documentElement( doc, &element );
2074     ok( r == S_OK, "should be a document element\n");
2075     ok( element != NULL, "should be an element\n");
2076
2077     ref = IXMLDOMDocument_AddRef( doc );
2078     ok( ref == 2, "ref %d\n", ref );
2079     IXMLDOMDocument_Release( doc );
2080
2081     r = IXMLDOMElement_get_childNodes( element, &node_list );
2082     ok( r == S_OK, "rets %08x\n", r);
2083
2084     ref = IXMLDOMNodeList_AddRef( node_list );
2085     ok( ref == 2, "ref %d\n", ref );
2086     IXMLDOMNodeList_Release( node_list );
2087
2088     IXMLDOMNodeList_get_item( node_list, 0, &node );
2089     ok( r == S_OK, "rets %08x\n", r);
2090
2091     IXMLDOMNodeList_get_item( node_list, 0, &node2 );
2092     ok( r == S_OK, "rets %08x\n", r);
2093
2094     ref = IXMLDOMNode_AddRef( node );
2095     ok( ref == 2, "ref %d\n", ref );
2096     IXMLDOMNode_Release( node );
2097
2098     ref = IXMLDOMNode_Release( node );
2099     ok( ref == 0, "ref %d\n", ref );
2100     ref = IXMLDOMNode_Release( node2 );
2101     ok( ref == 0, "ref %d\n", ref );
2102
2103     ref = IXMLDOMNodeList_Release( node_list );
2104     ok( ref == 0, "ref %d\n", ref );
2105
2106     ok( node != node2, "node %p node2 %p\n", node, node2 );
2107
2108     ref = IXMLDOMDocument_Release( doc );
2109     ok( ref == 0, "ref %d\n", ref );
2110
2111     ref = IXMLDOMElement_AddRef( element );
2112     todo_wine {
2113     ok( ref == 3, "ref %d\n", ref );
2114     }
2115     IXMLDOMElement_Release( element );
2116
2117     /* IUnknown must be unique however we obtain it */
2118     r = IXMLDOMElement_QueryInterface( element, &IID_IUnknown, (void**)&unk );
2119     ok( r == S_OK, "rets %08x\n", r );
2120     r = IXMLDOMElement_QueryInterface( element, &IID_IXMLDOMNode, (void**)&node );
2121     ok( r == S_OK, "rets %08x\n", r );
2122     r = IXMLDOMNode_QueryInterface( node, &IID_IUnknown, (void**)&unk2 );
2123     ok( r == S_OK, "rets %08x\n", r );
2124     ok( unk == unk2, "unk %p unk2 %p\n", unk, unk2 );
2125
2126     IUnknown_Release( unk2 );
2127     IUnknown_Release( unk );
2128     IXMLDOMNode_Release( node );
2129
2130     IXMLDOMElement_Release( element );
2131
2132 }
2133
2134 static void test_create(void)
2135 {
2136     static const WCHAR szOne[] = {'1',0};
2137     static const WCHAR szOneGarbage[] = {'1','G','a','r','b','a','g','e',0};
2138     HRESULT r;
2139     VARIANT var;
2140     BSTR str, name;
2141     IXMLDOMDocument *doc;
2142     IXMLDOMElement *element;
2143     IXMLDOMComment *comment;
2144     IXMLDOMText *text;
2145     IXMLDOMCDATASection *cdata;
2146     IXMLDOMNode *root, *node, *child;
2147     IXMLDOMNamedNodeMap *attr_map;
2148     IUnknown *unk;
2149     LONG ref;
2150     LONG num;
2151
2152     doc = create_document(&IID_IXMLDOMDocument);
2153     if (!doc) return;
2154
2155     /* types not supported for creation */
2156     V_VT(&var) = VT_I1;
2157     V_I1(&var) = NODE_DOCUMENT;
2158     node = (IXMLDOMNode*)0x1;
2159     r = IXMLDOMDocument_createNode( doc, var, NULL, NULL, &node );
2160     ok( r == E_INVALIDARG, "returns %08x\n", r );
2161     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
2162
2163     V_VT(&var) = VT_I1;
2164     V_I1(&var) = NODE_DOCUMENT_TYPE;
2165     node = (IXMLDOMNode*)0x1;
2166     r = IXMLDOMDocument_createNode( doc, var, NULL, NULL, &node );
2167     ok( r == E_INVALIDARG, "returns %08x\n", r );
2168     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
2169
2170     V_VT(&var) = VT_I1;
2171     V_I1(&var) = NODE_ENTITY;
2172     node = (IXMLDOMNode*)0x1;
2173     r = IXMLDOMDocument_createNode( doc, var, NULL, NULL, &node );
2174     ok( r == E_INVALIDARG, "returns %08x\n", r );
2175     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
2176
2177     V_VT(&var) = VT_I1;
2178     V_I1(&var) = NODE_NOTATION;
2179     node = (IXMLDOMNode*)0x1;
2180     r = IXMLDOMDocument_createNode( doc, var, NULL, NULL, &node );
2181     ok( r == E_INVALIDARG, "returns %08x\n", r );
2182     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
2183
2184     /* NODE_COMMENT */
2185     V_VT(&var) = VT_I1;
2186     V_I1(&var) = NODE_COMMENT;
2187     node = NULL;
2188     r = IXMLDOMDocument_createNode( doc, var, NULL, NULL, &node );
2189     ok( r == S_OK, "returns %08x\n", r );
2190     ok( node != NULL, "\n");
2191
2192     r = IXMLDOMNode_QueryInterface(node, &IID_IXMLDOMComment, (void**)&comment);
2193     ok( r == S_OK, "returns %08x\n", r );
2194     IXMLDOMNode_Release(node);
2195
2196     str = NULL;
2197     r = IXMLDOMComment_get_data(comment, &str);
2198     ok( r == S_OK, "returns %08x\n", r );
2199     ok( str && SysStringLen(str) == 0, "expected empty comment, %p\n", str);
2200     IXMLDOMComment_Release(comment);
2201     SysFreeString(str);
2202
2203     node = (IXMLDOMNode*)0x1;
2204     r = IXMLDOMDocument_createNode( doc, var, _bstr_(""), NULL, &node );
2205     ok( r == S_OK, "returns %08x\n", r );
2206
2207     r = IXMLDOMNode_QueryInterface(node, &IID_IXMLDOMComment, (void**)&comment);
2208     ok( r == S_OK, "returns %08x\n", r );
2209     IXMLDOMNode_Release(node);
2210
2211     str = NULL;
2212     r = IXMLDOMComment_get_data(comment, &str);
2213     ok( r == S_OK, "returns %08x\n", r );
2214     ok( str && SysStringLen(str) == 0, "expected empty comment, %p\n", str);
2215     IXMLDOMComment_Release(comment);
2216     SysFreeString(str);
2217
2218     node = (IXMLDOMNode*)0x1;
2219     r = IXMLDOMDocument_createNode( doc, var, _bstr_("blah"), NULL, &node );
2220     ok( r == S_OK, "returns %08x\n", r );
2221
2222     r = IXMLDOMNode_QueryInterface(node, &IID_IXMLDOMComment, (void**)&comment);
2223     ok( r == S_OK, "returns %08x\n", r );
2224     IXMLDOMNode_Release(node);
2225
2226     str = NULL;
2227     r = IXMLDOMComment_get_data(comment, &str);
2228     ok( r == S_OK, "returns %08x\n", r );
2229     ok( str && SysStringLen(str) == 0, "expected empty comment, %p\n", str);
2230     IXMLDOMComment_Release(comment);
2231     SysFreeString(str);
2232
2233     /* NODE_TEXT */
2234     V_VT(&var) = VT_I1;
2235     V_I1(&var) = NODE_TEXT;
2236     node = NULL;
2237     r = IXMLDOMDocument_createNode( doc, var, NULL, NULL, &node );
2238     ok( r == S_OK, "returns %08x\n", r );
2239     ok( node != NULL, "\n");
2240
2241     r = IXMLDOMNode_QueryInterface(node, &IID_IXMLDOMText, (void**)&text);
2242     ok( r == S_OK, "returns %08x\n", r );
2243     IXMLDOMNode_Release(node);
2244
2245     str = NULL;
2246     r = IXMLDOMText_get_data(text, &str);
2247     ok( r == S_OK, "returns %08x\n", r );
2248     ok( str && SysStringLen(str) == 0, "expected empty comment, %p\n", str);
2249     IXMLDOMText_Release(text);
2250     SysFreeString(str);
2251
2252     node = (IXMLDOMNode*)0x1;
2253     r = IXMLDOMDocument_createNode( doc, var, _bstr_(""), NULL, &node );
2254     ok( r == S_OK, "returns %08x\n", r );
2255
2256     r = IXMLDOMNode_QueryInterface(node, &IID_IXMLDOMText, (void**)&text);
2257     ok( r == S_OK, "returns %08x\n", r );
2258     IXMLDOMNode_Release(node);
2259
2260     str = NULL;
2261     r = IXMLDOMText_get_data(text, &str);
2262     ok( r == S_OK, "returns %08x\n", r );
2263     ok( str && SysStringLen(str) == 0, "expected empty comment, %p\n", str);
2264     IXMLDOMText_Release(text);
2265     SysFreeString(str);
2266
2267     node = (IXMLDOMNode*)0x1;
2268     r = IXMLDOMDocument_createNode( doc, var, _bstr_("blah"), NULL, &node );
2269     ok( r == S_OK, "returns %08x\n", r );
2270
2271     r = IXMLDOMNode_QueryInterface(node, &IID_IXMLDOMText, (void**)&text);
2272     ok( r == S_OK, "returns %08x\n", r );
2273     IXMLDOMNode_Release(node);
2274
2275     str = NULL;
2276     r = IXMLDOMText_get_data(text, &str);
2277     ok( r == S_OK, "returns %08x\n", r );
2278     ok( str && SysStringLen(str) == 0, "expected empty comment, %p\n", str);
2279     IXMLDOMText_Release(text);
2280     SysFreeString(str);
2281
2282     /* NODE_CDATA_SECTION */
2283     V_VT(&var) = VT_I1;
2284     V_I1(&var) = NODE_CDATA_SECTION;
2285     node = NULL;
2286     r = IXMLDOMDocument_createNode( doc, var, NULL, NULL, &node );
2287     ok( r == S_OK, "returns %08x\n", r );
2288     ok( node != NULL, "\n");
2289
2290     r = IXMLDOMNode_QueryInterface(node, &IID_IXMLDOMCDATASection, (void**)&cdata);
2291     ok( r == S_OK, "returns %08x\n", r );
2292     IXMLDOMNode_Release(node);
2293
2294     str = NULL;
2295     r = IXMLDOMCDATASection_get_data(cdata, &str);
2296     ok( r == S_OK, "returns %08x\n", r );
2297     ok( str && SysStringLen(str) == 0, "expected empty comment, %p\n", str);
2298     IXMLDOMCDATASection_Release(cdata);
2299     SysFreeString(str);
2300
2301     node = (IXMLDOMNode*)0x1;
2302     r = IXMLDOMDocument_createNode( doc, var, _bstr_(""), NULL, &node );
2303     ok( r == S_OK, "returns %08x\n", r );
2304
2305     r = IXMLDOMNode_QueryInterface(node, &IID_IXMLDOMCDATASection, (void**)&cdata);
2306     ok( r == S_OK, "returns %08x\n", r );
2307     IXMLDOMNode_Release(node);
2308
2309     str = NULL;
2310     r = IXMLDOMCDATASection_get_data(cdata, &str);
2311     ok( r == S_OK, "returns %08x\n", r );
2312     ok( str && SysStringLen(str) == 0, "expected empty comment, %p\n", str);
2313     IXMLDOMCDATASection_Release(cdata);
2314     SysFreeString(str);
2315
2316     node = (IXMLDOMNode*)0x1;
2317     r = IXMLDOMDocument_createNode( doc, var, _bstr_("blah"), NULL, &node );
2318     ok( r == S_OK, "returns %08x\n", r );
2319
2320     r = IXMLDOMNode_QueryInterface(node, &IID_IXMLDOMCDATASection, (void**)&cdata);
2321     ok( r == S_OK, "returns %08x\n", r );
2322     IXMLDOMNode_Release(node);
2323
2324     str = NULL;
2325     r = IXMLDOMCDATASection_get_data(cdata, &str);
2326     ok( r == S_OK, "returns %08x\n", r );
2327     ok( str && SysStringLen(str) == 0, "expected empty comment, %p\n", str);
2328     IXMLDOMCDATASection_Release(cdata);
2329     SysFreeString(str);
2330
2331     /* NODE_ATTRIBUTE */
2332     V_VT(&var) = VT_I1;
2333     V_I1(&var) = NODE_ATTRIBUTE;
2334     node = (IXMLDOMNode*)0x1;
2335     r = IXMLDOMDocument_createNode( doc, var, NULL, NULL, &node );
2336     ok( r == E_FAIL, "returns %08x\n", r );
2337     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
2338
2339     V_VT(&var) = VT_I1;
2340     V_I1(&var) = NODE_ATTRIBUTE;
2341     node = (IXMLDOMNode*)0x1;
2342     r = IXMLDOMDocument_createNode( doc, var, _bstr_(""), NULL, &node );
2343     ok( r == E_FAIL, "returns %08x\n", r );
2344     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
2345
2346     V_VT(&var) = VT_I1;
2347     V_I1(&var) = NODE_ATTRIBUTE;
2348     str = SysAllocString( szlc );
2349     r = IXMLDOMDocument_createNode( doc, var, str, NULL, &node );
2350     ok( r == S_OK, "returns %08x\n", r );
2351     if( SUCCEEDED(r) ) IXMLDOMNode_Release( node );
2352     SysFreeString(str);
2353
2354     /* a name is required for attribute, try a BSTR with first null wchar */
2355     V_VT(&var) = VT_I1;
2356     V_I1(&var) = NODE_ATTRIBUTE;
2357     str = SysAllocString( szstr1 );
2358     str[0] = 0;
2359     node = (IXMLDOMNode*)0x1;
2360     r = IXMLDOMDocument_createNode( doc, var, str, NULL, &node );
2361     ok( r == E_FAIL, "returns %08x\n", r );
2362     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
2363     SysFreeString(str);
2364
2365     /* NODE_PROCESSING_INSTRUCTION */
2366     V_VT(&var) = VT_I1;
2367     V_I1(&var) = NODE_PROCESSING_INSTRUCTION;
2368     node = (IXMLDOMNode*)0x1;
2369     r = IXMLDOMDocument_createNode( doc, var, NULL, NULL, &node );
2370     ok( r == E_FAIL, "returns %08x\n", r );
2371     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
2372
2373     V_VT(&var) = VT_I1;
2374     V_I1(&var) = NODE_PROCESSING_INSTRUCTION;
2375     node = (IXMLDOMNode*)0x1;
2376     r = IXMLDOMDocument_createNode( doc, var, _bstr_(""), NULL, &node );
2377     ok( r == E_FAIL, "returns %08x\n", r );
2378     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
2379
2380     V_VT(&var) = VT_I1;
2381     V_I1(&var) = NODE_PROCESSING_INSTRUCTION;
2382     r = IXMLDOMDocument_createNode( doc, var, _bstr_("pi"), NULL, NULL );
2383     ok( r == E_INVALIDARG, "returns %08x\n", r );
2384
2385     /* NODE_ENTITY_REFERENCE */
2386     V_VT(&var) = VT_I1;
2387     V_I1(&var) = NODE_ENTITY_REFERENCE;
2388     node = (IXMLDOMNode*)0x1;
2389     r = IXMLDOMDocument_createNode( doc, var, NULL, NULL, &node );
2390     ok( r == E_FAIL, "returns %08x\n", r );
2391     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
2392
2393     V_VT(&var) = VT_I1;
2394     V_I1(&var) = NODE_ENTITY_REFERENCE;
2395     node = (IXMLDOMNode*)0x1;
2396     r = IXMLDOMDocument_createNode( doc, var, _bstr_(""), NULL, &node );
2397     ok( r == E_FAIL, "returns %08x\n", r );
2398     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
2399
2400     /* NODE_ELEMENT */
2401     V_VT(&var) = VT_I1;
2402     V_I1(&var) = NODE_ELEMENT;
2403     node = (IXMLDOMNode*)0x1;
2404     r = IXMLDOMDocument_createNode( doc, var, NULL, NULL, &node );
2405     ok( r == E_FAIL, "returns %08x\n", r );
2406     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
2407
2408     V_VT(&var) = VT_I1;
2409     V_I1(&var) = NODE_ELEMENT;
2410     node = (IXMLDOMNode*)0x1;
2411     r = IXMLDOMDocument_createNode( doc, var, _bstr_(""), NULL, &node );
2412     ok( r == E_FAIL, "returns %08x\n", r );
2413     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
2414
2415     V_VT(&var) = VT_I1;
2416     V_I1(&var) = NODE_ELEMENT;
2417     str = SysAllocString( szlc );
2418     r = IXMLDOMDocument_createNode( doc, var, str, NULL, &node );
2419     ok( r == S_OK, "returns %08x\n", r );
2420     if( SUCCEEDED(r) ) IXMLDOMNode_Release( node );
2421
2422     V_VT(&var) = VT_I1;
2423     V_I1(&var) = NODE_ELEMENT;
2424     r = IXMLDOMDocument_createNode( doc, var, str, NULL, NULL );
2425     ok( r == E_INVALIDARG, "returns %08x\n", r );
2426
2427     V_VT(&var) = VT_R4;
2428     V_R4(&var) = NODE_ELEMENT;
2429     r = IXMLDOMDocument_createNode( doc, var, str, NULL, &node );
2430     ok( r == S_OK, "returns %08x\n", r );
2431     if( SUCCEEDED(r) ) IXMLDOMNode_Release( node );
2432
2433     V_VT(&var) = VT_BSTR;
2434     V_BSTR(&var) = SysAllocString( szOne );
2435     r = IXMLDOMDocument_createNode( doc, var, str, NULL, &node );
2436     ok( r == S_OK, "returns %08x\n", r );
2437     if( SUCCEEDED(r) ) IXMLDOMNode_Release( node );
2438     VariantClear(&var);
2439
2440     V_VT(&var) = VT_BSTR;
2441     V_BSTR(&var) = SysAllocString( szOneGarbage );
2442     r = IXMLDOMDocument_createNode( doc, var, str, NULL, &node );
2443     ok( r == E_INVALIDARG, "returns %08x\n", r );
2444     if( SUCCEEDED(r) ) IXMLDOMNode_Release( node );
2445     VariantClear(&var);
2446
2447     V_VT(&var) = VT_I4;
2448     V_I4(&var) = NODE_ELEMENT;
2449     r = IXMLDOMDocument_createNode( doc, var, str, NULL, &node );
2450     ok( r == S_OK, "returns %08x\n", r );
2451     r = IXMLDOMDocument_appendChild( doc, node, &root );
2452     ok( r == S_OK, "returns %08x\n", r );
2453     ok( node == root, "%p %p\n", node, root );
2454
2455     ref = IXMLDOMNode_AddRef( node );
2456     ok(ref == 3, "ref %d\n", ref);
2457     IXMLDOMNode_Release( node );
2458
2459     ref = IXMLDOMNode_Release( node );
2460     ok(ref == 1, "ref %d\n", ref);
2461     SysFreeString( str );
2462
2463     V_VT(&var) = VT_I4;
2464     V_I4(&var) = NODE_ELEMENT;
2465     str = SysAllocString( szbs );
2466     r = IXMLDOMDocument_createNode( doc, var, str, NULL, &node );
2467     ok( r == S_OK, "returns %08x\n", r );
2468     SysFreeString( str );
2469
2470     ref = IXMLDOMNode_AddRef( node );
2471     ok(ref == 2, "ref = %d\n", ref);
2472     IXMLDOMNode_Release( node );
2473
2474     r = IXMLDOMNode_QueryInterface( node, &IID_IUnknown, (void**)&unk );
2475     ok( r == S_OK, "returns %08x\n", r );
2476
2477     ref = IXMLDOMNode_AddRef( unk );
2478     ok(ref == 3, "ref = %d\n", ref);
2479     IXMLDOMNode_Release( unk );
2480
2481     V_VT(&var) = VT_EMPTY;
2482     r = IXMLDOMNode_insertBefore( root, (IXMLDOMNode*)unk, var, &child );
2483     ok( r == S_OK, "returns %08x\n", r );
2484     ok( unk == (IUnknown*)child, "%p %p\n", unk, child );
2485     IXMLDOMNode_Release( child );
2486     IUnknown_Release( unk );
2487
2488
2489     V_VT(&var) = VT_NULL;
2490     V_DISPATCH(&var) = (IDispatch*)node;
2491     r = IXMLDOMNode_insertBefore( root, node, var, &child );
2492     ok( r == S_OK, "returns %08x\n", r );
2493     ok( node == child, "%p %p\n", node, child );
2494     IXMLDOMNode_Release( child );
2495
2496
2497     V_VT(&var) = VT_NULL;
2498     V_DISPATCH(&var) = (IDispatch*)node;
2499     r = IXMLDOMNode_insertBefore( root, node, var, NULL );
2500     ok( r == S_OK, "returns %08x\n", r );
2501     IXMLDOMNode_Release( node );
2502
2503     r = IXMLDOMNode_QueryInterface( root, &IID_IXMLDOMElement, (void**)&element );
2504     ok( r == S_OK, "returns %08x\n", r );
2505
2506     r = IXMLDOMElement_get_attributes( element, &attr_map );
2507     ok( r == S_OK, "returns %08x\n", r );
2508     r = IXMLDOMNamedNodeMap_get_length( attr_map, &num );
2509     ok( r == S_OK, "returns %08x\n", r );
2510     ok( num == 0, "num %d\n", num );
2511     IXMLDOMNamedNodeMap_Release( attr_map );
2512
2513     V_VT(&var) = VT_BSTR;
2514     V_BSTR(&var) = SysAllocString( szstr1 );
2515     name = SysAllocString( szdl );
2516     r = IXMLDOMElement_setAttribute( element, name, var );
2517     ok( r == S_OK, "returns %08x\n", r );
2518     r = IXMLDOMElement_get_attributes( element, &attr_map );
2519     ok( r == S_OK, "returns %08x\n", r );
2520     r = IXMLDOMNamedNodeMap_get_length( attr_map, &num );
2521     ok( r == S_OK, "returns %08x\n", r );
2522     ok( num == 1, "num %d\n", num );
2523     IXMLDOMNamedNodeMap_Release( attr_map );
2524     VariantClear(&var);
2525
2526     V_VT(&var) = VT_BSTR;
2527     V_BSTR(&var) = SysAllocString( szstr2 );
2528     r = IXMLDOMElement_setAttribute( element, name, var );
2529     ok( r == S_OK, "returns %08x\n", r );
2530     r = IXMLDOMElement_get_attributes( element, &attr_map );
2531     ok( r == S_OK, "returns %08x\n", r );
2532     r = IXMLDOMNamedNodeMap_get_length( attr_map, &num );
2533     ok( r == S_OK, "returns %08x\n", r );
2534     ok( num == 1, "num %d\n", num );
2535     IXMLDOMNamedNodeMap_Release( attr_map );
2536     VariantClear(&var);
2537     r = IXMLDOMElement_getAttribute( element, name, &var );
2538     ok( r == S_OK, "returns %08x\n", r );
2539     ok( !lstrcmpW(V_BSTR(&var), szstr2), "wrong attr value\n");
2540     VariantClear(&var);
2541     SysFreeString(name);
2542
2543     V_VT(&var) = VT_BSTR;
2544     V_BSTR(&var) = SysAllocString( szstr1 );
2545     name = SysAllocString( szlc );
2546     r = IXMLDOMElement_setAttribute( element, name, var );
2547     ok( r == S_OK, "returns %08x\n", r );
2548     r = IXMLDOMElement_get_attributes( element, &attr_map );
2549     ok( r == S_OK, "returns %08x\n", r );
2550     r = IXMLDOMNamedNodeMap_get_length( attr_map, &num );
2551     ok( r == S_OK, "returns %08x\n", r );
2552     ok( num == 2, "num %d\n", num );
2553     IXMLDOMNamedNodeMap_Release( attr_map );
2554     VariantClear(&var);
2555     SysFreeString(name);
2556
2557     V_VT(&var) = VT_I4;
2558     V_I4(&var) = 10;
2559     name = SysAllocString( szbs );
2560     r = IXMLDOMElement_setAttribute( element, name, var );
2561     ok( r == S_OK, "returns %08x\n", r );
2562     VariantClear(&var);
2563     r = IXMLDOMElement_getAttribute( element, name, &var );
2564     ok( r == S_OK, "returns %08x\n", r );
2565     ok( V_VT(&var) == VT_BSTR, "variant type %x\n", V_VT(&var));
2566     VariantClear(&var);
2567     SysFreeString(name);
2568
2569     /* Create an Attribute */
2570     V_VT(&var) = VT_I4;
2571     V_I4(&var) = NODE_ATTRIBUTE;
2572     str = SysAllocString( szAttribute );
2573     r = IXMLDOMDocument_createNode( doc, var, str, NULL, &node );
2574     ok( r == S_OK, "returns %08x\n", r );
2575     ok( node != NULL, "node was null\n");
2576     SysFreeString(str);
2577
2578     r = IXMLDOMNode_get_nodeTypeString(node, &str);
2579     ok( r == S_OK, "returns %08x\n", r );
2580     ok( !lstrcmpW( str, _bstr_("attribute") ), "incorrect nodeTypeString string\n");
2581     SysFreeString(str);
2582     IXMLDOMNode_Release( node );
2583
2584     IXMLDOMElement_Release( element );
2585     IXMLDOMNode_Release( root );
2586     IXMLDOMDocument_Release( doc );
2587 }
2588
2589 static void test_getElementsByTagName(void)
2590 {
2591     IXMLDOMNodeList *node_list;
2592     IXMLDOMDocument *doc;
2593     IXMLDOMElement *elem;
2594     WCHAR buff[100];
2595     VARIANT_BOOL b;
2596     HRESULT r;
2597     LONG len;
2598     BSTR str;
2599
2600     doc = create_document(&IID_IXMLDOMDocument);
2601     if (!doc) return;
2602
2603     str = SysAllocString( szComplete4 );
2604     r = IXMLDOMDocument_loadXML( doc, str, &b );
2605     ok( r == S_OK, "loadXML failed\n");
2606     ok( b == VARIANT_TRUE, "failed to load XML string\n");
2607     SysFreeString( str );
2608
2609     str = SysAllocString( szstar );
2610
2611     /* null arguments cases */
2612     r = IXMLDOMDocument_getElementsByTagName(doc, NULL, &node_list);
2613     ok( r == E_INVALIDARG, "ret %08x\n", r );
2614     r = IXMLDOMDocument_getElementsByTagName(doc, str, NULL);
2615     ok( r == E_INVALIDARG, "ret %08x\n", r );
2616
2617     r = IXMLDOMDocument_getElementsByTagName(doc, str, &node_list);
2618     ok( r == S_OK, "ret %08x\n", r );
2619     r = IXMLDOMNodeList_get_length( node_list, &len );
2620     ok( r == S_OK, "ret %08x\n", r );
2621     ok( len == 6, "len %d\n", len );
2622
2623     test_disp((IUnknown*)node_list);
2624
2625     IXMLDOMNodeList_Release( node_list );
2626     SysFreeString( str );
2627
2628     /* broken query BSTR */
2629     memcpy(&buff[2], szstar, sizeof(szstar));
2630     /* just a big length */
2631     *(DWORD*)buff = 0xf0f0;
2632     r = IXMLDOMDocument_getElementsByTagName(doc, &buff[2], &node_list);
2633     ok( r == S_OK, "ret %08x\n", r );
2634     r = IXMLDOMNodeList_get_length( node_list, &len );
2635     ok( r == S_OK, "ret %08x\n", r );
2636     ok( len == 6, "len %d\n", len );
2637     IXMLDOMNodeList_Release( node_list );
2638
2639     str = SysAllocString( szbs );
2640     r = IXMLDOMDocument_getElementsByTagName(doc, str, &node_list);
2641     ok( r == S_OK, "ret %08x\n", r );
2642     r = IXMLDOMNodeList_get_length( node_list, &len );
2643     ok( r == S_OK, "ret %08x\n", r );
2644     ok( len == 1, "len %d\n", len );
2645     IXMLDOMNodeList_Release( node_list );
2646     SysFreeString( str );
2647
2648     str = SysAllocString( szdl );
2649     r = IXMLDOMDocument_getElementsByTagName(doc, str, &node_list);
2650     ok( r == S_OK, "ret %08x\n", r );
2651     r = IXMLDOMNodeList_get_length( node_list, &len );
2652     ok( r == S_OK, "ret %08x\n", r );
2653     ok( len == 0, "len %d\n", len );
2654     IXMLDOMNodeList_Release( node_list );
2655     SysFreeString( str );
2656
2657     str = SysAllocString( szstr1 );
2658     r = IXMLDOMDocument_getElementsByTagName(doc, str, &node_list);
2659     ok( r == S_OK, "ret %08x\n", r );
2660     r = IXMLDOMNodeList_get_length( node_list, &len );
2661     ok( r == S_OK, "ret %08x\n", r );
2662     ok( len == 0, "len %d\n", len );
2663     IXMLDOMNodeList_Release( node_list );
2664     SysFreeString( str );
2665
2666     /* test for element */
2667     r = IXMLDOMDocument_get_documentElement(doc, &elem);
2668     ok( r == S_OK, "ret %08x\n", r );
2669
2670     str = SysAllocString( szstar );
2671
2672     /* null arguments cases */
2673     r = IXMLDOMElement_getElementsByTagName(elem, NULL, &node_list);
2674     ok( r == E_INVALIDARG, "ret %08x\n", r );
2675     r = IXMLDOMElement_getElementsByTagName(elem, str, NULL);
2676     ok( r == E_INVALIDARG, "ret %08x\n", r );
2677
2678     r = IXMLDOMElement_getElementsByTagName(elem, str, &node_list);
2679     ok( r == S_OK, "ret %08x\n", r );
2680     r = IXMLDOMNodeList_get_length( node_list, &len );
2681     ok( r == S_OK, "ret %08x\n", r );
2682     ok( len == 5, "len %d\n", len );
2683     expect_list_and_release(node_list, "E1.E2.D1 E2.E2.D1 E3.E2.D1 E4.E2.D1 E1.E4.E2.D1");
2684     SysFreeString( str );
2685
2686     /* broken query BSTR */
2687     memcpy(&buff[2], szstar, sizeof(szstar));
2688     /* just a big length */
2689     *(DWORD*)buff = 0xf0f0;
2690     r = IXMLDOMElement_getElementsByTagName(elem, &buff[2], &node_list);
2691     ok( r == S_OK, "ret %08x\n", r );
2692     r = IXMLDOMNodeList_get_length( node_list, &len );
2693     ok( r == S_OK, "ret %08x\n", r );
2694     ok( len == 5, "len %d\n", len );
2695     IXMLDOMNodeList_Release( node_list );
2696
2697     IXMLDOMElement_Release(elem);
2698
2699     IXMLDOMDocument_Release( doc );
2700 }
2701
2702 static void test_get_text(void)
2703 {
2704     HRESULT r;
2705     BSTR str;
2706     VARIANT_BOOL b;
2707     IXMLDOMDocument *doc;
2708     IXMLDOMNode *node, *node2, *node3;
2709     IXMLDOMNode *nodeRoot;
2710     IXMLDOMNodeList *node_list;
2711     IXMLDOMNamedNodeMap *node_map;
2712     LONG len;
2713
2714     doc = create_document(&IID_IXMLDOMDocument);
2715     if (!doc) return;
2716
2717     str = SysAllocString( szComplete4 );
2718     r = IXMLDOMDocument_loadXML( doc, str, &b );
2719     ok( r == S_OK, "loadXML failed\n");
2720     ok( b == VARIANT_TRUE, "failed to load XML string\n");
2721     SysFreeString( str );
2722
2723     str = SysAllocString( szbs );
2724     r = IXMLDOMDocument_getElementsByTagName( doc, str, &node_list );
2725     ok( r == S_OK, "ret %08x\n", r );
2726     SysFreeString(str);
2727
2728     /* Test to get all child node text. */
2729     r = IXMLDOMDocument_QueryInterface(doc, &IID_IXMLDOMNode, (void**)&nodeRoot);
2730     ok( r == S_OK, "ret %08x\n", r );
2731     if(r == S_OK)
2732     {
2733         r = IXMLDOMNode_get_text( nodeRoot, &str );
2734         ok( r == S_OK, "ret %08x\n", r );
2735         ok( compareIgnoreReturns(str, _bstr_("fn1.txt\n\n fn2.txt \n\nf1\n")), "wrong get_text: %s\n", wine_dbgstr_w(str));
2736         SysFreeString(str);
2737
2738         IXMLDOMNode_Release(nodeRoot);
2739     }
2740
2741     if (0) {
2742     /* this test crashes on win9x */
2743     r = IXMLDOMNodeList_QueryInterface(node_list, &IID_IDispatch, NULL);
2744     ok( r == E_INVALIDARG, "ret %08x\n", r );
2745     }
2746
2747     r = IXMLDOMNodeList_get_length( node_list, NULL );
2748     ok( r == E_INVALIDARG, "ret %08x\n", r );
2749
2750     r = IXMLDOMNodeList_get_length( node_list, &len );
2751     ok( r == S_OK, "ret %08x\n", r );
2752     ok( len == 1, "expect 1 got %d\n", len );
2753
2754     r = IXMLDOMNodeList_get_item( node_list, 0, NULL );
2755     ok( r == E_INVALIDARG, "ret %08x\n", r );
2756
2757     r = IXMLDOMNodeList_nextNode( node_list, NULL );
2758     ok( r == E_INVALIDARG, "ret %08x\n", r );
2759
2760     r = IXMLDOMNodeList_get_item( node_list, 0, &node );
2761     ok( r == S_OK, "ret %08x\n", r );
2762     IXMLDOMNodeList_Release( node_list );
2763
2764     /* Invalid output parameter*/
2765     r = IXMLDOMNode_get_text( node, NULL );
2766     ok( r == E_INVALIDARG, "ret %08x\n", r );
2767
2768     r = IXMLDOMNode_get_text( node, &str );
2769     ok( r == S_OK, "ret %08x\n", r );
2770     ok( !memcmp(str, szfn1_txt, lstrlenW(szfn1_txt) ), "wrong string\n" );
2771     SysFreeString(str);
2772
2773     r = IXMLDOMNode_get_attributes( node, &node_map );
2774     ok( r == S_OK, "ret %08x\n", r );
2775
2776     str = SysAllocString( szvr );
2777     r = IXMLDOMNamedNodeMap_getNamedItem( node_map, str, &node2 );
2778     ok( r == S_OK, "ret %08x\n", r );
2779     SysFreeString(str);
2780
2781     r = IXMLDOMNode_get_text( node2, &str );
2782     ok( r == S_OK, "ret %08x\n", r );
2783     ok( !memcmp(str, szstr2, sizeof(szstr2)), "wrong string\n" );
2784     SysFreeString(str);
2785
2786     r = IXMLDOMNode_get_firstChild( node2, &node3 );
2787     ok( r == S_OK, "ret %08x\n", r );
2788
2789     r = IXMLDOMNode_get_text( node3, &str );
2790     ok( r == S_OK, "ret %08x\n", r );
2791     ok( !memcmp(str, szstr2, sizeof(szstr2)), "wrong string\n" );
2792     SysFreeString(str);
2793
2794
2795     IXMLDOMNode_Release( node3 );
2796     IXMLDOMNode_Release( node2 );
2797     IXMLDOMNamedNodeMap_Release( node_map );
2798     IXMLDOMNode_Release( node );
2799     IXMLDOMDocument_Release( doc );
2800 }
2801
2802 static void test_get_childNodes(void)
2803 {
2804     HRESULT r;
2805     BSTR str;
2806     VARIANT_BOOL b;
2807     IXMLDOMDocument *doc;
2808     IXMLDOMElement *element;
2809     IXMLDOMNode *node, *node2;
2810     IXMLDOMNodeList *node_list, *node_list2;
2811     LONG len;
2812
2813     doc = create_document(&IID_IXMLDOMDocument);
2814     if (!doc) return;
2815
2816     str = SysAllocString( szComplete4 );
2817     r = IXMLDOMDocument_loadXML( doc, str, &b );
2818     ok( r == S_OK, "loadXML failed\n");
2819     ok( b == VARIANT_TRUE, "failed to load XML string\n");
2820     SysFreeString( str );
2821
2822     r = IXMLDOMDocument_get_documentElement( doc, &element );
2823     ok( r == S_OK, "ret %08x\n", r);
2824
2825     r = IXMLDOMElement_get_childNodes( element, &node_list );
2826     ok( r == S_OK, "ret %08x\n", r);
2827
2828     r = IXMLDOMNodeList_get_length( node_list, &len );
2829     ok( r == S_OK, "ret %08x\n", r);
2830     ok( len == 4, "len %d\n", len);
2831
2832     r = IXMLDOMNodeList_get_item( node_list, 2, &node );
2833     ok( r == S_OK, "ret %08x\n", r);
2834
2835     r = IXMLDOMNode_get_childNodes( node, &node_list2 );
2836     ok( r == S_OK, "ret %08x\n", r);
2837
2838     r = IXMLDOMNodeList_get_length( node_list2, &len );
2839     ok( r == S_OK, "ret %08x\n", r);
2840     ok( len == 0, "len %d\n", len);
2841
2842     r = IXMLDOMNodeList_get_item( node_list2, 0, &node2);
2843     ok( r == S_FALSE, "ret %08x\n", r);
2844
2845     IXMLDOMNodeList_Release( node_list2 );
2846     IXMLDOMNode_Release( node );
2847     IXMLDOMNodeList_Release( node_list );
2848     IXMLDOMElement_Release( element );
2849     IXMLDOMDocument_Release( doc );
2850 }
2851
2852 static void test_get_firstChild(void)
2853 {
2854     static WCHAR xmlW[] = {'x','m','l',0};
2855     IXMLDOMDocument *doc;
2856     IXMLDOMNode *node;
2857     VARIANT_BOOL b;
2858     HRESULT r;
2859     BSTR str;
2860
2861     doc = create_document(&IID_IXMLDOMDocument);
2862     if (!doc) return;
2863
2864     str = SysAllocString( szComplete4 );
2865     r = IXMLDOMDocument_loadXML( doc, str, &b );
2866     ok( r == S_OK, "loadXML failed\n");
2867     ok( b == VARIANT_TRUE, "failed to load XML string\n");
2868     SysFreeString( str );
2869
2870     r = IXMLDOMDocument_get_firstChild( doc, &node );
2871     ok( r == S_OK, "ret %08x\n", r);
2872
2873     r = IXMLDOMNode_get_nodeName( node, &str );
2874     ok( r == S_OK, "ret %08x\n", r);
2875
2876     ok(memcmp(str, xmlW, sizeof(xmlW)) == 0, "expected \"xml\" node name\n");
2877
2878     SysFreeString(str);
2879     IXMLDOMNode_Release( node );
2880     IXMLDOMDocument_Release( doc );
2881 }
2882
2883 static void test_get_lastChild(void)
2884 {
2885     static WCHAR lcW[] = {'l','c',0};
2886     static WCHAR foW[] = {'f','o',0};
2887     IXMLDOMDocument *doc;
2888     IXMLDOMNode *node, *child;
2889     VARIANT_BOOL b;
2890     HRESULT r;
2891     BSTR str;
2892
2893     doc = create_document(&IID_IXMLDOMDocument);
2894     if (!doc) return;
2895
2896     str = SysAllocString( szComplete4 );
2897     r = IXMLDOMDocument_loadXML( doc, str, &b );
2898     ok( r == S_OK, "loadXML failed\n");
2899     ok( b == VARIANT_TRUE, "failed to load XML string\n");
2900     SysFreeString( str );
2901
2902     r = IXMLDOMDocument_get_lastChild( doc, &node );
2903     ok( r == S_OK, "ret %08x\n", r);
2904
2905     r = IXMLDOMNode_get_nodeName( node, &str );
2906     ok( r == S_OK, "ret %08x\n", r);
2907
2908     ok(memcmp(str, lcW, sizeof(lcW)) == 0, "expected \"lc\" node name\n");
2909     SysFreeString(str);
2910
2911     r = IXMLDOMNode_get_lastChild( node, &child );
2912     ok( r == S_OK, "ret %08x\n", r);
2913
2914     r = IXMLDOMNode_get_nodeName( child, &str );
2915     ok( r == S_OK, "ret %08x\n", r);
2916
2917     ok(memcmp(str, foW, sizeof(foW)) == 0, "expected \"fo\" node name\n");
2918     SysFreeString(str);
2919
2920     IXMLDOMNode_Release( child );
2921     IXMLDOMNode_Release( node );
2922     IXMLDOMDocument_Release( doc );
2923 }
2924
2925 static void test_removeChild(void)
2926 {
2927     HRESULT r;
2928     BSTR str;
2929     VARIANT_BOOL b;
2930     IXMLDOMDocument *doc;
2931     IXMLDOMElement *element, *lc_element;
2932     IXMLDOMNode *fo_node, *ba_node, *removed_node, *temp_node, *lc_node;
2933     IXMLDOMNodeList *root_list, *fo_list;
2934
2935     doc = create_document(&IID_IXMLDOMDocument);
2936     if (!doc) return;
2937
2938     str = SysAllocString( szComplete4 );
2939     r = IXMLDOMDocument_loadXML( doc, str, &b );
2940     ok( r == S_OK, "loadXML failed\n");
2941     ok( b == VARIANT_TRUE, "failed to load XML string\n");
2942     SysFreeString( str );
2943
2944     r = IXMLDOMDocument_get_documentElement( doc, &element );
2945     ok( r == S_OK, "ret %08x\n", r);
2946
2947     r = IXMLDOMElement_get_childNodes( element, &root_list );
2948     ok( r == S_OK, "ret %08x\n", r);
2949
2950     r = IXMLDOMNodeList_get_item( root_list, 3, &fo_node );
2951     ok( r == S_OK, "ret %08x\n", r);
2952
2953     r = IXMLDOMNode_get_childNodes( fo_node, &fo_list );
2954     ok( r == S_OK, "ret %08x\n", r);
2955
2956     r = IXMLDOMNodeList_get_item( fo_list, 0, &ba_node );
2957     ok( r == S_OK, "ret %08x\n", r);
2958
2959     /* invalid parameter: NULL ptr */
2960     removed_node = (void*)0xdeadbeef;
2961     r = IXMLDOMElement_removeChild( element, NULL, &removed_node );
2962     ok( r == E_INVALIDARG, "ret %08x\n", r );
2963     ok( removed_node == (void*)0xdeadbeef, "%p\n", removed_node );
2964
2965     /* ba_node is a descendant of element, but not a direct child. */
2966     removed_node = (void*)0xdeadbeef;
2967     r = IXMLDOMElement_removeChild( element, ba_node, &removed_node );
2968     ok( r == E_INVALIDARG, "ret %08x\n", r );
2969     ok( removed_node == NULL, "%p\n", removed_node );
2970
2971     r = IXMLDOMElement_removeChild( element, fo_node, &removed_node );
2972     ok( r == S_OK, "ret %08x\n", r);
2973     ok( fo_node == removed_node, "node %p node2 %p\n", fo_node, removed_node );
2974
2975     /* try removing already removed child */
2976     temp_node = (void*)0xdeadbeef;
2977     r = IXMLDOMElement_removeChild( element, fo_node, &temp_node );
2978     ok( r == E_INVALIDARG, "ret %08x\n", r);
2979     ok( temp_node == NULL, "%p\n", temp_node );
2980     IXMLDOMNode_Release( fo_node );
2981
2982     /* the removed node has no parent anymore */
2983     r = IXMLDOMNode_get_parentNode( removed_node, &temp_node );
2984     ok( r == S_FALSE, "ret %08x\n", r);
2985     ok( temp_node == NULL, "%p\n", temp_node );
2986
2987     IXMLDOMNode_Release( removed_node );
2988     IXMLDOMNode_Release( ba_node );
2989     IXMLDOMNodeList_Release( fo_list );
2990
2991     r = IXMLDOMNodeList_get_item( root_list, 0, &lc_node );
2992     ok( r == S_OK, "ret %08x\n", r);
2993
2994     r = IXMLDOMElement_QueryInterface( lc_node, &IID_IXMLDOMElement, (void**)&lc_element );
2995     ok( r == S_OK, "ret %08x\n", r);
2996
2997     /* MS quirk: passing wrong interface pointer works, too */
2998     r = IXMLDOMElement_removeChild( element, (IXMLDOMNode*)lc_element, NULL );
2999     ok( r == S_OK, "ret %08x\n", r);
3000     IXMLDOMElement_Release( lc_element );
3001
3002     r = IXMLDOMNode_get_parentNode( lc_node, &temp_node );
3003     ok( r == S_FALSE, "ret %08x\n", r);
3004     ok( temp_node == NULL, "%p\n", temp_node );
3005
3006     IXMLDOMNode_Release( lc_node );
3007     IXMLDOMNodeList_Release( root_list );
3008     IXMLDOMElement_Release( element );
3009     IXMLDOMDocument_Release( doc );
3010 }
3011
3012 static void test_replaceChild(void)
3013 {
3014     HRESULT r;
3015     BSTR str;
3016     VARIANT_BOOL b;
3017     IXMLDOMDocument *doc;
3018     IXMLDOMElement *element, *ba_element;
3019     IXMLDOMNode *fo_node, *ba_node, *lc_node, *removed_node, *temp_node;
3020     IXMLDOMNodeList *root_list, *fo_list;
3021     IUnknown * unk1, *unk2;
3022     LONG len;
3023
3024     doc = create_document(&IID_IXMLDOMDocument);
3025     if (!doc) return;
3026
3027     str = SysAllocString( szComplete4 );
3028     r = IXMLDOMDocument_loadXML( doc, str, &b );
3029     ok( r == S_OK, "loadXML failed\n");
3030     ok( b == VARIANT_TRUE, "failed to load XML string\n");
3031     SysFreeString( str );
3032
3033     r = IXMLDOMDocument_get_documentElement( doc, &element );
3034     ok( r == S_OK, "ret %08x\n", r);
3035
3036     r = IXMLDOMElement_get_childNodes( element, &root_list );
3037     ok( r == S_OK, "ret %08x\n", r);
3038
3039     r = IXMLDOMNodeList_get_item( root_list, 0, &lc_node );
3040     ok( r == S_OK, "ret %08x\n", r);
3041
3042     r = IXMLDOMNodeList_get_item( root_list, 3, &fo_node );
3043     ok( r == S_OK, "ret %08x\n", r);
3044
3045     r = IXMLDOMNode_get_childNodes( fo_node, &fo_list );
3046     ok( r == S_OK, "ret %08x\n", r);
3047
3048     r = IXMLDOMNodeList_get_item( fo_list, 0, &ba_node );
3049     ok( r == S_OK, "ret %08x\n", r);
3050
3051     IXMLDOMNodeList_Release( fo_list );
3052
3053     /* invalid parameter: NULL ptr for element to remove */
3054     removed_node = (void*)0xdeadbeef;
3055     r = IXMLDOMElement_replaceChild( element, ba_node, NULL, &removed_node );
3056     ok( r == E_INVALIDARG, "ret %08x\n", r );
3057     ok( removed_node == (void*)0xdeadbeef, "%p\n", removed_node );
3058
3059     /* invalid parameter: NULL for replacement element. (Sic!) */
3060     removed_node = (void*)0xdeadbeef;
3061     r = IXMLDOMElement_replaceChild( element, NULL, fo_node, &removed_node );
3062     ok( r == E_INVALIDARG, "ret %08x\n", r );
3063     ok( removed_node == (void*)0xdeadbeef, "%p\n", removed_node );
3064
3065     /* invalid parameter: OldNode is not a child */
3066     removed_node = (void*)0xdeadbeef;
3067     r = IXMLDOMElement_replaceChild( element, lc_node, ba_node, &removed_node );
3068     ok( r == E_INVALIDARG, "ret %08x\n", r );
3069     ok( removed_node == NULL, "%p\n", removed_node );
3070     IXMLDOMNode_Release( lc_node );
3071
3072     /* invalid parameter: would create loop */
3073     removed_node = (void*)0xdeadbeef;
3074     r = IXMLDOMNode_replaceChild( fo_node, fo_node, ba_node, &removed_node );
3075     ok( r == E_FAIL, "ret %08x\n", r );
3076     ok( removed_node == NULL, "%p\n", removed_node );
3077
3078     r = IXMLDOMElement_replaceChild( element, ba_node, fo_node, NULL );
3079     ok( r == S_OK, "ret %08x\n", r );
3080
3081     r = IXMLDOMNodeList_get_item( root_list, 3, &temp_node );
3082     ok( r == S_OK, "ret %08x\n", r );
3083
3084     /* ba_node and temp_node refer to the same node, yet they
3085        are different interface pointers */
3086     ok( ba_node != temp_node, "ba_node %p temp_node %p\n", ba_node, temp_node);
3087     r = IXMLDOMNode_QueryInterface( temp_node, &IID_IUnknown, (void**)&unk1);
3088     ok( r == S_OK, "ret %08x\n", r );
3089     r = IXMLDOMNode_QueryInterface( ba_node, &IID_IUnknown, (void**)&unk2);
3090     ok( r == S_OK, "ret %08x\n", r );
3091     todo_wine ok( unk1 == unk2, "unk1 %p unk2 %p\n", unk1, unk2);
3092
3093     IUnknown_Release( unk1 );
3094     IUnknown_Release( unk2 );
3095
3096     /* ba_node should have been removed from below fo_node */
3097     r = IXMLDOMNode_get_childNodes( fo_node, &fo_list );
3098     ok( r == S_OK, "ret %08x\n", r );
3099
3100     /* MS quirk: replaceChild also accepts elements instead of nodes */
3101     r = IXMLDOMNode_QueryInterface( ba_node, &IID_IXMLDOMElement, (void**)&ba_element);
3102     ok( r == S_OK, "ret %08x\n", r );
3103
3104     r = IXMLDOMElement_replaceChild( element, ba_node, (IXMLDOMNode*)ba_element, &removed_node );
3105     ok( r == S_OK, "ret %08x\n", r );
3106     IXMLDOMElement_Release( ba_element );
3107
3108     r = IXMLDOMNodeList_get_length( fo_list, &len);
3109     ok( r == S_OK, "ret %08x\n", r );
3110     ok( len == 0, "len %d\n", len);
3111
3112     IXMLDOMNodeList_Release( fo_list );
3113
3114     IXMLDOMNode_Release(ba_node);
3115     IXMLDOMNode_Release(fo_node);
3116     IXMLDOMNode_Release(temp_node);
3117     IXMLDOMNodeList_Release( root_list );
3118     IXMLDOMElement_Release( element );
3119     IXMLDOMDocument_Release( doc );
3120 }
3121
3122 static void test_removeNamedItem(void)
3123 {
3124     IXMLDOMDocument *doc;
3125     IXMLDOMElement *element;
3126     IXMLDOMNode *pr_node, *removed_node, *removed_node2;
3127     IXMLDOMNodeList *root_list;
3128     IXMLDOMNamedNodeMap * pr_attrs;
3129     VARIANT_BOOL b;
3130     BSTR str;
3131     LONG len;
3132     HRESULT r;
3133
3134     doc = create_document(&IID_IXMLDOMDocument);
3135     if (!doc) return;
3136
3137     str = SysAllocString( szComplete4 );
3138     r = IXMLDOMDocument_loadXML( doc, str, &b );
3139     ok( r == S_OK, "loadXML failed\n");
3140     ok( b == VARIANT_TRUE, "failed to load XML string\n");
3141     SysFreeString( str );
3142
3143     r = IXMLDOMDocument_get_documentElement( doc, &element );
3144     ok( r == S_OK, "ret %08x\n", r);
3145
3146     r = IXMLDOMElement_get_childNodes( element, &root_list );
3147     ok( r == S_OK, "ret %08x\n", r);
3148
3149     r = IXMLDOMNodeList_get_item( root_list, 1, &pr_node );
3150     ok( r == S_OK, "ret %08x\n", r);
3151
3152     r = IXMLDOMNode_get_attributes( pr_node, &pr_attrs );
3153     ok( r == S_OK, "ret %08x\n", r);
3154
3155     r = IXMLDOMNamedNodeMap_get_length( pr_attrs, &len );
3156     ok( r == S_OK, "ret %08x\n", r);
3157     ok( len == 3, "length %d\n", len);
3158
3159     removed_node = (void*)0xdeadbeef;
3160     r = IXMLDOMNamedNodeMap_removeNamedItem( pr_attrs, NULL, &removed_node);
3161     ok ( r == E_INVALIDARG, "ret %08x\n", r);
3162     ok ( removed_node == (void*)0xdeadbeef, "got %p\n", removed_node);
3163
3164     removed_node = (void*)0xdeadbeef;
3165     str = SysAllocString(szvr);
3166     r = IXMLDOMNamedNodeMap_removeNamedItem( pr_attrs, str, &removed_node);
3167     ok ( r == S_OK, "ret %08x\n", r);
3168
3169     removed_node2 = (void*)0xdeadbeef;
3170     r = IXMLDOMNamedNodeMap_removeNamedItem( pr_attrs, str, &removed_node2);
3171     ok ( r == S_FALSE, "ret %08x\n", r);
3172     ok ( removed_node2 == NULL, "got %p\n", removed_node2 );
3173
3174     r = IXMLDOMNamedNodeMap_get_length( pr_attrs, &len );
3175     ok( r == S_OK, "ret %08x\n", r);
3176     ok( len == 2, "length %d\n", len);
3177
3178     r = IXMLDOMNamedNodeMap_setNamedItem( pr_attrs, removed_node, NULL);
3179     ok ( r == S_OK, "ret %08x\n", r);
3180     IXMLDOMNode_Release(removed_node);
3181
3182     r = IXMLDOMNamedNodeMap_get_length( pr_attrs, &len );
3183     ok( r == S_OK, "ret %08x\n", r);
3184     ok( len == 3, "length %d\n", len);
3185
3186     r = IXMLDOMNamedNodeMap_removeNamedItem( pr_attrs, str, NULL);
3187     ok ( r == S_OK, "ret %08x\n", r);
3188
3189     r = IXMLDOMNamedNodeMap_get_length( pr_attrs, &len );
3190     ok( r == S_OK, "ret %08x\n", r);
3191     ok( len == 2, "length %d\n", len);
3192
3193     r = IXMLDOMNamedNodeMap_removeNamedItem( pr_attrs, str, NULL);
3194     ok ( r == S_FALSE, "ret %08x\n", r);
3195
3196     SysFreeString(str);
3197
3198     IXMLDOMNamedNodeMap_Release( pr_attrs );
3199     IXMLDOMNode_Release( pr_node );
3200     IXMLDOMNodeList_Release( root_list );
3201     IXMLDOMElement_Release( element );
3202     IXMLDOMDocument_Release( doc );
3203 }
3204
3205 #define test_IObjectSafety_set(p, r, r2, s, m, e, e2) _test_IObjectSafety_set(__LINE__,p, r, r2, s, m, e, e2)
3206 static void _test_IObjectSafety_set(unsigned line, IObjectSafety *safety, HRESULT result,
3207                                     HRESULT result2, DWORD set, DWORD mask, DWORD expected,
3208                                     DWORD expected2)
3209 {
3210     DWORD enabled, supported;
3211     HRESULT hr;
3212
3213     hr = IObjectSafety_SetInterfaceSafetyOptions(safety, NULL, set, mask);
3214     if (result == result2)
3215         ok_(__FILE__,line)(hr == result, "SetInterfaceSafetyOptions: expected %08x, returned %08x\n", result, hr );
3216     else
3217         ok_(__FILE__,line)(broken(hr == result) || hr == result2,
3218            "SetInterfaceSafetyOptions: expected %08x, got %08x\n", result2, hr );
3219
3220     supported = enabled = 0xCAFECAFE;
3221     hr = IObjectSafety_GetInterfaceSafetyOptions(safety, NULL, &supported, &enabled);
3222     ok(hr == S_OK, "ret %08x\n", hr );
3223     if (expected == expected2)
3224         ok_(__FILE__,line)(enabled == expected, "Expected %08x, got %08x\n", expected, enabled);
3225     else
3226         ok_(__FILE__,line)(broken(enabled == expected) || enabled == expected2,
3227            "Expected %08x, got %08x\n", expected2, enabled);
3228
3229     /* reset the safety options */
3230
3231     hr = IObjectSafety_SetInterfaceSafetyOptions(safety, NULL,
3232             INTERFACESAFE_FOR_UNTRUSTED_CALLER|INTERFACESAFE_FOR_UNTRUSTED_DATA|INTERFACE_USES_SECURITY_MANAGER,
3233             0);
3234     ok_(__FILE__,line)(hr == S_OK, "ret %08x\n", hr );
3235
3236     hr = IObjectSafety_GetInterfaceSafetyOptions(safety, NULL, &supported, &enabled);
3237     ok_(__FILE__,line)(hr == S_OK, "ret %08x\n", hr );
3238     ok_(__FILE__,line)(enabled == 0, "Expected 0, got %08x\n", enabled);
3239 }
3240
3241 #define test_IObjectSafety_common(s) _test_IObjectSafety_common(__LINE__,s)
3242 static void _test_IObjectSafety_common(unsigned line, IObjectSafety *safety)
3243 {
3244     DWORD enabled = 0, supported = 0;
3245     HRESULT hr;
3246
3247     /* get */
3248     hr = IObjectSafety_GetInterfaceSafetyOptions(safety, NULL, NULL, &enabled);
3249     ok_(__FILE__,line)(hr == E_POINTER, "ret %08x\n", hr );
3250     hr = IObjectSafety_GetInterfaceSafetyOptions(safety, NULL, &supported, NULL);
3251     ok_(__FILE__,line)(hr == E_POINTER, "ret %08x\n", hr );
3252
3253     hr = IObjectSafety_GetInterfaceSafetyOptions(safety, NULL, &supported, &enabled);
3254     ok_(__FILE__,line)(hr == S_OK, "ret %08x\n", hr );
3255     ok_(__FILE__,line)(broken(supported == (INTERFACESAFE_FOR_UNTRUSTED_CALLER | INTERFACESAFE_FOR_UNTRUSTED_DATA)) ||
3256        supported == (INTERFACESAFE_FOR_UNTRUSTED_CALLER | INTERFACESAFE_FOR_UNTRUSTED_DATA | INTERFACE_USES_SECURITY_MANAGER) /* msxml3 SP8+ */,
3257         "Expected (INTERFACESAFE_FOR_UNTRUSTED_CALLER | INTERFACESAFE_FOR_UNTRUSTED_DATA | INTERFACE_USES_SECURITY_MANAGER), "
3258              "got %08x\n", supported);
3259     ok_(__FILE__,line)(enabled == 0, "Expected 0, got %08x\n", enabled);
3260
3261     /* set -- individual flags */
3262
3263     test_IObjectSafety_set(safety, S_OK, S_OK,
3264         INTERFACESAFE_FOR_UNTRUSTED_CALLER, INTERFACESAFE_FOR_UNTRUSTED_CALLER,
3265         INTERFACESAFE_FOR_UNTRUSTED_CALLER, INTERFACESAFE_FOR_UNTRUSTED_CALLER);
3266
3267     test_IObjectSafety_set(safety, S_OK, S_OK,
3268         INTERFACESAFE_FOR_UNTRUSTED_DATA, INTERFACESAFE_FOR_UNTRUSTED_DATA,
3269         INTERFACESAFE_FOR_UNTRUSTED_DATA, INTERFACESAFE_FOR_UNTRUSTED_DATA);
3270
3271     test_IObjectSafety_set(safety, S_OK, S_OK,
3272         INTERFACE_USES_SECURITY_MANAGER, INTERFACE_USES_SECURITY_MANAGER,
3273         0, INTERFACE_USES_SECURITY_MANAGER /* msxml3 SP8+ */);
3274
3275     /* set INTERFACE_USES_DISPEX  */
3276
3277     test_IObjectSafety_set(safety, S_OK, E_FAIL /* msxml3 SP8+ */,
3278         INTERFACE_USES_DISPEX, INTERFACE_USES_DISPEX,
3279         0, 0);
3280
3281     test_IObjectSafety_set(safety, S_OK, E_FAIL /* msxml3 SP8+ */,
3282         INTERFACE_USES_DISPEX, 0,
3283         0, 0);
3284
3285     test_IObjectSafety_set(safety, S_OK, S_OK /* msxml3 SP8+ */,
3286         0, INTERFACE_USES_DISPEX,
3287         0, 0);
3288
3289     /* set option masking */
3290
3291     test_IObjectSafety_set(safety, S_OK, S_OK,
3292         INTERFACESAFE_FOR_UNTRUSTED_CALLER|INTERFACESAFE_FOR_UNTRUSTED_DATA,
3293         INTERFACESAFE_FOR_UNTRUSTED_CALLER,
3294         INTERFACESAFE_FOR_UNTRUSTED_CALLER,
3295         INTERFACESAFE_FOR_UNTRUSTED_CALLER);
3296
3297     test_IObjectSafety_set(safety, S_OK, S_OK,
3298         INTERFACESAFE_FOR_UNTRUSTED_CALLER|INTERFACESAFE_FOR_UNTRUSTED_DATA,
3299         INTERFACESAFE_FOR_UNTRUSTED_DATA,
3300         INTERFACESAFE_FOR_UNTRUSTED_DATA,
3301         INTERFACESAFE_FOR_UNTRUSTED_DATA);
3302
3303     test_IObjectSafety_set(safety, S_OK, S_OK,
3304         INTERFACESAFE_FOR_UNTRUSTED_CALLER|INTERFACESAFE_FOR_UNTRUSTED_DATA,
3305         INTERFACE_USES_SECURITY_MANAGER,
3306         0,
3307         0);
3308
3309     /* set -- inheriting previous settings */
3310
3311     hr = IObjectSafety_SetInterfaceSafetyOptions(safety, NULL,
3312                                                          INTERFACESAFE_FOR_UNTRUSTED_CALLER,
3313                                                          INTERFACESAFE_FOR_UNTRUSTED_CALLER);
3314     ok_(__FILE__,line)(hr == S_OK, "ret %08x\n", hr );
3315     hr = IObjectSafety_GetInterfaceSafetyOptions(safety, NULL, &supported, &enabled);
3316     ok_(__FILE__,line)(hr == S_OK, "ret %08x\n", hr );
3317     todo_wine
3318     ok_(__FILE__,line)(broken(enabled == INTERFACESAFE_FOR_UNTRUSTED_CALLER) ||
3319        enabled == (INTERFACESAFE_FOR_UNTRUSTED_CALLER | INTERFACE_USES_SECURITY_MANAGER) /* msxml3 SP8+ */,
3320          "Expected (INTERFACESAFE_FOR_UNTRUSTED_CALLER | INTERFACE_USES_SECURITY_MANAGER), "
3321          "got %08x\n", enabled);
3322
3323     hr = IObjectSafety_SetInterfaceSafetyOptions(safety, NULL,
3324                                                          INTERFACESAFE_FOR_UNTRUSTED_DATA,
3325                                                          INTERFACESAFE_FOR_UNTRUSTED_DATA);
3326     ok_(__FILE__,line)(hr == S_OK, "ret %08x\n", hr );
3327     hr = IObjectSafety_GetInterfaceSafetyOptions(safety, NULL, &supported, &enabled);
3328     ok_(__FILE__,line)(hr == S_OK, "ret %08x\n", hr );
3329     todo_wine
3330     ok_(__FILE__,line)(broken(enabled == INTERFACESAFE_FOR_UNTRUSTED_DATA) ||
3331        enabled == (INTERFACESAFE_FOR_UNTRUSTED_CALLER | INTERFACESAFE_FOR_UNTRUSTED_DATA) /* msxml3 SP8+ */,
3332         "Expected (INTERFACESAFE_FOR_UNTRUSTED_CALLER | INTERFACESAFE_FOR_UNTRUSTED_DATA), "
3333         "got %08x\n", enabled);
3334 }
3335
3336 static void test_XMLHTTP(void)
3337 {
3338     static const WCHAR wszBody[] = {'m','o','d','e','=','T','e','s','t',0};
3339     static const WCHAR wszPOST[] = {'P','O','S','T',0};
3340     static const WCHAR wszUrl[] = {'h','t','t','p',':','/','/',
3341         'c','r','o','s','s','o','v','e','r','.','c','o','d','e','w','e','a','v','e','r','s','.','c','o','m','/',
3342         'p','o','s','t','t','e','s','t','.','p','h','p',0};
3343     static const WCHAR xmltestW[] = {'h','t','t','p',':','/','/',
3344         'c','r','o','s','s','o','v','e','r','.','c','o','d','e','w','e','a','v','e','r','s','.','c','o','m','/',
3345         'x','m','l','t','e','s','t','.','x','m','l',0};
3346     static const WCHAR wszExpectedResponse[] = {'F','A','I','L','E','D',0};
3347     static const CHAR xmltestbodyA[] = "<?xml version=\"1.0\" encoding=\"utf-8\"?>\n<a>TEST</a>\n";
3348
3349     IXMLHttpRequest *pXMLHttpRequest;
3350     IObjectSafety *safety;
3351     IObjectWithSite *pSite;
3352     BSTR bstrResponse, method, url;
3353     VARIANT dummy;
3354     VARIANT async;
3355     VARIANT varbody;
3356     LONG state, status, ref, bound;
3357     void *ptr;
3358     IDispatch *event;
3359     HRESULT hr = CoCreateInstance(&CLSID_XMLHTTPRequest, NULL,
3360                                   CLSCTX_INPROC_SERVER, &IID_IXMLHttpRequest,
3361                                   (void **)&pXMLHttpRequest);
3362     if (FAILED(hr))
3363     {
3364         win_skip("IXMLHTTPRequest is not available (0x%08x)\n", hr);
3365         return;
3366     }
3367
3368     hr = IXMLHttpRequest_QueryInterface(pXMLHttpRequest, &IID_IObjectWithSite, (void**)&pSite);
3369     ok(hr == S_OK, "got 0x%08x\n", hr);
3370     if(hr == S_OK) IObjectWithSite_Release(pSite);
3371
3372     VariantInit(&dummy);
3373     V_VT(&dummy) = VT_ERROR;
3374     V_ERROR(&dummy) = DISP_E_MEMBERNOTFOUND;
3375     VariantInit(&async);
3376     V_VT(&async) = VT_BOOL;
3377     V_BOOL(&async) = VARIANT_FALSE;
3378     V_VT(&varbody) = VT_BSTR;
3379     V_BSTR(&varbody) = SysAllocString(wszBody);
3380
3381     method = SysAllocString(wszPOST);
3382     url = SysAllocString(wszUrl);
3383
3384 if (0)
3385 {
3386     /* crashes on win98 */
3387     hr = IXMLHttpRequest_put_onreadystatechange(pXMLHttpRequest, NULL);
3388     ok(hr == S_OK, "got 0x%08x\n", hr);
3389 }
3390
3391     hr = IXMLHttpRequest_abort(pXMLHttpRequest);
3392     ok(hr == S_OK, "got 0x%08x\n", hr);
3393
3394     /* send before open */
3395     hr = IXMLHttpRequest_send(pXMLHttpRequest, dummy);
3396     ok(hr == E_FAIL || broken(hr == E_UNEXPECTED) /* win9x, win2k */, "got 0x%08x\n", hr);
3397
3398     /* initial status code */
3399     hr = IXMLHttpRequest_get_status(pXMLHttpRequest, NULL);
3400     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
3401
3402     status = 0xdeadbeef;
3403     hr = IXMLHttpRequest_get_status(pXMLHttpRequest, &status);
3404     ok(hr == E_FAIL || broken(hr == E_UNEXPECTED) /* win9x, win2k */, "got 0x%08x\n", hr);
3405     ok(status == 0xdeadbeef, "got %d\n", status);
3406
3407     /* invalid parameters */
3408     hr = IXMLHttpRequest_open(pXMLHttpRequest, NULL, NULL, async, dummy, dummy);
3409     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
3410
3411     hr = IXMLHttpRequest_open(pXMLHttpRequest, method, NULL, async, dummy, dummy);
3412     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
3413
3414     hr = IXMLHttpRequest_open(pXMLHttpRequest, NULL, url, async, dummy, dummy);
3415     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
3416
3417     hr = IXMLHttpRequest_setRequestHeader(pXMLHttpRequest, NULL, NULL);
3418     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
3419
3420     hr = IXMLHttpRequest_setRequestHeader(pXMLHttpRequest, _bstr_("header1"), NULL);
3421     ok(hr == E_FAIL || broken(hr == E_UNEXPECTED) /* win9x, win2k */, "got 0x%08x\n", hr);
3422
3423     hr = IXMLHttpRequest_setRequestHeader(pXMLHttpRequest, NULL, _bstr_("value1"));
3424     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
3425
3426     hr = IXMLHttpRequest_setRequestHeader(pXMLHttpRequest, _bstr_("header1"), _bstr_("value1"));
3427     ok(hr == E_FAIL || broken(hr == E_UNEXPECTED) /* win9x, win2k */, "got 0x%08x\n", hr);
3428
3429     hr = IXMLHttpRequest_get_readyState(pXMLHttpRequest, NULL);
3430     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
3431
3432     state = -1;
3433     hr = IXMLHttpRequest_get_readyState(pXMLHttpRequest, &state);
3434     ok(hr == S_OK, "got 0x%08x\n", hr);
3435     ok(state == READYSTATE_UNINITIALIZED, "got %d, expected READYSTATE_UNINITIALIZED\n", state);
3436
3437     event = create_dispevent();
3438     ref = IDispatch_AddRef(event);
3439     ok(ref == 2, "got %d\n", ref);
3440     IDispatch_Release(event);
3441
3442     hr = IXMLHttpRequest_put_onreadystatechange(pXMLHttpRequest, event);
3443     ok(hr == S_OK, "got 0x%08x\n", hr);
3444
3445     ref = IDispatch_AddRef(event);
3446     ok(ref == 3, "got %d\n", ref);
3447     IDispatch_Release(event);
3448
3449     g_unexpectedcall = g_expectedcall = 0;
3450
3451     hr = IXMLHttpRequest_open(pXMLHttpRequest, method, url, async, dummy, dummy);
3452     ok(hr == S_OK, "got 0x%08x\n", hr);
3453
3454     ok(g_unexpectedcall == 0, "unexpected disp event call\n");
3455     ok(g_expectedcall == 1 || broken(g_expectedcall == 0) /* win2k */, "no expected disp event call\n");
3456
3457     /* status code after ::open() */
3458     status = 0xdeadbeef;
3459     hr = IXMLHttpRequest_get_status(pXMLHttpRequest, &status);
3460     ok(hr == E_FAIL || broken(hr == E_UNEXPECTED) /* win9x, win2k */, "got 0x%08x\n", hr);
3461     ok(status == 0xdeadbeef, "got %d\n", status);
3462
3463     state = -1;
3464     hr = IXMLHttpRequest_get_readyState(pXMLHttpRequest, &state);
3465     ok(hr == S_OK, "got 0x%08x\n", hr);
3466     ok(state == READYSTATE_LOADING, "got %d, expected READYSTATE_LOADING\n", state);
3467
3468     hr = IXMLHttpRequest_abort(pXMLHttpRequest);
3469     ok(hr == S_OK, "got 0x%08x\n", hr);
3470
3471     state = -1;
3472     hr = IXMLHttpRequest_get_readyState(pXMLHttpRequest, &state);
3473     ok(hr == S_OK, "got 0x%08x\n", hr);
3474     ok(state == READYSTATE_UNINITIALIZED || broken(state == READYSTATE_LOADING) /* win98, win2k */,
3475         "got %d, expected READYSTATE_UNINITIALIZED\n", state);
3476
3477     hr = IXMLHttpRequest_open(pXMLHttpRequest, method, url, async, dummy, dummy);
3478     ok(hr == S_OK, "got 0x%08x\n", hr);
3479
3480     hr = IXMLHttpRequest_setRequestHeader(pXMLHttpRequest, _bstr_("header1"), _bstr_("value1"));
3481     ok(hr == S_OK, "got 0x%08x\n", hr);
3482
3483     hr = IXMLHttpRequest_setRequestHeader(pXMLHttpRequest, NULL, _bstr_("value1"));
3484     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
3485
3486     hr = IXMLHttpRequest_setRequestHeader(pXMLHttpRequest, _bstr_(""), _bstr_("value1"));
3487     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
3488
3489     SysFreeString(method);
3490     SysFreeString(url);
3491
3492     hr = IXMLHttpRequest_send(pXMLHttpRequest, varbody);
3493     if (hr == INET_E_RESOURCE_NOT_FOUND)
3494     {
3495         skip("No connection could be made with crossover.codeweavers.com\n");
3496         IXMLHttpRequest_Release(pXMLHttpRequest);
3497         return;
3498     }
3499     ok(hr == S_OK, "got 0x%08x\n", hr);
3500
3501     /* status code after ::send() */
3502     status = 0xdeadbeef;
3503     hr = IXMLHttpRequest_get_status(pXMLHttpRequest, &status);
3504     ok(hr == S_OK, "got 0x%08x\n", hr);
3505     ok(status == 200, "got %d\n", status);
3506
3507     /* another ::send() after completed request */
3508     hr = IXMLHttpRequest_send(pXMLHttpRequest, varbody);
3509     ok(hr == E_FAIL || broken(hr == E_UNEXPECTED) /* win9x, win2k */, "got 0x%08x\n", hr);
3510
3511     VariantClear(&varbody);
3512
3513     hr = IXMLHttpRequest_get_responseText(pXMLHttpRequest, &bstrResponse);
3514     ok(hr == S_OK, "got 0x%08x\n", hr);
3515     /* the server currently returns "FAILED" because the Content-Type header is
3516      * not what the server expects */
3517     if(hr == S_OK)
3518     {
3519         ok(!memcmp(bstrResponse, wszExpectedResponse, sizeof(wszExpectedResponse)),
3520             "expected %s, got %s\n", wine_dbgstr_w(wszExpectedResponse), wine_dbgstr_w(bstrResponse));
3521         SysFreeString(bstrResponse);
3522     }
3523
3524     /* GET request */
3525     url = SysAllocString(xmltestW);
3526
3527     hr = IXMLHttpRequest_open(pXMLHttpRequest, _bstr_("GET"), url, async, dummy, dummy);
3528     ok(hr == S_OK, "got 0x%08x\n", hr);
3529
3530     V_VT(&varbody) = VT_EMPTY;
3531
3532     hr = IXMLHttpRequest_send(pXMLHttpRequest, varbody);
3533     if (hr == INET_E_RESOURCE_NOT_FOUND)
3534     {
3535         skip("No connection could be made with crossover.codeweavers.com\n");
3536         IXMLHttpRequest_Release(pXMLHttpRequest);
3537         return;
3538     }
3539     ok(hr == S_OK, "got 0x%08x\n", hr);
3540
3541     hr = IXMLHttpRequest_get_responseText(pXMLHttpRequest, NULL);
3542     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
3543
3544     hr = IXMLHttpRequest_get_responseText(pXMLHttpRequest, &bstrResponse);
3545     ok(hr == S_OK, "got 0x%08x\n", hr);
3546     if(hr == S_OK)
3547     {
3548         ok(!memcmp(bstrResponse, _bstr_(xmltestbodyA), sizeof(xmltestbodyA)*sizeof(WCHAR)),
3549             "expected %s, got %s\n", xmltestbodyA, wine_dbgstr_w(bstrResponse));
3550         SysFreeString(bstrResponse);
3551     }
3552
3553     hr = IXMLHttpRequest_get_responseBody(pXMLHttpRequest, NULL);
3554     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
3555
3556     V_VT(&varbody) = VT_EMPTY;
3557     hr = IXMLHttpRequest_get_responseBody(pXMLHttpRequest, &varbody);
3558     ok(hr == S_OK, "got 0x%08x\n", hr);
3559     ok(V_VT(&varbody) == (VT_ARRAY|VT_UI1), "got type %d, expected %d\n", V_VT(&varbody), VT_ARRAY|VT_UI1);
3560     ok(SafeArrayGetDim(V_ARRAY(&varbody)) == 1, "got %d, expected one dimension\n", SafeArrayGetDim(V_ARRAY(&varbody)));
3561
3562     bound = -1;
3563     hr = SafeArrayGetLBound(V_ARRAY(&varbody), 1, &bound);
3564     ok(hr == S_OK, "got 0x%08x\n", hr);
3565     ok(bound == 0, "got %d, expected zero bound\n", bound);
3566
3567     hr = SafeArrayAccessData(V_ARRAY(&varbody), &ptr);
3568     ok(hr == S_OK, "got 0x%08x\n", hr);
3569     ok(memcmp(ptr, xmltestbodyA, sizeof(xmltestbodyA)-1) == 0, "got wrond body data\n");
3570     SafeArrayUnaccessData(V_ARRAY(&varbody));
3571
3572     VariantClear(&varbody);
3573
3574     SysFreeString(url);
3575
3576     hr = IXMLHttpRequest_QueryInterface(pXMLHttpRequest, &IID_IObjectSafety, (void**)&safety);
3577     ok(hr == S_OK, "ret %08x\n", hr );
3578     if(hr == S_OK)
3579     {
3580         test_IObjectSafety_common(safety);
3581
3582         IObjectSafety_Release(safety);
3583     }
3584
3585
3586     IDispatch_Release(event);
3587     IXMLHttpRequest_Release(pXMLHttpRequest);
3588     free_bstrs();
3589 }
3590
3591 static void test_IXMLDOMDocument2(void)
3592 {
3593     static const WCHAR emptyW[] = {0};
3594     IXMLDOMDocument2 *doc2, *dtddoc2;
3595     IXMLDOMDocument *doc;
3596     IXMLDOMParseError* err;
3597     IDispatchEx *dispex;
3598     VARIANT_BOOL b;
3599     VARIANT var;
3600     HRESULT r;
3601     LONG ref, res;
3602     BSTR str;
3603
3604     doc = create_document(&IID_IXMLDOMDocument);
3605     if (!doc) return;
3606
3607     dtddoc2 = create_document(&IID_IXMLDOMDocument2);
3608     if (!dtddoc2)
3609     {
3610         IXMLDOMDocument_Release(doc);
3611         return;
3612     }
3613
3614     r = IXMLDOMDocument_QueryInterface( doc, &IID_IXMLDOMDocument2, (void**)&doc2 );
3615     ok( r == S_OK, "ret %08x\n", r );
3616     ok( doc == (IXMLDOMDocument*)doc2, "interfaces differ\n");
3617
3618     ole_expect(IXMLDOMDocument2_get_readyState(doc2, NULL), E_INVALIDARG);
3619     ole_check(IXMLDOMDocument2_get_readyState(doc2, &res));
3620     ok(res == READYSTATE_COMPLETE, "expected READYSTATE_COMPLETE (4), got %i\n", res);
3621
3622     err = NULL;
3623     ole_expect(IXMLDOMDocument2_validate(doc2, NULL), S_FALSE);
3624     ole_expect(IXMLDOMDocument2_validate(doc2, &err), S_FALSE);
3625     ok(err != NULL, "expected a pointer\n");
3626     if (err)
3627     {
3628         res = 0;
3629         ole_check(IXMLDOMParseError_get_errorCode(err, &res));
3630         /* XML_E_NOTWF */
3631         ok(res == E_XML_NOTWF, "got %08x\n", res);
3632         IXMLDOMParseError_Release(err);
3633     }
3634
3635     str = SysAllocString( szComplete4 );
3636     r = IXMLDOMDocument_loadXML( doc2, str, &b );
3637     ok( r == S_OK, "loadXML failed\n");
3638     ok( b == VARIANT_TRUE, "failed to load XML string\n");
3639     SysFreeString( str );
3640
3641     ole_check(IXMLDOMDocument2_get_readyState(doc, &res));
3642     ok(res == READYSTATE_COMPLETE, "expected READYSTATE_COMPLETE (4), got %i\n", res);
3643
3644     err = NULL;
3645     ole_expect(IXMLDOMDocument2_validate(doc2, &err), S_FALSE);
3646     ok(err != NULL, "expected a pointer\n");
3647     if (err)
3648     {
3649         res = 0;
3650         ole_check(IXMLDOMParseError_get_errorCode(err, &res));
3651         /* XML_E_NODTD */
3652         ok(res == E_XML_NODTD, "got %08x\n", res);
3653         IXMLDOMParseError_Release(err);
3654     }
3655
3656     r = IXMLDOMDocument_QueryInterface( doc, &IID_IDispatchEx, (void**)&dispex );
3657     ok( r == S_OK, "ret %08x\n", r );
3658     if(r == S_OK)
3659     {
3660         IDispatchEx_Release(dispex);
3661     }
3662
3663     /* we will check if the variant got cleared */
3664     ref = IXMLDOMDocument2_AddRef(doc2);
3665     expect_eq(ref, 3, int, "%d");  /* doc, doc2, AddRef*/
3666     V_VT(&var) = VT_UNKNOWN;
3667     V_UNKNOWN(&var) = (IUnknown *)doc2;
3668
3669     /* invalid calls */
3670     ole_expect(IXMLDOMDocument2_getProperty(doc2, _bstr_("askldhfaklsdf"), &var), E_FAIL);
3671     expect_eq(V_VT(&var), VT_UNKNOWN, int, "%x");
3672     ole_expect(IXMLDOMDocument2_getProperty(doc2, _bstr_("SelectionLanguage"), NULL), E_INVALIDARG);
3673
3674     /* valid call */
3675     ole_check(IXMLDOMDocument2_getProperty(doc2, _bstr_("SelectionLanguage"), &var));
3676     expect_eq(V_VT(&var), VT_BSTR, int, "%x");
3677     expect_bstr_eq_and_free(V_BSTR(&var), "XSLPattern");
3678     V_VT(&var) = VT_R4;
3679
3680     /* the variant didn't get cleared*/
3681     expect_eq(IXMLDOMDocument2_Release(doc2), 2, int, "%d");
3682
3683     /* setProperty tests */
3684     ole_expect(IXMLDOMDocument2_setProperty(doc2, _bstr_("askldhfaklsdf"), var), E_FAIL);
3685     ole_expect(IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionLanguage"), var), E_FAIL);
3686     ole_expect(IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionLanguage"), _variantbstr_("alskjdh faklsjd hfk")), E_FAIL);
3687     ole_check(IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionLanguage"), _variantbstr_("XSLPattern")));
3688     ole_check(IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionLanguage"), _variantbstr_("XPath")));
3689     ole_check(IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionLanguage"), _variantbstr_("XSLPattern")));
3690
3691     V_VT(&var) = VT_BSTR;
3692     V_BSTR(&var) = SysAllocString(emptyW);
3693     r = IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionNamespaces"), var);
3694     ok(r == S_OK, "got 0x%08x\n", r);
3695     VariantClear(&var);
3696
3697     V_VT(&var) = VT_I2;
3698     V_I2(&var) = 0;
3699     r = IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionNamespaces"), var);
3700     ok(r == E_FAIL, "got 0x%08x\n", r);
3701
3702     /* contrary to what MSDN claims you can switch back from XPath to XSLPattern */
3703     ole_check(IXMLDOMDocument2_getProperty(doc2, _bstr_("SelectionLanguage"), &var));
3704     expect_eq(V_VT(&var), VT_BSTR, int, "%x");
3705     expect_bstr_eq_and_free(V_BSTR(&var), "XSLPattern");
3706
3707     IXMLDOMDocument2_Release( doc2 );
3708     IXMLDOMDocument_Release( doc );
3709
3710     /* DTD validation */
3711     ole_check(IXMLDOMDocument2_put_validateOnParse(dtddoc2, VARIANT_FALSE));
3712     ole_check(IXMLDOMDocument2_loadXML(dtddoc2, _bstr_(szEmailXML), &b));
3713     ok( b == VARIANT_TRUE, "failed to load XML string\n");
3714     err = NULL;
3715     ole_check(IXMLDOMDocument2_validate(dtddoc2, &err));
3716     ok(err != NULL, "expected pointer\n");
3717     if (err)
3718     {
3719         res = 0;
3720         ole_expect(IXMLDOMParseError_get_errorCode(err, &res), S_FALSE);
3721         ok(res == 0, "got %08x\n", res);
3722         IXMLDOMParseError_Release(err);
3723     }
3724
3725     ole_check(IXMLDOMDocument2_loadXML(dtddoc2, _bstr_(szEmailXML_0D), &b));
3726     ok( b == VARIANT_TRUE, "failed to load XML string\n");
3727     err = NULL;
3728     ole_expect(IXMLDOMDocument2_validate(dtddoc2, &err), S_FALSE);
3729     ok(err != NULL, "expected pointer\n");
3730     if (err)
3731     {
3732         res = 0;
3733         ole_check(IXMLDOMParseError_get_errorCode(err, &res));
3734         /* XML_ELEMENT_UNDECLARED */
3735         todo_wine ok(res == 0xC00CE00D, "got %08x\n", res);
3736         IXMLDOMParseError_Release(err);
3737     }
3738
3739     ole_check(IXMLDOMDocument2_loadXML(dtddoc2, _bstr_(szEmailXML_0E), &b));
3740     ok( b == VARIANT_TRUE, "failed to load XML string\n");
3741     err = NULL;
3742     ole_expect(IXMLDOMDocument2_validate(dtddoc2, &err), S_FALSE);
3743     ok(err != NULL, "expected pointer\n");
3744     if (err)
3745     {
3746         res = 0;
3747         ole_check(IXMLDOMParseError_get_errorCode(err, &res));
3748         /* XML_ELEMENT_ID_NOT_FOUND */
3749         todo_wine ok(res == 0xC00CE00E, "got %08x\n", res);
3750         IXMLDOMParseError_Release(err);
3751     }
3752
3753     ole_check(IXMLDOMDocument2_loadXML(dtddoc2, _bstr_(szEmailXML_11), &b));
3754     ok( b == VARIANT_TRUE, "failed to load XML string\n");
3755     err = NULL;
3756     ole_expect(IXMLDOMDocument2_validate(dtddoc2, &err), S_FALSE);
3757     ok(err != NULL, "expected pointer\n");
3758     if (err)
3759     {
3760         res = 0;
3761         ole_check(IXMLDOMParseError_get_errorCode(err, &res));
3762         /* XML_EMPTY_NOT_ALLOWED */
3763         todo_wine ok(res == 0xC00CE011, "got %08x\n", res);
3764         IXMLDOMParseError_Release(err);
3765     }
3766
3767     ole_check(IXMLDOMDocument2_loadXML(dtddoc2, _bstr_(szEmailXML_13), &b));
3768     ok( b == VARIANT_TRUE, "failed to load XML string\n");
3769     err = NULL;
3770     ole_expect(IXMLDOMDocument2_validate(dtddoc2, &err), S_FALSE);
3771     ok(err != NULL, "expected pointer\n");
3772     if (err)
3773     {
3774         res = 0;
3775         ole_check(IXMLDOMParseError_get_errorCode(err, &res));
3776         /* XML_ROOT_NAME_MISMATCH */
3777         todo_wine ok(res == 0xC00CE013, "got %08x\n", res);
3778         IXMLDOMParseError_Release(err);
3779     }
3780
3781     ole_check(IXMLDOMDocument2_loadXML(dtddoc2, _bstr_(szEmailXML_14), &b));
3782     ok( b == VARIANT_TRUE, "failed to load XML string\n");
3783     err = NULL;
3784     ole_expect(IXMLDOMDocument2_validate(dtddoc2, &err), S_FALSE);
3785     ok(err != NULL, "expected pointer\n");
3786     if (err)
3787     {
3788         res = 0;
3789         ole_check(IXMLDOMParseError_get_errorCode(err, &res));
3790         /* XML_INVALID_CONTENT */
3791         todo_wine ok(res == 0xC00CE014, "got %08x\n", res);
3792         IXMLDOMParseError_Release(err);
3793     }
3794
3795     ole_check(IXMLDOMDocument2_loadXML(dtddoc2, _bstr_(szEmailXML_15), &b));
3796     ok( b == VARIANT_TRUE, "failed to load XML string\n");
3797     err = NULL;
3798     ole_expect(IXMLDOMDocument2_validate(dtddoc2, &err), S_FALSE);
3799     ok(err != NULL, "expected pointer\n");
3800     if (err)
3801     {
3802         res = 0;
3803         ole_check(IXMLDOMParseError_get_errorCode(err, &res));
3804         /* XML_ATTRIBUTE_NOT_DEFINED */
3805         todo_wine ok(res == 0xC00CE015, "got %08x\n", res);
3806         IXMLDOMParseError_Release(err);
3807     }
3808
3809     ole_check(IXMLDOMDocument2_loadXML(dtddoc2, _bstr_(szEmailXML_16), &b));
3810     ok( b == VARIANT_TRUE, "failed to load XML string\n");
3811     err = NULL;
3812     ole_expect(IXMLDOMDocument2_validate(dtddoc2, &err), S_FALSE);
3813     ok(err != NULL, "expected pointer\n");
3814     if (err)
3815     {
3816         res = 0;
3817         ole_check(IXMLDOMParseError_get_errorCode(err, &res));
3818         /* XML_ATTRIBUTE_FIXED */
3819         todo_wine ok(res == 0xC00CE016, "got %08x\n", res);
3820         IXMLDOMParseError_Release(err);
3821     }
3822
3823     ole_check(IXMLDOMDocument2_loadXML(dtddoc2, _bstr_(szEmailXML_17), &b));
3824     ok( b == VARIANT_TRUE, "failed to load XML string\n");
3825     err = NULL;
3826     ole_expect(IXMLDOMDocument2_validate(dtddoc2, &err), S_FALSE);
3827     ok(err != NULL, "expected pointer\n");
3828     if (err)
3829     {
3830         res = 0;
3831         ole_check(IXMLDOMParseError_get_errorCode(err, &res));
3832         /* XML_ATTRIBUTE_VALUE */
3833         todo_wine ok(res == 0xC00CE017, "got %08x\n", res);
3834         IXMLDOMParseError_Release(err);
3835     }
3836
3837     ole_check(IXMLDOMDocument2_loadXML(dtddoc2, _bstr_(szEmailXML_18), &b));
3838     ok( b == VARIANT_TRUE, "failed to load XML string\n");
3839     err = NULL;
3840     ole_expect(IXMLDOMDocument2_validate(dtddoc2, &err), S_FALSE);
3841     ok(err != NULL, "expected pointer\n");
3842     if (err)
3843     {
3844         res = 0;
3845         ole_check(IXMLDOMParseError_get_errorCode(err, &res));
3846         /* XML_ILLEGAL_TEXT */
3847         todo_wine ok(res == 0xC00CE018, "got %08x\n", res);
3848         IXMLDOMParseError_Release(err);
3849     }
3850
3851     ole_check(IXMLDOMDocument2_loadXML(dtddoc2, _bstr_(szEmailXML_20), &b));
3852     ok( b == VARIANT_TRUE, "failed to load XML string\n");
3853     err = NULL;
3854     ole_expect(IXMLDOMDocument2_validate(dtddoc2, &err), S_FALSE);
3855     ok(err != NULL, "expected pointer\n");
3856     if (err)
3857     {
3858         res = 0;
3859         ole_check(IXMLDOMParseError_get_errorCode(err, &res));
3860         /* XML_REQUIRED_ATTRIBUTE_MISSING */
3861         todo_wine ok(res == 0xC00CE020, "got %08x\n", res);
3862         IXMLDOMParseError_Release(err);
3863     }
3864
3865     IXMLDOMDocument2_Release( dtddoc2 );
3866     free_bstrs();
3867 }
3868
3869 #define helper_ole_check(expr) { \
3870     HRESULT r = expr; \
3871     ok_(__FILE__, line)(r == S_OK, "=> %i: " #expr " returned %08x\n", __LINE__, r); \
3872 }
3873
3874 #define helper_expect_list_and_release(list, expstr) { \
3875     char *str = list_to_string(list); \
3876     ok_(__FILE__, line)(strcmp(str, expstr)==0, "=> %i: Invalid node list: %s, expected %s\n", __LINE__, str, expstr); \
3877     if (list) IXMLDOMNodeList_Release(list); \
3878 }
3879
3880 #define helper_expect_bstr_and_release(bstr, str) { \
3881     ok_(__FILE__, line)(lstrcmpW(bstr, _bstr_(str)) == 0, \
3882        "=> %i: got %s\n", __LINE__, wine_dbgstr_w(bstr)); \
3883     SysFreeString(bstr); \
3884 }
3885
3886 #define check_ws_ignored(doc, str) _check_ws_ignored(__LINE__, doc, str)
3887 static inline void _check_ws_ignored(int line, IXMLDOMDocument2* doc, char const* str)
3888 {
3889     IXMLDOMNode *node1, *node2;
3890     IXMLDOMNodeList *list;
3891     BSTR bstr;
3892
3893     helper_ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//*[local-name()='html']"), &list));
3894     helper_ole_check(IXMLDOMNodeList_get_item(list, 0, &node1));
3895     helper_ole_check(IXMLDOMNodeList_get_item(list, 1, &node2));
3896     helper_ole_check(IXMLDOMNodeList_reset(list));
3897     helper_expect_list_and_release(list, "E1.E4.E1.E2.D1 E2.E4.E1.E2.D1");
3898
3899     helper_ole_check(IXMLDOMNode_get_childNodes(node1, &list));
3900     helper_expect_list_and_release(list, "T1.E1.E4.E1.E2.D1 E2.E1.E4.E1.E2.D1 E3.E1.E4.E1.E2.D1 T4.E1.E4.E1.E2.D1 E5.E1.E4.E1.E2.D1");
3901     helper_ole_check(IXMLDOMNode_get_text(node1, &bstr));
3902     if (str)
3903     {
3904         helper_expect_bstr_and_release(bstr, str);
3905     }
3906     else
3907     {
3908         helper_expect_bstr_and_release(bstr, "This is a description.");
3909     }
3910     IXMLDOMNode_Release(node1);
3911
3912     helper_ole_check(IXMLDOMNode_get_childNodes(node2, &list));
3913     helper_expect_list_and_release(list, "T1.E2.E4.E1.E2.D1 E2.E2.E4.E1.E2.D1 T3.E2.E4.E1.E2.D1 E4.E2.E4.E1.E2.D1 T5.E2.E4.E1.E2.D1 E6.E2.E4.E1.E2.D1 T7.E2.E4.E1.E2.D1");
3914     helper_ole_check(IXMLDOMNode_get_text(node2, &bstr));
3915     helper_expect_bstr_and_release(bstr, "\n                This is a description with preserved whitespace. \n            ");
3916     IXMLDOMNode_Release(node2);
3917 }
3918
3919 #define check_ws_preserved(doc, str) _check_ws_preserved(__LINE__, doc, str)
3920 static inline void _check_ws_preserved(int line, IXMLDOMDocument2* doc, char const* str)
3921 {
3922     IXMLDOMNode *node1, *node2;
3923     IXMLDOMNodeList *list;
3924     BSTR bstr;
3925
3926     helper_ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//*[local-name()='html']"), &list));
3927     helper_ole_check(IXMLDOMNodeList_get_item(list, 0, &node1));
3928     helper_ole_check(IXMLDOMNodeList_get_item(list, 1, &node2));
3929     helper_ole_check(IXMLDOMNodeList_reset(list));
3930     helper_expect_list_and_release(list, "E2.E8.E2.E2.D1 E4.E8.E2.E2.D1");
3931
3932     helper_ole_check(IXMLDOMNode_get_childNodes(node1, &list));
3933     helper_expect_list_and_release(list, "T1.E2.E8.E2.E2.D1 E2.E2.E8.E2.E2.D1 T3.E2.E8.E2.E2.D1 E4.E2.E8.E2.E2.D1 T5.E2.E8.E2.E2.D1 E6.E2.E8.E2.E2.D1 T7.E2.E8.E2.E2.D1");
3934     helper_ole_check(IXMLDOMNode_get_text(node1, &bstr));
3935     if (str)
3936     {
3937         helper_expect_bstr_and_release(bstr, str);
3938     }
3939     else
3940     {
3941         helper_expect_bstr_and_release(bstr, "\n                This is a description. \n            ");
3942     }
3943     IXMLDOMNode_Release(node1);
3944
3945     helper_ole_check(IXMLDOMNode_get_childNodes(node2, &list));
3946     helper_expect_list_and_release(list, "T1.E4.E8.E2.E2.D1 E2.E4.E8.E2.E2.D1 T3.E4.E8.E2.E2.D1 E4.E4.E8.E2.E2.D1 T5.E4.E8.E2.E2.D1 E6.E4.E8.E2.E2.D1 T7.E4.E8.E2.E2.D1");
3947     helper_ole_check(IXMLDOMNode_get_text(node2, &bstr));
3948     helper_expect_bstr_and_release(bstr, "\n                This is a description with preserved whitespace. \n            ");
3949     IXMLDOMNode_Release(node2);
3950 }
3951
3952 static void test_whitespace(void)
3953 {
3954     VARIANT_BOOL b;
3955     IXMLDOMDocument2 *doc1, *doc2, *doc3, *doc4;
3956
3957     doc1 = create_document(&IID_IXMLDOMDocument2);
3958     doc2 = create_document(&IID_IXMLDOMDocument2);
3959     if (!doc1 || !doc2) return;
3960
3961     ole_check(IXMLDOMDocument2_put_preserveWhiteSpace(doc2, VARIANT_TRUE));
3962     ole_check(IXMLDOMDocument2_get_preserveWhiteSpace(doc1, &b));
3963     ok(b == VARIANT_FALSE, "expected false\n");
3964     ole_check(IXMLDOMDocument2_get_preserveWhiteSpace(doc2, &b));
3965     ok(b == VARIANT_TRUE, "expected true\n");
3966
3967     ole_check(IXMLDOMDocument2_loadXML(doc1, _bstr_(szExampleXML), &b));
3968     ok(b == VARIANT_TRUE, "failed to load XML string\n");
3969     ole_check(IXMLDOMDocument2_loadXML(doc2, _bstr_(szExampleXML), &b));
3970     ok(b == VARIANT_TRUE, "failed to load XML string\n");
3971
3972     /* switch to XPath */
3973     ole_check(IXMLDOMDocument2_setProperty(doc1, _bstr_("SelectionLanguage"), _variantbstr_("XPath")));
3974     ole_check(IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionLanguage"), _variantbstr_("XPath")));
3975
3976     check_ws_ignored(doc1, NULL);
3977     check_ws_preserved(doc2, NULL);
3978
3979     /* new instances copy the property */
3980     ole_check(IXMLDOMDocument2_QueryInterface(doc1, &IID_IXMLDOMDocument2, (void**) &doc3));
3981     ole_check(IXMLDOMDocument2_QueryInterface(doc2, &IID_IXMLDOMDocument2, (void**) &doc4));
3982
3983     ole_check(IXMLDOMDocument2_get_preserveWhiteSpace(doc3, &b));
3984     ok(b == VARIANT_FALSE, "expected false\n");
3985     ole_check(IXMLDOMDocument2_get_preserveWhiteSpace(doc4, &b));
3986     ok(b == VARIANT_TRUE, "expected true\n");
3987
3988     check_ws_ignored(doc3, NULL);
3989     check_ws_preserved(doc4, NULL);
3990
3991     /* setting after loading xml affects trimming of leading/trailing ws only */
3992     ole_check(IXMLDOMDocument2_put_preserveWhiteSpace(doc1, VARIANT_TRUE));
3993     ole_check(IXMLDOMDocument2_put_preserveWhiteSpace(doc2, VARIANT_FALSE));
3994
3995     /* the trailing "\n            " isn't there, because it was ws-only node */
3996     check_ws_ignored(doc1, "\n                This is a description. ");
3997     check_ws_preserved(doc2, "This is a description.");
3998
3999     /* it takes effect on reload */
4000     ole_check(IXMLDOMDocument2_get_preserveWhiteSpace(doc1, &b));
4001     ok(b == VARIANT_TRUE, "expected true\n");
4002     ole_check(IXMLDOMDocument2_get_preserveWhiteSpace(doc2, &b));
4003     ok(b == VARIANT_FALSE, "expected false\n");
4004
4005     ole_check(IXMLDOMDocument2_loadXML(doc1, _bstr_(szExampleXML), &b));
4006     ok(b == VARIANT_TRUE, "failed to load XML string\n");
4007     ole_check(IXMLDOMDocument2_loadXML(doc2, _bstr_(szExampleXML), &b));
4008     ok(b == VARIANT_TRUE, "failed to load XML string\n");
4009
4010     check_ws_preserved(doc1, NULL);
4011     check_ws_ignored(doc2, NULL);
4012
4013     /* other instances follow suit */
4014     ole_check(IXMLDOMDocument2_get_preserveWhiteSpace(doc3, &b));
4015     ok(b == VARIANT_TRUE, "expected true\n");
4016     ole_check(IXMLDOMDocument2_get_preserveWhiteSpace(doc4, &b));
4017     ok(b == VARIANT_FALSE, "expected false\n");
4018
4019     check_ws_preserved(doc3, NULL);
4020     check_ws_ignored(doc4, NULL);
4021
4022     IXMLDOMDocument_Release(doc1);
4023     IXMLDOMDocument_Release(doc2);
4024     IXMLDOMDocument_Release(doc3);
4025     IXMLDOMDocument_Release(doc4);
4026     free_bstrs();
4027 }
4028
4029 static void test_XPath(void)
4030 {
4031     VARIANT var;
4032     VARIANT_BOOL b;
4033     IXMLDOMDocument2 *doc;
4034     IXMLDOMNode *rootNode;
4035     IXMLDOMNode *elem1Node;
4036     IXMLDOMNode *node;
4037     IXMLDOMNodeList *list;
4038
4039     doc = create_document(&IID_IXMLDOMDocument2);
4040     if (!doc) return;
4041
4042     ole_check(IXMLDOMDocument_loadXML(doc, _bstr_(szExampleXML), &b));
4043     ok(b == VARIANT_TRUE, "failed to load XML string\n");
4044
4045     /* switch to XPath */
4046     ole_check(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionLanguage"), _variantbstr_("XPath")));
4047
4048     /* some simple queries*/
4049     ole_check(IXMLDOMDocument_selectNodes(doc, _bstr_("root"), &list));
4050     ole_check(IXMLDOMNodeList_get_item(list, 0, &rootNode));
4051     ole_check(IXMLDOMNodeList_reset(list));
4052     expect_list_and_release(list, "E2.D1");
4053     if (rootNode == NULL)
4054         return;
4055
4056     ole_check(IXMLDOMDocument_selectNodes(doc, _bstr_("root//c"), &list));
4057     expect_list_and_release(list, "E3.E1.E2.D1 E3.E2.E2.D1");
4058
4059     ole_check(IXMLDOMDocument_selectNodes(doc, _bstr_("//c[@type]"), &list));
4060     expect_list_and_release(list, "E3.E2.E2.D1");
4061
4062     ole_check(IXMLDOMNode_selectNodes(rootNode, _bstr_("elem"), &list));
4063     /* using get_item for query results advances the position */
4064     ole_check(IXMLDOMNodeList_get_item(list, 1, &node));
4065     expect_node(node, "E2.E2.D1");
4066     IXMLDOMNode_Release(node);
4067     ole_check(IXMLDOMNodeList_nextNode(list, &node));
4068     expect_node(node, "E4.E2.D1");
4069     IXMLDOMNode_Release(node);
4070     ole_check(IXMLDOMNodeList_reset(list));
4071     expect_list_and_release(list, "E1.E2.D1 E2.E2.D1 E4.E2.D1");
4072
4073     ole_check(IXMLDOMNode_selectNodes(rootNode, _bstr_("."), &list));
4074     expect_list_and_release(list, "E2.D1");
4075
4076     ole_check(IXMLDOMNode_selectNodes(rootNode, _bstr_("elem[3]/preceding-sibling::*"), &list));
4077     ole_check(IXMLDOMNodeList_get_item(list, 0, &elem1Node));
4078     ole_check(IXMLDOMNodeList_reset(list));
4079     expect_list_and_release(list, "E1.E2.D1 E2.E2.D1 E3.E2.D1");
4080
4081     /* select an attribute */
4082     ole_check(IXMLDOMNode_selectNodes(rootNode, _bstr_(".//@type"), &list));
4083     expect_list_and_release(list, "A'type'.E3.E2.E2.D1");
4084
4085     /* would evaluate to a number */
4086     ole_expect(IXMLDOMNode_selectNodes(rootNode, _bstr_("count(*)"), &list), E_FAIL);
4087     /* would evaluate to a boolean */
4088     ole_expect(IXMLDOMNode_selectNodes(rootNode, _bstr_("position()>0"), &list), E_FAIL);
4089     /* would evaluate to a string */
4090     ole_expect(IXMLDOMNode_selectNodes(rootNode, _bstr_("name()"), &list), E_FAIL);
4091
4092     /* no results */
4093     ole_check(IXMLDOMNode_selectNodes(rootNode, _bstr_("c"), &list));
4094     expect_list_and_release(list, "");
4095     ole_check(IXMLDOMDocument_selectNodes(doc, _bstr_("elem//c"), &list));
4096     expect_list_and_release(list, "");
4097     ole_check(IXMLDOMDocument_selectNodes(doc, _bstr_("//elem[4]"), &list));
4098     expect_list_and_release(list, "");
4099     ole_check(IXMLDOMDocument_selectNodes(doc, _bstr_("root//elem[0]"), &list));
4100     expect_list_and_release(list, "");
4101
4102     /* foo undeclared in document node */
4103     ole_expect(IXMLDOMDocument_selectNodes(doc, _bstr_("root//foo:c"), &list), E_FAIL);
4104     /* undeclared in <root> node */
4105     ole_expect(IXMLDOMNode_selectNodes(rootNode, _bstr_(".//foo:c"), &list), E_FAIL);
4106     /* undeclared in <elem> node */
4107     ole_expect(IXMLDOMNode_selectNodes(elem1Node, _bstr_("//foo:c"), &list), E_FAIL);
4108     /* but this trick can be used */
4109     ole_check(IXMLDOMNode_selectNodes(elem1Node, _bstr_("//*[name()='foo:c']"), &list));
4110     expect_list_and_release(list, "E3.E4.E2.D1");
4111
4112     /* it has to be declared in SelectionNamespaces */
4113     ole_check(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionNamespaces"),
4114         _variantbstr_("xmlns:test='urn:uuid:86B2F87F-ACB6-45cd-8B77-9BDB92A01A29'")));
4115
4116     /* now the namespace can be used */
4117     ole_check(IXMLDOMDocument_selectNodes(doc, _bstr_("root//test:c"), &list));
4118     expect_list_and_release(list, "E3.E3.E2.D1 E3.E4.E2.D1");
4119     ole_check(IXMLDOMNode_selectNodes(rootNode, _bstr_(".//test:c"), &list));
4120     expect_list_and_release(list, "E3.E3.E2.D1 E3.E4.E2.D1");
4121     ole_check(IXMLDOMNode_selectNodes(elem1Node, _bstr_("//test:c"), &list));
4122     expect_list_and_release(list, "E3.E3.E2.D1 E3.E4.E2.D1");
4123     ole_check(IXMLDOMNode_selectNodes(elem1Node, _bstr_(".//test:x"), &list));
4124     expect_list_and_release(list, "E5.E1.E4.E1.E2.D1 E6.E2.E4.E1.E2.D1");
4125
4126     /* SelectionNamespaces syntax error - the namespaces doesn't work anymore but the value is stored */
4127     ole_expect(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionNamespaces"),
4128         _variantbstr_("xmlns:test='urn:uuid:86B2F87F-ACB6-45cd-8B77-9BDB92A01A29' xmlns:foo=###")), E_FAIL);
4129
4130     ole_expect(IXMLDOMDocument_selectNodes(doc, _bstr_("root//foo:c"), &list), E_FAIL);
4131
4132     VariantInit(&var);
4133     ole_check(IXMLDOMDocument2_getProperty(doc, _bstr_("SelectionNamespaces"), &var));
4134     expect_eq(V_VT(&var), VT_BSTR, int, "%x");
4135     if (V_VT(&var) == VT_BSTR)
4136         expect_bstr_eq_and_free(V_BSTR(&var), "xmlns:test='urn:uuid:86B2F87F-ACB6-45cd-8B77-9BDB92A01A29' xmlns:foo=###");
4137
4138     /* extra attributes - same thing*/
4139     ole_expect(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionNamespaces"),
4140         _variantbstr_("xmlns:test='urn:uuid:86B2F87F-ACB6-45cd-8B77-9BDB92A01A29' param='test'")), E_FAIL);
4141     ole_expect(IXMLDOMDocument_selectNodes(doc, _bstr_("root//foo:c"), &list), E_FAIL);
4142
4143     IXMLDOMNode_Release(rootNode);
4144     IXMLDOMNode_Release(elem1Node);
4145     IXMLDOMDocument_Release(doc);
4146     free_bstrs();
4147 }
4148
4149 static void test_cloneNode(void )
4150 {
4151     IXMLDOMDocument *doc;
4152     VARIANT_BOOL b;
4153     IXMLDOMNodeList *pList;
4154     IXMLDOMNamedNodeMap *mapAttr;
4155     LONG nLength = 0, nLength1 = 0;
4156     LONG nAttrCnt = 0, nAttrCnt1 = 0;
4157     IXMLDOMNode *node;
4158     IXMLDOMNode *node_clone;
4159     IXMLDOMNode *node_first;
4160     HRESULT r;
4161     BSTR str;
4162     static const WCHAR szSearch[] = { 'l', 'c', '/', 'p', 'r', 0 };
4163
4164     doc = create_document(&IID_IXMLDOMDocument);
4165     if (!doc) return;
4166
4167     str = SysAllocString( szComplete4 );
4168     ole_check(IXMLDOMDocument_loadXML(doc, str, &b));
4169     ok(b == VARIANT_TRUE, "failed to load XML string\n");
4170     SysFreeString(str);
4171
4172     if(!b)
4173         return;
4174
4175     str = SysAllocString( szSearch);
4176     r = IXMLDOMNode_selectSingleNode(doc, str, &node);
4177     ok( r == S_OK, "ret %08x\n", r );
4178     ok( node != NULL, "node %p\n", node );
4179     SysFreeString(str);
4180
4181     if(!node)
4182     {
4183         IXMLDOMDocument_Release(doc);
4184         return;
4185     }
4186
4187     /* Check invalid parameter */
4188     r = IXMLDOMNode_cloneNode(node, VARIANT_TRUE, NULL);
4189     ok( r == E_INVALIDARG, "ret %08x\n", r );
4190
4191     /* All Children */
4192     r = IXMLDOMNode_cloneNode(node, VARIANT_TRUE, &node_clone);
4193     ok( r == S_OK, "ret %08x\n", r );
4194     ok( node_clone != NULL, "node %p\n", node );
4195
4196     if(!node_clone)
4197     {
4198         IXMLDOMDocument_Release(doc);
4199         IXMLDOMNode_Release(node);
4200         return;
4201     }
4202
4203     r = IXMLDOMNode_get_firstChild(node_clone, &node_first);
4204     ok( r == S_OK, "ret %08x\n", r );
4205     if(r == S_OK)
4206     {
4207         IXMLDOMDocument *doc2;
4208
4209         r = IXMLDOMNode_get_ownerDocument(node_clone, &doc2);
4210         ok( r == S_OK, "ret %08x\n", r );
4211         if(r == S_OK)
4212             IXMLDOMDocument_Release(doc2);
4213
4214         IXMLDOMNode_Release(node_first);
4215     }
4216
4217     r = IXMLDOMNode_get_childNodes(node, &pList);
4218     ok( r == S_OK, "ret %08x\n", r );
4219     if (pList)
4220         {
4221                 IXMLDOMNodeList_get_length(pList, &nLength);
4222                 IXMLDOMNodeList_Release(pList);
4223         }
4224
4225     r = IXMLDOMNode_get_attributes(node, &mapAttr);
4226     ok( r == S_OK, "ret %08x\n", r );
4227     if(mapAttr)
4228     {
4229         IXMLDOMNamedNodeMap_get_length(mapAttr, &nAttrCnt);
4230         IXMLDOMNamedNodeMap_Release(mapAttr);
4231     }
4232
4233     r = IXMLDOMNode_get_childNodes(node_clone, &pList);
4234     ok( r == S_OK, "ret %08x\n", r );
4235     if (pList)
4236         {
4237                 IXMLDOMNodeList_get_length(pList, &nLength1);
4238                 IXMLDOMNodeList_Release(pList);
4239         }
4240
4241     r = IXMLDOMNode_get_attributes(node_clone, &mapAttr);
4242     ok( r == S_OK, "ret %08x\n", r );
4243     if(mapAttr)
4244     {
4245         IXMLDOMNamedNodeMap_get_length(mapAttr, &nAttrCnt1);
4246         IXMLDOMNamedNodeMap_Release(mapAttr);
4247     }
4248
4249     ok(nLength == nLength1, "wrong Child count (%d, %d)\n", nLength, nLength1);
4250     ok(nAttrCnt == nAttrCnt1, "wrong Attribute count (%d, %d)\n", nAttrCnt, nAttrCnt1);
4251     IXMLDOMNode_Release(node_clone);
4252
4253     /* No Children */
4254     r = IXMLDOMNode_cloneNode(node, VARIANT_FALSE, &node_clone);
4255     ok( r == S_OK, "ret %08x\n", r );
4256     ok( node_clone != NULL, "node %p\n", node );
4257
4258     if(!node_clone)
4259     {
4260         IXMLDOMDocument_Release(doc);
4261         IXMLDOMNode_Release(node);
4262         return;
4263     }
4264
4265     r = IXMLDOMNode_get_firstChild(node_clone, &node_first);
4266     ok( r == S_FALSE, "ret %08x\n", r );
4267     if(r == S_OK)
4268     {
4269         IXMLDOMDocument *doc2;
4270
4271         r = IXMLDOMNode_get_ownerDocument(node_clone, &doc2);
4272         ok( r == S_OK, "ret %08x\n", r );
4273         if(r == S_OK)
4274             IXMLDOMDocument_Release(doc2);
4275
4276         IXMLDOMNode_Release(node_first);
4277     }
4278
4279     r = IXMLDOMNode_get_childNodes(node_clone, &pList);
4280     ok( r == S_OK, "ret %08x\n", r );
4281     if (pList)
4282     {
4283         IXMLDOMNodeList_get_length(pList, &nLength1);
4284         ok( nLength1 == 0, "Length should be 0 (%d)\n", nLength1);
4285         IXMLDOMNodeList_Release(pList);
4286     }
4287
4288     r = IXMLDOMNode_get_attributes(node_clone, &mapAttr);
4289     ok( r == S_OK, "ret %08x\n", r );
4290     if(mapAttr)
4291     {
4292         IXMLDOMNamedNodeMap_get_length(mapAttr, &nAttrCnt1);
4293         ok( nAttrCnt1 == 3, "Attribute count should be 3 (%d)\n", nAttrCnt1);
4294         IXMLDOMNamedNodeMap_Release(mapAttr);
4295     }
4296
4297     ok(nLength != nLength1, "wrong Child count (%d, %d)\n", nLength, nLength1);
4298     ok(nAttrCnt == nAttrCnt1, "wrong Attribute count (%d, %d)\n", nAttrCnt, nAttrCnt1);
4299     IXMLDOMNode_Release(node_clone);
4300
4301
4302     IXMLDOMNode_Release(node);
4303     IXMLDOMDocument_Release(doc);
4304 }
4305
4306 static void test_xmlTypes(void)
4307 {
4308     IXMLDOMDocument *doc;
4309     IXMLDOMElement *pRoot;
4310     HRESULT hr;
4311     IXMLDOMComment *pComment;
4312     IXMLDOMElement *pElement;
4313     IXMLDOMAttribute *pAttribute;
4314     IXMLDOMNamedNodeMap *pAttribs;
4315     IXMLDOMCDATASection *pCDataSec;
4316     IXMLDOMImplementation *pIXMLDOMImplementation = NULL;
4317     IXMLDOMDocumentFragment *pDocFrag = NULL;
4318     IXMLDOMEntityReference *pEntityRef = NULL;
4319     BSTR str;
4320     IXMLDOMNode *pNextChild;
4321     VARIANT v;
4322     LONG len = 0;
4323
4324     doc = create_document(&IID_IXMLDOMDocument);
4325     if (!doc) return;
4326
4327     pNextChild = (void*)0xdeadbeef;
4328     hr = IXMLDOMDocument_get_nextSibling(doc, NULL);
4329     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4330
4331     pNextChild = (void*)0xdeadbeef;
4332     hr = IXMLDOMDocument_get_nextSibling(doc, &pNextChild);
4333     ok(hr == S_FALSE, "ret %08x\n", hr );
4334     ok(pNextChild == NULL, "pDocChild not NULL\n");
4335
4336     /* test previous Sibling */
4337     hr = IXMLDOMDocument_get_previousSibling(doc, NULL);
4338     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4339
4340     pNextChild = (void*)0xdeadbeef;
4341     hr = IXMLDOMDocument_get_previousSibling(doc, &pNextChild);
4342     ok(hr == S_FALSE, "ret %08x\n", hr );
4343     ok(pNextChild == NULL, "pNextChild not NULL\n");
4344
4345     /* test get_attributes */
4346     hr = IXMLDOMDocument_get_attributes( doc, NULL );
4347     ok( hr == E_INVALIDARG, "get_attributes returned wrong code\n");
4348
4349     pAttribs = (void*)0xdeadbeef;
4350     hr = IXMLDOMDocument_get_attributes( doc, &pAttribs);
4351     ok(hr == S_FALSE, "ret %08x\n", hr );
4352     ok( pAttribs == NULL, "pAttribs not NULL\n");
4353
4354     /* test get_dataType */
4355     V_VT(&v) = VT_EMPTY;
4356     hr = IXMLDOMDocument_get_dataType(doc, &v);
4357     ok(hr == S_FALSE, "ret %08x\n", hr );
4358     ok( V_VT(&v) == VT_NULL, "incorrect dataType type\n");
4359     VariantClear(&v);
4360
4361     /* test nodeTypeString */
4362     str = NULL;
4363     hr = IXMLDOMDocument_get_nodeTypeString(doc, &str);
4364     ok(hr == S_OK, "ret %08x\n", hr );
4365     ok( !lstrcmpW( str, _bstr_("document") ), "incorrect nodeTypeString string\n");
4366     SysFreeString(str);
4367
4368     /* test implementation */
4369     hr = IXMLDOMDocument_get_implementation(doc, NULL);
4370     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4371
4372     hr = IXMLDOMDocument_get_implementation(doc, &pIXMLDOMImplementation);
4373     ok(hr == S_OK, "ret %08x\n", hr );
4374     if(hr == S_OK)
4375     {
4376         VARIANT_BOOL hasFeature = VARIANT_TRUE;
4377         BSTR sEmpty = SysAllocStringLen(NULL, 0);
4378
4379         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, NULL, sEmpty, &hasFeature);
4380         ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4381
4382         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, sEmpty, sEmpty, NULL);
4383         ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4384
4385         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, _bstr_("DOM"), sEmpty, &hasFeature);
4386         ok(hr == S_OK, "ret %08x\n", hr );
4387         ok(hasFeature == VARIANT_FALSE, "hasFeature returned false\n");
4388
4389         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, sEmpty, sEmpty, &hasFeature);
4390         ok(hr == S_OK, "ret %08x\n", hr );
4391         ok(hasFeature == VARIANT_FALSE, "hasFeature returned true\n");
4392
4393         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, _bstr_("DOM"), NULL, &hasFeature);
4394         ok(hr == S_OK, "ret %08x\n", hr );
4395         ok(hasFeature == VARIANT_TRUE, "hasFeature returned false\n");
4396
4397         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, _bstr_("DOM"), sEmpty, &hasFeature);
4398         ok(hr == S_OK, "ret %08x\n", hr );
4399         ok(hasFeature == VARIANT_FALSE, "hasFeature returned false\n");
4400
4401         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, _bstr_("DOM"), _bstr_("1.0"), &hasFeature);
4402         ok(hr == S_OK, "ret %08x\n", hr );
4403         ok(hasFeature == VARIANT_TRUE, "hasFeature returned true\n");
4404
4405         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, _bstr_("XML"), _bstr_("1.0"), &hasFeature);
4406         ok(hr == S_OK, "ret %08x\n", hr );
4407         ok(hasFeature == VARIANT_TRUE, "hasFeature returned true\n");
4408
4409         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, _bstr_("MS-DOM"), _bstr_("1.0"), &hasFeature);
4410         ok(hr == S_OK, "ret %08x\n", hr );
4411         ok(hasFeature == VARIANT_TRUE, "hasFeature returned true\n");
4412
4413         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, _bstr_("SSS"), NULL, &hasFeature);
4414         ok(hr == S_OK, "ret %08x\n", hr );
4415         ok(hasFeature == VARIANT_FALSE, "hasFeature returned false\n");
4416
4417         SysFreeString(sEmpty);
4418         IXMLDOMImplementation_Release(pIXMLDOMImplementation);
4419     }
4420
4421     pRoot = (IXMLDOMElement*)0x1;
4422     hr = IXMLDOMDocument_createElement(doc, NULL, &pRoot);
4423     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4424     ok(pRoot == (void*)0x1, "Expect same ptr, got %p\n", pRoot);
4425
4426     pRoot = (IXMLDOMElement*)0x1;
4427     hr = IXMLDOMDocument_createElement(doc, _bstr_(""), &pRoot);
4428     ok(hr == E_FAIL, "ret %08x\n", hr );
4429     ok(pRoot == (void*)0x1, "Expect same ptr, got %p\n", pRoot);
4430
4431     hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing"), &pRoot);
4432     ok(hr == S_OK, "ret %08x\n", hr );
4433     if(hr == S_OK)
4434     {
4435         hr = IXMLDOMDocument_appendChild(doc, (IXMLDOMNode*)pRoot, NULL);
4436         ok(hr == S_OK, "ret %08x\n", hr );
4437         if(hr == S_OK)
4438         {
4439             /* Comment */
4440             str = SysAllocString(szComment);
4441             hr = IXMLDOMDocument_createComment(doc, str, &pComment);
4442             SysFreeString(str);
4443             ok(hr == S_OK, "ret %08x\n", hr );
4444             if(hr == S_OK)
4445             {
4446                 /* test get_attributes */
4447                 hr = IXMLDOMComment_get_attributes( pComment, NULL );
4448                 ok( hr == E_INVALIDARG, "get_attributes returned wrong code\n");
4449
4450                 pAttribs = (IXMLDOMNamedNodeMap*)0x1;
4451                 hr = IXMLDOMComment_get_attributes( pComment, &pAttribs);
4452                 ok(hr == S_FALSE, "ret %08x\n", hr );
4453                 ok( pAttribs == NULL, "pAttribs not NULL\n");
4454
4455                 /* test nodeTypeString */
4456                 hr = IXMLDOMComment_get_nodeTypeString(pComment, &str);
4457                 ok(hr == S_OK, "ret %08x\n", hr );
4458                 ok( !lstrcmpW( str, _bstr_("comment") ), "incorrect nodeTypeString string\n");
4459                 SysFreeString(str);
4460
4461                 hr = IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pComment, NULL);
4462                 ok(hr == S_OK, "ret %08x\n", hr );
4463
4464                 hr = IXMLDOMComment_get_nodeName(pComment, &str);
4465                 ok(hr == S_OK, "ret %08x\n", hr );
4466                 ok( !lstrcmpW( str, szCommentNodeText ), "incorrect comment node Name\n");
4467                 SysFreeString(str);
4468
4469                 hr = IXMLDOMComment_get_xml(pComment, &str);
4470                 ok(hr == S_OK, "ret %08x\n", hr );
4471                 ok( !lstrcmpW( str, szCommentXML ), "incorrect comment xml\n");
4472                 SysFreeString(str);
4473
4474                 hr = IXMLDOMComment_get_dataType(pComment, &v);
4475                 ok(hr == S_FALSE, "ret %08x\n", hr );
4476                 ok( V_VT(&v) == VT_NULL, "incorrect dataType type\n");
4477                 VariantClear(&v);
4478
4479                 /* put data Tests */
4480                 hr = IXMLDOMComment_put_data(pComment, _bstr_("This &is a ; test <>\\"));
4481                 ok(hr == S_OK, "ret %08x\n", hr );
4482
4483                 /* get data Tests */
4484                 hr = IXMLDOMComment_get_data(pComment, &str);
4485                 ok(hr == S_OK, "ret %08x\n", hr );
4486                 ok( !lstrcmpW( str, _bstr_("This &is a ; test <>\\") ), "incorrect get_data string\n");
4487                 SysFreeString(str);
4488
4489                 /* get data Tests */
4490                 hr = IXMLDOMComment_get_nodeValue(pComment, &v);
4491                 ok(hr == S_OK, "ret %08x\n", hr );
4492                 ok( V_VT(&v) == VT_BSTR, "incorrect dataType type\n");
4493                 ok( !lstrcmpW( V_BSTR(&v), _bstr_("This &is a ; test <>\\") ), "incorrect get_nodeValue string\n");
4494                 VariantClear(&v);
4495
4496                 /* Confirm XML text is good */
4497                 hr = IXMLDOMComment_get_xml(pComment, &str);
4498                 ok(hr == S_OK, "ret %08x\n", hr );
4499                 ok( !lstrcmpW( str, _bstr_("<!--This &is a ; test <>\\-->") ), "incorrect xml string\n");
4500                 SysFreeString(str);
4501
4502                 /* Confirm we get the put_data Text back */
4503                 hr = IXMLDOMComment_get_text(pComment, &str);
4504                 ok(hr == S_OK, "ret %08x\n", hr );
4505                 ok( !lstrcmpW( str, _bstr_("This &is a ; test <>\\") ), "incorrect xml string\n");
4506                 SysFreeString(str);
4507
4508                 /* test length property */
4509                 hr = IXMLDOMComment_get_length(pComment, &len);
4510                 ok(hr == S_OK, "ret %08x\n", hr );
4511                 ok(len == 21, "expected 21 got %d\n", len);
4512
4513                 /* test substringData */
4514                 hr = IXMLDOMComment_substringData(pComment, 0, 4, NULL);
4515                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4516
4517                 /* test substringData - Invalid offset */
4518                 str = (BSTR)&szElement;
4519                 hr = IXMLDOMComment_substringData(pComment, -1, 4, &str);
4520                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4521                 ok( str == NULL, "incorrect string\n");
4522
4523                 /* test substringData - Invalid offset */
4524                 str = (BSTR)&szElement;
4525                 hr = IXMLDOMComment_substringData(pComment, 30, 0, &str);
4526                 ok(hr == S_FALSE, "ret %08x\n", hr );
4527                 ok( str == NULL, "incorrect string\n");
4528
4529                 /* test substringData - Invalid size */
4530                 str = (BSTR)&szElement;
4531                 hr = IXMLDOMComment_substringData(pComment, 0, -1, &str);
4532                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4533                 ok( str == NULL, "incorrect string\n");
4534
4535                 /* test substringData - Invalid size */
4536                 str = (BSTR)&szElement;
4537                 hr = IXMLDOMComment_substringData(pComment, 2, 0, &str);
4538                 ok(hr == S_FALSE, "ret %08x\n", hr );
4539                 ok( str == NULL, "incorrect string\n");
4540
4541                 /* test substringData - Start of string */
4542                 hr = IXMLDOMComment_substringData(pComment, 0, 4, &str);
4543                 ok(hr == S_OK, "ret %08x\n", hr );
4544                 ok( !lstrcmpW( str, _bstr_("This") ), "incorrect substringData string\n");
4545                 SysFreeString(str);
4546
4547                 /* test substringData - Middle of string */
4548                 hr = IXMLDOMComment_substringData(pComment, 13, 4, &str);
4549                 ok(hr == S_OK, "ret %08x\n", hr );
4550                 ok( !lstrcmpW( str, _bstr_("test") ), "incorrect substringData string\n");
4551                 SysFreeString(str);
4552
4553                 /* test substringData - End of string */
4554                 hr = IXMLDOMComment_substringData(pComment, 20, 4, &str);
4555                 ok(hr == S_OK, "ret %08x\n", hr );
4556                 ok( !lstrcmpW( str, _bstr_("\\") ), "incorrect substringData string\n");
4557                 SysFreeString(str);
4558
4559                 /* test appendData */
4560                 hr = IXMLDOMComment_appendData(pComment, NULL);
4561                 ok(hr == S_OK, "ret %08x\n", hr );
4562
4563                 hr = IXMLDOMComment_appendData(pComment, _bstr_(""));
4564                 ok(hr == S_OK, "ret %08x\n", hr );
4565
4566                 hr = IXMLDOMComment_appendData(pComment, _bstr_("Append"));
4567                 ok(hr == S_OK, "ret %08x\n", hr );
4568
4569                 hr = IXMLDOMComment_get_text(pComment, &str);
4570                 ok(hr == S_OK, "ret %08x\n", hr );
4571                 ok( !lstrcmpW( str, _bstr_("This &is a ; test <>\\Append") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4572                 SysFreeString(str);
4573
4574                 /* test insertData */
4575                 str = SysAllocStringLen(NULL, 0);
4576                 hr = IXMLDOMComment_insertData(pComment, -1, str);
4577                 ok(hr == S_OK, "ret %08x\n", hr );
4578
4579                 hr = IXMLDOMComment_insertData(pComment, -1, NULL);
4580                 ok(hr == S_OK, "ret %08x\n", hr );
4581
4582                 hr = IXMLDOMComment_insertData(pComment, 1000, str);
4583                 ok(hr == S_OK, "ret %08x\n", hr );
4584
4585                 hr = IXMLDOMComment_insertData(pComment, 1000, NULL);
4586                 ok(hr == S_OK, "ret %08x\n", hr );
4587
4588                 hr = IXMLDOMComment_insertData(pComment, 0, NULL);
4589                 ok(hr == S_OK, "ret %08x\n", hr );
4590
4591                 hr = IXMLDOMComment_insertData(pComment, 0, str);
4592                 ok(hr == S_OK, "ret %08x\n", hr );
4593                 SysFreeString(str);
4594
4595                 hr = IXMLDOMComment_insertData(pComment, -1, _bstr_("Inserting"));
4596                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4597
4598                 hr = IXMLDOMComment_insertData(pComment, 1000, _bstr_("Inserting"));
4599                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4600
4601                 hr = IXMLDOMComment_insertData(pComment, 0, _bstr_("Begin "));
4602                 ok(hr == S_OK, "ret %08x\n", hr );
4603
4604                 hr = IXMLDOMComment_insertData(pComment, 17, _bstr_("Middle"));
4605                 ok(hr == S_OK, "ret %08x\n", hr );
4606
4607                 hr = IXMLDOMComment_insertData(pComment, 39, _bstr_(" End"));
4608                 ok(hr == S_OK, "ret %08x\n", hr );
4609
4610                 hr = IXMLDOMComment_get_text(pComment, &str);
4611                 ok(hr == S_OK, "ret %08x\n", hr );
4612                 ok( !lstrcmpW( str, _bstr_("Begin This &is a Middle; test <>\\Append End") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4613                 SysFreeString(str);
4614
4615                 /* delete data */
4616                 /* invalid arguments */
4617                 hr = IXMLDOMComment_deleteData(pComment, -1, 1);
4618                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4619
4620                 hr = IXMLDOMComment_deleteData(pComment, 0, 0);
4621                 ok(hr == S_OK, "ret %08x\n", hr );
4622
4623                 hr = IXMLDOMComment_deleteData(pComment, 0, -1);
4624                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4625
4626                 hr = IXMLDOMComment_get_length(pComment, &len);
4627                 ok(hr == S_OK, "ret %08x\n", hr );
4628                 ok(len == 43, "expected 43 got %d\n", len);
4629
4630                 hr = IXMLDOMComment_deleteData(pComment, len, 1);
4631                 ok(hr == S_OK, "ret %08x\n", hr );
4632
4633                 hr = IXMLDOMComment_deleteData(pComment, len+1, 1);
4634                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4635
4636                 /* delete from start */
4637                 hr = IXMLDOMComment_deleteData(pComment, 0, 5);
4638                 ok(hr == S_OK, "ret %08x\n", hr );
4639
4640                 hr = IXMLDOMComment_get_length(pComment, &len);
4641                 ok(hr == S_OK, "ret %08x\n", hr );
4642                 ok(len == 38, "expected 38 got %d\n", len);
4643
4644                 hr = IXMLDOMComment_get_text(pComment, &str);
4645                 ok(hr == S_OK, "ret %08x\n", hr );
4646                 ok( !lstrcmpW( str, _bstr_(" This &is a Middle; test <>\\Append End") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4647                 SysFreeString(str);
4648
4649                 /* delete from end */
4650                 hr = IXMLDOMComment_deleteData(pComment, 35, 3);
4651                 ok(hr == S_OK, "ret %08x\n", hr );
4652
4653                 hr = IXMLDOMComment_get_length(pComment, &len);
4654                 ok(hr == S_OK, "ret %08x\n", hr );
4655                 ok(len == 35, "expected 35 got %d\n", len);
4656
4657                 hr = IXMLDOMComment_get_text(pComment, &str);
4658                 ok(hr == S_OK, "ret %08x\n", hr );
4659                 ok( !lstrcmpW( str, _bstr_(" This &is a Middle; test <>\\Append ") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4660                 SysFreeString(str);
4661
4662                 /* delete from inside */
4663                 hr = IXMLDOMComment_deleteData(pComment, 1, 33);
4664                 ok(hr == S_OK, "ret %08x\n", hr );
4665
4666                 hr = IXMLDOMComment_get_length(pComment, &len);
4667                 ok(hr == S_OK, "ret %08x\n", hr );
4668                 ok(len == 2, "expected 2 got %d\n", len);
4669
4670                 hr = IXMLDOMComment_get_text(pComment, &str);
4671                 ok(hr == S_OK, "ret %08x\n", hr );
4672                 ok( !lstrcmpW( str, _bstr_("  ") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4673                 SysFreeString(str);
4674
4675                 /* delete whole data ... */
4676                 hr = IXMLDOMComment_get_length(pComment, &len);
4677                 ok(hr == S_OK, "ret %08x\n", hr );
4678
4679                 hr = IXMLDOMComment_deleteData(pComment, 0, len);
4680                 ok(hr == S_OK, "ret %08x\n", hr );
4681                 /* ... and try again with empty string */
4682                 hr = IXMLDOMComment_deleteData(pComment, 0, len);
4683                 ok(hr == S_OK, "ret %08x\n", hr );
4684
4685                 /* ::replaceData() */
4686                 V_VT(&v) = VT_BSTR;
4687                 V_BSTR(&v) = SysAllocString(szstr1);
4688                 hr = IXMLDOMComment_put_nodeValue(pComment, v);
4689                 ok(hr == S_OK, "ret %08x\n", hr );
4690                 VariantClear(&v);
4691
4692                 hr = IXMLDOMComment_replaceData(pComment, 6, 0, NULL);
4693                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4694                 hr = IXMLDOMComment_get_text(pComment, &str);
4695                 ok(hr == S_OK, "ret %08x\n", hr );
4696                 ok( !lstrcmpW( str, _bstr_("str1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4697                 SysFreeString(str);
4698
4699                 hr = IXMLDOMComment_replaceData(pComment, 0, 0, NULL);
4700                 ok(hr == S_OK, "ret %08x\n", hr );
4701                 hr = IXMLDOMComment_get_text(pComment, &str);
4702                 ok(hr == S_OK, "ret %08x\n", hr );
4703                 ok( !lstrcmpW( str, _bstr_("str1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4704                 SysFreeString(str);
4705
4706                 /* NULL pointer means delete */
4707                 hr = IXMLDOMComment_replaceData(pComment, 0, 1, NULL);
4708                 ok(hr == S_OK, "ret %08x\n", hr );
4709                 hr = IXMLDOMComment_get_text(pComment, &str);
4710                 ok(hr == S_OK, "ret %08x\n", hr );
4711                 ok( !lstrcmpW( str, _bstr_("tr1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4712                 SysFreeString(str);
4713
4714                 /* empty string means delete */
4715                 hr = IXMLDOMComment_replaceData(pComment, 0, 1, _bstr_(""));
4716                 ok(hr == S_OK, "ret %08x\n", hr );
4717                 hr = IXMLDOMComment_get_text(pComment, &str);
4718                 ok(hr == S_OK, "ret %08x\n", hr );
4719                 ok( !lstrcmpW( str, _bstr_("r1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4720                 SysFreeString(str);
4721
4722                 /* zero count means insert */
4723                 hr = IXMLDOMComment_replaceData(pComment, 0, 0, _bstr_("a"));
4724                 ok(hr == S_OK, "ret %08x\n", hr );
4725                 hr = IXMLDOMComment_get_text(pComment, &str);
4726                 ok(hr == S_OK, "ret %08x\n", hr );
4727                 ok( !lstrcmpW( str, _bstr_("ar1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4728                 SysFreeString(str);
4729
4730                 hr = IXMLDOMComment_replaceData(pComment, 0, 2, NULL);
4731                 ok(hr == S_OK, "ret %08x\n", hr );
4732
4733                 hr = IXMLDOMComment_insertData(pComment, 0, _bstr_("m"));
4734                 ok(hr == S_OK, "ret %08x\n", hr );
4735                 hr = IXMLDOMComment_get_text(pComment, &str);
4736                 ok(hr == S_OK, "ret %08x\n", hr );
4737                 ok( !lstrcmpW( str, _bstr_("m1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4738                 SysFreeString(str);
4739
4740                 /* nonempty string, count greater than its length */
4741                 hr = IXMLDOMComment_replaceData(pComment, 0, 2, _bstr_("a1.2"));
4742                 ok(hr == S_OK, "ret %08x\n", hr );
4743                 hr = IXMLDOMComment_get_text(pComment, &str);
4744                 ok(hr == S_OK, "ret %08x\n", hr );
4745                 ok( !lstrcmpW( str, _bstr_("a1.2") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4746                 SysFreeString(str);
4747
4748                 /* nonempty string, count less than its length */
4749                 hr = IXMLDOMComment_replaceData(pComment, 0, 1, _bstr_("wine"));
4750                 ok(hr == S_OK, "ret %08x\n", hr );
4751                 hr = IXMLDOMComment_get_text(pComment, &str);
4752                 ok(hr == S_OK, "ret %08x\n", hr );
4753                 ok( !lstrcmpW( str, _bstr_("wine1.2") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4754                 SysFreeString(str);
4755
4756                 IXMLDOMComment_Release(pComment);
4757             }
4758
4759             /* Element */
4760             str = SysAllocString(szElement);
4761             hr = IXMLDOMDocument_createElement(doc, str, &pElement);
4762             SysFreeString(str);
4763             ok(hr == S_OK, "ret %08x\n", hr );
4764             if(hr == S_OK)
4765             {
4766                 hr = IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
4767                 ok(hr == S_OK, "ret %08x\n", hr );
4768
4769                 /* test nodeTypeString */
4770                 hr = IXMLDOMDocument_get_nodeTypeString(pElement, &str);
4771                 ok(hr == S_OK, "ret %08x\n", hr );
4772                 ok( !lstrcmpW( str, _bstr_("element") ), "incorrect nodeTypeString string\n");
4773                 SysFreeString(str);
4774
4775                 hr = IXMLDOMElement_get_nodeName(pElement, &str);
4776                 ok(hr == S_OK, "ret %08x\n", hr );
4777                 ok( !lstrcmpW( str, szElement ), "incorrect element node Name\n");
4778                 SysFreeString(str);
4779
4780                 hr = IXMLDOMElement_get_xml(pElement, &str);
4781                 ok(hr == S_OK, "ret %08x\n", hr );
4782                 ok( !lstrcmpW( str, szElementXML ), "incorrect element xml\n");
4783                 SysFreeString(str);
4784
4785                 hr = IXMLDOMElement_get_dataType(pElement, &v);
4786                 ok(hr == S_FALSE, "ret %08x\n", hr );
4787                 ok( V_VT(&v) == VT_NULL, "incorrect dataType type\n");
4788                 VariantClear(&v);
4789
4790                 /* Attribute */
4791                 pAttribute = (IXMLDOMAttribute*)0x1;
4792                 hr = IXMLDOMDocument_createAttribute(doc, NULL, &pAttribute);
4793                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4794                 ok(pAttribute == (void*)0x1, "Expect same ptr, got %p\n", pAttribute);
4795
4796                 pAttribute = (IXMLDOMAttribute*)0x1;
4797                 hr = IXMLDOMDocument_createAttribute(doc, _bstr_(""), &pAttribute);
4798                 ok(hr == E_FAIL, "ret %08x\n", hr );
4799                 ok(pAttribute == (void*)0x1, "Expect same ptr, got %p\n", pAttribute);
4800
4801                 str = SysAllocString(szAttribute);
4802                 hr = IXMLDOMDocument_createAttribute(doc, str, &pAttribute);
4803                 SysFreeString(str);
4804                 ok(hr == S_OK, "ret %08x\n", hr );
4805                 if(hr == S_OK)
4806                 {
4807                     IXMLDOMNode *pNewChild = (IXMLDOMNode *)0x1;
4808
4809                     hr = IXMLDOMAttribute_get_nextSibling(pAttribute, NULL);
4810                     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4811
4812                     pNextChild = (IXMLDOMNode *)0x1;
4813                     hr = IXMLDOMAttribute_get_nextSibling(pAttribute, &pNextChild);
4814                     ok(hr == S_FALSE, "ret %08x\n", hr );
4815                     ok(pNextChild == NULL, "pNextChild not NULL\n");
4816
4817                     /* test Previous Sibling*/
4818                     hr = IXMLDOMAttribute_get_previousSibling(pAttribute, NULL);
4819                     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4820
4821                     pNextChild = (IXMLDOMNode *)0x1;
4822                     hr = IXMLDOMAttribute_get_previousSibling(pAttribute, &pNextChild);
4823                     ok(hr == S_FALSE, "ret %08x\n", hr );
4824                     ok(pNextChild == NULL, "pNextChild not NULL\n");
4825
4826                     /* test get_attributes */
4827                     hr = IXMLDOMAttribute_get_attributes( pAttribute, NULL );
4828                     ok( hr == E_INVALIDARG, "get_attributes returned wrong code\n");
4829
4830                     pAttribs = (IXMLDOMNamedNodeMap*)0x1;
4831                     hr = IXMLDOMAttribute_get_attributes( pAttribute, &pAttribs);
4832                     ok(hr == S_FALSE, "ret %08x\n", hr );
4833                     ok( pAttribs == NULL, "pAttribs not NULL\n");
4834
4835                     hr = IXMLDOMElement_appendChild(pElement, (IXMLDOMNode*)pAttribute, &pNewChild);
4836                     ok(hr == E_FAIL, "ret %08x\n", hr );
4837                     ok(pNewChild == NULL, "pNewChild not NULL\n");
4838
4839                     hr = IXMLDOMElement_get_attributes(pElement, &pAttribs);
4840                     ok(hr == S_OK, "ret %08x\n", hr );
4841                     if ( hr == S_OK )
4842                     {
4843                         hr = IXMLDOMNamedNodeMap_setNamedItem(pAttribs, (IXMLDOMNode*)pAttribute, NULL );
4844                         ok(hr == S_OK, "ret %08x\n", hr );
4845
4846                         IXMLDOMNamedNodeMap_Release(pAttribs);
4847                     }
4848
4849                     hr = IXMLDOMAttribute_get_nodeName(pAttribute, &str);
4850                     ok(hr == S_OK, "ret %08x\n", hr );
4851                     ok( !lstrcmpW( str, szAttribute ), "incorrect attribute node Name\n");
4852                     SysFreeString(str);
4853
4854                     /* test nodeTypeString */
4855                     hr = IXMLDOMAttribute_get_nodeTypeString(pAttribute, &str);
4856                     ok(hr == S_OK, "ret %08x\n", hr );
4857                     ok( !lstrcmpW( str, _bstr_("attribute") ), "incorrect nodeTypeString string\n");
4858                     SysFreeString(str);
4859
4860                     /* test nodeName */
4861                     hr = IXMLDOMAttribute_get_nodeName(pAttribute, &str);
4862                     ok(hr == S_OK, "ret %08x\n", hr );
4863                     ok( !lstrcmpW( str, szAttribute ), "incorrect nodeName string\n");
4864                     SysFreeString(str);
4865
4866                     /* test name property */
4867                     hr = IXMLDOMAttribute_get_name(pAttribute, &str);
4868                     ok(hr == S_OK, "ret %08x\n", hr );
4869                     ok( !lstrcmpW( str, szAttribute ), "incorrect name string\n");
4870                     SysFreeString(str);
4871
4872                     hr = IXMLDOMAttribute_get_xml(pAttribute, &str);
4873                     ok(hr == S_OK, "ret %08x\n", hr );
4874                     ok( !lstrcmpW( str, szAttributeXML ), "incorrect attribute xml\n");
4875                     SysFreeString(str);
4876
4877                     hr = IXMLDOMAttribute_get_dataType(pAttribute, &v);
4878                     ok(hr == S_FALSE, "ret %08x\n", hr );
4879                     ok( V_VT(&v) == VT_NULL, "incorrect dataType type\n");
4880                     VariantClear(&v);
4881
4882                     IXMLDOMAttribute_Release(pAttribute);
4883
4884                     /* Check Element again with the Add Attribute*/
4885                     hr = IXMLDOMElement_get_xml(pElement, &str);
4886                     ok(hr == S_OK, "ret %08x\n", hr );
4887                     ok( !lstrcmpW( str, szElementXML2 ), "incorrect element xml\n");
4888                     SysFreeString(str);
4889                 }
4890
4891                 hr = IXMLDOMElement_put_text(pElement, _bstr_("TestingNode"));
4892                 ok(hr == S_OK, "ret %08x\n", hr );
4893
4894                 hr = IXMLDOMElement_get_xml(pElement, &str);
4895                 ok(hr == S_OK, "ret %08x\n", hr );
4896                 ok( !lstrcmpW( str, szElementXML3 ), "incorrect element xml\n");
4897                 SysFreeString(str);
4898
4899                 /* Test for reversible escaping */
4900                 str = SysAllocString( szStrangeChars );
4901                 hr = IXMLDOMElement_put_text(pElement, str);
4902                 ok(hr == S_OK, "ret %08x\n", hr );
4903                 SysFreeString( str );
4904
4905                 hr = IXMLDOMElement_get_xml(pElement, &str);
4906                 ok(hr == S_OK, "ret %08x\n", hr );
4907                 ok( !lstrcmpW( str, szElementXML4 ), "incorrect element xml\n");
4908                 SysFreeString(str);
4909
4910                 hr = IXMLDOMElement_get_text(pElement, &str);
4911                 ok(hr == S_OK, "ret %08x\n", hr );
4912                 ok( !lstrcmpW( str, szStrangeChars ), "incorrect element text\n");
4913                 SysFreeString(str);
4914
4915                 IXMLDOMElement_Release(pElement);
4916             }
4917
4918             /* CData Section */
4919             str = SysAllocString(szCData);
4920             hr = IXMLDOMDocument_createCDATASection(doc, str, NULL);
4921             ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4922
4923             hr = IXMLDOMDocument_createCDATASection(doc, str, &pCDataSec);
4924             SysFreeString(str);
4925             ok(hr == S_OK, "ret %08x\n", hr );
4926             if(hr == S_OK)
4927             {
4928                 IXMLDOMNode *pNextChild = (IXMLDOMNode *)0x1;
4929                 VARIANT var;
4930
4931                 VariantInit(&var);
4932
4933                 hr = IXMLDOMCDATASection_QueryInterface(pCDataSec, &IID_IXMLDOMElement, (void**)&pElement);
4934                 ok(hr == E_NOINTERFACE, "ret %08x\n", hr);
4935
4936                 hr = IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pCDataSec, NULL);
4937                 ok(hr == S_OK, "ret %08x\n", hr );
4938
4939                 /* get Attribute Tests */
4940                 hr = IXMLDOMCDATASection_get_attributes(pCDataSec, NULL);
4941                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4942
4943                 pAttribs = (IXMLDOMNamedNodeMap*)0x1;
4944                 hr = IXMLDOMCDATASection_get_attributes(pCDataSec, &pAttribs);
4945                 ok(hr == S_FALSE, "ret %08x\n", hr );
4946                 ok(pAttribs == NULL, "pAttribs != NULL\n");
4947
4948                 hr = IXMLDOMCDATASection_get_nodeName(pCDataSec, &str);
4949                 ok(hr == S_OK, "ret %08x\n", hr );
4950                 ok( !lstrcmpW( str, szCDataNodeText ), "incorrect cdata node Name\n");
4951                 SysFreeString(str);
4952
4953                 hr = IXMLDOMCDATASection_get_xml(pCDataSec, &str);
4954                 ok(hr == S_OK, "ret %08x\n", hr );
4955                 ok( !lstrcmpW( str, szCDataXML ), "incorrect cdata xml\n");
4956                 SysFreeString(str);
4957
4958                 /* test lastChild */
4959                 pNextChild = (IXMLDOMNode*)0x1;
4960                 hr = IXMLDOMCDATASection_get_lastChild(pCDataSec, &pNextChild);
4961                 ok(hr == S_FALSE, "ret %08x\n", hr );
4962                 ok(pNextChild == NULL, "pNextChild not NULL\n");
4963
4964                 /* test get_dataType */
4965                 hr = IXMLDOMCDATASection_get_dataType(pCDataSec, NULL);
4966                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4967
4968                 hr = IXMLDOMCDATASection_get_dataType(pCDataSec, &v);
4969                 ok(hr == S_FALSE, "ret %08x\n", hr );
4970                 ok( V_VT(&v) == VT_NULL, "incorrect dataType type\n");
4971                 VariantClear(&v);
4972
4973                 /* test nodeTypeString */
4974                 hr = IXMLDOMCDATASection_get_nodeTypeString(pCDataSec, &str);
4975                 ok(hr == S_OK, "ret %08x\n", hr );
4976                 ok( !lstrcmpW( str, _bstr_("cdatasection") ), "incorrect nodeTypeString string\n");
4977                 SysFreeString(str);
4978
4979                 /* put data Tests */
4980                 hr = IXMLDOMCDATASection_put_data(pCDataSec, _bstr_("This &is a ; test <>\\"));
4981                 ok(hr == S_OK, "ret %08x\n", hr );
4982
4983                 /* Confirm XML text is good */
4984                 hr = IXMLDOMCDATASection_get_xml(pCDataSec, &str);
4985                 ok(hr == S_OK, "ret %08x\n", hr );
4986                 ok( !lstrcmpW( str, _bstr_("<![CDATA[This &is a ; test <>\\]]>") ), "incorrect xml string\n");
4987                 SysFreeString(str);
4988
4989                 /* Confirm we get the put_data Text back */
4990                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
4991                 ok(hr == S_OK, "ret %08x\n", hr );
4992                 ok( !lstrcmpW( str, _bstr_("This &is a ; test <>\\") ), "incorrect text string\n");
4993                 SysFreeString(str);
4994
4995                 /* test length property */
4996                 hr = IXMLDOMCDATASection_get_length(pCDataSec, &len);
4997                 ok(hr == S_OK, "ret %08x\n", hr );
4998                 ok(len == 21, "expected 21 got %d\n", len);
4999
5000                 /* test get nodeValue */
5001                 hr = IXMLDOMCDATASection_get_nodeValue(pCDataSec, &var);
5002                 ok(hr == S_OK, "ret %08x\n", hr );
5003                 ok(V_VT(&var) == VT_BSTR, "got vt %04x\n", V_VT(&var));
5004                 ok( !lstrcmpW( V_BSTR(&var), _bstr_("This &is a ; test <>\\") ), "incorrect text string\n");
5005                 VariantClear(&var);
5006
5007                 /* test get data */
5008                 hr = IXMLDOMCDATASection_get_data(pCDataSec, &str);
5009                 ok(hr == S_OK, "ret %08x\n", hr );
5010                 ok( !lstrcmpW( str, _bstr_("This &is a ; test <>\\") ), "incorrect text string\n");
5011                 SysFreeString(str);
5012
5013                 /* test substringData */
5014                 hr = IXMLDOMCDATASection_substringData(pCDataSec, 0, 4, NULL);
5015                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5016
5017                 /* test substringData - Invalid offset */
5018                 str = (BSTR)&szElement;
5019                 hr = IXMLDOMCDATASection_substringData(pCDataSec, -1, 4, &str);
5020                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5021                 ok( str == NULL, "incorrect string\n");
5022
5023                 /* test substringData - Invalid offset */
5024                 str = (BSTR)&szElement;
5025                 hr = IXMLDOMCDATASection_substringData(pCDataSec, 30, 0, &str);
5026                 ok(hr == S_FALSE, "ret %08x\n", hr );
5027                 ok( str == NULL, "incorrect string\n");
5028
5029                 /* test substringData - Invalid size */
5030                 str = (BSTR)&szElement;
5031                 hr = IXMLDOMCDATASection_substringData(pCDataSec, 0, -1, &str);
5032                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5033                 ok( str == NULL, "incorrect string\n");
5034
5035                 /* test substringData - Invalid size */
5036                 str = (BSTR)&szElement;
5037                 hr = IXMLDOMCDATASection_substringData(pCDataSec, 2, 0, &str);
5038                 ok(hr == S_FALSE, "ret %08x\n", hr );
5039                 ok( str == NULL, "incorrect string\n");
5040
5041                 /* test substringData - Start of string */
5042                 hr = IXMLDOMCDATASection_substringData(pCDataSec, 0, 4, &str);
5043                 ok(hr == S_OK, "ret %08x\n", hr );
5044                 ok( !lstrcmpW( str, _bstr_("This") ), "incorrect substringData string\n");
5045                 SysFreeString(str);
5046
5047                 /* test substringData - Middle of string */
5048                 hr = IXMLDOMCDATASection_substringData(pCDataSec, 13, 4, &str);
5049                 ok(hr == S_OK, "ret %08x\n", hr );
5050                 ok( !lstrcmpW( str, _bstr_("test") ), "incorrect substringData string\n");
5051                 SysFreeString(str);
5052
5053                 /* test substringData - End of string */
5054                 hr = IXMLDOMCDATASection_substringData(pCDataSec, 20, 4, &str);
5055                 ok(hr == S_OK, "ret %08x\n", hr );
5056                 ok( !lstrcmpW( str, _bstr_("\\") ), "incorrect substringData string\n");
5057                 SysFreeString(str);
5058
5059                 /* test appendData */
5060                 hr = IXMLDOMCDATASection_appendData(pCDataSec, NULL);
5061                 ok(hr == S_OK, "ret %08x\n", hr );
5062
5063                 hr = IXMLDOMCDATASection_appendData(pCDataSec, _bstr_(""));
5064                 ok(hr == S_OK, "ret %08x\n", hr );
5065
5066                 hr = IXMLDOMCDATASection_appendData(pCDataSec, _bstr_("Append"));
5067                 ok(hr == S_OK, "ret %08x\n", hr );
5068
5069                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
5070                 ok(hr == S_OK, "ret %08x\n", hr );
5071                 ok( !lstrcmpW( str, _bstr_("This &is a ; test <>\\Append") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
5072                 SysFreeString(str);
5073
5074                 /* test insertData */
5075                 str = SysAllocStringLen(NULL, 0);
5076                 hr = IXMLDOMCDATASection_insertData(pCDataSec, -1, str);
5077                 ok(hr == S_OK, "ret %08x\n", hr );
5078
5079                 hr = IXMLDOMCDATASection_insertData(pCDataSec, -1, NULL);
5080                 ok(hr == S_OK, "ret %08x\n", hr );
5081
5082                 hr = IXMLDOMCDATASection_insertData(pCDataSec, 1000, str);
5083                 ok(hr == S_OK, "ret %08x\n", hr );
5084
5085                 hr = IXMLDOMCDATASection_insertData(pCDataSec, 1000, NULL);
5086                 ok(hr == S_OK, "ret %08x\n", hr );
5087
5088                 hr = IXMLDOMCDATASection_insertData(pCDataSec, 0, NULL);
5089                 ok(hr == S_OK, "ret %08x\n", hr );
5090
5091                 hr = IXMLDOMCDATASection_insertData(pCDataSec, 0, str);
5092                 ok(hr == S_OK, "ret %08x\n", hr );
5093                 SysFreeString(str);
5094
5095                 hr = IXMLDOMCDATASection_insertData(pCDataSec, -1, _bstr_("Inserting"));
5096                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5097
5098                 hr = IXMLDOMCDATASection_insertData(pCDataSec, 1000, _bstr_("Inserting"));
5099                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5100
5101                 hr = IXMLDOMCDATASection_insertData(pCDataSec, 0, _bstr_("Begin "));
5102                 ok(hr == S_OK, "ret %08x\n", hr );
5103
5104                 hr = IXMLDOMCDATASection_insertData(pCDataSec, 17, _bstr_("Middle"));
5105                 ok(hr == S_OK, "ret %08x\n", hr );
5106
5107                 hr = IXMLDOMCDATASection_insertData(pCDataSec, 39, _bstr_(" End"));
5108                 ok(hr == S_OK, "ret %08x\n", hr );
5109
5110                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
5111                 ok(hr == S_OK, "ret %08x\n", hr );
5112                 ok( !lstrcmpW( str, _bstr_("Begin This &is a Middle; test <>\\Append End") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
5113                 SysFreeString(str);
5114
5115                 /* delete data */
5116                 /* invalid arguments */
5117                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, -1, 1);
5118                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5119
5120                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, 0, 0);
5121                 ok(hr == S_OK, "ret %08x\n", hr );
5122
5123                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, 0, -1);
5124                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5125
5126                 hr = IXMLDOMCDATASection_get_length(pCDataSec, &len);
5127                 ok(hr == S_OK, "ret %08x\n", hr );
5128                 ok(len == 43, "expected 43 got %d\n", len);
5129
5130                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, len, 1);
5131                 ok(hr == S_OK, "ret %08x\n", hr );
5132
5133                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, len+1, 1);
5134                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5135
5136                 /* delete from start */
5137                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, 0, 5);
5138                 ok(hr == S_OK, "ret %08x\n", hr );
5139
5140                 hr = IXMLDOMCDATASection_get_length(pCDataSec, &len);
5141                 ok(hr == S_OK, "ret %08x\n", hr );
5142                 ok(len == 38, "expected 38 got %d\n", len);
5143
5144                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
5145                 ok(hr == S_OK, "ret %08x\n", hr );
5146                 ok( !lstrcmpW( str, _bstr_(" This &is a Middle; test <>\\Append End") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
5147                 SysFreeString(str);
5148
5149                 /* delete from end */
5150                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, 35, 3);
5151                 ok(hr == S_OK, "ret %08x\n", hr );
5152
5153                 hr = IXMLDOMCDATASection_get_length(pCDataSec, &len);
5154                 ok(hr == S_OK, "ret %08x\n", hr );
5155                 ok(len == 35, "expected 35 got %d\n", len);
5156
5157                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
5158                 ok(hr == S_OK, "ret %08x\n", hr );
5159                 ok( !lstrcmpW( str, _bstr_(" This &is a Middle; test <>\\Append ") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
5160                 SysFreeString(str);
5161
5162                 /* delete from inside */
5163                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, 1, 33);
5164                 ok(hr == S_OK, "ret %08x\n", hr );
5165
5166                 hr = IXMLDOMCDATASection_get_length(pCDataSec, &len);
5167                 ok(hr == S_OK, "ret %08x\n", hr );
5168                 ok(len == 2, "expected 2 got %d\n", len);
5169
5170                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
5171                 ok(hr == S_OK, "ret %08x\n", hr );
5172                 ok( !lstrcmpW( str, _bstr_("  ") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
5173                 SysFreeString(str);
5174
5175                 /* delete whole data ... */
5176                 hr = IXMLDOMCDATASection_get_length(pCDataSec, &len);
5177                 ok(hr == S_OK, "ret %08x\n", hr );
5178
5179                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, 0, len);
5180                 ok(hr == S_OK, "ret %08x\n", hr );
5181
5182                 /* ... and try again with empty string */
5183                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, 0, len);
5184                 ok(hr == S_OK, "ret %08x\n", hr );
5185
5186                 /* ::replaceData() */
5187                 V_VT(&v) = VT_BSTR;
5188                 V_BSTR(&v) = SysAllocString(szstr1);
5189                 hr = IXMLDOMCDATASection_put_nodeValue(pCDataSec, v);
5190                 ok(hr == S_OK, "ret %08x\n", hr );
5191                 VariantClear(&v);
5192
5193                 hr = IXMLDOMCDATASection_replaceData(pCDataSec, 6, 0, NULL);
5194                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5195                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
5196                 ok(hr == S_OK, "ret %08x\n", hr );
5197                 ok( !lstrcmpW( str, _bstr_("str1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
5198                 SysFreeString(str);
5199
5200                 hr = IXMLDOMCDATASection_replaceData(pCDataSec, 0, 0, NULL);
5201                 ok(hr == S_OK, "ret %08x\n", hr );
5202                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
5203                 ok(hr == S_OK, "ret %08x\n", hr );
5204                 ok( !lstrcmpW( str, _bstr_("str1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
5205                 SysFreeString(str);
5206
5207                 /* NULL pointer means delete */
5208                 hr = IXMLDOMCDATASection_replaceData(pCDataSec, 0, 1, NULL);
5209                 ok(hr == S_OK, "ret %08x\n", hr );
5210                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
5211                 ok(hr == S_OK, "ret %08x\n", hr );
5212                 ok( !lstrcmpW( str, _bstr_("tr1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
5213                 SysFreeString(str);
5214
5215                 /* empty string means delete */
5216                 hr = IXMLDOMCDATASection_replaceData(pCDataSec, 0, 1, _bstr_(""));
5217                 ok(hr == S_OK, "ret %08x\n", hr );
5218                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
5219                 ok(hr == S_OK, "ret %08x\n", hr );
5220                 ok( !lstrcmpW( str, _bstr_("r1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
5221                 SysFreeString(str);
5222
5223                 /* zero count means insert */
5224                 hr = IXMLDOMCDATASection_replaceData(pCDataSec, 0, 0, _bstr_("a"));
5225                 ok(hr == S_OK, "ret %08x\n", hr );
5226                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
5227                 ok(hr == S_OK, "ret %08x\n", hr );
5228                 ok( !lstrcmpW( str, _bstr_("ar1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
5229                 SysFreeString(str);
5230
5231                 hr = IXMLDOMCDATASection_replaceData(pCDataSec, 0, 2, NULL);
5232                 ok(hr == S_OK, "ret %08x\n", hr );
5233
5234                 hr = IXMLDOMCDATASection_insertData(pCDataSec, 0, _bstr_("m"));
5235                 ok(hr == S_OK, "ret %08x\n", hr );
5236                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
5237                 ok(hr == S_OK, "ret %08x\n", hr );
5238                 ok( !lstrcmpW( str, _bstr_("m1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
5239                 SysFreeString(str);
5240
5241                 /* nonempty string, count greater than its length */
5242                 hr = IXMLDOMCDATASection_replaceData(pCDataSec, 0, 2, _bstr_("a1.2"));
5243                 ok(hr == S_OK, "ret %08x\n", hr );
5244                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
5245                 ok(hr == S_OK, "ret %08x\n", hr );
5246                 ok( !lstrcmpW( str, _bstr_("a1.2") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
5247                 SysFreeString(str);
5248
5249                 /* nonempty string, count less than its length */
5250                 hr = IXMLDOMCDATASection_replaceData(pCDataSec, 0, 1, _bstr_("wine"));
5251                 ok(hr == S_OK, "ret %08x\n", hr );
5252                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
5253                 ok(hr == S_OK, "ret %08x\n", hr );
5254                 ok( !lstrcmpW( str, _bstr_("wine1.2") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
5255                 SysFreeString(str);
5256
5257                 IXMLDOMCDATASection_Release(pCDataSec);
5258             }
5259
5260             /* Document Fragments */
5261             hr = IXMLDOMDocument_createDocumentFragment(doc, NULL);
5262             ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5263
5264             hr = IXMLDOMDocument_createDocumentFragment(doc, &pDocFrag);
5265             ok(hr == S_OK, "ret %08x\n", hr );
5266             if(hr == S_OK)
5267             {
5268                 IXMLDOMNode *node;
5269
5270                 hr = IXMLDOMDocumentFragment_get_parentNode(pDocFrag, NULL);
5271                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5272
5273                 node = (IXMLDOMNode *)0x1;
5274                 hr = IXMLDOMDocumentFragment_get_parentNode(pDocFrag, &node);
5275                 ok(hr == S_FALSE, "ret %08x\n", hr );
5276                 ok(node == NULL, "expected NULL, got %p\n", node);
5277
5278                 hr = IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pDocFrag, NULL);
5279                 ok(hr == S_OK, "ret %08x\n", hr );
5280
5281                 /* get Attribute Tests */
5282                 hr = IXMLDOMDocumentFragment_get_attributes(pDocFrag, NULL);
5283                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5284
5285                 pAttribs = (IXMLDOMNamedNodeMap*)0x1;
5286                 hr = IXMLDOMDocumentFragment_get_attributes(pDocFrag, &pAttribs);
5287                 ok(hr == S_FALSE, "ret %08x\n", hr );
5288                 ok(pAttribs == NULL, "pAttribs != NULL\n");
5289
5290                 hr = IXMLDOMDocumentFragment_get_nodeName(pDocFrag, &str);
5291                 ok(hr == S_OK, "ret %08x\n", hr );
5292                 ok( !lstrcmpW( str, szDocFragmentText ), "incorrect docfragment node Name\n");
5293                 SysFreeString(str);
5294
5295                 /* test next Sibling*/
5296                 hr = IXMLDOMDocumentFragment_get_nextSibling(pDocFrag, NULL);
5297                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5298
5299                 node = (IXMLDOMNode *)0x1;
5300                 hr = IXMLDOMDocumentFragment_get_nextSibling(pDocFrag, &node);
5301                 ok(hr == S_FALSE, "ret %08x\n", hr );
5302                 ok(node == NULL, "next sibling not NULL\n");
5303
5304                 /* test Previous Sibling*/
5305                 hr = IXMLDOMDocumentFragment_get_previousSibling(pDocFrag, NULL);
5306                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5307
5308                 node = (IXMLDOMNode *)0x1;
5309                 hr = IXMLDOMDocumentFragment_get_previousSibling(pDocFrag, &node);
5310                 ok(hr == S_FALSE, "ret %08x\n", hr );
5311                 ok(node == NULL, "previous sibling not NULL\n");
5312
5313                 /* test get_dataType */
5314                 hr = IXMLDOMDocumentFragment_get_dataType(pDocFrag, NULL);
5315                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5316
5317                 hr = IXMLDOMDocumentFragment_get_dataType(pDocFrag, &v);
5318                 ok(hr == S_FALSE, "ret %08x\n", hr );
5319                 ok( V_VT(&v) == VT_NULL, "incorrect dataType type\n");
5320                 VariantClear(&v);
5321
5322                 /* test nodeTypeString */
5323                 hr = IXMLDOMDocumentFragment_get_nodeTypeString(pDocFrag, &str);
5324                 ok(hr == S_OK, "ret %08x\n", hr );
5325                 ok( !lstrcmpW( str, _bstr_("documentfragment") ), "incorrect nodeTypeString string\n");
5326                 SysFreeString(str);
5327
5328                 IXMLDOMDocumentFragment_Release(pDocFrag);
5329             }
5330
5331             /* Entity References */
5332             hr = IXMLDOMDocument_createEntityReference(doc, NULL, &pEntityRef);
5333             ok(hr == E_FAIL, "ret %08x\n", hr );
5334             hr = IXMLDOMDocument_createEntityReference(doc, _bstr_(""), &pEntityRef);
5335             ok(hr == E_FAIL, "ret %08x\n", hr );
5336
5337             str = SysAllocString(szEntityRef);
5338             hr = IXMLDOMDocument_createEntityReference(doc, str, NULL);
5339             ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5340
5341             hr = IXMLDOMDocument_createEntityReference(doc, str, &pEntityRef);
5342             SysFreeString(str);
5343             ok(hr == S_OK, "ret %08x\n", hr );
5344             if(hr == S_OK)
5345             {
5346                 hr = IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pEntityRef, NULL);
5347                 ok(hr == S_OK, "ret %08x\n", hr );
5348
5349                 /* get Attribute Tests */
5350                 hr = IXMLDOMEntityReference_get_attributes(pEntityRef, NULL);
5351                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5352
5353                 pAttribs = (IXMLDOMNamedNodeMap*)0x1;
5354                 hr = IXMLDOMEntityReference_get_attributes(pEntityRef, &pAttribs);
5355                 ok(hr == S_FALSE, "ret %08x\n", hr );
5356                 ok(pAttribs == NULL, "pAttribs != NULL\n");
5357
5358                 /* test dataType */
5359                 hr = IXMLDOMEntityReference_get_dataType(pEntityRef, &v);
5360                 ok(hr == S_FALSE, "ret %08x\n", hr );
5361                 ok( V_VT(&v) == VT_NULL, "incorrect dataType type\n");
5362                 VariantClear(&v);
5363
5364                 /* test nodeTypeString */
5365                 hr = IXMLDOMEntityReference_get_nodeTypeString(pEntityRef, &str);
5366                 ok(hr == S_OK, "ret %08x\n", hr );
5367                 ok( !lstrcmpW( str, _bstr_("entityreference") ), "incorrect nodeTypeString string\n");
5368                 SysFreeString(str);
5369
5370                 /* test get_xml*/
5371                 hr = IXMLDOMEntityReference_get_xml(pEntityRef, &str);
5372                 ok(hr == S_OK, "ret %08x\n", hr );
5373                 ok( !lstrcmpW( str, szEntityRefXML ), "incorrect xml string\n");
5374                 SysFreeString(str);
5375
5376                 IXMLDOMEntityReference_Release(pEntityRef);
5377             }
5378
5379             IXMLDOMElement_Release( pRoot );
5380         }
5381     }
5382
5383     IXMLDOMDocument_Release(doc);
5384
5385     free_bstrs();
5386 }
5387
5388 static void test_nodeTypeTests( void )
5389 {
5390     IXMLDOMDocument *doc = NULL;
5391     IXMLDOMElement *pRoot;
5392     IXMLDOMElement *pElement;
5393     HRESULT hr;
5394
5395     doc = create_document(&IID_IXMLDOMDocument);
5396     if (!doc) return;
5397
5398     hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing"), NULL);
5399     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5400
5401     hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing"), &pRoot);
5402     ok(hr == S_OK, "ret %08x\n", hr );
5403     if(hr == S_OK)
5404     {
5405         hr = IXMLDOMDocument_appendChild(doc, (IXMLDOMNode*)pRoot, NULL);
5406         ok(hr == S_OK, "ret %08x\n", hr );
5407         if(hr == S_OK)
5408         {
5409             hr = IXMLDOMElement_put_dataType(pRoot, NULL);
5410             ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5411
5412             /* Invalid Value */
5413             hr = IXMLDOMElement_put_dataType(pRoot, _bstr_("abcdefg") );
5414             ok(hr == E_FAIL, "ret %08x\n", hr );
5415
5416             /* NOTE:
5417              *   The name passed into put_dataType is case-insensitive. So many of the names
5418              *     have been changed to reflect this.
5419              */
5420             /* Boolean */
5421             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_Boolean"), &pElement);
5422             ok(hr == S_OK, "ret %08x\n", hr );
5423             if(hr == S_OK)
5424             {
5425                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5426
5427                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("Boolean") );
5428                 ok(hr == S_OK, "ret %08x\n", hr );
5429
5430                 IXMLDOMElement_Release(pElement);
5431             }
5432
5433             /* String */
5434             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_String"), &pElement);
5435             ok(hr == S_OK, "ret %08x\n", hr );
5436             if(hr == S_OK)
5437             {
5438                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5439
5440                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("String") );
5441                 ok(hr == S_OK, "ret %08x\n", hr );
5442
5443                 IXMLDOMElement_Release(pElement);
5444             }
5445
5446             /* Number */
5447             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_Number"), &pElement);
5448             ok(hr == S_OK, "ret %08x\n", hr );
5449             if(hr == S_OK)
5450             {
5451                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5452
5453                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("number") );
5454                 ok(hr == S_OK, "ret %08x\n", hr );
5455
5456                 IXMLDOMElement_Release(pElement);
5457             }
5458
5459             /* Int */
5460             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_Int"), &pElement);
5461             ok(hr == S_OK, "ret %08x\n", hr );
5462             if(hr == S_OK)
5463             {
5464                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5465
5466                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("InT") );
5467                 ok(hr == S_OK, "ret %08x\n", hr );
5468
5469                 IXMLDOMElement_Release(pElement);
5470             }
5471
5472             /* Fixed */
5473             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_Fixed"), &pElement);
5474             ok(hr == S_OK, "ret %08x\n", hr );
5475             if(hr == S_OK)
5476             {
5477                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5478
5479                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("fixed.14.4") );
5480                 ok(hr == S_OK, "ret %08x\n", hr );
5481
5482                 IXMLDOMElement_Release(pElement);
5483             }
5484
5485             /* DateTime */
5486             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_DateTime"), &pElement);
5487             ok(hr == S_OK, "ret %08x\n", hr );
5488             if(hr == S_OK)
5489             {
5490                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5491
5492                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("DateTime") );
5493                 ok(hr == S_OK, "ret %08x\n", hr );
5494
5495                 IXMLDOMElement_Release(pElement);
5496             }
5497
5498             /* DateTime TZ */
5499             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_DateTime_tz"), &pElement);
5500             ok(hr == S_OK, "ret %08x\n", hr );
5501             if(hr == S_OK)
5502             {
5503                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5504
5505                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("DateTime.tz") );
5506                 ok(hr == S_OK, "ret %08x\n", hr );
5507
5508                 IXMLDOMElement_Release(pElement);
5509             }
5510
5511             /* Date */
5512             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_Date"), &pElement);
5513             ok(hr == S_OK, "ret %08x\n", hr );
5514             if(hr == S_OK)
5515             {
5516                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5517
5518                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("Date") );
5519                 ok(hr == S_OK, "ret %08x\n", hr );
5520
5521                 IXMLDOMElement_Release(pElement);
5522             }
5523
5524             /* Time */
5525             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_Time"), &pElement);
5526             ok(hr == S_OK, "ret %08x\n", hr );
5527             if(hr == S_OK)
5528             {
5529                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5530
5531                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("Time") );
5532                 ok(hr == S_OK, "ret %08x\n", hr );
5533
5534                 IXMLDOMElement_Release(pElement);
5535             }
5536
5537             /* Time.tz */
5538             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_Time_TZ"), &pElement);
5539             ok(hr == S_OK, "ret %08x\n", hr );
5540             if(hr == S_OK)
5541             {
5542                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5543
5544                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("Time.tz") );
5545                 ok(hr == S_OK, "ret %08x\n", hr );
5546
5547                 IXMLDOMElement_Release(pElement);
5548             }
5549
5550             /* I1 */
5551             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_I1"), &pElement);
5552             ok(hr == S_OK, "ret %08x\n", hr );
5553             if(hr == S_OK)
5554             {
5555                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5556
5557                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("I1") );
5558                 ok(hr == S_OK, "ret %08x\n", hr );
5559
5560                 IXMLDOMElement_Release(pElement);
5561             }
5562
5563             /* I2 */
5564             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_I2"), &pElement);
5565             ok(hr == S_OK, "ret %08x\n", hr );
5566             if(hr == S_OK)
5567             {
5568                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5569
5570                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("I2") );
5571                 ok(hr == S_OK, "ret %08x\n", hr );
5572
5573                 IXMLDOMElement_Release(pElement);
5574             }
5575
5576             /* I4 */
5577             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_I4"), &pElement);
5578             ok(hr == S_OK, "ret %08x\n", hr );
5579             if(hr == S_OK)
5580             {
5581                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5582
5583                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("I4") );
5584                 ok(hr == S_OK, "ret %08x\n", hr );
5585
5586                 IXMLDOMElement_Release(pElement);
5587             }
5588
5589             /* UI1 */
5590             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_UI1"), &pElement);
5591             ok(hr == S_OK, "ret %08x\n", hr );
5592             if(hr == S_OK)
5593             {
5594                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5595
5596                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("UI1") );
5597                 ok(hr == S_OK, "ret %08x\n", hr );
5598
5599                 IXMLDOMElement_Release(pElement);
5600             }
5601
5602             /* UI2 */
5603             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_UI2"), &pElement);
5604             ok(hr == S_OK, "ret %08x\n", hr );
5605             if(hr == S_OK)
5606             {
5607                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5608
5609                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("UI2") );
5610                 ok(hr == S_OK, "ret %08x\n", hr );
5611
5612                 IXMLDOMElement_Release(pElement);
5613             }
5614
5615             /* UI4 */
5616             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_UI4"), &pElement);
5617             ok(hr == S_OK, "ret %08x\n", hr );
5618             if(hr == S_OK)
5619             {
5620                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5621
5622                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("UI4") );
5623                 ok(hr == S_OK, "ret %08x\n", hr );
5624
5625                 IXMLDOMElement_Release(pElement);
5626             }
5627
5628             /* r4 */
5629             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_r4"), &pElement);
5630             ok(hr == S_OK, "ret %08x\n", hr );
5631             if(hr == S_OK)
5632             {
5633                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5634
5635                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("r4") );
5636                 ok(hr == S_OK, "ret %08x\n", hr );
5637
5638                 IXMLDOMElement_Release(pElement);
5639             }
5640
5641             /* r8 */
5642             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_r8"), &pElement);
5643             ok(hr == S_OK, "ret %08x\n", hr );
5644             if(hr == S_OK)
5645             {
5646                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5647
5648                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("r8") );
5649                 ok(hr == S_OK, "ret %08x\n", hr );
5650
5651                 IXMLDOMElement_Release(pElement);
5652             }
5653
5654             /* float */
5655             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_float"), &pElement);
5656             ok(hr == S_OK, "ret %08x\n", hr );
5657             if(hr == S_OK)
5658             {
5659                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5660
5661                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("float") );
5662                 ok(hr == S_OK, "ret %08x\n", hr );
5663
5664                 IXMLDOMElement_Release(pElement);
5665             }
5666
5667             /* uuid */
5668             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_uuid"), &pElement);
5669             ok(hr == S_OK, "ret %08x\n", hr );
5670             if(hr == S_OK)
5671             {
5672                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5673
5674                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("UuId") );
5675                 ok(hr == S_OK, "ret %08x\n", hr );
5676
5677                 IXMLDOMElement_Release(pElement);
5678             }
5679
5680             /* bin.hex */
5681             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_bin_hex"), &pElement);
5682             ok(hr == S_OK, "ret %08x\n", hr );
5683             if(hr == S_OK)
5684             {
5685                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5686
5687                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("bin.hex") );
5688                 ok(hr == S_OK, "ret %08x\n", hr );
5689
5690                 IXMLDOMElement_Release(pElement);
5691             }
5692
5693             /* bin.base64 */
5694             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_bin_base64"), &pElement);
5695             ok(hr == S_OK, "ret %08x\n", hr );
5696             if(hr == S_OK)
5697             {
5698                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5699
5700                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("bin.base64") );
5701                 ok(hr == S_OK, "ret %08x\n", hr );
5702
5703                 IXMLDOMElement_Release(pElement);
5704             }
5705
5706             /* Check changing types */
5707             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_Change"), &pElement);
5708             ok(hr == S_OK, "ret %08x\n", hr );
5709             if(hr == S_OK)
5710             {
5711                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5712
5713                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("DateTime.tz") );
5714                 ok(hr == S_OK, "ret %08x\n", hr );
5715
5716                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("string") );
5717                 ok(hr == S_OK, "ret %08x\n", hr );
5718
5719                 IXMLDOMElement_Release(pElement);
5720             }
5721
5722             IXMLDOMElement_Release(pRoot);
5723         }
5724     }
5725
5726     IXMLDOMDocument_Release(doc);
5727
5728     free_bstrs();
5729 }
5730
5731 static void test_DocumentSaveToDocument(void)
5732 {
5733     IXMLDOMDocument *doc, *doc2;
5734     IXMLDOMElement *pRoot;
5735     HRESULT hr;
5736
5737     doc = create_document(&IID_IXMLDOMDocument);
5738     if (!doc) return;
5739
5740     doc2 = create_document(&IID_IXMLDOMDocument);
5741     if (!doc2)
5742     {
5743         IXMLDOMDocument_Release(doc);
5744         return;
5745     }
5746
5747     hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing"), &pRoot);
5748     ok(hr == S_OK, "ret %08x\n", hr );
5749     if(hr == S_OK)
5750     {
5751         hr = IXMLDOMDocument_appendChild(doc, (IXMLDOMNode*)pRoot, NULL);
5752         ok(hr == S_OK, "ret %08x\n", hr );
5753         if(hr == S_OK)
5754         {
5755             VARIANT vDoc;
5756             BSTR sOrig;
5757             BSTR sNew;
5758
5759             V_VT(&vDoc) = VT_UNKNOWN;
5760             V_UNKNOWN(&vDoc) = (IUnknown*)doc2;
5761
5762             hr = IXMLDOMDocument_save(doc, vDoc);
5763             ok(hr == S_OK, "ret %08x\n", hr );
5764
5765             hr = IXMLDOMDocument_get_xml(doc, &sOrig);
5766             ok(hr == S_OK, "ret %08x\n", hr );
5767
5768             hr = IXMLDOMDocument_get_xml(doc2, &sNew);
5769             ok(hr == S_OK, "ret %08x\n", hr );
5770
5771             ok( !lstrcmpW( sOrig, sNew ), "New document is not the same as origial\n");
5772
5773             SysFreeString(sOrig);
5774             SysFreeString(sNew);
5775         }
5776         IXMLDOMElement_Release(pRoot);
5777     }
5778
5779     IXMLDOMDocument_Release(doc2);
5780     IXMLDOMDocument_Release(doc);
5781 }
5782
5783 static void test_DocumentSaveToFile(void)
5784 {
5785     IXMLDOMDocument *doc;
5786     IXMLDOMElement *pRoot;
5787     HANDLE file;
5788     char buffer[100];
5789     DWORD read = 0;
5790     HRESULT hr;
5791
5792     doc = create_document(&IID_IXMLDOMDocument);
5793     if (!doc) return;
5794
5795     hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing"), &pRoot);
5796     ok(hr == S_OK, "ret %08x\n", hr );
5797     if(hr == S_OK)
5798     {
5799         hr = IXMLDOMDocument_appendChild(doc, (IXMLDOMNode*)pRoot, NULL);
5800         ok(hr == S_OK, "ret %08x\n", hr );
5801         if(hr == S_OK)
5802         {
5803             VARIANT vFile;
5804
5805             V_VT(&vFile) = VT_BSTR;
5806             V_BSTR(&vFile) = _bstr_("test.xml");
5807
5808             hr = IXMLDOMDocument_save(doc, vFile);
5809             ok(hr == S_OK, "ret %08x\n", hr );
5810         }
5811     }
5812
5813     IXMLDOMElement_Release(pRoot);
5814     IXMLDOMDocument_Release(doc);
5815
5816     file = CreateFile("test.xml", GENERIC_READ, 0, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL );
5817     ok(file != INVALID_HANDLE_VALUE, "Could not open file: %u\n", GetLastError());
5818     if(file == INVALID_HANDLE_VALUE)
5819         return;
5820
5821     ReadFile(file, buffer, sizeof(buffer), &read, NULL);
5822     ok(read != 0, "could not read file\n");
5823     ok(buffer[0] != '<' || buffer[1] != '?', "File contains processing instruction\n");
5824
5825     CloseHandle(file);
5826     DeleteFile("test.xml");
5827 }
5828
5829 static void test_testTransforms(void)
5830 {
5831     IXMLDOMDocument *doc, *docSS;
5832     IXMLDOMNode *pNode;
5833     VARIANT_BOOL bSucc;
5834
5835     HRESULT hr;
5836
5837     doc = create_document(&IID_IXMLDOMDocument);
5838     if (!doc) return;
5839
5840     docSS = create_document(&IID_IXMLDOMDocument);
5841     if (!docSS)
5842     {
5843         IXMLDOMDocument_Release(doc);
5844         return;
5845     }
5846
5847     hr = IXMLDOMDocument_loadXML(doc, _bstr_(szTransformXML), &bSucc);
5848     ok(hr == S_OK, "ret %08x\n", hr );
5849
5850     hr = IXMLDOMDocument_loadXML(docSS, _bstr_(szTransformSSXML), &bSucc);
5851     ok(hr == S_OK, "ret %08x\n", hr );
5852
5853     hr = IXMLDOMDocument_QueryInterface(docSS, &IID_IXMLDOMNode, (void**)&pNode );
5854     ok(hr == S_OK, "ret %08x\n", hr );
5855     if(hr == S_OK)
5856     {
5857         BSTR bOut;
5858
5859         hr = IXMLDOMDocument_transformNode(doc, pNode, &bOut);
5860         ok(hr == S_OK, "ret %08x\n", hr );
5861         ok( compareIgnoreReturns( bOut, _bstr_(szTransformOutput)), "Stylesheet output not correct\n");
5862         SysFreeString(bOut);
5863
5864         IXMLDOMNode_Release(pNode);
5865     }
5866
5867     IXMLDOMDocument_Release(docSS);
5868     IXMLDOMDocument_Release(doc);
5869
5870     free_bstrs();
5871 }
5872
5873 static void test_Namespaces(void)
5874 {
5875     IXMLDOMDocument *doc;
5876     IXMLDOMNode *pNode;
5877     IXMLDOMNode *pNode2 = NULL;
5878     VARIANT_BOOL bSucc;
5879     HRESULT hr;
5880     BSTR str;
5881     static  const CHAR szNamespacesXML[] =
5882 "<?xml version=\"1.0\"?>\n"
5883 "<root xmlns:WEB='http://www.winehq.org'>\n"
5884 "<WEB:Site version=\"1.0\" />\n"
5885 "</root>";
5886
5887     doc = create_document(&IID_IXMLDOMDocument);
5888     if (!doc) return;
5889
5890     hr = IXMLDOMDocument_loadXML(doc, _bstr_(szNamespacesXML), &bSucc);
5891     ok(hr == S_OK, "ret %08x\n", hr );
5892     ok(bSucc == VARIANT_TRUE, "Expected VARIANT_TRUE got VARIANT_FALSE\n");
5893
5894     hr = IXMLDOMDocument_selectSingleNode(doc, _bstr_("root"), &pNode );
5895     ok(hr == S_OK, "ret %08x\n", hr );
5896     if(hr == S_OK)
5897     {
5898         hr = IXMLDOMNode_get_firstChild( pNode, &pNode2 );
5899         ok( hr == S_OK, "ret %08x\n", hr );
5900         ok( pNode2 != NULL, "pNode2 == NULL\n");
5901
5902         /* Test get_prefix */
5903         hr = IXMLDOMNode_get_prefix(pNode2, NULL);
5904         ok( hr == E_INVALIDARG, "ret %08x\n", hr );
5905         /* NOTE: Need to test that arg2 gets cleared on Error. */
5906
5907         hr = IXMLDOMNode_get_prefix(pNode2, &str);
5908         ok( hr == S_OK, "ret %08x\n", hr );
5909         ok( !lstrcmpW( str, _bstr_("WEB")), "incorrect prefix string\n");
5910         SysFreeString(str);
5911
5912         /* Test get_namespaceURI */
5913         hr = IXMLDOMNode_get_namespaceURI(pNode2, NULL);
5914         ok( hr == E_INVALIDARG, "ret %08x\n", hr );
5915         /* NOTE: Need to test that arg2 gets cleared on Error. */
5916
5917         hr = IXMLDOMNode_get_namespaceURI(pNode2, &str);
5918         ok( hr == S_OK, "ret %08x\n", hr );
5919         ok( !lstrcmpW( str, _bstr_("http://www.winehq.org")), "incorrect namespaceURI string\n");
5920         SysFreeString(str);
5921
5922         IXMLDOMNode_Release(pNode2);
5923         IXMLDOMNode_Release(pNode);
5924     }
5925
5926     IXMLDOMDocument_Release(doc);
5927
5928     free_bstrs();
5929 }
5930
5931 static void test_FormattingXML(void)
5932 {
5933     IXMLDOMDocument *doc;
5934     IXMLDOMElement *pElement;
5935     VARIANT_BOOL bSucc;
5936     HRESULT hr;
5937     BSTR str;
5938     static const CHAR szLinefeedXML[] = "<?xml version=\"1.0\"?>\n<Root>\n\t<Sub val=\"A\" />\n</Root>";
5939     static const CHAR szLinefeedRootXML[] = "<Root>\r\n\t<Sub val=\"A\"/>\r\n</Root>";
5940
5941     doc = create_document(&IID_IXMLDOMDocument);
5942     if (!doc) return;
5943
5944     hr = IXMLDOMDocument_loadXML(doc, _bstr_(szLinefeedXML), &bSucc);
5945     ok(hr == S_OK, "ret %08x\n", hr );
5946     ok(bSucc == VARIANT_TRUE, "Expected VARIANT_TRUE got VARIANT_FALSE\n");
5947
5948     if(bSucc == VARIANT_TRUE)
5949     {
5950         hr = IXMLDOMDocument_get_documentElement(doc, &pElement);
5951         ok(hr == S_OK, "ret %08x\n", hr );
5952         if(hr == S_OK)
5953         {
5954             hr = IXMLDOMElement_get_xml(pElement, &str);
5955             ok(hr == S_OK, "ret %08x\n", hr );
5956             ok( !lstrcmpW( str, _bstr_(szLinefeedRootXML) ), "incorrect element xml\n");
5957             SysFreeString(str);
5958
5959             IXMLDOMElement_Release(pElement);
5960         }
5961     }
5962
5963     IXMLDOMDocument_Release(doc);
5964
5965     free_bstrs();
5966 }
5967
5968 typedef struct _nodetypedvalue_t {
5969     const char *name;
5970     VARTYPE type;
5971     const char *value; /* value in string format */
5972 } nodetypedvalue_t;
5973
5974 static const nodetypedvalue_t get_nodetypedvalue[] = {
5975     { "root/string",    VT_BSTR, "Wine" },
5976     { "root/string2",   VT_BSTR, "String" },
5977     { "root/number",    VT_BSTR, "12.44" },
5978     { "root/number2",   VT_BSTR, "-3.71e3" },
5979     { "root/int",       VT_I4,   "-13" },
5980     { "root/fixed",     VT_CY,   "7322.9371" },
5981     { "root/bool",      VT_BOOL, "-1" },
5982     { "root/datetime",  VT_DATE, "40135.14" },
5983     { "root/datetimetz",VT_DATE, "37813.59" },
5984     { "root/date",      VT_DATE, "665413" },
5985     { "root/time",      VT_DATE, "0.5813889" },
5986     { "root/timetz",    VT_DATE, "1.112512" },
5987     { "root/i1",        VT_I1,   "-13" },
5988     { "root/i2",        VT_I2,   "31915" },
5989     { "root/i4",        VT_I4,   "-312232" },
5990     { "root/ui1",       VT_UI1,  "123" },
5991     { "root/ui2",       VT_UI2,  "48282" },
5992     { "root/ui4",       VT_UI4,  "949281" },
5993     { "root/r4",        VT_R4,   "213124" },
5994     { "root/r8",        VT_R8,   "0.412" },
5995     { "root/float",     VT_R8,   "41221.421" },
5996     { "root/uuid",      VT_BSTR, "333C7BC4-460F-11D0-BC04-0080C7055a83" },
5997     { 0 }
5998 };
5999
6000 static void test_nodeTypedValue(void)
6001 {
6002     const nodetypedvalue_t *entry = get_nodetypedvalue;
6003     IXMLDOMDocument *doc;
6004     IXMLDOMNode *node;
6005     VARIANT_BOOL b;
6006     VARIANT value;
6007     HRESULT hr;
6008
6009     doc = create_document(&IID_IXMLDOMDocument);
6010     if (!doc) return;
6011
6012     b = VARIANT_FALSE;
6013     hr = IXMLDOMDocument_loadXML(doc, _bstr_(szTypeValueXML), &b);
6014     ok(hr == S_OK, "ret %08x\n", hr );
6015     ok(b == VARIANT_TRUE, "got %d\n", b);
6016
6017     hr = IXMLDOMDocument_get_nodeValue(doc, NULL);
6018     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
6019
6020     V_VT(&value) = VT_BSTR;
6021     V_BSTR(&value) = NULL;
6022     hr = IXMLDOMDocument_get_nodeValue(doc, &value);
6023     ok(hr == S_FALSE, "ret %08x\n", hr );
6024     ok(V_VT(&value) == VT_NULL, "expect VT_NULL got %d\n", V_VT(&value));
6025
6026     hr = IXMLDOMDocument_get_nodeTypedValue(doc, NULL);
6027     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
6028
6029     hr = IXMLDOMDocument_get_nodeTypedValue(doc, &value);
6030     ok(hr == S_FALSE, "ret %08x\n", hr );
6031
6032     hr = IXMLDOMDocument_selectSingleNode(doc, _bstr_("root/string"), &node);
6033     ok(hr == S_OK, "ret %08x\n", hr );
6034
6035     V_VT(&value) = VT_BSTR;
6036     V_BSTR(&value) = NULL;
6037     hr = IXMLDOMNode_get_nodeValue(node, &value);
6038     ok(hr == S_FALSE, "ret %08x\n", hr );
6039     ok(V_VT(&value) == VT_NULL, "expect VT_NULL got %d\n", V_VT(&value));
6040
6041     hr = IXMLDOMNode_get_nodeTypedValue(node, NULL);
6042     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
6043
6044     IXMLDOMNode_Release(node);
6045
6046     hr = IXMLDOMDocument_selectSingleNode(doc, _bstr_("root/binhex"), &node);
6047     ok(hr == S_OK, "ret %08x\n", hr );
6048     {
6049         BYTE bytes[] = {0xff,0xfc,0xa0,0x12,0x00,0x3c};
6050
6051         hr = IXMLDOMNode_get_nodeTypedValue(node, &value);
6052         ok(hr == S_OK, "ret %08x\n", hr );
6053         ok(V_VT(&value) == (VT_ARRAY|VT_UI1), "incorrect type\n");
6054         ok(V_ARRAY(&value)->rgsabound[0].cElements == 6, "incorrect array size\n");
6055         if(V_ARRAY(&value)->rgsabound[0].cElements == 6)
6056             ok(!memcmp(bytes, V_ARRAY(&value)->pvData, sizeof(bytes)), "incorrect value\n");
6057         VariantClear(&value);
6058         IXMLDOMNode_Release(node);
6059     }
6060
6061     hr = IXMLDOMDocument_selectSingleNode(doc, _bstr_("root/binbase64"), &node);
6062     ok(hr == S_OK, "ret %08x\n", hr );
6063     {
6064         BYTE bytes[] = {0x62,0x61,0x73,0x65,0x36,0x34,0x20,0x74,0x65,0x73,0x74};
6065
6066         hr = IXMLDOMNode_get_nodeTypedValue(node, &value);
6067         ok(hr == S_OK, "ret %08x\n", hr );
6068         ok(V_VT(&value) == (VT_ARRAY|VT_UI1), "incorrect type\n");
6069         ok(V_ARRAY(&value)->rgsabound[0].cElements == 11, "incorrect array size\n");
6070         if(V_ARRAY(&value)->rgsabound[0].cElements == 11)
6071             ok(!memcmp(bytes, V_ARRAY(&value)->pvData, sizeof(bytes)), "incorrect value\n");
6072         VariantClear(&value);
6073         IXMLDOMNode_Release(node);
6074     }
6075
6076     while (entry->name)
6077     {
6078         hr = IXMLDOMDocument_selectSingleNode(doc, _bstr_(entry->name), &node);
6079         ok(hr == S_OK, "ret %08x\n", hr );
6080
6081         hr = IXMLDOMNode_get_nodeTypedValue(node, &value);
6082         ok(hr == S_OK, "ret %08x\n", hr );
6083         ok(V_VT(&value) == entry->type, "incorrect type, expected %d, got %d\n", entry->type, V_VT(&value));
6084
6085         if (entry->type != VT_BSTR)
6086         {
6087            if (entry->type == VT_DATE ||
6088                entry->type == VT_R8 ||
6089                entry->type == VT_CY)
6090            {
6091                if (entry->type == VT_DATE)
6092                {
6093                    hr = VariantChangeType(&value, &value, 0, VT_R4);
6094                    ok(hr == S_OK, "ret %08x\n", hr );
6095                }
6096                hr = VariantChangeTypeEx(&value, &value,
6097                                         MAKELCID(MAKELANGID(LANG_ENGLISH, SUBLANG_DEFAULT), SORT_DEFAULT),
6098                                         VARIANT_NOUSEROVERRIDE, VT_BSTR);
6099                ok(hr == S_OK, "ret %08x\n", hr );
6100            }
6101            else
6102            {
6103                hr = VariantChangeType(&value, &value, 0, VT_BSTR);
6104                ok(hr == S_OK, "ret %08x\n", hr );
6105            }
6106
6107            ok(lstrcmpW( V_BSTR(&value), _bstr_(entry->value)) == 0,
6108               "expected %s, got %s\n", entry->value, wine_dbgstr_w(V_BSTR(&value)));
6109         }
6110         else
6111            ok(lstrcmpW( V_BSTR(&value), _bstr_(entry->value)) == 0,
6112                "expected %s, got %s\n", entry->value, wine_dbgstr_w(V_BSTR(&value)));
6113
6114         VariantClear( &value );
6115         IXMLDOMNode_Release(node);
6116
6117         entry++;
6118     }
6119
6120     IXMLDOMDocument_Release(doc);
6121     free_bstrs();
6122 }
6123
6124 static void test_TransformWithLoadingLocalFile(void)
6125 {
6126     IXMLDOMDocument *doc;
6127     IXMLDOMDocument *xsl;
6128     IXMLDOMNode *pNode;
6129     VARIANT_BOOL bSucc;
6130     HRESULT hr;
6131     HANDLE file;
6132     DWORD dwWritten;
6133     char lpPathBuffer[MAX_PATH];
6134     int i;
6135
6136     /* Create a Temp File. */
6137     GetTempPathA(MAX_PATH, lpPathBuffer);
6138     strcat(lpPathBuffer, "customers.xml" );
6139
6140     file = CreateFile(lpPathBuffer, GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL );
6141     ok(file != INVALID_HANDLE_VALUE, "Could not create file: %u\n", GetLastError());
6142     if(file == INVALID_HANDLE_VALUE)
6143         return;
6144
6145     WriteFile(file, szBasicTransformXML, strlen(szBasicTransformXML), &dwWritten, NULL);
6146     CloseHandle(file);
6147
6148     /* Correct path to not include a escape character. */
6149     for(i=0; i < strlen(lpPathBuffer); i++)
6150     {
6151         if(lpPathBuffer[i] == '\\')
6152             lpPathBuffer[i] = '/';
6153     }
6154
6155     doc = create_document(&IID_IXMLDOMDocument);
6156     if (!doc) return;
6157
6158     xsl = create_document(&IID_IXMLDOMDocument);
6159     if (!xsl)
6160     {
6161         IXMLDOMDocument2_Release(doc);
6162         return;
6163     }
6164
6165     hr = IXMLDOMDocument_loadXML(doc, _bstr_(szTypeValueXML), &bSucc);
6166     ok(hr == S_OK, "ret %08x\n", hr );
6167     ok(bSucc == VARIANT_TRUE, "Expected VARIANT_TRUE got VARIANT_FALSE\n");
6168     if(bSucc == VARIANT_TRUE)
6169     {
6170         BSTR sXSL;
6171         BSTR sPart1 = _bstr_(szBasicTransformSSXMLPart1);
6172         BSTR sPart2 = _bstr_(szBasicTransformSSXMLPart2);
6173         BSTR sFileName = _bstr_(lpPathBuffer);
6174         int nLegnth = lstrlenW(sPart1) + lstrlenW(sPart2) + lstrlenW(sFileName) + 1;
6175
6176         sXSL = SysAllocStringLen(NULL, nLegnth);
6177         lstrcpyW(sXSL, sPart1);
6178         lstrcatW(sXSL, sFileName);
6179         lstrcatW(sXSL, sPart2);
6180
6181         hr = IXMLDOMDocument_loadXML(xsl, sXSL, &bSucc);
6182         ok(hr == S_OK, "ret %08x\n", hr );
6183         ok(bSucc == VARIANT_TRUE, "Expected VARIANT_TRUE got VARIANT_FALSE\n");
6184         if(bSucc == VARIANT_TRUE)
6185         {
6186             BSTR sResult;
6187
6188             hr = IXMLDOMDocument_QueryInterface(xsl, &IID_IXMLDOMNode, (void**)&pNode );
6189             ok(hr == S_OK, "ret %08x\n", hr );
6190             if(hr == S_OK)
6191             {
6192                 /* This will load the temp file via the XSL */
6193                 hr = IXMLDOMDocument_transformNode(doc, pNode, &sResult);
6194                 ok(hr == S_OK, "ret %08x\n", hr );
6195                 if(hr == S_OK)
6196                 {
6197                     ok( compareIgnoreReturns( sResult, _bstr_(szBasicTransformOutput)), "Stylesheet output not correct\n");
6198                     SysFreeString(sResult);
6199                 }
6200
6201                 IXMLDOMNode_Release(pNode);
6202             }
6203         }
6204
6205         SysFreeString(sXSL);
6206     }
6207
6208     IXMLDOMDocument_Release(doc);
6209     IXMLDOMDocument_Release(xsl);
6210
6211     DeleteFile(lpPathBuffer);
6212     free_bstrs();
6213 }
6214
6215 static void test_put_nodeValue(void)
6216 {
6217     static const WCHAR jeevesW[] = {'J','e','e','v','e','s',' ','&',' ','W','o','o','s','t','e','r',0};
6218     IXMLDOMDocument *doc;
6219     IXMLDOMText *text;
6220     IXMLDOMEntityReference *entityref;
6221     IXMLDOMAttribute *attr;
6222     IXMLDOMNode *node;
6223     HRESULT hr;
6224     VARIANT data, type;
6225
6226     doc = create_document(&IID_IXMLDOMDocument);
6227     if (!doc) return;
6228
6229     /* test for unsupported types */
6230     /* NODE_DOCUMENT */
6231     hr = IXMLDOMDocument_QueryInterface(doc, &IID_IXMLDOMNode, (void**)&node);
6232     ok(hr == S_OK, "ret %08x\n", hr );
6233     V_VT(&data) = VT_BSTR;
6234     V_BSTR(&data) = _bstr_("one two three");
6235     hr = IXMLDOMNode_put_nodeValue(node, data);
6236     ok(hr == E_FAIL, "ret %08x\n", hr );
6237     IXMLDOMNode_Release(node);
6238
6239     /* NODE_DOCUMENT_FRAGMENT */
6240     V_VT(&type) = VT_I1;
6241     V_I1(&type) = NODE_DOCUMENT_FRAGMENT;
6242     hr = IXMLDOMDocument_createNode(doc, type, _bstr_("test"), NULL, &node);
6243     ok(hr == S_OK, "ret %08x\n", hr );
6244     V_VT(&data) = VT_BSTR;
6245     V_BSTR(&data) = _bstr_("one two three");
6246     hr = IXMLDOMNode_put_nodeValue(node, data);
6247     ok(hr == E_FAIL, "ret %08x\n", hr );
6248     IXMLDOMNode_Release(node);
6249
6250     /* NODE_ELEMENT */
6251     V_VT(&type) = VT_I1;
6252     V_I1(&type) = NODE_ELEMENT;
6253     hr = IXMLDOMDocument_createNode(doc, type, _bstr_("test"), NULL, &node);
6254     ok(hr == S_OK, "ret %08x\n", hr );
6255     V_VT(&data) = VT_BSTR;
6256     V_BSTR(&data) = _bstr_("one two three");
6257     hr = IXMLDOMNode_put_nodeValue(node, data);
6258     ok(hr == E_FAIL, "ret %08x\n", hr );
6259     IXMLDOMNode_Release(node);
6260
6261     /* NODE_ENTITY_REFERENCE */
6262     hr = IXMLDOMDocument_createEntityReference(doc, _bstr_("ref"), &entityref);
6263     ok(hr == S_OK, "ret %08x\n", hr );
6264
6265     V_VT(&data) = VT_BSTR;
6266     V_BSTR(&data) = _bstr_("one two three");
6267     hr = IXMLDOMEntityReference_put_nodeValue(entityref, data);
6268     ok(hr == E_FAIL, "ret %08x\n", hr );
6269
6270     hr = IXMLDOMEntityReference_QueryInterface(entityref, &IID_IXMLDOMNode, (void**)&node);
6271     ok(hr == S_OK, "ret %08x\n", hr );
6272     V_VT(&data) = VT_BSTR;
6273     V_BSTR(&data) = _bstr_("one two three");
6274     hr = IXMLDOMNode_put_nodeValue(node, data);
6275     ok(hr == E_FAIL, "ret %08x\n", hr );
6276     IXMLDOMNode_Release(node);
6277     IXMLDOMEntityReference_Release(entityref);
6278
6279     /* supported types */
6280     hr = IXMLDOMDocument_createTextNode(doc, _bstr_(""), &text);
6281     ok(hr == S_OK, "ret %08x\n", hr );
6282     V_VT(&data) = VT_BSTR;
6283     V_BSTR(&data) = _bstr_("Jeeves & Wooster");
6284     hr = IXMLDOMText_put_nodeValue(text, data);
6285     ok(hr == S_OK, "ret %08x\n", hr );
6286     IXMLDOMText_Release(text);
6287
6288     hr = IXMLDOMDocument_createAttribute(doc, _bstr_("attr"), &attr);
6289     ok(hr == S_OK, "ret %08x\n", hr );
6290     V_VT(&data) = VT_BSTR;
6291     V_BSTR(&data) = _bstr_("Jeeves & Wooster");
6292     hr = IXMLDOMAttribute_put_nodeValue(attr, data);
6293     ok(hr == S_OK, "ret %08x\n", hr );
6294     hr = IXMLDOMAttribute_get_nodeValue(attr, &data);
6295     ok(hr == S_OK, "ret %08x\n", hr );
6296     ok(memcmp(V_BSTR(&data), jeevesW, sizeof(jeevesW)) == 0, "got %s\n",
6297         wine_dbgstr_w(V_BSTR(&data)));
6298     VariantClear(&data);
6299     IXMLDOMAttribute_Release(attr);
6300
6301     free_bstrs();
6302
6303     IXMLDOMDocument_Release(doc);
6304 }
6305
6306 static void test_document_IObjectSafety(void)
6307 {
6308     IXMLDOMDocument *doc;
6309     IObjectSafety *safety;
6310     HRESULT hr;
6311
6312     doc = create_document(&IID_IXMLDOMDocument);
6313     if (!doc) return;
6314
6315     hr = IXMLDOMDocument_QueryInterface(doc, &IID_IObjectSafety, (void**)&safety);
6316     ok(hr == S_OK, "ret %08x\n", hr );
6317
6318     test_IObjectSafety_common(safety);
6319
6320     IObjectSafety_Release(safety);
6321
6322     IXMLDOMDocument_Release(doc);
6323 }
6324
6325 typedef struct _property_test_t {
6326     const GUID *guid;
6327     const char *clsid;
6328     const char *property;
6329     const char *value;
6330 } property_test_t;
6331
6332 static const property_test_t properties_test_data[] = {
6333     { &CLSID_DOMDocument,  "CLSID_DOMDocument" , "SelectionLanguage", "XSLPattern" },
6334     { &CLSID_DOMDocument2,  "CLSID_DOMDocument2" , "SelectionLanguage", "XSLPattern" },
6335     { &CLSID_DOMDocument30, "CLSID_DOMDocument30", "SelectionLanguage", "XSLPattern" },
6336     { &CLSID_DOMDocument40, "CLSID_DOMDocument40", "SelectionLanguage", "XPath" },
6337     { &CLSID_DOMDocument60, "CLSID_DOMDocument60", "SelectionLanguage", "XPath" },
6338     { 0 }
6339 };
6340
6341 static void test_default_properties(void)
6342 {
6343     const property_test_t *entry = properties_test_data;
6344     IXMLDOMDocument2 *doc;
6345     VARIANT var;
6346     HRESULT hr;
6347
6348     while (entry->guid)
6349     {
6350         hr = CoCreateInstance(entry->guid, NULL, CLSCTX_INPROC_SERVER, &IID_IXMLDOMDocument2, (void**)&doc);
6351         if (hr != S_OK)
6352         {
6353             win_skip("can't create %s instance\n", entry->clsid);
6354             entry++;
6355             continue;
6356         }
6357
6358         hr = IXMLDOMDocument2_getProperty(doc, _bstr_(entry->property), &var);
6359         ok(hr == S_OK, "got 0x%08x\n", hr);
6360         ok(lstrcmpW(V_BSTR(&var), _bstr_(entry->value)) == 0, "expected %s, for %s\n",
6361            entry->value, entry->clsid);
6362         VariantClear(&var);
6363
6364         IXMLDOMDocument2_Release(doc);
6365
6366         entry++;
6367     }
6368 }
6369
6370 static void test_XSLPattern(void)
6371 {
6372     IXMLDOMDocument2 *doc;
6373     IXMLDOMNodeList *list;
6374     VARIANT_BOOL b;
6375     LONG len;
6376
6377     doc = create_document(&IID_IXMLDOMDocument2);
6378     if (!doc) return;
6379
6380     ole_check(IXMLDOMDocument2_loadXML(doc, _bstr_(szExampleXML), &b));
6381     ok(b == VARIANT_TRUE, "failed to load XML string\n");
6382
6383     /* switch to XSLPattern */
6384     ole_check(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionLanguage"), _variantbstr_("XSLPattern")));
6385
6386     /* XPath doesn't select elements with non-null default namespace with unqualified selectors, XSLPattern does */
6387     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//elem/c"), &list));
6388     len = 0;
6389     ole_check(IXMLDOMNodeList_get_length(list, &len));
6390     /* should select <elem><c> and <elem xmlns='...'><c> but not <elem><foo:c> */
6391     ok(len == 3, "expected 3 entries in list, got %d\n", len);
6392     IXMLDOMNodeList_Release(list);
6393
6394     /* for XSLPattern start index is 0, for XPath it's 1 */
6395     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[0]"), &list));
6396     len = 0;
6397     ole_check(IXMLDOMNodeList_get_length(list, &len));
6398     ok(len != 0, "expected filled list\n");
6399     if (len)
6400         expect_list_and_release(list, "E1.E2.D1");
6401
6402     /* index() */
6403     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index()=1]"), &list));
6404     len = 0;
6405     ole_check(IXMLDOMNodeList_get_length(list, &len));
6406     ok(len != 0, "expected filled list\n");
6407     if (len)
6408         expect_list_and_release(list, "E2.E2.D1");
6409
6410     /* $eq$ */
6411     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index() $eq$ 1]"), &list));
6412     len = 0;
6413     ole_check(IXMLDOMNodeList_get_length(list, &len));
6414     ok(len != 0, "expected filled list\n");
6415     if (len)
6416         expect_list_and_release(list, "E2.E2.D1");
6417
6418     /* end() */
6419     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[end()]"), &list));
6420     len = 0;
6421     ole_check(IXMLDOMNodeList_get_length(list, &len));
6422     ok(len != 0, "expected filled list\n");
6423     if (len)
6424         expect_list_and_release(list, "E4.E2.D1");
6425
6426     /* $not$ */
6427     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[$not$ end()]"), &list));
6428     len = 0;
6429     ole_check(IXMLDOMNodeList_get_length(list, &len));
6430     ok(len != 0, "expected filled list\n");
6431     if (len)
6432         expect_list_and_release(list, "E1.E2.D1 E2.E2.D1 E3.E2.D1");
6433
6434     /* !=/$ne$ */
6435     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index() != 0]"), &list));
6436     len = 0;
6437     ole_check(IXMLDOMNodeList_get_length(list, &len));
6438     ok(len != 0, "expected filled list\n");
6439     if (len)
6440         expect_list_and_release(list, "E2.E2.D1 E3.E2.D1 E4.E2.D1");
6441     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index() $ne$ 0]"), &list));
6442     len = 0;
6443     ole_check(IXMLDOMNodeList_get_length(list, &len));
6444     ok(len != 0, "expected filled list\n");
6445     if (len)
6446         expect_list_and_release(list, "E2.E2.D1 E3.E2.D1 E4.E2.D1");
6447
6448     /* </$lt$ */
6449     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index() < 2]"), &list));
6450     len = 0;
6451     ole_check(IXMLDOMNodeList_get_length(list, &len));
6452     ok(len != 0, "expected filled list\n");
6453     if (len)
6454         expect_list_and_release(list, "E1.E2.D1 E2.E2.D1");
6455     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index() $lt$ 2]"), &list));
6456     len = 0;
6457     ole_check(IXMLDOMNodeList_get_length(list, &len));
6458     ok(len != 0, "expected filled list\n");
6459     if (len)
6460         expect_list_and_release(list, "E1.E2.D1 E2.E2.D1");
6461
6462     /* <=/$le$ */
6463     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index() <= 1]"), &list));
6464     len = 0;
6465     ole_check(IXMLDOMNodeList_get_length(list, &len));
6466     ok(len != 0, "expected filled list\n");
6467     if (len)
6468         expect_list_and_release(list, "E1.E2.D1 E2.E2.D1");
6469     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index() $le$ 1]"), &list));
6470     len = 0;
6471     ole_check(IXMLDOMNodeList_get_length(list, &len));
6472     ok(len != 0, "expected filled list\n");
6473     if (len)
6474         expect_list_and_release(list, "E1.E2.D1 E2.E2.D1");
6475
6476     /* >/$gt$ */
6477     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index() > 1]"), &list));
6478     len = 0;
6479     ole_check(IXMLDOMNodeList_get_length(list, &len));
6480     ok(len != 0, "expected filled list\n");
6481     if (len)
6482         expect_list_and_release(list, "E3.E2.D1 E4.E2.D1");
6483     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index() $gt$ 1]"), &list));
6484     len = 0;
6485     ole_check(IXMLDOMNodeList_get_length(list, &len));
6486     ok(len != 0, "expected filled list\n");
6487     if (len)
6488         expect_list_and_release(list, "E3.E2.D1 E4.E2.D1");
6489
6490     /* >=/$ge$ */
6491     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index() >= 2]"), &list));
6492     len = 0;
6493     ole_check(IXMLDOMNodeList_get_length(list, &len));
6494     ok(len != 0, "expected filled list\n");
6495     if (len)
6496         expect_list_and_release(list, "E3.E2.D1 E4.E2.D1");
6497     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index() $ge$ 2]"), &list));
6498     len = 0;
6499     ole_check(IXMLDOMNodeList_get_length(list, &len));
6500     ok(len != 0, "expected filled list\n");
6501     if (len)
6502         expect_list_and_release(list, "E3.E2.D1 E4.E2.D1");
6503
6504     /* $ieq$ */
6505     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[a $ieq$ 'a2 field']"), &list));
6506     len = 0;
6507     ole_check(IXMLDOMNodeList_get_length(list, &len));
6508     ok(len != 0, "expected filled list\n");
6509     if (len)
6510         expect_list_and_release(list, "E2.E2.D1");
6511
6512     /* $ine$ */
6513     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[a $ine$ 'a2 field']"), &list));
6514     len = 0;
6515     ole_check(IXMLDOMNodeList_get_length(list, &len));
6516     ok(len != 0, "expected filled list\n");
6517     if (len)
6518         expect_list_and_release(list, "E1.E2.D1 E3.E2.D1 E4.E2.D1");
6519
6520     /* $ilt$ */
6521     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[a $ilt$ 'a3 field']"), &list));
6522     len = 0;
6523     ole_check(IXMLDOMNodeList_get_length(list, &len));
6524     ok(len != 0, "expected filled list\n");
6525     if (len)
6526         expect_list_and_release(list, "E1.E2.D1 E2.E2.D1");
6527
6528     /* $ile$ */
6529     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[a $ile$ 'a2 field']"), &list));
6530     len = 0;
6531     ole_check(IXMLDOMNodeList_get_length(list, &len));
6532     ok(len != 0, "expected filled list\n");
6533     if (len)
6534         expect_list_and_release(list, "E1.E2.D1 E2.E2.D1");
6535
6536     /* $igt$ */
6537     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[a $igt$ 'a2 field']"), &list));
6538     len = 0;
6539     ole_check(IXMLDOMNodeList_get_length(list, &len));
6540     ok(len != 0, "expected filled list\n");
6541     if (len)
6542         expect_list_and_release(list, "E3.E2.D1 E4.E2.D1");
6543
6544     /* $ige$ */
6545     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[a $ige$ 'a3 field']"), &list));
6546     len = 0;
6547     ole_check(IXMLDOMNodeList_get_length(list, &len));
6548     ok(len != 0, "expected filled list\n");
6549     if (len)
6550         expect_list_and_release(list, "E3.E2.D1 E4.E2.D1");
6551
6552     /* $any$ */
6553     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[$any$ *='B2 field']"), &list));
6554     len = 0;
6555     ole_check(IXMLDOMNodeList_get_length(list, &len));
6556     ok(len != 0, "expected filled list\n");
6557     if (len)
6558         expect_list_and_release(list, "E2.E2.D1");
6559
6560     /* $all$ */
6561     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[$all$ *!='B2 field']"), &list));
6562     len = 0;
6563     ole_check(IXMLDOMNodeList_get_length(list, &len));
6564     ok(len != 0, "expected filled list\n");
6565     if (len)
6566         expect_list_and_release(list, "E1.E2.D1 E3.E2.D1 E4.E2.D1");
6567
6568     /* or/$or$/|| */
6569     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index()=0 or end()]"), &list));
6570     len = 0;
6571     ole_check(IXMLDOMNodeList_get_length(list, &len));
6572     ok(len != 0, "expected filled list\n");
6573     if (len)
6574         expect_list_and_release(list, "E1.E2.D1 E4.E2.D1");
6575     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index()=0 $or$ end()]"), &list));
6576     len = 0;
6577     ole_check(IXMLDOMNodeList_get_length(list, &len));
6578     ok(len != 0, "expected filled list\n");
6579     if (len)
6580         expect_list_and_release(list, "E1.E2.D1 E4.E2.D1");
6581     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index()=0 || end()]"), &list));
6582     len = 0;
6583     ole_check(IXMLDOMNodeList_get_length(list, &len));
6584     ok(len != 0, "expected filled list\n");
6585     if (len)
6586         expect_list_and_release(list, "E1.E2.D1 E4.E2.D1");
6587
6588     /* and/$and$/&& */
6589     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index()>0 and $not$ end()]"), &list));
6590     len = 0;
6591     ole_check(IXMLDOMNodeList_get_length(list, &len));
6592     ok(len != 0, "expected filled list\n");
6593     if (len)
6594         expect_list_and_release(list, "E2.E2.D1 E3.E2.D1");
6595     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index()>0 $and$ $not$ end()]"), &list));
6596     len = 0;
6597     ole_check(IXMLDOMNodeList_get_length(list, &len));
6598     ok(len != 0, "expected filled list\n");
6599     if (len)
6600         expect_list_and_release(list, "E2.E2.D1 E3.E2.D1");
6601     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index()>0 && $not$ end()]"), &list));
6602     len = 0;
6603     ole_check(IXMLDOMNodeList_get_length(list, &len));
6604     ok(len != 0, "expected filled list\n");
6605     if (len)
6606         expect_list_and_release(list, "E2.E2.D1 E3.E2.D1");
6607
6608     /* namespace handling */
6609     /* no registered namespaces */
6610     ole_check(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionNamespaces"), _variantbstr_("")));
6611     list = NULL;
6612
6613     /* prefixes don't need to be registered, you may use them as they are in the doc */
6614     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//bar:x"), &list));
6615     len = 0;
6616     ole_check(IXMLDOMNodeList_get_length(list, &len));
6617     ok(len != 0, "expected filled list\n");
6618     if (len)
6619         expect_list_and_release(list, "E5.E1.E4.E1.E2.D1 E6.E2.E4.E1.E2.D1");
6620
6621     /* prefixes must be explicitly specified in the name */
6622     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//foo:elem"), &list));
6623     len = 0;
6624     ole_check(IXMLDOMNodeList_get_length(list, &len));
6625     ok(len == 0, "expected empty list\n");
6626     IXMLDOMNodeList_Release(list);
6627
6628     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//foo:c"), &list));
6629     len = 0;
6630     ole_check(IXMLDOMNodeList_get_length(list, &len));
6631     ok(len != 0, "expected filled list\n");
6632     if (len)
6633         expect_list_and_release(list, "E3.E4.E2.D1");
6634
6635     /* explicitly register prefix foo */
6636     ole_check(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionNamespaces"), _variantbstr_("xmlns:foo='urn:uuid:86B2F87F-ACB6-45cd-8B77-9BDB92A01A29'")));
6637
6638     /* now we get the same behavior as XPath */
6639     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//foo:c"), &list));
6640     len = 0;
6641     ole_check(IXMLDOMNodeList_get_length(list, &len));
6642     ok(len != 0, "expected filled list\n");
6643     if (len)
6644         expect_list_and_release(list, "E3.E3.E2.D1 E3.E4.E2.D1");
6645
6646     /* set prefix foo to some nonexistent namespace */
6647     ole_check(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionNamespaces"), _variantbstr_("xmlns:foo='urn:nonexistent-foo'")));
6648
6649     /* the registered prefix takes precedence */
6650     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//foo:c"), &list));
6651     len = 0;
6652     ole_check(IXMLDOMNodeList_get_length(list, &len));
6653     ok(len == 0, "expected empty list\n");
6654     IXMLDOMNodeList_Release(list);
6655
6656     IXMLDOMDocument2_Release(doc);
6657
6658     doc = create_document(&IID_IXMLDOMDocument2);
6659     if (!doc) return;
6660
6661     ole_check(IXMLDOMDocument2_loadXML(doc, _bstr_(szNodeTypesXML), &b));
6662     ok(b == VARIANT_TRUE, "failed to load XML string\n");
6663     list = NULL;
6664
6665     /* attribute() */
6666     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("attribute()"), &list));
6667     len = 0;
6668     ole_check(IXMLDOMNodeList_get_length(list, &len));
6669     ok(len == 0, "expected empty list\n");
6670     IXMLDOMNodeList_Release(list);
6671
6672     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("attribute('depth')"), &list));
6673     len = 0;
6674     ole_check(IXMLDOMNodeList_get_length(list, &len));
6675     ok(len == 0, "expected empty list\n");
6676     IXMLDOMNodeList_Release(list);
6677
6678     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root/attribute('depth')"), &list));
6679     len = 0;
6680     ole_check(IXMLDOMNodeList_get_length(list, &len));
6681     ok(len != 0, "expected filled list\n");
6682     expect_list_and_release(list, "A'depth'.E3.D1");
6683
6684     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//x/attribute()"), &list));
6685     len = 0;
6686     ole_check(IXMLDOMNodeList_get_length(list, &len));
6687     ok(len != 0, "expected filled list\n");
6688     expect_list_and_release(list, "A'id'.E3.E3.D1 A'depth'.E3.E3.D1");
6689
6690     list = NULL;
6691     ole_expect(IXMLDOMDocument2_selectNodes(doc, _bstr_("//x//attribute(id)"), &list), E_FAIL);
6692     if (list)
6693         IXMLDOMNodeList_Release(list);
6694     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//x//attribute('id')"), &list));
6695     len = 0;
6696     ole_check(IXMLDOMNodeList_get_length(list, &len));
6697     ok(len != 0, "expected filled list\n");
6698     if (len)
6699         expect_list_and_release(list, "A'id'.E3.E3.D1 A'id'.E4.E3.E3.D1 A'id'.E5.E3.E3.D1 A'id'.E6.E3.E3.D1");
6700
6701     /* comment() */
6702     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("comment()"), &list));
6703     len = 0;
6704     ole_check(IXMLDOMNodeList_get_length(list, &len));
6705     ok(len != 0, "expected filled list\n");
6706     if (len)
6707         expect_list_and_release(list, "C2.D1");
6708
6709     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//comment()"), &list));
6710     len = 0;
6711     ole_check(IXMLDOMNodeList_get_length(list, &len));
6712     ok(len != 0, "expected filled list\n");
6713     if (len)
6714         expect_list_and_release(list, "C2.D1 C1.E3.D1 C2.E3.E3.D1 C2.E4.E3.D1");
6715
6716     /* element() */
6717     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("element()"), &list));
6718     len = 0;
6719     ole_check(IXMLDOMNodeList_get_length(list, &len));
6720     ok(len != 0, "expected filled list\n");
6721     if (len)
6722         expect_list_and_release(list, "E3.D1");
6723
6724     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root/y/element()"), &list));
6725     len = 0;
6726     ole_check(IXMLDOMNodeList_get_length(list, &len));
6727     ok(len != 0, "expected filled list\n");
6728     if (len)
6729         expect_list_and_release(list, "E4.E4.E3.D1 E5.E4.E3.D1 E6.E4.E3.D1");
6730
6731     list = NULL;
6732     ole_expect(IXMLDOMDocument2_selectNodes(doc, _bstr_("//element(a)"), &list), E_FAIL);
6733     if (list)
6734         IXMLDOMNodeList_Release(list);
6735     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//element('a')"), &list));
6736     len = 0;
6737     ole_check(IXMLDOMNodeList_get_length(list, &len));
6738     ok(len != 0, "expected filled list\n");
6739     if (len)
6740         expect_list_and_release(list, "E4.E3.E3.D1 E4.E4.E3.D1");
6741
6742     /* node() */
6743     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("node()"), &list));
6744     len = 0;
6745     ole_check(IXMLDOMNodeList_get_length(list, &len));
6746     ok(len != 0, "expected filled list\n");
6747     if (len)
6748         expect_list_and_release(list, "P1.D1 C2.D1 E3.D1");
6749
6750     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//x/node()"), &list));
6751     len = 0;
6752     ole_check(IXMLDOMNodeList_get_length(list, &len));
6753     ok(len != 0, "expected filled list\n");
6754     if (len)
6755         expect_list_and_release(list, "P1.E3.E3.D1 C2.E3.E3.D1 T3.E3.E3.D1 E4.E3.E3.D1 E5.E3.E3.D1 E6.E3.E3.D1");
6756
6757     /* nodeType() */
6758     /* XML_ELEMENT_NODE */
6759     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//x/node()[nodeType()=1]"), &list));
6760     len = 0;
6761     ole_check(IXMLDOMNodeList_get_length(list, &len));
6762     ok(len != 0, "expected filled list\n");
6763     if (len)
6764         expect_list_and_release(list, "E4.E3.E3.D1 E5.E3.E3.D1 E6.E3.E3.D1");
6765     /* XML_TEXT_NODE */
6766     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//x/node()[nodeType()=3]"), &list));
6767     len = 0;
6768     ole_check(IXMLDOMNodeList_get_length(list, &len));
6769     ok(len != 0, "expected filled list\n");
6770     if (len)
6771         expect_list_and_release(list, "T3.E3.E3.D1");
6772     /* XML_PI_NODE */
6773     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//x/node()[nodeType()=7]"), &list));
6774     len = 0;
6775     ole_check(IXMLDOMNodeList_get_length(list, &len));
6776     ok(len != 0, "expected filled list\n");
6777     if (len)
6778         expect_list_and_release(list, "P1.E3.E3.D1");
6779     /* XML_COMMENT_NODE */
6780     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//x/node()[nodeType()=8]"), &list));
6781     len = 0;
6782     ole_check(IXMLDOMNodeList_get_length(list, &len));
6783     ok(len != 0, "expected filled list\n");
6784     if (len)
6785         expect_list_and_release(list, "C2.E3.E3.D1");
6786
6787     /* pi() */
6788     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("pi()"), &list));
6789     if (list)
6790     {
6791         len = 0;
6792         ole_check(IXMLDOMNodeList_get_length(list, &len));
6793         ok(len != 0, "expected filled list\n");
6794         if (len)
6795             expect_list_and_release(list, "P1.D1");
6796     }
6797
6798     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//y/pi()"), &list));
6799     len = 0;
6800     ole_check(IXMLDOMNodeList_get_length(list, &len));
6801     ok(len != 0, "expected filled list\n");
6802     if (len)
6803         expect_list_and_release(list, "P1.E4.E3.D1");
6804
6805     /* textnode() */
6806     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root/textnode()"), &list));
6807     len = 0;
6808     ole_check(IXMLDOMNodeList_get_length(list, &len));
6809     ok(len != 0, "expected filled list\n");
6810     if (len)
6811         expect_list_and_release(list, "T2.E3.D1");
6812
6813     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root/element()/textnode()"), &list));
6814     len = 0;
6815     ole_check(IXMLDOMNodeList_get_length(list, &len));
6816     ok(len != 0, "expected filled list\n");
6817     if (len)
6818         expect_list_and_release(list, "T3.E3.E3.D1 T3.E4.E3.D1");
6819
6820     IXMLDOMDocument2_Release(doc);
6821     free_bstrs();
6822 }
6823
6824 static void test_splitText(void)
6825 {
6826     IXMLDOMCDATASection *cdata;
6827     IXMLDOMElement *root;
6828     IXMLDOMDocument *doc;
6829     IXMLDOMText *text, *text2;
6830     IXMLDOMNode *node;
6831     VARIANT var;
6832     VARIANT_BOOL success;
6833     LONG length;
6834     HRESULT hr;
6835
6836     doc = create_document(&IID_IXMLDOMDocument);
6837     if (!doc) return;
6838
6839     hr = IXMLDOMDocument_loadXML(doc, _bstr_("<root></root>"), &success);
6840     ok(hr == S_OK, "got 0x%08x\n", hr);
6841
6842     hr = IXMLDOMDocument_get_documentElement(doc, &root);
6843     ok(hr == S_OK, "got 0x%08x\n", hr);
6844
6845     hr = IXMLDOMDocument_createCDATASection(doc, _bstr_("beautiful plumage"), &cdata);
6846     ok(hr == S_OK, "got 0x%08x\n", hr);
6847
6848     V_VT(&var) = VT_EMPTY;
6849     hr = IXMLDOMElement_appendChild(root, (IXMLDOMNode*)cdata, NULL);
6850     ok(hr == S_OK, "got 0x%08x\n", hr);
6851
6852     length = 0;
6853     hr = IXMLDOMCDATASection_get_length(cdata, &length);
6854     ok(hr == S_OK, "got 0x%08x\n", hr);
6855     ok(length > 0, "got %d\n", length);
6856
6857     hr = IXMLDOMCDATASection_splitText(cdata, 0, NULL);
6858     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
6859
6860     text = (void*)0xdeadbeef;
6861     /* negative offset */
6862     hr = IXMLDOMCDATASection_splitText(cdata, -1, &text);
6863     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
6864     ok(text == (void*)0xdeadbeef, "got %p\n", text);
6865
6866     text = (void*)0xdeadbeef;
6867     /* offset outside data */
6868     hr = IXMLDOMCDATASection_splitText(cdata, length + 1, &text);
6869     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
6870     ok(text == 0, "got %p\n", text);
6871
6872     text = (void*)0xdeadbeef;
6873     /* offset outside data */
6874     hr = IXMLDOMCDATASection_splitText(cdata, length, &text);
6875     ok(hr == S_FALSE, "got 0x%08x\n", hr);
6876     ok(text == 0, "got %p\n", text);
6877
6878     /* no empty node created */
6879     node = (void*)0xdeadbeef;
6880     hr = IXMLDOMCDATASection_get_nextSibling(cdata, &node);
6881     ok(hr == S_FALSE, "got 0x%08x\n", hr);
6882     ok(node == 0, "got %p\n", text);
6883
6884     hr = IXMLDOMCDATASection_splitText(cdata, 10, &text);
6885     ok(hr == S_OK, "got 0x%08x\n", hr);
6886
6887     length = 0;
6888     hr = IXMLDOMText_get_length(text, &length);
6889     ok(hr == S_OK, "got 0x%08x\n", hr);
6890     ok(length == 7, "got %d\n", length);
6891
6892     hr = IXMLDOMCDATASection_get_nextSibling(cdata, &node);
6893     ok(hr == S_OK, "got 0x%08x\n", hr);
6894     IXMLDOMNode_Release(node);
6895
6896     /* split new text node */
6897     hr = IXMLDOMText_get_length(text, &length);
6898     ok(hr == S_OK, "got 0x%08x\n", hr);
6899
6900     node = (void*)0xdeadbeef;
6901     hr = IXMLDOMText_get_nextSibling(text, &node);
6902     ok(hr == S_FALSE, "got 0x%08x\n", hr);
6903     ok(node == 0, "got %p\n", text);
6904
6905     hr = IXMLDOMText_splitText(text, 0, NULL);
6906     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
6907
6908     text2 = (void*)0xdeadbeef;
6909     /* negative offset */
6910     hr = IXMLDOMText_splitText(text, -1, &text2);
6911     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
6912     ok(text2 == (void*)0xdeadbeef, "got %p\n", text2);
6913
6914     text2 = (void*)0xdeadbeef;
6915     /* offset outside data */
6916     hr = IXMLDOMText_splitText(text, length + 1, &text2);
6917     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
6918     ok(text2 == 0, "got %p\n", text2);
6919
6920     text2 = (void*)0xdeadbeef;
6921     /* offset outside data */
6922     hr = IXMLDOMText_splitText(text, length, &text2);
6923     ok(hr == S_FALSE, "got 0x%08x\n", hr);
6924     ok(text2 == 0, "got %p\n", text);
6925
6926     text2 = 0;
6927     hr = IXMLDOMText_splitText(text, 4, &text2);
6928     todo_wine ok(hr == S_OK, "got 0x%08x\n", hr);
6929     if (text2) IXMLDOMText_Release(text2);
6930
6931     node = 0;
6932     hr = IXMLDOMText_get_nextSibling(text, &node);
6933     todo_wine ok(hr == S_OK, "got 0x%08x\n", hr);
6934     if (node) IXMLDOMNode_Release(node);
6935
6936     IXMLDOMText_Release(text);
6937     IXMLDOMElement_Release(root);
6938     IXMLDOMCDATASection_Release(cdata);
6939     free_bstrs();
6940 }
6941
6942 static void test_getQualifiedItem(void)
6943 {
6944     IXMLDOMDocument *doc;
6945     IXMLDOMElement *element;
6946     IXMLDOMNode *pr_node, *node;
6947     IXMLDOMNodeList *root_list;
6948     IXMLDOMNamedNodeMap *map;
6949     VARIANT_BOOL b;
6950     BSTR str;
6951     LONG len;
6952     HRESULT hr;
6953
6954     doc = create_document(&IID_IXMLDOMDocument);
6955     if (!doc) return;
6956
6957     str = SysAllocString( szComplete4 );
6958     hr = IXMLDOMDocument_loadXML( doc, str, &b );
6959     ok( hr == S_OK, "loadXML failed\n");
6960     ok( b == VARIANT_TRUE, "failed to load XML string\n");
6961     SysFreeString( str );
6962
6963     hr = IXMLDOMDocument_get_documentElement(doc, &element);
6964     ok( hr == S_OK, "ret %08x\n", hr);
6965
6966     hr = IXMLDOMElement_get_childNodes(element, &root_list);
6967     ok( hr == S_OK, "ret %08x\n", hr);
6968
6969     hr = IXMLDOMNodeList_get_item(root_list, 1, &pr_node);
6970     ok( hr == S_OK, "ret %08x\n", hr);
6971     IXMLDOMNodeList_Release(root_list);
6972
6973     hr = IXMLDOMNode_get_attributes(pr_node, &map);
6974     ok( hr == S_OK, "ret %08x\n", hr);
6975     IXMLDOMNode_Release(pr_node);
6976
6977     hr = IXMLDOMNamedNodeMap_get_length(map, &len);
6978     ok( hr == S_OK, "ret %08x\n", hr);
6979     ok( len == 3, "length %d\n", len);
6980
6981     hr = IXMLDOMNamedNodeMap_getQualifiedItem(map, NULL, NULL, NULL);
6982     ok( hr == E_INVALIDARG, "ret %08x\n", hr);
6983
6984     node = (void*)0xdeadbeef;
6985     hr = IXMLDOMNamedNodeMap_getQualifiedItem(map, NULL, NULL, &node);
6986     ok( hr == E_INVALIDARG, "ret %08x\n", hr);
6987     ok( node == (void*)0xdeadbeef, "got %p\n", node);
6988
6989     hr = IXMLDOMNamedNodeMap_getQualifiedItem(map, _bstr_("id"), NULL, NULL);
6990     ok( hr == E_INVALIDARG, "ret %08x\n", hr);
6991
6992     hr = IXMLDOMNamedNodeMap_getQualifiedItem(map, _bstr_("id"), NULL, &node);
6993     ok( hr == S_OK, "ret %08x\n", hr);
6994     IXMLDOMNode_Release(node);
6995
6996     IXMLDOMNamedNodeMap_Release( map );
6997     IXMLDOMElement_Release( element );
6998     IXMLDOMDocument_Release( doc );
6999     free_bstrs();
7000 }
7001
7002 static void test_removeQualifiedItem(void)
7003 {
7004     IXMLDOMDocument *doc;
7005     IXMLDOMElement *element;
7006     IXMLDOMNode *pr_node, *node;
7007     IXMLDOMNodeList *root_list;
7008     IXMLDOMNamedNodeMap *map;
7009     VARIANT_BOOL b;
7010     BSTR str;
7011     LONG len;
7012     HRESULT hr;
7013
7014     doc = create_document(&IID_IXMLDOMDocument);
7015     if (!doc) return;
7016
7017     str = SysAllocString( szComplete4 );
7018     hr = IXMLDOMDocument_loadXML( doc, str, &b );
7019     ok( hr == S_OK, "loadXML failed\n");
7020     ok( b == VARIANT_TRUE, "failed to load XML string\n");
7021     SysFreeString( str );
7022
7023     hr = IXMLDOMDocument_get_documentElement(doc, &element);
7024     ok( hr == S_OK, "ret %08x\n", hr);
7025
7026     hr = IXMLDOMElement_get_childNodes(element, &root_list);
7027     ok( hr == S_OK, "ret %08x\n", hr);
7028
7029     hr = IXMLDOMNodeList_get_item(root_list, 1, &pr_node);
7030     ok( hr == S_OK, "ret %08x\n", hr);
7031     IXMLDOMNodeList_Release(root_list);
7032
7033     hr = IXMLDOMNode_get_attributes(pr_node, &map);
7034     ok( hr == S_OK, "ret %08x\n", hr);
7035     IXMLDOMNode_Release(pr_node);
7036
7037     hr = IXMLDOMNamedNodeMap_get_length(map, &len);
7038     ok( hr == S_OK, "ret %08x\n", hr);
7039     ok( len == 3, "length %d\n", len);
7040
7041     hr = IXMLDOMNamedNodeMap_removeQualifiedItem(map, NULL, NULL, NULL);
7042     ok( hr == E_INVALIDARG, "ret %08x\n", hr);
7043
7044     node = (void*)0xdeadbeef;
7045     hr = IXMLDOMNamedNodeMap_removeQualifiedItem(map, NULL, NULL, &node);
7046     ok( hr == E_INVALIDARG, "ret %08x\n", hr);
7047     ok( node == (void*)0xdeadbeef, "got %p\n", node);
7048
7049     /* out pointer is optional */
7050     hr = IXMLDOMNamedNodeMap_removeQualifiedItem(map, _bstr_("id"), NULL, NULL);
7051     ok( hr == S_OK, "ret %08x\n", hr);
7052
7053     /* already removed */
7054     hr = IXMLDOMNamedNodeMap_removeQualifiedItem(map, _bstr_("id"), NULL, NULL);
7055     ok( hr == S_FALSE, "ret %08x\n", hr);
7056
7057     hr = IXMLDOMNamedNodeMap_removeQualifiedItem(map, _bstr_("vr"), NULL, &node);
7058     ok( hr == S_OK, "ret %08x\n", hr);
7059     IXMLDOMNode_Release(node);
7060
7061     IXMLDOMNamedNodeMap_Release( map );
7062     IXMLDOMElement_Release( element );
7063     IXMLDOMDocument_Release( doc );
7064     free_bstrs();
7065 }
7066
7067 #define check_default_props(doc) _check_default_props(__LINE__, doc)
7068 static inline void _check_default_props(int line, IXMLDOMDocument2* doc)
7069 {
7070     VARIANT_BOOL b;
7071     VARIANT var;
7072     HRESULT hr;
7073
7074     VariantInit(&var);
7075     helper_ole_check(IXMLDOMDocument2_getProperty(doc, _bstr_("SelectionLanguage"), &var));
7076     ok_(__FILE__, line)(lstrcmpW(V_BSTR(&var), _bstr_("XSLPattern")) == 0, "expected XSLPattern\n");
7077     VariantClear(&var);
7078
7079     helper_ole_check(IXMLDOMDocument2_getProperty(doc, _bstr_("SelectionNamespaces"), &var));
7080     ok_(__FILE__, line)(lstrcmpW(V_BSTR(&var), _bstr_("")) == 0, "expected empty string\n");
7081     VariantClear(&var);
7082
7083     helper_ole_check(IXMLDOMDocument2_get_preserveWhiteSpace(doc, &b));
7084     ok_(__FILE__, line)(b == VARIANT_FALSE, "expected FALSE\n");
7085
7086     hr = IXMLDOMDocument2_get_schemas(doc, &var);
7087     ok_(__FILE__, line)(hr == S_FALSE, "got %08x\n", hr);
7088     VariantClear(&var);
7089 }
7090
7091 #define check_set_props(doc) _check_set_props(__LINE__, doc)
7092 static inline void _check_set_props(int line, IXMLDOMDocument2* doc)
7093 {
7094     VARIANT_BOOL b;
7095     VARIANT var;
7096
7097     VariantInit(&var);
7098     helper_ole_check(IXMLDOMDocument2_getProperty(doc, _bstr_("SelectionLanguage"), &var));
7099     ok_(__FILE__, line)(lstrcmpW(V_BSTR(&var), _bstr_("XPath")) == 0, "expected XPath\n");
7100     VariantClear(&var);
7101
7102     helper_ole_check(IXMLDOMDocument2_getProperty(doc, _bstr_("SelectionNamespaces"), &var));
7103     ok_(__FILE__, line)(lstrcmpW(V_BSTR(&var), _bstr_("xmlns:wi=\'www.winehq.org\'")) == 0, "got %s\n", wine_dbgstr_w(V_BSTR(&var)));
7104     VariantClear(&var);
7105
7106     helper_ole_check(IXMLDOMDocument2_get_preserveWhiteSpace(doc, &b));
7107     ok_(__FILE__, line)(b == VARIANT_TRUE, "expected TRUE\n");
7108
7109     helper_ole_check(IXMLDOMDocument2_get_schemas(doc, &var));
7110     ok_(__FILE__, line)(V_VT(&var) != VT_NULL, "expected pointer\n");
7111     VariantClear(&var);
7112 }
7113
7114 #define set_props(doc, cache) _set_props(__LINE__, doc, cache)
7115 static inline void _set_props(int line, IXMLDOMDocument2* doc, IXMLDOMSchemaCollection* cache)
7116 {
7117     VARIANT var;
7118
7119     VariantInit(&var);
7120     helper_ole_check(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionLanguage"), _variantbstr_("XPath")));
7121     helper_ole_check(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionNamespaces"), _variantbstr_("xmlns:wi=\'www.winehq.org\'")));
7122     helper_ole_check(IXMLDOMDocument2_put_preserveWhiteSpace(doc, VARIANT_TRUE));
7123     V_VT(&var) = VT_DISPATCH;
7124     V_DISPATCH(&var) = NULL;
7125     helper_ole_check(IXMLDOMSchemaCollection_QueryInterface(cache, &IID_IDispatch, (void**)&V_DISPATCH(&var)));
7126     ok_(__FILE__, line)(V_DISPATCH(&var) != NULL, "expected pointer\n");
7127     helper_ole_check(IXMLDOMDocument2_putref_schemas(doc, var));
7128     VariantClear(&var);
7129 }
7130
7131 #define unset_props(doc) _unset_props(__LINE__, doc)
7132 static inline void _unset_props(int line, IXMLDOMDocument2* doc)
7133 {
7134     VARIANT var;
7135
7136     VariantInit(&var);
7137     helper_ole_check(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionLanguage"), _variantbstr_("XSLPattern")));
7138     helper_ole_check(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionNamespaces"), _variantbstr_("")));
7139     helper_ole_check(IXMLDOMDocument2_put_preserveWhiteSpace(doc, VARIANT_FALSE));
7140     V_VT(&var) = VT_NULL;
7141     helper_ole_check(IXMLDOMDocument2_putref_schemas(doc, var));
7142     VariantClear(&var);
7143 }
7144
7145 static void test_get_ownerDocument(void)
7146 {
7147     IXMLDOMDocument *doc1, *doc2, *doc3;
7148     IXMLDOMDocument2 *doc, *doc_owner;
7149     IXMLDOMNode *node;
7150     IXMLDOMSchemaCollection *cache;
7151     VARIANT_BOOL b;
7152     VARIANT var;
7153     BSTR str;
7154
7155     doc = create_document(&IID_IXMLDOMDocument2);
7156     cache = create_cache(&IID_IXMLDOMSchemaCollection);
7157     if (!doc || !cache)
7158     {
7159         if (doc) IXMLDOMDocument2_Release(doc);
7160         if (cache) IXMLDOMSchemaCollection_Release(cache);
7161         return;
7162     }
7163
7164     VariantInit(&var);
7165
7166     str = SysAllocString(szComplete4);
7167     ole_check(IXMLDOMDocument_loadXML(doc, str, &b));
7168     ok(b == VARIANT_TRUE, "failed to load XML string\n");
7169     SysFreeString(str);
7170
7171     check_default_props(doc);
7172
7173     /* set properties and check that new instances use them */
7174     set_props(doc, cache);
7175     check_set_props(doc);
7176
7177     ole_check(IXMLDOMDocument2_get_firstChild(doc, &node));
7178     ole_check(IXMLDOMNode_get_ownerDocument(node, &doc1));
7179
7180     /* new interface keeps props */
7181     ole_check(IXMLDOMDocument_QueryInterface(doc1, &IID_IXMLDOMDocument2, (void**)&doc_owner));
7182     ok( doc_owner != doc, "got %p, doc %p\n", doc_owner, doc);
7183     check_set_props(doc_owner);
7184     IXMLDOMDocument2_Release(doc_owner);
7185
7186     ole_check(IXMLDOMNode_get_ownerDocument(node, &doc2));
7187     IXMLDOMNode_Release(node);
7188
7189     ok(doc1 != doc2, "got %p, expected %p. original %p\n", doc2, doc1, doc);
7190
7191     /* reload */
7192     str = SysAllocString(szComplete4);
7193     ole_check(IXMLDOMDocument2_loadXML(doc, str, &b));
7194     ok(b == VARIANT_TRUE, "failed to load XML string\n");
7195     SysFreeString(str);
7196
7197     /* properties retained even after reload */
7198     check_set_props(doc);
7199
7200     ole_check(IXMLDOMDocument2_get_firstChild(doc, &node));
7201     ole_check(IXMLDOMNode_get_ownerDocument(node, &doc3));
7202     IXMLDOMNode_Release(node);
7203
7204     ole_check(IXMLDOMDocument_QueryInterface(doc3, &IID_IXMLDOMDocument2, (void**)&doc_owner));
7205     ok(doc3 != doc1 && doc3 != doc2 && doc_owner != doc, "got %p, (%p, %p, %p)\n", doc3, doc, doc1, doc2);
7206     check_set_props(doc_owner);
7207
7208     /* changing properties for one instance changes them for all */
7209     unset_props(doc_owner);
7210     check_default_props(doc_owner);
7211     check_default_props(doc);
7212
7213     IXMLDOMSchemaCollection_Release(cache);
7214     IXMLDOMDocument_Release(doc1);
7215     IXMLDOMDocument_Release(doc2);
7216     IXMLDOMDocument_Release(doc3);
7217     IXMLDOMDocument2_Release(doc);
7218     IXMLDOMDocument2_Release(doc_owner);
7219     free_bstrs();
7220 }
7221
7222 static void test_setAttributeNode(void)
7223 {
7224     IXMLDOMDocument *doc, *doc2;
7225     IXMLDOMElement *elem;
7226     IXMLDOMAttribute *attr, *attr2, *ret_attr;
7227     VARIANT_BOOL b;
7228     HRESULT hr;
7229     BSTR str;
7230
7231     doc = create_document(&IID_IXMLDOMDocument);
7232     if (!doc) return;
7233
7234     str = SysAllocString( szComplete4 );
7235     hr = IXMLDOMDocument2_loadXML( doc, str, &b );
7236     ok( hr == S_OK, "loadXML failed\n");
7237     ok( b == VARIANT_TRUE, "failed to load XML string\n");
7238     SysFreeString( str );
7239
7240     hr = IXMLDOMDocument_get_documentElement(doc, &elem);
7241     ok( hr == S_OK, "got 0x%08x\n", hr);
7242
7243     ret_attr = (void*)0xdeadbeef;
7244     hr = IXMLDOMElement_setAttributeNode(elem, NULL, &ret_attr);
7245     ok( hr == E_INVALIDARG, "got 0x%08x\n", hr);
7246     ok( ret_attr == (void*)0xdeadbeef, "got %p\n", ret_attr);
7247
7248     hr = IXMLDOMDocument_createAttribute(doc, _bstr_("attr"), &attr);
7249     ok( hr == S_OK, "got 0x%08x\n", hr);
7250
7251     ret_attr = (void*)0xdeadbeef;
7252     hr = IXMLDOMElement_setAttributeNode(elem, attr, &ret_attr);
7253     ok( hr == S_OK, "got 0x%08x\n", hr);
7254     ok( ret_attr == NULL, "got %p\n", ret_attr);
7255
7256     attr2 = NULL;
7257     hr = IXMLDOMElement_getAttributeNode(elem, _bstr_("attr"), &attr2);
7258     ok( hr == S_OK, "got 0x%08x\n", hr);
7259     IXMLDOMAttribute_Release(attr2);
7260
7261     /* try to add it another time */
7262     ret_attr = (void*)0xdeadbeef;
7263     hr = IXMLDOMElement_setAttributeNode(elem, attr, &ret_attr);
7264     todo_wine ok( hr == E_FAIL, "got 0x%08x\n", hr);
7265     todo_wine ok( ret_attr == (void*)0xdeadbeef, "got %p\n", ret_attr);
7266
7267     IXMLDOMElement_Release(elem);
7268
7269     /* initialy used element is released, attribute still 'has' a container */
7270     hr = IXMLDOMDocument_get_documentElement(doc, &elem);
7271     ok( hr == S_OK, "got 0x%08x\n", hr);
7272     ret_attr = (void*)0xdeadbeef;
7273     hr = IXMLDOMElement_setAttributeNode(elem, attr, &ret_attr);
7274     todo_wine ok( hr == E_FAIL, "got 0x%08x\n", hr);
7275     todo_wine ok( ret_attr == (void*)0xdeadbeef, "got %p\n", ret_attr);
7276     IXMLDOMElement_Release(elem);
7277
7278     /* add attribute already attached to another document */
7279     doc2 = create_document(&IID_IXMLDOMDocument);
7280
7281     str = SysAllocString( szComplete4 );
7282     hr = IXMLDOMDocument_loadXML( doc2, str, &b );
7283     ok( hr == S_OK, "loadXML failed\n");
7284     ok( b == VARIANT_TRUE, "failed to load XML string\n");
7285     SysFreeString( str );
7286
7287     hr = IXMLDOMDocument_get_documentElement(doc2, &elem);
7288     ok( hr == S_OK, "got 0x%08x\n", hr);
7289     hr = IXMLDOMElement_setAttributeNode(elem, attr, NULL);
7290     todo_wine ok( hr == E_FAIL, "got 0x%08x\n", hr);
7291     IXMLDOMElement_Release(elem);
7292
7293     IXMLDOMDocument_Release(doc2);
7294
7295     IXMLDOMAttribute_Release(attr);
7296     IXMLDOMDocument_Release(doc);
7297     free_bstrs();
7298 }
7299
7300 static void test_put_dataType(void)
7301 {
7302     IXMLDOMCDATASection *cdata;
7303     IXMLDOMDocument *doc;
7304     VARIANT_BOOL b;
7305     HRESULT hr;
7306     BSTR str;
7307
7308     doc = create_document(&IID_IXMLDOMDocument);
7309     if (!doc) return;
7310
7311     str = SysAllocString( szComplete4 );
7312     hr = IXMLDOMDocument_loadXML( doc, str, &b );
7313     ok( hr == S_OK, "loadXML failed\n");
7314     ok( b == VARIANT_TRUE, "failed to load XML string\n");
7315     SysFreeString( str );
7316
7317     hr = IXMLDOMDocument_createCDATASection(doc, _bstr_("test"), &cdata);
7318     ok( hr == S_OK, "got 0x%08x\n", hr);
7319     hr = IXMLDOMCDATASection_put_dataType(cdata, _bstr_("number"));
7320     ok( hr == E_FAIL, "got 0x%08x\n", hr);
7321     hr = IXMLDOMCDATASection_put_dataType(cdata, _bstr_("string"));
7322     ok( hr == E_FAIL, "got 0x%08x\n", hr);
7323     IXMLDOMCDATASection_Release(cdata);
7324
7325     IXMLDOMDocument_Release(doc);
7326     free_bstrs();
7327 }
7328
7329 static void test_createNode(void)
7330 {
7331     IXMLDOMDocument *doc;
7332     IXMLDOMElement *elem;
7333     IXMLDOMNode *node;
7334     VARIANT v, var;
7335     BSTR prefix, str;
7336     HRESULT hr;
7337
7338     doc = create_document(&IID_IXMLDOMDocument);
7339     if (!doc) return;
7340
7341     /* NODE_ELEMENT nodes */
7342     /* 1. specified namespace */
7343     V_VT(&v) = VT_I4;
7344     V_I4(&v) = NODE_ELEMENT;
7345
7346     hr = IXMLDOMDocument_createNode(doc, v, _bstr_("ns1:test"), _bstr_("http://winehq.org"), &node);
7347     ok( hr == S_OK, "got 0x%08x\n", hr);
7348     prefix = NULL;
7349     hr = IXMLDOMNode_get_prefix(node, &prefix);
7350     ok( hr == S_OK, "got 0x%08x\n", hr);
7351     ok(lstrcmpW(prefix, _bstr_("ns1")) == 0, "wrong prefix\n");
7352     SysFreeString(prefix);
7353     IXMLDOMNode_Release(node);
7354
7355     /* 2. default namespace */
7356     hr = IXMLDOMDocument_createNode(doc, v, _bstr_("test"), _bstr_("http://winehq.org/default"), &node);
7357     ok( hr == S_OK, "got 0x%08x\n", hr);
7358     prefix = (void*)0xdeadbeef;
7359     hr = IXMLDOMNode_get_prefix(node, &prefix);
7360     ok( hr == S_FALSE, "got 0x%08x\n", hr);
7361     ok(prefix == 0, "expected empty prefix, got %p\n", prefix);
7362
7363     hr = IXMLDOMNode_QueryInterface(node, &IID_IXMLDOMElement, (void**)&elem);
7364     ok( hr == S_OK, "got 0x%08x\n", hr);
7365
7366     V_VT(&var) = VT_BSTR;
7367     hr = IXMLDOMElement_getAttribute(elem, _bstr_("xmlns"), &var);
7368     ok( hr == S_FALSE, "got 0x%08x\n", hr);
7369     ok( V_VT(&var) == VT_NULL, "got %d\n", V_VT(&var));
7370
7371     str = NULL;
7372     hr = IXMLDOMElement_get_namespaceURI(elem, &str);
7373     ok( hr == S_OK, "got 0x%08x\n", hr);
7374     ok( lstrcmpW(str, _bstr_("http://winehq.org/default")) == 0, "expected default namespace\n");
7375     SysFreeString(str);
7376
7377     IXMLDOMElement_Release(elem);
7378     IXMLDOMNode_Release(node);
7379
7380     IXMLDOMDocument_Release(doc);
7381     free_bstrs();
7382 }
7383
7384 static void test_get_prefix(void)
7385 {
7386     IXMLDOMDocumentFragment *fragment;
7387     IXMLDOMCDATASection *cdata;
7388     IXMLDOMElement *element;
7389     IXMLDOMComment *comment;
7390     IXMLDOMDocument *doc;
7391     HRESULT hr;
7392     BSTR str;
7393
7394     doc = create_document(&IID_IXMLDOMDocument);
7395     if (!doc) return;
7396
7397     /* nodes that can't support prefix */
7398     /* 1. document */
7399     str = (void*)0xdeadbeef;
7400     hr = IXMLDOMDocument_get_prefix(doc, &str);
7401     ok( hr == S_FALSE, "got 0x%08x\n", hr);
7402     ok( str == 0, "got %p\n", str);
7403
7404     hr = IXMLDOMDocument_get_prefix(doc, NULL);
7405     ok( hr == E_INVALIDARG, "got 0x%08x\n", hr);
7406
7407     /* 2. cdata */
7408     hr = IXMLDOMDocument_createCDATASection(doc, NULL, &cdata);
7409     ok(hr == S_OK, "got %08x\n", hr );
7410
7411     str = (void*)0xdeadbeef;
7412     hr = IXMLDOMCDATASection_get_prefix(cdata, &str);
7413     ok(hr == S_FALSE, "got %08x\n", hr);
7414     ok( str == 0, "got %p\n", str);
7415
7416     hr = IXMLDOMCDATASection_get_prefix(cdata, NULL);
7417     ok(hr == E_INVALIDARG, "got %08x\n", hr);
7418     IXMLDOMCDATASection_Release(cdata);
7419
7420     /* 3. comment */
7421     hr = IXMLDOMDocument_createComment(doc, NULL, &comment);
7422     ok(hr == S_OK, "got %08x\n", hr );
7423
7424     str = (void*)0xdeadbeef;
7425     hr = IXMLDOMComment_get_prefix(comment, &str);
7426     ok(hr == S_FALSE, "got %08x\n", hr);
7427     ok( str == 0, "got %p\n", str);
7428
7429     hr = IXMLDOMComment_get_prefix(comment, NULL);
7430     ok(hr == E_INVALIDARG, "got %08x\n", hr);
7431     IXMLDOMComment_Release(comment);
7432
7433     /* 4. fragment */
7434     hr = IXMLDOMDocument_createDocumentFragment(doc, &fragment);
7435     ok(hr == S_OK, "got %08x\n", hr );
7436
7437     str = (void*)0xdeadbeef;
7438     hr = IXMLDOMDocumentFragment_get_prefix(fragment, &str);
7439     ok(hr == S_FALSE, "got %08x\n", hr);
7440     ok( str == 0, "got %p\n", str);
7441
7442     hr = IXMLDOMDocumentFragment_get_prefix(fragment, NULL);
7443     ok(hr == E_INVALIDARG, "got %08x\n", hr);
7444     IXMLDOMDocumentFragment_Release(fragment);
7445
7446     /* no prefix */
7447     hr = IXMLDOMDocument_createElement(doc, _bstr_("elem"), &element);
7448     ok( hr == S_OK, "got 0x%08x\n", hr);
7449
7450     hr = IXMLDOMElement_get_prefix(element, NULL);
7451     ok( hr == E_INVALIDARG, "got 0x%08x\n", hr);
7452
7453     str = (void*)0xdeadbeef;
7454     hr = IXMLDOMElement_get_prefix(element, &str);
7455     ok( hr == S_FALSE, "got 0x%08x\n", hr);
7456     ok( str == 0, "got %p\n", str);
7457
7458     IXMLDOMElement_Release(element);
7459
7460     /* with prefix */
7461     hr = IXMLDOMDocument_createElement(doc, _bstr_("a:elem"), &element);
7462     ok( hr == S_OK, "got 0x%08x\n", hr);
7463
7464     str = (void*)0xdeadbeef;
7465     hr = IXMLDOMElement_get_prefix(element, &str);
7466     ok( hr == S_OK, "got 0x%08x\n", hr);
7467     ok( lstrcmpW(str, _bstr_("a")) == 0, "expected prefix \"a\"\n");
7468     SysFreeString(str);
7469
7470     str = (void*)0xdeadbeef;
7471     hr = IXMLDOMElement_get_namespaceURI(element, &str);
7472     ok( hr == S_FALSE, "got 0x%08x\n", hr);
7473     ok( str == 0, "got %p\n", str);
7474
7475     IXMLDOMElement_Release(element);
7476
7477     IXMLDOMDocument_Release(doc);
7478     free_bstrs();
7479 }
7480
7481 static void test_selectSingleNode(void)
7482 {
7483     IXMLDOMDocument *doc;
7484     IXMLDOMNodeList *list;
7485     IXMLDOMNode *node;
7486     VARIANT_BOOL b;
7487     HRESULT hr;
7488     LONG len;
7489     BSTR str;
7490
7491     doc = create_document(&IID_IXMLDOMDocument);
7492     if (!doc) return;
7493
7494     hr = IXMLDOMDocument_selectSingleNode(doc, NULL, NULL);
7495     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
7496
7497     hr = IXMLDOMDocument_selectNodes(doc, NULL, NULL);
7498     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
7499
7500     str = SysAllocString( szComplete4 );
7501     hr = IXMLDOMDocument_loadXML( doc, str, &b );
7502     ok( hr == S_OK, "loadXML failed\n");
7503     ok( b == VARIANT_TRUE, "failed to load XML string\n");
7504     SysFreeString( str );
7505
7506     hr = IXMLDOMDocument_selectSingleNode(doc, NULL, NULL);
7507     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
7508
7509     hr = IXMLDOMDocument_selectNodes(doc, NULL, NULL);
7510     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
7511
7512     hr = IXMLDOMDocument_selectSingleNode(doc, _bstr_("lc"), NULL);
7513     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
7514
7515     hr = IXMLDOMDocument_selectNodes(doc, _bstr_("lc"), NULL);
7516     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
7517
7518     hr = IXMLDOMDocument_selectSingleNode(doc, _bstr_("lc"), &node);
7519     ok(hr == S_OK, "got 0x%08x\n", hr);
7520     IXMLDOMNode_Release(node);
7521
7522     hr = IXMLDOMDocument_selectNodes(doc, _bstr_("lc"), &list);
7523     ok(hr == S_OK, "got 0x%08x\n", hr);
7524     IXMLDOMNodeList_Release(list);
7525
7526     list = (void*)0xdeadbeef;
7527     hr = IXMLDOMDocument_selectNodes(doc, NULL, &list);
7528     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
7529     ok(list == (void*)0xdeadbeef, "got %p\n", list);
7530
7531     node = (void*)0xdeadbeef;
7532     hr = IXMLDOMDocument_selectSingleNode(doc, _bstr_("nonexistent"), &node);
7533     ok(hr == S_FALSE, "got 0x%08x\n", hr);
7534     ok(node == 0, "got %p\n", node);
7535
7536     list = (void*)0xdeadbeef;
7537     hr = IXMLDOMDocument_selectNodes(doc, _bstr_("nonexistent"), &list);
7538     ok(hr == S_OK, "got 0x%08x\n", hr);
7539     len = 1;
7540     hr = IXMLDOMNodeList_get_length(list, &len);
7541     ok(hr == S_OK, "got 0x%08x\n", hr);
7542     ok(len == 0, "got %d\n", len);
7543     IXMLDOMNodeList_Release(list);
7544
7545     IXMLDOMDocument_Release(doc);
7546     free_bstrs();
7547 }
7548
7549 static void test_events(void)
7550 {
7551     IConnectionPointContainer *conn;
7552     IConnectionPoint *point;
7553     IXMLDOMDocument *doc;
7554     HRESULT hr;
7555     VARIANT v;
7556     IDispatch *event;
7557     ULONG ref;
7558
7559     doc = create_document(&IID_IXMLDOMDocument);
7560     if (!doc) return;
7561
7562     hr = IXMLDOMDocument_QueryInterface(doc, &IID_IConnectionPointContainer, (void**)&conn);
7563     ok(hr == S_OK, "got 0x%08x\n", hr);
7564
7565     hr = IConnectionPointContainer_FindConnectionPoint(conn, &IID_IDispatch, &point);
7566     ok(hr == S_OK, "got 0x%08x\n", hr);
7567     IConnectionPoint_Release(point);
7568     hr = IConnectionPointContainer_FindConnectionPoint(conn, &IID_IPropertyNotifySink, &point);
7569     ok(hr == S_OK, "got 0x%08x\n", hr);
7570     IConnectionPoint_Release(point);
7571     hr = IConnectionPointContainer_FindConnectionPoint(conn, &DIID_XMLDOMDocumentEvents, &point);
7572     ok(hr == S_OK, "got 0x%08x\n", hr);
7573     IConnectionPoint_Release(point);
7574
7575     IConnectionPointContainer_Release(conn);
7576
7577     /* ready state callback */
7578     VariantInit(&v);
7579     hr = IXMLDOMDocument_put_onreadystatechange(doc, v);
7580     ok(hr == DISP_E_TYPEMISMATCH, "got 0x%08x\n", hr);
7581
7582     event = create_dispevent();
7583     V_VT(&v) = VT_UNKNOWN;
7584     V_UNKNOWN(&v) = (IUnknown*)event;
7585
7586     hr = IXMLDOMDocument_put_onreadystatechange(doc, v);
7587     ok(hr == S_OK, "got 0x%08x\n", hr);
7588     ref = IDispatch_AddRef(event);
7589     IDispatch_Release(event);
7590     ok(ref == 3, "got %d\n", ref);
7591
7592     V_VT(&v) = VT_DISPATCH;
7593     V_DISPATCH(&v) = event;
7594
7595     hr = IXMLDOMDocument_put_onreadystatechange(doc, v);
7596     ok(hr == S_OK, "got 0x%08x\n", hr);
7597     ref = IDispatch_AddRef(event);
7598     IDispatch_Release(event);
7599     ok(ref == 3, "got %d\n", ref);
7600
7601     /* VT_NULL doesn't reset event handler */
7602     V_VT(&v) = VT_NULL;
7603     hr = IXMLDOMDocument_put_onreadystatechange(doc, v);
7604     ok(hr == DISP_E_TYPEMISMATCH, "got 0x%08x\n", hr);
7605     ref = IDispatch_AddRef(event);
7606     IDispatch_Release(event);
7607     ok(ref == 3, "got %d\n", ref);
7608
7609     V_VT(&v) = VT_DISPATCH;
7610     V_DISPATCH(&v) = NULL;
7611
7612     hr = IXMLDOMDocument_put_onreadystatechange(doc, v);
7613     ok(hr == S_OK, "got 0x%08x\n", hr);
7614     ref = IDispatch_AddRef(event);
7615     IDispatch_Release(event);
7616     ok(ref == 2, "got %d\n", ref);
7617
7618     V_VT(&v) = VT_UNKNOWN;
7619     V_DISPATCH(&v) = NULL;
7620     hr = IXMLDOMDocument_put_onreadystatechange(doc, v);
7621     ok(hr == S_OK, "got 0x%08x\n", hr);
7622
7623     IDispatch_Release(event);
7624
7625     IXMLDOMDocument_Release(doc);
7626 }
7627
7628 static void test_createProcessingInstruction(void)
7629 {
7630     static const WCHAR bodyW[] = {'t','e','s','t',0};
7631     IXMLDOMProcessingInstruction *pi;
7632     IXMLDOMDocument *doc;
7633     WCHAR buff[10];
7634     HRESULT hr;
7635
7636     doc = create_document(&IID_IXMLDOMDocument);
7637     if (!doc) return;
7638
7639     /* test for BSTR handling, pass broken BSTR */
7640     memcpy(&buff[2], bodyW, sizeof(bodyW));
7641     /* just a big length */
7642     *(DWORD*)buff = 0xf0f0;
7643     hr = IXMLDOMDocument_createProcessingInstruction(doc, _bstr_("test"), &buff[2], &pi);
7644     ok(hr == S_OK, "got 0x%08x\n", hr);
7645
7646     IXMLDOMProcessingInstruction_Release(pi);
7647     IXMLDOMDocument_Release(doc);
7648 }
7649
7650 static void test_put_nodeTypedValue(void)
7651 {
7652     IXMLDOMDocument *doc;
7653     IXMLDOMElement *elem;
7654     VARIANT type;
7655     HRESULT hr;
7656
7657     doc = create_document(&IID_IXMLDOMDocument);
7658     if (!doc) return;
7659
7660     hr = IXMLDOMDocument_createElement(doc, _bstr_("Element"), &elem);
7661     ok(hr == S_OK, "got 0x%08x\n", hr);
7662
7663     V_VT(&type) = VT_EMPTY;
7664     hr = IXMLDOMElement_get_dataType(elem, &type);
7665     ok(hr == S_FALSE, "got 0x%08x\n", hr);
7666     ok(V_VT(&type) == VT_NULL, "got %d, expected VT_NULL\n", V_VT(&type));
7667
7668     /* set typed value for untyped node */
7669     V_VT(&type) = VT_I1;
7670     V_I1(&type) = 1;
7671     hr = IXMLDOMElement_put_nodeTypedValue(elem, type);
7672     ok(hr == S_OK, "got 0x%08x\n", hr);
7673
7674     V_VT(&type) = VT_EMPTY;
7675     hr = IXMLDOMElement_get_dataType(elem, &type);
7676     ok(hr == S_FALSE, "got 0x%08x\n", hr);
7677     ok(V_VT(&type) == VT_NULL, "got %d, expected VT_NULL\n", V_VT(&type));
7678
7679     /* no type info stored */
7680     V_VT(&type) = VT_EMPTY;
7681     hr = IXMLDOMElement_get_nodeTypedValue(elem, &type);
7682     ok(hr == S_OK, "got 0x%08x\n", hr);
7683     ok(V_VT(&type) == VT_BSTR, "got %d, expected VT_BSTR\n", V_VT(&type));
7684     ok(memcmp(V_BSTR(&type), _bstr_("1"), 2*sizeof(WCHAR)) == 0,
7685        "got %s, expected \"1\"\n", wine_dbgstr_w(V_BSTR(&type)));
7686     VariantClear(&type);
7687
7688     IXMLDOMElement_Release(elem);
7689     IXMLDOMDocument_Release(doc);
7690     free_bstrs();
7691 }
7692
7693 static void test_get_xml(void)
7694 {
7695     static const char xmlA[] = "<?xml version=\"1.0\" encoding=\"UTF-16\"?>\r\n<a>test</a>\r\n";
7696     IXMLDOMProcessingInstruction *pi;
7697     IXMLDOMNode *first;
7698     IXMLDOMDocument *doc;
7699     VARIANT_BOOL b;
7700     VARIANT v;
7701     BSTR xml;
7702     HRESULT hr;
7703
7704     doc = create_document(&IID_IXMLDOMDocument);
7705     if (!doc) return;
7706
7707     b = VARIANT_TRUE;
7708     hr = IXMLDOMDocument_loadXML( doc, _bstr_("<a>test</a>"), &b );
7709     ok(hr == S_OK, "got 0x%08x\n", hr);
7710     ok( b == VARIANT_TRUE, "got %d\n", b);
7711
7712     hr = IXMLDOMDocument_createProcessingInstruction(doc, _bstr_("xml"),
7713                              _bstr_("version=\"1.0\" encoding=\"UTF-16\""), &pi);
7714     ok(hr == S_OK, "got 0x%08x\n", hr);
7715
7716     hr = IXMLDOMDocument_get_firstChild(doc, &first);
7717     ok(hr == S_OK, "got 0x%08x\n", hr);
7718
7719     V_UNKNOWN(&v) = (IUnknown*)first;
7720     V_VT(&v) = VT_UNKNOWN;
7721
7722     hr = IXMLDOMDocument_insertBefore(doc, (IXMLDOMNode*)pi, v, NULL);
7723     ok(hr == S_OK, "got 0x%08x\n", hr);
7724
7725     IXMLDOMProcessingInstruction_Release(pi);
7726     IXMLDOMNode_Release(first);
7727
7728     hr = IXMLDOMDocument_get_xml(doc, &xml);
7729     ok(hr == S_OK, "got 0x%08x\n", hr);
7730
7731     ok(memcmp(xml, _bstr_(xmlA), sizeof(xmlA)*sizeof(WCHAR)) == 0,
7732         "got %s, expected %s\n", wine_dbgstr_w(xml), xmlA);
7733     SysFreeString(xml);
7734
7735     IXMLDOMDocument_Release(doc);
7736     free_bstrs();
7737 }
7738
7739 static void test_xsltemplate(void)
7740 {
7741     IXSLTemplate *template;
7742     IXSLProcessor *processor;
7743     IXMLDOMDocument *doc;
7744     IStream *stream;
7745     VARIANT_BOOL b;
7746     HRESULT hr;
7747     ULONG ref1, ref2, ref;
7748     VARIANT v;
7749
7750     template = create_xsltemplate(&IID_IXSLTemplate);
7751     if (!template) return;
7752
7753     /* works as reset */
7754     hr = IXSLTemplate_putref_stylesheet(template, NULL);
7755     ok(hr == S_OK, "got 0x%08x\n", hr);
7756
7757     doc = create_document(&IID_IXMLDOMDocument);
7758
7759     b = VARIANT_TRUE;
7760     hr = IXMLDOMDocument_loadXML( doc, _bstr_("<a>test</a>"), &b );
7761     ok(hr == S_OK, "got 0x%08x\n", hr);
7762     ok( b == VARIANT_TRUE, "got %d\n", b);
7763
7764     /* putref with non-xsl document */
7765     hr = IXSLTemplate_putref_stylesheet(template, (IXMLDOMNode*)doc);
7766     todo_wine ok(hr == E_FAIL, "got 0x%08x\n", hr);
7767
7768     b = VARIANT_TRUE;
7769     hr = IXMLDOMDocument_loadXML( doc, _bstr_(szTransformSSXML), &b );
7770     ok(hr == S_OK, "got 0x%08x\n", hr);
7771     ok( b == VARIANT_TRUE, "got %d\n", b);
7772
7773     /* not a freethreaded document */
7774     hr = IXSLTemplate_putref_stylesheet(template, (IXMLDOMNode*)doc);
7775     todo_wine ok(hr == E_FAIL, "got 0x%08x\n", hr);
7776
7777     IXMLDOMDocument_Release(doc);
7778
7779     hr = CoCreateInstance(&CLSID_FreeThreadedDOMDocument, NULL, CLSCTX_INPROC_SERVER, &IID_IXMLDOMDocument, (void**)&doc);
7780     if (hr != S_OK)
7781     {
7782         win_skip("failed to create free threaded document instance: 0x%08x\n", hr);
7783         IXSLTemplate_Release(template);
7784         return;
7785     }
7786
7787     b = VARIANT_TRUE;
7788     hr = IXMLDOMDocument_loadXML( doc, _bstr_(szTransformSSXML), &b );
7789     ok(hr == S_OK, "got 0x%08x\n", hr);
7790     ok( b == VARIANT_TRUE, "got %d\n", b);
7791
7792     /* freethreaded document */
7793     ref1 = IXMLDOMDocument_AddRef(doc);
7794     IXMLDOMDocument_Release(doc);
7795     hr = IXSLTemplate_putref_stylesheet(template, (IXMLDOMNode*)doc);
7796     ok(hr == S_OK, "got 0x%08x\n", hr);
7797     ref2 = IXMLDOMDocument_AddRef(doc);
7798     IXMLDOMDocument_Release(doc);
7799     ok(ref2 > ref1, "got %d\n", ref2);
7800
7801     /* processor */
7802     hr = IXSLTemplate_createProcessor(template, NULL);
7803     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
7804
7805     ref = IXSLTemplate_AddRef(template);
7806     IXSLTemplate_Release(template);
7807     ok(ref == 2, "got %d\n", ref);
7808
7809     hr = IXSLTemplate_createProcessor(template, &processor);
7810     ok(hr == S_OK, "got 0x%08x\n", hr);
7811
7812     ref = IXSLTemplate_AddRef(template);
7813     IXSLTemplate_Release(template);
7814     ok(ref == 3, "got %d\n", ref);
7815
7816     /* input no set yet */
7817     V_VT(&v) = VT_BSTR;
7818     V_BSTR(&v) = NULL;
7819     hr = IXSLProcessor_get_input(processor, &v);
7820 todo_wine {
7821     ok(hr == S_OK, "got 0x%08x\n", hr);
7822     ok(V_VT(&v) == VT_EMPTY, "got %d\n", V_VT(&v));
7823 }
7824
7825     /* reset before it was set */
7826     V_VT(&v) = VT_EMPTY;
7827     hr = IXSLProcessor_put_output(processor, v);
7828     ok(hr == S_OK, "got 0x%08x\n", hr);
7829
7830     CreateStreamOnHGlobal(NULL, TRUE, &stream);
7831     ref = IStream_AddRef(stream);
7832     IStream_Release(stream);
7833     ok(ref == 2, "got %d\n", ref);
7834
7835     V_VT(&v) = VT_UNKNOWN;
7836     V_UNKNOWN(&v) = (IUnknown*)stream;
7837     hr = IXSLProcessor_put_output(processor, v);
7838     ok(hr == S_OK, "got 0x%08x\n", hr);
7839
7840     /* it seems processor grabs 2 references */
7841     ref = IStream_AddRef(stream);
7842     IStream_Release(stream);
7843     todo_wine ok(ref == 4, "got %d\n", ref);
7844
7845     hr = IXSLProcessor_transform(processor, NULL);
7846     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
7847
7848     /* reset and check stream refcount */
7849     V_VT(&v) = VT_EMPTY;
7850     hr = IXSLProcessor_put_output(processor, v);
7851     ok(hr == S_OK, "got 0x%08x\n", hr);
7852
7853     ref = IStream_AddRef(stream);
7854     IStream_Release(stream);
7855     ok(ref == 2, "got %d\n", ref);
7856
7857     IStream_Release(stream);
7858     IXSLProcessor_Release(processor);
7859
7860     /* drop reference */
7861     hr = IXSLTemplate_putref_stylesheet(template, NULL);
7862     ok(hr == S_OK, "got 0x%08x\n", hr);
7863     ref2 = IXMLDOMDocument_AddRef(doc);
7864     IXMLDOMDocument_Release(doc);
7865     ok(ref2 == ref1, "got %d\n", ref2);
7866
7867     IXMLDOMDocument_Release(doc);
7868     IXSLTemplate_Release(template);
7869 }
7870
7871 START_TEST(domdoc)
7872 {
7873     IXMLDOMDocument *doc;
7874     HRESULT hr;
7875
7876     hr = CoInitialize( NULL );
7877     ok( hr == S_OK, "failed to init com\n");
7878     if (hr != S_OK)
7879         return;
7880
7881     test_XMLHTTP();
7882
7883     hr = CoCreateInstance( &CLSID_DOMDocument, NULL, CLSCTX_INPROC_SERVER, &IID_IXMLDOMDocument, (void**)&doc );
7884     if (hr != S_OK)
7885     {
7886         win_skip("IXMLDOMDocument is not available (0x%08x)\n", hr);
7887         return;
7888     }
7889
7890     IXMLDOMDocument_Release(doc);
7891
7892     test_domdoc();
7893     test_persiststreaminit();
7894     test_domnode();
7895     test_refs();
7896     test_create();
7897     test_getElementsByTagName();
7898     test_get_text();
7899     test_get_childNodes();
7900     test_get_firstChild();
7901     test_get_lastChild();
7902     test_removeChild();
7903     test_replaceChild();
7904     test_removeNamedItem();
7905     test_IXMLDOMDocument2();
7906     test_whitespace();
7907     test_XPath();
7908     test_XSLPattern();
7909     test_cloneNode();
7910     test_xmlTypes();
7911     test_nodeTypeTests();
7912     test_DocumentSaveToDocument();
7913     test_DocumentSaveToFile();
7914     test_testTransforms();
7915     test_Namespaces();
7916     test_FormattingXML();
7917     test_nodeTypedValue();
7918     test_TransformWithLoadingLocalFile();
7919     test_put_nodeValue();
7920     test_document_IObjectSafety();
7921     test_splitText();
7922     test_getQualifiedItem();
7923     test_removeQualifiedItem();
7924     test_get_ownerDocument();
7925     test_setAttributeNode();
7926     test_put_dataType();
7927     test_createNode();
7928     test_get_prefix();
7929     test_default_properties();
7930     test_selectSingleNode();
7931     test_events();
7932     test_createProcessingInstruction();
7933     test_put_nodeTypedValue();
7934     test_get_xml();
7935     test_xsltemplate();
7936
7937     CoUninitialize();
7938 }