mshtml: Fixed handling channels without container and necko channel.
[wine] / dlls / msxml3 / tests / domdoc.c
1 /*
2  * XML test
3  *
4  * Copyright 2005 Mike McCormack for CodeWeavers
5  *
6  * This library is free software; you can redistribute it and/or
7  * modify it under the terms of the GNU Lesser General Public
8  * License as published by the Free Software Foundation; either
9  * version 2.1 of the License, or (at your option) any later version.
10  *
11  * This library is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14  * Lesser General Public License for more details.
15  *
16  * You should have received a copy of the GNU Lesser General Public
17  * License along with this library; if not, write to the Free Software
18  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
19  */
20
21
22 #define COBJMACROS
23
24 #include "windows.h"
25 #include "ole2.h"
26 #include "xmldom.h"
27 #include "msxml2.h"
28 #include <stdio.h>
29 #include <assert.h>
30
31 #include "wine/test.h"
32
33 const CLSID CLSID_DOMDocument2 = {0xf6d90f11, 0x9c73, 0x11d3, {0xb3, 0x2e, 0x00, 0xc0, 0x4f, 0x99, 0x0b, 0xb4}};
34 const IID IID_IXMLDOMDocument2 = {0x2933bf95, 0x7b36, 0x11d2, {0xb2, 0x0e, 0x00, 0xc0, 0x4f, 0x98, 0x3e, 0x60}};
35
36 static const WCHAR szEmpty[] = { 0 };
37 static const WCHAR szIncomplete[] = {
38     '<','?','x','m','l',' ',
39     'v','e','r','s','i','o','n','=','\'','1','.','0','\'','?','>','\n',0
40 };
41 static const WCHAR szComplete1[] = {
42     '<','?','x','m','l',' ',
43     'v','e','r','s','i','o','n','=','\'','1','.','0','\'','?','>','\n',
44     '<','o','p','e','n','>','<','/','o','p','e','n','>','\n',0
45 };
46 static const WCHAR szComplete2[] = {
47     '<','?','x','m','l',' ',
48     'v','e','r','s','i','o','n','=','\'','1','.','0','\'','?','>','\n',
49     '<','o','>','<','/','o','>','\n',0
50 };
51 static const WCHAR szComplete3[] = {
52     '<','?','x','m','l',' ',
53     'v','e','r','s','i','o','n','=','\'','1','.','0','\'','?','>','\n',
54     '<','a','>','<','/','a','>','\n',0
55 };
56 static const WCHAR szComplete4[] = {
57     '<','?','x','m','l',' ','v','e','r','s','i','o','n','=','\'','1','.','0','\'','?','>','\n',
58     '<','l','c',' ','d','l','=','\'','s','t','r','1','\'','>','\n',
59         '<','b','s',' ','v','r','=','\'','s','t','r','2','\'',' ','s','z','=','\'','1','2','3','4','\'','>',
60             'f','n','1','.','t','x','t','\n',
61         '<','/','b','s','>','\n',
62         '<','p','r',' ','i','d','=','\'','s','t','r','3','\'',' ','v','r','=','\'','1','.','2','.','3','\'',' ',
63                     'p','n','=','\'','w','i','n','e',' ','2','0','0','5','0','8','0','4','\'','>','\n',
64             'f','n','2','.','t','x','t','\n',
65         '<','/','p','r','>','\n',
66         '<','e','m','p','t','y','>','<','/','e','m','p','t','y','>','\n',
67         '<','f','o','>','\n',
68             '<','b','a','>','\n',
69                 'f','1','\n',
70             '<','/','b','a','>','\n',
71         '<','/','f','o','>','\n',
72     '<','/','l','c','>','\n',0
73 };
74 static const WCHAR szComplete5[] = {
75     '<','S',':','s','e','a','r','c','h',' ','x','m','l','n','s',':','D','=','"','D','A','V',':','"',' ',
76     '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','"',
77     ' ','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','"','>',
78         '<','S',':','s','c','o','p','e','>',
79             '<','S',':','d','e','e','p','>','/','<','/','S',':','d','e','e','p','>',
80         '<','/','S',':','s','c','o','p','e','>',
81         '<','S',':','c','o','n','t','e','n','t','f','r','e','e','t','e','x','t','>',
82             '<','C',':','t','e','x','t','o','r','p','r','o','p','e','r','t','y','/','>',
83             'c','o','m','p','u','t','e','r',
84         '<','/','S',':','c','o','n','t','e','n','t','f','r','e','e','t','e','x','t','>',
85     '<','/','S',':','s','e','a','r','c','h','>',0
86 };
87
88 static const CHAR szExampleXML[] =
89 "<?xml version='1.0' encoding='utf-8'?>\n"
90 "<root xmlns:foo='urn:uuid:86B2F87F-ACB6-45cd-8B77-9BDB92A01A29'>\n"
91 "    <elem>\n"
92 "        <a>A1 field</a>\n"
93 "        <b>B1 field</b>\n"
94 "        <c>C1 field</c>\n"
95 "        <description xmlns:foo='http://www.winehq.org' xmlns:bar='urn:uuid:86B2F87F-ACB6-45cd-8B77-9BDB92A01A29'>\n"
96 "            <html xmlns='http://www.w3.org/1999/xhtml'>\n"
97 "                This is <strong>a</strong> <i>description</i>. <bar:x/>\n"
98 "            </html>\n"
99 "        </description>\n"
100 "    </elem>\n"
101 "\n"
102 "    <elem>\n"
103 "        <a>A2 field</a>\n"
104 "        <b>B2 field</b>\n"
105 "        <c type=\"old\">C2 field</c>\n"
106 "    </elem>\n"
107 "\n"
108 "    <elem xmlns='urn:uuid:86B2F87F-ACB6-45cd-8B77-9BDB92A01A29'>\n"
109 "        <a>A3 field</a>\n"
110 "        <b>B3 field</b>\n"
111 "        <c>C3 field</c>\n"
112 "    </elem>\n"
113 "\n"
114 "    <elem>\n"
115 "        <a>A4 field</a>\n"
116 "        <b>B4 field</b>\n"
117 "        <foo:c>C4 field</foo:c>\n"
118 "    </elem>\n"
119 "</root>\n";
120
121 static const WCHAR szNonExistentFile[] = {
122     'c', ':', '\\', 'N', 'o', 'n', 'e', 'x', 'i', 's', 't', 'e', 'n', 't', '.', 'x', 'm', 'l', 0
123 };
124 static const WCHAR szDocument[] = {
125     '#', 'd', 'o', 'c', 'u', 'm', 'e', 'n', 't', 0
126 };
127
128 static const WCHAR szOpen[] = { 'o','p','e','n',0 };
129 static WCHAR szdl[] = { 'd','l',0 };
130 static const WCHAR szvr[] = { 'v','r',0 };
131 static const WCHAR szlc[] = { 'l','c',0 };
132 static WCHAR szbs[] = { 'b','s',0 };
133 static const WCHAR szstr1[] = { 's','t','r','1',0 };
134 static const WCHAR szstr2[] = { 's','t','r','2',0 };
135 static const WCHAR szstar[] = { '*',0 };
136 static const WCHAR szfn1_txt[] = {'f','n','1','.','t','x','t',0};
137
138 #define expect_bstr_eq_and_free(bstr, expect) { \
139     BSTR bstrExp = alloc_str_from_narrow(expect); \
140     ok(lstrcmpW(bstr, bstrExp) == 0, "String differs\n"); \
141     SysFreeString(bstr); \
142     SysFreeString(bstrExp); \
143 }
144
145 #define expect_eq(expr, value, type, format) { type ret = (expr); ok((value) == ret, #expr " expected " format " got " format "\n", value, ret); }
146
147 #define ole_check(expr) { \
148     HRESULT r = expr; \
149     ok(r == S_OK, #expr " returned %x\n", r); \
150 }
151
152 #define ole_expect(expr, expect) { \
153     HRESULT r = expr; \
154     ok(r == (expect), #expr " returned %x, expected %x\n", r, expect); \
155 }
156
157 static BSTR alloc_str_from_narrow(const char *str)
158 {
159     int len = MultiByteToWideChar(CP_ACP, 0, str, -1, NULL, 0);
160     BSTR ret = SysAllocStringLen(NULL, len - 1);  /* NUL character added automatically */
161     MultiByteToWideChar(CP_ACP, 0, str, -1, ret, len);
162     return ret;
163 }
164
165 BSTR alloced_bstrs[256];
166 int alloced_bstrs_count = 0;
167
168 static BSTR _bstr_(const char *str)
169 {
170     assert(alloced_bstrs_count < sizeof(alloced_bstrs)/sizeof(alloced_bstrs[0]));
171     alloced_bstrs[alloced_bstrs_count] = alloc_str_from_narrow(str);
172     return alloced_bstrs[alloced_bstrs_count++];
173 }
174
175 static void free_bstrs(void)
176 {
177     int i;
178     for (i = 0; i < alloced_bstrs_count; i++)
179         SysFreeString(alloced_bstrs[i]);
180     alloced_bstrs_count = 0;
181 }
182
183 static VARIANT _variantbstr_(const char *str)
184 {
185     VARIANT v;
186     V_VT(&v) = VT_BSTR;
187     V_BSTR(&v) = _bstr_(str);
188     return v;
189 }
190
191 static void get_str_for_type(DOMNodeType type, char *buf)
192 {
193     switch (type)
194     {
195         case NODE_ATTRIBUTE:
196             strcpy(buf, "A");
197             break;
198         case NODE_ELEMENT:
199             strcpy(buf, "E");
200             break;
201         case NODE_DOCUMENT:
202             strcpy(buf, "D");
203             break;
204         default:
205             wsprintfA(buf, "[%d]", type);
206     }
207 }
208
209 static int get_node_position(IXMLDOMNode *node)
210 {
211     HRESULT r;
212     int pos = 0;
213
214     IXMLDOMNode_AddRef(node);
215     do
216     {
217         IXMLDOMNode *new_node;
218
219         pos++;
220         r = IXMLDOMNode_get_previousSibling(node, &new_node);
221         ok(!FAILED(r), "get_previousSibling failed\n");
222         IXMLDOMNode_Release(node);
223         node = new_node;
224     } while (r == S_OK);
225     return pos;
226 }
227
228 static void node_to_string(IXMLDOMNode *node, char *buf)
229 {
230     HRESULT r = S_OK;
231     DOMNodeType type;
232
233     if (node == NULL)
234     {
235         lstrcpyA(buf, "(null)");
236         return;
237     }
238
239     IXMLDOMNode_AddRef(node);
240     while (r == S_OK)
241     {
242         IXMLDOMNode *new_node;
243
244         ole_check(IXMLDOMNode_get_nodeType(node, &type));
245         get_str_for_type(type, buf);
246         buf+=strlen(buf);
247
248         if (type == NODE_ATTRIBUTE)
249         {
250             BSTR bstr;
251             ole_check(IXMLDOMNode_get_nodeName(node, &bstr));
252             *(buf++) = '\'';
253             wsprintfA(buf, "%ws", bstr);
254             buf += strlen(buf);
255             *(buf++) = '\'';
256             SysFreeString(bstr);
257
258             r = IXMLDOMNode_selectSingleNode(node, _bstr_(".."), &new_node);
259         }
260         else
261         {
262             int pos = get_node_position(node);
263             DOMNodeType parent_type = NODE_INVALID;
264             r = IXMLDOMNode_get_parentNode(node, &new_node);
265
266             /* currently wine doesn't create a node for the <?xml ... ?>. To be able to test query
267              * results we "fix" it */
268             if (r == S_OK)
269                 ole_check(IXMLDOMNode_get_nodeType(node, &parent_type));
270             /* we need also to workaround the no document node problem - see below */
271             if (((r == S_FALSE && type != NODE_DOCUMENT) || parent_type == NODE_DOCUMENT) && type != NODE_PROCESSING_INSTRUCTION && pos==1)
272             {
273                 todo_wine ok(FALSE, "The first child of the document node in MSXML is the <?xml ... ?> processing instruction\n");
274                 pos++;
275             }
276             wsprintf(buf, "%d", pos);
277             buf += strlen(buf);
278         }
279
280         ok(!FAILED(r), "get_parentNode failed (%08x)\n", r);
281         IXMLDOMNode_Release(node);
282         node = new_node;
283         if (r == S_OK)
284             *(buf++) = '.';
285     }
286
287     /* currently we can't access document node in wine. All our examples this is the
288      * root node so to be able to test query results we add it */
289     if (type != NODE_DOCUMENT)
290     {
291         todo_wine ok(FALSE, "Document node is not the last returned node!\n");
292         *(buf++) = '.';
293         *(buf++) = 'D';
294         *(buf++) = '1';
295     }
296     *buf = 0;
297 }
298
299 static char *list_to_string(IXMLDOMNodeList *list)
300 {
301     static char buf[4096];
302     char *pos = buf;
303     long len = 0;
304     int i;
305
306     if (list == NULL)
307     {
308         lstrcpyA(buf, "(null)");
309         return buf;
310     }
311     ole_check(IXMLDOMNodeList_get_length(list, &len));
312     for (i = 0; i < len; i++)
313     {
314         IXMLDOMNode *node;
315         if (i > 0)
316             *(pos++) = ' ';
317         ole_check(IXMLDOMNodeList_nextNode(list, &node));
318         node_to_string(node, pos);
319         pos += strlen(pos);
320         IXMLDOMNode_Release(node);
321     }
322     *pos = 0;
323     return buf;
324 }
325
326 #define expect_node(node, expstr) { char str[4096]; node_to_string(node, str); ok(strcmp(str, expstr)==0, "Invalid node: %s, exptected %s\n", str, expstr); }
327 #define expect_list_and_release(list, expstr) { char *str = list_to_string(list); ok(strcmp(str, expstr)==0, "Invalid node list: %s, exptected %s\n", str, expstr); if (list) IXMLDOMNodeList_Release(list); }
328
329 static void test_domdoc( void )
330 {
331     HRESULT r;
332     IXMLDOMDocument *doc = NULL;
333     IXMLDOMParseError *error;
334     IXMLDOMElement *element = NULL;
335     IXMLDOMNode *node;
336     VARIANT_BOOL b;
337     VARIANT var;
338     BSTR str;
339     long code;
340
341     r = CoCreateInstance( &CLSID_DOMDocument, NULL, 
342         CLSCTX_INPROC_SERVER, &IID_IXMLDOMDocument, (LPVOID*)&doc );
343     if( r != S_OK )
344         return;
345
346     /* try some stupid things */
347     r = IXMLDOMDocument_loadXML( doc, NULL, NULL );
348     ok( r == S_FALSE, "loadXML failed\n");
349
350     b = VARIANT_TRUE;
351     r = IXMLDOMDocument_loadXML( doc, NULL, &b );
352     ok( r == S_FALSE, "loadXML failed\n");
353     ok( b == VARIANT_FALSE, "failed to load XML string\n");
354
355     /* try to load a document from a nonexistent file */
356     b = VARIANT_TRUE;
357     str = SysAllocString( szNonExistentFile );
358     VariantInit(&var);
359     V_VT(&var) = VT_BSTR;
360     V_BSTR(&var) = str;
361
362     r = IXMLDOMDocument_load( doc, var, &b);
363     ok( r == S_FALSE, "load (from file) failed\n");
364     ok( b == VARIANT_FALSE, "failed to load XML file\n");
365     SysFreeString( str );
366
367     /* try load an empty document */
368     b = VARIANT_TRUE;
369     str = SysAllocString( szEmpty );
370     r = IXMLDOMDocument_loadXML( doc, str, &b );
371     ok( r == S_FALSE, "loadXML failed\n");
372     ok( b == VARIANT_FALSE, "failed to load XML string\n");
373     SysFreeString( str );
374
375     /* check that there's no document element */
376     element = NULL;
377     r = IXMLDOMDocument_get_documentElement( doc, &element );
378     ok( r == S_FALSE, "should be no document element\n");
379
380     /* try finding a node */
381     node = NULL;
382     str = SysAllocString( szstr1 );
383     r = IXMLDOMDocument_selectSingleNode( doc, str, &node );
384     ok( r == S_FALSE, "ret %08x\n", r );
385     SysFreeString( str );
386
387     b = VARIANT_TRUE;
388     str = SysAllocString( szIncomplete );
389     r = IXMLDOMDocument_loadXML( doc, str, &b );
390     ok( r == S_FALSE, "loadXML failed\n");
391     ok( b == VARIANT_FALSE, "failed to load XML string\n");
392     SysFreeString( str );
393
394     /* check that there's no document element */
395     element = (IXMLDOMElement*)1;
396     r = IXMLDOMDocument_get_documentElement( doc, &element );
397     ok( r == S_FALSE, "should be no document element\n");
398     ok( element == NULL, "Element should be NULL\n");
399
400     /* try to load something valid */
401     b = VARIANT_FALSE;
402     str = SysAllocString( szComplete1 );
403     r = IXMLDOMDocument_loadXML( doc, str, &b );
404     ok( r == S_OK, "loadXML failed\n");
405     ok( b == VARIANT_TRUE, "failed to load XML string\n");
406     SysFreeString( str );
407
408     /* try with a null out pointer */
409     r = IXMLDOMDocument_get_documentElement( doc, NULL );
410     ok( r == E_INVALIDARG, "should be no document element\n");
411
412     /* check if nodename is correct */
413     r = IXMLDOMDocument_get_nodeName( doc, NULL );
414     ok ( r == E_INVALIDARG, "get_nodeName (NULL) wrong code\n");
415
416     /* content doesn't matter here */
417     str = SysAllocString( szNonExistentFile );
418     r = IXMLDOMDocument_get_nodeName( doc, &str );
419     ok ( r == S_OK, "get_nodeName wrong code\n");
420     ok ( str != NULL, "str is null\n");
421     ok( !lstrcmpW( str, szDocument ), "incorrect nodeName\n");
422     SysFreeString( str );
423
424
425     /* check that there's a document element */
426     element = NULL;
427     r = IXMLDOMDocument_get_documentElement( doc, &element );
428     ok( r == S_OK, "should be a document element\n");
429     if( element )
430     {
431         BSTR tag = NULL;
432
433         /* check if the tag is correct */
434         r = IXMLDOMElement_get_tagName( element, &tag );
435         ok( r == S_OK, "couldn't get tag name\n");
436         ok( tag != NULL, "tag was null\n");
437         ok( !lstrcmpW( tag, szOpen ), "incorrect tag name\n");
438         SysFreeString( tag );
439
440         /* figure out what happens if we try to reload the document */
441         str = SysAllocString( szComplete2 );
442         r = IXMLDOMDocument_loadXML( doc, str, &b );
443         ok( r == S_OK, "loadXML failed\n");
444         ok( b == VARIANT_TRUE, "failed to load XML string\n");
445         SysFreeString( str );
446
447         /* check if the tag is still correct */
448         tag = NULL;
449         r = IXMLDOMElement_get_tagName( element, &tag );
450         ok( r == S_OK, "couldn't get tag name\n");
451         ok( tag != NULL, "tag was null\n");
452         ok( !lstrcmpW( tag, szOpen ), "incorrect tag name\n");
453         SysFreeString( tag );
454
455         IXMLDOMElement_Release( element );
456         element = NULL;
457     }
458
459     /* as soon as we call loadXML again, the document element will disappear */
460     b = 2;
461     r = IXMLDOMDocument_loadXML( doc, NULL, NULL );
462     ok( r == S_FALSE, "loadXML failed\n");
463     ok( b == 2, "variant modified\n");
464     r = IXMLDOMDocument_get_documentElement( doc, &element );
465     ok( r == S_FALSE, "should be no document element\n");
466
467     /* try to load something else simple and valid */
468     b = VARIANT_FALSE;
469     str = SysAllocString( szComplete3 );
470     r = IXMLDOMDocument_loadXML( doc, str, &b );
471     ok( r == S_OK, "loadXML failed\n");
472     ok( b == VARIANT_TRUE, "failed to load XML string\n");
473     SysFreeString( str );
474
475     /* try something a little more complicated */
476     b = FALSE;
477     str = SysAllocString( szComplete4 );
478     r = IXMLDOMDocument_loadXML( doc, str, &b );
479     ok( r == S_OK, "loadXML failed\n");
480     ok( b == VARIANT_TRUE, "failed to load XML string\n");
481     SysFreeString( str );
482
483     r = IXMLDOMDocument_get_parseError( doc, &error );
484     ok( r == S_OK, "returns %08x\n", r );
485
486     r = IXMLDOMParseError_get_errorCode( error, &code );
487     ok( r == S_FALSE, "returns %08x\n", r );
488     ok( code == 0, "code %ld\n", code );
489     IXMLDOMParseError_Release( error );
490
491     r = IXMLDOMDocument_Release( doc );
492     ok( r == 0, "document ref count incorrect\n");
493
494 }
495
496 static void test_domnode( void )
497 {
498     HRESULT r;
499     IXMLDOMDocument *doc = NULL, *owner = NULL;
500     IXMLDOMElement *element = NULL;
501     IXMLDOMNamedNodeMap *map = NULL;
502     IXMLDOMNode *node = NULL, *next = NULL;
503     IXMLDOMNodeList *list = NULL;
504     DOMNodeType type = NODE_INVALID;
505     VARIANT_BOOL b;
506     BSTR str;
507     VARIANT var;
508     long count;
509
510     r = CoCreateInstance( &CLSID_DOMDocument, NULL, 
511         CLSCTX_INPROC_SERVER, &IID_IXMLDOMDocument, (LPVOID*)&doc );
512     if( r != S_OK )
513         return;
514
515     b = FALSE;
516     str = SysAllocString( szComplete4 );
517     r = IXMLDOMDocument_loadXML( doc, str, &b );
518     ok( r == S_OK, "loadXML failed\n");
519     ok( b == VARIANT_TRUE, "failed to load XML string\n");
520     SysFreeString( str );
521
522     if (doc)
523     {
524         b = 1;
525         r = IXMLDOMNode_hasChildNodes( doc, &b );
526         ok( r == S_OK, "hasChildNoes bad return\n");
527         ok( b == VARIANT_TRUE, "hasChildNoes wrong result\n");
528
529         r = IXMLDOMDocument_get_documentElement( doc, &element );
530         ok( r == S_OK, "should be a document element\n");
531         ok( element != NULL, "should be an element\n");
532     }
533     else
534         ok( FALSE, "no document\n");
535
536     VariantInit(&var);
537     ok( V_VT(&var) == VT_EMPTY, "variant init failed\n");
538     r = IXMLDOMNode_get_nodeValue( doc, &var );
539     ok( r == S_FALSE, "nextNode returned wrong code\n");
540     ok( V_VT(&var) == VT_NULL, "variant wasn't empty\n");
541     ok( V_BSTR(&var) == NULL, "variant value wasn't null\n");
542
543     if (element)
544     {
545         owner = NULL;
546         r = IXMLDOMNode_get_ownerDocument( element, &owner );
547         todo_wine {
548         ok( r == S_OK, "get_ownerDocument return code\n");
549         }
550         ok( owner != doc, "get_ownerDocument return\n");
551
552         type = NODE_INVALID;
553         r = IXMLDOMNode_get_nodeType( element, &type);
554         ok( r == S_OK, "getNamedItem returned wrong code\n");
555         ok( type == NODE_ELEMENT, "node not an element\n");
556
557         str = NULL;
558         r = IXMLDOMNode_get_baseName( element, &str );
559         ok( r == S_OK, "get_baseName returned wrong code\n");
560         ok( lstrcmpW(str,szlc) == 0, "basename was wrong\n");
561
562         /* check if nodename is correct */
563         r = IXMLDOMElement_get_nodeName( element, NULL );
564         ok ( r == E_INVALIDARG, "get_nodeName (NULL) wrong code\n");
565     
566         /* content doesn't matter here */
567         str = SysAllocString( szNonExistentFile );
568         r = IXMLDOMElement_get_nodeName( element, &str );
569         ok ( r == S_OK, "get_nodeName wrong code\n");
570         ok ( str != NULL, "str is null\n");
571         ok( !lstrcmpW( str, szlc ), "incorrect nodeName\n");
572         SysFreeString( str );
573
574         str = SysAllocString( szNonExistentFile );      
575         V_VT(&var) = VT_I4;
576         V_I4(&var) = 0x1234;
577         r = IXMLDOMElement_getAttribute( element, str, &var );
578         ok( r == E_FAIL, "getAttribute ret %08x\n", r );
579         ok( V_VT(&var) == VT_EMPTY, "vt = %x\n", V_VT(&var));
580         VariantClear(&var);
581         SysFreeString( str );
582
583         str = SysAllocString( szdl );   
584         V_VT(&var) = VT_I4;
585         V_I4(&var) = 0x1234;
586         r = IXMLDOMElement_getAttribute( element, str, &var );
587         ok( r == S_OK, "getAttribute ret %08x\n", r );
588         ok( V_VT(&var) == VT_BSTR, "vt = %x\n", V_VT(&var));
589         ok( !lstrcmpW(V_BSTR(&var), szstr1), "wrong attr value\n");
590         VariantClear( &var );
591         SysFreeString( str );
592
593         r = IXMLDOMElement_get_attributes( element, &map );
594         ok( r == S_OK, "get_attributes returned wrong code\n");
595         ok( map != NULL, "should be attributes\n");
596
597         b = 1;
598         r = IXMLDOMNode_hasChildNodes( element, &b );
599         ok( r == S_OK, "hasChildNoes bad return\n");
600         ok( b == VARIANT_TRUE, "hasChildNoes wrong result\n");
601     }
602     else
603         ok( FALSE, "no element\n");
604
605     if (map)
606     {
607         ISupportErrorInfo *support_error;
608         r = IXMLDOMNamedNodeMap_QueryInterface( map, &IID_ISupportErrorInfo, (LPVOID*)&support_error );
609         ok( r == S_OK, "ret %08x\n", r );
610
611         r = ISupportErrorInfo_InterfaceSupportsErrorInfo( support_error, &IID_IXMLDOMNamedNodeMap );
612 todo_wine
613 {
614         ok( r == S_OK, "ret %08x\n", r );
615 }
616         ISupportErrorInfo_Release( support_error );
617
618         str = SysAllocString( szdl );
619         r = IXMLDOMNamedNodeMap_getNamedItem( map, str, &node );
620         ok( r == S_OK, "getNamedItem returned wrong code\n");
621         ok( node != NULL, "should be attributes\n");
622         IXMLDOMNode_Release(node);
623         SysFreeString( str );
624
625         str = SysAllocString( szdl );
626         r = IXMLDOMNamedNodeMap_getNamedItem( map, str, NULL );
627         ok( r == E_INVALIDARG, "getNamedItem should return E_INVALIDARG\n");
628         SysFreeString( str );
629
630         /* something that isn't in szComplete4 */
631         str = SysAllocString( szOpen );
632         node = (IXMLDOMNode *) 1;
633         r = IXMLDOMNamedNodeMap_getNamedItem( map, str, &node );
634         ok( r = S_FALSE, "getNamedItem found a node that wasn't there\n");
635         ok( node == NULL, "getNamedItem should have returned NULL\n");
636         SysFreeString( str );
637
638         /* test indexed access of attributes */
639         r = IXMLDOMNamedNodeMap_get_length( map, &count );
640         ok ( r == S_OK, "get_length wrong code\n");
641         ok ( count == 1, "get_length != 1\n");
642
643         node = NULL;
644         r = IXMLDOMNamedNodeMap_get_item( map, -1, &node);
645         ok ( r == S_FALSE, "get_item (-1) wrong code\n");
646         ok ( node == NULL, "there is no node\n");
647
648         node = NULL;
649         r = IXMLDOMNamedNodeMap_get_item( map, 1, &node);
650         ok ( r == S_FALSE, "get_item (1) wrong code\n");
651         ok ( node == NULL, "there is no attribute\n");
652
653         node = NULL;
654         r = IXMLDOMNamedNodeMap_get_item( map, 0, &node);
655         ok ( r == S_OK, "get_item (0) wrong code\n");
656         ok ( node != NULL, "should be attribute\n");
657
658         r = IXMLDOMNode_get_nodeName( node, NULL );
659         ok ( r == E_INVALIDARG, "get_nodeName (NULL) wrong code\n");
660
661         /* content doesn't matter here */
662         str = SysAllocString( szNonExistentFile );
663         r = IXMLDOMNode_get_nodeName( node, &str );
664         ok ( r == S_OK, "get_nodeName wrong code\n");
665         ok ( str != NULL, "str is null\n");
666         ok( !lstrcmpW( str, szdl ), "incorrect node name\n");
667         SysFreeString( str );
668
669         /* test sequential access of attributes */
670         node = NULL;
671         r = IXMLDOMNamedNodeMap_nextNode( map, &node );
672         ok ( r == S_OK, "nextNode (first time) wrong code\n");
673         ok ( node != NULL, "nextNode, should be attribute\n");
674
675         r = IXMLDOMNamedNodeMap_nextNode( map, &node );
676         ok ( r != S_OK, "nextNode (second time) wrong code\n");
677         ok ( node == NULL, "nextNode, there is no attribute\n");
678
679         r = IXMLDOMNamedNodeMap_reset( map );
680         ok ( r == S_OK, "reset should return S_OK\n");
681
682         r = IXMLDOMNamedNodeMap_nextNode( map, &node );
683         ok ( r == S_OK, "nextNode (third time) wrong code\n");
684         ok ( node != NULL, "nextNode, should be attribute\n");
685     }
686     else
687         ok( FALSE, "no map\n");
688
689     if (node)
690     {
691         type = NODE_INVALID;
692         r = IXMLDOMNode_get_nodeType( node, &type);
693         ok( r == S_OK, "getNamedItem returned wrong code\n");
694         ok( type == NODE_ATTRIBUTE, "node not an attribute\n");
695
696         str = NULL;
697         r = IXMLDOMNode_get_baseName( node, NULL );
698         ok( r == E_INVALIDARG, "get_baseName returned wrong code\n");
699
700         str = NULL;
701         r = IXMLDOMNode_get_baseName( node, &str );
702         ok( r == S_OK, "get_baseName returned wrong code\n");
703         ok( lstrcmpW(str,szdl) == 0, "basename was wrong\n");
704         SysFreeString( str );
705
706         r = IXMLDOMNode_get_nodeValue( node, &var );
707         ok( r == S_OK, "returns %08x\n", r );
708         ok( V_VT(&var) == VT_BSTR, "vt %x\n", V_VT(&var));
709         ok( !lstrcmpW(V_BSTR(&var), szstr1), "nodeValue incorrect\n");
710         VariantClear(&var);
711
712         r = IXMLDOMNode_get_childNodes( node, NULL );
713         ok( r == E_INVALIDARG, "get_childNodes returned wrong code\n");
714
715         r = IXMLDOMNode_get_childNodes( node, &list );
716         ok( r == S_OK, "get_childNodes returned wrong code\n");
717
718         if (list)
719         {
720             r = IXMLDOMNodeList_nextNode( list, &next );
721             ok( r == S_OK, "nextNode returned wrong code\n");
722         }
723         else
724             ok( FALSE, "no childlist\n");
725
726         if (next)
727         {
728             b = 1;
729             r = IXMLDOMNode_hasChildNodes( next, &b );
730             ok( r == S_FALSE, "hasChildNoes bad return\n");
731             ok( b == VARIANT_FALSE, "hasChildNoes wrong result\n");
732
733             type = NODE_INVALID;
734             r = IXMLDOMNode_get_nodeType( next, &type);
735             ok( r == S_OK, "getNamedItem returned wrong code\n");
736             ok( type == NODE_TEXT, "node not text\n");
737
738             str = (BSTR) 1;
739             r = IXMLDOMNode_get_baseName( next, &str );
740             ok( r == S_FALSE, "get_baseName returned wrong code\n");
741             ok( str == NULL, "basename was wrong\n");
742         }
743         else
744             ok( FALSE, "no next\n");
745
746         if (next)
747             IXMLDOMNode_Release( next );
748         next = NULL;
749         if (list)
750             IXMLDOMNodeList_Release( list );
751         list = NULL;
752         if (node)
753             IXMLDOMNode_Release( node );
754     }
755     else
756         ok( FALSE, "no node\n");
757     node = NULL;
758
759     if (map)
760         IXMLDOMNamedNodeMap_Release( map );
761
762     /* now traverse the tree from the root element */
763     if (element)
764     {
765         IXMLDOMNode *node;
766         r = IXMLDOMNode_get_childNodes( element, &list );
767         ok( r == S_OK, "get_childNodes returned wrong code\n");
768
769         /* using get_item for child list doesn't advance the position */
770         ole_check(IXMLDOMNodeList_get_item(list, 1, &node));
771         expect_node(node, "E2.E2.D1");
772         IXMLDOMNode_Release(node);
773         ole_check(IXMLDOMNodeList_nextNode(list, &node));
774         expect_node(node, "E1.E2.D1");
775         IXMLDOMNode_Release(node);
776         ole_check(IXMLDOMNodeList_reset(list));
777
778         IXMLDOMNodeList_AddRef(list);
779         expect_list_and_release(list, "E1.E2.D1 E2.E2.D1 E3.E2.D1 E4.E2.D1");
780         ole_check(IXMLDOMNodeList_reset(list));
781     }
782     else
783         ok( FALSE, "no element\n");
784
785     node = (void*)0xdeadbeef;
786     r = IXMLDOMNode_selectSingleNode( element, szdl, &node );
787     ok( r == S_FALSE, "ret %08x\n", r );
788     ok( node == NULL, "node %p\n", node );
789     r = IXMLDOMNode_selectSingleNode( element, szbs, &node );
790     ok( r == S_OK, "ret %08x\n", r );
791     r = IXMLDOMNode_Release( node );
792     ok( r == 0, "ret %08x\n", r );
793
794     if (list)
795     {
796         r = IXMLDOMNodeList_get_length( list, &count );
797         ok( r == S_OK, "get_length returns %08x\n", r );
798         ok( count == 4, "get_length got %ld\n", count );
799
800         r = IXMLDOMNodeList_nextNode( list, &node );
801         ok( r == S_OK, "nextNode returned wrong code\n");
802     }
803     else
804         ok( FALSE, "no list\n");
805
806     if (node)
807     {
808         type = NODE_INVALID;
809         r = IXMLDOMNode_get_nodeType( node, &type);
810         ok( r == S_OK, "getNamedItem returned wrong code\n");
811         ok( type == NODE_ELEMENT, "node not text\n");
812
813         VariantInit(&var);
814         ok( V_VT(&var) == VT_EMPTY, "variant init failed\n");
815         r = IXMLDOMNode_get_nodeValue( node, &var );
816         ok( r == S_FALSE, "nextNode returned wrong code\n");
817         ok( V_VT(&var) == VT_NULL, "variant wasn't empty\n");
818         ok( V_BSTR(&var) == NULL, "variant value wasn't null\n");
819
820         r = IXMLDOMNode_hasChildNodes( node, NULL );
821         ok( r == E_INVALIDARG, "hasChildNoes bad return\n");
822
823         b = 1;
824         r = IXMLDOMNode_hasChildNodes( node, &b );
825         ok( r == S_OK, "hasChildNoes bad return\n");
826         ok( b == VARIANT_TRUE, "hasChildNoes wrong result\n");
827
828         str = NULL;
829         r = IXMLDOMNode_get_baseName( node, &str );
830         ok( r == S_OK, "get_baseName returned wrong code\n");
831         ok( lstrcmpW(str,szbs) == 0, "basename was wrong\n");
832     }
833     else
834         ok( FALSE, "no node\n");
835
836     if (node)
837         IXMLDOMNode_Release( node );
838     if (list)
839         IXMLDOMNodeList_Release( list );
840     if (element)
841         IXMLDOMElement_Release( element );
842
843     b = FALSE;
844     str = SysAllocString( szComplete5 );
845     r = IXMLDOMDocument_loadXML( doc, str, &b );
846     ok( r == S_OK, "loadXML failed\n");
847     ok( b == VARIANT_TRUE, "failed to load XML string\n");
848     SysFreeString( str );
849
850     b = 1;
851     r = IXMLDOMNode_hasChildNodes( doc, &b );
852     ok( r == S_OK, "hasChildNoes bad return\n");
853     ok( b == VARIANT_TRUE, "hasChildNoes wrong result\n");
854
855     r = IXMLDOMDocument_get_documentElement( doc, &element );
856     ok( r == S_OK, "should be a document element\n");
857     ok( element != NULL, "should be an element\n");
858
859     if (element)
860     {
861         static const WCHAR szSSearch[] = {'S',':','s','e','a','r','c','h',0};
862         BSTR tag = NULL;
863
864         /* check if the tag is correct */
865         r = IXMLDOMElement_get_tagName( element, &tag );
866         ok( r == S_OK, "couldn't get tag name\n");
867         ok( tag != NULL, "tag was null\n");
868         ok( !lstrcmpW( tag, szSSearch ), "incorrect tag name\n");
869         SysFreeString( tag );
870     }
871
872     if (element)
873         IXMLDOMElement_Release( element );
874     if (doc)
875         IXMLDOMDocument_Release( doc );
876 }
877
878 static void test_refs(void)
879 {
880     HRESULT r;
881     BSTR str;
882     VARIANT_BOOL b;
883     IXMLDOMDocument *doc = NULL;
884     IXMLDOMElement *element = NULL;
885     IXMLDOMNode *node = NULL, *node2;
886     IXMLDOMNodeList *node_list = NULL;
887     LONG ref;
888     IUnknown *unk, *unk2;
889
890     r = CoCreateInstance( &CLSID_DOMDocument, NULL, 
891         CLSCTX_INPROC_SERVER, &IID_IXMLDOMDocument, (LPVOID*)&doc );
892     if( r != S_OK )
893         return;
894     ref = IXMLDOMDocument_Release(doc);
895     ok( ref == 0, "ref %d\n", ref);
896
897     r = CoCreateInstance( &CLSID_DOMDocument, NULL, 
898         CLSCTX_INPROC_SERVER, &IID_IXMLDOMDocument, (LPVOID*)&doc );
899     if( r != S_OK )
900         return;
901
902     str = SysAllocString( szComplete4 );
903     r = IXMLDOMDocument_loadXML( doc, str, &b );
904     ok( r == S_OK, "loadXML failed\n");
905     ok( b == VARIANT_TRUE, "failed to load XML string\n");
906     SysFreeString( str );
907
908     ref = IXMLDOMDocument_AddRef( doc );
909     ok( ref == 2, "ref %d\n", ref );
910     ref = IXMLDOMDocument_AddRef( doc );
911     ok( ref == 3, "ref %d\n", ref );
912     IXMLDOMDocument_Release( doc );
913     IXMLDOMDocument_Release( doc );
914
915     r = IXMLDOMDocument_get_documentElement( doc, &element );
916     ok( r == S_OK, "should be a document element\n");
917     ok( element != NULL, "should be an element\n");
918
919     ref = IXMLDOMDocument_AddRef( doc );
920     ok( ref == 2, "ref %d\n", ref );
921     IXMLDOMDocument_Release( doc );
922
923     r = IXMLDOMElement_get_childNodes( element, &node_list );
924     ok( r == S_OK, "rets %08x\n", r);
925     ref = IXMLDOMNodeList_AddRef( node_list );
926     ok( ref == 2, "ref %d\n", ref );
927     IXMLDOMNodeList_Release( node_list );
928
929     IXMLDOMNodeList_get_item( node_list, 0, &node );
930     ok( r == S_OK, "rets %08x\n", r);
931
932     IXMLDOMNodeList_get_item( node_list, 0, &node2 );
933     ok( r == S_OK, "rets %08x\n", r);
934
935     ref = IXMLDOMNode_AddRef( node );
936     ok( ref == 2, "ref %d\n", ref );
937     IXMLDOMNode_Release( node );
938
939     ref = IXMLDOMNode_Release( node );
940     ok( ref == 0, "ref %d\n", ref );
941     ref = IXMLDOMNode_Release( node2 );
942     ok( ref == 0, "ref %d\n", ref );
943
944     ref = IXMLDOMNodeList_Release( node_list );
945     ok( ref == 0, "ref %d\n", ref );
946
947     ok( node != node2, "node %p node2 %p\n", node, node2 );
948
949     ref = IXMLDOMDocument_Release( doc );
950     ok( ref == 0, "ref %d\n", ref );
951
952     ref = IXMLDOMElement_AddRef( element );
953     todo_wine {
954     ok( ref == 3, "ref %d\n", ref );
955     }
956     IXMLDOMElement_Release( element );
957
958     /* IUnknown must be unique however we obtain it */
959     r = IXMLDOMElement_QueryInterface( element, &IID_IUnknown, (LPVOID*)&unk );
960     ok( r == S_OK, "rets %08x\n", r );
961     r = IXMLDOMElement_QueryInterface( element, &IID_IXMLDOMNode, (LPVOID*)&node );
962     ok( r == S_OK, "rets %08x\n", r );
963     r = IXMLDOMNode_QueryInterface( node, &IID_IUnknown, (LPVOID*)&unk2 );
964     ok( r == S_OK, "rets %08x\n", r );
965     ok( unk == unk2, "unk %p unk2 %p\n", unk, unk2 );
966
967     IUnknown_Release( unk2 );
968     IUnknown_Release( unk );
969     IXMLDOMNode_Release( node );
970
971     IXMLDOMElement_Release( element );
972
973 }
974
975 static void test_create(void)
976 {
977     HRESULT r;
978     VARIANT var;
979     BSTR str, name;
980     IXMLDOMDocument *doc;
981     IXMLDOMElement *element;
982     IXMLDOMNode *root, *node, *child;
983     IXMLDOMNamedNodeMap *attr_map;
984     IUnknown *unk;
985     LONG ref;
986     long num;
987
988     r = CoCreateInstance( &CLSID_DOMDocument, NULL, 
989         CLSCTX_INPROC_SERVER, &IID_IXMLDOMDocument, (LPVOID*)&doc );
990     if( r != S_OK )
991         return;
992
993     V_VT(&var) = VT_I4;
994     V_I4(&var) = NODE_ELEMENT;
995     str = SysAllocString( szlc );
996     r = IXMLDOMDocument_createNode( doc, var, str, NULL, &node );
997     ok( r == S_OK, "returns %08x\n", r );
998     r = IXMLDOMDocument_appendChild( doc, node, &root );
999     ok( r == S_OK, "returns %08x\n", r );
1000     ok( node == root, "%p %p\n", node, root );
1001
1002     ref = IXMLDOMNode_AddRef( node );
1003     ok(ref == 3, "ref %d\n", ref);
1004     IXMLDOMNode_Release( node );
1005
1006     ref = IXMLDOMNode_Release( node );
1007     ok(ref == 1, "ref %d\n", ref);
1008     SysFreeString( str );
1009
1010     V_VT(&var) = VT_I4;
1011     V_I4(&var) = NODE_ELEMENT;
1012     str = SysAllocString( szbs );
1013     r = IXMLDOMDocument_createNode( doc, var, str, NULL, &node );
1014     ok( r == S_OK, "returns %08x\n", r );
1015
1016     ref = IXMLDOMNode_AddRef( node );
1017     ok(ref == 2, "ref = %d\n", ref);
1018     IXMLDOMNode_Release( node );
1019
1020     r = IXMLDOMNode_QueryInterface( node, &IID_IUnknown, (LPVOID*)&unk );
1021     ok( r == S_OK, "returns %08x\n", r );
1022
1023     ref = IXMLDOMNode_AddRef( unk );
1024     ok(ref == 3, "ref = %d\n", ref);
1025     IXMLDOMNode_Release( unk );
1026
1027     V_VT(&var) = VT_EMPTY;
1028     r = IXMLDOMNode_insertBefore( root, (IXMLDOMNode*)unk, var, &child );
1029     ok( r == S_OK, "returns %08x\n", r );
1030     ok( unk == (IUnknown*)child, "%p %p\n", unk, child );
1031     IXMLDOMNode_Release( child );
1032     IUnknown_Release( unk );
1033
1034
1035     V_VT(&var) = VT_NULL;
1036     V_DISPATCH(&var) = (IDispatch*)node;
1037     r = IXMLDOMNode_insertBefore( root, node, var, &child );
1038     ok( r == S_OK, "returns %08x\n", r );
1039     ok( node == child, "%p %p\n", node, child );
1040     IXMLDOMNode_Release( child );
1041
1042
1043     V_VT(&var) = VT_NULL;
1044     V_DISPATCH(&var) = (IDispatch*)node;
1045     r = IXMLDOMNode_insertBefore( root, node, var, NULL );
1046     ok( r == S_OK, "returns %08x\n", r );
1047     IXMLDOMNode_Release( node );
1048
1049     r = IXMLDOMNode_QueryInterface( root, &IID_IXMLDOMElement, (LPVOID*)&element );
1050     ok( r == S_OK, "returns %08x\n", r );
1051
1052     r = IXMLDOMElement_get_attributes( element, &attr_map );
1053     ok( r == S_OK, "returns %08x\n", r );
1054     r = IXMLDOMNamedNodeMap_get_length( attr_map, &num );
1055     ok( r == S_OK, "returns %08x\n", r );
1056     ok( num == 0, "num %ld\n", num );
1057     IXMLDOMNamedNodeMap_Release( attr_map );
1058
1059     V_VT(&var) = VT_BSTR;
1060     V_BSTR(&var) = SysAllocString( szstr1 );
1061     name = SysAllocString( szdl );
1062     r = IXMLDOMElement_setAttribute( element, name, var );
1063     ok( r == S_OK, "returns %08x\n", r );
1064     r = IXMLDOMElement_get_attributes( element, &attr_map );
1065     ok( r == S_OK, "returns %08x\n", r );
1066     r = IXMLDOMNamedNodeMap_get_length( attr_map, &num );
1067     ok( r == S_OK, "returns %08x\n", r );
1068     ok( num == 1, "num %ld\n", num );
1069     IXMLDOMNamedNodeMap_Release( attr_map );
1070     VariantClear(&var);
1071
1072     V_VT(&var) = VT_BSTR;
1073     V_BSTR(&var) = SysAllocString( szstr2 );
1074     r = IXMLDOMElement_setAttribute( element, name, var );
1075     ok( r == S_OK, "returns %08x\n", r );
1076     r = IXMLDOMElement_get_attributes( element, &attr_map );
1077     ok( r == S_OK, "returns %08x\n", r );
1078     r = IXMLDOMNamedNodeMap_get_length( attr_map, &num );
1079     ok( r == S_OK, "returns %08x\n", r );
1080     ok( num == 1, "num %ld\n", num );
1081     IXMLDOMNamedNodeMap_Release( attr_map );
1082     VariantClear(&var);
1083     r = IXMLDOMElement_getAttribute( element, name, &var );
1084     ok( r == S_OK, "returns %08x\n", r );
1085     ok( !lstrcmpW(V_BSTR(&var), szstr2), "wrong attr value\n");
1086     VariantClear(&var);
1087     SysFreeString(name);
1088
1089     V_VT(&var) = VT_BSTR;
1090     V_BSTR(&var) = SysAllocString( szstr1 );
1091     name = SysAllocString( szlc );
1092     r = IXMLDOMElement_setAttribute( element, name, var );
1093     ok( r == S_OK, "returns %08x\n", r );
1094     r = IXMLDOMElement_get_attributes( element, &attr_map );
1095     ok( r == S_OK, "returns %08x\n", r );
1096     r = IXMLDOMNamedNodeMap_get_length( attr_map, &num );
1097     ok( r == S_OK, "returns %08x\n", r );
1098     ok( num == 2, "num %ld\n", num );
1099     IXMLDOMNamedNodeMap_Release( attr_map );
1100     VariantClear(&var);
1101     SysFreeString(name);
1102
1103     V_VT(&var) = VT_I4;
1104     V_I4(&var) = 10;
1105     name = SysAllocString( szbs );
1106     r = IXMLDOMElement_setAttribute( element, name, var );
1107     ok( r == S_OK, "returns %08x\n", r );
1108     VariantClear(&var);
1109     r = IXMLDOMElement_getAttribute( element, name, &var );
1110     ok( r == S_OK, "returns %08x\n", r );
1111     ok( V_VT(&var) == VT_BSTR, "variant type %x\n", V_VT(&var));
1112     VariantClear(&var);
1113     SysFreeString(name);
1114
1115     IXMLDOMElement_Release( element );
1116     IXMLDOMNode_Release( root );
1117     IXMLDOMDocument_Release( doc );
1118 }
1119
1120 static void test_getElementsByTagName(void)
1121 {
1122     HRESULT r;
1123     BSTR str;
1124     VARIANT_BOOL b;
1125     IXMLDOMDocument *doc;
1126     IXMLDOMNodeList *node_list;
1127     long len;
1128
1129     r = CoCreateInstance( &CLSID_DOMDocument, NULL, 
1130         CLSCTX_INPROC_SERVER, &IID_IXMLDOMDocument, (LPVOID*)&doc );
1131     if( r != S_OK )
1132         return;
1133
1134     str = SysAllocString( szComplete4 );
1135     r = IXMLDOMDocument_loadXML( doc, str, &b );
1136     ok( r == S_OK, "loadXML failed\n");
1137     ok( b == VARIANT_TRUE, "failed to load XML string\n");
1138     SysFreeString( str );
1139
1140     str = SysAllocString( szstar );
1141     r = IXMLDOMDocument_getElementsByTagName(doc, str, &node_list);
1142     ok( r == S_OK, "ret %08x\n", r );
1143     r = IXMLDOMNodeList_get_length( node_list, &len );
1144     ok( r == S_OK, "ret %08x\n", r );
1145     ok( len == 6, "len %ld\n", len );
1146     IXMLDOMNodeList_Release( node_list );
1147     SysFreeString( str );
1148
1149     str = SysAllocString( szbs );
1150     r = IXMLDOMDocument_getElementsByTagName(doc, str, &node_list);
1151     ok( r == S_OK, "ret %08x\n", r );
1152     r = IXMLDOMNodeList_get_length( node_list, &len );
1153     ok( r == S_OK, "ret %08x\n", r );
1154     ok( len == 1, "len %ld\n", len );
1155     IXMLDOMNodeList_Release( node_list );
1156     SysFreeString( str );
1157
1158     str = SysAllocString( szdl );
1159     r = IXMLDOMDocument_getElementsByTagName(doc, str, &node_list);
1160     ok( r == S_OK, "ret %08x\n", r );
1161     r = IXMLDOMNodeList_get_length( node_list, &len );
1162     ok( r == S_OK, "ret %08x\n", r );
1163     ok( len == 0, "len %ld\n", len );
1164     IXMLDOMNodeList_Release( node_list );
1165     SysFreeString( str );
1166
1167     str = SysAllocString( szstr1 );
1168     r = IXMLDOMDocument_getElementsByTagName(doc, str, &node_list);
1169     ok( r == S_OK, "ret %08x\n", r );
1170     r = IXMLDOMNodeList_get_length( node_list, &len );
1171     ok( r == S_OK, "ret %08x\n", r );
1172     ok( len == 0, "len %ld\n", len );
1173     IXMLDOMNodeList_Release( node_list );
1174     SysFreeString( str );
1175
1176     IXMLDOMDocument_Release( doc );
1177 }
1178
1179 static void test_get_text(void)
1180 {
1181     HRESULT r;
1182     BSTR str;
1183     VARIANT_BOOL b;
1184     IXMLDOMDocument *doc;
1185     IXMLDOMNode *node, *node2, *node3;
1186     IXMLDOMNodeList *node_list;
1187     IXMLDOMNamedNodeMap *node_map;
1188
1189     r = CoCreateInstance( &CLSID_DOMDocument, NULL, 
1190         CLSCTX_INPROC_SERVER, &IID_IXMLDOMDocument, (LPVOID*)&doc );
1191     if( r != S_OK )
1192         return;
1193
1194     str = SysAllocString( szComplete4 );
1195     r = IXMLDOMDocument_loadXML( doc, str, &b );
1196     ok( r == S_OK, "loadXML failed\n");
1197     ok( b == VARIANT_TRUE, "failed to load XML string\n");
1198     SysFreeString( str );
1199
1200     str = SysAllocString( szbs );
1201     r = IXMLDOMDocument_getElementsByTagName( doc, str, &node_list );
1202     ok( r == S_OK, "ret %08x\n", r );
1203     SysFreeString(str);
1204     
1205     r = IXMLDOMNodeList_get_item( node_list, 0, &node );
1206     ok( r == S_OK, "ret %08x\n", r ); 
1207     IXMLDOMNodeList_Release( node_list );
1208
1209     r = IXMLDOMNode_get_text( node, &str );
1210     ok( r == S_OK, "ret %08x\n", r );
1211 todo_wine {
1212     ok( !memcmp(str, szfn1_txt, sizeof(szfn1_txt)), "wrong string\n" );
1213  }
1214     ok( !memcmp(str, szfn1_txt, sizeof(szfn1_txt)-4), "wrong string\n" );
1215     SysFreeString(str);
1216
1217     r = IXMLDOMNode_get_attributes( node, &node_map );
1218     ok( r == S_OK, "ret %08x\n", r );
1219     
1220     str = SysAllocString( szvr );
1221     r = IXMLDOMNamedNodeMap_getNamedItem( node_map, str, &node2 );
1222     ok( r == S_OK, "ret %08x\n", r );
1223     SysFreeString(str);
1224
1225     r = IXMLDOMNode_get_text( node2, &str );
1226     ok( r == S_OK, "ret %08x\n", r );
1227     ok( !memcmp(str, szstr2, sizeof(szstr2)), "wrong string\n" );
1228     SysFreeString(str);
1229
1230     r = IXMLDOMNode_get_firstChild( node2, &node3 );
1231     ok( r == S_OK, "ret %08x\n", r );
1232
1233     r = IXMLDOMNode_get_text( node3, &str );
1234     ok( r == S_OK, "ret %08x\n", r );
1235     ok( !memcmp(str, szstr2, sizeof(szstr2)), "wrong string\n" );
1236     SysFreeString(str);
1237
1238
1239     IXMLDOMNode_Release( node3 );
1240     IXMLDOMNode_Release( node2 );
1241     IXMLDOMNamedNodeMap_Release( node_map );
1242     IXMLDOMNode_Release( node );
1243     IXMLDOMDocument_Release( doc );
1244 }
1245
1246 static void test_get_childNodes(void)
1247 {
1248     HRESULT r;
1249     BSTR str;
1250     VARIANT_BOOL b;
1251     IXMLDOMDocument *doc;
1252     IXMLDOMElement *element;
1253     IXMLDOMNode *node, *node2;
1254     IXMLDOMNodeList *node_list, *node_list2;
1255     long len;
1256
1257     r = CoCreateInstance( &CLSID_DOMDocument, NULL, 
1258         CLSCTX_INPROC_SERVER, &IID_IXMLDOMDocument, (LPVOID*)&doc );
1259     if( r != S_OK )
1260         return;
1261
1262     str = SysAllocString( szComplete4 );
1263     r = IXMLDOMDocument_loadXML( doc, str, &b );
1264     ok( r == S_OK, "loadXML failed\n");
1265     ok( b == VARIANT_TRUE, "failed to load XML string\n");
1266     SysFreeString( str );
1267
1268     r = IXMLDOMDocument_get_documentElement( doc, &element );
1269     ok( r == S_OK, "ret %08x\n", r);
1270
1271     r = IXMLDOMElement_get_childNodes( element, &node_list );
1272     ok( r == S_OK, "ret %08x\n", r);
1273
1274     r = IXMLDOMNodeList_get_length( node_list, &len );
1275     ok( r == S_OK, "ret %08x\n", r);
1276     ok( len == 4, "len %ld\n", len);
1277
1278     r = IXMLDOMNodeList_get_item( node_list, 2, &node );
1279     ok( r == S_OK, "ret %08x\n", r);
1280
1281     r = IXMLDOMNode_get_childNodes( node, &node_list2 );
1282     ok( r == S_OK, "ret %08x\n", r);
1283
1284     r = IXMLDOMNodeList_get_length( node_list2, &len );
1285     ok( r == S_OK, "ret %08x\n", r);
1286     ok( len == 0, "len %ld\n", len);
1287
1288     r = IXMLDOMNodeList_get_item( node_list2, 0, &node2);
1289     ok( r == S_FALSE, "ret %08x\n", r);
1290
1291     IXMLDOMNodeList_Release( node_list2 );
1292     IXMLDOMNode_Release( node );
1293     IXMLDOMNodeList_Release( node_list );
1294     IXMLDOMElement_Release( element );
1295     IXMLDOMDocument_Release( doc );
1296 }
1297
1298 static void test_removeChild(void)
1299 {
1300     HRESULT r;
1301     BSTR str;
1302     VARIANT_BOOL b;
1303     IXMLDOMDocument *doc;
1304     IXMLDOMElement *element;
1305     IXMLDOMNode *node, *node2, *node3, *node4;
1306     IXMLDOMNodeList *node_list, *node_list2;
1307
1308     r = CoCreateInstance( &CLSID_DOMDocument, NULL, 
1309         CLSCTX_INPROC_SERVER, &IID_IXMLDOMDocument, (LPVOID*)&doc );
1310     if( r != S_OK )
1311         return;
1312
1313     str = SysAllocString( szComplete4 );
1314     r = IXMLDOMDocument_loadXML( doc, str, &b );
1315     ok( r == S_OK, "loadXML failed\n");
1316     ok( b == VARIANT_TRUE, "failed to load XML string\n");
1317     SysFreeString( str );
1318
1319     r = IXMLDOMDocument_get_documentElement( doc, &element );
1320     ok( r == S_OK, "ret %08x\n", r);
1321
1322     r = IXMLDOMElement_get_childNodes( element, &node_list );
1323     ok( r == S_OK, "ret %08x\n", r);
1324
1325     r = IXMLDOMNodeList_get_item( node_list, 3, &node );
1326     ok( r == S_OK, "ret %08x\n", r);
1327  
1328     r = IXMLDOMNode_get_childNodes( node, &node_list2 );
1329     ok( r == S_OK, "ret %08x\n", r);
1330  
1331     r = IXMLDOMNodeList_get_item( node_list, 0, &node4 );
1332     ok( r == S_OK, "ret %08x\n", r);
1333
1334     r = IXMLDOMElement_removeChild( element, NULL, &node2 );
1335     ok( r == E_INVALIDARG, "ret %08x\n", r );
1336
1337     r = IXMLDOMElement_removeChild( element, node4, &node2 );
1338     ok( r == S_OK, "ret %08x\n", r);
1339     ok( node4 == node2, "node %p node2 %p\n", node4, node2 );
1340
1341     r = IXMLDOMNode_get_parentNode( node4, &node3 );
1342     ok( r == S_FALSE, "ret %08x\n", r);
1343     ok( node3 == NULL, "%p\n", node3 );
1344
1345     IXMLDOMNode_Release( node2 );
1346     IXMLDOMNode_Release( node4 );
1347     IXMLDOMNodeList_Release( node_list2 );
1348     IXMLDOMNode_Release( node );
1349     IXMLDOMNodeList_Release( node_list );
1350     IXMLDOMElement_Release( element );
1351     IXMLDOMDocument_Release( doc );
1352 }
1353
1354 static void test_XMLHTTP(void)
1355 {
1356     static const WCHAR wszBody[] = {'m','o','d','e','=','T','e','s','t',0};
1357     static WCHAR wszPOST[] = {'P','O','S','T',0};
1358     static WCHAR wszUrl[] = {'h','t','t','p',':','/','/',
1359         'c','r','o','s','s','o','v','e','r','.','c','o','d','e','w','e','a','v','e','r','s','.','c','o','m','/',
1360         'p','o','s','t','t','e','s','t','.','p','h','p',0};
1361     static const WCHAR wszExpectedResponse[] = {'F','A','I','L','E','D',0};
1362     IXMLHttpRequest *pXMLHttpRequest;
1363     BSTR bstrResponse;
1364     VARIANT dummy;
1365     VARIANT varfalse;
1366     VARIANT varbody;
1367     HRESULT hr = CoCreateInstance(&CLSID_XMLHTTPRequest, NULL,
1368                                   CLSCTX_INPROC_SERVER, &IID_IXMLHttpRequest,
1369                                   (void **)&pXMLHttpRequest);
1370     todo_wine {
1371     ok(hr == S_OK, "CoCreateInstance(CLSID_XMLHTTPRequest) should have succeeded instead of failing with 0x%08x\n", hr);
1372     }
1373     if (hr != S_OK)
1374         return;
1375
1376     VariantInit(&dummy);
1377     V_VT(&dummy) = VT_ERROR;
1378     V_ERROR(&dummy) = DISP_E_MEMBERNOTFOUND;
1379     VariantInit(&varfalse);
1380     V_VT(&varfalse) = VT_BOOL;
1381     V_BOOL(&varfalse) = VARIANT_FALSE;
1382     V_VT(&varbody) = VT_BSTR;
1383     V_BSTR(&varbody) = SysAllocString(wszBody);
1384
1385     hr = IXMLHttpRequest_open(pXMLHttpRequest, wszPOST, wszUrl, varfalse, dummy, dummy);
1386     ok(hr == S_OK, "IXMLHttpRequest_open should have succeeded instead of failing with 0x%08x\n", hr);
1387
1388     hr = IXMLHttpRequest_send(pXMLHttpRequest, varbody);
1389     ok(hr == S_OK, "IXMLHttpRequest_send should have succeeded instead of failing with 0x%08x\n", hr);
1390     VariantClear(&varbody);
1391
1392     hr = IXMLHttpRequest_get_responseText(pXMLHttpRequest, &bstrResponse);
1393     ok(hr == S_OK, "IXMLHttpRequest_get_responseText should have succeeded instead of failing with 0x%08x\n", hr);
1394     /* the server currently returns "FAILED" because the Content-Type header is
1395      * not what the server expects */
1396     ok(!memcmp(bstrResponse, wszExpectedResponse, sizeof(wszExpectedResponse)), "bstrResponse differs from what was expected\n");
1397     SysFreeString(bstrResponse);
1398 }
1399
1400 static void test_IXMLDOMDocument2(void)
1401 {
1402     HRESULT r;
1403     VARIANT_BOOL b;
1404     BSTR str;
1405     IXMLDOMDocument *doc;
1406     IXMLDOMDocument2 *doc2;
1407     VARIANT var;
1408     int ref;
1409
1410     r = CoCreateInstance( &CLSID_DOMDocument, NULL,
1411         CLSCTX_INPROC_SERVER, &IID_IXMLDOMDocument, (LPVOID*)&doc );
1412     if( r != S_OK )
1413         return;
1414
1415     str = SysAllocString( szComplete4 );
1416     r = IXMLDOMDocument_loadXML( doc, str, &b );
1417     ok( r == S_OK, "loadXML failed\n");
1418     ok( b == VARIANT_TRUE, "failed to load XML string\n");
1419     SysFreeString( str );
1420
1421     r = IXMLDOMDocument_QueryInterface( doc, &IID_IXMLDOMDocument2, (void**)&doc2 );
1422     ok( r == S_OK, "ret %08x\n", r );
1423     ok( doc == (IXMLDOMDocument*)doc2, "interfaces differ\n");
1424
1425     /* we will check if the variant got cleared */
1426     ref = IXMLDOMDocument2_AddRef(doc2);
1427     expect_eq(ref, 3, int, "%d");  /* doc, doc2, AddRef*/
1428     V_VT(&var) = VT_UNKNOWN;
1429     V_UNKNOWN(&var) = (IUnknown *)doc2;
1430
1431     /* invalid calls */
1432     ole_expect(IXMLDOMDocument2_getProperty(doc2, _bstr_("askldhfaklsdf"), &var), E_FAIL);
1433     expect_eq(V_VT(&var), VT_UNKNOWN, int, "%x");
1434     ole_expect(IXMLDOMDocument2_getProperty(doc2, _bstr_("SelectionLanguage"), NULL), E_INVALIDARG);
1435
1436     /* valid call */
1437     ole_check(IXMLDOMDocument2_getProperty(doc2, _bstr_("SelectionLanguage"), &var));
1438     expect_eq(V_VT(&var), VT_BSTR, int, "%x");
1439     expect_bstr_eq_and_free(V_BSTR(&var), "XSLPattern");
1440     V_VT(&var) = VT_R4;
1441
1442     /* the variant didn't get cleared*/
1443     expect_eq(IXMLDOMDocument2_Release(doc2), 2, int, "%d");
1444
1445     /* setProperty tests */
1446     ole_expect(IXMLDOMDocument2_setProperty(doc2, _bstr_("askldhfaklsdf"), var), E_FAIL);
1447     ole_expect(IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionLanguage"), var), E_FAIL);
1448     ole_expect(IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionLanguage"), _variantbstr_("alskjdh faklsjd hfk")), E_FAIL);
1449     ole_check(IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionLanguage"), _variantbstr_("XSLPattern")));
1450     ole_check(IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionLanguage"), _variantbstr_("XPath")));
1451     ole_check(IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionLanguage"), _variantbstr_("XSLPattern")));
1452
1453     /* contrary to what MSDN calims you can switch back from XPath to XSLPattern */
1454     ole_check(IXMLDOMDocument2_getProperty(doc2, _bstr_("SelectionLanguage"), &var));
1455     expect_eq(V_VT(&var), VT_BSTR, int, "%x");
1456     expect_bstr_eq_and_free(V_BSTR(&var), "XSLPattern");
1457
1458     IXMLDOMDocument2_Release( doc2 );
1459     IXMLDOMDocument_Release( doc );
1460     free_bstrs();
1461 }
1462
1463 static void test_XPath(void)
1464 {
1465     HRESULT r;
1466     VARIANT var;
1467     VARIANT_BOOL b;
1468     IXMLDOMDocument2 *doc;
1469     IXMLDOMNode *rootNode;
1470     IXMLDOMNode *elem1Node;
1471     IXMLDOMNode *node;
1472     IXMLDOMNodeList *list;
1473
1474     r = CoCreateInstance( &CLSID_DOMDocument, NULL,
1475         CLSCTX_INPROC_SERVER, &IID_IXMLDOMDocument2, (LPVOID*)&doc );
1476     if( r != S_OK )
1477         return;
1478
1479     ole_check(IXMLDOMDocument_loadXML(doc, _bstr_(szExampleXML), &b));
1480     ok(b == VARIANT_TRUE, "failed to load XML string\n");
1481
1482     /* switch to XPath */
1483     ole_check(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionLanguage"), _variantbstr_("XPath")));
1484
1485     /* some simple queries*/
1486     ole_check(IXMLDOMDocument_selectNodes(doc, _bstr_("root"), &list));
1487     ole_check(IXMLDOMNodeList_get_item(list, 0, &rootNode));
1488     ole_check(IXMLDOMNodeList_reset(list));
1489     expect_list_and_release(list, "E2.D1");
1490     if (rootNode == NULL)
1491         return;
1492
1493     ole_check(IXMLDOMDocument_selectNodes(doc, _bstr_("root//c"), &list));
1494     expect_list_and_release(list, "E3.E1.E2.D1 E3.E2.E2.D1");
1495
1496     ole_check(IXMLDOMDocument_selectNodes(doc, _bstr_("//c[@type]"), &list));
1497     expect_list_and_release(list, "E3.E2.E2.D1");
1498
1499     ole_check(IXMLDOMNode_selectNodes(rootNode, _bstr_("elem"), &list));
1500     /* using get_item for query results advances the position */
1501     ole_check(IXMLDOMNodeList_get_item(list, 1, &node));
1502     expect_node(node, "E2.E2.D1");
1503     IXMLDOMNode_Release(node);
1504     ole_check(IXMLDOMNodeList_nextNode(list, &node));
1505     expect_node(node, "E4.E2.D1");
1506     IXMLDOMNode_Release(node);
1507     ole_check(IXMLDOMNodeList_reset(list));
1508     expect_list_and_release(list, "E1.E2.D1 E2.E2.D1 E4.E2.D1");
1509
1510     ole_check(IXMLDOMNode_selectNodes(rootNode, _bstr_("."), &list));
1511     expect_list_and_release(list, "E2.D1");
1512
1513     ole_check(IXMLDOMNode_selectNodes(rootNode, _bstr_("elem[3]/preceding-sibling::*"), &list));
1514     ole_check(IXMLDOMNodeList_get_item(list, 0, &elem1Node));
1515     ole_check(IXMLDOMNodeList_reset(list));
1516     expect_list_and_release(list, "E1.E2.D1 E2.E2.D1 E3.E2.D1");
1517
1518     /* select an attribute */
1519     ole_check(IXMLDOMNode_selectNodes(rootNode, _bstr_(".//@type"), &list));
1520     expect_list_and_release(list, "A'type'.E3.E2.E2.D1");
1521
1522     /* would evaluate to a number */
1523     ole_expect(IXMLDOMNode_selectNodes(rootNode, _bstr_("count(*)"), &list), E_FAIL);
1524     /* would evaluate to a boolean */
1525     ole_expect(IXMLDOMNode_selectNodes(rootNode, _bstr_("position()>0"), &list), E_FAIL);
1526     /* would evaluate to a string */
1527     ole_expect(IXMLDOMNode_selectNodes(rootNode, _bstr_("name()"), &list), E_FAIL);
1528
1529     /* no results */
1530     ole_check(IXMLDOMNode_selectNodes(rootNode, _bstr_("c"), &list));
1531     expect_list_and_release(list, "");
1532     ole_check(IXMLDOMDocument_selectNodes(doc, _bstr_("elem//c"), &list));
1533     expect_list_and_release(list, "");
1534     ole_check(IXMLDOMDocument_selectNodes(doc, _bstr_("//elem[4]"), &list));
1535     expect_list_and_release(list, "");
1536
1537     /* foo undeclared in document node */
1538     ole_expect(IXMLDOMDocument_selectNodes(doc, _bstr_("root//foo:c"), &list), E_FAIL);
1539     /* undeclared in <root> node */
1540     ole_expect(IXMLDOMNode_selectNodes(rootNode, _bstr_(".//foo:c"), &list), E_FAIL);
1541     /* undeclared in <elem> node */
1542     ole_expect(IXMLDOMNode_selectNodes(elem1Node, _bstr_("//foo:c"), &list), E_FAIL);
1543     /* but this trick can be used */
1544     ole_check(IXMLDOMNode_selectNodes(elem1Node, _bstr_("//*[name()='foo:c']"), &list));
1545     expect_list_and_release(list, "E3.E4.E2.D1");
1546
1547     /* it has to be declared in SelectionNamespaces */
1548     todo_wine ole_check(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionNamespaces"),
1549         _variantbstr_("xmlns:test='urn:uuid:86B2F87F-ACB6-45cd-8B77-9BDB92A01A29'")));
1550
1551     /* now the namespace can be used */
1552     todo_wine ole_check(IXMLDOMDocument_selectNodes(doc, _bstr_("root//test:c"), &list));
1553     todo_wine expect_list_and_release(list, "E3.E3.E2.D1 E3.E4.E2.D1");
1554     todo_wine ole_check(IXMLDOMNode_selectNodes(rootNode, _bstr_(".//test:c"), &list));
1555     todo_wine expect_list_and_release(list, "E3.E3.E2.D1 E3.E4.E2.D1");
1556     todo_wine ole_check(IXMLDOMNode_selectNodes(elem1Node, _bstr_("//test:c"), &list));
1557     todo_wine expect_list_and_release(list, "E3.E3.E2.D1 E3.E4.E2.D1");
1558     todo_wine ole_check(IXMLDOMNode_selectNodes(elem1Node, _bstr_(".//test:x"), &list));
1559     todo_wine expect_list_and_release(list, "E5.E1.E4.E1.E2.D1");
1560
1561     /* SelectionNamespaces syntax error - the namespaces doesn't work anymore but the value is stored */
1562     ole_expect(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionNamespaces"),
1563         _variantbstr_("xmlns:test='urn:uuid:86B2F87F-ACB6-45cd-8B77-9BDB92A01A29' xmlns:foo=###")), E_FAIL);
1564
1565     ole_expect(IXMLDOMDocument_selectNodes(doc, _bstr_("root//foo:c"), &list), E_FAIL);
1566
1567     todo_wine ole_check(IXMLDOMDocument2_getProperty(doc, _bstr_("SelectionNamespaces"), &var));
1568     todo_wine expect_eq(V_VT(&var), VT_BSTR, int, "%x");
1569     if (V_VT(&var) == VT_BSTR)
1570         expect_bstr_eq_and_free(V_BSTR(&var), "xmlns:test='urn:uuid:86B2F87F-ACB6-45cd-8B77-9BDB92A01A29' xmlns:foo=###");
1571
1572     /* extra attributes - same thing*/
1573     ole_expect(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionNamespaces"),
1574         _variantbstr_("xmlns:test='urn:uuid:86B2F87F-ACB6-45cd-8B77-9BDB92A01A29' param='test'")), E_FAIL);
1575     ole_expect(IXMLDOMDocument_selectNodes(doc, _bstr_("root//foo:c"), &list), E_FAIL);
1576
1577     IXMLDOMNode_Release(rootNode);
1578     IXMLDOMNode_Release(elem1Node);
1579     IXMLDOMDocument_Release(doc);
1580     free_bstrs();
1581 }
1582
1583 START_TEST(domdoc)
1584 {
1585     HRESULT r;
1586
1587     r = CoInitialize( NULL );
1588     ok( r == S_OK, "failed to init com\n");
1589
1590     test_domdoc();
1591     test_domnode();
1592     test_refs();
1593     test_create();
1594     test_getElementsByTagName();
1595     test_get_text();
1596     test_get_childNodes();
1597     test_removeChild();
1598     test_XMLHTTP();
1599     test_IXMLDOMDocument2();
1600     test_XPath();
1601
1602     CoUninitialize();
1603 }