dinput: Fix printing NULL strings.
[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-2011 Adam Martinson for CodeWeavers
7  * Copyright 2010-2011 Nikolay Sivov for CodeWeavers
8  *
9  * This library is free software; you can redistribute it and/or
10  * modify it under the terms of the GNU Lesser General Public
11  * License as published by the Free Software Foundation; either
12  * version 2.1 of the License, or (at your option) any later version.
13  *
14  * This library is distributed in the hope that it will be useful,
15  * but WITHOUT ANY WARRANTY; without even the implied warranty of
16  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
17  * Lesser General Public License for more details.
18  *
19  * You should have received a copy of the GNU Lesser General Public
20  * License along with this library; if not, write to the Free Software
21  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
22  */
23
24
25 #define COBJMACROS
26 #define CONST_VTABLE
27
28 #include <stdio.h>
29 #include <assert.h>
30
31 #include "windows.h"
32
33 #include "msxml2.h"
34 #include "msxml2did.h"
35 #include "ole2.h"
36 #include "dispex.h"
37
38 #include "initguid.h"
39 #include "objsafe.h"
40 #include "mshtml.h"
41
42 #include "wine/test.h"
43
44 DEFINE_GUID(SID_SContainerDispatch, 0xb722be00, 0x4e68, 0x101b, 0xa2, 0xbc, 0x00, 0xaa, 0x00, 0x40, 0x47, 0x70);
45 DEFINE_GUID(SID_UnknownSID, 0x75dd09cb, 0x6c40, 0x11d5, 0x85, 0x43, 0x00, 0xc0, 0x4f, 0xa0, 0xfb, 0xa3);
46
47 #define DEFINE_EXPECT(func) \
48     static BOOL expect_ ## func = FALSE, called_ ## func = FALSE
49
50 #define SET_EXPECT(func) \
51     expect_ ## func = TRUE
52
53 #define CHECK_EXPECT2(func) \
54     do { \
55         ok(expect_ ##func, "unexpected call " #func "\n"); \
56         called_ ## func = TRUE; \
57     }while(0)
58
59 #define CHECK_CALLED(func) \
60     do { \
61         ok(called_ ## func, "expected " #func "\n"); \
62         expect_ ## func = called_ ## func = FALSE; \
63     }while(0)
64
65 static const char *debugstr_guid(REFIID riid)
66 {
67     static char buf[50];
68
69     if(!riid)
70         return "(null)";
71
72     sprintf(buf, "{%08X-%04X-%04X-%02X%02X-%02X%02X%02X%02X%02X%02X}",
73             riid->Data1, riid->Data2, riid->Data3, riid->Data4[0],
74             riid->Data4[1], riid->Data4[2], riid->Data4[3], riid->Data4[4],
75             riid->Data4[5], riid->Data4[6], riid->Data4[7]);
76
77     return buf;
78 }
79
80 static int g_unexpectedcall, g_expectedcall;
81
82 typedef struct
83 {
84     IDispatch IDispatch_iface;
85     LONG ref;
86 } dispevent;
87
88 static inline dispevent *impl_from_IDispatch( IDispatch *iface )
89 {
90     return CONTAINING_RECORD(iface, dispevent, IDispatch_iface);
91 }
92
93 static HRESULT WINAPI dispevent_QueryInterface(IDispatch *iface, REFIID riid, void **ppvObject)
94 {
95     *ppvObject = NULL;
96
97     if ( IsEqualGUID( riid, &IID_IDispatch) ||
98          IsEqualGUID( riid, &IID_IUnknown) )
99     {
100         *ppvObject = iface;
101     }
102     else
103         return E_NOINTERFACE;
104
105     IDispatch_AddRef( iface );
106
107     return S_OK;
108 }
109
110 static ULONG WINAPI dispevent_AddRef(IDispatch *iface)
111 {
112     dispevent *This = impl_from_IDispatch( iface );
113     return InterlockedIncrement( &This->ref );
114 }
115
116 static ULONG WINAPI dispevent_Release(IDispatch *iface)
117 {
118     dispevent *This = impl_from_IDispatch( iface );
119     ULONG ref = InterlockedDecrement( &This->ref );
120
121     if (ref == 0)
122         HeapFree(GetProcessHeap(), 0, This);
123
124     return ref;
125 }
126
127 static HRESULT WINAPI dispevent_GetTypeInfoCount(IDispatch *iface, UINT *pctinfo)
128 {
129     g_unexpectedcall++;
130     *pctinfo = 0;
131     return S_OK;
132 }
133
134 static HRESULT WINAPI dispevent_GetTypeInfo(IDispatch *iface, UINT iTInfo,
135         LCID lcid, ITypeInfo **ppTInfo)
136 {
137     g_unexpectedcall++;
138     return S_OK;
139 }
140
141 static HRESULT WINAPI dispevent_GetIDsOfNames(IDispatch *iface, REFIID riid,
142         LPOLESTR *rgszNames, UINT cNames, LCID lcid, DISPID *rgDispId)
143 {
144     g_unexpectedcall++;
145     return S_OK;
146 }
147
148 static HRESULT WINAPI dispevent_Invoke(IDispatch *iface, DISPID member, REFIID riid,
149         LCID lcid, WORD flags, DISPPARAMS *params, VARIANT *result,
150         EXCEPINFO *excepInfo, UINT *argErr)
151 {
152     ok(member == 0, "expected 0 member, got %d\n", member);
153     ok(lcid == LOCALE_SYSTEM_DEFAULT, "expected LOCALE_SYSTEM_DEFAULT, got lcid %x\n", lcid);
154     ok(flags == DISPATCH_METHOD, "expected DISPATCH_METHOD, got %d\n", flags);
155
156     ok(params->cArgs == 0, "got %d\n", params->cArgs);
157     ok(params->cNamedArgs == 0, "got %d\n", params->cNamedArgs);
158     ok(params->rgvarg == NULL, "got %p\n", params->rgvarg);
159     ok(params->rgdispidNamedArgs == NULL, "got %p\n", params->rgdispidNamedArgs);
160
161     ok(result == NULL, "got %p\n", result);
162     ok(excepInfo == NULL, "got %p\n", excepInfo);
163     ok(argErr == NULL, "got %p\n", argErr);
164
165     g_expectedcall++;
166     return E_FAIL;
167 }
168
169 static const IDispatchVtbl dispeventVtbl =
170 {
171     dispevent_QueryInterface,
172     dispevent_AddRef,
173     dispevent_Release,
174     dispevent_GetTypeInfoCount,
175     dispevent_GetTypeInfo,
176     dispevent_GetIDsOfNames,
177     dispevent_Invoke
178 };
179
180 static IDispatch* create_dispevent(void)
181 {
182     dispevent *event = HeapAlloc(GetProcessHeap(), 0, sizeof(*event));
183
184     event->IDispatch_iface.lpVtbl = &dispeventVtbl;
185     event->ref = 1;
186
187     return (IDispatch*)&event->IDispatch_iface;
188 }
189
190 /* object site */
191 DEFINE_EXPECT(site_qi_IServiceProvider);
192 DEFINE_EXPECT(site_qi_IXMLDOMDocument);
193 DEFINE_EXPECT(site_qi_IOleClientSite);
194
195 DEFINE_EXPECT(sp_queryservice_SID_SBindHost);
196 DEFINE_EXPECT(sp_queryservice_SID_SContainerDispatch_htmldoc2);
197 DEFINE_EXPECT(sp_queryservice_SID_secmgr_htmldoc2);
198 DEFINE_EXPECT(sp_queryservice_SID_secmgr_xmldomdoc);
199 DEFINE_EXPECT(sp_queryservice_SID_secmgr_secmgr);
200
201 DEFINE_EXPECT(htmldoc2_get_all);
202 DEFINE_EXPECT(htmldoc2_get_url);
203 DEFINE_EXPECT(collection_get_length);
204
205 typedef struct
206 {
207     IServiceProvider IServiceProvider_iface;
208 } testprov_t;
209
210 testprov_t testprov;
211
212 static HRESULT WINAPI site_QueryInterface(IUnknown *iface, REFIID riid, void **ppvObject)
213 {
214     *ppvObject = NULL;
215
216     if (IsEqualGUID(riid, &IID_IServiceProvider))
217         CHECK_EXPECT2(site_qi_IServiceProvider);
218
219     if (IsEqualGUID(riid, &IID_IXMLDOMDocument))
220         CHECK_EXPECT2(site_qi_IXMLDOMDocument);
221
222     if (IsEqualGUID(riid, &IID_IOleClientSite))
223         CHECK_EXPECT2(site_qi_IOleClientSite);
224
225     if (IsEqualGUID(riid, &IID_IUnknown))
226          *ppvObject = iface;
227     else if (IsEqualGUID(riid, &IID_IServiceProvider))
228          *ppvObject = &testprov.IServiceProvider_iface;
229
230     if (*ppvObject) IUnknown_AddRef(iface);
231
232     return *ppvObject ? S_OK : E_NOINTERFACE;
233 }
234
235 static ULONG WINAPI site_AddRef(IUnknown *iface)
236 {
237     return 2;
238 }
239
240 static ULONG WINAPI site_Release(IUnknown *iface)
241 {
242     return 1;
243 }
244
245 static const IUnknownVtbl testsiteVtbl =
246 {
247     site_QueryInterface,
248     site_AddRef,
249     site_Release
250 };
251
252 typedef struct
253 {
254     IUnknown IUnknown_iface;
255 } testsite_t;
256
257 static testsite_t testsite = { { &testsiteVtbl } };
258
259 /* test IHTMLElementCollection */
260 static HRESULT WINAPI htmlecoll_QueryInterface(IHTMLElementCollection *iface, REFIID riid, void **ppvObject)
261 {
262     ok(0, "unexpected call\n");
263     *ppvObject = NULL;
264     return E_NOINTERFACE;
265 }
266
267 static ULONG WINAPI htmlecoll_AddRef(IHTMLElementCollection *iface)
268 {
269     return 2;
270 }
271
272 static ULONG WINAPI htmlecoll_Release(IHTMLElementCollection *iface)
273 {
274     return 1;
275 }
276
277 static HRESULT WINAPI htmlecoll_GetTypeInfoCount(IHTMLElementCollection *iface, UINT *pctinfo)
278 {
279     ok(0, "unexpected call\n");
280     return E_NOTIMPL;
281 }
282
283 static HRESULT WINAPI htmlecoll_GetTypeInfo(IHTMLElementCollection *iface, UINT iTInfo,
284                                                 LCID lcid, ITypeInfo **ppTInfo)
285 {
286     ok(0, "unexpected call\n");
287     return E_NOTIMPL;
288 }
289
290 static HRESULT WINAPI htmlecoll_GetIDsOfNames(IHTMLElementCollection *iface, REFIID riid,
291                                                 LPOLESTR *rgszNames, UINT cNames,
292                                                 LCID lcid, DISPID *rgDispId)
293 {
294     ok(0, "unexpected call\n");
295     return E_NOTIMPL;
296 }
297
298 static HRESULT WINAPI htmlecoll_Invoke(IHTMLElementCollection *iface, DISPID dispIdMember,
299                             REFIID riid, LCID lcid, WORD wFlags, DISPPARAMS *pDispParams,
300                             VARIANT *pVarResult, EXCEPINFO *pExcepInfo, UINT *puArgErr)
301 {
302     ok(0, "unexpected call\n");
303     return E_NOTIMPL;
304 }
305
306 static HRESULT WINAPI htmlecoll_toString(IHTMLElementCollection *iface, BSTR *String)
307 {
308     ok(0, "unexpected call\n");
309     return E_NOTIMPL;
310 }
311
312 static HRESULT WINAPI htmlecoll_put_length(IHTMLElementCollection *iface, LONG v)
313 {
314     ok(0, "unexpected call\n");
315     return E_NOTIMPL;
316 }
317
318 static HRESULT WINAPI htmlecoll_get_length(IHTMLElementCollection *iface, LONG *v)
319 {
320     CHECK_EXPECT2(collection_get_length);
321     return E_NOTIMPL;
322 }
323
324 static HRESULT WINAPI htmlecoll_get__newEnum(IHTMLElementCollection *iface, IUnknown **p)
325 {
326     ok(0, "unexpected call\n");
327     return E_NOTIMPL;
328 }
329
330 static HRESULT WINAPI htmlecoll_item(IHTMLElementCollection *iface, VARIANT name, VARIANT index, IDispatch **pdisp)
331 {
332     ok(0, "unexpected call\n");
333     return E_NOTIMPL;
334 }
335
336 static HRESULT WINAPI htmlecoll_tags(IHTMLElementCollection *iface, VARIANT tagName, IDispatch **pdisp)
337 {
338     ok(0, "unexpected call\n");
339     return E_NOTIMPL;
340 }
341
342 static const IHTMLElementCollectionVtbl TestHTMLECollectionVtbl = {
343     htmlecoll_QueryInterface,
344     htmlecoll_AddRef,
345     htmlecoll_Release,
346     htmlecoll_GetTypeInfoCount,
347     htmlecoll_GetTypeInfo,
348     htmlecoll_GetIDsOfNames,
349     htmlecoll_Invoke,
350
351     htmlecoll_toString,
352     htmlecoll_put_length,
353     htmlecoll_get_length,
354     htmlecoll_get__newEnum,
355     htmlecoll_item,
356     htmlecoll_tags
357 };
358
359 typedef struct
360 {
361     IHTMLElementCollection IHTMLElementCollection_iface;
362 } testhtmlecoll_t;
363
364 static testhtmlecoll_t htmlecoll = { { &TestHTMLECollectionVtbl } };
365
366 /* test IHTMLDocument2 */
367 static HRESULT WINAPI htmldoc2_QueryInterface(IHTMLDocument2 *iface, REFIID riid, void **ppvObject)
368 {
369    trace("\n");
370    *ppvObject = NULL;
371    return E_NOINTERFACE;
372 }
373
374 static ULONG WINAPI htmldoc2_AddRef(IHTMLDocument2 *iface)
375 {
376     return 2;
377 }
378
379 static ULONG WINAPI htmldoc2_Release(IHTMLDocument2 *iface)
380 {
381     return 1;
382 }
383
384 static HRESULT WINAPI htmldoc2_GetTypeInfoCount(IHTMLDocument2 *iface, UINT *pctinfo)
385 {
386     ok(0, "unexpected call\n");
387     return E_NOTIMPL;
388 }
389
390 static HRESULT WINAPI htmldoc2_GetTypeInfo(IHTMLDocument2 *iface, UINT iTInfo,
391                                                 LCID lcid, ITypeInfo **ppTInfo)
392 {
393     ok(0, "unexpected call\n");
394     return E_NOTIMPL;
395 }
396
397 static HRESULT WINAPI htmldoc2_GetIDsOfNames(IHTMLDocument2 *iface, REFIID riid,
398                                                 LPOLESTR *rgszNames, UINT cNames,
399                                                 LCID lcid, DISPID *rgDispId)
400 {
401     ok(0, "unexpected call\n");
402     return E_NOTIMPL;
403 }
404
405 static HRESULT WINAPI htmldoc2_Invoke(IHTMLDocument2 *iface, DISPID dispIdMember,
406                             REFIID riid, LCID lcid, WORD wFlags, DISPPARAMS *pDispParams,
407                             VARIANT *pVarResult, EXCEPINFO *pExcepInfo, UINT *puArgErr)
408 {
409     ok(0, "unexpected call\n");
410     return E_NOTIMPL;
411 }
412
413 static HRESULT WINAPI htmldoc2_get_Script(IHTMLDocument2 *iface, IDispatch **p)
414 {
415     ok(0, "unexpected call\n");
416     return E_NOTIMPL;
417 }
418
419 static HRESULT WINAPI htmldoc2_get_all(IHTMLDocument2 *iface, IHTMLElementCollection **p)
420 {
421     CHECK_EXPECT2(htmldoc2_get_all);
422     *p = &htmlecoll.IHTMLElementCollection_iface;
423     return S_OK;
424 }
425
426 static HRESULT WINAPI htmldoc2_get_body(IHTMLDocument2 *iface, IHTMLElement **p)
427 {
428     ok(0, "unexpected call\n");
429     return E_NOTIMPL;
430 }
431
432 static HRESULT WINAPI htmldoc2_get_activeElement(IHTMLDocument2 *iface, IHTMLElement **p)
433 {
434     ok(0, "unexpected call\n");
435     return E_NOTIMPL;
436 }
437
438 static HRESULT WINAPI htmldoc2_get_images(IHTMLDocument2 *iface, IHTMLElementCollection **p)
439 {
440     ok(0, "unexpected call\n");
441     return E_NOTIMPL;
442 }
443
444 static HRESULT WINAPI htmldoc2_get_applets(IHTMLDocument2 *iface, IHTMLElementCollection **p)
445 {
446     ok(0, "unexpected call\n");
447     return E_NOTIMPL;
448 }
449
450 static HRESULT WINAPI htmldoc2_get_links(IHTMLDocument2 *iface, IHTMLElementCollection **p)
451 {
452     ok(0, "unexpected call\n");
453     return E_NOTIMPL;
454 }
455
456 static HRESULT WINAPI htmldoc2_get_forms(IHTMLDocument2 *iface, IHTMLElementCollection **p)
457 {
458     ok(0, "unexpected call\n");
459     return E_NOTIMPL;
460 }
461
462 static HRESULT WINAPI htmldoc2_get_anchors(IHTMLDocument2 *iface, IHTMLElementCollection **p)
463 {
464     ok(0, "unexpected call\n");
465     return E_NOTIMPL;
466 }
467
468 static HRESULT WINAPI htmldoc2_put_title(IHTMLDocument2 *iface, BSTR v)
469 {
470     ok(0, "unexpected call\n");
471     return E_NOTIMPL;
472 }
473
474 static HRESULT WINAPI htmldoc2_get_title(IHTMLDocument2 *iface, BSTR *p)
475 {
476     ok(0, "unexpected call\n");
477     return E_NOTIMPL;
478 }
479
480 static HRESULT WINAPI htmldoc2_get_scripts(IHTMLDocument2 *iface, IHTMLElementCollection **p)
481 {
482     ok(0, "unexpected call\n");
483     return E_NOTIMPL;
484 }
485
486 static HRESULT WINAPI htmldoc2_put_designMode(IHTMLDocument2 *iface, BSTR v)
487 {
488     ok(0, "unexpected call\n");
489     return E_NOTIMPL;
490 }
491
492 static HRESULT WINAPI htmldoc2_get_designMode(IHTMLDocument2 *iface, BSTR *p)
493 {
494     ok(0, "unexpected call\n");
495     return E_NOTIMPL;
496 }
497
498 static HRESULT WINAPI htmldoc2_get_selection(IHTMLDocument2 *iface, IHTMLSelectionObject **p)
499 {
500     ok(0, "unexpected call\n");
501     return E_NOTIMPL;
502 }
503
504 static HRESULT WINAPI htmldoc2_get_readyState(IHTMLDocument2 *iface, BSTR *p)
505 {
506     ok(0, "unexpected call\n");
507     return E_NOTIMPL;
508 }
509
510 static HRESULT WINAPI htmldoc2_get_frames(IHTMLDocument2 *iface, IHTMLFramesCollection2 **p)
511 {
512     ok(0, "unexpected call\n");
513     return E_NOTIMPL;
514 }
515
516 static HRESULT WINAPI htmldoc2_get_embeds(IHTMLDocument2 *iface, IHTMLElementCollection **p)
517 {
518     ok(0, "unexpected call\n");
519     return E_NOTIMPL;
520 }
521
522 static HRESULT WINAPI htmldoc2_get_plugins(IHTMLDocument2 *iface, IHTMLElementCollection **p)
523 {
524     ok(0, "unexpected call\n");
525     return E_NOTIMPL;
526 }
527
528 static HRESULT WINAPI htmldoc2_put_alinkColor(IHTMLDocument2 *iface, VARIANT v)
529 {
530     ok(0, "unexpected call\n");
531     return E_NOTIMPL;
532 }
533
534 static HRESULT WINAPI htmldoc2_get_alinkColor(IHTMLDocument2 *iface, VARIANT *p)
535 {
536     ok(0, "unexpected call\n");
537     return E_NOTIMPL;
538 }
539
540 static HRESULT WINAPI htmldoc2_put_bgColor(IHTMLDocument2 *iface, VARIANT v)
541 {
542     ok(0, "unexpected call\n");
543     return E_NOTIMPL;
544 }
545
546 static HRESULT WINAPI htmldoc2_get_bgColor(IHTMLDocument2 *iface, VARIANT *p)
547 {
548     ok(0, "unexpected call\n");
549     return E_NOTIMPL;
550 }
551
552 static HRESULT WINAPI htmldoc2_put_fgColor(IHTMLDocument2 *iface, VARIANT v)
553 {
554     ok(0, "unexpected call\n");
555     return E_NOTIMPL;
556 }
557
558 static HRESULT WINAPI htmldoc2_get_fgColor(IHTMLDocument2 *iface, VARIANT *p)
559 {
560     ok(0, "unexpected call\n");
561     return E_NOTIMPL;
562 }
563
564 static HRESULT WINAPI htmldoc2_put_linkColor(IHTMLDocument2 *iface, VARIANT v)
565 {
566     ok(0, "unexpected call\n");
567     return E_NOTIMPL;
568 }
569
570 static HRESULT WINAPI htmldoc2_get_linkColor(IHTMLDocument2 *iface, VARIANT *p)
571 {
572     ok(0, "unexpected call\n");
573     return E_NOTIMPL;
574 }
575
576 static HRESULT WINAPI htmldoc2_put_vlinkColor(IHTMLDocument2 *iface, VARIANT v)
577 {
578     ok(0, "unexpected call\n");
579     return E_NOTIMPL;
580 }
581
582 static HRESULT WINAPI htmldoc2_get_vlinkColor(IHTMLDocument2 *iface, VARIANT *p)
583 {
584     ok(0, "unexpected call\n");
585     return E_NOTIMPL;
586 }
587
588 static HRESULT WINAPI htmldoc2_get_referrer(IHTMLDocument2 *iface, BSTR *p)
589 {
590     ok(0, "unexpected call\n");
591     return E_NOTIMPL;
592 }
593
594 static HRESULT WINAPI htmldoc2_get_location(IHTMLDocument2 *iface, IHTMLLocation **p)
595 {
596     ok(0, "unexpected call\n");
597     return E_NOTIMPL;
598 }
599
600 static HRESULT WINAPI htmldoc2_get_lastModified(IHTMLDocument2 *iface, BSTR *p)
601 {
602     ok(0, "unexpected call\n");
603     return E_NOTIMPL;
604 }
605
606 static HRESULT WINAPI htmldoc2_put_URL(IHTMLDocument2 *iface, BSTR v)
607 {
608     ok(0, "unexpected call\n");
609     return E_NOTIMPL;
610 }
611
612 static HRESULT WINAPI htmldoc2_get_URL(IHTMLDocument2 *iface, BSTR *p)
613 {
614     CHECK_EXPECT2(htmldoc2_get_url);
615     *p = SysAllocString(NULL);
616     return S_OK;
617 }
618
619 static HRESULT WINAPI htmldoc2_put_domain(IHTMLDocument2 *iface, BSTR v)
620 {
621     ok(0, "unexpected call\n");
622     return E_NOTIMPL;
623 }
624
625 static HRESULT WINAPI htmldoc2_get_domain(IHTMLDocument2 *iface, BSTR *p)
626 {
627     ok(0, "unexpected call\n");
628     return E_NOTIMPL;
629 }
630
631 static HRESULT WINAPI htmldoc2_put_cookie(IHTMLDocument2 *iface, BSTR v)
632 {
633     ok(0, "unexpected call\n");
634     return E_NOTIMPL;
635 }
636
637 static HRESULT WINAPI htmldoc2_get_cookie(IHTMLDocument2 *iface, BSTR *p)
638 {
639     ok(0, "unexpected call\n");
640     return E_NOTIMPL;
641 }
642
643 static HRESULT WINAPI htmldoc2_put_expando(IHTMLDocument2 *iface, VARIANT_BOOL v)
644 {
645     ok(0, "unexpected call\n");
646     return E_NOTIMPL;
647 }
648
649 static HRESULT WINAPI htmldoc2_get_expando(IHTMLDocument2 *iface, VARIANT_BOOL *p)
650 {
651     ok(0, "unexpected call\n");
652     return E_NOTIMPL;
653 }
654
655 static HRESULT WINAPI htmldoc2_put_charset(IHTMLDocument2 *iface, BSTR v)
656 {
657     ok(0, "unexpected call\n");
658     return E_NOTIMPL;
659 }
660
661 static HRESULT WINAPI htmldoc2_get_charset(IHTMLDocument2 *iface, BSTR *p)
662 {
663     ok(0, "unexpected call\n");
664     return E_NOTIMPL;
665 }
666
667 static HRESULT WINAPI htmldoc2_put_defaultCharset(IHTMLDocument2 *iface, BSTR v)
668 {
669     ok(0, "unexpected call\n");
670     return E_NOTIMPL;
671 }
672
673 static HRESULT WINAPI htmldoc2_get_defaultCharset(IHTMLDocument2 *iface, BSTR *p)
674 {
675     ok(0, "unexpected call\n");
676     return E_NOTIMPL;
677 }
678
679 static HRESULT WINAPI htmldoc2_get_mimeType(IHTMLDocument2 *iface, BSTR *p)
680 {
681     ok(0, "unexpected call\n");
682     return E_NOTIMPL;
683 }
684
685 static HRESULT WINAPI htmldoc2_get_fileSize(IHTMLDocument2 *iface, BSTR *p)
686 {
687     ok(0, "unexpected call\n");
688     return E_NOTIMPL;
689 }
690
691 static HRESULT WINAPI htmldoc2_get_fileCreatedDate(IHTMLDocument2 *iface, BSTR *p)
692 {
693     ok(0, "unexpected call\n");
694     return E_NOTIMPL;
695 }
696
697 static HRESULT WINAPI htmldoc2_get_fileModifiedDate(IHTMLDocument2 *iface, BSTR *p)
698 {
699     ok(0, "unexpected call\n");
700     return E_NOTIMPL;
701 }
702
703 static HRESULT WINAPI htmldoc2_get_fileUpdatedDate(IHTMLDocument2 *iface, BSTR *p)
704 {
705     ok(0, "unexpected call\n");
706     return E_NOTIMPL;
707 }
708
709 static HRESULT WINAPI htmldoc2_get_security(IHTMLDocument2 *iface, BSTR *p)
710 {
711     ok(0, "unexpected call\n");
712     return E_NOTIMPL;
713 }
714
715 static HRESULT WINAPI htmldoc2_get_protocol(IHTMLDocument2 *iface, BSTR *p)
716 {
717     ok(0, "unexpected call\n");
718     return E_NOTIMPL;
719 }
720
721 static HRESULT WINAPI htmldoc2_get_nameProp(IHTMLDocument2 *iface, BSTR *p)
722 {
723     ok(0, "unexpected call\n");
724     return E_NOTIMPL;
725 }
726
727 static HRESULT WINAPI htmldoc2_write(IHTMLDocument2 *iface, SAFEARRAY *psarray)
728 {
729     ok(0, "unexpected call\n");
730     return E_NOTIMPL;
731 }
732
733 static HRESULT WINAPI htmldoc2_writeln(IHTMLDocument2 *iface, SAFEARRAY *psarray)
734 {
735     ok(0, "unexpected call\n");
736     return E_NOTIMPL;
737 }
738
739 static HRESULT WINAPI htmldoc2_open(IHTMLDocument2 *iface, BSTR url, VARIANT name,
740                         VARIANT features, VARIANT replace, IDispatch **pomWindowResult)
741 {
742     ok(0, "unexpected call\n");
743     return E_NOTIMPL;
744 }
745
746 static HRESULT WINAPI htmldoc2_close(IHTMLDocument2 *iface)
747 {
748     ok(0, "unexpected call\n");
749     return E_NOTIMPL;
750 }
751
752 static HRESULT WINAPI htmldoc2_clear(IHTMLDocument2 *iface)
753 {
754     ok(0, "unexpected call\n");
755     return E_NOTIMPL;
756 }
757
758 static HRESULT WINAPI htmldoc2_queryCommandSupported(IHTMLDocument2 *iface, BSTR cmdID,
759                                                         VARIANT_BOOL *pfRet)
760 {
761     ok(0, "unexpected call\n");
762     return E_NOTIMPL;
763 }
764
765 static HRESULT WINAPI htmldoc2_queryCommandEnabled(IHTMLDocument2 *iface, BSTR cmdID,
766                                                         VARIANT_BOOL *pfRet)
767 {
768     ok(0, "unexpected call\n");
769     return E_NOTIMPL;
770 }
771
772 static HRESULT WINAPI htmldoc2_queryCommandState(IHTMLDocument2 *iface, BSTR cmdID,
773                                                         VARIANT_BOOL *pfRet)
774 {
775     ok(0, "unexpected call\n");
776     return E_NOTIMPL;
777 }
778
779 static HRESULT WINAPI htmldoc2_queryCommandIndeterm(IHTMLDocument2 *iface, BSTR cmdID,
780                                                         VARIANT_BOOL *pfRet)
781 {
782     ok(0, "unexpected call\n");
783     return E_NOTIMPL;
784 }
785
786 static HRESULT WINAPI htmldoc2_queryCommandText(IHTMLDocument2 *iface, BSTR cmdID,
787                                                         BSTR *pfRet)
788 {
789     ok(0, "unexpected call\n");
790     return E_NOTIMPL;
791 }
792
793 static HRESULT WINAPI htmldoc2_queryCommandValue(IHTMLDocument2 *iface, BSTR cmdID,
794                                                         VARIANT *pfRet)
795 {
796     ok(0, "unexpected call\n");
797     return E_NOTIMPL;
798 }
799
800 static HRESULT WINAPI htmldoc2_execCommand(IHTMLDocument2 *iface, BSTR cmdID,
801                                 VARIANT_BOOL showUI, VARIANT value, VARIANT_BOOL *pfRet)
802 {
803     ok(0, "unexpected call\n");
804     return E_NOTIMPL;
805 }
806
807 static HRESULT WINAPI htmldoc2_execCommandShowHelp(IHTMLDocument2 *iface, BSTR cmdID,
808                                                         VARIANT_BOOL *pfRet)
809 {
810     ok(0, "unexpected call\n");
811     return E_NOTIMPL;
812 }
813
814 static HRESULT WINAPI htmldoc2_createElement(IHTMLDocument2 *iface, BSTR eTag,
815                                                  IHTMLElement **newElem)
816 {
817     ok(0, "unexpected call\n");
818     return E_NOTIMPL;
819 }
820
821 static HRESULT WINAPI htmldoc2_put_onhelp(IHTMLDocument2 *iface, VARIANT v)
822 {
823     ok(0, "unexpected call\n");
824     return E_NOTIMPL;
825 }
826
827 static HRESULT WINAPI htmldoc2_get_onhelp(IHTMLDocument2 *iface, VARIANT *p)
828 {
829     ok(0, "unexpected call\n");
830     return E_NOTIMPL;
831 }
832
833 static HRESULT WINAPI htmldoc2_put_onclick(IHTMLDocument2 *iface, VARIANT v)
834 {
835     ok(0, "unexpected call\n");
836     return E_NOTIMPL;
837 }
838
839 static HRESULT WINAPI htmldoc2_get_onclick(IHTMLDocument2 *iface, VARIANT *p)
840 {
841     ok(0, "unexpected call\n");
842     return E_NOTIMPL;
843 }
844
845 static HRESULT WINAPI htmldoc2_put_ondblclick(IHTMLDocument2 *iface, VARIANT v)
846 {
847     ok(0, "unexpected call\n");
848     return E_NOTIMPL;
849 }
850
851 static HRESULT WINAPI htmldoc2_get_ondblclick(IHTMLDocument2 *iface, VARIANT *p)
852 {
853     ok(0, "unexpected call\n");
854     return E_NOTIMPL;
855 }
856
857 static HRESULT WINAPI htmldoc2_put_onkeyup(IHTMLDocument2 *iface, VARIANT v)
858 {
859     ok(0, "unexpected call\n");
860     return E_NOTIMPL;
861 }
862
863 static HRESULT WINAPI htmldoc2_get_onkeyup(IHTMLDocument2 *iface, VARIANT *p)
864 {
865     ok(0, "unexpected call\n");
866     return E_NOTIMPL;
867 }
868
869 static HRESULT WINAPI htmldoc2_put_onkeydown(IHTMLDocument2 *iface, VARIANT v)
870 {
871     ok(0, "unexpected call\n");
872     return E_NOTIMPL;
873 }
874
875 static HRESULT WINAPI htmldoc2_get_onkeydown(IHTMLDocument2 *iface, VARIANT *p)
876 {
877     ok(0, "unexpected call\n");
878     return E_NOTIMPL;
879 }
880
881 static HRESULT WINAPI htmldoc2_put_onkeypress(IHTMLDocument2 *iface, VARIANT v)
882 {
883     ok(0, "unexpected call\n");
884     return E_NOTIMPL;
885 }
886
887 static HRESULT WINAPI htmldoc2_get_onkeypress(IHTMLDocument2 *iface, VARIANT *p)
888 {
889     ok(0, "unexpected call\n");
890     return E_NOTIMPL;
891 }
892
893 static HRESULT WINAPI htmldoc2_put_onmouseup(IHTMLDocument2 *iface, VARIANT v)
894 {
895     ok(0, "unexpected call\n");
896     return E_NOTIMPL;
897 }
898
899 static HRESULT WINAPI htmldoc2_get_onmouseup(IHTMLDocument2 *iface, VARIANT *p)
900 {
901     ok(0, "unexpected call\n");
902     return E_NOTIMPL;
903 }
904
905 static HRESULT WINAPI htmldoc2_put_onmousedown(IHTMLDocument2 *iface, VARIANT v)
906 {
907     ok(0, "unexpected call\n");
908     return E_NOTIMPL;
909 }
910
911 static HRESULT WINAPI htmldoc2_get_onmousedown(IHTMLDocument2 *iface, VARIANT *p)
912 {
913     ok(0, "unexpected call\n");
914     return E_NOTIMPL;
915 }
916
917 static HRESULT WINAPI htmldoc2_put_onmousemove(IHTMLDocument2 *iface, VARIANT v)
918 {
919     ok(0, "unexpected call\n");
920     return E_NOTIMPL;
921 }
922
923 static HRESULT WINAPI htmldoc2_get_onmousemove(IHTMLDocument2 *iface, VARIANT *p)
924 {
925     ok(0, "unexpected call\n");
926     return E_NOTIMPL;
927 }
928
929 static HRESULT WINAPI htmldoc2_put_onmouseout(IHTMLDocument2 *iface, VARIANT v)
930 {
931     ok(0, "unexpected call\n");
932     return E_NOTIMPL;
933 }
934
935 static HRESULT WINAPI htmldoc2_get_onmouseout(IHTMLDocument2 *iface, VARIANT *p)
936 {
937     ok(0, "unexpected call\n");
938     return E_NOTIMPL;
939 }
940
941 static HRESULT WINAPI htmldoc2_put_onmouseover(IHTMLDocument2 *iface, VARIANT v)
942 {
943     ok(0, "unexpected call\n");
944     return E_NOTIMPL;
945 }
946
947 static HRESULT WINAPI htmldoc2_get_onmouseover(IHTMLDocument2 *iface, VARIANT *p)
948 {
949     ok(0, "unexpected call\n");
950     return E_NOTIMPL;
951 }
952
953 static HRESULT WINAPI htmldoc2_put_onreadystatechange(IHTMLDocument2 *iface, VARIANT v)
954 {
955     ok(0, "unexpected call\n");
956     return E_NOTIMPL;
957 }
958
959 static HRESULT WINAPI htmldoc2_get_onreadystatechange(IHTMLDocument2 *iface, VARIANT *p)
960 {
961     ok(0, "unexpected call\n");
962     return E_NOTIMPL;
963 }
964
965 static HRESULT WINAPI htmldoc2_put_onafterupdate(IHTMLDocument2 *iface, VARIANT v)
966 {
967     ok(0, "unexpected call\n");
968     return E_NOTIMPL;
969 }
970
971 static HRESULT WINAPI htmldoc2_get_onafterupdate(IHTMLDocument2 *iface, VARIANT *p)
972 {
973     ok(0, "unexpected call\n");
974     return E_NOTIMPL;
975 }
976
977 static HRESULT WINAPI htmldoc2_put_onrowexit(IHTMLDocument2 *iface, VARIANT v)
978 {
979     ok(0, "unexpected call\n");
980     return E_NOTIMPL;
981 }
982
983 static HRESULT WINAPI htmldoc2_get_onrowexit(IHTMLDocument2 *iface, VARIANT *p)
984 {
985     ok(0, "unexpected call\n");
986     return E_NOTIMPL;
987 }
988
989 static HRESULT WINAPI htmldoc2_put_onrowenter(IHTMLDocument2 *iface, VARIANT v)
990 {
991     ok(0, "unexpected call\n");
992     return E_NOTIMPL;
993 }
994
995 static HRESULT WINAPI htmldoc2_get_onrowenter(IHTMLDocument2 *iface, VARIANT *p)
996 {
997     ok(0, "unexpected call\n");
998     return E_NOTIMPL;
999 }
1000
1001 static HRESULT WINAPI htmldoc2_put_ondragstart(IHTMLDocument2 *iface, VARIANT v)
1002 {
1003     ok(0, "unexpected call\n");
1004     return E_NOTIMPL;
1005 }
1006
1007 static HRESULT WINAPI htmldoc2_get_ondragstart(IHTMLDocument2 *iface, VARIANT *p)
1008 {
1009     ok(0, "unexpected call\n");
1010     return E_NOTIMPL;
1011 }
1012
1013 static HRESULT WINAPI htmldoc2_put_onselectstart(IHTMLDocument2 *iface, VARIANT v)
1014 {
1015     ok(0, "unexpected call\n");
1016     return E_NOTIMPL;
1017 }
1018
1019 static HRESULT WINAPI htmldoc2_get_onselectstart(IHTMLDocument2 *iface, VARIANT *p)
1020 {
1021     ok(0, "unexpected call\n");
1022     return E_NOTIMPL;
1023 }
1024
1025 static HRESULT WINAPI htmldoc2_elementFromPoint(IHTMLDocument2 *iface, LONG x, LONG y,
1026                                                         IHTMLElement **elementHit)
1027 {
1028     ok(0, "unexpected call\n");
1029     return E_NOTIMPL;
1030 }
1031
1032 static HRESULT WINAPI htmldoc2_get_parentWindow(IHTMLDocument2 *iface, IHTMLWindow2 **p)
1033 {
1034     ok(0, "unexpected call\n");
1035     return E_NOTIMPL;
1036 }
1037
1038 static HRESULT WINAPI htmldoc2_get_styleSheets(IHTMLDocument2 *iface,
1039                                                    IHTMLStyleSheetsCollection **p)
1040 {
1041     ok(0, "unexpected call\n");
1042     return E_NOTIMPL;
1043 }
1044
1045 static HRESULT WINAPI htmldoc2_put_onbeforeupdate(IHTMLDocument2 *iface, VARIANT v)
1046 {
1047     ok(0, "unexpected call\n");
1048     return E_NOTIMPL;
1049 }
1050
1051 static HRESULT WINAPI htmldoc2_get_onbeforeupdate(IHTMLDocument2 *iface, VARIANT *p)
1052 {
1053     ok(0, "unexpected call\n");
1054     return E_NOTIMPL;
1055 }
1056
1057 static HRESULT WINAPI htmldoc2_put_onerrorupdate(IHTMLDocument2 *iface, VARIANT v)
1058 {
1059     ok(0, "unexpected call\n");
1060     return E_NOTIMPL;
1061 }
1062
1063 static HRESULT WINAPI htmldoc2_get_onerrorupdate(IHTMLDocument2 *iface, VARIANT *p)
1064 {
1065     ok(0, "unexpected call\n");
1066     return E_NOTIMPL;
1067 }
1068
1069 static HRESULT WINAPI htmldoc2_toString(IHTMLDocument2 *iface, BSTR *String)
1070 {
1071     ok(0, "unexpected call\n");
1072     return E_NOTIMPL;
1073 }
1074
1075 static HRESULT WINAPI htmldoc2_createStyleSheet(IHTMLDocument2 *iface, BSTR bstrHref,
1076                                             LONG lIndex, IHTMLStyleSheet **ppnewStyleSheet)
1077 {
1078     ok(0, "unexpected call\n");
1079     return E_NOTIMPL;
1080 }
1081
1082 static const IHTMLDocument2Vtbl TestHTMLDocumentVtbl = {
1083     htmldoc2_QueryInterface,
1084     htmldoc2_AddRef,
1085     htmldoc2_Release,
1086     htmldoc2_GetTypeInfoCount,
1087     htmldoc2_GetTypeInfo,
1088     htmldoc2_GetIDsOfNames,
1089     htmldoc2_Invoke,
1090     htmldoc2_get_Script,
1091     htmldoc2_get_all,
1092     htmldoc2_get_body,
1093     htmldoc2_get_activeElement,
1094     htmldoc2_get_images,
1095     htmldoc2_get_applets,
1096     htmldoc2_get_links,
1097     htmldoc2_get_forms,
1098     htmldoc2_get_anchors,
1099     htmldoc2_put_title,
1100     htmldoc2_get_title,
1101     htmldoc2_get_scripts,
1102     htmldoc2_put_designMode,
1103     htmldoc2_get_designMode,
1104     htmldoc2_get_selection,
1105     htmldoc2_get_readyState,
1106     htmldoc2_get_frames,
1107     htmldoc2_get_embeds,
1108     htmldoc2_get_plugins,
1109     htmldoc2_put_alinkColor,
1110     htmldoc2_get_alinkColor,
1111     htmldoc2_put_bgColor,
1112     htmldoc2_get_bgColor,
1113     htmldoc2_put_fgColor,
1114     htmldoc2_get_fgColor,
1115     htmldoc2_put_linkColor,
1116     htmldoc2_get_linkColor,
1117     htmldoc2_put_vlinkColor,
1118     htmldoc2_get_vlinkColor,
1119     htmldoc2_get_referrer,
1120     htmldoc2_get_location,
1121     htmldoc2_get_lastModified,
1122     htmldoc2_put_URL,
1123     htmldoc2_get_URL,
1124     htmldoc2_put_domain,
1125     htmldoc2_get_domain,
1126     htmldoc2_put_cookie,
1127     htmldoc2_get_cookie,
1128     htmldoc2_put_expando,
1129     htmldoc2_get_expando,
1130     htmldoc2_put_charset,
1131     htmldoc2_get_charset,
1132     htmldoc2_put_defaultCharset,
1133     htmldoc2_get_defaultCharset,
1134     htmldoc2_get_mimeType,
1135     htmldoc2_get_fileSize,
1136     htmldoc2_get_fileCreatedDate,
1137     htmldoc2_get_fileModifiedDate,
1138     htmldoc2_get_fileUpdatedDate,
1139     htmldoc2_get_security,
1140     htmldoc2_get_protocol,
1141     htmldoc2_get_nameProp,
1142     htmldoc2_write,
1143     htmldoc2_writeln,
1144     htmldoc2_open,
1145     htmldoc2_close,
1146     htmldoc2_clear,
1147     htmldoc2_queryCommandSupported,
1148     htmldoc2_queryCommandEnabled,
1149     htmldoc2_queryCommandState,
1150     htmldoc2_queryCommandIndeterm,
1151     htmldoc2_queryCommandText,
1152     htmldoc2_queryCommandValue,
1153     htmldoc2_execCommand,
1154     htmldoc2_execCommandShowHelp,
1155     htmldoc2_createElement,
1156     htmldoc2_put_onhelp,
1157     htmldoc2_get_onhelp,
1158     htmldoc2_put_onclick,
1159     htmldoc2_get_onclick,
1160     htmldoc2_put_ondblclick,
1161     htmldoc2_get_ondblclick,
1162     htmldoc2_put_onkeyup,
1163     htmldoc2_get_onkeyup,
1164     htmldoc2_put_onkeydown,
1165     htmldoc2_get_onkeydown,
1166     htmldoc2_put_onkeypress,
1167     htmldoc2_get_onkeypress,
1168     htmldoc2_put_onmouseup,
1169     htmldoc2_get_onmouseup,
1170     htmldoc2_put_onmousedown,
1171     htmldoc2_get_onmousedown,
1172     htmldoc2_put_onmousemove,
1173     htmldoc2_get_onmousemove,
1174     htmldoc2_put_onmouseout,
1175     htmldoc2_get_onmouseout,
1176     htmldoc2_put_onmouseover,
1177     htmldoc2_get_onmouseover,
1178     htmldoc2_put_onreadystatechange,
1179     htmldoc2_get_onreadystatechange,
1180     htmldoc2_put_onafterupdate,
1181     htmldoc2_get_onafterupdate,
1182     htmldoc2_put_onrowexit,
1183     htmldoc2_get_onrowexit,
1184     htmldoc2_put_onrowenter,
1185     htmldoc2_get_onrowenter,
1186     htmldoc2_put_ondragstart,
1187     htmldoc2_get_ondragstart,
1188     htmldoc2_put_onselectstart,
1189     htmldoc2_get_onselectstart,
1190     htmldoc2_elementFromPoint,
1191     htmldoc2_get_parentWindow,
1192     htmldoc2_get_styleSheets,
1193     htmldoc2_put_onbeforeupdate,
1194     htmldoc2_get_onbeforeupdate,
1195     htmldoc2_put_onerrorupdate,
1196     htmldoc2_get_onerrorupdate,
1197     htmldoc2_toString,
1198     htmldoc2_createStyleSheet
1199 };
1200
1201 typedef struct
1202 {
1203     IHTMLDocument2 IHTMLDocument2_iface;
1204 } testhtmldoc2_t;
1205
1206 static testhtmldoc2_t htmldoc2 = { { &TestHTMLDocumentVtbl } };
1207
1208 static HRESULT WINAPI sp_QueryInterface(IServiceProvider *iface, REFIID riid, void **ppvObject)
1209 {
1210     *ppvObject = NULL;
1211
1212     if (IsEqualGUID(riid, &IID_IUnknown) ||
1213         IsEqualGUID(riid, &IID_IServiceProvider))
1214     {
1215         *ppvObject = iface;
1216         IServiceProvider_AddRef(iface);
1217         return S_OK;
1218     }
1219
1220     ok(0, "unexpected query interface: %s\n", debugstr_guid(riid));
1221
1222     return E_NOINTERFACE;
1223 }
1224
1225 static ULONG WINAPI sp_AddRef(IServiceProvider *iface)
1226 {
1227     return 2;
1228 }
1229
1230 static ULONG WINAPI sp_Release(IServiceProvider *iface)
1231 {
1232     return 1;
1233 }
1234
1235 static HRESULT WINAPI sp_QueryService(IServiceProvider *iface, REFGUID service, REFIID riid, void **obj)
1236 {
1237     *obj = NULL;
1238
1239     if (IsEqualGUID(service, &SID_SBindHost) &&
1240         IsEqualGUID(riid, &IID_IBindHost))
1241     {
1242         CHECK_EXPECT2(sp_queryservice_SID_SBindHost);
1243     }
1244     else if (IsEqualGUID(service, &SID_SContainerDispatch) &&
1245              IsEqualGUID(riid, &IID_IHTMLDocument2))
1246     {
1247         CHECK_EXPECT2(sp_queryservice_SID_SContainerDispatch_htmldoc2);
1248     }
1249     else if (IsEqualGUID(service, &SID_SInternetHostSecurityManager) &&
1250              IsEqualGUID(riid, &IID_IHTMLDocument2))
1251     {
1252         CHECK_EXPECT2(sp_queryservice_SID_secmgr_htmldoc2);
1253         *obj = &htmldoc2.IHTMLDocument2_iface;
1254         return S_OK;
1255     }
1256     else if (IsEqualGUID(service, &SID_SInternetHostSecurityManager) &&
1257              IsEqualGUID(riid, &IID_IXMLDOMDocument))
1258     {
1259         CHECK_EXPECT2(sp_queryservice_SID_secmgr_xmldomdoc);
1260     }
1261     else if (IsEqualGUID(service, &SID_SInternetHostSecurityManager) &&
1262              IsEqualGUID(riid, &IID_IInternetHostSecurityManager))
1263     {
1264         CHECK_EXPECT2(sp_queryservice_SID_secmgr_secmgr);
1265     }
1266     else if (IsEqualGUID(service, &SID_UnknownSID) &&
1267              IsEqualGUID(riid, &IID_IStream))
1268     {
1269         /* FIXME: unidentified service id */
1270     }
1271     else
1272         ok(0, "unexpected request: sid %s, riid %s\n", debugstr_guid(service), debugstr_guid(riid));
1273
1274     return E_NOTIMPL;
1275 }
1276
1277 static const IServiceProviderVtbl testprovVtbl =
1278 {
1279     sp_QueryInterface,
1280     sp_AddRef,
1281     sp_Release,
1282     sp_QueryService
1283 };
1284
1285 testprov_t testprov = { { &testprovVtbl } };
1286
1287 #define EXPECT_CHILDREN(node) _expect_children((IXMLDOMNode*)node, __LINE__)
1288 static void _expect_children(IXMLDOMNode *node, int line)
1289 {
1290     VARIANT_BOOL b;
1291     HRESULT hr;
1292
1293     b = VARIANT_FALSE;
1294     hr = IXMLDOMNode_hasChildNodes(node, &b);
1295     ok_(__FILE__,line)(hr == S_OK, "hasChildNodes() failed, 0x%08x\n", hr);
1296     ok_(__FILE__,line)(b == VARIANT_TRUE, "no children, %d\n", b);
1297 }
1298
1299 #define EXPECT_NO_CHILDREN(node) _expect_no_children((IXMLDOMNode*)node, __LINE__)
1300 static void _expect_no_children(IXMLDOMNode *node, int line)
1301 {
1302     VARIANT_BOOL b;
1303     HRESULT hr;
1304
1305     b = VARIANT_TRUE;
1306     hr = IXMLDOMNode_hasChildNodes(node, &b);
1307     ok_(__FILE__,line)(hr == S_FALSE, "hasChildNodes() failed, 0x%08x\n", hr);
1308     ok_(__FILE__,line)(b == VARIANT_FALSE, "no children, %d\n", b);
1309 }
1310
1311 #define EXPECT_REF(node,ref) _expect_ref((IUnknown*)node, ref, __LINE__)
1312 static void _expect_ref(IUnknown* obj, ULONG ref, int line)
1313 {
1314     ULONG rc = IUnknown_AddRef(obj);
1315     IUnknown_Release(obj);
1316     ok_(__FILE__,line)(rc-1 == ref, "expected refcount %d, got %d\n", ref, rc-1);
1317 }
1318
1319 #define EXPECT_LIST_LEN(list,len) _expect_list_len(list, len, __LINE__)
1320 static void _expect_list_len(IXMLDOMNodeList *list, LONG len, int line)
1321 {
1322     LONG length;
1323     HRESULT hr;
1324
1325     length = 0;
1326     hr = IXMLDOMNodeList_get_length(list, &length);
1327     ok_(__FILE__,line)(hr == S_OK, "got 0x%08x\n", hr);
1328     ok_(__FILE__,line)(length == len, "got %d, expected %d\n", length, len);
1329 }
1330
1331 #define EXPECT_HR(hr,hr_exp) \
1332     ok(hr == hr_exp, "got 0x%08x, expected 0x%08x\n", hr, hr_exp)
1333
1334 static const WCHAR szEmpty[] = { 0 };
1335 static const WCHAR szIncomplete[] = {
1336     '<','?','x','m','l',' ',
1337     'v','e','r','s','i','o','n','=','\'','1','.','0','\'','?','>','\n',0
1338 };
1339 static const WCHAR szComplete1[] = {
1340     '<','?','x','m','l',' ',
1341     'v','e','r','s','i','o','n','=','\'','1','.','0','\'','?','>','\n',
1342     '<','o','p','e','n','>','<','/','o','p','e','n','>','\n',0
1343 };
1344 static const WCHAR szComplete2[] = {
1345     '<','?','x','m','l',' ',
1346     'v','e','r','s','i','o','n','=','\'','1','.','0','\'','?','>','\n',
1347     '<','o','>','<','/','o','>','\n',0
1348 };
1349 static const WCHAR szComplete3[] = {
1350     '<','?','x','m','l',' ',
1351     'v','e','r','s','i','o','n','=','\'','1','.','0','\'','?','>','\n',
1352     '<','a','>','<','/','a','>','\n',0
1353 };
1354 static const WCHAR szComplete4[] = {
1355     '<','?','x','m','l',' ','v','e','r','s','i','o','n','=','\'','1','.','0','\'','?','>','\n',
1356     '<','l','c',' ','d','l','=','\'','s','t','r','1','\'','>','\n',
1357         '<','b','s',' ','v','r','=','\'','s','t','r','2','\'',' ','s','z','=','\'','1','2','3','4','\'','>',
1358             'f','n','1','.','t','x','t','\n',
1359         '<','/','b','s','>','\n',
1360         '<','p','r',' ','i','d','=','\'','s','t','r','3','\'',' ','v','r','=','\'','1','.','2','.','3','\'',' ',
1361                     'p','n','=','\'','w','i','n','e',' ','2','0','0','5','0','8','0','4','\'','>','\n',
1362             'f','n','2','.','t','x','t','\n',
1363         '<','/','p','r','>','\n',
1364         '<','e','m','p','t','y','>','<','/','e','m','p','t','y','>','\n',
1365         '<','f','o','>','\n',
1366             '<','b','a','>','\n',
1367                 'f','1','\n',
1368             '<','/','b','a','>','\n',
1369         '<','/','f','o','>','\n',
1370     '<','/','l','c','>','\n',0
1371 };
1372 static const WCHAR szComplete5[] = {
1373     '<','S',':','s','e','a','r','c','h',' ','x','m','l','n','s',':','D','=','"','D','A','V',':','"',' ',
1374     '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','"',
1375     ' ','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','"','>',
1376         '<','S',':','s','c','o','p','e','>',
1377             '<','S',':','d','e','e','p','>','/','<','/','S',':','d','e','e','p','>',
1378         '<','/','S',':','s','c','o','p','e','>',
1379         '<','S',':','c','o','n','t','e','n','t','f','r','e','e','t','e','x','t','>',
1380             '<','C',':','t','e','x','t','o','r','p','r','o','p','e','r','t','y','/','>',
1381             'c','o','m','p','u','t','e','r',
1382         '<','/','S',':','c','o','n','t','e','n','t','f','r','e','e','t','e','x','t','>',
1383     '<','/','S',':','s','e','a','r','c','h','>',0
1384 };
1385
1386 static const WCHAR szComplete6[] = {
1387     '<','?','x','m','l',' ','v','e','r','s','i','o','n','=','\'','1','.','0','\'',' ',
1388     'e','n','c','o','d','i','n','g','=','\'','W','i','n','d','o','w','s','-','1','2','5','2','\'','?','>','\n',
1389     '<','o','p','e','n','>','<','/','o','p','e','n','>','\n',0
1390 };
1391
1392 static const CHAR szNonUnicodeXML[] =
1393 "<?xml version='1.0' encoding='Windows-1252'?>\n"
1394 "<open></open>\n";
1395
1396 static const CHAR szExampleXML[] =
1397 "<?xml version='1.0' encoding='utf-8'?>\n"
1398 "<root xmlns:foo='urn:uuid:86B2F87F-ACB6-45cd-8B77-9BDB92A01A29'>\n"
1399 "    <elem>\n"
1400 "        <a>A1 field</a>\n"
1401 "        <b>B1 field</b>\n"
1402 "        <c>C1 field</c>\n"
1403 "        <description xmlns:foo='http://www.winehq.org' xmlns:bar='urn:uuid:86B2F87F-ACB6-45cd-8B77-9BDB92A01A29'>\n"
1404 "            <html xmlns='http://www.w3.org/1999/xhtml'>\n"
1405 "                This is <strong>a</strong> <i>description</i>. <bar:x/>\n"
1406 "            </html>\n"
1407 "            <html xml:space='preserve' xmlns='http://www.w3.org/1999/xhtml'>\n"
1408 "                This is <strong>a</strong> <i>description</i> with preserved whitespace. <bar:x/>\n"
1409 "            </html>\n"
1410 "        </description>\n"
1411 "    </elem>\n"
1412 "\n"
1413 "    <elem>\n"
1414 "        <a>A2 field</a>\n"
1415 "        <b>B2 field</b>\n"
1416 "        <c type=\"old\">C2 field</c>\n"
1417 "    </elem>\n"
1418 "\n"
1419 "    <elem xmlns='urn:uuid:86B2F87F-ACB6-45cd-8B77-9BDB92A01A29'>\n"
1420 "        <a>A3 field</a>\n"
1421 "        <b>B3 field</b>\n"
1422 "        <c>C3 field</c>\n"
1423 "    </elem>\n"
1424 "\n"
1425 "    <elem>\n"
1426 "        <a>A4 field</a>\n"
1427 "        <b>B4 field</b>\n"
1428 "        <foo:c>C4 field</foo:c>\n"
1429 "    </elem>\n"
1430 "</root>\n";
1431
1432 static const CHAR szNodeTypesXML[] =
1433 "<?xml version='1.0'?>"
1434 "<!-- comment node 0 -->"
1435 "<root id='0' depth='0'>"
1436 "   <!-- comment node 1 -->"
1437 "   text node 0"
1438 "   <x id='1' depth='1'>"
1439 "       <?foo value='PI for x'?>"
1440 "       <!-- comment node 2 -->"
1441 "       text node 1"
1442 "       <a id='3' depth='2'/>"
1443 "       <b id='4' depth='2'/>"
1444 "       <c id='5' depth='2'/>"
1445 "   </x>"
1446 "   <y id='2' depth='1'>"
1447 "       <?bar value='PI for y'?>"
1448 "       <!-- comment node 3 -->"
1449 "       text node 2"
1450 "       <a id='6' depth='2'/>"
1451 "       <b id='7' depth='2'/>"
1452 "       <c id='8' depth='2'/>"
1453 "   </y>"
1454 "</root>";
1455
1456 static const CHAR szTransformXML[] =
1457 "<?xml version=\"1.0\"?>\n"
1458 "<greeting>\n"
1459 "Hello World\n"
1460 "</greeting>";
1461
1462 static  const CHAR szTransformSSXML[] =
1463 "<xsl:stylesheet xmlns:xsl=\"http://www.w3.org/1999/XSL/Transform\" version=\"1.0\">\n"
1464 "   <xsl:output method=\"html\"/>\n"
1465 "   <xsl:template match=\"/\">\n"
1466 "       <xsl:apply-templates select=\"greeting\"/>\n"
1467 "   </xsl:template>\n"
1468 "   <xsl:template match=\"greeting\">\n"
1469 "       <html>\n"
1470 "           <body>\n"
1471 "               <h1>\n"
1472 "                   <xsl:value-of select=\".\"/>\n"
1473 "               </h1>\n"
1474 "           </body>\n"
1475 "       </html>\n"
1476 "   </xsl:template>\n"
1477 "</xsl:stylesheet>";
1478
1479 static  const CHAR szTransformOutput[] =
1480 "<html><body><h1>"
1481 "Hello World"
1482 "</h1></body></html>";
1483
1484 static const CHAR szTypeValueXML[] =
1485 "<?xml version=\"1.0\" encoding=\"utf-8\"?>\n"
1486 "<root xmlns:dt=\"urn:schemas-microsoft-com:datatypes\">\n"
1487 "   <string>Wine</string>\n"
1488 "   <string2 dt:dt=\"string\">String</string2>\n"
1489 "   <number dt:dt=\"number\">12.44</number>\n"
1490 "   <number2 dt:dt=\"NUMbEr\">-3.71e3</number2>\n"
1491 "   <int dt:dt=\"int\">-13</int>\n"
1492 "   <fixed dt:dt=\"fixed.14.4\">7322.9371</fixed>\n"
1493 "   <bool dt:dt=\"boolean\">1</bool>\n"
1494 "   <datetime dt:dt=\"datetime\">2009-11-18T03:21:33.12</datetime>\n"
1495 "   <datetimetz dt:dt=\"datetime.tz\">2003-07-11T11:13:57+03:00</datetimetz>\n"
1496 "   <date dt:dt=\"date\">3721-11-01</date>\n"
1497 "   <time dt:dt=\"time\">13:57:12.31321</time>\n"
1498 "   <timetz dt:dt=\"time.tz\">23:21:01.13+03:21</timetz>\n"
1499 "   <i1 dt:dt=\"i1\">-13</i1>\n"
1500 "   <i2 dt:dt=\"i2\">31915</i2>\n"
1501 "   <i4 dt:dt=\"i4\">-312232</i4>\n"
1502 "   <ui1 dt:dt=\"ui1\">123</ui1>\n"
1503 "   <ui2 dt:dt=\"ui2\">48282</ui2>\n"
1504 "   <ui4 dt:dt=\"ui4\">949281</ui4>\n"
1505 "   <r4 dt:dt=\"r4\">213124.0</r4>\n"
1506 "   <r8 dt:dt=\"r8\">0.412</r8>\n"
1507 "   <float dt:dt=\"float\">41221.421</float>\n"
1508 "   <uuid dt:dt=\"uuid\">333C7BC4-460F-11D0-BC04-0080C7055a83</uuid>\n"
1509 "   <binhex dt:dt=\"bin.hex\">fffca012003c</binhex>\n"
1510 "   <binbase64 dt:dt=\"bin.base64\">YmFzZTY0IHRlc3Q=</binbase64>\n"
1511 "   <binbase64_1 dt:dt=\"bin.base64\">\nYmFzZTY0\nIHRlc3Q=\n</binbase64_1>\n"
1512 "   <binbase64_2 dt:dt=\"bin.base64\">\nYmF\r\t z  ZTY0\nIHRlc3Q=\n</binbase64_2>\n"
1513 "</root>";
1514
1515 static const CHAR szBasicTransformSSXMLPart1[] =
1516 "<?xml version=\"1.0\"?>"
1517 "<xsl:stylesheet version=\"1.0\" xmlns:xsl=\"http://www.w3.org/1999/XSL/Transform\" >"
1518 "<xsl:output method=\"html\"/>\n"
1519 "<xsl:template match=\"/\">"
1520 "<HTML><BODY><TABLE>"
1521 "        <xsl:apply-templates select='document(\"";
1522
1523 static const CHAR szBasicTransformSSXMLPart2[] =
1524 "\")/bottle/wine'>"
1525 "           <xsl:sort select=\"cost\"/><xsl:sort select=\"name\"/>"
1526 "        </xsl:apply-templates>"
1527 "</TABLE></BODY></HTML>"
1528 "</xsl:template>"
1529 "<xsl:template match=\"bottle\">"
1530 "   <TR><xsl:apply-templates select=\"name\" /><xsl:apply-templates select=\"cost\" /></TR>"
1531 "</xsl:template>"
1532 "<xsl:template match=\"name\">"
1533 "   <TD><xsl:apply-templates /></TD>"
1534 "</xsl:template>"
1535 "<xsl:template match=\"cost\">"
1536 "   <TD><xsl:apply-templates /></TD>"
1537 "</xsl:template>"
1538 "</xsl:stylesheet>";
1539
1540 static const CHAR szBasicTransformXML[] =
1541 "<?xml version=\"1.0\"?><bottle><wine><name>Wine</name><cost>$25.00</cost></wine></bottle>";
1542
1543 static const CHAR szBasicTransformOutput[] =
1544 "<HTML><BODY><TABLE><TD>Wine</TD><TD>$25.00</TD></TABLE></BODY></HTML>";
1545
1546 #define SZ_EMAIL_DTD \
1547 "<!DOCTYPE email ["\
1548 "   <!ELEMENT email         (recipients,from,reply-to?,subject,body,attachment*)>"\
1549 "       <!ATTLIST email attachments IDREFS #REQUIRED>"\
1550 "       <!ATTLIST email sent (yes|no) \"no\">"\
1551 "   <!ELEMENT recipients    (to+,cc*)>"\
1552 "   <!ELEMENT to            (#PCDATA)>"\
1553 "       <!ATTLIST to name CDATA #IMPLIED>"\
1554 "   <!ELEMENT cc            (#PCDATA)>"\
1555 "       <!ATTLIST cc name CDATA #IMPLIED>"\
1556 "   <!ELEMENT from          (#PCDATA)>"\
1557 "       <!ATTLIST from name CDATA #IMPLIED>"\
1558 "   <!ELEMENT reply-to      (#PCDATA)>"\
1559 "       <!ATTLIST reply-to name CDATA #IMPLIED>"\
1560 "   <!ELEMENT subject       ANY>"\
1561 "   <!ELEMENT body          ANY>"\
1562 "       <!ATTLIST body enc CDATA #FIXED \"UTF-8\">"\
1563 "   <!ELEMENT attachment    (#PCDATA)>"\
1564 "       <!ATTLIST attachment id ID #REQUIRED>"\
1565 "]>"
1566
1567 static const CHAR szEmailXML[] =
1568 "<?xml version=\"1.0\"?>"
1569 SZ_EMAIL_DTD
1570 "<email attachments=\"patch1\">"
1571 "   <recipients>"
1572 "       <to>wine-patches@winehq.org</to>"
1573 "   </recipients>"
1574 "   <from name=\"Anonymous\">user@localhost</from>"
1575 "   <subject>msxml3/tests: DTD validation (try 87)</subject>"
1576 "   <body>"
1577 "       It no longer causes spontaneous combustion..."
1578 "   </body>"
1579 "   <attachment id=\"patch1\">0001-msxml3-tests-DTD-validation.patch</attachment>"
1580 "</email>";
1581
1582 static const CHAR szEmailXML_0D[] =
1583 "<?xml version=\"1.0\"?>"
1584 SZ_EMAIL_DTD
1585 "<email attachments=\"patch1\">"
1586 "   <recipients>"
1587 "       <to>wine-patches@winehq.org</to>"
1588 "   </recipients>"
1589 "   <from name=\"Anonymous\">user@localhost</from>"
1590 "   <subject>msxml3/tests: DTD validation (try 88)</subject>"
1591 "   <body>"
1592 "       <undecl />"
1593 "       XML_ELEMENT_UNDECLARED 0xC00CE00D"
1594 "   </body>"
1595 "   <attachment id=\"patch1\">0001-msxml3-tests-DTD-validation.patch</attachment>"
1596 "</email>";
1597
1598 static const CHAR szEmailXML_0E[] =
1599 "<?xml version=\"1.0\"?>"
1600 SZ_EMAIL_DTD
1601 "<email attachments=\"patch1\">"
1602 "   <recipients>"
1603 "       <to>wine-patches@winehq.org</to>"
1604 "   </recipients>"
1605 "   <from name=\"Anonymous\">user@localhost</from>"
1606 "   <subject>msxml3/tests: DTD validation (try 89)</subject>"
1607 "   <body>"
1608 "       XML_ELEMENT_ID_NOT_FOUND 0xC00CE00E"
1609 "   </body>"
1610 "   <attachment id=\"patch\">0001-msxml3-tests-DTD-validation.patch</attachment>"
1611 "</email>";
1612
1613 static const CHAR szEmailXML_11[] =
1614 "<?xml version=\"1.0\"?>"
1615 SZ_EMAIL_DTD
1616 "<email attachments=\"patch1\">"
1617 "   <recipients>"
1618 "   </recipients>"
1619 "   <from name=\"Anonymous\">user@localhost</from>"
1620 "   <subject>msxml3/tests: DTD validation (try 90)</subject>"
1621 "   <body>"
1622 "       XML_EMPTY_NOT_ALLOWED 0xC00CE011"
1623 "   </body>"
1624 "   <attachment id=\"patch1\">0001-msxml3-tests-DTD-validation.patch</attachment>"
1625 "</email>";
1626
1627 static const CHAR szEmailXML_13[] =
1628 "<?xml version=\"1.0\"?>"
1629 SZ_EMAIL_DTD
1630 "<msg attachments=\"patch1\">"
1631 "   <recipients>"
1632 "       <to>wine-patches@winehq.org</to>"
1633 "   </recipients>"
1634 "   <from name=\"Anonymous\">user@localhost</from>"
1635 "   <subject>msxml3/tests: DTD validation (try 91)</subject>"
1636 "   <body>"
1637 "       XML_ROOT_NAME_MISMATCH 0xC00CE013"
1638 "   </body>"
1639 "   <attachment id=\"patch1\">0001-msxml3-tests-DTD-validation.patch</attachment>"
1640 "</msg>";
1641
1642 static const CHAR szEmailXML_14[] =
1643 "<?xml version=\"1.0\"?>"
1644 SZ_EMAIL_DTD
1645 "<email attachments=\"patch1\">"
1646 "   <to>wine-patches@winehq.org</to>"
1647 "   <from name=\"Anonymous\">user@localhost</from>"
1648 "   <subject>msxml3/tests: DTD validation (try 92)</subject>"
1649 "   <body>"
1650 "       XML_INVALID_CONTENT 0xC00CE014"
1651 "   </body>"
1652 "   <attachment id=\"patch1\">0001-msxml3-tests-DTD-validation.patch</attachment>"
1653 "</email>";
1654
1655 static const CHAR szEmailXML_15[] =
1656 "<?xml version=\"1.0\"?>"
1657 SZ_EMAIL_DTD
1658 "<email attachments=\"patch1\" ip=\"127.0.0.1\">"
1659 "   <recipients>"
1660 "       <to>wine-patches@winehq.org</to>"
1661 "   </recipients>"
1662 "   <from name=\"Anonymous\">user@localhost</from>"
1663 "   <subject>msxml3/tests: DTD validation (try 93)</subject>"
1664 "   <body>"
1665 "       XML_ATTRIBUTE_NOT_DEFINED 0xC00CE015"
1666 "   </body>"
1667 "   <attachment id=\"patch1\">0001-msxml3-tests-DTD-validation.patch</attachment>"
1668 "</email>";
1669
1670 static const CHAR szEmailXML_16[] =
1671 "<?xml version=\"1.0\"?>"
1672 SZ_EMAIL_DTD
1673 "<email attachments=\"patch1\">"
1674 "   <recipients>"
1675 "       <to>wine-patches@winehq.org</to>"
1676 "   </recipients>"
1677 "   <from name=\"Anonymous\">user@localhost</from>"
1678 "   <subject>msxml3/tests: DTD validation (try 94)</subject>"
1679 "   <body enc=\"ASCII\">"
1680 "       XML_ATTRIBUTE_FIXED 0xC00CE016"
1681 "   </body>"
1682 "   <attachment id=\"patch1\">0001-msxml3-tests-DTD-validation.patch</attachment>"
1683 "</email>";
1684
1685 static const CHAR szEmailXML_17[] =
1686 "<?xml version=\"1.0\"?>"
1687 SZ_EMAIL_DTD
1688 "<email attachments=\"patch1\" sent=\"true\">"
1689 "   <recipients>"
1690 "       <to>wine-patches@winehq.org</to>"
1691 "   </recipients>"
1692 "   <from name=\"Anonymous\">user@localhost</from>"
1693 "   <subject>msxml3/tests: DTD validation (try 95)</subject>"
1694 "   <body>"
1695 "       XML_ATTRIBUTE_VALUE 0xC00CE017"
1696 "   </body>"
1697 "   <attachment id=\"patch1\">0001-msxml3-tests-DTD-validation.patch</attachment>"
1698 "</email>";
1699
1700 static const CHAR szEmailXML_18[] =
1701 "<?xml version=\"1.0\"?>"
1702 SZ_EMAIL_DTD
1703 "<email attachments=\"patch1\">"
1704 "   oops"
1705 "   <recipients>"
1706 "       <to>wine-patches@winehq.org</to>"
1707 "   </recipients>"
1708 "   <from name=\"Anonymous\">user@localhost</from>"
1709 "   <subject>msxml3/tests: DTD validation (try 96)</subject>"
1710 "   <body>"
1711 "       XML_ILLEGAL_TEXT 0xC00CE018"
1712 "   </body>"
1713 "   <attachment id=\"patch1\">0001-msxml3-tests-DTD-validation.patch</attachment>"
1714 "</email>";
1715
1716 static const CHAR szEmailXML_20[] =
1717 "<?xml version=\"1.0\"?>"
1718 SZ_EMAIL_DTD
1719 "<email>"
1720 "   <recipients>"
1721 "       <to>wine-patches@winehq.org</to>"
1722 "   </recipients>"
1723 "   <from name=\"Anonymous\">user@localhost</from>"
1724 "   <subject>msxml3/tests: DTD validation (try 97)</subject>"
1725 "   <body>"
1726 "       XML_REQUIRED_ATTRIBUTE_MISSING 0xC00CE020"
1727 "   </body>"
1728 "   <attachment id=\"patch1\">0001-msxml3-tests-DTD-validation.patch</attachment>"
1729 "</email>";
1730
1731 static const WCHAR szNonExistentFile[] = {
1732     'c', ':', '\\', 'N', 'o', 'n', 'e', 'x', 'i', 's', 't', 'e', 'n', 't', '.', 'x', 'm', 'l', 0
1733 };
1734 static const WCHAR szNonExistentAttribute[] = {
1735     'n','o','n','E','x','i','s','i','t','i','n','g','A','t','t','r','i','b','u','t','e',0
1736 };
1737 static const WCHAR szDocument[] = {
1738     '#', 'd', 'o', 'c', 'u', 'm', 'e', 'n', 't', 0
1739 };
1740
1741 static const WCHAR szOpen[] = { 'o','p','e','n',0 };
1742 static WCHAR szdl[] = { 'd','l',0 };
1743 static const WCHAR szvr[] = { 'v','r',0 };
1744 static const WCHAR szlc[] = { 'l','c',0 };
1745 static WCHAR szbs[] = { 'b','s',0 };
1746 static const WCHAR szstr1[] = { 's','t','r','1',0 };
1747 static const WCHAR szstr2[] = { 's','t','r','2',0 };
1748 static const WCHAR szstar[] = { '*',0 };
1749 static const WCHAR szfn1_txt[] = {'f','n','1','.','t','x','t',0};
1750
1751 static WCHAR szComment[] = {'A',' ','C','o','m','m','e','n','t',0 };
1752 static WCHAR szCommentXML[] = {'<','!','-','-','A',' ','C','o','m','m','e','n','t','-','-','>',0 };
1753 static WCHAR szCommentNodeText[] = {'#','c','o','m','m','e','n','t',0 };
1754
1755 static WCHAR szElement[] = {'E','l','e','T','e','s','t', 0 };
1756 static WCHAR szElementXML[]  = {'<','E','l','e','T','e','s','t','/','>',0 };
1757 static WCHAR szElementXML2[] = {'<','E','l','e','T','e','s','t',' ','A','t','t','r','=','"','"','/','>',0 };
1758 static WCHAR szElementXML3[] = {'<','E','l','e','T','e','s','t',' ','A','t','t','r','=','"','"','>',
1759                                 'T','e','s','t','i','n','g','N','o','d','e','<','/','E','l','e','T','e','s','t','>',0 };
1760 static WCHAR szElementXML4[] = {'<','E','l','e','T','e','s','t',' ','A','t','t','r','=','"','"','>',
1761                                 '&','a','m','p',';','x',' ',0x2103,'<','/','E','l','e','T','e','s','t','>',0 };
1762
1763 static WCHAR szAttribute[] = {'A','t','t','r',0 };
1764 static WCHAR szAttributeXML[] = {'A','t','t','r','=','"','"',0 };
1765
1766 static WCHAR szCData[] = {'[','1',']','*','2','=','3',';',' ','&','g','e','e',' ','t','h','a','t','s',
1767                           ' ','n','o','t',' ','r','i','g','h','t','!', 0};
1768 static WCHAR szCDataXML[] = {'<','!','[','C','D','A','T','A','[','[','1',']','*','2','=','3',';',' ','&',
1769                              'g','e','e',' ','t','h','a','t','s',' ','n','o','t',' ','r','i','g','h','t',
1770                              '!',']',']','>',0};
1771 static WCHAR szCDataNodeText[] = {'#','c','d','a','t','a','-','s','e','c','t','i','o','n',0 };
1772 static WCHAR szDocFragmentText[] = {'#','d','o','c','u','m','e','n','t','-','f','r','a','g','m','e','n','t',0 };
1773
1774 static WCHAR szEntityRef[] = {'e','n','t','i','t','y','r','e','f',0 };
1775 static WCHAR szEntityRefXML[] = {'&','e','n','t','i','t','y','r','e','f',';',0 };
1776 static WCHAR szStrangeChars[] = {'&','x',' ',0x2103, 0};
1777
1778 #define expect_bstr_eq_and_free(bstr, expect) { \
1779     BSTR bstrExp = alloc_str_from_narrow(expect); \
1780     ok(lstrcmpW(bstr, bstrExp) == 0, "String differs\n"); \
1781     SysFreeString(bstr); \
1782     SysFreeString(bstrExp); \
1783 }
1784
1785 #define expect_eq(expr, value, type, format) { type ret = (expr); ok((value) == ret, #expr " expected " format " got " format "\n", value, ret); }
1786
1787 #define ole_check(expr) { \
1788     HRESULT r = expr; \
1789     ok(r == S_OK, #expr " returned %x\n", r); \
1790 }
1791
1792 #define ole_expect(expr, expect) { \
1793     HRESULT r = expr; \
1794     ok(r == (expect), #expr " returned %x, expected %x\n", r, expect); \
1795 }
1796
1797 #define double_eq(x, y) ok((x)-(y)<=1e-14*(x) && (x)-(y)>=-1e-14*(x), "expected %.16g, got %.16g\n", x, y)
1798
1799 static void* _create_object(const GUID *clsid, const char *name, const IID *iid, int line)
1800 {
1801     void *obj = NULL;
1802     HRESULT hr;
1803
1804     hr = CoCreateInstance(clsid, NULL, CLSCTX_INPROC_SERVER, iid, &obj);
1805     if (hr != S_OK)
1806         win_skip_(__FILE__,line)("failed to create %s instance: 0x%08x\n", name, hr);
1807
1808     return obj;
1809 }
1810
1811 #define _create(cls) cls, #cls
1812
1813 #define create_document(iid) _create_object(&_create(CLSID_DOMDocument), iid, __LINE__)
1814 #define create_document_version(v, iid) _create_object(&_create(CLSID_DOMDocument ## v), iid, __LINE__)
1815 #define create_cache(iid) _create_object(&_create(CLSID_XMLSchemaCache), iid, __LINE__)
1816 #define create_cache_version(v, iid) _create_object(&_create(CLSID_XMLSchemaCache ## v), iid, __LINE__)
1817 #define create_xsltemplate(iid) _create_object(&_create(CLSID_XSLTemplate), iid, __LINE__)
1818
1819 static BSTR alloc_str_from_narrow(const char *str)
1820 {
1821     int len = MultiByteToWideChar(CP_ACP, 0, str, -1, NULL, 0);
1822     BSTR ret = SysAllocStringLen(NULL, len - 1);  /* NUL character added automatically */
1823     MultiByteToWideChar(CP_ACP, 0, str, -1, ret, len);
1824     return ret;
1825 }
1826
1827 static BSTR alloced_bstrs[256];
1828 static int alloced_bstrs_count;
1829
1830 static BSTR _bstr_(const char *str)
1831 {
1832     assert(alloced_bstrs_count < sizeof(alloced_bstrs)/sizeof(alloced_bstrs[0]));
1833     alloced_bstrs[alloced_bstrs_count] = alloc_str_from_narrow(str);
1834     return alloced_bstrs[alloced_bstrs_count++];
1835 }
1836
1837 static void free_bstrs(void)
1838 {
1839     int i;
1840     for (i = 0; i < alloced_bstrs_count; i++)
1841         SysFreeString(alloced_bstrs[i]);
1842     alloced_bstrs_count = 0;
1843 }
1844
1845 static VARIANT _variantbstr_(const char *str)
1846 {
1847     VARIANT v;
1848     V_VT(&v) = VT_BSTR;
1849     V_BSTR(&v) = _bstr_(str);
1850     return v;
1851 }
1852
1853 static BOOL compareIgnoreReturns(BSTR sLeft, BSTR sRight)
1854 {
1855     for (;;)
1856     {
1857         while (*sLeft == '\r' || *sLeft == '\n') sLeft++;
1858         while (*sRight == '\r' || *sRight == '\n') sRight++;
1859         if (*sLeft != *sRight) return FALSE;
1860         if (!*sLeft) return TRUE;
1861         sLeft++;
1862         sRight++;
1863     }
1864 }
1865
1866 static void get_str_for_type(DOMNodeType type, char *buf)
1867 {
1868     switch (type)
1869     {
1870         case NODE_ATTRIBUTE:
1871             strcpy(buf, "A");
1872             break;
1873         case NODE_ELEMENT:
1874             strcpy(buf, "E");
1875             break;
1876         case NODE_DOCUMENT:
1877             strcpy(buf, "D");
1878             break;
1879         case NODE_TEXT:
1880             strcpy(buf, "T");
1881             break;
1882         case NODE_COMMENT:
1883             strcpy(buf, "C");
1884             break;
1885         case NODE_PROCESSING_INSTRUCTION:
1886             strcpy(buf, "P");
1887             break;
1888         default:
1889             wsprintfA(buf, "[%d]", type);
1890     }
1891 }
1892
1893 #define test_disp(u) _test_disp(__LINE__,u)
1894 static void _test_disp(unsigned line, IUnknown *unk)
1895 {
1896     DISPID dispid = DISPID_XMLDOM_NODELIST_RESET;
1897     IDispatchEx *dispex;
1898     DWORD dwProps = 0;
1899     BSTR sName;
1900     UINT ticnt;
1901     IUnknown *pUnk;
1902     HRESULT hres;
1903
1904     hres = IUnknown_QueryInterface(unk, &IID_IDispatchEx, (void**)&dispex);
1905     ok_(__FILE__,line) (hres == S_OK, "Could not get IDispatch: %08x\n", hres);
1906     if(FAILED(hres))
1907         return;
1908
1909     ticnt = 0xdeadbeef;
1910     hres = IDispatchEx_GetTypeInfoCount(dispex, &ticnt);
1911     ok_(__FILE__,line) (hres == S_OK, "GetTypeInfoCount failed: %08x\n", hres);
1912     ok_(__FILE__,line) (ticnt == 1, "ticnt=%u\n", ticnt);
1913
1914     sName = SysAllocString( szstar );
1915     hres = IDispatchEx_DeleteMemberByName(dispex, sName, fdexNameCaseSensitive);
1916     ok(hres == E_NOTIMPL, "expected E_NOTIMPL got %08x\n", hres);
1917     SysFreeString( sName );
1918
1919     hres = IDispatchEx_DeleteMemberByDispID(dispex, dispid);
1920     ok(hres == E_NOTIMPL, "expected E_NOTIMPL got %08x\n", hres);
1921
1922     hres = IDispatchEx_GetMemberProperties(dispex, dispid, grfdexPropCanAll, &dwProps);
1923     ok(hres == E_NOTIMPL, "expected E_NOTIMPL got %08x\n", hres);
1924     ok(dwProps == 0, "expected 0 got %d\n", dwProps);
1925
1926     hres = IDispatchEx_GetMemberName(dispex, dispid, &sName);
1927     ok(hres == E_NOTIMPL, "expected E_NOTIMPL got %08x\n", hres);
1928     if(SUCCEEDED(hres))
1929         SysFreeString(sName);
1930
1931     hres = IDispatchEx_GetNextDispID(dispex, fdexEnumDefault, DISPID_XMLDOM_NODELIST_RESET, &dispid);
1932     ok(hres == E_NOTIMPL, "expected E_NOTIMPL got %08x\n", hres);
1933
1934     hres = IDispatchEx_GetNameSpaceParent(dispex, &pUnk);
1935     ok(hres == E_NOTIMPL, "expected E_NOTIMPL got %08x\n", hres);
1936     if(hres == S_OK && pUnk)
1937         IUnknown_Release(pUnk);
1938
1939     IDispatchEx_Release(dispex);
1940 }
1941
1942 static int get_node_position(IXMLDOMNode *node)
1943 {
1944     HRESULT r;
1945     int pos = 0;
1946
1947     IXMLDOMNode_AddRef(node);
1948     do
1949     {
1950         IXMLDOMNode *new_node;
1951
1952         pos++;
1953         r = IXMLDOMNode_get_previousSibling(node, &new_node);
1954         ok(SUCCEEDED(r), "get_previousSibling failed\n");
1955         IXMLDOMNode_Release(node);
1956         node = new_node;
1957     } while (r == S_OK);
1958     return pos;
1959 }
1960
1961 static void node_to_string(IXMLDOMNode *node, char *buf)
1962 {
1963     HRESULT r = S_OK;
1964     DOMNodeType type;
1965
1966     if (node == NULL)
1967     {
1968         lstrcpyA(buf, "(null)");
1969         return;
1970     }
1971
1972     IXMLDOMNode_AddRef(node);
1973     while (r == S_OK)
1974     {
1975         IXMLDOMNode *new_node;
1976
1977         ole_check(IXMLDOMNode_get_nodeType(node, &type));
1978         get_str_for_type(type, buf);
1979         buf+=strlen(buf);
1980
1981         if (type == NODE_ATTRIBUTE)
1982         {
1983             BSTR bstr;
1984             ole_check(IXMLDOMNode_get_nodeName(node, &bstr));
1985             *(buf++) = '\'';
1986             wsprintfA(buf, "%ws", bstr);
1987             buf += strlen(buf);
1988             *(buf++) = '\'';
1989             SysFreeString(bstr);
1990
1991             r = IXMLDOMNode_selectSingleNode(node, _bstr_(".."), &new_node);
1992         }
1993         else
1994         {
1995             r = IXMLDOMNode_get_parentNode(node, &new_node);
1996             wsprintf(buf, "%d", get_node_position(node));
1997             buf += strlen(buf);
1998         }
1999
2000         ok(SUCCEEDED(r), "get_parentNode failed (%08x)\n", r);
2001         IXMLDOMNode_Release(node);
2002         node = new_node;
2003         if (r == S_OK)
2004             *(buf++) = '.';
2005     }
2006
2007     *buf = 0;
2008 }
2009
2010 static char *list_to_string(IXMLDOMNodeList *list)
2011 {
2012     static char buf[4096];
2013     char *pos = buf;
2014     LONG len = 0;
2015     int i;
2016
2017     if (list == NULL)
2018     {
2019         lstrcpyA(buf, "(null)");
2020         return buf;
2021     }
2022     ole_check(IXMLDOMNodeList_get_length(list, &len));
2023     for (i = 0; i < len; i++)
2024     {
2025         IXMLDOMNode *node;
2026         if (i > 0)
2027             *(pos++) = ' ';
2028         ole_check(IXMLDOMNodeList_nextNode(list, &node));
2029         node_to_string(node, pos);
2030         pos += strlen(pos);
2031         IXMLDOMNode_Release(node);
2032     }
2033     *pos = 0;
2034     return buf;
2035 }
2036
2037 #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); }
2038 #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); }
2039
2040 static void test_domdoc( void )
2041 {
2042     HRESULT r;
2043     IXMLDOMDocument *doc;
2044     IXMLDOMParseError *error;
2045     IXMLDOMElement *element = NULL;
2046     IXMLDOMNode *node;
2047     IXMLDOMText *nodetext = NULL;
2048     IXMLDOMComment *node_comment = NULL;
2049     IXMLDOMAttribute *node_attr = NULL;
2050     IXMLDOMNode *nodeChild = NULL;
2051     IXMLDOMProcessingInstruction *nodePI = NULL;
2052     ISupportErrorInfo *support_error = NULL;
2053     VARIANT_BOOL b;
2054     VARIANT var;
2055     BSTR str;
2056     LONG code;
2057     LONG nLength = 0;
2058     WCHAR buff[100];
2059
2060     doc = create_document(&IID_IXMLDOMDocument);
2061     if (!doc) return;
2062
2063     test_disp((IUnknown*)doc);
2064
2065 if (0)
2066 {
2067     /* crashes on native */
2068     IXMLDOMDocument_loadXML( doc, (BSTR)0x1, NULL );
2069 }
2070
2071     /* try some stupid things */
2072     r = IXMLDOMDocument_loadXML( doc, NULL, NULL );
2073     ok( r == S_FALSE, "loadXML succeeded\n");
2074
2075     b = VARIANT_TRUE;
2076     r = IXMLDOMDocument_loadXML( doc, NULL, &b );
2077     ok( r == S_FALSE, "loadXML succeeded\n");
2078     ok( b == VARIANT_FALSE, "failed to load XML string\n");
2079
2080     /* try to load a document from a nonexistent file */
2081     b = VARIANT_TRUE;
2082     str = SysAllocString( szNonExistentFile );
2083     VariantInit(&var);
2084     V_VT(&var) = VT_BSTR;
2085     V_BSTR(&var) = str;
2086
2087     r = IXMLDOMDocument_load( doc, var, &b);
2088     ok( r == S_FALSE, "loadXML succeeded\n");
2089     ok( b == VARIANT_FALSE, "succeeded in loading XML string\n");
2090     SysFreeString( str );
2091
2092     /* try load an empty document */
2093     b = VARIANT_TRUE;
2094     str = SysAllocString( szEmpty );
2095     r = IXMLDOMDocument_loadXML( doc, str, &b );
2096     ok( r == S_FALSE, "loadXML succeeded\n");
2097     ok( b == VARIANT_FALSE, "succeeded in loading XML string\n");
2098     SysFreeString( str );
2099
2100     r = IXMLDOMDocument_get_async( doc, &b );
2101     ok( r == S_OK, "get_async failed (%08x)\n", r);
2102     ok( b == VARIANT_TRUE, "Wrong default value\n");
2103
2104     /* check that there's no document element */
2105     element = NULL;
2106     r = IXMLDOMDocument_get_documentElement( doc, &element );
2107     ok( r == S_FALSE, "should be no document element\n");
2108
2109     /* try finding a node */
2110     node = NULL;
2111     str = SysAllocString( szstr1 );
2112     r = IXMLDOMDocument_selectSingleNode( doc, str, &node );
2113     ok( r == S_FALSE, "ret %08x\n", r );
2114     SysFreeString( str );
2115
2116     b = VARIANT_TRUE;
2117     str = SysAllocString( szIncomplete );
2118     r = IXMLDOMDocument_loadXML( doc, str, &b );
2119     ok( r == S_FALSE, "loadXML succeeded\n");
2120     ok( b == VARIANT_FALSE, "succeeded in loading XML string\n");
2121     SysFreeString( str );
2122
2123     /* check that there's no document element */
2124     element = (IXMLDOMElement*)1;
2125     r = IXMLDOMDocument_get_documentElement( doc, &element );
2126     ok( r == S_FALSE, "should be no document element\n");
2127     ok( element == NULL, "Element should be NULL\n");
2128
2129     /* test for BSTR handling, pass broken BSTR */
2130     memcpy(&buff[2], szComplete1, sizeof(szComplete1));
2131     /* just a big length */
2132     *(DWORD*)buff = 0xf0f0;
2133     b = VARIANT_FALSE;
2134     r = IXMLDOMDocument_loadXML( doc, &buff[2], &b );
2135     ok( r == S_OK, "loadXML failed\n");
2136     ok( b == VARIANT_TRUE, "failed to load XML string\n");
2137
2138     /* loadXML ignores the encoding attribute and always expects Unicode */
2139     b = VARIANT_FALSE;
2140     str = SysAllocString( szComplete6 );
2141     r = IXMLDOMDocument_loadXML( doc, str, &b );
2142     ok( r == S_OK, "loadXML failed\n");
2143     ok( b == VARIANT_TRUE, "failed to load XML string\n");
2144     SysFreeString( str );
2145
2146     /* try a BSTR containing a Windows-1252 document */
2147     b = VARIANT_TRUE;
2148     str = SysAllocStringByteLen( szNonUnicodeXML, sizeof(szNonUnicodeXML) - 1 );
2149     r = IXMLDOMDocument_loadXML( doc, str, &b );
2150     ok( r == S_FALSE, "loadXML succeeded\n");
2151     ok( b == VARIANT_FALSE, "succeeded in loading XML string\n");
2152     SysFreeString( str );
2153
2154     /* try to load something valid */
2155     b = VARIANT_FALSE;
2156     str = SysAllocString( szComplete1 );
2157     r = IXMLDOMDocument_loadXML( doc, str, &b );
2158     ok( r == S_OK, "loadXML failed\n");
2159     ok( b == VARIANT_TRUE, "failed to load XML string\n");
2160     SysFreeString( str );
2161
2162     /* check if nodename is correct */
2163     r = IXMLDOMDocument_get_nodeName( doc, NULL );
2164     ok ( r == E_INVALIDARG, "get_nodeName (NULL) wrong code\n");
2165
2166     str = (BSTR)0xdeadbeef;
2167     r = IXMLDOMDocument_get_baseName( doc, &str );
2168     ok ( r == S_FALSE, "got 0x%08x\n", r);
2169     ok (str == NULL, "got %p\n", str);
2170
2171     /* content doesn't matter here */
2172     str = NULL;
2173     r = IXMLDOMDocument_get_nodeName( doc, &str );
2174     ok ( r == S_OK, "get_nodeName wrong code\n");
2175     ok ( str != NULL, "str is null\n");
2176     ok( !lstrcmpW( str, szDocument ), "incorrect nodeName\n");
2177     SysFreeString( str );
2178
2179     /* test put_text */
2180     r = IXMLDOMDocument_put_text( doc, _bstr_("Should Fail") );
2181     ok( r == E_FAIL, "ret %08x\n", r );
2182
2183     /* check that there's a document element */
2184     element = NULL;
2185     r = IXMLDOMDocument_get_documentElement( doc, &element );
2186     ok( r == S_OK, "should be a document element\n");
2187     if( element )
2188     {
2189         IObjectIdentity *ident;
2190
2191         test_disp((IUnknown*)element);
2192
2193         r = IXMLDOMElement_QueryInterface( element, &IID_IObjectIdentity, (void**)&ident );
2194         ok( r == E_NOINTERFACE, "ret %08x\n", r);
2195
2196         IXMLDOMElement_Release( element );
2197         element = NULL;
2198     }
2199
2200     /* as soon as we call loadXML again, the document element will disappear */
2201     b = 2;
2202     r = IXMLDOMDocument_loadXML( doc, NULL, NULL );
2203     ok( r == S_FALSE, "loadXML failed\n");
2204     ok( b == 2, "variant modified\n");
2205     r = IXMLDOMDocument_get_documentElement( doc, &element );
2206     ok( r == S_FALSE, "should be no document element\n");
2207
2208     /* try to load something else simple and valid */
2209     b = VARIANT_FALSE;
2210     str = SysAllocString( szComplete3 );
2211     r = IXMLDOMDocument_loadXML( doc, str, &b );
2212     ok( r == S_OK, "loadXML failed\n");
2213     ok( b == VARIANT_TRUE, "failed to load XML string\n");
2214     SysFreeString( str );
2215
2216     /* try something a little more complicated */
2217     b = FALSE;
2218     str = SysAllocString( szComplete4 );
2219     r = IXMLDOMDocument_loadXML( doc, str, &b );
2220     ok( r == S_OK, "loadXML failed\n");
2221     ok( b == VARIANT_TRUE, "failed to load XML string\n");
2222     SysFreeString( str );
2223
2224     r = IXMLDOMDocument_get_parseError( doc, &error );
2225     ok( r == S_OK, "returns %08x\n", r );
2226
2227     r = IXMLDOMParseError_get_errorCode( error, &code );
2228     ok( r == S_FALSE, "returns %08x\n", r );
2229     ok( code == 0, "code %d\n", code );
2230     IXMLDOMParseError_Release( error );
2231
2232     /* test createTextNode */
2233     r = IXMLDOMDocument_createTextNode(doc, _bstr_(""), &nodetext);
2234     ok( r == S_OK, "returns %08x\n", r );
2235     IXMLDOMText_Release(nodetext);
2236
2237     str = SysAllocString( szOpen );
2238     r = IXMLDOMDocument_createTextNode(doc, str, NULL);
2239     ok( r == E_INVALIDARG, "returns %08x\n", r );
2240     r = IXMLDOMDocument_createTextNode(doc, str, &nodetext);
2241     ok( r == S_OK, "returns %08x\n", r );
2242     SysFreeString( str );
2243     if(nodetext)
2244     {
2245         r = IXMLDOMText_QueryInterface(nodetext, &IID_IXMLDOMElement, (void**)&element);
2246         ok(r == E_NOINTERFACE, "ret %08x\n", r );
2247
2248         /* Text Last Child Checks */
2249         r = IXMLDOMText_get_lastChild(nodetext, NULL);
2250         ok(r == E_INVALIDARG, "ret %08x\n", r );
2251
2252         nodeChild = (IXMLDOMNode*)0x1;
2253         r = IXMLDOMText_get_lastChild(nodetext, &nodeChild);
2254         ok(r == S_FALSE, "ret %08x\n", r );
2255         ok(nodeChild == NULL, "nodeChild not NULL\n");
2256
2257         /* test length property */
2258         r = IXMLDOMText_get_length(nodetext, NULL);
2259         ok(r == E_INVALIDARG, "ret %08x\n", r );
2260
2261         r = IXMLDOMText_get_length(nodetext, &nLength);
2262         ok(r == S_OK, "ret %08x\n", r );
2263         ok(nLength == 4, "expected 4 got %d\n", nLength);
2264
2265         /* put data Tests */
2266         r = IXMLDOMText_put_data(nodetext, _bstr_("This &is a ; test <>\\"));
2267         ok(r == S_OK, "ret %08x\n", r );
2268
2269         /* get data Tests */
2270         r = IXMLDOMText_get_data(nodetext, &str);
2271         ok(r == S_OK, "ret %08x\n", r );
2272         ok( !lstrcmpW( str, _bstr_("This &is a ; test <>\\") ), "incorrect put_data string\n");
2273         SysFreeString(str);
2274
2275         /* Confirm XML text is good */
2276         r = IXMLDOMText_get_xml(nodetext, &str);
2277         ok(r == S_OK, "ret %08x\n", r );
2278         ok( !lstrcmpW( str, _bstr_("This &amp;is a ; test &lt;&gt;\\") ), "incorrect xml string\n");
2279         SysFreeString(str);
2280
2281         /* Confirm we get the put_data Text back */
2282         r = IXMLDOMText_get_text(nodetext, &str);
2283         ok(r == S_OK, "ret %08x\n", r );
2284         ok( !lstrcmpW( str, _bstr_("This &is a ; test <>\\") ), "incorrect xml string\n");
2285         SysFreeString(str);
2286
2287         /* test substringData */
2288         r = IXMLDOMText_substringData(nodetext, 0, 4, NULL);
2289         ok(r == E_INVALIDARG, "ret %08x\n", r );
2290
2291         /* test substringData - Invalid offset */
2292         str = (BSTR)&szElement;
2293         r = IXMLDOMText_substringData(nodetext, -1, 4, &str);
2294         ok(r == E_INVALIDARG, "ret %08x\n", r );
2295         ok( str == NULL, "incorrect string\n");
2296
2297         /* test substringData - Invalid offset */
2298         str = (BSTR)&szElement;
2299         r = IXMLDOMText_substringData(nodetext, 30, 0, &str);
2300         ok(r == S_FALSE, "ret %08x\n", r );
2301         ok( str == NULL, "incorrect string\n");
2302
2303         /* test substringData - Invalid size */
2304         str = (BSTR)&szElement;
2305         r = IXMLDOMText_substringData(nodetext, 0, -1, &str);
2306         ok(r == E_INVALIDARG, "ret %08x\n", r );
2307         ok( str == NULL, "incorrect string\n");
2308
2309         /* test substringData - Invalid size */
2310         str = (BSTR)&szElement;
2311         r = IXMLDOMText_substringData(nodetext, 2, 0, &str);
2312         ok(r == S_FALSE, "ret %08x\n", r );
2313         ok( str == NULL, "incorrect string\n");
2314
2315         /* test substringData - Start of string */
2316         r = IXMLDOMText_substringData(nodetext, 0, 4, &str);
2317         ok(r == S_OK, "ret %08x\n", r );
2318         ok( !lstrcmpW( str, _bstr_("This") ), "incorrect substringData string\n");
2319         SysFreeString(str);
2320
2321         /* test substringData - Middle of string */
2322         r = IXMLDOMText_substringData(nodetext, 13, 4, &str);
2323         ok(r == S_OK, "ret %08x\n", r );
2324         ok( !lstrcmpW( str, _bstr_("test") ), "incorrect substringData string\n");
2325         SysFreeString(str);
2326
2327         /* test substringData - End of string */
2328         r = IXMLDOMText_substringData(nodetext, 20, 4, &str);
2329         ok(r == S_OK, "ret %08x\n", r );
2330         ok( !lstrcmpW( str, _bstr_("\\") ), "incorrect substringData string\n");
2331         SysFreeString(str);
2332
2333         /* test appendData */
2334         r = IXMLDOMText_appendData(nodetext, NULL);
2335         ok(r == S_OK, "ret %08x\n", r );
2336
2337         r = IXMLDOMText_appendData(nodetext, _bstr_(""));
2338         ok(r == S_OK, "ret %08x\n", r );
2339
2340         r = IXMLDOMText_appendData(nodetext, _bstr_("Append"));
2341         ok(r == S_OK, "ret %08x\n", r );
2342
2343         r = IXMLDOMText_get_text(nodetext, &str);
2344         ok(r == S_OK, "ret %08x\n", r );
2345         ok( !lstrcmpW( str, _bstr_("This &is a ; test <>\\Append") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
2346         SysFreeString(str);
2347
2348         /* test insertData */
2349         str = SysAllocStringLen(NULL, 0);
2350         r = IXMLDOMText_insertData(nodetext, -1, str);
2351         ok(r == S_OK, "ret %08x\n", r );
2352
2353         r = IXMLDOMText_insertData(nodetext, -1, NULL);
2354         ok(r == S_OK, "ret %08x\n", r );
2355
2356         r = IXMLDOMText_insertData(nodetext, 1000, str);
2357         ok(r == S_OK, "ret %08x\n", r );
2358
2359         r = IXMLDOMText_insertData(nodetext, 1000, NULL);
2360         ok(r == S_OK, "ret %08x\n", r );
2361
2362         r = IXMLDOMText_insertData(nodetext, 0, NULL);
2363         ok(r == S_OK, "ret %08x\n", r );
2364
2365         r = IXMLDOMText_insertData(nodetext, 0, str);
2366         ok(r == S_OK, "ret %08x\n", r );
2367         SysFreeString(str);
2368
2369         r = IXMLDOMText_insertData(nodetext, -1, _bstr_("Inserting"));
2370         ok(r == E_INVALIDARG, "ret %08x\n", r );
2371
2372         r = IXMLDOMText_insertData(nodetext, 1000, _bstr_("Inserting"));
2373         ok(r == E_INVALIDARG, "ret %08x\n", r );
2374
2375         r = IXMLDOMText_insertData(nodetext, 0, _bstr_("Begin "));
2376         ok(r == S_OK, "ret %08x\n", r );
2377
2378         r = IXMLDOMText_insertData(nodetext, 17, _bstr_("Middle"));
2379         ok(r == S_OK, "ret %08x\n", r );
2380
2381         r = IXMLDOMText_insertData(nodetext, 39, _bstr_(" End"));
2382         ok(r == S_OK, "ret %08x\n", r );
2383
2384         r = IXMLDOMText_get_text(nodetext, &str);
2385         ok(r == S_OK, "ret %08x\n", r );
2386         ok( !lstrcmpW( str, _bstr_("Begin This &is a Middle; test <>\\Append End") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
2387         SysFreeString(str);
2388
2389         /* delete data */
2390         /* invalid arguments */
2391         r = IXMLDOMText_deleteData(nodetext, -1, 1);
2392         ok(r == E_INVALIDARG, "ret %08x\n", r );
2393
2394         r = IXMLDOMText_deleteData(nodetext, 0, 0);
2395         ok(r == S_OK, "ret %08x\n", r );
2396
2397         r = IXMLDOMText_deleteData(nodetext, 0, -1);
2398         ok(r == E_INVALIDARG, "ret %08x\n", r );
2399
2400         r = IXMLDOMText_get_length(nodetext, &nLength);
2401         ok(r == S_OK, "ret %08x\n", r );
2402         ok(nLength == 43, "expected 43 got %d\n", nLength);
2403
2404         r = IXMLDOMText_deleteData(nodetext, nLength, 1);
2405         ok(r == S_OK, "ret %08x\n", r );
2406
2407         r = IXMLDOMText_deleteData(nodetext, nLength+1, 1);
2408         ok(r == E_INVALIDARG, "ret %08x\n", r );
2409
2410         /* delete from start */
2411         r = IXMLDOMText_deleteData(nodetext, 0, 5);
2412         ok(r == S_OK, "ret %08x\n", r );
2413
2414         r = IXMLDOMText_get_length(nodetext, &nLength);
2415         ok(r == S_OK, "ret %08x\n", r );
2416         ok(nLength == 38, "expected 38 got %d\n", nLength);
2417
2418         r = IXMLDOMText_get_text(nodetext, &str);
2419         ok(r == S_OK, "ret %08x\n", r );
2420         ok( !lstrcmpW( str, _bstr_("This &is a Middle; test <>\\Append End") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
2421         SysFreeString(str);
2422
2423         /* delete from end */
2424         r = IXMLDOMText_deleteData(nodetext, 35, 3);
2425         ok(r == S_OK, "ret %08x\n", r );
2426
2427         r = IXMLDOMText_get_length(nodetext, &nLength);
2428         ok(r == S_OK, "ret %08x\n", r );
2429         ok(nLength == 35, "expected 35 got %d\n", nLength);
2430
2431         r = IXMLDOMText_get_text(nodetext, &str);
2432         ok(r == S_OK, "ret %08x\n", r );
2433         ok( !lstrcmpW( str, _bstr_("This &is a Middle; test <>\\Append") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
2434         SysFreeString(str);
2435
2436         /* delete from inside */
2437         r = IXMLDOMText_deleteData(nodetext, 1, 33);
2438         ok(r == S_OK, "ret %08x\n", r );
2439
2440         r = IXMLDOMText_get_length(nodetext, &nLength);
2441         ok(r == S_OK, "ret %08x\n", r );
2442         ok(nLength == 2, "expected 2 got %d\n", nLength);
2443
2444         r = IXMLDOMText_get_text(nodetext, &str);
2445         ok(r == S_OK, "ret %08x\n", r );
2446         ok( !lstrcmpW( str, _bstr_("") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
2447         SysFreeString(str);
2448
2449         /* delete whole data ... */
2450         r = IXMLDOMText_get_length(nodetext, &nLength);
2451         ok(r == S_OK, "ret %08x\n", r );
2452
2453         r = IXMLDOMText_deleteData(nodetext, 0, nLength);
2454         ok(r == S_OK, "ret %08x\n", r );
2455         /* ... and try again with empty string */
2456         r = IXMLDOMText_deleteData(nodetext, 0, nLength);
2457         ok(r == S_OK, "ret %08x\n", r );
2458
2459         /* test put_data */
2460         V_VT(&var) = VT_BSTR;
2461         V_BSTR(&var) = SysAllocString(szstr1);
2462         r = IXMLDOMText_put_nodeValue(nodetext, var);
2463         ok(r == S_OK, "ret %08x\n", r );
2464         VariantClear(&var);
2465
2466         r = IXMLDOMText_get_text(nodetext, &str);
2467         ok(r == S_OK, "ret %08x\n", r );
2468         ok( !lstrcmpW( str, szstr1 ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
2469         SysFreeString(str);
2470
2471         /* test put_data */
2472         V_VT(&var) = VT_I4;
2473         V_I4(&var) = 99;
2474         r = IXMLDOMText_put_nodeValue(nodetext, var);
2475         ok(r == S_OK, "ret %08x\n", r );
2476         VariantClear(&var);
2477
2478         r = IXMLDOMText_get_text(nodetext, &str);
2479         ok(r == S_OK, "ret %08x\n", r );
2480         ok( !lstrcmpW( str, _bstr_("99") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
2481         SysFreeString(str);
2482
2483         /* ::replaceData() */
2484         V_VT(&var) = VT_BSTR;
2485         V_BSTR(&var) = SysAllocString(szstr1);
2486         r = IXMLDOMText_put_nodeValue(nodetext, var);
2487         ok(r == S_OK, "ret %08x\n", r );
2488         VariantClear(&var);
2489
2490         r = IXMLDOMText_replaceData(nodetext, 6, 0, NULL);
2491         ok(r == E_INVALIDARG, "ret %08x\n", r );
2492         r = IXMLDOMText_get_text(nodetext, &str);
2493         ok(r == S_OK, "ret %08x\n", r );
2494         ok( !lstrcmpW( str, _bstr_("str1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
2495         SysFreeString(str);
2496
2497         r = IXMLDOMText_replaceData(nodetext, 0, 0, NULL);
2498         ok(r == S_OK, "ret %08x\n", r );
2499         r = IXMLDOMText_get_text(nodetext, &str);
2500         ok(r == S_OK, "ret %08x\n", r );
2501         ok( !lstrcmpW( str, _bstr_("str1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
2502         SysFreeString(str);
2503
2504         /* NULL pointer means delete */
2505         r = IXMLDOMText_replaceData(nodetext, 0, 1, NULL);
2506         ok(r == S_OK, "ret %08x\n", r );
2507         r = IXMLDOMText_get_text(nodetext, &str);
2508         ok(r == S_OK, "ret %08x\n", r );
2509         ok( !lstrcmpW( str, _bstr_("tr1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
2510         SysFreeString(str);
2511
2512         /* empty string means delete */
2513         r = IXMLDOMText_replaceData(nodetext, 0, 1, _bstr_(""));
2514         ok(r == S_OK, "ret %08x\n", r );
2515         r = IXMLDOMText_get_text(nodetext, &str);
2516         ok(r == S_OK, "ret %08x\n", r );
2517         ok( !lstrcmpW( str, _bstr_("r1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
2518         SysFreeString(str);
2519
2520         /* zero count means insert */
2521         r = IXMLDOMText_replaceData(nodetext, 0, 0, _bstr_("a"));
2522         ok(r == S_OK, "ret %08x\n", r );
2523         r = IXMLDOMText_get_text(nodetext, &str);
2524         ok(r == S_OK, "ret %08x\n", r );
2525         ok( !lstrcmpW( str, _bstr_("ar1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
2526         SysFreeString(str);
2527
2528         r = IXMLDOMText_replaceData(nodetext, 0, 2, NULL);
2529         ok(r == S_OK, "ret %08x\n", r );
2530
2531         r = IXMLDOMText_insertData(nodetext, 0, _bstr_("m"));
2532         ok(r == S_OK, "ret %08x\n", r );
2533         r = IXMLDOMText_get_text(nodetext, &str);
2534         ok(r == S_OK, "ret %08x\n", r );
2535         ok( !lstrcmpW( str, _bstr_("m1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
2536         SysFreeString(str);
2537
2538         /* nonempty string, count greater than its length */
2539         r = IXMLDOMText_replaceData(nodetext, 0, 2, _bstr_("a1.2"));
2540         ok(r == S_OK, "ret %08x\n", r );
2541         r = IXMLDOMText_get_text(nodetext, &str);
2542         ok(r == S_OK, "ret %08x\n", r );
2543         ok( !lstrcmpW( str, _bstr_("a1.2") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
2544         SysFreeString(str);
2545
2546         /* nonempty string, count less than its length */
2547         r = IXMLDOMText_replaceData(nodetext, 0, 1, _bstr_("wine"));
2548         ok(r == S_OK, "ret %08x\n", r );
2549         r = IXMLDOMText_get_text(nodetext, &str);
2550         ok(r == S_OK, "ret %08x\n", r );
2551         ok( !lstrcmpW( str, _bstr_("wine1.2") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
2552         SysFreeString(str);
2553
2554         IXMLDOMText_Release( nodetext );
2555     }
2556
2557     /* test Create Comment */
2558     r = IXMLDOMDocument_createComment(doc, NULL, NULL);
2559     ok( r == E_INVALIDARG, "returns %08x\n", r );
2560     node_comment = (IXMLDOMComment*)0x1;
2561
2562     /* empty comment */
2563     r = IXMLDOMDocument_createComment(doc, _bstr_(""), &node_comment);
2564     ok( r == S_OK, "returns %08x\n", r );
2565     str = (BSTR)0x1;
2566     r = IXMLDOMComment_get_data(node_comment, &str);
2567     ok( r == S_OK, "returns %08x\n", r );
2568     ok( str && SysStringLen(str) == 0, "expected empty string data\n");
2569     IXMLDOMComment_Release(node_comment);
2570     SysFreeString(str);
2571
2572     r = IXMLDOMDocument_createComment(doc, NULL, &node_comment);
2573     ok( r == S_OK, "returns %08x\n", r );
2574     str = (BSTR)0x1;
2575     r = IXMLDOMComment_get_data(node_comment, &str);
2576     ok( r == S_OK, "returns %08x\n", r );
2577     ok( str && (SysStringLen(str) == 0), "expected empty string data\n");
2578     IXMLDOMComment_Release(node_comment);
2579     SysFreeString(str);
2580
2581     str = SysAllocString(szComment);
2582     r = IXMLDOMDocument_createComment(doc, str, &node_comment);
2583     SysFreeString(str);
2584     ok( r == S_OK, "returns %08x\n", r );
2585     if(node_comment)
2586     {
2587         /* Last Child Checks */
2588         r = IXMLDOMComment_get_lastChild(node_comment, NULL);
2589         ok(r == E_INVALIDARG, "ret %08x\n", r );
2590
2591         nodeChild = (IXMLDOMNode*)0x1;
2592         r = IXMLDOMComment_get_lastChild(node_comment, &nodeChild);
2593         ok(r == S_FALSE, "ret %08x\n", r );
2594         ok(nodeChild == NULL, "pLastChild not NULL\n");
2595
2596         /* baseName */
2597         str = (BSTR)0xdeadbeef;
2598         IXMLDOMComment_get_baseName(node_comment, &str);
2599         ok(r == S_FALSE, "ret %08x\n", r );
2600         ok(str == NULL, "Expected NULL\n");
2601
2602         IXMLDOMComment_Release( node_comment );
2603     }
2604
2605     /* test Create Attribute */
2606     str = SysAllocString(szAttribute);
2607     r = IXMLDOMDocument_createAttribute(doc, NULL, NULL);
2608     ok( r == E_INVALIDARG, "returns %08x\n", r );
2609     r = IXMLDOMDocument_createAttribute(doc, str, &node_attr);
2610     ok( r == S_OK, "returns %08x\n", r );
2611     IXMLDOMText_Release( node_attr);
2612     SysFreeString(str);
2613
2614     /* test Processing Instruction */
2615     str = SysAllocStringLen(NULL, 0);
2616     r = IXMLDOMDocument_createProcessingInstruction(doc, str, str, NULL);
2617     ok( r == E_INVALIDARG, "returns %08x\n", r );
2618     r = IXMLDOMDocument_createProcessingInstruction(doc, NULL, str, &nodePI);
2619     ok( r == E_FAIL, "returns %08x\n", r );
2620     r = IXMLDOMDocument_createProcessingInstruction(doc, str, str, &nodePI);
2621     ok( r == E_FAIL, "returns %08x\n", r );
2622     SysFreeString(str);
2623
2624     r = IXMLDOMDocument_createProcessingInstruction(doc, _bstr_("xml"), _bstr_("version=\"1.0\""), &nodePI);
2625     ok( r == S_OK, "returns %08x\n", r );
2626     if(nodePI)
2627     {
2628         /* Last Child Checks */
2629         r = IXMLDOMProcessingInstruction_get_lastChild(nodePI, NULL);
2630         ok(r == E_INVALIDARG, "ret %08x\n", r );
2631
2632         nodeChild = (IXMLDOMNode*)0x1;
2633         r = IXMLDOMProcessingInstruction_get_lastChild(nodePI, &nodeChild);
2634         ok(r == S_FALSE, "ret %08x\n", r );
2635         ok(nodeChild == NULL, "nodeChild not NULL\n");
2636
2637         /* test nodeName */
2638         r = IXMLDOMProcessingInstruction_get_nodeName(nodePI, &str);
2639         ok(r == S_OK, "ret %08x\n", r );
2640         ok( !lstrcmpW( str, _bstr_("xml") ), "incorrect nodeName string\n");
2641         SysFreeString(str);
2642
2643         /* test baseName */
2644         str = (BSTR)0x1;
2645         r = IXMLDOMProcessingInstruction_get_baseName(nodePI, &str);
2646         ok(r == S_OK, "ret %08x\n", r );
2647         ok( !lstrcmpW( str, _bstr_("xml") ), "incorrect nodeName string\n");
2648         SysFreeString(str);
2649
2650         /* test Target */
2651         r = IXMLDOMProcessingInstruction_get_target(nodePI, &str);
2652         ok(r == S_OK, "ret %08x\n", r );
2653         ok( !lstrcmpW( str, _bstr_("xml") ), "incorrect target string\n");
2654         SysFreeString(str);
2655
2656         /* test get_nodeValue */
2657         r = IXMLDOMProcessingInstruction_get_nodeValue(nodePI, &var);
2658         ok(r == S_OK, "ret %08x\n", r );
2659         ok( !lstrcmpW( V_BSTR(&var), _bstr_("version=\"1.0\"") ), "incorrect data string\n");
2660         VariantClear(&var);
2661
2662         /* test get_data */
2663         r = IXMLDOMProcessingInstruction_get_data(nodePI, &str);
2664         ok(r == S_OK, "ret %08x\n", r );
2665         ok( !lstrcmpW( str, _bstr_("version=\"1.0\"") ), "incorrect data string\n");
2666         SysFreeString(str);
2667
2668         /* test put_data */
2669         r = IXMLDOMProcessingInstruction_put_data(nodePI, _bstr_("version=\"1.0\" encoding=\"UTF-8\""));
2670         ok(r == E_FAIL, "ret %08x\n", r );
2671
2672         /* test put_data */
2673         V_VT(&var) = VT_BSTR;
2674         V_BSTR(&var) = SysAllocString(szOpen);  /* Doesn't matter what the string is, cannot set an xml node. */
2675         r = IXMLDOMProcessingInstruction_put_nodeValue(nodePI, var);
2676         ok(r == E_FAIL, "ret %08x\n", r );
2677         VariantClear(&var);
2678
2679         /* test get nodeName */
2680         r = IXMLDOMProcessingInstruction_get_nodeName(nodePI, &str);
2681         ok( !lstrcmpW( str, _bstr_("xml") ), "incorrect nodeName string\n");
2682         ok(r == S_OK, "ret %08x\n", r );
2683         SysFreeString(str);
2684
2685         IXMLDOMProcessingInstruction_Release(nodePI);
2686     }
2687
2688     r = IXMLDOMDocument_QueryInterface( doc, &IID_ISupportErrorInfo, (void**)&support_error );
2689     ok( r == S_OK, "ret %08x\n", r );
2690     if(r == S_OK)
2691     {
2692         r = ISupportErrorInfo_InterfaceSupportsErrorInfo( support_error, &IID_IXMLDOMDocument );
2693         todo_wine ok( r == S_OK, "ret %08x\n", r );
2694         ISupportErrorInfo_Release( support_error );
2695     }
2696
2697     r = IXMLDOMDocument_Release( doc );
2698     ok( r == 0, "document ref count incorrect\n");
2699
2700     free_bstrs();
2701 }
2702
2703 static void test_persiststreaminit(void)
2704 {
2705     IXMLDOMDocument *doc;
2706     IPersistStreamInit *streaminit;
2707     HRESULT hr;
2708
2709     doc = create_document(&IID_IXMLDOMDocument);
2710     if (!doc) return;
2711
2712     hr = IXMLDOMDocument_QueryInterface(doc, &IID_IPersistStreamInit, (void**)&streaminit);
2713     ok( hr == S_OK, "failed with 0x%08x\n", hr );
2714
2715     hr = IPersistStreamInit_InitNew(streaminit);
2716     ok( hr == S_OK, "failed with 0x%08x\n", hr );
2717
2718     IXMLDOMDocument_Release(doc);
2719 }
2720
2721 static void test_domnode( void )
2722 {
2723     HRESULT r;
2724     IXMLDOMDocument *doc, *owner = NULL;
2725     IXMLDOMElement *element = NULL;
2726     IXMLDOMNamedNodeMap *map = NULL;
2727     IXMLDOMNode *node = NULL, *next = NULL;
2728     IXMLDOMNodeList *list = NULL;
2729     IXMLDOMAttribute *attr = NULL;
2730     DOMNodeType type = NODE_INVALID;
2731     VARIANT_BOOL b;
2732     BSTR str;
2733     VARIANT var;
2734     LONG count;
2735
2736     doc = create_document(&IID_IXMLDOMDocument);
2737     if (!doc) return;
2738
2739     b = FALSE;
2740     str = SysAllocString( szComplete4 );
2741     r = IXMLDOMDocument_loadXML( doc, str, &b );
2742     ok( r == S_OK, "loadXML failed\n");
2743     ok( b == VARIANT_TRUE, "failed to load XML string\n");
2744     SysFreeString( str );
2745
2746     EXPECT_CHILDREN(doc);
2747
2748     r = IXMLDOMDocument_get_documentElement( doc, &element );
2749     ok( r == S_OK, "should be a document element\n");
2750     ok( element != NULL, "should be an element\n");
2751
2752     VariantInit(&var);
2753     ok( V_VT(&var) == VT_EMPTY, "variant init failed\n");
2754
2755     r = IXMLDOMNode_get_nodeValue( doc, NULL );
2756     ok(r == E_INVALIDARG, "get_nodeValue ret %08x\n", r );
2757
2758     r = IXMLDOMNode_get_nodeValue( doc, &var );
2759     ok( r == S_FALSE, "nextNode returned wrong code\n");
2760     ok( V_VT(&var) == VT_NULL, "variant wasn't empty\n");
2761     ok( V_BSTR(&var) == NULL, "variant value wasn't null\n");
2762
2763     if (element)
2764     {
2765         owner = NULL;
2766         r = IXMLDOMNode_get_ownerDocument( element, &owner );
2767         ok( r == S_OK, "get_ownerDocument return code\n");
2768         ok( owner != doc, "get_ownerDocument return\n");
2769         IXMLDOMDocument_Release(owner);
2770
2771         type = NODE_INVALID;
2772         r = IXMLDOMNode_get_nodeType( element, &type);
2773         ok( r == S_OK, "got %08x\n", r);
2774         ok( type == NODE_ELEMENT, "node not an element\n");
2775
2776         str = NULL;
2777         r = IXMLDOMNode_get_baseName( element, &str );
2778         ok( r == S_OK, "get_baseName returned wrong code\n");
2779         ok( lstrcmpW(str,szlc) == 0, "basename was wrong\n");
2780         SysFreeString(str);
2781
2782         /* check if nodename is correct */
2783         r = IXMLDOMElement_get_nodeName( element, NULL );
2784         ok ( r == E_INVALIDARG, "get_nodeName (NULL) wrong code\n");
2785
2786         /* content doesn't matter here */
2787         str = NULL;
2788         r = IXMLDOMElement_get_nodeName( element, &str );
2789         ok ( r == S_OK, "get_nodeName wrong code\n");
2790         ok ( str != NULL, "str is null\n");
2791         ok( !lstrcmpW( str, szlc ), "incorrect nodeName\n");
2792         SysFreeString( str );
2793
2794         str = SysAllocString( szNonExistentFile );      
2795         V_VT(&var) = VT_I4;
2796         V_I4(&var) = 0x1234;
2797         r = IXMLDOMElement_getAttribute( element, str, &var );
2798         ok( r == E_FAIL, "getAttribute ret %08x\n", r );
2799         ok( V_VT(&var) == VT_NULL || V_VT(&var) == VT_EMPTY, "vt = %x\n", V_VT(&var));
2800         VariantClear(&var);
2801
2802         r = IXMLDOMElement_getAttributeNode( element, str, NULL);
2803         ok( r == E_FAIL, "getAttributeNode ret %08x\n", r );
2804
2805         attr = (IXMLDOMAttribute*)0xdeadbeef;
2806         r = IXMLDOMElement_getAttributeNode( element, str, &attr);
2807         ok( r == E_FAIL, "getAttributeNode ret %08x\n", r );
2808         ok( attr == NULL, "getAttributeNode ret %p, expected NULL\n", attr );
2809         SysFreeString( str );
2810
2811         attr = (IXMLDOMAttribute*)0xdeadbeef;
2812         str = SysAllocString( szNonExistentAttribute );
2813         r = IXMLDOMElement_getAttributeNode( element, str, &attr);
2814         ok( r == S_FALSE, "getAttributeNode ret %08x\n", r );
2815         ok( attr == NULL, "getAttributeNode ret %p, expected NULL\n", attr );
2816         SysFreeString( str );
2817
2818         str = SysAllocString( szdl );   
2819         V_VT(&var) = VT_I4;
2820         V_I4(&var) = 0x1234;
2821         r = IXMLDOMElement_getAttribute( element, str, &var );
2822         ok( r == S_OK, "getAttribute ret %08x\n", r );
2823         ok( V_VT(&var) == VT_BSTR, "vt = %x\n", V_VT(&var));
2824         ok( !lstrcmpW(V_BSTR(&var), szstr1), "wrong attr value\n");
2825         VariantClear( &var );
2826
2827         r = IXMLDOMElement_getAttribute( element, NULL, &var );
2828         ok( r == E_INVALIDARG, "getAttribute ret %08x\n", r );
2829
2830         r = IXMLDOMElement_getAttribute( element, str, NULL );
2831         ok( r == E_INVALIDARG, "getAttribute ret %08x\n", r );
2832
2833         attr = NULL;
2834         r = IXMLDOMElement_getAttributeNode( element, str, &attr);
2835         ok( r == S_OK, "GetAttributeNode ret %08x\n", r );
2836         ok( attr != NULL, "getAttributeNode returned NULL\n" );
2837         if (attr)
2838         {
2839             r = IXMLDOMAttribute_get_parentNode( attr, NULL );
2840             ok( r == E_INVALIDARG, "Expected E_INVALIDARG, ret %08x\n", r );
2841
2842             /* attribute doesn't have a parent in msxml interpretation */
2843             node = (IXMLDOMNode*)0xdeadbeef;
2844             r = IXMLDOMAttribute_get_parentNode( attr, &node );
2845             ok( r == S_FALSE, "Expected S_FALSE, ret %08x\n", r );
2846             ok( node == NULL, "Expected NULL, got %p\n", node );
2847
2848             IXMLDOMAttribute_Release(attr);
2849         }
2850
2851         SysFreeString( str );
2852
2853         r = IXMLDOMElement_get_attributes( element, &map );
2854         ok( r == S_OK, "get_attributes returned wrong code\n");
2855         ok( map != NULL, "should be attributes\n");
2856
2857         EXPECT_CHILDREN(element);
2858     }
2859     else
2860         ok( FALSE, "no element\n");
2861
2862     if (map)
2863     {
2864         ISupportErrorInfo *support_error;
2865         r = IXMLDOMNamedNodeMap_QueryInterface( map, &IID_ISupportErrorInfo, (void**)&support_error );
2866         ok( r == S_OK, "ret %08x\n", r );
2867
2868         r = ISupportErrorInfo_InterfaceSupportsErrorInfo( support_error, &IID_IXMLDOMNamedNodeMap );
2869 todo_wine
2870 {
2871         ok( r == S_OK, "ret %08x\n", r );
2872 }
2873         ISupportErrorInfo_Release( support_error );
2874
2875         str = SysAllocString( szdl );
2876         r = IXMLDOMNamedNodeMap_getNamedItem( map, str, &node );
2877         ok( r == S_OK, "getNamedItem returned wrong code\n");
2878         ok( node != NULL, "should be attributes\n");
2879         IXMLDOMNode_Release(node);
2880         SysFreeString( str );
2881
2882         str = SysAllocString( szdl );
2883         r = IXMLDOMNamedNodeMap_getNamedItem( map, str, NULL );
2884         ok( r == E_INVALIDARG, "getNamedItem should return E_INVALIDARG\n");
2885         SysFreeString( str );
2886
2887         /* something that isn't in szComplete4 */
2888         str = SysAllocString( szOpen );
2889         node = (IXMLDOMNode *) 1;
2890         r = IXMLDOMNamedNodeMap_getNamedItem( map, str, &node );
2891         ok( r == S_FALSE, "getNamedItem found a node that wasn't there\n");
2892         ok( node == NULL, "getNamedItem should have returned NULL\n");
2893         SysFreeString( str );
2894
2895         /* test indexed access of attributes */
2896         r = IXMLDOMNamedNodeMap_get_length( map, NULL );
2897         ok ( r == E_INVALIDARG, "get_length should return E_INVALIDARG\n");
2898
2899         r = IXMLDOMNamedNodeMap_get_length( map, &count );
2900         ok ( r == S_OK, "get_length wrong code\n");
2901         ok ( count == 1, "get_length != 1\n");
2902
2903         node = NULL;
2904         r = IXMLDOMNamedNodeMap_get_item( map, -1, &node);
2905         ok ( r == S_FALSE, "get_item (-1) wrong code\n");
2906         ok ( node == NULL, "there is no node\n");
2907
2908         node = NULL;
2909         r = IXMLDOMNamedNodeMap_get_item( map, 1, &node);
2910         ok ( r == S_FALSE, "get_item (1) wrong code\n");
2911         ok ( node == NULL, "there is no attribute\n");
2912
2913         node = NULL;
2914         r = IXMLDOMNamedNodeMap_get_item( map, 0, &node);
2915         ok ( r == S_OK, "get_item (0) wrong code\n");
2916         ok ( node != NULL, "should be attribute\n");
2917
2918         r = IXMLDOMNode_get_nodeName( node, NULL );
2919         ok ( r == E_INVALIDARG, "get_nodeName (NULL) wrong code\n");
2920
2921         /* content doesn't matter here */
2922         str = NULL;
2923         r = IXMLDOMNode_get_nodeName( node, &str );
2924         ok ( r == S_OK, "get_nodeName wrong code\n");
2925         ok ( str != NULL, "str is null\n");
2926         ok( !lstrcmpW( str, szdl ), "incorrect node name\n");
2927         SysFreeString( str );
2928         IXMLDOMNode_Release( node );
2929
2930         /* test sequential access of attributes */
2931         node = NULL;
2932         r = IXMLDOMNamedNodeMap_nextNode( map, &node );
2933         ok ( r == S_OK, "nextNode (first time) wrong code\n");
2934         ok ( node != NULL, "nextNode, should be attribute\n");
2935         IXMLDOMNode_Release( node );
2936
2937         r = IXMLDOMNamedNodeMap_nextNode( map, &node );
2938         ok ( r != S_OK, "nextNode (second time) wrong code\n");
2939         ok ( node == NULL, "nextNode, there is no attribute\n");
2940
2941         r = IXMLDOMNamedNodeMap_reset( map );
2942         ok ( r == S_OK, "reset should return S_OK\n");
2943
2944         r = IXMLDOMNamedNodeMap_nextNode( map, &node );
2945         ok ( r == S_OK, "nextNode (third time) wrong code\n");
2946         ok ( node != NULL, "nextNode, should be attribute\n");
2947     }
2948     else
2949         ok( FALSE, "no map\n");
2950
2951     if (node)
2952     {
2953         type = NODE_INVALID;
2954         r = IXMLDOMNode_get_nodeType( node, &type);
2955         ok( r == S_OK, "getNamedItem returned wrong code\n");
2956         ok( type == NODE_ATTRIBUTE, "node not an attribute\n");
2957
2958         str = NULL;
2959         r = IXMLDOMNode_get_baseName( node, NULL );
2960         ok( r == E_INVALIDARG, "get_baseName returned wrong code\n");
2961
2962         str = NULL;
2963         r = IXMLDOMNode_get_baseName( node, &str );
2964         ok( r == S_OK, "get_baseName returned wrong code\n");
2965         ok( lstrcmpW(str,szdl) == 0, "basename was wrong\n");
2966         SysFreeString( str );
2967
2968         r = IXMLDOMNode_get_nodeValue( node, &var );
2969         ok( r == S_OK, "returns %08x\n", r );
2970         ok( V_VT(&var) == VT_BSTR, "vt %x\n", V_VT(&var));
2971         ok( !lstrcmpW(V_BSTR(&var), szstr1), "nodeValue incorrect\n");
2972         VariantClear(&var);
2973
2974         r = IXMLDOMNode_get_childNodes( node, NULL );
2975         ok( r == E_INVALIDARG, "get_childNodes returned wrong code\n");
2976
2977         r = IXMLDOMNode_get_childNodes( node, &list );
2978         ok( r == S_OK, "get_childNodes returned wrong code\n");
2979
2980         if (list)
2981         {
2982             r = IXMLDOMNodeList_nextNode( list, &next );
2983             ok( r == S_OK, "nextNode returned wrong code\n");
2984         }
2985         else
2986             ok( FALSE, "no childlist\n");
2987
2988         if (next)
2989         {
2990             EXPECT_NO_CHILDREN(next);
2991
2992             type = NODE_INVALID;
2993             r = IXMLDOMNode_get_nodeType( next, &type);
2994             ok( r == S_OK, "getNamedItem returned wrong code\n");
2995             ok( type == NODE_TEXT, "node not text\n");
2996
2997             str = (BSTR) 1;
2998             r = IXMLDOMNode_get_baseName( next, &str );
2999             ok( r == S_FALSE, "get_baseName returned wrong code\n");
3000             ok( str == NULL, "basename was wrong\n");
3001             SysFreeString(str);
3002         }
3003         else
3004             ok( FALSE, "no next\n");
3005
3006         if (next)
3007             IXMLDOMNode_Release( next );
3008         next = NULL;
3009         if (list)
3010             IXMLDOMNodeList_Release( list );
3011         list = NULL;
3012         if (node)
3013             IXMLDOMNode_Release( node );
3014     }
3015     else
3016         ok( FALSE, "no node\n");
3017     node = NULL;
3018
3019     if (map)
3020         IXMLDOMNamedNodeMap_Release( map );
3021
3022     /* now traverse the tree from the root element */
3023     if (element)
3024     {
3025         r = IXMLDOMNode_get_childNodes( element, &list );
3026         ok( r == S_OK, "get_childNodes returned wrong code\n");
3027
3028         /* using get_item for child list doesn't advance the position */
3029         ole_check(IXMLDOMNodeList_get_item(list, 1, &node));
3030         expect_node(node, "E2.E2.D1");
3031         IXMLDOMNode_Release(node);
3032         ole_check(IXMLDOMNodeList_nextNode(list, &node));
3033         expect_node(node, "E1.E2.D1");
3034         IXMLDOMNode_Release(node);
3035         ole_check(IXMLDOMNodeList_reset(list));
3036
3037         IXMLDOMNodeList_AddRef(list);
3038         expect_list_and_release(list, "E1.E2.D1 E2.E2.D1 E3.E2.D1 E4.E2.D1");
3039         ole_check(IXMLDOMNodeList_reset(list));
3040
3041         node = (void*)0xdeadbeef;
3042         str = SysAllocString(szdl);
3043         r = IXMLDOMNode_selectSingleNode( element, str, &node );
3044         SysFreeString(str);
3045         ok( r == S_FALSE, "ret %08x\n", r );
3046         ok( node == NULL, "node %p\n", node );
3047
3048         str = SysAllocString(szbs);
3049         r = IXMLDOMNode_selectSingleNode( element, str, &node );
3050         SysFreeString(str);
3051         ok( r == S_OK, "ret %08x\n", r );
3052         r = IXMLDOMNode_Release( node );
3053         ok( r == 0, "ret %08x\n", r );
3054     }
3055     else
3056         ok( FALSE, "no element\n");
3057
3058     if (list)
3059     {
3060         r = IXMLDOMNodeList_get_item(list, 0, NULL);
3061         ok(r == E_INVALIDARG, "Expected E_INVALIDARG got %08x\n", r);
3062
3063         r = IXMLDOMNodeList_get_length(list, NULL);
3064         ok(r == E_INVALIDARG, "Expected E_INVALIDARG got %08x\n", r);
3065
3066         r = IXMLDOMNodeList_get_length( list, &count );
3067         ok( r == S_OK, "get_length returns %08x\n", r );
3068         ok( count == 4, "get_length got %d\n", count );
3069
3070         r = IXMLDOMNodeList_nextNode(list, NULL);
3071         ok(r == E_INVALIDARG, "Expected E_INVALIDARG got %08x\n", r);
3072
3073         r = IXMLDOMNodeList_nextNode( list, &node );
3074         ok( r == S_OK, "nextNode returned wrong code\n");
3075     }
3076     else
3077         ok( FALSE, "no list\n");
3078
3079     if (node)
3080     {
3081         type = NODE_INVALID;
3082         r = IXMLDOMNode_get_nodeType( node, &type);
3083         ok( r == S_OK, "getNamedItem returned wrong code\n");
3084         ok( type == NODE_ELEMENT, "node not text\n");
3085
3086         VariantInit(&var);
3087         ok( V_VT(&var) == VT_EMPTY, "variant init failed\n");
3088         r = IXMLDOMNode_get_nodeValue( node, &var );
3089         ok( r == S_FALSE, "nextNode returned wrong code\n");
3090         ok( V_VT(&var) == VT_NULL, "variant wasn't empty\n");
3091         ok( V_BSTR(&var) == NULL, "variant value wasn't null\n");
3092
3093         r = IXMLDOMNode_hasChildNodes( node, NULL );
3094         ok( r == E_INVALIDARG, "hasChildNodes bad return\n");
3095
3096         EXPECT_CHILDREN(node);
3097
3098         str = NULL;
3099         r = IXMLDOMNode_get_baseName( node, &str );
3100         ok( r == S_OK, "get_baseName returned wrong code\n");
3101         ok( lstrcmpW(str,szbs) == 0, "basename was wrong\n");
3102         SysFreeString(str);
3103     }
3104     else
3105         ok( FALSE, "no node\n");
3106
3107     if (node)
3108         IXMLDOMNode_Release( node );
3109     if (list)
3110         IXMLDOMNodeList_Release( list );
3111     if (element)
3112         IXMLDOMElement_Release( element );
3113
3114     b = FALSE;
3115     str = SysAllocString( szComplete5 );
3116     r = IXMLDOMDocument_loadXML( doc, str, &b );
3117     ok( r == S_OK, "loadXML failed\n");
3118     ok( b == VARIANT_TRUE, "failed to load XML string\n");
3119     SysFreeString( str );
3120
3121     EXPECT_CHILDREN(doc);
3122
3123     r = IXMLDOMDocument_get_documentElement( doc, &element );
3124     ok( r == S_OK, "should be a document element\n");
3125     ok( element != NULL, "should be an element\n");
3126
3127     if (element)
3128     {
3129         static const WCHAR szSSearch[] = {'S',':','s','e','a','r','c','h',0};
3130         BSTR tag = NULL;
3131
3132         /* check if the tag is correct */
3133         r = IXMLDOMElement_get_tagName( element, &tag );
3134         ok( r == S_OK, "couldn't get tag name\n");
3135         ok( tag != NULL, "tag was null\n");
3136         ok( !lstrcmpW( tag, szSSearch ), "incorrect tag name\n");
3137         SysFreeString( tag );
3138     }
3139
3140     if (element)
3141         IXMLDOMElement_Release( element );
3142     ok(IXMLDOMDocument_Release( doc ) == 0, "document is not destroyed\n");
3143 }
3144
3145 static void test_refs(void)
3146 {
3147     HRESULT r;
3148     BSTR str;
3149     VARIANT_BOOL b;
3150     IXMLDOMDocument *doc;
3151     IXMLDOMElement *element = NULL;
3152     IXMLDOMNode *node = NULL, *node2;
3153     IXMLDOMNodeList *node_list = NULL;
3154     LONG ref;
3155     IUnknown *unk, *unk2;
3156
3157     doc = create_document(&IID_IXMLDOMDocument);
3158     if (!doc) return;
3159
3160     EXPECT_REF(doc, 1);
3161     ref = IXMLDOMDocument_Release(doc);
3162     ok( ref == 0, "ref %d\n", ref);
3163
3164     doc = create_document(&IID_IXMLDOMDocument);
3165     if (!doc) return;
3166
3167     str = SysAllocString( szComplete4 );
3168     r = IXMLDOMDocument_loadXML( doc, str, &b );
3169     ok( r == S_OK, "loadXML failed\n");
3170     ok( b == VARIANT_TRUE, "failed to load XML string\n");
3171     SysFreeString( str );
3172
3173     EXPECT_REF(doc, 1);
3174     IXMLDOMDocument_AddRef( doc );
3175     EXPECT_REF(doc, 2);
3176     IXMLDOMDocument_AddRef( doc );
3177     EXPECT_REF(doc, 3);
3178
3179     IXMLDOMDocument_Release( doc );
3180     IXMLDOMDocument_Release( doc );
3181
3182     r = IXMLDOMDocument_get_documentElement( doc, &element );
3183     ok( r == S_OK, "should be a document element\n");
3184     ok( element != NULL, "should be an element\n");
3185
3186     EXPECT_REF(doc, 1);
3187     todo_wine EXPECT_REF(element, 2);
3188
3189     IXMLDOMElement_AddRef(element);
3190     todo_wine EXPECT_REF(element, 3);
3191     IXMLDOMElement_Release(element);
3192
3193     r = IXMLDOMElement_get_childNodes( element, &node_list );
3194     ok( r == S_OK, "rets %08x\n", r);
3195
3196     todo_wine EXPECT_REF(element, 2);
3197     EXPECT_REF(node_list, 1);
3198
3199     IXMLDOMNodeList_get_item( node_list, 0, &node );
3200     ok( r == S_OK, "rets %08x\n", r);
3201     EXPECT_REF(node_list, 1);
3202     EXPECT_REF(node, 1);
3203
3204     IXMLDOMNodeList_get_item( node_list, 0, &node2 );
3205     ok( r == S_OK, "rets %08x\n", r);
3206     EXPECT_REF(node_list, 1);
3207     EXPECT_REF(node2, 1);
3208
3209     ref = IXMLDOMNode_Release( node );
3210     ok( ref == 0, "ref %d\n", ref );
3211     ref = IXMLDOMNode_Release( node2 );
3212     ok( ref == 0, "ref %d\n", ref );
3213
3214     ref = IXMLDOMNodeList_Release( node_list );
3215     ok( ref == 0, "ref %d\n", ref );
3216
3217     ok( node != node2, "node %p node2 %p\n", node, node2 );
3218
3219     ref = IXMLDOMDocument_Release( doc );
3220     ok( ref == 0, "ref %d\n", ref );
3221
3222     todo_wine EXPECT_REF(element, 2);
3223
3224     /* IUnknown must be unique however we obtain it */
3225     r = IXMLDOMElement_QueryInterface( element, &IID_IUnknown, (void**)&unk );
3226     ok( r == S_OK, "rets %08x\n", r );
3227     EXPECT_REF(element, 2);
3228     r = IXMLDOMElement_QueryInterface( element, &IID_IXMLDOMNode, (void**)&node );
3229     ok( r == S_OK, "rets %08x\n", r );
3230     todo_wine EXPECT_REF(element, 2);
3231     r = IXMLDOMNode_QueryInterface( node, &IID_IUnknown, (void**)&unk2 );
3232     ok( r == S_OK, "rets %08x\n", r );
3233     todo_wine EXPECT_REF(element, 2);
3234     ok( unk == unk2, "unk %p unk2 %p\n", unk, unk2 );
3235     todo_wine ok( element != (void*)node, "node %p element %p\n", node, element );
3236
3237     IUnknown_Release( unk2 );
3238     IUnknown_Release( unk );
3239     IXMLDOMNode_Release( node );
3240     todo_wine EXPECT_REF(element, 2);
3241
3242     IXMLDOMElement_Release( element );
3243 }
3244
3245 static void test_create(void)
3246 {
3247     static const WCHAR szOne[] = {'1',0};
3248     static const WCHAR szOneGarbage[] = {'1','G','a','r','b','a','g','e',0};
3249     HRESULT r;
3250     VARIANT var;
3251     BSTR str, name;
3252     IXMLDOMDocument *doc;
3253     IXMLDOMElement *element;
3254     IXMLDOMComment *comment;
3255     IXMLDOMText *text;
3256     IXMLDOMCDATASection *cdata;
3257     IXMLDOMNode *root, *node, *child;
3258     IXMLDOMNamedNodeMap *attr_map;
3259     IUnknown *unk;
3260     LONG ref;
3261     LONG num;
3262
3263     doc = create_document(&IID_IXMLDOMDocument);
3264     if (!doc) return;
3265
3266     EXPECT_REF(doc, 1);
3267
3268     /* types not supported for creation */
3269     V_VT(&var) = VT_I1;
3270     V_I1(&var) = NODE_DOCUMENT;
3271     node = (IXMLDOMNode*)0x1;
3272     r = IXMLDOMDocument_createNode( doc, var, NULL, NULL, &node );
3273     ok( r == E_INVALIDARG, "returns %08x\n", r );
3274     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
3275
3276     V_VT(&var) = VT_I1;
3277     V_I1(&var) = NODE_DOCUMENT_TYPE;
3278     node = (IXMLDOMNode*)0x1;
3279     r = IXMLDOMDocument_createNode( doc, var, NULL, NULL, &node );
3280     ok( r == E_INVALIDARG, "returns %08x\n", r );
3281     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
3282
3283     V_VT(&var) = VT_I1;
3284     V_I1(&var) = NODE_ENTITY;
3285     node = (IXMLDOMNode*)0x1;
3286     r = IXMLDOMDocument_createNode( doc, var, NULL, NULL, &node );
3287     ok( r == E_INVALIDARG, "returns %08x\n", r );
3288     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
3289
3290     V_VT(&var) = VT_I1;
3291     V_I1(&var) = NODE_NOTATION;
3292     node = (IXMLDOMNode*)0x1;
3293     r = IXMLDOMDocument_createNode( doc, var, NULL, NULL, &node );
3294     ok( r == E_INVALIDARG, "returns %08x\n", r );
3295     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
3296
3297     /* NODE_COMMENT */
3298     V_VT(&var) = VT_I1;
3299     V_I1(&var) = NODE_COMMENT;
3300     node = NULL;
3301     r = IXMLDOMDocument_createNode( doc, var, NULL, NULL, &node );
3302     ok( r == S_OK, "returns %08x\n", r );
3303     ok( node != NULL, "\n");
3304
3305     r = IXMLDOMNode_QueryInterface(node, &IID_IXMLDOMComment, (void**)&comment);
3306     ok( r == S_OK, "returns %08x\n", r );
3307     IXMLDOMNode_Release(node);
3308
3309     str = NULL;
3310     r = IXMLDOMComment_get_data(comment, &str);
3311     ok( r == S_OK, "returns %08x\n", r );
3312     ok( str && SysStringLen(str) == 0, "expected empty comment, %p\n", str);
3313     IXMLDOMComment_Release(comment);
3314     SysFreeString(str);
3315
3316     node = (IXMLDOMNode*)0x1;
3317     r = IXMLDOMDocument_createNode( doc, var, _bstr_(""), NULL, &node );
3318     ok( r == S_OK, "returns %08x\n", r );
3319
3320     r = IXMLDOMNode_QueryInterface(node, &IID_IXMLDOMComment, (void**)&comment);
3321     ok( r == S_OK, "returns %08x\n", r );
3322     IXMLDOMNode_Release(node);
3323
3324     str = NULL;
3325     r = IXMLDOMComment_get_data(comment, &str);
3326     ok( r == S_OK, "returns %08x\n", r );
3327     ok( str && SysStringLen(str) == 0, "expected empty comment, %p\n", str);
3328     IXMLDOMComment_Release(comment);
3329     SysFreeString(str);
3330
3331     node = (IXMLDOMNode*)0x1;
3332     r = IXMLDOMDocument_createNode( doc, var, _bstr_("blah"), NULL, &node );
3333     ok( r == S_OK, "returns %08x\n", r );
3334
3335     r = IXMLDOMNode_QueryInterface(node, &IID_IXMLDOMComment, (void**)&comment);
3336     ok( r == S_OK, "returns %08x\n", r );
3337     IXMLDOMNode_Release(node);
3338
3339     str = NULL;
3340     r = IXMLDOMComment_get_data(comment, &str);
3341     ok( r == S_OK, "returns %08x\n", r );
3342     ok( str && SysStringLen(str) == 0, "expected empty comment, %p\n", str);
3343     IXMLDOMComment_Release(comment);
3344     SysFreeString(str);
3345
3346     /* NODE_TEXT */
3347     V_VT(&var) = VT_I1;
3348     V_I1(&var) = NODE_TEXT;
3349     node = NULL;
3350     r = IXMLDOMDocument_createNode( doc, var, NULL, NULL, &node );
3351     ok( r == S_OK, "returns %08x\n", r );
3352     ok( node != NULL, "\n");
3353
3354     r = IXMLDOMNode_QueryInterface(node, &IID_IXMLDOMText, (void**)&text);
3355     ok( r == S_OK, "returns %08x\n", r );
3356     IXMLDOMNode_Release(node);
3357
3358     str = NULL;
3359     r = IXMLDOMText_get_data(text, &str);
3360     ok( r == S_OK, "returns %08x\n", r );
3361     ok( str && SysStringLen(str) == 0, "expected empty comment, %p\n", str);
3362     IXMLDOMText_Release(text);
3363     SysFreeString(str);
3364
3365     node = (IXMLDOMNode*)0x1;
3366     r = IXMLDOMDocument_createNode( doc, var, _bstr_(""), NULL, &node );
3367     ok( r == S_OK, "returns %08x\n", r );
3368
3369     r = IXMLDOMNode_QueryInterface(node, &IID_IXMLDOMText, (void**)&text);
3370     ok( r == S_OK, "returns %08x\n", r );
3371     IXMLDOMNode_Release(node);
3372
3373     str = NULL;
3374     r = IXMLDOMText_get_data(text, &str);
3375     ok( r == S_OK, "returns %08x\n", r );
3376     ok( str && SysStringLen(str) == 0, "expected empty comment, %p\n", str);
3377     IXMLDOMText_Release(text);
3378     SysFreeString(str);
3379
3380     node = (IXMLDOMNode*)0x1;
3381     r = IXMLDOMDocument_createNode( doc, var, _bstr_("blah"), NULL, &node );
3382     ok( r == S_OK, "returns %08x\n", r );
3383
3384     r = IXMLDOMNode_QueryInterface(node, &IID_IXMLDOMText, (void**)&text);
3385     ok( r == S_OK, "returns %08x\n", r );
3386     IXMLDOMNode_Release(node);
3387
3388     str = NULL;
3389     r = IXMLDOMText_get_data(text, &str);
3390     ok( r == S_OK, "returns %08x\n", r );
3391     ok( str && SysStringLen(str) == 0, "expected empty comment, %p\n", str);
3392     IXMLDOMText_Release(text);
3393     SysFreeString(str);
3394
3395     /* NODE_CDATA_SECTION */
3396     V_VT(&var) = VT_I1;
3397     V_I1(&var) = NODE_CDATA_SECTION;
3398     node = NULL;
3399     r = IXMLDOMDocument_createNode( doc, var, NULL, NULL, &node );
3400     ok( r == S_OK, "returns %08x\n", r );
3401     ok( node != NULL, "\n");
3402
3403     r = IXMLDOMNode_QueryInterface(node, &IID_IXMLDOMCDATASection, (void**)&cdata);
3404     ok( r == S_OK, "returns %08x\n", r );
3405     IXMLDOMNode_Release(node);
3406
3407     str = NULL;
3408     r = IXMLDOMCDATASection_get_data(cdata, &str);
3409     ok( r == S_OK, "returns %08x\n", r );
3410     ok( str && SysStringLen(str) == 0, "expected empty comment, %p\n", str);
3411     IXMLDOMCDATASection_Release(cdata);
3412     SysFreeString(str);
3413
3414     node = (IXMLDOMNode*)0x1;
3415     r = IXMLDOMDocument_createNode( doc, var, _bstr_(""), NULL, &node );
3416     ok( r == S_OK, "returns %08x\n", r );
3417
3418     r = IXMLDOMNode_QueryInterface(node, &IID_IXMLDOMCDATASection, (void**)&cdata);
3419     ok( r == S_OK, "returns %08x\n", r );
3420     IXMLDOMNode_Release(node);
3421
3422     str = NULL;
3423     r = IXMLDOMCDATASection_get_data(cdata, &str);
3424     ok( r == S_OK, "returns %08x\n", r );
3425     ok( str && SysStringLen(str) == 0, "expected empty comment, %p\n", str);
3426     IXMLDOMCDATASection_Release(cdata);
3427     SysFreeString(str);
3428
3429     node = (IXMLDOMNode*)0x1;
3430     r = IXMLDOMDocument_createNode( doc, var, _bstr_("blah"), NULL, &node );
3431     ok( r == S_OK, "returns %08x\n", r );
3432
3433     r = IXMLDOMNode_QueryInterface(node, &IID_IXMLDOMCDATASection, (void**)&cdata);
3434     ok( r == S_OK, "returns %08x\n", r );
3435     IXMLDOMNode_Release(node);
3436
3437     str = NULL;
3438     r = IXMLDOMCDATASection_get_data(cdata, &str);
3439     ok( r == S_OK, "returns %08x\n", r );
3440     ok( str && SysStringLen(str) == 0, "expected empty comment, %p\n", str);
3441     IXMLDOMCDATASection_Release(cdata);
3442     SysFreeString(str);
3443
3444     /* NODE_ATTRIBUTE */
3445     V_VT(&var) = VT_I1;
3446     V_I1(&var) = NODE_ATTRIBUTE;
3447     node = (IXMLDOMNode*)0x1;
3448     r = IXMLDOMDocument_createNode( doc, var, NULL, NULL, &node );
3449     ok( r == E_FAIL, "returns %08x\n", r );
3450     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
3451
3452     V_VT(&var) = VT_I1;
3453     V_I1(&var) = NODE_ATTRIBUTE;
3454     node = (IXMLDOMNode*)0x1;
3455     r = IXMLDOMDocument_createNode( doc, var, _bstr_(""), NULL, &node );
3456     ok( r == E_FAIL, "returns %08x\n", r );
3457     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
3458
3459     V_VT(&var) = VT_I1;
3460     V_I1(&var) = NODE_ATTRIBUTE;
3461     str = SysAllocString( szlc );
3462     r = IXMLDOMDocument_createNode( doc, var, str, NULL, &node );
3463     ok( r == S_OK, "returns %08x\n", r );
3464     if( SUCCEEDED(r) ) IXMLDOMNode_Release( node );
3465     SysFreeString(str);
3466
3467     /* a name is required for attribute, try a BSTR with first null wchar */
3468     V_VT(&var) = VT_I1;
3469     V_I1(&var) = NODE_ATTRIBUTE;
3470     str = SysAllocString( szstr1 );
3471     str[0] = 0;
3472     node = (IXMLDOMNode*)0x1;
3473     r = IXMLDOMDocument_createNode( doc, var, str, NULL, &node );
3474     ok( r == E_FAIL, "returns %08x\n", r );
3475     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
3476     SysFreeString(str);
3477
3478     /* NODE_PROCESSING_INSTRUCTION */
3479     V_VT(&var) = VT_I1;
3480     V_I1(&var) = NODE_PROCESSING_INSTRUCTION;
3481     node = (IXMLDOMNode*)0x1;
3482     r = IXMLDOMDocument_createNode( doc, var, NULL, NULL, &node );
3483     ok( r == E_FAIL, "returns %08x\n", r );
3484     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
3485
3486     V_VT(&var) = VT_I1;
3487     V_I1(&var) = NODE_PROCESSING_INSTRUCTION;
3488     node = (IXMLDOMNode*)0x1;
3489     r = IXMLDOMDocument_createNode( doc, var, _bstr_(""), NULL, &node );
3490     ok( r == E_FAIL, "returns %08x\n", r );
3491     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
3492
3493     V_VT(&var) = VT_I1;
3494     V_I1(&var) = NODE_PROCESSING_INSTRUCTION;
3495     r = IXMLDOMDocument_createNode( doc, var, _bstr_("pi"), NULL, NULL );
3496     ok( r == E_INVALIDARG, "returns %08x\n", r );
3497
3498     /* NODE_ENTITY_REFERENCE */
3499     V_VT(&var) = VT_I1;
3500     V_I1(&var) = NODE_ENTITY_REFERENCE;
3501     node = (IXMLDOMNode*)0x1;
3502     r = IXMLDOMDocument_createNode( doc, var, NULL, NULL, &node );
3503     ok( r == E_FAIL, "returns %08x\n", r );
3504     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
3505
3506     V_VT(&var) = VT_I1;
3507     V_I1(&var) = NODE_ENTITY_REFERENCE;
3508     node = (IXMLDOMNode*)0x1;
3509     r = IXMLDOMDocument_createNode( doc, var, _bstr_(""), NULL, &node );
3510     ok( r == E_FAIL, "returns %08x\n", r );
3511     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
3512
3513     /* NODE_ELEMENT */
3514     V_VT(&var) = VT_I1;
3515     V_I1(&var) = NODE_ELEMENT;
3516     node = (IXMLDOMNode*)0x1;
3517     r = IXMLDOMDocument_createNode( doc, var, NULL, NULL, &node );
3518     ok( r == E_FAIL, "returns %08x\n", r );
3519     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
3520
3521     V_VT(&var) = VT_I1;
3522     V_I1(&var) = NODE_ELEMENT;
3523     node = (IXMLDOMNode*)0x1;
3524     r = IXMLDOMDocument_createNode( doc, var, _bstr_(""), NULL, &node );
3525     ok( r == E_FAIL, "returns %08x\n", r );
3526     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
3527
3528     V_VT(&var) = VT_I1;
3529     V_I1(&var) = NODE_ELEMENT;
3530     str = SysAllocString( szlc );
3531     r = IXMLDOMDocument_createNode( doc, var, str, NULL, &node );
3532     ok( r == S_OK, "returns %08x\n", r );
3533     if( SUCCEEDED(r) ) IXMLDOMNode_Release( node );
3534
3535     V_VT(&var) = VT_I1;
3536     V_I1(&var) = NODE_ELEMENT;
3537     r = IXMLDOMDocument_createNode( doc, var, str, NULL, NULL );
3538     ok( r == E_INVALIDARG, "returns %08x\n", r );
3539
3540     V_VT(&var) = VT_R4;
3541     V_R4(&var) = NODE_ELEMENT;
3542     r = IXMLDOMDocument_createNode( doc, var, str, NULL, &node );
3543     ok( r == S_OK, "returns %08x\n", r );
3544     if( SUCCEEDED(r) ) IXMLDOMNode_Release( node );
3545
3546     V_VT(&var) = VT_BSTR;
3547     V_BSTR(&var) = SysAllocString( szOne );
3548     r = IXMLDOMDocument_createNode( doc, var, str, NULL, &node );
3549     ok( r == S_OK, "returns %08x\n", r );
3550     if( SUCCEEDED(r) ) IXMLDOMNode_Release( node );
3551     VariantClear(&var);
3552
3553     V_VT(&var) = VT_BSTR;
3554     V_BSTR(&var) = SysAllocString( szOneGarbage );
3555     r = IXMLDOMDocument_createNode( doc, var, str, NULL, &node );
3556     ok( r == E_INVALIDARG, "returns %08x\n", r );
3557     if( SUCCEEDED(r) ) IXMLDOMNode_Release( node );
3558     VariantClear(&var);
3559
3560     V_VT(&var) = VT_I4;
3561     V_I4(&var) = NODE_ELEMENT;
3562     r = IXMLDOMDocument_createNode( doc, var, str, NULL, &node );
3563     ok( r == S_OK, "returns %08x\n", r );
3564
3565     EXPECT_REF(doc, 1);
3566     r = IXMLDOMDocument_appendChild( doc, node, &root );
3567     ok( r == S_OK, "returns %08x\n", r );
3568     ok( node == root, "%p %p\n", node, root );
3569     EXPECT_REF(doc, 1);
3570
3571     EXPECT_REF(node, 2);
3572
3573     ref = IXMLDOMNode_Release( node );
3574     ok(ref == 1, "ref %d\n", ref);
3575     SysFreeString( str );
3576
3577     V_VT(&var) = VT_I4;
3578     V_I4(&var) = NODE_ELEMENT;
3579     str = SysAllocString( szbs );
3580     r = IXMLDOMDocument_createNode( doc, var, str, NULL, &node );
3581     ok( r == S_OK, "returns %08x\n", r );
3582     SysFreeString( str );
3583
3584     EXPECT_REF(node, 1);
3585
3586     r = IXMLDOMNode_QueryInterface( node, &IID_IUnknown, (void**)&unk );
3587     ok( r == S_OK, "returns %08x\n", r );
3588
3589     EXPECT_REF(unk, 2);
3590
3591     V_VT(&var) = VT_EMPTY;
3592     child = NULL;
3593     r = IXMLDOMNode_insertBefore( root, (IXMLDOMNode*)unk, var, &child );
3594     ok( r == S_OK, "returns %08x\n", r );
3595     ok( unk == (IUnknown*)child, "%p %p\n", unk, child );
3596
3597     todo_wine EXPECT_REF(unk, 4);
3598
3599     IXMLDOMNode_Release( child );
3600     IUnknown_Release( unk );
3601
3602     V_VT(&var) = VT_NULL;
3603     V_DISPATCH(&var) = (IDispatch*)node;
3604     r = IXMLDOMNode_insertBefore( root, node, var, &child );
3605     ok( r == S_OK, "returns %08x\n", r );
3606     ok( node == child, "%p %p\n", node, child );
3607     IXMLDOMNode_Release( child );
3608
3609     V_VT(&var) = VT_NULL;
3610     V_DISPATCH(&var) = (IDispatch*)node;
3611     r = IXMLDOMNode_insertBefore( root, node, var, NULL );
3612     ok( r == S_OK, "returns %08x\n", r );
3613     IXMLDOMNode_Release( node );
3614
3615     r = IXMLDOMNode_QueryInterface( root, &IID_IXMLDOMElement, (void**)&element );
3616     ok( r == S_OK, "returns %08x\n", r );
3617
3618     r = IXMLDOMElement_get_attributes( element, &attr_map );
3619     ok( r == S_OK, "returns %08x\n", r );
3620     r = IXMLDOMNamedNodeMap_get_length( attr_map, &num );
3621     ok( r == S_OK, "returns %08x\n", r );
3622     ok( num == 0, "num %d\n", num );
3623     IXMLDOMNamedNodeMap_Release( attr_map );
3624
3625     V_VT(&var) = VT_BSTR;
3626     V_BSTR(&var) = SysAllocString( szstr1 );
3627     name = SysAllocString( szdl );
3628     r = IXMLDOMElement_setAttribute( element, name, var );
3629     ok( r == S_OK, "returns %08x\n", r );
3630     r = IXMLDOMElement_get_attributes( element, &attr_map );
3631     ok( r == S_OK, "returns %08x\n", r );
3632     r = IXMLDOMNamedNodeMap_get_length( attr_map, &num );
3633     ok( r == S_OK, "returns %08x\n", r );
3634     ok( num == 1, "num %d\n", num );
3635     IXMLDOMNamedNodeMap_Release( attr_map );
3636     VariantClear(&var);
3637
3638     V_VT(&var) = VT_BSTR;
3639     V_BSTR(&var) = SysAllocString( szstr2 );
3640     r = IXMLDOMElement_setAttribute( element, name, var );
3641     ok( r == S_OK, "returns %08x\n", r );
3642     r = IXMLDOMElement_get_attributes( element, &attr_map );
3643     ok( r == S_OK, "returns %08x\n", r );
3644     r = IXMLDOMNamedNodeMap_get_length( attr_map, &num );
3645     ok( r == S_OK, "returns %08x\n", r );
3646     ok( num == 1, "num %d\n", num );
3647     IXMLDOMNamedNodeMap_Release( attr_map );
3648     VariantClear(&var);
3649     r = IXMLDOMElement_getAttribute( element, name, &var );
3650     ok( r == S_OK, "returns %08x\n", r );
3651     ok( !lstrcmpW(V_BSTR(&var), szstr2), "wrong attr value\n");
3652     VariantClear(&var);
3653     SysFreeString(name);
3654
3655     V_VT(&var) = VT_BSTR;
3656     V_BSTR(&var) = SysAllocString( szstr1 );
3657     name = SysAllocString( szlc );
3658     r = IXMLDOMElement_setAttribute( element, name, var );
3659     ok( r == S_OK, "returns %08x\n", r );
3660     r = IXMLDOMElement_get_attributes( element, &attr_map );
3661     ok( r == S_OK, "returns %08x\n", r );
3662     r = IXMLDOMNamedNodeMap_get_length( attr_map, &num );
3663     ok( r == S_OK, "returns %08x\n", r );
3664     ok( num == 2, "num %d\n", num );
3665     IXMLDOMNamedNodeMap_Release( attr_map );
3666     VariantClear(&var);
3667     SysFreeString(name);
3668
3669     V_VT(&var) = VT_I4;
3670     V_I4(&var) = 10;
3671     name = SysAllocString( szbs );
3672     r = IXMLDOMElement_setAttribute( element, name, var );
3673     ok( r == S_OK, "returns %08x\n", r );
3674     VariantClear(&var);
3675     r = IXMLDOMElement_getAttribute( element, name, &var );
3676     ok( r == S_OK, "returns %08x\n", r );
3677     ok( V_VT(&var) == VT_BSTR, "variant type %x\n", V_VT(&var));
3678     VariantClear(&var);
3679     SysFreeString(name);
3680
3681     /* Create an Attribute */
3682     V_VT(&var) = VT_I4;
3683     V_I4(&var) = NODE_ATTRIBUTE;
3684     str = SysAllocString( szAttribute );
3685     r = IXMLDOMDocument_createNode( doc, var, str, NULL, &node );
3686     ok( r == S_OK, "returns %08x\n", r );
3687     ok( node != NULL, "node was null\n");
3688     SysFreeString(str);
3689
3690     IXMLDOMElement_Release( element );
3691     IXMLDOMNode_Release( root );
3692     IXMLDOMDocument_Release( doc );
3693 }
3694
3695 static void test_getElementsByTagName(void)
3696 {
3697     IXMLDOMNodeList *node_list;
3698     IXMLDOMDocument *doc;
3699     IXMLDOMElement *elem;
3700     WCHAR buff[100];
3701     VARIANT_BOOL b;
3702     HRESULT r;
3703     LONG len;
3704     BSTR str;
3705
3706     doc = create_document(&IID_IXMLDOMDocument);
3707     if (!doc) return;
3708
3709     str = SysAllocString( szComplete4 );
3710     r = IXMLDOMDocument_loadXML( doc, str, &b );
3711     ok( r == S_OK, "loadXML failed\n");
3712     ok( b == VARIANT_TRUE, "failed to load XML string\n");
3713     SysFreeString( str );
3714
3715     str = SysAllocString( szstar );
3716
3717     /* null arguments cases */
3718     r = IXMLDOMDocument_getElementsByTagName(doc, NULL, &node_list);
3719     ok( r == E_INVALIDARG, "ret %08x\n", r );
3720     r = IXMLDOMDocument_getElementsByTagName(doc, str, NULL);
3721     ok( r == E_INVALIDARG, "ret %08x\n", r );
3722
3723     r = IXMLDOMDocument_getElementsByTagName(doc, str, &node_list);
3724     ok( r == S_OK, "ret %08x\n", r );
3725     r = IXMLDOMNodeList_get_length( node_list, &len );
3726     ok( r == S_OK, "ret %08x\n", r );
3727     ok( len == 6, "len %d\n", len );
3728
3729     test_disp((IUnknown*)node_list);
3730
3731     IXMLDOMNodeList_Release( node_list );
3732     SysFreeString( str );
3733
3734     /* broken query BSTR */
3735     memcpy(&buff[2], szstar, sizeof(szstar));
3736     /* just a big length */
3737     *(DWORD*)buff = 0xf0f0;
3738     r = IXMLDOMDocument_getElementsByTagName(doc, &buff[2], &node_list);
3739     ok( r == S_OK, "ret %08x\n", r );
3740     r = IXMLDOMNodeList_get_length( node_list, &len );
3741     ok( r == S_OK, "ret %08x\n", r );
3742     ok( len == 6, "len %d\n", len );
3743     IXMLDOMNodeList_Release( node_list );
3744
3745     str = SysAllocString( szbs );
3746     r = IXMLDOMDocument_getElementsByTagName(doc, str, &node_list);
3747     ok( r == S_OK, "ret %08x\n", r );
3748     r = IXMLDOMNodeList_get_length( node_list, &len );
3749     ok( r == S_OK, "ret %08x\n", r );
3750     ok( len == 1, "len %d\n", len );
3751     IXMLDOMNodeList_Release( node_list );
3752     SysFreeString( str );
3753
3754     str = SysAllocString( szdl );
3755     r = IXMLDOMDocument_getElementsByTagName(doc, str, &node_list);
3756     ok( r == S_OK, "ret %08x\n", r );
3757     r = IXMLDOMNodeList_get_length( node_list, &len );
3758     ok( r == S_OK, "ret %08x\n", r );
3759     ok( len == 0, "len %d\n", len );
3760     IXMLDOMNodeList_Release( node_list );
3761     SysFreeString( str );
3762
3763     str = SysAllocString( szstr1 );
3764     r = IXMLDOMDocument_getElementsByTagName(doc, str, &node_list);
3765     ok( r == S_OK, "ret %08x\n", r );
3766     r = IXMLDOMNodeList_get_length( node_list, &len );
3767     ok( r == S_OK, "ret %08x\n", r );
3768     ok( len == 0, "len %d\n", len );
3769     IXMLDOMNodeList_Release( node_list );
3770     SysFreeString( str );
3771
3772     /* test for element */
3773     r = IXMLDOMDocument_get_documentElement(doc, &elem);
3774     ok( r == S_OK, "ret %08x\n", r );
3775
3776     str = SysAllocString( szstar );
3777
3778     /* null arguments cases */
3779     r = IXMLDOMElement_getElementsByTagName(elem, NULL, &node_list);
3780     ok( r == E_INVALIDARG, "ret %08x\n", r );
3781     r = IXMLDOMElement_getElementsByTagName(elem, str, NULL);
3782     ok( r == E_INVALIDARG, "ret %08x\n", r );
3783
3784     r = IXMLDOMElement_getElementsByTagName(elem, str, &node_list);
3785     ok( r == S_OK, "ret %08x\n", r );
3786     r = IXMLDOMNodeList_get_length( node_list, &len );
3787     ok( r == S_OK, "ret %08x\n", r );
3788     ok( len == 5, "len %d\n", len );
3789     expect_list_and_release(node_list, "E1.E2.D1 E2.E2.D1 E3.E2.D1 E4.E2.D1 E1.E4.E2.D1");
3790     SysFreeString( str );
3791
3792     /* broken query BSTR */
3793     memcpy(&buff[2], szstar, sizeof(szstar));
3794     /* just a big length */
3795     *(DWORD*)buff = 0xf0f0;
3796     r = IXMLDOMElement_getElementsByTagName(elem, &buff[2], &node_list);
3797     ok( r == S_OK, "ret %08x\n", r );
3798     r = IXMLDOMNodeList_get_length( node_list, &len );
3799     ok( r == S_OK, "ret %08x\n", r );
3800     ok( len == 5, "len %d\n", len );
3801     IXMLDOMNodeList_Release( node_list );
3802
3803     IXMLDOMElement_Release(elem);
3804
3805     IXMLDOMDocument_Release( doc );
3806 }
3807
3808 static void test_get_text(void)
3809 {
3810     HRESULT r;
3811     BSTR str;
3812     VARIANT_BOOL b;
3813     IXMLDOMDocument *doc;
3814     IXMLDOMNode *node, *node2, *node3;
3815     IXMLDOMNode *nodeRoot;
3816     IXMLDOMNodeList *node_list;
3817     IXMLDOMNamedNodeMap *node_map;
3818     LONG len;
3819
3820     doc = create_document(&IID_IXMLDOMDocument);
3821     if (!doc) return;
3822
3823     str = SysAllocString( szComplete4 );
3824     r = IXMLDOMDocument_loadXML( doc, str, &b );
3825     ok( r == S_OK, "loadXML failed\n");
3826     ok( b == VARIANT_TRUE, "failed to load XML string\n");
3827     SysFreeString( str );
3828
3829     str = SysAllocString( szbs );
3830     r = IXMLDOMDocument_getElementsByTagName( doc, str, &node_list );
3831     ok( r == S_OK, "ret %08x\n", r );
3832     SysFreeString(str);
3833
3834     /* Test to get all child node text. */
3835     r = IXMLDOMDocument_QueryInterface(doc, &IID_IXMLDOMNode, (void**)&nodeRoot);
3836     ok( r == S_OK, "ret %08x\n", r );
3837     if(r == S_OK)
3838     {
3839         r = IXMLDOMNode_get_text( nodeRoot, &str );
3840         ok( r == S_OK, "ret %08x\n", r );
3841         ok( compareIgnoreReturns(str, _bstr_("fn1.txt\n\n fn2.txt \n\nf1\n")), "wrong get_text: %s\n", wine_dbgstr_w(str));
3842         SysFreeString(str);
3843
3844         IXMLDOMNode_Release(nodeRoot);
3845     }
3846
3847     r = IXMLDOMNodeList_get_length( node_list, NULL );
3848     ok( r == E_INVALIDARG, "ret %08x\n", r );
3849
3850     r = IXMLDOMNodeList_get_length( node_list, &len );
3851     ok( r == S_OK, "ret %08x\n", r );
3852     ok( len == 1, "expect 1 got %d\n", len );
3853
3854     r = IXMLDOMNodeList_get_item( node_list, 0, NULL );
3855     ok( r == E_INVALIDARG, "ret %08x\n", r );
3856
3857     r = IXMLDOMNodeList_nextNode( node_list, NULL );
3858     ok( r == E_INVALIDARG, "ret %08x\n", r );
3859
3860     r = IXMLDOMNodeList_get_item( node_list, 0, &node );
3861     ok( r == S_OK, "ret %08x\n", r );
3862     IXMLDOMNodeList_Release( node_list );
3863
3864     /* Invalid output parameter*/
3865     r = IXMLDOMNode_get_text( node, NULL );
3866     ok( r == E_INVALIDARG, "ret %08x\n", r );
3867
3868     r = IXMLDOMNode_get_text( node, &str );
3869     ok( r == S_OK, "ret %08x\n", r );
3870     ok( !memcmp(str, szfn1_txt, lstrlenW(szfn1_txt) ), "wrong string\n" );
3871     SysFreeString(str);
3872
3873     r = IXMLDOMNode_get_attributes( node, &node_map );
3874     ok( r == S_OK, "ret %08x\n", r );
3875
3876     str = SysAllocString( szvr );
3877     r = IXMLDOMNamedNodeMap_getNamedItem( node_map, str, &node2 );
3878     ok( r == S_OK, "ret %08x\n", r );
3879     SysFreeString(str);
3880
3881     r = IXMLDOMNode_get_text( node2, &str );
3882     ok( r == S_OK, "ret %08x\n", r );
3883     ok( !memcmp(str, szstr2, sizeof(szstr2)), "wrong string\n" );
3884     SysFreeString(str);
3885
3886     r = IXMLDOMNode_get_firstChild( node2, &node3 );
3887     ok( r == S_OK, "ret %08x\n", r );
3888
3889     r = IXMLDOMNode_get_text( node3, &str );
3890     ok( r == S_OK, "ret %08x\n", r );
3891     ok( !memcmp(str, szstr2, sizeof(szstr2)), "wrong string\n" );
3892     SysFreeString(str);
3893
3894
3895     IXMLDOMNode_Release( node3 );
3896     IXMLDOMNode_Release( node2 );
3897     IXMLDOMNamedNodeMap_Release( node_map );
3898     IXMLDOMNode_Release( node );
3899     IXMLDOMDocument_Release( doc );
3900 }
3901
3902 static void test_get_childNodes(void)
3903 {
3904     BSTR str;
3905     VARIANT_BOOL b;
3906     IXMLDOMDocument *doc;
3907     IXMLDOMElement *element;
3908     IXMLDOMNode *node, *node2;
3909     IXMLDOMNodeList *node_list, *node_list2;
3910     HRESULT hr;
3911     LONG len;
3912
3913     doc = create_document(&IID_IXMLDOMDocument);
3914     if (!doc) return;
3915
3916     str = SysAllocString( szComplete4 );
3917     hr = IXMLDOMDocument_loadXML( doc, str, &b );
3918     EXPECT_HR(hr, S_OK);
3919     ok( b == VARIANT_TRUE, "failed to load XML string\n");
3920     SysFreeString( str );
3921
3922     hr = IXMLDOMDocument_get_documentElement( doc, &element );
3923     EXPECT_HR(hr, S_OK);
3924
3925     hr = IXMLDOMElement_get_childNodes( element, &node_list );
3926     EXPECT_HR(hr, S_OK);
3927
3928     hr = IXMLDOMNodeList_get_length( node_list, &len );
3929     EXPECT_HR(hr, S_OK);
3930     ok( len == 4, "len %d\n", len);
3931
3932     hr = IXMLDOMNodeList_get_item( node_list, 2, &node );
3933     EXPECT_HR(hr, S_OK);
3934
3935     hr = IXMLDOMNode_get_childNodes( node, &node_list2 );
3936     EXPECT_HR(hr, S_OK);
3937
3938     hr = IXMLDOMNodeList_get_length( node_list2, &len );
3939     EXPECT_HR(hr, S_OK);
3940     ok( len == 0, "len %d\n", len);
3941
3942     hr = IXMLDOMNodeList_get_item( node_list2, 0, &node2);
3943     EXPECT_HR(hr, S_FALSE);
3944
3945     IXMLDOMNodeList_Release( node_list2 );
3946     IXMLDOMNode_Release( node );
3947     IXMLDOMNodeList_Release( node_list );
3948     IXMLDOMElement_Release( element );
3949
3950     /* test for children of <?xml ..?> node */
3951     hr = IXMLDOMDocument_get_firstChild(doc, &node);
3952     EXPECT_HR(hr, S_OK);
3953
3954     str = NULL;
3955     hr = IXMLDOMNode_get_nodeName(node, &str);
3956     EXPECT_HR(hr, S_OK);
3957     ok(!lstrcmpW(str, _bstr_("xml")), "got %s\n", wine_dbgstr_w(str));
3958     SysFreeString(str);
3959
3960     /* it returns empty but valid node list */
3961     node_list = (void*)0xdeadbeef;
3962     hr = IXMLDOMNode_get_childNodes(node, &node_list);
3963     EXPECT_HR(hr, S_OK);
3964
3965     len = -1;
3966     hr = IXMLDOMNodeList_get_length(node_list, &len);
3967     EXPECT_HR(hr, S_OK);
3968     ok(len == 0, "got %d\n", len);
3969
3970     IXMLDOMNodeList_Release( node_list );
3971     IXMLDOMNode_Release(node);
3972
3973     IXMLDOMDocument_Release( doc );
3974     free_bstrs();
3975 }
3976
3977 static void test_get_firstChild(void)
3978 {
3979     static WCHAR xmlW[] = {'x','m','l',0};
3980     IXMLDOMDocument *doc;
3981     IXMLDOMNode *node;
3982     VARIANT_BOOL b;
3983     HRESULT r;
3984     BSTR str;
3985
3986     doc = create_document(&IID_IXMLDOMDocument);
3987     if (!doc) return;
3988
3989     str = SysAllocString( szComplete4 );
3990     r = IXMLDOMDocument_loadXML( doc, str, &b );
3991     ok( r == S_OK, "loadXML failed\n");
3992     ok( b == VARIANT_TRUE, "failed to load XML string\n");
3993     SysFreeString( str );
3994
3995     r = IXMLDOMDocument_get_firstChild( doc, &node );
3996     ok( r == S_OK, "ret %08x\n", r);
3997
3998     r = IXMLDOMNode_get_nodeName( node, &str );
3999     ok( r == S_OK, "ret %08x\n", r);
4000
4001     ok(!lstrcmpW(str, xmlW), "expected \"xml\" node name, got %s\n", wine_dbgstr_w(str));
4002
4003     SysFreeString(str);
4004     IXMLDOMNode_Release( node );
4005     IXMLDOMDocument_Release( doc );
4006 }
4007
4008 static void test_get_lastChild(void)
4009 {
4010     static WCHAR lcW[] = {'l','c',0};
4011     static WCHAR foW[] = {'f','o',0};
4012     IXMLDOMDocument *doc;
4013     IXMLDOMNode *node, *child;
4014     VARIANT_BOOL b;
4015     HRESULT r;
4016     BSTR str;
4017
4018     doc = create_document(&IID_IXMLDOMDocument);
4019     if (!doc) return;
4020
4021     str = SysAllocString( szComplete4 );
4022     r = IXMLDOMDocument_loadXML( doc, str, &b );
4023     ok( r == S_OK, "loadXML failed\n");
4024     ok( b == VARIANT_TRUE, "failed to load XML string\n");
4025     SysFreeString( str );
4026
4027     r = IXMLDOMDocument_get_lastChild( doc, &node );
4028     ok( r == S_OK, "ret %08x\n", r);
4029
4030     r = IXMLDOMNode_get_nodeName( node, &str );
4031     ok( r == S_OK, "ret %08x\n", r);
4032
4033     ok(memcmp(str, lcW, sizeof(lcW)) == 0, "expected \"lc\" node name\n");
4034     SysFreeString(str);
4035
4036     r = IXMLDOMNode_get_lastChild( node, &child );
4037     ok( r == S_OK, "ret %08x\n", r);
4038
4039     r = IXMLDOMNode_get_nodeName( child, &str );
4040     ok( r == S_OK, "ret %08x\n", r);
4041
4042     ok(memcmp(str, foW, sizeof(foW)) == 0, "expected \"fo\" node name\n");
4043     SysFreeString(str);
4044
4045     IXMLDOMNode_Release( child );
4046     IXMLDOMNode_Release( node );
4047     IXMLDOMDocument_Release( doc );
4048 }
4049
4050 static void test_removeChild(void)
4051 {
4052     HRESULT r;
4053     BSTR str;
4054     VARIANT_BOOL b;
4055     IXMLDOMDocument *doc;
4056     IXMLDOMElement *element, *lc_element;
4057     IXMLDOMNode *fo_node, *ba_node, *removed_node, *temp_node, *lc_node;
4058     IXMLDOMNodeList *root_list, *fo_list;
4059
4060     doc = create_document(&IID_IXMLDOMDocument);
4061     if (!doc) return;
4062
4063     str = SysAllocString( szComplete4 );
4064     r = IXMLDOMDocument_loadXML( doc, str, &b );
4065     ok( r == S_OK, "loadXML failed\n");
4066     ok( b == VARIANT_TRUE, "failed to load XML string\n");
4067     SysFreeString( str );
4068
4069     r = IXMLDOMDocument_get_documentElement( doc, &element );
4070     ok( r == S_OK, "ret %08x\n", r);
4071     todo_wine EXPECT_REF(element, 2);
4072
4073     r = IXMLDOMElement_get_childNodes( element, &root_list );
4074     ok( r == S_OK, "ret %08x\n", r);
4075     EXPECT_REF(root_list, 1);
4076
4077     r = IXMLDOMNodeList_get_item( root_list, 3, &fo_node );
4078     ok( r == S_OK, "ret %08x\n", r);
4079     EXPECT_REF(fo_node, 1);
4080
4081     r = IXMLDOMNode_get_childNodes( fo_node, &fo_list );
4082     ok( r == S_OK, "ret %08x\n", r);
4083     EXPECT_REF(fo_list, 1);
4084
4085     r = IXMLDOMNodeList_get_item( fo_list, 0, &ba_node );
4086     ok( r == S_OK, "ret %08x\n", r);
4087     EXPECT_REF(ba_node, 1);
4088
4089     /* invalid parameter: NULL ptr */
4090     removed_node = (void*)0xdeadbeef;
4091     r = IXMLDOMElement_removeChild( element, NULL, &removed_node );
4092     ok( r == E_INVALIDARG, "ret %08x\n", r );
4093     ok( removed_node == (void*)0xdeadbeef, "%p\n", removed_node );
4094
4095     /* ba_node is a descendant of element, but not a direct child. */
4096     removed_node = (void*)0xdeadbeef;
4097     EXPECT_REF(ba_node, 1);
4098     EXPECT_CHILDREN(fo_node);
4099     r = IXMLDOMElement_removeChild( element, ba_node, &removed_node );
4100     ok( r == E_INVALIDARG, "ret %08x\n", r );
4101     ok( removed_node == NULL, "%p\n", removed_node );
4102     EXPECT_REF(ba_node, 1);
4103     EXPECT_CHILDREN(fo_node);
4104
4105     EXPECT_REF(ba_node, 1);
4106     EXPECT_REF(fo_node, 1);
4107     r = IXMLDOMElement_removeChild( element, fo_node, &removed_node );
4108     ok( r == S_OK, "ret %08x\n", r);
4109     ok( fo_node == removed_node, "node %p node2 %p\n", fo_node, removed_node );
4110     EXPECT_REF(fo_node, 2);
4111     EXPECT_REF(ba_node, 1);
4112
4113     /* try removing already removed child */
4114     temp_node = (void*)0xdeadbeef;
4115     r = IXMLDOMElement_removeChild( element, fo_node, &temp_node );
4116     ok( r == E_INVALIDARG, "ret %08x\n", r);
4117     ok( temp_node == NULL, "%p\n", temp_node );
4118     IXMLDOMNode_Release( fo_node );
4119
4120     /* the removed node has no parent anymore */
4121     r = IXMLDOMNode_get_parentNode( removed_node, &temp_node );
4122     ok( r == S_FALSE, "ret %08x\n", r);
4123     ok( temp_node == NULL, "%p\n", temp_node );
4124
4125     IXMLDOMNode_Release( removed_node );
4126     IXMLDOMNode_Release( ba_node );
4127     IXMLDOMNodeList_Release( fo_list );
4128
4129     r = IXMLDOMNodeList_get_item( root_list, 0, &lc_node );
4130     ok( r == S_OK, "ret %08x\n", r);
4131
4132     r = IXMLDOMElement_QueryInterface( lc_node, &IID_IXMLDOMElement, (void**)&lc_element );
4133     ok( r == S_OK, "ret %08x\n", r);
4134
4135     /* MS quirk: passing wrong interface pointer works, too */
4136     r = IXMLDOMElement_removeChild( element, (IXMLDOMNode*)lc_element, NULL );
4137     ok( r == S_OK, "ret %08x\n", r);
4138     IXMLDOMElement_Release( lc_element );
4139
4140     temp_node = (void*)0xdeadbeef;
4141     r = IXMLDOMNode_get_parentNode( lc_node, &temp_node );
4142     ok( r == S_FALSE, "ret %08x\n", r);
4143     ok( temp_node == NULL, "%p\n", temp_node );
4144
4145     IXMLDOMNode_Release( lc_node );
4146     IXMLDOMNodeList_Release( root_list );
4147     IXMLDOMElement_Release( element );
4148     IXMLDOMDocument_Release( doc );
4149 }
4150
4151 static void test_replaceChild(void)
4152 {
4153     HRESULT r;
4154     BSTR str;
4155     VARIANT_BOOL b;
4156     IXMLDOMDocument *doc;
4157     IXMLDOMElement *element, *ba_element;
4158     IXMLDOMNode *fo_node, *ba_node, *lc_node, *removed_node, *temp_node;
4159     IXMLDOMNodeList *root_list, *fo_list;
4160     IUnknown * unk1, *unk2;
4161     LONG len;
4162
4163     doc = create_document(&IID_IXMLDOMDocument);
4164     if (!doc) return;
4165
4166     str = SysAllocString( szComplete4 );
4167     r = IXMLDOMDocument_loadXML( doc, str, &b );
4168     ok( r == S_OK, "loadXML failed\n");
4169     ok( b == VARIANT_TRUE, "failed to load XML string\n");
4170     SysFreeString( str );
4171
4172     r = IXMLDOMDocument_get_documentElement( doc, &element );
4173     ok( r == S_OK, "ret %08x\n", r);
4174
4175     r = IXMLDOMElement_get_childNodes( element, &root_list );
4176     ok( r == S_OK, "ret %08x\n", r);
4177
4178     r = IXMLDOMNodeList_get_item( root_list, 0, &lc_node );
4179     ok( r == S_OK, "ret %08x\n", r);
4180
4181     r = IXMLDOMNodeList_get_item( root_list, 3, &fo_node );
4182     ok( r == S_OK, "ret %08x\n", r);
4183
4184     r = IXMLDOMNode_get_childNodes( fo_node, &fo_list );
4185     ok( r == S_OK, "ret %08x\n", r);
4186
4187     r = IXMLDOMNodeList_get_item( fo_list, 0, &ba_node );
4188     ok( r == S_OK, "ret %08x\n", r);
4189
4190     IXMLDOMNodeList_Release( fo_list );
4191
4192     /* invalid parameter: NULL ptr for element to remove */
4193     removed_node = (void*)0xdeadbeef;
4194     r = IXMLDOMElement_replaceChild( element, ba_node, NULL, &removed_node );
4195     ok( r == E_INVALIDARG, "ret %08x\n", r );
4196     ok( removed_node == (void*)0xdeadbeef, "%p\n", removed_node );
4197
4198     /* invalid parameter: NULL for replacement element. (Sic!) */
4199     removed_node = (void*)0xdeadbeef;
4200     r = IXMLDOMElement_replaceChild( element, NULL, fo_node, &removed_node );
4201     ok( r == E_INVALIDARG, "ret %08x\n", r );
4202     ok( removed_node == (void*)0xdeadbeef, "%p\n", removed_node );
4203
4204     /* invalid parameter: OldNode is not a child */
4205     removed_node = (void*)0xdeadbeef;
4206     r = IXMLDOMElement_replaceChild( element, lc_node, ba_node, &removed_node );
4207     ok( r == E_INVALIDARG, "ret %08x\n", r );
4208     ok( removed_node == NULL, "%p\n", removed_node );
4209     IXMLDOMNode_Release( lc_node );
4210
4211     /* invalid parameter: would create loop */
4212     removed_node = (void*)0xdeadbeef;
4213     r = IXMLDOMNode_replaceChild( fo_node, fo_node, ba_node, &removed_node );
4214     ok( r == E_FAIL, "ret %08x\n", r );
4215     ok( removed_node == NULL, "%p\n", removed_node );
4216
4217     r = IXMLDOMElement_replaceChild( element, ba_node, fo_node, NULL );
4218     ok( r == S_OK, "ret %08x\n", r );
4219
4220     r = IXMLDOMNodeList_get_item( root_list, 3, &temp_node );
4221     ok( r == S_OK, "ret %08x\n", r );
4222
4223     /* ba_node and temp_node refer to the same node, yet they
4224        are different interface pointers */
4225     ok( ba_node != temp_node, "ba_node %p temp_node %p\n", ba_node, temp_node);
4226     r = IXMLDOMNode_QueryInterface( temp_node, &IID_IUnknown, (void**)&unk1);
4227     ok( r == S_OK, "ret %08x\n", r );
4228     r = IXMLDOMNode_QueryInterface( ba_node, &IID_IUnknown, (void**)&unk2);
4229     ok( r == S_OK, "ret %08x\n", r );
4230     todo_wine ok( unk1 == unk2, "unk1 %p unk2 %p\n", unk1, unk2);
4231
4232     IUnknown_Release( unk1 );
4233     IUnknown_Release( unk2 );
4234
4235     /* ba_node should have been removed from below fo_node */
4236     r = IXMLDOMNode_get_childNodes( fo_node, &fo_list );
4237     ok( r == S_OK, "ret %08x\n", r );
4238
4239     /* MS quirk: replaceChild also accepts elements instead of nodes */
4240     r = IXMLDOMNode_QueryInterface( ba_node, &IID_IXMLDOMElement, (void**)&ba_element);
4241     ok( r == S_OK, "ret %08x\n", r );
4242     EXPECT_REF(ba_element, 2);
4243
4244     removed_node = NULL;
4245     r = IXMLDOMElement_replaceChild( element, ba_node, (IXMLDOMNode*)ba_element, &removed_node );
4246     ok( r == S_OK, "ret %08x\n", r );
4247     ok( removed_node != NULL, "got %p\n", removed_node);
4248     EXPECT_REF(ba_element, 3);
4249     IXMLDOMElement_Release( ba_element );
4250
4251     r = IXMLDOMNodeList_get_length( fo_list, &len);
4252     ok( r == S_OK, "ret %08x\n", r );
4253     ok( len == 0, "len %d\n", len);
4254
4255     IXMLDOMNodeList_Release( fo_list );
4256
4257     IXMLDOMNode_Release(ba_node);
4258     IXMLDOMNode_Release(fo_node);
4259     IXMLDOMNode_Release(temp_node);
4260     IXMLDOMNodeList_Release( root_list );
4261     IXMLDOMElement_Release( element );
4262     IXMLDOMDocument_Release( doc );
4263 }
4264
4265 static void test_removeNamedItem(void)
4266 {
4267     IXMLDOMDocument *doc;
4268     IXMLDOMElement *element;
4269     IXMLDOMNode *pr_node, *removed_node, *removed_node2;
4270     IXMLDOMNodeList *root_list;
4271     IXMLDOMNamedNodeMap * pr_attrs;
4272     VARIANT_BOOL b;
4273     BSTR str;
4274     LONG len;
4275     HRESULT r;
4276
4277     doc = create_document(&IID_IXMLDOMDocument);
4278     if (!doc) return;
4279
4280     str = SysAllocString( szComplete4 );
4281     r = IXMLDOMDocument_loadXML( doc, str, &b );
4282     ok( r == S_OK, "loadXML failed\n");
4283     ok( b == VARIANT_TRUE, "failed to load XML string\n");
4284     SysFreeString( str );
4285
4286     r = IXMLDOMDocument_get_documentElement( doc, &element );
4287     ok( r == S_OK, "ret %08x\n", r);
4288
4289     r = IXMLDOMElement_get_childNodes( element, &root_list );
4290     ok( r == S_OK, "ret %08x\n", r);
4291
4292     r = IXMLDOMNodeList_get_item( root_list, 1, &pr_node );
4293     ok( r == S_OK, "ret %08x\n", r);
4294
4295     r = IXMLDOMNode_get_attributes( pr_node, &pr_attrs );
4296     ok( r == S_OK, "ret %08x\n", r);
4297
4298     r = IXMLDOMNamedNodeMap_get_length( pr_attrs, &len );
4299     ok( r == S_OK, "ret %08x\n", r);
4300     ok( len == 3, "length %d\n", len);
4301
4302     removed_node = (void*)0xdeadbeef;
4303     r = IXMLDOMNamedNodeMap_removeNamedItem( pr_attrs, NULL, &removed_node);
4304     ok ( r == E_INVALIDARG, "ret %08x\n", r);
4305     ok ( removed_node == (void*)0xdeadbeef, "got %p\n", removed_node);
4306
4307     removed_node = (void*)0xdeadbeef;
4308     str = SysAllocString(szvr);
4309     r = IXMLDOMNamedNodeMap_removeNamedItem( pr_attrs, str, &removed_node);
4310     ok ( r == S_OK, "ret %08x\n", r);
4311
4312     removed_node2 = (void*)0xdeadbeef;
4313     r = IXMLDOMNamedNodeMap_removeNamedItem( pr_attrs, str, &removed_node2);
4314     ok ( r == S_FALSE, "ret %08x\n", r);
4315     ok ( removed_node2 == NULL, "got %p\n", removed_node2 );
4316
4317     r = IXMLDOMNamedNodeMap_get_length( pr_attrs, &len );
4318     ok( r == S_OK, "ret %08x\n", r);
4319     ok( len == 2, "length %d\n", len);
4320
4321     r = IXMLDOMNamedNodeMap_setNamedItem( pr_attrs, removed_node, NULL);
4322     ok ( r == S_OK, "ret %08x\n", r);
4323     IXMLDOMNode_Release(removed_node);
4324
4325     r = IXMLDOMNamedNodeMap_get_length( pr_attrs, &len );
4326     ok( r == S_OK, "ret %08x\n", r);
4327     ok( len == 3, "length %d\n", len);
4328
4329     r = IXMLDOMNamedNodeMap_removeNamedItem( pr_attrs, str, NULL);
4330     ok ( r == S_OK, "ret %08x\n", r);
4331
4332     r = IXMLDOMNamedNodeMap_get_length( pr_attrs, &len );
4333     ok( r == S_OK, "ret %08x\n", r);
4334     ok( len == 2, "length %d\n", len);
4335
4336     r = IXMLDOMNamedNodeMap_removeNamedItem( pr_attrs, str, NULL);
4337     ok ( r == S_FALSE, "ret %08x\n", r);
4338
4339     SysFreeString(str);
4340
4341     IXMLDOMNamedNodeMap_Release( pr_attrs );
4342     IXMLDOMNode_Release( pr_node );
4343     IXMLDOMNodeList_Release( root_list );
4344     IXMLDOMElement_Release( element );
4345     IXMLDOMDocument_Release( doc );
4346 }
4347
4348 #define test_IObjectSafety_set(p, r, r2, s, m, e, e2) _test_IObjectSafety_set(__LINE__,p, r, r2, s, m, e, e2)
4349 static void _test_IObjectSafety_set(unsigned line, IObjectSafety *safety, HRESULT result,
4350                                     HRESULT result2, DWORD set, DWORD mask, DWORD expected,
4351                                     DWORD expected2)
4352 {
4353     DWORD enabled, supported;
4354     HRESULT hr;
4355
4356     hr = IObjectSafety_SetInterfaceSafetyOptions(safety, NULL, set, mask);
4357     if (result == result2)
4358         ok_(__FILE__,line)(hr == result, "SetInterfaceSafetyOptions: expected %08x, returned %08x\n", result, hr );
4359     else
4360         ok_(__FILE__,line)(broken(hr == result) || hr == result2,
4361            "SetInterfaceSafetyOptions: expected %08x, got %08x\n", result2, hr );
4362
4363     supported = enabled = 0xCAFECAFE;
4364     hr = IObjectSafety_GetInterfaceSafetyOptions(safety, NULL, &supported, &enabled);
4365     ok(hr == S_OK, "ret %08x\n", hr );
4366     if (expected == expected2)
4367         ok_(__FILE__,line)(enabled == expected, "Expected %08x, got %08x\n", expected, enabled);
4368     else
4369         ok_(__FILE__,line)(broken(enabled == expected) || enabled == expected2,
4370            "Expected %08x, got %08x\n", expected2, enabled);
4371
4372     /* reset the safety options */
4373
4374     hr = IObjectSafety_SetInterfaceSafetyOptions(safety, NULL,
4375             INTERFACESAFE_FOR_UNTRUSTED_CALLER|INTERFACESAFE_FOR_UNTRUSTED_DATA|INTERFACE_USES_SECURITY_MANAGER,
4376             0);
4377     ok_(__FILE__,line)(hr == S_OK, "ret %08x\n", hr );
4378
4379     hr = IObjectSafety_GetInterfaceSafetyOptions(safety, NULL, &supported, &enabled);
4380     ok_(__FILE__,line)(hr == S_OK, "ret %08x\n", hr );
4381     ok_(__FILE__,line)(enabled == 0, "Expected 0, got %08x\n", enabled);
4382 }
4383
4384 #define test_IObjectSafety_common(s) _test_IObjectSafety_common(__LINE__,s)
4385 static void _test_IObjectSafety_common(unsigned line, IObjectSafety *safety)
4386 {
4387     DWORD enabled = 0, supported = 0;
4388     HRESULT hr;
4389
4390     /* get */
4391     hr = IObjectSafety_GetInterfaceSafetyOptions(safety, NULL, NULL, &enabled);
4392     ok_(__FILE__,line)(hr == E_POINTER, "ret %08x\n", hr );
4393     hr = IObjectSafety_GetInterfaceSafetyOptions(safety, NULL, &supported, NULL);
4394     ok_(__FILE__,line)(hr == E_POINTER, "ret %08x\n", hr );
4395
4396     hr = IObjectSafety_GetInterfaceSafetyOptions(safety, NULL, &supported, &enabled);
4397     ok_(__FILE__,line)(hr == S_OK, "ret %08x\n", hr );
4398     ok_(__FILE__,line)(broken(supported == (INTERFACESAFE_FOR_UNTRUSTED_CALLER | INTERFACESAFE_FOR_UNTRUSTED_DATA)) ||
4399        supported == (INTERFACESAFE_FOR_UNTRUSTED_CALLER | INTERFACESAFE_FOR_UNTRUSTED_DATA | INTERFACE_USES_SECURITY_MANAGER) /* msxml3 SP8+ */,
4400         "Expected (INTERFACESAFE_FOR_UNTRUSTED_CALLER | INTERFACESAFE_FOR_UNTRUSTED_DATA | INTERFACE_USES_SECURITY_MANAGER), "
4401              "got %08x\n", supported);
4402     ok_(__FILE__,line)(enabled == 0, "Expected 0, got %08x\n", enabled);
4403
4404     /* set -- individual flags */
4405
4406     test_IObjectSafety_set(safety, S_OK, S_OK,
4407         INTERFACESAFE_FOR_UNTRUSTED_CALLER, INTERFACESAFE_FOR_UNTRUSTED_CALLER,
4408         INTERFACESAFE_FOR_UNTRUSTED_CALLER, INTERFACESAFE_FOR_UNTRUSTED_CALLER);
4409
4410     test_IObjectSafety_set(safety, S_OK, S_OK,
4411         INTERFACESAFE_FOR_UNTRUSTED_DATA, INTERFACESAFE_FOR_UNTRUSTED_DATA,
4412         INTERFACESAFE_FOR_UNTRUSTED_DATA, INTERFACESAFE_FOR_UNTRUSTED_DATA);
4413
4414     test_IObjectSafety_set(safety, S_OK, S_OK,
4415         INTERFACE_USES_SECURITY_MANAGER, INTERFACE_USES_SECURITY_MANAGER,
4416         0, INTERFACE_USES_SECURITY_MANAGER /* msxml3 SP8+ */);
4417
4418     /* set INTERFACE_USES_DISPEX  */
4419
4420     test_IObjectSafety_set(safety, S_OK, E_FAIL /* msxml3 SP8+ */,
4421         INTERFACE_USES_DISPEX, INTERFACE_USES_DISPEX,
4422         0, 0);
4423
4424     test_IObjectSafety_set(safety, S_OK, E_FAIL /* msxml3 SP8+ */,
4425         INTERFACE_USES_DISPEX, 0,
4426         0, 0);
4427
4428     test_IObjectSafety_set(safety, S_OK, S_OK /* msxml3 SP8+ */,
4429         0, INTERFACE_USES_DISPEX,
4430         0, 0);
4431
4432     /* set option masking */
4433
4434     test_IObjectSafety_set(safety, S_OK, S_OK,
4435         INTERFACESAFE_FOR_UNTRUSTED_CALLER|INTERFACESAFE_FOR_UNTRUSTED_DATA,
4436         INTERFACESAFE_FOR_UNTRUSTED_CALLER,
4437         INTERFACESAFE_FOR_UNTRUSTED_CALLER,
4438         INTERFACESAFE_FOR_UNTRUSTED_CALLER);
4439
4440     test_IObjectSafety_set(safety, S_OK, S_OK,
4441         INTERFACESAFE_FOR_UNTRUSTED_CALLER|INTERFACESAFE_FOR_UNTRUSTED_DATA,
4442         INTERFACESAFE_FOR_UNTRUSTED_DATA,
4443         INTERFACESAFE_FOR_UNTRUSTED_DATA,
4444         INTERFACESAFE_FOR_UNTRUSTED_DATA);
4445
4446     test_IObjectSafety_set(safety, S_OK, S_OK,
4447         INTERFACESAFE_FOR_UNTRUSTED_CALLER|INTERFACESAFE_FOR_UNTRUSTED_DATA,
4448         INTERFACE_USES_SECURITY_MANAGER,
4449         0,
4450         0);
4451
4452     /* set -- inheriting previous settings */
4453
4454     hr = IObjectSafety_SetInterfaceSafetyOptions(safety, NULL,
4455                                                          INTERFACESAFE_FOR_UNTRUSTED_CALLER,
4456                                                          INTERFACESAFE_FOR_UNTRUSTED_CALLER);
4457     ok_(__FILE__,line)(hr == S_OK, "ret %08x\n", hr );
4458     hr = IObjectSafety_GetInterfaceSafetyOptions(safety, NULL, &supported, &enabled);
4459     ok_(__FILE__,line)(hr == S_OK, "ret %08x\n", hr );
4460     ok_(__FILE__,line)(enabled == INTERFACESAFE_FOR_UNTRUSTED_CALLER, "Expected INTERFACESAFE_FOR_UNTRUSTED_CALLER got %08x\n", enabled);
4461     ok_(__FILE__,line)(broken(supported == (INTERFACESAFE_FOR_UNTRUSTED_CALLER | INTERFACESAFE_FOR_UNTRUSTED_DATA)) ||
4462        supported == (INTERFACESAFE_FOR_UNTRUSTED_CALLER | INTERFACESAFE_FOR_UNTRUSTED_DATA | INTERFACE_USES_SECURITY_MANAGER) /* msxml3 SP8+ */,
4463         "Expected (INTERFACESAFE_FOR_UNTRUSTED_CALLER | INTERFACESAFE_FOR_UNTRUSTED_DATA | INTERFACE_USES_SECURITY_MANAGER), "
4464              "got %08x\n", supported);
4465
4466     hr = IObjectSafety_SetInterfaceSafetyOptions(safety, NULL,
4467                                                          INTERFACESAFE_FOR_UNTRUSTED_DATA,
4468                                                          INTERFACESAFE_FOR_UNTRUSTED_DATA);
4469     ok_(__FILE__,line)(hr == S_OK, "ret %08x\n", hr );
4470     hr = IObjectSafety_GetInterfaceSafetyOptions(safety, NULL, &supported, &enabled);
4471     ok_(__FILE__,line)(hr == S_OK, "ret %08x\n", hr );
4472     ok_(__FILE__,line)(broken(enabled == INTERFACESAFE_FOR_UNTRUSTED_DATA) ||
4473                        enabled == (INTERFACESAFE_FOR_UNTRUSTED_CALLER | INTERFACESAFE_FOR_UNTRUSTED_DATA),
4474                        "Expected (INTERFACESAFE_FOR_UNTRUSTED_CALLER | INTERFACESAFE_FOR_UNTRUSTED_DATA) got %08x\n", enabled);
4475     ok_(__FILE__,line)(broken(supported == (INTERFACESAFE_FOR_UNTRUSTED_CALLER | INTERFACESAFE_FOR_UNTRUSTED_DATA)) ||
4476        supported == (INTERFACESAFE_FOR_UNTRUSTED_CALLER | INTERFACESAFE_FOR_UNTRUSTED_DATA | INTERFACE_USES_SECURITY_MANAGER) /* msxml3 SP8+ */,
4477         "Expected (INTERFACESAFE_FOR_UNTRUSTED_CALLER | INTERFACESAFE_FOR_UNTRUSTED_DATA | INTERFACE_USES_SECURITY_MANAGER), "
4478              "got %08x\n", supported);
4479 }
4480
4481 static void test_XMLHTTP(void)
4482 {
4483     static const WCHAR wszBody[] = {'m','o','d','e','=','T','e','s','t',0};
4484     static const WCHAR wszUrl[] = {'h','t','t','p',':','/','/',
4485         'c','r','o','s','s','o','v','e','r','.','c','o','d','e','w','e','a','v','e','r','s','.','c','o','m','/',
4486         'p','o','s','t','t','e','s','t','.','p','h','p',0};
4487     static const WCHAR xmltestW[] = {'h','t','t','p',':','/','/',
4488         'c','r','o','s','s','o','v','e','r','.','c','o','d','e','w','e','a','v','e','r','s','.','c','o','m','/',
4489         'x','m','l','t','e','s','t','.','x','m','l',0};
4490     static const WCHAR wszExpectedResponse[] = {'F','A','I','L','E','D',0};
4491     static const CHAR xmltestbodyA[] = "<?xml version=\"1.0\" encoding=\"utf-8\"?>\n<a>TEST</a>\n";
4492
4493     IXMLHttpRequest *xhr;
4494     IObjectSafety *safety;
4495     IObjectWithSite *obj_site, *obj_site2;
4496     BSTR bstrResponse, url;
4497     VARIANT dummy;
4498     VARIANT async;
4499     VARIANT varbody;
4500     LONG state, status, bound;
4501     IDispatch *event;
4502     void *ptr;
4503     HRESULT hr;
4504
4505     hr = CoCreateInstance(&CLSID_XMLHTTPRequest, NULL, CLSCTX_INPROC_SERVER,
4506         &IID_IXMLHttpRequest, (void**)&xhr);
4507     if (FAILED(hr))
4508     {
4509         win_skip("IXMLHTTPRequest is not available (0x%08x)\n", hr);
4510         return;
4511     }
4512
4513     VariantInit(&dummy);
4514     V_VT(&dummy) = VT_ERROR;
4515     V_ERROR(&dummy) = DISP_E_MEMBERNOTFOUND;
4516     VariantInit(&async);
4517     V_VT(&async) = VT_BOOL;
4518     V_BOOL(&async) = VARIANT_FALSE;
4519     V_VT(&varbody) = VT_BSTR;
4520     V_BSTR(&varbody) = SysAllocString(wszBody);
4521
4522     url = SysAllocString(wszUrl);
4523
4524     hr = IXMLHttpRequest_put_onreadystatechange(xhr, NULL);
4525     EXPECT_HR(hr, S_OK);
4526
4527     hr = IXMLHttpRequest_abort(xhr);
4528     EXPECT_HR(hr, S_OK);
4529
4530     /* send before open */
4531     hr = IXMLHttpRequest_send(xhr, dummy);
4532     ok(hr == E_FAIL || broken(hr == E_UNEXPECTED) /* win2k */, "got 0x%08x\n", hr);
4533
4534     /* initial status code */
4535     hr = IXMLHttpRequest_get_status(xhr, NULL);
4536     EXPECT_HR(hr, E_INVALIDARG);
4537
4538     status = 0xdeadbeef;
4539     hr = IXMLHttpRequest_get_status(xhr, &status);
4540     ok(hr == E_FAIL || broken(hr == E_UNEXPECTED) /* win2k */, "got 0x%08x\n", hr);
4541     ok(status == 0xdeadbeef, "got %d\n", status);
4542
4543     /* invalid parameters */
4544     hr = IXMLHttpRequest_open(xhr, NULL, NULL, async, dummy, dummy);
4545     EXPECT_HR(hr, E_INVALIDARG);
4546
4547     hr = IXMLHttpRequest_open(xhr, _bstr_("POST"), NULL, async, dummy, dummy);
4548     EXPECT_HR(hr, E_INVALIDARG);
4549
4550     hr = IXMLHttpRequest_open(xhr, NULL, url, async, dummy, dummy);
4551     EXPECT_HR(hr, E_INVALIDARG);
4552
4553     hr = IXMLHttpRequest_setRequestHeader(xhr, NULL, NULL);
4554     EXPECT_HR(hr, E_INVALIDARG);
4555
4556     hr = IXMLHttpRequest_setRequestHeader(xhr, _bstr_("header1"), NULL);
4557     ok(hr == E_FAIL || broken(hr == E_UNEXPECTED) /* win2k */, "got 0x%08x\n", hr);
4558
4559     hr = IXMLHttpRequest_setRequestHeader(xhr, NULL, _bstr_("value1"));
4560     EXPECT_HR(hr, E_INVALIDARG);
4561
4562     hr = IXMLHttpRequest_setRequestHeader(xhr, _bstr_("header1"), _bstr_("value1"));
4563     ok(hr == E_FAIL || broken(hr == E_UNEXPECTED) /* win2k */, "got 0x%08x\n", hr);
4564
4565     hr = IXMLHttpRequest_get_readyState(xhr, NULL);
4566     EXPECT_HR(hr, E_INVALIDARG);
4567
4568     state = -1;
4569     hr = IXMLHttpRequest_get_readyState(xhr, &state);
4570     EXPECT_HR(hr, S_OK);
4571     ok(state == READYSTATE_UNINITIALIZED, "got %d, expected READYSTATE_UNINITIALIZED\n", state);
4572
4573     event = create_dispevent();
4574
4575     EXPECT_REF(event, 1);
4576     hr = IXMLHttpRequest_put_onreadystatechange(xhr, event);
4577     EXPECT_HR(hr, S_OK);
4578     EXPECT_REF(event, 2);
4579
4580     g_unexpectedcall = g_expectedcall = 0;
4581
4582     hr = IXMLHttpRequest_open(xhr, _bstr_("POST"), url, async, dummy, dummy);
4583     EXPECT_HR(hr, S_OK);
4584
4585     ok(g_unexpectedcall == 0, "unexpected disp event call\n");
4586     ok(g_expectedcall == 1 || broken(g_expectedcall == 0) /* win2k */, "no expected disp event call\n");
4587
4588     /* status code after ::open() */
4589     status = 0xdeadbeef;
4590     hr = IXMLHttpRequest_get_status(xhr, &status);
4591     ok(hr == E_FAIL || broken(hr == E_UNEXPECTED) /* win2k */, "got 0x%08x\n", hr);
4592     ok(status == 0xdeadbeef, "got %d\n", status);
4593
4594     state = -1;
4595     hr = IXMLHttpRequest_get_readyState(xhr, &state);
4596     EXPECT_HR(hr, S_OK);
4597     ok(state == READYSTATE_LOADING, "got %d, expected READYSTATE_LOADING\n", state);
4598
4599     hr = IXMLHttpRequest_abort(xhr);
4600     EXPECT_HR(hr, S_OK);
4601
4602     state = -1;
4603     hr = IXMLHttpRequest_get_readyState(xhr, &state);
4604     EXPECT_HR(hr, S_OK);
4605     ok(state == READYSTATE_UNINITIALIZED || broken(state == READYSTATE_LOADING) /* win2k */,
4606         "got %d, expected READYSTATE_UNINITIALIZED\n", state);
4607
4608     hr = IXMLHttpRequest_open(xhr, _bstr_("POST"), url, async, dummy, dummy);
4609     EXPECT_HR(hr, S_OK);
4610
4611     hr = IXMLHttpRequest_setRequestHeader(xhr, _bstr_("header1"), _bstr_("value1"));
4612     EXPECT_HR(hr, S_OK);
4613
4614     hr = IXMLHttpRequest_setRequestHeader(xhr, NULL, _bstr_("value1"));
4615     EXPECT_HR(hr, E_INVALIDARG);
4616
4617     hr = IXMLHttpRequest_setRequestHeader(xhr, _bstr_(""), _bstr_("value1"));
4618     EXPECT_HR(hr, E_INVALIDARG);
4619
4620     SysFreeString(url);
4621
4622     hr = IXMLHttpRequest_send(xhr, varbody);
4623     if (hr == INET_E_RESOURCE_NOT_FOUND)
4624     {
4625         skip("No connection could be made with crossover.codeweavers.com\n");
4626         IXMLHttpRequest_Release(xhr);
4627         return;
4628     }
4629     EXPECT_HR(hr, S_OK);
4630
4631     /* status code after ::send() */
4632     status = 0xdeadbeef;
4633     hr = IXMLHttpRequest_get_status(xhr, &status);
4634     EXPECT_HR(hr, S_OK);
4635     ok(status == 200, "got %d\n", status);
4636
4637     /* another ::send() after completed request */
4638     hr = IXMLHttpRequest_send(xhr, varbody);
4639     ok(hr == E_FAIL || broken(hr == E_UNEXPECTED) /* win2k */, "got 0x%08x\n", hr);
4640
4641     VariantClear(&varbody);
4642
4643     hr = IXMLHttpRequest_get_responseText(xhr, &bstrResponse);
4644     EXPECT_HR(hr, S_OK);
4645     /* the server currently returns "FAILED" because the Content-Type header is
4646      * not what the server expects */
4647     if(hr == S_OK)
4648     {
4649         ok(!memcmp(bstrResponse, wszExpectedResponse, sizeof(wszExpectedResponse)),
4650             "expected %s, got %s\n", wine_dbgstr_w(wszExpectedResponse), wine_dbgstr_w(bstrResponse));
4651         SysFreeString(bstrResponse);
4652     }
4653
4654     /* GET request */
4655     url = SysAllocString(xmltestW);
4656
4657     hr = IXMLHttpRequest_open(xhr, _bstr_("GET"), url, async, dummy, dummy);
4658     EXPECT_HR(hr, S_OK);
4659
4660     V_VT(&varbody) = VT_EMPTY;
4661
4662     hr = IXMLHttpRequest_send(xhr, varbody);
4663     if (hr == INET_E_RESOURCE_NOT_FOUND)
4664     {
4665         skip("No connection could be made with crossover.codeweavers.com\n");
4666         IXMLHttpRequest_Release(xhr);
4667         return;
4668     }
4669     EXPECT_HR(hr, S_OK);
4670
4671     hr = IXMLHttpRequest_get_responseText(xhr, NULL);
4672     EXPECT_HR(hr, E_INVALIDARG);
4673
4674     hr = IXMLHttpRequest_get_responseText(xhr, &bstrResponse);
4675     EXPECT_HR(hr, S_OK);
4676     ok(!memcmp(bstrResponse, _bstr_(xmltestbodyA), sizeof(xmltestbodyA)*sizeof(WCHAR)),
4677         "expected %s, got %s\n", xmltestbodyA, wine_dbgstr_w(bstrResponse));
4678     SysFreeString(bstrResponse);
4679
4680     hr = IXMLHttpRequest_get_responseBody(xhr, NULL);
4681     EXPECT_HR(hr, E_INVALIDARG);
4682
4683     V_VT(&varbody) = VT_EMPTY;
4684     hr = IXMLHttpRequest_get_responseBody(xhr, &varbody);
4685     EXPECT_HR(hr, S_OK);
4686     ok(V_VT(&varbody) == (VT_ARRAY|VT_UI1), "got type %d, expected %d\n", V_VT(&varbody), VT_ARRAY|VT_UI1);
4687     ok(SafeArrayGetDim(V_ARRAY(&varbody)) == 1, "got %d, expected one dimension\n", SafeArrayGetDim(V_ARRAY(&varbody)));
4688
4689     bound = -1;
4690     hr = SafeArrayGetLBound(V_ARRAY(&varbody), 1, &bound);
4691     EXPECT_HR(hr, S_OK);
4692     ok(bound == 0, "got %d, expected zero bound\n", bound);
4693
4694     hr = SafeArrayAccessData(V_ARRAY(&varbody), &ptr);
4695     EXPECT_HR(hr, S_OK);
4696     ok(memcmp(ptr, xmltestbodyA, sizeof(xmltestbodyA)-1) == 0, "got wrong body data\n");
4697     SafeArrayUnaccessData(V_ARRAY(&varbody));
4698
4699     VariantClear(&varbody);
4700
4701     SysFreeString(url);
4702
4703     hr = IXMLHttpRequest_QueryInterface(xhr, &IID_IObjectSafety, (void**)&safety);
4704     EXPECT_HR(hr, S_OK);
4705     if(hr == S_OK)
4706     {
4707         test_IObjectSafety_common(safety);
4708         IObjectSafety_Release(safety);
4709     }
4710
4711     IDispatch_Release(event);
4712
4713     /* interaction with object site */
4714     EXPECT_REF(xhr, 1);
4715     hr = IXMLHttpRequest_QueryInterface(xhr, &IID_IObjectWithSite, (void**)&obj_site);
4716     EXPECT_HR(hr, S_OK);
4717 todo_wine {
4718     EXPECT_REF(xhr, 1);
4719     EXPECT_REF(obj_site, 1);
4720 }
4721
4722     IObjectWithSite_AddRef(obj_site);
4723 todo_wine {
4724     EXPECT_REF(obj_site, 2);
4725     EXPECT_REF(xhr, 1);
4726 }
4727     IObjectWithSite_Release(obj_site);
4728
4729     hr = IXMLHttpRequest_QueryInterface(xhr, &IID_IObjectWithSite, (void**)&obj_site2);
4730     EXPECT_HR(hr, S_OK);
4731 todo_wine {
4732     EXPECT_REF(xhr, 1);
4733     EXPECT_REF(obj_site, 1);
4734     EXPECT_REF(obj_site2, 1);
4735     ok(obj_site != obj_site2, "expected new instance\n");
4736 }
4737     SET_EXPECT(site_qi_IServiceProvider);
4738     SET_EXPECT(sp_queryservice_SID_SBindHost);
4739     SET_EXPECT(sp_queryservice_SID_SContainerDispatch_htmldoc2);
4740     SET_EXPECT(sp_queryservice_SID_secmgr_htmldoc2);
4741     SET_EXPECT(sp_queryservice_SID_secmgr_xmldomdoc);
4742     SET_EXPECT(sp_queryservice_SID_secmgr_secmgr);
4743
4744     /* calls to IHTMLDocument2 */
4745     SET_EXPECT(htmldoc2_get_all);
4746     SET_EXPECT(collection_get_length);
4747     SET_EXPECT(htmldoc2_get_url);
4748
4749     SET_EXPECT(site_qi_IXMLDOMDocument);
4750     SET_EXPECT(site_qi_IOleClientSite);
4751
4752     hr = IObjectWithSite_SetSite(obj_site, &testsite.IUnknown_iface);
4753     EXPECT_HR(hr, S_OK);
4754
4755 todo_wine{
4756     CHECK_CALLED(site_qi_IServiceProvider);
4757
4758     CHECK_CALLED(sp_queryservice_SID_SBindHost);
4759     CHECK_CALLED(sp_queryservice_SID_SContainerDispatch_htmldoc2);
4760     CHECK_CALLED(sp_queryservice_SID_secmgr_htmldoc2);
4761     CHECK_CALLED(sp_queryservice_SID_secmgr_xmldomdoc);
4762     /* this one isn't very reliable
4763     CHECK_CALLED(sp_queryservice_SID_secmgr_secmgr); */
4764
4765     CHECK_CALLED(htmldoc2_get_all);
4766     CHECK_CALLED(collection_get_length);
4767     CHECK_CALLED(htmldoc2_get_url);
4768
4769     CHECK_CALLED(site_qi_IXMLDOMDocument);
4770     CHECK_CALLED(site_qi_IOleClientSite);
4771 }
4772     IObjectWithSite_Release(obj_site);
4773
4774     todo_wine EXPECT_REF(xhr, 1);
4775     IXMLHttpRequest_Release(xhr);
4776
4777     /* still works after request is released */
4778     SET_EXPECT(site_qi_IServiceProvider);
4779
4780     hr = IObjectWithSite_SetSite(obj_site2, &testsite.IUnknown_iface);
4781     EXPECT_HR(hr, S_OK);
4782     IObjectWithSite_Release(obj_site2);
4783
4784     free_bstrs();
4785 }
4786
4787 static void test_IXMLDOMDocument2(void)
4788 {
4789     static const WCHAR emptyW[] = {0};
4790     IXMLDOMDocument2 *doc2, *dtddoc2;
4791     IXMLDOMDocument *doc;
4792     IXMLDOMParseError* err;
4793     IDispatchEx *dispex;
4794     VARIANT_BOOL b;
4795     VARIANT var;
4796     HRESULT r;
4797     LONG res;
4798     BSTR str;
4799
4800     doc = create_document(&IID_IXMLDOMDocument);
4801     if (!doc) return;
4802
4803     dtddoc2 = create_document(&IID_IXMLDOMDocument2);
4804     if (!dtddoc2)
4805     {
4806         IXMLDOMDocument_Release(doc);
4807         return;
4808     }
4809
4810     r = IXMLDOMDocument_QueryInterface( doc, &IID_IXMLDOMDocument2, (void**)&doc2 );
4811     ok( r == S_OK, "ret %08x\n", r );
4812     ok( doc == (IXMLDOMDocument*)doc2, "interfaces differ\n");
4813
4814     ole_expect(IXMLDOMDocument2_get_readyState(doc2, NULL), E_INVALIDARG);
4815     ole_check(IXMLDOMDocument2_get_readyState(doc2, &res));
4816     ok(res == READYSTATE_COMPLETE, "expected READYSTATE_COMPLETE (4), got %i\n", res);
4817
4818     err = NULL;
4819     ole_expect(IXMLDOMDocument2_validate(doc2, NULL), S_FALSE);
4820     ole_expect(IXMLDOMDocument2_validate(doc2, &err), S_FALSE);
4821     ok(err != NULL, "expected a pointer\n");
4822     if (err)
4823     {
4824         res = 0;
4825         ole_check(IXMLDOMParseError_get_errorCode(err, &res));
4826         /* XML_E_NOTWF */
4827         ok(res == E_XML_NOTWF, "got %08x\n", res);
4828         IXMLDOMParseError_Release(err);
4829     }
4830
4831     str = SysAllocString( szComplete4 );
4832     r = IXMLDOMDocument_loadXML( doc2, str, &b );
4833     ok( r == S_OK, "loadXML failed\n");
4834     ok( b == VARIANT_TRUE, "failed to load XML string\n");
4835     SysFreeString( str );
4836
4837     ole_check(IXMLDOMDocument2_get_readyState(doc, &res));
4838     ok(res == READYSTATE_COMPLETE, "expected READYSTATE_COMPLETE (4), got %i\n", res);
4839
4840     err = NULL;
4841     ole_expect(IXMLDOMDocument2_validate(doc2, &err), S_FALSE);
4842     ok(err != NULL, "expected a pointer\n");
4843     if (err)
4844     {
4845         res = 0;
4846         ole_check(IXMLDOMParseError_get_errorCode(err, &res));
4847         /* XML_E_NODTD */
4848         ok(res == E_XML_NODTD, "got %08x\n", res);
4849         IXMLDOMParseError_Release(err);
4850     }
4851
4852     r = IXMLDOMDocument_QueryInterface( doc, &IID_IDispatchEx, (void**)&dispex );
4853     ok( r == S_OK, "ret %08x\n", r );
4854     if(r == S_OK)
4855     {
4856         IDispatchEx_Release(dispex);
4857     }
4858
4859     /* we will check if the variant got cleared */
4860     IXMLDOMDocument2_AddRef(doc2);
4861     EXPECT_REF(doc2, 3); /* doc, doc2, AddRef*/
4862
4863     V_VT(&var) = VT_UNKNOWN;
4864     V_UNKNOWN(&var) = (IUnknown *)doc2;
4865
4866     /* invalid calls */
4867     ole_expect(IXMLDOMDocument2_getProperty(doc2, _bstr_("askldhfaklsdf"), &var), E_FAIL);
4868     expect_eq(V_VT(&var), VT_UNKNOWN, int, "%x");
4869     ole_expect(IXMLDOMDocument2_getProperty(doc2, _bstr_("SelectionLanguage"), NULL), E_INVALIDARG);
4870
4871     /* valid call */
4872     ole_check(IXMLDOMDocument2_getProperty(doc2, _bstr_("SelectionLanguage"), &var));
4873     expect_eq(V_VT(&var), VT_BSTR, int, "%x");
4874     expect_bstr_eq_and_free(V_BSTR(&var), "XSLPattern");
4875     V_VT(&var) = VT_R4;
4876
4877     /* the variant didn't get cleared*/
4878     expect_eq(IXMLDOMDocument2_Release(doc2), 2, int, "%d");
4879
4880     /* setProperty tests */
4881     ole_expect(IXMLDOMDocument2_setProperty(doc2, _bstr_("askldhfaklsdf"), var), E_FAIL);
4882     ole_expect(IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionLanguage"), var), E_FAIL);
4883     ole_expect(IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionLanguage"), _variantbstr_("alskjdh faklsjd hfk")), E_FAIL);
4884     ole_check(IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionLanguage"), _variantbstr_("XSLPattern")));
4885     ole_check(IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionLanguage"), _variantbstr_("XPath")));
4886     ole_check(IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionLanguage"), _variantbstr_("XSLPattern")));
4887
4888     V_VT(&var) = VT_BSTR;
4889     V_BSTR(&var) = SysAllocString(emptyW);
4890     r = IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionNamespaces"), var);
4891     ok(r == S_OK, "got 0x%08x\n", r);
4892     VariantClear(&var);
4893
4894     V_VT(&var) = VT_I2;
4895     V_I2(&var) = 0;
4896     r = IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionNamespaces"), var);
4897     ok(r == E_FAIL, "got 0x%08x\n", r);
4898
4899     /* contrary to what MSDN claims you can switch back from XPath to XSLPattern */
4900     ole_check(IXMLDOMDocument2_getProperty(doc2, _bstr_("SelectionLanguage"), &var));
4901     expect_eq(V_VT(&var), VT_BSTR, int, "%x");
4902     expect_bstr_eq_and_free(V_BSTR(&var), "XSLPattern");
4903
4904     IXMLDOMDocument2_Release( doc2 );
4905     IXMLDOMDocument_Release( doc );
4906
4907     /* DTD validation */
4908     ole_check(IXMLDOMDocument2_put_validateOnParse(dtddoc2, VARIANT_FALSE));
4909     ole_check(IXMLDOMDocument2_loadXML(dtddoc2, _bstr_(szEmailXML), &b));
4910     ok( b == VARIANT_TRUE, "failed to load XML string\n");
4911     err = NULL;
4912     ole_check(IXMLDOMDocument2_validate(dtddoc2, &err));
4913     ok(err != NULL, "expected pointer\n");
4914     if (err)
4915     {
4916         res = 0;
4917         ole_expect(IXMLDOMParseError_get_errorCode(err, &res), S_FALSE);
4918         ok(res == 0, "got %08x\n", res);
4919         IXMLDOMParseError_Release(err);
4920     }
4921
4922     ole_check(IXMLDOMDocument2_loadXML(dtddoc2, _bstr_(szEmailXML_0D), &b));
4923     ok( b == VARIANT_TRUE, "failed to load XML string\n");
4924     err = NULL;
4925     ole_expect(IXMLDOMDocument2_validate(dtddoc2, &err), S_FALSE);
4926     ok(err != NULL, "expected pointer\n");
4927     if (err)
4928     {
4929         res = 0;
4930         ole_check(IXMLDOMParseError_get_errorCode(err, &res));
4931         /* XML_ELEMENT_UNDECLARED */
4932         todo_wine ok(res == 0xC00CE00D, "got %08x\n", res);
4933         IXMLDOMParseError_Release(err);
4934     }
4935
4936     ole_check(IXMLDOMDocument2_loadXML(dtddoc2, _bstr_(szEmailXML_0E), &b));
4937     ok( b == VARIANT_TRUE, "failed to load XML string\n");
4938     err = NULL;
4939     ole_expect(IXMLDOMDocument2_validate(dtddoc2, &err), S_FALSE);
4940     ok(err != NULL, "expected pointer\n");
4941     if (err)
4942     {
4943         res = 0;
4944         ole_check(IXMLDOMParseError_get_errorCode(err, &res));
4945         /* XML_ELEMENT_ID_NOT_FOUND */
4946         todo_wine ok(res == 0xC00CE00E, "got %08x\n", res);
4947         IXMLDOMParseError_Release(err);
4948     }
4949
4950     ole_check(IXMLDOMDocument2_loadXML(dtddoc2, _bstr_(szEmailXML_11), &b));
4951     ok( b == VARIANT_TRUE, "failed to load XML string\n");
4952     err = NULL;
4953     ole_expect(IXMLDOMDocument2_validate(dtddoc2, &err), S_FALSE);
4954     ok(err != NULL, "expected pointer\n");
4955     if (err)
4956     {
4957         res = 0;
4958         ole_check(IXMLDOMParseError_get_errorCode(err, &res));
4959         /* XML_EMPTY_NOT_ALLOWED */
4960         todo_wine ok(res == 0xC00CE011, "got %08x\n", res);
4961         IXMLDOMParseError_Release(err);
4962     }
4963
4964     ole_check(IXMLDOMDocument2_loadXML(dtddoc2, _bstr_(szEmailXML_13), &b));
4965     ok( b == VARIANT_TRUE, "failed to load XML string\n");
4966     err = NULL;
4967     ole_expect(IXMLDOMDocument2_validate(dtddoc2, &err), S_FALSE);
4968     ok(err != NULL, "expected pointer\n");
4969     if (err)
4970     {
4971         res = 0;
4972         ole_check(IXMLDOMParseError_get_errorCode(err, &res));
4973         /* XML_ROOT_NAME_MISMATCH */
4974         todo_wine ok(res == 0xC00CE013, "got %08x\n", res);
4975         IXMLDOMParseError_Release(err);
4976     }
4977
4978     ole_check(IXMLDOMDocument2_loadXML(dtddoc2, _bstr_(szEmailXML_14), &b));
4979     ok( b == VARIANT_TRUE, "failed to load XML string\n");
4980     err = NULL;
4981     ole_expect(IXMLDOMDocument2_validate(dtddoc2, &err), S_FALSE);
4982     ok(err != NULL, "expected pointer\n");
4983     if (err)
4984     {
4985         res = 0;
4986         ole_check(IXMLDOMParseError_get_errorCode(err, &res));
4987         /* XML_INVALID_CONTENT */
4988         todo_wine ok(res == 0xC00CE014, "got %08x\n", res);
4989         IXMLDOMParseError_Release(err);
4990     }
4991
4992     ole_check(IXMLDOMDocument2_loadXML(dtddoc2, _bstr_(szEmailXML_15), &b));
4993     ok( b == VARIANT_TRUE, "failed to load XML string\n");
4994     err = NULL;
4995     ole_expect(IXMLDOMDocument2_validate(dtddoc2, &err), S_FALSE);
4996     ok(err != NULL, "expected pointer\n");
4997     if (err)
4998     {
4999         res = 0;
5000         ole_check(IXMLDOMParseError_get_errorCode(err, &res));
5001         /* XML_ATTRIBUTE_NOT_DEFINED */
5002         todo_wine ok(res == 0xC00CE015, "got %08x\n", res);
5003         IXMLDOMParseError_Release(err);
5004     }
5005
5006     ole_check(IXMLDOMDocument2_loadXML(dtddoc2, _bstr_(szEmailXML_16), &b));
5007     ok( b == VARIANT_TRUE, "failed to load XML string\n");
5008     err = NULL;
5009     ole_expect(IXMLDOMDocument2_validate(dtddoc2, &err), S_FALSE);
5010     ok(err != NULL, "expected pointer\n");
5011     if (err)
5012     {
5013         res = 0;
5014         ole_check(IXMLDOMParseError_get_errorCode(err, &res));
5015         /* XML_ATTRIBUTE_FIXED */
5016         todo_wine ok(res == 0xC00CE016, "got %08x\n", res);
5017         IXMLDOMParseError_Release(err);
5018     }
5019
5020     ole_check(IXMLDOMDocument2_loadXML(dtddoc2, _bstr_(szEmailXML_17), &b));
5021     ok( b == VARIANT_TRUE, "failed to load XML string\n");
5022     err = NULL;
5023     ole_expect(IXMLDOMDocument2_validate(dtddoc2, &err), S_FALSE);
5024     ok(err != NULL, "expected pointer\n");
5025     if (err)
5026     {
5027         res = 0;
5028         ole_check(IXMLDOMParseError_get_errorCode(err, &res));
5029         /* XML_ATTRIBUTE_VALUE */
5030         todo_wine ok(res == 0xC00CE017, "got %08x\n", res);
5031         IXMLDOMParseError_Release(err);
5032     }
5033
5034     ole_check(IXMLDOMDocument2_loadXML(dtddoc2, _bstr_(szEmailXML_18), &b));
5035     ok( b == VARIANT_TRUE, "failed to load XML string\n");
5036     err = NULL;
5037     ole_expect(IXMLDOMDocument2_validate(dtddoc2, &err), S_FALSE);
5038     ok(err != NULL, "expected pointer\n");
5039     if (err)
5040     {
5041         res = 0;
5042         ole_check(IXMLDOMParseError_get_errorCode(err, &res));
5043         /* XML_ILLEGAL_TEXT */
5044         todo_wine ok(res == 0xC00CE018, "got %08x\n", res);
5045         IXMLDOMParseError_Release(err);
5046     }
5047
5048     ole_check(IXMLDOMDocument2_loadXML(dtddoc2, _bstr_(szEmailXML_20), &b));
5049     ok( b == VARIANT_TRUE, "failed to load XML string\n");
5050     err = NULL;
5051     ole_expect(IXMLDOMDocument2_validate(dtddoc2, &err), S_FALSE);
5052     ok(err != NULL, "expected pointer\n");
5053     if (err)
5054     {
5055         res = 0;
5056         ole_check(IXMLDOMParseError_get_errorCode(err, &res));
5057         /* XML_REQUIRED_ATTRIBUTE_MISSING */
5058         todo_wine ok(res == 0xC00CE020, "got %08x\n", res);
5059         IXMLDOMParseError_Release(err);
5060     }
5061
5062     IXMLDOMDocument2_Release( dtddoc2 );
5063     free_bstrs();
5064 }
5065
5066 #define helper_ole_check(expr) { \
5067     HRESULT r = expr; \
5068     ok_(__FILE__, line)(r == S_OK, "=> %i: " #expr " returned %08x\n", __LINE__, r); \
5069 }
5070
5071 #define helper_expect_list_and_release(list, expstr) { \
5072     char *str = list_to_string(list); \
5073     ok_(__FILE__, line)(strcmp(str, expstr)==0, "=> %i: Invalid node list: %s, expected %s\n", __LINE__, str, expstr); \
5074     if (list) IXMLDOMNodeList_Release(list); \
5075 }
5076
5077 #define helper_expect_bstr_and_release(bstr, str) { \
5078     ok_(__FILE__, line)(lstrcmpW(bstr, _bstr_(str)) == 0, \
5079        "=> %i: got %s\n", __LINE__, wine_dbgstr_w(bstr)); \
5080     SysFreeString(bstr); \
5081 }
5082
5083 #define check_ws_ignored(doc, str) _check_ws_ignored(__LINE__, doc, str)
5084 static inline void _check_ws_ignored(int line, IXMLDOMDocument2* doc, char const* str)
5085 {
5086     IXMLDOMNode *node1, *node2;
5087     IXMLDOMNodeList *list;
5088     BSTR bstr;
5089
5090     helper_ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//*[local-name()='html']"), &list));
5091     helper_ole_check(IXMLDOMNodeList_get_item(list, 0, &node1));
5092     helper_ole_check(IXMLDOMNodeList_get_item(list, 1, &node2));
5093     helper_ole_check(IXMLDOMNodeList_reset(list));
5094     helper_expect_list_and_release(list, "E1.E4.E1.E2.D1 E2.E4.E1.E2.D1");
5095
5096     helper_ole_check(IXMLDOMNode_get_childNodes(node1, &list));
5097     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");
5098     helper_ole_check(IXMLDOMNode_get_text(node1, &bstr));
5099     if (str)
5100     {
5101         helper_expect_bstr_and_release(bstr, str);
5102     }
5103     else
5104     {
5105         helper_expect_bstr_and_release(bstr, "This is a description.");
5106     }
5107     IXMLDOMNode_Release(node1);
5108
5109     helper_ole_check(IXMLDOMNode_get_childNodes(node2, &list));
5110     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");
5111     helper_ole_check(IXMLDOMNode_get_text(node2, &bstr));
5112     helper_expect_bstr_and_release(bstr, "\n                This is a description with preserved whitespace. \n            ");
5113     IXMLDOMNode_Release(node2);
5114 }
5115
5116 #define check_ws_preserved(doc, str) _check_ws_preserved(__LINE__, doc, str)
5117 static inline void _check_ws_preserved(int line, IXMLDOMDocument2* doc, char const* str)
5118 {
5119     IXMLDOMNode *node1, *node2;
5120     IXMLDOMNodeList *list;
5121     BSTR bstr;
5122
5123     helper_ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//*[local-name()='html']"), &list));
5124     helper_ole_check(IXMLDOMNodeList_get_item(list, 0, &node1));
5125     helper_ole_check(IXMLDOMNodeList_get_item(list, 1, &node2));
5126     helper_ole_check(IXMLDOMNodeList_reset(list));
5127     helper_expect_list_and_release(list, "E2.E8.E2.E2.D1 E4.E8.E2.E2.D1");
5128
5129     helper_ole_check(IXMLDOMNode_get_childNodes(node1, &list));
5130     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");
5131     helper_ole_check(IXMLDOMNode_get_text(node1, &bstr));
5132     if (str)
5133     {
5134         helper_expect_bstr_and_release(bstr, str);
5135     }
5136     else
5137     {
5138         helper_expect_bstr_and_release(bstr, "\n                This is a description. \n            ");
5139     }
5140     IXMLDOMNode_Release(node1);
5141
5142     helper_ole_check(IXMLDOMNode_get_childNodes(node2, &list));
5143     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");
5144     helper_ole_check(IXMLDOMNode_get_text(node2, &bstr));
5145     helper_expect_bstr_and_release(bstr, "\n                This is a description with preserved whitespace. \n            ");
5146     IXMLDOMNode_Release(node2);
5147 }
5148
5149 static void test_whitespace(void)
5150 {
5151     VARIANT_BOOL b;
5152     IXMLDOMDocument2 *doc1, *doc2, *doc3, *doc4;
5153
5154     doc1 = create_document(&IID_IXMLDOMDocument2);
5155     doc2 = create_document(&IID_IXMLDOMDocument2);
5156     if (!doc1 || !doc2) return;
5157
5158     ole_check(IXMLDOMDocument2_put_preserveWhiteSpace(doc2, VARIANT_TRUE));
5159     ole_check(IXMLDOMDocument2_get_preserveWhiteSpace(doc1, &b));
5160     ok(b == VARIANT_FALSE, "expected false\n");
5161     ole_check(IXMLDOMDocument2_get_preserveWhiteSpace(doc2, &b));
5162     ok(b == VARIANT_TRUE, "expected true\n");
5163
5164     ole_check(IXMLDOMDocument2_loadXML(doc1, _bstr_(szExampleXML), &b));
5165     ok(b == VARIANT_TRUE, "failed to load XML string\n");
5166     ole_check(IXMLDOMDocument2_loadXML(doc2, _bstr_(szExampleXML), &b));
5167     ok(b == VARIANT_TRUE, "failed to load XML string\n");
5168
5169     /* switch to XPath */
5170     ole_check(IXMLDOMDocument2_setProperty(doc1, _bstr_("SelectionLanguage"), _variantbstr_("XPath")));
5171     ole_check(IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionLanguage"), _variantbstr_("XPath")));
5172
5173     check_ws_ignored(doc1, NULL);
5174     check_ws_preserved(doc2, NULL);
5175
5176     /* new instances copy the property */
5177     ole_check(IXMLDOMDocument2_QueryInterface(doc1, &IID_IXMLDOMDocument2, (void**) &doc3));
5178     ole_check(IXMLDOMDocument2_QueryInterface(doc2, &IID_IXMLDOMDocument2, (void**) &doc4));
5179
5180     ole_check(IXMLDOMDocument2_get_preserveWhiteSpace(doc3, &b));
5181     ok(b == VARIANT_FALSE, "expected false\n");
5182     ole_check(IXMLDOMDocument2_get_preserveWhiteSpace(doc4, &b));
5183     ok(b == VARIANT_TRUE, "expected true\n");
5184
5185     check_ws_ignored(doc3, NULL);
5186     check_ws_preserved(doc4, NULL);
5187
5188     /* setting after loading xml affects trimming of leading/trailing ws only */
5189     ole_check(IXMLDOMDocument2_put_preserveWhiteSpace(doc1, VARIANT_TRUE));
5190     ole_check(IXMLDOMDocument2_put_preserveWhiteSpace(doc2, VARIANT_FALSE));
5191
5192     /* the trailing "\n            " isn't there, because it was ws-only node */
5193     check_ws_ignored(doc1, "\n                This is a description. ");
5194     check_ws_preserved(doc2, "This is a description.");
5195
5196     /* it takes effect on reload */
5197     ole_check(IXMLDOMDocument2_get_preserveWhiteSpace(doc1, &b));
5198     ok(b == VARIANT_TRUE, "expected true\n");
5199     ole_check(IXMLDOMDocument2_get_preserveWhiteSpace(doc2, &b));
5200     ok(b == VARIANT_FALSE, "expected false\n");
5201
5202     ole_check(IXMLDOMDocument2_loadXML(doc1, _bstr_(szExampleXML), &b));
5203     ok(b == VARIANT_TRUE, "failed to load XML string\n");
5204     ole_check(IXMLDOMDocument2_loadXML(doc2, _bstr_(szExampleXML), &b));
5205     ok(b == VARIANT_TRUE, "failed to load XML string\n");
5206
5207     check_ws_preserved(doc1, NULL);
5208     check_ws_ignored(doc2, NULL);
5209
5210     /* other instances follow suit */
5211     ole_check(IXMLDOMDocument2_get_preserveWhiteSpace(doc3, &b));
5212     ok(b == VARIANT_TRUE, "expected true\n");
5213     ole_check(IXMLDOMDocument2_get_preserveWhiteSpace(doc4, &b));
5214     ok(b == VARIANT_FALSE, "expected false\n");
5215
5216     check_ws_preserved(doc3, NULL);
5217     check_ws_ignored(doc4, NULL);
5218
5219     IXMLDOMDocument_Release(doc1);
5220     IXMLDOMDocument_Release(doc2);
5221     IXMLDOMDocument_Release(doc3);
5222     IXMLDOMDocument_Release(doc4);
5223     free_bstrs();
5224 }
5225
5226 static void test_XPath(void)
5227 {
5228     VARIANT var;
5229     VARIANT_BOOL b;
5230     IXMLDOMDocument2 *doc;
5231     IXMLDOMDocument *doc2;
5232     IXMLDOMNode *rootNode;
5233     IXMLDOMNode *elem1Node;
5234     IXMLDOMNode *node;
5235     IXMLDOMNodeList *list;
5236     IXMLDOMElement *elem;
5237     IXMLDOMAttribute *attr;
5238     HRESULT hr;
5239     BSTR str;
5240
5241     doc = create_document(&IID_IXMLDOMDocument2);
5242     if (!doc) return;
5243
5244     ole_check(IXMLDOMDocument2_loadXML(doc, _bstr_(szExampleXML), &b));
5245     ok(b == VARIANT_TRUE, "failed to load XML string\n");
5246
5247     /* switch to XPath */
5248     ole_check(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionLanguage"), _variantbstr_("XPath")));
5249
5250     /* some simple queries*/
5251     EXPECT_REF(doc, 1);
5252     hr = IXMLDOMDocument2_selectNodes(doc, _bstr_("root"), &list);
5253     EXPECT_HR(hr, S_OK);
5254     EXPECT_REF(doc, 1);
5255     EXPECT_LIST_LEN(list, 1);
5256
5257     EXPECT_REF(list, 1);
5258     hr = IXMLDOMNodeList_get_item(list, 0, &rootNode);
5259     EXPECT_HR(hr, S_OK);
5260     EXPECT_REF(list, 1);
5261     EXPECT_REF(rootNode, 1);
5262
5263     hr = IXMLDOMNodeList_reset(list);
5264     EXPECT_HR(hr, S_OK);
5265     expect_list_and_release(list, "E2.D1");
5266
5267     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//c"), &list));
5268     expect_list_and_release(list, "E3.E1.E2.D1 E3.E2.E2.D1");
5269
5270     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//c[@type]"), &list));
5271     expect_list_and_release(list, "E3.E2.E2.D1");
5272
5273     ole_check(IXMLDOMNode_selectNodes(rootNode, _bstr_("elem"), &list));
5274     /* using get_item for query results advances the position */
5275     ole_check(IXMLDOMNodeList_get_item(list, 1, &node));
5276     expect_node(node, "E2.E2.D1");
5277     IXMLDOMNode_Release(node);
5278     ole_check(IXMLDOMNodeList_nextNode(list, &node));
5279     expect_node(node, "E4.E2.D1");
5280     IXMLDOMNode_Release(node);
5281     ole_check(IXMLDOMNodeList_reset(list));
5282     expect_list_and_release(list, "E1.E2.D1 E2.E2.D1 E4.E2.D1");
5283
5284     ole_check(IXMLDOMNode_selectNodes(rootNode, _bstr_("."), &list));
5285     expect_list_and_release(list, "E2.D1");
5286
5287     ole_check(IXMLDOMNode_selectNodes(rootNode, _bstr_("elem[3]/preceding-sibling::*"), &list));
5288     ole_check(IXMLDOMNodeList_get_item(list, 0, &elem1Node));
5289     ole_check(IXMLDOMNodeList_reset(list));
5290     expect_list_and_release(list, "E1.E2.D1 E2.E2.D1 E3.E2.D1");
5291
5292     /* select an attribute */
5293     ole_check(IXMLDOMNode_selectNodes(rootNode, _bstr_(".//@type"), &list));
5294     expect_list_and_release(list, "A'type'.E3.E2.E2.D1");
5295
5296     /* would evaluate to a number */
5297     ole_expect(IXMLDOMNode_selectNodes(rootNode, _bstr_("count(*)"), &list), E_FAIL);
5298     /* would evaluate to a boolean */
5299     ole_expect(IXMLDOMNode_selectNodes(rootNode, _bstr_("position()>0"), &list), E_FAIL);
5300     /* would evaluate to a string */
5301     ole_expect(IXMLDOMNode_selectNodes(rootNode, _bstr_("name()"), &list), E_FAIL);
5302
5303     /* no results */
5304     ole_check(IXMLDOMNode_selectNodes(rootNode, _bstr_("c"), &list));
5305     expect_list_and_release(list, "");
5306     ole_check(IXMLDOMDocument_selectNodes(doc, _bstr_("elem//c"), &list));
5307     expect_list_and_release(list, "");
5308     ole_check(IXMLDOMDocument_selectNodes(doc, _bstr_("//elem[4]"), &list));
5309     expect_list_and_release(list, "");
5310     ole_check(IXMLDOMDocument_selectNodes(doc, _bstr_("root//elem[0]"), &list));
5311     expect_list_and_release(list, "");
5312
5313     /* foo undeclared in document node */
5314     ole_expect(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//foo:c"), &list), E_FAIL);
5315     /* undeclared in <root> node */
5316     ole_expect(IXMLDOMNode_selectNodes(rootNode, _bstr_(".//foo:c"), &list), E_FAIL);
5317     /* undeclared in <elem> node */
5318     ole_expect(IXMLDOMNode_selectNodes(elem1Node, _bstr_("//foo:c"), &list), E_FAIL);
5319     /* but this trick can be used */
5320     ole_check(IXMLDOMNode_selectNodes(elem1Node, _bstr_("//*[name()='foo:c']"), &list));
5321     expect_list_and_release(list, "E3.E4.E2.D1");
5322
5323     /* it has to be declared in SelectionNamespaces */
5324     ole_check(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionNamespaces"),
5325         _variantbstr_("xmlns:test='urn:uuid:86B2F87F-ACB6-45cd-8B77-9BDB92A01A29'")));
5326
5327     /* now the namespace can be used */
5328     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//test:c"), &list));
5329     expect_list_and_release(list, "E3.E3.E2.D1 E3.E4.E2.D1");
5330     ole_check(IXMLDOMNode_selectNodes(rootNode, _bstr_(".//test:c"), &list));
5331     expect_list_and_release(list, "E3.E3.E2.D1 E3.E4.E2.D1");
5332     ole_check(IXMLDOMNode_selectNodes(elem1Node, _bstr_("//test:c"), &list));
5333     expect_list_and_release(list, "E3.E3.E2.D1 E3.E4.E2.D1");
5334     ole_check(IXMLDOMNode_selectNodes(elem1Node, _bstr_(".//test:x"), &list));
5335     expect_list_and_release(list, "E5.E1.E4.E1.E2.D1 E6.E2.E4.E1.E2.D1");
5336
5337     /* SelectionNamespaces syntax error - the namespaces doesn't work anymore but the value is stored */
5338     ole_expect(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionNamespaces"),
5339         _variantbstr_("xmlns:test='urn:uuid:86B2F87F-ACB6-45cd-8B77-9BDB92A01A29' xmlns:foo=###")), E_FAIL);
5340
5341     ole_expect(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//foo:c"), &list), E_FAIL);
5342
5343     VariantInit(&var);
5344     ole_check(IXMLDOMDocument2_getProperty(doc, _bstr_("SelectionNamespaces"), &var));
5345     expect_eq(V_VT(&var), VT_BSTR, int, "%x");
5346     if (V_VT(&var) == VT_BSTR)
5347         expect_bstr_eq_and_free(V_BSTR(&var), "xmlns:test='urn:uuid:86B2F87F-ACB6-45cd-8B77-9BDB92A01A29' xmlns:foo=###");
5348
5349     /* extra attributes - same thing*/
5350     ole_expect(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionNamespaces"),
5351         _variantbstr_("xmlns:test='urn:uuid:86B2F87F-ACB6-45cd-8B77-9BDB92A01A29' param='test'")), E_FAIL);
5352     ole_expect(IXMLDOMDocument_selectNodes(doc, _bstr_("root//foo:c"), &list), E_FAIL);
5353
5354     IXMLDOMNode_Release(rootNode);
5355     IXMLDOMNode_Release(elem1Node);
5356
5357     /* alter document with already built list */
5358     hr = IXMLDOMDocument2_selectNodes(doc, _bstr_("root"), &list);
5359     EXPECT_HR(hr, S_OK);
5360     EXPECT_LIST_LEN(list, 1);
5361
5362     hr = IXMLDOMDocument2_get_lastChild(doc, &rootNode);
5363     EXPECT_HR(hr, S_OK);
5364     EXPECT_REF(rootNode, 1);
5365     EXPECT_REF(doc, 1);
5366
5367     hr = IXMLDOMDocument2_removeChild(doc, rootNode, NULL);
5368     EXPECT_HR(hr, S_OK);
5369     IXMLDOMNode_Release(rootNode);
5370
5371     EXPECT_LIST_LEN(list, 1);
5372
5373     hr = IXMLDOMNodeList_get_item(list, 0, &rootNode);
5374     EXPECT_HR(hr, S_OK);
5375     EXPECT_REF(rootNode, 1);
5376
5377     IXMLDOMNodeList_Release(list);
5378
5379     hr = IXMLDOMNode_get_nodeName(rootNode, &str);
5380     EXPECT_HR(hr, S_OK);
5381     ok(!lstrcmpW(str, _bstr_("root")), "got %s\n", wine_dbgstr_w(str));
5382     SysFreeString(str);
5383     IXMLDOMNode_Release(rootNode);
5384
5385     /* alter node from list and get it another time */
5386     hr = IXMLDOMDocument2_loadXML(doc, _bstr_(szExampleXML), &b);
5387     EXPECT_HR(hr, S_OK);
5388     ok(b == VARIANT_TRUE, "failed to load XML string\n");
5389
5390     hr = IXMLDOMDocument2_selectNodes(doc, _bstr_("root"), &list);
5391     EXPECT_HR(hr, S_OK);
5392     EXPECT_LIST_LEN(list, 1);
5393
5394     hr = IXMLDOMNodeList_get_item(list, 0, &rootNode);
5395     EXPECT_HR(hr, S_OK);
5396
5397     hr = IXMLDOMNode_QueryInterface(rootNode, &IID_IXMLDOMElement, (void**)&elem);
5398     EXPECT_HR(hr, S_OK);
5399
5400     V_VT(&var) = VT_I2;
5401     V_I2(&var) = 1;
5402     hr = IXMLDOMElement_setAttribute(elem, _bstr_("attrtest"), var);
5403     EXPECT_HR(hr, S_OK);
5404     IXMLDOMElement_Release(elem);
5405     IXMLDOMNode_Release(rootNode);
5406
5407     /* now check attribute to be present */
5408     hr = IXMLDOMNodeList_get_item(list, 0, &rootNode);
5409     EXPECT_HR(hr, S_OK);
5410
5411     hr = IXMLDOMNode_QueryInterface(rootNode, &IID_IXMLDOMElement, (void**)&elem);
5412     EXPECT_HR(hr, S_OK);
5413
5414     hr = IXMLDOMElement_getAttributeNode(elem, _bstr_("attrtest"), &attr);
5415     EXPECT_HR(hr, S_OK);
5416     IXMLDOMAttribute_Release(attr);
5417
5418     IXMLDOMElement_Release(elem);
5419     IXMLDOMNode_Release(rootNode);
5420
5421     /* and now check for attribute in original document */
5422     hr = IXMLDOMDocument_get_documentElement(doc, &elem);
5423     EXPECT_HR(hr, S_OK);
5424
5425     hr = IXMLDOMElement_getAttributeNode(elem, _bstr_("attrtest"), &attr);
5426     EXPECT_HR(hr, S_OK);
5427     IXMLDOMAttribute_Release(attr);
5428
5429     IXMLDOMElement_Release(elem);
5430
5431     /* attach node from list to another document */
5432     doc2 = create_document(&IID_IXMLDOMDocument);
5433
5434     hr = IXMLDOMDocument2_loadXML(doc, _bstr_(szExampleXML), &b);
5435     EXPECT_HR(hr, S_OK);
5436     ok(b == VARIANT_TRUE, "failed to load XML string\n");
5437
5438     hr = IXMLDOMDocument2_selectNodes(doc, _bstr_("root"), &list);
5439     EXPECT_HR(hr, S_OK);
5440     EXPECT_LIST_LEN(list, 1);
5441
5442     hr = IXMLDOMNodeList_get_item(list, 0, &rootNode);
5443     EXPECT_HR(hr, S_OK);
5444     EXPECT_REF(rootNode, 1);
5445
5446     hr = IXMLDOMDocument_appendChild(doc2, rootNode, NULL);
5447     EXPECT_HR(hr, S_OK);
5448     EXPECT_REF(rootNode, 1);
5449     EXPECT_REF(doc2, 1);
5450     EXPECT_REF(list, 1);
5451
5452     EXPECT_LIST_LEN(list, 1);
5453
5454     IXMLDOMNode_Release(rootNode);
5455     IXMLDOMNodeList_Release(list);
5456     IXMLDOMDocument_Release(doc2);
5457
5458     IXMLDOMDocument2_Release(doc);
5459     free_bstrs();
5460 }
5461
5462 static void test_cloneNode(void )
5463 {
5464     IXMLDOMDocument *doc, *doc2;
5465     VARIANT_BOOL b;
5466     IXMLDOMNodeList *pList;
5467     IXMLDOMNamedNodeMap *mapAttr;
5468     LONG length, length1;
5469     LONG attr_cnt, attr_cnt1;
5470     IXMLDOMNode *node;
5471     IXMLDOMNode *node_clone;
5472     IXMLDOMNode *node_first;
5473     HRESULT hr;
5474     BSTR str;
5475
5476     doc = create_document(&IID_IXMLDOMDocument);
5477     if (!doc) return;
5478
5479     str = SysAllocString( szComplete4 );
5480     ole_check(IXMLDOMDocument_loadXML(doc, str, &b));
5481     ok(b == VARIANT_TRUE, "failed to load XML string\n");
5482     SysFreeString(str);
5483
5484     hr = IXMLDOMNode_selectSingleNode(doc, _bstr_("lc/pr"), &node);
5485     ok( hr == S_OK, "ret %08x\n", hr );
5486     ok( node != NULL, "node %p\n", node );
5487
5488     /* Check invalid parameter */
5489     hr = IXMLDOMNode_cloneNode(node, VARIANT_TRUE, NULL);
5490     ok( hr == E_INVALIDARG, "ret %08x\n", hr );
5491
5492     /* All Children */
5493     hr = IXMLDOMNode_cloneNode(node, VARIANT_TRUE, &node_clone);
5494     ok( hr == S_OK, "ret %08x\n", hr );
5495     ok( node_clone != NULL, "node %p\n", node );
5496
5497     hr = IXMLDOMNode_get_firstChild(node_clone, &node_first);
5498     ok( hr == S_OK, "ret %08x\n", hr );
5499     hr = IXMLDOMNode_get_ownerDocument(node_clone, &doc2);
5500     ok( hr == S_OK, "ret %08x\n", hr );
5501     IXMLDOMDocument_Release(doc2);
5502     IXMLDOMNode_Release(node_first);
5503
5504     hr = IXMLDOMNode_get_childNodes(node, &pList);
5505     ok( hr == S_OK, "ret %08x\n", hr );
5506     length = 0;
5507     hr = IXMLDOMNodeList_get_length(pList, &length);
5508     ok( hr == S_OK, "ret %08x\n", hr );
5509     ok(length == 1, "got %d\n", length);
5510     IXMLDOMNodeList_Release(pList);
5511
5512     hr = IXMLDOMNode_get_attributes(node, &mapAttr);
5513     ok( hr == S_OK, "ret %08x\n", hr );
5514     attr_cnt = 0;
5515     hr = IXMLDOMNamedNodeMap_get_length(mapAttr, &attr_cnt);
5516     ok( hr == S_OK, "ret %08x\n", hr );
5517     ok(attr_cnt == 3, "got %d\n", attr_cnt);
5518     IXMLDOMNamedNodeMap_Release(mapAttr);
5519
5520     hr = IXMLDOMNode_get_childNodes(node_clone, &pList);
5521     ok( hr == S_OK, "ret %08x\n", hr );
5522     length1 = 0;
5523     hr = IXMLDOMNodeList_get_length(pList, &length1);
5524     ok(length1 == 1, "got %d\n", length1);
5525     ok( hr == S_OK, "ret %08x\n", hr );
5526     IXMLDOMNodeList_Release(pList);
5527
5528     hr = IXMLDOMNode_get_attributes(node_clone, &mapAttr);
5529     ok( hr == S_OK, "ret %08x\n", hr );
5530     attr_cnt1 = 0;
5531     hr = IXMLDOMNamedNodeMap_get_length(mapAttr, &attr_cnt1);
5532     ok( hr == S_OK, "ret %08x\n", hr );
5533     ok(attr_cnt1 == 3, "got %d\n", attr_cnt1);
5534     IXMLDOMNamedNodeMap_Release(mapAttr);
5535
5536     ok(length == length1, "wrong Child count (%d, %d)\n", length, length1);
5537     ok(attr_cnt == attr_cnt1, "wrong Attribute count (%d, %d)\n", attr_cnt, attr_cnt1);
5538     IXMLDOMNode_Release(node_clone);
5539
5540     /* No Children */
5541     hr = IXMLDOMNode_cloneNode(node, VARIANT_FALSE, &node_clone);
5542     ok( hr == S_OK, "ret %08x\n", hr );
5543     ok( node_clone != NULL, "node %p\n", node );
5544
5545     hr = IXMLDOMNode_get_firstChild(node_clone, &node_first);
5546     ok(hr == S_FALSE, "ret %08x\n", hr );
5547
5548     hr = IXMLDOMNode_get_childNodes(node_clone, &pList);
5549     ok(hr == S_OK, "ret %08x\n", hr );
5550     hr = IXMLDOMNodeList_get_length(pList, &length1);
5551     ok(hr == S_OK, "ret %08x\n", hr );
5552     ok( length1 == 0, "Length should be 0 (%d)\n", length1);
5553     IXMLDOMNodeList_Release(pList);
5554
5555     hr = IXMLDOMNode_get_attributes(node_clone, &mapAttr);
5556     ok(hr == S_OK, "ret %08x\n", hr );
5557     hr = IXMLDOMNamedNodeMap_get_length(mapAttr, &attr_cnt1);
5558     ok(hr == S_OK, "ret %08x\n", hr );
5559     ok(attr_cnt1 == 3, "Attribute count should be 3 (%d)\n", attr_cnt1);
5560     IXMLDOMNamedNodeMap_Release(mapAttr);
5561
5562     ok(length != length1, "wrong Child count (%d, %d)\n", length, length1);
5563     ok(attr_cnt == attr_cnt1, "wrong Attribute count (%d, %d)\n", attr_cnt, attr_cnt1);
5564     IXMLDOMNode_Release(node_clone);
5565
5566     IXMLDOMNode_Release(node);
5567     IXMLDOMDocument_Release(doc);
5568     free_bstrs();
5569 }
5570
5571 static void test_xmlTypes(void)
5572 {
5573     IXMLDOMDocument *doc;
5574     IXMLDOMElement *pRoot;
5575     HRESULT hr;
5576     IXMLDOMComment *pComment;
5577     IXMLDOMElement *pElement;
5578     IXMLDOMAttribute *pAttribute;
5579     IXMLDOMNamedNodeMap *pAttribs;
5580     IXMLDOMCDATASection *pCDataSec;
5581     IXMLDOMImplementation *pIXMLDOMImplementation = NULL;
5582     IXMLDOMDocumentFragment *pDocFrag = NULL;
5583     IXMLDOMEntityReference *pEntityRef = NULL;
5584     BSTR str;
5585     IXMLDOMNode *pNextChild;
5586     VARIANT v;
5587     LONG len = 0;
5588
5589     doc = create_document(&IID_IXMLDOMDocument);
5590     if (!doc) return;
5591
5592     pNextChild = (void*)0xdeadbeef;
5593     hr = IXMLDOMDocument_get_nextSibling(doc, NULL);
5594     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5595
5596     pNextChild = (void*)0xdeadbeef;
5597     hr = IXMLDOMDocument_get_nextSibling(doc, &pNextChild);
5598     ok(hr == S_FALSE, "ret %08x\n", hr );
5599     ok(pNextChild == NULL, "pDocChild not NULL\n");
5600
5601     /* test previous Sibling */
5602     hr = IXMLDOMDocument_get_previousSibling(doc, NULL);
5603     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5604
5605     pNextChild = (void*)0xdeadbeef;
5606     hr = IXMLDOMDocument_get_previousSibling(doc, &pNextChild);
5607     ok(hr == S_FALSE, "ret %08x\n", hr );
5608     ok(pNextChild == NULL, "pNextChild not NULL\n");
5609
5610     /* test get_dataType */
5611     V_VT(&v) = VT_EMPTY;
5612     hr = IXMLDOMDocument_get_dataType(doc, &v);
5613     ok(hr == S_FALSE, "ret %08x\n", hr );
5614     ok( V_VT(&v) == VT_NULL, "incorrect dataType type\n");
5615     VariantClear(&v);
5616
5617     /* test implementation */
5618     hr = IXMLDOMDocument_get_implementation(doc, NULL);
5619     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5620
5621     hr = IXMLDOMDocument_get_implementation(doc, &pIXMLDOMImplementation);
5622     ok(hr == S_OK, "ret %08x\n", hr );
5623     if(hr == S_OK)
5624     {
5625         VARIANT_BOOL hasFeature = VARIANT_TRUE;
5626         BSTR sEmpty = SysAllocStringLen(NULL, 0);
5627
5628         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, NULL, sEmpty, &hasFeature);
5629         ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5630
5631         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, sEmpty, sEmpty, NULL);
5632         ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5633
5634         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, _bstr_("DOM"), sEmpty, &hasFeature);
5635         ok(hr == S_OK, "ret %08x\n", hr );
5636         ok(hasFeature == VARIANT_FALSE, "hasFeature returned false\n");
5637
5638         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, sEmpty, sEmpty, &hasFeature);
5639         ok(hr == S_OK, "ret %08x\n", hr );
5640         ok(hasFeature == VARIANT_FALSE, "hasFeature returned true\n");
5641
5642         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, _bstr_("DOM"), NULL, &hasFeature);
5643         ok(hr == S_OK, "ret %08x\n", hr );
5644         ok(hasFeature == VARIANT_TRUE, "hasFeature returned false\n");
5645
5646         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, _bstr_("DOM"), sEmpty, &hasFeature);
5647         ok(hr == S_OK, "ret %08x\n", hr );
5648         ok(hasFeature == VARIANT_FALSE, "hasFeature returned false\n");
5649
5650         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, _bstr_("DOM"), _bstr_("1.0"), &hasFeature);
5651         ok(hr == S_OK, "ret %08x\n", hr );
5652         ok(hasFeature == VARIANT_TRUE, "hasFeature returned true\n");
5653
5654         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, _bstr_("XML"), _bstr_("1.0"), &hasFeature);
5655         ok(hr == S_OK, "ret %08x\n", hr );
5656         ok(hasFeature == VARIANT_TRUE, "hasFeature returned true\n");
5657
5658         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, _bstr_("MS-DOM"), _bstr_("1.0"), &hasFeature);
5659         ok(hr == S_OK, "ret %08x\n", hr );
5660         ok(hasFeature == VARIANT_TRUE, "hasFeature returned true\n");
5661
5662         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, _bstr_("SSS"), NULL, &hasFeature);
5663         ok(hr == S_OK, "ret %08x\n", hr );
5664         ok(hasFeature == VARIANT_FALSE, "hasFeature returned false\n");
5665
5666         SysFreeString(sEmpty);
5667         IXMLDOMImplementation_Release(pIXMLDOMImplementation);
5668     }
5669
5670     pRoot = (IXMLDOMElement*)0x1;
5671     hr = IXMLDOMDocument_createElement(doc, NULL, &pRoot);
5672     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5673     ok(pRoot == (void*)0x1, "Expect same ptr, got %p\n", pRoot);
5674
5675     pRoot = (IXMLDOMElement*)0x1;
5676     hr = IXMLDOMDocument_createElement(doc, _bstr_(""), &pRoot);
5677     ok(hr == E_FAIL, "ret %08x\n", hr );
5678     ok(pRoot == (void*)0x1, "Expect same ptr, got %p\n", pRoot);
5679
5680     hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing"), &pRoot);
5681     ok(hr == S_OK, "ret %08x\n", hr );
5682     if(hr == S_OK)
5683     {
5684         hr = IXMLDOMDocument_appendChild(doc, (IXMLDOMNode*)pRoot, NULL);
5685         ok(hr == S_OK, "ret %08x\n", hr );
5686         if(hr == S_OK)
5687         {
5688             /* Comment */
5689             str = SysAllocString(szComment);
5690             hr = IXMLDOMDocument_createComment(doc, str, &pComment);
5691             SysFreeString(str);
5692             ok(hr == S_OK, "ret %08x\n", hr );
5693             if(hr == S_OK)
5694             {
5695                 hr = IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pComment, NULL);
5696                 ok(hr == S_OK, "ret %08x\n", hr );
5697
5698                 hr = IXMLDOMComment_get_nodeName(pComment, &str);
5699                 ok(hr == S_OK, "ret %08x\n", hr );
5700                 ok( !lstrcmpW( str, szCommentNodeText ), "incorrect comment node Name\n");
5701                 SysFreeString(str);
5702
5703                 hr = IXMLDOMComment_get_xml(pComment, &str);
5704                 ok(hr == S_OK, "ret %08x\n", hr );
5705                 ok( !lstrcmpW( str, szCommentXML ), "incorrect comment xml\n");
5706                 SysFreeString(str);
5707
5708                 /* put data Tests */
5709                 hr = IXMLDOMComment_put_data(pComment, _bstr_("This &is a ; test <>\\"));
5710                 ok(hr == S_OK, "ret %08x\n", hr );
5711
5712                 /* get data Tests */
5713                 hr = IXMLDOMComment_get_data(pComment, &str);
5714                 ok(hr == S_OK, "ret %08x\n", hr );
5715                 ok( !lstrcmpW( str, _bstr_("This &is a ; test <>\\") ), "incorrect get_data string\n");
5716                 SysFreeString(str);
5717
5718                 /* get data Tests */
5719                 hr = IXMLDOMComment_get_nodeValue(pComment, &v);
5720                 ok(hr == S_OK, "ret %08x\n", hr );
5721                 ok( V_VT(&v) == VT_BSTR, "incorrect dataType type\n");
5722                 ok( !lstrcmpW( V_BSTR(&v), _bstr_("This &is a ; test <>\\") ), "incorrect get_nodeValue string\n");
5723                 VariantClear(&v);
5724
5725                 /* Confirm XML text is good */
5726                 hr = IXMLDOMComment_get_xml(pComment, &str);
5727                 ok(hr == S_OK, "ret %08x\n", hr );
5728                 ok( !lstrcmpW( str, _bstr_("<!--This &is a ; test <>\\-->") ), "incorrect xml string\n");
5729                 SysFreeString(str);
5730
5731                 /* Confirm we get the put_data Text back */
5732                 hr = IXMLDOMComment_get_text(pComment, &str);
5733                 ok(hr == S_OK, "ret %08x\n", hr );
5734                 ok( !lstrcmpW( str, _bstr_("This &is a ; test <>\\") ), "incorrect xml string\n");
5735                 SysFreeString(str);
5736
5737                 /* test length property */
5738                 hr = IXMLDOMComment_get_length(pComment, &len);
5739                 ok(hr == S_OK, "ret %08x\n", hr );
5740                 ok(len == 21, "expected 21 got %d\n", len);
5741
5742                 /* test substringData */
5743                 hr = IXMLDOMComment_substringData(pComment, 0, 4, NULL);
5744                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5745
5746                 /* test substringData - Invalid offset */
5747                 str = (BSTR)&szElement;
5748                 hr = IXMLDOMComment_substringData(pComment, -1, 4, &str);
5749                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5750                 ok( str == NULL, "incorrect string\n");
5751
5752                 /* test substringData - Invalid offset */
5753                 str = (BSTR)&szElement;
5754                 hr = IXMLDOMComment_substringData(pComment, 30, 0, &str);
5755                 ok(hr == S_FALSE, "ret %08x\n", hr );
5756                 ok( str == NULL, "incorrect string\n");
5757
5758                 /* test substringData - Invalid size */
5759                 str = (BSTR)&szElement;
5760                 hr = IXMLDOMComment_substringData(pComment, 0, -1, &str);
5761                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5762                 ok( str == NULL, "incorrect string\n");
5763
5764                 /* test substringData - Invalid size */
5765                 str = (BSTR)&szElement;
5766                 hr = IXMLDOMComment_substringData(pComment, 2, 0, &str);
5767                 ok(hr == S_FALSE, "ret %08x\n", hr );
5768                 ok( str == NULL, "incorrect string\n");
5769
5770                 /* test substringData - Start of string */
5771                 hr = IXMLDOMComment_substringData(pComment, 0, 4, &str);
5772                 ok(hr == S_OK, "ret %08x\n", hr );
5773                 ok( !lstrcmpW( str, _bstr_("This") ), "incorrect substringData string\n");
5774                 SysFreeString(str);
5775
5776                 /* test substringData - Middle of string */
5777                 hr = IXMLDOMComment_substringData(pComment, 13, 4, &str);
5778                 ok(hr == S_OK, "ret %08x\n", hr );
5779                 ok( !lstrcmpW( str, _bstr_("test") ), "incorrect substringData string\n");
5780                 SysFreeString(str);
5781
5782                 /* test substringData - End of string */
5783                 hr = IXMLDOMComment_substringData(pComment, 20, 4, &str);
5784                 ok(hr == S_OK, "ret %08x\n", hr );
5785                 ok( !lstrcmpW( str, _bstr_("\\") ), "incorrect substringData string\n");
5786                 SysFreeString(str);
5787
5788                 /* test appendData */
5789                 hr = IXMLDOMComment_appendData(pComment, NULL);
5790                 ok(hr == S_OK, "ret %08x\n", hr );
5791
5792                 hr = IXMLDOMComment_appendData(pComment, _bstr_(""));
5793                 ok(hr == S_OK, "ret %08x\n", hr );
5794
5795                 hr = IXMLDOMComment_appendData(pComment, _bstr_("Append"));
5796                 ok(hr == S_OK, "ret %08x\n", hr );
5797
5798                 hr = IXMLDOMComment_get_text(pComment, &str);
5799                 ok(hr == S_OK, "ret %08x\n", hr );
5800                 ok( !lstrcmpW( str, _bstr_("This &is a ; test <>\\Append") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
5801                 SysFreeString(str);
5802
5803                 /* test insertData */
5804                 str = SysAllocStringLen(NULL, 0);
5805                 hr = IXMLDOMComment_insertData(pComment, -1, str);
5806                 ok(hr == S_OK, "ret %08x\n", hr );
5807
5808                 hr = IXMLDOMComment_insertData(pComment, -1, NULL);
5809                 ok(hr == S_OK, "ret %08x\n", hr );
5810
5811                 hr = IXMLDOMComment_insertData(pComment, 1000, str);
5812                 ok(hr == S_OK, "ret %08x\n", hr );
5813
5814                 hr = IXMLDOMComment_insertData(pComment, 1000, NULL);
5815                 ok(hr == S_OK, "ret %08x\n", hr );
5816
5817                 hr = IXMLDOMComment_insertData(pComment, 0, NULL);
5818                 ok(hr == S_OK, "ret %08x\n", hr );
5819
5820                 hr = IXMLDOMComment_insertData(pComment, 0, str);
5821                 ok(hr == S_OK, "ret %08x\n", hr );
5822                 SysFreeString(str);
5823
5824                 hr = IXMLDOMComment_insertData(pComment, -1, _bstr_("Inserting"));
5825                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5826
5827                 hr = IXMLDOMComment_insertData(pComment, 1000, _bstr_("Inserting"));
5828                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5829
5830                 hr = IXMLDOMComment_insertData(pComment, 0, _bstr_("Begin "));
5831                 ok(hr == S_OK, "ret %08x\n", hr );
5832
5833                 hr = IXMLDOMComment_insertData(pComment, 17, _bstr_("Middle"));
5834                 ok(hr == S_OK, "ret %08x\n", hr );
5835
5836                 hr = IXMLDOMComment_insertData(pComment, 39, _bstr_(" End"));
5837                 ok(hr == S_OK, "ret %08x\n", hr );
5838
5839                 hr = IXMLDOMComment_get_text(pComment, &str);
5840                 ok(hr == S_OK, "ret %08x\n", hr );
5841                 ok( !lstrcmpW( str, _bstr_("Begin This &is a Middle; test <>\\Append End") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
5842                 SysFreeString(str);
5843
5844                 /* delete data */
5845                 /* invalid arguments */
5846                 hr = IXMLDOMComment_deleteData(pComment, -1, 1);
5847                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5848
5849                 hr = IXMLDOMComment_deleteData(pComment, 0, 0);
5850                 ok(hr == S_OK, "ret %08x\n", hr );
5851
5852                 hr = IXMLDOMComment_deleteData(pComment, 0, -1);
5853                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5854
5855                 hr = IXMLDOMComment_get_length(pComment, &len);
5856                 ok(hr == S_OK, "ret %08x\n", hr );
5857                 ok(len == 43, "expected 43 got %d\n", len);
5858
5859                 hr = IXMLDOMComment_deleteData(pComment, len, 1);
5860                 ok(hr == S_OK, "ret %08x\n", hr );
5861
5862                 hr = IXMLDOMComment_deleteData(pComment, len+1, 1);
5863                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5864
5865                 /* delete from start */
5866                 hr = IXMLDOMComment_deleteData(pComment, 0, 5);
5867                 ok(hr == S_OK, "ret %08x\n", hr );
5868
5869                 hr = IXMLDOMComment_get_length(pComment, &len);
5870                 ok(hr == S_OK, "ret %08x\n", hr );
5871                 ok(len == 38, "expected 38 got %d\n", len);
5872
5873                 hr = IXMLDOMComment_get_text(pComment, &str);
5874                 ok(hr == S_OK, "ret %08x\n", hr );
5875                 ok( !lstrcmpW( str, _bstr_(" This &is a Middle; test <>\\Append End") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
5876                 SysFreeString(str);
5877
5878                 /* delete from end */
5879                 hr = IXMLDOMComment_deleteData(pComment, 35, 3);
5880                 ok(hr == S_OK, "ret %08x\n", hr );
5881
5882                 hr = IXMLDOMComment_get_length(pComment, &len);
5883                 ok(hr == S_OK, "ret %08x\n", hr );
5884                 ok(len == 35, "expected 35 got %d\n", len);
5885
5886                 hr = IXMLDOMComment_get_text(pComment, &str);
5887                 ok(hr == S_OK, "ret %08x\n", hr );
5888                 ok( !lstrcmpW( str, _bstr_(" This &is a Middle; test <>\\Append ") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
5889                 SysFreeString(str);
5890
5891                 /* delete from inside */
5892                 hr = IXMLDOMComment_deleteData(pComment, 1, 33);
5893                 ok(hr == S_OK, "ret %08x\n", hr );
5894
5895                 hr = IXMLDOMComment_get_length(pComment, &len);
5896                 ok(hr == S_OK, "ret %08x\n", hr );
5897                 ok(len == 2, "expected 2 got %d\n", len);
5898
5899                 hr = IXMLDOMComment_get_text(pComment, &str);
5900                 ok(hr == S_OK, "ret %08x\n", hr );
5901                 ok( !lstrcmpW( str, _bstr_("  ") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
5902                 SysFreeString(str);
5903
5904                 /* delete whole data ... */
5905                 hr = IXMLDOMComment_get_length(pComment, &len);
5906                 ok(hr == S_OK, "ret %08x\n", hr );
5907
5908                 hr = IXMLDOMComment_deleteData(pComment, 0, len);
5909                 ok(hr == S_OK, "ret %08x\n", hr );
5910                 /* ... and try again with empty string */
5911                 hr = IXMLDOMComment_deleteData(pComment, 0, len);
5912                 ok(hr == S_OK, "ret %08x\n", hr );
5913
5914                 /* ::replaceData() */
5915                 V_VT(&v) = VT_BSTR;
5916                 V_BSTR(&v) = SysAllocString(szstr1);
5917                 hr = IXMLDOMComment_put_nodeValue(pComment, v);
5918                 ok(hr == S_OK, "ret %08x\n", hr );
5919                 VariantClear(&v);
5920
5921                 hr = IXMLDOMComment_replaceData(pComment, 6, 0, NULL);
5922                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5923                 hr = IXMLDOMComment_get_text(pComment, &str);
5924                 ok(hr == S_OK, "ret %08x\n", hr );
5925                 ok( !lstrcmpW( str, _bstr_("str1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
5926                 SysFreeString(str);
5927
5928                 hr = IXMLDOMComment_replaceData(pComment, 0, 0, NULL);
5929                 ok(hr == S_OK, "ret %08x\n", hr );
5930                 hr = IXMLDOMComment_get_text(pComment, &str);
5931                 ok(hr == S_OK, "ret %08x\n", hr );
5932                 ok( !lstrcmpW( str, _bstr_("str1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
5933                 SysFreeString(str);
5934
5935                 /* NULL pointer means delete */
5936                 hr = IXMLDOMComment_replaceData(pComment, 0, 1, NULL);
5937                 ok(hr == S_OK, "ret %08x\n", hr );
5938                 hr = IXMLDOMComment_get_text(pComment, &str);
5939                 ok(hr == S_OK, "ret %08x\n", hr );
5940                 ok( !lstrcmpW( str, _bstr_("tr1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
5941                 SysFreeString(str);
5942
5943                 /* empty string means delete */
5944                 hr = IXMLDOMComment_replaceData(pComment, 0, 1, _bstr_(""));
5945                 ok(hr == S_OK, "ret %08x\n", hr );
5946                 hr = IXMLDOMComment_get_text(pComment, &str);
5947                 ok(hr == S_OK, "ret %08x\n", hr );
5948                 ok( !lstrcmpW( str, _bstr_("r1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
5949                 SysFreeString(str);
5950
5951                 /* zero count means insert */
5952                 hr = IXMLDOMComment_replaceData(pComment, 0, 0, _bstr_("a"));
5953                 ok(hr == S_OK, "ret %08x\n", hr );
5954                 hr = IXMLDOMComment_get_text(pComment, &str);
5955                 ok(hr == S_OK, "ret %08x\n", hr );
5956                 ok( !lstrcmpW( str, _bstr_("ar1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
5957                 SysFreeString(str);
5958
5959                 hr = IXMLDOMComment_replaceData(pComment, 0, 2, NULL);
5960                 ok(hr == S_OK, "ret %08x\n", hr );
5961
5962                 hr = IXMLDOMComment_insertData(pComment, 0, _bstr_("m"));
5963                 ok(hr == S_OK, "ret %08x\n", hr );
5964                 hr = IXMLDOMComment_get_text(pComment, &str);
5965                 ok(hr == S_OK, "ret %08x\n", hr );
5966                 ok( !lstrcmpW( str, _bstr_("m1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
5967                 SysFreeString(str);
5968
5969                 /* nonempty string, count greater than its length */
5970                 hr = IXMLDOMComment_replaceData(pComment, 0, 2, _bstr_("a1.2"));
5971                 ok(hr == S_OK, "ret %08x\n", hr );
5972                 hr = IXMLDOMComment_get_text(pComment, &str);
5973                 ok(hr == S_OK, "ret %08x\n", hr );
5974                 ok( !lstrcmpW( str, _bstr_("a1.2") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
5975                 SysFreeString(str);
5976
5977                 /* nonempty string, count less than its length */
5978                 hr = IXMLDOMComment_replaceData(pComment, 0, 1, _bstr_("wine"));
5979                 ok(hr == S_OK, "ret %08x\n", hr );
5980                 hr = IXMLDOMComment_get_text(pComment, &str);
5981                 ok(hr == S_OK, "ret %08x\n", hr );
5982                 ok( !lstrcmpW( str, _bstr_("wine1.2") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
5983                 SysFreeString(str);
5984
5985                 IXMLDOMComment_Release(pComment);
5986             }
5987
5988             /* Element */
5989             str = SysAllocString(szElement);
5990             hr = IXMLDOMDocument_createElement(doc, str, &pElement);
5991             SysFreeString(str);
5992             ok(hr == S_OK, "ret %08x\n", hr );
5993             if(hr == S_OK)
5994             {
5995                 hr = IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
5996                 ok(hr == S_OK, "ret %08x\n", hr );
5997
5998                 hr = IXMLDOMElement_get_nodeName(pElement, &str);
5999                 ok(hr == S_OK, "ret %08x\n", hr );
6000                 ok( !lstrcmpW( str, szElement ), "incorrect element node Name\n");
6001                 SysFreeString(str);
6002
6003                 hr = IXMLDOMElement_get_xml(pElement, &str);
6004                 ok(hr == S_OK, "ret %08x\n", hr );
6005                 ok( !lstrcmpW( str, szElementXML ), "incorrect element xml\n");
6006                 SysFreeString(str);
6007
6008                 /* Attribute */
6009                 pAttribute = (IXMLDOMAttribute*)0x1;
6010                 hr = IXMLDOMDocument_createAttribute(doc, NULL, &pAttribute);
6011                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
6012                 ok(pAttribute == (void*)0x1, "Expect same ptr, got %p\n", pAttribute);
6013
6014                 pAttribute = (IXMLDOMAttribute*)0x1;
6015                 hr = IXMLDOMDocument_createAttribute(doc, _bstr_(""), &pAttribute);
6016                 ok(hr == E_FAIL, "ret %08x\n", hr );
6017                 ok(pAttribute == (void*)0x1, "Expect same ptr, got %p\n", pAttribute);
6018
6019                 str = SysAllocString(szAttribute);
6020                 hr = IXMLDOMDocument_createAttribute(doc, str, &pAttribute);
6021                 SysFreeString(str);
6022                 ok(hr == S_OK, "ret %08x\n", hr );
6023                 if(hr == S_OK)
6024                 {
6025                     IXMLDOMNode *pNewChild = (IXMLDOMNode *)0x1;
6026
6027                     hr = IXMLDOMAttribute_get_nextSibling(pAttribute, NULL);
6028                     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
6029
6030                     pNextChild = (IXMLDOMNode *)0x1;
6031                     hr = IXMLDOMAttribute_get_nextSibling(pAttribute, &pNextChild);
6032                     ok(hr == S_FALSE, "ret %08x\n", hr );
6033                     ok(pNextChild == NULL, "pNextChild not NULL\n");
6034
6035                     /* test Previous Sibling*/
6036                     hr = IXMLDOMAttribute_get_previousSibling(pAttribute, NULL);
6037                     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
6038
6039                     pNextChild = (IXMLDOMNode *)0x1;
6040                     hr = IXMLDOMAttribute_get_previousSibling(pAttribute, &pNextChild);
6041                     ok(hr == S_FALSE, "ret %08x\n", hr );
6042                     ok(pNextChild == NULL, "pNextChild not NULL\n");
6043
6044                     hr = IXMLDOMElement_appendChild(pElement, (IXMLDOMNode*)pAttribute, &pNewChild);
6045                     ok(hr == E_FAIL, "ret %08x\n", hr );
6046                     ok(pNewChild == NULL, "pNewChild not NULL\n");
6047
6048                     hr = IXMLDOMElement_get_attributes(pElement, &pAttribs);
6049                     ok(hr == S_OK, "ret %08x\n", hr );
6050                     if ( hr == S_OK )
6051                     {
6052                         hr = IXMLDOMNamedNodeMap_setNamedItem(pAttribs, (IXMLDOMNode*)pAttribute, NULL );
6053                         ok(hr == S_OK, "ret %08x\n", hr );
6054
6055                         IXMLDOMNamedNodeMap_Release(pAttribs);
6056                     }
6057
6058                     hr = IXMLDOMAttribute_get_nodeName(pAttribute, &str);
6059                     ok(hr == S_OK, "ret %08x\n", hr );
6060                     ok( !lstrcmpW( str, szAttribute ), "incorrect attribute node Name\n");
6061                     SysFreeString(str);
6062
6063                     /* test nodeName */
6064                     hr = IXMLDOMAttribute_get_nodeName(pAttribute, &str);
6065                     ok(hr == S_OK, "ret %08x\n", hr );
6066                     ok( !lstrcmpW( str, szAttribute ), "incorrect nodeName string\n");
6067                     SysFreeString(str);
6068
6069                     /* test name property */
6070                     hr = IXMLDOMAttribute_get_name(pAttribute, &str);
6071                     ok(hr == S_OK, "ret %08x\n", hr );
6072                     ok( !lstrcmpW( str, szAttribute ), "incorrect name string\n");
6073                     SysFreeString(str);
6074
6075                     hr = IXMLDOMAttribute_get_xml(pAttribute, &str);
6076                     ok(hr == S_OK, "ret %08x\n", hr );
6077                     ok( !lstrcmpW( str, szAttributeXML ), "incorrect attribute xml\n");
6078                     SysFreeString(str);
6079
6080                     IXMLDOMAttribute_Release(pAttribute);
6081
6082                     /* Check Element again with the Add Attribute*/
6083                     hr = IXMLDOMElement_get_xml(pElement, &str);
6084                     ok(hr == S_OK, "ret %08x\n", hr );
6085                     ok( !lstrcmpW( str, szElementXML2 ), "incorrect element xml\n");
6086                     SysFreeString(str);
6087                 }
6088
6089                 hr = IXMLDOMElement_put_text(pElement, _bstr_("TestingNode"));
6090                 ok(hr == S_OK, "ret %08x\n", hr );
6091
6092                 hr = IXMLDOMElement_get_xml(pElement, &str);
6093                 ok(hr == S_OK, "ret %08x\n", hr );
6094                 ok( !lstrcmpW( str, szElementXML3 ), "incorrect element xml\n");
6095                 SysFreeString(str);
6096
6097                 /* Test for reversible escaping */
6098                 str = SysAllocString( szStrangeChars );
6099                 hr = IXMLDOMElement_put_text(pElement, str);
6100                 ok(hr == S_OK, "ret %08x\n", hr );
6101                 SysFreeString( str );
6102
6103                 hr = IXMLDOMElement_get_xml(pElement, &str);
6104                 ok(hr == S_OK, "ret %08x\n", hr );
6105                 ok( !lstrcmpW( str, szElementXML4 ), "incorrect element xml\n");
6106                 SysFreeString(str);
6107
6108                 hr = IXMLDOMElement_get_text(pElement, &str);
6109                 ok(hr == S_OK, "ret %08x\n", hr );
6110                 ok( !lstrcmpW( str, szStrangeChars ), "incorrect element text\n");
6111                 SysFreeString(str);
6112
6113                 IXMLDOMElement_Release(pElement);
6114             }
6115
6116             /* CData Section */
6117             str = SysAllocString(szCData);
6118             hr = IXMLDOMDocument_createCDATASection(doc, str, NULL);
6119             ok(hr == E_INVALIDARG, "ret %08x\n", hr );
6120
6121             hr = IXMLDOMDocument_createCDATASection(doc, str, &pCDataSec);
6122             SysFreeString(str);
6123             ok(hr == S_OK, "ret %08x\n", hr );
6124             if(hr == S_OK)
6125             {
6126                 IXMLDOMNode *pNextChild = (IXMLDOMNode *)0x1;
6127                 VARIANT var;
6128
6129                 VariantInit(&var);
6130
6131                 hr = IXMLDOMCDATASection_QueryInterface(pCDataSec, &IID_IXMLDOMElement, (void**)&pElement);
6132                 ok(hr == E_NOINTERFACE, "ret %08x\n", hr);
6133
6134                 hr = IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pCDataSec, NULL);
6135                 ok(hr == S_OK, "ret %08x\n", hr );
6136
6137                 hr = IXMLDOMCDATASection_get_nodeName(pCDataSec, &str);
6138                 ok(hr == S_OK, "ret %08x\n", hr );
6139                 ok( !lstrcmpW( str, szCDataNodeText ), "incorrect cdata node Name\n");
6140                 SysFreeString(str);
6141
6142                 hr = IXMLDOMCDATASection_get_xml(pCDataSec, &str);
6143                 ok(hr == S_OK, "ret %08x\n", hr );
6144                 ok( !lstrcmpW( str, szCDataXML ), "incorrect cdata xml\n");
6145                 SysFreeString(str);
6146
6147                 /* test lastChild */
6148                 pNextChild = (IXMLDOMNode*)0x1;
6149                 hr = IXMLDOMCDATASection_get_lastChild(pCDataSec, &pNextChild);
6150                 ok(hr == S_FALSE, "ret %08x\n", hr );
6151                 ok(pNextChild == NULL, "pNextChild not NULL\n");
6152
6153                 /* put data Tests */
6154                 hr = IXMLDOMCDATASection_put_data(pCDataSec, _bstr_("This &is a ; test <>\\"));
6155                 ok(hr == S_OK, "ret %08x\n", hr );
6156
6157                 /* Confirm XML text is good */
6158                 hr = IXMLDOMCDATASection_get_xml(pCDataSec, &str);
6159                 ok(hr == S_OK, "ret %08x\n", hr );
6160                 ok( !lstrcmpW( str, _bstr_("<![CDATA[This &is a ; test <>\\]]>") ), "incorrect xml string\n");
6161                 SysFreeString(str);
6162
6163                 /* Confirm we get the put_data Text back */
6164                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
6165                 ok(hr == S_OK, "ret %08x\n", hr );
6166                 ok( !lstrcmpW( str, _bstr_("This &is a ; test <>\\") ), "incorrect text string\n");
6167                 SysFreeString(str);
6168
6169                 /* test length property */
6170                 hr = IXMLDOMCDATASection_get_length(pCDataSec, &len);
6171                 ok(hr == S_OK, "ret %08x\n", hr );
6172                 ok(len == 21, "expected 21 got %d\n", len);
6173
6174                 /* test get nodeValue */
6175                 hr = IXMLDOMCDATASection_get_nodeValue(pCDataSec, &var);
6176                 ok(hr == S_OK, "ret %08x\n", hr );
6177                 ok(V_VT(&var) == VT_BSTR, "got vt %04x\n", V_VT(&var));
6178                 ok( !lstrcmpW( V_BSTR(&var), _bstr_("This &is a ; test <>\\") ), "incorrect text string\n");
6179                 VariantClear(&var);
6180
6181                 /* test get data */
6182                 hr = IXMLDOMCDATASection_get_data(pCDataSec, &str);
6183                 ok(hr == S_OK, "ret %08x\n", hr );
6184                 ok( !lstrcmpW( str, _bstr_("This &is a ; test <>\\") ), "incorrect text string\n");
6185                 SysFreeString(str);
6186
6187                 /* test substringData */
6188                 hr = IXMLDOMCDATASection_substringData(pCDataSec, 0, 4, NULL);
6189                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
6190
6191                 /* test substringData - Invalid offset */
6192                 str = (BSTR)&szElement;
6193                 hr = IXMLDOMCDATASection_substringData(pCDataSec, -1, 4, &str);
6194                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
6195                 ok( str == NULL, "incorrect string\n");
6196
6197                 /* test substringData - Invalid offset */
6198                 str = (BSTR)&szElement;
6199                 hr = IXMLDOMCDATASection_substringData(pCDataSec, 30, 0, &str);
6200                 ok(hr == S_FALSE, "ret %08x\n", hr );
6201                 ok( str == NULL, "incorrect string\n");
6202
6203                 /* test substringData - Invalid size */
6204                 str = (BSTR)&szElement;
6205                 hr = IXMLDOMCDATASection_substringData(pCDataSec, 0, -1, &str);
6206                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
6207                 ok( str == NULL, "incorrect string\n");
6208
6209                 /* test substringData - Invalid size */
6210                 str = (BSTR)&szElement;
6211                 hr = IXMLDOMCDATASection_substringData(pCDataSec, 2, 0, &str);
6212                 ok(hr == S_FALSE, "ret %08x\n", hr );
6213                 ok( str == NULL, "incorrect string\n");
6214
6215                 /* test substringData - Start of string */
6216                 hr = IXMLDOMCDATASection_substringData(pCDataSec, 0, 4, &str);
6217                 ok(hr == S_OK, "ret %08x\n", hr );
6218                 ok( !lstrcmpW( str, _bstr_("This") ), "incorrect substringData string\n");
6219                 SysFreeString(str);
6220
6221                 /* test substringData - Middle of string */
6222                 hr = IXMLDOMCDATASection_substringData(pCDataSec, 13, 4, &str);
6223                 ok(hr == S_OK, "ret %08x\n", hr );
6224                 ok( !lstrcmpW( str, _bstr_("test") ), "incorrect substringData string\n");
6225                 SysFreeString(str);
6226
6227                 /* test substringData - End of string */
6228                 hr = IXMLDOMCDATASection_substringData(pCDataSec, 20, 4, &str);
6229                 ok(hr == S_OK, "ret %08x\n", hr );
6230                 ok( !lstrcmpW( str, _bstr_("\\") ), "incorrect substringData string\n");
6231                 SysFreeString(str);
6232
6233                 /* test appendData */
6234                 hr = IXMLDOMCDATASection_appendData(pCDataSec, NULL);
6235                 ok(hr == S_OK, "ret %08x\n", hr );
6236
6237                 hr = IXMLDOMCDATASection_appendData(pCDataSec, _bstr_(""));
6238                 ok(hr == S_OK, "ret %08x\n", hr );
6239
6240                 hr = IXMLDOMCDATASection_appendData(pCDataSec, _bstr_("Append"));
6241                 ok(hr == S_OK, "ret %08x\n", hr );
6242
6243                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
6244                 ok(hr == S_OK, "ret %08x\n", hr );
6245                 ok( !lstrcmpW( str, _bstr_("This &is a ; test <>\\Append") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
6246                 SysFreeString(str);
6247
6248                 /* test insertData */
6249                 str = SysAllocStringLen(NULL, 0);
6250                 hr = IXMLDOMCDATASection_insertData(pCDataSec, -1, str);
6251                 ok(hr == S_OK, "ret %08x\n", hr );
6252
6253                 hr = IXMLDOMCDATASection_insertData(pCDataSec, -1, NULL);
6254                 ok(hr == S_OK, "ret %08x\n", hr );
6255
6256                 hr = IXMLDOMCDATASection_insertData(pCDataSec, 1000, str);
6257                 ok(hr == S_OK, "ret %08x\n", hr );
6258
6259                 hr = IXMLDOMCDATASection_insertData(pCDataSec, 1000, NULL);
6260                 ok(hr == S_OK, "ret %08x\n", hr );
6261
6262                 hr = IXMLDOMCDATASection_insertData(pCDataSec, 0, NULL);
6263                 ok(hr == S_OK, "ret %08x\n", hr );
6264
6265                 hr = IXMLDOMCDATASection_insertData(pCDataSec, 0, str);
6266                 ok(hr == S_OK, "ret %08x\n", hr );
6267                 SysFreeString(str);
6268
6269                 hr = IXMLDOMCDATASection_insertData(pCDataSec, -1, _bstr_("Inserting"));
6270                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
6271
6272                 hr = IXMLDOMCDATASection_insertData(pCDataSec, 1000, _bstr_("Inserting"));
6273                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
6274
6275                 hr = IXMLDOMCDATASection_insertData(pCDataSec, 0, _bstr_("Begin "));
6276                 ok(hr == S_OK, "ret %08x\n", hr );
6277
6278                 hr = IXMLDOMCDATASection_insertData(pCDataSec, 17, _bstr_("Middle"));
6279                 ok(hr == S_OK, "ret %08x\n", hr );
6280
6281                 hr = IXMLDOMCDATASection_insertData(pCDataSec, 39, _bstr_(" End"));
6282                 ok(hr == S_OK, "ret %08x\n", hr );
6283
6284                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
6285                 ok(hr == S_OK, "ret %08x\n", hr );
6286                 ok( !lstrcmpW( str, _bstr_("Begin This &is a Middle; test <>\\Append End") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
6287                 SysFreeString(str);
6288
6289                 /* delete data */
6290                 /* invalid arguments */
6291                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, -1, 1);
6292                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
6293
6294                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, 0, 0);
6295                 ok(hr == S_OK, "ret %08x\n", hr );
6296
6297                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, 0, -1);
6298                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
6299
6300                 hr = IXMLDOMCDATASection_get_length(pCDataSec, &len);
6301                 ok(hr == S_OK, "ret %08x\n", hr );
6302                 ok(len == 43, "expected 43 got %d\n", len);
6303
6304                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, len, 1);
6305                 ok(hr == S_OK, "ret %08x\n", hr );
6306
6307                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, len+1, 1);
6308                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
6309
6310                 /* delete from start */
6311                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, 0, 5);
6312                 ok(hr == S_OK, "ret %08x\n", hr );
6313
6314                 hr = IXMLDOMCDATASection_get_length(pCDataSec, &len);
6315                 ok(hr == S_OK, "ret %08x\n", hr );
6316                 ok(len == 38, "expected 38 got %d\n", len);
6317
6318                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
6319                 ok(hr == S_OK, "ret %08x\n", hr );
6320                 ok( !lstrcmpW( str, _bstr_(" This &is a Middle; test <>\\Append End") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
6321                 SysFreeString(str);
6322
6323                 /* delete from end */
6324                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, 35, 3);
6325                 ok(hr == S_OK, "ret %08x\n", hr );
6326
6327                 hr = IXMLDOMCDATASection_get_length(pCDataSec, &len);
6328                 ok(hr == S_OK, "ret %08x\n", hr );
6329                 ok(len == 35, "expected 35 got %d\n", len);
6330
6331                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
6332                 ok(hr == S_OK, "ret %08x\n", hr );
6333                 ok( !lstrcmpW( str, _bstr_(" This &is a Middle; test <>\\Append ") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
6334                 SysFreeString(str);
6335
6336                 /* delete from inside */
6337                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, 1, 33);
6338                 ok(hr == S_OK, "ret %08x\n", hr );
6339
6340                 hr = IXMLDOMCDATASection_get_length(pCDataSec, &len);
6341                 ok(hr == S_OK, "ret %08x\n", hr );
6342                 ok(len == 2, "expected 2 got %d\n", len);
6343
6344                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
6345                 ok(hr == S_OK, "ret %08x\n", hr );
6346                 ok( !lstrcmpW( str, _bstr_("  ") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
6347                 SysFreeString(str);
6348
6349                 /* delete whole data ... */
6350                 hr = IXMLDOMCDATASection_get_length(pCDataSec, &len);
6351                 ok(hr == S_OK, "ret %08x\n", hr );
6352
6353                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, 0, len);
6354                 ok(hr == S_OK, "ret %08x\n", hr );
6355
6356                 /* ... and try again with empty string */
6357                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, 0, len);
6358                 ok(hr == S_OK, "ret %08x\n", hr );
6359
6360                 /* ::replaceData() */
6361                 V_VT(&v) = VT_BSTR;
6362                 V_BSTR(&v) = SysAllocString(szstr1);
6363                 hr = IXMLDOMCDATASection_put_nodeValue(pCDataSec, v);
6364                 ok(hr == S_OK, "ret %08x\n", hr );
6365                 VariantClear(&v);
6366
6367                 hr = IXMLDOMCDATASection_replaceData(pCDataSec, 6, 0, NULL);
6368                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
6369                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
6370                 ok(hr == S_OK, "ret %08x\n", hr );
6371                 ok( !lstrcmpW( str, _bstr_("str1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
6372                 SysFreeString(str);
6373
6374                 hr = IXMLDOMCDATASection_replaceData(pCDataSec, 0, 0, NULL);
6375                 ok(hr == S_OK, "ret %08x\n", hr );
6376                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
6377                 ok(hr == S_OK, "ret %08x\n", hr );
6378                 ok( !lstrcmpW( str, _bstr_("str1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
6379                 SysFreeString(str);
6380
6381                 /* NULL pointer means delete */
6382                 hr = IXMLDOMCDATASection_replaceData(pCDataSec, 0, 1, NULL);
6383                 ok(hr == S_OK, "ret %08x\n", hr );
6384                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
6385                 ok(hr == S_OK, "ret %08x\n", hr );
6386                 ok( !lstrcmpW( str, _bstr_("tr1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
6387                 SysFreeString(str);
6388
6389                 /* empty string means delete */
6390                 hr = IXMLDOMCDATASection_replaceData(pCDataSec, 0, 1, _bstr_(""));
6391                 ok(hr == S_OK, "ret %08x\n", hr );
6392                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
6393                 ok(hr == S_OK, "ret %08x\n", hr );
6394                 ok( !lstrcmpW( str, _bstr_("r1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
6395                 SysFreeString(str);
6396
6397                 /* zero count means insert */
6398                 hr = IXMLDOMCDATASection_replaceData(pCDataSec, 0, 0, _bstr_("a"));
6399                 ok(hr == S_OK, "ret %08x\n", hr );
6400                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
6401                 ok(hr == S_OK, "ret %08x\n", hr );
6402                 ok( !lstrcmpW( str, _bstr_("ar1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
6403                 SysFreeString(str);
6404
6405                 hr = IXMLDOMCDATASection_replaceData(pCDataSec, 0, 2, NULL);
6406                 ok(hr == S_OK, "ret %08x\n", hr );
6407
6408                 hr = IXMLDOMCDATASection_insertData(pCDataSec, 0, _bstr_("m"));
6409                 ok(hr == S_OK, "ret %08x\n", hr );
6410                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
6411                 ok(hr == S_OK, "ret %08x\n", hr );
6412                 ok( !lstrcmpW( str, _bstr_("m1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
6413                 SysFreeString(str);
6414
6415                 /* nonempty string, count greater than its length */
6416                 hr = IXMLDOMCDATASection_replaceData(pCDataSec, 0, 2, _bstr_("a1.2"));
6417                 ok(hr == S_OK, "ret %08x\n", hr );
6418                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
6419                 ok(hr == S_OK, "ret %08x\n", hr );
6420                 ok( !lstrcmpW( str, _bstr_("a1.2") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
6421                 SysFreeString(str);
6422
6423                 /* nonempty string, count less than its length */
6424                 hr = IXMLDOMCDATASection_replaceData(pCDataSec, 0, 1, _bstr_("wine"));
6425                 ok(hr == S_OK, "ret %08x\n", hr );
6426                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
6427                 ok(hr == S_OK, "ret %08x\n", hr );
6428                 ok( !lstrcmpW( str, _bstr_("wine1.2") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
6429                 SysFreeString(str);
6430
6431                 IXMLDOMCDATASection_Release(pCDataSec);
6432             }
6433
6434             /* Document Fragments */
6435             hr = IXMLDOMDocument_createDocumentFragment(doc, NULL);
6436             ok(hr == E_INVALIDARG, "ret %08x\n", hr );
6437
6438             hr = IXMLDOMDocument_createDocumentFragment(doc, &pDocFrag);
6439             ok(hr == S_OK, "ret %08x\n", hr );
6440             if(hr == S_OK)
6441             {
6442                 IXMLDOMNode *node;
6443
6444                 hr = IXMLDOMDocumentFragment_get_parentNode(pDocFrag, NULL);
6445                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
6446
6447                 node = (IXMLDOMNode *)0x1;
6448                 hr = IXMLDOMDocumentFragment_get_parentNode(pDocFrag, &node);
6449                 ok(hr == S_FALSE, "ret %08x\n", hr );
6450                 ok(node == NULL, "expected NULL, got %p\n", node);
6451
6452                 hr = IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pDocFrag, NULL);
6453                 ok(hr == S_OK, "ret %08x\n", hr );
6454
6455                 hr = IXMLDOMDocumentFragment_get_nodeName(pDocFrag, &str);
6456                 ok(hr == S_OK, "ret %08x\n", hr );
6457                 ok( !lstrcmpW( str, szDocFragmentText ), "incorrect docfragment node Name\n");
6458                 SysFreeString(str);
6459
6460                 /* test next Sibling*/
6461                 hr = IXMLDOMDocumentFragment_get_nextSibling(pDocFrag, NULL);
6462                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
6463
6464                 node = (IXMLDOMNode *)0x1;
6465                 hr = IXMLDOMDocumentFragment_get_nextSibling(pDocFrag, &node);
6466                 ok(hr == S_FALSE, "ret %08x\n", hr );
6467                 ok(node == NULL, "next sibling not NULL\n");
6468
6469                 /* test Previous Sibling*/
6470                 hr = IXMLDOMDocumentFragment_get_previousSibling(pDocFrag, NULL);
6471                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
6472
6473                 node = (IXMLDOMNode *)0x1;
6474                 hr = IXMLDOMDocumentFragment_get_previousSibling(pDocFrag, &node);
6475                 ok(hr == S_FALSE, "ret %08x\n", hr );
6476                 ok(node == NULL, "previous sibling not NULL\n");
6477
6478                 IXMLDOMDocumentFragment_Release(pDocFrag);
6479             }
6480
6481             /* Entity References */
6482             hr = IXMLDOMDocument_createEntityReference(doc, NULL, &pEntityRef);
6483             ok(hr == E_FAIL, "ret %08x\n", hr );
6484             hr = IXMLDOMDocument_createEntityReference(doc, _bstr_(""), &pEntityRef);
6485             ok(hr == E_FAIL, "ret %08x\n", hr );
6486
6487             str = SysAllocString(szEntityRef);
6488             hr = IXMLDOMDocument_createEntityReference(doc, str, NULL);
6489             ok(hr == E_INVALIDARG, "ret %08x\n", hr );
6490
6491             hr = IXMLDOMDocument_createEntityReference(doc, str, &pEntityRef);
6492             SysFreeString(str);
6493             ok(hr == S_OK, "ret %08x\n", hr );
6494             if(hr == S_OK)
6495             {
6496                 hr = IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pEntityRef, NULL);
6497                 ok(hr == S_OK, "ret %08x\n", hr );
6498
6499                 /* test get_xml*/
6500                 hr = IXMLDOMEntityReference_get_xml(pEntityRef, &str);
6501                 ok(hr == S_OK, "ret %08x\n", hr );
6502                 ok( !lstrcmpW( str, szEntityRefXML ), "incorrect xml string\n");
6503                 SysFreeString(str);
6504
6505                 IXMLDOMEntityReference_Release(pEntityRef);
6506             }
6507
6508             IXMLDOMElement_Release( pRoot );
6509         }
6510     }
6511
6512     IXMLDOMDocument_Release(doc);
6513
6514     free_bstrs();
6515 }
6516
6517 static void test_nodeTypeTests( void )
6518 {
6519     IXMLDOMDocument *doc = NULL;
6520     IXMLDOMElement *pRoot;
6521     IXMLDOMElement *pElement;
6522     HRESULT hr;
6523
6524     doc = create_document(&IID_IXMLDOMDocument);
6525     if (!doc) return;
6526
6527     hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing"), NULL);
6528     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
6529
6530     hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing"), &pRoot);
6531     ok(hr == S_OK, "ret %08x\n", hr );
6532     if(hr == S_OK)
6533     {
6534         hr = IXMLDOMDocument_appendChild(doc, (IXMLDOMNode*)pRoot, NULL);
6535         ok(hr == S_OK, "ret %08x\n", hr );
6536         if(hr == S_OK)
6537         {
6538             hr = IXMLDOMElement_put_dataType(pRoot, NULL);
6539             ok(hr == E_INVALIDARG, "ret %08x\n", hr );
6540
6541             /* Invalid Value */
6542             hr = IXMLDOMElement_put_dataType(pRoot, _bstr_("abcdefg") );
6543             ok(hr == E_FAIL, "ret %08x\n", hr );
6544
6545             /* NOTE:
6546              *   The name passed into put_dataType is case-insensitive. So many of the names
6547              *     have been changed to reflect this.
6548              */
6549             /* Boolean */
6550             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_Boolean"), &pElement);
6551             ok(hr == S_OK, "ret %08x\n", hr );
6552             if(hr == S_OK)
6553             {
6554                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
6555
6556                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("Boolean") );
6557                 ok(hr == S_OK, "ret %08x\n", hr );
6558
6559                 IXMLDOMElement_Release(pElement);
6560             }
6561
6562             /* String */
6563             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_String"), &pElement);
6564             ok(hr == S_OK, "ret %08x\n", hr );
6565             if(hr == S_OK)
6566             {
6567                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
6568
6569                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("String") );
6570                 ok(hr == S_OK, "ret %08x\n", hr );
6571
6572                 IXMLDOMElement_Release(pElement);
6573             }
6574
6575             /* Number */
6576             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_Number"), &pElement);
6577             ok(hr == S_OK, "ret %08x\n", hr );
6578             if(hr == S_OK)
6579             {
6580                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
6581
6582                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("number") );
6583                 ok(hr == S_OK, "ret %08x\n", hr );
6584
6585                 IXMLDOMElement_Release(pElement);
6586             }
6587
6588             /* Int */
6589             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_Int"), &pElement);
6590             ok(hr == S_OK, "ret %08x\n", hr );
6591             if(hr == S_OK)
6592             {
6593                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
6594
6595                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("InT") );
6596                 ok(hr == S_OK, "ret %08x\n", hr );
6597
6598                 IXMLDOMElement_Release(pElement);
6599             }
6600
6601             /* Fixed */
6602             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_Fixed"), &pElement);
6603             ok(hr == S_OK, "ret %08x\n", hr );
6604             if(hr == S_OK)
6605             {
6606                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
6607
6608                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("fixed.14.4") );
6609                 ok(hr == S_OK, "ret %08x\n", hr );
6610
6611                 IXMLDOMElement_Release(pElement);
6612             }
6613
6614             /* DateTime */
6615             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_DateTime"), &pElement);
6616             ok(hr == S_OK, "ret %08x\n", hr );
6617             if(hr == S_OK)
6618             {
6619                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
6620
6621                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("DateTime") );
6622                 ok(hr == S_OK, "ret %08x\n", hr );
6623
6624                 IXMLDOMElement_Release(pElement);
6625             }
6626
6627             /* DateTime TZ */
6628             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_DateTime_tz"), &pElement);
6629             ok(hr == S_OK, "ret %08x\n", hr );
6630             if(hr == S_OK)
6631             {
6632                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
6633
6634                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("DateTime.tz") );
6635                 ok(hr == S_OK, "ret %08x\n", hr );
6636
6637                 IXMLDOMElement_Release(pElement);
6638             }
6639
6640             /* Date */
6641             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_Date"), &pElement);
6642             ok(hr == S_OK, "ret %08x\n", hr );
6643             if(hr == S_OK)
6644             {
6645                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
6646
6647                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("Date") );
6648                 ok(hr == S_OK, "ret %08x\n", hr );
6649
6650                 IXMLDOMElement_Release(pElement);
6651             }
6652
6653             /* Time */
6654             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_Time"), &pElement);
6655             ok(hr == S_OK, "ret %08x\n", hr );
6656             if(hr == S_OK)
6657             {
6658                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
6659
6660                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("Time") );
6661                 ok(hr == S_OK, "ret %08x\n", hr );
6662
6663                 IXMLDOMElement_Release(pElement);
6664             }
6665
6666             /* Time.tz */
6667             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_Time_TZ"), &pElement);
6668             ok(hr == S_OK, "ret %08x\n", hr );
6669             if(hr == S_OK)
6670             {
6671                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
6672
6673                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("Time.tz") );
6674                 ok(hr == S_OK, "ret %08x\n", hr );
6675
6676                 IXMLDOMElement_Release(pElement);
6677             }
6678
6679             /* I1 */
6680             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_I1"), &pElement);
6681             ok(hr == S_OK, "ret %08x\n", hr );
6682             if(hr == S_OK)
6683             {
6684                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
6685
6686                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("I1") );
6687                 ok(hr == S_OK, "ret %08x\n", hr );
6688
6689                 IXMLDOMElement_Release(pElement);
6690             }
6691
6692             /* I2 */
6693             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_I2"), &pElement);
6694             ok(hr == S_OK, "ret %08x\n", hr );
6695             if(hr == S_OK)
6696             {
6697                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
6698
6699                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("I2") );
6700                 ok(hr == S_OK, "ret %08x\n", hr );
6701
6702                 IXMLDOMElement_Release(pElement);
6703             }
6704
6705             /* I4 */
6706             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_I4"), &pElement);
6707             ok(hr == S_OK, "ret %08x\n", hr );
6708             if(hr == S_OK)
6709             {
6710                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
6711
6712                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("I4") );
6713                 ok(hr == S_OK, "ret %08x\n", hr );
6714
6715                 IXMLDOMElement_Release(pElement);
6716             }
6717
6718             /* UI1 */
6719             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_UI1"), &pElement);
6720             ok(hr == S_OK, "ret %08x\n", hr );
6721             if(hr == S_OK)
6722             {
6723                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
6724
6725                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("UI1") );
6726                 ok(hr == S_OK, "ret %08x\n", hr );
6727
6728                 IXMLDOMElement_Release(pElement);
6729             }
6730
6731             /* UI2 */
6732             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_UI2"), &pElement);
6733             ok(hr == S_OK, "ret %08x\n", hr );
6734             if(hr == S_OK)
6735             {
6736                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
6737
6738                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("UI2") );
6739                 ok(hr == S_OK, "ret %08x\n", hr );
6740
6741                 IXMLDOMElement_Release(pElement);
6742             }
6743
6744             /* UI4 */
6745             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_UI4"), &pElement);
6746             ok(hr == S_OK, "ret %08x\n", hr );
6747             if(hr == S_OK)
6748             {
6749                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
6750
6751                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("UI4") );
6752                 ok(hr == S_OK, "ret %08x\n", hr );
6753
6754                 IXMLDOMElement_Release(pElement);
6755             }
6756
6757             /* r4 */
6758             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_r4"), &pElement);
6759             ok(hr == S_OK, "ret %08x\n", hr );
6760             if(hr == S_OK)
6761             {
6762                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
6763
6764                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("r4") );
6765                 ok(hr == S_OK, "ret %08x\n", hr );
6766
6767                 IXMLDOMElement_Release(pElement);
6768             }
6769
6770             /* r8 */
6771             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_r8"), &pElement);
6772             ok(hr == S_OK, "ret %08x\n", hr );
6773             if(hr == S_OK)
6774             {
6775                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
6776
6777                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("r8") );
6778                 ok(hr == S_OK, "ret %08x\n", hr );
6779
6780                 IXMLDOMElement_Release(pElement);
6781             }
6782
6783             /* float */
6784             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_float"), &pElement);
6785             ok(hr == S_OK, "ret %08x\n", hr );
6786             if(hr == S_OK)
6787             {
6788                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
6789
6790                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("float") );
6791                 ok(hr == S_OK, "ret %08x\n", hr );
6792
6793                 IXMLDOMElement_Release(pElement);
6794             }
6795
6796             /* uuid */
6797             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_uuid"), &pElement);
6798             ok(hr == S_OK, "ret %08x\n", hr );
6799             if(hr == S_OK)
6800             {
6801                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
6802
6803                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("UuId") );
6804                 ok(hr == S_OK, "ret %08x\n", hr );
6805
6806                 IXMLDOMElement_Release(pElement);
6807             }
6808
6809             /* bin.hex */
6810             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_bin_hex"), &pElement);
6811             ok(hr == S_OK, "ret %08x\n", hr );
6812             if(hr == S_OK)
6813             {
6814                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
6815
6816                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("bin.hex") );
6817                 ok(hr == S_OK, "ret %08x\n", hr );
6818
6819                 IXMLDOMElement_Release(pElement);
6820             }
6821
6822             /* bin.base64 */
6823             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_bin_base64"), &pElement);
6824             ok(hr == S_OK, "ret %08x\n", hr );
6825             if(hr == S_OK)
6826             {
6827                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
6828
6829                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("bin.base64") );
6830                 ok(hr == S_OK, "ret %08x\n", hr );
6831
6832                 IXMLDOMElement_Release(pElement);
6833             }
6834
6835             /* Check changing types */
6836             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_Change"), &pElement);
6837             ok(hr == S_OK, "ret %08x\n", hr );
6838             if(hr == S_OK)
6839             {
6840                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
6841
6842                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("DateTime.tz") );
6843                 ok(hr == S_OK, "ret %08x\n", hr );
6844
6845                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("string") );
6846                 ok(hr == S_OK, "ret %08x\n", hr );
6847
6848                 IXMLDOMElement_Release(pElement);
6849             }
6850
6851             IXMLDOMElement_Release(pRoot);
6852         }
6853     }
6854
6855     IXMLDOMDocument_Release(doc);
6856
6857     free_bstrs();
6858 }
6859
6860 static void test_save(void)
6861 {
6862     IXMLDOMDocument *doc, *doc2;
6863     IXMLDOMElement *root;
6864     VARIANT file, vDoc;
6865     BSTR sOrig, sNew, filename;
6866     char buffer[100];
6867     DWORD read = 0;
6868     HANDLE hfile;
6869     HRESULT hr;
6870
6871     doc = create_document(&IID_IXMLDOMDocument);
6872     if (!doc) return;
6873
6874     doc2 = create_document(&IID_IXMLDOMDocument);
6875     if (!doc2)
6876     {
6877         IXMLDOMDocument_Release(doc);
6878         return;
6879     }
6880
6881     /* save to IXMLDOMDocument */
6882     hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing"), &root);
6883     EXPECT_HR(hr, S_OK);
6884
6885     hr = IXMLDOMDocument_appendChild(doc, (IXMLDOMNode*)root, NULL);
6886     EXPECT_HR(hr, S_OK);
6887
6888     V_VT(&vDoc) = VT_UNKNOWN;
6889     V_UNKNOWN(&vDoc) = (IUnknown*)doc2;
6890
6891     hr = IXMLDOMDocument_save(doc, vDoc);
6892     EXPECT_HR(hr, S_OK);
6893
6894     hr = IXMLDOMDocument_get_xml(doc, &sOrig);
6895     EXPECT_HR(hr, S_OK);
6896
6897     hr = IXMLDOMDocument_get_xml(doc2, &sNew);
6898     EXPECT_HR(hr, S_OK);
6899
6900     ok( !lstrcmpW( sOrig, sNew ), "New document is not the same as original\n");
6901
6902     SysFreeString(sOrig);
6903     SysFreeString(sNew);
6904
6905     IXMLDOMElement_Release(root);
6906     IXMLDOMDocument_Release(doc2);
6907
6908     /* save to path */
6909     V_VT(&file) = VT_BSTR;
6910     V_BSTR(&file) = _bstr_("test.xml");
6911
6912     hr = IXMLDOMDocument_save(doc, file);
6913     EXPECT_HR(hr, S_OK);
6914
6915     hfile = CreateFileA("test.xml", GENERIC_READ, 0, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL );
6916     ok(hfile != INVALID_HANDLE_VALUE, "Could not open file: %u\n", GetLastError());
6917     if(hfile == INVALID_HANDLE_VALUE) return;
6918
6919     ReadFile(hfile, buffer, sizeof(buffer), &read, NULL);
6920     ok(read != 0, "could not read file\n");
6921     ok(buffer[0] != '<' || buffer[1] != '?', "File contains processing instruction\n");
6922
6923     CloseHandle(hfile);
6924     DeleteFile("test.xml");
6925
6926     /* save to path VT_BSTR | VT_BYREF */
6927     filename = _bstr_("test.xml");
6928     V_VT(&file) = VT_BSTR | VT_BYREF;
6929     V_BSTRREF(&file) = &filename;
6930
6931     hr = IXMLDOMDocument_save(doc, file);
6932     EXPECT_HR(hr, S_OK);
6933
6934     IXMLDOMDocument_Release(doc);
6935
6936     hfile = CreateFileA("test.xml", GENERIC_READ, 0, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL );
6937     ok(hfile != INVALID_HANDLE_VALUE, "Could not open file: %u\n", GetLastError());
6938     if(hfile == INVALID_HANDLE_VALUE) return;
6939
6940     ReadFile(hfile, buffer, sizeof(buffer), &read, NULL);
6941     ok(read != 0, "could not read file\n");
6942     ok(buffer[0] != '<' || buffer[1] != '?', "File contains processing instruction\n");
6943
6944     CloseHandle(hfile);
6945     DeleteFile("test.xml");
6946     free_bstrs();
6947 }
6948
6949 static void test_testTransforms(void)
6950 {
6951     IXMLDOMDocument *doc, *docSS;
6952     IXMLDOMNode *pNode;
6953     VARIANT_BOOL bSucc;
6954
6955     HRESULT hr;
6956
6957     doc = create_document(&IID_IXMLDOMDocument);
6958     if (!doc) return;
6959
6960     docSS = create_document(&IID_IXMLDOMDocument);
6961     if (!docSS)
6962     {
6963         IXMLDOMDocument_Release(doc);
6964         return;
6965     }
6966
6967     hr = IXMLDOMDocument_loadXML(doc, _bstr_(szTransformXML), &bSucc);
6968     ok(hr == S_OK, "ret %08x\n", hr );
6969
6970     hr = IXMLDOMDocument_loadXML(docSS, _bstr_(szTransformSSXML), &bSucc);
6971     ok(hr == S_OK, "ret %08x\n", hr );
6972
6973     hr = IXMLDOMDocument_QueryInterface(docSS, &IID_IXMLDOMNode, (void**)&pNode );
6974     ok(hr == S_OK, "ret %08x\n", hr );
6975     if(hr == S_OK)
6976     {
6977         BSTR bOut;
6978
6979         hr = IXMLDOMDocument_transformNode(doc, pNode, &bOut);
6980         ok(hr == S_OK, "ret %08x\n", hr );
6981         if(hr == S_OK)
6982         {
6983             ok( compareIgnoreReturns( bOut, _bstr_(szTransformOutput)), "Stylesheet output not correct\n");
6984             SysFreeString(bOut);
6985         }
6986
6987         IXMLDOMNode_Release(pNode);
6988     }
6989
6990     IXMLDOMDocument_Release(docSS);
6991     IXMLDOMDocument_Release(doc);
6992
6993     free_bstrs();
6994 }
6995
6996 static void test_Namespaces(void)
6997 {
6998     IXMLDOMDocument *doc;
6999     IXMLDOMNode *pNode;
7000     IXMLDOMNode *pNode2 = NULL;
7001     VARIANT_BOOL bSucc;
7002     HRESULT hr;
7003     BSTR str;
7004     static  const CHAR szNamespacesXML[] =
7005 "<?xml version=\"1.0\"?>\n"
7006 "<XMI xmi.version=\"1.1\" xmlns:Model=\"http://omg.org/mof.Model/1.3\">"
7007 "  <XMI.content>"
7008 "    <Model:Package name=\"WinePackage\" />"
7009 "  </XMI.content>"
7010 "</XMI>";
7011
7012     doc = create_document(&IID_IXMLDOMDocument);
7013     if (!doc) return;
7014
7015     hr = IXMLDOMDocument_loadXML(doc, _bstr_(szNamespacesXML), &bSucc);
7016     ok(hr == S_OK, "ret %08x\n", hr );
7017     ok(bSucc == VARIANT_TRUE, "Expected VARIANT_TRUE got VARIANT_FALSE\n");
7018
7019     hr = IXMLDOMDocument_selectSingleNode(doc, _bstr_("//XMI.content"), &pNode );
7020     ok(hr == S_OK, "ret %08x\n", hr );
7021     if(hr == S_OK)
7022     {
7023         hr = IXMLDOMNode_get_firstChild( pNode, &pNode2 );
7024         ok( hr == S_OK, "ret %08x\n", hr );
7025         ok( pNode2 != NULL, "pNode2 == NULL\n");
7026
7027         /* Test get_prefix */
7028         hr = IXMLDOMNode_get_prefix(pNode2, NULL);
7029         ok( hr == E_INVALIDARG, "ret %08x\n", hr );
7030         /* NOTE: Need to test that arg2 gets cleared on Error. */
7031
7032         hr = IXMLDOMNode_get_prefix(pNode2, &str);
7033         ok( hr == S_OK, "ret %08x\n", hr );
7034         ok( !lstrcmpW( str, _bstr_("Model")), "incorrect prefix string\n");
7035         SysFreeString(str);
7036
7037         hr = IXMLDOMNode_get_nodeName(pNode2, &str);
7038         ok( hr == S_OK, "ret %08x\n", hr );
7039         todo_wine ok( !lstrcmpW( str, _bstr_("Model:Package")), "incorrect nodeName string\n");
7040         SysFreeString(str);
7041
7042         /* Test get_namespaceURI */
7043         hr = IXMLDOMNode_get_namespaceURI(pNode2, NULL);
7044         ok( hr == E_INVALIDARG, "ret %08x\n", hr );
7045         /* NOTE: Need to test that arg2 gets cleared on Error. */
7046
7047         hr = IXMLDOMNode_get_namespaceURI(pNode2, &str);
7048         ok( hr == S_OK, "ret %08x\n", hr );
7049         ok( !lstrcmpW( str, _bstr_("http://omg.org/mof.Model/1.3")), "incorrect namespaceURI string\n");
7050         SysFreeString(str);
7051
7052         IXMLDOMNode_Release(pNode2);
7053         IXMLDOMNode_Release(pNode);
7054     }
7055
7056     IXMLDOMDocument_Release(doc);
7057
7058     free_bstrs();
7059 }
7060
7061 static void test_FormattingXML(void)
7062 {
7063     IXMLDOMDocument *doc;
7064     IXMLDOMElement *pElement;
7065     VARIANT_BOOL bSucc;
7066     HRESULT hr;
7067     BSTR str;
7068     static const CHAR szLinefeedXML[] = "<?xml version=\"1.0\"?>\n<Root>\n\t<Sub val=\"A\" />\n</Root>";
7069     static const CHAR szLinefeedRootXML[] = "<Root>\r\n\t<Sub val=\"A\"/>\r\n</Root>";
7070
7071     doc = create_document(&IID_IXMLDOMDocument);
7072     if (!doc) return;
7073
7074     hr = IXMLDOMDocument_loadXML(doc, _bstr_(szLinefeedXML), &bSucc);
7075     ok(hr == S_OK, "ret %08x\n", hr );
7076     ok(bSucc == VARIANT_TRUE, "Expected VARIANT_TRUE got VARIANT_FALSE\n");
7077
7078     if(bSucc == VARIANT_TRUE)
7079     {
7080         hr = IXMLDOMDocument_get_documentElement(doc, &pElement);
7081         ok(hr == S_OK, "ret %08x\n", hr );
7082         if(hr == S_OK)
7083         {
7084             hr = IXMLDOMElement_get_xml(pElement, &str);
7085             ok(hr == S_OK, "ret %08x\n", hr );
7086             ok( !lstrcmpW( str, _bstr_(szLinefeedRootXML) ), "incorrect element xml\n");
7087             SysFreeString(str);
7088
7089             IXMLDOMElement_Release(pElement);
7090         }
7091     }
7092
7093     IXMLDOMDocument_Release(doc);
7094
7095     free_bstrs();
7096 }
7097
7098 typedef struct _nodetypedvalue_t {
7099     const char *name;
7100     VARTYPE type;
7101     const char *value; /* value in string format */
7102 } nodetypedvalue_t;
7103
7104 static const nodetypedvalue_t get_nodetypedvalue[] = {
7105     { "root/string",    VT_BSTR, "Wine" },
7106     { "root/string2",   VT_BSTR, "String" },
7107     { "root/number",    VT_BSTR, "12.44" },
7108     { "root/number2",   VT_BSTR, "-3.71e3" },
7109     { "root/int",       VT_I4,   "-13" },
7110     { "root/fixed",     VT_CY,   "7322.9371" },
7111     { "root/bool",      VT_BOOL, "-1" },
7112     { "root/datetime",  VT_DATE, "40135.14" },
7113     { "root/datetimetz",VT_DATE, "37813.59" },
7114     { "root/date",      VT_DATE, "665413" },
7115     { "root/time",      VT_DATE, "0.5813889" },
7116     { "root/timetz",    VT_DATE, "1.112512" },
7117     { "root/i1",        VT_I1,   "-13" },
7118     { "root/i2",        VT_I2,   "31915" },
7119     { "root/i4",        VT_I4,   "-312232" },
7120     { "root/ui1",       VT_UI1,  "123" },
7121     { "root/ui2",       VT_UI2,  "48282" },
7122     { "root/ui4",       VT_UI4,  "949281" },
7123     { "root/r4",        VT_R4,   "213124" },
7124     { "root/r8",        VT_R8,   "0.412" },
7125     { "root/float",     VT_R8,   "41221.421" },
7126     { "root/uuid",      VT_BSTR, "333C7BC4-460F-11D0-BC04-0080C7055a83" },
7127     { "root/binbase64", VT_ARRAY|VT_UI1, "base64 test" },
7128     { "root/binbase64_1", VT_ARRAY|VT_UI1, "base64 test" },
7129     { "root/binbase64_2", VT_ARRAY|VT_UI1, "base64 test" },
7130     { 0 }
7131 };
7132
7133 static void test_nodeTypedValue(void)
7134 {
7135     const nodetypedvalue_t *entry = get_nodetypedvalue;
7136     IXMLDOMDocumentType *doctype, *doctype2;
7137     IXMLDOMProcessingInstruction *pi;
7138     IXMLDOMDocumentFragment *frag;
7139     IXMLDOMDocument *doc, *doc2;
7140     IXMLDOMCDATASection *cdata;
7141     IXMLDOMComment *comment;
7142     IXMLDOMNode *node;
7143     VARIANT_BOOL b;
7144     VARIANT value;
7145     HRESULT hr;
7146
7147     doc = create_document(&IID_IXMLDOMDocument);
7148     if (!doc) return;
7149
7150     b = VARIANT_FALSE;
7151     hr = IXMLDOMDocument_loadXML(doc, _bstr_(szTypeValueXML), &b);
7152     ok(hr == S_OK, "ret %08x\n", hr );
7153     ok(b == VARIANT_TRUE, "got %d\n", b);
7154
7155     hr = IXMLDOMDocument_get_nodeValue(doc, NULL);
7156     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
7157
7158     V_VT(&value) = VT_BSTR;
7159     V_BSTR(&value) = NULL;
7160     hr = IXMLDOMDocument_get_nodeValue(doc, &value);
7161     ok(hr == S_FALSE, "ret %08x\n", hr );
7162     ok(V_VT(&value) == VT_NULL, "expect VT_NULL got %d\n", V_VT(&value));
7163
7164     hr = IXMLDOMDocument_get_nodeTypedValue(doc, NULL);
7165     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
7166
7167     V_VT(&value) = VT_EMPTY;
7168     hr = IXMLDOMDocument_get_nodeTypedValue(doc, &value);
7169     ok(hr == S_FALSE, "ret %08x\n", hr );
7170     ok(V_VT(&value) == VT_NULL, "got %d\n", V_VT(&value));
7171
7172     hr = IXMLDOMDocument_selectSingleNode(doc, _bstr_("root/string"), &node);
7173     ok(hr == S_OK, "ret %08x\n", hr );
7174
7175     V_VT(&value) = VT_BSTR;
7176     V_BSTR(&value) = NULL;
7177     hr = IXMLDOMNode_get_nodeValue(node, &value);
7178     ok(hr == S_FALSE, "ret %08x\n", hr );
7179     ok(V_VT(&value) == VT_NULL, "expect VT_NULL got %d\n", V_VT(&value));
7180
7181     hr = IXMLDOMNode_get_nodeTypedValue(node, NULL);
7182     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
7183
7184     IXMLDOMNode_Release(node);
7185
7186     hr = IXMLDOMDocument_selectSingleNode(doc, _bstr_("root/binhex"), &node);
7187     ok(hr == S_OK, "ret %08x\n", hr );
7188     {
7189         BYTE bytes[] = {0xff,0xfc,0xa0,0x12,0x00,0x3c};
7190
7191         hr = IXMLDOMNode_get_nodeTypedValue(node, &value);
7192         ok(hr == S_OK, "ret %08x\n", hr );
7193         ok(V_VT(&value) == (VT_ARRAY|VT_UI1), "incorrect type\n");
7194         ok(V_ARRAY(&value)->rgsabound[0].cElements == 6, "incorrect array size\n");
7195         if(V_ARRAY(&value)->rgsabound[0].cElements == 6)
7196             ok(!memcmp(bytes, V_ARRAY(&value)->pvData, sizeof(bytes)), "incorrect value\n");
7197         VariantClear(&value);
7198         IXMLDOMNode_Release(node);
7199     }
7200
7201     hr = IXMLDOMDocument_createProcessingInstruction(doc, _bstr_("foo"), _bstr_("value"), &pi);
7202     ok(hr == S_OK, "ret %08x\n", hr );
7203     {
7204         V_VT(&value) = VT_NULL;
7205         V_BSTR(&value) = (void*)0xdeadbeef;
7206         hr = IXMLDOMProcessingInstruction_get_nodeTypedValue(pi, &value);
7207         ok(hr == S_OK, "ret %08x\n", hr );
7208         ok(V_VT(&value) == VT_BSTR, "got %d\n", V_VT(&value));
7209         ok(!lstrcmpW(V_BSTR(&value), _bstr_("value")), "got wrong value\n");
7210         IXMLDOMProcessingInstruction_Release(pi);
7211         VariantClear(&value);
7212     }
7213
7214     hr = IXMLDOMDocument_createCDATASection(doc, _bstr_("[1]*2=3; &gee that's not right!"), &cdata);
7215     ok(hr == S_OK, "ret %08x\n", hr );
7216     {
7217         V_VT(&value) = VT_NULL;
7218         V_BSTR(&value) = (void*)0xdeadbeef;
7219         hr = IXMLDOMCDATASection_get_nodeTypedValue(cdata, &value);
7220         ok(hr == S_OK, "ret %08x\n", hr );
7221         ok(V_VT(&value) == VT_BSTR, "got %d\n", V_VT(&value));
7222         ok(!lstrcmpW(V_BSTR(&value), _bstr_("[1]*2=3; &gee that's not right!")), "got wrong value\n");
7223         IXMLDOMCDATASection_Release(cdata);
7224         VariantClear(&value);
7225     }
7226
7227     hr = IXMLDOMDocument_createComment(doc, _bstr_("comment"), &comment);
7228     ok(hr == S_OK, "ret %08x\n", hr );
7229     {
7230         V_VT(&value) = VT_NULL;
7231         V_BSTR(&value) = (void*)0xdeadbeef;
7232         hr = IXMLDOMComment_get_nodeTypedValue(comment, &value);
7233         ok(hr == S_OK, "ret %08x\n", hr );
7234         ok(V_VT(&value) == VT_BSTR, "got %d\n", V_VT(&value));
7235         ok(!lstrcmpW(V_BSTR(&value), _bstr_("comment")), "got wrong value\n");
7236         IXMLDOMComment_Release(comment);
7237         VariantClear(&value);
7238     }
7239
7240     hr = IXMLDOMDocument_createDocumentFragment(doc, &frag);
7241     ok(hr == S_OK, "ret %08x\n", hr );
7242     {
7243         V_VT(&value) = VT_EMPTY;
7244         hr = IXMLDOMDocumentFragment_get_nodeTypedValue(frag, &value);
7245         ok(hr == S_FALSE, "ret %08x\n", hr );
7246         ok(V_VT(&value) == VT_NULL, "got %d\n", V_VT(&value));
7247         IXMLDOMDocumentFragment_Release(frag);
7248     }
7249
7250     doc2 = create_document(&IID_IXMLDOMDocument);
7251
7252     b = VARIANT_FALSE;
7253     hr = IXMLDOMDocument_loadXML(doc2, _bstr_(szEmailXML), &b);
7254     ok(hr == S_OK, "ret %08x\n", hr );
7255     ok(b == VARIANT_TRUE, "got %d\n", b);
7256
7257     EXPECT_REF(doc2, 1);
7258
7259     hr = IXMLDOMDocument_get_doctype(doc2, &doctype);
7260     ok(hr == S_OK, "ret %08x\n", hr );
7261
7262     EXPECT_REF(doc2, 1);
7263     todo_wine EXPECT_REF(doctype, 2);
7264
7265     {
7266         V_VT(&value) = VT_EMPTY;
7267         hr = IXMLDOMDocumentType_get_nodeTypedValue(doctype, &value);
7268         ok(hr == S_FALSE, "ret %08x\n", hr );
7269         ok(V_VT(&value) == VT_NULL, "got %d\n", V_VT(&value));
7270     }
7271
7272     hr = IXMLDOMDocument_get_doctype(doc2, &doctype2);
7273     ok(hr == S_OK, "ret %08x\n", hr );
7274     ok(doctype != doctype2, "got %p, was %p\n", doctype2, doctype);
7275
7276     IXMLDOMDocumentType_Release(doctype2);
7277     IXMLDOMDocumentType_Release(doctype);
7278
7279     IXMLDOMDocument_Release(doc2);
7280
7281     while (entry->name)
7282     {
7283         hr = IXMLDOMDocument_selectSingleNode(doc, _bstr_(entry->name), &node);
7284         ok(hr == S_OK, "ret %08x\n", hr );
7285
7286         hr = IXMLDOMNode_get_nodeTypedValue(node, &value);
7287         ok(hr == S_OK, "ret %08x\n", hr );
7288         ok(V_VT(&value) == entry->type, "incorrect type, expected %d, got %d\n", entry->type, V_VT(&value));
7289
7290         if (entry->type == (VT_ARRAY|VT_UI1))
7291         {
7292             ok(V_ARRAY(&value)->rgsabound[0].cElements == strlen(entry->value),
7293                "incorrect array size %d\n", V_ARRAY(&value)->rgsabound[0].cElements);
7294         }
7295
7296         if (entry->type != VT_BSTR)
7297         {
7298            if (entry->type == VT_DATE ||
7299                entry->type == VT_R8 ||
7300                entry->type == VT_CY)
7301            {
7302                if (entry->type == VT_DATE)
7303                {
7304                    hr = VariantChangeType(&value, &value, 0, VT_R4);
7305                    ok(hr == S_OK, "ret %08x\n", hr );
7306                }
7307                hr = VariantChangeTypeEx(&value, &value,
7308                                         MAKELCID(MAKELANGID(LANG_ENGLISH, SUBLANG_DEFAULT), SORT_DEFAULT),
7309                                         VARIANT_NOUSEROVERRIDE, VT_BSTR);
7310                ok(hr == S_OK, "ret %08x\n", hr );
7311            }
7312            else
7313            {
7314                hr = VariantChangeType(&value, &value, 0, VT_BSTR);
7315                ok(hr == S_OK, "ret %08x\n", hr );
7316            }
7317
7318            /* for byte array from VT_ARRAY|VT_UI1 it's not a WCHAR buffer */
7319            if (entry->type == (VT_ARRAY|VT_UI1))
7320            {
7321                ok(!memcmp( V_BSTR(&value), entry->value, strlen(entry->value)),
7322                   "expected %s\n", entry->value);
7323            }
7324            else
7325                ok(lstrcmpW( V_BSTR(&value), _bstr_(entry->value)) == 0,
7326                   "expected %s, got %s\n", entry->value, wine_dbgstr_w(V_BSTR(&value)));
7327         }
7328         else
7329            ok(lstrcmpW( V_BSTR(&value), _bstr_(entry->value)) == 0,
7330                "expected %s, got %s\n", entry->value, wine_dbgstr_w(V_BSTR(&value)));
7331
7332         VariantClear( &value );
7333         IXMLDOMNode_Release(node);
7334
7335         entry++;
7336     }
7337
7338     IXMLDOMDocument_Release(doc);
7339     free_bstrs();
7340 }
7341
7342 static void test_TransformWithLoadingLocalFile(void)
7343 {
7344     IXMLDOMDocument *doc;
7345     IXMLDOMDocument *xsl;
7346     IXMLDOMNode *pNode;
7347     VARIANT_BOOL bSucc;
7348     HRESULT hr;
7349     HANDLE file;
7350     DWORD dwWritten;
7351     char lpPathBuffer[MAX_PATH];
7352     int i;
7353
7354     /* Create a Temp File. */
7355     GetTempPathA(MAX_PATH, lpPathBuffer);
7356     strcat(lpPathBuffer, "customers.xml" );
7357
7358     file = CreateFileA(lpPathBuffer, GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL );
7359     ok(file != INVALID_HANDLE_VALUE, "Could not create file: %u\n", GetLastError());
7360     if(file == INVALID_HANDLE_VALUE)
7361         return;
7362
7363     WriteFile(file, szBasicTransformXML, strlen(szBasicTransformXML), &dwWritten, NULL);
7364     CloseHandle(file);
7365
7366     /* Correct path to not include a escape character. */
7367     for(i=0; i < strlen(lpPathBuffer); i++)
7368     {
7369         if(lpPathBuffer[i] == '\\')
7370             lpPathBuffer[i] = '/';
7371     }
7372
7373     doc = create_document(&IID_IXMLDOMDocument);
7374     if (!doc) return;
7375
7376     xsl = create_document(&IID_IXMLDOMDocument);
7377     if (!xsl)
7378     {
7379         IXMLDOMDocument2_Release(doc);
7380         return;
7381     }
7382
7383     hr = IXMLDOMDocument_loadXML(doc, _bstr_(szTypeValueXML), &bSucc);
7384     ok(hr == S_OK, "ret %08x\n", hr );
7385     ok(bSucc == VARIANT_TRUE, "Expected VARIANT_TRUE got VARIANT_FALSE\n");
7386     if(bSucc == VARIANT_TRUE)
7387     {
7388         BSTR sXSL;
7389         BSTR sPart1 = _bstr_(szBasicTransformSSXMLPart1);
7390         BSTR sPart2 = _bstr_(szBasicTransformSSXMLPart2);
7391         BSTR sFileName = _bstr_(lpPathBuffer);
7392         int nLegnth = lstrlenW(sPart1) + lstrlenW(sPart2) + lstrlenW(sFileName) + 1;
7393
7394         sXSL = SysAllocStringLen(NULL, nLegnth);
7395         lstrcpyW(sXSL, sPart1);
7396         lstrcatW(sXSL, sFileName);
7397         lstrcatW(sXSL, sPart2);
7398
7399         hr = IXMLDOMDocument_loadXML(xsl, sXSL, &bSucc);
7400         ok(hr == S_OK, "ret %08x\n", hr );
7401         ok(bSucc == VARIANT_TRUE, "Expected VARIANT_TRUE got VARIANT_FALSE\n");
7402         if(bSucc == VARIANT_TRUE)
7403         {
7404             BSTR sResult;
7405
7406             hr = IXMLDOMDocument_QueryInterface(xsl, &IID_IXMLDOMNode, (void**)&pNode );
7407             ok(hr == S_OK, "ret %08x\n", hr );
7408             if(hr == S_OK)
7409             {
7410                 /* This will load the temp file via the XSL */
7411                 hr = IXMLDOMDocument_transformNode(doc, pNode, &sResult);
7412                 ok(hr == S_OK, "ret %08x\n", hr );
7413                 if(hr == S_OK)
7414                 {
7415                     ok( compareIgnoreReturns( sResult, _bstr_(szBasicTransformOutput)), "Stylesheet output not correct\n");
7416                     SysFreeString(sResult);
7417                 }
7418
7419                 IXMLDOMNode_Release(pNode);
7420             }
7421         }
7422
7423         SysFreeString(sXSL);
7424     }
7425
7426     IXMLDOMDocument_Release(doc);
7427     IXMLDOMDocument_Release(xsl);
7428
7429     DeleteFile(lpPathBuffer);
7430     free_bstrs();
7431 }
7432
7433 static void test_put_nodeValue(void)
7434 {
7435     static const WCHAR jeevesW[] = {'J','e','e','v','e','s',' ','&',' ','W','o','o','s','t','e','r',0};
7436     IXMLDOMDocument *doc;
7437     IXMLDOMText *text;
7438     IXMLDOMEntityReference *entityref;
7439     IXMLDOMAttribute *attr;
7440     IXMLDOMNode *node;
7441     HRESULT hr;
7442     VARIANT data, type;
7443
7444     doc = create_document(&IID_IXMLDOMDocument);
7445     if (!doc) return;
7446
7447     /* test for unsupported types */
7448     /* NODE_DOCUMENT */
7449     hr = IXMLDOMDocument_QueryInterface(doc, &IID_IXMLDOMNode, (void**)&node);
7450     ok(hr == S_OK, "ret %08x\n", hr );
7451     V_VT(&data) = VT_BSTR;
7452     V_BSTR(&data) = _bstr_("one two three");
7453     hr = IXMLDOMNode_put_nodeValue(node, data);
7454     ok(hr == E_FAIL, "ret %08x\n", hr );
7455     IXMLDOMNode_Release(node);
7456
7457     /* NODE_DOCUMENT_FRAGMENT */
7458     V_VT(&type) = VT_I1;
7459     V_I1(&type) = NODE_DOCUMENT_FRAGMENT;
7460     hr = IXMLDOMDocument_createNode(doc, type, _bstr_("test"), NULL, &node);
7461     ok(hr == S_OK, "ret %08x\n", hr );
7462     V_VT(&data) = VT_BSTR;
7463     V_BSTR(&data) = _bstr_("one two three");
7464     hr = IXMLDOMNode_put_nodeValue(node, data);
7465     ok(hr == E_FAIL, "ret %08x\n", hr );
7466     IXMLDOMNode_Release(node);
7467
7468     /* NODE_ELEMENT */
7469     V_VT(&type) = VT_I1;
7470     V_I1(&type) = NODE_ELEMENT;
7471     hr = IXMLDOMDocument_createNode(doc, type, _bstr_("test"), NULL, &node);
7472     ok(hr == S_OK, "ret %08x\n", hr );
7473     V_VT(&data) = VT_BSTR;
7474     V_BSTR(&data) = _bstr_("one two three");
7475     hr = IXMLDOMNode_put_nodeValue(node, data);
7476     ok(hr == E_FAIL, "ret %08x\n", hr );
7477     IXMLDOMNode_Release(node);
7478
7479     /* NODE_ENTITY_REFERENCE */
7480     hr = IXMLDOMDocument_createEntityReference(doc, _bstr_("ref"), &entityref);
7481     ok(hr == S_OK, "ret %08x\n", hr );
7482
7483     V_VT(&data) = VT_BSTR;
7484     V_BSTR(&data) = _bstr_("one two three");
7485     hr = IXMLDOMEntityReference_put_nodeValue(entityref, data);
7486     ok(hr == E_FAIL, "ret %08x\n", hr );
7487
7488     hr = IXMLDOMEntityReference_QueryInterface(entityref, &IID_IXMLDOMNode, (void**)&node);
7489     ok(hr == S_OK, "ret %08x\n", hr );
7490     V_VT(&data) = VT_BSTR;
7491     V_BSTR(&data) = _bstr_("one two three");
7492     hr = IXMLDOMNode_put_nodeValue(node, data);
7493     ok(hr == E_FAIL, "ret %08x\n", hr );
7494     IXMLDOMNode_Release(node);
7495     IXMLDOMEntityReference_Release(entityref);
7496
7497     /* supported types */
7498     hr = IXMLDOMDocument_createTextNode(doc, _bstr_(""), &text);
7499     ok(hr == S_OK, "ret %08x\n", hr );
7500     V_VT(&data) = VT_BSTR;
7501     V_BSTR(&data) = _bstr_("Jeeves & Wooster");
7502     hr = IXMLDOMText_put_nodeValue(text, data);
7503     ok(hr == S_OK, "ret %08x\n", hr );
7504     IXMLDOMText_Release(text);
7505
7506     hr = IXMLDOMDocument_createAttribute(doc, _bstr_("attr"), &attr);
7507     ok(hr == S_OK, "ret %08x\n", hr );
7508     V_VT(&data) = VT_BSTR;
7509     V_BSTR(&data) = _bstr_("Jeeves & Wooster");
7510     hr = IXMLDOMAttribute_put_nodeValue(attr, data);
7511     ok(hr == S_OK, "ret %08x\n", hr );
7512     hr = IXMLDOMAttribute_get_nodeValue(attr, &data);
7513     ok(hr == S_OK, "ret %08x\n", hr );
7514     ok(memcmp(V_BSTR(&data), jeevesW, sizeof(jeevesW)) == 0, "got %s\n",
7515         wine_dbgstr_w(V_BSTR(&data)));
7516     VariantClear(&data);
7517     IXMLDOMAttribute_Release(attr);
7518
7519     free_bstrs();
7520
7521     IXMLDOMDocument_Release(doc);
7522 }
7523
7524 static void test_document_IObjectSafety(void)
7525 {
7526     IXMLDOMDocument *doc;
7527     IObjectSafety *safety;
7528     HRESULT hr;
7529
7530     doc = create_document(&IID_IXMLDOMDocument);
7531     if (!doc) return;
7532
7533     hr = IXMLDOMDocument_QueryInterface(doc, &IID_IObjectSafety, (void**)&safety);
7534     ok(hr == S_OK, "ret %08x\n", hr );
7535
7536     test_IObjectSafety_common(safety);
7537
7538     IObjectSafety_Release(safety);
7539
7540     IXMLDOMDocument_Release(doc);
7541 }
7542
7543 typedef struct _property_test_t {
7544     const GUID *guid;
7545     const char *clsid;
7546     const char *property;
7547     const char *value;
7548 } property_test_t;
7549
7550 static const property_test_t properties_test_data[] = {
7551     { &CLSID_DOMDocument,  "CLSID_DOMDocument" , "SelectionLanguage", "XSLPattern" },
7552     { &CLSID_DOMDocument2,  "CLSID_DOMDocument2" , "SelectionLanguage", "XSLPattern" },
7553     { &CLSID_DOMDocument30, "CLSID_DOMDocument30", "SelectionLanguage", "XSLPattern" },
7554     { &CLSID_DOMDocument40, "CLSID_DOMDocument40", "SelectionLanguage", "XPath" },
7555     { &CLSID_DOMDocument60, "CLSID_DOMDocument60", "SelectionLanguage", "XPath" },
7556     { 0 }
7557 };
7558
7559 static void test_default_properties(void)
7560 {
7561     const property_test_t *entry = properties_test_data;
7562     IXMLDOMDocument2 *doc;
7563     VARIANT var;
7564     HRESULT hr;
7565
7566     while (entry->guid)
7567     {
7568         hr = CoCreateInstance(entry->guid, NULL, CLSCTX_INPROC_SERVER, &IID_IXMLDOMDocument2, (void**)&doc);
7569         if (hr != S_OK)
7570         {
7571             win_skip("can't create %s instance\n", entry->clsid);
7572             entry++;
7573             continue;
7574         }
7575
7576         hr = IXMLDOMDocument2_getProperty(doc, _bstr_(entry->property), &var);
7577         ok(hr == S_OK, "got 0x%08x\n", hr);
7578         ok(lstrcmpW(V_BSTR(&var), _bstr_(entry->value)) == 0, "expected %s, for %s\n",
7579            entry->value, entry->clsid);
7580         VariantClear(&var);
7581
7582         IXMLDOMDocument2_Release(doc);
7583
7584         entry++;
7585     }
7586 }
7587
7588 static void test_XSLPattern(void)
7589 {
7590     IXMLDOMDocument2 *doc;
7591     IXMLDOMNodeList *list;
7592     VARIANT_BOOL b;
7593     LONG len;
7594
7595     doc = create_document(&IID_IXMLDOMDocument2);
7596     if (!doc) return;
7597
7598     ole_check(IXMLDOMDocument2_loadXML(doc, _bstr_(szExampleXML), &b));
7599     ok(b == VARIANT_TRUE, "failed to load XML string\n");
7600
7601     /* switch to XSLPattern */
7602     ole_check(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionLanguage"), _variantbstr_("XSLPattern")));
7603
7604     /* XPath doesn't select elements with non-null default namespace with unqualified selectors, XSLPattern does */
7605     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//elem/c"), &list));
7606     len = 0;
7607     ole_check(IXMLDOMNodeList_get_length(list, &len));
7608     /* should select <elem><c> and <elem xmlns='...'><c> but not <elem><foo:c> */
7609     ok(len == 3, "expected 3 entries in list, got %d\n", len);
7610     IXMLDOMNodeList_Release(list);
7611
7612     /* for XSLPattern start index is 0, for XPath it's 1 */
7613     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[0]"), &list));
7614     len = 0;
7615     ole_check(IXMLDOMNodeList_get_length(list, &len));
7616     ok(len != 0, "expected filled list\n");
7617     if (len)
7618         expect_list_and_release(list, "E1.E2.D1");
7619
7620     /* index() */
7621     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index()=1]"), &list));
7622     len = 0;
7623     ole_check(IXMLDOMNodeList_get_length(list, &len));
7624     ok(len != 0, "expected filled list\n");
7625     if (len)
7626         expect_list_and_release(list, "E2.E2.D1");
7627
7628     /* $eq$ */
7629     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index() $eq$ 1]"), &list));
7630     len = 0;
7631     ole_check(IXMLDOMNodeList_get_length(list, &len));
7632     ok(len != 0, "expected filled list\n");
7633     if (len)
7634         expect_list_and_release(list, "E2.E2.D1");
7635
7636     /* end() */
7637     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[end()]"), &list));
7638     len = 0;
7639     ole_check(IXMLDOMNodeList_get_length(list, &len));
7640     ok(len != 0, "expected filled list\n");
7641     if (len)
7642         expect_list_and_release(list, "E4.E2.D1");
7643
7644     /* $not$ */
7645     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[$not$ end()]"), &list));
7646     len = 0;
7647     ole_check(IXMLDOMNodeList_get_length(list, &len));
7648     ok(len != 0, "expected filled list\n");
7649     if (len)
7650         expect_list_and_release(list, "E1.E2.D1 E2.E2.D1 E3.E2.D1");
7651
7652     /* !=/$ne$ */
7653     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index() != 0]"), &list));
7654     len = 0;
7655     ole_check(IXMLDOMNodeList_get_length(list, &len));
7656     ok(len != 0, "expected filled list\n");
7657     if (len)
7658         expect_list_and_release(list, "E2.E2.D1 E3.E2.D1 E4.E2.D1");
7659     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index() $ne$ 0]"), &list));
7660     len = 0;
7661     ole_check(IXMLDOMNodeList_get_length(list, &len));
7662     ok(len != 0, "expected filled list\n");
7663     if (len)
7664         expect_list_and_release(list, "E2.E2.D1 E3.E2.D1 E4.E2.D1");
7665
7666     /* </$lt$ */
7667     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index() < 2]"), &list));
7668     len = 0;
7669     ole_check(IXMLDOMNodeList_get_length(list, &len));
7670     ok(len != 0, "expected filled list\n");
7671     if (len)
7672         expect_list_and_release(list, "E1.E2.D1 E2.E2.D1");
7673     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index() $lt$ 2]"), &list));
7674     len = 0;
7675     ole_check(IXMLDOMNodeList_get_length(list, &len));
7676     ok(len != 0, "expected filled list\n");
7677     if (len)
7678         expect_list_and_release(list, "E1.E2.D1 E2.E2.D1");
7679
7680     /* <=/$le$ */
7681     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index() <= 1]"), &list));
7682     len = 0;
7683     ole_check(IXMLDOMNodeList_get_length(list, &len));
7684     ok(len != 0, "expected filled list\n");
7685     if (len)
7686         expect_list_and_release(list, "E1.E2.D1 E2.E2.D1");
7687     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index() $le$ 1]"), &list));
7688     len = 0;
7689     ole_check(IXMLDOMNodeList_get_length(list, &len));
7690     ok(len != 0, "expected filled list\n");
7691     if (len)
7692         expect_list_and_release(list, "E1.E2.D1 E2.E2.D1");
7693
7694     /* >/$gt$ */
7695     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index() > 1]"), &list));
7696     len = 0;
7697     ole_check(IXMLDOMNodeList_get_length(list, &len));
7698     ok(len != 0, "expected filled list\n");
7699     if (len)
7700         expect_list_and_release(list, "E3.E2.D1 E4.E2.D1");
7701     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index() $gt$ 1]"), &list));
7702     len = 0;
7703     ole_check(IXMLDOMNodeList_get_length(list, &len));
7704     ok(len != 0, "expected filled list\n");
7705     if (len)
7706         expect_list_and_release(list, "E3.E2.D1 E4.E2.D1");
7707
7708     /* >=/$ge$ */
7709     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index() >= 2]"), &list));
7710     len = 0;
7711     ole_check(IXMLDOMNodeList_get_length(list, &len));
7712     ok(len != 0, "expected filled list\n");
7713     if (len)
7714         expect_list_and_release(list, "E3.E2.D1 E4.E2.D1");
7715     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index() $ge$ 2]"), &list));
7716     len = 0;
7717     ole_check(IXMLDOMNodeList_get_length(list, &len));
7718     ok(len != 0, "expected filled list\n");
7719     if (len)
7720         expect_list_and_release(list, "E3.E2.D1 E4.E2.D1");
7721
7722     /* $ieq$ */
7723     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[a $ieq$ 'a2 field']"), &list));
7724     len = 0;
7725     ole_check(IXMLDOMNodeList_get_length(list, &len));
7726     ok(len != 0, "expected filled list\n");
7727     if (len)
7728         expect_list_and_release(list, "E2.E2.D1");
7729
7730     /* $ine$ */
7731     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[a $ine$ 'a2 field']"), &list));
7732     len = 0;
7733     ole_check(IXMLDOMNodeList_get_length(list, &len));
7734     ok(len != 0, "expected filled list\n");
7735     if (len)
7736         expect_list_and_release(list, "E1.E2.D1 E3.E2.D1 E4.E2.D1");
7737
7738     /* $ilt$ */
7739     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[a $ilt$ 'a3 field']"), &list));
7740     len = 0;
7741     ole_check(IXMLDOMNodeList_get_length(list, &len));
7742     ok(len != 0, "expected filled list\n");
7743     if (len)
7744         expect_list_and_release(list, "E1.E2.D1 E2.E2.D1");
7745
7746     /* $ile$ */
7747     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[a $ile$ 'a2 field']"), &list));
7748     len = 0;
7749     ole_check(IXMLDOMNodeList_get_length(list, &len));
7750     ok(len != 0, "expected filled list\n");
7751     if (len)
7752         expect_list_and_release(list, "E1.E2.D1 E2.E2.D1");
7753
7754     /* $igt$ */
7755     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[a $igt$ 'a2 field']"), &list));
7756     len = 0;
7757     ole_check(IXMLDOMNodeList_get_length(list, &len));
7758     ok(len != 0, "expected filled list\n");
7759     if (len)
7760         expect_list_and_release(list, "E3.E2.D1 E4.E2.D1");
7761
7762     /* $ige$ */
7763     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[a $ige$ 'a3 field']"), &list));
7764     len = 0;
7765     ole_check(IXMLDOMNodeList_get_length(list, &len));
7766     ok(len != 0, "expected filled list\n");
7767     if (len)
7768         expect_list_and_release(list, "E3.E2.D1 E4.E2.D1");
7769
7770     /* $any$ */
7771     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[$any$ *='B2 field']"), &list));
7772     len = 0;
7773     ole_check(IXMLDOMNodeList_get_length(list, &len));
7774     ok(len != 0, "expected filled list\n");
7775     if (len)
7776         expect_list_and_release(list, "E2.E2.D1");
7777
7778     /* $all$ */
7779     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[$all$ *!='B2 field']"), &list));
7780     len = 0;
7781     ole_check(IXMLDOMNodeList_get_length(list, &len));
7782     ok(len != 0, "expected filled list\n");
7783     if (len)
7784         expect_list_and_release(list, "E1.E2.D1 E3.E2.D1 E4.E2.D1");
7785
7786     /* or/$or$/|| */
7787     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index()=0 or end()]"), &list));
7788     len = 0;
7789     ole_check(IXMLDOMNodeList_get_length(list, &len));
7790     ok(len != 0, "expected filled list\n");
7791     if (len)
7792         expect_list_and_release(list, "E1.E2.D1 E4.E2.D1");
7793     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index()=0 $or$ end()]"), &list));
7794     len = 0;
7795     ole_check(IXMLDOMNodeList_get_length(list, &len));
7796     ok(len != 0, "expected filled list\n");
7797     if (len)
7798         expect_list_and_release(list, "E1.E2.D1 E4.E2.D1");
7799     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index()=0 || end()]"), &list));
7800     len = 0;
7801     ole_check(IXMLDOMNodeList_get_length(list, &len));
7802     ok(len != 0, "expected filled list\n");
7803     if (len)
7804         expect_list_and_release(list, "E1.E2.D1 E4.E2.D1");
7805
7806     /* and/$and$/&& */
7807     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index()>0 and $not$ end()]"), &list));
7808     len = 0;
7809     ole_check(IXMLDOMNodeList_get_length(list, &len));
7810     ok(len != 0, "expected filled list\n");
7811     if (len)
7812         expect_list_and_release(list, "E2.E2.D1 E3.E2.D1");
7813     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index()>0 $and$ $not$ end()]"), &list));
7814     len = 0;
7815     ole_check(IXMLDOMNodeList_get_length(list, &len));
7816     ok(len != 0, "expected filled list\n");
7817     if (len)
7818         expect_list_and_release(list, "E2.E2.D1 E3.E2.D1");
7819     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[index()>0 && $not$ end()]"), &list));
7820     len = 0;
7821     ole_check(IXMLDOMNodeList_get_length(list, &len));
7822     ok(len != 0, "expected filled list\n");
7823     if (len)
7824         expect_list_and_release(list, "E2.E2.D1 E3.E2.D1");
7825
7826     /* namespace handling */
7827     /* no registered namespaces */
7828     ole_check(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionNamespaces"), _variantbstr_("")));
7829     list = NULL;
7830
7831     /* prefixes don't need to be registered, you may use them as they are in the doc */
7832     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//bar:x"), &list));
7833     len = 0;
7834     ole_check(IXMLDOMNodeList_get_length(list, &len));
7835     ok(len != 0, "expected filled list\n");
7836     if (len)
7837         expect_list_and_release(list, "E5.E1.E4.E1.E2.D1 E6.E2.E4.E1.E2.D1");
7838
7839     /* prefixes must be explicitly specified in the name */
7840     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//foo:elem"), &list));
7841     len = 0;
7842     ole_check(IXMLDOMNodeList_get_length(list, &len));
7843     ok(len == 0, "expected empty list\n");
7844     IXMLDOMNodeList_Release(list);
7845
7846     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//foo:c"), &list));
7847     len = 0;
7848     ole_check(IXMLDOMNodeList_get_length(list, &len));
7849     ok(len != 0, "expected filled list\n");
7850     if (len)
7851         expect_list_and_release(list, "E3.E4.E2.D1");
7852
7853     /* explicitly register prefix foo */
7854     ole_check(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionNamespaces"), _variantbstr_("xmlns:foo='urn:uuid:86B2F87F-ACB6-45cd-8B77-9BDB92A01A29'")));
7855
7856     /* now we get the same behavior as XPath */
7857     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//foo:c"), &list));
7858     len = 0;
7859     ole_check(IXMLDOMNodeList_get_length(list, &len));
7860     ok(len != 0, "expected filled list\n");
7861     if (len)
7862         expect_list_and_release(list, "E3.E3.E2.D1 E3.E4.E2.D1");
7863
7864     /* set prefix foo to some nonexistent namespace */
7865     ole_check(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionNamespaces"), _variantbstr_("xmlns:foo='urn:nonexistent-foo'")));
7866
7867     /* the registered prefix takes precedence */
7868     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//foo:c"), &list));
7869     len = 0;
7870     ole_check(IXMLDOMNodeList_get_length(list, &len));
7871     ok(len == 0, "expected empty list\n");
7872     IXMLDOMNodeList_Release(list);
7873
7874     IXMLDOMDocument2_Release(doc);
7875
7876     doc = create_document(&IID_IXMLDOMDocument2);
7877     if (!doc) return;
7878
7879     ole_check(IXMLDOMDocument2_loadXML(doc, _bstr_(szNodeTypesXML), &b));
7880     ok(b == VARIANT_TRUE, "failed to load XML string\n");
7881     list = NULL;
7882
7883     /* attribute() */
7884     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("attribute()"), &list));
7885     len = 0;
7886     ole_check(IXMLDOMNodeList_get_length(list, &len));
7887     ok(len == 0, "expected empty list\n");
7888     IXMLDOMNodeList_Release(list);
7889
7890     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("attribute('depth')"), &list));
7891     len = 0;
7892     ole_check(IXMLDOMNodeList_get_length(list, &len));
7893     ok(len == 0, "expected empty list\n");
7894     IXMLDOMNodeList_Release(list);
7895
7896     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root/attribute('depth')"), &list));
7897     len = 0;
7898     ole_check(IXMLDOMNodeList_get_length(list, &len));
7899     ok(len != 0, "expected filled list\n");
7900     expect_list_and_release(list, "A'depth'.E3.D1");
7901
7902     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//x/attribute()"), &list));
7903     len = 0;
7904     ole_check(IXMLDOMNodeList_get_length(list, &len));
7905     ok(len != 0, "expected filled list\n");
7906     expect_list_and_release(list, "A'id'.E3.E3.D1 A'depth'.E3.E3.D1");
7907
7908     list = NULL;
7909     ole_expect(IXMLDOMDocument2_selectNodes(doc, _bstr_("//x//attribute(id)"), &list), E_FAIL);
7910     if (list)
7911         IXMLDOMNodeList_Release(list);
7912     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//x//attribute('id')"), &list));
7913     len = 0;
7914     ole_check(IXMLDOMNodeList_get_length(list, &len));
7915     ok(len != 0, "expected filled list\n");
7916     if (len)
7917         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");
7918
7919     /* comment() */
7920     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("comment()"), &list));
7921     len = 0;
7922     ole_check(IXMLDOMNodeList_get_length(list, &len));
7923     ok(len != 0, "expected filled list\n");
7924     if (len)
7925         expect_list_and_release(list, "C2.D1");
7926
7927     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//comment()"), &list));
7928     len = 0;
7929     ole_check(IXMLDOMNodeList_get_length(list, &len));
7930     ok(len != 0, "expected filled list\n");
7931     if (len)
7932         expect_list_and_release(list, "C2.D1 C1.E3.D1 C2.E3.E3.D1 C2.E4.E3.D1");
7933
7934     /* element() */
7935     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("element()"), &list));
7936     len = 0;
7937     ole_check(IXMLDOMNodeList_get_length(list, &len));
7938     ok(len != 0, "expected filled list\n");
7939     if (len)
7940         expect_list_and_release(list, "E3.D1");
7941
7942     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root/y/element()"), &list));
7943     len = 0;
7944     ole_check(IXMLDOMNodeList_get_length(list, &len));
7945     ok(len != 0, "expected filled list\n");
7946     if (len)
7947         expect_list_and_release(list, "E4.E4.E3.D1 E5.E4.E3.D1 E6.E4.E3.D1");
7948
7949     list = NULL;
7950     ole_expect(IXMLDOMDocument2_selectNodes(doc, _bstr_("//element(a)"), &list), E_FAIL);
7951     if (list)
7952         IXMLDOMNodeList_Release(list);
7953     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//element('a')"), &list));
7954     len = 0;
7955     ole_check(IXMLDOMNodeList_get_length(list, &len));
7956     ok(len != 0, "expected filled list\n");
7957     if (len)
7958         expect_list_and_release(list, "E4.E3.E3.D1 E4.E4.E3.D1");
7959
7960     /* node() */
7961     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("node()"), &list));
7962     len = 0;
7963     ole_check(IXMLDOMNodeList_get_length(list, &len));
7964     ok(len != 0, "expected filled list\n");
7965     if (len)
7966         expect_list_and_release(list, "P1.D1 C2.D1 E3.D1");
7967
7968     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//x/node()"), &list));
7969     len = 0;
7970     ole_check(IXMLDOMNodeList_get_length(list, &len));
7971     ok(len != 0, "expected filled list\n");
7972     if (len)
7973         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");
7974
7975     /* nodeType() */
7976     /* XML_ELEMENT_NODE */
7977     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//x/node()[nodeType()=1]"), &list));
7978     len = 0;
7979     ole_check(IXMLDOMNodeList_get_length(list, &len));
7980     ok(len != 0, "expected filled list\n");
7981     if (len)
7982         expect_list_and_release(list, "E4.E3.E3.D1 E5.E3.E3.D1 E6.E3.E3.D1");
7983     /* XML_TEXT_NODE */
7984     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//x/node()[nodeType()=3]"), &list));
7985     len = 0;
7986     ole_check(IXMLDOMNodeList_get_length(list, &len));
7987     ok(len != 0, "expected filled list\n");
7988     if (len)
7989         expect_list_and_release(list, "T3.E3.E3.D1");
7990     /* XML_PI_NODE */
7991     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//x/node()[nodeType()=7]"), &list));
7992     len = 0;
7993     ole_check(IXMLDOMNodeList_get_length(list, &len));
7994     ok(len != 0, "expected filled list\n");
7995     if (len)
7996         expect_list_and_release(list, "P1.E3.E3.D1");
7997     /* XML_COMMENT_NODE */
7998     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//x/node()[nodeType()=8]"), &list));
7999     len = 0;
8000     ole_check(IXMLDOMNodeList_get_length(list, &len));
8001     ok(len != 0, "expected filled list\n");
8002     if (len)
8003         expect_list_and_release(list, "C2.E3.E3.D1");
8004
8005     /* pi() */
8006     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("pi()"), &list));
8007     if (list)
8008     {
8009         len = 0;
8010         ole_check(IXMLDOMNodeList_get_length(list, &len));
8011         ok(len != 0, "expected filled list\n");
8012         if (len)
8013             expect_list_and_release(list, "P1.D1");
8014     }
8015
8016     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//y/pi()"), &list));
8017     len = 0;
8018     ole_check(IXMLDOMNodeList_get_length(list, &len));
8019     ok(len != 0, "expected filled list\n");
8020     if (len)
8021         expect_list_and_release(list, "P1.E4.E3.D1");
8022
8023     /* textnode() */
8024     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root/textnode()"), &list));
8025     len = 0;
8026     ole_check(IXMLDOMNodeList_get_length(list, &len));
8027     ok(len != 0, "expected filled list\n");
8028     if (len)
8029         expect_list_and_release(list, "T2.E3.D1");
8030
8031     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root/element()/textnode()"), &list));
8032     len = 0;
8033     ole_check(IXMLDOMNodeList_get_length(list, &len));
8034     ok(len != 0, "expected filled list\n");
8035     if (len)
8036         expect_list_and_release(list, "T3.E3.E3.D1 T3.E4.E3.D1");
8037
8038     IXMLDOMDocument2_Release(doc);
8039     free_bstrs();
8040 }
8041
8042 static void test_splitText(void)
8043 {
8044     IXMLDOMCDATASection *cdata;
8045     IXMLDOMElement *root;
8046     IXMLDOMDocument *doc;
8047     IXMLDOMText *text, *text2;
8048     IXMLDOMNode *node;
8049     VARIANT var;
8050     VARIANT_BOOL success;
8051     LONG length;
8052     HRESULT hr;
8053
8054     doc = create_document(&IID_IXMLDOMDocument);
8055     if (!doc) return;
8056
8057     hr = IXMLDOMDocument_loadXML(doc, _bstr_("<root></root>"), &success);
8058     ok(hr == S_OK, "got 0x%08x\n", hr);
8059
8060     hr = IXMLDOMDocument_get_documentElement(doc, &root);
8061     ok(hr == S_OK, "got 0x%08x\n", hr);
8062
8063     hr = IXMLDOMDocument_createCDATASection(doc, _bstr_("beautiful plumage"), &cdata);
8064     ok(hr == S_OK, "got 0x%08x\n", hr);
8065
8066     V_VT(&var) = VT_EMPTY;
8067     hr = IXMLDOMElement_appendChild(root, (IXMLDOMNode*)cdata, NULL);
8068     ok(hr == S_OK, "got 0x%08x\n", hr);
8069
8070     length = 0;
8071     hr = IXMLDOMCDATASection_get_length(cdata, &length);
8072     ok(hr == S_OK, "got 0x%08x\n", hr);
8073     ok(length > 0, "got %d\n", length);
8074
8075     hr = IXMLDOMCDATASection_splitText(cdata, 0, NULL);
8076     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
8077
8078     text = (void*)0xdeadbeef;
8079     /* negative offset */
8080     hr = IXMLDOMCDATASection_splitText(cdata, -1, &text);
8081     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
8082     ok(text == (void*)0xdeadbeef, "got %p\n", text);
8083
8084     text = (void*)0xdeadbeef;
8085     /* offset outside data */
8086     hr = IXMLDOMCDATASection_splitText(cdata, length + 1, &text);
8087     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
8088     ok(text == 0, "got %p\n", text);
8089
8090     text = (void*)0xdeadbeef;
8091     /* offset outside data */
8092     hr = IXMLDOMCDATASection_splitText(cdata, length, &text);
8093     ok(hr == S_FALSE, "got 0x%08x\n", hr);
8094     ok(text == 0, "got %p\n", text);
8095
8096     /* no empty node created */
8097     node = (void*)0xdeadbeef;
8098     hr = IXMLDOMCDATASection_get_nextSibling(cdata, &node);
8099     ok(hr == S_FALSE, "got 0x%08x\n", hr);
8100     ok(node == 0, "got %p\n", text);
8101
8102     hr = IXMLDOMCDATASection_splitText(cdata, 10, &text);
8103     ok(hr == S_OK, "got 0x%08x\n", hr);
8104
8105     length = 0;
8106     hr = IXMLDOMText_get_length(text, &length);
8107     ok(hr == S_OK, "got 0x%08x\n", hr);
8108     ok(length == 7, "got %d\n", length);
8109
8110     hr = IXMLDOMCDATASection_get_nextSibling(cdata, &node);
8111     ok(hr == S_OK, "got 0x%08x\n", hr);
8112     IXMLDOMNode_Release(node);
8113
8114     /* split new text node */
8115     hr = IXMLDOMText_get_length(text, &length);
8116     ok(hr == S_OK, "got 0x%08x\n", hr);
8117
8118     node = (void*)0xdeadbeef;
8119     hr = IXMLDOMText_get_nextSibling(text, &node);
8120     ok(hr == S_FALSE, "got 0x%08x\n", hr);
8121     ok(node == 0, "got %p\n", text);
8122
8123     hr = IXMLDOMText_splitText(text, 0, NULL);
8124     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
8125
8126     text2 = (void*)0xdeadbeef;
8127     /* negative offset */
8128     hr = IXMLDOMText_splitText(text, -1, &text2);
8129     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
8130     ok(text2 == (void*)0xdeadbeef, "got %p\n", text2);
8131
8132     text2 = (void*)0xdeadbeef;
8133     /* offset outside data */
8134     hr = IXMLDOMText_splitText(text, length + 1, &text2);
8135     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
8136     ok(text2 == 0, "got %p\n", text2);
8137
8138     text2 = (void*)0xdeadbeef;
8139     /* offset outside data */
8140     hr = IXMLDOMText_splitText(text, length, &text2);
8141     ok(hr == S_FALSE, "got 0x%08x\n", hr);
8142     ok(text2 == 0, "got %p\n", text);
8143
8144     text2 = 0;
8145     hr = IXMLDOMText_splitText(text, 4, &text2);
8146     todo_wine ok(hr == S_OK, "got 0x%08x\n", hr);
8147     if (text2) IXMLDOMText_Release(text2);
8148
8149     node = 0;
8150     hr = IXMLDOMText_get_nextSibling(text, &node);
8151     todo_wine ok(hr == S_OK, "got 0x%08x\n", hr);
8152     if (node) IXMLDOMNode_Release(node);
8153
8154     IXMLDOMText_Release(text);
8155     IXMLDOMElement_Release(root);
8156     IXMLDOMCDATASection_Release(cdata);
8157     free_bstrs();
8158 }
8159
8160 static void test_getQualifiedItem(void)
8161 {
8162     IXMLDOMDocument *doc;
8163     IXMLDOMElement *element;
8164     IXMLDOMNode *pr_node, *node;
8165     IXMLDOMNodeList *root_list;
8166     IXMLDOMNamedNodeMap *map;
8167     VARIANT_BOOL b;
8168     BSTR str;
8169     LONG len;
8170     HRESULT hr;
8171
8172     doc = create_document(&IID_IXMLDOMDocument);
8173     if (!doc) return;
8174
8175     str = SysAllocString( szComplete4 );
8176     hr = IXMLDOMDocument_loadXML( doc, str, &b );
8177     ok( hr == S_OK, "loadXML failed\n");
8178     ok( b == VARIANT_TRUE, "failed to load XML string\n");
8179     SysFreeString( str );
8180
8181     hr = IXMLDOMDocument_get_documentElement(doc, &element);
8182     ok( hr == S_OK, "ret %08x\n", hr);
8183
8184     hr = IXMLDOMElement_get_childNodes(element, &root_list);
8185     ok( hr == S_OK, "ret %08x\n", hr);
8186
8187     hr = IXMLDOMNodeList_get_item(root_list, 1, &pr_node);
8188     ok( hr == S_OK, "ret %08x\n", hr);
8189     IXMLDOMNodeList_Release(root_list);
8190
8191     hr = IXMLDOMNode_get_attributes(pr_node, &map);
8192     ok( hr == S_OK, "ret %08x\n", hr);
8193     IXMLDOMNode_Release(pr_node);
8194
8195     hr = IXMLDOMNamedNodeMap_get_length(map, &len);
8196     ok( hr == S_OK, "ret %08x\n", hr);
8197     ok( len == 3, "length %d\n", len);
8198
8199     hr = IXMLDOMNamedNodeMap_getQualifiedItem(map, NULL, NULL, NULL);
8200     ok( hr == E_INVALIDARG, "ret %08x\n", hr);
8201
8202     node = (void*)0xdeadbeef;
8203     hr = IXMLDOMNamedNodeMap_getQualifiedItem(map, NULL, NULL, &node);
8204     ok( hr == E_INVALIDARG, "ret %08x\n", hr);
8205     ok( node == (void*)0xdeadbeef, "got %p\n", node);
8206
8207     hr = IXMLDOMNamedNodeMap_getQualifiedItem(map, _bstr_("id"), NULL, NULL);
8208     ok( hr == E_INVALIDARG, "ret %08x\n", hr);
8209
8210     hr = IXMLDOMNamedNodeMap_getQualifiedItem(map, _bstr_("id"), NULL, &node);
8211     ok( hr == S_OK, "ret %08x\n", hr);
8212     IXMLDOMNode_Release(node);
8213
8214     IXMLDOMNamedNodeMap_Release( map );
8215     IXMLDOMElement_Release( element );
8216     IXMLDOMDocument_Release( doc );
8217     free_bstrs();
8218 }
8219
8220 static void test_removeQualifiedItem(void)
8221 {
8222     IXMLDOMDocument *doc;
8223     IXMLDOMElement *element;
8224     IXMLDOMNode *pr_node, *node;
8225     IXMLDOMNodeList *root_list;
8226     IXMLDOMNamedNodeMap *map;
8227     VARIANT_BOOL b;
8228     BSTR str;
8229     LONG len;
8230     HRESULT hr;
8231
8232     doc = create_document(&IID_IXMLDOMDocument);
8233     if (!doc) return;
8234
8235     str = SysAllocString( szComplete4 );
8236     hr = IXMLDOMDocument_loadXML( doc, str, &b );
8237     ok( hr == S_OK, "loadXML failed\n");
8238     ok( b == VARIANT_TRUE, "failed to load XML string\n");
8239     SysFreeString( str );
8240
8241     hr = IXMLDOMDocument_get_documentElement(doc, &element);
8242     ok( hr == S_OK, "ret %08x\n", hr);
8243
8244     hr = IXMLDOMElement_get_childNodes(element, &root_list);
8245     ok( hr == S_OK, "ret %08x\n", hr);
8246
8247     hr = IXMLDOMNodeList_get_item(root_list, 1, &pr_node);
8248     ok( hr == S_OK, "ret %08x\n", hr);
8249     IXMLDOMNodeList_Release(root_list);
8250
8251     hr = IXMLDOMNode_get_attributes(pr_node, &map);
8252     ok( hr == S_OK, "ret %08x\n", hr);
8253     IXMLDOMNode_Release(pr_node);
8254
8255     hr = IXMLDOMNamedNodeMap_get_length(map, &len);
8256     ok( hr == S_OK, "ret %08x\n", hr);
8257     ok( len == 3, "length %d\n", len);
8258
8259     hr = IXMLDOMNamedNodeMap_removeQualifiedItem(map, NULL, NULL, NULL);
8260     ok( hr == E_INVALIDARG, "ret %08x\n", hr);
8261
8262     node = (void*)0xdeadbeef;
8263     hr = IXMLDOMNamedNodeMap_removeQualifiedItem(map, NULL, NULL, &node);
8264     ok( hr == E_INVALIDARG, "ret %08x\n", hr);
8265     ok( node == (void*)0xdeadbeef, "got %p\n", node);
8266
8267     /* out pointer is optional */
8268     hr = IXMLDOMNamedNodeMap_removeQualifiedItem(map, _bstr_("id"), NULL, NULL);
8269     ok( hr == S_OK, "ret %08x\n", hr);
8270
8271     /* already removed */
8272     hr = IXMLDOMNamedNodeMap_removeQualifiedItem(map, _bstr_("id"), NULL, NULL);
8273     ok( hr == S_FALSE, "ret %08x\n", hr);
8274
8275     hr = IXMLDOMNamedNodeMap_removeQualifiedItem(map, _bstr_("vr"), NULL, &node);
8276     ok( hr == S_OK, "ret %08x\n", hr);
8277     IXMLDOMNode_Release(node);
8278
8279     IXMLDOMNamedNodeMap_Release( map );
8280     IXMLDOMElement_Release( element );
8281     IXMLDOMDocument_Release( doc );
8282     free_bstrs();
8283 }
8284
8285 #define check_default_props(doc) _check_default_props(__LINE__, doc)
8286 static inline void _check_default_props(int line, IXMLDOMDocument2* doc)
8287 {
8288     VARIANT_BOOL b;
8289     VARIANT var;
8290     HRESULT hr;
8291
8292     VariantInit(&var);
8293     helper_ole_check(IXMLDOMDocument2_getProperty(doc, _bstr_("SelectionLanguage"), &var));
8294     ok_(__FILE__, line)(lstrcmpW(V_BSTR(&var), _bstr_("XSLPattern")) == 0, "expected XSLPattern\n");
8295     VariantClear(&var);
8296
8297     helper_ole_check(IXMLDOMDocument2_getProperty(doc, _bstr_("SelectionNamespaces"), &var));
8298     ok_(__FILE__, line)(lstrcmpW(V_BSTR(&var), _bstr_("")) == 0, "expected empty string\n");
8299     VariantClear(&var);
8300
8301     helper_ole_check(IXMLDOMDocument2_get_preserveWhiteSpace(doc, &b));
8302     ok_(__FILE__, line)(b == VARIANT_FALSE, "expected FALSE\n");
8303
8304     hr = IXMLDOMDocument2_get_schemas(doc, &var);
8305     ok_(__FILE__, line)(hr == S_FALSE, "got %08x\n", hr);
8306     VariantClear(&var);
8307 }
8308
8309 #define check_set_props(doc) _check_set_props(__LINE__, doc)
8310 static inline void _check_set_props(int line, IXMLDOMDocument2* doc)
8311 {
8312     VARIANT_BOOL b;
8313     VARIANT var;
8314
8315     VariantInit(&var);
8316     helper_ole_check(IXMLDOMDocument2_getProperty(doc, _bstr_("SelectionLanguage"), &var));
8317     ok_(__FILE__, line)(lstrcmpW(V_BSTR(&var), _bstr_("XPath")) == 0, "expected XPath\n");
8318     VariantClear(&var);
8319
8320     helper_ole_check(IXMLDOMDocument2_getProperty(doc, _bstr_("SelectionNamespaces"), &var));
8321     ok_(__FILE__, line)(lstrcmpW(V_BSTR(&var), _bstr_("xmlns:wi=\'www.winehq.org\'")) == 0, "got %s\n", wine_dbgstr_w(V_BSTR(&var)));
8322     VariantClear(&var);
8323
8324     helper_ole_check(IXMLDOMDocument2_get_preserveWhiteSpace(doc, &b));
8325     ok_(__FILE__, line)(b == VARIANT_TRUE, "expected TRUE\n");
8326
8327     helper_ole_check(IXMLDOMDocument2_get_schemas(doc, &var));
8328     ok_(__FILE__, line)(V_VT(&var) != VT_NULL, "expected pointer\n");
8329     VariantClear(&var);
8330 }
8331
8332 #define set_props(doc, cache) _set_props(__LINE__, doc, cache)
8333 static inline void _set_props(int line, IXMLDOMDocument2* doc, IXMLDOMSchemaCollection* cache)
8334 {
8335     VARIANT var;
8336
8337     VariantInit(&var);
8338     helper_ole_check(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionLanguage"), _variantbstr_("XPath")));
8339     helper_ole_check(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionNamespaces"), _variantbstr_("xmlns:wi=\'www.winehq.org\'")));
8340     helper_ole_check(IXMLDOMDocument2_put_preserveWhiteSpace(doc, VARIANT_TRUE));
8341     V_VT(&var) = VT_DISPATCH;
8342     V_DISPATCH(&var) = NULL;
8343     helper_ole_check(IXMLDOMSchemaCollection_QueryInterface(cache, &IID_IDispatch, (void**)&V_DISPATCH(&var)));
8344     ok_(__FILE__, line)(V_DISPATCH(&var) != NULL, "expected pointer\n");
8345     helper_ole_check(IXMLDOMDocument2_putref_schemas(doc, var));
8346     VariantClear(&var);
8347 }
8348
8349 #define unset_props(doc) _unset_props(__LINE__, doc)
8350 static inline void _unset_props(int line, IXMLDOMDocument2* doc)
8351 {
8352     VARIANT var;
8353
8354     VariantInit(&var);
8355     helper_ole_check(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionLanguage"), _variantbstr_("XSLPattern")));
8356     helper_ole_check(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionNamespaces"), _variantbstr_("")));
8357     helper_ole_check(IXMLDOMDocument2_put_preserveWhiteSpace(doc, VARIANT_FALSE));
8358     V_VT(&var) = VT_NULL;
8359     helper_ole_check(IXMLDOMDocument2_putref_schemas(doc, var));
8360     VariantClear(&var);
8361 }
8362
8363 static void test_get_ownerDocument(void)
8364 {
8365     IXMLDOMDocument *doc1, *doc2, *doc3;
8366     IXMLDOMDocument2 *doc, *doc_owner;
8367     IXMLDOMNode *node;
8368     IXMLDOMSchemaCollection *cache;
8369     VARIANT_BOOL b;
8370     VARIANT var;
8371     BSTR str;
8372
8373     doc = create_document(&IID_IXMLDOMDocument2);
8374     cache = create_cache(&IID_IXMLDOMSchemaCollection);
8375     if (!doc || !cache)
8376     {
8377         if (doc) IXMLDOMDocument2_Release(doc);
8378         if (cache) IXMLDOMSchemaCollection_Release(cache);
8379         return;
8380     }
8381
8382     VariantInit(&var);
8383
8384     str = SysAllocString(szComplete4);
8385     ole_check(IXMLDOMDocument_loadXML(doc, str, &b));
8386     ok(b == VARIANT_TRUE, "failed to load XML string\n");
8387     SysFreeString(str);
8388
8389     check_default_props(doc);
8390
8391     /* set properties and check that new instances use them */
8392     set_props(doc, cache);
8393     check_set_props(doc);
8394
8395     ole_check(IXMLDOMDocument2_get_firstChild(doc, &node));
8396     ole_check(IXMLDOMNode_get_ownerDocument(node, &doc1));
8397
8398     /* new interface keeps props */
8399     ole_check(IXMLDOMDocument_QueryInterface(doc1, &IID_IXMLDOMDocument2, (void**)&doc_owner));
8400     ok( doc_owner != doc, "got %p, doc %p\n", doc_owner, doc);
8401     check_set_props(doc_owner);
8402     IXMLDOMDocument2_Release(doc_owner);
8403
8404     ole_check(IXMLDOMNode_get_ownerDocument(node, &doc2));
8405     IXMLDOMNode_Release(node);
8406
8407     ok(doc1 != doc2, "got %p, expected %p. original %p\n", doc2, doc1, doc);
8408
8409     /* reload */
8410     str = SysAllocString(szComplete4);
8411     ole_check(IXMLDOMDocument2_loadXML(doc, str, &b));
8412     ok(b == VARIANT_TRUE, "failed to load XML string\n");
8413     SysFreeString(str);
8414
8415     /* properties retained even after reload */
8416     check_set_props(doc);
8417
8418     ole_check(IXMLDOMDocument2_get_firstChild(doc, &node));
8419     ole_check(IXMLDOMNode_get_ownerDocument(node, &doc3));
8420     IXMLDOMNode_Release(node);
8421
8422     ole_check(IXMLDOMDocument_QueryInterface(doc3, &IID_IXMLDOMDocument2, (void**)&doc_owner));
8423     ok(doc3 != doc1 && doc3 != doc2 && doc_owner != doc, "got %p, (%p, %p, %p)\n", doc3, doc, doc1, doc2);
8424     check_set_props(doc_owner);
8425
8426     /* changing properties for one instance changes them for all */
8427     unset_props(doc_owner);
8428     check_default_props(doc_owner);
8429     check_default_props(doc);
8430
8431     IXMLDOMSchemaCollection_Release(cache);
8432     IXMLDOMDocument_Release(doc1);
8433     IXMLDOMDocument_Release(doc2);
8434     IXMLDOMDocument_Release(doc3);
8435     IXMLDOMDocument2_Release(doc);
8436     IXMLDOMDocument2_Release(doc_owner);
8437     free_bstrs();
8438 }
8439
8440 static void test_setAttributeNode(void)
8441 {
8442     IXMLDOMDocument *doc, *doc2;
8443     IXMLDOMElement *elem, *elem2;
8444     IXMLDOMAttribute *attr, *attr2, *ret_attr;
8445     VARIANT_BOOL b;
8446     HRESULT hr;
8447     VARIANT v;
8448     BSTR str;
8449     ULONG ref1, ref2;
8450
8451     doc = create_document(&IID_IXMLDOMDocument);
8452     if (!doc) return;
8453
8454     str = SysAllocString( szComplete4 );
8455     hr = IXMLDOMDocument2_loadXML( doc, str, &b );
8456     ok( hr == S_OK, "loadXML failed\n");
8457     ok( b == VARIANT_TRUE, "failed to load XML string\n");
8458     SysFreeString( str );
8459
8460     hr = IXMLDOMDocument_get_documentElement(doc, &elem);
8461     ok( hr == S_OK, "got 0x%08x\n", hr);
8462
8463     hr = IXMLDOMDocument_get_documentElement(doc, &elem2);
8464     ok( hr == S_OK, "got 0x%08x\n", hr);
8465     ok( elem2 != elem, "got same instance\n");
8466
8467     ret_attr = (void*)0xdeadbeef;
8468     hr = IXMLDOMElement_setAttributeNode(elem, NULL, &ret_attr);
8469     ok( hr == E_INVALIDARG, "got 0x%08x\n", hr);
8470     ok( ret_attr == (void*)0xdeadbeef, "got %p\n", ret_attr);
8471
8472     hr = IXMLDOMDocument_createAttribute(doc, _bstr_("attr"), &attr);
8473     ok( hr == S_OK, "got 0x%08x\n", hr);
8474
8475     ref1 = IXMLDOMElement_AddRef(elem);
8476     IXMLDOMElement_Release(elem);
8477
8478     ret_attr = (void*)0xdeadbeef;
8479     hr = IXMLDOMElement_setAttributeNode(elem, attr, &ret_attr);
8480     ok( hr == S_OK, "got 0x%08x\n", hr);
8481     ok( ret_attr == NULL, "got %p\n", ret_attr);
8482
8483     /* no reference added */
8484     ref2 = IXMLDOMElement_AddRef(elem);
8485     IXMLDOMElement_Release(elem);
8486     ok(ref2 == ref1, "got %d, expected %d\n", ref2, ref1);
8487
8488     EXPECT_CHILDREN(elem);
8489     EXPECT_CHILDREN(elem2);
8490
8491     IXMLDOMElement_Release(elem2);
8492
8493     attr2 = NULL;
8494     hr = IXMLDOMElement_getAttributeNode(elem, _bstr_("attr"), &attr2);
8495     ok( hr == S_OK, "got 0x%08x\n", hr);
8496     ok( attr2 != attr, "got same instance %p\n", attr2);
8497     IXMLDOMAttribute_Release(attr2);
8498
8499     /* try to add it another time */
8500     ret_attr = (void*)0xdeadbeef;
8501     hr = IXMLDOMElement_setAttributeNode(elem, attr, &ret_attr);
8502     ok( hr == E_FAIL, "got 0x%08x\n", hr);
8503     ok( ret_attr == (void*)0xdeadbeef, "got %p\n", ret_attr);
8504
8505     IXMLDOMElement_Release(elem);
8506
8507     /* initially used element is released, attribute still 'has' a container */
8508     hr = IXMLDOMDocument_get_documentElement(doc, &elem);
8509     ok( hr == S_OK, "got 0x%08x\n", hr);
8510     ret_attr = (void*)0xdeadbeef;
8511     hr = IXMLDOMElement_setAttributeNode(elem, attr, &ret_attr);
8512     ok( hr == E_FAIL, "got 0x%08x\n", hr);
8513     ok( ret_attr == (void*)0xdeadbeef, "got %p\n", ret_attr);
8514     IXMLDOMElement_Release(elem);
8515
8516     /* add attribute already attached to another document */
8517     doc2 = create_document(&IID_IXMLDOMDocument);
8518
8519     str = SysAllocString( szComplete4 );
8520     hr = IXMLDOMDocument_loadXML( doc2, str, &b );
8521     ok( hr == S_OK, "loadXML failed\n");
8522     ok( b == VARIANT_TRUE, "failed to load XML string\n");
8523     SysFreeString( str );
8524
8525     hr = IXMLDOMDocument_get_documentElement(doc2, &elem);
8526     ok( hr == S_OK, "got 0x%08x\n", hr);
8527     hr = IXMLDOMElement_setAttributeNode(elem, attr, NULL);
8528     ok( hr == E_FAIL, "got 0x%08x\n", hr);
8529     IXMLDOMElement_Release(elem);
8530
8531     IXMLDOMAttribute_Release(attr);
8532
8533     /* create element, add attribute, see if it's copied or linked */
8534     hr = IXMLDOMDocument_createElement(doc, _bstr_("test"), &elem);
8535     ok( hr == S_OK, "got 0x%08x\n", hr);
8536
8537     attr = NULL;
8538     hr = IXMLDOMDocument_createAttribute(doc, _bstr_("attr"), &attr);
8539     ok(hr == S_OK, "got 0x%08x\n", hr);
8540     ok(attr != NULL, "got %p\n", attr);
8541
8542     ref1 = IXMLDOMAttribute_AddRef(attr);
8543     IXMLDOMAttribute_Release(attr);
8544
8545     V_VT(&v) = VT_BSTR;
8546     V_BSTR(&v) = _bstr_("attrvalue1");
8547     hr = IXMLDOMAttribute_put_nodeValue(attr, v);
8548     ok( hr == S_OK, "got 0x%08x\n", hr);
8549
8550     str = NULL;
8551     hr = IXMLDOMAttribute_get_xml(attr, &str);
8552     ok( hr == S_OK, "got 0x%08x\n", hr);
8553     ok( lstrcmpW(str, _bstr_("attr=\"attrvalue1\"")) == 0,
8554         "got %s\n", wine_dbgstr_w(str));
8555     SysFreeString(str);
8556
8557     ret_attr = (void*)0xdeadbeef;
8558     hr = IXMLDOMElement_setAttributeNode(elem, attr, &ret_attr);
8559     ok(hr == S_OK, "got 0x%08x\n", hr);
8560     ok(ret_attr == NULL, "got %p\n", ret_attr);
8561
8562     /* attribute reference increased */
8563     ref2 = IXMLDOMAttribute_AddRef(attr);
8564     IXMLDOMAttribute_Release(attr);
8565     ok(ref1 == ref2, "got %d, expected %d\n", ref2, ref1);
8566
8567     hr = IXMLDOMElement_get_xml(elem, &str);
8568     ok( hr == S_OK, "got 0x%08x\n", hr);
8569     ok( lstrcmpW(str, _bstr_("<test attr=\"attrvalue1\"/>")) == 0,
8570         "got %s\n", wine_dbgstr_w(str));
8571     SysFreeString(str);
8572
8573     V_VT(&v) = VT_BSTR;
8574     V_BSTR(&v) = _bstr_("attrvalue2");
8575     hr = IXMLDOMAttribute_put_nodeValue(attr, v);
8576     ok( hr == S_OK, "got 0x%08x\n", hr);
8577
8578     hr = IXMLDOMElement_get_xml(elem, &str);
8579     ok( hr == S_OK, "got 0x%08x\n", hr);
8580     todo_wine ok( lstrcmpW(str, _bstr_("<test attr=\"attrvalue2\"/>")) == 0,
8581         "got %s\n", wine_dbgstr_w(str));
8582     SysFreeString(str);
8583
8584     IXMLDOMElement_Release(elem);
8585     IXMLDOMAttribute_Release(attr);
8586     IXMLDOMDocument_Release(doc2);
8587     IXMLDOMDocument_Release(doc);
8588 }
8589
8590 static void test_put_dataType(void)
8591 {
8592     IXMLDOMCDATASection *cdata;
8593     IXMLDOMDocument *doc;
8594     VARIANT_BOOL b;
8595     HRESULT hr;
8596     BSTR str;
8597
8598     doc = create_document(&IID_IXMLDOMDocument);
8599     if (!doc) return;
8600
8601     str = SysAllocString( szComplete4 );
8602     hr = IXMLDOMDocument_loadXML( doc, str, &b );
8603     ok( hr == S_OK, "loadXML failed\n");
8604     ok( b == VARIANT_TRUE, "failed to load XML string\n");
8605     SysFreeString( str );
8606
8607     hr = IXMLDOMDocument_createCDATASection(doc, _bstr_("test"), &cdata);
8608     ok( hr == S_OK, "got 0x%08x\n", hr);
8609     hr = IXMLDOMCDATASection_put_dataType(cdata, _bstr_("number"));
8610     ok( hr == E_FAIL, "got 0x%08x\n", hr);
8611     hr = IXMLDOMCDATASection_put_dataType(cdata, _bstr_("string"));
8612     ok( hr == E_FAIL, "got 0x%08x\n", hr);
8613     IXMLDOMCDATASection_Release(cdata);
8614
8615     IXMLDOMDocument_Release(doc);
8616     free_bstrs();
8617 }
8618
8619 static void test_createNode(void)
8620 {
8621     IXMLDOMDocument *doc;
8622     IXMLDOMElement *elem;
8623     IXMLDOMNode *node;
8624     VARIANT v, var;
8625     BSTR prefix, str;
8626     HRESULT hr;
8627     ULONG ref;
8628
8629     doc = create_document(&IID_IXMLDOMDocument);
8630     if (!doc) return;
8631
8632     EXPECT_REF(doc, 1);
8633
8634     /* reference count tests */
8635     hr = IXMLDOMDocument_createElement(doc, _bstr_("elem"), &elem);
8636     ok( hr == S_OK, "got 0x%08x\n", hr);
8637
8638     /* initial reference is 2 */
8639 todo_wine {
8640     EXPECT_REF(elem, 2);
8641     ref = IXMLDOMElement_Release(elem);
8642     ok(ref == 1, "got %d\n", ref);
8643     /* it's released already, attempt to release now will crash it */
8644 }
8645
8646     hr = IXMLDOMDocument_createElement(doc, _bstr_("elem"), &elem);
8647     ok( hr == S_OK, "got 0x%08x\n", hr);
8648     todo_wine EXPECT_REF(elem, 2);
8649     IXMLDOMDocument_Release(doc);
8650     todo_wine EXPECT_REF(elem, 2);
8651     IXMLDOMElement_Release(elem);
8652
8653     doc = create_document(&IID_IXMLDOMDocument);
8654
8655     /* NODE_ELEMENT nodes */
8656     /* 1. specified namespace */
8657     V_VT(&v) = VT_I4;
8658     V_I4(&v) = NODE_ELEMENT;
8659
8660     hr = IXMLDOMDocument_createNode(doc, v, _bstr_("ns1:test"), _bstr_("http://winehq.org"), &node);
8661     ok( hr == S_OK, "got 0x%08x\n", hr);
8662     prefix = NULL;
8663     hr = IXMLDOMNode_get_prefix(node, &prefix);
8664     ok( hr == S_OK, "got 0x%08x\n", hr);
8665     ok(lstrcmpW(prefix, _bstr_("ns1")) == 0, "wrong prefix\n");
8666     SysFreeString(prefix);
8667     IXMLDOMNode_Release(node);
8668
8669     /* 2. default namespace */
8670     hr = IXMLDOMDocument_createNode(doc, v, _bstr_("test"), _bstr_("http://winehq.org/default"), &node);
8671     ok( hr == S_OK, "got 0x%08x\n", hr);
8672     prefix = (void*)0xdeadbeef;
8673     hr = IXMLDOMNode_get_prefix(node, &prefix);
8674     ok( hr == S_FALSE, "got 0x%08x\n", hr);
8675     ok(prefix == 0, "expected empty prefix, got %p\n", prefix);
8676     /* check dump */
8677     hr = IXMLDOMNode_get_xml(node, &str);
8678     ok( hr == S_OK, "got 0x%08x\n", hr);
8679     ok( lstrcmpW(str, _bstr_("<test xmlns=\"http://winehq.org/default\"/>")) == 0,
8680         "got %s\n", wine_dbgstr_w(str));
8681     SysFreeString(str);
8682
8683     hr = IXMLDOMNode_QueryInterface(node, &IID_IXMLDOMElement, (void**)&elem);
8684     ok( hr == S_OK, "got 0x%08x\n", hr);
8685
8686     V_VT(&var) = VT_BSTR;
8687     hr = IXMLDOMElement_getAttribute(elem, _bstr_("xmlns"), &var);
8688     ok( hr == S_FALSE, "got 0x%08x\n", hr);
8689     ok( V_VT(&var) == VT_NULL, "got %d\n", V_VT(&var));
8690
8691     str = NULL;
8692     hr = IXMLDOMElement_get_namespaceURI(elem, &str);
8693     ok( hr == S_OK, "got 0x%08x\n", hr);
8694     ok( lstrcmpW(str, _bstr_("http://winehq.org/default")) == 0, "expected default namespace\n");
8695     SysFreeString(str);
8696
8697     IXMLDOMElement_Release(elem);
8698     IXMLDOMNode_Release(node);
8699
8700     IXMLDOMDocument_Release(doc);
8701     free_bstrs();
8702 }
8703
8704 static void test_get_prefix(void)
8705 {
8706     IXMLDOMDocumentFragment *fragment;
8707     IXMLDOMCDATASection *cdata;
8708     IXMLDOMElement *element;
8709     IXMLDOMComment *comment;
8710     IXMLDOMDocument *doc;
8711     HRESULT hr;
8712     BSTR str;
8713
8714     doc = create_document(&IID_IXMLDOMDocument);
8715     if (!doc) return;
8716
8717     /* nodes that can't support prefix */
8718     /* 1. document */
8719     str = (void*)0xdeadbeef;
8720     hr = IXMLDOMDocument_get_prefix(doc, &str);
8721     ok( hr == S_FALSE, "got 0x%08x\n", hr);
8722     ok( str == 0, "got %p\n", str);
8723
8724     hr = IXMLDOMDocument_get_prefix(doc, NULL);
8725     ok( hr == E_INVALIDARG, "got 0x%08x\n", hr);
8726
8727     /* 2. cdata */
8728     hr = IXMLDOMDocument_createCDATASection(doc, NULL, &cdata);
8729     ok(hr == S_OK, "got %08x\n", hr );
8730
8731     str = (void*)0xdeadbeef;
8732     hr = IXMLDOMCDATASection_get_prefix(cdata, &str);
8733     ok(hr == S_FALSE, "got %08x\n", hr);
8734     ok( str == 0, "got %p\n", str);
8735
8736     hr = IXMLDOMCDATASection_get_prefix(cdata, NULL);
8737     ok(hr == E_INVALIDARG, "got %08x\n", hr);
8738     IXMLDOMCDATASection_Release(cdata);
8739
8740     /* 3. comment */
8741     hr = IXMLDOMDocument_createComment(doc, NULL, &comment);
8742     ok(hr == S_OK, "got %08x\n", hr );
8743
8744     str = (void*)0xdeadbeef;
8745     hr = IXMLDOMComment_get_prefix(comment, &str);
8746     ok(hr == S_FALSE, "got %08x\n", hr);
8747     ok( str == 0, "got %p\n", str);
8748
8749     hr = IXMLDOMComment_get_prefix(comment, NULL);
8750     ok(hr == E_INVALIDARG, "got %08x\n", hr);
8751     IXMLDOMComment_Release(comment);
8752
8753     /* 4. fragment */
8754     hr = IXMLDOMDocument_createDocumentFragment(doc, &fragment);
8755     ok(hr == S_OK, "got %08x\n", hr );
8756
8757     str = (void*)0xdeadbeef;
8758     hr = IXMLDOMDocumentFragment_get_prefix(fragment, &str);
8759     ok(hr == S_FALSE, "got %08x\n", hr);
8760     ok( str == 0, "got %p\n", str);
8761
8762     hr = IXMLDOMDocumentFragment_get_prefix(fragment, NULL);
8763     ok(hr == E_INVALIDARG, "got %08x\n", hr);
8764     IXMLDOMDocumentFragment_Release(fragment);
8765
8766     /* no prefix */
8767     hr = IXMLDOMDocument_createElement(doc, _bstr_("elem"), &element);
8768     ok( hr == S_OK, "got 0x%08x\n", hr);
8769
8770     hr = IXMLDOMElement_get_prefix(element, NULL);
8771     ok( hr == E_INVALIDARG, "got 0x%08x\n", hr);
8772
8773     str = (void*)0xdeadbeef;
8774     hr = IXMLDOMElement_get_prefix(element, &str);
8775     ok( hr == S_FALSE, "got 0x%08x\n", hr);
8776     ok( str == 0, "got %p\n", str);
8777
8778     IXMLDOMElement_Release(element);
8779
8780     /* with prefix */
8781     hr = IXMLDOMDocument_createElement(doc, _bstr_("a:elem"), &element);
8782     ok( hr == S_OK, "got 0x%08x\n", hr);
8783
8784     str = (void*)0xdeadbeef;
8785     hr = IXMLDOMElement_get_prefix(element, &str);
8786     ok( hr == S_OK, "got 0x%08x\n", hr);
8787     ok( lstrcmpW(str, _bstr_("a")) == 0, "expected prefix \"a\"\n");
8788     SysFreeString(str);
8789
8790     str = (void*)0xdeadbeef;
8791     hr = IXMLDOMElement_get_namespaceURI(element, &str);
8792     ok( hr == S_FALSE, "got 0x%08x\n", hr);
8793     ok( str == 0, "got %p\n", str);
8794
8795     IXMLDOMElement_Release(element);
8796
8797     IXMLDOMDocument_Release(doc);
8798     free_bstrs();
8799 }
8800
8801 static void test_selectSingleNode(void)
8802 {
8803     IXMLDOMDocument *doc;
8804     IXMLDOMNodeList *list;
8805     IXMLDOMNode *node;
8806     VARIANT_BOOL b;
8807     HRESULT hr;
8808     LONG len;
8809     BSTR str;
8810
8811     doc = create_document(&IID_IXMLDOMDocument);
8812     if (!doc) return;
8813
8814     hr = IXMLDOMDocument_selectSingleNode(doc, NULL, NULL);
8815     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
8816
8817     hr = IXMLDOMDocument_selectNodes(doc, NULL, NULL);
8818     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
8819
8820     str = SysAllocString( szComplete4 );
8821     hr = IXMLDOMDocument_loadXML( doc, str, &b );
8822     ok( hr == S_OK, "loadXML failed\n");
8823     ok( b == VARIANT_TRUE, "failed to load XML string\n");
8824     SysFreeString( str );
8825
8826     hr = IXMLDOMDocument_selectSingleNode(doc, NULL, NULL);
8827     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
8828
8829     hr = IXMLDOMDocument_selectNodes(doc, NULL, NULL);
8830     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
8831
8832     hr = IXMLDOMDocument_selectSingleNode(doc, _bstr_("lc"), NULL);
8833     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
8834
8835     hr = IXMLDOMDocument_selectNodes(doc, _bstr_("lc"), NULL);
8836     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
8837
8838     hr = IXMLDOMDocument_selectSingleNode(doc, _bstr_("lc"), &node);
8839     ok(hr == S_OK, "got 0x%08x\n", hr);
8840     IXMLDOMNode_Release(node);
8841
8842     hr = IXMLDOMDocument_selectNodes(doc, _bstr_("lc"), &list);
8843     ok(hr == S_OK, "got 0x%08x\n", hr);
8844     IXMLDOMNodeList_Release(list);
8845
8846     list = (void*)0xdeadbeef;
8847     hr = IXMLDOMDocument_selectNodes(doc, NULL, &list);
8848     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
8849     ok(list == (void*)0xdeadbeef, "got %p\n", list);
8850
8851     node = (void*)0xdeadbeef;
8852     hr = IXMLDOMDocument_selectSingleNode(doc, _bstr_("nonexistent"), &node);
8853     ok(hr == S_FALSE, "got 0x%08x\n", hr);
8854     ok(node == 0, "got %p\n", node);
8855
8856     list = (void*)0xdeadbeef;
8857     hr = IXMLDOMDocument_selectNodes(doc, _bstr_("nonexistent"), &list);
8858     ok(hr == S_OK, "got 0x%08x\n", hr);
8859     len = 1;
8860     hr = IXMLDOMNodeList_get_length(list, &len);
8861     ok(hr == S_OK, "got 0x%08x\n", hr);
8862     ok(len == 0, "got %d\n", len);
8863     IXMLDOMNodeList_Release(list);
8864
8865     IXMLDOMDocument_Release(doc);
8866     free_bstrs();
8867 }
8868
8869 static void test_events(void)
8870 {
8871     IConnectionPointContainer *conn;
8872     IConnectionPoint *point;
8873     IXMLDOMDocument *doc;
8874     HRESULT hr;
8875     VARIANT v;
8876     IDispatch *event;
8877
8878     doc = create_document(&IID_IXMLDOMDocument);
8879     if (!doc) return;
8880
8881     hr = IXMLDOMDocument_QueryInterface(doc, &IID_IConnectionPointContainer, (void**)&conn);
8882     ok(hr == S_OK, "got 0x%08x\n", hr);
8883
8884     hr = IConnectionPointContainer_FindConnectionPoint(conn, &IID_IDispatch, &point);
8885     ok(hr == S_OK, "got 0x%08x\n", hr);
8886     IConnectionPoint_Release(point);
8887     hr = IConnectionPointContainer_FindConnectionPoint(conn, &IID_IPropertyNotifySink, &point);
8888     ok(hr == S_OK, "got 0x%08x\n", hr);
8889     IConnectionPoint_Release(point);
8890     hr = IConnectionPointContainer_FindConnectionPoint(conn, &DIID_XMLDOMDocumentEvents, &point);
8891     ok(hr == S_OK, "got 0x%08x\n", hr);
8892     IConnectionPoint_Release(point);
8893
8894     IConnectionPointContainer_Release(conn);
8895
8896     /* ready state callback */
8897     VariantInit(&v);
8898     hr = IXMLDOMDocument_put_onreadystatechange(doc, v);
8899     ok(hr == DISP_E_TYPEMISMATCH, "got 0x%08x\n", hr);
8900
8901     event = create_dispevent();
8902     V_VT(&v) = VT_UNKNOWN;
8903     V_UNKNOWN(&v) = (IUnknown*)event;
8904
8905     hr = IXMLDOMDocument_put_onreadystatechange(doc, v);
8906     ok(hr == S_OK, "got 0x%08x\n", hr);
8907     EXPECT_REF(event, 2);
8908
8909     V_VT(&v) = VT_DISPATCH;
8910     V_DISPATCH(&v) = event;
8911
8912     hr = IXMLDOMDocument_put_onreadystatechange(doc, v);
8913     ok(hr == S_OK, "got 0x%08x\n", hr);
8914     EXPECT_REF(event, 2);
8915
8916     /* VT_NULL doesn't reset event handler */
8917     V_VT(&v) = VT_NULL;
8918     hr = IXMLDOMDocument_put_onreadystatechange(doc, v);
8919     ok(hr == DISP_E_TYPEMISMATCH, "got 0x%08x\n", hr);
8920     EXPECT_REF(event, 2);
8921
8922     V_VT(&v) = VT_DISPATCH;
8923     V_DISPATCH(&v) = NULL;
8924
8925     hr = IXMLDOMDocument_put_onreadystatechange(doc, v);
8926     ok(hr == S_OK, "got 0x%08x\n", hr);
8927     EXPECT_REF(event, 1);
8928
8929     V_VT(&v) = VT_UNKNOWN;
8930     V_DISPATCH(&v) = NULL;
8931     hr = IXMLDOMDocument_put_onreadystatechange(doc, v);
8932     ok(hr == S_OK, "got 0x%08x\n", hr);
8933
8934     IDispatch_Release(event);
8935
8936     IXMLDOMDocument_Release(doc);
8937 }
8938
8939 static void test_createProcessingInstruction(void)
8940 {
8941     static const WCHAR bodyW[] = {'t','e','s','t',0};
8942     IXMLDOMProcessingInstruction *pi;
8943     IXMLDOMDocument *doc;
8944     WCHAR buff[10];
8945     HRESULT hr;
8946
8947     doc = create_document(&IID_IXMLDOMDocument);
8948     if (!doc) return;
8949
8950     /* test for BSTR handling, pass broken BSTR */
8951     memcpy(&buff[2], bodyW, sizeof(bodyW));
8952     /* just a big length */
8953     *(DWORD*)buff = 0xf0f0;
8954     hr = IXMLDOMDocument_createProcessingInstruction(doc, _bstr_("test"), &buff[2], &pi);
8955     ok(hr == S_OK, "got 0x%08x\n", hr);
8956
8957     IXMLDOMProcessingInstruction_Release(pi);
8958     IXMLDOMDocument_Release(doc);
8959 }
8960
8961 static void test_put_nodeTypedValue(void)
8962 {
8963     IXMLDOMDocument *doc;
8964     IXMLDOMElement *elem;
8965     VARIANT type;
8966     HRESULT hr;
8967
8968     doc = create_document(&IID_IXMLDOMDocument);
8969     if (!doc) return;
8970
8971     hr = IXMLDOMDocument_createElement(doc, _bstr_("Element"), &elem);
8972     ok(hr == S_OK, "got 0x%08x\n", hr);
8973
8974     V_VT(&type) = VT_EMPTY;
8975     hr = IXMLDOMElement_get_dataType(elem, &type);
8976     ok(hr == S_FALSE, "got 0x%08x\n", hr);
8977     ok(V_VT(&type) == VT_NULL, "got %d, expected VT_NULL\n", V_VT(&type));
8978
8979     /* set typed value for untyped node */
8980     V_VT(&type) = VT_I1;
8981     V_I1(&type) = 1;
8982     hr = IXMLDOMElement_put_nodeTypedValue(elem, type);
8983     ok(hr == S_OK, "got 0x%08x\n", hr);
8984
8985     V_VT(&type) = VT_EMPTY;
8986     hr = IXMLDOMElement_get_dataType(elem, &type);
8987     ok(hr == S_FALSE, "got 0x%08x\n", hr);
8988     ok(V_VT(&type) == VT_NULL, "got %d, expected VT_NULL\n", V_VT(&type));
8989
8990     /* no type info stored */
8991     V_VT(&type) = VT_EMPTY;
8992     hr = IXMLDOMElement_get_nodeTypedValue(elem, &type);
8993     ok(hr == S_OK, "got 0x%08x\n", hr);
8994     ok(V_VT(&type) == VT_BSTR, "got %d, expected VT_BSTR\n", V_VT(&type));
8995     ok(memcmp(V_BSTR(&type), _bstr_("1"), 2*sizeof(WCHAR)) == 0,
8996        "got %s, expected \"1\"\n", wine_dbgstr_w(V_BSTR(&type)));
8997     VariantClear(&type);
8998
8999     IXMLDOMElement_Release(elem);
9000     IXMLDOMDocument_Release(doc);
9001     free_bstrs();
9002 }
9003
9004 static void test_get_xml(void)
9005 {
9006     static const char xmlA[] = "<?xml version=\"1.0\" encoding=\"UTF-16\"?>\r\n<a>test</a>\r\n";
9007     static const char fooA[] = "<foo/>";
9008     IXMLDOMProcessingInstruction *pi;
9009     IXMLDOMNode *first;
9010     IXMLDOMElement *elem = NULL;
9011     IXMLDOMDocument *doc;
9012     VARIANT_BOOL b;
9013     VARIANT v;
9014     BSTR xml;
9015     HRESULT hr;
9016
9017     doc = create_document(&IID_IXMLDOMDocument);
9018     if (!doc) return;
9019
9020     b = VARIANT_TRUE;
9021     hr = IXMLDOMDocument_loadXML( doc, _bstr_("<a>test</a>"), &b );
9022     ok(hr == S_OK, "got 0x%08x\n", hr);
9023     ok( b == VARIANT_TRUE, "got %d\n", b);
9024
9025     hr = IXMLDOMDocument_createProcessingInstruction(doc, _bstr_("xml"),
9026                              _bstr_("version=\"1.0\" encoding=\"UTF-16\""), &pi);
9027     ok(hr == S_OK, "got 0x%08x\n", hr);
9028
9029     hr = IXMLDOMDocument_get_firstChild(doc, &first);
9030     ok(hr == S_OK, "got 0x%08x\n", hr);
9031
9032     V_UNKNOWN(&v) = (IUnknown*)first;
9033     V_VT(&v) = VT_UNKNOWN;
9034
9035     hr = IXMLDOMDocument_insertBefore(doc, (IXMLDOMNode*)pi, v, NULL);
9036     ok(hr == S_OK, "got 0x%08x\n", hr);
9037
9038     IXMLDOMProcessingInstruction_Release(pi);
9039     IXMLDOMNode_Release(first);
9040
9041     hr = IXMLDOMDocument_get_xml(doc, &xml);
9042     ok(hr == S_OK, "got 0x%08x\n", hr);
9043
9044     ok(memcmp(xml, _bstr_(xmlA), sizeof(xmlA)*sizeof(WCHAR)) == 0,
9045         "got %s, expected %s\n", wine_dbgstr_w(xml), xmlA);
9046     SysFreeString(xml);
9047
9048     IXMLDOMDocument_Release(doc);
9049
9050     doc = create_document(&IID_IXMLDOMDocument);
9051
9052     hr = IXMLDOMDocument_createElement(doc, _bstr_("foo"), &elem);
9053     ok(hr == S_OK, "got 0x%08x\n", hr);
9054
9055     hr = IXMLDOMDocument_putref_documentElement(doc, elem);
9056     ok(hr == S_OK, "got 0x%08x\n", hr);
9057
9058     hr = IXMLDOMDocument_get_xml(doc, &xml);
9059     ok(hr == S_OK, "got 0x%08x\n", hr);
9060
9061     ok(memcmp(xml, _bstr_(fooA), (sizeof(fooA)-1)*sizeof(WCHAR)) == 0,
9062         "got %s, expected %s\n", wine_dbgstr_w(xml), fooA);
9063     SysFreeString(xml);
9064
9065     IXMLDOMElement_Release(elem);
9066     IXMLDOMDocument_Release(doc);
9067
9068     free_bstrs();
9069 }
9070
9071 static void test_xsltemplate(void)
9072 {
9073     IXSLTemplate *template;
9074     IXSLProcessor *processor;
9075     IXMLDOMDocument *doc, *doc2;
9076     IStream *stream;
9077     VARIANT_BOOL b;
9078     HRESULT hr;
9079     ULONG ref1, ref2;
9080     VARIANT v;
9081
9082     template = create_xsltemplate(&IID_IXSLTemplate);
9083     if (!template) return;
9084
9085     /* works as reset */
9086     hr = IXSLTemplate_putref_stylesheet(template, NULL);
9087     ok(hr == S_OK, "got 0x%08x\n", hr);
9088
9089     doc = create_document(&IID_IXMLDOMDocument);
9090
9091     b = VARIANT_TRUE;
9092     hr = IXMLDOMDocument_loadXML( doc, _bstr_("<a>test</a>"), &b );
9093     ok(hr == S_OK, "got 0x%08x\n", hr);
9094     ok( b == VARIANT_TRUE, "got %d\n", b);
9095
9096     /* putref with non-xsl document */
9097     hr = IXSLTemplate_putref_stylesheet(template, (IXMLDOMNode*)doc);
9098     todo_wine ok(hr == E_FAIL, "got 0x%08x\n", hr);
9099
9100     b = VARIANT_TRUE;
9101     hr = IXMLDOMDocument_loadXML( doc, _bstr_(szTransformSSXML), &b );
9102     ok(hr == S_OK, "got 0x%08x\n", hr);
9103     ok( b == VARIANT_TRUE, "got %d\n", b);
9104
9105     /* not a freethreaded document */
9106     hr = IXSLTemplate_putref_stylesheet(template, (IXMLDOMNode*)doc);
9107     todo_wine ok(hr == E_FAIL, "got 0x%08x\n", hr);
9108
9109     IXMLDOMDocument_Release(doc);
9110
9111     hr = CoCreateInstance(&CLSID_FreeThreadedDOMDocument, NULL, CLSCTX_INPROC_SERVER, &IID_IXMLDOMDocument, (void**)&doc);
9112     if (hr != S_OK)
9113     {
9114         win_skip("failed to create free threaded document instance: 0x%08x\n", hr);
9115         IXSLTemplate_Release(template);
9116         return;
9117     }
9118
9119     b = VARIANT_TRUE;
9120     hr = IXMLDOMDocument_loadXML( doc, _bstr_(szTransformSSXML), &b );
9121     ok(hr == S_OK, "got 0x%08x\n", hr);
9122     ok( b == VARIANT_TRUE, "got %d\n", b);
9123
9124     /* freethreaded document */
9125     ref1 = IXMLDOMDocument_AddRef(doc);
9126     IXMLDOMDocument_Release(doc);
9127     hr = IXSLTemplate_putref_stylesheet(template, (IXMLDOMNode*)doc);
9128     ok(hr == S_OK, "got 0x%08x\n", hr);
9129     ref2 = IXMLDOMDocument_AddRef(doc);
9130     IXMLDOMDocument_Release(doc);
9131     ok(ref2 > ref1, "got %d\n", ref2);
9132
9133     /* processor */
9134     hr = IXSLTemplate_createProcessor(template, NULL);
9135     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
9136
9137     EXPECT_REF(template, 1);
9138     hr = IXSLTemplate_createProcessor(template, &processor);
9139     ok(hr == S_OK, "got 0x%08x\n", hr);
9140     EXPECT_REF(template, 2);
9141
9142     /* input no set yet */
9143     V_VT(&v) = VT_BSTR;
9144     V_BSTR(&v) = NULL;
9145     hr = IXSLProcessor_get_input(processor, &v);
9146 todo_wine {
9147     ok(hr == S_OK, "got 0x%08x\n", hr);
9148     ok(V_VT(&v) == VT_EMPTY, "got %d\n", V_VT(&v));
9149 }
9150
9151     hr = IXSLProcessor_get_output(processor, NULL);
9152     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
9153
9154     /* reset before it was set */
9155     V_VT(&v) = VT_EMPTY;
9156     hr = IXSLProcessor_put_output(processor, v);
9157     ok(hr == S_OK, "got 0x%08x\n", hr);
9158
9159     CreateStreamOnHGlobal(NULL, TRUE, &stream);
9160     EXPECT_REF(stream, 1);
9161
9162     V_VT(&v) = VT_UNKNOWN;
9163     V_UNKNOWN(&v) = (IUnknown*)stream;
9164     hr = IXSLProcessor_put_output(processor, v);
9165     ok(hr == S_OK, "got 0x%08x\n", hr);
9166
9167     /* it seems processor grabs 2 references */
9168     todo_wine EXPECT_REF(stream, 3);
9169
9170     V_VT(&v) = VT_EMPTY;
9171     hr = IXSLProcessor_get_output(processor, &v);
9172     ok(hr == S_OK, "got 0x%08x\n", hr);
9173     ok(V_VT(&v) == VT_UNKNOWN, "got type %d\n", V_VT(&v));
9174     ok(V_UNKNOWN(&v) == (IUnknown*)stream, "got %p\n", V_UNKNOWN(&v));
9175
9176     todo_wine EXPECT_REF(stream, 4);
9177     VariantClear(&v);
9178
9179     hr = IXSLProcessor_transform(processor, NULL);
9180     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
9181
9182     /* reset and check stream refcount */
9183     V_VT(&v) = VT_EMPTY;
9184     hr = IXSLProcessor_put_output(processor, v);
9185     ok(hr == S_OK, "got 0x%08x\n", hr);
9186
9187     EXPECT_REF(stream, 1);
9188
9189     IStream_Release(stream);
9190
9191     /* no output interface set, check output */
9192     doc2 = create_document(&IID_IXMLDOMDocument);
9193
9194     b = VARIANT_TRUE;
9195     hr = IXMLDOMDocument_loadXML( doc2, _bstr_("<a>test</a>"), &b );
9196     ok(hr == S_OK, "got 0x%08x\n", hr);
9197     ok( b == VARIANT_TRUE, "got %d\n", b);
9198
9199     V_VT(&v) = VT_UNKNOWN;
9200     V_UNKNOWN(&v) = (IUnknown*)doc2;
9201     hr = IXSLProcessor_put_input(processor, v);
9202     ok(hr == S_OK, "got 0x%08x\n", hr);
9203
9204     hr = IXSLProcessor_transform(processor, &b);
9205     ok(hr == S_OK, "got 0x%08x\n", hr);
9206
9207     V_VT(&v) = VT_EMPTY;
9208     hr = IXSLProcessor_get_output(processor, &v);
9209     ok(hr == S_OK, "got 0x%08x\n", hr);
9210     ok(V_VT(&v) == VT_BSTR, "got type %d\n", V_VT(&v));
9211     /* we currently output one '\n' instead of empty string */
9212     todo_wine ok(lstrcmpW(V_BSTR(&v), _bstr_("")) == 0, "got %s\n", wine_dbgstr_w(V_BSTR(&v)));
9213     IXMLDOMDocument_Release(doc2);
9214     VariantClear(&v);
9215
9216     IXSLProcessor_Release(processor);
9217
9218     /* drop reference */
9219     hr = IXSLTemplate_putref_stylesheet(template, NULL);
9220     ok(hr == S_OK, "got 0x%08x\n", hr);
9221     ref2 = IXMLDOMDocument_AddRef(doc);
9222     IXMLDOMDocument_Release(doc);
9223     ok(ref2 == ref1, "got %d\n", ref2);
9224
9225     IXMLDOMDocument_Release(doc);
9226     IXSLTemplate_Release(template);
9227     free_bstrs();
9228 }
9229
9230 static void test_insertBefore(void)
9231 {
9232     IXMLDOMDocument *doc, *doc2;
9233     IXMLDOMAttribute *attr;
9234     IXMLDOMElement *elem1, *elem2, *elem3, *elem4, *elem5;
9235     IXMLDOMNode *node, *newnode;
9236     HRESULT hr;
9237     VARIANT v;
9238     BSTR p;
9239
9240     doc = create_document(&IID_IXMLDOMDocument);
9241
9242     /* insertBefore behaviour for attribute node */
9243     V_VT(&v) = VT_I4;
9244     V_I4(&v) = NODE_ATTRIBUTE;
9245
9246     attr = NULL;
9247     hr = IXMLDOMDocument_createNode(doc, v, _bstr_("attr"), NULL, (IXMLDOMNode**)&attr);
9248     ok(hr == S_OK, "got 0x%08x\n", hr);
9249     ok(attr != NULL, "got %p\n", attr);
9250
9251     /* attribute to attribute */
9252     V_VT(&v) = VT_I4;
9253     V_I4(&v) = NODE_ATTRIBUTE;
9254     newnode = NULL;
9255     hr = IXMLDOMDocument_createNode(doc, v, _bstr_("attr2"), NULL, &newnode);
9256     ok(hr == S_OK, "got 0x%08x\n", hr);
9257     ok(newnode != NULL, "got %p\n", newnode);
9258
9259     V_VT(&v) = VT_NULL;
9260     node = (void*)0xdeadbeef;
9261     hr = IXMLDOMAttribute_insertBefore(attr, newnode, v, &node);
9262     ok(hr == E_FAIL, "got 0x%08x\n", hr);
9263     ok(node == NULL, "got %p\n", node);
9264
9265     V_VT(&v) = VT_UNKNOWN;
9266     V_UNKNOWN(&v) = (IUnknown*)attr;
9267     node = (void*)0xdeadbeef;
9268     hr = IXMLDOMAttribute_insertBefore(attr, newnode, v, &node);
9269     todo_wine ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
9270     ok(node == NULL, "got %p\n", node);
9271     IXMLDOMNode_Release(newnode);
9272
9273     /* cdata to attribute */
9274     V_VT(&v) = VT_I4;
9275     V_I4(&v) = NODE_CDATA_SECTION;
9276     newnode = NULL;
9277     hr = IXMLDOMDocument_createNode(doc, v, _bstr_("cdata"), NULL, &newnode);
9278     ok(hr == S_OK, "got 0x%08x\n", hr);
9279     ok(newnode != NULL, "got %p\n", newnode);
9280
9281     V_VT(&v) = VT_NULL;
9282     node = (void*)0xdeadbeef;
9283     hr = IXMLDOMAttribute_insertBefore(attr, newnode, v, &node);
9284     ok(hr == E_FAIL, "got 0x%08x\n", hr);
9285     ok(node == NULL, "got %p\n", node);
9286     IXMLDOMNode_Release(newnode);
9287
9288     /* comment to attribute */
9289     V_VT(&v) = VT_I4;
9290     V_I4(&v) = NODE_COMMENT;
9291     newnode = NULL;
9292     hr = IXMLDOMDocument_createNode(doc, v, _bstr_("cdata"), NULL, &newnode);
9293     ok(hr == S_OK, "got 0x%08x\n", hr);
9294     ok(newnode != NULL, "got %p\n", newnode);
9295
9296     V_VT(&v) = VT_NULL;
9297     node = (void*)0xdeadbeef;
9298     hr = IXMLDOMAttribute_insertBefore(attr, newnode, v, &node);
9299     ok(hr == E_FAIL, "got 0x%08x\n", hr);
9300     ok(node == NULL, "got %p\n", node);
9301     IXMLDOMNode_Release(newnode);
9302
9303     /* element to attribute */
9304     V_VT(&v) = VT_I4;
9305     V_I4(&v) = NODE_ELEMENT;
9306     newnode = NULL;
9307     hr = IXMLDOMDocument_createNode(doc, v, _bstr_("cdata"), NULL, &newnode);
9308     ok(hr == S_OK, "got 0x%08x\n", hr);
9309     ok(newnode != NULL, "got %p\n", newnode);
9310
9311     V_VT(&v) = VT_NULL;
9312     node = (void*)0xdeadbeef;
9313     hr = IXMLDOMAttribute_insertBefore(attr, newnode, v, &node);
9314     ok(hr == E_FAIL, "got 0x%08x\n", hr);
9315     ok(node == NULL, "got %p\n", node);
9316     IXMLDOMNode_Release(newnode);
9317
9318     /* pi to attribute */
9319     V_VT(&v) = VT_I4;
9320     V_I4(&v) = NODE_PROCESSING_INSTRUCTION;
9321     newnode = NULL;
9322     hr = IXMLDOMDocument_createNode(doc, v, _bstr_("cdata"), NULL, &newnode);
9323     ok(hr == S_OK, "got 0x%08x\n", hr);
9324     ok(newnode != NULL, "got %p\n", newnode);
9325
9326     V_VT(&v) = VT_NULL;
9327     node = (void*)0xdeadbeef;
9328     hr = IXMLDOMAttribute_insertBefore(attr, newnode, v, &node);
9329     ok(hr == E_FAIL, "got 0x%08x\n", hr);
9330     ok(node == NULL, "got %p\n", node);
9331     IXMLDOMNode_Release(newnode);
9332     IXMLDOMAttribute_Release(attr);
9333
9334     /* insertBefore for elements */
9335     hr = IXMLDOMDocument_createElement(doc, _bstr_("elem"), &elem1);
9336     ok(hr == S_OK, "got 0x%08x\n", hr);
9337
9338     hr = IXMLDOMDocument_createElement(doc, _bstr_("elem2"), &elem2);
9339     ok(hr == S_OK, "got 0x%08x\n", hr);
9340
9341     hr = IXMLDOMDocument_createElement(doc, _bstr_("elem3"), &elem3);
9342     ok(hr == S_OK, "got 0x%08x\n", hr);
9343
9344     EXPECT_NO_CHILDREN(elem1);
9345     EXPECT_NO_CHILDREN(elem2);
9346     EXPECT_NO_CHILDREN(elem3);
9347
9348     todo_wine EXPECT_REF(elem2, 2);
9349
9350     V_VT(&v) = VT_NULL;
9351     node = NULL;
9352     hr = IXMLDOMElement_insertBefore(elem1, (IXMLDOMNode*)elem2, v, &node);
9353     ok(hr == S_OK, "got 0x%08x\n", hr);
9354     ok(node == (void*)elem2, "got %p\n", node);
9355
9356     EXPECT_CHILDREN(elem1);
9357     todo_wine EXPECT_REF(elem2, 3);
9358     IXMLDOMNode_Release(node);
9359
9360     /* again for already linked node */
9361     V_VT(&v) = VT_NULL;
9362     node = NULL;
9363     hr = IXMLDOMElement_insertBefore(elem1, (IXMLDOMNode*)elem2, v, &node);
9364     ok(hr == S_OK, "got 0x%08x\n", hr);
9365     ok(node == (void*)elem2, "got %p\n", node);
9366
9367     EXPECT_CHILDREN(elem1);
9368
9369     /* increments each time */
9370     todo_wine EXPECT_REF(elem2, 3);
9371     IXMLDOMNode_Release(node);
9372
9373     /* try to add to another element */
9374     V_VT(&v) = VT_NULL;
9375     node = (void*)0xdeadbeef;
9376     hr = IXMLDOMElement_insertBefore(elem3, (IXMLDOMNode*)elem2, v, &node);
9377     ok(hr == S_OK, "got 0x%08x\n", hr);
9378     ok(node == (void*)elem2, "got %p\n", node);
9379
9380     EXPECT_CHILDREN(elem3);
9381     EXPECT_NO_CHILDREN(elem1);
9382
9383     IXMLDOMNode_Release(node);
9384
9385     /* cross document case - try to add as child to a node created with other doc */
9386     doc2 = create_document(&IID_IXMLDOMDocument);
9387
9388     hr = IXMLDOMDocument_createElement(doc2, _bstr_("elem4"), &elem4);
9389     ok(hr == S_OK, "got 0x%08x\n", hr);
9390     todo_wine EXPECT_REF(elem4, 2);
9391
9392     /* same name, another instance */
9393     hr = IXMLDOMDocument_createElement(doc2, _bstr_("elem4"), &elem5);
9394     ok(hr == S_OK, "got 0x%08x\n", hr);
9395     todo_wine EXPECT_REF(elem5, 2);
9396
9397     todo_wine EXPECT_REF(elem3, 2);
9398     V_VT(&v) = VT_NULL;
9399     node = NULL;
9400     hr = IXMLDOMElement_insertBefore(elem3, (IXMLDOMNode*)elem4, v, &node);
9401     ok(hr == S_OK, "got 0x%08x\n", hr);
9402     ok(node == (void*)elem4, "got %p\n", node);
9403     todo_wine EXPECT_REF(elem4, 3);
9404     todo_wine EXPECT_REF(elem3, 2);
9405     IXMLDOMNode_Release(node);
9406
9407     V_VT(&v) = VT_NULL;
9408     node = NULL;
9409     hr = IXMLDOMElement_insertBefore(elem3, (IXMLDOMNode*)elem5, v, &node);
9410     ok(hr == S_OK, "got 0x%08x\n", hr);
9411     ok(node == (void*)elem5, "got %p\n", node);
9412     todo_wine EXPECT_REF(elem4, 2);
9413     todo_wine EXPECT_REF(elem5, 3);
9414     IXMLDOMNode_Release(node);
9415
9416     IXMLDOMDocument_Release(doc2);
9417
9418     IXMLDOMElement_Release(elem1);
9419     IXMLDOMElement_Release(elem2);
9420     IXMLDOMElement_Release(elem3);
9421     IXMLDOMElement_Release(elem4);
9422     IXMLDOMElement_Release(elem5);
9423
9424     /* elements with same default namespace */
9425     V_VT(&v) = VT_I4;
9426     V_I4(&v) = NODE_ELEMENT;
9427     elem1 = NULL;
9428     hr = IXMLDOMDocument_createNode(doc, v, _bstr_("elem1"), _bstr_("http://winehq.org/default"), (IXMLDOMNode**)&elem1);
9429     ok(hr == S_OK, "got 0x%08x\n", hr);
9430     ok(elem1 != NULL, "got %p\n", elem1);
9431
9432     V_VT(&v) = VT_I4;
9433     V_I4(&v) = NODE_ELEMENT;
9434     elem2 = NULL;
9435     hr = IXMLDOMDocument_createNode(doc, v, _bstr_("elem2"), _bstr_("http://winehq.org/default"), (IXMLDOMNode**)&elem2);
9436     ok(hr == S_OK, "got 0x%08x\n", hr);
9437     ok(elem2 != NULL, "got %p\n", elem2);
9438
9439     /* check contents so far */
9440     p = NULL;
9441     hr = IXMLDOMElement_get_xml(elem1, &p);
9442     ok(hr == S_OK, "got 0x%08x\n", hr);
9443     ok(!lstrcmpW(p, _bstr_("<elem1 xmlns=\"http://winehq.org/default\"/>")), "got %s\n", wine_dbgstr_w(p));
9444     SysFreeString(p);
9445
9446     p = NULL;
9447     hr = IXMLDOMElement_get_xml(elem2, &p);
9448     ok(hr == S_OK, "got 0x%08x\n", hr);
9449     ok(!lstrcmpW(p, _bstr_("<elem2 xmlns=\"http://winehq.org/default\"/>")), "got %s\n", wine_dbgstr_w(p));
9450     SysFreeString(p);
9451
9452     V_VT(&v) = VT_NULL;
9453     hr = IXMLDOMElement_insertBefore(elem1, (IXMLDOMNode*)elem2, v, NULL);
9454     ok(hr == S_OK, "got 0x%08x\n", hr);
9455
9456     /* get_xml depends on context, for top node it omits child namespace attribute,
9457        but at child level it's still returned */
9458     p = NULL;
9459     hr = IXMLDOMElement_get_xml(elem1, &p);
9460     ok(hr == S_OK, "got 0x%08x\n", hr);
9461     todo_wine ok(!lstrcmpW(p, _bstr_("<elem1 xmlns=\"http://winehq.org/default\"><elem2/></elem1>")),
9462         "got %s\n", wine_dbgstr_w(p));
9463     SysFreeString(p);
9464
9465     p = NULL;
9466     hr = IXMLDOMElement_get_xml(elem2, &p);
9467     ok(hr == S_OK, "got 0x%08x\n", hr);
9468     ok(!lstrcmpW(p, _bstr_("<elem2 xmlns=\"http://winehq.org/default\"/>")), "got %s\n", wine_dbgstr_w(p));
9469     SysFreeString(p);
9470
9471     IXMLDOMElement_Release(elem1);
9472     IXMLDOMElement_Release(elem2);
9473
9474     /* child without default namespace added to node with default namespace */
9475     V_VT(&v) = VT_I4;
9476     V_I4(&v) = NODE_ELEMENT;
9477     elem1 = NULL;
9478     hr = IXMLDOMDocument_createNode(doc, v, _bstr_("elem1"), _bstr_("http://winehq.org/default"), (IXMLDOMNode**)&elem1);
9479     ok(hr == S_OK, "got 0x%08x\n", hr);
9480     ok(elem1 != NULL, "got %p\n", elem1);
9481
9482     V_VT(&v) = VT_I4;
9483     V_I4(&v) = NODE_ELEMENT;
9484     elem2 = NULL;
9485     hr = IXMLDOMDocument_createNode(doc, v, _bstr_("elem2"), NULL, (IXMLDOMNode**)&elem2);
9486     ok(hr == S_OK, "got 0x%08x\n", hr);
9487     ok(elem2 != NULL, "got %p\n", elem2);
9488
9489     EXPECT_REF(elem2, 1);
9490     V_VT(&v) = VT_NULL;
9491     hr = IXMLDOMElement_insertBefore(elem1, (IXMLDOMNode*)elem2, v, NULL);
9492     ok(hr == S_OK, "got 0x%08x\n", hr);
9493     EXPECT_REF(elem2, 1);
9494
9495     p = NULL;
9496     hr = IXMLDOMElement_get_xml(elem2, &p);
9497     ok(hr == S_OK, "got 0x%08x\n", hr);
9498     ok(!lstrcmpW(p, _bstr_("<elem2/>")), "got %s\n", wine_dbgstr_w(p));
9499     SysFreeString(p);
9500
9501     hr = IXMLDOMElement_removeChild(elem1, (IXMLDOMNode*)elem2, NULL);
9502     ok(hr == S_OK, "got 0x%08x\n", hr);
9503
9504     p = NULL;
9505     hr = IXMLDOMElement_get_xml(elem2, &p);
9506     ok(hr == S_OK, "got 0x%08x\n", hr);
9507     ok(!lstrcmpW(p, _bstr_("<elem2/>")), "got %s\n", wine_dbgstr_w(p));
9508     SysFreeString(p);
9509
9510     IXMLDOMElement_Release(elem1);
9511     IXMLDOMElement_Release(elem2);
9512     IXMLDOMDocument_Release(doc);
9513 }
9514
9515 static void test_appendChild(void)
9516 {
9517     IXMLDOMDocument *doc, *doc2;
9518     IXMLDOMElement *elem, *elem2;
9519     HRESULT hr;
9520
9521     doc = create_document(&IID_IXMLDOMDocument);
9522     doc2 = create_document(&IID_IXMLDOMDocument);
9523
9524     hr = IXMLDOMDocument_createElement(doc, _bstr_("elem"), &elem);
9525     ok(hr == S_OK, "got 0x%08x\n", hr);
9526
9527     hr = IXMLDOMDocument_createElement(doc2, _bstr_("elem2"), &elem2);
9528     ok(hr == S_OK, "got 0x%08x\n", hr);
9529
9530     EXPECT_REF(doc, 1);
9531     todo_wine EXPECT_REF(elem, 2);
9532     EXPECT_REF(doc2, 1);
9533     todo_wine EXPECT_REF(elem2, 2);
9534     EXPECT_NO_CHILDREN(doc);
9535     EXPECT_NO_CHILDREN(doc2);
9536
9537     /* append from another document */
9538     hr = IXMLDOMDocument_appendChild(doc2, (IXMLDOMNode*)elem, NULL);
9539     ok(hr == S_OK, "got 0x%08x\n", hr);
9540
9541     EXPECT_REF(doc, 1);
9542     todo_wine EXPECT_REF(elem, 2);
9543     EXPECT_REF(doc2, 1);
9544     todo_wine EXPECT_REF(elem2, 2);
9545     EXPECT_NO_CHILDREN(doc);
9546     EXPECT_CHILDREN(doc2);
9547
9548     IXMLDOMElement_Release(elem);
9549     IXMLDOMElement_Release(elem2);
9550     IXMLDOMDocument_Release(doc);
9551     IXMLDOMDocument_Release(doc2);
9552 }
9553
9554 static void test_get_doctype(void)
9555 {
9556     IXMLDOMDocumentType *doctype;
9557     IXMLDOMDocument *doc;
9558     HRESULT hr;
9559
9560     doc = create_document(&IID_IXMLDOMDocument);
9561
9562     hr = IXMLDOMDocument_get_doctype(doc, NULL);
9563     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
9564
9565     doctype = (void*)0xdeadbeef;
9566     hr = IXMLDOMDocument_get_doctype(doc, &doctype);
9567     ok(hr == S_FALSE, "got 0x%08x\n", hr);
9568     ok(doctype == NULL, "got %p\n", doctype);
9569
9570     IXMLDOMDocument_Release(doc);
9571 }
9572
9573 static void test_get_tagName(void)
9574 {
9575     IXMLDOMDocument *doc;
9576     IXMLDOMElement *elem, *elem2;
9577     HRESULT hr;
9578     BSTR str;
9579
9580     doc = create_document(&IID_IXMLDOMDocument);
9581
9582     hr = IXMLDOMDocument_createElement(doc, _bstr_("element"), &elem);
9583     ok(hr == S_OK, "got 0x%08x\n", hr);
9584
9585     hr = IXMLDOMElement_get_tagName(elem, NULL);
9586     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
9587
9588     str = NULL;
9589     hr = IXMLDOMElement_get_tagName(elem, &str);
9590     ok(hr == S_OK, "got 0x%08x\n", hr);
9591     ok(!lstrcmpW(str, _bstr_("element")), "got %s\n", wine_dbgstr_w(str));
9592     SysFreeString(str);
9593
9594     hr = IXMLDOMDocument_createElement(doc, _bstr_("s:element"), &elem2);
9595     ok(hr == S_OK, "got 0x%08x\n", hr);
9596
9597     str = NULL;
9598     hr = IXMLDOMElement_get_tagName(elem2, &str);
9599     ok(hr == S_OK, "got 0x%08x\n", hr);
9600     ok(!lstrcmpW(str, _bstr_("s:element")), "got %s\n", wine_dbgstr_w(str));
9601     SysFreeString(str);
9602
9603     IXMLDOMDocument_Release(elem);
9604     IXMLDOMDocument_Release(elem2);
9605     IXMLDOMDocument_Release(doc);
9606     free_bstrs();
9607 }
9608
9609 typedef struct _get_datatype_t {
9610     DOMNodeType type;
9611     const char *name;
9612     VARTYPE vt;
9613     HRESULT hr;
9614 } get_datatype_t;
9615
9616 static const get_datatype_t get_datatype[] = {
9617     { NODE_ELEMENT,                "element",   VT_NULL, S_FALSE },
9618     { NODE_ATTRIBUTE,              "attr",      VT_NULL, S_FALSE },
9619     { NODE_TEXT,                   "text",      VT_NULL, S_FALSE },
9620     { NODE_CDATA_SECTION ,         "cdata",     VT_NULL, S_FALSE },
9621     { NODE_ENTITY_REFERENCE,       "entityref", VT_NULL, S_FALSE },
9622     { NODE_PROCESSING_INSTRUCTION, "pi",        VT_NULL, S_FALSE },
9623     { NODE_COMMENT,                "comment",   VT_NULL, S_FALSE },
9624     { NODE_DOCUMENT_FRAGMENT,      "docfrag",   VT_NULL, S_FALSE },
9625     { 0 }
9626 };
9627
9628 static void test_get_dataType(void)
9629 {
9630     IXMLDOMDocument *doc;
9631     const get_datatype_t *entry = get_datatype;
9632
9633     doc = create_document(&IID_IXMLDOMDocument);
9634
9635     while (entry->type)
9636     {
9637         IXMLDOMNode *node = NULL;
9638         VARIANT var, type;
9639         HRESULT hr;
9640
9641         V_VT(&var) = VT_I4;
9642         V_I4(&var) = entry->type;
9643         hr = IXMLDOMDocument_createNode(doc, var, _bstr_(entry->name), NULL, &node);
9644         ok(hr == S_OK, "failed to create node, type %d\n", entry->type);
9645
9646         hr = IXMLDOMNode_get_dataType(node, NULL);
9647         ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
9648
9649         VariantInit(&type);
9650         hr = IXMLDOMNode_get_dataType(node, &type);
9651         ok(hr == entry->hr, "got 0x%08x, expected 0x%08x. node type %d\n",
9652             hr, entry->hr, entry->type);
9653         ok(V_VT(&type) == entry->vt, "got %d, expected %d. node type %d\n",
9654             V_VT(&type), entry->vt, entry->type);
9655         VariantClear(&type);
9656
9657         IXMLDOMNode_Release(node);
9658
9659         entry++;
9660     }
9661
9662     IXMLDOMDocument_Release(doc);
9663     free_bstrs();
9664 }
9665
9666 typedef struct _get_node_typestring_t {
9667     DOMNodeType type;
9668     const char *string;
9669 } get_node_typestring_t;
9670
9671 static const get_node_typestring_t get_node_typestring[] = {
9672     { NODE_ELEMENT,                "element"               },
9673     { NODE_ATTRIBUTE,              "attribute"             },
9674     { NODE_TEXT,                   "text"                  },
9675     { NODE_CDATA_SECTION ,         "cdatasection"          },
9676     { NODE_ENTITY_REFERENCE,       "entityreference"       },
9677     { NODE_PROCESSING_INSTRUCTION, "processinginstruction" },
9678     { NODE_COMMENT,                "comment"               },
9679     { NODE_DOCUMENT_FRAGMENT,      "documentfragment"      },
9680     { 0 }
9681 };
9682
9683 static void test_get_nodeTypeString(void)
9684 {
9685     const get_node_typestring_t *entry = get_node_typestring;
9686     IXMLDOMDocument *doc;
9687     HRESULT hr;
9688     BSTR str;
9689
9690     doc = create_document(&IID_IXMLDOMDocument);
9691
9692     hr = IXMLDOMDocument_get_nodeTypeString(doc, &str);
9693     ok(hr == S_OK, "got 0x%08x\n", hr);
9694     ok(!lstrcmpW(str, _bstr_("document")), "got string %s\n", wine_dbgstr_w(str));
9695     SysFreeString(str);
9696
9697     while (entry->type)
9698     {
9699         IXMLDOMNode *node = NULL;
9700         VARIANT var;
9701
9702         V_VT(&var) = VT_I4;
9703         V_I4(&var) = entry->type;
9704         hr = IXMLDOMDocument_createNode(doc, var, _bstr_("node"), NULL, &node);
9705         ok(hr == S_OK, "failed to create node, type %d\n", entry->type);
9706
9707         hr = IXMLDOMNode_get_nodeTypeString(node, NULL);
9708         ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
9709
9710         hr = IXMLDOMNode_get_nodeTypeString(node, &str);
9711         ok(hr == S_OK, "got 0x%08x\n", hr);
9712         ok(!lstrcmpW(str, _bstr_(entry->string)), "got string %s, expected %s. node type %d\n",
9713             wine_dbgstr_w(str), entry->string, entry->type);
9714         SysFreeString(str);
9715         IXMLDOMNode_Release(node);
9716
9717         entry++;
9718     }
9719
9720     IXMLDOMDocument_Release(doc);
9721     free_bstrs();
9722 }
9723
9724 typedef struct _get_attributes_t {
9725     DOMNodeType type;
9726     HRESULT hr;
9727 } get_attributes_t;
9728
9729 static const get_attributes_t get_attributes[] = {
9730     { NODE_ATTRIBUTE,              S_FALSE },
9731     { NODE_TEXT,                   S_FALSE },
9732     { NODE_CDATA_SECTION ,         S_FALSE },
9733     { NODE_ENTITY_REFERENCE,       S_FALSE },
9734     { NODE_PROCESSING_INSTRUCTION, S_FALSE },
9735     { NODE_COMMENT,                S_FALSE },
9736     { NODE_DOCUMENT_FRAGMENT,      S_FALSE },
9737     { 0 }
9738 };
9739
9740 static void test_get_attributes(void)
9741 {
9742     const get_attributes_t *entry = get_attributes;
9743     IXMLDOMNamedNodeMap *map;
9744     IXMLDOMDocument *doc;
9745     IXMLDOMNode *node, *node2;
9746     VARIANT_BOOL b;
9747     HRESULT hr;
9748     BSTR str;
9749     LONG length;
9750
9751     doc = create_document(&IID_IXMLDOMDocument);
9752
9753     str = SysAllocString( szComplete4 );
9754     hr = IXMLDOMDocument_loadXML(doc, str, &b);
9755     ok(hr == S_OK, "got %08x\n", hr);
9756     SysFreeString(str);
9757
9758     hr = IXMLDOMDocument_get_attributes(doc, NULL);
9759     ok(hr == E_INVALIDARG, "got %08x\n", hr);
9760
9761     map = (void*)0xdeadbeef;
9762     hr = IXMLDOMDocument_get_attributes(doc, &map);
9763     ok(hr == S_FALSE, "got %08x\n", hr);
9764     ok(map == NULL, "got %p\n", map);
9765
9766     /* first child is <?xml ?> */
9767     hr = IXMLDOMDocument_get_firstChild(doc, &node);
9768     ok(hr == S_OK, "got %08x\n", hr);
9769
9770     hr = IXMLDOMNode_get_attributes(node, &map);
9771     ok(hr == S_OK, "got %08x\n", hr);
9772
9773     length = -1;
9774     hr = IXMLDOMNamedNodeMap_get_length(map, &length);
9775     EXPECT_HR(hr, S_OK);
9776     todo_wine ok(length == 1, "got %d\n", length);
9777
9778     if (hr == S_OK && length == 1)
9779     {
9780         IXMLDOMAttribute *attr;
9781         DOMNodeType type;
9782         VARIANT v;
9783
9784         node2 = NULL;
9785         hr = IXMLDOMNamedNodeMap_get_item(map, 0, &node2);
9786         EXPECT_HR(hr, S_OK);
9787         ok(node != NULL, "got %p\n", node2);
9788
9789         hr = IXMLDOMNode_get_nodeName(node2, &str);
9790         EXPECT_HR(hr, S_OK);
9791         ok(!lstrcmpW(str, _bstr_("version")), "got %s\n", wine_dbgstr_w(str));
9792         SysFreeString(str);
9793
9794         length = -1;
9795         hr = IXMLDOMNamedNodeMap_get_length(map, &length);
9796         EXPECT_HR(hr, S_OK);
9797         ok(length == 1, "got %d\n", length);
9798
9799         type = -1;
9800         hr = IXMLDOMNode_get_nodeType(node2, &type);
9801         EXPECT_HR(hr, S_OK);
9802         ok(type == NODE_ATTRIBUTE, "got %d\n", type);
9803
9804         hr = IXMLDOMNode_get_xml(node, &str);
9805         EXPECT_HR(hr, S_OK);
9806         ok(!lstrcmpW(str, _bstr_("<?xml version=\"1.0\"?>")), "got %s\n", wine_dbgstr_w(str));
9807         SysFreeString(str);
9808
9809         hr = IXMLDOMNode_get_text(node, &str);
9810         EXPECT_HR(hr, S_OK);
9811         ok(!lstrcmpW(str, _bstr_("version=\"1.0\"")), "got %s\n", wine_dbgstr_w(str));
9812         SysFreeString(str);
9813
9814         hr = IXMLDOMNamedNodeMap_removeNamedItem(map, _bstr_("version"), NULL);
9815         EXPECT_HR(hr, S_OK);
9816
9817         length = -1;
9818         hr = IXMLDOMNamedNodeMap_get_length(map, &length);
9819         EXPECT_HR(hr, S_OK);
9820         ok(length == 0, "got %d\n", length);
9821
9822         hr = IXMLDOMNode_get_xml(node, &str);
9823         EXPECT_HR(hr, S_OK);
9824         ok(!lstrcmpW(str, _bstr_("<?xml version=\"1.0\"?>")), "got %s\n", wine_dbgstr_w(str));
9825         SysFreeString(str);
9826
9827         hr = IXMLDOMNode_get_text(node, &str);
9828         EXPECT_HR(hr, S_OK);
9829         ok(!lstrcmpW(str, _bstr_("")), "got %s\n", wine_dbgstr_w(str));
9830         SysFreeString(str);
9831
9832         IXMLDOMNamedNodeMap_Release(map);
9833
9834         hr = IXMLDOMNode_get_attributes(node, &map);
9835         ok(hr == S_OK, "got %08x\n", hr);
9836
9837         length = -1;
9838         hr = IXMLDOMNamedNodeMap_get_length(map, &length);
9839         EXPECT_HR(hr, S_OK);
9840         ok(length == 0, "got %d\n", length);
9841
9842         hr = IXMLDOMDocument_createAttribute(doc, _bstr_("encoding"), &attr);
9843         EXPECT_HR(hr, S_OK);
9844
9845         V_VT(&v) = VT_BSTR;
9846         V_BSTR(&v) = _bstr_("UTF-8");
9847         hr = IXMLDOMAttribute_put_nodeValue(attr, v);
9848         EXPECT_HR(hr, S_OK);
9849
9850         EXPECT_REF(attr, 2);
9851         hr = IXMLDOMNamedNodeMap_setNamedItem(map, (IXMLDOMNode*)attr, NULL);
9852         EXPECT_HR(hr, S_OK);
9853         EXPECT_REF(attr, 2);
9854
9855         hr = IXMLDOMNode_get_attributes(node, &map);
9856         ok(hr == S_OK, "got %08x\n", hr);
9857
9858         length = -1;
9859         hr = IXMLDOMNamedNodeMap_get_length(map, &length);
9860         EXPECT_HR(hr, S_OK);
9861         ok(length == 1, "got %d\n", length);
9862
9863         hr = IXMLDOMNode_get_xml(node, &str);
9864         EXPECT_HR(hr, S_OK);
9865         ok(!lstrcmpW(str, _bstr_("<?xml version=\"1.0\"?>")), "got %s\n", wine_dbgstr_w(str));
9866         SysFreeString(str);
9867
9868         hr = IXMLDOMNode_get_text(node, &str);
9869         EXPECT_HR(hr, S_OK);
9870         ok(!lstrcmpW(str, _bstr_("encoding=\"UTF-8\"")), "got %s\n", wine_dbgstr_w(str));
9871         SysFreeString(str);
9872
9873         IXMLDOMNamedNodeMap_Release(map);
9874         IXMLDOMNode_Release(node2);
9875     }
9876
9877     IXMLDOMNode_Release(node);
9878
9879     /* last child is element */
9880     EXPECT_REF(doc, 1);
9881     hr = IXMLDOMDocument_get_lastChild(doc, &node);
9882     ok(hr == S_OK, "got %08x\n", hr);
9883     EXPECT_REF(doc, 1);
9884
9885     EXPECT_REF(node, 1);
9886     hr = IXMLDOMNode_get_attributes(node, &map);
9887     ok(hr == S_OK, "got %08x\n", hr);
9888     EXPECT_REF(node, 1);
9889     EXPECT_REF(doc, 1);
9890
9891     EXPECT_REF(map, 1);
9892     hr = IXMLDOMNamedNodeMap_get_item(map, 0, &node2);
9893     ok(hr == S_OK, "got %08x\n", hr);
9894     EXPECT_REF(node, 1);
9895     EXPECT_REF(node2, 1);
9896     EXPECT_REF(map, 1);
9897     EXPECT_REF(doc, 1);
9898     IXMLDOMNode_Release(node2);
9899
9900     /* release node before map release, map still works */
9901     IXMLDOMNode_Release(node);
9902
9903     length = 0;
9904     hr = IXMLDOMNamedNodeMap_get_length(map, &length);
9905     ok(hr == S_OK, "got %08x\n", hr);
9906     ok(length == 1, "got %d\n", length);
9907
9908     node2 = NULL;
9909     hr = IXMLDOMNamedNodeMap_get_item(map, 0, &node2);
9910     ok(hr == S_OK, "got %08x\n", hr);
9911     EXPECT_REF(node2, 1);
9912     IXMLDOMNode_Release(node2);
9913
9914     IXMLDOMNamedNodeMap_Release(map);
9915
9916     while (entry->type)
9917     {
9918         VARIANT var;
9919
9920         node = NULL;
9921
9922         V_VT(&var) = VT_I4;
9923         V_I4(&var) = entry->type;
9924         hr = IXMLDOMDocument_createNode(doc, var, _bstr_("node"), NULL, &node);
9925         ok(hr == S_OK, "failed to create node, type %d\n", entry->type);
9926
9927         hr = IXMLDOMNode_get_attributes(node, NULL);
9928         ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
9929
9930         map = (void*)0xdeadbeef;
9931         hr = IXMLDOMNode_get_attributes(node, &map);
9932         ok(hr == entry->hr, "got 0x%08x, expected 0x%08x. node type %d\n",
9933             hr, entry->hr, entry->type);
9934         ok(map == NULL, "got %p\n", map);
9935
9936         IXMLDOMNode_Release(node);
9937
9938         entry++;
9939     }
9940
9941     IXMLDOMDocument_Release(doc);
9942     free_bstrs();
9943 }
9944
9945 static void test_selection(void)
9946 {
9947     IXMLDOMSelection *selection;
9948     IXMLDOMNodeList *list;
9949     IXMLDOMDocument *doc;
9950     VARIANT_BOOL b;
9951     HRESULT hr;
9952
9953     doc = create_document(&IID_IXMLDOMDocument);
9954
9955     hr = IXMLDOMDocument2_loadXML(doc, _bstr_(szExampleXML), &b);
9956     EXPECT_HR(hr, S_OK);
9957
9958     hr = IXMLDOMDocument_selectNodes(doc, _bstr_("root"), &list);
9959     EXPECT_HR(hr, S_OK);
9960
9961     hr = IXMLDOMNodeList_QueryInterface(list, &IID_IXMLDOMSelection, (void**)&selection);
9962     EXPECT_HR(hr, S_OK);
9963     IXMLDOMSelection_Release(selection);
9964
9965     IXMLDOMNodeList_Release(list);
9966
9967     hr = IXMLDOMDocument_get_childNodes(doc, &list);
9968     EXPECT_HR(hr, S_OK);
9969
9970     hr = IXMLDOMNodeList_QueryInterface(list, &IID_IXMLDOMSelection, (void**)&selection);
9971     EXPECT_HR(hr, E_NOINTERFACE);
9972
9973     IXMLDOMNodeList_Release(list);
9974
9975     IXMLDOMDocument_Release(doc);
9976     free_bstrs();
9977 }
9978
9979 static void test_load(void)
9980 {
9981     IXMLDOMDocument *doc;
9982     IXMLDOMNodeList *list;
9983     VARIANT_BOOL b;
9984     HANDLE hfile;
9985     VARIANT src;
9986     HRESULT hr;
9987     BOOL ret;
9988     BSTR path, bstr1, bstr2;
9989     DWORD written;
9990     void* ptr;
9991
9992     /* prepare a file */
9993     hfile = CreateFileA("test.xml", GENERIC_WRITE|GENERIC_READ, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL );
9994     ok(hfile != INVALID_HANDLE_VALUE, "failed to create test file\n");
9995     if(hfile == INVALID_HANDLE_VALUE) return;
9996
9997     ret = WriteFile(hfile, szNonUnicodeXML, sizeof(szNonUnicodeXML)-1, &written, NULL);
9998     ok(ret, "WriteFile failed\n");
9999
10000     CloseHandle(hfile);
10001
10002     doc = create_document(&IID_IXMLDOMDocument);
10003
10004     path = _bstr_("test.xml");
10005
10006     /* load from path: VT_BSTR */
10007     V_VT(&src) = VT_BSTR;
10008     V_BSTR(&src) = path;
10009     hr = IXMLDOMDocument_load(doc, src, &b);
10010     EXPECT_HR(hr, S_OK);
10011     ok(b == VARIANT_TRUE, "got %d\n", b);
10012
10013     /* load from a path: VT_BSTR|VT_BYREF */
10014     V_VT(&src) = VT_BSTR | VT_BYREF;
10015     V_BSTRREF(&src) = &path;
10016     hr = IXMLDOMDocument_load(doc, src, &b);
10017     EXPECT_HR(hr, S_OK);
10018     ok(b == VARIANT_TRUE, "got %d\n", b);
10019
10020     /* load from a path: VT_BSTR|VT_BYREF, null ptr */
10021     V_VT(&src) = VT_BSTR | VT_BYREF;
10022     V_BSTRREF(&src) = NULL;
10023     hr = IXMLDOMDocument_load(doc, src, &b);
10024     EXPECT_HR(hr, E_INVALIDARG);
10025     ok(b == VARIANT_FALSE, "got %d\n", b);
10026
10027     IXMLDOMDocument_Release(doc);
10028
10029     DeleteFileA("test.xml");
10030
10031     doc = create_document(&IID_IXMLDOMDocument);
10032
10033     hr = IXMLDOMDocument_loadXML(doc, _bstr_(szExampleXML), &b);
10034     EXPECT_HR(hr, S_OK);
10035     ok(b == VARIANT_TRUE, "got %d\n", b);
10036
10037     hr = IXMLDOMDocument_selectNodes(doc, _bstr_("//*"), &list);
10038     EXPECT_HR(hr, S_OK);
10039     bstr1 = _bstr_(list_to_string(list));
10040
10041     hr = IXMLDOMNodeList_reset(list);
10042     EXPECT_HR(hr, S_OK);
10043
10044     IXMLDOMDocument_Release(doc);
10045
10046     doc = create_document(&IID_IXMLDOMDocument);
10047
10048     VariantInit(&src);
10049     V_ARRAY(&src) = SafeArrayCreateVector(VT_UI1, 0, lstrlenA(szExampleXML));
10050     V_VT(&src) = VT_ARRAY|VT_UI1;
10051     ok(V_ARRAY(&src) != NULL, "SafeArrayCreateVector() returned NULL\n");
10052     ptr = NULL;
10053     hr = SafeArrayAccessData(V_ARRAY(&src), &ptr);
10054     EXPECT_HR(hr, S_OK);
10055     ok(ptr != NULL, "SafeArrayAccessData() returned NULL\n");
10056
10057     memcpy(ptr, szExampleXML, lstrlenA(szExampleXML));
10058     hr = SafeArrayUnlock(V_ARRAY(&src));
10059     EXPECT_HR(hr, S_OK);
10060
10061     hr = IXMLDOMDocument_load(doc, src, &b);
10062     EXPECT_HR(hr, S_OK);
10063     ok(b == VARIANT_TRUE, "got %d\n", b);
10064
10065     hr = IXMLDOMDocument_selectNodes(doc, _bstr_("//*"), &list);
10066     EXPECT_HR(hr, S_OK);
10067     bstr2 = _bstr_(list_to_string(list));
10068
10069     hr = IXMLDOMNodeList_reset(list);
10070     EXPECT_HR(hr, S_OK);
10071
10072     ok(lstrcmpW(bstr1, bstr2) == 0, "strings not equal: %s : %s\n",
10073        wine_dbgstr_w(bstr1), wine_dbgstr_w(bstr2));
10074
10075     IXMLDOMDocument_Release(doc);
10076     IXMLDOMNodeList_Release(list);
10077     VariantClear(&src);
10078
10079     /* UTF-16 isn't accepted */
10080     doc = create_document(&IID_IXMLDOMDocument);
10081
10082     V_ARRAY(&src) = SafeArrayCreateVector(VT_UI1, 0, lstrlenW(szComplete1) * sizeof(WCHAR));
10083     V_VT(&src) = VT_ARRAY|VT_UI1;
10084     ok(V_ARRAY(&src) != NULL, "SafeArrayCreateVector() returned NULL\n");
10085     ptr = NULL;
10086     hr = SafeArrayAccessData(V_ARRAY(&src), &ptr);
10087     EXPECT_HR(hr, S_OK);
10088     ok(ptr != NULL, "SafeArrayAccessData() returned NULL\n");
10089
10090     memcpy(ptr, szComplete1, lstrlenW(szComplete1) * sizeof(WCHAR));
10091     hr = SafeArrayUnlock(V_ARRAY(&src));
10092     EXPECT_HR(hr, S_OK);
10093
10094     hr = IXMLDOMDocument_load(doc, src, &b);
10095     todo_wine EXPECT_HR(hr, S_FALSE);
10096     todo_wine ok(b == VARIANT_FALSE, "got %d\n", b);
10097
10098     VariantClear(&src);
10099
10100     /* it doesn't like it as a VT_ARRAY|VT_UI2 either */
10101     V_ARRAY(&src) = SafeArrayCreateVector(VT_UI2, 0, lstrlenW(szComplete1));
10102     V_VT(&src) = VT_ARRAY|VT_UI2;
10103     ok(V_ARRAY(&src) != NULL, "SafeArrayCreateVector() returned NULL\n");
10104     ptr = NULL;
10105     hr = SafeArrayAccessData(V_ARRAY(&src), &ptr);
10106     EXPECT_HR(hr, S_OK);
10107     ok(ptr != NULL, "SafeArrayAccessData() returned NULL\n");
10108
10109     memcpy(ptr, szComplete1, lstrlenW(szComplete1) * sizeof(WCHAR));
10110     hr = SafeArrayUnlock(V_ARRAY(&src));
10111     EXPECT_HR(hr, S_OK);
10112
10113     hr = IXMLDOMDocument_load(doc, src, &b);
10114     todo_wine EXPECT_HR(hr, E_INVALIDARG);
10115     ok(b == VARIANT_FALSE, "got %d\n", b);
10116
10117     VariantClear(&src);
10118     IXMLDOMDocument_Release(doc);
10119
10120     free_bstrs();
10121 }
10122
10123 static void test_nsnamespacemanager(void)
10124 {
10125     static const char xmluriA[] = "http://www.w3.org/XML/1998/namespace";
10126     IMXNamespaceManager *nsmgr;
10127     IVBMXNamespaceManager *mgr2;
10128     IDispatch *disp;
10129     HRESULT hr;
10130     WCHAR buffW[250];
10131     INT len;
10132
10133     hr = CoCreateInstance(&CLSID_MXNamespaceManager40, NULL, CLSCTX_INPROC_SERVER,
10134         &IID_IMXNamespaceManager, (void**)&nsmgr);
10135     if (hr != S_OK)
10136     {
10137         win_skip("MXNamespaceManager is not available\n");
10138         return;
10139     }
10140
10141     /* IMXNamespaceManager inherits from IUnknown */
10142     hr = IMXNamespaceManager_QueryInterface(nsmgr, &IID_IDispatch, (void**)&disp);
10143     EXPECT_HR(hr, S_OK);
10144     IDispatch_Release(disp);
10145
10146     hr = IMXNamespaceManager_QueryInterface(nsmgr, &IID_IVBMXNamespaceManager, (void**)&mgr2);
10147     EXPECT_HR(hr, S_OK);
10148     IVBMXNamespaceManager_Release(mgr2);
10149
10150     hr = IMXNamespaceManager_declarePrefix(nsmgr, NULL, NULL);
10151     EXPECT_HR(hr, S_OK);
10152
10153     /* prefix already added */
10154     hr = IMXNamespaceManager_declarePrefix(nsmgr, NULL, _bstr_("ns0 uri"));
10155     EXPECT_HR(hr, S_FALSE);
10156
10157     hr = IMXNamespaceManager_declarePrefix(nsmgr, _bstr_("ns0"), NULL);
10158     EXPECT_HR(hr, E_INVALIDARG);
10159
10160     /* "xml" and "xmlns" are not allowed here */
10161     hr = IMXNamespaceManager_declarePrefix(nsmgr, _bstr_("xml"), _bstr_("uri1"));
10162     EXPECT_HR(hr, E_INVALIDARG);
10163
10164     hr = IMXNamespaceManager_declarePrefix(nsmgr, _bstr_("xmlns"), _bstr_("uri1"));
10165     EXPECT_HR(hr, E_INVALIDARG);
10166 todo_wine {
10167     hr = IMXNamespaceManager_getDeclaredPrefix(nsmgr, -1, NULL, NULL);
10168     EXPECT_HR(hr, E_FAIL);
10169 }
10170     hr = IMXNamespaceManager_getDeclaredPrefix(nsmgr, 0, NULL, NULL);
10171     EXPECT_HR(hr, E_POINTER);
10172
10173     len = -1;
10174     hr = IMXNamespaceManager_getDeclaredPrefix(nsmgr, 0, NULL, &len);
10175     EXPECT_HR(hr, S_OK);
10176     ok(len == 3, "got %d\n", len);
10177
10178     len = -1;
10179     buffW[0] = 0x1;
10180     hr = IMXNamespaceManager_getDeclaredPrefix(nsmgr, 0, buffW, &len);
10181     EXPECT_HR(hr, E_XML_BUFFERTOOSMALL);
10182     ok(len == -1, "got %d\n", len);
10183     ok(buffW[0] == 0x1, "got %x\n", buffW[0]);
10184
10185     len = 10;
10186     buffW[0] = 0x1;
10187     hr = IMXNamespaceManager_getDeclaredPrefix(nsmgr, 0, buffW, &len);
10188     EXPECT_HR(hr, S_OK);
10189     ok(len == 3, "got %d\n", len);
10190     ok(!lstrcmpW(buffW, _bstr_("xml")), "got prefix %s\n", wine_dbgstr_w(buffW));
10191
10192     /* getURI */
10193     hr = IMXNamespaceManager_getURI(nsmgr, NULL, NULL, NULL, NULL);
10194     EXPECT_HR(hr, E_INVALIDARG);
10195
10196     len = -1;
10197     hr = IMXNamespaceManager_getURI(nsmgr, NULL, NULL, NULL, &len);
10198     EXPECT_HR(hr, E_INVALIDARG);
10199     ok(len == -1, "got %d\n", len);
10200
10201     hr = IMXNamespaceManager_getURI(nsmgr, _bstr_("xml"), NULL, NULL, NULL);
10202     EXPECT_HR(hr, E_POINTER);
10203
10204     len = -1;
10205     hr = IMXNamespaceManager_getURI(nsmgr, _bstr_("xml"), NULL, NULL, &len);
10206     EXPECT_HR(hr, S_OK);
10207     /* length of "xml" uri is constant */
10208     ok(len == strlen(xmluriA), "got %d\n", len);
10209
10210     len = 100;
10211     hr = IMXNamespaceManager_getURI(nsmgr, _bstr_("xml"), NULL, buffW, &len);
10212     EXPECT_HR(hr, S_OK);
10213     ok(len == strlen(xmluriA), "got %d\n", len);
10214     ok(!lstrcmpW(buffW, _bstr_(xmluriA)), "got prefix %s\n", wine_dbgstr_w(buffW));
10215
10216     len = strlen(xmluriA)-1;
10217     buffW[0] = 0x1;
10218     hr = IMXNamespaceManager_getURI(nsmgr, _bstr_("xml"), NULL, buffW, &len);
10219     EXPECT_HR(hr, E_XML_BUFFERTOOSMALL);
10220     ok(len == strlen(xmluriA)-1, "got %d\n", len);
10221     ok(buffW[0] == 0x1, "got %x\n", buffW[0]);
10222
10223     /* prefix xml1 not defined */
10224     len = -1;
10225     hr = IMXNamespaceManager_getURI(nsmgr, _bstr_("xml1"), NULL, NULL, &len);
10226     EXPECT_HR(hr, S_FALSE);
10227     ok(len == 0, "got %d\n", len);
10228
10229     len = 100;
10230     buffW[0] = 0x1;
10231     hr = IMXNamespaceManager_getURI(nsmgr, _bstr_("xml1"), NULL, buffW, &len);
10232     EXPECT_HR(hr, S_FALSE);
10233     ok(buffW[0] == 0, "got %x\n", buffW[0]);
10234     ok(len == 0, "got %d\n", len);
10235
10236     IMXNamespaceManager_Release(nsmgr);
10237
10238     free_bstrs();
10239 }
10240
10241 static void test_nsnamespacemanager_override(void)
10242 {
10243     IMXNamespaceManager *nsmgr;
10244     WCHAR buffW[250];
10245     VARIANT_BOOL b;
10246     HRESULT hr;
10247     INT len;
10248
10249     hr = CoCreateInstance(&CLSID_MXNamespaceManager40, NULL, CLSCTX_INPROC_SERVER,
10250         &IID_IMXNamespaceManager, (void**)&nsmgr);
10251     if (hr != S_OK)
10252     {
10253         win_skip("MXNamespaceManager is not available\n");
10254         return;
10255     }
10256
10257     len = sizeof(buffW)/sizeof(WCHAR);
10258     buffW[0] = 0;
10259     hr = IMXNamespaceManager_getDeclaredPrefix(nsmgr, 0, buffW, &len);
10260     EXPECT_HR(hr, S_OK);
10261     ok(!lstrcmpW(buffW, _bstr_("xml")), "got prefix %s\n", wine_dbgstr_w(buffW));
10262
10263     len = sizeof(buffW)/sizeof(WCHAR);
10264     buffW[0] = 0;
10265     hr = IMXNamespaceManager_getDeclaredPrefix(nsmgr, 1, buffW, &len);
10266     EXPECT_HR(hr, E_FAIL);
10267
10268     hr = IMXNamespaceManager_getAllowOverride(nsmgr, NULL);
10269     EXPECT_HR(hr, E_POINTER);
10270
10271     b = VARIANT_FALSE;
10272     hr = IMXNamespaceManager_getAllowOverride(nsmgr, &b);
10273     EXPECT_HR(hr, S_OK);
10274     ok(b == VARIANT_TRUE, "got %d\n", b);
10275
10276     hr = IMXNamespaceManager_putAllowOverride(nsmgr, VARIANT_FALSE);
10277     EXPECT_HR(hr, S_OK);
10278
10279     hr = IMXNamespaceManager_declarePrefix(nsmgr, NULL, _bstr_("ns0 uri"));
10280     EXPECT_HR(hr, S_OK);
10281
10282     len = sizeof(buffW)/sizeof(WCHAR);
10283     buffW[0] = 0;
10284     hr = IMXNamespaceManager_getURI(nsmgr, _bstr_(""), NULL, buffW, &len);
10285     EXPECT_HR(hr, S_OK);
10286     ok(!lstrcmpW(buffW, _bstr_("ns0 uri")), "got uri %s\n", wine_dbgstr_w(buffW));
10287
10288     hr = IMXNamespaceManager_declarePrefix(nsmgr, _bstr_("ns0"), _bstr_("ns0 uri"));
10289     EXPECT_HR(hr, S_OK);
10290
10291     len = sizeof(buffW)/sizeof(WCHAR);
10292     buffW[0] = 0;
10293     hr = IMXNamespaceManager_getDeclaredPrefix(nsmgr, 0, buffW, &len);
10294     EXPECT_HR(hr, S_OK);
10295     ok(!lstrcmpW(buffW, _bstr_("xml")), "got prefix %s\n", wine_dbgstr_w(buffW));
10296
10297     len = sizeof(buffW)/sizeof(WCHAR);
10298     buffW[0] = 0;
10299     hr = IMXNamespaceManager_getDeclaredPrefix(nsmgr, 1, buffW, &len);
10300     EXPECT_HR(hr, S_OK);
10301     ok(!lstrcmpW(buffW, _bstr_("ns0")), "got prefix %s\n", wine_dbgstr_w(buffW));
10302
10303     len = sizeof(buffW)/sizeof(WCHAR);
10304     buffW[0] = 0;
10305     hr = IMXNamespaceManager_getDeclaredPrefix(nsmgr, 2, buffW, &len);
10306     EXPECT_HR(hr, S_OK);
10307     ok(!lstrcmpW(buffW, _bstr_("")), "got prefix %s\n", wine_dbgstr_w(buffW));
10308
10309     /* new prefix placed at index 1 always */
10310     hr = IMXNamespaceManager_declarePrefix(nsmgr, _bstr_("ns1"), _bstr_("ns1 uri"));
10311     EXPECT_HR(hr, S_OK);
10312
10313     len = sizeof(buffW)/sizeof(WCHAR);
10314     buffW[0] = 0;
10315     hr = IMXNamespaceManager_getDeclaredPrefix(nsmgr, 1, buffW, &len);
10316     EXPECT_HR(hr, S_OK);
10317     ok(!lstrcmpW(buffW, _bstr_("ns1")), "got prefix %s\n", wine_dbgstr_w(buffW));
10318
10319     hr = IMXNamespaceManager_declarePrefix(nsmgr, _bstr_(""), NULL);
10320     todo_wine EXPECT_HR(hr, E_FAIL);
10321
10322     hr = IMXNamespaceManager_declarePrefix(nsmgr, NULL, NULL);
10323     EXPECT_HR(hr, E_FAIL);
10324
10325     hr = IMXNamespaceManager_declarePrefix(nsmgr, NULL, _bstr_("ns0 uri"));
10326     EXPECT_HR(hr, E_FAIL);
10327
10328     hr = IMXNamespaceManager_putAllowOverride(nsmgr, VARIANT_TRUE);
10329     EXPECT_HR(hr, S_OK);
10330
10331     hr = IMXNamespaceManager_declarePrefix(nsmgr, NULL, _bstr_("ns0 uri override"));
10332     EXPECT_HR(hr, S_FALSE);
10333
10334     len = sizeof(buffW)/sizeof(WCHAR);
10335     buffW[0] = 0;
10336     hr = IMXNamespaceManager_getURI(nsmgr, _bstr_(""), NULL, buffW, &len);
10337     EXPECT_HR(hr, S_OK);
10338     ok(!lstrcmpW(buffW, _bstr_("ns0 uri override")), "got uri %s\n", wine_dbgstr_w(buffW));
10339
10340     len = sizeof(buffW)/sizeof(WCHAR);
10341     buffW[0] = 0;
10342     hr = IMXNamespaceManager_getDeclaredPrefix(nsmgr, 3, buffW, &len);
10343     EXPECT_HR(hr, S_OK);
10344     ok(!lstrcmpW(buffW, _bstr_("")), "got prefix %s\n", wine_dbgstr_w(buffW));
10345
10346     IMXNamespaceManager_Release(nsmgr);
10347
10348     free_bstrs();
10349 }
10350
10351 START_TEST(domdoc)
10352 {
10353     IXMLDOMDocument *doc;
10354     HRESULT hr;
10355
10356     hr = CoInitialize( NULL );
10357     ok( hr == S_OK, "failed to init com\n");
10358     if (hr != S_OK) return;
10359
10360     test_XMLHTTP();
10361
10362     hr = CoCreateInstance( &CLSID_DOMDocument, NULL, CLSCTX_INPROC_SERVER, &IID_IXMLDOMDocument, (void**)&doc );
10363     if (hr != S_OK)
10364     {
10365         win_skip("IXMLDOMDocument is not available (0x%08x)\n", hr);
10366         return;
10367     }
10368
10369     IXMLDOMDocument_Release(doc);
10370
10371     test_domdoc();
10372     test_persiststreaminit();
10373     test_domnode();
10374     test_refs();
10375     test_create();
10376     test_getElementsByTagName();
10377     test_get_text();
10378     test_get_childNodes();
10379     test_get_firstChild();
10380     test_get_lastChild();
10381     test_removeChild();
10382     test_replaceChild();
10383     test_removeNamedItem();
10384     test_IXMLDOMDocument2();
10385     test_whitespace();
10386     test_XPath();
10387     test_XSLPattern();
10388     test_cloneNode();
10389     test_xmlTypes();
10390     test_nodeTypeTests();
10391     test_save();
10392     test_testTransforms();
10393     test_Namespaces();
10394     test_FormattingXML();
10395     test_nodeTypedValue();
10396     test_TransformWithLoadingLocalFile();
10397     test_put_nodeValue();
10398     test_document_IObjectSafety();
10399     test_splitText();
10400     test_getQualifiedItem();
10401     test_removeQualifiedItem();
10402     test_get_ownerDocument();
10403     test_setAttributeNode();
10404     test_put_dataType();
10405     test_createNode();
10406     test_get_prefix();
10407     test_default_properties();
10408     test_selectSingleNode();
10409     test_events();
10410     test_createProcessingInstruction();
10411     test_put_nodeTypedValue();
10412     test_get_xml();
10413     test_insertBefore();
10414     test_appendChild();
10415     test_get_doctype();
10416     test_get_tagName();
10417     test_get_dataType();
10418     test_get_nodeTypeString();
10419     test_get_attributes();
10420     test_selection();
10421     test_load();
10422
10423     test_xsltemplate();
10424
10425     test_nsnamespacemanager();
10426     test_nsnamespacemanager_override();
10427
10428     CoUninitialize();
10429 }