mshtml: Added IHTMLWindow2::focus implementation.
[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         IXMLDOMNode_Release( node );
1813
1814         /* test sequential access of attributes */
1815         node = NULL;
1816         r = IXMLDOMNamedNodeMap_nextNode( map, &node );
1817         ok ( r == S_OK, "nextNode (first time) wrong code\n");
1818         ok ( node != NULL, "nextNode, should be attribute\n");
1819         IXMLDOMNode_Release( node );
1820
1821         r = IXMLDOMNamedNodeMap_nextNode( map, &node );
1822         ok ( r != S_OK, "nextNode (second time) wrong code\n");
1823         ok ( node == NULL, "nextNode, there is no attribute\n");
1824
1825         r = IXMLDOMNamedNodeMap_reset( map );
1826         ok ( r == S_OK, "reset should return S_OK\n");
1827
1828         r = IXMLDOMNamedNodeMap_nextNode( map, &node );
1829         ok ( r == S_OK, "nextNode (third time) wrong code\n");
1830         ok ( node != NULL, "nextNode, should be attribute\n");
1831     }
1832     else
1833         ok( FALSE, "no map\n");
1834
1835     if (node)
1836     {
1837         type = NODE_INVALID;
1838         r = IXMLDOMNode_get_nodeType( node, &type);
1839         ok( r == S_OK, "getNamedItem returned wrong code\n");
1840         ok( type == NODE_ATTRIBUTE, "node not an attribute\n");
1841
1842         str = NULL;
1843         r = IXMLDOMNode_get_baseName( node, NULL );
1844         ok( r == E_INVALIDARG, "get_baseName returned wrong code\n");
1845
1846         str = NULL;
1847         r = IXMLDOMNode_get_baseName( node, &str );
1848         ok( r == S_OK, "get_baseName returned wrong code\n");
1849         ok( lstrcmpW(str,szdl) == 0, "basename was wrong\n");
1850         SysFreeString( str );
1851
1852         r = IXMLDOMNode_get_nodeValue( node, &var );
1853         ok( r == S_OK, "returns %08x\n", r );
1854         ok( V_VT(&var) == VT_BSTR, "vt %x\n", V_VT(&var));
1855         ok( !lstrcmpW(V_BSTR(&var), szstr1), "nodeValue incorrect\n");
1856         VariantClear(&var);
1857
1858         r = IXMLDOMNode_get_childNodes( node, NULL );
1859         ok( r == E_INVALIDARG, "get_childNodes returned wrong code\n");
1860
1861         r = IXMLDOMNode_get_childNodes( node, &list );
1862         ok( r == S_OK, "get_childNodes returned wrong code\n");
1863
1864         if (list)
1865         {
1866             r = IXMLDOMNodeList_nextNode( list, &next );
1867             ok( r == S_OK, "nextNode returned wrong code\n");
1868         }
1869         else
1870             ok( FALSE, "no childlist\n");
1871
1872         if (next)
1873         {
1874             b = 1;
1875             r = IXMLDOMNode_hasChildNodes( next, &b );
1876             ok( r == S_FALSE, "hasChildNoes bad return\n");
1877             ok( b == VARIANT_FALSE, "hasChildNoes wrong result\n");
1878
1879             type = NODE_INVALID;
1880             r = IXMLDOMNode_get_nodeType( next, &type);
1881             ok( r == S_OK, "getNamedItem returned wrong code\n");
1882             ok( type == NODE_TEXT, "node not text\n");
1883
1884             str = (BSTR) 1;
1885             r = IXMLDOMNode_get_baseName( next, &str );
1886             ok( r == S_FALSE, "get_baseName returned wrong code\n");
1887             ok( str == NULL, "basename was wrong\n");
1888             SysFreeString(str);
1889         }
1890         else
1891             ok( FALSE, "no next\n");
1892
1893         if (next)
1894             IXMLDOMNode_Release( next );
1895         next = NULL;
1896         if (list)
1897             IXMLDOMNodeList_Release( list );
1898         list = NULL;
1899         if (node)
1900             IXMLDOMNode_Release( node );
1901     }
1902     else
1903         ok( FALSE, "no node\n");
1904     node = NULL;
1905
1906     if (map)
1907         IXMLDOMNamedNodeMap_Release( map );
1908
1909     /* now traverse the tree from the root element */
1910     if (element)
1911     {
1912         r = IXMLDOMNode_get_childNodes( element, &list );
1913         ok( r == S_OK, "get_childNodes returned wrong code\n");
1914
1915         /* using get_item for child list doesn't advance the position */
1916         ole_check(IXMLDOMNodeList_get_item(list, 1, &node));
1917         expect_node(node, "E2.E2.D1");
1918         IXMLDOMNode_Release(node);
1919         ole_check(IXMLDOMNodeList_nextNode(list, &node));
1920         expect_node(node, "E1.E2.D1");
1921         IXMLDOMNode_Release(node);
1922         ole_check(IXMLDOMNodeList_reset(list));
1923
1924         IXMLDOMNodeList_AddRef(list);
1925         expect_list_and_release(list, "E1.E2.D1 E2.E2.D1 E3.E2.D1 E4.E2.D1");
1926         ole_check(IXMLDOMNodeList_reset(list));
1927
1928         node = (void*)0xdeadbeef;
1929         str = SysAllocString(szdl);
1930         r = IXMLDOMNode_selectSingleNode( element, str, &node );
1931         SysFreeString(str);
1932         ok( r == S_FALSE, "ret %08x\n", r );
1933         ok( node == NULL, "node %p\n", node );
1934
1935         str = SysAllocString(szbs);
1936         r = IXMLDOMNode_selectSingleNode( element, str, &node );
1937         SysFreeString(str);
1938         ok( r == S_OK, "ret %08x\n", r );
1939         r = IXMLDOMNode_Release( node );
1940         ok( r == 0, "ret %08x\n", r );
1941     }
1942     else
1943         ok( FALSE, "no element\n");
1944
1945     if (list)
1946     {
1947         r = IXMLDOMNodeList_QueryInterface(list, &IID_IDispatch, NULL);
1948         ok( r == E_INVALIDARG || r == E_POINTER, "ret %08x\n", r );
1949
1950         r = IXMLDOMNodeList_get_item(list, 0, NULL);
1951         ok(r == E_INVALIDARG, "Exected E_INVALIDARG got %08x\n", r);
1952
1953         r = IXMLDOMNodeList_get_length(list, NULL);
1954         ok(r == E_INVALIDARG, "Exected E_INVALIDARG got %08x\n", r);
1955
1956         r = IXMLDOMNodeList_get_length( list, &count );
1957         ok( r == S_OK, "get_length returns %08x\n", r );
1958         ok( count == 4, "get_length got %d\n", count );
1959
1960         r = IXMLDOMNodeList_nextNode(list, NULL);
1961         ok(r == E_INVALIDARG, "Exected E_INVALIDARG got %08x\n", r);
1962
1963         r = IXMLDOMNodeList_nextNode( list, &node );
1964         ok( r == S_OK, "nextNode returned wrong code\n");
1965     }
1966     else
1967         ok( FALSE, "no list\n");
1968
1969     if (node)
1970     {
1971         type = NODE_INVALID;
1972         r = IXMLDOMNode_get_nodeType( node, &type);
1973         ok( r == S_OK, "getNamedItem returned wrong code\n");
1974         ok( type == NODE_ELEMENT, "node not text\n");
1975
1976         VariantInit(&var);
1977         ok( V_VT(&var) == VT_EMPTY, "variant init failed\n");
1978         r = IXMLDOMNode_get_nodeValue( node, &var );
1979         ok( r == S_FALSE, "nextNode returned wrong code\n");
1980         ok( V_VT(&var) == VT_NULL, "variant wasn't empty\n");
1981         ok( V_BSTR(&var) == NULL, "variant value wasn't null\n");
1982
1983         r = IXMLDOMNode_hasChildNodes( node, NULL );
1984         ok( r == E_INVALIDARG, "hasChildNoes bad return\n");
1985
1986         b = 1;
1987         r = IXMLDOMNode_hasChildNodes( node, &b );
1988         ok( r == S_OK, "hasChildNoes bad return\n");
1989         ok( b == VARIANT_TRUE, "hasChildNoes wrong result\n");
1990
1991         str = NULL;
1992         r = IXMLDOMNode_get_baseName( node, &str );
1993         ok( r == S_OK, "get_baseName returned wrong code\n");
1994         ok( lstrcmpW(str,szbs) == 0, "basename was wrong\n");
1995         SysFreeString(str);
1996     }
1997     else
1998         ok( FALSE, "no node\n");
1999
2000     if (node)
2001         IXMLDOMNode_Release( node );
2002     if (list)
2003         IXMLDOMNodeList_Release( list );
2004     if (element)
2005         IXMLDOMElement_Release( element );
2006
2007     b = FALSE;
2008     str = SysAllocString( szComplete5 );
2009     r = IXMLDOMDocument_loadXML( doc, str, &b );
2010     ok( r == S_OK, "loadXML failed\n");
2011     ok( b == VARIANT_TRUE, "failed to load XML string\n");
2012     SysFreeString( str );
2013
2014     b = 1;
2015     r = IXMLDOMNode_hasChildNodes( doc, &b );
2016     ok( r == S_OK, "hasChildNoes bad return\n");
2017     ok( b == VARIANT_TRUE, "hasChildNoes wrong result\n");
2018
2019     r = IXMLDOMDocument_get_documentElement( doc, &element );
2020     ok( r == S_OK, "should be a document element\n");
2021     ok( element != NULL, "should be an element\n");
2022
2023     if (element)
2024     {
2025         static const WCHAR szSSearch[] = {'S',':','s','e','a','r','c','h',0};
2026         BSTR tag = NULL;
2027
2028         /* check if the tag is correct */
2029         r = IXMLDOMElement_get_tagName( element, &tag );
2030         ok( r == S_OK, "couldn't get tag name\n");
2031         ok( tag != NULL, "tag was null\n");
2032         ok( !lstrcmpW( tag, szSSearch ), "incorrect tag name\n");
2033         SysFreeString( tag );
2034     }
2035
2036     if (element)
2037         IXMLDOMElement_Release( element );
2038     ok(IXMLDOMDocument_Release( doc ) == 0, "document is not destroyed\n");
2039 }
2040
2041 static void test_refs(void)
2042 {
2043     HRESULT r;
2044     BSTR str;
2045     VARIANT_BOOL b;
2046     IXMLDOMDocument *doc;
2047     IXMLDOMElement *element = NULL;
2048     IXMLDOMNode *node = NULL, *node2;
2049     IXMLDOMNodeList *node_list = NULL;
2050     LONG ref;
2051     IUnknown *unk, *unk2;
2052
2053     doc = create_document(&IID_IXMLDOMDocument);
2054     if (!doc) return;
2055
2056     ref = IXMLDOMDocument_Release(doc);
2057     ok( ref == 0, "ref %d\n", ref);
2058
2059     doc = create_document(&IID_IXMLDOMDocument);
2060     if (!doc) return;
2061
2062     str = SysAllocString( szComplete4 );
2063     r = IXMLDOMDocument_loadXML( doc, str, &b );
2064     ok( r == S_OK, "loadXML failed\n");
2065     ok( b == VARIANT_TRUE, "failed to load XML string\n");
2066     SysFreeString( str );
2067
2068     ref = IXMLDOMDocument_AddRef( doc );
2069     ok( ref == 2, "ref %d\n", ref );
2070     ref = IXMLDOMDocument_AddRef( doc );
2071     ok( ref == 3, "ref %d\n", ref );
2072     IXMLDOMDocument_Release( doc );
2073     IXMLDOMDocument_Release( doc );
2074
2075     r = IXMLDOMDocument_get_documentElement( doc, &element );
2076     ok( r == S_OK, "should be a document element\n");
2077     ok( element != NULL, "should be an element\n");
2078
2079     ref = IXMLDOMDocument_AddRef( doc );
2080     ok( ref == 2, "ref %d\n", ref );
2081     IXMLDOMDocument_Release( doc );
2082
2083     r = IXMLDOMElement_get_childNodes( element, &node_list );
2084     ok( r == S_OK, "rets %08x\n", r);
2085
2086     ref = IXMLDOMNodeList_AddRef( node_list );
2087     ok( ref == 2, "ref %d\n", ref );
2088     IXMLDOMNodeList_Release( node_list );
2089
2090     IXMLDOMNodeList_get_item( node_list, 0, &node );
2091     ok( r == S_OK, "rets %08x\n", r);
2092
2093     IXMLDOMNodeList_get_item( node_list, 0, &node2 );
2094     ok( r == S_OK, "rets %08x\n", r);
2095
2096     ref = IXMLDOMNode_AddRef( node );
2097     ok( ref == 2, "ref %d\n", ref );
2098     IXMLDOMNode_Release( node );
2099
2100     ref = IXMLDOMNode_Release( node );
2101     ok( ref == 0, "ref %d\n", ref );
2102     ref = IXMLDOMNode_Release( node2 );
2103     ok( ref == 0, "ref %d\n", ref );
2104
2105     ref = IXMLDOMNodeList_Release( node_list );
2106     ok( ref == 0, "ref %d\n", ref );
2107
2108     ok( node != node2, "node %p node2 %p\n", node, node2 );
2109
2110     ref = IXMLDOMDocument_Release( doc );
2111     ok( ref == 0, "ref %d\n", ref );
2112
2113     ref = IXMLDOMElement_AddRef( element );
2114     todo_wine {
2115     ok( ref == 3, "ref %d\n", ref );
2116     }
2117     IXMLDOMElement_Release( element );
2118
2119     /* IUnknown must be unique however we obtain it */
2120     r = IXMLDOMElement_QueryInterface( element, &IID_IUnknown, (void**)&unk );
2121     ok( r == S_OK, "rets %08x\n", r );
2122     r = IXMLDOMElement_QueryInterface( element, &IID_IXMLDOMNode, (void**)&node );
2123     ok( r == S_OK, "rets %08x\n", r );
2124     r = IXMLDOMNode_QueryInterface( node, &IID_IUnknown, (void**)&unk2 );
2125     ok( r == S_OK, "rets %08x\n", r );
2126     ok( unk == unk2, "unk %p unk2 %p\n", unk, unk2 );
2127
2128     IUnknown_Release( unk2 );
2129     IUnknown_Release( unk );
2130     IXMLDOMNode_Release( node );
2131
2132     IXMLDOMElement_Release( element );
2133
2134 }
2135
2136 static void test_create(void)
2137 {
2138     static const WCHAR szOne[] = {'1',0};
2139     static const WCHAR szOneGarbage[] = {'1','G','a','r','b','a','g','e',0};
2140     HRESULT r;
2141     VARIANT var;
2142     BSTR str, name;
2143     IXMLDOMDocument *doc;
2144     IXMLDOMElement *element;
2145     IXMLDOMComment *comment;
2146     IXMLDOMText *text;
2147     IXMLDOMCDATASection *cdata;
2148     IXMLDOMNode *root, *node, *child;
2149     IXMLDOMNamedNodeMap *attr_map;
2150     IUnknown *unk;
2151     LONG ref;
2152     LONG num;
2153
2154     doc = create_document(&IID_IXMLDOMDocument);
2155     if (!doc) return;
2156
2157     /* types not supported for creation */
2158     V_VT(&var) = VT_I1;
2159     V_I1(&var) = NODE_DOCUMENT;
2160     node = (IXMLDOMNode*)0x1;
2161     r = IXMLDOMDocument_createNode( doc, var, NULL, NULL, &node );
2162     ok( r == E_INVALIDARG, "returns %08x\n", r );
2163     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
2164
2165     V_VT(&var) = VT_I1;
2166     V_I1(&var) = NODE_DOCUMENT_TYPE;
2167     node = (IXMLDOMNode*)0x1;
2168     r = IXMLDOMDocument_createNode( doc, var, NULL, NULL, &node );
2169     ok( r == E_INVALIDARG, "returns %08x\n", r );
2170     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
2171
2172     V_VT(&var) = VT_I1;
2173     V_I1(&var) = NODE_ENTITY;
2174     node = (IXMLDOMNode*)0x1;
2175     r = IXMLDOMDocument_createNode( doc, var, NULL, NULL, &node );
2176     ok( r == E_INVALIDARG, "returns %08x\n", r );
2177     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
2178
2179     V_VT(&var) = VT_I1;
2180     V_I1(&var) = NODE_NOTATION;
2181     node = (IXMLDOMNode*)0x1;
2182     r = IXMLDOMDocument_createNode( doc, var, NULL, NULL, &node );
2183     ok( r == E_INVALIDARG, "returns %08x\n", r );
2184     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
2185
2186     /* NODE_COMMENT */
2187     V_VT(&var) = VT_I1;
2188     V_I1(&var) = NODE_COMMENT;
2189     node = NULL;
2190     r = IXMLDOMDocument_createNode( doc, var, NULL, NULL, &node );
2191     ok( r == S_OK, "returns %08x\n", r );
2192     ok( node != NULL, "\n");
2193
2194     r = IXMLDOMNode_QueryInterface(node, &IID_IXMLDOMComment, (void**)&comment);
2195     ok( r == S_OK, "returns %08x\n", r );
2196     IXMLDOMNode_Release(node);
2197
2198     str = NULL;
2199     r = IXMLDOMComment_get_data(comment, &str);
2200     ok( r == S_OK, "returns %08x\n", r );
2201     ok( str && SysStringLen(str) == 0, "expected empty comment, %p\n", str);
2202     IXMLDOMComment_Release(comment);
2203     SysFreeString(str);
2204
2205     node = (IXMLDOMNode*)0x1;
2206     r = IXMLDOMDocument_createNode( doc, var, _bstr_(""), NULL, &node );
2207     ok( r == S_OK, "returns %08x\n", r );
2208
2209     r = IXMLDOMNode_QueryInterface(node, &IID_IXMLDOMComment, (void**)&comment);
2210     ok( r == S_OK, "returns %08x\n", r );
2211     IXMLDOMNode_Release(node);
2212
2213     str = NULL;
2214     r = IXMLDOMComment_get_data(comment, &str);
2215     ok( r == S_OK, "returns %08x\n", r );
2216     ok( str && SysStringLen(str) == 0, "expected empty comment, %p\n", str);
2217     IXMLDOMComment_Release(comment);
2218     SysFreeString(str);
2219
2220     node = (IXMLDOMNode*)0x1;
2221     r = IXMLDOMDocument_createNode( doc, var, _bstr_("blah"), NULL, &node );
2222     ok( r == S_OK, "returns %08x\n", r );
2223
2224     r = IXMLDOMNode_QueryInterface(node, &IID_IXMLDOMComment, (void**)&comment);
2225     ok( r == S_OK, "returns %08x\n", r );
2226     IXMLDOMNode_Release(node);
2227
2228     str = NULL;
2229     r = IXMLDOMComment_get_data(comment, &str);
2230     ok( r == S_OK, "returns %08x\n", r );
2231     ok( str && SysStringLen(str) == 0, "expected empty comment, %p\n", str);
2232     IXMLDOMComment_Release(comment);
2233     SysFreeString(str);
2234
2235     /* NODE_TEXT */
2236     V_VT(&var) = VT_I1;
2237     V_I1(&var) = NODE_TEXT;
2238     node = NULL;
2239     r = IXMLDOMDocument_createNode( doc, var, NULL, NULL, &node );
2240     ok( r == S_OK, "returns %08x\n", r );
2241     ok( node != NULL, "\n");
2242
2243     r = IXMLDOMNode_QueryInterface(node, &IID_IXMLDOMText, (void**)&text);
2244     ok( r == S_OK, "returns %08x\n", r );
2245     IXMLDOMNode_Release(node);
2246
2247     str = NULL;
2248     r = IXMLDOMText_get_data(text, &str);
2249     ok( r == S_OK, "returns %08x\n", r );
2250     ok( str && SysStringLen(str) == 0, "expected empty comment, %p\n", str);
2251     IXMLDOMText_Release(text);
2252     SysFreeString(str);
2253
2254     node = (IXMLDOMNode*)0x1;
2255     r = IXMLDOMDocument_createNode( doc, var, _bstr_(""), NULL, &node );
2256     ok( r == S_OK, "returns %08x\n", r );
2257
2258     r = IXMLDOMNode_QueryInterface(node, &IID_IXMLDOMText, (void**)&text);
2259     ok( r == S_OK, "returns %08x\n", r );
2260     IXMLDOMNode_Release(node);
2261
2262     str = NULL;
2263     r = IXMLDOMText_get_data(text, &str);
2264     ok( r == S_OK, "returns %08x\n", r );
2265     ok( str && SysStringLen(str) == 0, "expected empty comment, %p\n", str);
2266     IXMLDOMText_Release(text);
2267     SysFreeString(str);
2268
2269     node = (IXMLDOMNode*)0x1;
2270     r = IXMLDOMDocument_createNode( doc, var, _bstr_("blah"), NULL, &node );
2271     ok( r == S_OK, "returns %08x\n", r );
2272
2273     r = IXMLDOMNode_QueryInterface(node, &IID_IXMLDOMText, (void**)&text);
2274     ok( r == S_OK, "returns %08x\n", r );
2275     IXMLDOMNode_Release(node);
2276
2277     str = NULL;
2278     r = IXMLDOMText_get_data(text, &str);
2279     ok( r == S_OK, "returns %08x\n", r );
2280     ok( str && SysStringLen(str) == 0, "expected empty comment, %p\n", str);
2281     IXMLDOMText_Release(text);
2282     SysFreeString(str);
2283
2284     /* NODE_CDATA_SECTION */
2285     V_VT(&var) = VT_I1;
2286     V_I1(&var) = NODE_CDATA_SECTION;
2287     node = NULL;
2288     r = IXMLDOMDocument_createNode( doc, var, NULL, NULL, &node );
2289     ok( r == S_OK, "returns %08x\n", r );
2290     ok( node != NULL, "\n");
2291
2292     r = IXMLDOMNode_QueryInterface(node, &IID_IXMLDOMCDATASection, (void**)&cdata);
2293     ok( r == S_OK, "returns %08x\n", r );
2294     IXMLDOMNode_Release(node);
2295
2296     str = NULL;
2297     r = IXMLDOMCDATASection_get_data(cdata, &str);
2298     ok( r == S_OK, "returns %08x\n", r );
2299     ok( str && SysStringLen(str) == 0, "expected empty comment, %p\n", str);
2300     IXMLDOMCDATASection_Release(cdata);
2301     SysFreeString(str);
2302
2303     node = (IXMLDOMNode*)0x1;
2304     r = IXMLDOMDocument_createNode( doc, var, _bstr_(""), NULL, &node );
2305     ok( r == S_OK, "returns %08x\n", r );
2306
2307     r = IXMLDOMNode_QueryInterface(node, &IID_IXMLDOMCDATASection, (void**)&cdata);
2308     ok( r == S_OK, "returns %08x\n", r );
2309     IXMLDOMNode_Release(node);
2310
2311     str = NULL;
2312     r = IXMLDOMCDATASection_get_data(cdata, &str);
2313     ok( r == S_OK, "returns %08x\n", r );
2314     ok( str && SysStringLen(str) == 0, "expected empty comment, %p\n", str);
2315     IXMLDOMCDATASection_Release(cdata);
2316     SysFreeString(str);
2317
2318     node = (IXMLDOMNode*)0x1;
2319     r = IXMLDOMDocument_createNode( doc, var, _bstr_("blah"), NULL, &node );
2320     ok( r == S_OK, "returns %08x\n", r );
2321
2322     r = IXMLDOMNode_QueryInterface(node, &IID_IXMLDOMCDATASection, (void**)&cdata);
2323     ok( r == S_OK, "returns %08x\n", r );
2324     IXMLDOMNode_Release(node);
2325
2326     str = NULL;
2327     r = IXMLDOMCDATASection_get_data(cdata, &str);
2328     ok( r == S_OK, "returns %08x\n", r );
2329     ok( str && SysStringLen(str) == 0, "expected empty comment, %p\n", str);
2330     IXMLDOMCDATASection_Release(cdata);
2331     SysFreeString(str);
2332
2333     /* NODE_ATTRIBUTE */
2334     V_VT(&var) = VT_I1;
2335     V_I1(&var) = NODE_ATTRIBUTE;
2336     node = (IXMLDOMNode*)0x1;
2337     r = IXMLDOMDocument_createNode( doc, var, NULL, NULL, &node );
2338     ok( r == E_FAIL, "returns %08x\n", r );
2339     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
2340
2341     V_VT(&var) = VT_I1;
2342     V_I1(&var) = NODE_ATTRIBUTE;
2343     node = (IXMLDOMNode*)0x1;
2344     r = IXMLDOMDocument_createNode( doc, var, _bstr_(""), NULL, &node );
2345     ok( r == E_FAIL, "returns %08x\n", r );
2346     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
2347
2348     V_VT(&var) = VT_I1;
2349     V_I1(&var) = NODE_ATTRIBUTE;
2350     str = SysAllocString( szlc );
2351     r = IXMLDOMDocument_createNode( doc, var, str, NULL, &node );
2352     ok( r == S_OK, "returns %08x\n", r );
2353     if( SUCCEEDED(r) ) IXMLDOMNode_Release( node );
2354     SysFreeString(str);
2355
2356     /* a name is required for attribute, try a BSTR with first null wchar */
2357     V_VT(&var) = VT_I1;
2358     V_I1(&var) = NODE_ATTRIBUTE;
2359     str = SysAllocString( szstr1 );
2360     str[0] = 0;
2361     node = (IXMLDOMNode*)0x1;
2362     r = IXMLDOMDocument_createNode( doc, var, str, NULL, &node );
2363     ok( r == E_FAIL, "returns %08x\n", r );
2364     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
2365     SysFreeString(str);
2366
2367     /* NODE_PROCESSING_INSTRUCTION */
2368     V_VT(&var) = VT_I1;
2369     V_I1(&var) = NODE_PROCESSING_INSTRUCTION;
2370     node = (IXMLDOMNode*)0x1;
2371     r = IXMLDOMDocument_createNode( doc, var, NULL, NULL, &node );
2372     ok( r == E_FAIL, "returns %08x\n", r );
2373     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
2374
2375     V_VT(&var) = VT_I1;
2376     V_I1(&var) = NODE_PROCESSING_INSTRUCTION;
2377     node = (IXMLDOMNode*)0x1;
2378     r = IXMLDOMDocument_createNode( doc, var, _bstr_(""), NULL, &node );
2379     ok( r == E_FAIL, "returns %08x\n", r );
2380     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
2381
2382     V_VT(&var) = VT_I1;
2383     V_I1(&var) = NODE_PROCESSING_INSTRUCTION;
2384     r = IXMLDOMDocument_createNode( doc, var, _bstr_("pi"), NULL, NULL );
2385     ok( r == E_INVALIDARG, "returns %08x\n", r );
2386
2387     /* NODE_ENTITY_REFERENCE */
2388     V_VT(&var) = VT_I1;
2389     V_I1(&var) = NODE_ENTITY_REFERENCE;
2390     node = (IXMLDOMNode*)0x1;
2391     r = IXMLDOMDocument_createNode( doc, var, NULL, NULL, &node );
2392     ok( r == E_FAIL, "returns %08x\n", r );
2393     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
2394
2395     V_VT(&var) = VT_I1;
2396     V_I1(&var) = NODE_ENTITY_REFERENCE;
2397     node = (IXMLDOMNode*)0x1;
2398     r = IXMLDOMDocument_createNode( doc, var, _bstr_(""), NULL, &node );
2399     ok( r == E_FAIL, "returns %08x\n", r );
2400     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
2401
2402     /* NODE_ELEMENT */
2403     V_VT(&var) = VT_I1;
2404     V_I1(&var) = NODE_ELEMENT;
2405     node = (IXMLDOMNode*)0x1;
2406     r = IXMLDOMDocument_createNode( doc, var, NULL, NULL, &node );
2407     ok( r == E_FAIL, "returns %08x\n", r );
2408     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
2409
2410     V_VT(&var) = VT_I1;
2411     V_I1(&var) = NODE_ELEMENT;
2412     node = (IXMLDOMNode*)0x1;
2413     r = IXMLDOMDocument_createNode( doc, var, _bstr_(""), NULL, &node );
2414     ok( r == E_FAIL, "returns %08x\n", r );
2415     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
2416
2417     V_VT(&var) = VT_I1;
2418     V_I1(&var) = NODE_ELEMENT;
2419     str = SysAllocString( szlc );
2420     r = IXMLDOMDocument_createNode( doc, var, str, NULL, &node );
2421     ok( r == S_OK, "returns %08x\n", r );
2422     if( SUCCEEDED(r) ) IXMLDOMNode_Release( node );
2423
2424     V_VT(&var) = VT_I1;
2425     V_I1(&var) = NODE_ELEMENT;
2426     r = IXMLDOMDocument_createNode( doc, var, str, NULL, NULL );
2427     ok( r == E_INVALIDARG, "returns %08x\n", r );
2428
2429     V_VT(&var) = VT_R4;
2430     V_R4(&var) = NODE_ELEMENT;
2431     r = IXMLDOMDocument_createNode( doc, var, str, NULL, &node );
2432     ok( r == S_OK, "returns %08x\n", r );
2433     if( SUCCEEDED(r) ) IXMLDOMNode_Release( node );
2434
2435     V_VT(&var) = VT_BSTR;
2436     V_BSTR(&var) = SysAllocString( szOne );
2437     r = IXMLDOMDocument_createNode( doc, var, str, NULL, &node );
2438     ok( r == S_OK, "returns %08x\n", r );
2439     if( SUCCEEDED(r) ) IXMLDOMNode_Release( node );
2440     VariantClear(&var);
2441
2442     V_VT(&var) = VT_BSTR;
2443     V_BSTR(&var) = SysAllocString( szOneGarbage );
2444     r = IXMLDOMDocument_createNode( doc, var, str, NULL, &node );
2445     ok( r == E_INVALIDARG, "returns %08x\n", r );
2446     if( SUCCEEDED(r) ) IXMLDOMNode_Release( node );
2447     VariantClear(&var);
2448
2449     V_VT(&var) = VT_I4;
2450     V_I4(&var) = NODE_ELEMENT;
2451     r = IXMLDOMDocument_createNode( doc, var, str, NULL, &node );
2452     ok( r == S_OK, "returns %08x\n", r );
2453     r = IXMLDOMDocument_appendChild( doc, node, &root );
2454     ok( r == S_OK, "returns %08x\n", r );
2455     ok( node == root, "%p %p\n", node, root );
2456
2457     ref = IXMLDOMNode_AddRef( node );
2458     ok(ref == 3, "ref %d\n", ref);
2459     IXMLDOMNode_Release( node );
2460
2461     ref = IXMLDOMNode_Release( node );
2462     ok(ref == 1, "ref %d\n", ref);
2463     SysFreeString( str );
2464
2465     V_VT(&var) = VT_I4;
2466     V_I4(&var) = NODE_ELEMENT;
2467     str = SysAllocString( szbs );
2468     r = IXMLDOMDocument_createNode( doc, var, str, NULL, &node );
2469     ok( r == S_OK, "returns %08x\n", r );
2470     SysFreeString( str );
2471
2472     ref = IXMLDOMNode_AddRef( node );
2473     ok(ref == 2, "ref = %d\n", ref);
2474     IXMLDOMNode_Release( node );
2475
2476     r = IXMLDOMNode_QueryInterface( node, &IID_IUnknown, (void**)&unk );
2477     ok( r == S_OK, "returns %08x\n", r );
2478
2479     ref = IXMLDOMNode_AddRef( unk );
2480     ok(ref == 3, "ref = %d\n", ref);
2481     IXMLDOMNode_Release( unk );
2482
2483     V_VT(&var) = VT_EMPTY;
2484     r = IXMLDOMNode_insertBefore( root, (IXMLDOMNode*)unk, var, &child );
2485     ok( r == S_OK, "returns %08x\n", r );
2486     ok( unk == (IUnknown*)child, "%p %p\n", unk, child );
2487     IXMLDOMNode_Release( child );
2488     IUnknown_Release( unk );
2489
2490
2491     V_VT(&var) = VT_NULL;
2492     V_DISPATCH(&var) = (IDispatch*)node;
2493     r = IXMLDOMNode_insertBefore( root, node, var, &child );
2494     ok( r == S_OK, "returns %08x\n", r );
2495     ok( node == child, "%p %p\n", node, child );
2496     IXMLDOMNode_Release( child );
2497
2498
2499     V_VT(&var) = VT_NULL;
2500     V_DISPATCH(&var) = (IDispatch*)node;
2501     r = IXMLDOMNode_insertBefore( root, node, var, NULL );
2502     ok( r == S_OK, "returns %08x\n", r );
2503     IXMLDOMNode_Release( node );
2504
2505     r = IXMLDOMNode_QueryInterface( root, &IID_IXMLDOMElement, (void**)&element );
2506     ok( r == S_OK, "returns %08x\n", r );
2507
2508     r = IXMLDOMElement_get_attributes( element, &attr_map );
2509     ok( r == S_OK, "returns %08x\n", r );
2510     r = IXMLDOMNamedNodeMap_get_length( attr_map, &num );
2511     ok( r == S_OK, "returns %08x\n", r );
2512     ok( num == 0, "num %d\n", num );
2513     IXMLDOMNamedNodeMap_Release( attr_map );
2514
2515     V_VT(&var) = VT_BSTR;
2516     V_BSTR(&var) = SysAllocString( szstr1 );
2517     name = SysAllocString( szdl );
2518     r = IXMLDOMElement_setAttribute( element, name, var );
2519     ok( r == S_OK, "returns %08x\n", r );
2520     r = IXMLDOMElement_get_attributes( element, &attr_map );
2521     ok( r == S_OK, "returns %08x\n", r );
2522     r = IXMLDOMNamedNodeMap_get_length( attr_map, &num );
2523     ok( r == S_OK, "returns %08x\n", r );
2524     ok( num == 1, "num %d\n", num );
2525     IXMLDOMNamedNodeMap_Release( attr_map );
2526     VariantClear(&var);
2527
2528     V_VT(&var) = VT_BSTR;
2529     V_BSTR(&var) = SysAllocString( szstr2 );
2530     r = IXMLDOMElement_setAttribute( element, name, var );
2531     ok( r == S_OK, "returns %08x\n", r );
2532     r = IXMLDOMElement_get_attributes( element, &attr_map );
2533     ok( r == S_OK, "returns %08x\n", r );
2534     r = IXMLDOMNamedNodeMap_get_length( attr_map, &num );
2535     ok( r == S_OK, "returns %08x\n", r );
2536     ok( num == 1, "num %d\n", num );
2537     IXMLDOMNamedNodeMap_Release( attr_map );
2538     VariantClear(&var);
2539     r = IXMLDOMElement_getAttribute( element, name, &var );
2540     ok( r == S_OK, "returns %08x\n", r );
2541     ok( !lstrcmpW(V_BSTR(&var), szstr2), "wrong attr value\n");
2542     VariantClear(&var);
2543     SysFreeString(name);
2544
2545     V_VT(&var) = VT_BSTR;
2546     V_BSTR(&var) = SysAllocString( szstr1 );
2547     name = SysAllocString( szlc );
2548     r = IXMLDOMElement_setAttribute( element, name, var );
2549     ok( r == S_OK, "returns %08x\n", r );
2550     r = IXMLDOMElement_get_attributes( element, &attr_map );
2551     ok( r == S_OK, "returns %08x\n", r );
2552     r = IXMLDOMNamedNodeMap_get_length( attr_map, &num );
2553     ok( r == S_OK, "returns %08x\n", r );
2554     ok( num == 2, "num %d\n", num );
2555     IXMLDOMNamedNodeMap_Release( attr_map );
2556     VariantClear(&var);
2557     SysFreeString(name);
2558
2559     V_VT(&var) = VT_I4;
2560     V_I4(&var) = 10;
2561     name = SysAllocString( szbs );
2562     r = IXMLDOMElement_setAttribute( element, name, var );
2563     ok( r == S_OK, "returns %08x\n", r );
2564     VariantClear(&var);
2565     r = IXMLDOMElement_getAttribute( element, name, &var );
2566     ok( r == S_OK, "returns %08x\n", r );
2567     ok( V_VT(&var) == VT_BSTR, "variant type %x\n", V_VT(&var));
2568     VariantClear(&var);
2569     SysFreeString(name);
2570
2571     /* Create an Attribute */
2572     V_VT(&var) = VT_I4;
2573     V_I4(&var) = NODE_ATTRIBUTE;
2574     str = SysAllocString( szAttribute );
2575     r = IXMLDOMDocument_createNode( doc, var, str, NULL, &node );
2576     ok( r == S_OK, "returns %08x\n", r );
2577     ok( node != NULL, "node was null\n");
2578     SysFreeString(str);
2579
2580     r = IXMLDOMNode_get_nodeTypeString(node, &str);
2581     ok( r == S_OK, "returns %08x\n", r );
2582     ok( !lstrcmpW( str, _bstr_("attribute") ), "incorrect nodeTypeString string\n");
2583     SysFreeString(str);
2584     IXMLDOMNode_Release( node );
2585
2586     IXMLDOMElement_Release( element );
2587     IXMLDOMNode_Release( root );
2588     IXMLDOMDocument_Release( doc );
2589 }
2590
2591 static void test_getElementsByTagName(void)
2592 {
2593     IXMLDOMNodeList *node_list;
2594     IXMLDOMDocument *doc;
2595     IXMLDOMElement *elem;
2596     WCHAR buff[100];
2597     VARIANT_BOOL b;
2598     HRESULT r;
2599     LONG len;
2600     BSTR str;
2601
2602     doc = create_document(&IID_IXMLDOMDocument);
2603     if (!doc) return;
2604
2605     str = SysAllocString( szComplete4 );
2606     r = IXMLDOMDocument_loadXML( doc, str, &b );
2607     ok( r == S_OK, "loadXML failed\n");
2608     ok( b == VARIANT_TRUE, "failed to load XML string\n");
2609     SysFreeString( str );
2610
2611     str = SysAllocString( szstar );
2612
2613     /* null arguments cases */
2614     r = IXMLDOMDocument_getElementsByTagName(doc, NULL, &node_list);
2615     ok( r == E_INVALIDARG, "ret %08x\n", r );
2616     r = IXMLDOMDocument_getElementsByTagName(doc, str, NULL);
2617     ok( r == E_INVALIDARG, "ret %08x\n", r );
2618
2619     r = IXMLDOMDocument_getElementsByTagName(doc, str, &node_list);
2620     ok( r == S_OK, "ret %08x\n", r );
2621     r = IXMLDOMNodeList_get_length( node_list, &len );
2622     ok( r == S_OK, "ret %08x\n", r );
2623     ok( len == 6, "len %d\n", len );
2624
2625     test_disp((IUnknown*)node_list);
2626
2627     IXMLDOMNodeList_Release( node_list );
2628     SysFreeString( str );
2629
2630     /* broken query BSTR */
2631     memcpy(&buff[2], szstar, sizeof(szstar));
2632     /* just a big length */
2633     *(DWORD*)buff = 0xf0f0;
2634     r = IXMLDOMDocument_getElementsByTagName(doc, &buff[2], &node_list);
2635     ok( r == S_OK, "ret %08x\n", r );
2636     r = IXMLDOMNodeList_get_length( node_list, &len );
2637     ok( r == S_OK, "ret %08x\n", r );
2638     ok( len == 6, "len %d\n", len );
2639     IXMLDOMNodeList_Release( node_list );
2640
2641     str = SysAllocString( szbs );
2642     r = IXMLDOMDocument_getElementsByTagName(doc, str, &node_list);
2643     ok( r == S_OK, "ret %08x\n", r );
2644     r = IXMLDOMNodeList_get_length( node_list, &len );
2645     ok( r == S_OK, "ret %08x\n", r );
2646     ok( len == 1, "len %d\n", len );
2647     IXMLDOMNodeList_Release( node_list );
2648     SysFreeString( str );
2649
2650     str = SysAllocString( szdl );
2651     r = IXMLDOMDocument_getElementsByTagName(doc, str, &node_list);
2652     ok( r == S_OK, "ret %08x\n", r );
2653     r = IXMLDOMNodeList_get_length( node_list, &len );
2654     ok( r == S_OK, "ret %08x\n", r );
2655     ok( len == 0, "len %d\n", len );
2656     IXMLDOMNodeList_Release( node_list );
2657     SysFreeString( str );
2658
2659     str = SysAllocString( szstr1 );
2660     r = IXMLDOMDocument_getElementsByTagName(doc, str, &node_list);
2661     ok( r == S_OK, "ret %08x\n", r );
2662     r = IXMLDOMNodeList_get_length( node_list, &len );
2663     ok( r == S_OK, "ret %08x\n", r );
2664     ok( len == 0, "len %d\n", len );
2665     IXMLDOMNodeList_Release( node_list );
2666     SysFreeString( str );
2667
2668     /* test for element */
2669     r = IXMLDOMDocument_get_documentElement(doc, &elem);
2670     ok( r == S_OK, "ret %08x\n", r );
2671
2672     str = SysAllocString( szstar );
2673
2674     /* null arguments cases */
2675     r = IXMLDOMElement_getElementsByTagName(elem, NULL, &node_list);
2676     ok( r == E_INVALIDARG, "ret %08x\n", r );
2677     r = IXMLDOMElement_getElementsByTagName(elem, str, NULL);
2678     ok( r == E_INVALIDARG, "ret %08x\n", r );
2679
2680     r = IXMLDOMElement_getElementsByTagName(elem, str, &node_list);
2681     ok( r == S_OK, "ret %08x\n", r );
2682     r = IXMLDOMNodeList_get_length( node_list, &len );
2683     ok( r == S_OK, "ret %08x\n", r );
2684     ok( len == 5, "len %d\n", len );
2685     expect_list_and_release(node_list, "E1.E2.D1 E2.E2.D1 E3.E2.D1 E4.E2.D1 E1.E4.E2.D1");
2686     SysFreeString( str );
2687
2688     /* broken query BSTR */
2689     memcpy(&buff[2], szstar, sizeof(szstar));
2690     /* just a big length */
2691     *(DWORD*)buff = 0xf0f0;
2692     r = IXMLDOMElement_getElementsByTagName(elem, &buff[2], &node_list);
2693     ok( r == S_OK, "ret %08x\n", r );
2694     r = IXMLDOMNodeList_get_length( node_list, &len );
2695     ok( r == S_OK, "ret %08x\n", r );
2696     ok( len == 5, "len %d\n", len );
2697     IXMLDOMNodeList_Release( node_list );
2698
2699     IXMLDOMElement_Release(elem);
2700
2701     IXMLDOMDocument_Release( doc );
2702 }
2703
2704 static void test_get_text(void)
2705 {
2706     HRESULT r;
2707     BSTR str;
2708     VARIANT_BOOL b;
2709     IXMLDOMDocument *doc;
2710     IXMLDOMNode *node, *node2, *node3;
2711     IXMLDOMNode *nodeRoot;
2712     IXMLDOMNodeList *node_list;
2713     IXMLDOMNamedNodeMap *node_map;
2714     LONG len;
2715
2716     doc = create_document(&IID_IXMLDOMDocument);
2717     if (!doc) return;
2718
2719     str = SysAllocString( szComplete4 );
2720     r = IXMLDOMDocument_loadXML( doc, str, &b );
2721     ok( r == S_OK, "loadXML failed\n");
2722     ok( b == VARIANT_TRUE, "failed to load XML string\n");
2723     SysFreeString( str );
2724
2725     str = SysAllocString( szbs );
2726     r = IXMLDOMDocument_getElementsByTagName( doc, str, &node_list );
2727     ok( r == S_OK, "ret %08x\n", r );
2728     SysFreeString(str);
2729
2730     /* Test to get all child node text. */
2731     r = IXMLDOMDocument_QueryInterface(doc, &IID_IXMLDOMNode, (void**)&nodeRoot);
2732     ok( r == S_OK, "ret %08x\n", r );
2733     if(r == S_OK)
2734     {
2735         r = IXMLDOMNode_get_text( nodeRoot, &str );
2736         ok( r == S_OK, "ret %08x\n", r );
2737         ok( compareIgnoreReturns(str, _bstr_("fn1.txt\n\n fn2.txt \n\nf1\n")), "wrong get_text: %s\n", wine_dbgstr_w(str));
2738         SysFreeString(str);
2739
2740         IXMLDOMNode_Release(nodeRoot);
2741     }
2742
2743     if (0) {
2744     /* this test crashes on win9x */
2745     r = IXMLDOMNodeList_QueryInterface(node_list, &IID_IDispatch, NULL);
2746     ok( r == E_INVALIDARG, "ret %08x\n", r );
2747     }
2748
2749     r = IXMLDOMNodeList_get_length( node_list, NULL );
2750     ok( r == E_INVALIDARG, "ret %08x\n", r );
2751
2752     r = IXMLDOMNodeList_get_length( node_list, &len );
2753     ok( r == S_OK, "ret %08x\n", r );
2754     ok( len == 1, "expect 1 got %d\n", len );
2755
2756     r = IXMLDOMNodeList_get_item( node_list, 0, NULL );
2757     ok( r == E_INVALIDARG, "ret %08x\n", r );
2758
2759     r = IXMLDOMNodeList_nextNode( node_list, NULL );
2760     ok( r == E_INVALIDARG, "ret %08x\n", r );
2761
2762     r = IXMLDOMNodeList_get_item( node_list, 0, &node );
2763     ok( r == S_OK, "ret %08x\n", r );
2764     IXMLDOMNodeList_Release( node_list );
2765
2766     /* Invalid output parameter*/
2767     r = IXMLDOMNode_get_text( node, NULL );
2768     ok( r == E_INVALIDARG, "ret %08x\n", r );
2769
2770     r = IXMLDOMNode_get_text( node, &str );
2771     ok( r == S_OK, "ret %08x\n", r );
2772     ok( !memcmp(str, szfn1_txt, lstrlenW(szfn1_txt) ), "wrong string\n" );
2773     SysFreeString(str);
2774
2775     r = IXMLDOMNode_get_attributes( node, &node_map );
2776     ok( r == S_OK, "ret %08x\n", r );
2777
2778     str = SysAllocString( szvr );
2779     r = IXMLDOMNamedNodeMap_getNamedItem( node_map, str, &node2 );
2780     ok( r == S_OK, "ret %08x\n", r );
2781     SysFreeString(str);
2782
2783     r = IXMLDOMNode_get_text( node2, &str );
2784     ok( r == S_OK, "ret %08x\n", r );
2785     ok( !memcmp(str, szstr2, sizeof(szstr2)), "wrong string\n" );
2786     SysFreeString(str);
2787
2788     r = IXMLDOMNode_get_firstChild( node2, &node3 );
2789     ok( r == S_OK, "ret %08x\n", r );
2790
2791     r = IXMLDOMNode_get_text( node3, &str );
2792     ok( r == S_OK, "ret %08x\n", r );
2793     ok( !memcmp(str, szstr2, sizeof(szstr2)), "wrong string\n" );
2794     SysFreeString(str);
2795
2796
2797     IXMLDOMNode_Release( node3 );
2798     IXMLDOMNode_Release( node2 );
2799     IXMLDOMNamedNodeMap_Release( node_map );
2800     IXMLDOMNode_Release( node );
2801     IXMLDOMDocument_Release( doc );
2802 }
2803
2804 static void test_get_childNodes(void)
2805 {
2806     HRESULT r;
2807     BSTR str;
2808     VARIANT_BOOL b;
2809     IXMLDOMDocument *doc;
2810     IXMLDOMElement *element;
2811     IXMLDOMNode *node, *node2;
2812     IXMLDOMNodeList *node_list, *node_list2;
2813     LONG len;
2814
2815     doc = create_document(&IID_IXMLDOMDocument);
2816     if (!doc) return;
2817
2818     str = SysAllocString( szComplete4 );
2819     r = IXMLDOMDocument_loadXML( doc, str, &b );
2820     ok( r == S_OK, "loadXML failed\n");
2821     ok( b == VARIANT_TRUE, "failed to load XML string\n");
2822     SysFreeString( str );
2823
2824     r = IXMLDOMDocument_get_documentElement( doc, &element );
2825     ok( r == S_OK, "ret %08x\n", r);
2826
2827     r = IXMLDOMElement_get_childNodes( element, &node_list );
2828     ok( r == S_OK, "ret %08x\n", r);
2829
2830     r = IXMLDOMNodeList_get_length( node_list, &len );
2831     ok( r == S_OK, "ret %08x\n", r);
2832     ok( len == 4, "len %d\n", len);
2833
2834     r = IXMLDOMNodeList_get_item( node_list, 2, &node );
2835     ok( r == S_OK, "ret %08x\n", r);
2836
2837     r = IXMLDOMNode_get_childNodes( node, &node_list2 );
2838     ok( r == S_OK, "ret %08x\n", r);
2839
2840     r = IXMLDOMNodeList_get_length( node_list2, &len );
2841     ok( r == S_OK, "ret %08x\n", r);
2842     ok( len == 0, "len %d\n", len);
2843
2844     r = IXMLDOMNodeList_get_item( node_list2, 0, &node2);
2845     ok( r == S_FALSE, "ret %08x\n", r);
2846
2847     IXMLDOMNodeList_Release( node_list2 );
2848     IXMLDOMNode_Release( node );
2849     IXMLDOMNodeList_Release( node_list );
2850     IXMLDOMElement_Release( element );
2851     IXMLDOMDocument_Release( doc );
2852 }
2853
2854 static void test_get_firstChild(void)
2855 {
2856     static WCHAR xmlW[] = {'x','m','l',0};
2857     IXMLDOMDocument *doc;
2858     IXMLDOMNode *node;
2859     VARIANT_BOOL b;
2860     HRESULT r;
2861     BSTR str;
2862
2863     doc = create_document(&IID_IXMLDOMDocument);
2864     if (!doc) return;
2865
2866     str = SysAllocString( szComplete4 );
2867     r = IXMLDOMDocument_loadXML( doc, str, &b );
2868     ok( r == S_OK, "loadXML failed\n");
2869     ok( b == VARIANT_TRUE, "failed to load XML string\n");
2870     SysFreeString( str );
2871
2872     r = IXMLDOMDocument_get_firstChild( doc, &node );
2873     ok( r == S_OK, "ret %08x\n", r);
2874
2875     r = IXMLDOMNode_get_nodeName( node, &str );
2876     ok( r == S_OK, "ret %08x\n", r);
2877
2878     ok(memcmp(str, xmlW, sizeof(xmlW)) == 0, "expected \"xml\" node name\n");
2879
2880     SysFreeString(str);
2881     IXMLDOMNode_Release( node );
2882     IXMLDOMDocument_Release( doc );
2883 }
2884
2885 static void test_get_lastChild(void)
2886 {
2887     static WCHAR lcW[] = {'l','c',0};
2888     static WCHAR foW[] = {'f','o',0};
2889     IXMLDOMDocument *doc;
2890     IXMLDOMNode *node, *child;
2891     VARIANT_BOOL b;
2892     HRESULT r;
2893     BSTR str;
2894
2895     doc = create_document(&IID_IXMLDOMDocument);
2896     if (!doc) return;
2897
2898     str = SysAllocString( szComplete4 );
2899     r = IXMLDOMDocument_loadXML( doc, str, &b );
2900     ok( r == S_OK, "loadXML failed\n");
2901     ok( b == VARIANT_TRUE, "failed to load XML string\n");
2902     SysFreeString( str );
2903
2904     r = IXMLDOMDocument_get_lastChild( doc, &node );
2905     ok( r == S_OK, "ret %08x\n", r);
2906
2907     r = IXMLDOMNode_get_nodeName( node, &str );
2908     ok( r == S_OK, "ret %08x\n", r);
2909
2910     ok(memcmp(str, lcW, sizeof(lcW)) == 0, "expected \"lc\" node name\n");
2911     SysFreeString(str);
2912
2913     r = IXMLDOMNode_get_lastChild( node, &child );
2914     ok( r == S_OK, "ret %08x\n", r);
2915
2916     r = IXMLDOMNode_get_nodeName( child, &str );
2917     ok( r == S_OK, "ret %08x\n", r);
2918
2919     ok(memcmp(str, foW, sizeof(foW)) == 0, "expected \"fo\" node name\n");
2920     SysFreeString(str);
2921
2922     IXMLDOMNode_Release( child );
2923     IXMLDOMNode_Release( node );
2924     IXMLDOMDocument_Release( doc );
2925 }
2926
2927 static void test_removeChild(void)
2928 {
2929     HRESULT r;
2930     BSTR str;
2931     VARIANT_BOOL b;
2932     IXMLDOMDocument *doc;
2933     IXMLDOMElement *element, *lc_element;
2934     IXMLDOMNode *fo_node, *ba_node, *removed_node, *temp_node, *lc_node;
2935     IXMLDOMNodeList *root_list, *fo_list;
2936
2937     doc = create_document(&IID_IXMLDOMDocument);
2938     if (!doc) return;
2939
2940     str = SysAllocString( szComplete4 );
2941     r = IXMLDOMDocument_loadXML( doc, str, &b );
2942     ok( r == S_OK, "loadXML failed\n");
2943     ok( b == VARIANT_TRUE, "failed to load XML string\n");
2944     SysFreeString( str );
2945
2946     r = IXMLDOMDocument_get_documentElement( doc, &element );
2947     ok( r == S_OK, "ret %08x\n", r);
2948
2949     r = IXMLDOMElement_get_childNodes( element, &root_list );
2950     ok( r == S_OK, "ret %08x\n", r);
2951
2952     r = IXMLDOMNodeList_get_item( root_list, 3, &fo_node );
2953     ok( r == S_OK, "ret %08x\n", r);
2954
2955     r = IXMLDOMNode_get_childNodes( fo_node, &fo_list );
2956     ok( r == S_OK, "ret %08x\n", r);
2957
2958     r = IXMLDOMNodeList_get_item( fo_list, 0, &ba_node );
2959     ok( r == S_OK, "ret %08x\n", r);
2960
2961     /* invalid parameter: NULL ptr */
2962     removed_node = (void*)0xdeadbeef;
2963     r = IXMLDOMElement_removeChild( element, NULL, &removed_node );
2964     ok( r == E_INVALIDARG, "ret %08x\n", r );
2965     ok( removed_node == (void*)0xdeadbeef, "%p\n", removed_node );
2966
2967     /* ba_node is a descendant of element, but not a direct child. */
2968     removed_node = (void*)0xdeadbeef;
2969     r = IXMLDOMElement_removeChild( element, ba_node, &removed_node );
2970     ok( r == E_INVALIDARG, "ret %08x\n", r );
2971     ok( removed_node == NULL, "%p\n", removed_node );
2972
2973     r = IXMLDOMElement_removeChild( element, fo_node, &removed_node );
2974     ok( r == S_OK, "ret %08x\n", r);
2975     ok( fo_node == removed_node, "node %p node2 %p\n", fo_node, removed_node );
2976
2977     /* try removing already removed child */
2978     temp_node = (void*)0xdeadbeef;
2979     r = IXMLDOMElement_removeChild( element, fo_node, &temp_node );
2980     ok( r == E_INVALIDARG, "ret %08x\n", r);
2981     ok( temp_node == NULL, "%p\n", temp_node );
2982     IXMLDOMNode_Release( fo_node );
2983
2984     /* the removed node has no parent anymore */
2985     r = IXMLDOMNode_get_parentNode( removed_node, &temp_node );
2986     ok( r == S_FALSE, "ret %08x\n", r);
2987     ok( temp_node == NULL, "%p\n", temp_node );
2988
2989     IXMLDOMNode_Release( removed_node );
2990     IXMLDOMNode_Release( ba_node );
2991     IXMLDOMNodeList_Release( fo_list );
2992
2993     r = IXMLDOMNodeList_get_item( root_list, 0, &lc_node );
2994     ok( r == S_OK, "ret %08x\n", r);
2995
2996     r = IXMLDOMElement_QueryInterface( lc_node, &IID_IXMLDOMElement, (void**)&lc_element );
2997     ok( r == S_OK, "ret %08x\n", r);
2998
2999     /* MS quirk: passing wrong interface pointer works, too */
3000     r = IXMLDOMElement_removeChild( element, (IXMLDOMNode*)lc_element, NULL );
3001     ok( r == S_OK, "ret %08x\n", r);
3002     IXMLDOMElement_Release( lc_element );
3003
3004     r = IXMLDOMNode_get_parentNode( lc_node, &temp_node );
3005     ok( r == S_FALSE, "ret %08x\n", r);
3006     ok( temp_node == NULL, "%p\n", temp_node );
3007
3008     IXMLDOMNode_Release( lc_node );
3009     IXMLDOMNodeList_Release( root_list );
3010     IXMLDOMElement_Release( element );
3011     IXMLDOMDocument_Release( doc );
3012 }
3013
3014 static void test_replaceChild(void)
3015 {
3016     HRESULT r;
3017     BSTR str;
3018     VARIANT_BOOL b;
3019     IXMLDOMDocument *doc;
3020     IXMLDOMElement *element, *ba_element;
3021     IXMLDOMNode *fo_node, *ba_node, *lc_node, *removed_node, *temp_node;
3022     IXMLDOMNodeList *root_list, *fo_list;
3023     IUnknown * unk1, *unk2;
3024     LONG len;
3025
3026     doc = create_document(&IID_IXMLDOMDocument);
3027     if (!doc) return;
3028
3029     str = SysAllocString( szComplete4 );
3030     r = IXMLDOMDocument_loadXML( doc, str, &b );
3031     ok( r == S_OK, "loadXML failed\n");
3032     ok( b == VARIANT_TRUE, "failed to load XML string\n");
3033     SysFreeString( str );
3034
3035     r = IXMLDOMDocument_get_documentElement( doc, &element );
3036     ok( r == S_OK, "ret %08x\n", r);
3037
3038     r = IXMLDOMElement_get_childNodes( element, &root_list );
3039     ok( r == S_OK, "ret %08x\n", r);
3040
3041     r = IXMLDOMNodeList_get_item( root_list, 0, &lc_node );
3042     ok( r == S_OK, "ret %08x\n", r);
3043
3044     r = IXMLDOMNodeList_get_item( root_list, 3, &fo_node );
3045     ok( r == S_OK, "ret %08x\n", r);
3046
3047     r = IXMLDOMNode_get_childNodes( fo_node, &fo_list );
3048     ok( r == S_OK, "ret %08x\n", r);
3049
3050     r = IXMLDOMNodeList_get_item( fo_list, 0, &ba_node );
3051     ok( r == S_OK, "ret %08x\n", r);
3052
3053     IXMLDOMNodeList_Release( fo_list );
3054
3055     /* invalid parameter: NULL ptr for element to remove */
3056     removed_node = (void*)0xdeadbeef;
3057     r = IXMLDOMElement_replaceChild( element, ba_node, NULL, &removed_node );
3058     ok( r == E_INVALIDARG, "ret %08x\n", r );
3059     ok( removed_node == (void*)0xdeadbeef, "%p\n", removed_node );
3060
3061     /* invalid parameter: NULL for replacement element. (Sic!) */
3062     removed_node = (void*)0xdeadbeef;
3063     r = IXMLDOMElement_replaceChild( element, NULL, fo_node, &removed_node );
3064     ok( r == E_INVALIDARG, "ret %08x\n", r );
3065     ok( removed_node == (void*)0xdeadbeef, "%p\n", removed_node );
3066
3067     /* invalid parameter: OldNode is not a child */
3068     removed_node = (void*)0xdeadbeef;
3069     r = IXMLDOMElement_replaceChild( element, lc_node, ba_node, &removed_node );
3070     ok( r == E_INVALIDARG, "ret %08x\n", r );
3071     ok( removed_node == NULL, "%p\n", removed_node );
3072     IXMLDOMNode_Release( lc_node );
3073
3074     /* invalid parameter: would create loop */
3075     removed_node = (void*)0xdeadbeef;
3076     r = IXMLDOMNode_replaceChild( fo_node, fo_node, ba_node, &removed_node );
3077     ok( r == E_FAIL, "ret %08x\n", r );
3078     ok( removed_node == NULL, "%p\n", removed_node );
3079
3080     r = IXMLDOMElement_replaceChild( element, ba_node, fo_node, NULL );
3081     ok( r == S_OK, "ret %08x\n", r );
3082
3083     r = IXMLDOMNodeList_get_item( root_list, 3, &temp_node );
3084     ok( r == S_OK, "ret %08x\n", r );
3085
3086     /* ba_node and temp_node refer to the same node, yet they
3087        are different interface pointers */
3088     ok( ba_node != temp_node, "ba_node %p temp_node %p\n", ba_node, temp_node);
3089     r = IXMLDOMNode_QueryInterface( temp_node, &IID_IUnknown, (void**)&unk1);
3090     ok( r == S_OK, "ret %08x\n", r );
3091     r = IXMLDOMNode_QueryInterface( ba_node, &IID_IUnknown, (void**)&unk2);
3092     ok( r == S_OK, "ret %08x\n", r );
3093     todo_wine ok( unk1 == unk2, "unk1 %p unk2 %p\n", unk1, unk2);
3094
3095     IUnknown_Release( unk1 );
3096     IUnknown_Release( unk2 );
3097
3098     /* ba_node should have been removed from below fo_node */
3099     r = IXMLDOMNode_get_childNodes( fo_node, &fo_list );
3100     ok( r == S_OK, "ret %08x\n", r );
3101
3102     /* MS quirk: replaceChild also accepts elements instead of nodes */
3103     r = IXMLDOMNode_QueryInterface( ba_node, &IID_IXMLDOMElement, (void**)&ba_element);
3104     ok( r == S_OK, "ret %08x\n", r );
3105
3106     r = IXMLDOMElement_replaceChild( element, ba_node, (IXMLDOMNode*)ba_element, &removed_node );
3107     ok( r == S_OK, "ret %08x\n", r );
3108     IXMLDOMElement_Release( ba_element );
3109
3110     r = IXMLDOMNodeList_get_length( fo_list, &len);
3111     ok( r == S_OK, "ret %08x\n", r );
3112     ok( len == 0, "len %d\n", len);
3113
3114     IXMLDOMNodeList_Release( fo_list );
3115
3116     IXMLDOMNode_Release(ba_node);
3117     IXMLDOMNode_Release(fo_node);
3118     IXMLDOMNode_Release(temp_node);
3119     IXMLDOMNodeList_Release( root_list );
3120     IXMLDOMElement_Release( element );
3121     IXMLDOMDocument_Release( doc );
3122 }
3123
3124 static void test_removeNamedItem(void)
3125 {
3126     IXMLDOMDocument *doc;
3127     IXMLDOMElement *element;
3128     IXMLDOMNode *pr_node, *removed_node, *removed_node2;
3129     IXMLDOMNodeList *root_list;
3130     IXMLDOMNamedNodeMap * pr_attrs;
3131     VARIANT_BOOL b;
3132     BSTR str;
3133     LONG len;
3134     HRESULT r;
3135
3136     doc = create_document(&IID_IXMLDOMDocument);
3137     if (!doc) return;
3138
3139     str = SysAllocString( szComplete4 );
3140     r = IXMLDOMDocument_loadXML( doc, str, &b );
3141     ok( r == S_OK, "loadXML failed\n");
3142     ok( b == VARIANT_TRUE, "failed to load XML string\n");
3143     SysFreeString( str );
3144
3145     r = IXMLDOMDocument_get_documentElement( doc, &element );
3146     ok( r == S_OK, "ret %08x\n", r);
3147
3148     r = IXMLDOMElement_get_childNodes( element, &root_list );
3149     ok( r == S_OK, "ret %08x\n", r);
3150
3151     r = IXMLDOMNodeList_get_item( root_list, 1, &pr_node );
3152     ok( r == S_OK, "ret %08x\n", r);
3153
3154     r = IXMLDOMNode_get_attributes( pr_node, &pr_attrs );
3155     ok( r == S_OK, "ret %08x\n", r);
3156
3157     r = IXMLDOMNamedNodeMap_get_length( pr_attrs, &len );
3158     ok( r == S_OK, "ret %08x\n", r);
3159     ok( len == 3, "length %d\n", len);
3160
3161     removed_node = (void*)0xdeadbeef;
3162     r = IXMLDOMNamedNodeMap_removeNamedItem( pr_attrs, NULL, &removed_node);
3163     ok ( r == E_INVALIDARG, "ret %08x\n", r);
3164     ok ( removed_node == (void*)0xdeadbeef, "got %p\n", removed_node);
3165
3166     removed_node = (void*)0xdeadbeef;
3167     str = SysAllocString(szvr);
3168     r = IXMLDOMNamedNodeMap_removeNamedItem( pr_attrs, str, &removed_node);
3169     ok ( r == S_OK, "ret %08x\n", r);
3170
3171     removed_node2 = (void*)0xdeadbeef;
3172     r = IXMLDOMNamedNodeMap_removeNamedItem( pr_attrs, str, &removed_node2);
3173     ok ( r == S_FALSE, "ret %08x\n", r);
3174     ok ( removed_node2 == NULL, "got %p\n", removed_node2 );
3175
3176     r = IXMLDOMNamedNodeMap_get_length( pr_attrs, &len );
3177     ok( r == S_OK, "ret %08x\n", r);
3178     ok( len == 2, "length %d\n", len);
3179
3180     r = IXMLDOMNamedNodeMap_setNamedItem( pr_attrs, removed_node, NULL);
3181     ok ( r == S_OK, "ret %08x\n", r);
3182     IXMLDOMNode_Release(removed_node);
3183
3184     r = IXMLDOMNamedNodeMap_get_length( pr_attrs, &len );
3185     ok( r == S_OK, "ret %08x\n", r);
3186     ok( len == 3, "length %d\n", len);
3187
3188     r = IXMLDOMNamedNodeMap_removeNamedItem( pr_attrs, str, NULL);
3189     ok ( r == S_OK, "ret %08x\n", r);
3190
3191     r = IXMLDOMNamedNodeMap_get_length( pr_attrs, &len );
3192     ok( r == S_OK, "ret %08x\n", r);
3193     ok( len == 2, "length %d\n", len);
3194
3195     r = IXMLDOMNamedNodeMap_removeNamedItem( pr_attrs, str, NULL);
3196     ok ( r == S_FALSE, "ret %08x\n", r);
3197
3198     SysFreeString(str);
3199
3200     IXMLDOMNamedNodeMap_Release( pr_attrs );
3201     IXMLDOMNode_Release( pr_node );
3202     IXMLDOMNodeList_Release( root_list );
3203     IXMLDOMElement_Release( element );
3204     IXMLDOMDocument_Release( doc );
3205 }
3206
3207 static void test_XMLHTTP(void)
3208 {
3209     static const WCHAR wszBody[] = {'m','o','d','e','=','T','e','s','t',0};
3210     static const WCHAR wszPOST[] = {'P','O','S','T',0};
3211     static const WCHAR wszUrl[] = {'h','t','t','p',':','/','/',
3212         'c','r','o','s','s','o','v','e','r','.','c','o','d','e','w','e','a','v','e','r','s','.','c','o','m','/',
3213         'p','o','s','t','t','e','s','t','.','p','h','p',0};
3214     static const WCHAR xmltestW[] = {'h','t','t','p',':','/','/',
3215         'c','r','o','s','s','o','v','e','r','.','c','o','d','e','w','e','a','v','e','r','s','.','c','o','m','/',
3216         'x','m','l','t','e','s','t','.','x','m','l',0};
3217     static const WCHAR wszExpectedResponse[] = {'F','A','I','L','E','D',0};
3218     static const CHAR xmltestbodyA[] = "<?xml version=\"1.0\" encoding=\"utf-8\"?>\n<a>TEST</a>\n";
3219
3220     IXMLHttpRequest *pXMLHttpRequest;
3221     BSTR bstrResponse, method, url;
3222     VARIANT dummy;
3223     VARIANT async;
3224     VARIANT varbody;
3225     LONG state, status, ref, bound;
3226     void *ptr;
3227     IDispatch *event;
3228     HRESULT hr = CoCreateInstance(&CLSID_XMLHTTPRequest, NULL,
3229                                   CLSCTX_INPROC_SERVER, &IID_IXMLHttpRequest,
3230                                   (void **)&pXMLHttpRequest);
3231     if (FAILED(hr))
3232     {
3233         win_skip("IXMLHTTPRequest is not available (0x%08x)\n", hr);
3234         return;
3235     }
3236
3237     VariantInit(&dummy);
3238     V_VT(&dummy) = VT_ERROR;
3239     V_ERROR(&dummy) = DISP_E_MEMBERNOTFOUND;
3240     VariantInit(&async);
3241     V_VT(&async) = VT_BOOL;
3242     V_BOOL(&async) = VARIANT_FALSE;
3243     V_VT(&varbody) = VT_BSTR;
3244     V_BSTR(&varbody) = SysAllocString(wszBody);
3245
3246     method = SysAllocString(wszPOST);
3247     url = SysAllocString(wszUrl);
3248
3249 if (0)
3250 {
3251     /* crashes on win98 */
3252     hr = IXMLHttpRequest_put_onreadystatechange(pXMLHttpRequest, NULL);
3253     ok(hr == S_OK, "got 0x%08x\n", hr);
3254 }
3255
3256     hr = IXMLHttpRequest_abort(pXMLHttpRequest);
3257     ok(hr == S_OK, "got 0x%08x\n", hr);
3258
3259     /* send before open */
3260     hr = IXMLHttpRequest_send(pXMLHttpRequest, dummy);
3261     ok(hr == E_FAIL || broken(hr == E_UNEXPECTED) /* win9x, win2k */, "got 0x%08x\n", hr);
3262
3263     /* initial status code */
3264     hr = IXMLHttpRequest_get_status(pXMLHttpRequest, NULL);
3265     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
3266
3267     status = 0xdeadbeef;
3268     hr = IXMLHttpRequest_get_status(pXMLHttpRequest, &status);
3269     ok(hr == E_FAIL || broken(hr == E_UNEXPECTED) /* win9x, win2k */, "got 0x%08x\n", hr);
3270     ok(status == 0xdeadbeef, "got %d\n", status);
3271
3272     /* invalid parameters */
3273     hr = IXMLHttpRequest_open(pXMLHttpRequest, NULL, NULL, async, dummy, dummy);
3274     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
3275
3276     hr = IXMLHttpRequest_open(pXMLHttpRequest, method, NULL, async, dummy, dummy);
3277     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
3278
3279     hr = IXMLHttpRequest_open(pXMLHttpRequest, NULL, url, async, dummy, dummy);
3280     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
3281
3282     hr = IXMLHttpRequest_setRequestHeader(pXMLHttpRequest, NULL, NULL);
3283     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
3284
3285     hr = IXMLHttpRequest_setRequestHeader(pXMLHttpRequest, _bstr_("header1"), NULL);
3286     ok(hr == E_FAIL || broken(hr == E_UNEXPECTED) /* win9x, win2k */, "got 0x%08x\n", hr);
3287
3288     hr = IXMLHttpRequest_setRequestHeader(pXMLHttpRequest, NULL, _bstr_("value1"));
3289     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
3290
3291     hr = IXMLHttpRequest_setRequestHeader(pXMLHttpRequest, _bstr_("header1"), _bstr_("value1"));
3292     ok(hr == E_FAIL || broken(hr == E_UNEXPECTED) /* win9x, win2k */, "got 0x%08x\n", hr);
3293
3294     hr = IXMLHttpRequest_get_readyState(pXMLHttpRequest, NULL);
3295     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
3296
3297     state = -1;
3298     hr = IXMLHttpRequest_get_readyState(pXMLHttpRequest, &state);
3299     ok(hr == S_OK, "got 0x%08x\n", hr);
3300     ok(state == READYSTATE_UNINITIALIZED, "got %d, expected READYSTATE_UNINITIALIZED\n", state);
3301
3302     event = create_dispevent();
3303     ref = IDispatch_AddRef(event);
3304     ok(ref == 2, "got %d\n", ref);
3305     IDispatch_Release(event);
3306
3307     hr = IXMLHttpRequest_put_onreadystatechange(pXMLHttpRequest, event);
3308     ok(hr == S_OK, "got 0x%08x\n", hr);
3309
3310     ref = IDispatch_AddRef(event);
3311     ok(ref == 3, "got %d\n", ref);
3312     IDispatch_Release(event);
3313
3314     g_unexpectedcall = g_expectedcall = 0;
3315
3316     hr = IXMLHttpRequest_open(pXMLHttpRequest, method, url, async, dummy, dummy);
3317     ok(hr == S_OK, "got 0x%08x\n", hr);
3318
3319     ok(g_unexpectedcall == 0, "unexpected disp event call\n");
3320     ok(g_expectedcall == 1 || broken(g_expectedcall == 0) /* win2k */, "no expected disp event call\n");
3321
3322     /* status code after ::open() */
3323     status = 0xdeadbeef;
3324     hr = IXMLHttpRequest_get_status(pXMLHttpRequest, &status);
3325     ok(hr == E_FAIL || broken(hr == E_UNEXPECTED) /* win9x, win2k */, "got 0x%08x\n", hr);
3326     ok(status == 0xdeadbeef, "got %d\n", status);
3327
3328     state = -1;
3329     hr = IXMLHttpRequest_get_readyState(pXMLHttpRequest, &state);
3330     ok(hr == S_OK, "got 0x%08x\n", hr);
3331     ok(state == READYSTATE_LOADING, "got %d, expected READYSTATE_LOADING\n", state);
3332
3333     hr = IXMLHttpRequest_abort(pXMLHttpRequest);
3334     ok(hr == S_OK, "got 0x%08x\n", hr);
3335
3336     state = -1;
3337     hr = IXMLHttpRequest_get_readyState(pXMLHttpRequest, &state);
3338     ok(hr == S_OK, "got 0x%08x\n", hr);
3339     ok(state == READYSTATE_UNINITIALIZED || broken(state == READYSTATE_LOADING) /* win98, win2k */,
3340         "got %d, expected READYSTATE_UNINITIALIZED\n", state);
3341
3342     hr = IXMLHttpRequest_open(pXMLHttpRequest, method, url, async, dummy, dummy);
3343     ok(hr == S_OK, "got 0x%08x\n", hr);
3344
3345     hr = IXMLHttpRequest_setRequestHeader(pXMLHttpRequest, _bstr_("header1"), _bstr_("value1"));
3346     ok(hr == S_OK, "got 0x%08x\n", hr);
3347
3348     hr = IXMLHttpRequest_setRequestHeader(pXMLHttpRequest, NULL, _bstr_("value1"));
3349     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
3350
3351     hr = IXMLHttpRequest_setRequestHeader(pXMLHttpRequest, _bstr_(""), _bstr_("value1"));
3352     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
3353
3354     SysFreeString(method);
3355     SysFreeString(url);
3356
3357     hr = IXMLHttpRequest_send(pXMLHttpRequest, varbody);
3358     if (hr == INET_E_RESOURCE_NOT_FOUND)
3359     {
3360         skip("No connection could be made with crossover.codeweavers.com\n");
3361         IXMLHttpRequest_Release(pXMLHttpRequest);
3362         return;
3363     }
3364     ok(hr == S_OK, "got 0x%08x\n", hr);
3365
3366     /* status code after ::send() */
3367     status = 0xdeadbeef;
3368     hr = IXMLHttpRequest_get_status(pXMLHttpRequest, &status);
3369     ok(hr == S_OK, "got 0x%08x\n", hr);
3370     ok(status == 200, "got %d\n", status);
3371
3372     /* another ::send() after completed request */
3373     hr = IXMLHttpRequest_send(pXMLHttpRequest, varbody);
3374     ok(hr == E_FAIL || broken(hr == E_UNEXPECTED) /* win9x, win2k */, "got 0x%08x\n", hr);
3375
3376     VariantClear(&varbody);
3377
3378     hr = IXMLHttpRequest_get_responseText(pXMLHttpRequest, &bstrResponse);
3379     ok(hr == S_OK, "got 0x%08x\n", hr);
3380     /* the server currently returns "FAILED" because the Content-Type header is
3381      * not what the server expects */
3382     if(hr == S_OK)
3383     {
3384         ok(!memcmp(bstrResponse, wszExpectedResponse, sizeof(wszExpectedResponse)),
3385             "expected %s, got %s\n", wine_dbgstr_w(wszExpectedResponse), wine_dbgstr_w(bstrResponse));
3386         SysFreeString(bstrResponse);
3387     }
3388
3389     /* GET request */
3390     url = SysAllocString(xmltestW);
3391
3392     hr = IXMLHttpRequest_open(pXMLHttpRequest, _bstr_("GET"), url, async, dummy, dummy);
3393     ok(hr == S_OK, "got 0x%08x\n", hr);
3394
3395     V_VT(&varbody) = VT_EMPTY;
3396
3397     hr = IXMLHttpRequest_send(pXMLHttpRequest, varbody);
3398     if (hr == INET_E_RESOURCE_NOT_FOUND)
3399     {
3400         skip("No connection could be made with crossover.codeweavers.com\n");
3401         IXMLHttpRequest_Release(pXMLHttpRequest);
3402         return;
3403     }
3404     ok(hr == S_OK, "got 0x%08x\n", hr);
3405
3406     hr = IXMLHttpRequest_get_responseText(pXMLHttpRequest, NULL);
3407     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
3408
3409     hr = IXMLHttpRequest_get_responseText(pXMLHttpRequest, &bstrResponse);
3410     ok(hr == S_OK, "got 0x%08x\n", hr);
3411     if(hr == S_OK)
3412     {
3413         ok(!memcmp(bstrResponse, _bstr_(xmltestbodyA), sizeof(xmltestbodyA)*sizeof(WCHAR)),
3414             "expected %s, got %s\n", xmltestbodyA, wine_dbgstr_w(bstrResponse));
3415         SysFreeString(bstrResponse);
3416     }
3417
3418     hr = IXMLHttpRequest_get_responseBody(pXMLHttpRequest, NULL);
3419     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
3420
3421     V_VT(&varbody) = VT_EMPTY;
3422     hr = IXMLHttpRequest_get_responseBody(pXMLHttpRequest, &varbody);
3423     ok(hr == S_OK, "got 0x%08x\n", hr);
3424     ok(V_VT(&varbody) == (VT_ARRAY|VT_UI1), "got type %d, expected %d\n", V_VT(&varbody), VT_ARRAY|VT_UI1);
3425     ok(SafeArrayGetDim(V_ARRAY(&varbody)) == 1, "got %d, expected one dimension\n", SafeArrayGetDim(V_ARRAY(&varbody)));
3426
3427     bound = -1;
3428     hr = SafeArrayGetLBound(V_ARRAY(&varbody), 1, &bound);
3429     ok(hr == S_OK, "got 0x%08x\n", hr);
3430     ok(bound == 0, "got %d, expected zero bound\n", bound);
3431
3432     hr = SafeArrayAccessData(V_ARRAY(&varbody), &ptr);
3433     ok(hr == S_OK, "got 0x%08x\n", hr);
3434     ok(memcmp(ptr, xmltestbodyA, sizeof(xmltestbodyA)-1) == 0, "got wrond body data\n");
3435     SafeArrayUnaccessData(V_ARRAY(&varbody));
3436
3437     VariantClear(&varbody);
3438
3439     SysFreeString(url);
3440
3441     IDispatch_Release(event);
3442     IXMLHttpRequest_Release(pXMLHttpRequest);
3443     free_bstrs();
3444 }
3445
3446 static void test_IXMLDOMDocument2(void)
3447 {
3448     static const WCHAR emptyW[] = {0};
3449     IXMLDOMDocument2 *doc2, *dtddoc2;
3450     IXMLDOMDocument *doc;
3451     IXMLDOMParseError* err;
3452     IDispatchEx *dispex;
3453     VARIANT_BOOL b;
3454     VARIANT var;
3455     HRESULT r;
3456     LONG ref, res;
3457     BSTR str;
3458
3459     doc = create_document(&IID_IXMLDOMDocument);
3460     if (!doc) return;
3461
3462     dtddoc2 = create_document(&IID_IXMLDOMDocument2);
3463     if (!dtddoc2)
3464     {
3465         IXMLDOMDocument_Release(doc);
3466         return;
3467     }
3468
3469     r = IXMLDOMDocument_QueryInterface( doc, &IID_IXMLDOMDocument2, (void**)&doc2 );
3470     ok( r == S_OK, "ret %08x\n", r );
3471     ok( doc == (IXMLDOMDocument*)doc2, "interfaces differ\n");
3472
3473     ole_expect(IXMLDOMDocument2_get_readyState(doc2, NULL), E_INVALIDARG);
3474     ole_check(IXMLDOMDocument2_get_readyState(doc2, &res));
3475     ok(res == READYSTATE_COMPLETE, "expected READYSTATE_COMPLETE (4), got %i\n", res);
3476
3477     err = NULL;
3478     ole_expect(IXMLDOMDocument2_validate(doc2, NULL), S_FALSE);
3479     ole_expect(IXMLDOMDocument2_validate(doc2, &err), S_FALSE);
3480     ok(err != NULL, "expected a pointer\n");
3481     if (err)
3482     {
3483         res = 0;
3484         ole_check(IXMLDOMParseError_get_errorCode(err, &res));
3485         /* XML_E_NOTWF */
3486         ok(res == E_XML_NOTWF, "got %08x\n", res);
3487         IXMLDOMParseError_Release(err);
3488     }
3489
3490     str = SysAllocString( szComplete4 );
3491     r = IXMLDOMDocument_loadXML( doc2, str, &b );
3492     ok( r == S_OK, "loadXML failed\n");
3493     ok( b == VARIANT_TRUE, "failed to load XML string\n");
3494     SysFreeString( str );
3495
3496     ole_check(IXMLDOMDocument2_get_readyState(doc, &res));
3497     ok(res == READYSTATE_COMPLETE, "expected READYSTATE_COMPLETE (4), got %i\n", res);
3498
3499     err = NULL;
3500     ole_expect(IXMLDOMDocument2_validate(doc2, &err), S_FALSE);
3501     ok(err != NULL, "expected a pointer\n");
3502     if (err)
3503     {
3504         res = 0;
3505         ole_check(IXMLDOMParseError_get_errorCode(err, &res));
3506         /* XML_E_NODTD */
3507         ok(res == E_XML_NODTD, "got %08x\n", res);
3508         IXMLDOMParseError_Release(err);
3509     }
3510
3511     r = IXMLDOMDocument_QueryInterface( doc, &IID_IDispatchEx, (void**)&dispex );
3512     ok( r == S_OK, "ret %08x\n", r );
3513     if(r == S_OK)
3514     {
3515         IDispatchEx_Release(dispex);
3516     }
3517
3518     /* we will check if the variant got cleared */
3519     ref = IXMLDOMDocument2_AddRef(doc2);
3520     expect_eq(ref, 3, int, "%d");  /* doc, doc2, AddRef*/
3521     V_VT(&var) = VT_UNKNOWN;
3522     V_UNKNOWN(&var) = (IUnknown *)doc2;
3523
3524     /* invalid calls */
3525     ole_expect(IXMLDOMDocument2_getProperty(doc2, _bstr_("askldhfaklsdf"), &var), E_FAIL);
3526     expect_eq(V_VT(&var), VT_UNKNOWN, int, "%x");
3527     ole_expect(IXMLDOMDocument2_getProperty(doc2, _bstr_("SelectionLanguage"), NULL), E_INVALIDARG);
3528
3529     /* valid call */
3530     ole_check(IXMLDOMDocument2_getProperty(doc2, _bstr_("SelectionLanguage"), &var));
3531     expect_eq(V_VT(&var), VT_BSTR, int, "%x");
3532     expect_bstr_eq_and_free(V_BSTR(&var), "XSLPattern");
3533     V_VT(&var) = VT_R4;
3534
3535     /* the variant didn't get cleared*/
3536     expect_eq(IXMLDOMDocument2_Release(doc2), 2, int, "%d");
3537
3538     /* setProperty tests */
3539     ole_expect(IXMLDOMDocument2_setProperty(doc2, _bstr_("askldhfaklsdf"), var), E_FAIL);
3540     ole_expect(IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionLanguage"), var), E_FAIL);
3541     ole_expect(IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionLanguage"), _variantbstr_("alskjdh faklsjd hfk")), E_FAIL);
3542     ole_check(IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionLanguage"), _variantbstr_("XSLPattern")));
3543     ole_check(IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionLanguage"), _variantbstr_("XPath")));
3544     ole_check(IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionLanguage"), _variantbstr_("XSLPattern")));
3545
3546     V_VT(&var) = VT_BSTR;
3547     V_BSTR(&var) = SysAllocString(emptyW);
3548     r = IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionNamespaces"), var);
3549     ok(r == S_OK, "got 0x%08x\n", r);
3550     VariantClear(&var);
3551
3552     V_VT(&var) = VT_I2;
3553     V_I2(&var) = 0;
3554     r = IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionNamespaces"), var);
3555     ok(r == E_FAIL, "got 0x%08x\n", r);
3556
3557     /* contrary to what MSDN claims you can switch back from XPath to XSLPattern */
3558     ole_check(IXMLDOMDocument2_getProperty(doc2, _bstr_("SelectionLanguage"), &var));
3559     expect_eq(V_VT(&var), VT_BSTR, int, "%x");
3560     expect_bstr_eq_and_free(V_BSTR(&var), "XSLPattern");
3561
3562     IXMLDOMDocument2_Release( doc2 );
3563     IXMLDOMDocument_Release( doc );
3564
3565     /* DTD validation */
3566     ole_check(IXMLDOMDocument2_put_validateOnParse(dtddoc2, VARIANT_FALSE));
3567     ole_check(IXMLDOMDocument2_loadXML(dtddoc2, _bstr_(szEmailXML), &b));
3568     ok( b == VARIANT_TRUE, "failed to load XML string\n");
3569     err = NULL;
3570     ole_check(IXMLDOMDocument2_validate(dtddoc2, &err));
3571     ok(err != NULL, "expected pointer\n");
3572     if (err)
3573     {
3574         res = 0;
3575         ole_expect(IXMLDOMParseError_get_errorCode(err, &res), S_FALSE);
3576         ok(res == 0, "got %08x\n", res);
3577         IXMLDOMParseError_Release(err);
3578     }
3579
3580     ole_check(IXMLDOMDocument2_loadXML(dtddoc2, _bstr_(szEmailXML_0D), &b));
3581     ok( b == VARIANT_TRUE, "failed to load XML string\n");
3582     err = NULL;
3583     ole_expect(IXMLDOMDocument2_validate(dtddoc2, &err), S_FALSE);
3584     ok(err != NULL, "expected pointer\n");
3585     if (err)
3586     {
3587         res = 0;
3588         ole_check(IXMLDOMParseError_get_errorCode(err, &res));
3589         /* XML_ELEMENT_UNDECLARED */
3590         todo_wine ok(res == 0xC00CE00D, "got %08x\n", res);
3591         IXMLDOMParseError_Release(err);
3592     }
3593
3594     ole_check(IXMLDOMDocument2_loadXML(dtddoc2, _bstr_(szEmailXML_0E), &b));
3595     ok( b == VARIANT_TRUE, "failed to load XML string\n");
3596     err = NULL;
3597     ole_expect(IXMLDOMDocument2_validate(dtddoc2, &err), S_FALSE);
3598     ok(err != NULL, "expected pointer\n");
3599     if (err)
3600     {
3601         res = 0;
3602         ole_check(IXMLDOMParseError_get_errorCode(err, &res));
3603         /* XML_ELEMENT_ID_NOT_FOUND */
3604         todo_wine ok(res == 0xC00CE00E, "got %08x\n", res);
3605         IXMLDOMParseError_Release(err);
3606     }
3607
3608     ole_check(IXMLDOMDocument2_loadXML(dtddoc2, _bstr_(szEmailXML_11), &b));
3609     ok( b == VARIANT_TRUE, "failed to load XML string\n");
3610     err = NULL;
3611     ole_expect(IXMLDOMDocument2_validate(dtddoc2, &err), S_FALSE);
3612     ok(err != NULL, "expected pointer\n");
3613     if (err)
3614     {
3615         res = 0;
3616         ole_check(IXMLDOMParseError_get_errorCode(err, &res));
3617         /* XML_EMPTY_NOT_ALLOWED */
3618         todo_wine ok(res == 0xC00CE011, "got %08x\n", res);
3619         IXMLDOMParseError_Release(err);
3620     }
3621
3622     ole_check(IXMLDOMDocument2_loadXML(dtddoc2, _bstr_(szEmailXML_13), &b));
3623     ok( b == VARIANT_TRUE, "failed to load XML string\n");
3624     err = NULL;
3625     ole_expect(IXMLDOMDocument2_validate(dtddoc2, &err), S_FALSE);
3626     ok(err != NULL, "expected pointer\n");
3627     if (err)
3628     {
3629         res = 0;
3630         ole_check(IXMLDOMParseError_get_errorCode(err, &res));
3631         /* XML_ROOT_NAME_MISMATCH */
3632         todo_wine ok(res == 0xC00CE013, "got %08x\n", res);
3633         IXMLDOMParseError_Release(err);
3634     }
3635
3636     ole_check(IXMLDOMDocument2_loadXML(dtddoc2, _bstr_(szEmailXML_14), &b));
3637     ok( b == VARIANT_TRUE, "failed to load XML string\n");
3638     err = NULL;
3639     ole_expect(IXMLDOMDocument2_validate(dtddoc2, &err), S_FALSE);
3640     ok(err != NULL, "expected pointer\n");
3641     if (err)
3642     {
3643         res = 0;
3644         ole_check(IXMLDOMParseError_get_errorCode(err, &res));
3645         /* XML_INVALID_CONTENT */
3646         todo_wine ok(res == 0xC00CE014, "got %08x\n", res);
3647         IXMLDOMParseError_Release(err);
3648     }
3649
3650     ole_check(IXMLDOMDocument2_loadXML(dtddoc2, _bstr_(szEmailXML_15), &b));
3651     ok( b == VARIANT_TRUE, "failed to load XML string\n");
3652     err = NULL;
3653     ole_expect(IXMLDOMDocument2_validate(dtddoc2, &err), S_FALSE);
3654     ok(err != NULL, "expected pointer\n");
3655     if (err)
3656     {
3657         res = 0;
3658         ole_check(IXMLDOMParseError_get_errorCode(err, &res));
3659         /* XML_ATTRIBUTE_NOT_DEFINED */
3660         todo_wine ok(res == 0xC00CE015, "got %08x\n", res);
3661         IXMLDOMParseError_Release(err);
3662     }
3663
3664     ole_check(IXMLDOMDocument2_loadXML(dtddoc2, _bstr_(szEmailXML_16), &b));
3665     ok( b == VARIANT_TRUE, "failed to load XML string\n");
3666     err = NULL;
3667     ole_expect(IXMLDOMDocument2_validate(dtddoc2, &err), S_FALSE);
3668     ok(err != NULL, "expected pointer\n");
3669     if (err)
3670     {
3671         res = 0;
3672         ole_check(IXMLDOMParseError_get_errorCode(err, &res));
3673         /* XML_ATTRIBUTE_FIXED */
3674         todo_wine ok(res == 0xC00CE016, "got %08x\n", res);
3675         IXMLDOMParseError_Release(err);
3676     }
3677
3678     ole_check(IXMLDOMDocument2_loadXML(dtddoc2, _bstr_(szEmailXML_17), &b));
3679     ok( b == VARIANT_TRUE, "failed to load XML string\n");
3680     err = NULL;
3681     ole_expect(IXMLDOMDocument2_validate(dtddoc2, &err), S_FALSE);
3682     ok(err != NULL, "expected pointer\n");
3683     if (err)
3684     {
3685         res = 0;
3686         ole_check(IXMLDOMParseError_get_errorCode(err, &res));
3687         /* XML_ATTRIBUTE_VALUE */
3688         todo_wine ok(res == 0xC00CE017, "got %08x\n", res);
3689         IXMLDOMParseError_Release(err);
3690     }
3691
3692     ole_check(IXMLDOMDocument2_loadXML(dtddoc2, _bstr_(szEmailXML_18), &b));
3693     ok( b == VARIANT_TRUE, "failed to load XML string\n");
3694     err = NULL;
3695     ole_expect(IXMLDOMDocument2_validate(dtddoc2, &err), S_FALSE);
3696     ok(err != NULL, "expected pointer\n");
3697     if (err)
3698     {
3699         res = 0;
3700         ole_check(IXMLDOMParseError_get_errorCode(err, &res));
3701         /* XML_ILLEGAL_TEXT */
3702         todo_wine ok(res == 0xC00CE018, "got %08x\n", res);
3703         IXMLDOMParseError_Release(err);
3704     }
3705
3706     ole_check(IXMLDOMDocument2_loadXML(dtddoc2, _bstr_(szEmailXML_20), &b));
3707     ok( b == VARIANT_TRUE, "failed to load XML string\n");
3708     err = NULL;
3709     ole_expect(IXMLDOMDocument2_validate(dtddoc2, &err), S_FALSE);
3710     ok(err != NULL, "expected pointer\n");
3711     if (err)
3712     {
3713         res = 0;
3714         ole_check(IXMLDOMParseError_get_errorCode(err, &res));
3715         /* XML_REQUIRED_ATTRIBUTE_MISSING */
3716         todo_wine ok(res == 0xC00CE020, "got %08x\n", res);
3717         IXMLDOMParseError_Release(err);
3718     }
3719
3720     IXMLDOMDocument2_Release( dtddoc2 );
3721     free_bstrs();
3722 }
3723
3724 #define helper_ole_check(expr) { \
3725     HRESULT r = expr; \
3726     ok_(__FILE__, line)(r == S_OK, "=> %i: " #expr " returned %08x\n", __LINE__, r); \
3727 }
3728
3729 #define helper_expect_list_and_release(list, expstr) { \
3730     char *str = list_to_string(list); \
3731     ok_(__FILE__, line)(strcmp(str, expstr)==0, "=> %i: Invalid node list: %s, expected %s\n", __LINE__, str, expstr); \
3732     if (list) IXMLDOMNodeList_Release(list); \
3733 }
3734
3735 #define helper_expect_bstr_and_release(bstr, str) { \
3736     ok_(__FILE__, line)(lstrcmpW(bstr, _bstr_(str)) == 0, \
3737        "=> %i: got %s\n", __LINE__, wine_dbgstr_w(bstr)); \
3738     SysFreeString(bstr); \
3739 }
3740
3741 #define check_ws_ignored(doc, str) _check_ws_ignored(__LINE__, doc, str)
3742 static inline void _check_ws_ignored(int line, IXMLDOMDocument2* doc, char const* str)
3743 {
3744     IXMLDOMNode *node1, *node2;
3745     IXMLDOMNodeList *list;
3746     BSTR bstr;
3747
3748     helper_ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//*[local-name()='html']"), &list));
3749     helper_ole_check(IXMLDOMNodeList_get_item(list, 0, &node1));
3750     helper_ole_check(IXMLDOMNodeList_get_item(list, 1, &node2));
3751     helper_ole_check(IXMLDOMNodeList_reset(list));
3752     helper_expect_list_and_release(list, "E1.E4.E1.E2.D1 E2.E4.E1.E2.D1");
3753
3754     helper_ole_check(IXMLDOMNode_get_childNodes(node1, &list));
3755     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");
3756     helper_ole_check(IXMLDOMNode_get_text(node1, &bstr));
3757     if (str)
3758     {
3759         helper_expect_bstr_and_release(bstr, str);
3760     }
3761     else
3762     {
3763         helper_expect_bstr_and_release(bstr, "This is a description.");
3764     }
3765     IXMLDOMNode_Release(node1);
3766
3767     helper_ole_check(IXMLDOMNode_get_childNodes(node2, &list));
3768     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");
3769     helper_ole_check(IXMLDOMNode_get_text(node2, &bstr));
3770     helper_expect_bstr_and_release(bstr, "\n                This is a description with preserved whitespace. \n            ");
3771     IXMLDOMNode_Release(node2);
3772 }
3773
3774 #define check_ws_preserved(doc, str) _check_ws_preserved(__LINE__, doc, str)
3775 static inline void _check_ws_preserved(int line, IXMLDOMDocument2* doc, char const* str)
3776 {
3777     IXMLDOMNode *node1, *node2;
3778     IXMLDOMNodeList *list;
3779     BSTR bstr;
3780
3781     helper_ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//*[local-name()='html']"), &list));
3782     helper_ole_check(IXMLDOMNodeList_get_item(list, 0, &node1));
3783     helper_ole_check(IXMLDOMNodeList_get_item(list, 1, &node2));
3784     helper_ole_check(IXMLDOMNodeList_reset(list));
3785     helper_expect_list_and_release(list, "E2.E8.E2.E2.D1 E4.E8.E2.E2.D1");
3786
3787     helper_ole_check(IXMLDOMNode_get_childNodes(node1, &list));
3788     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");
3789     helper_ole_check(IXMLDOMNode_get_text(node1, &bstr));
3790     if (str)
3791     {
3792         helper_expect_bstr_and_release(bstr, str);
3793     }
3794     else
3795     {
3796         helper_expect_bstr_and_release(bstr, "\n                This is a description. \n            ");
3797     }
3798     IXMLDOMNode_Release(node1);
3799
3800     helper_ole_check(IXMLDOMNode_get_childNodes(node2, &list));
3801     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");
3802     helper_ole_check(IXMLDOMNode_get_text(node2, &bstr));
3803     helper_expect_bstr_and_release(bstr, "\n                This is a description with preserved whitespace. \n            ");
3804     IXMLDOMNode_Release(node2);
3805 }
3806
3807 static void test_whitespace(void)
3808 {
3809     VARIANT_BOOL b;
3810     IXMLDOMDocument2 *doc1, *doc2, *doc3, *doc4;
3811
3812     doc1 = create_document(&IID_IXMLDOMDocument2);
3813     doc2 = create_document(&IID_IXMLDOMDocument2);
3814     if (!doc1 || !doc2) return;
3815
3816     ole_check(IXMLDOMDocument2_put_preserveWhiteSpace(doc2, VARIANT_TRUE));
3817     ole_check(IXMLDOMDocument2_get_preserveWhiteSpace(doc1, &b));
3818     ok(b == VARIANT_FALSE, "expected false\n");
3819     ole_check(IXMLDOMDocument2_get_preserveWhiteSpace(doc2, &b));
3820     ok(b == VARIANT_TRUE, "expected true\n");
3821
3822     ole_check(IXMLDOMDocument2_loadXML(doc1, _bstr_(szExampleXML), &b));
3823     ok(b == VARIANT_TRUE, "failed to load XML string\n");
3824     ole_check(IXMLDOMDocument2_loadXML(doc2, _bstr_(szExampleXML), &b));
3825     ok(b == VARIANT_TRUE, "failed to load XML string\n");
3826
3827     /* switch to XPath */
3828     ole_check(IXMLDOMDocument2_setProperty(doc1, _bstr_("SelectionLanguage"), _variantbstr_("XPath")));
3829     ole_check(IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionLanguage"), _variantbstr_("XPath")));
3830
3831     check_ws_ignored(doc1, NULL);
3832     check_ws_preserved(doc2, NULL);
3833
3834     /* new instances copy the property */
3835     ole_check(IXMLDOMDocument2_QueryInterface(doc1, &IID_IXMLDOMDocument2, (void**) &doc3));
3836     ole_check(IXMLDOMDocument2_QueryInterface(doc2, &IID_IXMLDOMDocument2, (void**) &doc4));
3837
3838     ole_check(IXMLDOMDocument2_get_preserveWhiteSpace(doc3, &b));
3839     ok(b == VARIANT_FALSE, "expected false\n");
3840     ole_check(IXMLDOMDocument2_get_preserveWhiteSpace(doc4, &b));
3841     ok(b == VARIANT_TRUE, "expected true\n");
3842
3843     check_ws_ignored(doc3, NULL);
3844     check_ws_preserved(doc4, NULL);
3845
3846     /* setting after loading xml affects trimming of leading/trailing ws only */
3847     ole_check(IXMLDOMDocument2_put_preserveWhiteSpace(doc1, VARIANT_TRUE));
3848     ole_check(IXMLDOMDocument2_put_preserveWhiteSpace(doc2, VARIANT_FALSE));
3849
3850     /* the trailing "\n            " isn't there, because it was ws-only node */
3851     check_ws_ignored(doc1, "\n                This is a description. ");
3852     check_ws_preserved(doc2, "This is a description.");
3853
3854     /* it takes effect on reload */
3855     ole_check(IXMLDOMDocument2_get_preserveWhiteSpace(doc1, &b));
3856     ok(b == VARIANT_TRUE, "expected true\n");
3857     ole_check(IXMLDOMDocument2_get_preserveWhiteSpace(doc2, &b));
3858     ok(b == VARIANT_FALSE, "expected false\n");
3859
3860     ole_check(IXMLDOMDocument2_loadXML(doc1, _bstr_(szExampleXML), &b));
3861     ok(b == VARIANT_TRUE, "failed to load XML string\n");
3862     ole_check(IXMLDOMDocument2_loadXML(doc2, _bstr_(szExampleXML), &b));
3863     ok(b == VARIANT_TRUE, "failed to load XML string\n");
3864
3865     check_ws_preserved(doc1, NULL);
3866     check_ws_ignored(doc2, NULL);
3867
3868     /* other instances follow suit */
3869     ole_check(IXMLDOMDocument2_get_preserveWhiteSpace(doc3, &b));
3870     ok(b == VARIANT_TRUE, "expected true\n");
3871     ole_check(IXMLDOMDocument2_get_preserveWhiteSpace(doc4, &b));
3872     ok(b == VARIANT_FALSE, "expected false\n");
3873
3874     check_ws_preserved(doc3, NULL);
3875     check_ws_ignored(doc4, NULL);
3876
3877     IXMLDOMDocument_Release(doc1);
3878     IXMLDOMDocument_Release(doc2);
3879     IXMLDOMDocument_Release(doc3);
3880     IXMLDOMDocument_Release(doc4);
3881     free_bstrs();
3882 }
3883
3884 static void test_XPath(void)
3885 {
3886     VARIANT var;
3887     VARIANT_BOOL b;
3888     IXMLDOMDocument2 *doc;
3889     IXMLDOMNode *rootNode;
3890     IXMLDOMNode *elem1Node;
3891     IXMLDOMNode *node;
3892     IXMLDOMNodeList *list;
3893
3894     doc = create_document(&IID_IXMLDOMDocument2);
3895     if (!doc) return;
3896
3897     ole_check(IXMLDOMDocument_loadXML(doc, _bstr_(szExampleXML), &b));
3898     ok(b == VARIANT_TRUE, "failed to load XML string\n");
3899
3900     /* switch to XPath */
3901     ole_check(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionLanguage"), _variantbstr_("XPath")));
3902
3903     /* some simple queries*/
3904     ole_check(IXMLDOMDocument_selectNodes(doc, _bstr_("root"), &list));
3905     ole_check(IXMLDOMNodeList_get_item(list, 0, &rootNode));
3906     ole_check(IXMLDOMNodeList_reset(list));
3907     expect_list_and_release(list, "E2.D1");
3908     if (rootNode == NULL)
3909         return;
3910
3911     ole_check(IXMLDOMDocument_selectNodes(doc, _bstr_("root//c"), &list));
3912     expect_list_and_release(list, "E3.E1.E2.D1 E3.E2.E2.D1");
3913
3914     ole_check(IXMLDOMDocument_selectNodes(doc, _bstr_("//c[@type]"), &list));
3915     expect_list_and_release(list, "E3.E2.E2.D1");
3916
3917     ole_check(IXMLDOMNode_selectNodes(rootNode, _bstr_("elem"), &list));
3918     /* using get_item for query results advances the position */
3919     ole_check(IXMLDOMNodeList_get_item(list, 1, &node));
3920     expect_node(node, "E2.E2.D1");
3921     IXMLDOMNode_Release(node);
3922     ole_check(IXMLDOMNodeList_nextNode(list, &node));
3923     expect_node(node, "E4.E2.D1");
3924     IXMLDOMNode_Release(node);
3925     ole_check(IXMLDOMNodeList_reset(list));
3926     expect_list_and_release(list, "E1.E2.D1 E2.E2.D1 E4.E2.D1");
3927
3928     ole_check(IXMLDOMNode_selectNodes(rootNode, _bstr_("."), &list));
3929     expect_list_and_release(list, "E2.D1");
3930
3931     ole_check(IXMLDOMNode_selectNodes(rootNode, _bstr_("elem[3]/preceding-sibling::*"), &list));
3932     ole_check(IXMLDOMNodeList_get_item(list, 0, &elem1Node));
3933     ole_check(IXMLDOMNodeList_reset(list));
3934     expect_list_and_release(list, "E1.E2.D1 E2.E2.D1 E3.E2.D1");
3935
3936     /* select an attribute */
3937     ole_check(IXMLDOMNode_selectNodes(rootNode, _bstr_(".//@type"), &list));
3938     expect_list_and_release(list, "A'type'.E3.E2.E2.D1");
3939
3940     /* would evaluate to a number */
3941     ole_expect(IXMLDOMNode_selectNodes(rootNode, _bstr_("count(*)"), &list), E_FAIL);
3942     /* would evaluate to a boolean */
3943     ole_expect(IXMLDOMNode_selectNodes(rootNode, _bstr_("position()>0"), &list), E_FAIL);
3944     /* would evaluate to a string */
3945     ole_expect(IXMLDOMNode_selectNodes(rootNode, _bstr_("name()"), &list), E_FAIL);
3946
3947     /* no results */
3948     ole_check(IXMLDOMNode_selectNodes(rootNode, _bstr_("c"), &list));
3949     expect_list_and_release(list, "");
3950     ole_check(IXMLDOMDocument_selectNodes(doc, _bstr_("elem//c"), &list));
3951     expect_list_and_release(list, "");
3952     ole_check(IXMLDOMDocument_selectNodes(doc, _bstr_("//elem[4]"), &list));
3953     expect_list_and_release(list, "");
3954     ole_check(IXMLDOMDocument_selectNodes(doc, _bstr_("root//elem[0]"), &list));
3955     expect_list_and_release(list, "");
3956
3957     /* foo undeclared in document node */
3958     ole_expect(IXMLDOMDocument_selectNodes(doc, _bstr_("root//foo:c"), &list), E_FAIL);
3959     /* undeclared in <root> node */
3960     ole_expect(IXMLDOMNode_selectNodes(rootNode, _bstr_(".//foo:c"), &list), E_FAIL);
3961     /* undeclared in <elem> node */
3962     ole_expect(IXMLDOMNode_selectNodes(elem1Node, _bstr_("//foo:c"), &list), E_FAIL);
3963     /* but this trick can be used */
3964     ole_check(IXMLDOMNode_selectNodes(elem1Node, _bstr_("//*[name()='foo:c']"), &list));
3965     expect_list_and_release(list, "E3.E4.E2.D1");
3966
3967     /* it has to be declared in SelectionNamespaces */
3968     ole_check(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionNamespaces"),
3969         _variantbstr_("xmlns:test='urn:uuid:86B2F87F-ACB6-45cd-8B77-9BDB92A01A29'")));
3970
3971     /* now the namespace can be used */
3972     ole_check(IXMLDOMDocument_selectNodes(doc, _bstr_("root//test:c"), &list));
3973     expect_list_and_release(list, "E3.E3.E2.D1 E3.E4.E2.D1");
3974     ole_check(IXMLDOMNode_selectNodes(rootNode, _bstr_(".//test:c"), &list));
3975     expect_list_and_release(list, "E3.E3.E2.D1 E3.E4.E2.D1");
3976     ole_check(IXMLDOMNode_selectNodes(elem1Node, _bstr_("//test:c"), &list));
3977     expect_list_and_release(list, "E3.E3.E2.D1 E3.E4.E2.D1");
3978     ole_check(IXMLDOMNode_selectNodes(elem1Node, _bstr_(".//test:x"), &list));
3979     expect_list_and_release(list, "E5.E1.E4.E1.E2.D1 E6.E2.E4.E1.E2.D1");
3980
3981     /* SelectionNamespaces syntax error - the namespaces doesn't work anymore but the value is stored */
3982     ole_expect(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionNamespaces"),
3983         _variantbstr_("xmlns:test='urn:uuid:86B2F87F-ACB6-45cd-8B77-9BDB92A01A29' xmlns:foo=###")), E_FAIL);
3984
3985     ole_expect(IXMLDOMDocument_selectNodes(doc, _bstr_("root//foo:c"), &list), E_FAIL);
3986
3987     VariantInit(&var);
3988     ole_check(IXMLDOMDocument2_getProperty(doc, _bstr_("SelectionNamespaces"), &var));
3989     expect_eq(V_VT(&var), VT_BSTR, int, "%x");
3990     if (V_VT(&var) == VT_BSTR)
3991         expect_bstr_eq_and_free(V_BSTR(&var), "xmlns:test='urn:uuid:86B2F87F-ACB6-45cd-8B77-9BDB92A01A29' xmlns:foo=###");
3992
3993     /* extra attributes - same thing*/
3994     ole_expect(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionNamespaces"),
3995         _variantbstr_("xmlns:test='urn:uuid:86B2F87F-ACB6-45cd-8B77-9BDB92A01A29' param='test'")), E_FAIL);
3996     ole_expect(IXMLDOMDocument_selectNodes(doc, _bstr_("root//foo:c"), &list), E_FAIL);
3997
3998     IXMLDOMNode_Release(rootNode);
3999     IXMLDOMNode_Release(elem1Node);
4000     IXMLDOMDocument_Release(doc);
4001     free_bstrs();
4002 }
4003
4004 static void test_cloneNode(void )
4005 {
4006     IXMLDOMDocument *doc;
4007     VARIANT_BOOL b;
4008     IXMLDOMNodeList *pList;
4009     IXMLDOMNamedNodeMap *mapAttr;
4010     LONG nLength = 0, nLength1 = 0;
4011     LONG nAttrCnt = 0, nAttrCnt1 = 0;
4012     IXMLDOMNode *node;
4013     IXMLDOMNode *node_clone;
4014     IXMLDOMNode *node_first;
4015     HRESULT r;
4016     BSTR str;
4017     static const WCHAR szSearch[] = { 'l', 'c', '/', 'p', 'r', 0 };
4018
4019     doc = create_document(&IID_IXMLDOMDocument);
4020     if (!doc) return;
4021
4022     str = SysAllocString( szComplete4 );
4023     ole_check(IXMLDOMDocument_loadXML(doc, str, &b));
4024     ok(b == VARIANT_TRUE, "failed to load XML string\n");
4025     SysFreeString(str);
4026
4027     if(!b)
4028         return;
4029
4030     str = SysAllocString( szSearch);
4031     r = IXMLDOMNode_selectSingleNode(doc, str, &node);
4032     ok( r == S_OK, "ret %08x\n", r );
4033     ok( node != NULL, "node %p\n", node );
4034     SysFreeString(str);
4035
4036     if(!node)
4037     {
4038         IXMLDOMDocument_Release(doc);
4039         return;
4040     }
4041
4042     /* Check invalid parameter */
4043     r = IXMLDOMNode_cloneNode(node, VARIANT_TRUE, NULL);
4044     ok( r == E_INVALIDARG, "ret %08x\n", r );
4045
4046     /* All Children */
4047     r = IXMLDOMNode_cloneNode(node, VARIANT_TRUE, &node_clone);
4048     ok( r == S_OK, "ret %08x\n", r );
4049     ok( node_clone != NULL, "node %p\n", node );
4050
4051     if(!node_clone)
4052     {
4053         IXMLDOMDocument_Release(doc);
4054         IXMLDOMNode_Release(node);
4055         return;
4056     }
4057
4058     r = IXMLDOMNode_get_firstChild(node_clone, &node_first);
4059     ok( r == S_OK, "ret %08x\n", r );
4060     if(r == S_OK)
4061     {
4062         IXMLDOMDocument *doc2;
4063
4064         r = IXMLDOMNode_get_ownerDocument(node_clone, &doc2);
4065         ok( r == S_OK, "ret %08x\n", r );
4066         if(r == S_OK)
4067             IXMLDOMDocument_Release(doc2);
4068
4069         IXMLDOMNode_Release(node_first);
4070     }
4071
4072     r = IXMLDOMNode_get_childNodes(node, &pList);
4073     ok( r == S_OK, "ret %08x\n", r );
4074     if (pList)
4075         {
4076                 IXMLDOMNodeList_get_length(pList, &nLength);
4077                 IXMLDOMNodeList_Release(pList);
4078         }
4079
4080     r = IXMLDOMNode_get_attributes(node, &mapAttr);
4081     ok( r == S_OK, "ret %08x\n", r );
4082     if(mapAttr)
4083     {
4084         IXMLDOMNamedNodeMap_get_length(mapAttr, &nAttrCnt);
4085         IXMLDOMNamedNodeMap_Release(mapAttr);
4086     }
4087
4088     r = IXMLDOMNode_get_childNodes(node_clone, &pList);
4089     ok( r == S_OK, "ret %08x\n", r );
4090     if (pList)
4091         {
4092                 IXMLDOMNodeList_get_length(pList, &nLength1);
4093                 IXMLDOMNodeList_Release(pList);
4094         }
4095
4096     r = IXMLDOMNode_get_attributes(node_clone, &mapAttr);
4097     ok( r == S_OK, "ret %08x\n", r );
4098     if(mapAttr)
4099     {
4100         IXMLDOMNamedNodeMap_get_length(mapAttr, &nAttrCnt1);
4101         IXMLDOMNamedNodeMap_Release(mapAttr);
4102     }
4103
4104     ok(nLength == nLength1, "wrong Child count (%d, %d)\n", nLength, nLength1);
4105     ok(nAttrCnt == nAttrCnt1, "wrong Attribute count (%d, %d)\n", nAttrCnt, nAttrCnt1);
4106     IXMLDOMNode_Release(node_clone);
4107
4108     /* No Children */
4109     r = IXMLDOMNode_cloneNode(node, VARIANT_FALSE, &node_clone);
4110     ok( r == S_OK, "ret %08x\n", r );
4111     ok( node_clone != NULL, "node %p\n", node );
4112
4113     if(!node_clone)
4114     {
4115         IXMLDOMDocument_Release(doc);
4116         IXMLDOMNode_Release(node);
4117         return;
4118     }
4119
4120     r = IXMLDOMNode_get_firstChild(node_clone, &node_first);
4121     ok( r == S_FALSE, "ret %08x\n", r );
4122     if(r == S_OK)
4123     {
4124         IXMLDOMDocument *doc2;
4125
4126         r = IXMLDOMNode_get_ownerDocument(node_clone, &doc2);
4127         ok( r == S_OK, "ret %08x\n", r );
4128         if(r == S_OK)
4129             IXMLDOMDocument_Release(doc2);
4130
4131         IXMLDOMNode_Release(node_first);
4132     }
4133
4134     r = IXMLDOMNode_get_childNodes(node_clone, &pList);
4135     ok( r == S_OK, "ret %08x\n", r );
4136     if (pList)
4137     {
4138         IXMLDOMNodeList_get_length(pList, &nLength1);
4139         ok( nLength1 == 0, "Length should be 0 (%d)\n", nLength1);
4140         IXMLDOMNodeList_Release(pList);
4141     }
4142
4143     r = IXMLDOMNode_get_attributes(node_clone, &mapAttr);
4144     ok( r == S_OK, "ret %08x\n", r );
4145     if(mapAttr)
4146     {
4147         IXMLDOMNamedNodeMap_get_length(mapAttr, &nAttrCnt1);
4148         ok( nAttrCnt1 == 3, "Attribute count should be 3 (%d)\n", nAttrCnt1);
4149         IXMLDOMNamedNodeMap_Release(mapAttr);
4150     }
4151
4152     ok(nLength != nLength1, "wrong Child count (%d, %d)\n", nLength, nLength1);
4153     ok(nAttrCnt == nAttrCnt1, "wrong Attribute count (%d, %d)\n", nAttrCnt, nAttrCnt1);
4154     IXMLDOMNode_Release(node_clone);
4155
4156
4157     IXMLDOMNode_Release(node);
4158     IXMLDOMDocument_Release(doc);
4159 }
4160
4161 static void test_xmlTypes(void)
4162 {
4163     IXMLDOMDocument *doc;
4164     IXMLDOMElement *pRoot;
4165     HRESULT hr;
4166     IXMLDOMComment *pComment;
4167     IXMLDOMElement *pElement;
4168     IXMLDOMAttribute *pAttribute;
4169     IXMLDOMNamedNodeMap *pAttribs;
4170     IXMLDOMCDATASection *pCDataSec;
4171     IXMLDOMImplementation *pIXMLDOMImplementation = NULL;
4172     IXMLDOMDocumentFragment *pDocFrag = NULL;
4173     IXMLDOMEntityReference *pEntityRef = NULL;
4174     BSTR str;
4175     IXMLDOMNode *pNextChild;
4176     VARIANT v;
4177     LONG len = 0;
4178
4179     doc = create_document(&IID_IXMLDOMDocument);
4180     if (!doc) return;
4181
4182     pNextChild = (void*)0xdeadbeef;
4183     hr = IXMLDOMDocument_get_nextSibling(doc, NULL);
4184     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4185
4186     pNextChild = (void*)0xdeadbeef;
4187     hr = IXMLDOMDocument_get_nextSibling(doc, &pNextChild);
4188     ok(hr == S_FALSE, "ret %08x\n", hr );
4189     ok(pNextChild == NULL, "pDocChild not NULL\n");
4190
4191     /* test previous Sibling */
4192     hr = IXMLDOMDocument_get_previousSibling(doc, NULL);
4193     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4194
4195     pNextChild = (void*)0xdeadbeef;
4196     hr = IXMLDOMDocument_get_previousSibling(doc, &pNextChild);
4197     ok(hr == S_FALSE, "ret %08x\n", hr );
4198     ok(pNextChild == NULL, "pNextChild not NULL\n");
4199
4200     /* test get_attributes */
4201     hr = IXMLDOMDocument_get_attributes( doc, NULL );
4202     ok( hr == E_INVALIDARG, "get_attributes returned wrong code\n");
4203
4204     pAttribs = (void*)0xdeadbeef;
4205     hr = IXMLDOMDocument_get_attributes( doc, &pAttribs);
4206     ok(hr == S_FALSE, "ret %08x\n", hr );
4207     ok( pAttribs == NULL, "pAttribs not NULL\n");
4208
4209     /* test get_dataType */
4210     V_VT(&v) = VT_EMPTY;
4211     hr = IXMLDOMDocument_get_dataType(doc, &v);
4212     ok(hr == S_FALSE, "ret %08x\n", hr );
4213     ok( V_VT(&v) == VT_NULL, "incorrect dataType type\n");
4214     VariantClear(&v);
4215
4216     /* test nodeTypeString */
4217     str = NULL;
4218     hr = IXMLDOMDocument_get_nodeTypeString(doc, &str);
4219     ok(hr == S_OK, "ret %08x\n", hr );
4220     ok( !lstrcmpW( str, _bstr_("document") ), "incorrect nodeTypeString string\n");
4221     SysFreeString(str);
4222
4223     /* test implementation */
4224     hr = IXMLDOMDocument_get_implementation(doc, NULL);
4225     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4226
4227     hr = IXMLDOMDocument_get_implementation(doc, &pIXMLDOMImplementation);
4228     ok(hr == S_OK, "ret %08x\n", hr );
4229     if(hr == S_OK)
4230     {
4231         VARIANT_BOOL hasFeature = VARIANT_TRUE;
4232         BSTR sEmpty = SysAllocStringLen(NULL, 0);
4233
4234         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, NULL, sEmpty, &hasFeature);
4235         ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4236
4237         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, sEmpty, sEmpty, NULL);
4238         ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4239
4240         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, _bstr_("DOM"), sEmpty, &hasFeature);
4241         ok(hr == S_OK, "ret %08x\n", hr );
4242         ok(hasFeature == VARIANT_FALSE, "hasFeature returned false\n");
4243
4244         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, sEmpty, sEmpty, &hasFeature);
4245         ok(hr == S_OK, "ret %08x\n", hr );
4246         ok(hasFeature == VARIANT_FALSE, "hasFeature returned true\n");
4247
4248         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, _bstr_("DOM"), NULL, &hasFeature);
4249         ok(hr == S_OK, "ret %08x\n", hr );
4250         ok(hasFeature == VARIANT_TRUE, "hasFeature returned false\n");
4251
4252         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, _bstr_("DOM"), sEmpty, &hasFeature);
4253         ok(hr == S_OK, "ret %08x\n", hr );
4254         ok(hasFeature == VARIANT_FALSE, "hasFeature returned false\n");
4255
4256         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, _bstr_("DOM"), _bstr_("1.0"), &hasFeature);
4257         ok(hr == S_OK, "ret %08x\n", hr );
4258         ok(hasFeature == VARIANT_TRUE, "hasFeature returned true\n");
4259
4260         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, _bstr_("XML"), _bstr_("1.0"), &hasFeature);
4261         ok(hr == S_OK, "ret %08x\n", hr );
4262         ok(hasFeature == VARIANT_TRUE, "hasFeature returned true\n");
4263
4264         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, _bstr_("MS-DOM"), _bstr_("1.0"), &hasFeature);
4265         ok(hr == S_OK, "ret %08x\n", hr );
4266         ok(hasFeature == VARIANT_TRUE, "hasFeature returned true\n");
4267
4268         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, _bstr_("SSS"), NULL, &hasFeature);
4269         ok(hr == S_OK, "ret %08x\n", hr );
4270         ok(hasFeature == VARIANT_FALSE, "hasFeature returned false\n");
4271
4272         SysFreeString(sEmpty);
4273         IXMLDOMImplementation_Release(pIXMLDOMImplementation);
4274     }
4275
4276     pRoot = (IXMLDOMElement*)0x1;
4277     hr = IXMLDOMDocument_createElement(doc, NULL, &pRoot);
4278     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4279     ok(pRoot == (void*)0x1, "Expect same ptr, got %p\n", pRoot);
4280
4281     pRoot = (IXMLDOMElement*)0x1;
4282     hr = IXMLDOMDocument_createElement(doc, _bstr_(""), &pRoot);
4283     ok(hr == E_FAIL, "ret %08x\n", hr );
4284     ok(pRoot == (void*)0x1, "Expect same ptr, got %p\n", pRoot);
4285
4286     hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing"), &pRoot);
4287     ok(hr == S_OK, "ret %08x\n", hr );
4288     if(hr == S_OK)
4289     {
4290         hr = IXMLDOMDocument_appendChild(doc, (IXMLDOMNode*)pRoot, NULL);
4291         ok(hr == S_OK, "ret %08x\n", hr );
4292         if(hr == S_OK)
4293         {
4294             /* Comment */
4295             str = SysAllocString(szComment);
4296             hr = IXMLDOMDocument_createComment(doc, str, &pComment);
4297             SysFreeString(str);
4298             ok(hr == S_OK, "ret %08x\n", hr );
4299             if(hr == S_OK)
4300             {
4301                 /* test get_attributes */
4302                 hr = IXMLDOMComment_get_attributes( pComment, NULL );
4303                 ok( hr == E_INVALIDARG, "get_attributes returned wrong code\n");
4304
4305                 pAttribs = (IXMLDOMNamedNodeMap*)0x1;
4306                 hr = IXMLDOMComment_get_attributes( pComment, &pAttribs);
4307                 ok(hr == S_FALSE, "ret %08x\n", hr );
4308                 ok( pAttribs == NULL, "pAttribs not NULL\n");
4309
4310                 /* test nodeTypeString */
4311                 hr = IXMLDOMComment_get_nodeTypeString(pComment, &str);
4312                 ok(hr == S_OK, "ret %08x\n", hr );
4313                 ok( !lstrcmpW( str, _bstr_("comment") ), "incorrect nodeTypeString string\n");
4314                 SysFreeString(str);
4315
4316                 hr = IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pComment, NULL);
4317                 ok(hr == S_OK, "ret %08x\n", hr );
4318
4319                 hr = IXMLDOMComment_get_nodeName(pComment, &str);
4320                 ok(hr == S_OK, "ret %08x\n", hr );
4321                 ok( !lstrcmpW( str, szCommentNodeText ), "incorrect comment node Name\n");
4322                 SysFreeString(str);
4323
4324                 hr = IXMLDOMComment_get_xml(pComment, &str);
4325                 ok(hr == S_OK, "ret %08x\n", hr );
4326                 ok( !lstrcmpW( str, szCommentXML ), "incorrect comment xml\n");
4327                 SysFreeString(str);
4328
4329                 hr = IXMLDOMComment_get_dataType(pComment, &v);
4330                 ok(hr == S_FALSE, "ret %08x\n", hr );
4331                 ok( V_VT(&v) == VT_NULL, "incorrect dataType type\n");
4332                 VariantClear(&v);
4333
4334                 /* put data Tests */
4335                 hr = IXMLDOMComment_put_data(pComment, _bstr_("This &is a ; test <>\\"));
4336                 ok(hr == S_OK, "ret %08x\n", hr );
4337
4338                 /* get data Tests */
4339                 hr = IXMLDOMComment_get_data(pComment, &str);
4340                 ok(hr == S_OK, "ret %08x\n", hr );
4341                 ok( !lstrcmpW( str, _bstr_("This &is a ; test <>\\") ), "incorrect get_data string\n");
4342                 SysFreeString(str);
4343
4344                 /* get data Tests */
4345                 hr = IXMLDOMComment_get_nodeValue(pComment, &v);
4346                 ok(hr == S_OK, "ret %08x\n", hr );
4347                 ok( V_VT(&v) == VT_BSTR, "incorrect dataType type\n");
4348                 ok( !lstrcmpW( V_BSTR(&v), _bstr_("This &is a ; test <>\\") ), "incorrect get_nodeValue string\n");
4349                 VariantClear(&v);
4350
4351                 /* Confirm XML text is good */
4352                 hr = IXMLDOMComment_get_xml(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                 /* Confirm we get the put_data Text back */
4358                 hr = IXMLDOMComment_get_text(pComment, &str);
4359                 ok(hr == S_OK, "ret %08x\n", hr );
4360                 ok( !lstrcmpW( str, _bstr_("This &is a ; test <>\\") ), "incorrect xml string\n");
4361                 SysFreeString(str);
4362
4363                 /* test length property */
4364                 hr = IXMLDOMComment_get_length(pComment, &len);
4365                 ok(hr == S_OK, "ret %08x\n", hr );
4366                 ok(len == 21, "expected 21 got %d\n", len);
4367
4368                 /* test substringData */
4369                 hr = IXMLDOMComment_substringData(pComment, 0, 4, NULL);
4370                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4371
4372                 /* test substringData - Invalid offset */
4373                 str = (BSTR)&szElement;
4374                 hr = IXMLDOMComment_substringData(pComment, -1, 4, &str);
4375                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4376                 ok( str == NULL, "incorrect string\n");
4377
4378                 /* test substringData - Invalid offset */
4379                 str = (BSTR)&szElement;
4380                 hr = IXMLDOMComment_substringData(pComment, 30, 0, &str);
4381                 ok(hr == S_FALSE, "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, 0, -1, &str);
4387                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4388                 ok( str == NULL, "incorrect string\n");
4389
4390                 /* test substringData - Invalid size */
4391                 str = (BSTR)&szElement;
4392                 hr = IXMLDOMComment_substringData(pComment, 2, 0, &str);
4393                 ok(hr == S_FALSE, "ret %08x\n", hr );
4394                 ok( str == NULL, "incorrect string\n");
4395
4396                 /* test substringData - Start of string */
4397                 hr = IXMLDOMComment_substringData(pComment, 0, 4, &str);
4398                 ok(hr == S_OK, "ret %08x\n", hr );
4399                 ok( !lstrcmpW( str, _bstr_("This") ), "incorrect substringData string\n");
4400                 SysFreeString(str);
4401
4402                 /* test substringData - Middle of string */
4403                 hr = IXMLDOMComment_substringData(pComment, 13, 4, &str);
4404                 ok(hr == S_OK, "ret %08x\n", hr );
4405                 ok( !lstrcmpW( str, _bstr_("test") ), "incorrect substringData string\n");
4406                 SysFreeString(str);
4407
4408                 /* test substringData - End of string */
4409                 hr = IXMLDOMComment_substringData(pComment, 20, 4, &str);
4410                 ok(hr == S_OK, "ret %08x\n", hr );
4411                 ok( !lstrcmpW( str, _bstr_("\\") ), "incorrect substringData string\n");
4412                 SysFreeString(str);
4413
4414                 /* test appendData */
4415                 hr = IXMLDOMComment_appendData(pComment, NULL);
4416                 ok(hr == S_OK, "ret %08x\n", hr );
4417
4418                 hr = IXMLDOMComment_appendData(pComment, _bstr_(""));
4419                 ok(hr == S_OK, "ret %08x\n", hr );
4420
4421                 hr = IXMLDOMComment_appendData(pComment, _bstr_("Append"));
4422                 ok(hr == S_OK, "ret %08x\n", hr );
4423
4424                 hr = IXMLDOMComment_get_text(pComment, &str);
4425                 ok(hr == S_OK, "ret %08x\n", hr );
4426                 ok( !lstrcmpW( str, _bstr_("This &is a ; test <>\\Append") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4427                 SysFreeString(str);
4428
4429                 /* test insertData */
4430                 str = SysAllocStringLen(NULL, 0);
4431                 hr = IXMLDOMComment_insertData(pComment, -1, str);
4432                 ok(hr == S_OK, "ret %08x\n", hr );
4433
4434                 hr = IXMLDOMComment_insertData(pComment, -1, NULL);
4435                 ok(hr == S_OK, "ret %08x\n", hr );
4436
4437                 hr = IXMLDOMComment_insertData(pComment, 1000, str);
4438                 ok(hr == S_OK, "ret %08x\n", hr );
4439
4440                 hr = IXMLDOMComment_insertData(pComment, 1000, NULL);
4441                 ok(hr == S_OK, "ret %08x\n", hr );
4442
4443                 hr = IXMLDOMComment_insertData(pComment, 0, NULL);
4444                 ok(hr == S_OK, "ret %08x\n", hr );
4445
4446                 hr = IXMLDOMComment_insertData(pComment, 0, str);
4447                 ok(hr == S_OK, "ret %08x\n", hr );
4448                 SysFreeString(str);
4449
4450                 hr = IXMLDOMComment_insertData(pComment, -1, _bstr_("Inserting"));
4451                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4452
4453                 hr = IXMLDOMComment_insertData(pComment, 1000, _bstr_("Inserting"));
4454                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4455
4456                 hr = IXMLDOMComment_insertData(pComment, 0, _bstr_("Begin "));
4457                 ok(hr == S_OK, "ret %08x\n", hr );
4458
4459                 hr = IXMLDOMComment_insertData(pComment, 17, _bstr_("Middle"));
4460                 ok(hr == S_OK, "ret %08x\n", hr );
4461
4462                 hr = IXMLDOMComment_insertData(pComment, 39, _bstr_(" End"));
4463                 ok(hr == S_OK, "ret %08x\n", hr );
4464
4465                 hr = IXMLDOMComment_get_text(pComment, &str);
4466                 ok(hr == S_OK, "ret %08x\n", hr );
4467                 ok( !lstrcmpW( str, _bstr_("Begin This &is a Middle; test <>\\Append End") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4468                 SysFreeString(str);
4469
4470                 /* delete data */
4471                 /* invalid arguments */
4472                 hr = IXMLDOMComment_deleteData(pComment, -1, 1);
4473                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4474
4475                 hr = IXMLDOMComment_deleteData(pComment, 0, 0);
4476                 ok(hr == S_OK, "ret %08x\n", hr );
4477
4478                 hr = IXMLDOMComment_deleteData(pComment, 0, -1);
4479                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4480
4481                 hr = IXMLDOMComment_get_length(pComment, &len);
4482                 ok(hr == S_OK, "ret %08x\n", hr );
4483                 ok(len == 43, "expected 43 got %d\n", len);
4484
4485                 hr = IXMLDOMComment_deleteData(pComment, len, 1);
4486                 ok(hr == S_OK, "ret %08x\n", hr );
4487
4488                 hr = IXMLDOMComment_deleteData(pComment, len+1, 1);
4489                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4490
4491                 /* delete from start */
4492                 hr = IXMLDOMComment_deleteData(pComment, 0, 5);
4493                 ok(hr == S_OK, "ret %08x\n", hr );
4494
4495                 hr = IXMLDOMComment_get_length(pComment, &len);
4496                 ok(hr == S_OK, "ret %08x\n", hr );
4497                 ok(len == 38, "expected 38 got %d\n", len);
4498
4499                 hr = IXMLDOMComment_get_text(pComment, &str);
4500                 ok(hr == S_OK, "ret %08x\n", hr );
4501                 ok( !lstrcmpW( str, _bstr_(" This &is a Middle; test <>\\Append End") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4502                 SysFreeString(str);
4503
4504                 /* delete from end */
4505                 hr = IXMLDOMComment_deleteData(pComment, 35, 3);
4506                 ok(hr == S_OK, "ret %08x\n", hr );
4507
4508                 hr = IXMLDOMComment_get_length(pComment, &len);
4509                 ok(hr == S_OK, "ret %08x\n", hr );
4510                 ok(len == 35, "expected 35 got %d\n", len);
4511
4512                 hr = IXMLDOMComment_get_text(pComment, &str);
4513                 ok(hr == S_OK, "ret %08x\n", hr );
4514                 ok( !lstrcmpW( str, _bstr_(" This &is a Middle; test <>\\Append ") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4515                 SysFreeString(str);
4516
4517                 /* delete from inside */
4518                 hr = IXMLDOMComment_deleteData(pComment, 1, 33);
4519                 ok(hr == S_OK, "ret %08x\n", hr );
4520
4521                 hr = IXMLDOMComment_get_length(pComment, &len);
4522                 ok(hr == S_OK, "ret %08x\n", hr );
4523                 ok(len == 2, "expected 2 got %d\n", len);
4524
4525                 hr = IXMLDOMComment_get_text(pComment, &str);
4526                 ok(hr == S_OK, "ret %08x\n", hr );
4527                 ok( !lstrcmpW( str, _bstr_("  ") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4528                 SysFreeString(str);
4529
4530                 /* delete whole data ... */
4531                 hr = IXMLDOMComment_get_length(pComment, &len);
4532                 ok(hr == S_OK, "ret %08x\n", hr );
4533
4534                 hr = IXMLDOMComment_deleteData(pComment, 0, len);
4535                 ok(hr == S_OK, "ret %08x\n", hr );
4536                 /* ... and try again with empty string */
4537                 hr = IXMLDOMComment_deleteData(pComment, 0, len);
4538                 ok(hr == S_OK, "ret %08x\n", hr );
4539
4540                 /* ::replaceData() */
4541                 V_VT(&v) = VT_BSTR;
4542                 V_BSTR(&v) = SysAllocString(szstr1);
4543                 hr = IXMLDOMComment_put_nodeValue(pComment, v);
4544                 ok(hr == S_OK, "ret %08x\n", hr );
4545                 VariantClear(&v);
4546
4547                 hr = IXMLDOMComment_replaceData(pComment, 6, 0, NULL);
4548                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4549                 hr = IXMLDOMComment_get_text(pComment, &str);
4550                 ok(hr == S_OK, "ret %08x\n", hr );
4551                 ok( !lstrcmpW( str, _bstr_("str1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4552                 SysFreeString(str);
4553
4554                 hr = IXMLDOMComment_replaceData(pComment, 0, 0, NULL);
4555                 ok(hr == S_OK, "ret %08x\n", hr );
4556                 hr = IXMLDOMComment_get_text(pComment, &str);
4557                 ok(hr == S_OK, "ret %08x\n", hr );
4558                 ok( !lstrcmpW( str, _bstr_("str1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4559                 SysFreeString(str);
4560
4561                 /* NULL pointer means delete */
4562                 hr = IXMLDOMComment_replaceData(pComment, 0, 1, NULL);
4563                 ok(hr == S_OK, "ret %08x\n", hr );
4564                 hr = IXMLDOMComment_get_text(pComment, &str);
4565                 ok(hr == S_OK, "ret %08x\n", hr );
4566                 ok( !lstrcmpW( str, _bstr_("tr1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4567                 SysFreeString(str);
4568
4569                 /* empty string means delete */
4570                 hr = IXMLDOMComment_replaceData(pComment, 0, 1, _bstr_(""));
4571                 ok(hr == S_OK, "ret %08x\n", hr );
4572                 hr = IXMLDOMComment_get_text(pComment, &str);
4573                 ok(hr == S_OK, "ret %08x\n", hr );
4574                 ok( !lstrcmpW( str, _bstr_("r1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4575                 SysFreeString(str);
4576
4577                 /* zero count means insert */
4578                 hr = IXMLDOMComment_replaceData(pComment, 0, 0, _bstr_("a"));
4579                 ok(hr == S_OK, "ret %08x\n", hr );
4580                 hr = IXMLDOMComment_get_text(pComment, &str);
4581                 ok(hr == S_OK, "ret %08x\n", hr );
4582                 ok( !lstrcmpW( str, _bstr_("ar1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4583                 SysFreeString(str);
4584
4585                 hr = IXMLDOMComment_replaceData(pComment, 0, 2, NULL);
4586                 ok(hr == S_OK, "ret %08x\n", hr );
4587
4588                 hr = IXMLDOMComment_insertData(pComment, 0, _bstr_("m"));
4589                 ok(hr == S_OK, "ret %08x\n", hr );
4590                 hr = IXMLDOMComment_get_text(pComment, &str);
4591                 ok(hr == S_OK, "ret %08x\n", hr );
4592                 ok( !lstrcmpW( str, _bstr_("m1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4593                 SysFreeString(str);
4594
4595                 /* nonempty string, count greater than its length */
4596                 hr = IXMLDOMComment_replaceData(pComment, 0, 2, _bstr_("a1.2"));
4597                 ok(hr == S_OK, "ret %08x\n", hr );
4598                 hr = IXMLDOMComment_get_text(pComment, &str);
4599                 ok(hr == S_OK, "ret %08x\n", hr );
4600                 ok( !lstrcmpW( str, _bstr_("a1.2") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4601                 SysFreeString(str);
4602
4603                 /* nonempty string, count less than its length */
4604                 hr = IXMLDOMComment_replaceData(pComment, 0, 1, _bstr_("wine"));
4605                 ok(hr == S_OK, "ret %08x\n", hr );
4606                 hr = IXMLDOMComment_get_text(pComment, &str);
4607                 ok(hr == S_OK, "ret %08x\n", hr );
4608                 ok( !lstrcmpW( str, _bstr_("wine1.2") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4609                 SysFreeString(str);
4610
4611                 IXMLDOMComment_Release(pComment);
4612             }
4613
4614             /* Element */
4615             str = SysAllocString(szElement);
4616             hr = IXMLDOMDocument_createElement(doc, str, &pElement);
4617             SysFreeString(str);
4618             ok(hr == S_OK, "ret %08x\n", hr );
4619             if(hr == S_OK)
4620             {
4621                 hr = IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
4622                 ok(hr == S_OK, "ret %08x\n", hr );
4623
4624                 /* test nodeTypeString */
4625                 hr = IXMLDOMDocument_get_nodeTypeString(pElement, &str);
4626                 ok(hr == S_OK, "ret %08x\n", hr );
4627                 ok( !lstrcmpW( str, _bstr_("element") ), "incorrect nodeTypeString string\n");
4628                 SysFreeString(str);
4629
4630                 hr = IXMLDOMElement_get_nodeName(pElement, &str);
4631                 ok(hr == S_OK, "ret %08x\n", hr );
4632                 ok( !lstrcmpW( str, szElement ), "incorrect element node Name\n");
4633                 SysFreeString(str);
4634
4635                 hr = IXMLDOMElement_get_xml(pElement, &str);
4636                 ok(hr == S_OK, "ret %08x\n", hr );
4637                 ok( !lstrcmpW( str, szElementXML ), "incorrect element xml\n");
4638                 SysFreeString(str);
4639
4640                 hr = IXMLDOMElement_get_dataType(pElement, &v);
4641                 ok(hr == S_FALSE, "ret %08x\n", hr );
4642                 ok( V_VT(&v) == VT_NULL, "incorrect dataType type\n");
4643                 VariantClear(&v);
4644
4645                 /* Attribute */
4646                 pAttribute = (IXMLDOMAttribute*)0x1;
4647                 hr = IXMLDOMDocument_createAttribute(doc, NULL, &pAttribute);
4648                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4649                 ok(pAttribute == (void*)0x1, "Expect same ptr, got %p\n", pAttribute);
4650
4651                 pAttribute = (IXMLDOMAttribute*)0x1;
4652                 hr = IXMLDOMDocument_createAttribute(doc, _bstr_(""), &pAttribute);
4653                 ok(hr == E_FAIL, "ret %08x\n", hr );
4654                 ok(pAttribute == (void*)0x1, "Expect same ptr, got %p\n", pAttribute);
4655
4656                 str = SysAllocString(szAttribute);
4657                 hr = IXMLDOMDocument_createAttribute(doc, str, &pAttribute);
4658                 SysFreeString(str);
4659                 ok(hr == S_OK, "ret %08x\n", hr );
4660                 if(hr == S_OK)
4661                 {
4662                     IXMLDOMNode *pNewChild = (IXMLDOMNode *)0x1;
4663
4664                     hr = IXMLDOMAttribute_get_nextSibling(pAttribute, NULL);
4665                     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4666
4667                     pNextChild = (IXMLDOMNode *)0x1;
4668                     hr = IXMLDOMAttribute_get_nextSibling(pAttribute, &pNextChild);
4669                     ok(hr == S_FALSE, "ret %08x\n", hr );
4670                     ok(pNextChild == NULL, "pNextChild not NULL\n");
4671
4672                     /* test Previous Sibling*/
4673                     hr = IXMLDOMAttribute_get_previousSibling(pAttribute, NULL);
4674                     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4675
4676                     pNextChild = (IXMLDOMNode *)0x1;
4677                     hr = IXMLDOMAttribute_get_previousSibling(pAttribute, &pNextChild);
4678                     ok(hr == S_FALSE, "ret %08x\n", hr );
4679                     ok(pNextChild == NULL, "pNextChild not NULL\n");
4680
4681                     /* test get_attributes */
4682                     hr = IXMLDOMAttribute_get_attributes( pAttribute, NULL );
4683                     ok( hr == E_INVALIDARG, "get_attributes returned wrong code\n");
4684
4685                     pAttribs = (IXMLDOMNamedNodeMap*)0x1;
4686                     hr = IXMLDOMAttribute_get_attributes( pAttribute, &pAttribs);
4687                     ok(hr == S_FALSE, "ret %08x\n", hr );
4688                     ok( pAttribs == NULL, "pAttribs not NULL\n");
4689
4690                     hr = IXMLDOMElement_appendChild(pElement, (IXMLDOMNode*)pAttribute, &pNewChild);
4691                     ok(hr == E_FAIL, "ret %08x\n", hr );
4692                     ok(pNewChild == NULL, "pNewChild not NULL\n");
4693
4694                     hr = IXMLDOMElement_get_attributes(pElement, &pAttribs);
4695                     ok(hr == S_OK, "ret %08x\n", hr );
4696                     if ( hr == S_OK )
4697                     {
4698                         hr = IXMLDOMNamedNodeMap_setNamedItem(pAttribs, (IXMLDOMNode*)pAttribute, NULL );
4699                         ok(hr == S_OK, "ret %08x\n", hr );
4700
4701                         IXMLDOMNamedNodeMap_Release(pAttribs);
4702                     }
4703
4704                     hr = IXMLDOMAttribute_get_nodeName(pAttribute, &str);
4705                     ok(hr == S_OK, "ret %08x\n", hr );
4706                     ok( !lstrcmpW( str, szAttribute ), "incorrect attribute node Name\n");
4707                     SysFreeString(str);
4708
4709                     /* test nodeTypeString */
4710                     hr = IXMLDOMAttribute_get_nodeTypeString(pAttribute, &str);
4711                     ok(hr == S_OK, "ret %08x\n", hr );
4712                     ok( !lstrcmpW( str, _bstr_("attribute") ), "incorrect nodeTypeString string\n");
4713                     SysFreeString(str);
4714
4715                     /* test nodeName */
4716                     hr = IXMLDOMAttribute_get_nodeName(pAttribute, &str);
4717                     ok(hr == S_OK, "ret %08x\n", hr );
4718                     ok( !lstrcmpW( str, szAttribute ), "incorrect nodeName string\n");
4719                     SysFreeString(str);
4720
4721                     /* test name property */
4722                     hr = IXMLDOMAttribute_get_name(pAttribute, &str);
4723                     ok(hr == S_OK, "ret %08x\n", hr );
4724                     ok( !lstrcmpW( str, szAttribute ), "incorrect name string\n");
4725                     SysFreeString(str);
4726
4727                     hr = IXMLDOMAttribute_get_xml(pAttribute, &str);
4728                     ok(hr == S_OK, "ret %08x\n", hr );
4729                     ok( !lstrcmpW( str, szAttributeXML ), "incorrect attribute xml\n");
4730                     SysFreeString(str);
4731
4732                     hr = IXMLDOMAttribute_get_dataType(pAttribute, &v);
4733                     ok(hr == S_FALSE, "ret %08x\n", hr );
4734                     ok( V_VT(&v) == VT_NULL, "incorrect dataType type\n");
4735                     VariantClear(&v);
4736
4737                     IXMLDOMAttribute_Release(pAttribute);
4738
4739                     /* Check Element again with the Add Attribute*/
4740                     hr = IXMLDOMElement_get_xml(pElement, &str);
4741                     ok(hr == S_OK, "ret %08x\n", hr );
4742                     ok( !lstrcmpW( str, szElementXML2 ), "incorrect element xml\n");
4743                     SysFreeString(str);
4744                 }
4745
4746                 hr = IXMLDOMElement_put_text(pElement, _bstr_("TestingNode"));
4747                 ok(hr == S_OK, "ret %08x\n", hr );
4748
4749                 hr = IXMLDOMElement_get_xml(pElement, &str);
4750                 ok(hr == S_OK, "ret %08x\n", hr );
4751                 ok( !lstrcmpW( str, szElementXML3 ), "incorrect element xml\n");
4752                 SysFreeString(str);
4753
4754                 /* Test for reversible escaping */
4755                 str = SysAllocString( szStrangeChars );
4756                 hr = IXMLDOMElement_put_text(pElement, str);
4757                 ok(hr == S_OK, "ret %08x\n", hr );
4758                 SysFreeString( str );
4759
4760                 hr = IXMLDOMElement_get_xml(pElement, &str);
4761                 ok(hr == S_OK, "ret %08x\n", hr );
4762                 ok( !lstrcmpW( str, szElementXML4 ), "incorrect element xml\n");
4763                 SysFreeString(str);
4764
4765                 hr = IXMLDOMElement_get_text(pElement, &str);
4766                 ok(hr == S_OK, "ret %08x\n", hr );
4767                 ok( !lstrcmpW( str, szStrangeChars ), "incorrect element text\n");
4768                 SysFreeString(str);
4769
4770                 IXMLDOMElement_Release(pElement);
4771             }
4772
4773             /* CData Section */
4774             str = SysAllocString(szCData);
4775             hr = IXMLDOMDocument_createCDATASection(doc, str, NULL);
4776             ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4777
4778             hr = IXMLDOMDocument_createCDATASection(doc, str, &pCDataSec);
4779             SysFreeString(str);
4780             ok(hr == S_OK, "ret %08x\n", hr );
4781             if(hr == S_OK)
4782             {
4783                 IXMLDOMNode *pNextChild = (IXMLDOMNode *)0x1;
4784                 VARIANT var;
4785
4786                 VariantInit(&var);
4787
4788                 hr = IXMLDOMCDATASection_QueryInterface(pCDataSec, &IID_IXMLDOMElement, (void**)&pElement);
4789                 ok(hr == E_NOINTERFACE, "ret %08x\n", hr);
4790
4791                 hr = IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pCDataSec, NULL);
4792                 ok(hr == S_OK, "ret %08x\n", hr );
4793
4794                 /* get Attribute Tests */
4795                 hr = IXMLDOMCDATASection_get_attributes(pCDataSec, NULL);
4796                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4797
4798                 pAttribs = (IXMLDOMNamedNodeMap*)0x1;
4799                 hr = IXMLDOMCDATASection_get_attributes(pCDataSec, &pAttribs);
4800                 ok(hr == S_FALSE, "ret %08x\n", hr );
4801                 ok(pAttribs == NULL, "pAttribs != NULL\n");
4802
4803                 hr = IXMLDOMCDATASection_get_nodeName(pCDataSec, &str);
4804                 ok(hr == S_OK, "ret %08x\n", hr );
4805                 ok( !lstrcmpW( str, szCDataNodeText ), "incorrect cdata node Name\n");
4806                 SysFreeString(str);
4807
4808                 hr = IXMLDOMCDATASection_get_xml(pCDataSec, &str);
4809                 ok(hr == S_OK, "ret %08x\n", hr );
4810                 ok( !lstrcmpW( str, szCDataXML ), "incorrect cdata xml\n");
4811                 SysFreeString(str);
4812
4813                 /* test lastChild */
4814                 pNextChild = (IXMLDOMNode*)0x1;
4815                 hr = IXMLDOMCDATASection_get_lastChild(pCDataSec, &pNextChild);
4816                 ok(hr == S_FALSE, "ret %08x\n", hr );
4817                 ok(pNextChild == NULL, "pNextChild not NULL\n");
4818
4819                 /* test get_dataType */
4820                 hr = IXMLDOMCDATASection_get_dataType(pCDataSec, NULL);
4821                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4822
4823                 hr = IXMLDOMCDATASection_get_dataType(pCDataSec, &v);
4824                 ok(hr == S_FALSE, "ret %08x\n", hr );
4825                 ok( V_VT(&v) == VT_NULL, "incorrect dataType type\n");
4826                 VariantClear(&v);
4827
4828                 /* test nodeTypeString */
4829                 hr = IXMLDOMCDATASection_get_nodeTypeString(pCDataSec, &str);
4830                 ok(hr == S_OK, "ret %08x\n", hr );
4831                 ok( !lstrcmpW( str, _bstr_("cdatasection") ), "incorrect nodeTypeString string\n");
4832                 SysFreeString(str);
4833
4834                 /* put data Tests */
4835                 hr = IXMLDOMCDATASection_put_data(pCDataSec, _bstr_("This &is a ; test <>\\"));
4836                 ok(hr == S_OK, "ret %08x\n", hr );
4837
4838                 /* Confirm XML text is good */
4839                 hr = IXMLDOMCDATASection_get_xml(pCDataSec, &str);
4840                 ok(hr == S_OK, "ret %08x\n", hr );
4841                 ok( !lstrcmpW( str, _bstr_("<![CDATA[This &is a ; test <>\\]]>") ), "incorrect xml string\n");
4842                 SysFreeString(str);
4843
4844                 /* Confirm we get the put_data Text back */
4845                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
4846                 ok(hr == S_OK, "ret %08x\n", hr );
4847                 ok( !lstrcmpW( str, _bstr_("This &is a ; test <>\\") ), "incorrect text string\n");
4848                 SysFreeString(str);
4849
4850                 /* test length property */
4851                 hr = IXMLDOMCDATASection_get_length(pCDataSec, &len);
4852                 ok(hr == S_OK, "ret %08x\n", hr );
4853                 ok(len == 21, "expected 21 got %d\n", len);
4854
4855                 /* test get nodeValue */
4856                 hr = IXMLDOMCDATASection_get_nodeValue(pCDataSec, &var);
4857                 ok(hr == S_OK, "ret %08x\n", hr );
4858                 ok(V_VT(&var) == VT_BSTR, "got vt %04x\n", V_VT(&var));
4859                 ok( !lstrcmpW( V_BSTR(&var), _bstr_("This &is a ; test <>\\") ), "incorrect text string\n");
4860                 VariantClear(&var);
4861
4862                 /* test get data */
4863                 hr = IXMLDOMCDATASection_get_data(pCDataSec, &str);
4864                 ok(hr == S_OK, "ret %08x\n", hr );
4865                 ok( !lstrcmpW( str, _bstr_("This &is a ; test <>\\") ), "incorrect text string\n");
4866                 SysFreeString(str);
4867
4868                 /* test substringData */
4869                 hr = IXMLDOMCDATASection_substringData(pCDataSec, 0, 4, NULL);
4870                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4871
4872                 /* test substringData - Invalid offset */
4873                 str = (BSTR)&szElement;
4874                 hr = IXMLDOMCDATASection_substringData(pCDataSec, -1, 4, &str);
4875                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4876                 ok( str == NULL, "incorrect string\n");
4877
4878                 /* test substringData - Invalid offset */
4879                 str = (BSTR)&szElement;
4880                 hr = IXMLDOMCDATASection_substringData(pCDataSec, 30, 0, &str);
4881                 ok(hr == S_FALSE, "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, 0, -1, &str);
4887                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4888                 ok( str == NULL, "incorrect string\n");
4889
4890                 /* test substringData - Invalid size */
4891                 str = (BSTR)&szElement;
4892                 hr = IXMLDOMCDATASection_substringData(pCDataSec, 2, 0, &str);
4893                 ok(hr == S_FALSE, "ret %08x\n", hr );
4894                 ok( str == NULL, "incorrect string\n");
4895
4896                 /* test substringData - Start of string */
4897                 hr = IXMLDOMCDATASection_substringData(pCDataSec, 0, 4, &str);
4898                 ok(hr == S_OK, "ret %08x\n", hr );
4899                 ok( !lstrcmpW( str, _bstr_("This") ), "incorrect substringData string\n");
4900                 SysFreeString(str);
4901
4902                 /* test substringData - Middle of string */
4903                 hr = IXMLDOMCDATASection_substringData(pCDataSec, 13, 4, &str);
4904                 ok(hr == S_OK, "ret %08x\n", hr );
4905                 ok( !lstrcmpW( str, _bstr_("test") ), "incorrect substringData string\n");
4906                 SysFreeString(str);
4907
4908                 /* test substringData - End of string */
4909                 hr = IXMLDOMCDATASection_substringData(pCDataSec, 20, 4, &str);
4910                 ok(hr == S_OK, "ret %08x\n", hr );
4911                 ok( !lstrcmpW( str, _bstr_("\\") ), "incorrect substringData string\n");
4912                 SysFreeString(str);
4913
4914                 /* test appendData */
4915                 hr = IXMLDOMCDATASection_appendData(pCDataSec, NULL);
4916                 ok(hr == S_OK, "ret %08x\n", hr );
4917
4918                 hr = IXMLDOMCDATASection_appendData(pCDataSec, _bstr_(""));
4919                 ok(hr == S_OK, "ret %08x\n", hr );
4920
4921                 hr = IXMLDOMCDATASection_appendData(pCDataSec, _bstr_("Append"));
4922                 ok(hr == S_OK, "ret %08x\n", hr );
4923
4924                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
4925                 ok(hr == S_OK, "ret %08x\n", hr );
4926                 ok( !lstrcmpW( str, _bstr_("This &is a ; test <>\\Append") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4927                 SysFreeString(str);
4928
4929                 /* test insertData */
4930                 str = SysAllocStringLen(NULL, 0);
4931                 hr = IXMLDOMCDATASection_insertData(pCDataSec, -1, str);
4932                 ok(hr == S_OK, "ret %08x\n", hr );
4933
4934                 hr = IXMLDOMCDATASection_insertData(pCDataSec, -1, NULL);
4935                 ok(hr == S_OK, "ret %08x\n", hr );
4936
4937                 hr = IXMLDOMCDATASection_insertData(pCDataSec, 1000, str);
4938                 ok(hr == S_OK, "ret %08x\n", hr );
4939
4940                 hr = IXMLDOMCDATASection_insertData(pCDataSec, 1000, NULL);
4941                 ok(hr == S_OK, "ret %08x\n", hr );
4942
4943                 hr = IXMLDOMCDATASection_insertData(pCDataSec, 0, NULL);
4944                 ok(hr == S_OK, "ret %08x\n", hr );
4945
4946                 hr = IXMLDOMCDATASection_insertData(pCDataSec, 0, str);
4947                 ok(hr == S_OK, "ret %08x\n", hr );
4948                 SysFreeString(str);
4949
4950                 hr = IXMLDOMCDATASection_insertData(pCDataSec, -1, _bstr_("Inserting"));
4951                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4952
4953                 hr = IXMLDOMCDATASection_insertData(pCDataSec, 1000, _bstr_("Inserting"));
4954                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4955
4956                 hr = IXMLDOMCDATASection_insertData(pCDataSec, 0, _bstr_("Begin "));
4957                 ok(hr == S_OK, "ret %08x\n", hr );
4958
4959                 hr = IXMLDOMCDATASection_insertData(pCDataSec, 17, _bstr_("Middle"));
4960                 ok(hr == S_OK, "ret %08x\n", hr );
4961
4962                 hr = IXMLDOMCDATASection_insertData(pCDataSec, 39, _bstr_(" End"));
4963                 ok(hr == S_OK, "ret %08x\n", hr );
4964
4965                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
4966                 ok(hr == S_OK, "ret %08x\n", hr );
4967                 ok( !lstrcmpW( str, _bstr_("Begin This &is a Middle; test <>\\Append End") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4968                 SysFreeString(str);
4969
4970                 /* delete data */
4971                 /* invalid arguments */
4972                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, -1, 1);
4973                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4974
4975                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, 0, 0);
4976                 ok(hr == S_OK, "ret %08x\n", hr );
4977
4978                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, 0, -1);
4979                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4980
4981                 hr = IXMLDOMCDATASection_get_length(pCDataSec, &len);
4982                 ok(hr == S_OK, "ret %08x\n", hr );
4983                 ok(len == 43, "expected 43 got %d\n", len);
4984
4985                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, len, 1);
4986                 ok(hr == S_OK, "ret %08x\n", hr );
4987
4988                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, len+1, 1);
4989                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4990
4991                 /* delete from start */
4992                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, 0, 5);
4993                 ok(hr == S_OK, "ret %08x\n", hr );
4994
4995                 hr = IXMLDOMCDATASection_get_length(pCDataSec, &len);
4996                 ok(hr == S_OK, "ret %08x\n", hr );
4997                 ok(len == 38, "expected 38 got %d\n", len);
4998
4999                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
5000                 ok(hr == S_OK, "ret %08x\n", hr );
5001                 ok( !lstrcmpW( str, _bstr_(" This &is a Middle; test <>\\Append End") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
5002                 SysFreeString(str);
5003
5004                 /* delete from end */
5005                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, 35, 3);
5006                 ok(hr == S_OK, "ret %08x\n", hr );
5007
5008                 hr = IXMLDOMCDATASection_get_length(pCDataSec, &len);
5009                 ok(hr == S_OK, "ret %08x\n", hr );
5010                 ok(len == 35, "expected 35 got %d\n", len);
5011
5012                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
5013                 ok(hr == S_OK, "ret %08x\n", hr );
5014                 ok( !lstrcmpW( str, _bstr_(" This &is a Middle; test <>\\Append ") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
5015                 SysFreeString(str);
5016
5017                 /* delete from inside */
5018                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, 1, 33);
5019                 ok(hr == S_OK, "ret %08x\n", hr );
5020
5021                 hr = IXMLDOMCDATASection_get_length(pCDataSec, &len);
5022                 ok(hr == S_OK, "ret %08x\n", hr );
5023                 ok(len == 2, "expected 2 got %d\n", len);
5024
5025                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
5026                 ok(hr == S_OK, "ret %08x\n", hr );
5027                 ok( !lstrcmpW( str, _bstr_("  ") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
5028                 SysFreeString(str);
5029
5030                 /* delete whole data ... */
5031                 hr = IXMLDOMCDATASection_get_length(pCDataSec, &len);
5032                 ok(hr == S_OK, "ret %08x\n", hr );
5033
5034                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, 0, len);
5035                 ok(hr == S_OK, "ret %08x\n", hr );
5036
5037                 /* ... and try again with empty string */
5038                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, 0, len);
5039                 ok(hr == S_OK, "ret %08x\n", hr );
5040
5041                 /* ::replaceData() */
5042                 V_VT(&v) = VT_BSTR;
5043                 V_BSTR(&v) = SysAllocString(szstr1);
5044                 hr = IXMLDOMCDATASection_put_nodeValue(pCDataSec, v);
5045                 ok(hr == S_OK, "ret %08x\n", hr );
5046                 VariantClear(&v);
5047
5048                 hr = IXMLDOMCDATASection_replaceData(pCDataSec, 6, 0, NULL);
5049                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5050                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
5051                 ok(hr == S_OK, "ret %08x\n", hr );
5052                 ok( !lstrcmpW( str, _bstr_("str1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
5053                 SysFreeString(str);
5054
5055                 hr = IXMLDOMCDATASection_replaceData(pCDataSec, 0, 0, NULL);
5056                 ok(hr == S_OK, "ret %08x\n", hr );
5057                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
5058                 ok(hr == S_OK, "ret %08x\n", hr );
5059                 ok( !lstrcmpW( str, _bstr_("str1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
5060                 SysFreeString(str);
5061
5062                 /* NULL pointer means delete */
5063                 hr = IXMLDOMCDATASection_replaceData(pCDataSec, 0, 1, NULL);
5064                 ok(hr == S_OK, "ret %08x\n", hr );
5065                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
5066                 ok(hr == S_OK, "ret %08x\n", hr );
5067                 ok( !lstrcmpW( str, _bstr_("tr1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
5068                 SysFreeString(str);
5069
5070                 /* empty string means delete */
5071                 hr = IXMLDOMCDATASection_replaceData(pCDataSec, 0, 1, _bstr_(""));
5072                 ok(hr == S_OK, "ret %08x\n", hr );
5073                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
5074                 ok(hr == S_OK, "ret %08x\n", hr );
5075                 ok( !lstrcmpW( str, _bstr_("r1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
5076                 SysFreeString(str);
5077
5078                 /* zero count means insert */
5079                 hr = IXMLDOMCDATASection_replaceData(pCDataSec, 0, 0, _bstr_("a"));
5080                 ok(hr == S_OK, "ret %08x\n", hr );
5081                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
5082                 ok(hr == S_OK, "ret %08x\n", hr );
5083                 ok( !lstrcmpW( str, _bstr_("ar1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
5084                 SysFreeString(str);
5085
5086                 hr = IXMLDOMCDATASection_replaceData(pCDataSec, 0, 2, NULL);
5087                 ok(hr == S_OK, "ret %08x\n", hr );
5088
5089                 hr = IXMLDOMCDATASection_insertData(pCDataSec, 0, _bstr_("m"));
5090                 ok(hr == S_OK, "ret %08x\n", hr );
5091                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
5092                 ok(hr == S_OK, "ret %08x\n", hr );
5093                 ok( !lstrcmpW( str, _bstr_("m1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
5094                 SysFreeString(str);
5095
5096                 /* nonempty string, count greater than its length */
5097                 hr = IXMLDOMCDATASection_replaceData(pCDataSec, 0, 2, _bstr_("a1.2"));
5098                 ok(hr == S_OK, "ret %08x\n", hr );
5099                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
5100                 ok(hr == S_OK, "ret %08x\n", hr );
5101                 ok( !lstrcmpW( str, _bstr_("a1.2") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
5102                 SysFreeString(str);
5103
5104                 /* nonempty string, count less than its length */
5105                 hr = IXMLDOMCDATASection_replaceData(pCDataSec, 0, 1, _bstr_("wine"));
5106                 ok(hr == S_OK, "ret %08x\n", hr );
5107                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
5108                 ok(hr == S_OK, "ret %08x\n", hr );
5109                 ok( !lstrcmpW( str, _bstr_("wine1.2") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
5110                 SysFreeString(str);
5111
5112                 IXMLDOMCDATASection_Release(pCDataSec);
5113             }
5114
5115             /* Document Fragments */
5116             hr = IXMLDOMDocument_createDocumentFragment(doc, NULL);
5117             ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5118
5119             hr = IXMLDOMDocument_createDocumentFragment(doc, &pDocFrag);
5120             ok(hr == S_OK, "ret %08x\n", hr );
5121             if(hr == S_OK)
5122             {
5123                 IXMLDOMNode *node;
5124
5125                 hr = IXMLDOMDocumentFragment_get_parentNode(pDocFrag, NULL);
5126                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5127
5128                 node = (IXMLDOMNode *)0x1;
5129                 hr = IXMLDOMDocumentFragment_get_parentNode(pDocFrag, &node);
5130                 ok(hr == S_FALSE, "ret %08x\n", hr );
5131                 ok(node == NULL, "expected NULL, got %p\n", node);
5132
5133                 hr = IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pDocFrag, NULL);
5134                 ok(hr == S_OK, "ret %08x\n", hr );
5135
5136                 /* get Attribute Tests */
5137                 hr = IXMLDOMDocumentFragment_get_attributes(pDocFrag, NULL);
5138                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5139
5140                 pAttribs = (IXMLDOMNamedNodeMap*)0x1;
5141                 hr = IXMLDOMDocumentFragment_get_attributes(pDocFrag, &pAttribs);
5142                 ok(hr == S_FALSE, "ret %08x\n", hr );
5143                 ok(pAttribs == NULL, "pAttribs != NULL\n");
5144
5145                 hr = IXMLDOMDocumentFragment_get_nodeName(pDocFrag, &str);
5146                 ok(hr == S_OK, "ret %08x\n", hr );
5147                 ok( !lstrcmpW( str, szDocFragmentText ), "incorrect docfragment node Name\n");
5148                 SysFreeString(str);
5149
5150                 /* test next Sibling*/
5151                 hr = IXMLDOMDocumentFragment_get_nextSibling(pDocFrag, NULL);
5152                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5153
5154                 node = (IXMLDOMNode *)0x1;
5155                 hr = IXMLDOMDocumentFragment_get_nextSibling(pDocFrag, &node);
5156                 ok(hr == S_FALSE, "ret %08x\n", hr );
5157                 ok(node == NULL, "next sibling not NULL\n");
5158
5159                 /* test Previous Sibling*/
5160                 hr = IXMLDOMDocumentFragment_get_previousSibling(pDocFrag, NULL);
5161                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5162
5163                 node = (IXMLDOMNode *)0x1;
5164                 hr = IXMLDOMDocumentFragment_get_previousSibling(pDocFrag, &node);
5165                 ok(hr == S_FALSE, "ret %08x\n", hr );
5166                 ok(node == NULL, "previous sibling not NULL\n");
5167
5168                 /* test get_dataType */
5169                 hr = IXMLDOMDocumentFragment_get_dataType(pDocFrag, NULL);
5170                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5171
5172                 hr = IXMLDOMDocumentFragment_get_dataType(pDocFrag, &v);
5173                 ok(hr == S_FALSE, "ret %08x\n", hr );
5174                 ok( V_VT(&v) == VT_NULL, "incorrect dataType type\n");
5175                 VariantClear(&v);
5176
5177                 /* test nodeTypeString */
5178                 hr = IXMLDOMDocumentFragment_get_nodeTypeString(pDocFrag, &str);
5179                 ok(hr == S_OK, "ret %08x\n", hr );
5180                 ok( !lstrcmpW( str, _bstr_("documentfragment") ), "incorrect nodeTypeString string\n");
5181                 SysFreeString(str);
5182
5183                 IXMLDOMDocumentFragment_Release(pDocFrag);
5184             }
5185
5186             /* Entity References */
5187             hr = IXMLDOMDocument_createEntityReference(doc, NULL, &pEntityRef);
5188             ok(hr == E_FAIL, "ret %08x\n", hr );
5189             hr = IXMLDOMDocument_createEntityReference(doc, _bstr_(""), &pEntityRef);
5190             ok(hr == E_FAIL, "ret %08x\n", hr );
5191
5192             str = SysAllocString(szEntityRef);
5193             hr = IXMLDOMDocument_createEntityReference(doc, str, NULL);
5194             ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5195
5196             hr = IXMLDOMDocument_createEntityReference(doc, str, &pEntityRef);
5197             SysFreeString(str);
5198             ok(hr == S_OK, "ret %08x\n", hr );
5199             if(hr == S_OK)
5200             {
5201                 hr = IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pEntityRef, NULL);
5202                 ok(hr == S_OK, "ret %08x\n", hr );
5203
5204                 /* get Attribute Tests */
5205                 hr = IXMLDOMEntityReference_get_attributes(pEntityRef, NULL);
5206                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5207
5208                 pAttribs = (IXMLDOMNamedNodeMap*)0x1;
5209                 hr = IXMLDOMEntityReference_get_attributes(pEntityRef, &pAttribs);
5210                 ok(hr == S_FALSE, "ret %08x\n", hr );
5211                 ok(pAttribs == NULL, "pAttribs != NULL\n");
5212
5213                 /* test dataType */
5214                 hr = IXMLDOMEntityReference_get_dataType(pEntityRef, &v);
5215                 ok(hr == S_FALSE, "ret %08x\n", hr );
5216                 ok( V_VT(&v) == VT_NULL, "incorrect dataType type\n");
5217                 VariantClear(&v);
5218
5219                 /* test nodeTypeString */
5220                 hr = IXMLDOMEntityReference_get_nodeTypeString(pEntityRef, &str);
5221                 ok(hr == S_OK, "ret %08x\n", hr );
5222                 ok( !lstrcmpW( str, _bstr_("entityreference") ), "incorrect nodeTypeString string\n");
5223                 SysFreeString(str);
5224
5225                 /* test get_xml*/
5226                 hr = IXMLDOMEntityReference_get_xml(pEntityRef, &str);
5227                 ok(hr == S_OK, "ret %08x\n", hr );
5228                 ok( !lstrcmpW( str, szEntityRefXML ), "incorrect xml string\n");
5229                 SysFreeString(str);
5230
5231                 IXMLDOMEntityReference_Release(pEntityRef);
5232             }
5233
5234             IXMLDOMElement_Release( pRoot );
5235         }
5236     }
5237
5238     IXMLDOMDocument_Release(doc);
5239
5240     free_bstrs();
5241 }
5242
5243 static void test_nodeTypeTests( void )
5244 {
5245     IXMLDOMDocument *doc = NULL;
5246     IXMLDOMElement *pRoot;
5247     IXMLDOMElement *pElement;
5248     HRESULT hr;
5249
5250     doc = create_document(&IID_IXMLDOMDocument);
5251     if (!doc) return;
5252
5253     hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing"), NULL);
5254     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5255
5256     hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing"), &pRoot);
5257     ok(hr == S_OK, "ret %08x\n", hr );
5258     if(hr == S_OK)
5259     {
5260         hr = IXMLDOMDocument_appendChild(doc, (IXMLDOMNode*)pRoot, NULL);
5261         ok(hr == S_OK, "ret %08x\n", hr );
5262         if(hr == S_OK)
5263         {
5264             hr = IXMLDOMElement_put_dataType(pRoot, NULL);
5265             ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5266
5267             /* Invalid Value */
5268             hr = IXMLDOMElement_put_dataType(pRoot, _bstr_("abcdefg") );
5269             ok(hr == E_FAIL, "ret %08x\n", hr );
5270
5271             /* NOTE:
5272              *   The name passed into put_dataType is case-insensitive. So many of the names
5273              *     have been changed to reflect this.
5274              */
5275             /* Boolean */
5276             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_Boolean"), &pElement);
5277             ok(hr == S_OK, "ret %08x\n", hr );
5278             if(hr == S_OK)
5279             {
5280                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5281
5282                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("Boolean") );
5283                 ok(hr == S_OK, "ret %08x\n", hr );
5284
5285                 IXMLDOMElement_Release(pElement);
5286             }
5287
5288             /* String */
5289             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_String"), &pElement);
5290             ok(hr == S_OK, "ret %08x\n", hr );
5291             if(hr == S_OK)
5292             {
5293                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5294
5295                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("String") );
5296                 ok(hr == S_OK, "ret %08x\n", hr );
5297
5298                 IXMLDOMElement_Release(pElement);
5299             }
5300
5301             /* Number */
5302             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_Number"), &pElement);
5303             ok(hr == S_OK, "ret %08x\n", hr );
5304             if(hr == S_OK)
5305             {
5306                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5307
5308                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("number") );
5309                 ok(hr == S_OK, "ret %08x\n", hr );
5310
5311                 IXMLDOMElement_Release(pElement);
5312             }
5313
5314             /* Int */
5315             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_Int"), &pElement);
5316             ok(hr == S_OK, "ret %08x\n", hr );
5317             if(hr == S_OK)
5318             {
5319                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5320
5321                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("InT") );
5322                 ok(hr == S_OK, "ret %08x\n", hr );
5323
5324                 IXMLDOMElement_Release(pElement);
5325             }
5326
5327             /* Fixed */
5328             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_Fixed"), &pElement);
5329             ok(hr == S_OK, "ret %08x\n", hr );
5330             if(hr == S_OK)
5331             {
5332                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5333
5334                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("fixed.14.4") );
5335                 ok(hr == S_OK, "ret %08x\n", hr );
5336
5337                 IXMLDOMElement_Release(pElement);
5338             }
5339
5340             /* DateTime */
5341             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_DateTime"), &pElement);
5342             ok(hr == S_OK, "ret %08x\n", hr );
5343             if(hr == S_OK)
5344             {
5345                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5346
5347                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("DateTime") );
5348                 ok(hr == S_OK, "ret %08x\n", hr );
5349
5350                 IXMLDOMElement_Release(pElement);
5351             }
5352
5353             /* DateTime TZ */
5354             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_DateTime_tz"), &pElement);
5355             ok(hr == S_OK, "ret %08x\n", hr );
5356             if(hr == S_OK)
5357             {
5358                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5359
5360                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("DateTime.tz") );
5361                 ok(hr == S_OK, "ret %08x\n", hr );
5362
5363                 IXMLDOMElement_Release(pElement);
5364             }
5365
5366             /* Date */
5367             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_Date"), &pElement);
5368             ok(hr == S_OK, "ret %08x\n", hr );
5369             if(hr == S_OK)
5370             {
5371                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5372
5373                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("Date") );
5374                 ok(hr == S_OK, "ret %08x\n", hr );
5375
5376                 IXMLDOMElement_Release(pElement);
5377             }
5378
5379             /* Time */
5380             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_Time"), &pElement);
5381             ok(hr == S_OK, "ret %08x\n", hr );
5382             if(hr == S_OK)
5383             {
5384                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5385
5386                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("Time") );
5387                 ok(hr == S_OK, "ret %08x\n", hr );
5388
5389                 IXMLDOMElement_Release(pElement);
5390             }
5391
5392             /* Time.tz */
5393             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_Time_TZ"), &pElement);
5394             ok(hr == S_OK, "ret %08x\n", hr );
5395             if(hr == S_OK)
5396             {
5397                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5398
5399                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("Time.tz") );
5400                 ok(hr == S_OK, "ret %08x\n", hr );
5401
5402                 IXMLDOMElement_Release(pElement);
5403             }
5404
5405             /* I1 */
5406             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_I1"), &pElement);
5407             ok(hr == S_OK, "ret %08x\n", hr );
5408             if(hr == S_OK)
5409             {
5410                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5411
5412                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("I1") );
5413                 ok(hr == S_OK, "ret %08x\n", hr );
5414
5415                 IXMLDOMElement_Release(pElement);
5416             }
5417
5418             /* I2 */
5419             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_I2"), &pElement);
5420             ok(hr == S_OK, "ret %08x\n", hr );
5421             if(hr == S_OK)
5422             {
5423                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5424
5425                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("I2") );
5426                 ok(hr == S_OK, "ret %08x\n", hr );
5427
5428                 IXMLDOMElement_Release(pElement);
5429             }
5430
5431             /* I4 */
5432             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_I4"), &pElement);
5433             ok(hr == S_OK, "ret %08x\n", hr );
5434             if(hr == S_OK)
5435             {
5436                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5437
5438                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("I4") );
5439                 ok(hr == S_OK, "ret %08x\n", hr );
5440
5441                 IXMLDOMElement_Release(pElement);
5442             }
5443
5444             /* UI1 */
5445             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_UI1"), &pElement);
5446             ok(hr == S_OK, "ret %08x\n", hr );
5447             if(hr == S_OK)
5448             {
5449                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5450
5451                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("UI1") );
5452                 ok(hr == S_OK, "ret %08x\n", hr );
5453
5454                 IXMLDOMElement_Release(pElement);
5455             }
5456
5457             /* UI2 */
5458             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_UI2"), &pElement);
5459             ok(hr == S_OK, "ret %08x\n", hr );
5460             if(hr == S_OK)
5461             {
5462                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5463
5464                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("UI2") );
5465                 ok(hr == S_OK, "ret %08x\n", hr );
5466
5467                 IXMLDOMElement_Release(pElement);
5468             }
5469
5470             /* UI4 */
5471             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_UI4"), &pElement);
5472             ok(hr == S_OK, "ret %08x\n", hr );
5473             if(hr == S_OK)
5474             {
5475                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5476
5477                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("UI4") );
5478                 ok(hr == S_OK, "ret %08x\n", hr );
5479
5480                 IXMLDOMElement_Release(pElement);
5481             }
5482
5483             /* r4 */
5484             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_r4"), &pElement);
5485             ok(hr == S_OK, "ret %08x\n", hr );
5486             if(hr == S_OK)
5487             {
5488                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5489
5490                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("r4") );
5491                 ok(hr == S_OK, "ret %08x\n", hr );
5492
5493                 IXMLDOMElement_Release(pElement);
5494             }
5495
5496             /* r8 */
5497             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_r8"), &pElement);
5498             ok(hr == S_OK, "ret %08x\n", hr );
5499             if(hr == S_OK)
5500             {
5501                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5502
5503                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("r8") );
5504                 ok(hr == S_OK, "ret %08x\n", hr );
5505
5506                 IXMLDOMElement_Release(pElement);
5507             }
5508
5509             /* float */
5510             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_float"), &pElement);
5511             ok(hr == S_OK, "ret %08x\n", hr );
5512             if(hr == S_OK)
5513             {
5514                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5515
5516                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("float") );
5517                 ok(hr == S_OK, "ret %08x\n", hr );
5518
5519                 IXMLDOMElement_Release(pElement);
5520             }
5521
5522             /* uuid */
5523             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_uuid"), &pElement);
5524             ok(hr == S_OK, "ret %08x\n", hr );
5525             if(hr == S_OK)
5526             {
5527                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5528
5529                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("UuId") );
5530                 ok(hr == S_OK, "ret %08x\n", hr );
5531
5532                 IXMLDOMElement_Release(pElement);
5533             }
5534
5535             /* bin.hex */
5536             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_bin_hex"), &pElement);
5537             ok(hr == S_OK, "ret %08x\n", hr );
5538             if(hr == S_OK)
5539             {
5540                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5541
5542                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("bin.hex") );
5543                 ok(hr == S_OK, "ret %08x\n", hr );
5544
5545                 IXMLDOMElement_Release(pElement);
5546             }
5547
5548             /* bin.base64 */
5549             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_bin_base64"), &pElement);
5550             ok(hr == S_OK, "ret %08x\n", hr );
5551             if(hr == S_OK)
5552             {
5553                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5554
5555                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("bin.base64") );
5556                 ok(hr == S_OK, "ret %08x\n", hr );
5557
5558                 IXMLDOMElement_Release(pElement);
5559             }
5560
5561             /* Check changing types */
5562             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_Change"), &pElement);
5563             ok(hr == S_OK, "ret %08x\n", hr );
5564             if(hr == S_OK)
5565             {
5566                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5567
5568                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("DateTime.tz") );
5569                 ok(hr == S_OK, "ret %08x\n", hr );
5570
5571                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("string") );
5572                 ok(hr == S_OK, "ret %08x\n", hr );
5573
5574                 IXMLDOMElement_Release(pElement);
5575             }
5576
5577             IXMLDOMElement_Release(pRoot);
5578         }
5579     }
5580
5581     IXMLDOMDocument_Release(doc);
5582
5583     free_bstrs();
5584 }
5585
5586 static void test_DocumentSaveToDocument(void)
5587 {
5588     IXMLDOMDocument *doc, *doc2;
5589     IXMLDOMElement *pRoot;
5590     HRESULT hr;
5591
5592     doc = create_document(&IID_IXMLDOMDocument);
5593     if (!doc) return;
5594
5595     doc2 = create_document(&IID_IXMLDOMDocument);
5596     if (!doc2)
5597     {
5598         IXMLDOMDocument_Release(doc);
5599         return;
5600     }
5601
5602     hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing"), &pRoot);
5603     ok(hr == S_OK, "ret %08x\n", hr );
5604     if(hr == S_OK)
5605     {
5606         hr = IXMLDOMDocument_appendChild(doc, (IXMLDOMNode*)pRoot, NULL);
5607         ok(hr == S_OK, "ret %08x\n", hr );
5608         if(hr == S_OK)
5609         {
5610             VARIANT vDoc;
5611             BSTR sOrig;
5612             BSTR sNew;
5613
5614             V_VT(&vDoc) = VT_UNKNOWN;
5615             V_UNKNOWN(&vDoc) = (IUnknown*)doc2;
5616
5617             hr = IXMLDOMDocument_save(doc, vDoc);
5618             ok(hr == S_OK, "ret %08x\n", hr );
5619
5620             hr = IXMLDOMDocument_get_xml(doc, &sOrig);
5621             ok(hr == S_OK, "ret %08x\n", hr );
5622
5623             hr = IXMLDOMDocument_get_xml(doc2, &sNew);
5624             ok(hr == S_OK, "ret %08x\n", hr );
5625
5626             ok( !lstrcmpW( sOrig, sNew ), "New document is not the same as origial\n");
5627
5628             SysFreeString(sOrig);
5629             SysFreeString(sNew);
5630         }
5631         IXMLDOMElement_Release(pRoot);
5632     }
5633
5634     IXMLDOMDocument_Release(doc2);
5635     IXMLDOMDocument_Release(doc);
5636 }
5637
5638 static void test_DocumentSaveToFile(void)
5639 {
5640     IXMLDOMDocument *doc;
5641     IXMLDOMElement *pRoot;
5642     HANDLE file;
5643     char buffer[100];
5644     DWORD read = 0;
5645     HRESULT hr;
5646
5647     doc = create_document(&IID_IXMLDOMDocument);
5648     if (!doc) return;
5649
5650     hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing"), &pRoot);
5651     ok(hr == S_OK, "ret %08x\n", hr );
5652     if(hr == S_OK)
5653     {
5654         hr = IXMLDOMDocument_appendChild(doc, (IXMLDOMNode*)pRoot, NULL);
5655         ok(hr == S_OK, "ret %08x\n", hr );
5656         if(hr == S_OK)
5657         {
5658             VARIANT vFile;
5659
5660             V_VT(&vFile) = VT_BSTR;
5661             V_BSTR(&vFile) = _bstr_("test.xml");
5662
5663             hr = IXMLDOMDocument_save(doc, vFile);
5664             ok(hr == S_OK, "ret %08x\n", hr );
5665         }
5666     }
5667
5668     IXMLDOMElement_Release(pRoot);
5669     IXMLDOMDocument_Release(doc);
5670
5671     file = CreateFile("test.xml", GENERIC_READ, 0, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL );
5672     ok(file != INVALID_HANDLE_VALUE, "Could not open file: %u\n", GetLastError());
5673     if(file == INVALID_HANDLE_VALUE)
5674         return;
5675
5676     ReadFile(file, buffer, sizeof(buffer), &read, NULL);
5677     ok(read != 0, "could not read file\n");
5678     ok(buffer[0] != '<' || buffer[1] != '?', "File contains processing instruction\n");
5679
5680     CloseHandle(file);
5681     DeleteFile("test.xml");
5682 }
5683
5684 static void test_testTransforms(void)
5685 {
5686     IXMLDOMDocument *doc, *docSS;
5687     IXMLDOMNode *pNode;
5688     VARIANT_BOOL bSucc;
5689
5690     HRESULT hr;
5691
5692     doc = create_document(&IID_IXMLDOMDocument);
5693     if (!doc) return;
5694
5695     docSS = create_document(&IID_IXMLDOMDocument);
5696     if (!docSS)
5697     {
5698         IXMLDOMDocument_Release(doc);
5699         return;
5700     }
5701
5702     hr = IXMLDOMDocument_loadXML(doc, _bstr_(szTransformXML), &bSucc);
5703     ok(hr == S_OK, "ret %08x\n", hr );
5704
5705     hr = IXMLDOMDocument_loadXML(docSS, _bstr_(szTransformSSXML), &bSucc);
5706     ok(hr == S_OK, "ret %08x\n", hr );
5707
5708     hr = IXMLDOMDocument_QueryInterface(docSS, &IID_IXMLDOMNode, (void**)&pNode );
5709     ok(hr == S_OK, "ret %08x\n", hr );
5710     if(hr == S_OK)
5711     {
5712         BSTR bOut;
5713
5714         hr = IXMLDOMDocument_transformNode(doc, pNode, &bOut);
5715         ok(hr == S_OK, "ret %08x\n", hr );
5716         ok( compareIgnoreReturns( bOut, _bstr_(szTransformOutput)), "Stylesheet output not correct\n");
5717         SysFreeString(bOut);
5718
5719         IXMLDOMNode_Release(pNode);
5720     }
5721
5722     IXMLDOMDocument_Release(docSS);
5723     IXMLDOMDocument_Release(doc);
5724
5725     free_bstrs();
5726 }
5727
5728 static void test_Namespaces(void)
5729 {
5730     IXMLDOMDocument *doc;
5731     IXMLDOMNode *pNode;
5732     IXMLDOMNode *pNode2 = NULL;
5733     VARIANT_BOOL bSucc;
5734     HRESULT hr;
5735     BSTR str;
5736     static  const CHAR szNamespacesXML[] =
5737 "<?xml version=\"1.0\"?>\n"
5738 "<root xmlns:WEB='http://www.winehq.org'>\n"
5739 "<WEB:Site version=\"1.0\" />\n"
5740 "</root>";
5741
5742     doc = create_document(&IID_IXMLDOMDocument);
5743     if (!doc) return;
5744
5745     hr = IXMLDOMDocument_loadXML(doc, _bstr_(szNamespacesXML), &bSucc);
5746     ok(hr == S_OK, "ret %08x\n", hr );
5747     ok(bSucc == VARIANT_TRUE, "Expected VARIANT_TRUE got VARIANT_FALSE\n");
5748
5749     hr = IXMLDOMDocument_selectSingleNode(doc, _bstr_("root"), &pNode );
5750     ok(hr == S_OK, "ret %08x\n", hr );
5751     if(hr == S_OK)
5752     {
5753         hr = IXMLDOMNode_get_firstChild( pNode, &pNode2 );
5754         ok( hr == S_OK, "ret %08x\n", hr );
5755         ok( pNode2 != NULL, "pNode2 == NULL\n");
5756
5757         /* Test get_prefix */
5758         hr = IXMLDOMNode_get_prefix(pNode2, NULL);
5759         ok( hr == E_INVALIDARG, "ret %08x\n", hr );
5760         /* NOTE: Need to test that arg2 gets cleared on Error. */
5761
5762         hr = IXMLDOMNode_get_prefix(pNode2, &str);
5763         ok( hr == S_OK, "ret %08x\n", hr );
5764         ok( !lstrcmpW( str, _bstr_("WEB")), "incorrect prefix string\n");
5765         SysFreeString(str);
5766
5767         /* Test get_namespaceURI */
5768         hr = IXMLDOMNode_get_namespaceURI(pNode2, NULL);
5769         ok( hr == E_INVALIDARG, "ret %08x\n", hr );
5770         /* NOTE: Need to test that arg2 gets cleared on Error. */
5771
5772         hr = IXMLDOMNode_get_namespaceURI(pNode2, &str);
5773         ok( hr == S_OK, "ret %08x\n", hr );
5774         ok( !lstrcmpW( str, _bstr_("http://www.winehq.org")), "incorrect namespaceURI string\n");
5775         SysFreeString(str);
5776
5777         IXMLDOMNode_Release(pNode2);
5778         IXMLDOMNode_Release(pNode);
5779     }
5780
5781     IXMLDOMDocument_Release(doc);
5782
5783     free_bstrs();
5784 }
5785
5786 static void test_FormattingXML(void)
5787 {
5788     IXMLDOMDocument *doc;
5789     IXMLDOMElement *pElement;
5790     VARIANT_BOOL bSucc;
5791     HRESULT hr;
5792     BSTR str;
5793     static const CHAR szLinefeedXML[] = "<?xml version=\"1.0\"?>\n<Root>\n\t<Sub val=\"A\" />\n</Root>";
5794     static const CHAR szLinefeedRootXML[] = "<Root>\r\n\t<Sub val=\"A\"/>\r\n</Root>";
5795
5796     doc = create_document(&IID_IXMLDOMDocument);
5797     if (!doc) return;
5798
5799     hr = IXMLDOMDocument_loadXML(doc, _bstr_(szLinefeedXML), &bSucc);
5800     ok(hr == S_OK, "ret %08x\n", hr );
5801     ok(bSucc == VARIANT_TRUE, "Expected VARIANT_TRUE got VARIANT_FALSE\n");
5802
5803     if(bSucc == VARIANT_TRUE)
5804     {
5805         hr = IXMLDOMDocument_get_documentElement(doc, &pElement);
5806         ok(hr == S_OK, "ret %08x\n", hr );
5807         if(hr == S_OK)
5808         {
5809             hr = IXMLDOMElement_get_xml(pElement, &str);
5810             ok(hr == S_OK, "ret %08x\n", hr );
5811             ok( !lstrcmpW( str, _bstr_(szLinefeedRootXML) ), "incorrect element xml\n");
5812             SysFreeString(str);
5813
5814             IXMLDOMElement_Release(pElement);
5815         }
5816     }
5817
5818     IXMLDOMDocument_Release(doc);
5819
5820     free_bstrs();
5821 }
5822
5823 typedef struct _nodetypedvalue_t {
5824     const char *name;
5825     VARTYPE type;
5826     const char *value; /* value in string format */
5827 } nodetypedvalue_t;
5828
5829 static const nodetypedvalue_t get_nodetypedvalue[] = {
5830     { "root/string",    VT_BSTR, "Wine" },
5831     { "root/string2",   VT_BSTR, "String" },
5832     { "root/number",    VT_BSTR, "12.44" },
5833     { "root/number2",   VT_BSTR, "-3.71e3" },
5834     { "root/int",       VT_I4,   "-13" },
5835     { "root/fixed",     VT_CY,   "7322.9371" },
5836     { "root/bool",      VT_BOOL, "-1" },
5837     { "root/datetime",  VT_DATE, "40135.14" },
5838     { "root/datetimetz",VT_DATE, "37813.59" },
5839     { "root/date",      VT_DATE, "665413" },
5840     { "root/time",      VT_DATE, "0.5813889" },
5841     { "root/timetz",    VT_DATE, "1.112512" },
5842     { "root/i1",        VT_I1,   "-13" },
5843     { "root/i2",        VT_I2,   "31915" },
5844     { "root/i4",        VT_I4,   "-312232" },
5845     { "root/ui1",       VT_UI1,  "123" },
5846     { "root/ui2",       VT_UI2,  "48282" },
5847     { "root/ui4",       VT_UI4,  "949281" },
5848     { "root/r4",        VT_R4,   "213124" },
5849     { "root/r8",        VT_R8,   "0.412" },
5850     { "root/float",     VT_R8,   "41221.421" },
5851     { "root/uuid",      VT_BSTR, "333C7BC4-460F-11D0-BC04-0080C7055a83" },
5852     { 0 }
5853 };
5854
5855 static void test_nodeTypedValue(void)
5856 {
5857     const nodetypedvalue_t *entry = get_nodetypedvalue;
5858     IXMLDOMDocument *doc;
5859     IXMLDOMNode *node;
5860     VARIANT_BOOL b;
5861     VARIANT value;
5862     HRESULT hr;
5863
5864     doc = create_document(&IID_IXMLDOMDocument);
5865     if (!doc) return;
5866
5867     b = VARIANT_FALSE;
5868     hr = IXMLDOMDocument_loadXML(doc, _bstr_(szTypeValueXML), &b);
5869     ok(hr == S_OK, "ret %08x\n", hr );
5870     ok(b == VARIANT_TRUE, "got %d\n", b);
5871
5872     hr = IXMLDOMDocument_get_nodeValue(doc, NULL);
5873     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5874
5875     V_VT(&value) = VT_BSTR;
5876     V_BSTR(&value) = NULL;
5877     hr = IXMLDOMDocument_get_nodeValue(doc, &value);
5878     ok(hr == S_FALSE, "ret %08x\n", hr );
5879     ok(V_VT(&value) == VT_NULL, "expect VT_NULL got %d\n", V_VT(&value));
5880
5881     hr = IXMLDOMDocument_get_nodeTypedValue(doc, NULL);
5882     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5883
5884     hr = IXMLDOMDocument_get_nodeTypedValue(doc, &value);
5885     ok(hr == S_FALSE, "ret %08x\n", hr );
5886
5887     hr = IXMLDOMDocument_selectSingleNode(doc, _bstr_("root/string"), &node);
5888     ok(hr == S_OK, "ret %08x\n", hr );
5889
5890     V_VT(&value) = VT_BSTR;
5891     V_BSTR(&value) = NULL;
5892     hr = IXMLDOMNode_get_nodeValue(node, &value);
5893     ok(hr == S_FALSE, "ret %08x\n", hr );
5894     ok(V_VT(&value) == VT_NULL, "expect VT_NULL got %d\n", V_VT(&value));
5895
5896     hr = IXMLDOMNode_get_nodeTypedValue(node, NULL);
5897     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5898
5899     IXMLDOMNode_Release(node);
5900
5901     hr = IXMLDOMDocument_selectSingleNode(doc, _bstr_("root/binhex"), &node);
5902     ok(hr == S_OK, "ret %08x\n", hr );
5903     {
5904         BYTE bytes[] = {0xff,0xfc,0xa0,0x12,0x00,0x3c};
5905
5906         hr = IXMLDOMNode_get_nodeTypedValue(node, &value);
5907         ok(hr == S_OK, "ret %08x\n", hr );
5908         ok(V_VT(&value) == (VT_ARRAY|VT_UI1), "incorrect type\n");
5909         ok(V_ARRAY(&value)->rgsabound[0].cElements == 6, "incorrect array size\n");
5910         if(V_ARRAY(&value)->rgsabound[0].cElements == 6)
5911             ok(!memcmp(bytes, V_ARRAY(&value)->pvData, sizeof(bytes)), "incorrect value\n");
5912         VariantClear(&value);
5913         IXMLDOMNode_Release(node);
5914     }
5915
5916     hr = IXMLDOMDocument_selectSingleNode(doc, _bstr_("root/binbase64"), &node);
5917     ok(hr == S_OK, "ret %08x\n", hr );
5918     {
5919         BYTE bytes[] = {0x62,0x61,0x73,0x65,0x36,0x34,0x20,0x74,0x65,0x73,0x74};
5920
5921         hr = IXMLDOMNode_get_nodeTypedValue(node, &value);
5922         ok(hr == S_OK, "ret %08x\n", hr );
5923         ok(V_VT(&value) == (VT_ARRAY|VT_UI1), "incorrect type\n");
5924         ok(V_ARRAY(&value)->rgsabound[0].cElements == 11, "incorrect array size\n");
5925         if(V_ARRAY(&value)->rgsabound[0].cElements == 11)
5926             ok(!memcmp(bytes, V_ARRAY(&value)->pvData, sizeof(bytes)), "incorrect value\n");
5927         VariantClear(&value);
5928         IXMLDOMNode_Release(node);
5929     }
5930
5931     while (entry->name)
5932     {
5933         hr = IXMLDOMDocument_selectSingleNode(doc, _bstr_(entry->name), &node);
5934         ok(hr == S_OK, "ret %08x\n", hr );
5935
5936         hr = IXMLDOMNode_get_nodeTypedValue(node, &value);
5937         ok(hr == S_OK, "ret %08x\n", hr );
5938         ok(V_VT(&value) == entry->type, "incorrect type, expected %d, got %d\n", entry->type, V_VT(&value));
5939
5940         if (entry->type != VT_BSTR)
5941         {
5942            if (entry->type == VT_DATE ||
5943                entry->type == VT_R8 ||
5944                entry->type == VT_CY)
5945            {
5946                if (entry->type == VT_DATE)
5947                {
5948                    hr = VariantChangeType(&value, &value, 0, VT_R4);
5949                    ok(hr == S_OK, "ret %08x\n", hr );
5950                }
5951                hr = VariantChangeTypeEx(&value, &value,
5952                                         MAKELCID(MAKELANGID(LANG_ENGLISH, SUBLANG_DEFAULT), SORT_DEFAULT),
5953                                         VARIANT_NOUSEROVERRIDE, VT_BSTR);
5954                ok(hr == S_OK, "ret %08x\n", hr );
5955            }
5956            else
5957            {
5958                hr = VariantChangeType(&value, &value, 0, VT_BSTR);
5959                ok(hr == S_OK, "ret %08x\n", hr );
5960            }
5961
5962            ok(lstrcmpW( V_BSTR(&value), _bstr_(entry->value)) == 0,
5963               "expected %s, got %s\n", entry->value, wine_dbgstr_w(V_BSTR(&value)));
5964         }
5965         else
5966            ok(lstrcmpW( V_BSTR(&value), _bstr_(entry->value)) == 0,
5967                "expected %s, got %s\n", entry->value, wine_dbgstr_w(V_BSTR(&value)));
5968
5969         VariantClear( &value );
5970         IXMLDOMNode_Release(node);
5971
5972         entry++;
5973     }
5974
5975     IXMLDOMDocument_Release(doc);
5976     free_bstrs();
5977 }
5978
5979 static void test_TransformWithLoadingLocalFile(void)
5980 {
5981     IXMLDOMDocument *doc;
5982     IXMLDOMDocument *xsl;
5983     IXMLDOMNode *pNode;
5984     VARIANT_BOOL bSucc;
5985     HRESULT hr;
5986     HANDLE file;
5987     DWORD dwWritten;
5988     char lpPathBuffer[MAX_PATH];
5989     int i;
5990
5991     /* Create a Temp File. */
5992     GetTempPathA(MAX_PATH, lpPathBuffer);
5993     strcat(lpPathBuffer, "customers.xml" );
5994
5995     file = CreateFile(lpPathBuffer, GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL );
5996     ok(file != INVALID_HANDLE_VALUE, "Could not create file: %u\n", GetLastError());
5997     if(file == INVALID_HANDLE_VALUE)
5998         return;
5999
6000     WriteFile(file, szBasicTransformXML, strlen(szBasicTransformXML), &dwWritten, NULL);
6001     CloseHandle(file);
6002
6003     /* Correct path to not include a escape character. */
6004     for(i=0; i < strlen(lpPathBuffer); i++)
6005     {
6006         if(lpPathBuffer[i] == '\\')
6007             lpPathBuffer[i] = '/';
6008     }
6009
6010     doc = create_document(&IID_IXMLDOMDocument);
6011     if (!doc) return;
6012
6013     xsl = create_document(&IID_IXMLDOMDocument);
6014     if (!xsl)
6015     {
6016         IXMLDOMDocument2_Release(doc);
6017         return;
6018     }
6019
6020     hr = IXMLDOMDocument_loadXML(doc, _bstr_(szTypeValueXML), &bSucc);
6021     ok(hr == S_OK, "ret %08x\n", hr );
6022     ok(bSucc == VARIANT_TRUE, "Expected VARIANT_TRUE got VARIANT_FALSE\n");
6023     if(bSucc == VARIANT_TRUE)
6024     {
6025         BSTR sXSL;
6026         BSTR sPart1 = _bstr_(szBasicTransformSSXMLPart1);
6027         BSTR sPart2 = _bstr_(szBasicTransformSSXMLPart2);
6028         BSTR sFileName = _bstr_(lpPathBuffer);
6029         int nLegnth = lstrlenW(sPart1) + lstrlenW(sPart2) + lstrlenW(sFileName) + 1;
6030
6031         sXSL = SysAllocStringLen(NULL, nLegnth);
6032         lstrcpyW(sXSL, sPart1);
6033         lstrcatW(sXSL, sFileName);
6034         lstrcatW(sXSL, sPart2);
6035
6036         hr = IXMLDOMDocument_loadXML(xsl, sXSL, &bSucc);
6037         ok(hr == S_OK, "ret %08x\n", hr );
6038         ok(bSucc == VARIANT_TRUE, "Expected VARIANT_TRUE got VARIANT_FALSE\n");
6039         if(bSucc == VARIANT_TRUE)
6040         {
6041             BSTR sResult;
6042
6043             hr = IXMLDOMDocument_QueryInterface(xsl, &IID_IXMLDOMNode, (void**)&pNode );
6044             ok(hr == S_OK, "ret %08x\n", hr );
6045             if(hr == S_OK)
6046             {
6047                 /* This will load the temp file via the XSL */
6048                 hr = IXMLDOMDocument_transformNode(doc, pNode, &sResult);
6049                 ok(hr == S_OK, "ret %08x\n", hr );
6050                 if(hr == S_OK)
6051                 {
6052                     ok( compareIgnoreReturns( sResult, _bstr_(szBasicTransformOutput)), "Stylesheet output not correct\n");
6053                     SysFreeString(sResult);
6054                 }
6055
6056                 IXMLDOMNode_Release(pNode);
6057             }
6058         }
6059
6060         SysFreeString(sXSL);
6061     }
6062
6063     IXMLDOMDocument_Release(doc);
6064     IXMLDOMDocument_Release(xsl);
6065
6066     DeleteFile(lpPathBuffer);
6067     free_bstrs();
6068 }
6069
6070 static void test_put_nodeValue(void)
6071 {
6072     static const WCHAR jeevesW[] = {'J','e','e','v','e','s',' ','&',' ','W','o','o','s','t','e','r',0};
6073     IXMLDOMDocument *doc;
6074     IXMLDOMText *text;
6075     IXMLDOMEntityReference *entityref;
6076     IXMLDOMAttribute *attr;
6077     IXMLDOMNode *node;
6078     HRESULT hr;
6079     VARIANT data, type;
6080
6081     doc = create_document(&IID_IXMLDOMDocument);
6082     if (!doc) return;
6083
6084     /* test for unsupported types */
6085     /* NODE_DOCUMENT */
6086     hr = IXMLDOMDocument_QueryInterface(doc, &IID_IXMLDOMNode, (void**)&node);
6087     ok(hr == S_OK, "ret %08x\n", hr );
6088     V_VT(&data) = VT_BSTR;
6089     V_BSTR(&data) = _bstr_("one two three");
6090     hr = IXMLDOMNode_put_nodeValue(node, data);
6091     ok(hr == E_FAIL, "ret %08x\n", hr );
6092     IXMLDOMNode_Release(node);
6093
6094     /* NODE_DOCUMENT_FRAGMENT */
6095     V_VT(&type) = VT_I1;
6096     V_I1(&type) = NODE_DOCUMENT_FRAGMENT;
6097     hr = IXMLDOMDocument_createNode(doc, type, _bstr_("test"), NULL, &node);
6098     ok(hr == S_OK, "ret %08x\n", hr );
6099     V_VT(&data) = VT_BSTR;
6100     V_BSTR(&data) = _bstr_("one two three");
6101     hr = IXMLDOMNode_put_nodeValue(node, data);
6102     ok(hr == E_FAIL, "ret %08x\n", hr );
6103     IXMLDOMNode_Release(node);
6104
6105     /* NODE_ELEMENT */
6106     V_VT(&type) = VT_I1;
6107     V_I1(&type) = NODE_ELEMENT;
6108     hr = IXMLDOMDocument_createNode(doc, type, _bstr_("test"), NULL, &node);
6109     ok(hr == S_OK, "ret %08x\n", hr );
6110     V_VT(&data) = VT_BSTR;
6111     V_BSTR(&data) = _bstr_("one two three");
6112     hr = IXMLDOMNode_put_nodeValue(node, data);
6113     ok(hr == E_FAIL, "ret %08x\n", hr );
6114     IXMLDOMNode_Release(node);
6115
6116     /* NODE_ENTITY_REFERENCE */
6117     hr = IXMLDOMDocument_createEntityReference(doc, _bstr_("ref"), &entityref);
6118     ok(hr == S_OK, "ret %08x\n", hr );
6119
6120     V_VT(&data) = VT_BSTR;
6121     V_BSTR(&data) = _bstr_("one two three");
6122     hr = IXMLDOMEntityReference_put_nodeValue(entityref, data);
6123     ok(hr == E_FAIL, "ret %08x\n", hr );
6124
6125     hr = IXMLDOMEntityReference_QueryInterface(entityref, &IID_IXMLDOMNode, (void**)&node);
6126     ok(hr == S_OK, "ret %08x\n", hr );
6127     V_VT(&data) = VT_BSTR;
6128     V_BSTR(&data) = _bstr_("one two three");
6129     hr = IXMLDOMNode_put_nodeValue(node, data);
6130     ok(hr == E_FAIL, "ret %08x\n", hr );
6131     IXMLDOMNode_Release(node);
6132     IXMLDOMEntityReference_Release(entityref);
6133
6134     /* supported types */
6135     hr = IXMLDOMDocument_createTextNode(doc, _bstr_(""), &text);
6136     ok(hr == S_OK, "ret %08x\n", hr );
6137     V_VT(&data) = VT_BSTR;
6138     V_BSTR(&data) = _bstr_("Jeeves & Wooster");
6139     hr = IXMLDOMText_put_nodeValue(text, data);
6140     ok(hr == S_OK, "ret %08x\n", hr );
6141     IXMLDOMText_Release(text);
6142
6143     hr = IXMLDOMDocument_createAttribute(doc, _bstr_("attr"), &attr);
6144     ok(hr == S_OK, "ret %08x\n", hr );
6145     V_VT(&data) = VT_BSTR;
6146     V_BSTR(&data) = _bstr_("Jeeves & Wooster");
6147     hr = IXMLDOMAttribute_put_nodeValue(attr, data);
6148     ok(hr == S_OK, "ret %08x\n", hr );
6149     hr = IXMLDOMAttribute_get_nodeValue(attr, &data);
6150     ok(hr == S_OK, "ret %08x\n", hr );
6151     ok(memcmp(V_BSTR(&data), jeevesW, sizeof(jeevesW)) == 0, "got %s\n",
6152         wine_dbgstr_w(V_BSTR(&data)));
6153     VariantClear(&data);
6154     IXMLDOMAttribute_Release(attr);
6155
6156     free_bstrs();
6157
6158     IXMLDOMDocument_Release(doc);
6159 }
6160
6161 static void test_IObjectSafety_set(IObjectSafety *safety, HRESULT result, HRESULT result2, DWORD set, DWORD mask, DWORD expected, DWORD expected2)
6162 {
6163     DWORD enabled, supported;
6164     HRESULT hr;
6165
6166     hr = IObjectSafety_SetInterfaceSafetyOptions(safety, NULL, set, mask);
6167     if (result == result2)
6168         ok(hr == result, "SetInterfaceSafetyOptions: expected %08x, returned %08x\n", result, hr );
6169     else
6170         ok(broken(hr == result) || hr == result2,
6171            "SetInterfaceSafetyOptions: expected %08x, got %08x\n", result2, hr );
6172
6173     supported = enabled = 0xCAFECAFE;
6174     hr = IObjectSafety_GetInterfaceSafetyOptions(safety, NULL, &supported, &enabled);
6175     ok(hr == S_OK, "ret %08x\n", hr );
6176     if (expected == expected2)
6177         ok(enabled == expected, "Expected %08x, got %08x\n", expected, enabled);
6178     else
6179         ok(broken(enabled == expected) || enabled == expected2,
6180            "Expected %08x, got %08x\n", expected2, enabled);
6181
6182     /* reset the safety options */
6183
6184     hr = IObjectSafety_SetInterfaceSafetyOptions(safety, NULL,
6185             INTERFACESAFE_FOR_UNTRUSTED_CALLER|INTERFACESAFE_FOR_UNTRUSTED_DATA|INTERFACE_USES_SECURITY_MANAGER,
6186             0);
6187     ok(hr == S_OK, "ret %08x\n", hr );
6188
6189     hr = IObjectSafety_GetInterfaceSafetyOptions(safety, NULL, &supported, &enabled);
6190     ok(hr == S_OK, "ret %08x\n", hr );
6191     ok(enabled == 0, "Expected 0, got %08x\n", enabled);
6192 }
6193
6194 static void test_document_IObjectSafety(void)
6195 {
6196     IXMLDOMDocument *doc;
6197     IObjectSafety *safety;
6198     DWORD enabled = 0, supported = 0;
6199     HRESULT hr;
6200
6201     doc = create_document(&IID_IXMLDOMDocument);
6202     if (!doc) return;
6203
6204     hr = IXMLDOMDocument_QueryInterface(doc, &IID_IObjectSafety, (void**)&safety);
6205     ok(hr == S_OK, "ret %08x\n", hr );
6206
6207     /* get */
6208     hr = IObjectSafety_GetInterfaceSafetyOptions(safety, NULL, NULL, &enabled);
6209     ok(hr == E_POINTER, "ret %08x\n", hr );
6210     hr = IObjectSafety_GetInterfaceSafetyOptions(safety, NULL, &supported, NULL);
6211     ok(hr == E_POINTER, "ret %08x\n", hr );
6212
6213     hr = IObjectSafety_GetInterfaceSafetyOptions(safety, NULL, &supported, &enabled);
6214     ok(hr == S_OK, "ret %08x\n", hr );
6215     ok(broken(supported == (INTERFACESAFE_FOR_UNTRUSTED_CALLER | INTERFACESAFE_FOR_UNTRUSTED_DATA)) ||
6216        supported == (INTERFACESAFE_FOR_UNTRUSTED_CALLER | INTERFACESAFE_FOR_UNTRUSTED_DATA | INTERFACE_USES_SECURITY_MANAGER) /* msxml3 SP8+ */,
6217         "Expected (INTERFACESAFE_FOR_UNTRUSTED_CALLER | INTERFACESAFE_FOR_UNTRUSTED_DATA | INTERFACE_USES_SECURITY_MANAGER), "
6218              "got %08x\n", supported);
6219     ok(enabled == 0, "Expected 0, got %08x\n", enabled);
6220
6221     /* set -- individual flags */
6222
6223     test_IObjectSafety_set(safety, S_OK, S_OK,
6224         INTERFACESAFE_FOR_UNTRUSTED_CALLER, INTERFACESAFE_FOR_UNTRUSTED_CALLER,
6225         INTERFACESAFE_FOR_UNTRUSTED_CALLER, INTERFACESAFE_FOR_UNTRUSTED_CALLER);
6226
6227     test_IObjectSafety_set(safety, S_OK, S_OK,
6228         INTERFACESAFE_FOR_UNTRUSTED_DATA, INTERFACESAFE_FOR_UNTRUSTED_DATA,
6229         INTERFACESAFE_FOR_UNTRUSTED_DATA, INTERFACESAFE_FOR_UNTRUSTED_DATA);
6230
6231     test_IObjectSafety_set(safety, S_OK, S_OK,
6232         INTERFACE_USES_SECURITY_MANAGER, INTERFACE_USES_SECURITY_MANAGER,
6233         0, INTERFACE_USES_SECURITY_MANAGER /* msxml3 SP8+ */);
6234
6235     /* set INTERFACE_USES_DISPEX  */
6236
6237     test_IObjectSafety_set(safety, S_OK, E_FAIL /* msxml3 SP8+ */,
6238         INTERFACE_USES_DISPEX, INTERFACE_USES_DISPEX,
6239         0, 0);
6240
6241     test_IObjectSafety_set(safety, S_OK, E_FAIL /* msxml3 SP8+ */,
6242         INTERFACE_USES_DISPEX, 0,
6243         0, 0);
6244
6245     test_IObjectSafety_set(safety, S_OK, S_OK /* msxml3 SP8+ */,
6246         0, INTERFACE_USES_DISPEX,
6247         0, 0);
6248
6249     /* set option masking */
6250
6251     test_IObjectSafety_set(safety, S_OK, S_OK,
6252         INTERFACESAFE_FOR_UNTRUSTED_CALLER|INTERFACESAFE_FOR_UNTRUSTED_DATA,
6253         INTERFACESAFE_FOR_UNTRUSTED_CALLER,
6254         INTERFACESAFE_FOR_UNTRUSTED_CALLER,
6255         INTERFACESAFE_FOR_UNTRUSTED_CALLER);
6256
6257     test_IObjectSafety_set(safety, S_OK, S_OK,
6258         INTERFACESAFE_FOR_UNTRUSTED_CALLER|INTERFACESAFE_FOR_UNTRUSTED_DATA,
6259         INTERFACESAFE_FOR_UNTRUSTED_DATA,
6260         INTERFACESAFE_FOR_UNTRUSTED_DATA,
6261         INTERFACESAFE_FOR_UNTRUSTED_DATA);
6262
6263     test_IObjectSafety_set(safety, S_OK, S_OK,
6264         INTERFACESAFE_FOR_UNTRUSTED_CALLER|INTERFACESAFE_FOR_UNTRUSTED_DATA,
6265         INTERFACE_USES_SECURITY_MANAGER,
6266         0,
6267         0);
6268
6269     /* set -- inheriting previous settings */
6270
6271     hr = IObjectSafety_SetInterfaceSafetyOptions(safety, NULL,
6272                                                          INTERFACESAFE_FOR_UNTRUSTED_CALLER,
6273                                                          INTERFACESAFE_FOR_UNTRUSTED_CALLER);
6274     ok(hr == S_OK, "ret %08x\n", hr );
6275     hr = IObjectSafety_GetInterfaceSafetyOptions(safety, NULL, &supported, &enabled);
6276     ok(hr == S_OK, "ret %08x\n", hr );
6277     todo_wine
6278     ok(broken(enabled == INTERFACESAFE_FOR_UNTRUSTED_CALLER) ||
6279        enabled == (INTERFACESAFE_FOR_UNTRUSTED_CALLER | INTERFACE_USES_SECURITY_MANAGER) /* msxml3 SP8+ */,
6280          "Expected (INTERFACESAFE_FOR_UNTRUSTED_CALLER | INTERFACE_USES_SECURITY_MANAGER), "
6281          "got %08x\n", enabled);
6282
6283     hr = IObjectSafety_SetInterfaceSafetyOptions(safety, NULL,
6284                                                          INTERFACESAFE_FOR_UNTRUSTED_DATA,
6285                                                          INTERFACESAFE_FOR_UNTRUSTED_DATA);
6286     ok(hr == S_OK, "ret %08x\n", hr );
6287     hr = IObjectSafety_GetInterfaceSafetyOptions(safety, NULL, &supported, &enabled);
6288     ok(hr == S_OK, "ret %08x\n", hr );
6289     todo_wine
6290     ok(broken(enabled == INTERFACESAFE_FOR_UNTRUSTED_DATA) ||
6291        enabled == (INTERFACESAFE_FOR_UNTRUSTED_CALLER | INTERFACESAFE_FOR_UNTRUSTED_DATA) /* msxml3 SP8+ */,
6292         "Expected (INTERFACESAFE_FOR_UNTRUSTED_CALLER | INTERFACESAFE_FOR_UNTRUSTED_DATA), "
6293         "got %08x\n", enabled);
6294
6295     IObjectSafety_Release(safety);
6296
6297     IXMLDOMDocument_Release(doc);
6298 }
6299
6300 typedef struct _property_test_t {
6301     const GUID *guid;
6302     const char *clsid;
6303     const char *property;
6304     const char *value;
6305 } property_test_t;
6306
6307 static const property_test_t properties_test_data[] = {
6308     { &CLSID_DOMDocument,  "CLSID_DOMDocument" , "SelectionLanguage", "XSLPattern" },
6309     { &CLSID_DOMDocument2,  "CLSID_DOMDocument2" , "SelectionLanguage", "XSLPattern" },
6310     { &CLSID_DOMDocument30, "CLSID_DOMDocument30", "SelectionLanguage", "XSLPattern" },
6311     { &CLSID_DOMDocument40, "CLSID_DOMDocument40", "SelectionLanguage", "XPath" },
6312     { &CLSID_DOMDocument60, "CLSID_DOMDocument60", "SelectionLanguage", "XPath" },
6313     { 0 }
6314 };
6315
6316 static void test_default_properties(void)
6317 {
6318     const property_test_t *entry = properties_test_data;
6319     IXMLDOMDocument2 *doc;
6320     VARIANT var;
6321     HRESULT hr;
6322
6323     while (entry->guid)
6324     {
6325         hr = CoCreateInstance(entry->guid, NULL, CLSCTX_INPROC_SERVER, &IID_IXMLDOMDocument2, (void**)&doc);
6326         if (hr != S_OK)
6327         {
6328             win_skip("can't create %s instance\n", entry->clsid);
6329             entry++;
6330             continue;
6331         }
6332
6333         hr = IXMLDOMDocument2_getProperty(doc, _bstr_(entry->property), &var);
6334         ok(hr == S_OK, "got 0x%08x\n", hr);
6335         ok(lstrcmpW(V_BSTR(&var), _bstr_(entry->value)) == 0, "expected %s, for %s\n",
6336            entry->value, entry->clsid);
6337         VariantClear(&var);
6338
6339         IXMLDOMDocument2_Release(doc);
6340
6341         entry++;
6342     }
6343 }
6344
6345 static void test_XSLPattern(void)
6346 {
6347     IXMLDOMDocument2 *doc;
6348     IXMLDOMNodeList *list;
6349     VARIANT_BOOL b;
6350     LONG len;
6351
6352     doc = create_document(&IID_IXMLDOMDocument2);
6353     if (!doc) return;
6354
6355     ole_check(IXMLDOMDocument2_loadXML(doc, _bstr_(szExampleXML), &b));
6356     ok(b == VARIANT_TRUE, "failed to load XML string\n");
6357
6358     /* switch to XSLPattern */
6359     ole_check(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionLanguage"), _variantbstr_("XSLPattern")));
6360
6361     /* XPath doesn't select elements with non-null default namespace with unqualified selectors, XSLPattern does */
6362     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//elem/c"), &list));
6363     len = 0;
6364     ole_check(IXMLDOMNodeList_get_length(list, &len));
6365     /* should select <elem><c> and <elem xmlns='...'><c> but not <elem><foo:c> */
6366     ok(len == 3, "expected 3 entries in list, got %d\n", len);
6367     IXMLDOMNodeList_Release(list);
6368
6369     /* for XSLPattern start index is 0, for XPath it's 1 */
6370     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[0]"), &list));
6371     len = 0;
6372     ole_check(IXMLDOMNodeList_get_length(list, &len));
6373     ok(len != 0, "expected filled list\n");
6374     if (len)
6375         expect_list_and_release(list, "E1.E2.D1");
6376
6377     /* index() */
6378     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index()=1]"), &list));
6379     len = 0;
6380     ole_check(IXMLDOMNodeList_get_length(list, &len));
6381     ok(len != 0, "expected filled list\n");
6382     if (len)
6383         expect_list_and_release(list, "E2.E2.D1");
6384
6385     /* $eq$ */
6386     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index() $eq$ 1]"), &list));
6387     len = 0;
6388     ole_check(IXMLDOMNodeList_get_length(list, &len));
6389     ok(len != 0, "expected filled list\n");
6390     if (len)
6391         expect_list_and_release(list, "E2.E2.D1");
6392
6393     /* end() */
6394     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[end()]"), &list));
6395     len = 0;
6396     ole_check(IXMLDOMNodeList_get_length(list, &len));
6397     ok(len != 0, "expected filled list\n");
6398     if (len)
6399         expect_list_and_release(list, "E4.E2.D1");
6400
6401     /* $not$ */
6402     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[$not$ end()]"), &list));
6403     len = 0;
6404     ole_check(IXMLDOMNodeList_get_length(list, &len));
6405     ok(len != 0, "expected filled list\n");
6406     if (len)
6407         expect_list_and_release(list, "E1.E2.D1 E2.E2.D1 E3.E2.D1");
6408
6409     /* !=/$ne$ */
6410     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index() != 0]"), &list));
6411     len = 0;
6412     ole_check(IXMLDOMNodeList_get_length(list, &len));
6413     ok(len != 0, "expected filled list\n");
6414     if (len)
6415         expect_list_and_release(list, "E2.E2.D1 E3.E2.D1 E4.E2.D1");
6416     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index() $ne$ 0]"), &list));
6417     len = 0;
6418     ole_check(IXMLDOMNodeList_get_length(list, &len));
6419     ok(len != 0, "expected filled list\n");
6420     if (len)
6421         expect_list_and_release(list, "E2.E2.D1 E3.E2.D1 E4.E2.D1");
6422
6423     /* </$lt$ */
6424     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index() < 2]"), &list));
6425     len = 0;
6426     ole_check(IXMLDOMNodeList_get_length(list, &len));
6427     ok(len != 0, "expected filled list\n");
6428     if (len)
6429         expect_list_and_release(list, "E1.E2.D1 E2.E2.D1");
6430     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index() $lt$ 2]"), &list));
6431     len = 0;
6432     ole_check(IXMLDOMNodeList_get_length(list, &len));
6433     ok(len != 0, "expected filled list\n");
6434     if (len)
6435         expect_list_and_release(list, "E1.E2.D1 E2.E2.D1");
6436
6437     /* <=/$le$ */
6438     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index() <= 1]"), &list));
6439     len = 0;
6440     ole_check(IXMLDOMNodeList_get_length(list, &len));
6441     ok(len != 0, "expected filled list\n");
6442     if (len)
6443         expect_list_and_release(list, "E1.E2.D1 E2.E2.D1");
6444     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index() $le$ 1]"), &list));
6445     len = 0;
6446     ole_check(IXMLDOMNodeList_get_length(list, &len));
6447     ok(len != 0, "expected filled list\n");
6448     if (len)
6449         expect_list_and_release(list, "E1.E2.D1 E2.E2.D1");
6450
6451     /* >/$gt$ */
6452     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index() > 1]"), &list));
6453     len = 0;
6454     ole_check(IXMLDOMNodeList_get_length(list, &len));
6455     ok(len != 0, "expected filled list\n");
6456     if (len)
6457         expect_list_and_release(list, "E3.E2.D1 E4.E2.D1");
6458     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index() $gt$ 1]"), &list));
6459     len = 0;
6460     ole_check(IXMLDOMNodeList_get_length(list, &len));
6461     ok(len != 0, "expected filled list\n");
6462     if (len)
6463         expect_list_and_release(list, "E3.E2.D1 E4.E2.D1");
6464
6465     /* >=/$ge$ */
6466     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index() >= 2]"), &list));
6467     len = 0;
6468     ole_check(IXMLDOMNodeList_get_length(list, &len));
6469     ok(len != 0, "expected filled list\n");
6470     if (len)
6471         expect_list_and_release(list, "E3.E2.D1 E4.E2.D1");
6472     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index() $ge$ 2]"), &list));
6473     len = 0;
6474     ole_check(IXMLDOMNodeList_get_length(list, &len));
6475     ok(len != 0, "expected filled list\n");
6476     if (len)
6477         expect_list_and_release(list, "E3.E2.D1 E4.E2.D1");
6478
6479     /* $ieq$ */
6480     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[a $ieq$ 'a2 field']"), &list));
6481     len = 0;
6482     ole_check(IXMLDOMNodeList_get_length(list, &len));
6483     ok(len != 0, "expected filled list\n");
6484     if (len)
6485         expect_list_and_release(list, "E2.E2.D1");
6486
6487     /* $ine$ */
6488     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[a $ine$ 'a2 field']"), &list));
6489     len = 0;
6490     ole_check(IXMLDOMNodeList_get_length(list, &len));
6491     ok(len != 0, "expected filled list\n");
6492     if (len)
6493         expect_list_and_release(list, "E1.E2.D1 E3.E2.D1 E4.E2.D1");
6494
6495     /* $ilt$ */
6496     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[a $ilt$ 'a3 field']"), &list));
6497     len = 0;
6498     ole_check(IXMLDOMNodeList_get_length(list, &len));
6499     ok(len != 0, "expected filled list\n");
6500     if (len)
6501         expect_list_and_release(list, "E1.E2.D1 E2.E2.D1");
6502
6503     /* $ile$ */
6504     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[a $ile$ 'a2 field']"), &list));
6505     len = 0;
6506     ole_check(IXMLDOMNodeList_get_length(list, &len));
6507     ok(len != 0, "expected filled list\n");
6508     if (len)
6509         expect_list_and_release(list, "E1.E2.D1 E2.E2.D1");
6510
6511     /* $igt$ */
6512     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[a $igt$ 'a2 field']"), &list));
6513     len = 0;
6514     ole_check(IXMLDOMNodeList_get_length(list, &len));
6515     ok(len != 0, "expected filled list\n");
6516     if (len)
6517         expect_list_and_release(list, "E3.E2.D1 E4.E2.D1");
6518
6519     /* $ige$ */
6520     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[a $ige$ 'a3 field']"), &list));
6521     len = 0;
6522     ole_check(IXMLDOMNodeList_get_length(list, &len));
6523     ok(len != 0, "expected filled list\n");
6524     if (len)
6525         expect_list_and_release(list, "E3.E2.D1 E4.E2.D1");
6526
6527     /* $any$ */
6528     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[$any$ *='B2 field']"), &list));
6529     len = 0;
6530     ole_check(IXMLDOMNodeList_get_length(list, &len));
6531     ok(len != 0, "expected filled list\n");
6532     if (len)
6533         expect_list_and_release(list, "E2.E2.D1");
6534
6535     /* $all$ */
6536     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[$all$ *!='B2 field']"), &list));
6537     len = 0;
6538     ole_check(IXMLDOMNodeList_get_length(list, &len));
6539     ok(len != 0, "expected filled list\n");
6540     if (len)
6541         expect_list_and_release(list, "E1.E2.D1 E3.E2.D1 E4.E2.D1");
6542
6543     /* or/$or$/|| */
6544     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index()=0 or end()]"), &list));
6545     len = 0;
6546     ole_check(IXMLDOMNodeList_get_length(list, &len));
6547     ok(len != 0, "expected filled list\n");
6548     if (len)
6549         expect_list_and_release(list, "E1.E2.D1 E4.E2.D1");
6550     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index()=0 $or$ end()]"), &list));
6551     len = 0;
6552     ole_check(IXMLDOMNodeList_get_length(list, &len));
6553     ok(len != 0, "expected filled list\n");
6554     if (len)
6555         expect_list_and_release(list, "E1.E2.D1 E4.E2.D1");
6556     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index()=0 || end()]"), &list));
6557     len = 0;
6558     ole_check(IXMLDOMNodeList_get_length(list, &len));
6559     ok(len != 0, "expected filled list\n");
6560     if (len)
6561         expect_list_and_release(list, "E1.E2.D1 E4.E2.D1");
6562
6563     /* and/$and$/&& */
6564     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index()>0 and $not$ end()]"), &list));
6565     len = 0;
6566     ole_check(IXMLDOMNodeList_get_length(list, &len));
6567     ok(len != 0, "expected filled list\n");
6568     if (len)
6569         expect_list_and_release(list, "E2.E2.D1 E3.E2.D1");
6570     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index()>0 $and$ $not$ end()]"), &list));
6571     len = 0;
6572     ole_check(IXMLDOMNodeList_get_length(list, &len));
6573     ok(len != 0, "expected filled list\n");
6574     if (len)
6575         expect_list_and_release(list, "E2.E2.D1 E3.E2.D1");
6576     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index()>0 && $not$ end()]"), &list));
6577     len = 0;
6578     ole_check(IXMLDOMNodeList_get_length(list, &len));
6579     ok(len != 0, "expected filled list\n");
6580     if (len)
6581         expect_list_and_release(list, "E2.E2.D1 E3.E2.D1");
6582
6583     /* namespace handling */
6584     /* no registered namespaces */
6585     ole_check(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionNamespaces"), _variantbstr_("")));
6586     list = NULL;
6587
6588     /* prefixes don't need to be registered, you may use them as they are in the doc */
6589     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//bar:x"), &list));
6590     len = 0;
6591     ole_check(IXMLDOMNodeList_get_length(list, &len));
6592     ok(len != 0, "expected filled list\n");
6593     if (len)
6594         expect_list_and_release(list, "E5.E1.E4.E1.E2.D1 E6.E2.E4.E1.E2.D1");
6595
6596     /* prefixes must be explicitly specified in the name */
6597     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//foo:elem"), &list));
6598     len = 0;
6599     ole_check(IXMLDOMNodeList_get_length(list, &len));
6600     ok(len == 0, "expected empty list\n");
6601     IXMLDOMNodeList_Release(list);
6602
6603     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//foo:c"), &list));
6604     len = 0;
6605     ole_check(IXMLDOMNodeList_get_length(list, &len));
6606     ok(len != 0, "expected filled list\n");
6607     if (len)
6608         expect_list_and_release(list, "E3.E4.E2.D1");
6609
6610     /* explicitly register prefix foo */
6611     ole_check(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionNamespaces"), _variantbstr_("xmlns:foo='urn:uuid:86B2F87F-ACB6-45cd-8B77-9BDB92A01A29'")));
6612
6613     /* now we get the same behavior as XPath */
6614     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//foo:c"), &list));
6615     len = 0;
6616     ole_check(IXMLDOMNodeList_get_length(list, &len));
6617     ok(len != 0, "expected filled list\n");
6618     if (len)
6619         expect_list_and_release(list, "E3.E3.E2.D1 E3.E4.E2.D1");
6620
6621     /* set prefix foo to some nonexistent namespace */
6622     ole_check(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionNamespaces"), _variantbstr_("xmlns:foo='urn:nonexistent-foo'")));
6623
6624     /* the registered prefix takes precedence */
6625     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//foo:c"), &list));
6626     len = 0;
6627     ole_check(IXMLDOMNodeList_get_length(list, &len));
6628     ok(len == 0, "expected empty list\n");
6629     IXMLDOMNodeList_Release(list);
6630
6631     IXMLDOMDocument2_Release(doc);
6632
6633     doc = create_document(&IID_IXMLDOMDocument2);
6634     if (!doc) return;
6635
6636     ole_check(IXMLDOMDocument2_loadXML(doc, _bstr_(szNodeTypesXML), &b));
6637     ok(b == VARIANT_TRUE, "failed to load XML string\n");
6638     list = NULL;
6639
6640     /* attribute() */
6641     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("attribute()"), &list));
6642     len = 0;
6643     ole_check(IXMLDOMNodeList_get_length(list, &len));
6644     ok(len == 0, "expected empty list\n");
6645     IXMLDOMNodeList_Release(list);
6646
6647     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("attribute('depth')"), &list));
6648     len = 0;
6649     ole_check(IXMLDOMNodeList_get_length(list, &len));
6650     ok(len == 0, "expected empty list\n");
6651     IXMLDOMNodeList_Release(list);
6652
6653     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root/attribute('depth')"), &list));
6654     len = 0;
6655     ole_check(IXMLDOMNodeList_get_length(list, &len));
6656     ok(len != 0, "expected filled list\n");
6657     expect_list_and_release(list, "A'depth'.E3.D1");
6658
6659     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//x/attribute()"), &list));
6660     len = 0;
6661     ole_check(IXMLDOMNodeList_get_length(list, &len));
6662     ok(len != 0, "expected filled list\n");
6663     expect_list_and_release(list, "A'id'.E3.E3.D1 A'depth'.E3.E3.D1");
6664
6665     list = NULL;
6666     ole_expect(IXMLDOMDocument2_selectNodes(doc, _bstr_("//x//attribute(id)"), &list), E_FAIL);
6667     if (list)
6668         IXMLDOMNodeList_Release(list);
6669     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//x//attribute('id')"), &list));
6670     len = 0;
6671     ole_check(IXMLDOMNodeList_get_length(list, &len));
6672     ok(len != 0, "expected filled list\n");
6673     if (len)
6674         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");
6675
6676     /* comment() */
6677     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("comment()"), &list));
6678     len = 0;
6679     ole_check(IXMLDOMNodeList_get_length(list, &len));
6680     ok(len != 0, "expected filled list\n");
6681     if (len)
6682         expect_list_and_release(list, "C2.D1");
6683
6684     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//comment()"), &list));
6685     len = 0;
6686     ole_check(IXMLDOMNodeList_get_length(list, &len));
6687     ok(len != 0, "expected filled list\n");
6688     if (len)
6689         expect_list_and_release(list, "C2.D1 C1.E3.D1 C2.E3.E3.D1 C2.E4.E3.D1");
6690
6691     /* element() */
6692     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("element()"), &list));
6693     len = 0;
6694     ole_check(IXMLDOMNodeList_get_length(list, &len));
6695     ok(len != 0, "expected filled list\n");
6696     if (len)
6697         expect_list_and_release(list, "E3.D1");
6698
6699     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root/y/element()"), &list));
6700     len = 0;
6701     ole_check(IXMLDOMNodeList_get_length(list, &len));
6702     ok(len != 0, "expected filled list\n");
6703     if (len)
6704         expect_list_and_release(list, "E4.E4.E3.D1 E5.E4.E3.D1 E6.E4.E3.D1");
6705
6706     list = NULL;
6707     ole_expect(IXMLDOMDocument2_selectNodes(doc, _bstr_("//element(a)"), &list), E_FAIL);
6708     if (list)
6709         IXMLDOMNodeList_Release(list);
6710     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//element('a')"), &list));
6711     len = 0;
6712     ole_check(IXMLDOMNodeList_get_length(list, &len));
6713     ok(len != 0, "expected filled list\n");
6714     if (len)
6715         expect_list_and_release(list, "E4.E3.E3.D1 E4.E4.E3.D1");
6716
6717     /* node() */
6718     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("node()"), &list));
6719     len = 0;
6720     ole_check(IXMLDOMNodeList_get_length(list, &len));
6721     ok(len != 0, "expected filled list\n");
6722     if (len)
6723         expect_list_and_release(list, "P1.D1 C2.D1 E3.D1");
6724
6725     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//x/node()"), &list));
6726     len = 0;
6727     ole_check(IXMLDOMNodeList_get_length(list, &len));
6728     ok(len != 0, "expected filled list\n");
6729     if (len)
6730         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");
6731
6732     /* nodeType() */
6733     /* XML_ELEMENT_NODE */
6734     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//x/node()[nodeType()=1]"), &list));
6735     len = 0;
6736     ole_check(IXMLDOMNodeList_get_length(list, &len));
6737     ok(len != 0, "expected filled list\n");
6738     if (len)
6739         expect_list_and_release(list, "E4.E3.E3.D1 E5.E3.E3.D1 E6.E3.E3.D1");
6740     /* XML_TEXT_NODE */
6741     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//x/node()[nodeType()=3]"), &list));
6742     len = 0;
6743     ole_check(IXMLDOMNodeList_get_length(list, &len));
6744     ok(len != 0, "expected filled list\n");
6745     if (len)
6746         expect_list_and_release(list, "T3.E3.E3.D1");
6747     /* XML_PI_NODE */
6748     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//x/node()[nodeType()=7]"), &list));
6749     len = 0;
6750     ole_check(IXMLDOMNodeList_get_length(list, &len));
6751     ok(len != 0, "expected filled list\n");
6752     if (len)
6753         expect_list_and_release(list, "P1.E3.E3.D1");
6754     /* XML_COMMENT_NODE */
6755     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//x/node()[nodeType()=8]"), &list));
6756     len = 0;
6757     ole_check(IXMLDOMNodeList_get_length(list, &len));
6758     ok(len != 0, "expected filled list\n");
6759     if (len)
6760         expect_list_and_release(list, "C2.E3.E3.D1");
6761
6762     /* pi() */
6763     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("pi()"), &list));
6764     if (list)
6765     {
6766         len = 0;
6767         ole_check(IXMLDOMNodeList_get_length(list, &len));
6768         ok(len != 0, "expected filled list\n");
6769         if (len)
6770             expect_list_and_release(list, "P1.D1");
6771     }
6772
6773     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//y/pi()"), &list));
6774     len = 0;
6775     ole_check(IXMLDOMNodeList_get_length(list, &len));
6776     ok(len != 0, "expected filled list\n");
6777     if (len)
6778         expect_list_and_release(list, "P1.E4.E3.D1");
6779
6780     /* textnode() */
6781     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root/textnode()"), &list));
6782     len = 0;
6783     ole_check(IXMLDOMNodeList_get_length(list, &len));
6784     ok(len != 0, "expected filled list\n");
6785     if (len)
6786         expect_list_and_release(list, "T2.E3.D1");
6787
6788     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root/element()/textnode()"), &list));
6789     len = 0;
6790     ole_check(IXMLDOMNodeList_get_length(list, &len));
6791     ok(len != 0, "expected filled list\n");
6792     if (len)
6793         expect_list_and_release(list, "T3.E3.E3.D1 T3.E4.E3.D1");
6794
6795     IXMLDOMDocument2_Release(doc);
6796     free_bstrs();
6797 }
6798
6799 static void test_splitText(void)
6800 {
6801     IXMLDOMCDATASection *cdata;
6802     IXMLDOMElement *root;
6803     IXMLDOMDocument *doc;
6804     IXMLDOMText *text, *text2;
6805     IXMLDOMNode *node;
6806     VARIANT var;
6807     VARIANT_BOOL success;
6808     LONG length;
6809     HRESULT hr;
6810
6811     doc = create_document(&IID_IXMLDOMDocument);
6812     if (!doc) return;
6813
6814     hr = IXMLDOMDocument_loadXML(doc, _bstr_("<root></root>"), &success);
6815     ok(hr == S_OK, "got 0x%08x\n", hr);
6816
6817     hr = IXMLDOMDocument_get_documentElement(doc, &root);
6818     ok(hr == S_OK, "got 0x%08x\n", hr);
6819
6820     hr = IXMLDOMDocument_createCDATASection(doc, _bstr_("beautiful plumage"), &cdata);
6821     ok(hr == S_OK, "got 0x%08x\n", hr);
6822
6823     V_VT(&var) = VT_EMPTY;
6824     hr = IXMLDOMElement_appendChild(root, (IXMLDOMNode*)cdata, NULL);
6825     ok(hr == S_OK, "got 0x%08x\n", hr);
6826
6827     length = 0;
6828     hr = IXMLDOMCDATASection_get_length(cdata, &length);
6829     ok(hr == S_OK, "got 0x%08x\n", hr);
6830     ok(length > 0, "got %d\n", length);
6831
6832     hr = IXMLDOMCDATASection_splitText(cdata, 0, NULL);
6833     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
6834
6835     text = (void*)0xdeadbeef;
6836     /* negative offset */
6837     hr = IXMLDOMCDATASection_splitText(cdata, -1, &text);
6838     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
6839     ok(text == (void*)0xdeadbeef, "got %p\n", text);
6840
6841     text = (void*)0xdeadbeef;
6842     /* offset outside data */
6843     hr = IXMLDOMCDATASection_splitText(cdata, length + 1, &text);
6844     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
6845     ok(text == 0, "got %p\n", text);
6846
6847     text = (void*)0xdeadbeef;
6848     /* offset outside data */
6849     hr = IXMLDOMCDATASection_splitText(cdata, length, &text);
6850     ok(hr == S_FALSE, "got 0x%08x\n", hr);
6851     ok(text == 0, "got %p\n", text);
6852
6853     /* no empty node created */
6854     node = (void*)0xdeadbeef;
6855     hr = IXMLDOMCDATASection_get_nextSibling(cdata, &node);
6856     ok(hr == S_FALSE, "got 0x%08x\n", hr);
6857     ok(node == 0, "got %p\n", text);
6858
6859     hr = IXMLDOMCDATASection_splitText(cdata, 10, &text);
6860     ok(hr == S_OK, "got 0x%08x\n", hr);
6861
6862     length = 0;
6863     hr = IXMLDOMText_get_length(text, &length);
6864     ok(hr == S_OK, "got 0x%08x\n", hr);
6865     ok(length == 7, "got %d\n", length);
6866
6867     hr = IXMLDOMCDATASection_get_nextSibling(cdata, &node);
6868     ok(hr == S_OK, "got 0x%08x\n", hr);
6869     IXMLDOMNode_Release(node);
6870
6871     /* split new text node */
6872     hr = IXMLDOMText_get_length(text, &length);
6873     ok(hr == S_OK, "got 0x%08x\n", hr);
6874
6875     node = (void*)0xdeadbeef;
6876     hr = IXMLDOMText_get_nextSibling(text, &node);
6877     ok(hr == S_FALSE, "got 0x%08x\n", hr);
6878     ok(node == 0, "got %p\n", text);
6879
6880     hr = IXMLDOMText_splitText(text, 0, NULL);
6881     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
6882
6883     text2 = (void*)0xdeadbeef;
6884     /* negative offset */
6885     hr = IXMLDOMText_splitText(text, -1, &text2);
6886     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
6887     ok(text2 == (void*)0xdeadbeef, "got %p\n", text2);
6888
6889     text2 = (void*)0xdeadbeef;
6890     /* offset outside data */
6891     hr = IXMLDOMText_splitText(text, length + 1, &text2);
6892     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
6893     ok(text2 == 0, "got %p\n", text2);
6894
6895     text2 = (void*)0xdeadbeef;
6896     /* offset outside data */
6897     hr = IXMLDOMText_splitText(text, length, &text2);
6898     ok(hr == S_FALSE, "got 0x%08x\n", hr);
6899     ok(text2 == 0, "got %p\n", text);
6900
6901     text2 = 0;
6902     hr = IXMLDOMText_splitText(text, 4, &text2);
6903     todo_wine ok(hr == S_OK, "got 0x%08x\n", hr);
6904     if (text2) IXMLDOMText_Release(text2);
6905
6906     node = 0;
6907     hr = IXMLDOMText_get_nextSibling(text, &node);
6908     todo_wine ok(hr == S_OK, "got 0x%08x\n", hr);
6909     if (node) IXMLDOMNode_Release(node);
6910
6911     IXMLDOMText_Release(text);
6912     IXMLDOMElement_Release(root);
6913     IXMLDOMCDATASection_Release(cdata);
6914     free_bstrs();
6915 }
6916
6917 static void test_getQualifiedItem(void)
6918 {
6919     IXMLDOMDocument *doc;
6920     IXMLDOMElement *element;
6921     IXMLDOMNode *pr_node, *node;
6922     IXMLDOMNodeList *root_list;
6923     IXMLDOMNamedNodeMap *map;
6924     VARIANT_BOOL b;
6925     BSTR str;
6926     LONG len;
6927     HRESULT hr;
6928
6929     doc = create_document(&IID_IXMLDOMDocument);
6930     if (!doc) return;
6931
6932     str = SysAllocString( szComplete4 );
6933     hr = IXMLDOMDocument_loadXML( doc, str, &b );
6934     ok( hr == S_OK, "loadXML failed\n");
6935     ok( b == VARIANT_TRUE, "failed to load XML string\n");
6936     SysFreeString( str );
6937
6938     hr = IXMLDOMDocument_get_documentElement(doc, &element);
6939     ok( hr == S_OK, "ret %08x\n", hr);
6940
6941     hr = IXMLDOMElement_get_childNodes(element, &root_list);
6942     ok( hr == S_OK, "ret %08x\n", hr);
6943
6944     hr = IXMLDOMNodeList_get_item(root_list, 1, &pr_node);
6945     ok( hr == S_OK, "ret %08x\n", hr);
6946     IXMLDOMNodeList_Release(root_list);
6947
6948     hr = IXMLDOMNode_get_attributes(pr_node, &map);
6949     ok( hr == S_OK, "ret %08x\n", hr);
6950     IXMLDOMNode_Release(pr_node);
6951
6952     hr = IXMLDOMNamedNodeMap_get_length(map, &len);
6953     ok( hr == S_OK, "ret %08x\n", hr);
6954     ok( len == 3, "length %d\n", len);
6955
6956     hr = IXMLDOMNamedNodeMap_getQualifiedItem(map, NULL, NULL, NULL);
6957     ok( hr == E_INVALIDARG, "ret %08x\n", hr);
6958
6959     node = (void*)0xdeadbeef;
6960     hr = IXMLDOMNamedNodeMap_getQualifiedItem(map, NULL, NULL, &node);
6961     ok( hr == E_INVALIDARG, "ret %08x\n", hr);
6962     ok( node == (void*)0xdeadbeef, "got %p\n", node);
6963
6964     hr = IXMLDOMNamedNodeMap_getQualifiedItem(map, _bstr_("id"), NULL, NULL);
6965     ok( hr == E_INVALIDARG, "ret %08x\n", hr);
6966
6967     hr = IXMLDOMNamedNodeMap_getQualifiedItem(map, _bstr_("id"), NULL, &node);
6968     ok( hr == S_OK, "ret %08x\n", hr);
6969     IXMLDOMNode_Release(node);
6970
6971     IXMLDOMNamedNodeMap_Release( map );
6972     IXMLDOMElement_Release( element );
6973     IXMLDOMDocument_Release( doc );
6974     free_bstrs();
6975 }
6976
6977 static void test_removeQualifiedItem(void)
6978 {
6979     IXMLDOMDocument *doc;
6980     IXMLDOMElement *element;
6981     IXMLDOMNode *pr_node, *node;
6982     IXMLDOMNodeList *root_list;
6983     IXMLDOMNamedNodeMap *map;
6984     VARIANT_BOOL b;
6985     BSTR str;
6986     LONG len;
6987     HRESULT hr;
6988
6989     doc = create_document(&IID_IXMLDOMDocument);
6990     if (!doc) return;
6991
6992     str = SysAllocString( szComplete4 );
6993     hr = IXMLDOMDocument_loadXML( doc, str, &b );
6994     ok( hr == S_OK, "loadXML failed\n");
6995     ok( b == VARIANT_TRUE, "failed to load XML string\n");
6996     SysFreeString( str );
6997
6998     hr = IXMLDOMDocument_get_documentElement(doc, &element);
6999     ok( hr == S_OK, "ret %08x\n", hr);
7000
7001     hr = IXMLDOMElement_get_childNodes(element, &root_list);
7002     ok( hr == S_OK, "ret %08x\n", hr);
7003
7004     hr = IXMLDOMNodeList_get_item(root_list, 1, &pr_node);
7005     ok( hr == S_OK, "ret %08x\n", hr);
7006     IXMLDOMNodeList_Release(root_list);
7007
7008     hr = IXMLDOMNode_get_attributes(pr_node, &map);
7009     ok( hr == S_OK, "ret %08x\n", hr);
7010     IXMLDOMNode_Release(pr_node);
7011
7012     hr = IXMLDOMNamedNodeMap_get_length(map, &len);
7013     ok( hr == S_OK, "ret %08x\n", hr);
7014     ok( len == 3, "length %d\n", len);
7015
7016     hr = IXMLDOMNamedNodeMap_removeQualifiedItem(map, NULL, NULL, NULL);
7017     ok( hr == E_INVALIDARG, "ret %08x\n", hr);
7018
7019     node = (void*)0xdeadbeef;
7020     hr = IXMLDOMNamedNodeMap_removeQualifiedItem(map, NULL, NULL, &node);
7021     ok( hr == E_INVALIDARG, "ret %08x\n", hr);
7022     ok( node == (void*)0xdeadbeef, "got %p\n", node);
7023
7024     /* out pointer is optional */
7025     hr = IXMLDOMNamedNodeMap_removeQualifiedItem(map, _bstr_("id"), NULL, NULL);
7026     ok( hr == S_OK, "ret %08x\n", hr);
7027
7028     /* already removed */
7029     hr = IXMLDOMNamedNodeMap_removeQualifiedItem(map, _bstr_("id"), NULL, NULL);
7030     ok( hr == S_FALSE, "ret %08x\n", hr);
7031
7032     hr = IXMLDOMNamedNodeMap_removeQualifiedItem(map, _bstr_("vr"), NULL, &node);
7033     ok( hr == S_OK, "ret %08x\n", hr);
7034     IXMLDOMNode_Release(node);
7035
7036     IXMLDOMNamedNodeMap_Release( map );
7037     IXMLDOMElement_Release( element );
7038     IXMLDOMDocument_Release( doc );
7039     free_bstrs();
7040 }
7041
7042 #define check_default_props(doc) _check_default_props(__LINE__, doc)
7043 static inline void _check_default_props(int line, IXMLDOMDocument2* doc)
7044 {
7045     VARIANT_BOOL b;
7046     VARIANT var;
7047     HRESULT hr;
7048
7049     VariantInit(&var);
7050     helper_ole_check(IXMLDOMDocument2_getProperty(doc, _bstr_("SelectionLanguage"), &var));
7051     ok_(__FILE__, line)(lstrcmpW(V_BSTR(&var), _bstr_("XSLPattern")) == 0, "expected XSLPattern\n");
7052     VariantClear(&var);
7053
7054     helper_ole_check(IXMLDOMDocument2_getProperty(doc, _bstr_("SelectionNamespaces"), &var));
7055     ok_(__FILE__, line)(lstrcmpW(V_BSTR(&var), _bstr_("")) == 0, "expected empty string\n");
7056     VariantClear(&var);
7057
7058     helper_ole_check(IXMLDOMDocument2_get_preserveWhiteSpace(doc, &b));
7059     ok_(__FILE__, line)(b == VARIANT_FALSE, "expected FALSE\n");
7060
7061     hr = IXMLDOMDocument2_get_schemas(doc, &var);
7062     ok_(__FILE__, line)(hr == S_FALSE, "got %08x\n", hr);
7063     VariantClear(&var);
7064 }
7065
7066 #define check_set_props(doc) _check_set_props(__LINE__, doc)
7067 static inline void _check_set_props(int line, IXMLDOMDocument2* doc)
7068 {
7069     VARIANT_BOOL b;
7070     VARIANT var;
7071
7072     VariantInit(&var);
7073     helper_ole_check(IXMLDOMDocument2_getProperty(doc, _bstr_("SelectionLanguage"), &var));
7074     ok_(__FILE__, line)(lstrcmpW(V_BSTR(&var), _bstr_("XPath")) == 0, "expected XPath\n");
7075     VariantClear(&var);
7076
7077     helper_ole_check(IXMLDOMDocument2_getProperty(doc, _bstr_("SelectionNamespaces"), &var));
7078     ok_(__FILE__, line)(lstrcmpW(V_BSTR(&var), _bstr_("xmlns:wi=\'www.winehq.org\'")) == 0, "got %s\n", wine_dbgstr_w(V_BSTR(&var)));
7079     VariantClear(&var);
7080
7081     helper_ole_check(IXMLDOMDocument2_get_preserveWhiteSpace(doc, &b));
7082     ok_(__FILE__, line)(b == VARIANT_TRUE, "expected TRUE\n");
7083
7084     helper_ole_check(IXMLDOMDocument2_get_schemas(doc, &var));
7085     ok_(__FILE__, line)(V_VT(&var) != VT_NULL, "expected pointer\n");
7086     VariantClear(&var);
7087 }
7088
7089 #define set_props(doc, cache) _set_props(__LINE__, doc, cache)
7090 static inline void _set_props(int line, IXMLDOMDocument2* doc, IXMLDOMSchemaCollection* cache)
7091 {
7092     VARIANT var;
7093
7094     VariantInit(&var);
7095     helper_ole_check(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionLanguage"), _variantbstr_("XPath")));
7096     helper_ole_check(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionNamespaces"), _variantbstr_("xmlns:wi=\'www.winehq.org\'")));
7097     helper_ole_check(IXMLDOMDocument2_put_preserveWhiteSpace(doc, VARIANT_TRUE));
7098     V_VT(&var) = VT_DISPATCH;
7099     V_DISPATCH(&var) = NULL;
7100     helper_ole_check(IXMLDOMSchemaCollection_QueryInterface(cache, &IID_IDispatch, (void**)&V_DISPATCH(&var)));
7101     ok_(__FILE__, line)(V_DISPATCH(&var) != NULL, "expected pointer\n");
7102     helper_ole_check(IXMLDOMDocument2_putref_schemas(doc, var));
7103     VariantClear(&var);
7104 }
7105
7106 #define unset_props(doc) _unset_props(__LINE__, doc)
7107 static inline void _unset_props(int line, IXMLDOMDocument2* doc)
7108 {
7109     VARIANT var;
7110
7111     VariantInit(&var);
7112     helper_ole_check(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionLanguage"), _variantbstr_("XSLPattern")));
7113     helper_ole_check(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionNamespaces"), _variantbstr_("")));
7114     helper_ole_check(IXMLDOMDocument2_put_preserveWhiteSpace(doc, VARIANT_FALSE));
7115     V_VT(&var) = VT_NULL;
7116     helper_ole_check(IXMLDOMDocument2_putref_schemas(doc, var));
7117     VariantClear(&var);
7118 }
7119
7120 static void test_get_ownerDocument(void)
7121 {
7122     IXMLDOMDocument *doc1, *doc2, *doc3;
7123     IXMLDOMDocument2 *doc, *doc_owner;
7124     IXMLDOMNode *node;
7125     IXMLDOMSchemaCollection *cache;
7126     VARIANT_BOOL b;
7127     VARIANT var;
7128     BSTR str;
7129
7130     doc = create_document(&IID_IXMLDOMDocument2);
7131     cache = create_cache(&IID_IXMLDOMSchemaCollection);
7132     if (!doc || !cache)
7133     {
7134         if (doc) IXMLDOMDocument2_Release(doc);
7135         if (cache) IXMLDOMSchemaCollection_Release(cache);
7136         return;
7137     }
7138
7139     VariantInit(&var);
7140
7141     str = SysAllocString(szComplete4);
7142     ole_check(IXMLDOMDocument_loadXML(doc, str, &b));
7143     ok(b == VARIANT_TRUE, "failed to load XML string\n");
7144     SysFreeString(str);
7145
7146     check_default_props(doc);
7147
7148     /* set properties and check that new instances use them */
7149     set_props(doc, cache);
7150     check_set_props(doc);
7151
7152     ole_check(IXMLDOMDocument2_get_firstChild(doc, &node));
7153     ole_check(IXMLDOMNode_get_ownerDocument(node, &doc1));
7154
7155     /* new interface keeps props */
7156     ole_check(IXMLDOMDocument_QueryInterface(doc1, &IID_IXMLDOMDocument2, (void**)&doc_owner));
7157     ok( doc_owner != doc, "got %p, doc %p\n", doc_owner, doc);
7158     check_set_props(doc_owner);
7159     IXMLDOMDocument2_Release(doc_owner);
7160
7161     ole_check(IXMLDOMNode_get_ownerDocument(node, &doc2));
7162     IXMLDOMNode_Release(node);
7163
7164     ok(doc1 != doc2, "got %p, expected %p. original %p\n", doc2, doc1, doc);
7165
7166     /* reload */
7167     str = SysAllocString(szComplete4);
7168     ole_check(IXMLDOMDocument2_loadXML(doc, str, &b));
7169     ok(b == VARIANT_TRUE, "failed to load XML string\n");
7170     SysFreeString(str);
7171
7172     /* properties retained even after reload */
7173     check_set_props(doc);
7174
7175     ole_check(IXMLDOMDocument2_get_firstChild(doc, &node));
7176     ole_check(IXMLDOMNode_get_ownerDocument(node, &doc3));
7177     IXMLDOMNode_Release(node);
7178
7179     ole_check(IXMLDOMDocument_QueryInterface(doc3, &IID_IXMLDOMDocument2, (void**)&doc_owner));
7180     ok(doc3 != doc1 && doc3 != doc2 && doc_owner != doc, "got %p, (%p, %p, %p)\n", doc3, doc, doc1, doc2);
7181     check_set_props(doc_owner);
7182
7183     /* changing properties for one instance changes them for all */
7184     unset_props(doc_owner);
7185     check_default_props(doc_owner);
7186     check_default_props(doc);
7187
7188     IXMLDOMSchemaCollection_Release(cache);
7189     IXMLDOMDocument_Release(doc1);
7190     IXMLDOMDocument_Release(doc2);
7191     IXMLDOMDocument_Release(doc3);
7192     IXMLDOMDocument2_Release(doc);
7193     IXMLDOMDocument2_Release(doc_owner);
7194     free_bstrs();
7195 }
7196
7197 static void test_setAttributeNode(void)
7198 {
7199     IXMLDOMDocument *doc, *doc2;
7200     IXMLDOMElement *elem;
7201     IXMLDOMAttribute *attr, *attr2, *ret_attr;
7202     VARIANT_BOOL b;
7203     HRESULT hr;
7204     BSTR str;
7205
7206     doc = create_document(&IID_IXMLDOMDocument);
7207     if (!doc) return;
7208
7209     str = SysAllocString( szComplete4 );
7210     hr = IXMLDOMDocument2_loadXML( doc, str, &b );
7211     ok( hr == S_OK, "loadXML failed\n");
7212     ok( b == VARIANT_TRUE, "failed to load XML string\n");
7213     SysFreeString( str );
7214
7215     hr = IXMLDOMDocument_get_documentElement(doc, &elem);
7216     ok( hr == S_OK, "got 0x%08x\n", hr);
7217
7218     ret_attr = (void*)0xdeadbeef;
7219     hr = IXMLDOMElement_setAttributeNode(elem, NULL, &ret_attr);
7220     ok( hr == E_INVALIDARG, "got 0x%08x\n", hr);
7221     ok( ret_attr == (void*)0xdeadbeef, "got %p\n", ret_attr);
7222
7223     hr = IXMLDOMDocument_createAttribute(doc, _bstr_("attr"), &attr);
7224     ok( hr == S_OK, "got 0x%08x\n", hr);
7225
7226     ret_attr = (void*)0xdeadbeef;
7227     hr = IXMLDOMElement_setAttributeNode(elem, attr, &ret_attr);
7228     ok( hr == S_OK, "got 0x%08x\n", hr);
7229     ok( ret_attr == NULL, "got %p\n", ret_attr);
7230
7231     attr2 = NULL;
7232     hr = IXMLDOMElement_getAttributeNode(elem, _bstr_("attr"), &attr2);
7233     ok( hr == S_OK, "got 0x%08x\n", hr);
7234     IXMLDOMAttribute_Release(attr2);
7235
7236     /* try to add it another time */
7237     ret_attr = (void*)0xdeadbeef;
7238     hr = IXMLDOMElement_setAttributeNode(elem, attr, &ret_attr);
7239     todo_wine ok( hr == E_FAIL, "got 0x%08x\n", hr);
7240     todo_wine ok( ret_attr == (void*)0xdeadbeef, "got %p\n", ret_attr);
7241
7242     IXMLDOMElement_Release(elem);
7243
7244     /* initialy used element is released, attribute still 'has' a container */
7245     hr = IXMLDOMDocument_get_documentElement(doc, &elem);
7246     ok( hr == S_OK, "got 0x%08x\n", hr);
7247     ret_attr = (void*)0xdeadbeef;
7248     hr = IXMLDOMElement_setAttributeNode(elem, attr, &ret_attr);
7249     todo_wine ok( hr == E_FAIL, "got 0x%08x\n", hr);
7250     todo_wine ok( ret_attr == (void*)0xdeadbeef, "got %p\n", ret_attr);
7251     IXMLDOMElement_Release(elem);
7252
7253     /* add attribute already attached to another document */
7254     doc2 = create_document(&IID_IXMLDOMDocument);
7255
7256     str = SysAllocString( szComplete4 );
7257     hr = IXMLDOMDocument_loadXML( doc2, str, &b );
7258     ok( hr == S_OK, "loadXML failed\n");
7259     ok( b == VARIANT_TRUE, "failed to load XML string\n");
7260     SysFreeString( str );
7261
7262     hr = IXMLDOMDocument_get_documentElement(doc2, &elem);
7263     ok( hr == S_OK, "got 0x%08x\n", hr);
7264     hr = IXMLDOMElement_setAttributeNode(elem, attr, NULL);
7265     todo_wine ok( hr == E_FAIL, "got 0x%08x\n", hr);
7266     IXMLDOMElement_Release(elem);
7267
7268     IXMLDOMDocument_Release(doc2);
7269
7270     IXMLDOMAttribute_Release(attr);
7271     IXMLDOMDocument_Release(doc);
7272     free_bstrs();
7273 }
7274
7275 static void test_put_dataType(void)
7276 {
7277     IXMLDOMCDATASection *cdata;
7278     IXMLDOMDocument *doc;
7279     VARIANT_BOOL b;
7280     HRESULT hr;
7281     BSTR str;
7282
7283     doc = create_document(&IID_IXMLDOMDocument);
7284     if (!doc) return;
7285
7286     str = SysAllocString( szComplete4 );
7287     hr = IXMLDOMDocument_loadXML( doc, str, &b );
7288     ok( hr == S_OK, "loadXML failed\n");
7289     ok( b == VARIANT_TRUE, "failed to load XML string\n");
7290     SysFreeString( str );
7291
7292     hr = IXMLDOMDocument_createCDATASection(doc, _bstr_("test"), &cdata);
7293     ok( hr == S_OK, "got 0x%08x\n", hr);
7294     hr = IXMLDOMCDATASection_put_dataType(cdata, _bstr_("number"));
7295     ok( hr == E_FAIL, "got 0x%08x\n", hr);
7296     hr = IXMLDOMCDATASection_put_dataType(cdata, _bstr_("string"));
7297     ok( hr == E_FAIL, "got 0x%08x\n", hr);
7298     IXMLDOMCDATASection_Release(cdata);
7299
7300     IXMLDOMDocument_Release(doc);
7301     free_bstrs();
7302 }
7303
7304 static void test_createNode(void)
7305 {
7306     IXMLDOMDocument *doc;
7307     IXMLDOMElement *elem;
7308     IXMLDOMNode *node;
7309     VARIANT v, var;
7310     BSTR prefix, str;
7311     HRESULT hr;
7312
7313     doc = create_document(&IID_IXMLDOMDocument);
7314     if (!doc) return;
7315
7316     /* NODE_ELEMENT nodes */
7317     /* 1. specified namespace */
7318     V_VT(&v) = VT_I4;
7319     V_I4(&v) = NODE_ELEMENT;
7320
7321     hr = IXMLDOMDocument_createNode(doc, v, _bstr_("ns1:test"), _bstr_("http://winehq.org"), &node);
7322     ok( hr == S_OK, "got 0x%08x\n", hr);
7323     prefix = NULL;
7324     hr = IXMLDOMNode_get_prefix(node, &prefix);
7325     ok( hr == S_OK, "got 0x%08x\n", hr);
7326     ok(lstrcmpW(prefix, _bstr_("ns1")) == 0, "wrong prefix\n");
7327     SysFreeString(prefix);
7328     IXMLDOMNode_Release(node);
7329
7330     /* 2. default namespace */
7331     hr = IXMLDOMDocument_createNode(doc, v, _bstr_("test"), _bstr_("http://winehq.org/default"), &node);
7332     ok( hr == S_OK, "got 0x%08x\n", hr);
7333     prefix = (void*)0xdeadbeef;
7334     hr = IXMLDOMNode_get_prefix(node, &prefix);
7335     ok( hr == S_FALSE, "got 0x%08x\n", hr);
7336     ok(prefix == 0, "expected empty prefix, got %p\n", prefix);
7337
7338     hr = IXMLDOMNode_QueryInterface(node, &IID_IXMLDOMElement, (void**)&elem);
7339     ok( hr == S_OK, "got 0x%08x\n", hr);
7340
7341     V_VT(&var) = VT_BSTR;
7342     hr = IXMLDOMElement_getAttribute(elem, _bstr_("xmlns"), &var);
7343     ok( hr == S_FALSE, "got 0x%08x\n", hr);
7344     ok( V_VT(&var) == VT_NULL, "got %d\n", V_VT(&var));
7345
7346     str = NULL;
7347     hr = IXMLDOMElement_get_namespaceURI(elem, &str);
7348     ok( hr == S_OK, "got 0x%08x\n", hr);
7349     ok( lstrcmpW(str, _bstr_("http://winehq.org/default")) == 0, "expected default namespace\n");
7350     SysFreeString(str);
7351
7352     IXMLDOMElement_Release(elem);
7353     IXMLDOMNode_Release(node);
7354
7355     IXMLDOMDocument_Release(doc);
7356     free_bstrs();
7357 }
7358
7359 static void test_get_prefix(void)
7360 {
7361     IXMLDOMDocumentFragment *fragment;
7362     IXMLDOMCDATASection *cdata;
7363     IXMLDOMElement *element;
7364     IXMLDOMComment *comment;
7365     IXMLDOMDocument *doc;
7366     HRESULT hr;
7367     BSTR str;
7368
7369     doc = create_document(&IID_IXMLDOMDocument);
7370     if (!doc) return;
7371
7372     /* nodes that can't support prefix */
7373     /* 1. document */
7374     str = (void*)0xdeadbeef;
7375     hr = IXMLDOMDocument_get_prefix(doc, &str);
7376     ok( hr == S_FALSE, "got 0x%08x\n", hr);
7377     ok( str == 0, "got %p\n", str);
7378
7379     hr = IXMLDOMDocument_get_prefix(doc, NULL);
7380     ok( hr == E_INVALIDARG, "got 0x%08x\n", hr);
7381
7382     /* 2. cdata */
7383     hr = IXMLDOMDocument_createCDATASection(doc, NULL, &cdata);
7384     ok(hr == S_OK, "got %08x\n", hr );
7385
7386     str = (void*)0xdeadbeef;
7387     hr = IXMLDOMCDATASection_get_prefix(cdata, &str);
7388     ok(hr == S_FALSE, "got %08x\n", hr);
7389     ok( str == 0, "got %p\n", str);
7390
7391     hr = IXMLDOMCDATASection_get_prefix(cdata, NULL);
7392     ok(hr == E_INVALIDARG, "got %08x\n", hr);
7393     IXMLDOMCDATASection_Release(cdata);
7394
7395     /* 3. comment */
7396     hr = IXMLDOMDocument_createComment(doc, NULL, &comment);
7397     ok(hr == S_OK, "got %08x\n", hr );
7398
7399     str = (void*)0xdeadbeef;
7400     hr = IXMLDOMComment_get_prefix(comment, &str);
7401     ok(hr == S_FALSE, "got %08x\n", hr);
7402     ok( str == 0, "got %p\n", str);
7403
7404     hr = IXMLDOMComment_get_prefix(comment, NULL);
7405     ok(hr == E_INVALIDARG, "got %08x\n", hr);
7406     IXMLDOMComment_Release(comment);
7407
7408     /* 4. fragment */
7409     hr = IXMLDOMDocument_createDocumentFragment(doc, &fragment);
7410     ok(hr == S_OK, "got %08x\n", hr );
7411
7412     str = (void*)0xdeadbeef;
7413     hr = IXMLDOMDocumentFragment_get_prefix(fragment, &str);
7414     ok(hr == S_FALSE, "got %08x\n", hr);
7415     ok( str == 0, "got %p\n", str);
7416
7417     hr = IXMLDOMDocumentFragment_get_prefix(fragment, NULL);
7418     ok(hr == E_INVALIDARG, "got %08x\n", hr);
7419     IXMLDOMDocumentFragment_Release(fragment);
7420
7421     /* no prefix */
7422     hr = IXMLDOMDocument_createElement(doc, _bstr_("elem"), &element);
7423     ok( hr == S_OK, "got 0x%08x\n", hr);
7424
7425     hr = IXMLDOMElement_get_prefix(element, NULL);
7426     ok( hr == E_INVALIDARG, "got 0x%08x\n", hr);
7427
7428     str = (void*)0xdeadbeef;
7429     hr = IXMLDOMElement_get_prefix(element, &str);
7430     ok( hr == S_FALSE, "got 0x%08x\n", hr);
7431     ok( str == 0, "got %p\n", str);
7432
7433     IXMLDOMElement_Release(element);
7434
7435     /* with prefix */
7436     hr = IXMLDOMDocument_createElement(doc, _bstr_("a:elem"), &element);
7437     ok( hr == S_OK, "got 0x%08x\n", hr);
7438
7439     str = (void*)0xdeadbeef;
7440     hr = IXMLDOMElement_get_prefix(element, &str);
7441     ok( hr == S_OK, "got 0x%08x\n", hr);
7442     ok( lstrcmpW(str, _bstr_("a")) == 0, "expected prefix \"a\"\n");
7443     SysFreeString(str);
7444
7445     str = (void*)0xdeadbeef;
7446     hr = IXMLDOMElement_get_namespaceURI(element, &str);
7447     ok( hr == S_FALSE, "got 0x%08x\n", hr);
7448     ok( str == 0, "got %p\n", str);
7449
7450     IXMLDOMElement_Release(element);
7451
7452     IXMLDOMDocument_Release(doc);
7453     free_bstrs();
7454 }
7455
7456 static void test_selectSingleNode(void)
7457 {
7458     IXMLDOMDocument *doc;
7459     IXMLDOMNodeList *list;
7460     IXMLDOMNode *node;
7461     VARIANT_BOOL b;
7462     HRESULT hr;
7463     LONG len;
7464     BSTR str;
7465
7466     doc = create_document(&IID_IXMLDOMDocument);
7467     if (!doc) return;
7468
7469     hr = IXMLDOMDocument_selectSingleNode(doc, NULL, NULL);
7470     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
7471
7472     hr = IXMLDOMDocument_selectNodes(doc, NULL, NULL);
7473     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
7474
7475     str = SysAllocString( szComplete4 );
7476     hr = IXMLDOMDocument_loadXML( doc, str, &b );
7477     ok( hr == S_OK, "loadXML failed\n");
7478     ok( b == VARIANT_TRUE, "failed to load XML string\n");
7479     SysFreeString( str );
7480
7481     hr = IXMLDOMDocument_selectSingleNode(doc, NULL, NULL);
7482     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
7483
7484     hr = IXMLDOMDocument_selectNodes(doc, NULL, NULL);
7485     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
7486
7487     hr = IXMLDOMDocument_selectSingleNode(doc, _bstr_("lc"), NULL);
7488     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
7489
7490     hr = IXMLDOMDocument_selectNodes(doc, _bstr_("lc"), NULL);
7491     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
7492
7493     hr = IXMLDOMDocument_selectSingleNode(doc, _bstr_("lc"), &node);
7494     ok(hr == S_OK, "got 0x%08x\n", hr);
7495     IXMLDOMNode_Release(node);
7496
7497     hr = IXMLDOMDocument_selectNodes(doc, _bstr_("lc"), &list);
7498     ok(hr == S_OK, "got 0x%08x\n", hr);
7499     IXMLDOMNodeList_Release(list);
7500
7501     list = (void*)0xdeadbeef;
7502     hr = IXMLDOMDocument_selectNodes(doc, NULL, &list);
7503     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
7504     ok(list == (void*)0xdeadbeef, "got %p\n", list);
7505
7506     node = (void*)0xdeadbeef;
7507     hr = IXMLDOMDocument_selectSingleNode(doc, _bstr_("nonexistent"), &node);
7508     ok(hr == S_FALSE, "got 0x%08x\n", hr);
7509     ok(node == 0, "got %p\n", node);
7510
7511     list = (void*)0xdeadbeef;
7512     hr = IXMLDOMDocument_selectNodes(doc, _bstr_("nonexistent"), &list);
7513     ok(hr == S_OK, "got 0x%08x\n", hr);
7514     len = 1;
7515     hr = IXMLDOMNodeList_get_length(list, &len);
7516     ok(hr == S_OK, "got 0x%08x\n", hr);
7517     ok(len == 0, "got %d\n", len);
7518     IXMLDOMNodeList_Release(list);
7519
7520     IXMLDOMDocument_Release(doc);
7521     free_bstrs();
7522 }
7523
7524 static void test_events(void)
7525 {
7526     IConnectionPointContainer *conn;
7527     IConnectionPoint *point;
7528     IXMLDOMDocument *doc;
7529     HRESULT hr;
7530
7531     doc = create_document(&IID_IXMLDOMDocument);
7532     if (!doc) return;
7533
7534     hr = IXMLDOMDocument_QueryInterface(doc, &IID_IConnectionPointContainer, (void**)&conn);
7535     ok(hr == S_OK, "got 0x%08x\n", hr);
7536
7537     hr = IConnectionPointContainer_FindConnectionPoint(conn, &IID_IDispatch, &point);
7538     ok(hr == S_OK, "got 0x%08x\n", hr);
7539     IConnectionPoint_Release(point);
7540     hr = IConnectionPointContainer_FindConnectionPoint(conn, &IID_IPropertyNotifySink, &point);
7541     ok(hr == S_OK, "got 0x%08x\n", hr);
7542     IConnectionPoint_Release(point);
7543     hr = IConnectionPointContainer_FindConnectionPoint(conn, &DIID_XMLDOMDocumentEvents, &point);
7544     ok(hr == S_OK, "got 0x%08x\n", hr);
7545     IConnectionPoint_Release(point);
7546
7547     IConnectionPointContainer_Release(conn);
7548
7549     IXMLDOMDocument_Release(doc);
7550 }
7551
7552 static void test_createProcessingInstruction(void)
7553 {
7554     static const WCHAR bodyW[] = {'t','e','s','t',0};
7555     IXMLDOMProcessingInstruction *pi;
7556     IXMLDOMDocument *doc;
7557     WCHAR buff[10];
7558     HRESULT hr;
7559
7560     doc = create_document(&IID_IXMLDOMDocument);
7561     if (!doc) return;
7562
7563     /* test for BSTR handling, pass broken BSTR */
7564     memcpy(&buff[2], bodyW, sizeof(bodyW));
7565     /* just a big length */
7566     *(DWORD*)buff = 0xf0f0;
7567     hr = IXMLDOMDocument_createProcessingInstruction(doc, _bstr_("test"), &buff[2], &pi);
7568     ok(hr == S_OK, "got 0x%08x\n", hr);
7569
7570     IXMLDOMProcessingInstruction_Release(pi);
7571     IXMLDOMDocument_Release(doc);
7572 }
7573
7574 static void test_put_nodeTypedValue(void)
7575 {
7576     IXMLDOMDocument *doc;
7577     IXMLDOMElement *elem;
7578     VARIANT type;
7579     HRESULT hr;
7580
7581     doc = create_document(&IID_IXMLDOMDocument);
7582     if (!doc) return;
7583
7584     hr = IXMLDOMDocument_createElement(doc, _bstr_("Element"), &elem);
7585     ok(hr == S_OK, "got 0x%08x\n", hr);
7586
7587     V_VT(&type) = VT_EMPTY;
7588     hr = IXMLDOMElement_get_dataType(elem, &type);
7589     ok(hr == S_FALSE, "got 0x%08x\n", hr);
7590     ok(V_VT(&type) == VT_NULL, "got %d, expected VT_NULL\n", V_VT(&type));
7591
7592     /* set typed value for untyped node */
7593     V_VT(&type) = VT_I1;
7594     V_I1(&type) = 1;
7595     hr = IXMLDOMElement_put_nodeTypedValue(elem, type);
7596     ok(hr == S_OK, "got 0x%08x\n", hr);
7597
7598     V_VT(&type) = VT_EMPTY;
7599     hr = IXMLDOMElement_get_dataType(elem, &type);
7600     ok(hr == S_FALSE, "got 0x%08x\n", hr);
7601     ok(V_VT(&type) == VT_NULL, "got %d, expected VT_NULL\n", V_VT(&type));
7602
7603     /* no type info stored */
7604     V_VT(&type) = VT_EMPTY;
7605     hr = IXMLDOMElement_get_nodeTypedValue(elem, &type);
7606     ok(hr == S_OK, "got 0x%08x\n", hr);
7607     ok(V_VT(&type) == VT_BSTR, "got %d, expected VT_BSTR\n", V_VT(&type));
7608     ok(memcmp(V_BSTR(&type), _bstr_("1"), 2*sizeof(WCHAR)) == 0,
7609        "got %s, expected \"1\"\n", wine_dbgstr_w(V_BSTR(&type)));
7610     VariantClear(&type);
7611
7612     IXMLDOMElement_Release(elem);
7613     IXMLDOMDocument_Release(doc);
7614     free_bstrs();
7615 }
7616
7617 static void test_get_xml(void)
7618 {
7619     static const char xmlA[] = "<?xml version=\"1.0\" encoding=\"UTF-16\"?>\r\n<a>test</a>\r\n";
7620     IXMLDOMProcessingInstruction *pi;
7621     IXMLDOMNode *first;
7622     IXMLDOMDocument *doc;
7623     VARIANT_BOOL b;
7624     VARIANT v;
7625     BSTR xml;
7626     HRESULT hr;
7627
7628     doc = create_document(&IID_IXMLDOMDocument);
7629     if (!doc) return;
7630
7631     b = VARIANT_TRUE;
7632     hr = IXMLDOMDocument_loadXML( doc, _bstr_("<a>test</a>"), &b );
7633     ok(hr == S_OK, "got 0x%08x\n", hr);
7634     ok( b == VARIANT_TRUE, "got %d\n", b);
7635
7636     hr = IXMLDOMDocument_createProcessingInstruction(doc, _bstr_("xml"),
7637                              _bstr_("version=\"1.0\" encoding=\"UTF-16\""), &pi);
7638     ok(hr == S_OK, "got 0x%08x\n", hr);
7639
7640     hr = IXMLDOMDocument_get_firstChild(doc, &first);
7641     ok(hr == S_OK, "got 0x%08x\n", hr);
7642
7643     V_UNKNOWN(&v) = (IUnknown*)first;
7644     V_VT(&v) = VT_UNKNOWN;
7645
7646     hr = IXMLDOMDocument_insertBefore(doc, (IXMLDOMNode*)pi, v, NULL);
7647     ok(hr == S_OK, "got 0x%08x\n", hr);
7648
7649     IXMLDOMProcessingInstruction_Release(pi);
7650     IXMLDOMNode_Release(first);
7651
7652     hr = IXMLDOMDocument_get_xml(doc, &xml);
7653     ok(hr == S_OK, "got 0x%08x\n", hr);
7654
7655     ok(memcmp(xml, _bstr_(xmlA), sizeof(xmlA)*sizeof(WCHAR)) == 0,
7656         "got %s, expected %s\n", wine_dbgstr_w(xml), xmlA);
7657     SysFreeString(xml);
7658
7659     IXMLDOMDocument_Release(doc);
7660     free_bstrs();
7661 }
7662
7663 START_TEST(domdoc)
7664 {
7665     IXMLDOMDocument *doc;
7666     HRESULT hr;
7667
7668     hr = CoInitialize( NULL );
7669     ok( hr == S_OK, "failed to init com\n");
7670     if (hr != S_OK)
7671         return;
7672
7673     test_XMLHTTP();
7674
7675     hr = CoCreateInstance( &CLSID_DOMDocument, NULL, CLSCTX_INPROC_SERVER, &IID_IXMLDOMDocument, (void**)&doc );
7676     if (hr != S_OK)
7677     {
7678         win_skip("IXMLDOMDocument is not available (0x%08x)\n", hr);
7679         return;
7680     }
7681
7682     IXMLDOMDocument_Release(doc);
7683
7684     test_domdoc();
7685     test_persiststreaminit();
7686     test_domnode();
7687     test_refs();
7688     test_create();
7689     test_getElementsByTagName();
7690     test_get_text();
7691     test_get_childNodes();
7692     test_get_firstChild();
7693     test_get_lastChild();
7694     test_removeChild();
7695     test_replaceChild();
7696     test_removeNamedItem();
7697     test_IXMLDOMDocument2();
7698     test_whitespace();
7699     test_XPath();
7700     test_XSLPattern();
7701     test_cloneNode();
7702     test_xmlTypes();
7703     test_nodeTypeTests();
7704     test_DocumentSaveToDocument();
7705     test_DocumentSaveToFile();
7706     test_testTransforms();
7707     test_Namespaces();
7708     test_FormattingXML();
7709     test_nodeTypedValue();
7710     test_TransformWithLoadingLocalFile();
7711     test_put_nodeValue();
7712     test_document_IObjectSafety();
7713     test_splitText();
7714     test_getQualifiedItem();
7715     test_removeQualifiedItem();
7716     test_get_ownerDocument();
7717     test_setAttributeNode();
7718     test_put_dataType();
7719     test_createNode();
7720     test_get_prefix();
7721     test_default_properties();
7722     test_selectSingleNode();
7723     test_events();
7724     test_createProcessingInstruction();
7725     test_put_nodeTypedValue();
7726     test_get_xml();
7727
7728     CoUninitialize();
7729 }