msxml3: nextSibling not supported for Attribute, Document and Document Fragment Nodes.
[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 static WCHAR szComment[] = {'A',' ','C','o','m','m','e','n','t',0 };
139 static WCHAR szCommentXML[] = {'<','!','-','-','A',' ','C','o','m','m','e','n','t','-','-','>',0 };
140 static WCHAR szCommentNodeText[] = {'#','c','o','m','m','e','n','t',0 };
141
142 static WCHAR szElement[] = {'E','l','e','T','e','s','t', 0 };
143 static WCHAR szElementXML[]  = {'<','E','l','e','T','e','s','t','/','>',0 };
144 static WCHAR szElementXML2[] = {'<','E','l','e','T','e','s','t',' ','A','t','t','r','=','"','"','/','>',0 };
145
146 static WCHAR szAttribute[] = {'A','t','t','r',0 };
147 static WCHAR szAttributeXML[] = {'A','t','t','r','=','"','"',0 };
148
149 #define expect_bstr_eq_and_free(bstr, expect) { \
150     BSTR bstrExp = alloc_str_from_narrow(expect); \
151     ok(lstrcmpW(bstr, bstrExp) == 0, "String differs\n"); \
152     SysFreeString(bstr); \
153     SysFreeString(bstrExp); \
154 }
155
156 #define expect_eq(expr, value, type, format) { type ret = (expr); ok((value) == ret, #expr " expected " format " got " format "\n", value, ret); }
157
158 #define ole_check(expr) { \
159     HRESULT r = expr; \
160     ok(r == S_OK, #expr " returned %x\n", r); \
161 }
162
163 #define ole_expect(expr, expect) { \
164     HRESULT r = expr; \
165     ok(r == (expect), #expr " returned %x, expected %x\n", r, expect); \
166 }
167
168 static BSTR alloc_str_from_narrow(const char *str)
169 {
170     int len = MultiByteToWideChar(CP_ACP, 0, str, -1, NULL, 0);
171     BSTR ret = SysAllocStringLen(NULL, len - 1);  /* NUL character added automatically */
172     MultiByteToWideChar(CP_ACP, 0, str, -1, ret, len);
173     return ret;
174 }
175
176 BSTR alloced_bstrs[256];
177 int alloced_bstrs_count = 0;
178
179 static BSTR _bstr_(const char *str)
180 {
181     assert(alloced_bstrs_count < sizeof(alloced_bstrs)/sizeof(alloced_bstrs[0]));
182     alloced_bstrs[alloced_bstrs_count] = alloc_str_from_narrow(str);
183     return alloced_bstrs[alloced_bstrs_count++];
184 }
185
186 static void free_bstrs(void)
187 {
188     int i;
189     for (i = 0; i < alloced_bstrs_count; i++)
190         SysFreeString(alloced_bstrs[i]);
191     alloced_bstrs_count = 0;
192 }
193
194 static VARIANT _variantbstr_(const char *str)
195 {
196     VARIANT v;
197     V_VT(&v) = VT_BSTR;
198     V_BSTR(&v) = _bstr_(str);
199     return v;
200 }
201
202 static void get_str_for_type(DOMNodeType type, char *buf)
203 {
204     switch (type)
205     {
206         case NODE_ATTRIBUTE:
207             strcpy(buf, "A");
208             break;
209         case NODE_ELEMENT:
210             strcpy(buf, "E");
211             break;
212         case NODE_DOCUMENT:
213             strcpy(buf, "D");
214             break;
215         default:
216             wsprintfA(buf, "[%d]", type);
217     }
218 }
219
220 static int get_node_position(IXMLDOMNode *node)
221 {
222     HRESULT r;
223     int pos = 0;
224
225     IXMLDOMNode_AddRef(node);
226     do
227     {
228         IXMLDOMNode *new_node;
229
230         pos++;
231         r = IXMLDOMNode_get_previousSibling(node, &new_node);
232         ok(!FAILED(r), "get_previousSibling failed\n");
233         IXMLDOMNode_Release(node);
234         node = new_node;
235     } while (r == S_OK);
236     return pos;
237 }
238
239 static void node_to_string(IXMLDOMNode *node, char *buf)
240 {
241     HRESULT r = S_OK;
242     DOMNodeType type;
243
244     if (node == NULL)
245     {
246         lstrcpyA(buf, "(null)");
247         return;
248     }
249
250     IXMLDOMNode_AddRef(node);
251     while (r == S_OK)
252     {
253         IXMLDOMNode *new_node;
254
255         ole_check(IXMLDOMNode_get_nodeType(node, &type));
256         get_str_for_type(type, buf);
257         buf+=strlen(buf);
258
259         if (type == NODE_ATTRIBUTE)
260         {
261             BSTR bstr;
262             ole_check(IXMLDOMNode_get_nodeName(node, &bstr));
263             *(buf++) = '\'';
264             wsprintfA(buf, "%ws", bstr);
265             buf += strlen(buf);
266             *(buf++) = '\'';
267             SysFreeString(bstr);
268
269             r = IXMLDOMNode_selectSingleNode(node, _bstr_(".."), &new_node);
270         }
271         else
272         {
273             int pos = get_node_position(node);
274             DOMNodeType parent_type = NODE_INVALID;
275             r = IXMLDOMNode_get_parentNode(node, &new_node);
276
277             /* currently wine doesn't create a node for the <?xml ... ?>. To be able to test query
278              * results we "fix" it */
279             if (r == S_OK)
280                 ole_check(IXMLDOMNode_get_nodeType(node, &parent_type));
281             /* we need also to workaround the no document node problem - see below */
282             if (((r == S_FALSE && type != NODE_DOCUMENT) || parent_type == NODE_DOCUMENT) && type != NODE_PROCESSING_INSTRUCTION && pos==1)
283             {
284                 todo_wine ok(FALSE, "The first child of the document node in MSXML is the <?xml ... ?> processing instruction\n");
285                 pos++;
286             }
287             wsprintf(buf, "%d", pos);
288             buf += strlen(buf);
289         }
290
291         ok(!FAILED(r), "get_parentNode failed (%08x)\n", r);
292         IXMLDOMNode_Release(node);
293         node = new_node;
294         if (r == S_OK)
295             *(buf++) = '.';
296     }
297
298     /* currently we can't access document node in wine. All our examples this is the
299      * root node so to be able to test query results we add it */
300     if (type != NODE_DOCUMENT)
301     {
302         todo_wine ok(FALSE, "Document node is not the last returned node!\n");
303         *(buf++) = '.';
304         *(buf++) = 'D';
305         *(buf++) = '1';
306     }
307     *buf = 0;
308 }
309
310 static char *list_to_string(IXMLDOMNodeList *list)
311 {
312     static char buf[4096];
313     char *pos = buf;
314     long len = 0;
315     int i;
316
317     if (list == NULL)
318     {
319         lstrcpyA(buf, "(null)");
320         return buf;
321     }
322     ole_check(IXMLDOMNodeList_get_length(list, &len));
323     for (i = 0; i < len; i++)
324     {
325         IXMLDOMNode *node;
326         if (i > 0)
327             *(pos++) = ' ';
328         ole_check(IXMLDOMNodeList_nextNode(list, &node));
329         node_to_string(node, pos);
330         pos += strlen(pos);
331         IXMLDOMNode_Release(node);
332     }
333     *pos = 0;
334     return buf;
335 }
336
337 #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); }
338 #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); }
339
340 static void test_domdoc( void )
341 {
342     HRESULT r;
343     IXMLDOMDocument *doc = NULL;
344     IXMLDOMParseError *error;
345     IXMLDOMElement *element = NULL;
346     IXMLDOMNode *node;
347     IXMLDOMText *nodetext = NULL;
348     IXMLDOMComment *node_comment = NULL;
349     IXMLDOMAttribute *node_attr = NULL;
350     VARIANT_BOOL b;
351     VARIANT var;
352     BSTR str;
353     long code;
354
355     r = CoCreateInstance( &CLSID_DOMDocument, NULL, 
356         CLSCTX_INPROC_SERVER, &IID_IXMLDOMDocument, (LPVOID*)&doc );
357     if( r != S_OK )
358         return;
359
360     /* try some stupid things */
361     r = IXMLDOMDocument_loadXML( doc, NULL, NULL );
362     ok( r == S_FALSE, "loadXML failed\n");
363
364     b = VARIANT_TRUE;
365     r = IXMLDOMDocument_loadXML( doc, NULL, &b );
366     ok( r == S_FALSE, "loadXML failed\n");
367     ok( b == VARIANT_FALSE, "failed to load XML string\n");
368
369     /* try to load a document from a nonexistent file */
370     b = VARIANT_TRUE;
371     str = SysAllocString( szNonExistentFile );
372     VariantInit(&var);
373     V_VT(&var) = VT_BSTR;
374     V_BSTR(&var) = str;
375
376     r = IXMLDOMDocument_load( doc, var, &b);
377     ok( r == S_FALSE, "load (from file) failed\n");
378     ok( b == VARIANT_FALSE, "failed to load XML file\n");
379     SysFreeString( str );
380
381     /* try load an empty document */
382     b = VARIANT_TRUE;
383     str = SysAllocString( szEmpty );
384     r = IXMLDOMDocument_loadXML( doc, str, &b );
385     ok( r == S_FALSE, "loadXML failed\n");
386     ok( b == VARIANT_FALSE, "failed to load XML string\n");
387     SysFreeString( str );
388
389     /* check that there's no document element */
390     element = NULL;
391     r = IXMLDOMDocument_get_documentElement( doc, &element );
392     ok( r == S_FALSE, "should be no document element\n");
393
394     /* try finding a node */
395     node = NULL;
396     str = SysAllocString( szstr1 );
397     r = IXMLDOMDocument_selectSingleNode( doc, str, &node );
398     ok( r == S_FALSE, "ret %08x\n", r );
399     SysFreeString( str );
400
401     b = VARIANT_TRUE;
402     str = SysAllocString( szIncomplete );
403     r = IXMLDOMDocument_loadXML( doc, str, &b );
404     ok( r == S_FALSE, "loadXML failed\n");
405     ok( b == VARIANT_FALSE, "failed to load XML string\n");
406     SysFreeString( str );
407
408     /* check that there's no document element */
409     element = (IXMLDOMElement*)1;
410     r = IXMLDOMDocument_get_documentElement( doc, &element );
411     ok( r == S_FALSE, "should be no document element\n");
412     ok( element == NULL, "Element should be NULL\n");
413
414     /* try to load something valid */
415     b = VARIANT_FALSE;
416     str = SysAllocString( szComplete1 );
417     r = IXMLDOMDocument_loadXML( doc, str, &b );
418     ok( r == S_OK, "loadXML failed\n");
419     ok( b == VARIANT_TRUE, "failed to load XML string\n");
420     SysFreeString( str );
421
422     /* try with a null out pointer */
423     r = IXMLDOMDocument_get_documentElement( doc, NULL );
424     ok( r == E_INVALIDARG, "should be no document element\n");
425
426     /* check if nodename is correct */
427     r = IXMLDOMDocument_get_nodeName( doc, NULL );
428     ok ( r == E_INVALIDARG, "get_nodeName (NULL) wrong code\n");
429
430     /* content doesn't matter here */
431     str = NULL;
432     r = IXMLDOMDocument_get_nodeName( doc, &str );
433     ok ( r == S_OK, "get_nodeName wrong code\n");
434     ok ( str != NULL, "str is null\n");
435     ok( !lstrcmpW( str, szDocument ), "incorrect nodeName\n");
436     SysFreeString( str );
437
438
439     /* check that there's a document element */
440     element = NULL;
441     r = IXMLDOMDocument_get_documentElement( doc, &element );
442     ok( r == S_OK, "should be a document element\n");
443     if( element )
444     {
445         BSTR tag = NULL;
446
447         /* check if the tag is correct */
448         r = IXMLDOMElement_get_tagName( element, &tag );
449         ok( r == S_OK, "couldn't get tag name\n");
450         ok( tag != NULL, "tag was null\n");
451         ok( !lstrcmpW( tag, szOpen ), "incorrect tag name\n");
452         SysFreeString( tag );
453
454         /* figure out what happens if we try to reload the document */
455         str = SysAllocString( szComplete2 );
456         r = IXMLDOMDocument_loadXML( doc, str, &b );
457         ok( r == S_OK, "loadXML failed\n");
458         ok( b == VARIANT_TRUE, "failed to load XML string\n");
459         SysFreeString( str );
460
461         /* check if the tag is still correct */
462         tag = NULL;
463         r = IXMLDOMElement_get_tagName( element, &tag );
464         ok( r == S_OK, "couldn't get tag name\n");
465         ok( tag != NULL, "tag was null\n");
466         ok( !lstrcmpW( tag, szOpen ), "incorrect tag name\n");
467         SysFreeString( tag );
468
469         IXMLDOMElement_Release( element );
470         element = NULL;
471     }
472
473     /* as soon as we call loadXML again, the document element will disappear */
474     b = 2;
475     r = IXMLDOMDocument_loadXML( doc, NULL, NULL );
476     ok( r == S_FALSE, "loadXML failed\n");
477     ok( b == 2, "variant modified\n");
478     r = IXMLDOMDocument_get_documentElement( doc, &element );
479     ok( r == S_FALSE, "should be no document element\n");
480
481     /* try to load something else simple and valid */
482     b = VARIANT_FALSE;
483     str = SysAllocString( szComplete3 );
484     r = IXMLDOMDocument_loadXML( doc, str, &b );
485     ok( r == S_OK, "loadXML failed\n");
486     ok( b == VARIANT_TRUE, "failed to load XML string\n");
487     SysFreeString( str );
488
489     /* try something a little more complicated */
490     b = FALSE;
491     str = SysAllocString( szComplete4 );
492     r = IXMLDOMDocument_loadXML( doc, str, &b );
493     ok( r == S_OK, "loadXML failed\n");
494     ok( b == VARIANT_TRUE, "failed to load XML string\n");
495     SysFreeString( str );
496
497     r = IXMLDOMDocument_get_parseError( doc, &error );
498     ok( r == S_OK, "returns %08x\n", r );
499
500     r = IXMLDOMParseError_get_errorCode( error, &code );
501     ok( r == S_FALSE, "returns %08x\n", r );
502     ok( code == 0, "code %ld\n", code );
503     IXMLDOMParseError_Release( error );
504
505      /* test createTextNode */
506     str = SysAllocString( szOpen );
507     r = IXMLDOMDocument_createTextNode(doc, str, NULL);
508     ok( r == E_INVALIDARG, "returns %08x\n", r );
509     r = IXMLDOMDocument_createTextNode(doc, str, &nodetext);
510     ok( r == S_OK, "returns %08x\n", r );
511     IXMLDOMText_Release( nodetext );
512     SysFreeString( str );
513
514     /* test Create Comment */
515     r = IXMLDOMDocument_createComment(doc, NULL, NULL);
516     ok( r == E_INVALIDARG, "returns %08x\n", r );
517     r = IXMLDOMDocument_createComment(doc, szComment, &node_comment);
518     ok( r == S_OK, "returns %08x\n", r );
519     IXMLDOMText_Release( node_comment );
520
521     /* test Create Attribute */
522     r = IXMLDOMDocument_createAttribute(doc, NULL, NULL);
523     ok( r == E_INVALIDARG, "returns %08x\n", r );
524     r = IXMLDOMDocument_createAttribute(doc, szAttribute, &node_attr);
525     ok( r == S_OK, "returns %08x\n", r );
526     IXMLDOMText_Release( node_attr);
527
528     r = IXMLDOMDocument_Release( doc );
529     ok( r == 0, "document ref count incorrect\n");
530
531 }
532
533 static void test_domnode( void )
534 {
535     HRESULT r;
536     IXMLDOMDocument *doc = NULL, *owner = NULL;
537     IXMLDOMElement *element = NULL;
538     IXMLDOMNamedNodeMap *map = NULL;
539     IXMLDOMNode *node = NULL, *next = NULL;
540     IXMLDOMNodeList *list = NULL;
541     DOMNodeType type = NODE_INVALID;
542     VARIANT_BOOL b;
543     BSTR str;
544     VARIANT var;
545     long count;
546
547     r = CoCreateInstance( &CLSID_DOMDocument, NULL, 
548         CLSCTX_INPROC_SERVER, &IID_IXMLDOMDocument, (LPVOID*)&doc );
549     if( r != S_OK )
550         return;
551
552     b = FALSE;
553     str = SysAllocString( szComplete4 );
554     r = IXMLDOMDocument_loadXML( doc, str, &b );
555     ok( r == S_OK, "loadXML failed\n");
556     ok( b == VARIANT_TRUE, "failed to load XML string\n");
557     SysFreeString( str );
558
559     if (doc)
560     {
561         b = 1;
562         r = IXMLDOMNode_hasChildNodes( doc, &b );
563         ok( r == S_OK, "hasChildNoes bad return\n");
564         ok( b == VARIANT_TRUE, "hasChildNoes wrong result\n");
565
566         r = IXMLDOMDocument_get_documentElement( doc, &element );
567         ok( r == S_OK, "should be a document element\n");
568         ok( element != NULL, "should be an element\n");
569     }
570     else
571         ok( FALSE, "no document\n");
572
573     VariantInit(&var);
574     ok( V_VT(&var) == VT_EMPTY, "variant init failed\n");
575
576     r = IXMLDOMNode_get_nodeValue( doc, NULL );
577     ok(r == E_INVALIDARG, "get_nodeValue ret %08x\n", r );
578
579     r = IXMLDOMNode_get_nodeValue( doc, &var );
580     ok( r == S_FALSE, "nextNode returned wrong code\n");
581     ok( V_VT(&var) == VT_NULL, "variant wasn't empty\n");
582     ok( V_BSTR(&var) == NULL, "variant value wasn't null\n");
583
584     if (element)
585     {
586         owner = NULL;
587         r = IXMLDOMNode_get_ownerDocument( element, &owner );
588         todo_wine {
589         ok( r == S_OK, "get_ownerDocument return code\n");
590         }
591         ok( owner != doc, "get_ownerDocument return\n");
592
593         type = NODE_INVALID;
594         r = IXMLDOMNode_get_nodeType( element, &type);
595         ok( r == S_OK, "getNamedItem returned wrong code\n");
596         ok( type == NODE_ELEMENT, "node not an element\n");
597
598         str = NULL;
599         r = IXMLDOMNode_get_baseName( element, &str );
600         ok( r == S_OK, "get_baseName returned wrong code\n");
601         ok( lstrcmpW(str,szlc) == 0, "basename was wrong\n");
602         SysFreeString(str);
603
604         /* check if nodename is correct */
605         r = IXMLDOMElement_get_nodeName( element, NULL );
606         ok ( r == E_INVALIDARG, "get_nodeName (NULL) wrong code\n");
607     
608         /* content doesn't matter here */
609         str = NULL;
610         r = IXMLDOMElement_get_nodeName( element, &str );
611         ok ( r == S_OK, "get_nodeName wrong code\n");
612         ok ( str != NULL, "str is null\n");
613         ok( !lstrcmpW( str, szlc ), "incorrect nodeName\n");
614         SysFreeString( str );
615
616         str = SysAllocString( szNonExistentFile );      
617         V_VT(&var) = VT_I4;
618         V_I4(&var) = 0x1234;
619         r = IXMLDOMElement_getAttribute( element, str, &var );
620         ok( r == E_FAIL, "getAttribute ret %08x\n", r );
621         ok( V_VT(&var) == VT_EMPTY, "vt = %x\n", V_VT(&var));
622         VariantClear(&var);
623         SysFreeString( str );
624
625         str = SysAllocString( szdl );   
626         V_VT(&var) = VT_I4;
627         V_I4(&var) = 0x1234;
628         r = IXMLDOMElement_getAttribute( element, str, &var );
629         ok( r == S_OK, "getAttribute ret %08x\n", r );
630         ok( V_VT(&var) == VT_BSTR, "vt = %x\n", V_VT(&var));
631         ok( !lstrcmpW(V_BSTR(&var), szstr1), "wrong attr value\n");
632         VariantClear( &var );
633         SysFreeString( str );
634
635         r = IXMLDOMElement_get_attributes( element, &map );
636         ok( r == S_OK, "get_attributes returned wrong code\n");
637         ok( map != NULL, "should be attributes\n");
638
639         b = 1;
640         r = IXMLDOMNode_hasChildNodes( element, &b );
641         ok( r == S_OK, "hasChildNoes bad return\n");
642         ok( b == VARIANT_TRUE, "hasChildNoes wrong result\n");
643     }
644     else
645         ok( FALSE, "no element\n");
646
647     if (map)
648     {
649         ISupportErrorInfo *support_error;
650         r = IXMLDOMNamedNodeMap_QueryInterface( map, &IID_ISupportErrorInfo, (LPVOID*)&support_error );
651         ok( r == S_OK, "ret %08x\n", r );
652
653         r = ISupportErrorInfo_InterfaceSupportsErrorInfo( support_error, &IID_IXMLDOMNamedNodeMap );
654 todo_wine
655 {
656         ok( r == S_OK, "ret %08x\n", r );
657 }
658         ISupportErrorInfo_Release( support_error );
659
660         str = SysAllocString( szdl );
661         r = IXMLDOMNamedNodeMap_getNamedItem( map, str, &node );
662         ok( r == S_OK, "getNamedItem returned wrong code\n");
663         ok( node != NULL, "should be attributes\n");
664         IXMLDOMNode_Release(node);
665         SysFreeString( str );
666
667         str = SysAllocString( szdl );
668         r = IXMLDOMNamedNodeMap_getNamedItem( map, str, NULL );
669         ok( r == E_INVALIDARG, "getNamedItem should return E_INVALIDARG\n");
670         SysFreeString( str );
671
672         /* something that isn't in szComplete4 */
673         str = SysAllocString( szOpen );
674         node = (IXMLDOMNode *) 1;
675         r = IXMLDOMNamedNodeMap_getNamedItem( map, str, &node );
676         ok( r = S_FALSE, "getNamedItem found a node that wasn't there\n");
677         ok( node == NULL, "getNamedItem should have returned NULL\n");
678         SysFreeString( str );
679
680         /* test indexed access of attributes */
681         r = IXMLDOMNamedNodeMap_get_length( map, &count );
682         ok ( r == S_OK, "get_length wrong code\n");
683         ok ( count == 1, "get_length != 1\n");
684
685         node = NULL;
686         r = IXMLDOMNamedNodeMap_get_item( map, -1, &node);
687         ok ( r == S_FALSE, "get_item (-1) wrong code\n");
688         ok ( node == NULL, "there is no node\n");
689
690         node = NULL;
691         r = IXMLDOMNamedNodeMap_get_item( map, 1, &node);
692         ok ( r == S_FALSE, "get_item (1) wrong code\n");
693         ok ( node == NULL, "there is no attribute\n");
694
695         node = NULL;
696         r = IXMLDOMNamedNodeMap_get_item( map, 0, &node);
697         ok ( r == S_OK, "get_item (0) wrong code\n");
698         ok ( node != NULL, "should be attribute\n");
699
700         r = IXMLDOMNode_get_nodeName( node, NULL );
701         ok ( r == E_INVALIDARG, "get_nodeName (NULL) wrong code\n");
702
703         /* content doesn't matter here */
704         str = NULL;
705         r = IXMLDOMNode_get_nodeName( node, &str );
706         ok ( r == S_OK, "get_nodeName wrong code\n");
707         ok ( str != NULL, "str is null\n");
708         ok( !lstrcmpW( str, szdl ), "incorrect node name\n");
709         SysFreeString( str );
710
711         /* test sequential access of attributes */
712         node = NULL;
713         r = IXMLDOMNamedNodeMap_nextNode( map, &node );
714         ok ( r == S_OK, "nextNode (first time) wrong code\n");
715         ok ( node != NULL, "nextNode, should be attribute\n");
716
717         r = IXMLDOMNamedNodeMap_nextNode( map, &node );
718         ok ( r != S_OK, "nextNode (second time) wrong code\n");
719         ok ( node == NULL, "nextNode, there is no attribute\n");
720
721         r = IXMLDOMNamedNodeMap_reset( map );
722         ok ( r == S_OK, "reset should return S_OK\n");
723
724         r = IXMLDOMNamedNodeMap_nextNode( map, &node );
725         ok ( r == S_OK, "nextNode (third time) wrong code\n");
726         ok ( node != NULL, "nextNode, should be attribute\n");
727     }
728     else
729         ok( FALSE, "no map\n");
730
731     if (node)
732     {
733         type = NODE_INVALID;
734         r = IXMLDOMNode_get_nodeType( node, &type);
735         ok( r == S_OK, "getNamedItem returned wrong code\n");
736         ok( type == NODE_ATTRIBUTE, "node not an attribute\n");
737
738         str = NULL;
739         r = IXMLDOMNode_get_baseName( node, NULL );
740         ok( r == E_INVALIDARG, "get_baseName returned wrong code\n");
741
742         str = NULL;
743         r = IXMLDOMNode_get_baseName( node, &str );
744         ok( r == S_OK, "get_baseName returned wrong code\n");
745         ok( lstrcmpW(str,szdl) == 0, "basename was wrong\n");
746         SysFreeString( str );
747
748         r = IXMLDOMNode_get_nodeValue( node, &var );
749         ok( r == S_OK, "returns %08x\n", r );
750         ok( V_VT(&var) == VT_BSTR, "vt %x\n", V_VT(&var));
751         ok( !lstrcmpW(V_BSTR(&var), szstr1), "nodeValue incorrect\n");
752         VariantClear(&var);
753
754         r = IXMLDOMNode_get_childNodes( node, NULL );
755         ok( r == E_INVALIDARG, "get_childNodes returned wrong code\n");
756
757         r = IXMLDOMNode_get_childNodes( node, &list );
758         ok( r == S_OK, "get_childNodes returned wrong code\n");
759
760         if (list)
761         {
762             r = IXMLDOMNodeList_nextNode( list, &next );
763             ok( r == S_OK, "nextNode returned wrong code\n");
764         }
765         else
766             ok( FALSE, "no childlist\n");
767
768         if (next)
769         {
770             b = 1;
771             r = IXMLDOMNode_hasChildNodes( next, &b );
772             ok( r == S_FALSE, "hasChildNoes bad return\n");
773             ok( b == VARIANT_FALSE, "hasChildNoes wrong result\n");
774
775             type = NODE_INVALID;
776             r = IXMLDOMNode_get_nodeType( next, &type);
777             ok( r == S_OK, "getNamedItem returned wrong code\n");
778             ok( type == NODE_TEXT, "node not text\n");
779
780             str = (BSTR) 1;
781             r = IXMLDOMNode_get_baseName( next, &str );
782             ok( r == S_FALSE, "get_baseName returned wrong code\n");
783             ok( str == NULL, "basename was wrong\n");
784             SysFreeString(str);
785         }
786         else
787             ok( FALSE, "no next\n");
788
789         if (next)
790             IXMLDOMNode_Release( next );
791         next = NULL;
792         if (list)
793             IXMLDOMNodeList_Release( list );
794         list = NULL;
795         if (node)
796             IXMLDOMNode_Release( node );
797     }
798     else
799         ok( FALSE, "no node\n");
800     node = NULL;
801
802     if (map)
803         IXMLDOMNamedNodeMap_Release( map );
804
805     /* now traverse the tree from the root element */
806     if (element)
807     {
808         IXMLDOMNode *node;
809         r = IXMLDOMNode_get_childNodes( element, &list );
810         ok( r == S_OK, "get_childNodes returned wrong code\n");
811
812         /* using get_item for child list doesn't advance the position */
813         ole_check(IXMLDOMNodeList_get_item(list, 1, &node));
814         expect_node(node, "E2.E2.D1");
815         IXMLDOMNode_Release(node);
816         ole_check(IXMLDOMNodeList_nextNode(list, &node));
817         expect_node(node, "E1.E2.D1");
818         IXMLDOMNode_Release(node);
819         ole_check(IXMLDOMNodeList_reset(list));
820
821         IXMLDOMNodeList_AddRef(list);
822         expect_list_and_release(list, "E1.E2.D1 E2.E2.D1 E3.E2.D1 E4.E2.D1");
823         ole_check(IXMLDOMNodeList_reset(list));
824     }
825     else
826         ok( FALSE, "no element\n");
827
828     node = (void*)0xdeadbeef;
829     r = IXMLDOMNode_selectSingleNode( element, szdl, &node );
830     ok( r == S_FALSE, "ret %08x\n", r );
831     ok( node == NULL, "node %p\n", node );
832     r = IXMLDOMNode_selectSingleNode( element, szbs, &node );
833     ok( r == S_OK, "ret %08x\n", r );
834     r = IXMLDOMNode_Release( node );
835     ok( r == 0, "ret %08x\n", r );
836
837     if (list)
838     {
839         r = IXMLDOMNodeList_get_length( list, &count );
840         ok( r == S_OK, "get_length returns %08x\n", r );
841         ok( count == 4, "get_length got %ld\n", count );
842
843         r = IXMLDOMNodeList_nextNode( list, &node );
844         ok( r == S_OK, "nextNode returned wrong code\n");
845     }
846     else
847         ok( FALSE, "no list\n");
848
849     if (node)
850     {
851         type = NODE_INVALID;
852         r = IXMLDOMNode_get_nodeType( node, &type);
853         ok( r == S_OK, "getNamedItem returned wrong code\n");
854         ok( type == NODE_ELEMENT, "node not text\n");
855
856         VariantInit(&var);
857         ok( V_VT(&var) == VT_EMPTY, "variant init failed\n");
858         r = IXMLDOMNode_get_nodeValue( node, &var );
859         ok( r == S_FALSE, "nextNode returned wrong code\n");
860         ok( V_VT(&var) == VT_NULL, "variant wasn't empty\n");
861         ok( V_BSTR(&var) == NULL, "variant value wasn't null\n");
862
863         r = IXMLDOMNode_hasChildNodes( node, NULL );
864         ok( r == E_INVALIDARG, "hasChildNoes bad return\n");
865
866         b = 1;
867         r = IXMLDOMNode_hasChildNodes( node, &b );
868         ok( r == S_OK, "hasChildNoes bad return\n");
869         ok( b == VARIANT_TRUE, "hasChildNoes wrong result\n");
870
871         str = NULL;
872         r = IXMLDOMNode_get_baseName( node, &str );
873         ok( r == S_OK, "get_baseName returned wrong code\n");
874         ok( lstrcmpW(str,szbs) == 0, "basename was wrong\n");
875         SysFreeString(str);
876     }
877     else
878         ok( FALSE, "no node\n");
879
880     if (node)
881         IXMLDOMNode_Release( node );
882     if (list)
883         IXMLDOMNodeList_Release( list );
884     if (element)
885         IXMLDOMElement_Release( element );
886
887     b = FALSE;
888     str = SysAllocString( szComplete5 );
889     r = IXMLDOMDocument_loadXML( doc, str, &b );
890     ok( r == S_OK, "loadXML failed\n");
891     ok( b == VARIANT_TRUE, "failed to load XML string\n");
892     SysFreeString( str );
893
894     b = 1;
895     r = IXMLDOMNode_hasChildNodes( doc, &b );
896     ok( r == S_OK, "hasChildNoes bad return\n");
897     ok( b == VARIANT_TRUE, "hasChildNoes wrong result\n");
898
899     r = IXMLDOMDocument_get_documentElement( doc, &element );
900     ok( r == S_OK, "should be a document element\n");
901     ok( element != NULL, "should be an element\n");
902
903     if (element)
904     {
905         static const WCHAR szSSearch[] = {'S',':','s','e','a','r','c','h',0};
906         BSTR tag = NULL;
907
908         /* check if the tag is correct */
909         r = IXMLDOMElement_get_tagName( element, &tag );
910         ok( r == S_OK, "couldn't get tag name\n");
911         ok( tag != NULL, "tag was null\n");
912         ok( !lstrcmpW( tag, szSSearch ), "incorrect tag name\n");
913         SysFreeString( tag );
914     }
915
916     if (element)
917         IXMLDOMElement_Release( element );
918     if (doc)
919         IXMLDOMDocument_Release( doc );
920 }
921
922 static void test_refs(void)
923 {
924     HRESULT r;
925     BSTR str;
926     VARIANT_BOOL b;
927     IXMLDOMDocument *doc = NULL;
928     IXMLDOMElement *element = NULL;
929     IXMLDOMNode *node = NULL, *node2;
930     IXMLDOMNodeList *node_list = NULL;
931     LONG ref;
932     IUnknown *unk, *unk2;
933
934     r = CoCreateInstance( &CLSID_DOMDocument, NULL, 
935         CLSCTX_INPROC_SERVER, &IID_IXMLDOMDocument, (LPVOID*)&doc );
936     if( r != S_OK )
937         return;
938     ref = IXMLDOMDocument_Release(doc);
939     ok( ref == 0, "ref %d\n", ref);
940
941     r = CoCreateInstance( &CLSID_DOMDocument, NULL, 
942         CLSCTX_INPROC_SERVER, &IID_IXMLDOMDocument, (LPVOID*)&doc );
943     if( r != S_OK )
944         return;
945
946     str = SysAllocString( szComplete4 );
947     r = IXMLDOMDocument_loadXML( doc, str, &b );
948     ok( r == S_OK, "loadXML failed\n");
949     ok( b == VARIANT_TRUE, "failed to load XML string\n");
950     SysFreeString( str );
951
952     ref = IXMLDOMDocument_AddRef( doc );
953     ok( ref == 2, "ref %d\n", ref );
954     ref = IXMLDOMDocument_AddRef( doc );
955     ok( ref == 3, "ref %d\n", ref );
956     IXMLDOMDocument_Release( doc );
957     IXMLDOMDocument_Release( doc );
958
959     r = IXMLDOMDocument_get_documentElement( doc, &element );
960     ok( r == S_OK, "should be a document element\n");
961     ok( element != NULL, "should be an element\n");
962
963     ref = IXMLDOMDocument_AddRef( doc );
964     ok( ref == 2, "ref %d\n", ref );
965     IXMLDOMDocument_Release( doc );
966
967     r = IXMLDOMElement_get_childNodes( element, &node_list );
968     ok( r == S_OK, "rets %08x\n", r);
969     ref = IXMLDOMNodeList_AddRef( node_list );
970     ok( ref == 2, "ref %d\n", ref );
971     IXMLDOMNodeList_Release( node_list );
972
973     IXMLDOMNodeList_get_item( node_list, 0, &node );
974     ok( r == S_OK, "rets %08x\n", r);
975
976     IXMLDOMNodeList_get_item( node_list, 0, &node2 );
977     ok( r == S_OK, "rets %08x\n", r);
978
979     ref = IXMLDOMNode_AddRef( node );
980     ok( ref == 2, "ref %d\n", ref );
981     IXMLDOMNode_Release( node );
982
983     ref = IXMLDOMNode_Release( node );
984     ok( ref == 0, "ref %d\n", ref );
985     ref = IXMLDOMNode_Release( node2 );
986     ok( ref == 0, "ref %d\n", ref );
987
988     ref = IXMLDOMNodeList_Release( node_list );
989     ok( ref == 0, "ref %d\n", ref );
990
991     ok( node != node2, "node %p node2 %p\n", node, node2 );
992
993     ref = IXMLDOMDocument_Release( doc );
994     ok( ref == 0, "ref %d\n", ref );
995
996     ref = IXMLDOMElement_AddRef( element );
997     todo_wine {
998     ok( ref == 3, "ref %d\n", ref );
999     }
1000     IXMLDOMElement_Release( element );
1001
1002     /* IUnknown must be unique however we obtain it */
1003     r = IXMLDOMElement_QueryInterface( element, &IID_IUnknown, (LPVOID*)&unk );
1004     ok( r == S_OK, "rets %08x\n", r );
1005     r = IXMLDOMElement_QueryInterface( element, &IID_IXMLDOMNode, (LPVOID*)&node );
1006     ok( r == S_OK, "rets %08x\n", r );
1007     r = IXMLDOMNode_QueryInterface( node, &IID_IUnknown, (LPVOID*)&unk2 );
1008     ok( r == S_OK, "rets %08x\n", r );
1009     ok( unk == unk2, "unk %p unk2 %p\n", unk, unk2 );
1010
1011     IUnknown_Release( unk2 );
1012     IUnknown_Release( unk );
1013     IXMLDOMNode_Release( node );
1014
1015     IXMLDOMElement_Release( element );
1016
1017 }
1018
1019 static void test_create(void)
1020 {
1021     HRESULT r;
1022     VARIANT var;
1023     BSTR str, name;
1024     IXMLDOMDocument *doc;
1025     IXMLDOMElement *element;
1026     IXMLDOMNode *root, *node, *child;
1027     IXMLDOMNamedNodeMap *attr_map;
1028     IUnknown *unk;
1029     LONG ref;
1030     long num;
1031
1032     r = CoCreateInstance( &CLSID_DOMDocument, NULL, 
1033         CLSCTX_INPROC_SERVER, &IID_IXMLDOMDocument, (LPVOID*)&doc );
1034     if( r != S_OK )
1035         return;
1036
1037     V_VT(&var) = VT_I4;
1038     V_I4(&var) = NODE_ELEMENT;
1039     str = SysAllocString( szlc );
1040     r = IXMLDOMDocument_createNode( doc, var, str, NULL, &node );
1041     ok( r == S_OK, "returns %08x\n", r );
1042     r = IXMLDOMDocument_appendChild( doc, node, &root );
1043     ok( r == S_OK, "returns %08x\n", r );
1044     ok( node == root, "%p %p\n", node, root );
1045
1046     ref = IXMLDOMNode_AddRef( node );
1047     ok(ref == 3, "ref %d\n", ref);
1048     IXMLDOMNode_Release( node );
1049
1050     ref = IXMLDOMNode_Release( node );
1051     ok(ref == 1, "ref %d\n", ref);
1052     SysFreeString( str );
1053
1054     V_VT(&var) = VT_I4;
1055     V_I4(&var) = NODE_ELEMENT;
1056     str = SysAllocString( szbs );
1057     r = IXMLDOMDocument_createNode( doc, var, str, NULL, &node );
1058     ok( r == S_OK, "returns %08x\n", r );
1059
1060     ref = IXMLDOMNode_AddRef( node );
1061     ok(ref == 2, "ref = %d\n", ref);
1062     IXMLDOMNode_Release( node );
1063
1064     r = IXMLDOMNode_QueryInterface( node, &IID_IUnknown, (LPVOID*)&unk );
1065     ok( r == S_OK, "returns %08x\n", r );
1066
1067     ref = IXMLDOMNode_AddRef( unk );
1068     ok(ref == 3, "ref = %d\n", ref);
1069     IXMLDOMNode_Release( unk );
1070
1071     V_VT(&var) = VT_EMPTY;
1072     r = IXMLDOMNode_insertBefore( root, (IXMLDOMNode*)unk, var, &child );
1073     ok( r == S_OK, "returns %08x\n", r );
1074     ok( unk == (IUnknown*)child, "%p %p\n", unk, child );
1075     IXMLDOMNode_Release( child );
1076     IUnknown_Release( unk );
1077
1078
1079     V_VT(&var) = VT_NULL;
1080     V_DISPATCH(&var) = (IDispatch*)node;
1081     r = IXMLDOMNode_insertBefore( root, node, var, &child );
1082     ok( r == S_OK, "returns %08x\n", r );
1083     ok( node == child, "%p %p\n", node, child );
1084     IXMLDOMNode_Release( child );
1085
1086
1087     V_VT(&var) = VT_NULL;
1088     V_DISPATCH(&var) = (IDispatch*)node;
1089     r = IXMLDOMNode_insertBefore( root, node, var, NULL );
1090     ok( r == S_OK, "returns %08x\n", r );
1091     IXMLDOMNode_Release( node );
1092
1093     r = IXMLDOMNode_QueryInterface( root, &IID_IXMLDOMElement, (LPVOID*)&element );
1094     ok( r == S_OK, "returns %08x\n", r );
1095
1096     r = IXMLDOMElement_get_attributes( element, &attr_map );
1097     ok( r == S_OK, "returns %08x\n", r );
1098     r = IXMLDOMNamedNodeMap_get_length( attr_map, &num );
1099     ok( r == S_OK, "returns %08x\n", r );
1100     ok( num == 0, "num %ld\n", num );
1101     IXMLDOMNamedNodeMap_Release( attr_map );
1102
1103     V_VT(&var) = VT_BSTR;
1104     V_BSTR(&var) = SysAllocString( szstr1 );
1105     name = SysAllocString( szdl );
1106     r = IXMLDOMElement_setAttribute( element, name, var );
1107     ok( r == S_OK, "returns %08x\n", r );
1108     r = IXMLDOMElement_get_attributes( element, &attr_map );
1109     ok( r == S_OK, "returns %08x\n", r );
1110     r = IXMLDOMNamedNodeMap_get_length( attr_map, &num );
1111     ok( r == S_OK, "returns %08x\n", r );
1112     ok( num == 1, "num %ld\n", num );
1113     IXMLDOMNamedNodeMap_Release( attr_map );
1114     VariantClear(&var);
1115
1116     V_VT(&var) = VT_BSTR;
1117     V_BSTR(&var) = SysAllocString( szstr2 );
1118     r = IXMLDOMElement_setAttribute( element, name, var );
1119     ok( r == S_OK, "returns %08x\n", r );
1120     r = IXMLDOMElement_get_attributes( element, &attr_map );
1121     ok( r == S_OK, "returns %08x\n", r );
1122     r = IXMLDOMNamedNodeMap_get_length( attr_map, &num );
1123     ok( r == S_OK, "returns %08x\n", r );
1124     ok( num == 1, "num %ld\n", num );
1125     IXMLDOMNamedNodeMap_Release( attr_map );
1126     VariantClear(&var);
1127     r = IXMLDOMElement_getAttribute( element, name, &var );
1128     ok( r == S_OK, "returns %08x\n", r );
1129     ok( !lstrcmpW(V_BSTR(&var), szstr2), "wrong attr value\n");
1130     VariantClear(&var);
1131     SysFreeString(name);
1132
1133     V_VT(&var) = VT_BSTR;
1134     V_BSTR(&var) = SysAllocString( szstr1 );
1135     name = SysAllocString( szlc );
1136     r = IXMLDOMElement_setAttribute( element, name, var );
1137     ok( r == S_OK, "returns %08x\n", r );
1138     r = IXMLDOMElement_get_attributes( element, &attr_map );
1139     ok( r == S_OK, "returns %08x\n", r );
1140     r = IXMLDOMNamedNodeMap_get_length( attr_map, &num );
1141     ok( r == S_OK, "returns %08x\n", r );
1142     ok( num == 2, "num %ld\n", num );
1143     IXMLDOMNamedNodeMap_Release( attr_map );
1144     VariantClear(&var);
1145     SysFreeString(name);
1146
1147     V_VT(&var) = VT_I4;
1148     V_I4(&var) = 10;
1149     name = SysAllocString( szbs );
1150     r = IXMLDOMElement_setAttribute( element, name, var );
1151     ok( r == S_OK, "returns %08x\n", r );
1152     VariantClear(&var);
1153     r = IXMLDOMElement_getAttribute( element, name, &var );
1154     ok( r == S_OK, "returns %08x\n", r );
1155     ok( V_VT(&var) == VT_BSTR, "variant type %x\n", V_VT(&var));
1156     VariantClear(&var);
1157     SysFreeString(name);
1158
1159     IXMLDOMElement_Release( element );
1160     IXMLDOMNode_Release( root );
1161     IXMLDOMDocument_Release( doc );
1162 }
1163
1164 static void test_getElementsByTagName(void)
1165 {
1166     HRESULT r;
1167     BSTR str;
1168     VARIANT_BOOL b;
1169     IXMLDOMDocument *doc;
1170     IXMLDOMNodeList *node_list;
1171     long len;
1172
1173     r = CoCreateInstance( &CLSID_DOMDocument, NULL, 
1174         CLSCTX_INPROC_SERVER, &IID_IXMLDOMDocument, (LPVOID*)&doc );
1175     if( r != S_OK )
1176         return;
1177
1178     str = SysAllocString( szComplete4 );
1179     r = IXMLDOMDocument_loadXML( doc, str, &b );
1180     ok( r == S_OK, "loadXML failed\n");
1181     ok( b == VARIANT_TRUE, "failed to load XML string\n");
1182     SysFreeString( str );
1183
1184     str = SysAllocString( szstar );
1185     r = IXMLDOMDocument_getElementsByTagName(doc, str, &node_list);
1186     ok( r == S_OK, "ret %08x\n", r );
1187     r = IXMLDOMNodeList_get_length( node_list, &len );
1188     ok( r == S_OK, "ret %08x\n", r );
1189     ok( len == 6, "len %ld\n", len );
1190     IXMLDOMNodeList_Release( node_list );
1191     SysFreeString( str );
1192
1193     str = SysAllocString( szbs );
1194     r = IXMLDOMDocument_getElementsByTagName(doc, str, &node_list);
1195     ok( r == S_OK, "ret %08x\n", r );
1196     r = IXMLDOMNodeList_get_length( node_list, &len );
1197     ok( r == S_OK, "ret %08x\n", r );
1198     ok( len == 1, "len %ld\n", len );
1199     IXMLDOMNodeList_Release( node_list );
1200     SysFreeString( str );
1201
1202     str = SysAllocString( szdl );
1203     r = IXMLDOMDocument_getElementsByTagName(doc, str, &node_list);
1204     ok( r == S_OK, "ret %08x\n", r );
1205     r = IXMLDOMNodeList_get_length( node_list, &len );
1206     ok( r == S_OK, "ret %08x\n", r );
1207     ok( len == 0, "len %ld\n", len );
1208     IXMLDOMNodeList_Release( node_list );
1209     SysFreeString( str );
1210
1211     str = SysAllocString( szstr1 );
1212     r = IXMLDOMDocument_getElementsByTagName(doc, str, &node_list);
1213     ok( r == S_OK, "ret %08x\n", r );
1214     r = IXMLDOMNodeList_get_length( node_list, &len );
1215     ok( r == S_OK, "ret %08x\n", r );
1216     ok( len == 0, "len %ld\n", len );
1217     IXMLDOMNodeList_Release( node_list );
1218     SysFreeString( str );
1219
1220     IXMLDOMDocument_Release( doc );
1221 }
1222
1223 static void test_get_text(void)
1224 {
1225     HRESULT r;
1226     BSTR str;
1227     VARIANT_BOOL b;
1228     IXMLDOMDocument *doc;
1229     IXMLDOMNode *node, *node2, *node3;
1230     IXMLDOMNodeList *node_list;
1231     IXMLDOMNamedNodeMap *node_map;
1232
1233     r = CoCreateInstance( &CLSID_DOMDocument, NULL, 
1234         CLSCTX_INPROC_SERVER, &IID_IXMLDOMDocument, (LPVOID*)&doc );
1235     if( r != S_OK )
1236         return;
1237
1238     str = SysAllocString( szComplete4 );
1239     r = IXMLDOMDocument_loadXML( doc, str, &b );
1240     ok( r == S_OK, "loadXML failed\n");
1241     ok( b == VARIANT_TRUE, "failed to load XML string\n");
1242     SysFreeString( str );
1243
1244     str = SysAllocString( szbs );
1245     r = IXMLDOMDocument_getElementsByTagName( doc, str, &node_list );
1246     ok( r == S_OK, "ret %08x\n", r );
1247     SysFreeString(str);
1248     
1249     r = IXMLDOMNodeList_get_item( node_list, 0, &node );
1250     ok( r == S_OK, "ret %08x\n", r ); 
1251     IXMLDOMNodeList_Release( node_list );
1252
1253     /* Invalid output parameter*/
1254     r = IXMLDOMNode_get_text( node, NULL );
1255     ok( r == E_INVALIDARG, "ret %08x\n", r );
1256
1257     r = IXMLDOMNode_get_text( node, &str );
1258     ok( r == S_OK, "ret %08x\n", r );
1259 todo_wine {
1260     ok( !memcmp(str, szfn1_txt, sizeof(szfn1_txt)), "wrong string\n" );
1261  }
1262     ok( !memcmp(str, szfn1_txt, sizeof(szfn1_txt)-4), "wrong string\n" );
1263     SysFreeString(str);
1264
1265     r = IXMLDOMNode_get_attributes( node, &node_map );
1266     ok( r == S_OK, "ret %08x\n", r );
1267     
1268     str = SysAllocString( szvr );
1269     r = IXMLDOMNamedNodeMap_getNamedItem( node_map, str, &node2 );
1270     ok( r == S_OK, "ret %08x\n", r );
1271     SysFreeString(str);
1272
1273     r = IXMLDOMNode_get_text( node2, &str );
1274     ok( r == S_OK, "ret %08x\n", r );
1275     ok( !memcmp(str, szstr2, sizeof(szstr2)), "wrong string\n" );
1276     SysFreeString(str);
1277
1278     r = IXMLDOMNode_get_firstChild( node2, &node3 );
1279     ok( r == S_OK, "ret %08x\n", r );
1280
1281     r = IXMLDOMNode_get_text( node3, &str );
1282     ok( r == S_OK, "ret %08x\n", r );
1283     ok( !memcmp(str, szstr2, sizeof(szstr2)), "wrong string\n" );
1284     SysFreeString(str);
1285
1286
1287     IXMLDOMNode_Release( node3 );
1288     IXMLDOMNode_Release( node2 );
1289     IXMLDOMNamedNodeMap_Release( node_map );
1290     IXMLDOMNode_Release( node );
1291     IXMLDOMDocument_Release( doc );
1292 }
1293
1294 static void test_get_childNodes(void)
1295 {
1296     HRESULT r;
1297     BSTR str;
1298     VARIANT_BOOL b;
1299     IXMLDOMDocument *doc;
1300     IXMLDOMElement *element;
1301     IXMLDOMNode *node, *node2;
1302     IXMLDOMNodeList *node_list, *node_list2;
1303     long len;
1304
1305     r = CoCreateInstance( &CLSID_DOMDocument, NULL, 
1306         CLSCTX_INPROC_SERVER, &IID_IXMLDOMDocument, (LPVOID*)&doc );
1307     if( r != S_OK )
1308         return;
1309
1310     str = SysAllocString( szComplete4 );
1311     r = IXMLDOMDocument_loadXML( doc, str, &b );
1312     ok( r == S_OK, "loadXML failed\n");
1313     ok( b == VARIANT_TRUE, "failed to load XML string\n");
1314     SysFreeString( str );
1315
1316     r = IXMLDOMDocument_get_documentElement( doc, &element );
1317     ok( r == S_OK, "ret %08x\n", r);
1318
1319     r = IXMLDOMElement_get_childNodes( element, &node_list );
1320     ok( r == S_OK, "ret %08x\n", r);
1321
1322     r = IXMLDOMNodeList_get_length( node_list, &len );
1323     ok( r == S_OK, "ret %08x\n", r);
1324     ok( len == 4, "len %ld\n", len);
1325
1326     r = IXMLDOMNodeList_get_item( node_list, 2, &node );
1327     ok( r == S_OK, "ret %08x\n", r);
1328
1329     r = IXMLDOMNode_get_childNodes( node, &node_list2 );
1330     ok( r == S_OK, "ret %08x\n", r);
1331
1332     r = IXMLDOMNodeList_get_length( node_list2, &len );
1333     ok( r == S_OK, "ret %08x\n", r);
1334     ok( len == 0, "len %ld\n", len);
1335
1336     r = IXMLDOMNodeList_get_item( node_list2, 0, &node2);
1337     ok( r == S_FALSE, "ret %08x\n", r);
1338
1339     IXMLDOMNodeList_Release( node_list2 );
1340     IXMLDOMNode_Release( node );
1341     IXMLDOMNodeList_Release( node_list );
1342     IXMLDOMElement_Release( element );
1343     IXMLDOMDocument_Release( doc );
1344 }
1345
1346 static void test_removeChild(void)
1347 {
1348     HRESULT r;
1349     BSTR str;
1350     VARIANT_BOOL b;
1351     IXMLDOMDocument *doc;
1352     IXMLDOMElement *element;
1353     IXMLDOMNode *node, *node2, *node3, *node4;
1354     IXMLDOMNodeList *node_list, *node_list2;
1355
1356     r = CoCreateInstance( &CLSID_DOMDocument, NULL, 
1357         CLSCTX_INPROC_SERVER, &IID_IXMLDOMDocument, (LPVOID*)&doc );
1358     if( r != S_OK )
1359         return;
1360
1361     str = SysAllocString( szComplete4 );
1362     r = IXMLDOMDocument_loadXML( doc, str, &b );
1363     ok( r == S_OK, "loadXML failed\n");
1364     ok( b == VARIANT_TRUE, "failed to load XML string\n");
1365     SysFreeString( str );
1366
1367     r = IXMLDOMDocument_get_documentElement( doc, &element );
1368     ok( r == S_OK, "ret %08x\n", r);
1369
1370     r = IXMLDOMElement_get_childNodes( element, &node_list );
1371     ok( r == S_OK, "ret %08x\n", r);
1372
1373     r = IXMLDOMNodeList_get_item( node_list, 3, &node );
1374     ok( r == S_OK, "ret %08x\n", r);
1375  
1376     r = IXMLDOMNode_get_childNodes( node, &node_list2 );
1377     ok( r == S_OK, "ret %08x\n", r);
1378  
1379     r = IXMLDOMNodeList_get_item( node_list, 0, &node4 );
1380     ok( r == S_OK, "ret %08x\n", r);
1381
1382     r = IXMLDOMElement_removeChild( element, NULL, &node2 );
1383     ok( r == E_INVALIDARG, "ret %08x\n", r );
1384
1385     r = IXMLDOMElement_removeChild( element, node4, &node2 );
1386     ok( r == S_OK, "ret %08x\n", r);
1387     ok( node4 == node2, "node %p node2 %p\n", node4, node2 );
1388
1389     r = IXMLDOMNode_get_parentNode( node4, &node3 );
1390     ok( r == S_FALSE, "ret %08x\n", r);
1391     ok( node3 == NULL, "%p\n", node3 );
1392
1393     IXMLDOMNode_Release( node2 );
1394     IXMLDOMNode_Release( node4 );
1395     IXMLDOMNodeList_Release( node_list2 );
1396     IXMLDOMNode_Release( node );
1397     IXMLDOMNodeList_Release( node_list );
1398     IXMLDOMElement_Release( element );
1399     IXMLDOMDocument_Release( doc );
1400 }
1401
1402 static void test_XMLHTTP(void)
1403 {
1404     static const WCHAR wszBody[] = {'m','o','d','e','=','T','e','s','t',0};
1405     static WCHAR wszPOST[] = {'P','O','S','T',0};
1406     static WCHAR wszUrl[] = {'h','t','t','p',':','/','/',
1407         'c','r','o','s','s','o','v','e','r','.','c','o','d','e','w','e','a','v','e','r','s','.','c','o','m','/',
1408         'p','o','s','t','t','e','s','t','.','p','h','p',0};
1409     static const WCHAR wszExpectedResponse[] = {'F','A','I','L','E','D',0};
1410     IXMLHttpRequest *pXMLHttpRequest;
1411     BSTR bstrResponse;
1412     VARIANT dummy;
1413     VARIANT varfalse;
1414     VARIANT varbody;
1415     HRESULT hr = CoCreateInstance(&CLSID_XMLHTTPRequest, NULL,
1416                                   CLSCTX_INPROC_SERVER, &IID_IXMLHttpRequest,
1417                                   (void **)&pXMLHttpRequest);
1418     todo_wine {
1419     ok(hr == S_OK, "CoCreateInstance(CLSID_XMLHTTPRequest) should have succeeded instead of failing with 0x%08x\n", hr);
1420     }
1421     if (hr != S_OK)
1422         return;
1423
1424     VariantInit(&dummy);
1425     V_VT(&dummy) = VT_ERROR;
1426     V_ERROR(&dummy) = DISP_E_MEMBERNOTFOUND;
1427     VariantInit(&varfalse);
1428     V_VT(&varfalse) = VT_BOOL;
1429     V_BOOL(&varfalse) = VARIANT_FALSE;
1430     V_VT(&varbody) = VT_BSTR;
1431     V_BSTR(&varbody) = SysAllocString(wszBody);
1432
1433     hr = IXMLHttpRequest_open(pXMLHttpRequest, wszPOST, wszUrl, varfalse, dummy, dummy);
1434     ok(hr == S_OK, "IXMLHttpRequest_open should have succeeded instead of failing with 0x%08x\n", hr);
1435
1436     hr = IXMLHttpRequest_send(pXMLHttpRequest, varbody);
1437     ok(hr == S_OK, "IXMLHttpRequest_send should have succeeded instead of failing with 0x%08x\n", hr);
1438     VariantClear(&varbody);
1439
1440     hr = IXMLHttpRequest_get_responseText(pXMLHttpRequest, &bstrResponse);
1441     ok(hr == S_OK, "IXMLHttpRequest_get_responseText should have succeeded instead of failing with 0x%08x\n", hr);
1442     /* the server currently returns "FAILED" because the Content-Type header is
1443      * not what the server expects */
1444     ok(!memcmp(bstrResponse, wszExpectedResponse, sizeof(wszExpectedResponse)), "bstrResponse differs from what was expected\n");
1445     SysFreeString(bstrResponse);
1446 }
1447
1448 static void test_IXMLDOMDocument2(void)
1449 {
1450     HRESULT r;
1451     VARIANT_BOOL b;
1452     BSTR str;
1453     IXMLDOMDocument *doc;
1454     IXMLDOMDocument2 *doc2;
1455     VARIANT var;
1456     int ref;
1457
1458     r = CoCreateInstance( &CLSID_DOMDocument, NULL,
1459         CLSCTX_INPROC_SERVER, &IID_IXMLDOMDocument, (LPVOID*)&doc );
1460     if( r != S_OK )
1461         return;
1462
1463     str = SysAllocString( szComplete4 );
1464     r = IXMLDOMDocument_loadXML( doc, str, &b );
1465     ok( r == S_OK, "loadXML failed\n");
1466     ok( b == VARIANT_TRUE, "failed to load XML string\n");
1467     SysFreeString( str );
1468
1469     r = IXMLDOMDocument_QueryInterface( doc, &IID_IXMLDOMDocument2, (void**)&doc2 );
1470     ok( r == S_OK, "ret %08x\n", r );
1471     ok( doc == (IXMLDOMDocument*)doc2, "interfaces differ\n");
1472
1473     /* we will check if the variant got cleared */
1474     ref = IXMLDOMDocument2_AddRef(doc2);
1475     expect_eq(ref, 3, int, "%d");  /* doc, doc2, AddRef*/
1476     V_VT(&var) = VT_UNKNOWN;
1477     V_UNKNOWN(&var) = (IUnknown *)doc2;
1478
1479     /* invalid calls */
1480     ole_expect(IXMLDOMDocument2_getProperty(doc2, _bstr_("askldhfaklsdf"), &var), E_FAIL);
1481     expect_eq(V_VT(&var), VT_UNKNOWN, int, "%x");
1482     ole_expect(IXMLDOMDocument2_getProperty(doc2, _bstr_("SelectionLanguage"), NULL), E_INVALIDARG);
1483
1484     /* valid call */
1485     ole_check(IXMLDOMDocument2_getProperty(doc2, _bstr_("SelectionLanguage"), &var));
1486     expect_eq(V_VT(&var), VT_BSTR, int, "%x");
1487     expect_bstr_eq_and_free(V_BSTR(&var), "XSLPattern");
1488     V_VT(&var) = VT_R4;
1489
1490     /* the variant didn't get cleared*/
1491     expect_eq(IXMLDOMDocument2_Release(doc2), 2, int, "%d");
1492
1493     /* setProperty tests */
1494     ole_expect(IXMLDOMDocument2_setProperty(doc2, _bstr_("askldhfaklsdf"), var), E_FAIL);
1495     ole_expect(IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionLanguage"), var), E_FAIL);
1496     ole_expect(IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionLanguage"), _variantbstr_("alskjdh faklsjd hfk")), E_FAIL);
1497     ole_check(IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionLanguage"), _variantbstr_("XSLPattern")));
1498     ole_check(IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionLanguage"), _variantbstr_("XPath")));
1499     ole_check(IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionLanguage"), _variantbstr_("XSLPattern")));
1500
1501     /* contrary to what MSDN calims you can switch back from XPath to XSLPattern */
1502     ole_check(IXMLDOMDocument2_getProperty(doc2, _bstr_("SelectionLanguage"), &var));
1503     expect_eq(V_VT(&var), VT_BSTR, int, "%x");
1504     expect_bstr_eq_and_free(V_BSTR(&var), "XSLPattern");
1505
1506     IXMLDOMDocument2_Release( doc2 );
1507     IXMLDOMDocument_Release( doc );
1508     free_bstrs();
1509 }
1510
1511 static void test_XPath(void)
1512 {
1513     HRESULT r;
1514     VARIANT var;
1515     VARIANT_BOOL b;
1516     IXMLDOMDocument2 *doc;
1517     IXMLDOMNode *rootNode;
1518     IXMLDOMNode *elem1Node;
1519     IXMLDOMNode *node;
1520     IXMLDOMNodeList *list;
1521
1522     r = CoCreateInstance( &CLSID_DOMDocument, NULL,
1523         CLSCTX_INPROC_SERVER, &IID_IXMLDOMDocument2, (LPVOID*)&doc );
1524     if( r != S_OK )
1525         return;
1526
1527     ole_check(IXMLDOMDocument_loadXML(doc, _bstr_(szExampleXML), &b));
1528     ok(b == VARIANT_TRUE, "failed to load XML string\n");
1529
1530     /* switch to XPath */
1531     ole_check(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionLanguage"), _variantbstr_("XPath")));
1532
1533     /* some simple queries*/
1534     ole_check(IXMLDOMDocument_selectNodes(doc, _bstr_("root"), &list));
1535     ole_check(IXMLDOMNodeList_get_item(list, 0, &rootNode));
1536     ole_check(IXMLDOMNodeList_reset(list));
1537     expect_list_and_release(list, "E2.D1");
1538     if (rootNode == NULL)
1539         return;
1540
1541     ole_check(IXMLDOMDocument_selectNodes(doc, _bstr_("root//c"), &list));
1542     expect_list_and_release(list, "E3.E1.E2.D1 E3.E2.E2.D1");
1543
1544     ole_check(IXMLDOMDocument_selectNodes(doc, _bstr_("//c[@type]"), &list));
1545     expect_list_and_release(list, "E3.E2.E2.D1");
1546
1547     ole_check(IXMLDOMNode_selectNodes(rootNode, _bstr_("elem"), &list));
1548     /* using get_item for query results advances the position */
1549     ole_check(IXMLDOMNodeList_get_item(list, 1, &node));
1550     expect_node(node, "E2.E2.D1");
1551     IXMLDOMNode_Release(node);
1552     ole_check(IXMLDOMNodeList_nextNode(list, &node));
1553     expect_node(node, "E4.E2.D1");
1554     IXMLDOMNode_Release(node);
1555     ole_check(IXMLDOMNodeList_reset(list));
1556     expect_list_and_release(list, "E1.E2.D1 E2.E2.D1 E4.E2.D1");
1557
1558     ole_check(IXMLDOMNode_selectNodes(rootNode, _bstr_("."), &list));
1559     expect_list_and_release(list, "E2.D1");
1560
1561     ole_check(IXMLDOMNode_selectNodes(rootNode, _bstr_("elem[3]/preceding-sibling::*"), &list));
1562     ole_check(IXMLDOMNodeList_get_item(list, 0, &elem1Node));
1563     ole_check(IXMLDOMNodeList_reset(list));
1564     expect_list_and_release(list, "E1.E2.D1 E2.E2.D1 E3.E2.D1");
1565
1566     /* select an attribute */
1567     ole_check(IXMLDOMNode_selectNodes(rootNode, _bstr_(".//@type"), &list));
1568     expect_list_and_release(list, "A'type'.E3.E2.E2.D1");
1569
1570     /* would evaluate to a number */
1571     ole_expect(IXMLDOMNode_selectNodes(rootNode, _bstr_("count(*)"), &list), E_FAIL);
1572     /* would evaluate to a boolean */
1573     ole_expect(IXMLDOMNode_selectNodes(rootNode, _bstr_("position()>0"), &list), E_FAIL);
1574     /* would evaluate to a string */
1575     ole_expect(IXMLDOMNode_selectNodes(rootNode, _bstr_("name()"), &list), E_FAIL);
1576
1577     /* no results */
1578     ole_check(IXMLDOMNode_selectNodes(rootNode, _bstr_("c"), &list));
1579     expect_list_and_release(list, "");
1580     ole_check(IXMLDOMDocument_selectNodes(doc, _bstr_("elem//c"), &list));
1581     expect_list_and_release(list, "");
1582     ole_check(IXMLDOMDocument_selectNodes(doc, _bstr_("//elem[4]"), &list));
1583     expect_list_and_release(list, "");
1584
1585     /* foo undeclared in document node */
1586     ole_expect(IXMLDOMDocument_selectNodes(doc, _bstr_("root//foo:c"), &list), E_FAIL);
1587     /* undeclared in <root> node */
1588     ole_expect(IXMLDOMNode_selectNodes(rootNode, _bstr_(".//foo:c"), &list), E_FAIL);
1589     /* undeclared in <elem> node */
1590     ole_expect(IXMLDOMNode_selectNodes(elem1Node, _bstr_("//foo:c"), &list), E_FAIL);
1591     /* but this trick can be used */
1592     ole_check(IXMLDOMNode_selectNodes(elem1Node, _bstr_("//*[name()='foo:c']"), &list));
1593     expect_list_and_release(list, "E3.E4.E2.D1");
1594
1595     /* it has to be declared in SelectionNamespaces */
1596     todo_wine ole_check(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionNamespaces"),
1597         _variantbstr_("xmlns:test='urn:uuid:86B2F87F-ACB6-45cd-8B77-9BDB92A01A29'")));
1598
1599     /* now the namespace can be used */
1600     todo_wine ole_check(IXMLDOMDocument_selectNodes(doc, _bstr_("root//test:c"), &list));
1601     todo_wine expect_list_and_release(list, "E3.E3.E2.D1 E3.E4.E2.D1");
1602     todo_wine ole_check(IXMLDOMNode_selectNodes(rootNode, _bstr_(".//test:c"), &list));
1603     todo_wine expect_list_and_release(list, "E3.E3.E2.D1 E3.E4.E2.D1");
1604     todo_wine ole_check(IXMLDOMNode_selectNodes(elem1Node, _bstr_("//test:c"), &list));
1605     todo_wine expect_list_and_release(list, "E3.E3.E2.D1 E3.E4.E2.D1");
1606     todo_wine ole_check(IXMLDOMNode_selectNodes(elem1Node, _bstr_(".//test:x"), &list));
1607     todo_wine expect_list_and_release(list, "E5.E1.E4.E1.E2.D1");
1608
1609     /* SelectionNamespaces syntax error - the namespaces doesn't work anymore but the value is stored */
1610     ole_expect(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionNamespaces"),
1611         _variantbstr_("xmlns:test='urn:uuid:86B2F87F-ACB6-45cd-8B77-9BDB92A01A29' xmlns:foo=###")), E_FAIL);
1612
1613     ole_expect(IXMLDOMDocument_selectNodes(doc, _bstr_("root//foo:c"), &list), E_FAIL);
1614
1615     todo_wine ole_check(IXMLDOMDocument2_getProperty(doc, _bstr_("SelectionNamespaces"), &var));
1616     todo_wine expect_eq(V_VT(&var), VT_BSTR, int, "%x");
1617     if (V_VT(&var) == VT_BSTR)
1618         expect_bstr_eq_and_free(V_BSTR(&var), "xmlns:test='urn:uuid:86B2F87F-ACB6-45cd-8B77-9BDB92A01A29' xmlns:foo=###");
1619
1620     /* extra attributes - same thing*/
1621     ole_expect(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionNamespaces"),
1622         _variantbstr_("xmlns:test='urn:uuid:86B2F87F-ACB6-45cd-8B77-9BDB92A01A29' param='test'")), E_FAIL);
1623     ole_expect(IXMLDOMDocument_selectNodes(doc, _bstr_("root//foo:c"), &list), E_FAIL);
1624
1625     IXMLDOMNode_Release(rootNode);
1626     IXMLDOMNode_Release(elem1Node);
1627     IXMLDOMDocument_Release(doc);
1628     free_bstrs();
1629 }
1630
1631 static void test_cloneNode(void )
1632 {
1633     IXMLDOMDocument *doc = NULL;
1634     VARIANT_BOOL b;
1635     IXMLDOMNodeList *pList;
1636     IXMLDOMNamedNodeMap *mapAttr;
1637     long nLength = 0, nLength1 = 0;
1638     long nAttrCnt = 0, nAttrCnt1 = 0;
1639     IXMLDOMNode *node;
1640     IXMLDOMNode *node_clone;
1641     HRESULT r;
1642     BSTR str;
1643     static const WCHAR szSearch[] = { 'l', 'c', '/', 'p', 'r', 0 };
1644
1645     r = CoCreateInstance( &CLSID_DOMDocument, NULL, CLSCTX_INPROC_SERVER, &IID_IXMLDOMDocument2, (LPVOID*)&doc );
1646     if( r != S_OK )
1647         return;
1648
1649     str = SysAllocString( szComplete4 );
1650     ole_check(IXMLDOMDocument_loadXML(doc, str, &b));
1651     ok(b == VARIANT_TRUE, "failed to load XML string\n");
1652     SysFreeString(str);
1653
1654     if(!b)
1655         return;
1656
1657     str = SysAllocString( szSearch);
1658     r = IXMLDOMNode_selectSingleNode(doc, str, &node);
1659     ok( r == S_OK, "ret %08x\n", r );
1660     ok( node != NULL, "node %p\n", node );
1661     SysFreeString(str);
1662
1663     if(!node)
1664     {
1665         IXMLDOMDocument_Release(doc);
1666         return;
1667     }
1668
1669     /* Check invalid parameter */
1670     r = IXMLDOMNode_cloneNode(node, VARIANT_TRUE, NULL);
1671     ok( r == E_INVALIDARG, "ret %08x\n", r );
1672
1673     /* All Children */
1674     r = IXMLDOMNode_cloneNode(node, VARIANT_TRUE, &node_clone);
1675     ok( r == S_OK, "ret %08x\n", r );
1676     ok( node_clone != NULL, "node %p\n", node );
1677
1678     if(!node_clone)
1679     {
1680         IXMLDOMDocument_Release(doc);
1681         IXMLDOMNode_Release(node);
1682         return;
1683     }
1684
1685     r = IXMLDOMNode_get_childNodes(node, &pList);
1686     ok( r == S_OK, "ret %08x\n", r );
1687     if (pList)
1688         {
1689                 IXMLDOMNodeList_get_length(pList, &nLength);
1690                 IXMLDOMNodeList_Release(pList);
1691         }
1692
1693     r = IXMLDOMNode_get_attributes(node, &mapAttr);
1694     ok( r == S_OK, "ret %08x\n", r );
1695     if(mapAttr)
1696     {
1697         IXMLDOMNamedNodeMap_get_length(mapAttr, &nAttrCnt);
1698         IXMLDOMNamedNodeMap_Release(mapAttr);
1699     }
1700
1701     r = IXMLDOMNode_get_childNodes(node_clone, &pList);
1702     ok( r == S_OK, "ret %08x\n", r );
1703     if (pList)
1704         {
1705                 IXMLDOMNodeList_get_length(pList, &nLength1);
1706                 IXMLDOMNodeList_Release(pList);
1707         }
1708
1709     r = IXMLDOMNode_get_attributes(node_clone, &mapAttr);
1710     ok( r == S_OK, "ret %08x\n", r );
1711     if(mapAttr)
1712     {
1713         IXMLDOMNamedNodeMap_get_length(mapAttr, &nAttrCnt1);
1714         IXMLDOMNamedNodeMap_Release(mapAttr);
1715     }
1716
1717     ok(nLength == nLength1, "wrong Child count (%ld, %ld)\n", nLength, nLength1);
1718     ok(nAttrCnt == nAttrCnt1, "wrong Attribute count (%ld, %ld)\n", nAttrCnt, nAttrCnt1);
1719     IXMLDOMNode_Release(node_clone);
1720
1721     /* No Children */
1722     r = IXMLDOMNode_cloneNode(node, VARIANT_FALSE, &node_clone);
1723     ok( r == S_OK, "ret %08x\n", r );
1724     ok( node_clone != NULL, "node %p\n", node );
1725
1726     if(!node_clone)
1727     {
1728         IXMLDOMDocument_Release(doc);
1729         IXMLDOMNode_Release(node);
1730         return;
1731     }
1732
1733     r = IXMLDOMNode_get_childNodes(node_clone, &pList);
1734     ok( r == S_OK, "ret %08x\n", r );
1735     if (pList)
1736         {
1737                 IXMLDOMNodeList_get_length(pList, &nLength1);
1738         ok( nLength1 == 0, "Length should be 0 (%ld)\n", nLength1);
1739                 IXMLDOMNodeList_Release(pList);
1740         }
1741
1742     r = IXMLDOMNode_get_attributes(node_clone, &mapAttr);
1743     ok( r == S_OK, "ret %08x\n", r );
1744     if(mapAttr)
1745     {
1746         IXMLDOMNamedNodeMap_get_length(mapAttr, &nAttrCnt1);
1747         ok( nAttrCnt1 == 3, "Attribute count should be 3 (%ld)\n", nAttrCnt1);
1748         IXMLDOMNamedNodeMap_Release(mapAttr);
1749     }
1750
1751     ok(nLength != nLength1, "wrong Child count (%ld, %ld)\n", nLength, nLength1);
1752     ok(nAttrCnt == nAttrCnt1, "wrong Attribute count (%ld, %ld)\n", nAttrCnt, nAttrCnt1);
1753     IXMLDOMNode_Release(node_clone);
1754
1755
1756     IXMLDOMNode_Release(node);
1757     IXMLDOMDocument_Release(doc);
1758 }
1759
1760 static void test_xmlTypes(void)
1761 {
1762     IXMLDOMDocument *doc = NULL;
1763     IXMLDOMElement *pRoot;
1764     HRESULT hr;
1765     IXMLDOMComment *pComment;
1766     IXMLDOMElement *pElement;
1767     IXMLDOMAttribute *pAttrubute;
1768     IXMLDOMNamedNodeMap *pAttribs;
1769     BSTR str;
1770     IXMLDOMNode *pNextChild = (IXMLDOMNode *)0x1;   /* Used for testing Siblings */
1771
1772     hr = CoCreateInstance( &CLSID_DOMDocument, NULL, CLSCTX_INPROC_SERVER, &IID_IXMLDOMDocument2, (LPVOID*)&doc );
1773     if( hr != S_OK )
1774         return;
1775
1776     hr = IXMLDOMDocument_get_nextSibling(doc, NULL);
1777     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
1778
1779     hr = IXMLDOMDocument_get_nextSibling(doc, &pNextChild);
1780     ok(hr == S_FALSE, "ret %08x\n", hr );
1781     ok(pNextChild == NULL, "pDocChild not NULL\n");
1782
1783     hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing"), &pRoot);
1784     ok(hr == S_OK, "ret %08x\n", hr );
1785     if(hr == S_OK)
1786     {
1787         hr = IXMLDOMDocument_appendChild(doc, (IXMLDOMNode*)pRoot, NULL);
1788         ok(hr == S_OK, "ret %08x\n", hr );
1789         if(hr == S_OK)
1790         {
1791             /* Comment */
1792             hr = IXMLDOMDocument_createComment(doc, szComment, &pComment);
1793             ok(hr == S_OK, "ret %08x\n", hr );
1794             if(hr == S_OK)
1795             {
1796                 hr = IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pComment, NULL);
1797                 ok(hr == S_OK, "ret %08x\n", hr );
1798
1799                 hr = IXMLDOMComment_get_nodeName(pComment, &str);
1800                 ok(hr == S_OK, "ret %08x\n", hr );
1801                 ok( !lstrcmpW( str, szCommentNodeText ), "incorrect comment node Name\n");
1802                 SysFreeString(str);
1803
1804                 hr = IXMLDOMComment_get_xml(pComment, &str);
1805                 ok(hr == S_OK, "ret %08x\n", hr );
1806                 ok( !lstrcmpW( str, szCommentXML ), "incorrect comment xml\n");
1807                 SysFreeString(str);
1808
1809                 IXMLDOMComment_Release(pComment);
1810             }
1811
1812             /* Element */
1813             hr = IXMLDOMDocument_createElement(doc, szElement, &pElement);
1814             ok(hr == S_OK, "ret %08x\n", hr );
1815             if(hr == S_OK)
1816             {
1817                 hr = IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
1818                 ok(hr == S_OK, "ret %08x\n", hr );
1819
1820                 hr = IXMLDOMElement_get_nodeName(pElement, &str);
1821                 ok(hr == S_OK, "ret %08x\n", hr );
1822                 ok( !lstrcmpW( str, szElement ), "incorrect element node Name\n");
1823                 SysFreeString(str);
1824
1825                 hr = IXMLDOMElement_get_xml(pElement, &str);
1826                 ok(hr == S_OK, "ret %08x\n", hr );
1827                 ok( !lstrcmpW( str, szElementXML ), "incorrect element xml\n");
1828                 SysFreeString(str);
1829
1830                  /* Attribute */
1831                 hr = IXMLDOMDocument_createAttribute(doc, szAttribute, &pAttrubute);
1832                 ok(hr == S_OK, "ret %08x\n", hr );
1833                 if(hr == S_OK)
1834                 {
1835                     IXMLDOMNode *pNewChild = (IXMLDOMNode *)0x1;
1836
1837                     hr = IXMLDOMAttribute_get_nextSibling(pAttrubute, NULL);
1838                     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
1839
1840                     pNextChild = (IXMLDOMNode *)0x1;
1841                     hr = IXMLDOMAttribute_get_nextSibling(pAttrubute, &pNextChild);
1842                     ok(hr == S_FALSE, "ret %08x\n", hr );
1843                     ok(pNextChild == NULL, "pNextChild not NULL\n");
1844
1845                     hr = IXMLDOMElement_appendChild(pElement, (IXMLDOMNode*)pAttrubute, &pNewChild);
1846                     ok(hr == E_FAIL, "ret %08x\n", hr );
1847                     ok(pNewChild == NULL, "pNewChild not NULL\n");
1848
1849                     hr = IXMLDOMElement_get_attributes(pElement, &pAttribs);
1850                     ok(hr == S_OK, "ret %08x\n", hr );
1851                     if ( hr == S_OK )
1852                     {
1853                         hr = IXMLDOMNamedNodeMap_setNamedItem(pAttribs, (IXMLDOMNode*)pAttrubute, NULL );
1854                         ok(hr == S_OK, "ret %08x\n", hr );
1855
1856                         IXMLDOMNamedNodeMap_Release(pAttribs);
1857                     }
1858
1859                     hr = IXMLDOMAttribute_get_nodeName(pAttrubute, &str);
1860                     ok(hr == S_OK, "ret %08x\n", hr );
1861                     ok( !lstrcmpW( str, szAttribute ), "incorrect attribute node Name\n");
1862                     SysFreeString(str);
1863
1864                     hr = IXMLDOMAttribute_get_xml(pAttrubute, &str);
1865                     ok(hr == S_OK, "ret %08x\n", hr );
1866                     ok( !lstrcmpW( str, szAttributeXML ), "incorrect attribute xml\n");
1867                     SysFreeString(str);
1868
1869                     IXMLDOMAttribute_Release(pAttrubute);
1870
1871                     /* Check Element again with the Add Attribute*/
1872                     hr = IXMLDOMElement_get_xml(pElement, &str);
1873                     ok(hr == S_OK, "ret %08x\n", hr );
1874                     ok( !lstrcmpW( str, szElementXML2 ), "incorrect element xml\n");
1875                     SysFreeString(str);
1876                 }
1877
1878                 IXMLDOMElement_Release(pElement);
1879             }
1880
1881             IXMLDOMElement_Release( pRoot );
1882         }
1883     }
1884
1885     IXMLDOMDocument_Release(doc);
1886 }
1887
1888 START_TEST(domdoc)
1889 {
1890     HRESULT r;
1891
1892     r = CoInitialize( NULL );
1893     ok( r == S_OK, "failed to init com\n");
1894
1895     test_domdoc();
1896     test_domnode();
1897     test_refs();
1898     test_create();
1899     test_getElementsByTagName();
1900     test_get_text();
1901     test_get_childNodes();
1902     test_removeChild();
1903     test_XMLHTTP();
1904     test_IXMLDOMDocument2();
1905     test_XPath();
1906     test_cloneNode();
1907     test_xmlTypes();
1908
1909     CoUninitialize();
1910 }