hlink: Use ifaces instead of vtbl pointers in ExtensionService.
[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
631 #define create_document_version(v, iid) _create_object(&_create(CLSID_DOMDocument ## v), iid, __LINE__)
632
633 #define create_cache(iid) _create_object(&_create(CLSID_XMLSchemaCache), iid, __LINE__)
634
635 #define create_cache_version(v, iid) _create_object(&_create(CLSID_XMLSchemaCache ## v), iid, __LINE__)
636
637 static BSTR alloc_str_from_narrow(const char *str)
638 {
639     int len = MultiByteToWideChar(CP_ACP, 0, str, -1, NULL, 0);
640     BSTR ret = SysAllocStringLen(NULL, len - 1);  /* NUL character added automatically */
641     MultiByteToWideChar(CP_ACP, 0, str, -1, ret, len);
642     return ret;
643 }
644
645 static BSTR alloced_bstrs[256];
646 static int alloced_bstrs_count;
647
648 static BSTR _bstr_(const char *str)
649 {
650     assert(alloced_bstrs_count < sizeof(alloced_bstrs)/sizeof(alloced_bstrs[0]));
651     alloced_bstrs[alloced_bstrs_count] = alloc_str_from_narrow(str);
652     return alloced_bstrs[alloced_bstrs_count++];
653 }
654
655 static void free_bstrs(void)
656 {
657     int i;
658     for (i = 0; i < alloced_bstrs_count; i++)
659         SysFreeString(alloced_bstrs[i]);
660     alloced_bstrs_count = 0;
661 }
662
663 static VARIANT _variantbstr_(const char *str)
664 {
665     VARIANT v;
666     V_VT(&v) = VT_BSTR;
667     V_BSTR(&v) = _bstr_(str);
668     return v;
669 }
670
671 static BOOL compareIgnoreReturns(BSTR sLeft, BSTR sRight)
672 {
673     for (;;)
674     {
675         while (*sLeft == '\r' || *sLeft == '\n') sLeft++;
676         while (*sRight == '\r' || *sRight == '\n') sRight++;
677         if (*sLeft != *sRight) return FALSE;
678         if (!*sLeft) return TRUE;
679         sLeft++;
680         sRight++;
681     }
682 }
683
684 static void get_str_for_type(DOMNodeType type, char *buf)
685 {
686     switch (type)
687     {
688         case NODE_ATTRIBUTE:
689             strcpy(buf, "A");
690             break;
691         case NODE_ELEMENT:
692             strcpy(buf, "E");
693             break;
694         case NODE_DOCUMENT:
695             strcpy(buf, "D");
696             break;
697         case NODE_TEXT:
698             strcpy(buf, "T");
699             break;
700         case NODE_COMMENT:
701             strcpy(buf, "C");
702             break;
703         case NODE_PROCESSING_INSTRUCTION:
704             strcpy(buf, "P");
705             break;
706         default:
707             wsprintfA(buf, "[%d]", type);
708     }
709 }
710
711 #define test_disp(u) _test_disp(__LINE__,u)
712 static void _test_disp(unsigned line, IUnknown *unk)
713 {
714     DISPID dispid = DISPID_XMLDOM_NODELIST_RESET;
715     IDispatchEx *dispex;
716     DWORD dwProps = 0;
717     BSTR sName;
718     UINT ticnt;
719     IUnknown *pUnk;
720     HRESULT hres;
721
722     hres = IUnknown_QueryInterface(unk, &IID_IDispatchEx, (void**)&dispex);
723     ok_(__FILE__,line) (hres == S_OK, "Could not get IDispatch: %08x\n", hres);
724     if(FAILED(hres))
725         return;
726
727     ticnt = 0xdeadbeef;
728     hres = IDispatchEx_GetTypeInfoCount(dispex, &ticnt);
729     ok_(__FILE__,line) (hres == S_OK, "GetTypeInfoCount failed: %08x\n", hres);
730     ok_(__FILE__,line) (ticnt == 1, "ticnt=%u\n", ticnt);
731
732     sName = SysAllocString( szstar );
733     hres = IDispatchEx_DeleteMemberByName(dispex, sName, fdexNameCaseSensitive);
734     ok(hres == E_NOTIMPL, "expected E_NOTIMPL got %08x\n", hres);
735     SysFreeString( sName );
736
737     hres = IDispatchEx_DeleteMemberByDispID(dispex, dispid);
738     ok(hres == E_NOTIMPL, "expected E_NOTIMPL got %08x\n", hres);
739
740     hres = IDispatchEx_GetMemberProperties(dispex, dispid, grfdexPropCanAll, &dwProps);
741     ok(hres == E_NOTIMPL, "expected E_NOTIMPL got %08x\n", hres);
742     ok(dwProps == 0, "expected 0 got %d\n", dwProps);
743
744     hres = IDispatchEx_GetMemberName(dispex, dispid, &sName);
745     ok(hres == E_NOTIMPL, "expected E_NOTIMPL got %08x\n", hres);
746     if(SUCCEEDED(hres))
747         SysFreeString(sName);
748
749     hres = IDispatchEx_GetNextDispID(dispex, fdexEnumDefault, DISPID_XMLDOM_NODELIST_RESET, &dispid);
750     ok(hres == E_NOTIMPL, "expected E_NOTIMPL got %08x\n", hres);
751
752     hres = IDispatchEx_GetNameSpaceParent(dispex, &pUnk);
753     ok(hres == E_NOTIMPL, "expected E_NOTIMPL got %08x\n", hres);
754     if(hres == S_OK && pUnk)
755         IUnknown_Release(pUnk);
756
757     IDispatchEx_Release(dispex);
758 }
759
760 static int get_node_position(IXMLDOMNode *node)
761 {
762     HRESULT r;
763     int pos = 0;
764
765     IXMLDOMNode_AddRef(node);
766     do
767     {
768         IXMLDOMNode *new_node;
769
770         pos++;
771         r = IXMLDOMNode_get_previousSibling(node, &new_node);
772         ok(SUCCEEDED(r), "get_previousSibling failed\n");
773         IXMLDOMNode_Release(node);
774         node = new_node;
775     } while (r == S_OK);
776     return pos;
777 }
778
779 static void node_to_string(IXMLDOMNode *node, char *buf)
780 {
781     HRESULT r = S_OK;
782     DOMNodeType type;
783
784     if (node == NULL)
785     {
786         lstrcpyA(buf, "(null)");
787         return;
788     }
789
790     IXMLDOMNode_AddRef(node);
791     while (r == S_OK)
792     {
793         IXMLDOMNode *new_node;
794
795         ole_check(IXMLDOMNode_get_nodeType(node, &type));
796         get_str_for_type(type, buf);
797         buf+=strlen(buf);
798
799         if (type == NODE_ATTRIBUTE)
800         {
801             BSTR bstr;
802             ole_check(IXMLDOMNode_get_nodeName(node, &bstr));
803             *(buf++) = '\'';
804             wsprintfA(buf, "%ws", bstr);
805             buf += strlen(buf);
806             *(buf++) = '\'';
807             SysFreeString(bstr);
808
809             r = IXMLDOMNode_selectSingleNode(node, _bstr_(".."), &new_node);
810         }
811         else
812         {
813             r = IXMLDOMNode_get_parentNode(node, &new_node);
814             wsprintf(buf, "%d", get_node_position(node));
815             buf += strlen(buf);
816         }
817
818         ok(SUCCEEDED(r), "get_parentNode failed (%08x)\n", r);
819         IXMLDOMNode_Release(node);
820         node = new_node;
821         if (r == S_OK)
822             *(buf++) = '.';
823     }
824
825     *buf = 0;
826 }
827
828 static char *list_to_string(IXMLDOMNodeList *list)
829 {
830     static char buf[4096];
831     char *pos = buf;
832     LONG len = 0;
833     int i;
834
835     if (list == NULL)
836     {
837         lstrcpyA(buf, "(null)");
838         return buf;
839     }
840     ole_check(IXMLDOMNodeList_get_length(list, &len));
841     for (i = 0; i < len; i++)
842     {
843         IXMLDOMNode *node;
844         if (i > 0)
845             *(pos++) = ' ';
846         ole_check(IXMLDOMNodeList_nextNode(list, &node));
847         node_to_string(node, pos);
848         pos += strlen(pos);
849         IXMLDOMNode_Release(node);
850     }
851     *pos = 0;
852     return buf;
853 }
854
855 #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); }
856 #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); }
857
858 static void test_domdoc( void )
859 {
860     HRESULT r;
861     IXMLDOMDocument *doc;
862     IXMLDOMParseError *error;
863     IXMLDOMElement *element = NULL;
864     IXMLDOMNode *node;
865     IXMLDOMText *nodetext = NULL;
866     IXMLDOMComment *node_comment = NULL;
867     IXMLDOMAttribute *node_attr = NULL;
868     IXMLDOMNode *nodeChild = NULL;
869     IXMLDOMProcessingInstruction *nodePI = NULL;
870     ISupportErrorInfo *support_error = NULL;
871     VARIANT_BOOL b;
872     VARIANT var;
873     BSTR str;
874     LONG code;
875     LONG nLength = 0;
876     WCHAR buff[100];
877
878     doc = create_document(&IID_IXMLDOMDocument);
879     if (!doc) return;
880
881     test_disp((IUnknown*)doc);
882
883 if (0)
884 {
885     /* crashes on native */
886     r = IXMLDOMDocument_loadXML( doc, (BSTR)0x1, NULL );
887 }
888
889     /* try some stupid things */
890     r = IXMLDOMDocument_loadXML( doc, NULL, NULL );
891     ok( r == S_FALSE, "loadXML succeeded\n");
892
893     b = VARIANT_TRUE;
894     r = IXMLDOMDocument_loadXML( doc, NULL, &b );
895     ok( r == S_FALSE, "loadXML succeeded\n");
896     ok( b == VARIANT_FALSE, "failed to load XML string\n");
897
898     /* try to load a document from a nonexistent file */
899     b = VARIANT_TRUE;
900     str = SysAllocString( szNonExistentFile );
901     VariantInit(&var);
902     V_VT(&var) = VT_BSTR;
903     V_BSTR(&var) = str;
904
905     r = IXMLDOMDocument_load( doc, var, &b);
906     ok( r == S_FALSE, "loadXML succeeded\n");
907     ok( b == VARIANT_FALSE, "succeeded in loading XML string\n");
908     SysFreeString( str );
909
910     /* try load an empty document */
911     b = VARIANT_TRUE;
912     str = SysAllocString( szEmpty );
913     r = IXMLDOMDocument_loadXML( doc, str, &b );
914     ok( r == S_FALSE, "loadXML succeeded\n");
915     ok( b == VARIANT_FALSE, "succeeded in loading XML string\n");
916     SysFreeString( str );
917
918     r = IXMLDOMDocument_get_async( doc, &b );
919     ok( r == S_OK, "get_async failed (%08x)\n", r);
920     ok( b == VARIANT_TRUE, "Wrong default value\n");
921
922     /* check that there's no document element */
923     element = NULL;
924     r = IXMLDOMDocument_get_documentElement( doc, &element );
925     ok( r == S_FALSE, "should be no document element\n");
926
927     /* try finding a node */
928     node = NULL;
929     str = SysAllocString( szstr1 );
930     r = IXMLDOMDocument_selectSingleNode( doc, str, &node );
931     ok( r == S_FALSE, "ret %08x\n", r );
932     SysFreeString( str );
933
934     b = VARIANT_TRUE;
935     str = SysAllocString( szIncomplete );
936     r = IXMLDOMDocument_loadXML( doc, str, &b );
937     ok( r == S_FALSE, "loadXML succeeded\n");
938     ok( b == VARIANT_FALSE, "succeeded in loading XML string\n");
939     SysFreeString( str );
940
941     /* check that there's no document element */
942     element = (IXMLDOMElement*)1;
943     r = IXMLDOMDocument_get_documentElement( doc, &element );
944     ok( r == S_FALSE, "should be no document element\n");
945     ok( element == NULL, "Element should be NULL\n");
946
947     /* test for BSTR handling, pass broken BSTR */
948     memcpy(&buff[2], szComplete1, sizeof(szComplete1));
949     /* just a big length */
950     *(DWORD*)buff = 0xf0f0;
951     b = VARIANT_FALSE;
952     r = IXMLDOMDocument_loadXML( doc, &buff[2], &b );
953     ok( r == S_OK, "loadXML failed\n");
954     ok( b == VARIANT_TRUE, "failed to load XML string\n");
955
956     /* loadXML ignores the encoding attribute and always expects Unicode */
957     b = VARIANT_FALSE;
958     str = SysAllocString( szComplete6 );
959     r = IXMLDOMDocument_loadXML( doc, str, &b );
960     ok( r == S_OK, "loadXML failed\n");
961     ok( b == VARIANT_TRUE, "failed to load XML string\n");
962     SysFreeString( str );
963
964     /* try a BSTR containing a Windows-1252 document */
965     b = VARIANT_TRUE;
966     str = SysAllocStringByteLen( szNonUnicodeXML, sizeof(szNonUnicodeXML) - 1 );
967     r = IXMLDOMDocument_loadXML( doc, str, &b );
968     ok( r == S_FALSE, "loadXML succeeded\n");
969     ok( b == VARIANT_FALSE, "succeeded in loading XML string\n");
970     SysFreeString( str );
971
972     /* try to load something valid */
973     b = VARIANT_FALSE;
974     str = SysAllocString( szComplete1 );
975     r = IXMLDOMDocument_loadXML( doc, str, &b );
976     ok( r == S_OK, "loadXML failed\n");
977     ok( b == VARIANT_TRUE, "failed to load XML string\n");
978     SysFreeString( str );
979
980     /* check if nodename is correct */
981     r = IXMLDOMDocument_get_nodeName( doc, NULL );
982     ok ( r == E_INVALIDARG, "get_nodeName (NULL) wrong code\n");
983
984     str = (BSTR)0xdeadbeef;
985     r = IXMLDOMDocument_get_baseName( doc, &str );
986     ok ( r == S_FALSE, "got 0x%08x\n", r);
987     ok (str == NULL, "got %p\n", str);
988
989     /* content doesn't matter here */
990     str = NULL;
991     r = IXMLDOMDocument_get_nodeName( doc, &str );
992     ok ( r == S_OK, "get_nodeName wrong code\n");
993     ok ( str != NULL, "str is null\n");
994     ok( !lstrcmpW( str, szDocument ), "incorrect nodeName\n");
995     SysFreeString( str );
996
997     /* test put_text */
998     r = IXMLDOMDocument_put_text( doc, _bstr_("Should Fail") );
999     ok( r == E_FAIL, "ret %08x\n", r );
1000
1001     /* check that there's a document element */
1002     element = NULL;
1003     r = IXMLDOMDocument_get_documentElement( doc, &element );
1004     ok( r == S_OK, "should be a document element\n");
1005     if( element )
1006     {
1007         IObjectIdentity *ident;
1008         BSTR tag = NULL;
1009
1010         test_disp((IUnknown*)element);
1011
1012         r = IXMLDOMElement_QueryInterface( element, &IID_IObjectIdentity, (void**)&ident );
1013         ok( r == E_NOINTERFACE, "ret %08x\n", r);
1014
1015         r = IXMLDOMElement_get_tagName( element, NULL );
1016         ok( r == E_INVALIDARG, "ret %08x\n", r);
1017
1018         /* check if the tag is correct */
1019         r = IXMLDOMElement_get_tagName( element, &tag );
1020         ok( r == S_OK, "couldn't get tag name\n");
1021         ok( tag != NULL, "tag was null\n");
1022         ok( !lstrcmpW( tag, szOpen ), "incorrect tag name\n");
1023         SysFreeString( tag );
1024
1025         /* figure out what happens if we try to reload the document */
1026         str = SysAllocString( szComplete2 );
1027         r = IXMLDOMDocument_loadXML( doc, str, &b );
1028         ok( r == S_OK, "loadXML failed\n");
1029         ok( b == VARIANT_TRUE, "failed to load XML string\n");
1030         SysFreeString( str );
1031
1032         /* check if the tag is still correct */
1033         tag = NULL;
1034         r = IXMLDOMElement_get_tagName( element, &tag );
1035         ok( r == S_OK, "couldn't get tag name\n");
1036         ok( tag != NULL, "tag was null\n");
1037         ok( !lstrcmpW( tag, szOpen ), "incorrect tag name\n");
1038         SysFreeString( tag );
1039
1040         IXMLDOMElement_Release( element );
1041         element = NULL;
1042     }
1043
1044     /* as soon as we call loadXML again, the document element will disappear */
1045     b = 2;
1046     r = IXMLDOMDocument_loadXML( doc, NULL, NULL );
1047     ok( r == S_FALSE, "loadXML failed\n");
1048     ok( b == 2, "variant modified\n");
1049     r = IXMLDOMDocument_get_documentElement( doc, &element );
1050     ok( r == S_FALSE, "should be no document element\n");
1051
1052     /* try to load something else simple and valid */
1053     b = VARIANT_FALSE;
1054     str = SysAllocString( szComplete3 );
1055     r = IXMLDOMDocument_loadXML( doc, str, &b );
1056     ok( r == S_OK, "loadXML failed\n");
1057     ok( b == VARIANT_TRUE, "failed to load XML string\n");
1058     SysFreeString( str );
1059
1060     /* try something a little more complicated */
1061     b = FALSE;
1062     str = SysAllocString( szComplete4 );
1063     r = IXMLDOMDocument_loadXML( doc, str, &b );
1064     ok( r == S_OK, "loadXML failed\n");
1065     ok( b == VARIANT_TRUE, "failed to load XML string\n");
1066     SysFreeString( str );
1067
1068     r = IXMLDOMDocument_get_parseError( doc, &error );
1069     ok( r == S_OK, "returns %08x\n", r );
1070
1071     r = IXMLDOMParseError_get_errorCode( error, &code );
1072     ok( r == S_FALSE, "returns %08x\n", r );
1073     ok( code == 0, "code %d\n", code );
1074     IXMLDOMParseError_Release( error );
1075
1076     /* test createTextNode */
1077     r = IXMLDOMDocument_createTextNode(doc, _bstr_(""), &nodetext);
1078     ok( r == S_OK, "returns %08x\n", r );
1079     IXMLDOMText_Release(nodetext);
1080
1081     str = SysAllocString( szOpen );
1082     r = IXMLDOMDocument_createTextNode(doc, str, NULL);
1083     ok( r == E_INVALIDARG, "returns %08x\n", r );
1084     r = IXMLDOMDocument_createTextNode(doc, str, &nodetext);
1085     ok( r == S_OK, "returns %08x\n", r );
1086     SysFreeString( str );
1087     if(nodetext)
1088     {
1089         IXMLDOMNamedNodeMap *pAttribs;
1090
1091         r = IXMLDOMText_QueryInterface(nodetext, &IID_IXMLDOMElement, (void**)&element);
1092         ok(r == E_NOINTERFACE, "ret %08x\n", r );
1093
1094         /* Text Last Child Checks */
1095         r = IXMLDOMText_get_lastChild(nodetext, NULL);
1096         ok(r == E_INVALIDARG, "ret %08x\n", r );
1097
1098         nodeChild = (IXMLDOMNode*)0x1;
1099         r = IXMLDOMText_get_lastChild(nodetext, &nodeChild);
1100         ok(r == S_FALSE, "ret %08x\n", r );
1101         ok(nodeChild == NULL, "nodeChild not NULL\n");
1102
1103         /* test get_attributes */
1104         r = IXMLDOMText_get_attributes( nodetext, NULL );
1105         ok( r == E_INVALIDARG, "get_attributes returned wrong code\n");
1106
1107         pAttribs = (IXMLDOMNamedNodeMap*)0x1;
1108         r = IXMLDOMText_get_attributes( nodetext, &pAttribs);
1109         ok(r == S_FALSE, "ret %08x\n", r );
1110         ok( pAttribs == NULL, "pAttribs not NULL\n");
1111
1112         /* test get_dataType */
1113         r = IXMLDOMText_get_dataType(nodetext, &var);
1114         ok(r == S_FALSE, "ret %08x\n", r );
1115         ok( V_VT(&var) == VT_NULL, "incorrect dataType type\n");
1116         VariantClear(&var);
1117
1118         /* test length property */
1119         r = IXMLDOMText_get_length(nodetext, NULL);
1120         ok(r == E_INVALIDARG, "ret %08x\n", r );
1121
1122         r = IXMLDOMText_get_length(nodetext, &nLength);
1123         ok(r == S_OK, "ret %08x\n", r );
1124         ok(nLength == 4, "expected 4 got %d\n", nLength);
1125
1126         /* test nodeTypeString */
1127         r = IXMLDOMText_get_nodeTypeString(nodetext, &str);
1128         ok(r == S_OK, "ret %08x\n", r );
1129         ok( !lstrcmpW( str, _bstr_("text") ), "incorrect nodeTypeString string\n");
1130         SysFreeString(str);
1131
1132         /* put data Tests */
1133         r = IXMLDOMText_put_data(nodetext, _bstr_("This &is a ; test <>\\"));
1134         ok(r == S_OK, "ret %08x\n", r );
1135
1136         /* get data Tests */
1137         r = IXMLDOMText_get_data(nodetext, &str);
1138         ok(r == S_OK, "ret %08x\n", r );
1139         ok( !lstrcmpW( str, _bstr_("This &is a ; test <>\\") ), "incorrect put_data string\n");
1140         SysFreeString(str);
1141
1142         /* Confirm XML text is good */
1143         r = IXMLDOMText_get_xml(nodetext, &str);
1144         ok(r == S_OK, "ret %08x\n", r );
1145         ok( !lstrcmpW( str, _bstr_("This &amp;is a ; test &lt;&gt;\\") ), "incorrect xml string\n");
1146         SysFreeString(str);
1147
1148         /* Confirm we get the put_data Text back */
1149         r = IXMLDOMText_get_text(nodetext, &str);
1150         ok(r == S_OK, "ret %08x\n", r );
1151         ok( !lstrcmpW( str, _bstr_("This &is a ; test <>\\") ), "incorrect xml string\n");
1152         SysFreeString(str);
1153
1154         /* test substringData */
1155         r = IXMLDOMText_substringData(nodetext, 0, 4, NULL);
1156         ok(r == E_INVALIDARG, "ret %08x\n", r );
1157
1158         /* test substringData - Invalid offset */
1159         str = (BSTR)&szElement;
1160         r = IXMLDOMText_substringData(nodetext, -1, 4, &str);
1161         ok(r == E_INVALIDARG, "ret %08x\n", r );
1162         ok( str == NULL, "incorrect string\n");
1163
1164         /* test substringData - Invalid offset */
1165         str = (BSTR)&szElement;
1166         r = IXMLDOMText_substringData(nodetext, 30, 0, &str);
1167         ok(r == S_FALSE, "ret %08x\n", r );
1168         ok( str == NULL, "incorrect string\n");
1169
1170         /* test substringData - Invalid size */
1171         str = (BSTR)&szElement;
1172         r = IXMLDOMText_substringData(nodetext, 0, -1, &str);
1173         ok(r == E_INVALIDARG, "ret %08x\n", r );
1174         ok( str == NULL, "incorrect string\n");
1175
1176         /* test substringData - Invalid size */
1177         str = (BSTR)&szElement;
1178         r = IXMLDOMText_substringData(nodetext, 2, 0, &str);
1179         ok(r == S_FALSE, "ret %08x\n", r );
1180         ok( str == NULL, "incorrect string\n");
1181
1182         /* test substringData - Start of string */
1183         r = IXMLDOMText_substringData(nodetext, 0, 4, &str);
1184         ok(r == S_OK, "ret %08x\n", r );
1185         ok( !lstrcmpW( str, _bstr_("This") ), "incorrect substringData string\n");
1186         SysFreeString(str);
1187
1188         /* test substringData - Middle of string */
1189         r = IXMLDOMText_substringData(nodetext, 13, 4, &str);
1190         ok(r == S_OK, "ret %08x\n", r );
1191         ok( !lstrcmpW( str, _bstr_("test") ), "incorrect substringData string\n");
1192         SysFreeString(str);
1193
1194         /* test substringData - End of string */
1195         r = IXMLDOMText_substringData(nodetext, 20, 4, &str);
1196         ok(r == S_OK, "ret %08x\n", r );
1197         ok( !lstrcmpW( str, _bstr_("\\") ), "incorrect substringData string\n");
1198         SysFreeString(str);
1199
1200         /* test appendData */
1201         r = IXMLDOMText_appendData(nodetext, NULL);
1202         ok(r == S_OK, "ret %08x\n", r );
1203
1204         r = IXMLDOMText_appendData(nodetext, _bstr_(""));
1205         ok(r == S_OK, "ret %08x\n", r );
1206
1207         r = IXMLDOMText_appendData(nodetext, _bstr_("Append"));
1208         ok(r == S_OK, "ret %08x\n", r );
1209
1210         r = IXMLDOMText_get_text(nodetext, &str);
1211         ok(r == S_OK, "ret %08x\n", r );
1212         ok( !lstrcmpW( str, _bstr_("This &is a ; test <>\\Append") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
1213         SysFreeString(str);
1214
1215         /* test insertData */
1216         str = SysAllocStringLen(NULL, 0);
1217         r = IXMLDOMText_insertData(nodetext, -1, str);
1218         ok(r == S_OK, "ret %08x\n", r );
1219
1220         r = IXMLDOMText_insertData(nodetext, -1, NULL);
1221         ok(r == S_OK, "ret %08x\n", r );
1222
1223         r = IXMLDOMText_insertData(nodetext, 1000, str);
1224         ok(r == S_OK, "ret %08x\n", r );
1225
1226         r = IXMLDOMText_insertData(nodetext, 1000, NULL);
1227         ok(r == S_OK, "ret %08x\n", r );
1228
1229         r = IXMLDOMText_insertData(nodetext, 0, NULL);
1230         ok(r == S_OK, "ret %08x\n", r );
1231
1232         r = IXMLDOMText_insertData(nodetext, 0, str);
1233         ok(r == S_OK, "ret %08x\n", r );
1234         SysFreeString(str);
1235
1236         r = IXMLDOMText_insertData(nodetext, -1, _bstr_("Inserting"));
1237         ok(r == E_INVALIDARG, "ret %08x\n", r );
1238
1239         r = IXMLDOMText_insertData(nodetext, 1000, _bstr_("Inserting"));
1240         ok(r == E_INVALIDARG, "ret %08x\n", r );
1241
1242         r = IXMLDOMText_insertData(nodetext, 0, _bstr_("Begin "));
1243         ok(r == S_OK, "ret %08x\n", r );
1244
1245         r = IXMLDOMText_insertData(nodetext, 17, _bstr_("Middle"));
1246         ok(r == S_OK, "ret %08x\n", r );
1247
1248         r = IXMLDOMText_insertData(nodetext, 39, _bstr_(" End"));
1249         ok(r == S_OK, "ret %08x\n", r );
1250
1251         r = IXMLDOMText_get_text(nodetext, &str);
1252         ok(r == S_OK, "ret %08x\n", r );
1253         ok( !lstrcmpW( str, _bstr_("Begin This &is a Middle; test <>\\Append End") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
1254         SysFreeString(str);
1255
1256         /* delete data */
1257         /* invalid arguments */
1258         r = IXMLDOMText_deleteData(nodetext, -1, 1);
1259         ok(r == E_INVALIDARG, "ret %08x\n", r );
1260
1261         r = IXMLDOMText_deleteData(nodetext, 0, 0);
1262         ok(r == S_OK, "ret %08x\n", r );
1263
1264         r = IXMLDOMText_deleteData(nodetext, 0, -1);
1265         ok(r == E_INVALIDARG, "ret %08x\n", r );
1266
1267         r = IXMLDOMText_get_length(nodetext, &nLength);
1268         ok(r == S_OK, "ret %08x\n", r );
1269         ok(nLength == 43, "expected 43 got %d\n", nLength);
1270
1271         r = IXMLDOMText_deleteData(nodetext, nLength, 1);
1272         ok(r == S_OK, "ret %08x\n", r );
1273
1274         r = IXMLDOMText_deleteData(nodetext, nLength+1, 1);
1275         ok(r == E_INVALIDARG, "ret %08x\n", r );
1276
1277         /* delete from start */
1278         r = IXMLDOMText_deleteData(nodetext, 0, 5);
1279         ok(r == S_OK, "ret %08x\n", r );
1280
1281         r = IXMLDOMText_get_length(nodetext, &nLength);
1282         ok(r == S_OK, "ret %08x\n", r );
1283         ok(nLength == 38, "expected 38 got %d\n", nLength);
1284
1285         r = IXMLDOMText_get_text(nodetext, &str);
1286         ok(r == S_OK, "ret %08x\n", r );
1287         ok( !lstrcmpW( str, _bstr_("This &is a Middle; test <>\\Append End") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
1288         SysFreeString(str);
1289
1290         /* delete from end */
1291         r = IXMLDOMText_deleteData(nodetext, 35, 3);
1292         ok(r == S_OK, "ret %08x\n", r );
1293
1294         r = IXMLDOMText_get_length(nodetext, &nLength);
1295         ok(r == S_OK, "ret %08x\n", r );
1296         ok(nLength == 35, "expected 35 got %d\n", nLength);
1297
1298         r = IXMLDOMText_get_text(nodetext, &str);
1299         ok(r == S_OK, "ret %08x\n", r );
1300         ok( !lstrcmpW( str, _bstr_("This &is a Middle; test <>\\Append") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
1301         SysFreeString(str);
1302
1303         /* delete from inside */
1304         r = IXMLDOMText_deleteData(nodetext, 1, 33);
1305         ok(r == S_OK, "ret %08x\n", r );
1306
1307         r = IXMLDOMText_get_length(nodetext, &nLength);
1308         ok(r == S_OK, "ret %08x\n", r );
1309         ok(nLength == 2, "expected 2 got %d\n", nLength);
1310
1311         r = IXMLDOMText_get_text(nodetext, &str);
1312         ok(r == S_OK, "ret %08x\n", r );
1313         ok( !lstrcmpW( str, _bstr_("") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
1314         SysFreeString(str);
1315
1316         /* delete whole data ... */
1317         r = IXMLDOMText_get_length(nodetext, &nLength);
1318         ok(r == S_OK, "ret %08x\n", r );
1319
1320         r = IXMLDOMText_deleteData(nodetext, 0, nLength);
1321         ok(r == S_OK, "ret %08x\n", r );
1322         /* ... and try again with empty string */
1323         r = IXMLDOMText_deleteData(nodetext, 0, nLength);
1324         ok(r == S_OK, "ret %08x\n", r );
1325
1326         /* test put_data */
1327         V_VT(&var) = VT_BSTR;
1328         V_BSTR(&var) = SysAllocString(szstr1);
1329         r = IXMLDOMText_put_nodeValue(nodetext, var);
1330         ok(r == S_OK, "ret %08x\n", r );
1331         VariantClear(&var);
1332
1333         r = IXMLDOMText_get_text(nodetext, &str);
1334         ok(r == S_OK, "ret %08x\n", r );
1335         ok( !lstrcmpW( str, szstr1 ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
1336         SysFreeString(str);
1337
1338         /* test put_data */
1339         V_VT(&var) = VT_I4;
1340         V_I4(&var) = 99;
1341         r = IXMLDOMText_put_nodeValue(nodetext, var);
1342         ok(r == S_OK, "ret %08x\n", r );
1343         VariantClear(&var);
1344
1345         r = IXMLDOMText_get_text(nodetext, &str);
1346         ok(r == S_OK, "ret %08x\n", r );
1347         ok( !lstrcmpW( str, _bstr_("99") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
1348         SysFreeString(str);
1349
1350         /* ::replaceData() */
1351         V_VT(&var) = VT_BSTR;
1352         V_BSTR(&var) = SysAllocString(szstr1);
1353         r = IXMLDOMText_put_nodeValue(nodetext, var);
1354         ok(r == S_OK, "ret %08x\n", r );
1355         VariantClear(&var);
1356
1357         r = IXMLDOMText_replaceData(nodetext, 6, 0, NULL);
1358         ok(r == E_INVALIDARG, "ret %08x\n", r );
1359         r = IXMLDOMText_get_text(nodetext, &str);
1360         ok(r == S_OK, "ret %08x\n", r );
1361         ok( !lstrcmpW( str, _bstr_("str1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
1362         SysFreeString(str);
1363
1364         r = IXMLDOMText_replaceData(nodetext, 0, 0, NULL);
1365         ok(r == S_OK, "ret %08x\n", r );
1366         r = IXMLDOMText_get_text(nodetext, &str);
1367         ok(r == S_OK, "ret %08x\n", r );
1368         ok( !lstrcmpW( str, _bstr_("str1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
1369         SysFreeString(str);
1370
1371         /* NULL pointer means delete */
1372         r = IXMLDOMText_replaceData(nodetext, 0, 1, NULL);
1373         ok(r == S_OK, "ret %08x\n", r );
1374         r = IXMLDOMText_get_text(nodetext, &str);
1375         ok(r == S_OK, "ret %08x\n", r );
1376         ok( !lstrcmpW( str, _bstr_("tr1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
1377         SysFreeString(str);
1378
1379         /* empty string means delete */
1380         r = IXMLDOMText_replaceData(nodetext, 0, 1, _bstr_(""));
1381         ok(r == S_OK, "ret %08x\n", r );
1382         r = IXMLDOMText_get_text(nodetext, &str);
1383         ok(r == S_OK, "ret %08x\n", r );
1384         ok( !lstrcmpW( str, _bstr_("r1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
1385         SysFreeString(str);
1386
1387         /* zero count means insert */
1388         r = IXMLDOMText_replaceData(nodetext, 0, 0, _bstr_("a"));
1389         ok(r == S_OK, "ret %08x\n", r );
1390         r = IXMLDOMText_get_text(nodetext, &str);
1391         ok(r == S_OK, "ret %08x\n", r );
1392         ok( !lstrcmpW( str, _bstr_("ar1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
1393         SysFreeString(str);
1394
1395         r = IXMLDOMText_replaceData(nodetext, 0, 2, NULL);
1396         ok(r == S_OK, "ret %08x\n", r );
1397
1398         r = IXMLDOMText_insertData(nodetext, 0, _bstr_("m"));
1399         ok(r == S_OK, "ret %08x\n", r );
1400         r = IXMLDOMText_get_text(nodetext, &str);
1401         ok(r == S_OK, "ret %08x\n", r );
1402         ok( !lstrcmpW( str, _bstr_("m1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
1403         SysFreeString(str);
1404
1405         /* nonempty string, count greater than its length */
1406         r = IXMLDOMText_replaceData(nodetext, 0, 2, _bstr_("a1.2"));
1407         ok(r == S_OK, "ret %08x\n", r );
1408         r = IXMLDOMText_get_text(nodetext, &str);
1409         ok(r == S_OK, "ret %08x\n", r );
1410         ok( !lstrcmpW( str, _bstr_("a1.2") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
1411         SysFreeString(str);
1412
1413         /* nonempty string, count less than its length */
1414         r = IXMLDOMText_replaceData(nodetext, 0, 1, _bstr_("wine"));
1415         ok(r == S_OK, "ret %08x\n", r );
1416         r = IXMLDOMText_get_text(nodetext, &str);
1417         ok(r == S_OK, "ret %08x\n", r );
1418         ok( !lstrcmpW( str, _bstr_("wine1.2") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
1419         SysFreeString(str);
1420
1421         IXMLDOMText_Release( nodetext );
1422     }
1423
1424     /* test Create Comment */
1425     r = IXMLDOMDocument_createComment(doc, NULL, NULL);
1426     ok( r == E_INVALIDARG, "returns %08x\n", r );
1427     node_comment = (IXMLDOMComment*)0x1;
1428
1429     /* empty comment */
1430     r = IXMLDOMDocument_createComment(doc, _bstr_(""), &node_comment);
1431     ok( r == S_OK, "returns %08x\n", r );
1432     str = (BSTR)0x1;
1433     r = IXMLDOMComment_get_data(node_comment, &str);
1434     ok( r == S_OK, "returns %08x\n", r );
1435     ok( str && SysStringLen(str) == 0, "expected empty string data\n");
1436     IXMLDOMComment_Release(node_comment);
1437     SysFreeString(str);
1438
1439     r = IXMLDOMDocument_createComment(doc, NULL, &node_comment);
1440     ok( r == S_OK, "returns %08x\n", r );
1441     str = (BSTR)0x1;
1442     r = IXMLDOMComment_get_data(node_comment, &str);
1443     ok( r == S_OK, "returns %08x\n", r );
1444     ok( str && (SysStringLen(str) == 0), "expected empty string data\n");
1445     IXMLDOMComment_Release(node_comment);
1446     SysFreeString(str);
1447
1448     str = SysAllocString(szComment);
1449     r = IXMLDOMDocument_createComment(doc, str, &node_comment);
1450     SysFreeString(str);
1451     ok( r == S_OK, "returns %08x\n", r );
1452     if(node_comment)
1453     {
1454         /* Last Child Checks */
1455         r = IXMLDOMComment_get_lastChild(node_comment, NULL);
1456         ok(r == E_INVALIDARG, "ret %08x\n", r );
1457
1458         nodeChild = (IXMLDOMNode*)0x1;
1459         r = IXMLDOMComment_get_lastChild(node_comment, &nodeChild);
1460         ok(r == S_FALSE, "ret %08x\n", r );
1461         ok(nodeChild == NULL, "pLastChild not NULL\n");
1462
1463         /* baseName */
1464         str = (BSTR)0xdeadbeef;
1465         IXMLDOMComment_get_baseName(node_comment, &str);
1466         ok(r == S_FALSE, "ret %08x\n", r );
1467         ok(str == NULL, "Expected NULL\n");
1468
1469         IXMLDOMComment_Release( node_comment );
1470     }
1471
1472     /* test Create Attribute */
1473     str = SysAllocString(szAttribute);
1474     r = IXMLDOMDocument_createAttribute(doc, NULL, NULL);
1475     ok( r == E_INVALIDARG, "returns %08x\n", r );
1476     r = IXMLDOMDocument_createAttribute(doc, str, &node_attr);
1477     ok( r == S_OK, "returns %08x\n", r );
1478     IXMLDOMText_Release( node_attr);
1479     SysFreeString(str);
1480
1481     /* test Processing Instruction */
1482     str = SysAllocStringLen(NULL, 0);
1483     r = IXMLDOMDocument_createProcessingInstruction(doc, str, str, NULL);
1484     ok( r == E_INVALIDARG, "returns %08x\n", r );
1485     r = IXMLDOMDocument_createProcessingInstruction(doc, NULL, str, &nodePI);
1486     ok( r == E_FAIL, "returns %08x\n", r );
1487     r = IXMLDOMDocument_createProcessingInstruction(doc, str, str, &nodePI);
1488     ok( r == E_FAIL, "returns %08x\n", r );
1489     SysFreeString(str);
1490
1491     r = IXMLDOMDocument_createProcessingInstruction(doc, _bstr_("xml"), _bstr_("version=\"1.0\""), &nodePI);
1492     ok( r == S_OK, "returns %08x\n", r );
1493     if(nodePI)
1494     {
1495         /* Last Child Checks */
1496         r = IXMLDOMProcessingInstruction_get_lastChild(nodePI, NULL);
1497         ok(r == E_INVALIDARG, "ret %08x\n", r );
1498
1499         nodeChild = (IXMLDOMNode*)0x1;
1500         r = IXMLDOMProcessingInstruction_get_lastChild(nodePI, &nodeChild);
1501         ok(r == S_FALSE, "ret %08x\n", r );
1502         ok(nodeChild == NULL, "nodeChild not NULL\n");
1503
1504         r = IXMLDOMProcessingInstruction_get_dataType(nodePI, &var);
1505         ok(r == S_FALSE, "ret %08x\n", r );
1506         ok( V_VT(&var) == VT_NULL, "incorrect dataType type\n");
1507         VariantClear(&var);
1508
1509         /* test nodeName */
1510         r = IXMLDOMProcessingInstruction_get_nodeName(nodePI, &str);
1511         ok(r == S_OK, "ret %08x\n", r );
1512         ok( !lstrcmpW( str, _bstr_("xml") ), "incorrect nodeName string\n");
1513         SysFreeString(str);
1514
1515         /* test baseName */
1516         str = (BSTR)0x1;
1517         r = IXMLDOMProcessingInstruction_get_baseName(nodePI, &str);
1518         ok(r == S_OK, "ret %08x\n", r );
1519         ok( !lstrcmpW( str, _bstr_("xml") ), "incorrect nodeName string\n");
1520         SysFreeString(str);
1521
1522         /* test Target */
1523         r = IXMLDOMProcessingInstruction_get_target(nodePI, &str);
1524         ok(r == S_OK, "ret %08x\n", r );
1525         ok( !lstrcmpW( str, _bstr_("xml") ), "incorrect target string\n");
1526         SysFreeString(str);
1527
1528         /* test nodeTypeString */
1529         r = IXMLDOMProcessingInstruction_get_nodeTypeString(nodePI, &str);
1530         ok(r == S_OK, "ret %08x\n", r );
1531         ok( !lstrcmpW( str, _bstr_("processinginstruction") ), "incorrect nodeTypeString string\n");
1532         SysFreeString(str);
1533
1534         /* test get_nodeValue */
1535         r = IXMLDOMProcessingInstruction_get_nodeValue(nodePI, &var);
1536         ok(r == S_OK, "ret %08x\n", r );
1537         ok( !lstrcmpW( V_BSTR(&var), _bstr_("version=\"1.0\"") ), "incorrect data string\n");
1538         VariantClear(&var);
1539
1540         /* test get_data */
1541         r = IXMLDOMProcessingInstruction_get_data(nodePI, &str);
1542         ok(r == S_OK, "ret %08x\n", r );
1543         ok( !lstrcmpW( str, _bstr_("version=\"1.0\"") ), "incorrect data string\n");
1544         SysFreeString(str);
1545
1546         /* test put_data */
1547         r = IXMLDOMProcessingInstruction_put_data(nodePI, _bstr_("version=\"1.0\" encoding=\"UTF-8\""));
1548         ok(r == E_FAIL, "ret %08x\n", r );
1549
1550         /* test put_data */
1551         V_VT(&var) = VT_BSTR;
1552         V_BSTR(&var) = SysAllocString(szOpen);  /* Doesn't matter what the string is, cannot set an xml node. */
1553         r = IXMLDOMProcessingInstruction_put_nodeValue(nodePI, var);
1554         ok(r == E_FAIL, "ret %08x\n", r );
1555         VariantClear(&var);
1556
1557         /* test get nodeName */
1558         r = IXMLDOMProcessingInstruction_get_nodeName(nodePI, &str);
1559         ok( !lstrcmpW( str, _bstr_("xml") ), "incorrect nodeName string\n");
1560         ok(r == S_OK, "ret %08x\n", r );
1561         SysFreeString(str);
1562
1563         IXMLDOMProcessingInstruction_Release(nodePI);
1564     }
1565
1566     r = IXMLDOMDocument_QueryInterface( doc, &IID_ISupportErrorInfo, (void**)&support_error );
1567     ok( r == S_OK, "ret %08x\n", r );
1568     if(r == S_OK)
1569     {
1570         r = ISupportErrorInfo_InterfaceSupportsErrorInfo( support_error, &IID_IXMLDOMDocument );
1571         todo_wine ok( r == S_OK, "ret %08x\n", r );
1572         ISupportErrorInfo_Release( support_error );
1573     }
1574
1575     r = IXMLDOMDocument_Release( doc );
1576     ok( r == 0, "document ref count incorrect\n");
1577
1578     free_bstrs();
1579 }
1580
1581 static void test_persiststreaminit(void)
1582 {
1583     IXMLDOMDocument *doc;
1584     IPersistStreamInit *streaminit;
1585     HRESULT hr;
1586
1587     doc = create_document(&IID_IXMLDOMDocument);
1588     if (!doc) return;
1589
1590     hr = IXMLDOMDocument_QueryInterface(doc, &IID_IPersistStreamInit, (void**)&streaminit);
1591     ok( hr == S_OK, "failed with 0x%08x\n", hr );
1592
1593     hr = IPersistStreamInit_InitNew(streaminit);
1594     ok( hr == S_OK, "failed with 0x%08x\n", hr );
1595
1596     IXMLDOMDocument_Release(doc);
1597 }
1598
1599 static void test_domnode( void )
1600 {
1601     HRESULT r;
1602     IXMLDOMDocument *doc, *owner = NULL;
1603     IXMLDOMElement *element = NULL;
1604     IXMLDOMNamedNodeMap *map = NULL;
1605     IXMLDOMNode *node = NULL, *next = NULL;
1606     IXMLDOMNodeList *list = NULL;
1607     IXMLDOMAttribute *attr = NULL;
1608     DOMNodeType type = NODE_INVALID;
1609     VARIANT_BOOL b;
1610     BSTR str;
1611     VARIANT var;
1612     LONG count;
1613
1614     doc = create_document(&IID_IXMLDOMDocument);
1615     if (!doc) return;
1616
1617     b = FALSE;
1618     str = SysAllocString( szComplete4 );
1619     r = IXMLDOMDocument_loadXML( doc, str, &b );
1620     ok( r == S_OK, "loadXML failed\n");
1621     ok( b == VARIANT_TRUE, "failed to load XML string\n");
1622     SysFreeString( str );
1623
1624     b = 1;
1625     r = IXMLDOMNode_hasChildNodes( doc, &b );
1626     ok( r == S_OK, "hasChildNoes bad return\n");
1627     ok( b == VARIANT_TRUE, "hasChildNoes wrong result\n");
1628
1629     r = IXMLDOMDocument_get_documentElement( doc, &element );
1630     ok( r == S_OK, "should be a document element\n");
1631     ok( element != NULL, "should be an element\n");
1632
1633     VariantInit(&var);
1634     ok( V_VT(&var) == VT_EMPTY, "variant init failed\n");
1635
1636     r = IXMLDOMNode_get_nodeValue( doc, NULL );
1637     ok(r == E_INVALIDARG, "get_nodeValue ret %08x\n", r );
1638
1639     r = IXMLDOMNode_get_nodeValue( doc, &var );
1640     ok( r == S_FALSE, "nextNode returned wrong code\n");
1641     ok( V_VT(&var) == VT_NULL, "variant wasn't empty\n");
1642     ok( V_BSTR(&var) == NULL, "variant value wasn't null\n");
1643
1644     if (element)
1645     {
1646         owner = NULL;
1647         r = IXMLDOMNode_get_ownerDocument( element, &owner );
1648         ok( r == S_OK, "get_ownerDocument return code\n");
1649         ok( owner != doc, "get_ownerDocument return\n");
1650         IXMLDOMDocument_Release(owner);
1651
1652         type = NODE_INVALID;
1653         r = IXMLDOMNode_get_nodeType( element, &type);
1654         ok( r == S_OK, "got %08x\n", r);
1655         ok( type == NODE_ELEMENT, "node not an element\n");
1656
1657         str = NULL;
1658         r = IXMLDOMNode_get_baseName( element, &str );
1659         ok( r == S_OK, "get_baseName returned wrong code\n");
1660         ok( lstrcmpW(str,szlc) == 0, "basename was wrong\n");
1661         SysFreeString(str);
1662
1663         /* check if nodename is correct */
1664         r = IXMLDOMElement_get_nodeName( element, NULL );
1665         ok ( r == E_INVALIDARG, "get_nodeName (NULL) wrong code\n");
1666
1667         /* content doesn't matter here */
1668         str = NULL;
1669         r = IXMLDOMElement_get_nodeName( element, &str );
1670         ok ( r == S_OK, "get_nodeName wrong code\n");
1671         ok ( str != NULL, "str is null\n");
1672         ok( !lstrcmpW( str, szlc ), "incorrect nodeName\n");
1673         SysFreeString( str );
1674
1675         str = SysAllocString( szNonExistentFile );      
1676         V_VT(&var) = VT_I4;
1677         V_I4(&var) = 0x1234;
1678         r = IXMLDOMElement_getAttribute( element, str, &var );
1679         ok( r == E_FAIL, "getAttribute ret %08x\n", r );
1680         ok( V_VT(&var) == VT_NULL || V_VT(&var) == VT_EMPTY, "vt = %x\n", V_VT(&var));
1681         VariantClear(&var);
1682
1683         r = IXMLDOMElement_getAttributeNode( element, str, NULL);
1684         ok( r == E_FAIL, "getAttributeNode ret %08x\n", r );
1685
1686         attr = (IXMLDOMAttribute*)0xdeadbeef;
1687         r = IXMLDOMElement_getAttributeNode( element, str, &attr);
1688         ok( r == E_FAIL, "getAttributeNode ret %08x\n", r );
1689         ok( attr == NULL, "getAttributeNode ret %p, expected NULL\n", attr );
1690         SysFreeString( str );
1691
1692         attr = (IXMLDOMAttribute*)0xdeadbeef;
1693         str = SysAllocString( szNonExistentAttribute );
1694         r = IXMLDOMElement_getAttributeNode( element, str, &attr);
1695         ok( r == S_FALSE, "getAttributeNode ret %08x\n", r );
1696         ok( attr == NULL, "getAttributeNode ret %p, expected NULL\n", attr );
1697         SysFreeString( str );
1698
1699         str = SysAllocString( szdl );   
1700         V_VT(&var) = VT_I4;
1701         V_I4(&var) = 0x1234;
1702         r = IXMLDOMElement_getAttribute( element, str, &var );
1703         ok( r == S_OK, "getAttribute ret %08x\n", r );
1704         ok( V_VT(&var) == VT_BSTR, "vt = %x\n", V_VT(&var));
1705         ok( !lstrcmpW(V_BSTR(&var), szstr1), "wrong attr value\n");
1706         VariantClear( &var );
1707
1708         r = IXMLDOMElement_getAttribute( element, NULL, &var );
1709         ok( r == E_INVALIDARG, "getAttribute ret %08x\n", r );
1710
1711         r = IXMLDOMElement_getAttribute( element, str, NULL );
1712         ok( r == E_INVALIDARG, "getAttribute ret %08x\n", r );
1713
1714         attr = NULL;
1715         r = IXMLDOMElement_getAttributeNode( element, str, &attr);
1716         ok( r == S_OK, "GetAttributeNode ret %08x\n", r );
1717         ok( attr != NULL, "getAttributeNode returned NULL\n" );
1718         if (attr)
1719         {
1720             r = IXMLDOMAttribute_get_parentNode( attr, NULL );
1721             ok( r == E_INVALIDARG, "Expected E_INVALIDARG, ret %08x\n", r );
1722
1723             /* attribute doesn't have a parent in msxml interpretation */
1724             node = (IXMLDOMNode*)0xdeadbeef;
1725             r = IXMLDOMAttribute_get_parentNode( attr, &node );
1726             ok( r == S_FALSE, "Expected S_FALSE, ret %08x\n", r );
1727             ok( node == NULL, "Expected NULL, got %p\n", node );
1728
1729             IXMLDOMAttribute_Release(attr);
1730         }
1731
1732         SysFreeString( str );
1733
1734         r = IXMLDOMElement_get_attributes( element, &map );
1735         ok( r == S_OK, "get_attributes returned wrong code\n");
1736         ok( map != NULL, "should be attributes\n");
1737
1738         b = 1;
1739         r = IXMLDOMNode_hasChildNodes( element, &b );
1740         ok( r == S_OK, "hasChildNoes bad return\n");
1741         ok( b == VARIANT_TRUE, "hasChildNoes wrong result\n");
1742     }
1743     else
1744         ok( FALSE, "no element\n");
1745
1746     if (map)
1747     {
1748         ISupportErrorInfo *support_error;
1749         r = IXMLDOMNamedNodeMap_QueryInterface( map, &IID_ISupportErrorInfo, (void**)&support_error );
1750         ok( r == S_OK, "ret %08x\n", r );
1751
1752         r = ISupportErrorInfo_InterfaceSupportsErrorInfo( support_error, &IID_IXMLDOMNamedNodeMap );
1753 todo_wine
1754 {
1755         ok( r == S_OK, "ret %08x\n", r );
1756 }
1757         ISupportErrorInfo_Release( support_error );
1758
1759         str = SysAllocString( szdl );
1760         r = IXMLDOMNamedNodeMap_getNamedItem( map, str, &node );
1761         ok( r == S_OK, "getNamedItem returned wrong code\n");
1762         ok( node != NULL, "should be attributes\n");
1763         IXMLDOMNode_Release(node);
1764         SysFreeString( str );
1765
1766         str = SysAllocString( szdl );
1767         r = IXMLDOMNamedNodeMap_getNamedItem( map, str, NULL );
1768         ok( r == E_INVALIDARG, "getNamedItem should return E_INVALIDARG\n");
1769         SysFreeString( str );
1770
1771         /* something that isn't in szComplete4 */
1772         str = SysAllocString( szOpen );
1773         node = (IXMLDOMNode *) 1;
1774         r = IXMLDOMNamedNodeMap_getNamedItem( map, str, &node );
1775         ok( r == S_FALSE, "getNamedItem found a node that wasn't there\n");
1776         ok( node == NULL, "getNamedItem should have returned NULL\n");
1777         SysFreeString( str );
1778
1779         /* test indexed access of attributes */
1780         r = IXMLDOMNamedNodeMap_get_length( map, NULL );
1781         ok ( r == E_INVALIDARG, "get_length should return E_INVALIDARG\n");
1782
1783         r = IXMLDOMNamedNodeMap_get_length( map, &count );
1784         ok ( r == S_OK, "get_length wrong code\n");
1785         ok ( count == 1, "get_length != 1\n");
1786
1787         node = NULL;
1788         r = IXMLDOMNamedNodeMap_get_item( map, -1, &node);
1789         ok ( r == S_FALSE, "get_item (-1) wrong code\n");
1790         ok ( node == NULL, "there is no node\n");
1791
1792         node = NULL;
1793         r = IXMLDOMNamedNodeMap_get_item( map, 1, &node);
1794         ok ( r == S_FALSE, "get_item (1) wrong code\n");
1795         ok ( node == NULL, "there is no attribute\n");
1796
1797         node = NULL;
1798         r = IXMLDOMNamedNodeMap_get_item( map, 0, &node);
1799         ok ( r == S_OK, "get_item (0) wrong code\n");
1800         ok ( node != NULL, "should be attribute\n");
1801
1802         r = IXMLDOMNode_get_nodeName( node, NULL );
1803         ok ( r == E_INVALIDARG, "get_nodeName (NULL) wrong code\n");
1804
1805         /* content doesn't matter here */
1806         str = NULL;
1807         r = IXMLDOMNode_get_nodeName( node, &str );
1808         ok ( r == S_OK, "get_nodeName wrong code\n");
1809         ok ( str != NULL, "str is null\n");
1810         ok( !lstrcmpW( str, szdl ), "incorrect node name\n");
1811         SysFreeString( str );
1812
1813         /* test sequential access of attributes */
1814         node = NULL;
1815         r = IXMLDOMNamedNodeMap_nextNode( map, &node );
1816         ok ( r == S_OK, "nextNode (first time) wrong code\n");
1817         ok ( node != NULL, "nextNode, should be attribute\n");
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
2981     /* the removed node has no parent anymore */
2982     r = IXMLDOMNode_get_parentNode( removed_node, &temp_node );
2983     ok( r == S_FALSE, "ret %08x\n", r);
2984     ok( temp_node == NULL, "%p\n", temp_node );
2985
2986     IXMLDOMNode_Release( removed_node );
2987     IXMLDOMNode_Release( ba_node );
2988     IXMLDOMNodeList_Release( fo_list );
2989
2990     r = IXMLDOMNodeList_get_item( root_list, 0, &lc_node );
2991     ok( r == S_OK, "ret %08x\n", r);
2992
2993     r = IXMLDOMElement_QueryInterface( lc_node, &IID_IXMLDOMElement, (void**)&lc_element );
2994     ok( r == S_OK, "ret %08x\n", r);
2995
2996     /* MS quirk: passing wrong interface pointer works, too */
2997     r = IXMLDOMElement_removeChild( element, (IXMLDOMNode*)lc_element, NULL );
2998     ok( r == S_OK, "ret %08x\n", r);
2999
3000     r = IXMLDOMNode_get_parentNode( lc_node, &temp_node );
3001     ok( r == S_FALSE, "ret %08x\n", r);
3002     ok( temp_node == NULL, "%p\n", temp_node );
3003
3004     IXMLDOMNode_Release( lc_node );
3005     IXMLDOMNodeList_Release( root_list );
3006     IXMLDOMElement_Release( element );
3007     IXMLDOMDocument_Release( doc );
3008 }
3009
3010 static void test_replaceChild(void)
3011 {
3012     HRESULT r;
3013     BSTR str;
3014     VARIANT_BOOL b;
3015     IXMLDOMDocument *doc;
3016     IXMLDOMElement *element, *ba_element;
3017     IXMLDOMNode *fo_node, *ba_node, *lc_node, *removed_node, *temp_node;
3018     IXMLDOMNodeList *root_list, *fo_list;
3019     IUnknown * unk1, *unk2;
3020     LONG len;
3021
3022     doc = create_document(&IID_IXMLDOMDocument);
3023     if (!doc) return;
3024
3025     str = SysAllocString( szComplete4 );
3026     r = IXMLDOMDocument_loadXML( doc, str, &b );
3027     ok( r == S_OK, "loadXML failed\n");
3028     ok( b == VARIANT_TRUE, "failed to load XML string\n");
3029     SysFreeString( str );
3030
3031     r = IXMLDOMDocument_get_documentElement( doc, &element );
3032     ok( r == S_OK, "ret %08x\n", r);
3033
3034     r = IXMLDOMElement_get_childNodes( element, &root_list );
3035     ok( r == S_OK, "ret %08x\n", r);
3036
3037     r = IXMLDOMNodeList_get_item( root_list, 0, &lc_node );
3038     ok( r == S_OK, "ret %08x\n", r);
3039
3040     r = IXMLDOMNodeList_get_item( root_list, 3, &fo_node );
3041     ok( r == S_OK, "ret %08x\n", r);
3042
3043     r = IXMLDOMNode_get_childNodes( fo_node, &fo_list );
3044     ok( r == S_OK, "ret %08x\n", r);
3045
3046     r = IXMLDOMNodeList_get_item( fo_list, 0, &ba_node );
3047     ok( r == S_OK, "ret %08x\n", r);
3048
3049     IXMLDOMNodeList_Release( fo_list );
3050
3051     /* invalid parameter: NULL ptr for element to remove */
3052     removed_node = (void*)0xdeadbeef;
3053     r = IXMLDOMElement_replaceChild( element, ba_node, NULL, &removed_node );
3054     ok( r == E_INVALIDARG, "ret %08x\n", r );
3055     ok( removed_node == (void*)0xdeadbeef, "%p\n", removed_node );
3056
3057     /* invalid parameter: NULL for replacement element. (Sic!) */
3058     removed_node = (void*)0xdeadbeef;
3059     r = IXMLDOMElement_replaceChild( element, NULL, fo_node, &removed_node );
3060     ok( r == E_INVALIDARG, "ret %08x\n", r );
3061     ok( removed_node == (void*)0xdeadbeef, "%p\n", removed_node );
3062
3063     /* invalid parameter: OldNode is not a child */
3064     removed_node = (void*)0xdeadbeef;
3065     r = IXMLDOMElement_replaceChild( element, lc_node, ba_node, &removed_node );
3066     ok( r == E_INVALIDARG, "ret %08x\n", r );
3067     ok( removed_node == NULL, "%p\n", removed_node );
3068
3069     /* invalid parameter: would create loop */
3070     removed_node = (void*)0xdeadbeef;
3071     r = IXMLDOMNode_replaceChild( fo_node, fo_node, ba_node, &removed_node );
3072     ok( r == E_FAIL, "ret %08x\n", r );
3073     ok( removed_node == NULL, "%p\n", removed_node );
3074
3075     r = IXMLDOMElement_replaceChild( element, ba_node, fo_node, NULL );
3076     ok( r == S_OK, "ret %08x\n", r );
3077
3078     r = IXMLDOMNodeList_get_item( root_list, 3, &temp_node );
3079     ok( r == S_OK, "ret %08x\n", r );
3080
3081     /* ba_node and temp_node refer to the same node, yet they
3082        are different interface pointers */
3083     ok( ba_node != temp_node, "ba_node %p temp_node %p\n", ba_node, temp_node);
3084     r = IXMLDOMNode_QueryInterface( temp_node, &IID_IUnknown, (void**)&unk1);
3085     ok( r == S_OK, "ret %08x\n", r );
3086     r = IXMLDOMNode_QueryInterface( ba_node, &IID_IUnknown, (void**)&unk2);
3087     ok( r == S_OK, "ret %08x\n", r );
3088     todo_wine ok( unk1 == unk2, "unk1 %p unk2 %p\n", unk1, unk2);
3089
3090     IUnknown_Release( unk1 );
3091     IUnknown_Release( unk2 );
3092
3093     /* ba_node should have been removed from below fo_node */
3094     r = IXMLDOMNode_get_childNodes( fo_node, &fo_list );
3095     ok( r == S_OK, "ret %08x\n", r );
3096
3097     /* MS quirk: replaceChild also accepts elements instead of nodes */
3098     r = IXMLDOMNode_QueryInterface( ba_node, &IID_IXMLDOMElement, (void**)&ba_element);
3099     ok( r == S_OK, "ret %08x\n", r );
3100
3101     r = IXMLDOMElement_replaceChild( element, ba_node, (IXMLDOMNode*)ba_element, &removed_node );
3102     ok( r == S_OK, "ret %08x\n", r );
3103
3104     r = IXMLDOMNodeList_get_length( fo_list, &len);
3105     ok( r == S_OK, "ret %08x\n", r );
3106     ok( len == 0, "len %d\n", len);
3107
3108     IXMLDOMNodeList_Release( fo_list );
3109
3110     IXMLDOMNode_Release(ba_node);
3111     IXMLDOMNode_Release(fo_node);
3112     IXMLDOMNode_Release(temp_node);
3113     IXMLDOMNodeList_Release( root_list );
3114     IXMLDOMElement_Release( element );
3115     IXMLDOMDocument_Release( doc );
3116 }
3117
3118 static void test_removeNamedItem(void)
3119 {
3120     IXMLDOMDocument *doc;
3121     IXMLDOMElement *element;
3122     IXMLDOMNode *pr_node, *removed_node, *removed_node2;
3123     IXMLDOMNodeList *root_list;
3124     IXMLDOMNamedNodeMap * pr_attrs;
3125     VARIANT_BOOL b;
3126     BSTR str;
3127     LONG len;
3128     HRESULT r;
3129
3130     doc = create_document(&IID_IXMLDOMDocument);
3131     if (!doc) return;
3132
3133     str = SysAllocString( szComplete4 );
3134     r = IXMLDOMDocument_loadXML( doc, str, &b );
3135     ok( r == S_OK, "loadXML failed\n");
3136     ok( b == VARIANT_TRUE, "failed to load XML string\n");
3137     SysFreeString( str );
3138
3139     r = IXMLDOMDocument_get_documentElement( doc, &element );
3140     ok( r == S_OK, "ret %08x\n", r);
3141
3142     r = IXMLDOMElement_get_childNodes( element, &root_list );
3143     ok( r == S_OK, "ret %08x\n", r);
3144
3145     r = IXMLDOMNodeList_get_item( root_list, 1, &pr_node );
3146     ok( r == S_OK, "ret %08x\n", r);
3147
3148     r = IXMLDOMNode_get_attributes( pr_node, &pr_attrs );
3149     ok( r == S_OK, "ret %08x\n", r);
3150
3151     r = IXMLDOMNamedNodeMap_get_length( pr_attrs, &len );
3152     ok( r == S_OK, "ret %08x\n", r);
3153     ok( len == 3, "length %d\n", len);
3154
3155     removed_node = (void*)0xdeadbeef;
3156     r = IXMLDOMNamedNodeMap_removeNamedItem( pr_attrs, NULL, &removed_node);
3157     ok ( r == E_INVALIDARG, "ret %08x\n", r);
3158     ok ( removed_node == (void*)0xdeadbeef, "got %p\n", removed_node);
3159
3160     removed_node = (void*)0xdeadbeef;
3161     str = SysAllocString(szvr);
3162     r = IXMLDOMNamedNodeMap_removeNamedItem( pr_attrs, str, &removed_node);
3163     ok ( r == S_OK, "ret %08x\n", r);
3164
3165     removed_node2 = (void*)0xdeadbeef;
3166     r = IXMLDOMNamedNodeMap_removeNamedItem( pr_attrs, str, &removed_node2);
3167     ok ( r == S_FALSE, "ret %08x\n", r);
3168     ok ( removed_node2 == NULL, "got %p\n", removed_node2 );
3169
3170     r = IXMLDOMNamedNodeMap_get_length( pr_attrs, &len );
3171     ok( r == S_OK, "ret %08x\n", r);
3172     ok( len == 2, "length %d\n", len);
3173
3174     r = IXMLDOMNamedNodeMap_setNamedItem( pr_attrs, removed_node, NULL);
3175     ok ( r == S_OK, "ret %08x\n", r);
3176     IXMLDOMNode_Release(removed_node);
3177
3178     r = IXMLDOMNamedNodeMap_get_length( pr_attrs, &len );
3179     ok( r == S_OK, "ret %08x\n", r);
3180     ok( len == 3, "length %d\n", len);
3181
3182     r = IXMLDOMNamedNodeMap_removeNamedItem( pr_attrs, str, NULL);
3183     ok ( r == S_OK, "ret %08x\n", r);
3184
3185     r = IXMLDOMNamedNodeMap_get_length( pr_attrs, &len );
3186     ok( r == S_OK, "ret %08x\n", r);
3187     ok( len == 2, "length %d\n", len);
3188
3189     r = IXMLDOMNamedNodeMap_removeNamedItem( pr_attrs, str, NULL);
3190     ok ( r == S_FALSE, "ret %08x\n", r);
3191
3192     SysFreeString(str);
3193
3194     IXMLDOMNamedNodeMap_Release( pr_attrs );
3195     IXMLDOMNode_Release( pr_node );
3196     IXMLDOMNodeList_Release( root_list );
3197     IXMLDOMElement_Release( element );
3198     IXMLDOMDocument_Release( doc );
3199 }
3200
3201 static void test_XMLHTTP(void)
3202 {
3203     static const WCHAR wszBody[] = {'m','o','d','e','=','T','e','s','t',0};
3204     static const WCHAR wszPOST[] = {'P','O','S','T',0};
3205     static const WCHAR wszUrl[] = {'h','t','t','p',':','/','/',
3206         'c','r','o','s','s','o','v','e','r','.','c','o','d','e','w','e','a','v','e','r','s','.','c','o','m','/',
3207         'p','o','s','t','t','e','s','t','.','p','h','p',0};
3208     static const WCHAR xmltestW[] = {'h','t','t','p',':','/','/',
3209         'c','r','o','s','s','o','v','e','r','.','c','o','d','e','w','e','a','v','e','r','s','.','c','o','m','/',
3210         'x','m','l','t','e','s','t','.','x','m','l',0};
3211     static const WCHAR wszExpectedResponse[] = {'F','A','I','L','E','D',0};
3212     static const CHAR xmltestbodyA[] = "<?xml version=\"1.0\" encoding=\"utf-8\"?>\n<a>TEST</a>\n";
3213
3214     IXMLHttpRequest *pXMLHttpRequest;
3215     BSTR bstrResponse, method, url;
3216     VARIANT dummy;
3217     VARIANT async;
3218     VARIANT varbody;
3219     LONG state, status, ref, bound;
3220     void *ptr;
3221     IDispatch *event;
3222     HRESULT hr = CoCreateInstance(&CLSID_XMLHTTPRequest, NULL,
3223                                   CLSCTX_INPROC_SERVER, &IID_IXMLHttpRequest,
3224                                   (void **)&pXMLHttpRequest);
3225     if (FAILED(hr))
3226     {
3227         win_skip("IXMLHTTPRequest is not available (0x%08x)\n", hr);
3228         return;
3229     }
3230
3231     VariantInit(&dummy);
3232     V_VT(&dummy) = VT_ERROR;
3233     V_ERROR(&dummy) = DISP_E_MEMBERNOTFOUND;
3234     VariantInit(&async);
3235     V_VT(&async) = VT_BOOL;
3236     V_BOOL(&async) = VARIANT_FALSE;
3237     V_VT(&varbody) = VT_BSTR;
3238     V_BSTR(&varbody) = SysAllocString(wszBody);
3239
3240     method = SysAllocString(wszPOST);
3241     url = SysAllocString(wszUrl);
3242
3243 if (0)
3244 {
3245     /* crashes on win98 */
3246     hr = IXMLHttpRequest_put_onreadystatechange(pXMLHttpRequest, NULL);
3247     ok(hr == S_OK, "got 0x%08x\n", hr);
3248 }
3249
3250     hr = IXMLHttpRequest_abort(pXMLHttpRequest);
3251     ok(hr == S_OK, "got 0x%08x\n", hr);
3252
3253     /* send before open */
3254     hr = IXMLHttpRequest_send(pXMLHttpRequest, dummy);
3255     ok(hr == E_FAIL || broken(hr == E_UNEXPECTED) /* win9x, win2k */, "got 0x%08x\n", hr);
3256
3257     /* initial status code */
3258     hr = IXMLHttpRequest_get_status(pXMLHttpRequest, NULL);
3259     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
3260
3261     status = 0xdeadbeef;
3262     hr = IXMLHttpRequest_get_status(pXMLHttpRequest, &status);
3263     ok(hr == E_FAIL || broken(hr == E_UNEXPECTED) /* win9x, win2k */, "got 0x%08x\n", hr);
3264     ok(status == 0xdeadbeef, "got %d\n", status);
3265
3266     /* invalid parameters */
3267     hr = IXMLHttpRequest_open(pXMLHttpRequest, NULL, NULL, async, dummy, dummy);
3268     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
3269
3270     hr = IXMLHttpRequest_open(pXMLHttpRequest, method, NULL, async, dummy, dummy);
3271     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
3272
3273     hr = IXMLHttpRequest_open(pXMLHttpRequest, NULL, url, async, dummy, dummy);
3274     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
3275
3276     hr = IXMLHttpRequest_setRequestHeader(pXMLHttpRequest, NULL, NULL);
3277     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
3278
3279     hr = IXMLHttpRequest_setRequestHeader(pXMLHttpRequest, _bstr_("header1"), NULL);
3280     ok(hr == E_FAIL || broken(hr == E_UNEXPECTED) /* win9x, win2k */, "got 0x%08x\n", hr);
3281
3282     hr = IXMLHttpRequest_setRequestHeader(pXMLHttpRequest, NULL, _bstr_("value1"));
3283     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
3284
3285     hr = IXMLHttpRequest_setRequestHeader(pXMLHttpRequest, _bstr_("header1"), _bstr_("value1"));
3286     ok(hr == E_FAIL || broken(hr == E_UNEXPECTED) /* win9x, win2k */, "got 0x%08x\n", hr);
3287
3288     hr = IXMLHttpRequest_get_readyState(pXMLHttpRequest, NULL);
3289     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
3290
3291     state = -1;
3292     hr = IXMLHttpRequest_get_readyState(pXMLHttpRequest, &state);
3293     ok(hr == S_OK, "got 0x%08x\n", hr);
3294     ok(state == READYSTATE_UNINITIALIZED, "got %d, expected READYSTATE_UNINITIALIZED\n", state);
3295
3296     event = create_dispevent();
3297     ref = IDispatch_AddRef(event);
3298     ok(ref == 2, "got %d\n", ref);
3299     IDispatch_Release(event);
3300
3301     hr = IXMLHttpRequest_put_onreadystatechange(pXMLHttpRequest, event);
3302     ok(hr == S_OK, "got 0x%08x\n", hr);
3303
3304     ref = IDispatch_AddRef(event);
3305     ok(ref == 3, "got %d\n", ref);
3306     IDispatch_Release(event);
3307
3308     g_unexpectedcall = g_expectedcall = 0;
3309
3310     hr = IXMLHttpRequest_open(pXMLHttpRequest, method, url, async, dummy, dummy);
3311     ok(hr == S_OK, "got 0x%08x\n", hr);
3312
3313     ok(g_unexpectedcall == 0, "unexpected disp event call\n");
3314     ok(g_expectedcall == 1 || broken(g_expectedcall == 0) /* win2k */, "no expected disp event call\n");
3315
3316     /* status code after ::open() */
3317     status = 0xdeadbeef;
3318     hr = IXMLHttpRequest_get_status(pXMLHttpRequest, &status);
3319     ok(hr == E_FAIL || broken(hr == E_UNEXPECTED) /* win9x, win2k */, "got 0x%08x\n", hr);
3320     ok(status == 0xdeadbeef, "got %d\n", status);
3321
3322     state = -1;
3323     hr = IXMLHttpRequest_get_readyState(pXMLHttpRequest, &state);
3324     ok(hr == S_OK, "got 0x%08x\n", hr);
3325     ok(state == READYSTATE_LOADING, "got %d, expected READYSTATE_LOADING\n", state);
3326
3327     hr = IXMLHttpRequest_abort(pXMLHttpRequest);
3328     ok(hr == S_OK, "got 0x%08x\n", hr);
3329
3330     state = -1;
3331     hr = IXMLHttpRequest_get_readyState(pXMLHttpRequest, &state);
3332     ok(hr == S_OK, "got 0x%08x\n", hr);
3333     ok(state == READYSTATE_UNINITIALIZED || broken(state == READYSTATE_LOADING) /* win98, win2k */,
3334         "got %d, expected READYSTATE_UNINITIALIZED\n", state);
3335
3336     hr = IXMLHttpRequest_open(pXMLHttpRequest, method, url, async, dummy, dummy);
3337     ok(hr == S_OK, "got 0x%08x\n", hr);
3338
3339     hr = IXMLHttpRequest_setRequestHeader(pXMLHttpRequest, _bstr_("header1"), _bstr_("value1"));
3340     ok(hr == S_OK, "got 0x%08x\n", hr);
3341
3342     hr = IXMLHttpRequest_setRequestHeader(pXMLHttpRequest, NULL, _bstr_("value1"));
3343     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
3344
3345     hr = IXMLHttpRequest_setRequestHeader(pXMLHttpRequest, _bstr_(""), _bstr_("value1"));
3346     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
3347
3348     SysFreeString(method);
3349     SysFreeString(url);
3350
3351     hr = IXMLHttpRequest_send(pXMLHttpRequest, varbody);
3352     if (hr == INET_E_RESOURCE_NOT_FOUND)
3353     {
3354         skip("No connection could be made with crossover.codeweavers.com\n");
3355         IXMLHttpRequest_Release(pXMLHttpRequest);
3356         return;
3357     }
3358     ok(hr == S_OK, "got 0x%08x\n", hr);
3359
3360     /* status code after ::send() */
3361     status = 0xdeadbeef;
3362     hr = IXMLHttpRequest_get_status(pXMLHttpRequest, &status);
3363     ok(hr == S_OK, "got 0x%08x\n", hr);
3364     ok(status == 200, "got %d\n", status);
3365
3366     /* another ::send() after completed request */
3367     hr = IXMLHttpRequest_send(pXMLHttpRequest, varbody);
3368     ok(hr == E_FAIL || broken(hr == E_UNEXPECTED) /* win9x, win2k */, "got 0x%08x\n", hr);
3369
3370     VariantClear(&varbody);
3371
3372     hr = IXMLHttpRequest_get_responseText(pXMLHttpRequest, &bstrResponse);
3373     ok(hr == S_OK, "got 0x%08x\n", hr);
3374     /* the server currently returns "FAILED" because the Content-Type header is
3375      * not what the server expects */
3376     if(hr == S_OK)
3377     {
3378         ok(!memcmp(bstrResponse, wszExpectedResponse, sizeof(wszExpectedResponse)),
3379             "expected %s, got %s\n", wine_dbgstr_w(wszExpectedResponse), wine_dbgstr_w(bstrResponse));
3380         SysFreeString(bstrResponse);
3381     }
3382
3383     /* GET request */
3384     url = SysAllocString(xmltestW);
3385
3386     hr = IXMLHttpRequest_open(pXMLHttpRequest, _bstr_("GET"), url, async, dummy, dummy);
3387     ok(hr == S_OK, "got 0x%08x\n", hr);
3388
3389     V_VT(&varbody) = VT_EMPTY;
3390
3391     hr = IXMLHttpRequest_send(pXMLHttpRequest, varbody);
3392     if (hr == INET_E_RESOURCE_NOT_FOUND)
3393     {
3394         skip("No connection could be made with crossover.codeweavers.com\n");
3395         IXMLHttpRequest_Release(pXMLHttpRequest);
3396         return;
3397     }
3398     ok(hr == S_OK, "got 0x%08x\n", hr);
3399
3400     hr = IXMLHttpRequest_get_responseText(pXMLHttpRequest, NULL);
3401     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
3402
3403     hr = IXMLHttpRequest_get_responseText(pXMLHttpRequest, &bstrResponse);
3404     ok(hr == S_OK, "got 0x%08x\n", hr);
3405     if(hr == S_OK)
3406     {
3407         ok(!memcmp(bstrResponse, _bstr_(xmltestbodyA), sizeof(xmltestbodyA)*sizeof(WCHAR)),
3408             "expected %s, got %s\n", xmltestbodyA, wine_dbgstr_w(bstrResponse));
3409         SysFreeString(bstrResponse);
3410     }
3411
3412     hr = IXMLHttpRequest_get_responseBody(pXMLHttpRequest, NULL);
3413     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
3414
3415     V_VT(&varbody) = VT_EMPTY;
3416     hr = IXMLHttpRequest_get_responseBody(pXMLHttpRequest, &varbody);
3417     ok(hr == S_OK, "got 0x%08x\n", hr);
3418     ok(V_VT(&varbody) == (VT_ARRAY|VT_UI1), "got type %d, expected %d\n", V_VT(&varbody), VT_ARRAY|VT_UI1);
3419     ok(SafeArrayGetDim(V_ARRAY(&varbody)) == 1, "got %d, expected one dimension\n", SafeArrayGetDim(V_ARRAY(&varbody)));
3420
3421     bound = -1;
3422     hr = SafeArrayGetLBound(V_ARRAY(&varbody), 1, &bound);
3423     ok(hr == S_OK, "got 0x%08x\n", hr);
3424     ok(bound == 0, "got %d, expected zero bound\n", bound);
3425
3426     hr = SafeArrayAccessData(V_ARRAY(&varbody), &ptr);
3427     ok(hr == S_OK, "got 0x%08x\n", hr);
3428     ok(memcmp(ptr, xmltestbodyA, sizeof(xmltestbodyA)-1) == 0, "got wrond body data\n");
3429     SafeArrayUnaccessData(V_ARRAY(&varbody));
3430
3431     VariantClear(&varbody);
3432
3433     SysFreeString(url);
3434
3435     IDispatch_Release(event);
3436     IXMLHttpRequest_Release(pXMLHttpRequest);
3437     free_bstrs();
3438 }
3439
3440 static void test_IXMLDOMDocument2(void)
3441 {
3442     static const WCHAR emptyW[] = {0};
3443     IXMLDOMDocument2 *doc2, *dtddoc2;
3444     IXMLDOMDocument *doc;
3445     IXMLDOMParseError* err;
3446     IDispatchEx *dispex;
3447     VARIANT_BOOL b;
3448     VARIANT var;
3449     HRESULT r;
3450     LONG ref, res;
3451     BSTR str;
3452
3453     doc = create_document(&IID_IXMLDOMDocument);
3454     if (!doc) return;
3455
3456     dtddoc2 = create_document(&IID_IXMLDOMDocument2);
3457     if (!dtddoc2)
3458     {
3459         IXMLDOMDocument_Release(doc);
3460         return;
3461     }
3462
3463     r = IXMLDOMDocument_QueryInterface( doc, &IID_IXMLDOMDocument2, (void**)&doc2 );
3464     ok( r == S_OK, "ret %08x\n", r );
3465     ok( doc == (IXMLDOMDocument*)doc2, "interfaces differ\n");
3466
3467     ole_expect(IXMLDOMDocument2_get_readyState(doc2, NULL), E_INVALIDARG);
3468     ole_check(IXMLDOMDocument2_get_readyState(doc2, &res));
3469     ok(res == READYSTATE_COMPLETE, "expected READYSTATE_COMPLETE (4), got %i\n", res);
3470
3471     err = NULL;
3472     ole_expect(IXMLDOMDocument2_validate(doc2, NULL), S_FALSE);
3473     ole_expect(IXMLDOMDocument2_validate(doc2, &err), S_FALSE);
3474     ok(err != NULL, "expected a pointer\n");
3475     if (err)
3476     {
3477         res = 0;
3478         ole_check(IXMLDOMParseError_get_errorCode(err, &res));
3479         /* XML_E_NOTWF */
3480         ok(res == E_XML_NOTWF, "got %08x\n", res);
3481         IXMLDOMParseError_Release(err);
3482     }
3483
3484     str = SysAllocString( szComplete4 );
3485     r = IXMLDOMDocument_loadXML( doc2, str, &b );
3486     ok( r == S_OK, "loadXML failed\n");
3487     ok( b == VARIANT_TRUE, "failed to load XML string\n");
3488     SysFreeString( str );
3489
3490     ole_check(IXMLDOMDocument2_get_readyState(doc, &res));
3491     ok(res == READYSTATE_COMPLETE, "expected READYSTATE_COMPLETE (4), got %i\n", res);
3492
3493     err = NULL;
3494     ole_expect(IXMLDOMDocument2_validate(doc2, &err), S_FALSE);
3495     ok(err != NULL, "expected a pointer\n");
3496     if (err)
3497     {
3498         res = 0;
3499         ole_check(IXMLDOMParseError_get_errorCode(err, &res));
3500         /* XML_E_NODTD */
3501         ok(res == E_XML_NODTD, "got %08x\n", res);
3502         IXMLDOMParseError_Release(err);
3503     }
3504
3505     r = IXMLDOMDocument_QueryInterface( doc, &IID_IDispatchEx, (void**)&dispex );
3506     ok( r == S_OK, "ret %08x\n", r );
3507     if(r == S_OK)
3508     {
3509         IDispatchEx_Release(dispex);
3510     }
3511
3512     /* we will check if the variant got cleared */
3513     ref = IXMLDOMDocument2_AddRef(doc2);
3514     expect_eq(ref, 3, int, "%d");  /* doc, doc2, AddRef*/
3515     V_VT(&var) = VT_UNKNOWN;
3516     V_UNKNOWN(&var) = (IUnknown *)doc2;
3517
3518     /* invalid calls */
3519     ole_expect(IXMLDOMDocument2_getProperty(doc2, _bstr_("askldhfaklsdf"), &var), E_FAIL);
3520     expect_eq(V_VT(&var), VT_UNKNOWN, int, "%x");
3521     ole_expect(IXMLDOMDocument2_getProperty(doc2, _bstr_("SelectionLanguage"), NULL), E_INVALIDARG);
3522
3523     /* valid call */
3524     ole_check(IXMLDOMDocument2_getProperty(doc2, _bstr_("SelectionLanguage"), &var));
3525     expect_eq(V_VT(&var), VT_BSTR, int, "%x");
3526     expect_bstr_eq_and_free(V_BSTR(&var), "XSLPattern");
3527     V_VT(&var) = VT_R4;
3528
3529     /* the variant didn't get cleared*/
3530     expect_eq(IXMLDOMDocument2_Release(doc2), 2, int, "%d");
3531
3532     /* setProperty tests */
3533     ole_expect(IXMLDOMDocument2_setProperty(doc2, _bstr_("askldhfaklsdf"), var), E_FAIL);
3534     ole_expect(IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionLanguage"), var), E_FAIL);
3535     ole_expect(IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionLanguage"), _variantbstr_("alskjdh faklsjd hfk")), E_FAIL);
3536     ole_check(IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionLanguage"), _variantbstr_("XSLPattern")));
3537     ole_check(IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionLanguage"), _variantbstr_("XPath")));
3538     ole_check(IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionLanguage"), _variantbstr_("XSLPattern")));
3539
3540     V_VT(&var) = VT_BSTR;
3541     V_BSTR(&var) = SysAllocString(emptyW);
3542     r = IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionNamespaces"), var);
3543     ok(r == S_OK, "got 0x%08x\n", r);
3544     VariantClear(&var);
3545
3546     V_VT(&var) = VT_I2;
3547     V_I2(&var) = 0;
3548     r = IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionNamespaces"), var);
3549     ok(r == E_FAIL, "got 0x%08x\n", r);
3550
3551     /* contrary to what MSDN claims you can switch back from XPath to XSLPattern */
3552     ole_check(IXMLDOMDocument2_getProperty(doc2, _bstr_("SelectionLanguage"), &var));
3553     expect_eq(V_VT(&var), VT_BSTR, int, "%x");
3554     expect_bstr_eq_and_free(V_BSTR(&var), "XSLPattern");
3555
3556     IXMLDOMDocument2_Release( doc2 );
3557     IXMLDOMDocument_Release( doc );
3558
3559     /* DTD validation */
3560     ole_check(IXMLDOMDocument2_put_validateOnParse(dtddoc2, VARIANT_FALSE));
3561     ole_check(IXMLDOMDocument2_loadXML(dtddoc2, _bstr_(szEmailXML), &b));
3562     ok( b == VARIANT_TRUE, "failed to load XML string\n");
3563     err = NULL;
3564     ole_check(IXMLDOMDocument2_validate(dtddoc2, &err));
3565     ok(err != NULL, "expected pointer\n");
3566     if (err)
3567     {
3568         res = 0;
3569         ole_expect(IXMLDOMParseError_get_errorCode(err, &res), S_FALSE);
3570         ok(res == 0, "got %08x\n", res);
3571         IXMLDOMParseError_Release(err);
3572     }
3573
3574     ole_check(IXMLDOMDocument2_loadXML(dtddoc2, _bstr_(szEmailXML_0D), &b));
3575     ok( b == VARIANT_TRUE, "failed to load XML string\n");
3576     err = NULL;
3577     ole_expect(IXMLDOMDocument2_validate(dtddoc2, &err), S_FALSE);
3578     ok(err != NULL, "expected pointer\n");
3579     if (err)
3580     {
3581         res = 0;
3582         ole_check(IXMLDOMParseError_get_errorCode(err, &res));
3583         /* XML_ELEMENT_UNDECLARED */
3584         todo_wine ok(res == 0xC00CE00D, "got %08x\n", res);
3585         IXMLDOMParseError_Release(err);
3586     }
3587
3588     ole_check(IXMLDOMDocument2_loadXML(dtddoc2, _bstr_(szEmailXML_0E), &b));
3589     ok( b == VARIANT_TRUE, "failed to load XML string\n");
3590     err = NULL;
3591     ole_expect(IXMLDOMDocument2_validate(dtddoc2, &err), S_FALSE);
3592     ok(err != NULL, "expected pointer\n");
3593     if (err)
3594     {
3595         res = 0;
3596         ole_check(IXMLDOMParseError_get_errorCode(err, &res));
3597         /* XML_ELEMENT_ID_NOT_FOUND */
3598         todo_wine ok(res == 0xC00CE00E, "got %08x\n", res);
3599         IXMLDOMParseError_Release(err);
3600     }
3601
3602     ole_check(IXMLDOMDocument2_loadXML(dtddoc2, _bstr_(szEmailXML_11), &b));
3603     ok( b == VARIANT_TRUE, "failed to load XML string\n");
3604     err = NULL;
3605     ole_expect(IXMLDOMDocument2_validate(dtddoc2, &err), S_FALSE);
3606     ok(err != NULL, "expected pointer\n");
3607     if (err)
3608     {
3609         res = 0;
3610         ole_check(IXMLDOMParseError_get_errorCode(err, &res));
3611         /* XML_EMPTY_NOT_ALLOWED */
3612         todo_wine ok(res == 0xC00CE011, "got %08x\n", res);
3613         IXMLDOMParseError_Release(err);
3614     }
3615
3616     ole_check(IXMLDOMDocument2_loadXML(dtddoc2, _bstr_(szEmailXML_13), &b));
3617     ok( b == VARIANT_TRUE, "failed to load XML string\n");
3618     err = NULL;
3619     ole_expect(IXMLDOMDocument2_validate(dtddoc2, &err), S_FALSE);
3620     ok(err != NULL, "expected pointer\n");
3621     if (err)
3622     {
3623         res = 0;
3624         ole_check(IXMLDOMParseError_get_errorCode(err, &res));
3625         /* XML_ROOT_NAME_MISMATCH */
3626         todo_wine ok(res == 0xC00CE013, "got %08x\n", res);
3627         IXMLDOMParseError_Release(err);
3628     }
3629
3630     ole_check(IXMLDOMDocument2_loadXML(dtddoc2, _bstr_(szEmailXML_14), &b));
3631     ok( b == VARIANT_TRUE, "failed to load XML string\n");
3632     err = NULL;
3633     ole_expect(IXMLDOMDocument2_validate(dtddoc2, &err), S_FALSE);
3634     ok(err != NULL, "expected pointer\n");
3635     if (err)
3636     {
3637         res = 0;
3638         ole_check(IXMLDOMParseError_get_errorCode(err, &res));
3639         /* XML_INVALID_CONTENT */
3640         todo_wine ok(res == 0xC00CE014, "got %08x\n", res);
3641         IXMLDOMParseError_Release(err);
3642     }
3643
3644     ole_check(IXMLDOMDocument2_loadXML(dtddoc2, _bstr_(szEmailXML_15), &b));
3645     ok( b == VARIANT_TRUE, "failed to load XML string\n");
3646     err = NULL;
3647     ole_expect(IXMLDOMDocument2_validate(dtddoc2, &err), S_FALSE);
3648     ok(err != NULL, "expected pointer\n");
3649     if (err)
3650     {
3651         res = 0;
3652         ole_check(IXMLDOMParseError_get_errorCode(err, &res));
3653         /* XML_ATTRIBUTE_NOT_DEFINED */
3654         todo_wine ok(res == 0xC00CE015, "got %08x\n", res);
3655         IXMLDOMParseError_Release(err);
3656     }
3657
3658     ole_check(IXMLDOMDocument2_loadXML(dtddoc2, _bstr_(szEmailXML_16), &b));
3659     ok( b == VARIANT_TRUE, "failed to load XML string\n");
3660     err = NULL;
3661     ole_expect(IXMLDOMDocument2_validate(dtddoc2, &err), S_FALSE);
3662     ok(err != NULL, "expected pointer\n");
3663     if (err)
3664     {
3665         res = 0;
3666         ole_check(IXMLDOMParseError_get_errorCode(err, &res));
3667         /* XML_ATTRIBUTE_FIXED */
3668         todo_wine ok(res == 0xC00CE016, "got %08x\n", res);
3669         IXMLDOMParseError_Release(err);
3670     }
3671
3672     ole_check(IXMLDOMDocument2_loadXML(dtddoc2, _bstr_(szEmailXML_17), &b));
3673     ok( b == VARIANT_TRUE, "failed to load XML string\n");
3674     err = NULL;
3675     ole_expect(IXMLDOMDocument2_validate(dtddoc2, &err), S_FALSE);
3676     ok(err != NULL, "expected pointer\n");
3677     if (err)
3678     {
3679         res = 0;
3680         ole_check(IXMLDOMParseError_get_errorCode(err, &res));
3681         /* XML_ATTRIBUTE_VALUE */
3682         todo_wine ok(res == 0xC00CE017, "got %08x\n", res);
3683         IXMLDOMParseError_Release(err);
3684     }
3685
3686     ole_check(IXMLDOMDocument2_loadXML(dtddoc2, _bstr_(szEmailXML_18), &b));
3687     ok( b == VARIANT_TRUE, "failed to load XML string\n");
3688     err = NULL;
3689     ole_expect(IXMLDOMDocument2_validate(dtddoc2, &err), S_FALSE);
3690     ok(err != NULL, "expected pointer\n");
3691     if (err)
3692     {
3693         res = 0;
3694         ole_check(IXMLDOMParseError_get_errorCode(err, &res));
3695         /* XML_ILLEGAL_TEXT */
3696         todo_wine ok(res == 0xC00CE018, "got %08x\n", res);
3697         IXMLDOMParseError_Release(err);
3698     }
3699
3700     ole_check(IXMLDOMDocument2_loadXML(dtddoc2, _bstr_(szEmailXML_20), &b));
3701     ok( b == VARIANT_TRUE, "failed to load XML string\n");
3702     err = NULL;
3703     ole_expect(IXMLDOMDocument2_validate(dtddoc2, &err), S_FALSE);
3704     ok(err != NULL, "expected pointer\n");
3705     if (err)
3706     {
3707         res = 0;
3708         ole_check(IXMLDOMParseError_get_errorCode(err, &res));
3709         /* XML_REQUIRED_ATTRIBUTE_MISSING */
3710         todo_wine ok(res == 0xC00CE020, "got %08x\n", res);
3711         IXMLDOMParseError_Release(err);
3712     }
3713
3714     IXMLDOMDocument2_Release( dtddoc2 );
3715     free_bstrs();
3716 }
3717
3718 #define helper_ole_check(expr) { \
3719     HRESULT r = expr; \
3720     ok_(__FILE__, line)(r == S_OK, "=> %i: " #expr " returned %08x\n", __LINE__, r); \
3721 }
3722
3723 #define helper_expect_list_and_release(list, expstr) { \
3724     char *str = list_to_string(list); \
3725     ok_(__FILE__, line)(strcmp(str, expstr)==0, "=> %i: Invalid node list: %s, expected %s\n", __LINE__, str, expstr); \
3726     if (list) IXMLDOMNodeList_Release(list); \
3727 }
3728
3729 #define helper_expect_bstr_and_release(bstr, str) { \
3730     ok_(__FILE__, line)(lstrcmpW(bstr, _bstr_(str)) == 0, \
3731        "=> %i: got %s\n", __LINE__, wine_dbgstr_w(bstr)); \
3732     SysFreeString(bstr); \
3733 }
3734
3735 #define check_ws_ignored(doc, str) _check_ws_ignored(__LINE__, doc, str)
3736 static inline void _check_ws_ignored(int line, IXMLDOMDocument2* doc, char const* str)
3737 {
3738     IXMLDOMNode *node1, *node2;
3739     IXMLDOMNodeList *list;
3740     BSTR bstr;
3741
3742     helper_ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//*[local-name()='html']"), &list));
3743     helper_ole_check(IXMLDOMNodeList_get_item(list, 0, &node1));
3744     helper_ole_check(IXMLDOMNodeList_get_item(list, 1, &node2));
3745     helper_ole_check(IXMLDOMNodeList_reset(list));
3746     helper_expect_list_and_release(list, "E1.E4.E1.E2.D1 E2.E4.E1.E2.D1");
3747
3748     helper_ole_check(IXMLDOMNode_get_childNodes(node1, &list));
3749     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");
3750     helper_ole_check(IXMLDOMNode_get_text(node1, &bstr));
3751     if (str)
3752     {
3753         helper_expect_bstr_and_release(bstr, str);
3754     }
3755     else
3756     {
3757         helper_expect_bstr_and_release(bstr, "This is a description.");
3758     }
3759     IXMLDOMNode_Release(node1);
3760
3761     helper_ole_check(IXMLDOMNode_get_childNodes(node2, &list));
3762     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");
3763     helper_ole_check(IXMLDOMNode_get_text(node2, &bstr));
3764     helper_expect_bstr_and_release(bstr, "\n                This is a description with preserved whitespace. \n            ");
3765     IXMLDOMNode_Release(node2);
3766 }
3767
3768 #define check_ws_preserved(doc, str) _check_ws_preserved(__LINE__, doc, str)
3769 static inline void _check_ws_preserved(int line, IXMLDOMDocument2* doc, char const* str)
3770 {
3771     IXMLDOMNode *node1, *node2;
3772     IXMLDOMNodeList *list;
3773     BSTR bstr;
3774
3775     helper_ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//*[local-name()='html']"), &list));
3776     helper_ole_check(IXMLDOMNodeList_get_item(list, 0, &node1));
3777     helper_ole_check(IXMLDOMNodeList_get_item(list, 1, &node2));
3778     helper_ole_check(IXMLDOMNodeList_reset(list));
3779     helper_expect_list_and_release(list, "E2.E8.E2.E2.D1 E4.E8.E2.E2.D1");
3780
3781     helper_ole_check(IXMLDOMNode_get_childNodes(node1, &list));
3782     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");
3783     helper_ole_check(IXMLDOMNode_get_text(node1, &bstr));
3784     if (str)
3785     {
3786         helper_expect_bstr_and_release(bstr, str);
3787     }
3788     else
3789     {
3790         helper_expect_bstr_and_release(bstr, "\n                This is a description. \n            ");
3791     }
3792     IXMLDOMNode_Release(node1);
3793
3794     helper_ole_check(IXMLDOMNode_get_childNodes(node2, &list));
3795     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");
3796     helper_ole_check(IXMLDOMNode_get_text(node2, &bstr));
3797     helper_expect_bstr_and_release(bstr, "\n                This is a description with preserved whitespace. \n            ");
3798     IXMLDOMNode_Release(node2);
3799 }
3800
3801 static void test_whitespace(void)
3802 {
3803     VARIANT_BOOL b;
3804     IXMLDOMDocument2 *doc1, *doc2, *doc3, *doc4;
3805
3806     doc1 = create_document(&IID_IXMLDOMDocument2);
3807     doc2 = create_document(&IID_IXMLDOMDocument2);
3808     if (!doc1 || !doc2) return;
3809
3810     ole_check(IXMLDOMDocument2_put_preserveWhiteSpace(doc2, VARIANT_TRUE));
3811     ole_check(IXMLDOMDocument2_get_preserveWhiteSpace(doc1, &b));
3812     ok(b == VARIANT_FALSE, "expected false\n");
3813     ole_check(IXMLDOMDocument2_get_preserveWhiteSpace(doc2, &b));
3814     ok(b == VARIANT_TRUE, "expected true\n");
3815
3816     ole_check(IXMLDOMDocument2_loadXML(doc1, _bstr_(szExampleXML), &b));
3817     ok(b == VARIANT_TRUE, "failed to load XML string\n");
3818     ole_check(IXMLDOMDocument2_loadXML(doc2, _bstr_(szExampleXML), &b));
3819     ok(b == VARIANT_TRUE, "failed to load XML string\n");
3820
3821     /* switch to XPath */
3822     ole_check(IXMLDOMDocument2_setProperty(doc1, _bstr_("SelectionLanguage"), _variantbstr_("XPath")));
3823     ole_check(IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionLanguage"), _variantbstr_("XPath")));
3824
3825     check_ws_ignored(doc1, NULL);
3826     check_ws_preserved(doc2, NULL);
3827
3828     /* new instances copy the property */
3829     ole_check(IXMLDOMDocument2_QueryInterface(doc1, &IID_IXMLDOMDocument2, (void**) &doc3));
3830     ole_check(IXMLDOMDocument2_QueryInterface(doc2, &IID_IXMLDOMDocument2, (void**) &doc4));
3831
3832     ole_check(IXMLDOMDocument2_get_preserveWhiteSpace(doc3, &b));
3833     ok(b == VARIANT_FALSE, "expected false\n");
3834     ole_check(IXMLDOMDocument2_get_preserveWhiteSpace(doc4, &b));
3835     ok(b == VARIANT_TRUE, "expected true\n");
3836
3837     check_ws_ignored(doc3, NULL);
3838     check_ws_preserved(doc4, NULL);
3839
3840     /* setting after loading xml affects trimming of leading/trailing ws only */
3841     ole_check(IXMLDOMDocument2_put_preserveWhiteSpace(doc1, VARIANT_TRUE));
3842     ole_check(IXMLDOMDocument2_put_preserveWhiteSpace(doc2, VARIANT_FALSE));
3843
3844     /* the trailing "\n            " isn't there, because it was ws-only node */
3845     check_ws_ignored(doc1, "\n                This is a description. ");
3846     check_ws_preserved(doc2, "This is a description.");
3847
3848     /* it takes effect on reload */
3849     ole_check(IXMLDOMDocument2_get_preserveWhiteSpace(doc1, &b));
3850     ok(b == VARIANT_TRUE, "expected true\n");
3851     ole_check(IXMLDOMDocument2_get_preserveWhiteSpace(doc2, &b));
3852     ok(b == VARIANT_FALSE, "expected false\n");
3853
3854     ole_check(IXMLDOMDocument2_loadXML(doc1, _bstr_(szExampleXML), &b));
3855     ok(b == VARIANT_TRUE, "failed to load XML string\n");
3856     ole_check(IXMLDOMDocument2_loadXML(doc2, _bstr_(szExampleXML), &b));
3857     ok(b == VARIANT_TRUE, "failed to load XML string\n");
3858
3859     check_ws_preserved(doc1, NULL);
3860     check_ws_ignored(doc2, NULL);
3861
3862     /* other instances follow suit */
3863     ole_check(IXMLDOMDocument2_get_preserveWhiteSpace(doc3, &b));
3864     ok(b == VARIANT_TRUE, "expected true\n");
3865     ole_check(IXMLDOMDocument2_get_preserveWhiteSpace(doc4, &b));
3866     ok(b == VARIANT_FALSE, "expected false\n");
3867
3868     check_ws_preserved(doc3, NULL);
3869     check_ws_ignored(doc4, NULL);
3870
3871     IXMLDOMDocument_Release(doc1);
3872     IXMLDOMDocument_Release(doc2);
3873     IXMLDOMDocument_Release(doc3);
3874     IXMLDOMDocument_Release(doc4);
3875     free_bstrs();
3876 }
3877
3878 static void test_XPath(void)
3879 {
3880     VARIANT var;
3881     VARIANT_BOOL b;
3882     IXMLDOMDocument2 *doc;
3883     IXMLDOMNode *rootNode;
3884     IXMLDOMNode *elem1Node;
3885     IXMLDOMNode *node;
3886     IXMLDOMNodeList *list;
3887
3888     doc = create_document(&IID_IXMLDOMDocument2);
3889     if (!doc) return;
3890
3891     ole_check(IXMLDOMDocument_loadXML(doc, _bstr_(szExampleXML), &b));
3892     ok(b == VARIANT_TRUE, "failed to load XML string\n");
3893
3894     /* switch to XPath */
3895     ole_check(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionLanguage"), _variantbstr_("XPath")));
3896
3897     /* some simple queries*/
3898     ole_check(IXMLDOMDocument_selectNodes(doc, _bstr_("root"), &list));
3899     ole_check(IXMLDOMNodeList_get_item(list, 0, &rootNode));
3900     ole_check(IXMLDOMNodeList_reset(list));
3901     expect_list_and_release(list, "E2.D1");
3902     if (rootNode == NULL)
3903         return;
3904
3905     ole_check(IXMLDOMDocument_selectNodes(doc, _bstr_("root//c"), &list));
3906     expect_list_and_release(list, "E3.E1.E2.D1 E3.E2.E2.D1");
3907
3908     ole_check(IXMLDOMDocument_selectNodes(doc, _bstr_("//c[@type]"), &list));
3909     expect_list_and_release(list, "E3.E2.E2.D1");
3910
3911     ole_check(IXMLDOMNode_selectNodes(rootNode, _bstr_("elem"), &list));
3912     /* using get_item for query results advances the position */
3913     ole_check(IXMLDOMNodeList_get_item(list, 1, &node));
3914     expect_node(node, "E2.E2.D1");
3915     IXMLDOMNode_Release(node);
3916     ole_check(IXMLDOMNodeList_nextNode(list, &node));
3917     expect_node(node, "E4.E2.D1");
3918     IXMLDOMNode_Release(node);
3919     ole_check(IXMLDOMNodeList_reset(list));
3920     expect_list_and_release(list, "E1.E2.D1 E2.E2.D1 E4.E2.D1");
3921
3922     ole_check(IXMLDOMNode_selectNodes(rootNode, _bstr_("."), &list));
3923     expect_list_and_release(list, "E2.D1");
3924
3925     ole_check(IXMLDOMNode_selectNodes(rootNode, _bstr_("elem[3]/preceding-sibling::*"), &list));
3926     ole_check(IXMLDOMNodeList_get_item(list, 0, &elem1Node));
3927     ole_check(IXMLDOMNodeList_reset(list));
3928     expect_list_and_release(list, "E1.E2.D1 E2.E2.D1 E3.E2.D1");
3929
3930     /* select an attribute */
3931     ole_check(IXMLDOMNode_selectNodes(rootNode, _bstr_(".//@type"), &list));
3932     expect_list_and_release(list, "A'type'.E3.E2.E2.D1");
3933
3934     /* would evaluate to a number */
3935     ole_expect(IXMLDOMNode_selectNodes(rootNode, _bstr_("count(*)"), &list), E_FAIL);
3936     /* would evaluate to a boolean */
3937     ole_expect(IXMLDOMNode_selectNodes(rootNode, _bstr_("position()>0"), &list), E_FAIL);
3938     /* would evaluate to a string */
3939     ole_expect(IXMLDOMNode_selectNodes(rootNode, _bstr_("name()"), &list), E_FAIL);
3940
3941     /* no results */
3942     ole_check(IXMLDOMNode_selectNodes(rootNode, _bstr_("c"), &list));
3943     expect_list_and_release(list, "");
3944     ole_check(IXMLDOMDocument_selectNodes(doc, _bstr_("elem//c"), &list));
3945     expect_list_and_release(list, "");
3946     ole_check(IXMLDOMDocument_selectNodes(doc, _bstr_("//elem[4]"), &list));
3947     expect_list_and_release(list, "");
3948     ole_check(IXMLDOMDocument_selectNodes(doc, _bstr_("root//elem[0]"), &list));
3949     expect_list_and_release(list, "");
3950
3951     /* foo undeclared in document node */
3952     ole_expect(IXMLDOMDocument_selectNodes(doc, _bstr_("root//foo:c"), &list), E_FAIL);
3953     /* undeclared in <root> node */
3954     ole_expect(IXMLDOMNode_selectNodes(rootNode, _bstr_(".//foo:c"), &list), E_FAIL);
3955     /* undeclared in <elem> node */
3956     ole_expect(IXMLDOMNode_selectNodes(elem1Node, _bstr_("//foo:c"), &list), E_FAIL);
3957     /* but this trick can be used */
3958     ole_check(IXMLDOMNode_selectNodes(elem1Node, _bstr_("//*[name()='foo:c']"), &list));
3959     expect_list_and_release(list, "E3.E4.E2.D1");
3960
3961     /* it has to be declared in SelectionNamespaces */
3962     ole_check(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionNamespaces"),
3963         _variantbstr_("xmlns:test='urn:uuid:86B2F87F-ACB6-45cd-8B77-9BDB92A01A29'")));
3964
3965     /* now the namespace can be used */
3966     ole_check(IXMLDOMDocument_selectNodes(doc, _bstr_("root//test:c"), &list));
3967     expect_list_and_release(list, "E3.E3.E2.D1 E3.E4.E2.D1");
3968     ole_check(IXMLDOMNode_selectNodes(rootNode, _bstr_(".//test:c"), &list));
3969     expect_list_and_release(list, "E3.E3.E2.D1 E3.E4.E2.D1");
3970     ole_check(IXMLDOMNode_selectNodes(elem1Node, _bstr_("//test:c"), &list));
3971     expect_list_and_release(list, "E3.E3.E2.D1 E3.E4.E2.D1");
3972     ole_check(IXMLDOMNode_selectNodes(elem1Node, _bstr_(".//test:x"), &list));
3973     expect_list_and_release(list, "E5.E1.E4.E1.E2.D1 E6.E2.E4.E1.E2.D1");
3974
3975     /* SelectionNamespaces syntax error - the namespaces doesn't work anymore but the value is stored */
3976     ole_expect(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionNamespaces"),
3977         _variantbstr_("xmlns:test='urn:uuid:86B2F87F-ACB6-45cd-8B77-9BDB92A01A29' xmlns:foo=###")), E_FAIL);
3978
3979     ole_expect(IXMLDOMDocument_selectNodes(doc, _bstr_("root//foo:c"), &list), E_FAIL);
3980
3981     VariantInit(&var);
3982     ole_check(IXMLDOMDocument2_getProperty(doc, _bstr_("SelectionNamespaces"), &var));
3983     expect_eq(V_VT(&var), VT_BSTR, int, "%x");
3984     if (V_VT(&var) == VT_BSTR)
3985         expect_bstr_eq_and_free(V_BSTR(&var), "xmlns:test='urn:uuid:86B2F87F-ACB6-45cd-8B77-9BDB92A01A29' xmlns:foo=###");
3986
3987     /* extra attributes - same thing*/
3988     ole_expect(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionNamespaces"),
3989         _variantbstr_("xmlns:test='urn:uuid:86B2F87F-ACB6-45cd-8B77-9BDB92A01A29' param='test'")), E_FAIL);
3990     ole_expect(IXMLDOMDocument_selectNodes(doc, _bstr_("root//foo:c"), &list), E_FAIL);
3991
3992     IXMLDOMNode_Release(rootNode);
3993     IXMLDOMNode_Release(elem1Node);
3994     IXMLDOMDocument_Release(doc);
3995     free_bstrs();
3996 }
3997
3998 static void test_cloneNode(void )
3999 {
4000     IXMLDOMDocument *doc;
4001     VARIANT_BOOL b;
4002     IXMLDOMNodeList *pList;
4003     IXMLDOMNamedNodeMap *mapAttr;
4004     LONG nLength = 0, nLength1 = 0;
4005     LONG nAttrCnt = 0, nAttrCnt1 = 0;
4006     IXMLDOMNode *node;
4007     IXMLDOMNode *node_clone;
4008     IXMLDOMNode *node_first;
4009     HRESULT r;
4010     BSTR str;
4011     static const WCHAR szSearch[] = { 'l', 'c', '/', 'p', 'r', 0 };
4012
4013     doc = create_document(&IID_IXMLDOMDocument);
4014     if (!doc) return;
4015
4016     str = SysAllocString( szComplete4 );
4017     ole_check(IXMLDOMDocument_loadXML(doc, str, &b));
4018     ok(b == VARIANT_TRUE, "failed to load XML string\n");
4019     SysFreeString(str);
4020
4021     if(!b)
4022         return;
4023
4024     str = SysAllocString( szSearch);
4025     r = IXMLDOMNode_selectSingleNode(doc, str, &node);
4026     ok( r == S_OK, "ret %08x\n", r );
4027     ok( node != NULL, "node %p\n", node );
4028     SysFreeString(str);
4029
4030     if(!node)
4031     {
4032         IXMLDOMDocument_Release(doc);
4033         return;
4034     }
4035
4036     /* Check invalid parameter */
4037     r = IXMLDOMNode_cloneNode(node, VARIANT_TRUE, NULL);
4038     ok( r == E_INVALIDARG, "ret %08x\n", r );
4039
4040     /* All Children */
4041     r = IXMLDOMNode_cloneNode(node, VARIANT_TRUE, &node_clone);
4042     ok( r == S_OK, "ret %08x\n", r );
4043     ok( node_clone != NULL, "node %p\n", node );
4044
4045     if(!node_clone)
4046     {
4047         IXMLDOMDocument_Release(doc);
4048         IXMLDOMNode_Release(node);
4049         return;
4050     }
4051
4052     r = IXMLDOMNode_get_firstChild(node_clone, &node_first);
4053     ok( r == S_OK, "ret %08x\n", r );
4054     if(r == S_OK)
4055     {
4056         IXMLDOMDocument *doc2;
4057
4058         r = IXMLDOMNode_get_ownerDocument(node_clone, &doc2);
4059         ok( r == S_OK, "ret %08x\n", r );
4060         if(r == S_OK)
4061             IXMLDOMDocument_Release(doc2);
4062
4063         IXMLDOMNode_Release(node_first);
4064     }
4065
4066     r = IXMLDOMNode_get_childNodes(node, &pList);
4067     ok( r == S_OK, "ret %08x\n", r );
4068     if (pList)
4069         {
4070                 IXMLDOMNodeList_get_length(pList, &nLength);
4071                 IXMLDOMNodeList_Release(pList);
4072         }
4073
4074     r = IXMLDOMNode_get_attributes(node, &mapAttr);
4075     ok( r == S_OK, "ret %08x\n", r );
4076     if(mapAttr)
4077     {
4078         IXMLDOMNamedNodeMap_get_length(mapAttr, &nAttrCnt);
4079         IXMLDOMNamedNodeMap_Release(mapAttr);
4080     }
4081
4082     r = IXMLDOMNode_get_childNodes(node_clone, &pList);
4083     ok( r == S_OK, "ret %08x\n", r );
4084     if (pList)
4085         {
4086                 IXMLDOMNodeList_get_length(pList, &nLength1);
4087                 IXMLDOMNodeList_Release(pList);
4088         }
4089
4090     r = IXMLDOMNode_get_attributes(node_clone, &mapAttr);
4091     ok( r == S_OK, "ret %08x\n", r );
4092     if(mapAttr)
4093     {
4094         IXMLDOMNamedNodeMap_get_length(mapAttr, &nAttrCnt1);
4095         IXMLDOMNamedNodeMap_Release(mapAttr);
4096     }
4097
4098     ok(nLength == nLength1, "wrong Child count (%d, %d)\n", nLength, nLength1);
4099     ok(nAttrCnt == nAttrCnt1, "wrong Attribute count (%d, %d)\n", nAttrCnt, nAttrCnt1);
4100     IXMLDOMNode_Release(node_clone);
4101
4102     /* No Children */
4103     r = IXMLDOMNode_cloneNode(node, VARIANT_FALSE, &node_clone);
4104     ok( r == S_OK, "ret %08x\n", r );
4105     ok( node_clone != NULL, "node %p\n", node );
4106
4107     if(!node_clone)
4108     {
4109         IXMLDOMDocument_Release(doc);
4110         IXMLDOMNode_Release(node);
4111         return;
4112     }
4113
4114     r = IXMLDOMNode_get_firstChild(node_clone, &node_first);
4115     ok( r == S_FALSE, "ret %08x\n", r );
4116     if(r == S_OK)
4117     {
4118         IXMLDOMDocument *doc2;
4119
4120         r = IXMLDOMNode_get_ownerDocument(node_clone, &doc2);
4121         ok( r == S_OK, "ret %08x\n", r );
4122         if(r == S_OK)
4123             IXMLDOMDocument_Release(doc2);
4124
4125         IXMLDOMNode_Release(node_first);
4126     }
4127
4128     r = IXMLDOMNode_get_childNodes(node_clone, &pList);
4129     ok( r == S_OK, "ret %08x\n", r );
4130     if (pList)
4131     {
4132         IXMLDOMNodeList_get_length(pList, &nLength1);
4133         ok( nLength1 == 0, "Length should be 0 (%d)\n", nLength1);
4134         IXMLDOMNodeList_Release(pList);
4135     }
4136
4137     r = IXMLDOMNode_get_attributes(node_clone, &mapAttr);
4138     ok( r == S_OK, "ret %08x\n", r );
4139     if(mapAttr)
4140     {
4141         IXMLDOMNamedNodeMap_get_length(mapAttr, &nAttrCnt1);
4142         ok( nAttrCnt1 == 3, "Attribute count should be 3 (%d)\n", nAttrCnt1);
4143         IXMLDOMNamedNodeMap_Release(mapAttr);
4144     }
4145
4146     ok(nLength != nLength1, "wrong Child count (%d, %d)\n", nLength, nLength1);
4147     ok(nAttrCnt == nAttrCnt1, "wrong Attribute count (%d, %d)\n", nAttrCnt, nAttrCnt1);
4148     IXMLDOMNode_Release(node_clone);
4149
4150
4151     IXMLDOMNode_Release(node);
4152     IXMLDOMDocument_Release(doc);
4153 }
4154
4155 static void test_xmlTypes(void)
4156 {
4157     IXMLDOMDocument *doc;
4158     IXMLDOMElement *pRoot;
4159     HRESULT hr;
4160     IXMLDOMComment *pComment;
4161     IXMLDOMElement *pElement;
4162     IXMLDOMAttribute *pAttribute;
4163     IXMLDOMNamedNodeMap *pAttribs;
4164     IXMLDOMCDATASection *pCDataSec;
4165     IXMLDOMImplementation *pIXMLDOMImplementation = NULL;
4166     IXMLDOMDocumentFragment *pDocFrag = NULL;
4167     IXMLDOMEntityReference *pEntityRef = NULL;
4168     BSTR str;
4169     IXMLDOMNode *pNextChild;
4170     VARIANT v;
4171     LONG len = 0;
4172
4173     doc = create_document(&IID_IXMLDOMDocument);
4174     if (!doc) return;
4175
4176     pNextChild = (void*)0xdeadbeef;
4177     hr = IXMLDOMDocument_get_nextSibling(doc, NULL);
4178     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4179
4180     pNextChild = (void*)0xdeadbeef;
4181     hr = IXMLDOMDocument_get_nextSibling(doc, &pNextChild);
4182     ok(hr == S_FALSE, "ret %08x\n", hr );
4183     ok(pNextChild == NULL, "pDocChild not NULL\n");
4184
4185     /* test previous Sibling */
4186     hr = IXMLDOMDocument_get_previousSibling(doc, NULL);
4187     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4188
4189     pNextChild = (void*)0xdeadbeef;
4190     hr = IXMLDOMDocument_get_previousSibling(doc, &pNextChild);
4191     ok(hr == S_FALSE, "ret %08x\n", hr );
4192     ok(pNextChild == NULL, "pNextChild not NULL\n");
4193
4194     /* test get_attributes */
4195     hr = IXMLDOMDocument_get_attributes( doc, NULL );
4196     ok( hr == E_INVALIDARG, "get_attributes returned wrong code\n");
4197
4198     pAttribs = (void*)0xdeadbeef;
4199     hr = IXMLDOMDocument_get_attributes( doc, &pAttribs);
4200     ok(hr == S_FALSE, "ret %08x\n", hr );
4201     ok( pAttribs == NULL, "pAttribs not NULL\n");
4202
4203     /* test get_dataType */
4204     V_VT(&v) = VT_EMPTY;
4205     hr = IXMLDOMDocument_get_dataType(doc, &v);
4206     ok(hr == S_FALSE, "ret %08x\n", hr );
4207     ok( V_VT(&v) == VT_NULL, "incorrect dataType type\n");
4208     VariantClear(&v);
4209
4210     /* test nodeTypeString */
4211     str = NULL;
4212     hr = IXMLDOMDocument_get_nodeTypeString(doc, &str);
4213     ok(hr == S_OK, "ret %08x\n", hr );
4214     ok( !lstrcmpW( str, _bstr_("document") ), "incorrect nodeTypeString string\n");
4215     SysFreeString(str);
4216
4217     /* test implementation */
4218     hr = IXMLDOMDocument_get_implementation(doc, NULL);
4219     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4220
4221     hr = IXMLDOMDocument_get_implementation(doc, &pIXMLDOMImplementation);
4222     ok(hr == S_OK, "ret %08x\n", hr );
4223     if(hr == S_OK)
4224     {
4225         VARIANT_BOOL hasFeature = VARIANT_TRUE;
4226         BSTR sEmpty = SysAllocStringLen(NULL, 0);
4227
4228         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, NULL, sEmpty, &hasFeature);
4229         ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4230
4231         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, sEmpty, sEmpty, NULL);
4232         ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4233
4234         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, _bstr_("DOM"), sEmpty, &hasFeature);
4235         ok(hr == S_OK, "ret %08x\n", hr );
4236         ok(hasFeature == VARIANT_FALSE, "hasFeature returned false\n");
4237
4238         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, sEmpty, sEmpty, &hasFeature);
4239         ok(hr == S_OK, "ret %08x\n", hr );
4240         ok(hasFeature == VARIANT_FALSE, "hasFeature returned true\n");
4241
4242         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, _bstr_("DOM"), NULL, &hasFeature);
4243         ok(hr == S_OK, "ret %08x\n", hr );
4244         ok(hasFeature == VARIANT_TRUE, "hasFeature returned false\n");
4245
4246         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, _bstr_("DOM"), sEmpty, &hasFeature);
4247         ok(hr == S_OK, "ret %08x\n", hr );
4248         ok(hasFeature == VARIANT_FALSE, "hasFeature returned false\n");
4249
4250         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, _bstr_("DOM"), _bstr_("1.0"), &hasFeature);
4251         ok(hr == S_OK, "ret %08x\n", hr );
4252         ok(hasFeature == VARIANT_TRUE, "hasFeature returned true\n");
4253
4254         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, _bstr_("XML"), _bstr_("1.0"), &hasFeature);
4255         ok(hr == S_OK, "ret %08x\n", hr );
4256         ok(hasFeature == VARIANT_TRUE, "hasFeature returned true\n");
4257
4258         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, _bstr_("MS-DOM"), _bstr_("1.0"), &hasFeature);
4259         ok(hr == S_OK, "ret %08x\n", hr );
4260         ok(hasFeature == VARIANT_TRUE, "hasFeature returned true\n");
4261
4262         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, _bstr_("SSS"), NULL, &hasFeature);
4263         ok(hr == S_OK, "ret %08x\n", hr );
4264         ok(hasFeature == VARIANT_FALSE, "hasFeature returned false\n");
4265
4266         SysFreeString(sEmpty);
4267         IXMLDOMImplementation_Release(pIXMLDOMImplementation);
4268     }
4269
4270     pRoot = (IXMLDOMElement*)0x1;
4271     hr = IXMLDOMDocument_createElement(doc, NULL, &pRoot);
4272     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4273     ok(pRoot == (void*)0x1, "Expect same ptr, got %p\n", pRoot);
4274
4275     pRoot = (IXMLDOMElement*)0x1;
4276     hr = IXMLDOMDocument_createElement(doc, _bstr_(""), &pRoot);
4277     ok(hr == E_FAIL, "ret %08x\n", hr );
4278     ok(pRoot == (void*)0x1, "Expect same ptr, got %p\n", pRoot);
4279
4280     hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing"), &pRoot);
4281     ok(hr == S_OK, "ret %08x\n", hr );
4282     if(hr == S_OK)
4283     {
4284         hr = IXMLDOMDocument_appendChild(doc, (IXMLDOMNode*)pRoot, NULL);
4285         ok(hr == S_OK, "ret %08x\n", hr );
4286         if(hr == S_OK)
4287         {
4288             /* Comment */
4289             str = SysAllocString(szComment);
4290             hr = IXMLDOMDocument_createComment(doc, str, &pComment);
4291             SysFreeString(str);
4292             ok(hr == S_OK, "ret %08x\n", hr );
4293             if(hr == S_OK)
4294             {
4295                 /* test get_attributes */
4296                 hr = IXMLDOMComment_get_attributes( pComment, NULL );
4297                 ok( hr == E_INVALIDARG, "get_attributes returned wrong code\n");
4298
4299                 pAttribs = (IXMLDOMNamedNodeMap*)0x1;
4300                 hr = IXMLDOMComment_get_attributes( pComment, &pAttribs);
4301                 ok(hr == S_FALSE, "ret %08x\n", hr );
4302                 ok( pAttribs == NULL, "pAttribs not NULL\n");
4303
4304                 /* test nodeTypeString */
4305                 hr = IXMLDOMComment_get_nodeTypeString(pComment, &str);
4306                 ok(hr == S_OK, "ret %08x\n", hr );
4307                 ok( !lstrcmpW( str, _bstr_("comment") ), "incorrect nodeTypeString string\n");
4308                 SysFreeString(str);
4309
4310                 hr = IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pComment, NULL);
4311                 ok(hr == S_OK, "ret %08x\n", hr );
4312
4313                 hr = IXMLDOMComment_get_nodeName(pComment, &str);
4314                 ok(hr == S_OK, "ret %08x\n", hr );
4315                 ok( !lstrcmpW( str, szCommentNodeText ), "incorrect comment node Name\n");
4316                 SysFreeString(str);
4317
4318                 hr = IXMLDOMComment_get_xml(pComment, &str);
4319                 ok(hr == S_OK, "ret %08x\n", hr );
4320                 ok( !lstrcmpW( str, szCommentXML ), "incorrect comment xml\n");
4321                 SysFreeString(str);
4322
4323                 hr = IXMLDOMComment_get_dataType(pComment, &v);
4324                 ok(hr == S_FALSE, "ret %08x\n", hr );
4325                 ok( V_VT(&v) == VT_NULL, "incorrect dataType type\n");
4326                 VariantClear(&v);
4327
4328                 /* put data Tests */
4329                 hr = IXMLDOMComment_put_data(pComment, _bstr_("This &is a ; test <>\\"));
4330                 ok(hr == S_OK, "ret %08x\n", hr );
4331
4332                 /* get data Tests */
4333                 hr = IXMLDOMComment_get_data(pComment, &str);
4334                 ok(hr == S_OK, "ret %08x\n", hr );
4335                 ok( !lstrcmpW( str, _bstr_("This &is a ; test <>\\") ), "incorrect get_data string\n");
4336                 SysFreeString(str);
4337
4338                 /* get data Tests */
4339                 hr = IXMLDOMComment_get_nodeValue(pComment, &v);
4340                 ok(hr == S_OK, "ret %08x\n", hr );
4341                 ok( V_VT(&v) == VT_BSTR, "incorrect dataType type\n");
4342                 ok( !lstrcmpW( V_BSTR(&v), _bstr_("This &is a ; test <>\\") ), "incorrect get_nodeValue string\n");
4343                 VariantClear(&v);
4344
4345                 /* Confirm XML text is good */
4346                 hr = IXMLDOMComment_get_xml(pComment, &str);
4347                 ok(hr == S_OK, "ret %08x\n", hr );
4348                 ok( !lstrcmpW( str, _bstr_("<!--This &is a ; test <>\\-->") ), "incorrect xml string\n");
4349                 SysFreeString(str);
4350
4351                 /* Confirm we get the put_data Text back */
4352                 hr = IXMLDOMComment_get_text(pComment, &str);
4353                 ok(hr == S_OK, "ret %08x\n", hr );
4354                 ok( !lstrcmpW( str, _bstr_("This &is a ; test <>\\") ), "incorrect xml string\n");
4355                 SysFreeString(str);
4356
4357                 /* test length property */
4358                 hr = IXMLDOMComment_get_length(pComment, &len);
4359                 ok(hr == S_OK, "ret %08x\n", hr );
4360                 ok(len == 21, "expected 21 got %d\n", len);
4361
4362                 /* test substringData */
4363                 hr = IXMLDOMComment_substringData(pComment, 0, 4, NULL);
4364                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4365
4366                 /* test substringData - Invalid offset */
4367                 str = (BSTR)&szElement;
4368                 hr = IXMLDOMComment_substringData(pComment, -1, 4, &str);
4369                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4370                 ok( str == NULL, "incorrect string\n");
4371
4372                 /* test substringData - Invalid offset */
4373                 str = (BSTR)&szElement;
4374                 hr = IXMLDOMComment_substringData(pComment, 30, 0, &str);
4375                 ok(hr == S_FALSE, "ret %08x\n", hr );
4376                 ok( str == NULL, "incorrect string\n");
4377
4378                 /* test substringData - Invalid size */
4379                 str = (BSTR)&szElement;
4380                 hr = IXMLDOMComment_substringData(pComment, 0, -1, &str);
4381                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4382                 ok( str == NULL, "incorrect string\n");
4383
4384                 /* test substringData - Invalid size */
4385                 str = (BSTR)&szElement;
4386                 hr = IXMLDOMComment_substringData(pComment, 2, 0, &str);
4387                 ok(hr == S_FALSE, "ret %08x\n", hr );
4388                 ok( str == NULL, "incorrect string\n");
4389
4390                 /* test substringData - Start of string */
4391                 hr = IXMLDOMComment_substringData(pComment, 0, 4, &str);
4392                 ok(hr == S_OK, "ret %08x\n", hr );
4393                 ok( !lstrcmpW( str, _bstr_("This") ), "incorrect substringData string\n");
4394                 SysFreeString(str);
4395
4396                 /* test substringData - Middle of string */
4397                 hr = IXMLDOMComment_substringData(pComment, 13, 4, &str);
4398                 ok(hr == S_OK, "ret %08x\n", hr );
4399                 ok( !lstrcmpW( str, _bstr_("test") ), "incorrect substringData string\n");
4400                 SysFreeString(str);
4401
4402                 /* test substringData - End of string */
4403                 hr = IXMLDOMComment_substringData(pComment, 20, 4, &str);
4404                 ok(hr == S_OK, "ret %08x\n", hr );
4405                 ok( !lstrcmpW( str, _bstr_("\\") ), "incorrect substringData string\n");
4406                 SysFreeString(str);
4407
4408                 /* test appendData */
4409                 hr = IXMLDOMComment_appendData(pComment, NULL);
4410                 ok(hr == S_OK, "ret %08x\n", hr );
4411
4412                 hr = IXMLDOMComment_appendData(pComment, _bstr_(""));
4413                 ok(hr == S_OK, "ret %08x\n", hr );
4414
4415                 hr = IXMLDOMComment_appendData(pComment, _bstr_("Append"));
4416                 ok(hr == S_OK, "ret %08x\n", hr );
4417
4418                 hr = IXMLDOMComment_get_text(pComment, &str);
4419                 ok(hr == S_OK, "ret %08x\n", hr );
4420                 ok( !lstrcmpW( str, _bstr_("This &is a ; test <>\\Append") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4421                 SysFreeString(str);
4422
4423                 /* test insertData */
4424                 str = SysAllocStringLen(NULL, 0);
4425                 hr = IXMLDOMComment_insertData(pComment, -1, str);
4426                 ok(hr == S_OK, "ret %08x\n", hr );
4427
4428                 hr = IXMLDOMComment_insertData(pComment, -1, NULL);
4429                 ok(hr == S_OK, "ret %08x\n", hr );
4430
4431                 hr = IXMLDOMComment_insertData(pComment, 1000, str);
4432                 ok(hr == S_OK, "ret %08x\n", hr );
4433
4434                 hr = IXMLDOMComment_insertData(pComment, 1000, NULL);
4435                 ok(hr == S_OK, "ret %08x\n", hr );
4436
4437                 hr = IXMLDOMComment_insertData(pComment, 0, NULL);
4438                 ok(hr == S_OK, "ret %08x\n", hr );
4439
4440                 hr = IXMLDOMComment_insertData(pComment, 0, str);
4441                 ok(hr == S_OK, "ret %08x\n", hr );
4442                 SysFreeString(str);
4443
4444                 hr = IXMLDOMComment_insertData(pComment, -1, _bstr_("Inserting"));
4445                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4446
4447                 hr = IXMLDOMComment_insertData(pComment, 1000, _bstr_("Inserting"));
4448                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4449
4450                 hr = IXMLDOMComment_insertData(pComment, 0, _bstr_("Begin "));
4451                 ok(hr == S_OK, "ret %08x\n", hr );
4452
4453                 hr = IXMLDOMComment_insertData(pComment, 17, _bstr_("Middle"));
4454                 ok(hr == S_OK, "ret %08x\n", hr );
4455
4456                 hr = IXMLDOMComment_insertData(pComment, 39, _bstr_(" End"));
4457                 ok(hr == S_OK, "ret %08x\n", hr );
4458
4459                 hr = IXMLDOMComment_get_text(pComment, &str);
4460                 ok(hr == S_OK, "ret %08x\n", hr );
4461                 ok( !lstrcmpW( str, _bstr_("Begin This &is a Middle; test <>\\Append End") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4462                 SysFreeString(str);
4463
4464                 /* delete data */
4465                 /* invalid arguments */
4466                 hr = IXMLDOMComment_deleteData(pComment, -1, 1);
4467                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4468
4469                 hr = IXMLDOMComment_deleteData(pComment, 0, 0);
4470                 ok(hr == S_OK, "ret %08x\n", hr );
4471
4472                 hr = IXMLDOMComment_deleteData(pComment, 0, -1);
4473                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4474
4475                 hr = IXMLDOMComment_get_length(pComment, &len);
4476                 ok(hr == S_OK, "ret %08x\n", hr );
4477                 ok(len == 43, "expected 43 got %d\n", len);
4478
4479                 hr = IXMLDOMComment_deleteData(pComment, len, 1);
4480                 ok(hr == S_OK, "ret %08x\n", hr );
4481
4482                 hr = IXMLDOMComment_deleteData(pComment, len+1, 1);
4483                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4484
4485                 /* delete from start */
4486                 hr = IXMLDOMComment_deleteData(pComment, 0, 5);
4487                 ok(hr == S_OK, "ret %08x\n", hr );
4488
4489                 hr = IXMLDOMComment_get_length(pComment, &len);
4490                 ok(hr == S_OK, "ret %08x\n", hr );
4491                 ok(len == 38, "expected 38 got %d\n", len);
4492
4493                 hr = IXMLDOMComment_get_text(pComment, &str);
4494                 ok(hr == S_OK, "ret %08x\n", hr );
4495                 ok( !lstrcmpW( str, _bstr_(" This &is a Middle; test <>\\Append End") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4496                 SysFreeString(str);
4497
4498                 /* delete from end */
4499                 hr = IXMLDOMComment_deleteData(pComment, 35, 3);
4500                 ok(hr == S_OK, "ret %08x\n", hr );
4501
4502                 hr = IXMLDOMComment_get_length(pComment, &len);
4503                 ok(hr == S_OK, "ret %08x\n", hr );
4504                 ok(len == 35, "expected 35 got %d\n", len);
4505
4506                 hr = IXMLDOMComment_get_text(pComment, &str);
4507                 ok(hr == S_OK, "ret %08x\n", hr );
4508                 ok( !lstrcmpW( str, _bstr_(" This &is a Middle; test <>\\Append ") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4509                 SysFreeString(str);
4510
4511                 /* delete from inside */
4512                 hr = IXMLDOMComment_deleteData(pComment, 1, 33);
4513                 ok(hr == S_OK, "ret %08x\n", hr );
4514
4515                 hr = IXMLDOMComment_get_length(pComment, &len);
4516                 ok(hr == S_OK, "ret %08x\n", hr );
4517                 ok(len == 2, "expected 2 got %d\n", len);
4518
4519                 hr = IXMLDOMComment_get_text(pComment, &str);
4520                 ok(hr == S_OK, "ret %08x\n", hr );
4521                 ok( !lstrcmpW( str, _bstr_("  ") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4522                 SysFreeString(str);
4523
4524                 /* delete whole data ... */
4525                 hr = IXMLDOMComment_get_length(pComment, &len);
4526                 ok(hr == S_OK, "ret %08x\n", hr );
4527
4528                 hr = IXMLDOMComment_deleteData(pComment, 0, len);
4529                 ok(hr == S_OK, "ret %08x\n", hr );
4530                 /* ... and try again with empty string */
4531                 hr = IXMLDOMComment_deleteData(pComment, 0, len);
4532                 ok(hr == S_OK, "ret %08x\n", hr );
4533
4534                 /* ::replaceData() */
4535                 V_VT(&v) = VT_BSTR;
4536                 V_BSTR(&v) = SysAllocString(szstr1);
4537                 hr = IXMLDOMComment_put_nodeValue(pComment, v);
4538                 ok(hr == S_OK, "ret %08x\n", hr );
4539                 VariantClear(&v);
4540
4541                 hr = IXMLDOMComment_replaceData(pComment, 6, 0, NULL);
4542                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4543                 hr = IXMLDOMComment_get_text(pComment, &str);
4544                 ok(hr == S_OK, "ret %08x\n", hr );
4545                 ok( !lstrcmpW( str, _bstr_("str1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4546                 SysFreeString(str);
4547
4548                 hr = IXMLDOMComment_replaceData(pComment, 0, 0, NULL);
4549                 ok(hr == S_OK, "ret %08x\n", hr );
4550                 hr = IXMLDOMComment_get_text(pComment, &str);
4551                 ok(hr == S_OK, "ret %08x\n", hr );
4552                 ok( !lstrcmpW( str, _bstr_("str1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4553                 SysFreeString(str);
4554
4555                 /* NULL pointer means delete */
4556                 hr = IXMLDOMComment_replaceData(pComment, 0, 1, NULL);
4557                 ok(hr == S_OK, "ret %08x\n", hr );
4558                 hr = IXMLDOMComment_get_text(pComment, &str);
4559                 ok(hr == S_OK, "ret %08x\n", hr );
4560                 ok( !lstrcmpW( str, _bstr_("tr1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4561                 SysFreeString(str);
4562
4563                 /* empty string means delete */
4564                 hr = IXMLDOMComment_replaceData(pComment, 0, 1, _bstr_(""));
4565                 ok(hr == S_OK, "ret %08x\n", hr );
4566                 hr = IXMLDOMComment_get_text(pComment, &str);
4567                 ok(hr == S_OK, "ret %08x\n", hr );
4568                 ok( !lstrcmpW( str, _bstr_("r1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4569                 SysFreeString(str);
4570
4571                 /* zero count means insert */
4572                 hr = IXMLDOMComment_replaceData(pComment, 0, 0, _bstr_("a"));
4573                 ok(hr == S_OK, "ret %08x\n", hr );
4574                 hr = IXMLDOMComment_get_text(pComment, &str);
4575                 ok(hr == S_OK, "ret %08x\n", hr );
4576                 ok( !lstrcmpW( str, _bstr_("ar1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4577                 SysFreeString(str);
4578
4579                 hr = IXMLDOMComment_replaceData(pComment, 0, 2, NULL);
4580                 ok(hr == S_OK, "ret %08x\n", hr );
4581
4582                 hr = IXMLDOMComment_insertData(pComment, 0, _bstr_("m"));
4583                 ok(hr == S_OK, "ret %08x\n", hr );
4584                 hr = IXMLDOMComment_get_text(pComment, &str);
4585                 ok(hr == S_OK, "ret %08x\n", hr );
4586                 ok( !lstrcmpW( str, _bstr_("m1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4587                 SysFreeString(str);
4588
4589                 /* nonempty string, count greater than its length */
4590                 hr = IXMLDOMComment_replaceData(pComment, 0, 2, _bstr_("a1.2"));
4591                 ok(hr == S_OK, "ret %08x\n", hr );
4592                 hr = IXMLDOMComment_get_text(pComment, &str);
4593                 ok(hr == S_OK, "ret %08x\n", hr );
4594                 ok( !lstrcmpW( str, _bstr_("a1.2") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4595                 SysFreeString(str);
4596
4597                 /* nonempty string, count less than its length */
4598                 hr = IXMLDOMComment_replaceData(pComment, 0, 1, _bstr_("wine"));
4599                 ok(hr == S_OK, "ret %08x\n", hr );
4600                 hr = IXMLDOMComment_get_text(pComment, &str);
4601                 ok(hr == S_OK, "ret %08x\n", hr );
4602                 ok( !lstrcmpW( str, _bstr_("wine1.2") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4603                 SysFreeString(str);
4604
4605                 IXMLDOMComment_Release(pComment);
4606             }
4607
4608             /* Element */
4609             str = SysAllocString(szElement);
4610             hr = IXMLDOMDocument_createElement(doc, str, &pElement);
4611             SysFreeString(str);
4612             ok(hr == S_OK, "ret %08x\n", hr );
4613             if(hr == S_OK)
4614             {
4615                 hr = IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
4616                 ok(hr == S_OK, "ret %08x\n", hr );
4617
4618                 /* test nodeTypeString */
4619                 hr = IXMLDOMDocument_get_nodeTypeString(pElement, &str);
4620                 ok(hr == S_OK, "ret %08x\n", hr );
4621                 ok( !lstrcmpW( str, _bstr_("element") ), "incorrect nodeTypeString string\n");
4622                 SysFreeString(str);
4623
4624                 hr = IXMLDOMElement_get_nodeName(pElement, &str);
4625                 ok(hr == S_OK, "ret %08x\n", hr );
4626                 ok( !lstrcmpW( str, szElement ), "incorrect element node Name\n");
4627                 SysFreeString(str);
4628
4629                 hr = IXMLDOMElement_get_xml(pElement, &str);
4630                 ok(hr == S_OK, "ret %08x\n", hr );
4631                 ok( !lstrcmpW( str, szElementXML ), "incorrect element xml\n");
4632                 SysFreeString(str);
4633
4634                 hr = IXMLDOMElement_get_dataType(pElement, &v);
4635                 ok(hr == S_FALSE, "ret %08x\n", hr );
4636                 ok( V_VT(&v) == VT_NULL, "incorrect dataType type\n");
4637                 VariantClear(&v);
4638
4639                 /* Attribute */
4640                 pAttribute = (IXMLDOMAttribute*)0x1;
4641                 hr = IXMLDOMDocument_createAttribute(doc, NULL, &pAttribute);
4642                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4643                 ok(pAttribute == (void*)0x1, "Expect same ptr, got %p\n", pAttribute);
4644
4645                 pAttribute = (IXMLDOMAttribute*)0x1;
4646                 hr = IXMLDOMDocument_createAttribute(doc, _bstr_(""), &pAttribute);
4647                 ok(hr == E_FAIL, "ret %08x\n", hr );
4648                 ok(pAttribute == (void*)0x1, "Expect same ptr, got %p\n", pAttribute);
4649
4650                 str = SysAllocString(szAttribute);
4651                 hr = IXMLDOMDocument_createAttribute(doc, str, &pAttribute);
4652                 SysFreeString(str);
4653                 ok(hr == S_OK, "ret %08x\n", hr );
4654                 if(hr == S_OK)
4655                 {
4656                     IXMLDOMNode *pNewChild = (IXMLDOMNode *)0x1;
4657
4658                     hr = IXMLDOMAttribute_get_nextSibling(pAttribute, NULL);
4659                     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4660
4661                     pNextChild = (IXMLDOMNode *)0x1;
4662                     hr = IXMLDOMAttribute_get_nextSibling(pAttribute, &pNextChild);
4663                     ok(hr == S_FALSE, "ret %08x\n", hr );
4664                     ok(pNextChild == NULL, "pNextChild not NULL\n");
4665
4666                     /* test Previous Sibling*/
4667                     hr = IXMLDOMAttribute_get_previousSibling(pAttribute, NULL);
4668                     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4669
4670                     pNextChild = (IXMLDOMNode *)0x1;
4671                     hr = IXMLDOMAttribute_get_previousSibling(pAttribute, &pNextChild);
4672                     ok(hr == S_FALSE, "ret %08x\n", hr );
4673                     ok(pNextChild == NULL, "pNextChild not NULL\n");
4674
4675                     /* test get_attributes */
4676                     hr = IXMLDOMAttribute_get_attributes( pAttribute, NULL );
4677                     ok( hr == E_INVALIDARG, "get_attributes returned wrong code\n");
4678
4679                     pAttribs = (IXMLDOMNamedNodeMap*)0x1;
4680                     hr = IXMLDOMAttribute_get_attributes( pAttribute, &pAttribs);
4681                     ok(hr == S_FALSE, "ret %08x\n", hr );
4682                     ok( pAttribs == NULL, "pAttribs not NULL\n");
4683
4684                     hr = IXMLDOMElement_appendChild(pElement, (IXMLDOMNode*)pAttribute, &pNewChild);
4685                     ok(hr == E_FAIL, "ret %08x\n", hr );
4686                     ok(pNewChild == NULL, "pNewChild not NULL\n");
4687
4688                     hr = IXMLDOMElement_get_attributes(pElement, &pAttribs);
4689                     ok(hr == S_OK, "ret %08x\n", hr );
4690                     if ( hr == S_OK )
4691                     {
4692                         hr = IXMLDOMNamedNodeMap_setNamedItem(pAttribs, (IXMLDOMNode*)pAttribute, NULL );
4693                         ok(hr == S_OK, "ret %08x\n", hr );
4694
4695                         IXMLDOMNamedNodeMap_Release(pAttribs);
4696                     }
4697
4698                     hr = IXMLDOMAttribute_get_nodeName(pAttribute, &str);
4699                     ok(hr == S_OK, "ret %08x\n", hr );
4700                     ok( !lstrcmpW( str, szAttribute ), "incorrect attribute node Name\n");
4701                     SysFreeString(str);
4702
4703                     /* test nodeTypeString */
4704                     hr = IXMLDOMAttribute_get_nodeTypeString(pAttribute, &str);
4705                     ok(hr == S_OK, "ret %08x\n", hr );
4706                     ok( !lstrcmpW( str, _bstr_("attribute") ), "incorrect nodeTypeString string\n");
4707                     SysFreeString(str);
4708
4709                     /* test nodeName */
4710                     hr = IXMLDOMAttribute_get_nodeName(pAttribute, &str);
4711                     ok(hr == S_OK, "ret %08x\n", hr );
4712                     ok( !lstrcmpW( str, szAttribute ), "incorrect nodeName string\n");
4713                     SysFreeString(str);
4714
4715                     /* test name property */
4716                     hr = IXMLDOMAttribute_get_name(pAttribute, &str);
4717                     ok(hr == S_OK, "ret %08x\n", hr );
4718                     ok( !lstrcmpW( str, szAttribute ), "incorrect name string\n");
4719                     SysFreeString(str);
4720
4721                     hr = IXMLDOMAttribute_get_xml(pAttribute, &str);
4722                     ok(hr == S_OK, "ret %08x\n", hr );
4723                     ok( !lstrcmpW( str, szAttributeXML ), "incorrect attribute xml\n");
4724                     SysFreeString(str);
4725
4726                     hr = IXMLDOMAttribute_get_dataType(pAttribute, &v);
4727                     ok(hr == S_FALSE, "ret %08x\n", hr );
4728                     ok( V_VT(&v) == VT_NULL, "incorrect dataType type\n");
4729                     VariantClear(&v);
4730
4731                     IXMLDOMAttribute_Release(pAttribute);
4732
4733                     /* Check Element again with the Add Attribute*/
4734                     hr = IXMLDOMElement_get_xml(pElement, &str);
4735                     ok(hr == S_OK, "ret %08x\n", hr );
4736                     ok( !lstrcmpW( str, szElementXML2 ), "incorrect element xml\n");
4737                     SysFreeString(str);
4738                 }
4739
4740                 hr = IXMLDOMElement_put_text(pElement, _bstr_("TestingNode"));
4741                 ok(hr == S_OK, "ret %08x\n", hr );
4742
4743                 hr = IXMLDOMElement_get_xml(pElement, &str);
4744                 ok(hr == S_OK, "ret %08x\n", hr );
4745                 ok( !lstrcmpW( str, szElementXML3 ), "incorrect element xml\n");
4746                 SysFreeString(str);
4747
4748                 /* Test for reversible escaping */
4749                 str = SysAllocString( szStrangeChars );
4750                 hr = IXMLDOMElement_put_text(pElement, str);
4751                 ok(hr == S_OK, "ret %08x\n", hr );
4752                 SysFreeString( str );
4753
4754                 hr = IXMLDOMElement_get_xml(pElement, &str);
4755                 ok(hr == S_OK, "ret %08x\n", hr );
4756                 ok( !lstrcmpW( str, szElementXML4 ), "incorrect element xml\n");
4757                 SysFreeString(str);
4758
4759                 hr = IXMLDOMElement_get_text(pElement, &str);
4760                 ok(hr == S_OK, "ret %08x\n", hr );
4761                 ok( !lstrcmpW( str, szStrangeChars ), "incorrect element text\n");
4762                 SysFreeString(str);
4763
4764                 IXMLDOMElement_Release(pElement);
4765             }
4766
4767             /* CData Section */
4768             str = SysAllocString(szCData);
4769             hr = IXMLDOMDocument_createCDATASection(doc, str, NULL);
4770             ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4771
4772             hr = IXMLDOMDocument_createCDATASection(doc, str, &pCDataSec);
4773             SysFreeString(str);
4774             ok(hr == S_OK, "ret %08x\n", hr );
4775             if(hr == S_OK)
4776             {
4777                 IXMLDOMNode *pNextChild = (IXMLDOMNode *)0x1;
4778                 VARIANT var;
4779
4780                 VariantInit(&var);
4781
4782                 hr = IXMLDOMCDATASection_QueryInterface(pCDataSec, &IID_IXMLDOMElement, (void**)&pElement);
4783                 ok(hr == E_NOINTERFACE, "ret %08x\n", hr);
4784
4785                 hr = IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pCDataSec, NULL);
4786                 ok(hr == S_OK, "ret %08x\n", hr );
4787
4788                 /* get Attribute Tests */
4789                 hr = IXMLDOMCDATASection_get_attributes(pCDataSec, NULL);
4790                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4791
4792                 pAttribs = (IXMLDOMNamedNodeMap*)0x1;
4793                 hr = IXMLDOMCDATASection_get_attributes(pCDataSec, &pAttribs);
4794                 ok(hr == S_FALSE, "ret %08x\n", hr );
4795                 ok(pAttribs == NULL, "pAttribs != NULL\n");
4796
4797                 hr = IXMLDOMCDATASection_get_nodeName(pCDataSec, &str);
4798                 ok(hr == S_OK, "ret %08x\n", hr );
4799                 ok( !lstrcmpW( str, szCDataNodeText ), "incorrect cdata node Name\n");
4800                 SysFreeString(str);
4801
4802                 hr = IXMLDOMCDATASection_get_xml(pCDataSec, &str);
4803                 ok(hr == S_OK, "ret %08x\n", hr );
4804                 ok( !lstrcmpW( str, szCDataXML ), "incorrect cdata xml\n");
4805                 SysFreeString(str);
4806
4807                 /* test lastChild */
4808                 pNextChild = (IXMLDOMNode*)0x1;
4809                 hr = IXMLDOMCDATASection_get_lastChild(pCDataSec, &pNextChild);
4810                 ok(hr == S_FALSE, "ret %08x\n", hr );
4811                 ok(pNextChild == NULL, "pNextChild not NULL\n");
4812
4813                 /* test get_dataType */
4814                 hr = IXMLDOMCDATASection_get_dataType(pCDataSec, NULL);
4815                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4816
4817                 hr = IXMLDOMCDATASection_get_dataType(pCDataSec, &v);
4818                 ok(hr == S_FALSE, "ret %08x\n", hr );
4819                 ok( V_VT(&v) == VT_NULL, "incorrect dataType type\n");
4820                 VariantClear(&v);
4821
4822                 /* test nodeTypeString */
4823                 hr = IXMLDOMCDATASection_get_nodeTypeString(pCDataSec, &str);
4824                 ok(hr == S_OK, "ret %08x\n", hr );
4825                 ok( !lstrcmpW( str, _bstr_("cdatasection") ), "incorrect nodeTypeString string\n");
4826                 SysFreeString(str);
4827
4828                 /* put data Tests */
4829                 hr = IXMLDOMCDATASection_put_data(pCDataSec, _bstr_("This &is a ; test <>\\"));
4830                 ok(hr == S_OK, "ret %08x\n", hr );
4831
4832                 /* Confirm XML text is good */
4833                 hr = IXMLDOMCDATASection_get_xml(pCDataSec, &str);
4834                 ok(hr == S_OK, "ret %08x\n", hr );
4835                 ok( !lstrcmpW( str, _bstr_("<![CDATA[This &is a ; test <>\\]]>") ), "incorrect xml string\n");
4836                 SysFreeString(str);
4837
4838                 /* Confirm we get the put_data Text back */
4839                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
4840                 ok(hr == S_OK, "ret %08x\n", hr );
4841                 ok( !lstrcmpW( str, _bstr_("This &is a ; test <>\\") ), "incorrect text string\n");
4842                 SysFreeString(str);
4843
4844                 /* test length property */
4845                 hr = IXMLDOMCDATASection_get_length(pCDataSec, &len);
4846                 ok(hr == S_OK, "ret %08x\n", hr );
4847                 ok(len == 21, "expected 21 got %d\n", len);
4848
4849                 /* test get nodeValue */
4850                 hr = IXMLDOMCDATASection_get_nodeValue(pCDataSec, &var);
4851                 ok(hr == S_OK, "ret %08x\n", hr );
4852                 ok(V_VT(&var) == VT_BSTR, "got vt %04x\n", V_VT(&var));
4853                 ok( !lstrcmpW( V_BSTR(&var), _bstr_("This &is a ; test <>\\") ), "incorrect text string\n");
4854                 VariantClear(&var);
4855
4856                 /* test get data */
4857                 hr = IXMLDOMCDATASection_get_data(pCDataSec, &str);
4858                 ok(hr == S_OK, "ret %08x\n", hr );
4859                 ok( !lstrcmpW( str, _bstr_("This &is a ; test <>\\") ), "incorrect text string\n");
4860                 SysFreeString(str);
4861
4862                 /* test substringData */
4863                 hr = IXMLDOMCDATASection_substringData(pCDataSec, 0, 4, NULL);
4864                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4865
4866                 /* test substringData - Invalid offset */
4867                 str = (BSTR)&szElement;
4868                 hr = IXMLDOMCDATASection_substringData(pCDataSec, -1, 4, &str);
4869                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4870                 ok( str == NULL, "incorrect string\n");
4871
4872                 /* test substringData - Invalid offset */
4873                 str = (BSTR)&szElement;
4874                 hr = IXMLDOMCDATASection_substringData(pCDataSec, 30, 0, &str);
4875                 ok(hr == S_FALSE, "ret %08x\n", hr );
4876                 ok( str == NULL, "incorrect string\n");
4877
4878                 /* test substringData - Invalid size */
4879                 str = (BSTR)&szElement;
4880                 hr = IXMLDOMCDATASection_substringData(pCDataSec, 0, -1, &str);
4881                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4882                 ok( str == NULL, "incorrect string\n");
4883
4884                 /* test substringData - Invalid size */
4885                 str = (BSTR)&szElement;
4886                 hr = IXMLDOMCDATASection_substringData(pCDataSec, 2, 0, &str);
4887                 ok(hr == S_FALSE, "ret %08x\n", hr );
4888                 ok( str == NULL, "incorrect string\n");
4889
4890                 /* test substringData - Start of string */
4891                 hr = IXMLDOMCDATASection_substringData(pCDataSec, 0, 4, &str);
4892                 ok(hr == S_OK, "ret %08x\n", hr );
4893                 ok( !lstrcmpW( str, _bstr_("This") ), "incorrect substringData string\n");
4894                 SysFreeString(str);
4895
4896                 /* test substringData - Middle of string */
4897                 hr = IXMLDOMCDATASection_substringData(pCDataSec, 13, 4, &str);
4898                 ok(hr == S_OK, "ret %08x\n", hr );
4899                 ok( !lstrcmpW( str, _bstr_("test") ), "incorrect substringData string\n");
4900                 SysFreeString(str);
4901
4902                 /* test substringData - End of string */
4903                 hr = IXMLDOMCDATASection_substringData(pCDataSec, 20, 4, &str);
4904                 ok(hr == S_OK, "ret %08x\n", hr );
4905                 ok( !lstrcmpW( str, _bstr_("\\") ), "incorrect substringData string\n");
4906                 SysFreeString(str);
4907
4908                 /* test appendData */
4909                 hr = IXMLDOMCDATASection_appendData(pCDataSec, NULL);
4910                 ok(hr == S_OK, "ret %08x\n", hr );
4911
4912                 hr = IXMLDOMCDATASection_appendData(pCDataSec, _bstr_(""));
4913                 ok(hr == S_OK, "ret %08x\n", hr );
4914
4915                 hr = IXMLDOMCDATASection_appendData(pCDataSec, _bstr_("Append"));
4916                 ok(hr == S_OK, "ret %08x\n", hr );
4917
4918                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
4919                 ok(hr == S_OK, "ret %08x\n", hr );
4920                 ok( !lstrcmpW( str, _bstr_("This &is a ; test <>\\Append") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4921                 SysFreeString(str);
4922
4923                 /* test insertData */
4924                 str = SysAllocStringLen(NULL, 0);
4925                 hr = IXMLDOMCDATASection_insertData(pCDataSec, -1, str);
4926                 ok(hr == S_OK, "ret %08x\n", hr );
4927
4928                 hr = IXMLDOMCDATASection_insertData(pCDataSec, -1, NULL);
4929                 ok(hr == S_OK, "ret %08x\n", hr );
4930
4931                 hr = IXMLDOMCDATASection_insertData(pCDataSec, 1000, str);
4932                 ok(hr == S_OK, "ret %08x\n", hr );
4933
4934                 hr = IXMLDOMCDATASection_insertData(pCDataSec, 1000, NULL);
4935                 ok(hr == S_OK, "ret %08x\n", hr );
4936
4937                 hr = IXMLDOMCDATASection_insertData(pCDataSec, 0, NULL);
4938                 ok(hr == S_OK, "ret %08x\n", hr );
4939
4940                 hr = IXMLDOMCDATASection_insertData(pCDataSec, 0, str);
4941                 ok(hr == S_OK, "ret %08x\n", hr );
4942                 SysFreeString(str);
4943
4944                 hr = IXMLDOMCDATASection_insertData(pCDataSec, -1, _bstr_("Inserting"));
4945                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4946
4947                 hr = IXMLDOMCDATASection_insertData(pCDataSec, 1000, _bstr_("Inserting"));
4948                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4949
4950                 hr = IXMLDOMCDATASection_insertData(pCDataSec, 0, _bstr_("Begin "));
4951                 ok(hr == S_OK, "ret %08x\n", hr );
4952
4953                 hr = IXMLDOMCDATASection_insertData(pCDataSec, 17, _bstr_("Middle"));
4954                 ok(hr == S_OK, "ret %08x\n", hr );
4955
4956                 hr = IXMLDOMCDATASection_insertData(pCDataSec, 39, _bstr_(" End"));
4957                 ok(hr == S_OK, "ret %08x\n", hr );
4958
4959                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
4960                 ok(hr == S_OK, "ret %08x\n", hr );
4961                 ok( !lstrcmpW( str, _bstr_("Begin This &is a Middle; test <>\\Append End") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4962                 SysFreeString(str);
4963
4964                 /* delete data */
4965                 /* invalid arguments */
4966                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, -1, 1);
4967                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4968
4969                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, 0, 0);
4970                 ok(hr == S_OK, "ret %08x\n", hr );
4971
4972                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, 0, -1);
4973                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4974
4975                 hr = IXMLDOMCDATASection_get_length(pCDataSec, &len);
4976                 ok(hr == S_OK, "ret %08x\n", hr );
4977                 ok(len == 43, "expected 43 got %d\n", len);
4978
4979                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, len, 1);
4980                 ok(hr == S_OK, "ret %08x\n", hr );
4981
4982                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, len+1, 1);
4983                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4984
4985                 /* delete from start */
4986                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, 0, 5);
4987                 ok(hr == S_OK, "ret %08x\n", hr );
4988
4989                 hr = IXMLDOMCDATASection_get_length(pCDataSec, &len);
4990                 ok(hr == S_OK, "ret %08x\n", hr );
4991                 ok(len == 38, "expected 38 got %d\n", len);
4992
4993                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
4994                 ok(hr == S_OK, "ret %08x\n", hr );
4995                 ok( !lstrcmpW( str, _bstr_(" This &is a Middle; test <>\\Append End") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4996                 SysFreeString(str);
4997
4998                 /* delete from end */
4999                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, 35, 3);
5000                 ok(hr == S_OK, "ret %08x\n", hr );
5001
5002                 hr = IXMLDOMCDATASection_get_length(pCDataSec, &len);
5003                 ok(hr == S_OK, "ret %08x\n", hr );
5004                 ok(len == 35, "expected 35 got %d\n", len);
5005
5006                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
5007                 ok(hr == S_OK, "ret %08x\n", hr );
5008                 ok( !lstrcmpW( str, _bstr_(" This &is a Middle; test <>\\Append ") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
5009                 SysFreeString(str);
5010
5011                 /* delete from inside */
5012                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, 1, 33);
5013                 ok(hr == S_OK, "ret %08x\n", hr );
5014
5015                 hr = IXMLDOMCDATASection_get_length(pCDataSec, &len);
5016                 ok(hr == S_OK, "ret %08x\n", hr );
5017                 ok(len == 2, "expected 2 got %d\n", len);
5018
5019                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
5020                 ok(hr == S_OK, "ret %08x\n", hr );
5021                 ok( !lstrcmpW( str, _bstr_("  ") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
5022                 SysFreeString(str);
5023
5024                 /* delete whole data ... */
5025                 hr = IXMLDOMCDATASection_get_length(pCDataSec, &len);
5026                 ok(hr == S_OK, "ret %08x\n", hr );
5027
5028                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, 0, len);
5029                 ok(hr == S_OK, "ret %08x\n", hr );
5030
5031                 /* ... and try again with empty string */
5032                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, 0, len);
5033                 ok(hr == S_OK, "ret %08x\n", hr );
5034
5035                 /* ::replaceData() */
5036                 V_VT(&v) = VT_BSTR;
5037                 V_BSTR(&v) = SysAllocString(szstr1);
5038                 hr = IXMLDOMCDATASection_put_nodeValue(pCDataSec, v);
5039                 ok(hr == S_OK, "ret %08x\n", hr );
5040                 VariantClear(&v);
5041
5042                 hr = IXMLDOMCDATASection_replaceData(pCDataSec, 6, 0, NULL);
5043                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5044                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
5045                 ok(hr == S_OK, "ret %08x\n", hr );
5046                 ok( !lstrcmpW( str, _bstr_("str1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
5047                 SysFreeString(str);
5048
5049                 hr = IXMLDOMCDATASection_replaceData(pCDataSec, 0, 0, NULL);
5050                 ok(hr == S_OK, "ret %08x\n", hr );
5051                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
5052                 ok(hr == S_OK, "ret %08x\n", hr );
5053                 ok( !lstrcmpW( str, _bstr_("str1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
5054                 SysFreeString(str);
5055
5056                 /* NULL pointer means delete */
5057                 hr = IXMLDOMCDATASection_replaceData(pCDataSec, 0, 1, NULL);
5058                 ok(hr == S_OK, "ret %08x\n", hr );
5059                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
5060                 ok(hr == S_OK, "ret %08x\n", hr );
5061                 ok( !lstrcmpW( str, _bstr_("tr1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
5062                 SysFreeString(str);
5063
5064                 /* empty string means delete */
5065                 hr = IXMLDOMCDATASection_replaceData(pCDataSec, 0, 1, _bstr_(""));
5066                 ok(hr == S_OK, "ret %08x\n", hr );
5067                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
5068                 ok(hr == S_OK, "ret %08x\n", hr );
5069                 ok( !lstrcmpW( str, _bstr_("r1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
5070                 SysFreeString(str);
5071
5072                 /* zero count means insert */
5073                 hr = IXMLDOMCDATASection_replaceData(pCDataSec, 0, 0, _bstr_("a"));
5074                 ok(hr == S_OK, "ret %08x\n", hr );
5075                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
5076                 ok(hr == S_OK, "ret %08x\n", hr );
5077                 ok( !lstrcmpW( str, _bstr_("ar1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
5078                 SysFreeString(str);
5079
5080                 hr = IXMLDOMCDATASection_replaceData(pCDataSec, 0, 2, NULL);
5081                 ok(hr == S_OK, "ret %08x\n", hr );
5082
5083                 hr = IXMLDOMCDATASection_insertData(pCDataSec, 0, _bstr_("m"));
5084                 ok(hr == S_OK, "ret %08x\n", hr );
5085                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
5086                 ok(hr == S_OK, "ret %08x\n", hr );
5087                 ok( !lstrcmpW( str, _bstr_("m1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
5088                 SysFreeString(str);
5089
5090                 /* nonempty string, count greater than its length */
5091                 hr = IXMLDOMCDATASection_replaceData(pCDataSec, 0, 2, _bstr_("a1.2"));
5092                 ok(hr == S_OK, "ret %08x\n", hr );
5093                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
5094                 ok(hr == S_OK, "ret %08x\n", hr );
5095                 ok( !lstrcmpW( str, _bstr_("a1.2") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
5096                 SysFreeString(str);
5097
5098                 /* nonempty string, count less than its length */
5099                 hr = IXMLDOMCDATASection_replaceData(pCDataSec, 0, 1, _bstr_("wine"));
5100                 ok(hr == S_OK, "ret %08x\n", hr );
5101                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
5102                 ok(hr == S_OK, "ret %08x\n", hr );
5103                 ok( !lstrcmpW( str, _bstr_("wine1.2") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
5104                 SysFreeString(str);
5105
5106                 IXMLDOMCDATASection_Release(pCDataSec);
5107             }
5108
5109             /* Document Fragments */
5110             hr = IXMLDOMDocument_createDocumentFragment(doc, NULL);
5111             ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5112
5113             hr = IXMLDOMDocument_createDocumentFragment(doc, &pDocFrag);
5114             ok(hr == S_OK, "ret %08x\n", hr );
5115             if(hr == S_OK)
5116             {
5117                 IXMLDOMNode *node;
5118
5119                 hr = IXMLDOMDocumentFragment_get_parentNode(pDocFrag, NULL);
5120                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5121
5122                 node = (IXMLDOMNode *)0x1;
5123                 hr = IXMLDOMDocumentFragment_get_parentNode(pDocFrag, &node);
5124                 ok(hr == S_FALSE, "ret %08x\n", hr );
5125                 ok(node == NULL, "expected NULL, got %p\n", node);
5126
5127                 hr = IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pDocFrag, NULL);
5128                 ok(hr == S_OK, "ret %08x\n", hr );
5129
5130                 /* get Attribute Tests */
5131                 hr = IXMLDOMDocumentFragment_get_attributes(pDocFrag, NULL);
5132                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5133
5134                 pAttribs = (IXMLDOMNamedNodeMap*)0x1;
5135                 hr = IXMLDOMDocumentFragment_get_attributes(pDocFrag, &pAttribs);
5136                 ok(hr == S_FALSE, "ret %08x\n", hr );
5137                 ok(pAttribs == NULL, "pAttribs != NULL\n");
5138
5139                 hr = IXMLDOMDocumentFragment_get_nodeName(pDocFrag, &str);
5140                 ok(hr == S_OK, "ret %08x\n", hr );
5141                 ok( !lstrcmpW( str, szDocFragmentText ), "incorrect docfragment node Name\n");
5142                 SysFreeString(str);
5143
5144                 /* test next Sibling*/
5145                 hr = IXMLDOMDocumentFragment_get_nextSibling(pDocFrag, NULL);
5146                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5147
5148                 node = (IXMLDOMNode *)0x1;
5149                 hr = IXMLDOMDocumentFragment_get_nextSibling(pDocFrag, &node);
5150                 ok(hr == S_FALSE, "ret %08x\n", hr );
5151                 ok(node == NULL, "next sibling not NULL\n");
5152
5153                 /* test Previous Sibling*/
5154                 hr = IXMLDOMDocumentFragment_get_previousSibling(pDocFrag, NULL);
5155                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5156
5157                 node = (IXMLDOMNode *)0x1;
5158                 hr = IXMLDOMDocumentFragment_get_previousSibling(pDocFrag, &node);
5159                 ok(hr == S_FALSE, "ret %08x\n", hr );
5160                 ok(node == NULL, "previous sibling not NULL\n");
5161
5162                 /* test get_dataType */
5163                 hr = IXMLDOMDocumentFragment_get_dataType(pDocFrag, NULL);
5164                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5165
5166                 hr = IXMLDOMDocumentFragment_get_dataType(pDocFrag, &v);
5167                 ok(hr == S_FALSE, "ret %08x\n", hr );
5168                 ok( V_VT(&v) == VT_NULL, "incorrect dataType type\n");
5169                 VariantClear(&v);
5170
5171                 /* test nodeTypeString */
5172                 hr = IXMLDOMDocumentFragment_get_nodeTypeString(pDocFrag, &str);
5173                 ok(hr == S_OK, "ret %08x\n", hr );
5174                 ok( !lstrcmpW( str, _bstr_("documentfragment") ), "incorrect nodeTypeString string\n");
5175                 SysFreeString(str);
5176
5177                 IXMLDOMDocumentFragment_Release(pDocFrag);
5178             }
5179
5180             /* Entity References */
5181             hr = IXMLDOMDocument_createEntityReference(doc, NULL, &pEntityRef);
5182             ok(hr == E_FAIL, "ret %08x\n", hr );
5183             hr = IXMLDOMDocument_createEntityReference(doc, _bstr_(""), &pEntityRef);
5184             ok(hr == E_FAIL, "ret %08x\n", hr );
5185
5186             str = SysAllocString(szEntityRef);
5187             hr = IXMLDOMDocument_createEntityReference(doc, str, NULL);
5188             ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5189
5190             hr = IXMLDOMDocument_createEntityReference(doc, str, &pEntityRef);
5191             SysFreeString(str);
5192             ok(hr == S_OK, "ret %08x\n", hr );
5193             if(hr == S_OK)
5194             {
5195                 hr = IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pEntityRef, NULL);
5196                 ok(hr == S_OK, "ret %08x\n", hr );
5197
5198                 /* get Attribute Tests */
5199                 hr = IXMLDOMEntityReference_get_attributes(pEntityRef, NULL);
5200                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5201
5202                 pAttribs = (IXMLDOMNamedNodeMap*)0x1;
5203                 hr = IXMLDOMEntityReference_get_attributes(pEntityRef, &pAttribs);
5204                 ok(hr == S_FALSE, "ret %08x\n", hr );
5205                 ok(pAttribs == NULL, "pAttribs != NULL\n");
5206
5207                 /* test dataType */
5208                 hr = IXMLDOMEntityReference_get_dataType(pEntityRef, &v);
5209                 ok(hr == S_FALSE, "ret %08x\n", hr );
5210                 ok( V_VT(&v) == VT_NULL, "incorrect dataType type\n");
5211                 VariantClear(&v);
5212
5213                 /* test nodeTypeString */
5214                 hr = IXMLDOMEntityReference_get_nodeTypeString(pEntityRef, &str);
5215                 ok(hr == S_OK, "ret %08x\n", hr );
5216                 ok( !lstrcmpW( str, _bstr_("entityreference") ), "incorrect nodeTypeString string\n");
5217                 SysFreeString(str);
5218
5219                 /* test get_xml*/
5220                 hr = IXMLDOMEntityReference_get_xml(pEntityRef, &str);
5221                 ok(hr == S_OK, "ret %08x\n", hr );
5222                 ok( !lstrcmpW( str, szEntityRefXML ), "incorrect xml string\n");
5223                 SysFreeString(str);
5224
5225                 IXMLDOMEntityReference_Release(pEntityRef);
5226             }
5227
5228             IXMLDOMElement_Release( pRoot );
5229         }
5230     }
5231
5232     IXMLDOMDocument_Release(doc);
5233
5234     free_bstrs();
5235 }
5236
5237 static void test_nodeTypeTests( void )
5238 {
5239     IXMLDOMDocument *doc = NULL;
5240     IXMLDOMElement *pRoot;
5241     IXMLDOMElement *pElement;
5242     HRESULT hr;
5243
5244     doc = create_document(&IID_IXMLDOMDocument);
5245     if (!doc) return;
5246
5247     hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing"), NULL);
5248     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5249
5250     hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing"), &pRoot);
5251     ok(hr == S_OK, "ret %08x\n", hr );
5252     if(hr == S_OK)
5253     {
5254         hr = IXMLDOMDocument_appendChild(doc, (IXMLDOMNode*)pRoot, NULL);
5255         ok(hr == S_OK, "ret %08x\n", hr );
5256         if(hr == S_OK)
5257         {
5258             hr = IXMLDOMElement_put_dataType(pRoot, NULL);
5259             ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5260
5261             /* Invalid Value */
5262             hr = IXMLDOMElement_put_dataType(pRoot, _bstr_("abcdefg") );
5263             ok(hr == E_FAIL, "ret %08x\n", hr );
5264
5265             /* NOTE:
5266              *   The name passed into put_dataType is case-insensitive. So many of the names
5267              *     have been changed to reflect this.
5268              */
5269             /* Boolean */
5270             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_Boolean"), &pElement);
5271             ok(hr == S_OK, "ret %08x\n", hr );
5272             if(hr == S_OK)
5273             {
5274                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5275
5276                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("Boolean") );
5277                 ok(hr == S_OK, "ret %08x\n", hr );
5278
5279                 IXMLDOMElement_Release(pElement);
5280             }
5281
5282             /* String */
5283             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_String"), &pElement);
5284             ok(hr == S_OK, "ret %08x\n", hr );
5285             if(hr == S_OK)
5286             {
5287                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5288
5289                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("String") );
5290                 ok(hr == S_OK, "ret %08x\n", hr );
5291
5292                 IXMLDOMElement_Release(pElement);
5293             }
5294
5295             /* Number */
5296             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_Number"), &pElement);
5297             ok(hr == S_OK, "ret %08x\n", hr );
5298             if(hr == S_OK)
5299             {
5300                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5301
5302                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("number") );
5303                 ok(hr == S_OK, "ret %08x\n", hr );
5304
5305                 IXMLDOMElement_Release(pElement);
5306             }
5307
5308             /* Int */
5309             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_Int"), &pElement);
5310             ok(hr == S_OK, "ret %08x\n", hr );
5311             if(hr == S_OK)
5312             {
5313                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5314
5315                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("InT") );
5316                 ok(hr == S_OK, "ret %08x\n", hr );
5317
5318                 IXMLDOMElement_Release(pElement);
5319             }
5320
5321             /* Fixed */
5322             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_Fixed"), &pElement);
5323             ok(hr == S_OK, "ret %08x\n", hr );
5324             if(hr == S_OK)
5325             {
5326                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5327
5328                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("fixed.14.4") );
5329                 ok(hr == S_OK, "ret %08x\n", hr );
5330
5331                 IXMLDOMElement_Release(pElement);
5332             }
5333
5334             /* DateTime */
5335             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_DateTime"), &pElement);
5336             ok(hr == S_OK, "ret %08x\n", hr );
5337             if(hr == S_OK)
5338             {
5339                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5340
5341                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("DateTime") );
5342                 ok(hr == S_OK, "ret %08x\n", hr );
5343
5344                 IXMLDOMElement_Release(pElement);
5345             }
5346
5347             /* DateTime TZ */
5348             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_DateTime_tz"), &pElement);
5349             ok(hr == S_OK, "ret %08x\n", hr );
5350             if(hr == S_OK)
5351             {
5352                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5353
5354                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("DateTime.tz") );
5355                 ok(hr == S_OK, "ret %08x\n", hr );
5356
5357                 IXMLDOMElement_Release(pElement);
5358             }
5359
5360             /* Date */
5361             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_Date"), &pElement);
5362             ok(hr == S_OK, "ret %08x\n", hr );
5363             if(hr == S_OK)
5364             {
5365                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5366
5367                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("Date") );
5368                 ok(hr == S_OK, "ret %08x\n", hr );
5369
5370                 IXMLDOMElement_Release(pElement);
5371             }
5372
5373             /* Time */
5374             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_Time"), &pElement);
5375             ok(hr == S_OK, "ret %08x\n", hr );
5376             if(hr == S_OK)
5377             {
5378                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5379
5380                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("Time") );
5381                 ok(hr == S_OK, "ret %08x\n", hr );
5382
5383                 IXMLDOMElement_Release(pElement);
5384             }
5385
5386             /* Time.tz */
5387             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_Time_TZ"), &pElement);
5388             ok(hr == S_OK, "ret %08x\n", hr );
5389             if(hr == S_OK)
5390             {
5391                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5392
5393                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("Time.tz") );
5394                 ok(hr == S_OK, "ret %08x\n", hr );
5395
5396                 IXMLDOMElement_Release(pElement);
5397             }
5398
5399             /* I1 */
5400             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_I1"), &pElement);
5401             ok(hr == S_OK, "ret %08x\n", hr );
5402             if(hr == S_OK)
5403             {
5404                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5405
5406                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("I1") );
5407                 ok(hr == S_OK, "ret %08x\n", hr );
5408
5409                 IXMLDOMElement_Release(pElement);
5410             }
5411
5412             /* I2 */
5413             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_I2"), &pElement);
5414             ok(hr == S_OK, "ret %08x\n", hr );
5415             if(hr == S_OK)
5416             {
5417                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5418
5419                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("I2") );
5420                 ok(hr == S_OK, "ret %08x\n", hr );
5421
5422                 IXMLDOMElement_Release(pElement);
5423             }
5424
5425             /* I4 */
5426             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_I4"), &pElement);
5427             ok(hr == S_OK, "ret %08x\n", hr );
5428             if(hr == S_OK)
5429             {
5430                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5431
5432                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("I4") );
5433                 ok(hr == S_OK, "ret %08x\n", hr );
5434
5435                 IXMLDOMElement_Release(pElement);
5436             }
5437
5438             /* UI1 */
5439             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_UI1"), &pElement);
5440             ok(hr == S_OK, "ret %08x\n", hr );
5441             if(hr == S_OK)
5442             {
5443                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5444
5445                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("UI1") );
5446                 ok(hr == S_OK, "ret %08x\n", hr );
5447
5448                 IXMLDOMElement_Release(pElement);
5449             }
5450
5451             /* UI2 */
5452             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_UI2"), &pElement);
5453             ok(hr == S_OK, "ret %08x\n", hr );
5454             if(hr == S_OK)
5455             {
5456                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5457
5458                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("UI2") );
5459                 ok(hr == S_OK, "ret %08x\n", hr );
5460
5461                 IXMLDOMElement_Release(pElement);
5462             }
5463
5464             /* UI4 */
5465             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_UI4"), &pElement);
5466             ok(hr == S_OK, "ret %08x\n", hr );
5467             if(hr == S_OK)
5468             {
5469                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5470
5471                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("UI4") );
5472                 ok(hr == S_OK, "ret %08x\n", hr );
5473
5474                 IXMLDOMElement_Release(pElement);
5475             }
5476
5477             /* r4 */
5478             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_r4"), &pElement);
5479             ok(hr == S_OK, "ret %08x\n", hr );
5480             if(hr == S_OK)
5481             {
5482                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5483
5484                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("r4") );
5485                 ok(hr == S_OK, "ret %08x\n", hr );
5486
5487                 IXMLDOMElement_Release(pElement);
5488             }
5489
5490             /* r8 */
5491             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_r8"), &pElement);
5492             ok(hr == S_OK, "ret %08x\n", hr );
5493             if(hr == S_OK)
5494             {
5495                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5496
5497                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("r8") );
5498                 ok(hr == S_OK, "ret %08x\n", hr );
5499
5500                 IXMLDOMElement_Release(pElement);
5501             }
5502
5503             /* float */
5504             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_float"), &pElement);
5505             ok(hr == S_OK, "ret %08x\n", hr );
5506             if(hr == S_OK)
5507             {
5508                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5509
5510                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("float") );
5511                 ok(hr == S_OK, "ret %08x\n", hr );
5512
5513                 IXMLDOMElement_Release(pElement);
5514             }
5515
5516             /* uuid */
5517             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_uuid"), &pElement);
5518             ok(hr == S_OK, "ret %08x\n", hr );
5519             if(hr == S_OK)
5520             {
5521                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5522
5523                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("UuId") );
5524                 ok(hr == S_OK, "ret %08x\n", hr );
5525
5526                 IXMLDOMElement_Release(pElement);
5527             }
5528
5529             /* bin.hex */
5530             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_bin_hex"), &pElement);
5531             ok(hr == S_OK, "ret %08x\n", hr );
5532             if(hr == S_OK)
5533             {
5534                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5535
5536                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("bin.hex") );
5537                 ok(hr == S_OK, "ret %08x\n", hr );
5538
5539                 IXMLDOMElement_Release(pElement);
5540             }
5541
5542             /* bin.base64 */
5543             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_bin_base64"), &pElement);
5544             ok(hr == S_OK, "ret %08x\n", hr );
5545             if(hr == S_OK)
5546             {
5547                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5548
5549                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("bin.base64") );
5550                 ok(hr == S_OK, "ret %08x\n", hr );
5551
5552                 IXMLDOMElement_Release(pElement);
5553             }
5554
5555             /* Check changing types */
5556             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_Change"), &pElement);
5557             ok(hr == S_OK, "ret %08x\n", hr );
5558             if(hr == S_OK)
5559             {
5560                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5561
5562                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("DateTime.tz") );
5563                 ok(hr == S_OK, "ret %08x\n", hr );
5564
5565                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("string") );
5566                 ok(hr == S_OK, "ret %08x\n", hr );
5567
5568                 IXMLDOMElement_Release(pElement);
5569             }
5570
5571             IXMLDOMElement_Release(pRoot);
5572         }
5573     }
5574
5575     IXMLDOMDocument_Release(doc);
5576
5577     free_bstrs();
5578 }
5579
5580 static void test_DocumentSaveToDocument(void)
5581 {
5582     IXMLDOMDocument *doc, *doc2;
5583     IXMLDOMElement *pRoot;
5584     HRESULT hr;
5585
5586     doc = create_document(&IID_IXMLDOMDocument);
5587     if (!doc) return;
5588
5589     doc2 = create_document(&IID_IXMLDOMDocument);
5590     if (!doc2)
5591     {
5592         IXMLDOMDocument_Release(doc);
5593         return;
5594     }
5595
5596     hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing"), &pRoot);
5597     ok(hr == S_OK, "ret %08x\n", hr );
5598     if(hr == S_OK)
5599     {
5600         hr = IXMLDOMDocument_appendChild(doc, (IXMLDOMNode*)pRoot, NULL);
5601         ok(hr == S_OK, "ret %08x\n", hr );
5602         if(hr == S_OK)
5603         {
5604             VARIANT vDoc;
5605             BSTR sOrig;
5606             BSTR sNew;
5607
5608             V_VT(&vDoc) = VT_UNKNOWN;
5609             V_UNKNOWN(&vDoc) = (IUnknown*)doc2;
5610
5611             hr = IXMLDOMDocument_save(doc, vDoc);
5612             ok(hr == S_OK, "ret %08x\n", hr );
5613
5614             hr = IXMLDOMDocument_get_xml(doc, &sOrig);
5615             ok(hr == S_OK, "ret %08x\n", hr );
5616
5617             hr = IXMLDOMDocument_get_xml(doc2, &sNew);
5618             ok(hr == S_OK, "ret %08x\n", hr );
5619
5620             ok( !lstrcmpW( sOrig, sNew ), "New document is not the same as origial\n");
5621
5622             SysFreeString(sOrig);
5623             SysFreeString(sNew);
5624         }
5625     }
5626
5627     IXMLDOMDocument_Release(doc2);
5628     IXMLDOMDocument_Release(doc);
5629 }
5630
5631 static void test_DocumentSaveToFile(void)
5632 {
5633     IXMLDOMDocument *doc;
5634     IXMLDOMElement *pRoot;
5635     HANDLE file;
5636     char buffer[100];
5637     DWORD read = 0;
5638     HRESULT hr;
5639
5640     doc = create_document(&IID_IXMLDOMDocument);
5641     if (!doc) return;
5642
5643     hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing"), &pRoot);
5644     ok(hr == S_OK, "ret %08x\n", hr );
5645     if(hr == S_OK)
5646     {
5647         hr = IXMLDOMDocument_appendChild(doc, (IXMLDOMNode*)pRoot, NULL);
5648         ok(hr == S_OK, "ret %08x\n", hr );
5649         if(hr == S_OK)
5650         {
5651             VARIANT vFile;
5652
5653             V_VT(&vFile) = VT_BSTR;
5654             V_BSTR(&vFile) = _bstr_("test.xml");
5655
5656             hr = IXMLDOMDocument_save(doc, vFile);
5657             ok(hr == S_OK, "ret %08x\n", hr );
5658         }
5659     }
5660
5661     IXMLDOMElement_Release(pRoot);
5662     IXMLDOMDocument_Release(doc);
5663
5664     file = CreateFile("test.xml", GENERIC_READ, 0, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL );
5665     ok(file != INVALID_HANDLE_VALUE, "Could not open file: %u\n", GetLastError());
5666     if(file == INVALID_HANDLE_VALUE)
5667         return;
5668
5669     ReadFile(file, buffer, sizeof(buffer), &read, NULL);
5670     ok(read != 0, "could not read file\n");
5671     ok(buffer[0] != '<' || buffer[1] != '?', "File contains processing instruction\n");
5672
5673     CloseHandle(file);
5674     DeleteFile("test.xml");
5675 }
5676
5677 static void test_testTransforms(void)
5678 {
5679     IXMLDOMDocument *doc, *docSS;
5680     IXMLDOMNode *pNode;
5681     VARIANT_BOOL bSucc;
5682
5683     HRESULT hr;
5684
5685     doc = create_document(&IID_IXMLDOMDocument);
5686     if (!doc) return;
5687
5688     docSS = create_document(&IID_IXMLDOMDocument);
5689     if (!docSS)
5690     {
5691         IXMLDOMDocument_Release(doc);
5692         return;
5693     }
5694
5695     hr = IXMLDOMDocument_loadXML(doc, _bstr_(szTransformXML), &bSucc);
5696     ok(hr == S_OK, "ret %08x\n", hr );
5697
5698     hr = IXMLDOMDocument_loadXML(docSS, _bstr_(szTransformSSXML), &bSucc);
5699     ok(hr == S_OK, "ret %08x\n", hr );
5700
5701     hr = IXMLDOMDocument_QueryInterface(docSS, &IID_IXMLDOMNode, (void**)&pNode );
5702     ok(hr == S_OK, "ret %08x\n", hr );
5703     if(hr == S_OK)
5704     {
5705         BSTR bOut;
5706
5707         hr = IXMLDOMDocument_transformNode(doc, pNode, &bOut);
5708         ok(hr == S_OK, "ret %08x\n", hr );
5709         ok( compareIgnoreReturns( bOut, _bstr_(szTransformOutput)), "Stylesheet output not correct\n");
5710         SysFreeString(bOut);
5711
5712         IXMLDOMNode_Release(pNode);
5713     }
5714
5715     IXMLDOMDocument_Release(docSS);
5716     IXMLDOMDocument_Release(doc);
5717
5718     free_bstrs();
5719 }
5720
5721 static void test_Namespaces(void)
5722 {
5723     IXMLDOMDocument *doc;
5724     IXMLDOMNode *pNode;
5725     IXMLDOMNode *pNode2 = NULL;
5726     VARIANT_BOOL bSucc;
5727     HRESULT hr;
5728     BSTR str;
5729     static  const CHAR szNamespacesXML[] =
5730 "<?xml version=\"1.0\"?>\n"
5731 "<root xmlns:WEB='http://www.winehq.org'>\n"
5732 "<WEB:Site version=\"1.0\" />\n"
5733 "</root>";
5734
5735     doc = create_document(&IID_IXMLDOMDocument);
5736     if (!doc) return;
5737
5738     hr = IXMLDOMDocument_loadXML(doc, _bstr_(szNamespacesXML), &bSucc);
5739     ok(hr == S_OK, "ret %08x\n", hr );
5740     ok(bSucc == VARIANT_TRUE, "Expected VARIANT_TRUE got VARIANT_FALSE\n");
5741
5742     hr = IXMLDOMDocument_selectSingleNode(doc, _bstr_("root"), &pNode );
5743     ok(hr == S_OK, "ret %08x\n", hr );
5744     if(hr == S_OK)
5745     {
5746         hr = IXMLDOMNode_get_firstChild( pNode, &pNode2 );
5747         ok( hr == S_OK, "ret %08x\n", hr );
5748         ok( pNode2 != NULL, "pNode2 == NULL\n");
5749
5750         /* Test get_prefix */
5751         hr = IXMLDOMNode_get_prefix(pNode2, NULL);
5752         ok( hr == E_INVALIDARG, "ret %08x\n", hr );
5753         /* NOTE: Need to test that arg2 gets cleared on Error. */
5754
5755         hr = IXMLDOMNode_get_prefix(pNode2, &str);
5756         ok( hr == S_OK, "ret %08x\n", hr );
5757         ok( !lstrcmpW( str, _bstr_("WEB")), "incorrect prefix string\n");
5758         SysFreeString(str);
5759
5760         /* Test get_namespaceURI */
5761         hr = IXMLDOMNode_get_namespaceURI(pNode2, NULL);
5762         ok( hr == E_INVALIDARG, "ret %08x\n", hr );
5763         /* NOTE: Need to test that arg2 gets cleared on Error. */
5764
5765         hr = IXMLDOMNode_get_namespaceURI(pNode2, &str);
5766         ok( hr == S_OK, "ret %08x\n", hr );
5767         ok( !lstrcmpW( str, _bstr_("http://www.winehq.org")), "incorrect namespaceURI string\n");
5768         SysFreeString(str);
5769
5770         IXMLDOMNode_Release(pNode2);
5771         IXMLDOMNode_Release(pNode);
5772     }
5773
5774     IXMLDOMDocument_Release(doc);
5775
5776     free_bstrs();
5777 }
5778
5779 static void test_FormattingXML(void)
5780 {
5781     IXMLDOMDocument *doc;
5782     IXMLDOMElement *pElement;
5783     VARIANT_BOOL bSucc;
5784     HRESULT hr;
5785     BSTR str;
5786     static const CHAR szLinefeedXML[] = "<?xml version=\"1.0\"?>\n<Root>\n\t<Sub val=\"A\" />\n</Root>";
5787     static const CHAR szLinefeedRootXML[] = "<Root>\r\n\t<Sub val=\"A\"/>\r\n</Root>";
5788
5789     doc = create_document(&IID_IXMLDOMDocument);
5790     if (!doc) return;
5791
5792     hr = IXMLDOMDocument_loadXML(doc, _bstr_(szLinefeedXML), &bSucc);
5793     ok(hr == S_OK, "ret %08x\n", hr );
5794     ok(bSucc == VARIANT_TRUE, "Expected VARIANT_TRUE got VARIANT_FALSE\n");
5795
5796     if(bSucc == VARIANT_TRUE)
5797     {
5798         hr = IXMLDOMDocument_get_documentElement(doc, &pElement);
5799         ok(hr == S_OK, "ret %08x\n", hr );
5800         if(hr == S_OK)
5801         {
5802             hr = IXMLDOMElement_get_xml(pElement, &str);
5803             ok(hr == S_OK, "ret %08x\n", hr );
5804             ok( !lstrcmpW( str, _bstr_(szLinefeedRootXML) ), "incorrect element xml\n");
5805             SysFreeString(str);
5806
5807             IXMLDOMElement_Release(pElement);
5808         }
5809     }
5810
5811     IXMLDOMDocument_Release(doc);
5812
5813     free_bstrs();
5814 }
5815
5816 typedef struct _nodetypedvalue_t {
5817     const char *name;
5818     VARTYPE type;
5819     const char *value; /* value in string format */
5820 } nodetypedvalue_t;
5821
5822 static const nodetypedvalue_t get_nodetypedvalue[] = {
5823     { "root/string",    VT_BSTR, "Wine" },
5824     { "root/string2",   VT_BSTR, "String" },
5825     { "root/number",    VT_BSTR, "12.44" },
5826     { "root/number2",   VT_BSTR, "-3.71e3" },
5827     { "root/int",       VT_I4,   "-13" },
5828     { "root/fixed",     VT_CY,   "7322.9371" },
5829     { "root/bool",      VT_BOOL, "-1" },
5830     { "root/datetime",  VT_DATE, "40135.14" },
5831     { "root/datetimetz",VT_DATE, "37813.59" },
5832     { "root/date",      VT_DATE, "665413" },
5833     { "root/time",      VT_DATE, "0.5813889" },
5834     { "root/timetz",    VT_DATE, "1.112512" },
5835     { "root/i1",        VT_I1,   "-13" },
5836     { "root/i2",        VT_I2,   "31915" },
5837     { "root/i4",        VT_I4,   "-312232" },
5838     { "root/ui1",       VT_UI1,  "123" },
5839     { "root/ui2",       VT_UI2,  "48282" },
5840     { "root/ui4",       VT_UI4,  "949281" },
5841     { "root/r4",        VT_R4,   "213124" },
5842     { "root/r8",        VT_R8,   "0.412" },
5843     { "root/float",     VT_R8,   "41221.421" },
5844     { "root/uuid",      VT_BSTR, "333C7BC4-460F-11D0-BC04-0080C7055a83" },
5845     { 0 }
5846 };
5847
5848 static void test_nodeTypedValue(void)
5849 {
5850     const nodetypedvalue_t *entry = get_nodetypedvalue;
5851     IXMLDOMDocument *doc;
5852     IXMLDOMNode *node;
5853     VARIANT_BOOL b;
5854     VARIANT value;
5855     HRESULT hr;
5856
5857     doc = create_document(&IID_IXMLDOMDocument);
5858     if (!doc) return;
5859
5860     b = VARIANT_FALSE;
5861     hr = IXMLDOMDocument_loadXML(doc, _bstr_(szTypeValueXML), &b);
5862     ok(hr == S_OK, "ret %08x\n", hr );
5863     ok(b == VARIANT_TRUE, "got %d\n", b);
5864
5865     hr = IXMLDOMDocument_get_nodeValue(doc, NULL);
5866     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5867
5868     V_VT(&value) = VT_BSTR;
5869     V_BSTR(&value) = NULL;
5870     hr = IXMLDOMDocument_get_nodeValue(doc, &value);
5871     ok(hr == S_FALSE, "ret %08x\n", hr );
5872     ok(V_VT(&value) == VT_NULL, "expect VT_NULL got %d\n", V_VT(&value));
5873
5874     hr = IXMLDOMDocument_get_nodeTypedValue(doc, NULL);
5875     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5876
5877     hr = IXMLDOMDocument_get_nodeTypedValue(doc, &value);
5878     ok(hr == S_FALSE, "ret %08x\n", hr );
5879
5880     hr = IXMLDOMDocument_selectSingleNode(doc, _bstr_("root/string"), &node);
5881     ok(hr == S_OK, "ret %08x\n", hr );
5882
5883     V_VT(&value) = VT_BSTR;
5884     V_BSTR(&value) = NULL;
5885     hr = IXMLDOMNode_get_nodeValue(node, &value);
5886     ok(hr == S_FALSE, "ret %08x\n", hr );
5887     ok(V_VT(&value) == VT_NULL, "expect VT_NULL got %d\n", V_VT(&value));
5888
5889     hr = IXMLDOMNode_get_nodeTypedValue(node, NULL);
5890     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5891
5892     IXMLDOMNode_Release(node);
5893
5894     hr = IXMLDOMDocument_selectSingleNode(doc, _bstr_("root/binhex"), &node);
5895     ok(hr == S_OK, "ret %08x\n", hr );
5896     {
5897         BYTE bytes[] = {0xff,0xfc,0xa0,0x12,0x00,0x3c};
5898
5899         hr = IXMLDOMNode_get_nodeTypedValue(node, &value);
5900         ok(hr == S_OK, "ret %08x\n", hr );
5901         ok(V_VT(&value) == (VT_ARRAY|VT_UI1), "incorrect type\n");
5902         ok(V_ARRAY(&value)->rgsabound[0].cElements == 6, "incorrect array size\n");
5903         if(V_ARRAY(&value)->rgsabound[0].cElements == 6)
5904             ok(!memcmp(bytes, V_ARRAY(&value)->pvData, sizeof(bytes)), "incorrect value\n");
5905         VariantClear(&value);
5906         IXMLDOMNode_Release(node);
5907     }
5908
5909     hr = IXMLDOMDocument_selectSingleNode(doc, _bstr_("root/binbase64"), &node);
5910     ok(hr == S_OK, "ret %08x\n", hr );
5911     {
5912         BYTE bytes[] = {0x62,0x61,0x73,0x65,0x36,0x34,0x20,0x74,0x65,0x73,0x74};
5913
5914         hr = IXMLDOMNode_get_nodeTypedValue(node, &value);
5915         ok(hr == S_OK, "ret %08x\n", hr );
5916         ok(V_VT(&value) == (VT_ARRAY|VT_UI1), "incorrect type\n");
5917         ok(V_ARRAY(&value)->rgsabound[0].cElements == 11, "incorrect array size\n");
5918         if(V_ARRAY(&value)->rgsabound[0].cElements == 11)
5919             ok(!memcmp(bytes, V_ARRAY(&value)->pvData, sizeof(bytes)), "incorrect value\n");
5920         VariantClear(&value);
5921         IXMLDOMNode_Release(node);
5922     }
5923
5924     while (entry->name)
5925     {
5926         hr = IXMLDOMDocument_selectSingleNode(doc, _bstr_(entry->name), &node);
5927         ok(hr == S_OK, "ret %08x\n", hr );
5928
5929         hr = IXMLDOMNode_get_nodeTypedValue(node, &value);
5930         ok(hr == S_OK, "ret %08x\n", hr );
5931         ok(V_VT(&value) == entry->type, "incorrect type, expected %d, got %d\n", entry->type, V_VT(&value));
5932
5933         if (entry->type != VT_BSTR)
5934         {
5935            if (entry->type == VT_DATE ||
5936                entry->type == VT_R8 ||
5937                entry->type == VT_CY)
5938            {
5939                if (entry->type == VT_DATE)
5940                {
5941                    hr = VariantChangeType(&value, &value, 0, VT_R4);
5942                    ok(hr == S_OK, "ret %08x\n", hr );
5943                }
5944                hr = VariantChangeTypeEx(&value, &value,
5945                                         MAKELCID(MAKELANGID(LANG_ENGLISH, SUBLANG_DEFAULT), SORT_DEFAULT),
5946                                         VARIANT_NOUSEROVERRIDE, VT_BSTR);
5947                ok(hr == S_OK, "ret %08x\n", hr );
5948            }
5949            else
5950            {
5951                hr = VariantChangeType(&value, &value, 0, VT_BSTR);
5952                ok(hr == S_OK, "ret %08x\n", hr );
5953            }
5954
5955            ok(lstrcmpW( V_BSTR(&value), _bstr_(entry->value)) == 0,
5956               "expected %s, got %s\n", entry->value, wine_dbgstr_w(V_BSTR(&value)));
5957         }
5958         else
5959            ok(lstrcmpW( V_BSTR(&value), _bstr_(entry->value)) == 0,
5960                "expected %s, got %s\n", entry->value, wine_dbgstr_w(V_BSTR(&value)));
5961
5962         VariantClear( &value );
5963         IXMLDOMNode_Release(node);
5964
5965         entry++;
5966     }
5967
5968     IXMLDOMDocument_Release(doc);
5969     free_bstrs();
5970 }
5971
5972 static void test_TransformWithLoadingLocalFile(void)
5973 {
5974     IXMLDOMDocument *doc;
5975     IXMLDOMDocument *xsl;
5976     IXMLDOMNode *pNode;
5977     VARIANT_BOOL bSucc;
5978     HRESULT hr;
5979     HANDLE file;
5980     DWORD dwWritten;
5981     char lpPathBuffer[MAX_PATH];
5982     int i;
5983
5984     /* Create a Temp File. */
5985     GetTempPathA(MAX_PATH, lpPathBuffer);
5986     strcat(lpPathBuffer, "customers.xml" );
5987
5988     file = CreateFile(lpPathBuffer, GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL );
5989     ok(file != INVALID_HANDLE_VALUE, "Could not create file: %u\n", GetLastError());
5990     if(file == INVALID_HANDLE_VALUE)
5991         return;
5992
5993     WriteFile(file, szBasicTransformXML, strlen(szBasicTransformXML), &dwWritten, NULL);
5994     CloseHandle(file);
5995
5996     /* Correct path to not include a escape character. */
5997     for(i=0; i < strlen(lpPathBuffer); i++)
5998     {
5999         if(lpPathBuffer[i] == '\\')
6000             lpPathBuffer[i] = '/';
6001     }
6002
6003     doc = create_document(&IID_IXMLDOMDocument);
6004     if (!doc) return;
6005
6006     xsl = create_document(&IID_IXMLDOMDocument);
6007     if (!xsl)
6008     {
6009         IXMLDOMDocument2_Release(doc);
6010         return;
6011     }
6012
6013     hr = IXMLDOMDocument_loadXML(doc, _bstr_(szTypeValueXML), &bSucc);
6014     ok(hr == S_OK, "ret %08x\n", hr );
6015     ok(bSucc == VARIANT_TRUE, "Expected VARIANT_TRUE got VARIANT_FALSE\n");
6016     if(bSucc == VARIANT_TRUE)
6017     {
6018         BSTR sXSL;
6019         BSTR sPart1 = _bstr_(szBasicTransformSSXMLPart1);
6020         BSTR sPart2 = _bstr_(szBasicTransformSSXMLPart2);
6021         BSTR sFileName = _bstr_(lpPathBuffer);
6022         int nLegnth = lstrlenW(sPart1) + lstrlenW(sPart2) + lstrlenW(sFileName) + 1;
6023
6024         sXSL = SysAllocStringLen(NULL, nLegnth);
6025         lstrcpyW(sXSL, sPart1);
6026         lstrcatW(sXSL, sFileName);
6027         lstrcatW(sXSL, sPart2);
6028
6029         hr = IXMLDOMDocument_loadXML(xsl, sXSL, &bSucc);
6030         ok(hr == S_OK, "ret %08x\n", hr );
6031         ok(bSucc == VARIANT_TRUE, "Expected VARIANT_TRUE got VARIANT_FALSE\n");
6032         if(bSucc == VARIANT_TRUE)
6033         {
6034             BSTR sResult;
6035
6036             hr = IXMLDOMDocument_QueryInterface(xsl, &IID_IXMLDOMNode, (void**)&pNode );
6037             ok(hr == S_OK, "ret %08x\n", hr );
6038             if(hr == S_OK)
6039             {
6040                 /* This will load the temp file via the XSL */
6041                 hr = IXMLDOMDocument_transformNode(doc, pNode, &sResult);
6042                 ok(hr == S_OK, "ret %08x\n", hr );
6043                 if(hr == S_OK)
6044                 {
6045                     ok( compareIgnoreReturns( sResult, _bstr_(szBasicTransformOutput)), "Stylesheet output not correct\n");
6046                     SysFreeString(sResult);
6047                 }
6048
6049                 IXMLDOMNode_Release(pNode);
6050             }
6051         }
6052
6053         SysFreeString(sXSL);
6054     }
6055
6056     IXMLDOMDocument_Release(doc);
6057     IXMLDOMDocument_Release(xsl);
6058
6059     DeleteFile(lpPathBuffer);
6060     free_bstrs();
6061 }
6062
6063 static void test_put_nodeValue(void)
6064 {
6065     static const WCHAR jeevesW[] = {'J','e','e','v','e','s',' ','&',' ','W','o','o','s','t','e','r',0};
6066     IXMLDOMDocument *doc;
6067     IXMLDOMText *text;
6068     IXMLDOMEntityReference *entityref;
6069     IXMLDOMAttribute *attr;
6070     IXMLDOMNode *node;
6071     HRESULT hr;
6072     VARIANT data, type;
6073
6074     doc = create_document(&IID_IXMLDOMDocument);
6075     if (!doc) return;
6076
6077     /* test for unsupported types */
6078     /* NODE_DOCUMENT */
6079     hr = IXMLDOMDocument_QueryInterface(doc, &IID_IXMLDOMNode, (void**)&node);
6080     ok(hr == S_OK, "ret %08x\n", hr );
6081     V_VT(&data) = VT_BSTR;
6082     V_BSTR(&data) = _bstr_("one two three");
6083     hr = IXMLDOMNode_put_nodeValue(node, data);
6084     ok(hr == E_FAIL, "ret %08x\n", hr );
6085     IXMLDOMNode_Release(node);
6086
6087     /* NODE_DOCUMENT_FRAGMENT */
6088     V_VT(&type) = VT_I1;
6089     V_I1(&type) = NODE_DOCUMENT_FRAGMENT;
6090     hr = IXMLDOMDocument_createNode(doc, type, _bstr_("test"), NULL, &node);
6091     ok(hr == S_OK, "ret %08x\n", hr );
6092     V_VT(&data) = VT_BSTR;
6093     V_BSTR(&data) = _bstr_("one two three");
6094     hr = IXMLDOMNode_put_nodeValue(node, data);
6095     ok(hr == E_FAIL, "ret %08x\n", hr );
6096     IXMLDOMNode_Release(node);
6097
6098     /* NODE_ELEMENT */
6099     V_VT(&type) = VT_I1;
6100     V_I1(&type) = NODE_ELEMENT;
6101     hr = IXMLDOMDocument_createNode(doc, type, _bstr_("test"), NULL, &node);
6102     ok(hr == S_OK, "ret %08x\n", hr );
6103     V_VT(&data) = VT_BSTR;
6104     V_BSTR(&data) = _bstr_("one two three");
6105     hr = IXMLDOMNode_put_nodeValue(node, data);
6106     ok(hr == E_FAIL, "ret %08x\n", hr );
6107     IXMLDOMNode_Release(node);
6108
6109     /* NODE_ENTITY_REFERENCE */
6110     hr = IXMLDOMDocument_createEntityReference(doc, _bstr_("ref"), &entityref);
6111     ok(hr == S_OK, "ret %08x\n", hr );
6112
6113     V_VT(&data) = VT_BSTR;
6114     V_BSTR(&data) = _bstr_("one two three");
6115     hr = IXMLDOMEntityReference_put_nodeValue(entityref, data);
6116     ok(hr == E_FAIL, "ret %08x\n", hr );
6117
6118     hr = IXMLDOMEntityReference_QueryInterface(entityref, &IID_IXMLDOMNode, (void**)&node);
6119     ok(hr == S_OK, "ret %08x\n", hr );
6120     V_VT(&data) = VT_BSTR;
6121     V_BSTR(&data) = _bstr_("one two three");
6122     hr = IXMLDOMNode_put_nodeValue(node, data);
6123     ok(hr == E_FAIL, "ret %08x\n", hr );
6124     IXMLDOMNode_Release(node);
6125     IXMLDOMEntityReference_Release(entityref);
6126
6127     /* supported types */
6128     hr = IXMLDOMDocument_createTextNode(doc, _bstr_(""), &text);
6129     ok(hr == S_OK, "ret %08x\n", hr );
6130     V_VT(&data) = VT_BSTR;
6131     V_BSTR(&data) = _bstr_("Jeeves & Wooster");
6132     hr = IXMLDOMText_put_nodeValue(text, data);
6133     ok(hr == S_OK, "ret %08x\n", hr );
6134     IXMLDOMText_Release(text);
6135
6136     hr = IXMLDOMDocument_createAttribute(doc, _bstr_("attr"), &attr);
6137     ok(hr == S_OK, "ret %08x\n", hr );
6138     V_VT(&data) = VT_BSTR;
6139     V_BSTR(&data) = _bstr_("Jeeves & Wooster");
6140     hr = IXMLDOMAttribute_put_nodeValue(attr, data);
6141     ok(hr == S_OK, "ret %08x\n", hr );
6142     hr = IXMLDOMAttribute_get_nodeValue(attr, &data);
6143     ok(hr == S_OK, "ret %08x\n", hr );
6144     ok(memcmp(V_BSTR(&data), jeevesW, sizeof(jeevesW)) == 0, "got %s\n",
6145         wine_dbgstr_w(V_BSTR(&data)));
6146     VariantClear(&data);
6147     IXMLDOMAttribute_Release(attr);
6148
6149     free_bstrs();
6150
6151     IXMLDOMDocument_Release(doc);
6152 }
6153
6154 static void test_IObjectSafety_set(IObjectSafety *safety, HRESULT result, HRESULT result2, DWORD set, DWORD mask, DWORD expected, DWORD expected2)
6155 {
6156     DWORD enabled, supported;
6157     HRESULT hr;
6158
6159     hr = IObjectSafety_SetInterfaceSafetyOptions(safety, NULL, set, mask);
6160     if (result == result2)
6161         ok(hr == result, "SetInterfaceSafetyOptions: expected %08x, returned %08x\n", result, hr );
6162     else
6163         ok(broken(hr == result) || hr == result2,
6164            "SetInterfaceSafetyOptions: expected %08x, got %08x\n", result2, hr );
6165
6166     supported = enabled = 0xCAFECAFE;
6167     hr = IObjectSafety_GetInterfaceSafetyOptions(safety, NULL, &supported, &enabled);
6168     ok(hr == S_OK, "ret %08x\n", hr );
6169     if (expected == expected2)
6170         ok(enabled == expected, "Expected %08x, got %08x\n", expected, enabled);
6171     else
6172         ok(broken(enabled == expected) || enabled == expected2,
6173            "Expected %08x, got %08x\n", expected2, enabled);
6174
6175     /* reset the safety options */
6176
6177     hr = IObjectSafety_SetInterfaceSafetyOptions(safety, NULL,
6178             INTERFACESAFE_FOR_UNTRUSTED_CALLER|INTERFACESAFE_FOR_UNTRUSTED_DATA|INTERFACE_USES_SECURITY_MANAGER,
6179             0);
6180     ok(hr == S_OK, "ret %08x\n", hr );
6181
6182     hr = IObjectSafety_GetInterfaceSafetyOptions(safety, NULL, &supported, &enabled);
6183     ok(hr == S_OK, "ret %08x\n", hr );
6184     ok(enabled == 0, "Expected 0, got %08x\n", enabled);
6185 }
6186
6187 static void test_document_IObjectSafety(void)
6188 {
6189     IXMLDOMDocument *doc;
6190     IObjectSafety *safety;
6191     DWORD enabled = 0, supported = 0;
6192     HRESULT hr;
6193
6194     doc = create_document(&IID_IXMLDOMDocument);
6195     if (!doc) return;
6196
6197     hr = IXMLDOMDocument_QueryInterface(doc, &IID_IObjectSafety, (void**)&safety);
6198     ok(hr == S_OK, "ret %08x\n", hr );
6199
6200     /* get */
6201     hr = IObjectSafety_GetInterfaceSafetyOptions(safety, NULL, NULL, &enabled);
6202     ok(hr == E_POINTER, "ret %08x\n", hr );
6203     hr = IObjectSafety_GetInterfaceSafetyOptions(safety, NULL, &supported, NULL);
6204     ok(hr == E_POINTER, "ret %08x\n", hr );
6205
6206     hr = IObjectSafety_GetInterfaceSafetyOptions(safety, NULL, &supported, &enabled);
6207     ok(hr == S_OK, "ret %08x\n", hr );
6208     ok(broken(supported == (INTERFACESAFE_FOR_UNTRUSTED_CALLER | INTERFACESAFE_FOR_UNTRUSTED_DATA)) ||
6209        supported == (INTERFACESAFE_FOR_UNTRUSTED_CALLER | INTERFACESAFE_FOR_UNTRUSTED_DATA | INTERFACE_USES_SECURITY_MANAGER) /* msxml3 SP8+ */,
6210         "Expected (INTERFACESAFE_FOR_UNTRUSTED_CALLER | INTERFACESAFE_FOR_UNTRUSTED_DATA | INTERFACE_USES_SECURITY_MANAGER), "
6211              "got %08x\n", supported);
6212     ok(enabled == 0, "Expected 0, got %08x\n", enabled);
6213
6214     /* set -- individual flags */
6215
6216     test_IObjectSafety_set(safety, S_OK, S_OK,
6217         INTERFACESAFE_FOR_UNTRUSTED_CALLER, INTERFACESAFE_FOR_UNTRUSTED_CALLER,
6218         INTERFACESAFE_FOR_UNTRUSTED_CALLER, INTERFACESAFE_FOR_UNTRUSTED_CALLER);
6219
6220     test_IObjectSafety_set(safety, S_OK, S_OK,
6221         INTERFACESAFE_FOR_UNTRUSTED_DATA, INTERFACESAFE_FOR_UNTRUSTED_DATA,
6222         INTERFACESAFE_FOR_UNTRUSTED_DATA, INTERFACESAFE_FOR_UNTRUSTED_DATA);
6223
6224     test_IObjectSafety_set(safety, S_OK, S_OK,
6225         INTERFACE_USES_SECURITY_MANAGER, INTERFACE_USES_SECURITY_MANAGER,
6226         0, INTERFACE_USES_SECURITY_MANAGER /* msxml3 SP8+ */);
6227
6228     /* set INTERFACE_USES_DISPEX  */
6229
6230     test_IObjectSafety_set(safety, S_OK, E_FAIL /* msxml3 SP8+ */,
6231         INTERFACE_USES_DISPEX, INTERFACE_USES_DISPEX,
6232         0, 0);
6233
6234     test_IObjectSafety_set(safety, S_OK, E_FAIL /* msxml3 SP8+ */,
6235         INTERFACE_USES_DISPEX, 0,
6236         0, 0);
6237
6238     test_IObjectSafety_set(safety, S_OK, S_OK /* msxml3 SP8+ */,
6239         0, INTERFACE_USES_DISPEX,
6240         0, 0);
6241
6242     /* set option masking */
6243
6244     test_IObjectSafety_set(safety, S_OK, S_OK,
6245         INTERFACESAFE_FOR_UNTRUSTED_CALLER|INTERFACESAFE_FOR_UNTRUSTED_DATA,
6246         INTERFACESAFE_FOR_UNTRUSTED_CALLER,
6247         INTERFACESAFE_FOR_UNTRUSTED_CALLER,
6248         INTERFACESAFE_FOR_UNTRUSTED_CALLER);
6249
6250     test_IObjectSafety_set(safety, S_OK, S_OK,
6251         INTERFACESAFE_FOR_UNTRUSTED_CALLER|INTERFACESAFE_FOR_UNTRUSTED_DATA,
6252         INTERFACESAFE_FOR_UNTRUSTED_DATA,
6253         INTERFACESAFE_FOR_UNTRUSTED_DATA,
6254         INTERFACESAFE_FOR_UNTRUSTED_DATA);
6255
6256     test_IObjectSafety_set(safety, S_OK, S_OK,
6257         INTERFACESAFE_FOR_UNTRUSTED_CALLER|INTERFACESAFE_FOR_UNTRUSTED_DATA,
6258         INTERFACE_USES_SECURITY_MANAGER,
6259         0,
6260         0);
6261
6262     /* set -- inheriting previous settings */
6263
6264     hr = IObjectSafety_SetInterfaceSafetyOptions(safety, NULL,
6265                                                          INTERFACESAFE_FOR_UNTRUSTED_CALLER,
6266                                                          INTERFACESAFE_FOR_UNTRUSTED_CALLER);
6267     ok(hr == S_OK, "ret %08x\n", hr );
6268     hr = IObjectSafety_GetInterfaceSafetyOptions(safety, NULL, &supported, &enabled);
6269     ok(hr == S_OK, "ret %08x\n", hr );
6270     todo_wine
6271     ok(broken(enabled == INTERFACESAFE_FOR_UNTRUSTED_CALLER) ||
6272        enabled == (INTERFACESAFE_FOR_UNTRUSTED_CALLER | INTERFACE_USES_SECURITY_MANAGER) /* msxml3 SP8+ */,
6273          "Expected (INTERFACESAFE_FOR_UNTRUSTED_CALLER | INTERFACE_USES_SECURITY_MANAGER), "
6274          "got %08x\n", enabled);
6275
6276     hr = IObjectSafety_SetInterfaceSafetyOptions(safety, NULL,
6277                                                          INTERFACESAFE_FOR_UNTRUSTED_DATA,
6278                                                          INTERFACESAFE_FOR_UNTRUSTED_DATA);
6279     ok(hr == S_OK, "ret %08x\n", hr );
6280     hr = IObjectSafety_GetInterfaceSafetyOptions(safety, NULL, &supported, &enabled);
6281     ok(hr == S_OK, "ret %08x\n", hr );
6282     todo_wine
6283     ok(broken(enabled == INTERFACESAFE_FOR_UNTRUSTED_DATA) ||
6284        enabled == (INTERFACESAFE_FOR_UNTRUSTED_CALLER | INTERFACESAFE_FOR_UNTRUSTED_DATA) /* msxml3 SP8+ */,
6285         "Expected (INTERFACESAFE_FOR_UNTRUSTED_CALLER | INTERFACESAFE_FOR_UNTRUSTED_DATA), "
6286         "got %08x\n", enabled);
6287
6288     IObjectSafety_Release(safety);
6289
6290     IXMLDOMDocument_Release(doc);
6291 }
6292
6293 typedef struct _property_test_t {
6294     const GUID *guid;
6295     const char *clsid;
6296     const char *property;
6297     const char *value;
6298 } property_test_t;
6299
6300 static const property_test_t properties_test_data[] = {
6301     { &CLSID_DOMDocument,  "CLSID_DOMDocument" , "SelectionLanguage", "XSLPattern" },
6302     { &CLSID_DOMDocument2,  "CLSID_DOMDocument2" , "SelectionLanguage", "XSLPattern" },
6303     { &CLSID_DOMDocument30, "CLSID_DOMDocument30", "SelectionLanguage", "XSLPattern" },
6304     { &CLSID_DOMDocument40, "CLSID_DOMDocument40", "SelectionLanguage", "XPath" },
6305     { &CLSID_DOMDocument60, "CLSID_DOMDocument60", "SelectionLanguage", "XPath" },
6306     { 0 }
6307 };
6308
6309 static void test_default_properties(void)
6310 {
6311     const property_test_t *entry = properties_test_data;
6312     IXMLDOMDocument2 *doc;
6313     VARIANT var;
6314     HRESULT hr;
6315
6316     while (entry->guid)
6317     {
6318         hr = CoCreateInstance(entry->guid, NULL, CLSCTX_INPROC_SERVER, &IID_IXMLDOMDocument2, (void**)&doc);
6319         if (hr != S_OK)
6320         {
6321             win_skip("can't create %s instance\n", entry->clsid);
6322             entry++;
6323             continue;
6324         }
6325
6326         hr = IXMLDOMDocument2_getProperty(doc, _bstr_(entry->property), &var);
6327         ok(hr == S_OK, "got 0x%08x\n", hr);
6328         ok(lstrcmpW(V_BSTR(&var), _bstr_(entry->value)) == 0, "expected %s, for %s\n",
6329            entry->value, entry->clsid);
6330         VariantClear(&var);
6331
6332         IXMLDOMDocument2_Release(doc);
6333
6334         entry++;
6335     }
6336 }
6337
6338 static void test_XSLPattern(void)
6339 {
6340     IXMLDOMDocument2 *doc;
6341     IXMLDOMNodeList *list;
6342     VARIANT_BOOL b;
6343     LONG len;
6344
6345     doc = create_document(&IID_IXMLDOMDocument2);
6346     if (!doc) return;
6347
6348     ole_check(IXMLDOMDocument2_loadXML(doc, _bstr_(szExampleXML), &b));
6349     ok(b == VARIANT_TRUE, "failed to load XML string\n");
6350
6351     /* switch to XSLPattern */
6352     ole_check(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionLanguage"), _variantbstr_("XSLPattern")));
6353
6354     /* XPath doesn't select elements with non-null default namespace with unqualified selectors, XSLPattern does */
6355     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//elem/c"), &list));
6356     len = 0;
6357     ole_check(IXMLDOMNodeList_get_length(list, &len));
6358     /* should select <elem><c> and <elem xmlns='...'><c> but not <elem><foo:c> */
6359     ok(len == 3, "expected 3 entries in list, got %d\n", len);
6360     IXMLDOMNodeList_Release(list);
6361
6362     /* for XSLPattern start index is 0, for XPath it's 1 */
6363     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[0]"), &list));
6364     len = 0;
6365     ole_check(IXMLDOMNodeList_get_length(list, &len));
6366     ok(len != 0, "expected filled list\n");
6367     if (len)
6368         expect_list_and_release(list, "E1.E2.D1");
6369
6370     /* index() */
6371     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index()=1]"), &list));
6372     len = 0;
6373     ole_check(IXMLDOMNodeList_get_length(list, &len));
6374     ok(len != 0, "expected filled list\n");
6375     if (len)
6376         expect_list_and_release(list, "E2.E2.D1");
6377
6378     /* $eq$ */
6379     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index() $eq$ 1]"), &list));
6380     len = 0;
6381     ole_check(IXMLDOMNodeList_get_length(list, &len));
6382     ok(len != 0, "expected filled list\n");
6383     if (len)
6384         expect_list_and_release(list, "E2.E2.D1");
6385
6386     /* end() */
6387     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[end()]"), &list));
6388     len = 0;
6389     ole_check(IXMLDOMNodeList_get_length(list, &len));
6390     ok(len != 0, "expected filled list\n");
6391     if (len)
6392         expect_list_and_release(list, "E4.E2.D1");
6393
6394     /* $not$ */
6395     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[$not$ end()]"), &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 E2.E2.D1 E3.E2.D1");
6401
6402     /* !=/$ne$ */
6403     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index() != 0]"), &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 E3.E2.D1 E4.E2.D1");
6409     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index() $ne$ 0]"), &list));
6410     len = 0;
6411     ole_check(IXMLDOMNodeList_get_length(list, &len));
6412     ok(len != 0, "expected filled list\n");
6413     if (len)
6414         expect_list_and_release(list, "E2.E2.D1 E3.E2.D1 E4.E2.D1");
6415
6416     /* </$lt$ */
6417     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index() < 2]"), &list));
6418     len = 0;
6419     ole_check(IXMLDOMNodeList_get_length(list, &len));
6420     ok(len != 0, "expected filled list\n");
6421     if (len)
6422         expect_list_and_release(list, "E1.E2.D1 E2.E2.D1");
6423     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index() $lt$ 2]"), &list));
6424     len = 0;
6425     ole_check(IXMLDOMNodeList_get_length(list, &len));
6426     ok(len != 0, "expected filled list\n");
6427     if (len)
6428         expect_list_and_release(list, "E1.E2.D1 E2.E2.D1");
6429
6430     /* <=/$le$ */
6431     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index() <= 1]"), &list));
6432     len = 0;
6433     ole_check(IXMLDOMNodeList_get_length(list, &len));
6434     ok(len != 0, "expected filled list\n");
6435     if (len)
6436         expect_list_and_release(list, "E1.E2.D1 E2.E2.D1");
6437     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index() $le$ 1]"), &list));
6438     len = 0;
6439     ole_check(IXMLDOMNodeList_get_length(list, &len));
6440     ok(len != 0, "expected filled list\n");
6441     if (len)
6442         expect_list_and_release(list, "E1.E2.D1 E2.E2.D1");
6443
6444     /* >/$gt$ */
6445     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index() > 1]"), &list));
6446     len = 0;
6447     ole_check(IXMLDOMNodeList_get_length(list, &len));
6448     ok(len != 0, "expected filled list\n");
6449     if (len)
6450         expect_list_and_release(list, "E3.E2.D1 E4.E2.D1");
6451     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index() $gt$ 1]"), &list));
6452     len = 0;
6453     ole_check(IXMLDOMNodeList_get_length(list, &len));
6454     ok(len != 0, "expected filled list\n");
6455     if (len)
6456         expect_list_and_release(list, "E3.E2.D1 E4.E2.D1");
6457
6458     /* >=/$ge$ */
6459     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index() >= 2]"), &list));
6460     len = 0;
6461     ole_check(IXMLDOMNodeList_get_length(list, &len));
6462     ok(len != 0, "expected filled list\n");
6463     if (len)
6464         expect_list_and_release(list, "E3.E2.D1 E4.E2.D1");
6465     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index() $ge$ 2]"), &list));
6466     len = 0;
6467     ole_check(IXMLDOMNodeList_get_length(list, &len));
6468     ok(len != 0, "expected filled list\n");
6469     if (len)
6470         expect_list_and_release(list, "E3.E2.D1 E4.E2.D1");
6471
6472     /* $ieq$ */
6473     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[a $ieq$ 'a2 field']"), &list));
6474     len = 0;
6475     ole_check(IXMLDOMNodeList_get_length(list, &len));
6476     ok(len != 0, "expected filled list\n");
6477     if (len)
6478         expect_list_and_release(list, "E2.E2.D1");
6479
6480     /* $ine$ */
6481     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[a $ine$ 'a2 field']"), &list));
6482     len = 0;
6483     ole_check(IXMLDOMNodeList_get_length(list, &len));
6484     ok(len != 0, "expected filled list\n");
6485     if (len)
6486         expect_list_and_release(list, "E1.E2.D1 E3.E2.D1 E4.E2.D1");
6487
6488     /* $ilt$ */
6489     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[a $ilt$ 'a3 field']"), &list));
6490     len = 0;
6491     ole_check(IXMLDOMNodeList_get_length(list, &len));
6492     ok(len != 0, "expected filled list\n");
6493     if (len)
6494         expect_list_and_release(list, "E1.E2.D1 E2.E2.D1");
6495
6496     /* $ile$ */
6497     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[a $ile$ 'a2 field']"), &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, "E1.E2.D1 E2.E2.D1");
6503
6504     /* $igt$ */
6505     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[a $igt$ '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, "E3.E2.D1 E4.E2.D1");
6511
6512     /* $ige$ */
6513     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[a $ige$ 'a3 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, "E3.E2.D1 E4.E2.D1");
6519
6520     /* $any$ */
6521     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[$any$ *='B2 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, "E2.E2.D1");
6527
6528     /* $all$ */
6529     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[$all$ *!='B2 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 E3.E2.D1 E4.E2.D1");
6535
6536     /* or/$or$/|| */
6537     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index()=0 or end()]"), &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, "E1.E2.D1 E4.E2.D1");
6543     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index()=0 $or$ end()]"), &list));
6544     len = 0;
6545     ole_check(IXMLDOMNodeList_get_length(list, &len));
6546     ok(len != 0, "expected filled list\n");
6547     if (len)
6548         expect_list_and_release(list, "E1.E2.D1 E4.E2.D1");
6549     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index()=0 || end()]"), &list));
6550     len = 0;
6551     ole_check(IXMLDOMNodeList_get_length(list, &len));
6552     ok(len != 0, "expected filled list\n");
6553     if (len)
6554         expect_list_and_release(list, "E1.E2.D1 E4.E2.D1");
6555
6556     /* and/$and$/&& */
6557     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index()>0 and $not$ end()]"), &list));
6558     len = 0;
6559     ole_check(IXMLDOMNodeList_get_length(list, &len));
6560     ok(len != 0, "expected filled list\n");
6561     if (len)
6562         expect_list_and_release(list, "E2.E2.D1 E3.E2.D1");
6563     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index()>0 $and$ $not$ end()]"), &list));
6564     len = 0;
6565     ole_check(IXMLDOMNodeList_get_length(list, &len));
6566     ok(len != 0, "expected filled list\n");
6567     if (len)
6568         expect_list_and_release(list, "E2.E2.D1 E3.E2.D1");
6569     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index()>0 && $not$ 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, "E2.E2.D1 E3.E2.D1");
6575
6576     /* namespace handling */
6577     /* no registered namespaces */
6578     ole_check(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionNamespaces"), _variantbstr_("")));
6579     list = NULL;
6580
6581     /* prefixes don't need to be registered, you may use them as they are in the doc */
6582     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//bar:x"), &list));
6583     len = 0;
6584     ole_check(IXMLDOMNodeList_get_length(list, &len));
6585     ok(len != 0, "expected filled list\n");
6586     if (len)
6587         expect_list_and_release(list, "E5.E1.E4.E1.E2.D1 E6.E2.E4.E1.E2.D1");
6588
6589     /* prefixes must be explicitly specified in the name */
6590     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//foo:elem"), &list));
6591     len = 0;
6592     ole_check(IXMLDOMNodeList_get_length(list, &len));
6593     ok(len == 0, "expected empty list\n");
6594     if (len)
6595         IXMLDOMNodeList_Release(list);
6596
6597     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//foo:c"), &list));
6598     len = 0;
6599     ole_check(IXMLDOMNodeList_get_length(list, &len));
6600     ok(len != 0, "expected filled list\n");
6601     if (len)
6602         expect_list_and_release(list, "E3.E4.E2.D1");
6603
6604     /* explicitly register prefix foo */
6605     ole_check(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionNamespaces"), _variantbstr_("xmlns:foo='urn:uuid:86B2F87F-ACB6-45cd-8B77-9BDB92A01A29'")));
6606
6607     /* now we get the same behavior as XPath */
6608     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//foo:c"), &list));
6609     len = 0;
6610     ole_check(IXMLDOMNodeList_get_length(list, &len));
6611     ok(len != 0, "expected filled list\n");
6612     if (len)
6613         expect_list_and_release(list, "E3.E3.E2.D1 E3.E4.E2.D1");
6614
6615     /* set prefix foo to some nonexistent namespace */
6616     ole_check(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionNamespaces"), _variantbstr_("xmlns:foo='urn:nonexistent-foo'")));
6617
6618     /* the registered prefix takes precedence */
6619     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//foo:c"), &list));
6620     len = 0;
6621     ole_check(IXMLDOMNodeList_get_length(list, &len));
6622     ok(len == 0, "expected empty list\n");
6623     if (len)
6624         IXMLDOMNodeList_Release(list);
6625
6626     IXMLDOMDocument2_Release(doc);
6627
6628     doc = create_document(&IID_IXMLDOMDocument2);
6629     if (!doc) return;
6630
6631     ole_check(IXMLDOMDocument2_loadXML(doc, _bstr_(szNodeTypesXML), &b));
6632     ok(b == VARIANT_TRUE, "failed to load XML string\n");
6633     list = NULL;
6634
6635     /* attribute() */
6636     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("attribute()"), &list));
6637     len = 0;
6638     ole_check(IXMLDOMNodeList_get_length(list, &len));
6639     ok(len == 0, "expected empty list\n");
6640     if (len)
6641         IXMLDOMNodeList_Release(list);
6642
6643     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("attribute('depth')"), &list));
6644     len = 0;
6645     ole_check(IXMLDOMNodeList_get_length(list, &len));
6646     ok(len == 0, "expected empty list\n");
6647     if (len)
6648         IXMLDOMNodeList_Release(list);
6649
6650     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root/attribute('depth')"), &list));
6651     len = 0;
6652     ole_check(IXMLDOMNodeList_get_length(list, &len));
6653     ok(len != 0, "expected filled list\n");
6654     if (len)
6655         expect_list_and_release(list, "A'depth'.E3.D1");
6656
6657     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//x/attribute()"), &list));
6658     len = 0;
6659     ole_check(IXMLDOMNodeList_get_length(list, &len));
6660     ok(len != 0, "expected filled list\n");
6661     if (len)
6662         expect_list_and_release(list, "A'id'.E3.E3.D1 A'depth'.E3.E3.D1");
6663
6664     list = NULL;
6665     ole_expect(IXMLDOMDocument2_selectNodes(doc, _bstr_("//x//attribute(id)"), &list), E_FAIL);
6666     if (list)
6667         IXMLDOMNodeList_Release(list);
6668     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//x//attribute('id')"), &list));
6669     len = 0;
6670     ole_check(IXMLDOMNodeList_get_length(list, &len));
6671     ok(len != 0, "expected filled list\n");
6672     if (len)
6673         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");
6674
6675     /* comment() */
6676     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("comment()"), &list));
6677     len = 0;
6678     ole_check(IXMLDOMNodeList_get_length(list, &len));
6679     ok(len != 0, "expected filled list\n");
6680     if (len)
6681         expect_list_and_release(list, "C2.D1");
6682
6683     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//comment()"), &list));
6684     len = 0;
6685     ole_check(IXMLDOMNodeList_get_length(list, &len));
6686     ok(len != 0, "expected filled list\n");
6687     if (len)
6688         expect_list_and_release(list, "C2.D1 C1.E3.D1 C2.E3.E3.D1 C2.E4.E3.D1");
6689
6690     /* element() */
6691     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("element()"), &list));
6692     len = 0;
6693     ole_check(IXMLDOMNodeList_get_length(list, &len));
6694     ok(len != 0, "expected filled list\n");
6695     if (len)
6696         expect_list_and_release(list, "E3.D1");
6697
6698     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root/y/element()"), &list));
6699     len = 0;
6700     ole_check(IXMLDOMNodeList_get_length(list, &len));
6701     ok(len != 0, "expected filled list\n");
6702     if (len)
6703         expect_list_and_release(list, "E4.E4.E3.D1 E5.E4.E3.D1 E6.E4.E3.D1");
6704
6705     list = NULL;
6706     ole_expect(IXMLDOMDocument2_selectNodes(doc, _bstr_("//element(a)"), &list), E_FAIL);
6707     if (list)
6708         IXMLDOMNodeList_Release(list);
6709     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//element('a')"), &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, "E4.E3.E3.D1 E4.E4.E3.D1");
6715
6716     /* node() */
6717     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("node()"), &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, "P1.D1 C2.D1 E3.D1");
6723
6724     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//x/node()"), &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, "P1.E3.E3.D1 C2.E3.E3.D1 T3.E3.E3.D1 E4.E3.E3.D1 E5.E3.E3.D1 E6.E3.E3.D1");
6730
6731     /* nodeType() */
6732     /* XML_ELEMENT_NODE */
6733     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//x/node()[nodeType()=1]"), &list));
6734     len = 0;
6735     ole_check(IXMLDOMNodeList_get_length(list, &len));
6736     ok(len != 0, "expected filled list\n");
6737     if (len)
6738         expect_list_and_release(list, "E4.E3.E3.D1 E5.E3.E3.D1 E6.E3.E3.D1");
6739     /* XML_TEXT_NODE */
6740     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//x/node()[nodeType()=3]"), &list));
6741     len = 0;
6742     ole_check(IXMLDOMNodeList_get_length(list, &len));
6743     ok(len != 0, "expected filled list\n");
6744     if (len)
6745         expect_list_and_release(list, "T3.E3.E3.D1");
6746     /* XML_PI_NODE */
6747     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//x/node()[nodeType()=7]"), &list));
6748     len = 0;
6749     ole_check(IXMLDOMNodeList_get_length(list, &len));
6750     ok(len != 0, "expected filled list\n");
6751     if (len)
6752         expect_list_and_release(list, "P1.E3.E3.D1");
6753     /* XML_COMMENT_NODE */
6754     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//x/node()[nodeType()=8]"), &list));
6755     len = 0;
6756     ole_check(IXMLDOMNodeList_get_length(list, &len));
6757     ok(len != 0, "expected filled list\n");
6758     if (len)
6759         expect_list_and_release(list, "C2.E3.E3.D1");
6760
6761     /* pi() */
6762     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("pi()"), &list));
6763     if (list)
6764     {
6765         len = 0;
6766         ole_check(IXMLDOMNodeList_get_length(list, &len));
6767         ok(len != 0, "expected filled list\n");
6768         if (len)
6769             expect_list_and_release(list, "P1.D1");
6770     }
6771
6772     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//y/pi()"), &list));
6773     len = 0;
6774     ole_check(IXMLDOMNodeList_get_length(list, &len));
6775     ok(len != 0, "expected filled list\n");
6776     if (len)
6777         expect_list_and_release(list, "P1.E4.E3.D1");
6778
6779     /* textnode() */
6780     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root/textnode()"), &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, "T2.E3.D1");
6786
6787     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root/element()/textnode()"), &list));
6788     len = 0;
6789     ole_check(IXMLDOMNodeList_get_length(list, &len));
6790     ok(len != 0, "expected filled list\n");
6791     if (len)
6792         expect_list_and_release(list, "T3.E3.E3.D1 T3.E4.E3.D1");
6793
6794     IXMLDOMDocument2_Release(doc);
6795     free_bstrs();
6796 }
6797
6798 static void test_splitText(void)
6799 {
6800     IXMLDOMCDATASection *cdata;
6801     IXMLDOMElement *root;
6802     IXMLDOMDocument *doc;
6803     IXMLDOMText *text, *text2;
6804     IXMLDOMNode *node;
6805     VARIANT var;
6806     VARIANT_BOOL success;
6807     LONG length;
6808     HRESULT hr;
6809
6810     doc = create_document(&IID_IXMLDOMDocument);
6811     if (!doc) return;
6812
6813     hr = IXMLDOMDocument_loadXML(doc, _bstr_("<root></root>"), &success);
6814     ok(hr == S_OK, "got 0x%08x\n", hr);
6815
6816     hr = IXMLDOMDocument_get_documentElement(doc, &root);
6817     ok(hr == S_OK, "got 0x%08x\n", hr);
6818
6819     hr = IXMLDOMDocument_createCDATASection(doc, _bstr_("beautiful plumage"), &cdata);
6820     ok(hr == S_OK, "got 0x%08x\n", hr);
6821
6822     V_VT(&var) = VT_EMPTY;
6823     hr = IXMLDOMElement_appendChild(root, (IXMLDOMNode*)cdata, NULL);
6824     ok(hr == S_OK, "got 0x%08x\n", hr);
6825
6826     length = 0;
6827     hr = IXMLDOMCDATASection_get_length(cdata, &length);
6828     ok(hr == S_OK, "got 0x%08x\n", hr);
6829     ok(length > 0, "got %d\n", length);
6830
6831     hr = IXMLDOMCDATASection_splitText(cdata, 0, NULL);
6832     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
6833
6834     text = (void*)0xdeadbeef;
6835     /* negative offset */
6836     hr = IXMLDOMCDATASection_splitText(cdata, -1, &text);
6837     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
6838     ok(text == (void*)0xdeadbeef, "got %p\n", text);
6839
6840     text = (void*)0xdeadbeef;
6841     /* offset outside data */
6842     hr = IXMLDOMCDATASection_splitText(cdata, length + 1, &text);
6843     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
6844     ok(text == 0, "got %p\n", text);
6845
6846     text = (void*)0xdeadbeef;
6847     /* offset outside data */
6848     hr = IXMLDOMCDATASection_splitText(cdata, length, &text);
6849     ok(hr == S_FALSE, "got 0x%08x\n", hr);
6850     ok(text == 0, "got %p\n", text);
6851
6852     /* no empty node created */
6853     node = (void*)0xdeadbeef;
6854     hr = IXMLDOMCDATASection_get_nextSibling(cdata, &node);
6855     ok(hr == S_FALSE, "got 0x%08x\n", hr);
6856     ok(node == 0, "got %p\n", text);
6857
6858     hr = IXMLDOMCDATASection_splitText(cdata, 10, &text);
6859     ok(hr == S_OK, "got 0x%08x\n", hr);
6860
6861     length = 0;
6862     hr = IXMLDOMText_get_length(text, &length);
6863     ok(hr == S_OK, "got 0x%08x\n", hr);
6864     ok(length == 7, "got %d\n", length);
6865
6866     hr = IXMLDOMCDATASection_get_nextSibling(cdata, &node);
6867     ok(hr == S_OK, "got 0x%08x\n", hr);
6868     IXMLDOMNode_Release(node);
6869
6870     /* split new text node */
6871     hr = IXMLDOMText_get_length(text, &length);
6872     ok(hr == S_OK, "got 0x%08x\n", hr);
6873
6874     node = (void*)0xdeadbeef;
6875     hr = IXMLDOMText_get_nextSibling(text, &node);
6876     ok(hr == S_FALSE, "got 0x%08x\n", hr);
6877     ok(node == 0, "got %p\n", text);
6878
6879     hr = IXMLDOMText_splitText(text, 0, NULL);
6880     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
6881
6882     text2 = (void*)0xdeadbeef;
6883     /* negative offset */
6884     hr = IXMLDOMText_splitText(text, -1, &text2);
6885     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
6886     ok(text2 == (void*)0xdeadbeef, "got %p\n", text2);
6887
6888     text2 = (void*)0xdeadbeef;
6889     /* offset outside data */
6890     hr = IXMLDOMText_splitText(text, length + 1, &text2);
6891     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
6892     ok(text2 == 0, "got %p\n", text2);
6893
6894     text2 = (void*)0xdeadbeef;
6895     /* offset outside data */
6896     hr = IXMLDOMText_splitText(text, length, &text2);
6897     ok(hr == S_FALSE, "got 0x%08x\n", hr);
6898     ok(text2 == 0, "got %p\n", text);
6899
6900     text2 = 0;
6901     hr = IXMLDOMText_splitText(text, 4, &text2);
6902     todo_wine ok(hr == S_OK, "got 0x%08x\n", hr);
6903     if (text2) IXMLDOMText_Release(text2);
6904
6905     node = 0;
6906     hr = IXMLDOMText_get_nextSibling(text, &node);
6907     todo_wine ok(hr == S_OK, "got 0x%08x\n", hr);
6908     if (node) IXMLDOMNode_Release(node);
6909
6910     IXMLDOMText_Release(text);
6911     IXMLDOMElement_Release(root);
6912     IXMLDOMCDATASection_Release(cdata);
6913     free_bstrs();
6914 }
6915
6916 static void test_getQualifiedItem(void)
6917 {
6918     IXMLDOMDocument *doc;
6919     IXMLDOMElement *element;
6920     IXMLDOMNode *pr_node, *node;
6921     IXMLDOMNodeList *root_list;
6922     IXMLDOMNamedNodeMap *map;
6923     VARIANT_BOOL b;
6924     BSTR str;
6925     LONG len;
6926     HRESULT hr;
6927
6928     doc = create_document(&IID_IXMLDOMDocument);
6929     if (!doc) return;
6930
6931     str = SysAllocString( szComplete4 );
6932     hr = IXMLDOMDocument_loadXML( doc, str, &b );
6933     ok( hr == S_OK, "loadXML failed\n");
6934     ok( b == VARIANT_TRUE, "failed to load XML string\n");
6935     SysFreeString( str );
6936
6937     hr = IXMLDOMDocument_get_documentElement(doc, &element);
6938     ok( hr == S_OK, "ret %08x\n", hr);
6939
6940     hr = IXMLDOMElement_get_childNodes(element, &root_list);
6941     ok( hr == S_OK, "ret %08x\n", hr);
6942
6943     hr = IXMLDOMNodeList_get_item(root_list, 1, &pr_node);
6944     ok( hr == S_OK, "ret %08x\n", hr);
6945     IXMLDOMNodeList_Release(root_list);
6946
6947     hr = IXMLDOMNode_get_attributes(pr_node, &map);
6948     ok( hr == S_OK, "ret %08x\n", hr);
6949     IXMLDOMNode_Release(pr_node);
6950
6951     hr = IXMLDOMNamedNodeMap_get_length(map, &len);
6952     ok( hr == S_OK, "ret %08x\n", hr);
6953     ok( len == 3, "length %d\n", len);
6954
6955     hr = IXMLDOMNamedNodeMap_getQualifiedItem(map, NULL, NULL, NULL);
6956     ok( hr == E_INVALIDARG, "ret %08x\n", hr);
6957
6958     node = (void*)0xdeadbeef;
6959     hr = IXMLDOMNamedNodeMap_getQualifiedItem(map, NULL, NULL, &node);
6960     ok( hr == E_INVALIDARG, "ret %08x\n", hr);
6961     ok( node == (void*)0xdeadbeef, "got %p\n", node);
6962
6963     hr = IXMLDOMNamedNodeMap_getQualifiedItem(map, _bstr_("id"), NULL, NULL);
6964     ok( hr == E_INVALIDARG, "ret %08x\n", hr);
6965
6966     hr = IXMLDOMNamedNodeMap_getQualifiedItem(map, _bstr_("id"), NULL, &node);
6967     ok( hr == S_OK, "ret %08x\n", hr);
6968     IXMLDOMNode_Release(node);
6969
6970     IXMLDOMNamedNodeMap_Release( map );
6971     IXMLDOMElement_Release( element );
6972     IXMLDOMDocument_Release( doc );
6973     free_bstrs();
6974 }
6975
6976 static void test_removeQualifiedItem(void)
6977 {
6978     IXMLDOMDocument *doc;
6979     IXMLDOMElement *element;
6980     IXMLDOMNode *pr_node, *node;
6981     IXMLDOMNodeList *root_list;
6982     IXMLDOMNamedNodeMap *map;
6983     VARIANT_BOOL b;
6984     BSTR str;
6985     LONG len;
6986     HRESULT hr;
6987
6988     doc = create_document(&IID_IXMLDOMDocument);
6989     if (!doc) return;
6990
6991     str = SysAllocString( szComplete4 );
6992     hr = IXMLDOMDocument_loadXML( doc, str, &b );
6993     ok( hr == S_OK, "loadXML failed\n");
6994     ok( b == VARIANT_TRUE, "failed to load XML string\n");
6995     SysFreeString( str );
6996
6997     hr = IXMLDOMDocument_get_documentElement(doc, &element);
6998     ok( hr == S_OK, "ret %08x\n", hr);
6999
7000     hr = IXMLDOMElement_get_childNodes(element, &root_list);
7001     ok( hr == S_OK, "ret %08x\n", hr);
7002
7003     hr = IXMLDOMNodeList_get_item(root_list, 1, &pr_node);
7004     ok( hr == S_OK, "ret %08x\n", hr);
7005     IXMLDOMNodeList_Release(root_list);
7006
7007     hr = IXMLDOMNode_get_attributes(pr_node, &map);
7008     ok( hr == S_OK, "ret %08x\n", hr);
7009     IXMLDOMNode_Release(pr_node);
7010
7011     hr = IXMLDOMNamedNodeMap_get_length(map, &len);
7012     ok( hr == S_OK, "ret %08x\n", hr);
7013     ok( len == 3, "length %d\n", len);
7014
7015     hr = IXMLDOMNamedNodeMap_removeQualifiedItem(map, NULL, NULL, NULL);
7016     ok( hr == E_INVALIDARG, "ret %08x\n", hr);
7017
7018     node = (void*)0xdeadbeef;
7019     hr = IXMLDOMNamedNodeMap_removeQualifiedItem(map, NULL, NULL, &node);
7020     ok( hr == E_INVALIDARG, "ret %08x\n", hr);
7021     ok( node == (void*)0xdeadbeef, "got %p\n", node);
7022
7023     /* out pointer is optional */
7024     hr = IXMLDOMNamedNodeMap_removeQualifiedItem(map, _bstr_("id"), NULL, NULL);
7025     ok( hr == S_OK, "ret %08x\n", hr);
7026
7027     /* already removed */
7028     hr = IXMLDOMNamedNodeMap_removeQualifiedItem(map, _bstr_("id"), NULL, NULL);
7029     ok( hr == S_FALSE, "ret %08x\n", hr);
7030
7031     hr = IXMLDOMNamedNodeMap_removeQualifiedItem(map, _bstr_("vr"), NULL, &node);
7032     ok( hr == S_OK, "ret %08x\n", hr);
7033     IXMLDOMNode_Release(node);
7034
7035     IXMLDOMNamedNodeMap_Release( map );
7036     IXMLDOMElement_Release( element );
7037     IXMLDOMDocument_Release( doc );
7038     free_bstrs();
7039 }
7040
7041 #define check_default_props(doc) _check_default_props(__LINE__, doc)
7042 static inline void _check_default_props(int line, IXMLDOMDocument2* doc)
7043 {
7044     VARIANT_BOOL b;
7045     VARIANT var;
7046     HRESULT hr;
7047
7048     VariantInit(&var);
7049     helper_ole_check(IXMLDOMDocument2_getProperty(doc, _bstr_("SelectionLanguage"), &var));
7050     ok_(__FILE__, line)(lstrcmpW(V_BSTR(&var), _bstr_("XSLPattern")) == 0, "expected XSLPattern\n");
7051     VariantClear(&var);
7052
7053     helper_ole_check(IXMLDOMDocument2_getProperty(doc, _bstr_("SelectionNamespaces"), &var));
7054     ok_(__FILE__, line)(lstrcmpW(V_BSTR(&var), _bstr_("")) == 0, "expected empty string\n");
7055     VariantClear(&var);
7056
7057     helper_ole_check(IXMLDOMDocument2_get_preserveWhiteSpace(doc, &b));
7058     ok_(__FILE__, line)(b == VARIANT_FALSE, "expected FALSE\n");
7059
7060     hr = IXMLDOMDocument2_get_schemas(doc, &var);
7061     ok_(__FILE__, line)(hr == S_FALSE, "got %08x\n", hr);
7062     VariantClear(&var);
7063 }
7064
7065 #define check_set_props(doc) _check_set_props(__LINE__, doc)
7066 static inline void _check_set_props(int line, IXMLDOMDocument2* doc)
7067 {
7068     VARIANT_BOOL b;
7069     VARIANT var;
7070
7071     VariantInit(&var);
7072     helper_ole_check(IXMLDOMDocument2_getProperty(doc, _bstr_("SelectionLanguage"), &var));
7073     ok_(__FILE__, line)(lstrcmpW(V_BSTR(&var), _bstr_("XPath")) == 0, "expected XPath\n");
7074     VariantClear(&var);
7075
7076     helper_ole_check(IXMLDOMDocument2_getProperty(doc, _bstr_("SelectionNamespaces"), &var));
7077     ok_(__FILE__, line)(lstrcmpW(V_BSTR(&var), _bstr_("xmlns:wi=\'www.winehq.org\'")) == 0, "got %s\n", wine_dbgstr_w(V_BSTR(&var)));
7078     VariantClear(&var);
7079
7080     helper_ole_check(IXMLDOMDocument2_get_preserveWhiteSpace(doc, &b));
7081     ok_(__FILE__, line)(b == VARIANT_TRUE, "expected TRUE\n");
7082
7083     helper_ole_check(IXMLDOMDocument2_get_schemas(doc, &var));
7084     ok_(__FILE__, line)(V_VT(&var) != VT_NULL, "expected pointer\n");
7085     VariantClear(&var);
7086 }
7087
7088 #define set_props(doc, cache) _set_props(__LINE__, doc, cache)
7089 static inline void _set_props(int line, IXMLDOMDocument2* doc, IXMLDOMSchemaCollection* cache)
7090 {
7091     VARIANT var;
7092
7093     VariantInit(&var);
7094     helper_ole_check(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionLanguage"), _variantbstr_("XPath")));
7095     helper_ole_check(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionNamespaces"), _variantbstr_("xmlns:wi=\'www.winehq.org\'")));
7096     helper_ole_check(IXMLDOMDocument2_put_preserveWhiteSpace(doc, VARIANT_TRUE));
7097     V_VT(&var) = VT_DISPATCH;
7098     V_DISPATCH(&var) = NULL;
7099     helper_ole_check(IXMLDOMSchemaCollection_QueryInterface(cache, &IID_IDispatch, (void**)&V_DISPATCH(&var)));
7100     ok_(__FILE__, line)(V_DISPATCH(&var) != NULL, "expected pointer\n");
7101     helper_ole_check(IXMLDOMDocument2_putref_schemas(doc, var));
7102     VariantClear(&var);
7103 }
7104
7105 #define unset_props(doc) _unset_props(__LINE__, doc)
7106 static inline void _unset_props(int line, IXMLDOMDocument2* doc)
7107 {
7108     VARIANT var;
7109
7110     VariantInit(&var);
7111     helper_ole_check(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionLanguage"), _variantbstr_("XSLPattern")));
7112     helper_ole_check(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionNamespaces"), _variantbstr_("")));
7113     helper_ole_check(IXMLDOMDocument2_put_preserveWhiteSpace(doc, VARIANT_FALSE));
7114     V_VT(&var) = VT_NULL;
7115     helper_ole_check(IXMLDOMDocument2_putref_schemas(doc, var));
7116     VariantClear(&var);
7117 }
7118
7119 static void test_get_ownerDocument(void)
7120 {
7121     IXMLDOMDocument *doc1, *doc2, *doc3;
7122     IXMLDOMDocument2 *doc, *doc_owner;
7123     IXMLDOMNode *node;
7124     IXMLDOMSchemaCollection *cache;
7125     VARIANT_BOOL b;
7126     VARIANT var;
7127     BSTR str;
7128
7129     doc = create_document(&IID_IXMLDOMDocument2);
7130     cache = create_cache(&IID_IXMLDOMSchemaCollection);
7131     if (!doc || !cache)
7132     {
7133         if (doc) IXMLDOMDocument2_Release(doc);
7134         if (cache) IXMLDOMSchemaCollection_Release(cache);
7135         return;
7136     }
7137
7138     VariantInit(&var);
7139
7140     str = SysAllocString(szComplete4);
7141     ole_check(IXMLDOMDocument_loadXML(doc, str, &b));
7142     ok(b == VARIANT_TRUE, "failed to load XML string\n");
7143     SysFreeString(str);
7144
7145     check_default_props(doc);
7146
7147     /* set properties and check that new instances use them */
7148     set_props(doc, cache);
7149     check_set_props(doc);
7150
7151     ole_check(IXMLDOMDocument2_get_firstChild(doc, &node));
7152     ole_check(IXMLDOMNode_get_ownerDocument(node, &doc1));
7153
7154     /* new interface keeps props */
7155     ole_check(IXMLDOMDocument_QueryInterface(doc1, &IID_IXMLDOMDocument2, (void**)&doc_owner));
7156     ok( doc_owner != doc, "got %p, doc %p\n", doc_owner, doc);
7157     check_set_props(doc_owner);
7158     IXMLDOMDocument2_Release(doc_owner);
7159
7160     ole_check(IXMLDOMNode_get_ownerDocument(node, &doc2));
7161     IXMLDOMNode_Release(node);
7162
7163     ok(doc1 != doc2, "got %p, expected %p. original %p\n", doc2, doc1, doc);
7164
7165     /* reload */
7166     str = SysAllocString(szComplete4);
7167     ole_check(IXMLDOMDocument2_loadXML(doc, str, &b));
7168     ok(b == VARIANT_TRUE, "failed to load XML string\n");
7169     SysFreeString(str);
7170
7171     /* properties retained even after reload */
7172     check_set_props(doc);
7173
7174     ole_check(IXMLDOMDocument2_get_firstChild(doc, &node));
7175     ole_check(IXMLDOMNode_get_ownerDocument(node, &doc3));
7176     IXMLDOMNode_Release(node);
7177
7178     ole_check(IXMLDOMDocument_QueryInterface(doc3, &IID_IXMLDOMDocument2, (void**)&doc_owner));
7179     ok(doc3 != doc1 && doc3 != doc2 && doc_owner != doc, "got %p, (%p, %p, %p)\n", doc3, doc, doc1, doc2);
7180     check_set_props(doc_owner);
7181
7182     /* changing properties for one instance changes them for all */
7183     unset_props(doc_owner);
7184     check_default_props(doc_owner);
7185     check_default_props(doc);
7186
7187
7188     IXMLDOMDocument_Release(doc1);
7189     IXMLDOMDocument_Release(doc2);
7190     IXMLDOMDocument_Release(doc3);
7191     IXMLDOMDocument2_Release(doc);
7192     IXMLDOMDocument2_Release(doc_owner);
7193     free_bstrs();
7194 }
7195
7196 static void test_setAttributeNode(void)
7197 {
7198     IXMLDOMDocument *doc, *doc2;
7199     IXMLDOMElement *elem;
7200     IXMLDOMAttribute *attr, *attr2, *ret_attr;
7201     VARIANT_BOOL b;
7202     HRESULT hr;
7203     BSTR str;
7204
7205     doc = create_document(&IID_IXMLDOMDocument);
7206     if (!doc) return;
7207
7208     str = SysAllocString( szComplete4 );
7209     hr = IXMLDOMDocument2_loadXML( doc, str, &b );
7210     ok( hr == S_OK, "loadXML failed\n");
7211     ok( b == VARIANT_TRUE, "failed to load XML string\n");
7212     SysFreeString( str );
7213
7214     hr = IXMLDOMDocument_get_documentElement(doc, &elem);
7215     ok( hr == S_OK, "got 0x%08x\n", hr);
7216
7217     ret_attr = (void*)0xdeadbeef;
7218     hr = IXMLDOMElement_setAttributeNode(elem, NULL, &ret_attr);
7219     ok( hr == E_INVALIDARG, "got 0x%08x\n", hr);
7220     ok( ret_attr == (void*)0xdeadbeef, "got %p\n", ret_attr);
7221
7222     hr = IXMLDOMDocument_createAttribute(doc, _bstr_("attr"), &attr);
7223     ok( hr == S_OK, "got 0x%08x\n", hr);
7224
7225     ret_attr = (void*)0xdeadbeef;
7226     hr = IXMLDOMElement_setAttributeNode(elem, attr, &ret_attr);
7227     ok( hr == S_OK, "got 0x%08x\n", hr);
7228     ok( ret_attr == NULL, "got %p\n", ret_attr);
7229
7230     attr2 = NULL;
7231     hr = IXMLDOMElement_getAttributeNode(elem, _bstr_("attr"), &attr2);
7232     ok( hr == S_OK, "got 0x%08x\n", hr);
7233     IXMLDOMAttribute_Release(attr2);
7234
7235     /* try to add it another time */
7236     ret_attr = (void*)0xdeadbeef;
7237     hr = IXMLDOMElement_setAttributeNode(elem, attr, &ret_attr);
7238     todo_wine ok( hr == E_FAIL, "got 0x%08x\n", hr);
7239     todo_wine ok( ret_attr == (void*)0xdeadbeef, "got %p\n", ret_attr);
7240
7241     IXMLDOMElement_Release(elem);
7242
7243     /* initialy used element is released, attribute still 'has' a container */
7244     hr = IXMLDOMDocument_get_documentElement(doc, &elem);
7245     ok( hr == S_OK, "got 0x%08x\n", hr);
7246     ret_attr = (void*)0xdeadbeef;
7247     hr = IXMLDOMElement_setAttributeNode(elem, attr, &ret_attr);
7248     todo_wine ok( hr == E_FAIL, "got 0x%08x\n", hr);
7249     todo_wine ok( ret_attr == (void*)0xdeadbeef, "got %p\n", ret_attr);
7250     IXMLDOMElement_Release(elem);
7251
7252     /* add attribute already attached to another document */
7253     doc2 = create_document(&IID_IXMLDOMDocument);
7254
7255     str = SysAllocString( szComplete4 );
7256     hr = IXMLDOMDocument_loadXML( doc2, str, &b );
7257     ok( hr == S_OK, "loadXML failed\n");
7258     ok( b == VARIANT_TRUE, "failed to load XML string\n");
7259     SysFreeString( str );
7260
7261     hr = IXMLDOMDocument_get_documentElement(doc2, &elem);
7262     ok( hr == S_OK, "got 0x%08x\n", hr);
7263     hr = IXMLDOMElement_setAttributeNode(elem, attr, NULL);
7264     todo_wine ok( hr == E_FAIL, "got 0x%08x\n", hr);
7265     IXMLDOMElement_Release(elem);
7266
7267     IXMLDOMDocument_Release(doc2);
7268
7269     IXMLDOMAttribute_Release(attr);
7270     IXMLDOMDocument_Release(doc);
7271     free_bstrs();
7272 }
7273
7274 static void test_put_dataType(void)
7275 {
7276     IXMLDOMCDATASection *cdata;
7277     IXMLDOMDocument *doc;
7278     VARIANT_BOOL b;
7279     HRESULT hr;
7280     BSTR str;
7281
7282     doc = create_document(&IID_IXMLDOMDocument);
7283     if (!doc) return;
7284
7285     str = SysAllocString( szComplete4 );
7286     hr = IXMLDOMDocument_loadXML( doc, str, &b );
7287     ok( hr == S_OK, "loadXML failed\n");
7288     ok( b == VARIANT_TRUE, "failed to load XML string\n");
7289     SysFreeString( str );
7290
7291     hr = IXMLDOMDocument_createCDATASection(doc, _bstr_("test"), &cdata);
7292     ok( hr == S_OK, "got 0x%08x\n", hr);
7293     hr = IXMLDOMCDATASection_put_dataType(cdata, _bstr_("number"));
7294     ok( hr == E_FAIL, "got 0x%08x\n", hr);
7295     hr = IXMLDOMCDATASection_put_dataType(cdata, _bstr_("string"));
7296     ok( hr == E_FAIL, "got 0x%08x\n", hr);
7297     IXMLDOMCDATASection_Release(cdata);
7298
7299     IXMLDOMDocument_Release(doc);
7300     free_bstrs();
7301 }
7302
7303 static void test_createNode(void)
7304 {
7305     IXMLDOMDocument *doc;
7306     IXMLDOMElement *elem;
7307     IXMLDOMNode *node;
7308     VARIANT v, var;
7309     BSTR prefix, str;
7310     HRESULT hr;
7311
7312     doc = create_document(&IID_IXMLDOMDocument);
7313     if (!doc) return;
7314
7315     /* NODE_ELEMENT nodes */
7316     /* 1. specified namespace */
7317     V_VT(&v) = VT_I4;
7318     V_I4(&v) = NODE_ELEMENT;
7319
7320     hr = IXMLDOMDocument_createNode(doc, v, _bstr_("ns1:test"), _bstr_("http://winehq.org"), &node);
7321     ok( hr == S_OK, "got 0x%08x\n", hr);
7322     prefix = NULL;
7323     hr = IXMLDOMNode_get_prefix(node, &prefix);
7324     ok( hr == S_OK, "got 0x%08x\n", hr);
7325     ok(lstrcmpW(prefix, _bstr_("ns1")) == 0, "wrong prefix\n");
7326     SysFreeString(prefix);
7327     IXMLDOMNode_Release(node);
7328
7329     /* 2. default namespace */
7330     hr = IXMLDOMDocument_createNode(doc, v, _bstr_("test"), _bstr_("http://winehq.org/default"), &node);
7331     ok( hr == S_OK, "got 0x%08x\n", hr);
7332     prefix = (void*)0xdeadbeef;
7333     hr = IXMLDOMNode_get_prefix(node, &prefix);
7334     ok( hr == S_FALSE, "got 0x%08x\n", hr);
7335     ok(prefix == 0, "expected empty prefix, got %p\n", prefix);
7336
7337     hr = IXMLDOMNode_QueryInterface(node, &IID_IXMLDOMElement, (void**)&elem);
7338     ok( hr == S_OK, "got 0x%08x\n", hr);
7339
7340     V_VT(&var) = VT_BSTR;
7341     hr = IXMLDOMElement_getAttribute(elem, _bstr_("xmlns"), &var);
7342     ok( hr == S_FALSE, "got 0x%08x\n", hr);
7343     ok( V_VT(&var) == VT_NULL, "got %d\n", V_VT(&var));
7344
7345     str = NULL;
7346     hr = IXMLDOMElement_get_namespaceURI(elem, &str);
7347     ok( hr == S_OK, "got 0x%08x\n", hr);
7348     ok( lstrcmpW(str, _bstr_("http://winehq.org/default")) == 0, "expected default namespace\n");
7349     SysFreeString(str);
7350
7351     IXMLDOMElement_Release(elem);
7352     IXMLDOMNode_Release(node);
7353
7354     IXMLDOMDocument_Release(doc);
7355     free_bstrs();
7356 }
7357
7358 static void test_get_prefix(void)
7359 {
7360     IXMLDOMDocumentFragment *fragment;
7361     IXMLDOMCDATASection *cdata;
7362     IXMLDOMElement *element;
7363     IXMLDOMComment *comment;
7364     IXMLDOMDocument *doc;
7365     HRESULT hr;
7366     BSTR str;
7367
7368     doc = create_document(&IID_IXMLDOMDocument);
7369     if (!doc) return;
7370
7371     /* nodes that can't support prefix */
7372     /* 1. document */
7373     str = (void*)0xdeadbeef;
7374     hr = IXMLDOMDocument_get_prefix(doc, &str);
7375     ok( hr == S_FALSE, "got 0x%08x\n", hr);
7376     ok( str == 0, "got %p\n", str);
7377
7378     hr = IXMLDOMDocument_get_prefix(doc, NULL);
7379     ok( hr == E_INVALIDARG, "got 0x%08x\n", hr);
7380
7381     /* 2. cdata */
7382     hr = IXMLDOMDocument_createCDATASection(doc, NULL, &cdata);
7383     ok(hr == S_OK, "got %08x\n", hr );
7384
7385     str = (void*)0xdeadbeef;
7386     hr = IXMLDOMCDATASection_get_prefix(cdata, &str);
7387     ok(hr == S_FALSE, "got %08x\n", hr);
7388     ok( str == 0, "got %p\n", str);
7389
7390     hr = IXMLDOMCDATASection_get_prefix(cdata, NULL);
7391     ok(hr == E_INVALIDARG, "got %08x\n", hr);
7392     IXMLDOMCDATASection_Release(cdata);
7393
7394     /* 3. comment */
7395     hr = IXMLDOMDocument_createComment(doc, NULL, &comment);
7396     ok(hr == S_OK, "got %08x\n", hr );
7397
7398     str = (void*)0xdeadbeef;
7399     hr = IXMLDOMComment_get_prefix(comment, &str);
7400     ok(hr == S_FALSE, "got %08x\n", hr);
7401     ok( str == 0, "got %p\n", str);
7402
7403     hr = IXMLDOMComment_get_prefix(comment, NULL);
7404     ok(hr == E_INVALIDARG, "got %08x\n", hr);
7405     IXMLDOMComment_Release(comment);
7406
7407     /* 4. fragment */
7408     hr = IXMLDOMDocument_createDocumentFragment(doc, &fragment);
7409     ok(hr == S_OK, "got %08x\n", hr );
7410
7411     str = (void*)0xdeadbeef;
7412     hr = IXMLDOMDocumentFragment_get_prefix(fragment, &str);
7413     ok(hr == S_FALSE, "got %08x\n", hr);
7414     ok( str == 0, "got %p\n", str);
7415
7416     hr = IXMLDOMDocumentFragment_get_prefix(fragment, NULL);
7417     ok(hr == E_INVALIDARG, "got %08x\n", hr);
7418     IXMLDOMDocumentFragment_Release(fragment);
7419
7420     /* no prefix */
7421     hr = IXMLDOMDocument_createElement(doc, _bstr_("elem"), &element);
7422     ok( hr == S_OK, "got 0x%08x\n", hr);
7423
7424     hr = IXMLDOMElement_get_prefix(element, NULL);
7425     ok( hr == E_INVALIDARG, "got 0x%08x\n", hr);
7426
7427     str = (void*)0xdeadbeef;
7428     hr = IXMLDOMElement_get_prefix(element, &str);
7429     ok( hr == S_FALSE, "got 0x%08x\n", hr);
7430     ok( str == 0, "got %p\n", str);
7431
7432     IXMLDOMElement_Release(element);
7433
7434     /* with prefix */
7435     hr = IXMLDOMDocument_createElement(doc, _bstr_("a:elem"), &element);
7436     ok( hr == S_OK, "got 0x%08x\n", hr);
7437
7438     str = (void*)0xdeadbeef;
7439     hr = IXMLDOMElement_get_prefix(element, &str);
7440     ok( hr == S_OK, "got 0x%08x\n", hr);
7441     ok( lstrcmpW(str, _bstr_("a")) == 0, "expected prefix \"a\"\n");
7442     SysFreeString(str);
7443
7444     str = (void*)0xdeadbeef;
7445     hr = IXMLDOMElement_get_namespaceURI(element, &str);
7446     ok( hr == S_FALSE, "got 0x%08x\n", hr);
7447     ok( str == 0, "got %p\n", str);
7448
7449     IXMLDOMElement_Release(element);
7450
7451     IXMLDOMDocument_Release(doc);
7452     free_bstrs();
7453 }
7454
7455 static void test_selectSingleNode(void)
7456 {
7457     IXMLDOMDocument *doc;
7458     IXMLDOMNodeList *list;
7459     IXMLDOMNode *node;
7460     VARIANT_BOOL b;
7461     HRESULT hr;
7462     LONG len;
7463     BSTR str;
7464
7465     doc = create_document(&IID_IXMLDOMDocument);
7466     if (!doc) return;
7467
7468     hr = IXMLDOMDocument_selectSingleNode(doc, NULL, NULL);
7469     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
7470
7471     hr = IXMLDOMDocument_selectNodes(doc, NULL, NULL);
7472     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
7473
7474     str = SysAllocString( szComplete4 );
7475     hr = IXMLDOMDocument_loadXML( doc, str, &b );
7476     ok( hr == S_OK, "loadXML failed\n");
7477     ok( b == VARIANT_TRUE, "failed to load XML string\n");
7478     SysFreeString( str );
7479
7480     hr = IXMLDOMDocument_selectSingleNode(doc, NULL, NULL);
7481     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
7482
7483     hr = IXMLDOMDocument_selectNodes(doc, NULL, NULL);
7484     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
7485
7486     hr = IXMLDOMDocument_selectSingleNode(doc, _bstr_("lc"), NULL);
7487     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
7488
7489     hr = IXMLDOMDocument_selectNodes(doc, _bstr_("lc"), NULL);
7490     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
7491
7492     hr = IXMLDOMDocument_selectSingleNode(doc, _bstr_("lc"), &node);
7493     ok(hr == S_OK, "got 0x%08x\n", hr);
7494     IXMLDOMNode_Release(node);
7495
7496     hr = IXMLDOMDocument_selectNodes(doc, _bstr_("lc"), &list);
7497     ok(hr == S_OK, "got 0x%08x\n", hr);
7498     IXMLDOMNodeList_Release(list);
7499
7500     list = (void*)0xdeadbeef;
7501     hr = IXMLDOMDocument_selectNodes(doc, NULL, &list);
7502     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
7503     ok(list == (void*)0xdeadbeef, "got %p\n", list);
7504
7505     node = (void*)0xdeadbeef;
7506     hr = IXMLDOMDocument_selectSingleNode(doc, _bstr_("nonexistent"), &node);
7507     ok(hr == S_FALSE, "got 0x%08x\n", hr);
7508     ok(node == 0, "got %p\n", node);
7509
7510     list = (void*)0xdeadbeef;
7511     hr = IXMLDOMDocument_selectNodes(doc, _bstr_("nonexistent"), &list);
7512     ok(hr == S_OK, "got 0x%08x\n", hr);
7513     len = 1;
7514     hr = IXMLDOMNodeList_get_length(list, &len);
7515     ok(hr == S_OK, "got 0x%08x\n", hr);
7516     ok(len == 0, "got %d\n", len);
7517     IXMLDOMNodeList_Release(list);
7518
7519     IXMLDOMDocument_Release(doc);
7520     free_bstrs();
7521 }
7522
7523 static void test_events(void)
7524 {
7525     IConnectionPointContainer *conn;
7526     IConnectionPoint *point;
7527     IXMLDOMDocument *doc;
7528     HRESULT hr;
7529
7530     doc = create_document(&IID_IXMLDOMDocument);
7531     if (!doc) return;
7532
7533     hr = IXMLDOMDocument_QueryInterface(doc, &IID_IConnectionPointContainer, (void**)&conn);
7534     ok(hr == S_OK, "got 0x%08x\n", hr);
7535
7536     hr = IConnectionPointContainer_FindConnectionPoint(conn, &IID_IDispatch, &point);
7537     ok(hr == S_OK, "got 0x%08x\n", hr);
7538     IConnectionPoint_Release(point);
7539     hr = IConnectionPointContainer_FindConnectionPoint(conn, &IID_IPropertyNotifySink, &point);
7540     ok(hr == S_OK, "got 0x%08x\n", hr);
7541     IConnectionPoint_Release(point);
7542     hr = IConnectionPointContainer_FindConnectionPoint(conn, &DIID_XMLDOMDocumentEvents, &point);
7543     ok(hr == S_OK, "got 0x%08x\n", hr);
7544     IConnectionPoint_Release(point);
7545
7546     IConnectionPointContainer_Release(conn);
7547
7548     IXMLDOMDocument_Release(doc);
7549 }
7550
7551 static void test_createProcessingInstruction(void)
7552 {
7553     static const WCHAR bodyW[] = {'t','e','s','t',0};
7554     IXMLDOMProcessingInstruction *pi;
7555     IXMLDOMDocument *doc;
7556     WCHAR buff[10];
7557     HRESULT hr;
7558
7559     doc = create_document(&IID_IXMLDOMDocument);
7560     if (!doc) return;
7561
7562     /* test for BSTR handling, pass broken BSTR */
7563     memcpy(&buff[2], bodyW, sizeof(bodyW));
7564     /* just a big length */
7565     *(DWORD*)buff = 0xf0f0;
7566     hr = IXMLDOMDocument_createProcessingInstruction(doc, _bstr_("test"), &buff[2], &pi);
7567     ok(hr == S_OK, "got 0x%08x\n", hr);
7568
7569     IXMLDOMProcessingInstruction_Release(pi);
7570     IXMLDOMDocument_Release(doc);
7571 }
7572
7573 static void test_put_nodeTypedValue(void)
7574 {
7575     IXMLDOMDocument *doc;
7576     IXMLDOMElement *elem;
7577     VARIANT type;
7578     HRESULT hr;
7579
7580     doc = create_document(&IID_IXMLDOMDocument);
7581     if (!doc) return;
7582
7583     hr = IXMLDOMDocument_createElement(doc, _bstr_("Element"), &elem);
7584     ok(hr == S_OK, "got 0x%08x\n", hr);
7585
7586     V_VT(&type) = VT_EMPTY;
7587     hr = IXMLDOMElement_get_dataType(elem, &type);
7588     ok(hr == S_FALSE, "got 0x%08x\n", hr);
7589     ok(V_VT(&type) == VT_NULL, "got %d, expected VT_NULL\n", V_VT(&type));
7590
7591     /* set typed value for untyped node */
7592     V_VT(&type) = VT_I1;
7593     V_I1(&type) = 1;
7594     hr = IXMLDOMElement_put_nodeTypedValue(elem, type);
7595     ok(hr == S_OK, "got 0x%08x\n", hr);
7596
7597     V_VT(&type) = VT_EMPTY;
7598     hr = IXMLDOMElement_get_dataType(elem, &type);
7599     ok(hr == S_FALSE, "got 0x%08x\n", hr);
7600     ok(V_VT(&type) == VT_NULL, "got %d, expected VT_NULL\n", V_VT(&type));
7601
7602     /* no type info stored */
7603     V_VT(&type) = VT_EMPTY;
7604     hr = IXMLDOMElement_get_nodeTypedValue(elem, &type);
7605     ok(hr == S_OK, "got 0x%08x\n", hr);
7606     ok(V_VT(&type) == VT_BSTR, "got %d, expected VT_BSTR\n", V_VT(&type));
7607     ok(memcmp(V_BSTR(&type), _bstr_("1"), 2*sizeof(WCHAR)) == 0,
7608        "got %s, expected \"1\"\n", wine_dbgstr_w(V_BSTR(&type)));
7609     VariantClear(&type);
7610
7611     IXMLDOMDocument_Release(doc);
7612     free_bstrs();
7613 }
7614
7615 static void test_get_xml(void)
7616 {
7617     static const char xmlA[] = "<?xml version=\"1.0\" encoding=\"UTF-16\"?>\r\n<a>test</a>\r\n";
7618     IXMLDOMProcessingInstruction *pi;
7619     IXMLDOMNode *first;
7620     IXMLDOMDocument *doc;
7621     VARIANT_BOOL b;
7622     VARIANT v;
7623     BSTR xml;
7624     HRESULT hr;
7625
7626     doc = create_document(&IID_IXMLDOMDocument);
7627     if (!doc) return;
7628
7629     b = VARIANT_TRUE;
7630     hr = IXMLDOMDocument_loadXML( doc, _bstr_("<a>test</a>"), &b );
7631     ok(hr == S_OK, "got 0x%08x\n", hr);
7632     ok( b == VARIANT_TRUE, "got %d\n", b);
7633
7634     hr = IXMLDOMDocument_createProcessingInstruction(doc, _bstr_("xml"),
7635                              _bstr_("version=\"1.0\" encoding=\"UTF-16\""), &pi);
7636     ok(hr == S_OK, "got 0x%08x\n", hr);
7637
7638     hr = IXMLDOMDocument_get_firstChild(doc, &first);
7639     ok(hr == S_OK, "got 0x%08x\n", hr);
7640
7641     V_UNKNOWN(&v) = (IUnknown*)first;
7642     V_VT(&v) = VT_UNKNOWN;
7643
7644     hr = IXMLDOMDocument_insertBefore(doc, (IXMLDOMNode*)pi, v, NULL);
7645     ok(hr == S_OK, "got 0x%08x\n", hr);
7646
7647     IXMLDOMProcessingInstruction_Release(pi);
7648     IXMLDOMNode_Release(first);
7649
7650     hr = IXMLDOMDocument_get_xml(doc, &xml);
7651     ok(hr == S_OK, "got 0x%08x\n", hr);
7652
7653     ok(memcmp(xml, _bstr_(xmlA), sizeof(xmlA)*sizeof(WCHAR)) == 0,
7654         "got %s, expected %s\n", wine_dbgstr_w(xml), xmlA);
7655     SysFreeString(xml);
7656
7657     IXMLDOMDocument_Release(doc);
7658 }
7659
7660 START_TEST(domdoc)
7661 {
7662     IXMLDOMDocument *doc;
7663     HRESULT hr;
7664
7665     hr = CoInitialize( NULL );
7666     ok( hr == S_OK, "failed to init com\n");
7667     if (hr != S_OK)
7668         return;
7669
7670     test_XMLHTTP();
7671
7672     hr = CoCreateInstance( &CLSID_DOMDocument, NULL, CLSCTX_INPROC_SERVER, &IID_IXMLDOMDocument, (void**)&doc );
7673     if (hr != S_OK)
7674     {
7675         win_skip("IXMLDOMDocument is not available (0x%08x)\n", hr);
7676         return;
7677     }
7678
7679     IXMLDOMDocument_Release(doc);
7680
7681     test_domdoc();
7682     test_persiststreaminit();
7683     test_domnode();
7684     test_refs();
7685     test_create();
7686     test_getElementsByTagName();
7687     test_get_text();
7688     test_get_childNodes();
7689     test_get_firstChild();
7690     test_get_lastChild();
7691     test_removeChild();
7692     test_replaceChild();
7693     test_removeNamedItem();
7694     test_IXMLDOMDocument2();
7695     test_whitespace();
7696     test_XPath();
7697     test_XSLPattern();
7698     test_cloneNode();
7699     test_xmlTypes();
7700     test_nodeTypeTests();
7701     test_DocumentSaveToDocument();
7702     test_DocumentSaveToFile();
7703     test_testTransforms();
7704     test_Namespaces();
7705     test_FormattingXML();
7706     test_nodeTypedValue();
7707     test_TransformWithLoadingLocalFile();
7708     test_put_nodeValue();
7709     test_document_IObjectSafety();
7710     test_splitText();
7711     test_getQualifiedItem();
7712     test_removeQualifiedItem();
7713     test_get_ownerDocument();
7714     test_setAttributeNode();
7715     test_put_dataType();
7716     test_createNode();
7717     test_get_prefix();
7718     test_default_properties();
7719     test_selectSingleNode();
7720     test_events();
7721     test_createProcessingInstruction();
7722     test_put_nodeTypedValue();
7723     test_get_xml();
7724
7725     CoUninitialize();
7726 }