makefiles: Rename the SRCDIR, TOPSRCDIR and TOPOBJDIR variables to follow autoconf...
[wine] / dlls / msxml3 / tests / domdoc.c
1 /*
2  * XML test
3  *
4  * Copyright 2005 Mike McCormack for CodeWeavers
5  * Copyright 2007-2008 Alistair Leslie-Hughes
6  *
7  * This library is free software; you can redistribute it and/or
8  * modify it under the terms of the GNU Lesser General Public
9  * License as published by the Free Software Foundation; either
10  * version 2.1 of the License, or (at your option) any later version.
11  *
12  * This library is distributed in the hope that it will be useful,
13  * but WITHOUT ANY WARRANTY; without even the implied warranty of
14  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
15  * Lesser General Public License for more details.
16  *
17  * You should have received a copy of the GNU Lesser General Public
18  * License along with this library; if not, write to the Free Software
19  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
20  */
21
22
23 #define COBJMACROS
24
25 #include "windows.h"
26 #include "ole2.h"
27 #include "objsafe.h"
28 #include "xmldom.h"
29 #include "msxml2.h"
30 #include "msxml2did.h"
31 #include "dispex.h"
32 #include <stdio.h>
33 #include <assert.h>
34
35 #include "wine/test.h"
36
37 #include "initguid.h"
38
39 DEFINE_GUID(IID_IObjectSafety, 0xcb5bdc81, 0x93c1, 0x11cf, 0x8f,0x20, 0x00,0x80,0x5f,0x2c,0xd0,0x64);
40 DEFINE_GUID(CLSID_DOMDocument60, 0x88d96a05, 0xf192, 0x11d4, 0xa6,0x5f, 0x00,0x40,0x96,0x32,0x51,0xe5);
41
42 static const WCHAR szEmpty[] = { 0 };
43 static const WCHAR szIncomplete[] = {
44     '<','?','x','m','l',' ',
45     'v','e','r','s','i','o','n','=','\'','1','.','0','\'','?','>','\n',0
46 };
47 static const WCHAR szComplete1[] = {
48     '<','?','x','m','l',' ',
49     'v','e','r','s','i','o','n','=','\'','1','.','0','\'','?','>','\n',
50     '<','o','p','e','n','>','<','/','o','p','e','n','>','\n',0
51 };
52 static const WCHAR szComplete2[] = {
53     '<','?','x','m','l',' ',
54     'v','e','r','s','i','o','n','=','\'','1','.','0','\'','?','>','\n',
55     '<','o','>','<','/','o','>','\n',0
56 };
57 static const WCHAR szComplete3[] = {
58     '<','?','x','m','l',' ',
59     'v','e','r','s','i','o','n','=','\'','1','.','0','\'','?','>','\n',
60     '<','a','>','<','/','a','>','\n',0
61 };
62 static const WCHAR szComplete4[] = {
63     '<','?','x','m','l',' ','v','e','r','s','i','o','n','=','\'','1','.','0','\'','?','>','\n',
64     '<','l','c',' ','d','l','=','\'','s','t','r','1','\'','>','\n',
65         '<','b','s',' ','v','r','=','\'','s','t','r','2','\'',' ','s','z','=','\'','1','2','3','4','\'','>',
66             'f','n','1','.','t','x','t','\n',
67         '<','/','b','s','>','\n',
68         '<','p','r',' ','i','d','=','\'','s','t','r','3','\'',' ','v','r','=','\'','1','.','2','.','3','\'',' ',
69                     'p','n','=','\'','w','i','n','e',' ','2','0','0','5','0','8','0','4','\'','>','\n',
70             'f','n','2','.','t','x','t','\n',
71         '<','/','p','r','>','\n',
72         '<','e','m','p','t','y','>','<','/','e','m','p','t','y','>','\n',
73         '<','f','o','>','\n',
74             '<','b','a','>','\n',
75                 'f','1','\n',
76             '<','/','b','a','>','\n',
77         '<','/','f','o','>','\n',
78     '<','/','l','c','>','\n',0
79 };
80 static const WCHAR szComplete5[] = {
81     '<','S',':','s','e','a','r','c','h',' ','x','m','l','n','s',':','D','=','"','D','A','V',':','"',' ',
82     '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','"',
83     ' ','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','"','>',
84         '<','S',':','s','c','o','p','e','>',
85             '<','S',':','d','e','e','p','>','/','<','/','S',':','d','e','e','p','>',
86         '<','/','S',':','s','c','o','p','e','>',
87         '<','S',':','c','o','n','t','e','n','t','f','r','e','e','t','e','x','t','>',
88             '<','C',':','t','e','x','t','o','r','p','r','o','p','e','r','t','y','/','>',
89             'c','o','m','p','u','t','e','r',
90         '<','/','S',':','c','o','n','t','e','n','t','f','r','e','e','t','e','x','t','>',
91     '<','/','S',':','s','e','a','r','c','h','>',0
92 };
93
94 static const WCHAR szComplete6[] = {
95     '<','?','x','m','l',' ','v','e','r','s','i','o','n','=','\'','1','.','0','\'',' ',
96     'e','n','c','o','d','i','n','g','=','\'','W','i','n','d','o','w','s','-','1','2','5','2','\'','?','>','\n',
97     '<','o','p','e','n','>','<','/','o','p','e','n','>','\n',0
98 };
99
100 static const CHAR szNonUnicodeXML[] =
101 "<?xml version='1.0' encoding='Windows-1252'?>\n"
102 "<open></open>\n";
103
104 static const CHAR szExampleXML[] =
105 "<?xml version='1.0' encoding='utf-8'?>\n"
106 "<root xmlns:foo='urn:uuid:86B2F87F-ACB6-45cd-8B77-9BDB92A01A29'>\n"
107 "    <elem>\n"
108 "        <a>A1 field</a>\n"
109 "        <b>B1 field</b>\n"
110 "        <c>C1 field</c>\n"
111 "        <description xmlns:foo='http://www.winehq.org' xmlns:bar='urn:uuid:86B2F87F-ACB6-45cd-8B77-9BDB92A01A29'>\n"
112 "            <html xmlns='http://www.w3.org/1999/xhtml'>\n"
113 "                This is <strong>a</strong> <i>description</i>. <bar:x/>\n"
114 "            </html>\n"
115 "        </description>\n"
116 "    </elem>\n"
117 "\n"
118 "    <elem>\n"
119 "        <a>A2 field</a>\n"
120 "        <b>B2 field</b>\n"
121 "        <c type=\"old\">C2 field</c>\n"
122 "    </elem>\n"
123 "\n"
124 "    <elem xmlns='urn:uuid:86B2F87F-ACB6-45cd-8B77-9BDB92A01A29'>\n"
125 "        <a>A3 field</a>\n"
126 "        <b>B3 field</b>\n"
127 "        <c>C3 field</c>\n"
128 "    </elem>\n"
129 "\n"
130 "    <elem>\n"
131 "        <a>A4 field</a>\n"
132 "        <b>B4 field</b>\n"
133 "        <foo:c>C4 field</foo:c>\n"
134 "    </elem>\n"
135 "</root>\n";
136
137 static  const CHAR szTransformXML[] =
138 "<?xml version=\"1.0\"?>\n"
139 "<greeting>\n"
140 "Hello World\n"
141 "</greeting>";
142
143 static  const CHAR szTransformSSXML[] =
144 "<xsl:stylesheet xmlns:xsl=\"http://www.w3.org/1999/XSL/Transform\" version=\"1.0\">\n"
145 "   <xsl:output method=\"html\"/>\n"
146 "   <xsl:template match=\"/\">\n"
147 "       <xsl:apply-templates select=\"greeting\"/>\n"
148 "   </xsl:template>\n"
149 "   <xsl:template match=\"greeting\">\n"
150 "       <html>\n"
151 "           <body>\n"
152 "               <h1>\n"
153 "                   <xsl:value-of select=\".\"/>\n"
154 "               </h1>\n"
155 "           </body>\n"
156 "       </html>\n"
157 "   </xsl:template>\n"
158 "</xsl:stylesheet>";
159
160 static  const CHAR szTransformOutput[] =
161 "<html><body><h1>"
162 "Hello World"
163 "</h1></body></html>";
164
165 static const CHAR szTypeValueXML[] =
166 "<?xml version=\"1.0\" encoding=\"utf-8\"?>\n"
167 "<root xmlns:dt=\"urn:schemas-microsoft-com:datatypes\">\n"
168 "   <string>Wine</string>\n"
169 "   <string2 dt:dt=\"string\">String</string2>\n"
170 "   <number dt:dt=\"number\">12.44</number>\n"
171 "   <number2 dt:dt=\"NUMbEr\">-3.71e3</number2>\n"
172 "   <int dt:dt=\"int\">-13</int>\n"
173 "   <fixed dt:dt=\"fixed.14.4\">7322.9371</fixed>\n"
174 "   <bool dt:dt=\"boolean\">1</bool>\n"
175 "   <datetime dt:dt=\"datetime\">2009-11-18T03:21:33.12</datetime>\n"
176 "   <datetimetz dt:dt=\"datetime.tz\">2003-07-11T11:13:57+03:00</datetimetz>\n"
177 "   <date dt:dt=\"date\">3721-11-01</date>\n"
178 "   <time dt:dt=\"time\">13:57:12.31321</time>\n"
179 "   <timetz dt:dt=\"time.tz\">23:21:01.13+03:21</timetz>\n"
180 "   <i1 dt:dt=\"i1\">-13</i1>\n"
181 "   <i2 dt:dt=\"i2\">31915</i2>\n"
182 "   <i4 dt:dt=\"i4\">-312232</i4>\n"
183 "   <ui1 dt:dt=\"ui1\">123</ui1>\n"
184 "   <ui2 dt:dt=\"ui2\">48282</ui2>\n"
185 "   <ui4 dt:dt=\"ui4\">949281</ui4>\n"
186 "   <r4 dt:dt=\"r4\">213124.0</r4>\n"
187 "   <r8 dt:dt=\"r8\">0.412</r8>\n"
188 "   <float dt:dt=\"float\">41221.421</float>\n"
189 "   <uuid dt:dt=\"uuid\">333C7BC4-460F-11D0-BC04-0080C7055a83</uuid>\n"
190 "   <binhex dt:dt=\"bin.hex\">fffca012003c</binhex>\n"
191 "   <binbase64 dt:dt=\"bin.base64\">YmFzZTY0IHRlc3Q=</binbase64>\n"
192 "</root>";
193
194 static const CHAR szBasicTransformSSXMLPart1[] =
195 "<?xml version=\"1.0\"?>"
196 "<xsl:stylesheet version=\"1.0\" xmlns:xsl=\"http://www.w3.org/1999/XSL/Transform\" >"
197 "<xsl:output method=\"html\"/>\n"
198 "<xsl:template match=\"/\">"
199 "<HTML><BODY><TABLE>"
200 "        <xsl:apply-templates select='document(\"";
201
202 static const CHAR szBasicTransformSSXMLPart2[] =
203 "\")/bottle/wine'>"
204 "           <xsl:sort select=\"cost\"/><xsl:sort select=\"name\"/>"
205 "        </xsl:apply-templates>"
206 "</TABLE></BODY></HTML>"
207 "</xsl:template>"
208 "<xsl:template match=\"bottle\">"
209 "   <TR><xsl:apply-templates select=\"name\" /><xsl:apply-templates select=\"cost\" /></TR>"
210 "</xsl:template>"
211 "<xsl:template match=\"name\">"
212 "   <TD><xsl:apply-templates /></TD>"
213 "</xsl:template>"
214 "<xsl:template match=\"cost\">"
215 "   <TD><xsl:apply-templates /></TD>"
216 "</xsl:template>"
217 "</xsl:stylesheet>";
218
219 static const CHAR szBasicTransformXML[] =
220 "<?xml version=\"1.0\"?><bottle><wine><name>Wine</name><cost>$25.00</cost></wine></bottle>";
221
222 static const CHAR szBasicTransformOutput[] =
223 "<HTML><BODY><TABLE><TD>Wine</TD><TD>$25.00</TD></TABLE></BODY></HTML>";
224
225 static const WCHAR szNonExistentFile[] = {
226     'c', ':', '\\', 'N', 'o', 'n', 'e', 'x', 'i', 's', 't', 'e', 'n', 't', '.', 'x', 'm', 'l', 0
227 };
228 static const WCHAR szNonExistentAttribute[] = {
229     'n','o','n','E','x','i','s','i','t','i','n','g','A','t','t','r','i','b','u','t','e',0
230 };
231 static const WCHAR szDocument[] = {
232     '#', 'd', 'o', 'c', 'u', 'm', 'e', 'n', 't', 0
233 };
234
235 static const WCHAR szOpen[] = { 'o','p','e','n',0 };
236 static WCHAR szdl[] = { 'd','l',0 };
237 static const WCHAR szvr[] = { 'v','r',0 };
238 static const WCHAR szlc[] = { 'l','c',0 };
239 static WCHAR szbs[] = { 'b','s',0 };
240 static const WCHAR szstr1[] = { 's','t','r','1',0 };
241 static const WCHAR szstr2[] = { 's','t','r','2',0 };
242 static const WCHAR szstar[] = { '*',0 };
243 static const WCHAR szfn1_txt[] = {'f','n','1','.','t','x','t',0};
244
245 static WCHAR szComment[] = {'A',' ','C','o','m','m','e','n','t',0 };
246 static WCHAR szCommentXML[] = {'<','!','-','-','A',' ','C','o','m','m','e','n','t','-','-','>',0 };
247 static WCHAR szCommentNodeText[] = {'#','c','o','m','m','e','n','t',0 };
248
249 static WCHAR szElement[] = {'E','l','e','T','e','s','t', 0 };
250 static WCHAR szElementXML[]  = {'<','E','l','e','T','e','s','t','/','>',0 };
251 static WCHAR szElementXML2[] = {'<','E','l','e','T','e','s','t',' ','A','t','t','r','=','"','"','/','>',0 };
252 static WCHAR szElementXML3[] = {'<','E','l','e','T','e','s','t',' ','A','t','t','r','=','"','"','>',
253                                 'T','e','s','t','i','n','g','N','o','d','e','<','/','E','l','e','T','e','s','t','>',0 };
254 static WCHAR szElementXML4[] = {'<','E','l','e','T','e','s','t',' ','A','t','t','r','=','"','"','>',
255                                 '&','a','m','p',';','x',' ',0x2103,'<','/','E','l','e','T','e','s','t','>',0 };
256
257 static WCHAR szAttribute[] = {'A','t','t','r',0 };
258 static WCHAR szAttributeXML[] = {'A','t','t','r','=','"','"',0 };
259
260 static WCHAR szCData[] = {'[','1',']','*','2','=','3',';',' ','&','g','e','e',' ','t','h','a','t','s',
261                           ' ','n','o','t',' ','r','i','g','h','t','!', 0};
262 static WCHAR szCDataXML[] = {'<','!','[','C','D','A','T','A','[','[','1',']','*','2','=','3',';',' ','&',
263                              'g','e','e',' ','t','h','a','t','s',' ','n','o','t',' ','r','i','g','h','t',
264                              '!',']',']','>',0};
265 static WCHAR szCDataNodeText[] = {'#','c','d','a','t','a','-','s','e','c','t','i','o','n',0 };
266 static WCHAR szDocFragmentText[] = {'#','d','o','c','u','m','e','n','t','-','f','r','a','g','m','e','n','t',0 };
267
268 static WCHAR szEntityRef[] = {'e','n','t','i','t','y','r','e','f',0 };
269 static WCHAR szEntityRefXML[] = {'&','e','n','t','i','t','y','r','e','f',';',0 };
270 static WCHAR szStrangeChars[] = {'&','x',' ',0x2103, 0};
271
272 #define expect_bstr_eq_and_free(bstr, expect) { \
273     BSTR bstrExp = alloc_str_from_narrow(expect); \
274     ok(lstrcmpW(bstr, bstrExp) == 0, "String differs\n"); \
275     SysFreeString(bstr); \
276     SysFreeString(bstrExp); \
277 }
278
279 #define expect_eq(expr, value, type, format) { type ret = (expr); ok((value) == ret, #expr " expected " format " got " format "\n", value, ret); }
280
281 #define ole_check(expr) { \
282     HRESULT r = expr; \
283     ok(r == S_OK, #expr " returned %x\n", r); \
284 }
285
286 #define ole_expect(expr, expect) { \
287     HRESULT r = expr; \
288     ok(r == (expect), #expr " returned %x, expected %x\n", r, expect); \
289 }
290
291 #define double_eq(x, y) ok((x)-(y)<=1e-14*(x) && (x)-(y)>=-1e-14*(x), "expected %.16g, got %.16g\n", x, y)
292
293 static void* _create_document(const IID *iid, int line)
294 {
295     void *doc = NULL;
296     HRESULT hr;
297
298     hr = CoCreateInstance( &CLSID_DOMDocument, NULL, CLSCTX_INPROC_SERVER, iid, (void**)&doc );
299     ok_(__FILE__,line)( hr == S_OK, "failed to create CLSID_DOMDocument instance: 0x%08x\n", hr );
300
301     return doc;
302 }
303
304 #define create_document(iid) _create_document(iid, __LINE__)
305
306 static BSTR alloc_str_from_narrow(const char *str)
307 {
308     int len = MultiByteToWideChar(CP_ACP, 0, str, -1, NULL, 0);
309     BSTR ret = SysAllocStringLen(NULL, len - 1);  /* NUL character added automatically */
310     MultiByteToWideChar(CP_ACP, 0, str, -1, ret, len);
311     return ret;
312 }
313
314 BSTR alloced_bstrs[256];
315 int alloced_bstrs_count = 0;
316
317 static BSTR _bstr_(const char *str)
318 {
319     assert(alloced_bstrs_count < sizeof(alloced_bstrs)/sizeof(alloced_bstrs[0]));
320     alloced_bstrs[alloced_bstrs_count] = alloc_str_from_narrow(str);
321     return alloced_bstrs[alloced_bstrs_count++];
322 }
323
324 static void free_bstrs(void)
325 {
326     int i;
327     for (i = 0; i < alloced_bstrs_count; i++)
328         SysFreeString(alloced_bstrs[i]);
329     alloced_bstrs_count = 0;
330 }
331
332 static VARIANT _variantbstr_(const char *str)
333 {
334     VARIANT v;
335     V_VT(&v) = VT_BSTR;
336     V_BSTR(&v) = _bstr_(str);
337     return v;
338 }
339
340 static BOOL compareIgnoreReturns(BSTR sLeft, BSTR sRight)
341 {
342     for (;;)
343     {
344         while (*sLeft == '\r' || *sLeft == '\n') sLeft++;
345         while (*sRight == '\r' || *sRight == '\n') sRight++;
346         if (*sLeft != *sRight) return FALSE;
347         if (!*sLeft) return TRUE;
348         sLeft++;
349         sRight++;
350     }
351 }
352
353 static BOOL compareIgnoreReturnsWhitespace(BSTR sLeft, BSTR sRight)
354 {
355     /* MSXML3 inserts whitespace where as libxml doesn't. */
356     for (;;)
357     {
358         while (*sLeft == '\r' || *sLeft == '\n' || *sLeft == ' ') sLeft++;
359         while (*sRight == '\r' || *sRight == '\n' || *sRight == ' ') sRight++;
360         if (*sLeft != *sRight) return FALSE;
361         if (!*sLeft) return TRUE;
362         sLeft++;
363         sRight++;
364     }
365 }
366
367 static void get_str_for_type(DOMNodeType type, char *buf)
368 {
369     switch (type)
370     {
371         case NODE_ATTRIBUTE:
372             strcpy(buf, "A");
373             break;
374         case NODE_ELEMENT:
375             strcpy(buf, "E");
376             break;
377         case NODE_DOCUMENT:
378             strcpy(buf, "D");
379             break;
380         default:
381             wsprintfA(buf, "[%d]", type);
382     }
383 }
384
385 #define test_disp(u) _test_disp(__LINE__,u)
386 static void _test_disp(unsigned line, IUnknown *unk)
387 {
388     DISPID dispid = DISPID_XMLDOM_NODELIST_RESET;
389     IDispatchEx *dispex;
390     DWORD dwProps = 0;
391     BSTR sName;
392     UINT ticnt;
393     IUnknown *pUnk;
394     HRESULT hres;
395
396     hres = IUnknown_QueryInterface(unk, &IID_IDispatchEx, (void**)&dispex);
397     ok_(__FILE__,line) (hres == S_OK, "Could not get IDispatch: %08x\n", hres);
398     if(FAILED(hres))
399         return;
400
401     ticnt = 0xdeadbeef;
402     hres = IDispatchEx_GetTypeInfoCount(dispex, &ticnt);
403     ok_(__FILE__,line) (hres == S_OK, "GetTypeInfoCount failed: %08x\n", hres);
404     ok_(__FILE__,line) (ticnt == 1, "ticnt=%u\n", ticnt);
405
406     sName = SysAllocString( szstar );
407     hres = IDispatchEx_DeleteMemberByName(dispex, sName, fdexNameCaseSensitive);
408     ok(hres == E_NOTIMPL, "expected E_NOTIMPL got %08x\n", hres);
409     SysFreeString( sName );
410
411     hres = IDispatchEx_DeleteMemberByDispID(dispex, dispid);
412     ok(hres == E_NOTIMPL, "expected E_NOTIMPL got %08x\n", hres);
413
414     hres = IDispatchEx_GetMemberProperties(dispex, dispid, grfdexPropCanAll, &dwProps);
415     ok(hres == E_NOTIMPL, "expected E_NOTIMPL got %08x\n", hres);
416     ok(dwProps == 0, "expected 0 got %d\n", dwProps);
417
418     hres = IDispatchEx_GetMemberName(dispex, dispid, &sName);
419     ok(hres == E_NOTIMPL, "expected E_NOTIMPL got %08x\n", hres);
420     if(SUCCEEDED(hres))
421         SysFreeString(sName);
422
423     hres = IDispatchEx_GetNextDispID(dispex, fdexEnumDefault, DISPID_XMLDOM_NODELIST_RESET, &dispid);
424     ok(hres == E_NOTIMPL, "expected E_NOTIMPL got %08x\n", hres);
425
426     hres = IDispatchEx_GetNameSpaceParent(dispex, &pUnk);
427     ok(hres == E_NOTIMPL, "expected E_NOTIMPL got %08x\n", hres);
428     if(hres == S_OK && pUnk)
429         IUnknown_Release(pUnk);
430
431     IDispatchEx_Release(dispex);
432 }
433
434 static int get_node_position(IXMLDOMNode *node)
435 {
436     HRESULT r;
437     int pos = 0;
438
439     IXMLDOMNode_AddRef(node);
440     do
441     {
442         IXMLDOMNode *new_node;
443
444         pos++;
445         r = IXMLDOMNode_get_previousSibling(node, &new_node);
446         ok(SUCCEEDED(r), "get_previousSibling failed\n");
447         IXMLDOMNode_Release(node);
448         node = new_node;
449     } while (r == S_OK);
450     return pos;
451 }
452
453 static void node_to_string(IXMLDOMNode *node, char *buf)
454 {
455     HRESULT r = S_OK;
456     DOMNodeType type;
457
458     if (node == NULL)
459     {
460         lstrcpyA(buf, "(null)");
461         return;
462     }
463
464     IXMLDOMNode_AddRef(node);
465     while (r == S_OK)
466     {
467         IXMLDOMNode *new_node;
468
469         ole_check(IXMLDOMNode_get_nodeType(node, &type));
470         get_str_for_type(type, buf);
471         buf+=strlen(buf);
472
473         if (type == NODE_ATTRIBUTE)
474         {
475             BSTR bstr;
476             ole_check(IXMLDOMNode_get_nodeName(node, &bstr));
477             *(buf++) = '\'';
478             wsprintfA(buf, "%ws", bstr);
479             buf += strlen(buf);
480             *(buf++) = '\'';
481             SysFreeString(bstr);
482
483             r = IXMLDOMNode_selectSingleNode(node, _bstr_(".."), &new_node);
484         }
485         else
486         {
487             r = IXMLDOMNode_get_parentNode(node, &new_node);
488             wsprintf(buf, "%d", get_node_position(node));
489             buf += strlen(buf);
490         }
491
492         ok(SUCCEEDED(r), "get_parentNode failed (%08x)\n", r);
493         IXMLDOMNode_Release(node);
494         node = new_node;
495         if (r == S_OK)
496             *(buf++) = '.';
497     }
498
499     *buf = 0;
500 }
501
502 static char *list_to_string(IXMLDOMNodeList *list)
503 {
504     static char buf[4096];
505     char *pos = buf;
506     LONG len = 0;
507     int i;
508
509     if (list == NULL)
510     {
511         lstrcpyA(buf, "(null)");
512         return buf;
513     }
514     ole_check(IXMLDOMNodeList_get_length(list, &len));
515     for (i = 0; i < len; i++)
516     {
517         IXMLDOMNode *node;
518         if (i > 0)
519             *(pos++) = ' ';
520         ole_check(IXMLDOMNodeList_nextNode(list, &node));
521         node_to_string(node, pos);
522         pos += strlen(pos);
523         IXMLDOMNode_Release(node);
524     }
525     *pos = 0;
526     return buf;
527 }
528
529 #define expect_node(node, expstr) { char str[4096]; node_to_string(node, str); ok(strcmp(str, expstr)==0, "Invalid node: %s, expected %s\n", str, expstr); }
530 #define expect_list_and_release(list, expstr) { char *str = list_to_string(list); ok(strcmp(str, expstr)==0, "Invalid node list: %s, expected %s\n", str, expstr); if (list) IXMLDOMNodeList_Release(list); }
531
532 static void test_domdoc( void )
533 {
534     HRESULT r;
535     IXMLDOMDocument *doc;
536     IXMLDOMParseError *error;
537     IXMLDOMElement *element = NULL;
538     IXMLDOMNode *node;
539     IXMLDOMText *nodetext = NULL;
540     IXMLDOMComment *node_comment = NULL;
541     IXMLDOMAttribute *node_attr = NULL;
542     IXMLDOMNode *nodeChild = NULL;
543     IXMLDOMProcessingInstruction *nodePI = NULL;
544     ISupportErrorInfo *support_error = NULL;
545     VARIANT_BOOL b;
546     VARIANT var;
547     BSTR str;
548     LONG code;
549     LONG nLength = 0;
550     WCHAR buff[100];
551
552     doc = create_document(&IID_IXMLDOMDocument);
553     if (!doc) return;
554
555     test_disp((IUnknown*)doc);
556
557 if (0)
558 {
559     /* crashes on native */
560     r = IXMLDOMDocument_loadXML( doc, (BSTR)0x1, NULL );
561 }
562
563     /* try some stupid things */
564     r = IXMLDOMDocument_loadXML( doc, NULL, NULL );
565     ok( r == S_FALSE, "loadXML succeeded\n");
566
567     b = VARIANT_TRUE;
568     r = IXMLDOMDocument_loadXML( doc, NULL, &b );
569     ok( r == S_FALSE, "loadXML succeeded\n");
570     ok( b == VARIANT_FALSE, "failed to load XML string\n");
571
572     /* try to load a document from a nonexistent file */
573     b = VARIANT_TRUE;
574     str = SysAllocString( szNonExistentFile );
575     VariantInit(&var);
576     V_VT(&var) = VT_BSTR;
577     V_BSTR(&var) = str;
578
579     r = IXMLDOMDocument_load( doc, var, &b);
580     ok( r == S_FALSE, "loadXML succeeded\n");
581     ok( b == VARIANT_FALSE, "succeeded in loading XML string\n");
582     SysFreeString( str );
583
584     /* try load an empty document */
585     b = VARIANT_TRUE;
586     str = SysAllocString( szEmpty );
587     r = IXMLDOMDocument_loadXML( doc, str, &b );
588     ok( r == S_FALSE, "loadXML succeeded\n");
589     ok( b == VARIANT_FALSE, "succeeded in loading XML string\n");
590     SysFreeString( str );
591
592     r = IXMLDOMDocument_get_async( doc, &b );
593     ok( r == S_OK, "get_async failed (%08x)\n", r);
594     ok( b == VARIANT_TRUE, "Wrong default value\n");
595
596     /* check that there's no document element */
597     element = NULL;
598     r = IXMLDOMDocument_get_documentElement( doc, &element );
599     ok( r == S_FALSE, "should be no document element\n");
600
601     /* try finding a node */
602     node = NULL;
603     str = SysAllocString( szstr1 );
604     r = IXMLDOMDocument_selectSingleNode( doc, str, &node );
605     ok( r == S_FALSE, "ret %08x\n", r );
606     SysFreeString( str );
607
608     b = VARIANT_TRUE;
609     str = SysAllocString( szIncomplete );
610     r = IXMLDOMDocument_loadXML( doc, str, &b );
611     ok( r == S_FALSE, "loadXML succeeded\n");
612     ok( b == VARIANT_FALSE, "succeeded in loading XML string\n");
613     SysFreeString( str );
614
615     /* check that there's no document element */
616     element = (IXMLDOMElement*)1;
617     r = IXMLDOMDocument_get_documentElement( doc, &element );
618     ok( r == S_FALSE, "should be no document element\n");
619     ok( element == NULL, "Element should be NULL\n");
620
621     /* test for BSTR handling, pass broken BSTR */
622     memcpy(&buff[2], szComplete1, sizeof(szComplete1));
623     /* just a big length */
624     *(DWORD*)buff = 0xf0f0;
625     b = VARIANT_FALSE;
626     r = IXMLDOMDocument_loadXML( doc, &buff[2], &b );
627     ok( r == S_OK, "loadXML failed\n");
628     ok( b == VARIANT_TRUE, "failed to load XML string\n");
629
630     /* loadXML ignores the encoding attribute and always expects Unicode */
631     b = VARIANT_FALSE;
632     str = SysAllocString( szComplete6 );
633     r = IXMLDOMDocument_loadXML( doc, str, &b );
634     ok( r == S_OK, "loadXML failed\n");
635     ok( b == VARIANT_TRUE, "failed to load XML string\n");
636     SysFreeString( str );
637
638     /* try a BSTR containing a Windows-1252 document */
639     b = VARIANT_TRUE;
640     str = SysAllocStringByteLen( szNonUnicodeXML, sizeof(szNonUnicodeXML) - 1 );
641     r = IXMLDOMDocument_loadXML( doc, str, &b );
642     ok( r == S_FALSE, "loadXML succeeded\n");
643     ok( b == VARIANT_FALSE, "succeeded in loading XML string\n");
644     SysFreeString( str );
645
646     /* try to load something valid */
647     b = VARIANT_FALSE;
648     str = SysAllocString( szComplete1 );
649     r = IXMLDOMDocument_loadXML( doc, str, &b );
650     ok( r == S_OK, "loadXML failed\n");
651     ok( b == VARIANT_TRUE, "failed to load XML string\n");
652     SysFreeString( str );
653
654     /* check if nodename is correct */
655     r = IXMLDOMDocument_get_nodeName( doc, NULL );
656     ok ( r == E_INVALIDARG, "get_nodeName (NULL) wrong code\n");
657
658     str = (BSTR)0xdeadbeef;
659     r = IXMLDOMDocument_get_baseName( doc, &str );
660     ok ( r == S_FALSE, "got 0x%08x\n", r);
661     ok (str == NULL, "got %p\n", str);
662
663     /* content doesn't matter here */
664     str = NULL;
665     r = IXMLDOMDocument_get_nodeName( doc, &str );
666     ok ( r == S_OK, "get_nodeName wrong code\n");
667     ok ( str != NULL, "str is null\n");
668     ok( !lstrcmpW( str, szDocument ), "incorrect nodeName\n");
669     SysFreeString( str );
670
671     /* test put_text */
672     r = IXMLDOMDocument_put_text( doc, _bstr_("Should Fail") );
673     ok( r == E_FAIL, "ret %08x\n", r );
674
675     /* check that there's a document element */
676     element = NULL;
677     r = IXMLDOMDocument_get_documentElement( doc, &element );
678     ok( r == S_OK, "should be a document element\n");
679     if( element )
680     {
681         IObjectIdentity *ident;
682         BSTR tag = NULL;
683
684         test_disp((IUnknown*)element);
685
686         r = IXMLDOMElement_QueryInterface( element, &IID_IObjectIdentity, (void**)&ident );
687         ok( r == E_NOINTERFACE, "ret %08x\n", r);
688
689         r = IXMLDOMElement_get_tagName( element, NULL );
690         ok( r == E_INVALIDARG, "ret %08x\n", r);
691
692         /* check if the tag is correct */
693         r = IXMLDOMElement_get_tagName( element, &tag );
694         ok( r == S_OK, "couldn't get tag name\n");
695         ok( tag != NULL, "tag was null\n");
696         ok( !lstrcmpW( tag, szOpen ), "incorrect tag name\n");
697         SysFreeString( tag );
698
699         /* figure out what happens if we try to reload the document */
700         str = SysAllocString( szComplete2 );
701         r = IXMLDOMDocument_loadXML( doc, str, &b );
702         ok( r == S_OK, "loadXML failed\n");
703         ok( b == VARIANT_TRUE, "failed to load XML string\n");
704         SysFreeString( str );
705
706         /* check if the tag is still correct */
707         tag = NULL;
708         r = IXMLDOMElement_get_tagName( element, &tag );
709         ok( r == S_OK, "couldn't get tag name\n");
710         ok( tag != NULL, "tag was null\n");
711         ok( !lstrcmpW( tag, szOpen ), "incorrect tag name\n");
712         SysFreeString( tag );
713
714         IXMLDOMElement_Release( element );
715         element = NULL;
716     }
717
718     /* as soon as we call loadXML again, the document element will disappear */
719     b = 2;
720     r = IXMLDOMDocument_loadXML( doc, NULL, NULL );
721     ok( r == S_FALSE, "loadXML failed\n");
722     ok( b == 2, "variant modified\n");
723     r = IXMLDOMDocument_get_documentElement( doc, &element );
724     ok( r == S_FALSE, "should be no document element\n");
725
726     /* try to load something else simple and valid */
727     b = VARIANT_FALSE;
728     str = SysAllocString( szComplete3 );
729     r = IXMLDOMDocument_loadXML( doc, str, &b );
730     ok( r == S_OK, "loadXML failed\n");
731     ok( b == VARIANT_TRUE, "failed to load XML string\n");
732     SysFreeString( str );
733
734     /* try something a little more complicated */
735     b = FALSE;
736     str = SysAllocString( szComplete4 );
737     r = IXMLDOMDocument_loadXML( doc, str, &b );
738     ok( r == S_OK, "loadXML failed\n");
739     ok( b == VARIANT_TRUE, "failed to load XML string\n");
740     SysFreeString( str );
741
742     r = IXMLDOMDocument_get_parseError( doc, &error );
743     ok( r == S_OK, "returns %08x\n", r );
744
745     r = IXMLDOMParseError_get_errorCode( error, &code );
746     ok( r == S_FALSE, "returns %08x\n", r );
747     ok( code == 0, "code %d\n", code );
748     IXMLDOMParseError_Release( error );
749
750     /* test createTextNode */
751     r = IXMLDOMDocument_createTextNode(doc, _bstr_(""), &nodetext);
752     ok( r == S_OK, "returns %08x\n", r );
753     IXMLDOMText_Release(nodetext);
754
755     str = SysAllocString( szOpen );
756     r = IXMLDOMDocument_createTextNode(doc, str, NULL);
757     ok( r == E_INVALIDARG, "returns %08x\n", r );
758     r = IXMLDOMDocument_createTextNode(doc, str, &nodetext);
759     ok( r == S_OK, "returns %08x\n", r );
760     SysFreeString( str );
761     if(nodetext)
762     {
763         IXMLDOMNamedNodeMap *pAttribs;
764
765         r = IXMLDOMText_QueryInterface(nodetext, &IID_IXMLDOMElement, (void**)&element);
766         ok(r == E_NOINTERFACE, "ret %08x\n", r );
767
768         /* Text Last Child Checks */
769         r = IXMLDOMText_get_lastChild(nodetext, NULL);
770         ok(r == E_INVALIDARG, "ret %08x\n", r );
771
772         nodeChild = (IXMLDOMNode*)0x1;
773         r = IXMLDOMText_get_lastChild(nodetext, &nodeChild);
774         ok(r == S_FALSE, "ret %08x\n", r );
775         ok(nodeChild == NULL, "nodeChild not NULL\n");
776
777         /* test get_attributes */
778         r = IXMLDOMText_get_attributes( nodetext, NULL );
779         ok( r == E_INVALIDARG, "get_attributes returned wrong code\n");
780
781         pAttribs = (IXMLDOMNamedNodeMap*)0x1;
782         r = IXMLDOMText_get_attributes( nodetext, &pAttribs);
783         ok(r == S_FALSE, "ret %08x\n", r );
784         ok( pAttribs == NULL, "pAttribs not NULL\n");
785
786         /* test get_dataType */
787         r = IXMLDOMText_get_dataType(nodetext, &var);
788         ok(r == S_FALSE, "ret %08x\n", r );
789         ok( V_VT(&var) == VT_NULL, "incorrect dataType type\n");
790         VariantClear(&var);
791
792         /* test length property */
793         r = IXMLDOMText_get_length(nodetext, NULL);
794         ok(r == E_INVALIDARG, "ret %08x\n", r );
795
796         r = IXMLDOMText_get_length(nodetext, &nLength);
797         ok(r == S_OK, "ret %08x\n", r );
798         ok(nLength == 4, "expected 4 got %d\n", nLength);
799
800         /* test nodeTypeString */
801         r = IXMLDOMText_get_nodeTypeString(nodetext, &str);
802         ok(r == S_OK, "ret %08x\n", r );
803         ok( !lstrcmpW( str, _bstr_("text") ), "incorrect nodeTypeString string\n");
804         SysFreeString(str);
805
806         /* put data Tests */
807         r = IXMLDOMText_put_data(nodetext, _bstr_("This &is a ; test <>\\"));
808         ok(r == S_OK, "ret %08x\n", r );
809
810         /* get data Tests */
811         r = IXMLDOMText_get_data(nodetext, &str);
812         ok(r == S_OK, "ret %08x\n", r );
813         ok( !lstrcmpW( str, _bstr_("This &is a ; test <>\\") ), "incorrect put_data string\n");
814         SysFreeString(str);
815
816         /* Confirm XML text is good */
817         r = IXMLDOMText_get_xml(nodetext, &str);
818         ok(r == S_OK, "ret %08x\n", r );
819         ok( !lstrcmpW( str, _bstr_("This &amp;is a ; test &lt;&gt;\\") ), "incorrect xml string\n");
820         SysFreeString(str);
821
822         /* Confirm we get the put_data Text back */
823         r = IXMLDOMText_get_text(nodetext, &str);
824         ok(r == S_OK, "ret %08x\n", r );
825         ok( !lstrcmpW( str, _bstr_("This &is a ; test <>\\") ), "incorrect xml string\n");
826         SysFreeString(str);
827
828         /* test substringData */
829         r = IXMLDOMText_substringData(nodetext, 0, 4, NULL);
830         ok(r == E_INVALIDARG, "ret %08x\n", r );
831
832         /* test substringData - Invalid offset */
833         str = (BSTR)&szElement;
834         r = IXMLDOMText_substringData(nodetext, -1, 4, &str);
835         ok(r == E_INVALIDARG, "ret %08x\n", r );
836         ok( str == NULL, "incorrect string\n");
837
838         /* test substringData - Invalid offset */
839         str = (BSTR)&szElement;
840         r = IXMLDOMText_substringData(nodetext, 30, 0, &str);
841         ok(r == S_FALSE, "ret %08x\n", r );
842         ok( str == NULL, "incorrect string\n");
843
844         /* test substringData - Invalid size */
845         str = (BSTR)&szElement;
846         r = IXMLDOMText_substringData(nodetext, 0, -1, &str);
847         ok(r == E_INVALIDARG, "ret %08x\n", r );
848         ok( str == NULL, "incorrect string\n");
849
850         /* test substringData - Invalid size */
851         str = (BSTR)&szElement;
852         r = IXMLDOMText_substringData(nodetext, 2, 0, &str);
853         ok(r == S_FALSE, "ret %08x\n", r );
854         ok( str == NULL, "incorrect string\n");
855
856         /* test substringData - Start of string */
857         r = IXMLDOMText_substringData(nodetext, 0, 4, &str);
858         ok(r == S_OK, "ret %08x\n", r );
859         ok( !lstrcmpW( str, _bstr_("This") ), "incorrect substringData string\n");
860         SysFreeString(str);
861
862         /* test substringData - Middle of string */
863         r = IXMLDOMText_substringData(nodetext, 13, 4, &str);
864         ok(r == S_OK, "ret %08x\n", r );
865         ok( !lstrcmpW( str, _bstr_("test") ), "incorrect substringData string\n");
866         SysFreeString(str);
867
868         /* test substringData - End of string */
869         r = IXMLDOMText_substringData(nodetext, 20, 4, &str);
870         ok(r == S_OK, "ret %08x\n", r );
871         ok( !lstrcmpW( str, _bstr_("\\") ), "incorrect substringData string\n");
872         SysFreeString(str);
873
874         /* test appendData */
875         r = IXMLDOMText_appendData(nodetext, NULL);
876         ok(r == S_OK, "ret %08x\n", r );
877
878         r = IXMLDOMText_appendData(nodetext, _bstr_(""));
879         ok(r == S_OK, "ret %08x\n", r );
880
881         r = IXMLDOMText_appendData(nodetext, _bstr_("Append"));
882         ok(r == S_OK, "ret %08x\n", r );
883
884         r = IXMLDOMText_get_text(nodetext, &str);
885         ok(r == S_OK, "ret %08x\n", r );
886         ok( !lstrcmpW( str, _bstr_("This &is a ; test <>\\Append") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
887         SysFreeString(str);
888
889         /* test insertData */
890         str = SysAllocStringLen(NULL, 0);
891         r = IXMLDOMText_insertData(nodetext, -1, str);
892         ok(r == S_OK, "ret %08x\n", r );
893
894         r = IXMLDOMText_insertData(nodetext, -1, NULL);
895         ok(r == S_OK, "ret %08x\n", r );
896
897         r = IXMLDOMText_insertData(nodetext, 1000, str);
898         ok(r == S_OK, "ret %08x\n", r );
899
900         r = IXMLDOMText_insertData(nodetext, 1000, NULL);
901         ok(r == S_OK, "ret %08x\n", r );
902
903         r = IXMLDOMText_insertData(nodetext, 0, NULL);
904         ok(r == S_OK, "ret %08x\n", r );
905
906         r = IXMLDOMText_insertData(nodetext, 0, str);
907         ok(r == S_OK, "ret %08x\n", r );
908         SysFreeString(str);
909
910         r = IXMLDOMText_insertData(nodetext, -1, _bstr_("Inserting"));
911         ok(r == E_INVALIDARG, "ret %08x\n", r );
912
913         r = IXMLDOMText_insertData(nodetext, 1000, _bstr_("Inserting"));
914         ok(r == E_INVALIDARG, "ret %08x\n", r );
915
916         r = IXMLDOMText_insertData(nodetext, 0, _bstr_("Begin "));
917         ok(r == S_OK, "ret %08x\n", r );
918
919         r = IXMLDOMText_insertData(nodetext, 17, _bstr_("Middle"));
920         ok(r == S_OK, "ret %08x\n", r );
921
922         r = IXMLDOMText_insertData(nodetext, 39, _bstr_(" End"));
923         ok(r == S_OK, "ret %08x\n", r );
924
925         r = IXMLDOMText_get_text(nodetext, &str);
926         ok(r == S_OK, "ret %08x\n", r );
927         ok( !lstrcmpW( str, _bstr_("Begin This &is a Middle; test <>\\Append End") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
928         SysFreeString(str);
929
930         /* delete data */
931         /* invalid arguments */
932         r = IXMLDOMText_deleteData(nodetext, -1, 1);
933         ok(r == E_INVALIDARG, "ret %08x\n", r );
934
935         r = IXMLDOMText_deleteData(nodetext, 0, 0);
936         ok(r == S_OK, "ret %08x\n", r );
937
938         r = IXMLDOMText_deleteData(nodetext, 0, -1);
939         ok(r == E_INVALIDARG, "ret %08x\n", r );
940
941         r = IXMLDOMText_get_length(nodetext, &nLength);
942         ok(r == S_OK, "ret %08x\n", r );
943         ok(nLength == 43, "expected 43 got %d\n", nLength);
944
945         r = IXMLDOMText_deleteData(nodetext, nLength, 1);
946         ok(r == S_OK, "ret %08x\n", r );
947
948         r = IXMLDOMText_deleteData(nodetext, nLength+1, 1);
949         ok(r == E_INVALIDARG, "ret %08x\n", r );
950
951         /* delete from start */
952         r = IXMLDOMText_deleteData(nodetext, 0, 5);
953         ok(r == S_OK, "ret %08x\n", r );
954
955         r = IXMLDOMText_get_length(nodetext, &nLength);
956         ok(r == S_OK, "ret %08x\n", r );
957         ok(nLength == 38, "expected 38 got %d\n", nLength);
958
959         r = IXMLDOMText_get_text(nodetext, &str);
960         ok(r == S_OK, "ret %08x\n", r );
961         /* whitespace preserving needs to be handled here */
962         todo_wine ok( !lstrcmpW( str, _bstr_("This &is a Middle; test <>\\Append End") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
963         SysFreeString(str);
964
965         /* delete from end */
966         r = IXMLDOMText_deleteData(nodetext, 35, 3);
967         ok(r == S_OK, "ret %08x\n", r );
968
969         r = IXMLDOMText_get_length(nodetext, &nLength);
970         ok(r == S_OK, "ret %08x\n", r );
971         ok(nLength == 35, "expected 35 got %d\n", nLength);
972
973         r = IXMLDOMText_get_text(nodetext, &str);
974         ok(r == S_OK, "ret %08x\n", r );
975         todo_wine ok( !lstrcmpW( str, _bstr_("This &is a Middle; test <>\\Append") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
976         SysFreeString(str);
977
978         /* delete from inside */
979         r = IXMLDOMText_deleteData(nodetext, 1, 33);
980         ok(r == S_OK, "ret %08x\n", r );
981
982         r = IXMLDOMText_get_length(nodetext, &nLength);
983         ok(r == S_OK, "ret %08x\n", r );
984         ok(nLength == 2, "expected 2 got %d\n", nLength);
985
986         r = IXMLDOMText_get_text(nodetext, &str);
987         ok(r == S_OK, "ret %08x\n", r );
988         todo_wine ok( !lstrcmpW( str, _bstr_("") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
989         SysFreeString(str);
990
991         /* delete whole data ... */
992         r = IXMLDOMText_get_length(nodetext, &nLength);
993         ok(r == S_OK, "ret %08x\n", r );
994
995         r = IXMLDOMText_deleteData(nodetext, 0, nLength);
996         ok(r == S_OK, "ret %08x\n", r );
997         /* ... and try again with empty string */
998         r = IXMLDOMText_deleteData(nodetext, 0, nLength);
999         ok(r == S_OK, "ret %08x\n", r );
1000
1001         /* test put_data */
1002         V_VT(&var) = VT_BSTR;
1003         V_BSTR(&var) = SysAllocString(szstr1);
1004         r = IXMLDOMText_put_nodeValue(nodetext, var);
1005         ok(r == S_OK, "ret %08x\n", r );
1006         VariantClear(&var);
1007
1008         r = IXMLDOMText_get_text(nodetext, &str);
1009         ok(r == S_OK, "ret %08x\n", r );
1010         ok( !lstrcmpW( str, szstr1 ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
1011         SysFreeString(str);
1012
1013         /* test put_data */
1014         V_VT(&var) = VT_I4;
1015         V_I4(&var) = 99;
1016         r = IXMLDOMText_put_nodeValue(nodetext, var);
1017         ok(r == S_OK, "ret %08x\n", r );
1018         VariantClear(&var);
1019
1020         r = IXMLDOMText_get_text(nodetext, &str);
1021         ok(r == S_OK, "ret %08x\n", r );
1022         ok( !lstrcmpW( str, _bstr_("99") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
1023         SysFreeString(str);
1024
1025         /* ::replaceData() */
1026         V_VT(&var) = VT_BSTR;
1027         V_BSTR(&var) = SysAllocString(szstr1);
1028         r = IXMLDOMText_put_nodeValue(nodetext, var);
1029         ok(r == S_OK, "ret %08x\n", r );
1030         VariantClear(&var);
1031
1032         r = IXMLDOMText_replaceData(nodetext, 6, 0, NULL);
1033         ok(r == E_INVALIDARG, "ret %08x\n", r );
1034         r = IXMLDOMText_get_text(nodetext, &str);
1035         ok(r == S_OK, "ret %08x\n", r );
1036         ok( !lstrcmpW( str, _bstr_("str1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
1037         SysFreeString(str);
1038
1039         r = IXMLDOMText_replaceData(nodetext, 0, 0, NULL);
1040         ok(r == S_OK, "ret %08x\n", r );
1041         r = IXMLDOMText_get_text(nodetext, &str);
1042         ok(r == S_OK, "ret %08x\n", r );
1043         ok( !lstrcmpW( str, _bstr_("str1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
1044         SysFreeString(str);
1045
1046         /* NULL pointer means delete */
1047         r = IXMLDOMText_replaceData(nodetext, 0, 1, NULL);
1048         ok(r == S_OK, "ret %08x\n", r );
1049         r = IXMLDOMText_get_text(nodetext, &str);
1050         ok(r == S_OK, "ret %08x\n", r );
1051         ok( !lstrcmpW( str, _bstr_("tr1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
1052         SysFreeString(str);
1053
1054         /* empty string means delete */
1055         r = IXMLDOMText_replaceData(nodetext, 0, 1, _bstr_(""));
1056         ok(r == S_OK, "ret %08x\n", r );
1057         r = IXMLDOMText_get_text(nodetext, &str);
1058         ok(r == S_OK, "ret %08x\n", r );
1059         ok( !lstrcmpW( str, _bstr_("r1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
1060         SysFreeString(str);
1061
1062         /* zero count means insert */
1063         r = IXMLDOMText_replaceData(nodetext, 0, 0, _bstr_("a"));
1064         ok(r == S_OK, "ret %08x\n", r );
1065         r = IXMLDOMText_get_text(nodetext, &str);
1066         ok(r == S_OK, "ret %08x\n", r );
1067         ok( !lstrcmpW( str, _bstr_("ar1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
1068         SysFreeString(str);
1069
1070         r = IXMLDOMText_replaceData(nodetext, 0, 2, NULL);
1071         ok(r == S_OK, "ret %08x\n", r );
1072
1073         r = IXMLDOMText_insertData(nodetext, 0, _bstr_("m"));
1074         ok(r == S_OK, "ret %08x\n", r );
1075         r = IXMLDOMText_get_text(nodetext, &str);
1076         ok(r == S_OK, "ret %08x\n", r );
1077         ok( !lstrcmpW( str, _bstr_("m1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
1078         SysFreeString(str);
1079
1080         /* nonempty string, count greater than its length */
1081         r = IXMLDOMText_replaceData(nodetext, 0, 2, _bstr_("a1.2"));
1082         ok(r == S_OK, "ret %08x\n", r );
1083         r = IXMLDOMText_get_text(nodetext, &str);
1084         ok(r == S_OK, "ret %08x\n", r );
1085         ok( !lstrcmpW( str, _bstr_("a1.2") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
1086         SysFreeString(str);
1087
1088         /* nonempty string, count less than its length */
1089         r = IXMLDOMText_replaceData(nodetext, 0, 1, _bstr_("wine"));
1090         ok(r == S_OK, "ret %08x\n", r );
1091         r = IXMLDOMText_get_text(nodetext, &str);
1092         ok(r == S_OK, "ret %08x\n", r );
1093         ok( !lstrcmpW( str, _bstr_("wine1.2") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
1094         SysFreeString(str);
1095
1096         IXMLDOMText_Release( nodetext );
1097     }
1098
1099     /* test Create Comment */
1100     r = IXMLDOMDocument_createComment(doc, NULL, NULL);
1101     ok( r == E_INVALIDARG, "returns %08x\n", r );
1102     node_comment = (IXMLDOMComment*)0x1;
1103
1104     /* empty comment */
1105     r = IXMLDOMDocument_createComment(doc, _bstr_(""), &node_comment);
1106     ok( r == S_OK, "returns %08x\n", r );
1107     str = (BSTR)0x1;
1108     r = IXMLDOMComment_get_data(node_comment, &str);
1109     ok( r == S_OK, "returns %08x\n", r );
1110     ok( str && SysStringLen(str) == 0, "expected empty string data\n");
1111     IXMLDOMComment_Release(node_comment);
1112     SysFreeString(str);
1113
1114     r = IXMLDOMDocument_createComment(doc, NULL, &node_comment);
1115     ok( r == S_OK, "returns %08x\n", r );
1116     str = (BSTR)0x1;
1117     r = IXMLDOMComment_get_data(node_comment, &str);
1118     ok( r == S_OK, "returns %08x\n", r );
1119     ok( str && (SysStringLen(str) == 0), "expected empty string data\n");
1120     IXMLDOMComment_Release(node_comment);
1121     SysFreeString(str);
1122
1123     str = SysAllocString(szComment);
1124     r = IXMLDOMDocument_createComment(doc, str, &node_comment);
1125     SysFreeString(str);
1126     ok( r == S_OK, "returns %08x\n", r );
1127     if(node_comment)
1128     {
1129         /* Last Child Checks */
1130         r = IXMLDOMComment_get_lastChild(node_comment, NULL);
1131         ok(r == E_INVALIDARG, "ret %08x\n", r );
1132
1133         nodeChild = (IXMLDOMNode*)0x1;
1134         r = IXMLDOMComment_get_lastChild(node_comment, &nodeChild);
1135         ok(r == S_FALSE, "ret %08x\n", r );
1136         ok(nodeChild == NULL, "pLastChild not NULL\n");
1137
1138         /* baseName */
1139         str = (BSTR)0xdeadbeef;
1140         IXMLDOMComment_get_baseName(node_comment, &str);
1141         ok(r == S_FALSE, "ret %08x\n", r );
1142         ok(str == NULL, "Expected NULL\n");
1143
1144         IXMLDOMComment_Release( node_comment );
1145     }
1146
1147     /* test Create Attribute */
1148     str = SysAllocString(szAttribute);
1149     r = IXMLDOMDocument_createAttribute(doc, NULL, NULL);
1150     ok( r == E_INVALIDARG, "returns %08x\n", r );
1151     r = IXMLDOMDocument_createAttribute(doc, str, &node_attr);
1152     ok( r == S_OK, "returns %08x\n", r );
1153     IXMLDOMText_Release( node_attr);
1154     SysFreeString(str);
1155
1156     /* test Processing Instruction */
1157     str = SysAllocStringLen(NULL, 0);
1158     r = IXMLDOMDocument_createProcessingInstruction(doc, str, str, NULL);
1159     ok( r == E_INVALIDARG, "returns %08x\n", r );
1160     r = IXMLDOMDocument_createProcessingInstruction(doc, NULL, str, &nodePI);
1161     ok( r == E_FAIL, "returns %08x\n", r );
1162     r = IXMLDOMDocument_createProcessingInstruction(doc, str, str, &nodePI);
1163     ok( r == E_FAIL, "returns %08x\n", r );
1164     SysFreeString(str);
1165
1166     r = IXMLDOMDocument_createProcessingInstruction(doc, _bstr_("xml"), _bstr_("version=\"1.0\""), &nodePI);
1167     ok( r == S_OK, "returns %08x\n", r );
1168     if(nodePI)
1169     {
1170         /* Last Child Checks */
1171         r = IXMLDOMProcessingInstruction_get_lastChild(nodePI, NULL);
1172         ok(r == E_INVALIDARG, "ret %08x\n", r );
1173
1174         nodeChild = (IXMLDOMNode*)0x1;
1175         r = IXMLDOMProcessingInstruction_get_lastChild(nodePI, &nodeChild);
1176         ok(r == S_FALSE, "ret %08x\n", r );
1177         ok(nodeChild == NULL, "nodeChild not NULL\n");
1178
1179         r = IXMLDOMProcessingInstruction_get_dataType(nodePI, &var);
1180         ok(r == S_FALSE, "ret %08x\n", r );
1181         ok( V_VT(&var) == VT_NULL, "incorrect dataType type\n");
1182         VariantClear(&var);
1183
1184         /* test nodeName */
1185         r = IXMLDOMProcessingInstruction_get_nodeName(nodePI, &str);
1186         ok(r == S_OK, "ret %08x\n", r );
1187         ok( !lstrcmpW( str, _bstr_("xml") ), "incorrect nodeName string\n");
1188         SysFreeString(str);
1189
1190         /* test baseName */
1191         str = (BSTR)0x1;
1192         r = IXMLDOMProcessingInstruction_get_baseName(nodePI, &str);
1193         ok(r == S_OK, "ret %08x\n", r );
1194         ok( !lstrcmpW( str, _bstr_("xml") ), "incorrect nodeName string\n");
1195         SysFreeString(str);
1196
1197         /* test Target */
1198         r = IXMLDOMProcessingInstruction_get_target(nodePI, &str);
1199         ok(r == S_OK, "ret %08x\n", r );
1200         ok( !lstrcmpW( str, _bstr_("xml") ), "incorrect target string\n");
1201         SysFreeString(str);
1202
1203         /* test nodeTypeString */
1204         r = IXMLDOMProcessingInstruction_get_nodeTypeString(nodePI, &str);
1205         ok(r == S_OK, "ret %08x\n", r );
1206         ok( !lstrcmpW( str, _bstr_("processinginstruction") ), "incorrect nodeTypeString string\n");
1207         SysFreeString(str);
1208
1209         /* test get_nodeValue */
1210         r = IXMLDOMProcessingInstruction_get_nodeValue(nodePI, &var);
1211         ok(r == S_OK, "ret %08x\n", r );
1212         ok( !lstrcmpW( V_BSTR(&var), _bstr_("version=\"1.0\"") ), "incorrect data string\n");
1213         VariantClear(&var);
1214
1215         /* test get_data */
1216         r = IXMLDOMProcessingInstruction_get_data(nodePI, &str);
1217         ok(r == S_OK, "ret %08x\n", r );
1218         ok( !lstrcmpW( str, _bstr_("version=\"1.0\"") ), "incorrect data string\n");
1219         SysFreeString(str);
1220
1221         /* test put_data */
1222         r = IXMLDOMProcessingInstruction_put_data(nodePI, _bstr_("version=\"1.0\" encoding=\"UTF-8\""));
1223         ok(r == E_FAIL, "ret %08x\n", r );
1224
1225         /* test put_data */
1226         V_VT(&var) = VT_BSTR;
1227         V_BSTR(&var) = SysAllocString(szOpen);  /* Doesn't matter what the string is, cannot set an xml node. */
1228         r = IXMLDOMProcessingInstruction_put_nodeValue(nodePI, var);
1229         ok(r == E_FAIL, "ret %08x\n", r );
1230         VariantClear(&var);
1231
1232         /* test get nodeName */
1233         r = IXMLDOMProcessingInstruction_get_nodeName(nodePI, &str);
1234         ok( !lstrcmpW( str, _bstr_("xml") ), "incorrect nodeName string\n");
1235         ok(r == S_OK, "ret %08x\n", r );
1236         SysFreeString(str);
1237
1238         IXMLDOMProcessingInstruction_Release(nodePI);
1239     }
1240
1241     r = IXMLDOMDocument_QueryInterface( doc, &IID_ISupportErrorInfo, (void**)&support_error );
1242     ok( r == S_OK, "ret %08x\n", r );
1243     if(r == S_OK)
1244     {
1245         r = ISupportErrorInfo_InterfaceSupportsErrorInfo( support_error, &IID_IXMLDOMDocument );
1246         todo_wine ok( r == S_OK, "ret %08x\n", r );
1247         ISupportErrorInfo_Release( support_error );
1248     }
1249
1250     r = IXMLDOMDocument_Release( doc );
1251     ok( r == 0, "document ref count incorrect\n");
1252
1253     free_bstrs();
1254 }
1255
1256 static void test_persiststreaminit(void)
1257 {
1258     IXMLDOMDocument *doc;
1259     IPersistStreamInit *streaminit;
1260     HRESULT hr;
1261
1262     doc = create_document(&IID_IXMLDOMDocument);
1263     if (!doc) return;
1264
1265     hr = IXMLDOMDocument_QueryInterface(doc, &IID_IPersistStreamInit, (void**)&streaminit);
1266     ok( hr == S_OK, "failed with 0x%08x\n", hr );
1267
1268     hr = IPersistStreamInit_InitNew(streaminit);
1269     ok( hr == S_OK, "failed with 0x%08x\n", hr );
1270
1271     IXMLDOMDocument_Release(doc);
1272 }
1273
1274 static void test_domnode( void )
1275 {
1276     HRESULT r;
1277     IXMLDOMDocument *doc, *owner = NULL;
1278     IXMLDOMElement *element = NULL;
1279     IXMLDOMNamedNodeMap *map = NULL;
1280     IXMLDOMNode *node = NULL, *next = NULL;
1281     IXMLDOMNodeList *list = NULL;
1282     IXMLDOMAttribute *attr = NULL;
1283     DOMNodeType type = NODE_INVALID;
1284     VARIANT_BOOL b;
1285     BSTR str;
1286     VARIANT var;
1287     LONG count;
1288
1289     doc = create_document(&IID_IXMLDOMDocument);
1290     if (!doc) return;
1291
1292     b = FALSE;
1293     str = SysAllocString( szComplete4 );
1294     r = IXMLDOMDocument_loadXML( doc, str, &b );
1295     ok( r == S_OK, "loadXML failed\n");
1296     ok( b == VARIANT_TRUE, "failed to load XML string\n");
1297     SysFreeString( str );
1298
1299     b = 1;
1300     r = IXMLDOMNode_hasChildNodes( doc, &b );
1301     ok( r == S_OK, "hasChildNoes bad return\n");
1302     ok( b == VARIANT_TRUE, "hasChildNoes wrong result\n");
1303
1304     r = IXMLDOMDocument_get_documentElement( doc, &element );
1305     ok( r == S_OK, "should be a document element\n");
1306     ok( element != NULL, "should be an element\n");
1307
1308     VariantInit(&var);
1309     ok( V_VT(&var) == VT_EMPTY, "variant init failed\n");
1310
1311     r = IXMLDOMNode_get_nodeValue( doc, NULL );
1312     ok(r == E_INVALIDARG, "get_nodeValue ret %08x\n", r );
1313
1314     r = IXMLDOMNode_get_nodeValue( doc, &var );
1315     ok( r == S_FALSE, "nextNode returned wrong code\n");
1316     ok( V_VT(&var) == VT_NULL, "variant wasn't empty\n");
1317     ok( V_BSTR(&var) == NULL, "variant value wasn't null\n");
1318
1319     if (element)
1320     {
1321         owner = NULL;
1322         r = IXMLDOMNode_get_ownerDocument( element, &owner );
1323         ok( r == S_OK, "get_ownerDocument return code\n");
1324         ok( owner != doc, "get_ownerDocument return\n");
1325         IXMLDOMDocument_Release(owner);
1326
1327         type = NODE_INVALID;
1328         r = IXMLDOMNode_get_nodeType( element, &type);
1329         ok( r == S_OK, "got %08x\n", r);
1330         ok( type == NODE_ELEMENT, "node not an element\n");
1331
1332         str = NULL;
1333         r = IXMLDOMNode_get_baseName( element, &str );
1334         ok( r == S_OK, "get_baseName returned wrong code\n");
1335         ok( lstrcmpW(str,szlc) == 0, "basename was wrong\n");
1336         SysFreeString(str);
1337
1338         /* check if nodename is correct */
1339         r = IXMLDOMElement_get_nodeName( element, NULL );
1340         ok ( r == E_INVALIDARG, "get_nodeName (NULL) wrong code\n");
1341     
1342         /* content doesn't matter here */
1343         str = NULL;
1344         r = IXMLDOMElement_get_nodeName( element, &str );
1345         ok ( r == S_OK, "get_nodeName wrong code\n");
1346         ok ( str != NULL, "str is null\n");
1347         ok( !lstrcmpW( str, szlc ), "incorrect nodeName\n");
1348         SysFreeString( str );
1349
1350         str = SysAllocString( szNonExistentFile );      
1351         V_VT(&var) = VT_I4;
1352         V_I4(&var) = 0x1234;
1353         r = IXMLDOMElement_getAttribute( element, str, &var );
1354         ok( r == E_FAIL, "getAttribute ret %08x\n", r );
1355         ok( V_VT(&var) == VT_NULL || V_VT(&var) == VT_EMPTY, "vt = %x\n", V_VT(&var));
1356         VariantClear(&var);
1357
1358         r = IXMLDOMElement_getAttributeNode( element, str, NULL);
1359         ok( r == E_FAIL, "getAttributeNode ret %08x\n", r );
1360
1361         attr = (IXMLDOMAttribute*)0xdeadbeef;
1362         r = IXMLDOMElement_getAttributeNode( element, str, &attr);
1363         ok( r == E_FAIL, "getAttributeNode ret %08x\n", r );
1364         ok( attr == NULL, "getAttributeNode ret %p, expected NULL\n", attr );
1365         SysFreeString( str );
1366
1367         attr = (IXMLDOMAttribute*)0xdeadbeef;
1368         str = SysAllocString( szNonExistentAttribute );
1369         r = IXMLDOMElement_getAttributeNode( element, str, &attr);
1370         ok( r == S_FALSE, "getAttributeNode ret %08x\n", r );
1371         ok( attr == NULL, "getAttributeNode ret %p, expected NULL\n", attr );
1372         SysFreeString( str );
1373
1374         str = SysAllocString( szdl );   
1375         V_VT(&var) = VT_I4;
1376         V_I4(&var) = 0x1234;
1377         r = IXMLDOMElement_getAttribute( element, str, &var );
1378         ok( r == S_OK, "getAttribute ret %08x\n", r );
1379         ok( V_VT(&var) == VT_BSTR, "vt = %x\n", V_VT(&var));
1380         ok( !lstrcmpW(V_BSTR(&var), szstr1), "wrong attr value\n");
1381         VariantClear( &var );
1382
1383         r = IXMLDOMElement_getAttribute( element, NULL, &var );
1384         ok( r == E_INVALIDARG, "getAttribute ret %08x\n", r );
1385
1386         r = IXMLDOMElement_getAttribute( element, str, NULL );
1387         ok( r == E_INVALIDARG, "getAttribute ret %08x\n", r );
1388
1389         attr = NULL;
1390         r = IXMLDOMElement_getAttributeNode( element, str, &attr);
1391         ok( r == S_OK, "GetAttributeNode ret %08x\n", r );
1392         ok( attr != NULL, "getAttributeNode returned NULL\n" );
1393         if (attr)
1394         {
1395             r = IXMLDOMAttribute_get_parentNode( attr, NULL );
1396             ok( r == E_INVALIDARG, "Expected E_INVALIDARG, ret %08x\n", r );
1397
1398             /* attribute doesn't have a parent in msxml interpretation */
1399             node = (IXMLDOMNode*)0xdeadbeef;
1400             r = IXMLDOMAttribute_get_parentNode( attr, &node );
1401             ok( r == S_FALSE, "Expected S_FALSE, ret %08x\n", r );
1402             ok( node == NULL, "Expected NULL, got %p\n", node );
1403
1404             IXMLDOMAttribute_Release(attr);
1405         }
1406
1407         SysFreeString( str );
1408
1409         r = IXMLDOMElement_get_attributes( element, &map );
1410         ok( r == S_OK, "get_attributes returned wrong code\n");
1411         ok( map != NULL, "should be attributes\n");
1412
1413         b = 1;
1414         r = IXMLDOMNode_hasChildNodes( element, &b );
1415         ok( r == S_OK, "hasChildNoes bad return\n");
1416         ok( b == VARIANT_TRUE, "hasChildNoes wrong result\n");
1417     }
1418     else
1419         ok( FALSE, "no element\n");
1420
1421     if (map)
1422     {
1423         ISupportErrorInfo *support_error;
1424         r = IXMLDOMNamedNodeMap_QueryInterface( map, &IID_ISupportErrorInfo, (void**)&support_error );
1425         ok( r == S_OK, "ret %08x\n", r );
1426
1427         r = ISupportErrorInfo_InterfaceSupportsErrorInfo( support_error, &IID_IXMLDOMNamedNodeMap );
1428 todo_wine
1429 {
1430         ok( r == S_OK, "ret %08x\n", r );
1431 }
1432         ISupportErrorInfo_Release( support_error );
1433
1434         str = SysAllocString( szdl );
1435         r = IXMLDOMNamedNodeMap_getNamedItem( map, str, &node );
1436         ok( r == S_OK, "getNamedItem returned wrong code\n");
1437         ok( node != NULL, "should be attributes\n");
1438         IXMLDOMNode_Release(node);
1439         SysFreeString( str );
1440
1441         str = SysAllocString( szdl );
1442         r = IXMLDOMNamedNodeMap_getNamedItem( map, str, NULL );
1443         ok( r == E_INVALIDARG, "getNamedItem should return E_INVALIDARG\n");
1444         SysFreeString( str );
1445
1446         /* something that isn't in szComplete4 */
1447         str = SysAllocString( szOpen );
1448         node = (IXMLDOMNode *) 1;
1449         r = IXMLDOMNamedNodeMap_getNamedItem( map, str, &node );
1450         ok( r == S_FALSE, "getNamedItem found a node that wasn't there\n");
1451         ok( node == NULL, "getNamedItem should have returned NULL\n");
1452         SysFreeString( str );
1453
1454         /* test indexed access of attributes */
1455         r = IXMLDOMNamedNodeMap_get_length( map, NULL );
1456         ok ( r == E_INVALIDARG, "get_length should return E_INVALIDARG\n");
1457
1458         r = IXMLDOMNamedNodeMap_get_length( map, &count );
1459         ok ( r == S_OK, "get_length wrong code\n");
1460         ok ( count == 1, "get_length != 1\n");
1461
1462         node = NULL;
1463         r = IXMLDOMNamedNodeMap_get_item( map, -1, &node);
1464         ok ( r == S_FALSE, "get_item (-1) wrong code\n");
1465         ok ( node == NULL, "there is no node\n");
1466
1467         node = NULL;
1468         r = IXMLDOMNamedNodeMap_get_item( map, 1, &node);
1469         ok ( r == S_FALSE, "get_item (1) wrong code\n");
1470         ok ( node == NULL, "there is no attribute\n");
1471
1472         node = NULL;
1473         r = IXMLDOMNamedNodeMap_get_item( map, 0, &node);
1474         ok ( r == S_OK, "get_item (0) wrong code\n");
1475         ok ( node != NULL, "should be attribute\n");
1476
1477         r = IXMLDOMNode_get_nodeName( node, NULL );
1478         ok ( r == E_INVALIDARG, "get_nodeName (NULL) wrong code\n");
1479
1480         /* content doesn't matter here */
1481         str = NULL;
1482         r = IXMLDOMNode_get_nodeName( node, &str );
1483         ok ( r == S_OK, "get_nodeName wrong code\n");
1484         ok ( str != NULL, "str is null\n");
1485         ok( !lstrcmpW( str, szdl ), "incorrect node name\n");
1486         SysFreeString( str );
1487
1488         /* test sequential access of attributes */
1489         node = NULL;
1490         r = IXMLDOMNamedNodeMap_nextNode( map, &node );
1491         ok ( r == S_OK, "nextNode (first time) wrong code\n");
1492         ok ( node != NULL, "nextNode, should be attribute\n");
1493
1494         r = IXMLDOMNamedNodeMap_nextNode( map, &node );
1495         ok ( r != S_OK, "nextNode (second time) wrong code\n");
1496         ok ( node == NULL, "nextNode, there is no attribute\n");
1497
1498         r = IXMLDOMNamedNodeMap_reset( map );
1499         ok ( r == S_OK, "reset should return S_OK\n");
1500
1501         r = IXMLDOMNamedNodeMap_nextNode( map, &node );
1502         ok ( r == S_OK, "nextNode (third time) wrong code\n");
1503         ok ( node != NULL, "nextNode, should be attribute\n");
1504     }
1505     else
1506         ok( FALSE, "no map\n");
1507
1508     if (node)
1509     {
1510         type = NODE_INVALID;
1511         r = IXMLDOMNode_get_nodeType( node, &type);
1512         ok( r == S_OK, "getNamedItem returned wrong code\n");
1513         ok( type == NODE_ATTRIBUTE, "node not an attribute\n");
1514
1515         str = NULL;
1516         r = IXMLDOMNode_get_baseName( node, NULL );
1517         ok( r == E_INVALIDARG, "get_baseName returned wrong code\n");
1518
1519         str = NULL;
1520         r = IXMLDOMNode_get_baseName( node, &str );
1521         ok( r == S_OK, "get_baseName returned wrong code\n");
1522         ok( lstrcmpW(str,szdl) == 0, "basename was wrong\n");
1523         SysFreeString( str );
1524
1525         r = IXMLDOMNode_get_nodeValue( node, &var );
1526         ok( r == S_OK, "returns %08x\n", r );
1527         ok( V_VT(&var) == VT_BSTR, "vt %x\n", V_VT(&var));
1528         ok( !lstrcmpW(V_BSTR(&var), szstr1), "nodeValue incorrect\n");
1529         VariantClear(&var);
1530
1531         r = IXMLDOMNode_get_childNodes( node, NULL );
1532         ok( r == E_INVALIDARG, "get_childNodes returned wrong code\n");
1533
1534         r = IXMLDOMNode_get_childNodes( node, &list );
1535         ok( r == S_OK, "get_childNodes returned wrong code\n");
1536
1537         if (list)
1538         {
1539             r = IXMLDOMNodeList_nextNode( list, &next );
1540             ok( r == S_OK, "nextNode returned wrong code\n");
1541         }
1542         else
1543             ok( FALSE, "no childlist\n");
1544
1545         if (next)
1546         {
1547             b = 1;
1548             r = IXMLDOMNode_hasChildNodes( next, &b );
1549             ok( r == S_FALSE, "hasChildNoes bad return\n");
1550             ok( b == VARIANT_FALSE, "hasChildNoes wrong result\n");
1551
1552             type = NODE_INVALID;
1553             r = IXMLDOMNode_get_nodeType( next, &type);
1554             ok( r == S_OK, "getNamedItem returned wrong code\n");
1555             ok( type == NODE_TEXT, "node not text\n");
1556
1557             str = (BSTR) 1;
1558             r = IXMLDOMNode_get_baseName( next, &str );
1559             ok( r == S_FALSE, "get_baseName returned wrong code\n");
1560             ok( str == NULL, "basename was wrong\n");
1561             SysFreeString(str);
1562         }
1563         else
1564             ok( FALSE, "no next\n");
1565
1566         if (next)
1567             IXMLDOMNode_Release( next );
1568         next = NULL;
1569         if (list)
1570             IXMLDOMNodeList_Release( list );
1571         list = NULL;
1572         if (node)
1573             IXMLDOMNode_Release( node );
1574     }
1575     else
1576         ok( FALSE, "no node\n");
1577     node = NULL;
1578
1579     if (map)
1580         IXMLDOMNamedNodeMap_Release( map );
1581
1582     /* now traverse the tree from the root element */
1583     if (element)
1584     {
1585         r = IXMLDOMNode_get_childNodes( element, &list );
1586         ok( r == S_OK, "get_childNodes returned wrong code\n");
1587
1588         /* using get_item for child list doesn't advance the position */
1589         ole_check(IXMLDOMNodeList_get_item(list, 1, &node));
1590         expect_node(node, "E2.E2.D1");
1591         IXMLDOMNode_Release(node);
1592         ole_check(IXMLDOMNodeList_nextNode(list, &node));
1593         expect_node(node, "E1.E2.D1");
1594         IXMLDOMNode_Release(node);
1595         ole_check(IXMLDOMNodeList_reset(list));
1596
1597         IXMLDOMNodeList_AddRef(list);
1598         expect_list_and_release(list, "E1.E2.D1 E2.E2.D1 E3.E2.D1 E4.E2.D1");
1599         ole_check(IXMLDOMNodeList_reset(list));
1600
1601         node = (void*)0xdeadbeef;
1602         str = SysAllocString(szdl);
1603         r = IXMLDOMNode_selectSingleNode( element, str, &node );
1604         SysFreeString(str);
1605         ok( r == S_FALSE, "ret %08x\n", r );
1606         ok( node == NULL, "node %p\n", node );
1607
1608         str = SysAllocString(szbs);
1609         r = IXMLDOMNode_selectSingleNode( element, str, &node );
1610         SysFreeString(str);
1611         ok( r == S_OK, "ret %08x\n", r );
1612         r = IXMLDOMNode_Release( node );
1613         ok( r == 0, "ret %08x\n", r );
1614     }
1615     else
1616         ok( FALSE, "no element\n");
1617
1618     if (list)
1619     {
1620         r = IXMLDOMNodeList_QueryInterface(list, &IID_IDispatch, NULL);
1621         ok( r == E_INVALIDARG || r == E_POINTER, "ret %08x\n", r );
1622
1623         r = IXMLDOMNodeList_get_item(list, 0, NULL);
1624         ok(r == E_INVALIDARG, "Exected E_INVALIDARG got %08x\n", r);
1625
1626         r = IXMLDOMNodeList_get_length(list, NULL);
1627         ok(r == E_INVALIDARG, "Exected E_INVALIDARG got %08x\n", r);
1628
1629         r = IXMLDOMNodeList_get_length( list, &count );
1630         ok( r == S_OK, "get_length returns %08x\n", r );
1631         ok( count == 4, "get_length got %d\n", count );
1632
1633         r = IXMLDOMNodeList_nextNode(list, NULL);
1634         ok(r == E_INVALIDARG, "Exected E_INVALIDARG got %08x\n", r);
1635
1636         r = IXMLDOMNodeList_nextNode( list, &node );
1637         ok( r == S_OK, "nextNode returned wrong code\n");
1638     }
1639     else
1640         ok( FALSE, "no list\n");
1641
1642     if (node)
1643     {
1644         type = NODE_INVALID;
1645         r = IXMLDOMNode_get_nodeType( node, &type);
1646         ok( r == S_OK, "getNamedItem returned wrong code\n");
1647         ok( type == NODE_ELEMENT, "node not text\n");
1648
1649         VariantInit(&var);
1650         ok( V_VT(&var) == VT_EMPTY, "variant init failed\n");
1651         r = IXMLDOMNode_get_nodeValue( node, &var );
1652         ok( r == S_FALSE, "nextNode returned wrong code\n");
1653         ok( V_VT(&var) == VT_NULL, "variant wasn't empty\n");
1654         ok( V_BSTR(&var) == NULL, "variant value wasn't null\n");
1655
1656         r = IXMLDOMNode_hasChildNodes( node, NULL );
1657         ok( r == E_INVALIDARG, "hasChildNoes bad return\n");
1658
1659         b = 1;
1660         r = IXMLDOMNode_hasChildNodes( node, &b );
1661         ok( r == S_OK, "hasChildNoes bad return\n");
1662         ok( b == VARIANT_TRUE, "hasChildNoes wrong result\n");
1663
1664         str = NULL;
1665         r = IXMLDOMNode_get_baseName( node, &str );
1666         ok( r == S_OK, "get_baseName returned wrong code\n");
1667         ok( lstrcmpW(str,szbs) == 0, "basename was wrong\n");
1668         SysFreeString(str);
1669     }
1670     else
1671         ok( FALSE, "no node\n");
1672
1673     if (node)
1674         IXMLDOMNode_Release( node );
1675     if (list)
1676         IXMLDOMNodeList_Release( list );
1677     if (element)
1678         IXMLDOMElement_Release( element );
1679
1680     b = FALSE;
1681     str = SysAllocString( szComplete5 );
1682     r = IXMLDOMDocument_loadXML( doc, str, &b );
1683     ok( r == S_OK, "loadXML failed\n");
1684     ok( b == VARIANT_TRUE, "failed to load XML string\n");
1685     SysFreeString( str );
1686
1687     b = 1;
1688     r = IXMLDOMNode_hasChildNodes( doc, &b );
1689     ok( r == S_OK, "hasChildNoes bad return\n");
1690     ok( b == VARIANT_TRUE, "hasChildNoes wrong result\n");
1691
1692     r = IXMLDOMDocument_get_documentElement( doc, &element );
1693     ok( r == S_OK, "should be a document element\n");
1694     ok( element != NULL, "should be an element\n");
1695
1696     if (element)
1697     {
1698         static const WCHAR szSSearch[] = {'S',':','s','e','a','r','c','h',0};
1699         BSTR tag = NULL;
1700
1701         /* check if the tag is correct */
1702         r = IXMLDOMElement_get_tagName( element, &tag );
1703         ok( r == S_OK, "couldn't get tag name\n");
1704         ok( tag != NULL, "tag was null\n");
1705         ok( !lstrcmpW( tag, szSSearch ), "incorrect tag name\n");
1706         SysFreeString( tag );
1707     }
1708
1709     if (element)
1710         IXMLDOMElement_Release( element );
1711     ok(IXMLDOMDocument_Release( doc ) == 0, "document is not destroyed\n");
1712 }
1713
1714 static void test_refs(void)
1715 {
1716     HRESULT r;
1717     BSTR str;
1718     VARIANT_BOOL b;
1719     IXMLDOMDocument *doc;
1720     IXMLDOMElement *element = NULL;
1721     IXMLDOMNode *node = NULL, *node2;
1722     IXMLDOMNodeList *node_list = NULL;
1723     LONG ref;
1724     IUnknown *unk, *unk2;
1725
1726     doc = create_document(&IID_IXMLDOMDocument);
1727     if (!doc) return;
1728
1729     ref = IXMLDOMDocument_Release(doc);
1730     ok( ref == 0, "ref %d\n", ref);
1731
1732     doc = create_document(&IID_IXMLDOMDocument);
1733     if (!doc) return;
1734
1735     str = SysAllocString( szComplete4 );
1736     r = IXMLDOMDocument_loadXML( doc, str, &b );
1737     ok( r == S_OK, "loadXML failed\n");
1738     ok( b == VARIANT_TRUE, "failed to load XML string\n");
1739     SysFreeString( str );
1740
1741     ref = IXMLDOMDocument_AddRef( doc );
1742     ok( ref == 2, "ref %d\n", ref );
1743     ref = IXMLDOMDocument_AddRef( doc );
1744     ok( ref == 3, "ref %d\n", ref );
1745     IXMLDOMDocument_Release( doc );
1746     IXMLDOMDocument_Release( doc );
1747
1748     r = IXMLDOMDocument_get_documentElement( doc, &element );
1749     ok( r == S_OK, "should be a document element\n");
1750     ok( element != NULL, "should be an element\n");
1751
1752     ref = IXMLDOMDocument_AddRef( doc );
1753     ok( ref == 2, "ref %d\n", ref );
1754     IXMLDOMDocument_Release( doc );
1755
1756     r = IXMLDOMElement_get_childNodes( element, &node_list );
1757     ok( r == S_OK, "rets %08x\n", r);
1758
1759     ref = IXMLDOMNodeList_AddRef( node_list );
1760     ok( ref == 2, "ref %d\n", ref );
1761     IXMLDOMNodeList_Release( node_list );
1762
1763     IXMLDOMNodeList_get_item( node_list, 0, &node );
1764     ok( r == S_OK, "rets %08x\n", r);
1765
1766     IXMLDOMNodeList_get_item( node_list, 0, &node2 );
1767     ok( r == S_OK, "rets %08x\n", r);
1768
1769     ref = IXMLDOMNode_AddRef( node );
1770     ok( ref == 2, "ref %d\n", ref );
1771     IXMLDOMNode_Release( node );
1772
1773     ref = IXMLDOMNode_Release( node );
1774     ok( ref == 0, "ref %d\n", ref );
1775     ref = IXMLDOMNode_Release( node2 );
1776     ok( ref == 0, "ref %d\n", ref );
1777
1778     ref = IXMLDOMNodeList_Release( node_list );
1779     ok( ref == 0, "ref %d\n", ref );
1780
1781     ok( node != node2, "node %p node2 %p\n", node, node2 );
1782
1783     ref = IXMLDOMDocument_Release( doc );
1784     ok( ref == 0, "ref %d\n", ref );
1785
1786     ref = IXMLDOMElement_AddRef( element );
1787     todo_wine {
1788     ok( ref == 3, "ref %d\n", ref );
1789     }
1790     IXMLDOMElement_Release( element );
1791
1792     /* IUnknown must be unique however we obtain it */
1793     r = IXMLDOMElement_QueryInterface( element, &IID_IUnknown, (void**)&unk );
1794     ok( r == S_OK, "rets %08x\n", r );
1795     r = IXMLDOMElement_QueryInterface( element, &IID_IXMLDOMNode, (void**)&node );
1796     ok( r == S_OK, "rets %08x\n", r );
1797     r = IXMLDOMNode_QueryInterface( node, &IID_IUnknown, (void**)&unk2 );
1798     ok( r == S_OK, "rets %08x\n", r );
1799     ok( unk == unk2, "unk %p unk2 %p\n", unk, unk2 );
1800
1801     IUnknown_Release( unk2 );
1802     IUnknown_Release( unk );
1803     IXMLDOMNode_Release( node );
1804
1805     IXMLDOMElement_Release( element );
1806
1807 }
1808
1809 static void test_create(void)
1810 {
1811     static const WCHAR szOne[] = {'1',0};
1812     static const WCHAR szOneGarbage[] = {'1','G','a','r','b','a','g','e',0};
1813     HRESULT r;
1814     VARIANT var;
1815     BSTR str, name;
1816     IXMLDOMDocument *doc;
1817     IXMLDOMElement *element;
1818     IXMLDOMComment *comment;
1819     IXMLDOMText *text;
1820     IXMLDOMCDATASection *cdata;
1821     IXMLDOMNode *root, *node, *child;
1822     IXMLDOMNamedNodeMap *attr_map;
1823     IUnknown *unk;
1824     LONG ref;
1825     LONG num;
1826
1827     doc = create_document(&IID_IXMLDOMDocument);
1828     if (!doc) return;
1829
1830     /* types not supported for creation */
1831     V_VT(&var) = VT_I1;
1832     V_I1(&var) = NODE_DOCUMENT;
1833     node = (IXMLDOMNode*)0x1;
1834     r = IXMLDOMDocument_createNode( doc, var, NULL, NULL, &node );
1835     ok( r == E_INVALIDARG, "returns %08x\n", r );
1836     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
1837
1838     V_VT(&var) = VT_I1;
1839     V_I1(&var) = NODE_DOCUMENT_TYPE;
1840     node = (IXMLDOMNode*)0x1;
1841     r = IXMLDOMDocument_createNode( doc, var, NULL, NULL, &node );
1842     ok( r == E_INVALIDARG, "returns %08x\n", r );
1843     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
1844
1845     V_VT(&var) = VT_I1;
1846     V_I1(&var) = NODE_ENTITY;
1847     node = (IXMLDOMNode*)0x1;
1848     r = IXMLDOMDocument_createNode( doc, var, NULL, NULL, &node );
1849     ok( r == E_INVALIDARG, "returns %08x\n", r );
1850     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
1851
1852     V_VT(&var) = VT_I1;
1853     V_I1(&var) = NODE_NOTATION;
1854     node = (IXMLDOMNode*)0x1;
1855     r = IXMLDOMDocument_createNode( doc, var, NULL, NULL, &node );
1856     ok( r == E_INVALIDARG, "returns %08x\n", r );
1857     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
1858
1859     /* NODE_COMMENT */
1860     V_VT(&var) = VT_I1;
1861     V_I1(&var) = NODE_COMMENT;
1862     node = NULL;
1863     r = IXMLDOMDocument_createNode( doc, var, NULL, NULL, &node );
1864     ok( r == S_OK, "returns %08x\n", r );
1865     ok( node != NULL, "\n");
1866
1867     r = IXMLDOMNode_QueryInterface(node, &IID_IXMLDOMComment, (void**)&comment);
1868     ok( r == S_OK, "returns %08x\n", r );
1869     IXMLDOMNode_Release(node);
1870
1871     str = NULL;
1872     r = IXMLDOMComment_get_data(comment, &str);
1873     ok( r == S_OK, "returns %08x\n", r );
1874     ok( str && SysStringLen(str) == 0, "expected empty comment, %p\n", str);
1875     IXMLDOMComment_Release(comment);
1876     SysFreeString(str);
1877
1878     node = (IXMLDOMNode*)0x1;
1879     r = IXMLDOMDocument_createNode( doc, var, _bstr_(""), NULL, &node );
1880     ok( r == S_OK, "returns %08x\n", r );
1881
1882     r = IXMLDOMNode_QueryInterface(node, &IID_IXMLDOMComment, (void**)&comment);
1883     ok( r == S_OK, "returns %08x\n", r );
1884     IXMLDOMNode_Release(node);
1885
1886     str = NULL;
1887     r = IXMLDOMComment_get_data(comment, &str);
1888     ok( r == S_OK, "returns %08x\n", r );
1889     ok( str && SysStringLen(str) == 0, "expected empty comment, %p\n", str);
1890     IXMLDOMComment_Release(comment);
1891     SysFreeString(str);
1892
1893     node = (IXMLDOMNode*)0x1;
1894     r = IXMLDOMDocument_createNode( doc, var, _bstr_("blah"), NULL, &node );
1895     ok( r == S_OK, "returns %08x\n", r );
1896
1897     r = IXMLDOMNode_QueryInterface(node, &IID_IXMLDOMComment, (void**)&comment);
1898     ok( r == S_OK, "returns %08x\n", r );
1899     IXMLDOMNode_Release(node);
1900
1901     str = NULL;
1902     r = IXMLDOMComment_get_data(comment, &str);
1903     ok( r == S_OK, "returns %08x\n", r );
1904     ok( str && SysStringLen(str) == 0, "expected empty comment, %p\n", str);
1905     IXMLDOMComment_Release(comment);
1906     SysFreeString(str);
1907
1908     /* NODE_TEXT */
1909     V_VT(&var) = VT_I1;
1910     V_I1(&var) = NODE_TEXT;
1911     node = NULL;
1912     r = IXMLDOMDocument_createNode( doc, var, NULL, NULL, &node );
1913     ok( r == S_OK, "returns %08x\n", r );
1914     ok( node != NULL, "\n");
1915
1916     r = IXMLDOMNode_QueryInterface(node, &IID_IXMLDOMText, (void**)&text);
1917     ok( r == S_OK, "returns %08x\n", r );
1918     IXMLDOMNode_Release(node);
1919
1920     str = NULL;
1921     r = IXMLDOMText_get_data(text, &str);
1922     ok( r == S_OK, "returns %08x\n", r );
1923     ok( str && SysStringLen(str) == 0, "expected empty comment, %p\n", str);
1924     IXMLDOMText_Release(text);
1925     SysFreeString(str);
1926
1927     node = (IXMLDOMNode*)0x1;
1928     r = IXMLDOMDocument_createNode( doc, var, _bstr_(""), NULL, &node );
1929     ok( r == S_OK, "returns %08x\n", r );
1930
1931     r = IXMLDOMNode_QueryInterface(node, &IID_IXMLDOMText, (void**)&text);
1932     ok( r == S_OK, "returns %08x\n", r );
1933     IXMLDOMNode_Release(node);
1934
1935     str = NULL;
1936     r = IXMLDOMText_get_data(text, &str);
1937     ok( r == S_OK, "returns %08x\n", r );
1938     ok( str && SysStringLen(str) == 0, "expected empty comment, %p\n", str);
1939     IXMLDOMText_Release(text);
1940     SysFreeString(str);
1941
1942     node = (IXMLDOMNode*)0x1;
1943     r = IXMLDOMDocument_createNode( doc, var, _bstr_("blah"), NULL, &node );
1944     ok( r == S_OK, "returns %08x\n", r );
1945
1946     r = IXMLDOMNode_QueryInterface(node, &IID_IXMLDOMText, (void**)&text);
1947     ok( r == S_OK, "returns %08x\n", r );
1948     IXMLDOMNode_Release(node);
1949
1950     str = NULL;
1951     r = IXMLDOMText_get_data(text, &str);
1952     ok( r == S_OK, "returns %08x\n", r );
1953     ok( str && SysStringLen(str) == 0, "expected empty comment, %p\n", str);
1954     IXMLDOMText_Release(text);
1955     SysFreeString(str);
1956
1957     /* NODE_CDATA_SECTION */
1958     V_VT(&var) = VT_I1;
1959     V_I1(&var) = NODE_CDATA_SECTION;
1960     node = NULL;
1961     r = IXMLDOMDocument_createNode( doc, var, NULL, NULL, &node );
1962     ok( r == S_OK, "returns %08x\n", r );
1963     ok( node != NULL, "\n");
1964
1965     r = IXMLDOMNode_QueryInterface(node, &IID_IXMLDOMCDATASection, (void**)&cdata);
1966     ok( r == S_OK, "returns %08x\n", r );
1967     IXMLDOMNode_Release(node);
1968
1969     str = NULL;
1970     r = IXMLDOMCDATASection_get_data(cdata, &str);
1971     ok( r == S_OK, "returns %08x\n", r );
1972     ok( str && SysStringLen(str) == 0, "expected empty comment, %p\n", str);
1973     IXMLDOMCDATASection_Release(cdata);
1974     SysFreeString(str);
1975
1976     node = (IXMLDOMNode*)0x1;
1977     r = IXMLDOMDocument_createNode( doc, var, _bstr_(""), NULL, &node );
1978     ok( r == S_OK, "returns %08x\n", r );
1979
1980     r = IXMLDOMNode_QueryInterface(node, &IID_IXMLDOMCDATASection, (void**)&cdata);
1981     ok( r == S_OK, "returns %08x\n", r );
1982     IXMLDOMNode_Release(node);
1983
1984     str = NULL;
1985     r = IXMLDOMCDATASection_get_data(cdata, &str);
1986     ok( r == S_OK, "returns %08x\n", r );
1987     ok( str && SysStringLen(str) == 0, "expected empty comment, %p\n", str);
1988     IXMLDOMCDATASection_Release(cdata);
1989     SysFreeString(str);
1990
1991     node = (IXMLDOMNode*)0x1;
1992     r = IXMLDOMDocument_createNode( doc, var, _bstr_("blah"), NULL, &node );
1993     ok( r == S_OK, "returns %08x\n", r );
1994
1995     r = IXMLDOMNode_QueryInterface(node, &IID_IXMLDOMCDATASection, (void**)&cdata);
1996     ok( r == S_OK, "returns %08x\n", r );
1997     IXMLDOMNode_Release(node);
1998
1999     str = NULL;
2000     r = IXMLDOMCDATASection_get_data(cdata, &str);
2001     ok( r == S_OK, "returns %08x\n", r );
2002     ok( str && SysStringLen(str) == 0, "expected empty comment, %p\n", str);
2003     IXMLDOMCDATASection_Release(cdata);
2004     SysFreeString(str);
2005
2006     /* NODE_ATTRIBUTE */
2007     V_VT(&var) = VT_I1;
2008     V_I1(&var) = NODE_ATTRIBUTE;
2009     node = (IXMLDOMNode*)0x1;
2010     r = IXMLDOMDocument_createNode( doc, var, NULL, NULL, &node );
2011     ok( r == E_FAIL, "returns %08x\n", r );
2012     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
2013
2014     V_VT(&var) = VT_I1;
2015     V_I1(&var) = NODE_ATTRIBUTE;
2016     node = (IXMLDOMNode*)0x1;
2017     r = IXMLDOMDocument_createNode( doc, var, _bstr_(""), NULL, &node );
2018     ok( r == E_FAIL, "returns %08x\n", r );
2019     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
2020
2021     V_VT(&var) = VT_I1;
2022     V_I1(&var) = NODE_ATTRIBUTE;
2023     str = SysAllocString( szlc );
2024     r = IXMLDOMDocument_createNode( doc, var, str, NULL, &node );
2025     ok( r == S_OK, "returns %08x\n", r );
2026     if( SUCCEEDED(r) ) IXMLDOMNode_Release( node );
2027     SysFreeString(str);
2028
2029     /* a name is required for attribute, try a BSTR with first null wchar */
2030     V_VT(&var) = VT_I1;
2031     V_I1(&var) = NODE_ATTRIBUTE;
2032     str = SysAllocString( szstr1 );
2033     str[0] = 0;
2034     node = (IXMLDOMNode*)0x1;
2035     r = IXMLDOMDocument_createNode( doc, var, str, NULL, &node );
2036     ok( r == E_FAIL, "returns %08x\n", r );
2037     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
2038     SysFreeString(str);
2039
2040     /* NODE_PROCESSING_INSTRUCTION */
2041     V_VT(&var) = VT_I1;
2042     V_I1(&var) = NODE_PROCESSING_INSTRUCTION;
2043     node = (IXMLDOMNode*)0x1;
2044     r = IXMLDOMDocument_createNode( doc, var, NULL, NULL, &node );
2045     ok( r == E_FAIL, "returns %08x\n", r );
2046     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
2047
2048     V_VT(&var) = VT_I1;
2049     V_I1(&var) = NODE_PROCESSING_INSTRUCTION;
2050     node = (IXMLDOMNode*)0x1;
2051     r = IXMLDOMDocument_createNode( doc, var, _bstr_(""), NULL, &node );
2052     ok( r == E_FAIL, "returns %08x\n", r );
2053     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
2054
2055     V_VT(&var) = VT_I1;
2056     V_I1(&var) = NODE_PROCESSING_INSTRUCTION;
2057     r = IXMLDOMDocument_createNode( doc, var, _bstr_("pi"), NULL, NULL );
2058     ok( r == E_INVALIDARG, "returns %08x\n", r );
2059
2060     /* NODE_ENTITY_REFERENCE */
2061     V_VT(&var) = VT_I1;
2062     V_I1(&var) = NODE_ENTITY_REFERENCE;
2063     node = (IXMLDOMNode*)0x1;
2064     r = IXMLDOMDocument_createNode( doc, var, NULL, NULL, &node );
2065     ok( r == E_FAIL, "returns %08x\n", r );
2066     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
2067
2068     V_VT(&var) = VT_I1;
2069     V_I1(&var) = NODE_ENTITY_REFERENCE;
2070     node = (IXMLDOMNode*)0x1;
2071     r = IXMLDOMDocument_createNode( doc, var, _bstr_(""), NULL, &node );
2072     ok( r == E_FAIL, "returns %08x\n", r );
2073     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
2074
2075     /* NODE_ELEMENT */
2076     V_VT(&var) = VT_I1;
2077     V_I1(&var) = NODE_ELEMENT;
2078     node = (IXMLDOMNode*)0x1;
2079     r = IXMLDOMDocument_createNode( doc, var, NULL, NULL, &node );
2080     ok( r == E_FAIL, "returns %08x\n", r );
2081     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
2082
2083     V_VT(&var) = VT_I1;
2084     V_I1(&var) = NODE_ELEMENT;
2085     node = (IXMLDOMNode*)0x1;
2086     r = IXMLDOMDocument_createNode( doc, var, _bstr_(""), NULL, &node );
2087     ok( r == E_FAIL, "returns %08x\n", r );
2088     ok( node == (void*)0x1, "expected same ptr, got %p\n", node);
2089
2090     V_VT(&var) = VT_I1;
2091     V_I1(&var) = NODE_ELEMENT;
2092     str = SysAllocString( szlc );
2093     r = IXMLDOMDocument_createNode( doc, var, str, NULL, &node );
2094     ok( r == S_OK, "returns %08x\n", r );
2095     if( SUCCEEDED(r) ) IXMLDOMNode_Release( node );
2096
2097     V_VT(&var) = VT_I1;
2098     V_I1(&var) = NODE_ELEMENT;
2099     r = IXMLDOMDocument_createNode( doc, var, str, NULL, NULL );
2100     ok( r == E_INVALIDARG, "returns %08x\n", r );
2101
2102     V_VT(&var) = VT_R4;
2103     V_R4(&var) = NODE_ELEMENT;
2104     r = IXMLDOMDocument_createNode( doc, var, str, NULL, &node );
2105     ok( r == S_OK, "returns %08x\n", r );
2106     if( SUCCEEDED(r) ) IXMLDOMNode_Release( node );
2107
2108     V_VT(&var) = VT_BSTR;
2109     V_BSTR(&var) = SysAllocString( szOne );
2110     r = IXMLDOMDocument_createNode( doc, var, str, NULL, &node );
2111     ok( r == S_OK, "returns %08x\n", r );
2112     if( SUCCEEDED(r) ) IXMLDOMNode_Release( node );
2113     VariantClear(&var);
2114
2115     V_VT(&var) = VT_BSTR;
2116     V_BSTR(&var) = SysAllocString( szOneGarbage );
2117     r = IXMLDOMDocument_createNode( doc, var, str, NULL, &node );
2118     ok( r == E_INVALIDARG, "returns %08x\n", r );
2119     if( SUCCEEDED(r) ) IXMLDOMNode_Release( node );
2120     VariantClear(&var);
2121
2122     V_VT(&var) = VT_I4;
2123     V_I4(&var) = NODE_ELEMENT;
2124     r = IXMLDOMDocument_createNode( doc, var, str, NULL, &node );
2125     ok( r == S_OK, "returns %08x\n", r );
2126     r = IXMLDOMDocument_appendChild( doc, node, &root );
2127     ok( r == S_OK, "returns %08x\n", r );
2128     ok( node == root, "%p %p\n", node, root );
2129
2130     ref = IXMLDOMNode_AddRef( node );
2131     ok(ref == 3, "ref %d\n", ref);
2132     IXMLDOMNode_Release( node );
2133
2134     ref = IXMLDOMNode_Release( node );
2135     ok(ref == 1, "ref %d\n", ref);
2136     SysFreeString( str );
2137
2138     V_VT(&var) = VT_I4;
2139     V_I4(&var) = NODE_ELEMENT;
2140     str = SysAllocString( szbs );
2141     r = IXMLDOMDocument_createNode( doc, var, str, NULL, &node );
2142     ok( r == S_OK, "returns %08x\n", r );
2143     SysFreeString( str );
2144
2145     ref = IXMLDOMNode_AddRef( node );
2146     ok(ref == 2, "ref = %d\n", ref);
2147     IXMLDOMNode_Release( node );
2148
2149     r = IXMLDOMNode_QueryInterface( node, &IID_IUnknown, (void**)&unk );
2150     ok( r == S_OK, "returns %08x\n", r );
2151
2152     ref = IXMLDOMNode_AddRef( unk );
2153     ok(ref == 3, "ref = %d\n", ref);
2154     IXMLDOMNode_Release( unk );
2155
2156     V_VT(&var) = VT_EMPTY;
2157     r = IXMLDOMNode_insertBefore( root, (IXMLDOMNode*)unk, var, &child );
2158     ok( r == S_OK, "returns %08x\n", r );
2159     ok( unk == (IUnknown*)child, "%p %p\n", unk, child );
2160     IXMLDOMNode_Release( child );
2161     IUnknown_Release( unk );
2162
2163
2164     V_VT(&var) = VT_NULL;
2165     V_DISPATCH(&var) = (IDispatch*)node;
2166     r = IXMLDOMNode_insertBefore( root, node, var, &child );
2167     ok( r == S_OK, "returns %08x\n", r );
2168     ok( node == child, "%p %p\n", node, child );
2169     IXMLDOMNode_Release( child );
2170
2171
2172     V_VT(&var) = VT_NULL;
2173     V_DISPATCH(&var) = (IDispatch*)node;
2174     r = IXMLDOMNode_insertBefore( root, node, var, NULL );
2175     ok( r == S_OK, "returns %08x\n", r );
2176     IXMLDOMNode_Release( node );
2177
2178     r = IXMLDOMNode_QueryInterface( root, &IID_IXMLDOMElement, (void**)&element );
2179     ok( r == S_OK, "returns %08x\n", r );
2180
2181     r = IXMLDOMElement_get_attributes( element, &attr_map );
2182     ok( r == S_OK, "returns %08x\n", r );
2183     r = IXMLDOMNamedNodeMap_get_length( attr_map, &num );
2184     ok( r == S_OK, "returns %08x\n", r );
2185     ok( num == 0, "num %d\n", num );
2186     IXMLDOMNamedNodeMap_Release( attr_map );
2187
2188     V_VT(&var) = VT_BSTR;
2189     V_BSTR(&var) = SysAllocString( szstr1 );
2190     name = SysAllocString( szdl );
2191     r = IXMLDOMElement_setAttribute( element, name, var );
2192     ok( r == S_OK, "returns %08x\n", r );
2193     r = IXMLDOMElement_get_attributes( element, &attr_map );
2194     ok( r == S_OK, "returns %08x\n", r );
2195     r = IXMLDOMNamedNodeMap_get_length( attr_map, &num );
2196     ok( r == S_OK, "returns %08x\n", r );
2197     ok( num == 1, "num %d\n", num );
2198     IXMLDOMNamedNodeMap_Release( attr_map );
2199     VariantClear(&var);
2200
2201     V_VT(&var) = VT_BSTR;
2202     V_BSTR(&var) = SysAllocString( szstr2 );
2203     r = IXMLDOMElement_setAttribute( element, name, var );
2204     ok( r == S_OK, "returns %08x\n", r );
2205     r = IXMLDOMElement_get_attributes( element, &attr_map );
2206     ok( r == S_OK, "returns %08x\n", r );
2207     r = IXMLDOMNamedNodeMap_get_length( attr_map, &num );
2208     ok( r == S_OK, "returns %08x\n", r );
2209     ok( num == 1, "num %d\n", num );
2210     IXMLDOMNamedNodeMap_Release( attr_map );
2211     VariantClear(&var);
2212     r = IXMLDOMElement_getAttribute( element, name, &var );
2213     ok( r == S_OK, "returns %08x\n", r );
2214     ok( !lstrcmpW(V_BSTR(&var), szstr2), "wrong attr value\n");
2215     VariantClear(&var);
2216     SysFreeString(name);
2217
2218     V_VT(&var) = VT_BSTR;
2219     V_BSTR(&var) = SysAllocString( szstr1 );
2220     name = SysAllocString( szlc );
2221     r = IXMLDOMElement_setAttribute( element, name, var );
2222     ok( r == S_OK, "returns %08x\n", r );
2223     r = IXMLDOMElement_get_attributes( element, &attr_map );
2224     ok( r == S_OK, "returns %08x\n", r );
2225     r = IXMLDOMNamedNodeMap_get_length( attr_map, &num );
2226     ok( r == S_OK, "returns %08x\n", r );
2227     ok( num == 2, "num %d\n", num );
2228     IXMLDOMNamedNodeMap_Release( attr_map );
2229     VariantClear(&var);
2230     SysFreeString(name);
2231
2232     V_VT(&var) = VT_I4;
2233     V_I4(&var) = 10;
2234     name = SysAllocString( szbs );
2235     r = IXMLDOMElement_setAttribute( element, name, var );
2236     ok( r == S_OK, "returns %08x\n", r );
2237     VariantClear(&var);
2238     r = IXMLDOMElement_getAttribute( element, name, &var );
2239     ok( r == S_OK, "returns %08x\n", r );
2240     ok( V_VT(&var) == VT_BSTR, "variant type %x\n", V_VT(&var));
2241     VariantClear(&var);
2242     SysFreeString(name);
2243
2244     /* Create an Attribute */
2245     V_VT(&var) = VT_I4;
2246     V_I4(&var) = NODE_ATTRIBUTE;
2247     str = SysAllocString( szAttribute );
2248     r = IXMLDOMDocument_createNode( doc, var, str, NULL, &node );
2249     ok( r == S_OK, "returns %08x\n", r );
2250     ok( node != NULL, "node was null\n");
2251     SysFreeString(str);
2252
2253     r = IXMLDOMNode_get_nodeTypeString(node, &str);
2254     ok( r == S_OK, "returns %08x\n", r );
2255     ok( !lstrcmpW( str, _bstr_("attribute") ), "incorrect nodeTypeString string\n");
2256     SysFreeString(str);
2257     IXMLDOMNode_Release( node );
2258
2259     IXMLDOMElement_Release( element );
2260     IXMLDOMNode_Release( root );
2261     IXMLDOMDocument_Release( doc );
2262 }
2263
2264 static void test_getElementsByTagName(void)
2265 {
2266     IXMLDOMNodeList *node_list;
2267     IXMLDOMDocument *doc;
2268     IXMLDOMElement *elem;
2269     WCHAR buff[100];
2270     VARIANT_BOOL b;
2271     HRESULT r;
2272     LONG len;
2273     BSTR str;
2274
2275     doc = create_document(&IID_IXMLDOMDocument);
2276     if (!doc) return;
2277
2278     str = SysAllocString( szComplete4 );
2279     r = IXMLDOMDocument_loadXML( doc, str, &b );
2280     ok( r == S_OK, "loadXML failed\n");
2281     ok( b == VARIANT_TRUE, "failed to load XML string\n");
2282     SysFreeString( str );
2283
2284     str = SysAllocString( szstar );
2285
2286     /* null arguments cases */
2287     r = IXMLDOMDocument_getElementsByTagName(doc, NULL, &node_list);
2288     ok( r == E_INVALIDARG, "ret %08x\n", r );
2289     r = IXMLDOMDocument_getElementsByTagName(doc, str, NULL);
2290     ok( r == E_INVALIDARG, "ret %08x\n", r );
2291
2292     r = IXMLDOMDocument_getElementsByTagName(doc, str, &node_list);
2293     ok( r == S_OK, "ret %08x\n", r );
2294     r = IXMLDOMNodeList_get_length( node_list, &len );
2295     ok( r == S_OK, "ret %08x\n", r );
2296     ok( len == 6, "len %d\n", len );
2297
2298     test_disp((IUnknown*)node_list);
2299
2300     IXMLDOMNodeList_Release( node_list );
2301     SysFreeString( str );
2302
2303     /* broken query BSTR */
2304     memcpy(&buff[2], szstar, sizeof(szstar));
2305     /* just a big length */
2306     *(DWORD*)buff = 0xf0f0;
2307     r = IXMLDOMDocument_getElementsByTagName(doc, &buff[2], &node_list);
2308     ok( r == S_OK, "ret %08x\n", r );
2309     r = IXMLDOMNodeList_get_length( node_list, &len );
2310     ok( r == S_OK, "ret %08x\n", r );
2311     ok( len == 6, "len %d\n", len );
2312     IXMLDOMNodeList_Release( node_list );
2313
2314     str = SysAllocString( szbs );
2315     r = IXMLDOMDocument_getElementsByTagName(doc, str, &node_list);
2316     ok( r == S_OK, "ret %08x\n", r );
2317     r = IXMLDOMNodeList_get_length( node_list, &len );
2318     ok( r == S_OK, "ret %08x\n", r );
2319     ok( len == 1, "len %d\n", len );
2320     IXMLDOMNodeList_Release( node_list );
2321     SysFreeString( str );
2322
2323     str = SysAllocString( szdl );
2324     r = IXMLDOMDocument_getElementsByTagName(doc, str, &node_list);
2325     ok( r == S_OK, "ret %08x\n", r );
2326     r = IXMLDOMNodeList_get_length( node_list, &len );
2327     ok( r == S_OK, "ret %08x\n", r );
2328     ok( len == 0, "len %d\n", len );
2329     IXMLDOMNodeList_Release( node_list );
2330     SysFreeString( str );
2331
2332     str = SysAllocString( szstr1 );
2333     r = IXMLDOMDocument_getElementsByTagName(doc, str, &node_list);
2334     ok( r == S_OK, "ret %08x\n", r );
2335     r = IXMLDOMNodeList_get_length( node_list, &len );
2336     ok( r == S_OK, "ret %08x\n", r );
2337     ok( len == 0, "len %d\n", len );
2338     IXMLDOMNodeList_Release( node_list );
2339     SysFreeString( str );
2340
2341     /* test for element */
2342     r = IXMLDOMDocument_get_documentElement(doc, &elem);
2343     ok( r == S_OK, "ret %08x\n", r );
2344
2345     str = SysAllocString( szstar );
2346
2347     /* null arguments cases */
2348     r = IXMLDOMElement_getElementsByTagName(elem, NULL, &node_list);
2349     ok( r == E_INVALIDARG, "ret %08x\n", r );
2350     r = IXMLDOMElement_getElementsByTagName(elem, str, NULL);
2351     ok( r == E_INVALIDARG, "ret %08x\n", r );
2352
2353     r = IXMLDOMElement_getElementsByTagName(elem, str, &node_list);
2354     ok( r == S_OK, "ret %08x\n", r );
2355     r = IXMLDOMNodeList_get_length( node_list, &len );
2356     ok( r == S_OK, "ret %08x\n", r );
2357     todo_wine ok( len == 5, "len %d\n", len );
2358
2359     IXMLDOMNodeList_Release( node_list );
2360     SysFreeString( str );
2361
2362     /* broken query BSTR */
2363     memcpy(&buff[2], szstar, sizeof(szstar));
2364     /* just a big length */
2365     *(DWORD*)buff = 0xf0f0;
2366     r = IXMLDOMElement_getElementsByTagName(elem, &buff[2], &node_list);
2367     ok( r == S_OK, "ret %08x\n", r );
2368     r = IXMLDOMNodeList_get_length( node_list, &len );
2369     ok( r == S_OK, "ret %08x\n", r );
2370     todo_wine ok( len == 5, "len %d\n", len );
2371     IXMLDOMNodeList_Release( node_list );
2372
2373     IXMLDOMElement_Release(elem);
2374
2375     IXMLDOMDocument_Release( doc );
2376 }
2377
2378 static void test_get_text(void)
2379 {
2380     HRESULT r;
2381     BSTR str;
2382     VARIANT_BOOL b;
2383     IXMLDOMDocument *doc;
2384     IXMLDOMNode *node, *node2, *node3;
2385     IXMLDOMNode *nodeRoot;
2386     IXMLDOMNodeList *node_list;
2387     IXMLDOMNamedNodeMap *node_map;
2388     LONG len;
2389
2390     doc = create_document(&IID_IXMLDOMDocument);
2391     if (!doc) return;
2392
2393     str = SysAllocString( szComplete4 );
2394     r = IXMLDOMDocument_loadXML( doc, str, &b );
2395     ok( r == S_OK, "loadXML failed\n");
2396     ok( b == VARIANT_TRUE, "failed to load XML string\n");
2397     SysFreeString( str );
2398
2399     str = SysAllocString( szbs );
2400     r = IXMLDOMDocument_getElementsByTagName( doc, str, &node_list );
2401     ok( r == S_OK, "ret %08x\n", r );
2402     SysFreeString(str);
2403
2404     /* Test to get all child node text. */
2405     r = IXMLDOMDocument_QueryInterface(doc, &IID_IXMLDOMNode, (void**)&nodeRoot);
2406     ok( r == S_OK, "ret %08x\n", r );
2407     if(r == S_OK)
2408     {
2409         r = IXMLDOMNode_get_text( nodeRoot, &str );
2410         ok( r == S_OK, "ret %08x\n", r );
2411         ok( compareIgnoreReturnsWhitespace(str, _bstr_("fn1.txt\n\n fn2.txt \n\nf1\n")), "wrong get_text\n");
2412         SysFreeString(str);
2413
2414         IXMLDOMNode_Release(nodeRoot);
2415     }
2416
2417     if (0) {
2418     /* this test crashes on win9x */
2419     r = IXMLDOMNodeList_QueryInterface(node_list, &IID_IDispatch, NULL);
2420     ok( r == E_INVALIDARG, "ret %08x\n", r );
2421     }
2422
2423     r = IXMLDOMNodeList_get_length( node_list, NULL );
2424     ok( r == E_INVALIDARG, "ret %08x\n", r );
2425
2426     r = IXMLDOMNodeList_get_length( node_list, &len );
2427     ok( r == S_OK, "ret %08x\n", r );
2428     ok( len == 1, "expect 1 got %d\n", len );
2429
2430     r = IXMLDOMNodeList_get_item( node_list, 0, NULL );
2431     ok( r == E_INVALIDARG, "ret %08x\n", r );
2432
2433     r = IXMLDOMNodeList_nextNode( node_list, NULL );
2434     ok( r == E_INVALIDARG, "ret %08x\n", r );
2435
2436     r = IXMLDOMNodeList_get_item( node_list, 0, &node );
2437     ok( r == S_OK, "ret %08x\n", r ); 
2438     IXMLDOMNodeList_Release( node_list );
2439
2440     /* Invalid output parameter*/
2441     r = IXMLDOMNode_get_text( node, NULL );
2442     ok( r == E_INVALIDARG, "ret %08x\n", r );
2443
2444     r = IXMLDOMNode_get_text( node, &str );
2445     ok( r == S_OK, "ret %08x\n", r );
2446     ok( !memcmp(str, szfn1_txt, lstrlenW(szfn1_txt) ), "wrong string\n" );
2447     SysFreeString(str);
2448
2449     r = IXMLDOMNode_get_attributes( node, &node_map );
2450     ok( r == S_OK, "ret %08x\n", r );
2451     
2452     str = SysAllocString( szvr );
2453     r = IXMLDOMNamedNodeMap_getNamedItem( node_map, str, &node2 );
2454     ok( r == S_OK, "ret %08x\n", r );
2455     SysFreeString(str);
2456
2457     r = IXMLDOMNode_get_text( node2, &str );
2458     ok( r == S_OK, "ret %08x\n", r );
2459     ok( !memcmp(str, szstr2, sizeof(szstr2)), "wrong string\n" );
2460     SysFreeString(str);
2461
2462     r = IXMLDOMNode_get_firstChild( node2, &node3 );
2463     ok( r == S_OK, "ret %08x\n", r );
2464
2465     r = IXMLDOMNode_get_text( node3, &str );
2466     ok( r == S_OK, "ret %08x\n", r );
2467     ok( !memcmp(str, szstr2, sizeof(szstr2)), "wrong string\n" );
2468     SysFreeString(str);
2469
2470
2471     IXMLDOMNode_Release( node3 );
2472     IXMLDOMNode_Release( node2 );
2473     IXMLDOMNamedNodeMap_Release( node_map );
2474     IXMLDOMNode_Release( node );
2475     IXMLDOMDocument_Release( doc );
2476 }
2477
2478 static void test_get_childNodes(void)
2479 {
2480     HRESULT r;
2481     BSTR str;
2482     VARIANT_BOOL b;
2483     IXMLDOMDocument *doc;
2484     IXMLDOMElement *element;
2485     IXMLDOMNode *node, *node2;
2486     IXMLDOMNodeList *node_list, *node_list2;
2487     LONG len;
2488
2489     doc = create_document(&IID_IXMLDOMDocument);
2490     if (!doc) return;
2491
2492     str = SysAllocString( szComplete4 );
2493     r = IXMLDOMDocument_loadXML( doc, str, &b );
2494     ok( r == S_OK, "loadXML failed\n");
2495     ok( b == VARIANT_TRUE, "failed to load XML string\n");
2496     SysFreeString( str );
2497
2498     r = IXMLDOMDocument_get_documentElement( doc, &element );
2499     ok( r == S_OK, "ret %08x\n", r);
2500
2501     r = IXMLDOMElement_get_childNodes( element, &node_list );
2502     ok( r == S_OK, "ret %08x\n", r);
2503
2504     r = IXMLDOMNodeList_get_length( node_list, &len );
2505     ok( r == S_OK, "ret %08x\n", r);
2506     ok( len == 4, "len %d\n", len);
2507
2508     r = IXMLDOMNodeList_get_item( node_list, 2, &node );
2509     ok( r == S_OK, "ret %08x\n", r);
2510
2511     r = IXMLDOMNode_get_childNodes( node, &node_list2 );
2512     ok( r == S_OK, "ret %08x\n", r);
2513
2514     r = IXMLDOMNodeList_get_length( node_list2, &len );
2515     ok( r == S_OK, "ret %08x\n", r);
2516     ok( len == 0, "len %d\n", len);
2517
2518     r = IXMLDOMNodeList_get_item( node_list2, 0, &node2);
2519     ok( r == S_FALSE, "ret %08x\n", r);
2520
2521     IXMLDOMNodeList_Release( node_list2 );
2522     IXMLDOMNode_Release( node );
2523     IXMLDOMNodeList_Release( node_list );
2524     IXMLDOMElement_Release( element );
2525     IXMLDOMDocument_Release( doc );
2526 }
2527
2528 static void test_get_firstChild(void)
2529 {
2530     static WCHAR xmlW[] = {'x','m','l',0};
2531     IXMLDOMDocument *doc;
2532     IXMLDOMNode *node;
2533     VARIANT_BOOL b;
2534     HRESULT r;
2535     BSTR str;
2536
2537     doc = create_document(&IID_IXMLDOMDocument);
2538     if (!doc) return;
2539
2540     str = SysAllocString( szComplete4 );
2541     r = IXMLDOMDocument_loadXML( doc, str, &b );
2542     ok( r == S_OK, "loadXML failed\n");
2543     ok( b == VARIANT_TRUE, "failed to load XML string\n");
2544     SysFreeString( str );
2545
2546     r = IXMLDOMDocument_get_firstChild( doc, &node );
2547     ok( r == S_OK, "ret %08x\n", r);
2548
2549     r = IXMLDOMNode_get_nodeName( node, &str );
2550     ok( r == S_OK, "ret %08x\n", r);
2551
2552     ok(memcmp(str, xmlW, sizeof(xmlW)) == 0, "expected \"xml\" node name\n");
2553
2554     SysFreeString(str);
2555     IXMLDOMNode_Release( node );
2556     IXMLDOMDocument_Release( doc );
2557 }
2558
2559 static void test_get_lastChild(void)
2560 {
2561     static WCHAR lcW[] = {'l','c',0};
2562     static WCHAR foW[] = {'f','o',0};
2563     IXMLDOMDocument *doc;
2564     IXMLDOMNode *node, *child;
2565     VARIANT_BOOL b;
2566     HRESULT r;
2567     BSTR str;
2568
2569     doc = create_document(&IID_IXMLDOMDocument);
2570     if (!doc) return;
2571
2572     str = SysAllocString( szComplete4 );
2573     r = IXMLDOMDocument_loadXML( doc, str, &b );
2574     ok( r == S_OK, "loadXML failed\n");
2575     ok( b == VARIANT_TRUE, "failed to load XML string\n");
2576     SysFreeString( str );
2577
2578     r = IXMLDOMDocument_get_lastChild( doc, &node );
2579     ok( r == S_OK, "ret %08x\n", r);
2580
2581     r = IXMLDOMNode_get_nodeName( node, &str );
2582     ok( r == S_OK, "ret %08x\n", r);
2583
2584     ok(memcmp(str, lcW, sizeof(lcW)) == 0, "expected \"lc\" node name\n");
2585     SysFreeString(str);
2586
2587     r = IXMLDOMNode_get_lastChild( node, &child );
2588     ok( r == S_OK, "ret %08x\n", r);
2589
2590     r = IXMLDOMNode_get_nodeName( child, &str );
2591     ok( r == S_OK, "ret %08x\n", r);
2592
2593     ok(memcmp(str, foW, sizeof(foW)) == 0, "expected \"fo\" node name\n");
2594     SysFreeString(str);
2595
2596     IXMLDOMNode_Release( child );
2597     IXMLDOMNode_Release( node );
2598     IXMLDOMDocument_Release( doc );
2599 }
2600
2601 static void test_removeChild(void)
2602 {
2603     HRESULT r;
2604     BSTR str;
2605     VARIANT_BOOL b;
2606     IXMLDOMDocument *doc;
2607     IXMLDOMElement *element, *lc_element;
2608     IXMLDOMNode *fo_node, *ba_node, *removed_node, *temp_node, *lc_node;
2609     IXMLDOMNodeList *root_list, *fo_list;
2610
2611     doc = create_document(&IID_IXMLDOMDocument);
2612     if (!doc) return;
2613
2614     str = SysAllocString( szComplete4 );
2615     r = IXMLDOMDocument_loadXML( doc, str, &b );
2616     ok( r == S_OK, "loadXML failed\n");
2617     ok( b == VARIANT_TRUE, "failed to load XML string\n");
2618     SysFreeString( str );
2619
2620     r = IXMLDOMDocument_get_documentElement( doc, &element );
2621     ok( r == S_OK, "ret %08x\n", r);
2622
2623     r = IXMLDOMElement_get_childNodes( element, &root_list );
2624     ok( r == S_OK, "ret %08x\n", r);
2625
2626     r = IXMLDOMNodeList_get_item( root_list, 3, &fo_node );
2627     ok( r == S_OK, "ret %08x\n", r);
2628  
2629     r = IXMLDOMNode_get_childNodes( fo_node, &fo_list );
2630     ok( r == S_OK, "ret %08x\n", r);
2631  
2632     r = IXMLDOMNodeList_get_item( fo_list, 0, &ba_node );
2633     ok( r == S_OK, "ret %08x\n", r);
2634
2635     /* invalid parameter: NULL ptr */
2636     removed_node = (void*)0xdeadbeef;
2637     r = IXMLDOMElement_removeChild( element, NULL, &removed_node );
2638     ok( r == E_INVALIDARG, "ret %08x\n", r );
2639     ok( removed_node == (void*)0xdeadbeef, "%p\n", removed_node );
2640
2641     /* ba_node is a descendant of element, but not a direct child. */
2642     removed_node = (void*)0xdeadbeef;
2643     r = IXMLDOMElement_removeChild( element, ba_node, &removed_node );
2644     ok( r == E_INVALIDARG, "ret %08x\n", r );
2645     ok( removed_node == NULL, "%p\n", removed_node );
2646
2647     r = IXMLDOMElement_removeChild( element, fo_node, &removed_node );
2648     ok( r == S_OK, "ret %08x\n", r);
2649     ok( fo_node == removed_node, "node %p node2 %p\n", fo_node, removed_node );
2650
2651     /* try removing already removed child */
2652     temp_node = (void*)0xdeadbeef;
2653     r = IXMLDOMElement_removeChild( element, fo_node, &temp_node );
2654     ok( r == E_INVALIDARG, "ret %08x\n", r);
2655     ok( temp_node == NULL, "%p\n", temp_node );
2656
2657     /* the removed node has no parent anymore */
2658     r = IXMLDOMNode_get_parentNode( removed_node, &temp_node );
2659     ok( r == S_FALSE, "ret %08x\n", r);
2660     ok( temp_node == NULL, "%p\n", temp_node );
2661
2662     IXMLDOMNode_Release( removed_node );
2663     IXMLDOMNode_Release( ba_node );
2664     IXMLDOMNodeList_Release( fo_list );
2665
2666     r = IXMLDOMNodeList_get_item( root_list, 0, &lc_node );
2667     ok( r == S_OK, "ret %08x\n", r);
2668
2669     r = IXMLDOMElement_QueryInterface( lc_node, &IID_IXMLDOMElement, (void**)&lc_element );
2670     ok( r == S_OK, "ret %08x\n", r);
2671
2672     /* MS quirk: passing wrong interface pointer works, too */
2673     r = IXMLDOMElement_removeChild( element, (IXMLDOMNode*)lc_element, NULL );
2674     ok( r == S_OK, "ret %08x\n", r);
2675
2676     r = IXMLDOMNode_get_parentNode( lc_node, &temp_node );
2677     ok( r == S_FALSE, "ret %08x\n", r);
2678     ok( temp_node == NULL, "%p\n", temp_node );
2679
2680     IXMLDOMNode_Release( lc_node );
2681     IXMLDOMNodeList_Release( root_list );
2682     IXMLDOMElement_Release( element );
2683     IXMLDOMDocument_Release( doc );
2684 }
2685
2686 static void test_replaceChild(void)
2687 {
2688     HRESULT r;
2689     BSTR str;
2690     VARIANT_BOOL b;
2691     IXMLDOMDocument *doc;
2692     IXMLDOMElement *element, *ba_element;
2693     IXMLDOMNode *fo_node, *ba_node, *lc_node, *removed_node, *temp_node;
2694     IXMLDOMNodeList *root_list, *fo_list;
2695     IUnknown * unk1, *unk2;
2696     LONG len;
2697
2698     doc = create_document(&IID_IXMLDOMDocument);
2699     if (!doc) return;
2700
2701     str = SysAllocString( szComplete4 );
2702     r = IXMLDOMDocument_loadXML( doc, str, &b );
2703     ok( r == S_OK, "loadXML failed\n");
2704     ok( b == VARIANT_TRUE, "failed to load XML string\n");
2705     SysFreeString( str );
2706
2707     r = IXMLDOMDocument_get_documentElement( doc, &element );
2708     ok( r == S_OK, "ret %08x\n", r);
2709
2710     r = IXMLDOMElement_get_childNodes( element, &root_list );
2711     ok( r == S_OK, "ret %08x\n", r);
2712
2713     r = IXMLDOMNodeList_get_item( root_list, 0, &lc_node );
2714     ok( r == S_OK, "ret %08x\n", r);
2715
2716     r = IXMLDOMNodeList_get_item( root_list, 3, &fo_node );
2717     ok( r == S_OK, "ret %08x\n", r);
2718
2719     r = IXMLDOMNode_get_childNodes( fo_node, &fo_list );
2720     ok( r == S_OK, "ret %08x\n", r);
2721
2722     r = IXMLDOMNodeList_get_item( fo_list, 0, &ba_node );
2723     ok( r == S_OK, "ret %08x\n", r);
2724
2725     IXMLDOMNodeList_Release( fo_list );
2726
2727     /* invalid parameter: NULL ptr for element to remove */
2728     removed_node = (void*)0xdeadbeef;
2729     r = IXMLDOMElement_replaceChild( element, ba_node, NULL, &removed_node );
2730     ok( r == E_INVALIDARG, "ret %08x\n", r );
2731     ok( removed_node == (void*)0xdeadbeef, "%p\n", removed_node );
2732
2733     /* invalid parameter: NULL for replacement element. (Sic!) */
2734     removed_node = (void*)0xdeadbeef;
2735     r = IXMLDOMElement_replaceChild( element, NULL, fo_node, &removed_node );
2736     ok( r == E_INVALIDARG, "ret %08x\n", r );
2737     ok( removed_node == (void*)0xdeadbeef, "%p\n", removed_node );
2738
2739     /* invalid parameter: OldNode is not a child */
2740     removed_node = (void*)0xdeadbeef;
2741     r = IXMLDOMElement_replaceChild( element, lc_node, ba_node, &removed_node );
2742     ok( r == E_INVALIDARG, "ret %08x\n", r );
2743     ok( removed_node == NULL, "%p\n", removed_node );
2744
2745     /* invalid parameter: would create loop */
2746     removed_node = (void*)0xdeadbeef;
2747     r = IXMLDOMNode_replaceChild( fo_node, fo_node, ba_node, &removed_node );
2748     ok( r == E_FAIL, "ret %08x\n", r );
2749     ok( removed_node == NULL, "%p\n", removed_node );
2750
2751     r = IXMLDOMElement_replaceChild( element, ba_node, fo_node, NULL );
2752     ok( r == S_OK, "ret %08x\n", r );
2753
2754     r = IXMLDOMNodeList_get_item( root_list, 3, &temp_node );
2755     ok( r == S_OK, "ret %08x\n", r );
2756
2757     /* ba_node and temp_node refer to the same node, yet they
2758        are different interface pointers */
2759     ok( ba_node != temp_node, "ba_node %p temp_node %p\n", ba_node, temp_node);
2760     r = IXMLDOMNode_QueryInterface( temp_node, &IID_IUnknown, (void**)&unk1);
2761     ok( r == S_OK, "ret %08x\n", r );
2762     r = IXMLDOMNode_QueryInterface( ba_node, &IID_IUnknown, (void**)&unk2);
2763     ok( r == S_OK, "ret %08x\n", r );
2764     todo_wine ok( unk1 == unk2, "unk1 %p unk2 %p\n", unk1, unk2);
2765
2766     IUnknown_Release( unk1 );
2767     IUnknown_Release( unk2 );
2768
2769     /* ba_node should have been removed from below fo_node */
2770     r = IXMLDOMNode_get_childNodes( fo_node, &fo_list );
2771     ok( r == S_OK, "ret %08x\n", r );
2772
2773     /* MS quirk: replaceChild also accepts elements instead of nodes */
2774     r = IXMLDOMNode_QueryInterface( ba_node, &IID_IXMLDOMElement, (void**)&ba_element);
2775     ok( r == S_OK, "ret %08x\n", r );
2776
2777     r = IXMLDOMElement_replaceChild( element, ba_node, (IXMLDOMNode*)ba_element, &removed_node );
2778     ok( r == S_OK, "ret %08x\n", r );
2779
2780     r = IXMLDOMNodeList_get_length( fo_list, &len);
2781     ok( r == S_OK, "ret %08x\n", r );
2782     ok( len == 0, "len %d\n", len);
2783
2784     IXMLDOMNodeList_Release( fo_list );
2785
2786     IXMLDOMNode_Release(ba_node);
2787     IXMLDOMNode_Release(fo_node);
2788     IXMLDOMNode_Release(temp_node);
2789     IXMLDOMNodeList_Release( root_list );
2790     IXMLDOMElement_Release( element );
2791     IXMLDOMDocument_Release( doc );
2792 }
2793
2794 static void test_removeNamedItem(void)
2795 {
2796     IXMLDOMDocument *doc;
2797     IXMLDOMElement *element;
2798     IXMLDOMNode *pr_node, *removed_node, *removed_node2;
2799     IXMLDOMNodeList *root_list;
2800     IXMLDOMNamedNodeMap * pr_attrs;
2801     VARIANT_BOOL b;
2802     BSTR str;
2803     LONG len;
2804     HRESULT r;
2805
2806     doc = create_document(&IID_IXMLDOMDocument);
2807     if (!doc) return;
2808
2809     str = SysAllocString( szComplete4 );
2810     r = IXMLDOMDocument_loadXML( doc, str, &b );
2811     ok( r == S_OK, "loadXML failed\n");
2812     ok( b == VARIANT_TRUE, "failed to load XML string\n");
2813     SysFreeString( str );
2814
2815     r = IXMLDOMDocument_get_documentElement( doc, &element );
2816     ok( r == S_OK, "ret %08x\n", r);
2817
2818     r = IXMLDOMElement_get_childNodes( element, &root_list );
2819     ok( r == S_OK, "ret %08x\n", r);
2820
2821     r = IXMLDOMNodeList_get_item( root_list, 1, &pr_node );
2822     ok( r == S_OK, "ret %08x\n", r);
2823
2824     r = IXMLDOMNode_get_attributes( pr_node, &pr_attrs );
2825     ok( r == S_OK, "ret %08x\n", r);
2826
2827     r = IXMLDOMNamedNodeMap_get_length( pr_attrs, &len );
2828     ok( r == S_OK, "ret %08x\n", r);
2829     ok( len == 3, "length %d\n", len);
2830
2831     removed_node = (void*)0xdeadbeef;
2832     r = IXMLDOMNamedNodeMap_removeNamedItem( pr_attrs, NULL, &removed_node);
2833     ok ( r == E_INVALIDARG, "ret %08x\n", r);
2834     ok ( removed_node == (void*)0xdeadbeef, "got %p\n", removed_node);
2835
2836     removed_node = (void*)0xdeadbeef;
2837     str = SysAllocString(szvr);
2838     r = IXMLDOMNamedNodeMap_removeNamedItem( pr_attrs, str, &removed_node);
2839     ok ( r == S_OK, "ret %08x\n", r);
2840
2841     removed_node2 = (void*)0xdeadbeef;
2842     r = IXMLDOMNamedNodeMap_removeNamedItem( pr_attrs, str, &removed_node2);
2843     ok ( r == S_FALSE, "ret %08x\n", r);
2844     ok ( removed_node2 == NULL, "got %p\n", removed_node2 );
2845
2846     r = IXMLDOMNamedNodeMap_get_length( pr_attrs, &len );
2847     ok( r == S_OK, "ret %08x\n", r);
2848     ok( len == 2, "length %d\n", len);
2849
2850     r = IXMLDOMNamedNodeMap_setNamedItem( pr_attrs, removed_node, NULL);
2851     ok ( r == S_OK, "ret %08x\n", r);
2852     IXMLDOMNode_Release(removed_node);
2853
2854     r = IXMLDOMNamedNodeMap_get_length( pr_attrs, &len );
2855     ok( r == S_OK, "ret %08x\n", r);
2856     ok( len == 3, "length %d\n", len);
2857
2858     r = IXMLDOMNamedNodeMap_removeNamedItem( pr_attrs, str, NULL);
2859     ok ( r == S_OK, "ret %08x\n", r);
2860
2861     r = IXMLDOMNamedNodeMap_get_length( pr_attrs, &len );
2862     ok( r == S_OK, "ret %08x\n", r);
2863     ok( len == 2, "length %d\n", len);
2864
2865     r = IXMLDOMNamedNodeMap_removeNamedItem( pr_attrs, str, NULL);
2866     ok ( r == S_FALSE, "ret %08x\n", r);
2867
2868     SysFreeString(str);
2869
2870     IXMLDOMNamedNodeMap_Release( pr_attrs );
2871     IXMLDOMNode_Release( pr_node );
2872     IXMLDOMNodeList_Release( root_list );
2873     IXMLDOMElement_Release( element );
2874     IXMLDOMDocument_Release( doc );
2875 }
2876
2877 static void test_XMLHTTP(void)
2878 {
2879     static const WCHAR wszBody[] = {'m','o','d','e','=','T','e','s','t',0};
2880     static WCHAR wszPOST[] = {'P','O','S','T',0};
2881     static WCHAR wszUrl[] = {'h','t','t','p',':','/','/',
2882         'c','r','o','s','s','o','v','e','r','.','c','o','d','e','w','e','a','v','e','r','s','.','c','o','m','/',
2883         'p','o','s','t','t','e','s','t','.','p','h','p',0};
2884     static const WCHAR wszExpectedResponse[] = {'F','A','I','L','E','D',0};
2885     IXMLHttpRequest *pXMLHttpRequest;
2886     BSTR bstrResponse, str1, str2;
2887     VARIANT dummy;
2888     VARIANT varfalse;
2889     VARIANT varbody;
2890     HRESULT hr = CoCreateInstance(&CLSID_XMLHTTPRequest, NULL,
2891                                   CLSCTX_INPROC_SERVER, &IID_IXMLHttpRequest,
2892                                   (void **)&pXMLHttpRequest);
2893     if (FAILED(hr))
2894     {
2895         win_skip("IXMLHTTPRequest is not available (0x%08x)\n", hr);
2896         return;
2897     }
2898
2899     VariantInit(&dummy);
2900     V_VT(&dummy) = VT_ERROR;
2901     V_ERROR(&dummy) = DISP_E_MEMBERNOTFOUND;
2902     VariantInit(&varfalse);
2903     V_VT(&varfalse) = VT_BOOL;
2904     V_BOOL(&varfalse) = VARIANT_FALSE;
2905     V_VT(&varbody) = VT_BSTR;
2906     V_BSTR(&varbody) = SysAllocString(wszBody);
2907
2908     str1 = SysAllocString(wszPOST);
2909     str2 = SysAllocString(wszUrl);
2910     hr = IXMLHttpRequest_open(pXMLHttpRequest, str1, str2, varfalse, dummy, dummy);
2911     todo_wine ok(hr == S_OK, "IXMLHttpRequest_open should have succeeded instead of failing with 0x%08x\n", hr);
2912     SysFreeString(str1);
2913     SysFreeString(str2);
2914
2915     hr = IXMLHttpRequest_send(pXMLHttpRequest, varbody);
2916     if (hr == INET_E_RESOURCE_NOT_FOUND)
2917     {
2918         skip("No connection could be made with crossover.codeweavers.com\n");
2919         IXMLHttpRequest_Release(pXMLHttpRequest);
2920         return;
2921     }
2922     todo_wine ok(hr == S_OK, "IXMLHttpRequest_send should have succeeded instead of failing with 0x%08x\n", hr);
2923     VariantClear(&varbody);
2924
2925     hr = IXMLHttpRequest_get_responseText(pXMLHttpRequest, &bstrResponse);
2926     todo_wine ok(hr == S_OK, "IXMLHttpRequest_get_responseText should have succeeded instead of failing with 0x%08x\n", hr);
2927     /* the server currently returns "FAILED" because the Content-Type header is
2928      * not what the server expects */
2929     if(hr == S_OK)
2930     {
2931         ok(!memcmp(bstrResponse, wszExpectedResponse, sizeof(wszExpectedResponse)), "bstrResponse differs from what was expected\n");
2932         SysFreeString(bstrResponse);
2933     }
2934
2935     IXMLHttpRequest_Release(pXMLHttpRequest);
2936 }
2937
2938 static void test_IXMLDOMDocument2(void)
2939 {
2940     static const WCHAR emptyW[] = {0};
2941     IXMLDOMDocument2 *doc2;
2942     IXMLDOMDocument *doc;
2943     IDispatchEx *dispex;
2944     VARIANT_BOOL b;
2945     VARIANT var;
2946     HRESULT r;
2947     LONG ref;
2948     BSTR str;
2949
2950     doc = create_document(&IID_IXMLDOMDocument);
2951     if (!doc) return;
2952
2953     str = SysAllocString( szComplete4 );
2954     r = IXMLDOMDocument_loadXML( doc, str, &b );
2955     ok( r == S_OK, "loadXML failed\n");
2956     ok( b == VARIANT_TRUE, "failed to load XML string\n");
2957     SysFreeString( str );
2958
2959     r = IXMLDOMDocument_QueryInterface( doc, &IID_IXMLDOMDocument2, (void**)&doc2 );
2960     ok( r == S_OK, "ret %08x\n", r );
2961     ok( doc == (IXMLDOMDocument*)doc2, "interfaces differ\n");
2962
2963     r = IXMLDOMDocument_QueryInterface( doc, &IID_IDispatchEx, (void**)&dispex );
2964     ok( r == S_OK, "ret %08x\n", r );
2965     if(r == S_OK)
2966     {
2967         IDispatchEx_Release(dispex);
2968     }
2969
2970     /* we will check if the variant got cleared */
2971     ref = IXMLDOMDocument2_AddRef(doc2);
2972     expect_eq(ref, 3, int, "%d");  /* doc, doc2, AddRef*/
2973     V_VT(&var) = VT_UNKNOWN;
2974     V_UNKNOWN(&var) = (IUnknown *)doc2;
2975
2976     /* invalid calls */
2977     ole_expect(IXMLDOMDocument2_getProperty(doc2, _bstr_("askldhfaklsdf"), &var), E_FAIL);
2978     expect_eq(V_VT(&var), VT_UNKNOWN, int, "%x");
2979     ole_expect(IXMLDOMDocument2_getProperty(doc2, _bstr_("SelectionLanguage"), NULL), E_INVALIDARG);
2980
2981     /* valid call */
2982     ole_check(IXMLDOMDocument2_getProperty(doc2, _bstr_("SelectionLanguage"), &var));
2983     expect_eq(V_VT(&var), VT_BSTR, int, "%x");
2984     expect_bstr_eq_and_free(V_BSTR(&var), "XSLPattern");
2985     V_VT(&var) = VT_R4;
2986
2987     /* the variant didn't get cleared*/
2988     expect_eq(IXMLDOMDocument2_Release(doc2), 2, int, "%d");
2989
2990     /* setProperty tests */
2991     ole_expect(IXMLDOMDocument2_setProperty(doc2, _bstr_("askldhfaklsdf"), var), E_FAIL);
2992     ole_expect(IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionLanguage"), var), E_FAIL);
2993     ole_expect(IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionLanguage"), _variantbstr_("alskjdh faklsjd hfk")), E_FAIL);
2994     ole_check(IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionLanguage"), _variantbstr_("XSLPattern")));
2995     ole_check(IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionLanguage"), _variantbstr_("XPath")));
2996     ole_check(IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionLanguage"), _variantbstr_("XSLPattern")));
2997
2998     V_VT(&var) = VT_BSTR;
2999     V_BSTR(&var) = SysAllocString(emptyW);
3000     r = IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionNamespaces"), var);
3001     todo_wine ok(r == S_OK, "got 0x%08x\n", r);
3002     VariantClear(&var);
3003
3004     V_VT(&var) = VT_I2;
3005     V_I2(&var) = 0;
3006     r = IXMLDOMDocument2_setProperty(doc2, _bstr_("SelectionNamespaces"), var);
3007     ok(r == E_FAIL, "got 0x%08x\n", r);
3008
3009     /* contrary to what MSDN claims you can switch back from XPath to XSLPattern */
3010     ole_check(IXMLDOMDocument2_getProperty(doc2, _bstr_("SelectionLanguage"), &var));
3011     expect_eq(V_VT(&var), VT_BSTR, int, "%x");
3012     expect_bstr_eq_and_free(V_BSTR(&var), "XSLPattern");
3013
3014     IXMLDOMDocument2_Release( doc2 );
3015     IXMLDOMDocument_Release( doc );
3016     free_bstrs();
3017 }
3018
3019 static void test_XPath(void)
3020 {
3021     VARIANT var;
3022     VARIANT_BOOL b;
3023     IXMLDOMDocument2 *doc;
3024     IXMLDOMNode *rootNode;
3025     IXMLDOMNode *elem1Node;
3026     IXMLDOMNode *node;
3027     IXMLDOMNodeList *list;
3028
3029     doc = create_document(&IID_IXMLDOMDocument2);
3030     if (!doc) return;
3031
3032     ole_check(IXMLDOMDocument_loadXML(doc, _bstr_(szExampleXML), &b));
3033     ok(b == VARIANT_TRUE, "failed to load XML string\n");
3034
3035     /* switch to XPath */
3036     ole_check(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionLanguage"), _variantbstr_("XPath")));
3037
3038     /* some simple queries*/
3039     ole_check(IXMLDOMDocument_selectNodes(doc, _bstr_("root"), &list));
3040     ole_check(IXMLDOMNodeList_get_item(list, 0, &rootNode));
3041     ole_check(IXMLDOMNodeList_reset(list));
3042     expect_list_and_release(list, "E2.D1");
3043     if (rootNode == NULL)
3044         return;
3045
3046     ole_check(IXMLDOMDocument_selectNodes(doc, _bstr_("root//c"), &list));
3047     expect_list_and_release(list, "E3.E1.E2.D1 E3.E2.E2.D1");
3048
3049     ole_check(IXMLDOMDocument_selectNodes(doc, _bstr_("//c[@type]"), &list));
3050     expect_list_and_release(list, "E3.E2.E2.D1");
3051
3052     ole_check(IXMLDOMNode_selectNodes(rootNode, _bstr_("elem"), &list));
3053     /* using get_item for query results advances the position */
3054     ole_check(IXMLDOMNodeList_get_item(list, 1, &node));
3055     expect_node(node, "E2.E2.D1");
3056     IXMLDOMNode_Release(node);
3057     ole_check(IXMLDOMNodeList_nextNode(list, &node));
3058     expect_node(node, "E4.E2.D1");
3059     IXMLDOMNode_Release(node);
3060     ole_check(IXMLDOMNodeList_reset(list));
3061     expect_list_and_release(list, "E1.E2.D1 E2.E2.D1 E4.E2.D1");
3062
3063     ole_check(IXMLDOMNode_selectNodes(rootNode, _bstr_("."), &list));
3064     expect_list_and_release(list, "E2.D1");
3065
3066     ole_check(IXMLDOMNode_selectNodes(rootNode, _bstr_("elem[3]/preceding-sibling::*"), &list));
3067     ole_check(IXMLDOMNodeList_get_item(list, 0, &elem1Node));
3068     ole_check(IXMLDOMNodeList_reset(list));
3069     expect_list_and_release(list, "E1.E2.D1 E2.E2.D1 E3.E2.D1");
3070
3071     /* select an attribute */
3072     ole_check(IXMLDOMNode_selectNodes(rootNode, _bstr_(".//@type"), &list));
3073     expect_list_and_release(list, "A'type'.E3.E2.E2.D1");
3074
3075     /* would evaluate to a number */
3076     ole_expect(IXMLDOMNode_selectNodes(rootNode, _bstr_("count(*)"), &list), E_FAIL);
3077     /* would evaluate to a boolean */
3078     ole_expect(IXMLDOMNode_selectNodes(rootNode, _bstr_("position()>0"), &list), E_FAIL);
3079     /* would evaluate to a string */
3080     ole_expect(IXMLDOMNode_selectNodes(rootNode, _bstr_("name()"), &list), E_FAIL);
3081
3082     /* no results */
3083     ole_check(IXMLDOMNode_selectNodes(rootNode, _bstr_("c"), &list));
3084     expect_list_and_release(list, "");
3085     ole_check(IXMLDOMDocument_selectNodes(doc, _bstr_("elem//c"), &list));
3086     expect_list_and_release(list, "");
3087     ole_check(IXMLDOMDocument_selectNodes(doc, _bstr_("//elem[4]"), &list));
3088     expect_list_and_release(list, "");
3089     ole_check(IXMLDOMDocument_selectNodes(doc, _bstr_("root//elem[0]"), &list));
3090     expect_list_and_release(list, "");
3091
3092     /* foo undeclared in document node */
3093     ole_expect(IXMLDOMDocument_selectNodes(doc, _bstr_("root//foo:c"), &list), E_FAIL);
3094     /* undeclared in <root> node */
3095     ole_expect(IXMLDOMNode_selectNodes(rootNode, _bstr_(".//foo:c"), &list), E_FAIL);
3096     /* undeclared in <elem> node */
3097     ole_expect(IXMLDOMNode_selectNodes(elem1Node, _bstr_("//foo:c"), &list), E_FAIL);
3098     /* but this trick can be used */
3099     ole_check(IXMLDOMNode_selectNodes(elem1Node, _bstr_("//*[name()='foo:c']"), &list));
3100     expect_list_and_release(list, "E3.E4.E2.D1");
3101
3102     /* it has to be declared in SelectionNamespaces */
3103     todo_wine ole_check(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionNamespaces"),
3104         _variantbstr_("xmlns:test='urn:uuid:86B2F87F-ACB6-45cd-8B77-9BDB92A01A29'")));
3105
3106     /* now the namespace can be used */
3107     todo_wine ole_check(IXMLDOMDocument_selectNodes(doc, _bstr_("root//test:c"), &list));
3108     todo_wine expect_list_and_release(list, "E3.E3.E2.D1 E3.E4.E2.D1");
3109     todo_wine ole_check(IXMLDOMNode_selectNodes(rootNode, _bstr_(".//test:c"), &list));
3110     todo_wine expect_list_and_release(list, "E3.E3.E2.D1 E3.E4.E2.D1");
3111     todo_wine ole_check(IXMLDOMNode_selectNodes(elem1Node, _bstr_("//test:c"), &list));
3112     todo_wine expect_list_and_release(list, "E3.E3.E2.D1 E3.E4.E2.D1");
3113     todo_wine ole_check(IXMLDOMNode_selectNodes(elem1Node, _bstr_(".//test:x"), &list));
3114     todo_wine expect_list_and_release(list, "E5.E1.E4.E1.E2.D1");
3115
3116     /* SelectionNamespaces syntax error - the namespaces doesn't work anymore but the value is stored */
3117     ole_expect(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionNamespaces"),
3118         _variantbstr_("xmlns:test='urn:uuid:86B2F87F-ACB6-45cd-8B77-9BDB92A01A29' xmlns:foo=###")), E_FAIL);
3119
3120     ole_expect(IXMLDOMDocument_selectNodes(doc, _bstr_("root//foo:c"), &list), E_FAIL);
3121
3122     VariantInit(&var);
3123     todo_wine ole_check(IXMLDOMDocument2_getProperty(doc, _bstr_("SelectionNamespaces"), &var));
3124     todo_wine expect_eq(V_VT(&var), VT_BSTR, int, "%x");
3125     if (V_VT(&var) == VT_BSTR)
3126         expect_bstr_eq_and_free(V_BSTR(&var), "xmlns:test='urn:uuid:86B2F87F-ACB6-45cd-8B77-9BDB92A01A29' xmlns:foo=###");
3127
3128     /* extra attributes - same thing*/
3129     ole_expect(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionNamespaces"),
3130         _variantbstr_("xmlns:test='urn:uuid:86B2F87F-ACB6-45cd-8B77-9BDB92A01A29' param='test'")), E_FAIL);
3131     ole_expect(IXMLDOMDocument_selectNodes(doc, _bstr_("root//foo:c"), &list), E_FAIL);
3132
3133     IXMLDOMNode_Release(rootNode);
3134     IXMLDOMNode_Release(elem1Node);
3135     IXMLDOMDocument_Release(doc);
3136     free_bstrs();
3137 }
3138
3139 static void test_cloneNode(void )
3140 {
3141     IXMLDOMDocument *doc;
3142     VARIANT_BOOL b;
3143     IXMLDOMNodeList *pList;
3144     IXMLDOMNamedNodeMap *mapAttr;
3145     LONG nLength = 0, nLength1 = 0;
3146     LONG nAttrCnt = 0, nAttrCnt1 = 0;
3147     IXMLDOMNode *node;
3148     IXMLDOMNode *node_clone;
3149     IXMLDOMNode *node_first;
3150     HRESULT r;
3151     BSTR str;
3152     static const WCHAR szSearch[] = { 'l', 'c', '/', 'p', 'r', 0 };
3153
3154     doc = create_document(&IID_IXMLDOMDocument);
3155     if (!doc) return;
3156
3157     str = SysAllocString( szComplete4 );
3158     ole_check(IXMLDOMDocument_loadXML(doc, str, &b));
3159     ok(b == VARIANT_TRUE, "failed to load XML string\n");
3160     SysFreeString(str);
3161
3162     if(!b)
3163         return;
3164
3165     str = SysAllocString( szSearch);
3166     r = IXMLDOMNode_selectSingleNode(doc, str, &node);
3167     ok( r == S_OK, "ret %08x\n", r );
3168     ok( node != NULL, "node %p\n", node );
3169     SysFreeString(str);
3170
3171     if(!node)
3172     {
3173         IXMLDOMDocument_Release(doc);
3174         return;
3175     }
3176
3177     /* Check invalid parameter */
3178     r = IXMLDOMNode_cloneNode(node, VARIANT_TRUE, NULL);
3179     ok( r == E_INVALIDARG, "ret %08x\n", r );
3180
3181     /* All Children */
3182     r = IXMLDOMNode_cloneNode(node, VARIANT_TRUE, &node_clone);
3183     ok( r == S_OK, "ret %08x\n", r );
3184     ok( node_clone != NULL, "node %p\n", node );
3185
3186     if(!node_clone)
3187     {
3188         IXMLDOMDocument_Release(doc);
3189         IXMLDOMNode_Release(node);
3190         return;
3191     }
3192
3193     r = IXMLDOMNode_get_firstChild(node_clone, &node_first);
3194     ok( r == S_OK, "ret %08x\n", r );
3195     if(r == S_OK)
3196     {
3197         IXMLDOMDocument *doc2;
3198
3199         r = IXMLDOMNode_get_ownerDocument(node_clone, &doc2);
3200         ok( r == S_OK, "ret %08x\n", r );
3201         if(r == S_OK)
3202             IXMLDOMDocument_Release(doc2);
3203
3204         IXMLDOMNode_Release(node_first);
3205     }
3206
3207     r = IXMLDOMNode_get_childNodes(node, &pList);
3208     ok( r == S_OK, "ret %08x\n", r );
3209     if (pList)
3210         {
3211                 IXMLDOMNodeList_get_length(pList, &nLength);
3212                 IXMLDOMNodeList_Release(pList);
3213         }
3214
3215     r = IXMLDOMNode_get_attributes(node, &mapAttr);
3216     ok( r == S_OK, "ret %08x\n", r );
3217     if(mapAttr)
3218     {
3219         IXMLDOMNamedNodeMap_get_length(mapAttr, &nAttrCnt);
3220         IXMLDOMNamedNodeMap_Release(mapAttr);
3221     }
3222
3223     r = IXMLDOMNode_get_childNodes(node_clone, &pList);
3224     ok( r == S_OK, "ret %08x\n", r );
3225     if (pList)
3226         {
3227                 IXMLDOMNodeList_get_length(pList, &nLength1);
3228                 IXMLDOMNodeList_Release(pList);
3229         }
3230
3231     r = IXMLDOMNode_get_attributes(node_clone, &mapAttr);
3232     ok( r == S_OK, "ret %08x\n", r );
3233     if(mapAttr)
3234     {
3235         IXMLDOMNamedNodeMap_get_length(mapAttr, &nAttrCnt1);
3236         IXMLDOMNamedNodeMap_Release(mapAttr);
3237     }
3238
3239     ok(nLength == nLength1, "wrong Child count (%d, %d)\n", nLength, nLength1);
3240     ok(nAttrCnt == nAttrCnt1, "wrong Attribute count (%d, %d)\n", nAttrCnt, nAttrCnt1);
3241     IXMLDOMNode_Release(node_clone);
3242
3243     /* No Children */
3244     r = IXMLDOMNode_cloneNode(node, VARIANT_FALSE, &node_clone);
3245     ok( r == S_OK, "ret %08x\n", r );
3246     ok( node_clone != NULL, "node %p\n", node );
3247
3248     if(!node_clone)
3249     {
3250         IXMLDOMDocument_Release(doc);
3251         IXMLDOMNode_Release(node);
3252         return;
3253     }
3254
3255     r = IXMLDOMNode_get_firstChild(node_clone, &node_first);
3256     ok( r == S_FALSE, "ret %08x\n", r );
3257     if(r == S_OK)
3258     {
3259         IXMLDOMDocument *doc2;
3260
3261         r = IXMLDOMNode_get_ownerDocument(node_clone, &doc2);
3262         ok( r == S_OK, "ret %08x\n", r );
3263         if(r == S_OK)
3264             IXMLDOMDocument_Release(doc2);
3265
3266         IXMLDOMNode_Release(node_first);
3267     }
3268
3269     r = IXMLDOMNode_get_childNodes(node_clone, &pList);
3270     ok( r == S_OK, "ret %08x\n", r );
3271     if (pList)
3272     {
3273         IXMLDOMNodeList_get_length(pList, &nLength1);
3274         ok( nLength1 == 0, "Length should be 0 (%d)\n", nLength1);
3275         IXMLDOMNodeList_Release(pList);
3276     }
3277
3278     r = IXMLDOMNode_get_attributes(node_clone, &mapAttr);
3279     ok( r == S_OK, "ret %08x\n", r );
3280     if(mapAttr)
3281     {
3282         IXMLDOMNamedNodeMap_get_length(mapAttr, &nAttrCnt1);
3283         ok( nAttrCnt1 == 3, "Attribute count should be 3 (%d)\n", nAttrCnt1);
3284         IXMLDOMNamedNodeMap_Release(mapAttr);
3285     }
3286
3287     ok(nLength != nLength1, "wrong Child count (%d, %d)\n", nLength, nLength1);
3288     ok(nAttrCnt == nAttrCnt1, "wrong Attribute count (%d, %d)\n", nAttrCnt, nAttrCnt1);
3289     IXMLDOMNode_Release(node_clone);
3290
3291
3292     IXMLDOMNode_Release(node);
3293     IXMLDOMDocument_Release(doc);
3294 }
3295
3296 static void test_xmlTypes(void)
3297 {
3298     IXMLDOMDocument *doc;
3299     IXMLDOMElement *pRoot;
3300     HRESULT hr;
3301     IXMLDOMComment *pComment;
3302     IXMLDOMElement *pElement;
3303     IXMLDOMAttribute *pAttribute;
3304     IXMLDOMNamedNodeMap *pAttribs;
3305     IXMLDOMCDATASection *pCDataSec;
3306     IXMLDOMImplementation *pIXMLDOMImplementation = NULL;
3307     IXMLDOMDocumentFragment *pDocFrag = NULL;
3308     IXMLDOMEntityReference *pEntityRef = NULL;
3309     BSTR str;
3310     IXMLDOMNode *pNextChild;
3311     VARIANT v;
3312     LONG len = 0;
3313
3314     doc = create_document(&IID_IXMLDOMDocument);
3315     if (!doc) return;
3316
3317     pNextChild = (void*)0xdeadbeef;
3318     hr = IXMLDOMDocument_get_nextSibling(doc, NULL);
3319     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
3320
3321     pNextChild = (void*)0xdeadbeef;
3322     hr = IXMLDOMDocument_get_nextSibling(doc, &pNextChild);
3323     ok(hr == S_FALSE, "ret %08x\n", hr );
3324     ok(pNextChild == NULL, "pDocChild not NULL\n");
3325
3326     /* test previous Sibling */
3327     hr = IXMLDOMDocument_get_previousSibling(doc, NULL);
3328     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
3329
3330     pNextChild = (void*)0xdeadbeef;
3331     hr = IXMLDOMDocument_get_previousSibling(doc, &pNextChild);
3332     ok(hr == S_FALSE, "ret %08x\n", hr );
3333     ok(pNextChild == NULL, "pNextChild not NULL\n");
3334
3335     /* test get_attributes */
3336     hr = IXMLDOMDocument_get_attributes( doc, NULL );
3337     ok( hr == E_INVALIDARG, "get_attributes returned wrong code\n");
3338
3339     pAttribs = (void*)0xdeadbeef;
3340     hr = IXMLDOMDocument_get_attributes( doc, &pAttribs);
3341     ok(hr == S_FALSE, "ret %08x\n", hr );
3342     ok( pAttribs == NULL, "pAttribs not NULL\n");
3343
3344     /* test get_dataType */
3345     V_VT(&v) = VT_EMPTY;
3346     hr = IXMLDOMDocument_get_dataType(doc, &v);
3347     ok(hr == S_FALSE, "ret %08x\n", hr );
3348     ok( V_VT(&v) == VT_NULL, "incorrect dataType type\n");
3349     VariantClear(&v);
3350
3351     /* test nodeTypeString */
3352     str = NULL;
3353     hr = IXMLDOMDocument_get_nodeTypeString(doc, &str);
3354     ok(hr == S_OK, "ret %08x\n", hr );
3355     ok( !lstrcmpW( str, _bstr_("document") ), "incorrect nodeTypeString string\n");
3356     SysFreeString(str);
3357
3358     /* test implementation */
3359     hr = IXMLDOMDocument_get_implementation(doc, NULL);
3360     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
3361
3362     hr = IXMLDOMDocument_get_implementation(doc, &pIXMLDOMImplementation);
3363     ok(hr == S_OK, "ret %08x\n", hr );
3364     if(hr == S_OK)
3365     {
3366         VARIANT_BOOL hasFeature = VARIANT_TRUE;
3367         BSTR sEmpty = SysAllocStringLen(NULL, 0);
3368
3369         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, NULL, sEmpty, &hasFeature);
3370         ok(hr == E_INVALIDARG, "ret %08x\n", hr );
3371
3372         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, sEmpty, sEmpty, NULL);
3373         ok(hr == E_INVALIDARG, "ret %08x\n", hr );
3374
3375         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, _bstr_("DOM"), sEmpty, &hasFeature);
3376         ok(hr == S_OK, "ret %08x\n", hr );
3377         ok(hasFeature == VARIANT_FALSE, "hasFeature returned false\n");
3378
3379         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, sEmpty, sEmpty, &hasFeature);
3380         ok(hr == S_OK, "ret %08x\n", hr );
3381         ok(hasFeature == VARIANT_FALSE, "hasFeature returned true\n");
3382
3383         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, _bstr_("DOM"), NULL, &hasFeature);
3384         ok(hr == S_OK, "ret %08x\n", hr );
3385         ok(hasFeature == VARIANT_TRUE, "hasFeature returned false\n");
3386
3387         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, _bstr_("DOM"), sEmpty, &hasFeature);
3388         ok(hr == S_OK, "ret %08x\n", hr );
3389         ok(hasFeature == VARIANT_FALSE, "hasFeature returned false\n");
3390
3391         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, _bstr_("DOM"), _bstr_("1.0"), &hasFeature);
3392         ok(hr == S_OK, "ret %08x\n", hr );
3393         ok(hasFeature == VARIANT_TRUE, "hasFeature returned true\n");
3394
3395         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, _bstr_("XML"), _bstr_("1.0"), &hasFeature);
3396         ok(hr == S_OK, "ret %08x\n", hr );
3397         ok(hasFeature == VARIANT_TRUE, "hasFeature returned true\n");
3398
3399         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, _bstr_("MS-DOM"), _bstr_("1.0"), &hasFeature);
3400         ok(hr == S_OK, "ret %08x\n", hr );
3401         ok(hasFeature == VARIANT_TRUE, "hasFeature returned true\n");
3402
3403         hr = IXMLDOMImplementation_hasFeature(pIXMLDOMImplementation, _bstr_("SSS"), NULL, &hasFeature);
3404         ok(hr == S_OK, "ret %08x\n", hr );
3405         ok(hasFeature == VARIANT_FALSE, "hasFeature returned false\n");
3406
3407         SysFreeString(sEmpty);
3408         IXMLDOMImplementation_Release(pIXMLDOMImplementation);
3409     }
3410
3411     pRoot = (IXMLDOMElement*)0x1;
3412     hr = IXMLDOMDocument_createElement(doc, NULL, &pRoot);
3413     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
3414     ok(pRoot == (void*)0x1, "Expect same ptr, got %p\n", pRoot);
3415
3416     pRoot = (IXMLDOMElement*)0x1;
3417     hr = IXMLDOMDocument_createElement(doc, _bstr_(""), &pRoot);
3418     ok(hr == E_FAIL, "ret %08x\n", hr );
3419     ok(pRoot == (void*)0x1, "Expect same ptr, got %p\n", pRoot);
3420
3421     hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing"), &pRoot);
3422     ok(hr == S_OK, "ret %08x\n", hr );
3423     if(hr == S_OK)
3424     {
3425         hr = IXMLDOMDocument_appendChild(doc, (IXMLDOMNode*)pRoot, NULL);
3426         ok(hr == S_OK, "ret %08x\n", hr );
3427         if(hr == S_OK)
3428         {
3429             /* Comment */
3430             str = SysAllocString(szComment);
3431             hr = IXMLDOMDocument_createComment(doc, str, &pComment);
3432             SysFreeString(str);
3433             ok(hr == S_OK, "ret %08x\n", hr );
3434             if(hr == S_OK)
3435             {
3436                 /* test get_attributes */
3437                 hr = IXMLDOMComment_get_attributes( pComment, NULL );
3438                 ok( hr == E_INVALIDARG, "get_attributes returned wrong code\n");
3439
3440                 pAttribs = (IXMLDOMNamedNodeMap*)0x1;
3441                 hr = IXMLDOMComment_get_attributes( pComment, &pAttribs);
3442                 ok(hr == S_FALSE, "ret %08x\n", hr );
3443                 ok( pAttribs == NULL, "pAttribs not NULL\n");
3444
3445                 /* test nodeTypeString */
3446                 hr = IXMLDOMComment_get_nodeTypeString(pComment, &str);
3447                 ok(hr == S_OK, "ret %08x\n", hr );
3448                 ok( !lstrcmpW( str, _bstr_("comment") ), "incorrect nodeTypeString string\n");
3449                 SysFreeString(str);
3450
3451                 hr = IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pComment, NULL);
3452                 ok(hr == S_OK, "ret %08x\n", hr );
3453
3454                 hr = IXMLDOMComment_get_nodeName(pComment, &str);
3455                 ok(hr == S_OK, "ret %08x\n", hr );
3456                 ok( !lstrcmpW( str, szCommentNodeText ), "incorrect comment node Name\n");
3457                 SysFreeString(str);
3458
3459                 hr = IXMLDOMComment_get_xml(pComment, &str);
3460                 ok(hr == S_OK, "ret %08x\n", hr );
3461                 ok( !lstrcmpW( str, szCommentXML ), "incorrect comment xml\n");
3462                 SysFreeString(str);
3463
3464                 hr = IXMLDOMComment_get_dataType(pComment, &v);
3465                 ok(hr == S_FALSE, "ret %08x\n", hr );
3466                 ok( V_VT(&v) == VT_NULL, "incorrect dataType type\n");
3467                 VariantClear(&v);
3468
3469                 /* put data Tests */
3470                 hr = IXMLDOMComment_put_data(pComment, _bstr_("This &is a ; test <>\\"));
3471                 ok(hr == S_OK, "ret %08x\n", hr );
3472
3473                 /* get data Tests */
3474                 hr = IXMLDOMComment_get_data(pComment, &str);
3475                 ok(hr == S_OK, "ret %08x\n", hr );
3476                 ok( !lstrcmpW( str, _bstr_("This &is a ; test <>\\") ), "incorrect get_data string\n");
3477                 SysFreeString(str);
3478
3479                 /* get data Tests */
3480                 hr = IXMLDOMComment_get_nodeValue(pComment, &v);
3481                 ok(hr == S_OK, "ret %08x\n", hr );
3482                 ok( V_VT(&v) == VT_BSTR, "incorrect dataType type\n");
3483                 ok( !lstrcmpW( V_BSTR(&v), _bstr_("This &is a ; test <>\\") ), "incorrect get_nodeValue string\n");
3484                 VariantClear(&v);
3485
3486                 /* Confirm XML text is good */
3487                 hr = IXMLDOMComment_get_xml(pComment, &str);
3488                 ok(hr == S_OK, "ret %08x\n", hr );
3489                 ok( !lstrcmpW( str, _bstr_("<!--This &is a ; test <>\\-->") ), "incorrect xml string\n");
3490                 SysFreeString(str);
3491
3492                 /* Confirm we get the put_data Text back */
3493                 hr = IXMLDOMComment_get_text(pComment, &str);
3494                 ok(hr == S_OK, "ret %08x\n", hr );
3495                 ok( !lstrcmpW( str, _bstr_("This &is a ; test <>\\") ), "incorrect xml string\n");
3496                 SysFreeString(str);
3497
3498                 /* test length property */
3499                 hr = IXMLDOMComment_get_length(pComment, &len);
3500                 ok(hr == S_OK, "ret %08x\n", hr );
3501                 ok(len == 21, "expected 21 got %d\n", len);
3502
3503                 /* test substringData */
3504                 hr = IXMLDOMComment_substringData(pComment, 0, 4, NULL);
3505                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
3506
3507                 /* test substringData - Invalid offset */
3508                 str = (BSTR)&szElement;
3509                 hr = IXMLDOMComment_substringData(pComment, -1, 4, &str);
3510                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
3511                 ok( str == NULL, "incorrect string\n");
3512
3513                 /* test substringData - Invalid offset */
3514                 str = (BSTR)&szElement;
3515                 hr = IXMLDOMComment_substringData(pComment, 30, 0, &str);
3516                 ok(hr == S_FALSE, "ret %08x\n", hr );
3517                 ok( str == NULL, "incorrect string\n");
3518
3519                 /* test substringData - Invalid size */
3520                 str = (BSTR)&szElement;
3521                 hr = IXMLDOMComment_substringData(pComment, 0, -1, &str);
3522                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
3523                 ok( str == NULL, "incorrect string\n");
3524
3525                 /* test substringData - Invalid size */
3526                 str = (BSTR)&szElement;
3527                 hr = IXMLDOMComment_substringData(pComment, 2, 0, &str);
3528                 ok(hr == S_FALSE, "ret %08x\n", hr );
3529                 ok( str == NULL, "incorrect string\n");
3530
3531                 /* test substringData - Start of string */
3532                 hr = IXMLDOMComment_substringData(pComment, 0, 4, &str);
3533                 ok(hr == S_OK, "ret %08x\n", hr );
3534                 ok( !lstrcmpW( str, _bstr_("This") ), "incorrect substringData string\n");
3535                 SysFreeString(str);
3536
3537                 /* test substringData - Middle of string */
3538                 hr = IXMLDOMComment_substringData(pComment, 13, 4, &str);
3539                 ok(hr == S_OK, "ret %08x\n", hr );
3540                 ok( !lstrcmpW( str, _bstr_("test") ), "incorrect substringData string\n");
3541                 SysFreeString(str);
3542
3543                 /* test substringData - End of string */
3544                 hr = IXMLDOMComment_substringData(pComment, 20, 4, &str);
3545                 ok(hr == S_OK, "ret %08x\n", hr );
3546                 ok( !lstrcmpW( str, _bstr_("\\") ), "incorrect substringData string\n");
3547                 SysFreeString(str);
3548
3549                 /* test appendData */
3550                 hr = IXMLDOMComment_appendData(pComment, NULL);
3551                 ok(hr == S_OK, "ret %08x\n", hr );
3552
3553                 hr = IXMLDOMComment_appendData(pComment, _bstr_(""));
3554                 ok(hr == S_OK, "ret %08x\n", hr );
3555
3556                 hr = IXMLDOMComment_appendData(pComment, _bstr_("Append"));
3557                 ok(hr == S_OK, "ret %08x\n", hr );
3558
3559                 hr = IXMLDOMComment_get_text(pComment, &str);
3560                 ok(hr == S_OK, "ret %08x\n", hr );
3561                 ok( !lstrcmpW( str, _bstr_("This &is a ; test <>\\Append") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
3562                 SysFreeString(str);
3563
3564                 /* test insertData */
3565                 str = SysAllocStringLen(NULL, 0);
3566                 hr = IXMLDOMComment_insertData(pComment, -1, str);
3567                 ok(hr == S_OK, "ret %08x\n", hr );
3568
3569                 hr = IXMLDOMComment_insertData(pComment, -1, NULL);
3570                 ok(hr == S_OK, "ret %08x\n", hr );
3571
3572                 hr = IXMLDOMComment_insertData(pComment, 1000, str);
3573                 ok(hr == S_OK, "ret %08x\n", hr );
3574
3575                 hr = IXMLDOMComment_insertData(pComment, 1000, NULL);
3576                 ok(hr == S_OK, "ret %08x\n", hr );
3577
3578                 hr = IXMLDOMComment_insertData(pComment, 0, NULL);
3579                 ok(hr == S_OK, "ret %08x\n", hr );
3580
3581                 hr = IXMLDOMComment_insertData(pComment, 0, str);
3582                 ok(hr == S_OK, "ret %08x\n", hr );
3583                 SysFreeString(str);
3584
3585                 hr = IXMLDOMComment_insertData(pComment, -1, _bstr_("Inserting"));
3586                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
3587
3588                 hr = IXMLDOMComment_insertData(pComment, 1000, _bstr_("Inserting"));
3589                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
3590
3591                 hr = IXMLDOMComment_insertData(pComment, 0, _bstr_("Begin "));
3592                 ok(hr == S_OK, "ret %08x\n", hr );
3593
3594                 hr = IXMLDOMComment_insertData(pComment, 17, _bstr_("Middle"));
3595                 ok(hr == S_OK, "ret %08x\n", hr );
3596
3597                 hr = IXMLDOMComment_insertData(pComment, 39, _bstr_(" End"));
3598                 ok(hr == S_OK, "ret %08x\n", hr );
3599
3600                 hr = IXMLDOMComment_get_text(pComment, &str);
3601                 ok(hr == S_OK, "ret %08x\n", hr );
3602                 ok( !lstrcmpW( str, _bstr_("Begin This &is a Middle; test <>\\Append End") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
3603                 SysFreeString(str);
3604
3605                 /* delete data */
3606                 /* invalid arguments */
3607                 hr = IXMLDOMComment_deleteData(pComment, -1, 1);
3608                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
3609
3610                 hr = IXMLDOMComment_deleteData(pComment, 0, 0);
3611                 ok(hr == S_OK, "ret %08x\n", hr );
3612
3613                 hr = IXMLDOMComment_deleteData(pComment, 0, -1);
3614                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
3615
3616                 hr = IXMLDOMComment_get_length(pComment, &len);
3617                 ok(hr == S_OK, "ret %08x\n", hr );
3618                 ok(len == 43, "expected 43 got %d\n", len);
3619
3620                 hr = IXMLDOMComment_deleteData(pComment, len, 1);
3621                 ok(hr == S_OK, "ret %08x\n", hr );
3622
3623                 hr = IXMLDOMComment_deleteData(pComment, len+1, 1);
3624                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
3625
3626                 /* delete from start */
3627                 hr = IXMLDOMComment_deleteData(pComment, 0, 5);
3628                 ok(hr == S_OK, "ret %08x\n", hr );
3629
3630                 hr = IXMLDOMComment_get_length(pComment, &len);
3631                 ok(hr == S_OK, "ret %08x\n", hr );
3632                 ok(len == 38, "expected 38 got %d\n", len);
3633
3634                 hr = IXMLDOMComment_get_text(pComment, &str);
3635                 ok(hr == S_OK, "ret %08x\n", hr );
3636                 ok( !lstrcmpW( str, _bstr_(" This &is a Middle; test <>\\Append End") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
3637                 SysFreeString(str);
3638
3639                 /* delete from end */
3640                 hr = IXMLDOMComment_deleteData(pComment, 35, 3);
3641                 ok(hr == S_OK, "ret %08x\n", hr );
3642
3643                 hr = IXMLDOMComment_get_length(pComment, &len);
3644                 ok(hr == S_OK, "ret %08x\n", hr );
3645                 ok(len == 35, "expected 35 got %d\n", len);
3646
3647                 hr = IXMLDOMComment_get_text(pComment, &str);
3648                 ok(hr == S_OK, "ret %08x\n", hr );
3649                 ok( !lstrcmpW( str, _bstr_(" This &is a Middle; test <>\\Append ") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
3650                 SysFreeString(str);
3651
3652                 /* delete from inside */
3653                 hr = IXMLDOMComment_deleteData(pComment, 1, 33);
3654                 ok(hr == S_OK, "ret %08x\n", hr );
3655
3656                 hr = IXMLDOMComment_get_length(pComment, &len);
3657                 ok(hr == S_OK, "ret %08x\n", hr );
3658                 ok(len == 2, "expected 2 got %d\n", len);
3659
3660                 hr = IXMLDOMComment_get_text(pComment, &str);
3661                 ok(hr == S_OK, "ret %08x\n", hr );
3662                 ok( !lstrcmpW( str, _bstr_("  ") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
3663                 SysFreeString(str);
3664
3665                 /* delete whole data ... */
3666                 hr = IXMLDOMComment_get_length(pComment, &len);
3667                 ok(hr == S_OK, "ret %08x\n", hr );
3668
3669                 hr = IXMLDOMComment_deleteData(pComment, 0, len);
3670                 ok(hr == S_OK, "ret %08x\n", hr );
3671                 /* ... and try again with empty string */
3672                 hr = IXMLDOMComment_deleteData(pComment, 0, len);
3673                 ok(hr == S_OK, "ret %08x\n", hr );
3674
3675                 /* ::replaceData() */
3676                 V_VT(&v) = VT_BSTR;
3677                 V_BSTR(&v) = SysAllocString(szstr1);
3678                 hr = IXMLDOMComment_put_nodeValue(pComment, v);
3679                 ok(hr == S_OK, "ret %08x\n", hr );
3680                 VariantClear(&v);
3681
3682                 hr = IXMLDOMComment_replaceData(pComment, 6, 0, NULL);
3683                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
3684                 hr = IXMLDOMComment_get_text(pComment, &str);
3685                 ok(hr == S_OK, "ret %08x\n", hr );
3686                 ok( !lstrcmpW( str, _bstr_("str1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
3687                 SysFreeString(str);
3688
3689                 hr = IXMLDOMComment_replaceData(pComment, 0, 0, NULL);
3690                 ok(hr == S_OK, "ret %08x\n", hr );
3691                 hr = IXMLDOMComment_get_text(pComment, &str);
3692                 ok(hr == S_OK, "ret %08x\n", hr );
3693                 ok( !lstrcmpW( str, _bstr_("str1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
3694                 SysFreeString(str);
3695
3696                 /* NULL pointer means delete */
3697                 hr = IXMLDOMComment_replaceData(pComment, 0, 1, NULL);
3698                 ok(hr == S_OK, "ret %08x\n", hr );
3699                 hr = IXMLDOMComment_get_text(pComment, &str);
3700                 ok(hr == S_OK, "ret %08x\n", hr );
3701                 ok( !lstrcmpW( str, _bstr_("tr1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
3702                 SysFreeString(str);
3703
3704                 /* empty string means delete */
3705                 hr = IXMLDOMComment_replaceData(pComment, 0, 1, _bstr_(""));
3706                 ok(hr == S_OK, "ret %08x\n", hr );
3707                 hr = IXMLDOMComment_get_text(pComment, &str);
3708                 ok(hr == S_OK, "ret %08x\n", hr );
3709                 ok( !lstrcmpW( str, _bstr_("r1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
3710                 SysFreeString(str);
3711
3712                 /* zero count means insert */
3713                 hr = IXMLDOMComment_replaceData(pComment, 0, 0, _bstr_("a"));
3714                 ok(hr == S_OK, "ret %08x\n", hr );
3715                 hr = IXMLDOMComment_get_text(pComment, &str);
3716                 ok(hr == S_OK, "ret %08x\n", hr );
3717                 ok( !lstrcmpW( str, _bstr_("ar1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
3718                 SysFreeString(str);
3719
3720                 hr = IXMLDOMComment_replaceData(pComment, 0, 2, NULL);
3721                 ok(hr == S_OK, "ret %08x\n", hr );
3722
3723                 hr = IXMLDOMComment_insertData(pComment, 0, _bstr_("m"));
3724                 ok(hr == S_OK, "ret %08x\n", hr );
3725                 hr = IXMLDOMComment_get_text(pComment, &str);
3726                 ok(hr == S_OK, "ret %08x\n", hr );
3727                 ok( !lstrcmpW( str, _bstr_("m1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
3728                 SysFreeString(str);
3729
3730                 /* nonempty string, count greater than its length */
3731                 hr = IXMLDOMComment_replaceData(pComment, 0, 2, _bstr_("a1.2"));
3732                 ok(hr == S_OK, "ret %08x\n", hr );
3733                 hr = IXMLDOMComment_get_text(pComment, &str);
3734                 ok(hr == S_OK, "ret %08x\n", hr );
3735                 ok( !lstrcmpW( str, _bstr_("a1.2") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
3736                 SysFreeString(str);
3737
3738                 /* nonempty string, count less than its length */
3739                 hr = IXMLDOMComment_replaceData(pComment, 0, 1, _bstr_("wine"));
3740                 ok(hr == S_OK, "ret %08x\n", hr );
3741                 hr = IXMLDOMComment_get_text(pComment, &str);
3742                 ok(hr == S_OK, "ret %08x\n", hr );
3743                 ok( !lstrcmpW( str, _bstr_("wine1.2") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
3744                 SysFreeString(str);
3745
3746                 IXMLDOMComment_Release(pComment);
3747             }
3748
3749             /* Element */
3750             str = SysAllocString(szElement);
3751             hr = IXMLDOMDocument_createElement(doc, str, &pElement);
3752             SysFreeString(str);
3753             ok(hr == S_OK, "ret %08x\n", hr );
3754             if(hr == S_OK)
3755             {
3756                 hr = IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
3757                 ok(hr == S_OK, "ret %08x\n", hr );
3758
3759                 /* test nodeTypeString */
3760                 hr = IXMLDOMDocument_get_nodeTypeString(pElement, &str);
3761                 ok(hr == S_OK, "ret %08x\n", hr );
3762                 ok( !lstrcmpW( str, _bstr_("element") ), "incorrect nodeTypeString string\n");
3763                 SysFreeString(str);
3764
3765                 hr = IXMLDOMElement_get_nodeName(pElement, &str);
3766                 ok(hr == S_OK, "ret %08x\n", hr );
3767                 ok( !lstrcmpW( str, szElement ), "incorrect element node Name\n");
3768                 SysFreeString(str);
3769
3770                 hr = IXMLDOMElement_get_xml(pElement, &str);
3771                 ok(hr == S_OK, "ret %08x\n", hr );
3772                 ok( !lstrcmpW( str, szElementXML ), "incorrect element xml\n");
3773                 SysFreeString(str);
3774
3775                 hr = IXMLDOMElement_get_dataType(pElement, &v);
3776                 ok(hr == S_FALSE, "ret %08x\n", hr );
3777                 ok( V_VT(&v) == VT_NULL, "incorrect dataType type\n");
3778                 VariantClear(&v);
3779
3780                 /* Attribute */
3781                 pAttribute = (IXMLDOMAttribute*)0x1;
3782                 hr = IXMLDOMDocument_createAttribute(doc, NULL, &pAttribute);
3783                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
3784                 ok(pAttribute == (void*)0x1, "Expect same ptr, got %p\n", pAttribute);
3785
3786                 pAttribute = (IXMLDOMAttribute*)0x1;
3787                 hr = IXMLDOMDocument_createAttribute(doc, _bstr_(""), &pAttribute);
3788                 ok(hr == E_FAIL, "ret %08x\n", hr );
3789                 ok(pAttribute == (void*)0x1, "Expect same ptr, got %p\n", pAttribute);
3790
3791                 str = SysAllocString(szAttribute);
3792                 hr = IXMLDOMDocument_createAttribute(doc, str, &pAttribute);
3793                 SysFreeString(str);
3794                 ok(hr == S_OK, "ret %08x\n", hr );
3795                 if(hr == S_OK)
3796                 {
3797                     IXMLDOMNode *pNewChild = (IXMLDOMNode *)0x1;
3798
3799                     hr = IXMLDOMAttribute_get_nextSibling(pAttribute, NULL);
3800                     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
3801
3802                     pNextChild = (IXMLDOMNode *)0x1;
3803                     hr = IXMLDOMAttribute_get_nextSibling(pAttribute, &pNextChild);
3804                     ok(hr == S_FALSE, "ret %08x\n", hr );
3805                     ok(pNextChild == NULL, "pNextChild not NULL\n");
3806
3807                     /* test Previous Sibling*/
3808                     hr = IXMLDOMAttribute_get_previousSibling(pAttribute, NULL);
3809                     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
3810
3811                     pNextChild = (IXMLDOMNode *)0x1;
3812                     hr = IXMLDOMAttribute_get_previousSibling(pAttribute, &pNextChild);
3813                     ok(hr == S_FALSE, "ret %08x\n", hr );
3814                     ok(pNextChild == NULL, "pNextChild not NULL\n");
3815
3816                     /* test get_attributes */
3817                     hr = IXMLDOMAttribute_get_attributes( pAttribute, NULL );
3818                     ok( hr == E_INVALIDARG, "get_attributes returned wrong code\n");
3819
3820                     pAttribs = (IXMLDOMNamedNodeMap*)0x1;
3821                     hr = IXMLDOMAttribute_get_attributes( pAttribute, &pAttribs);
3822                     ok(hr == S_FALSE, "ret %08x\n", hr );
3823                     ok( pAttribs == NULL, "pAttribs not NULL\n");
3824
3825                     hr = IXMLDOMElement_appendChild(pElement, (IXMLDOMNode*)pAttribute, &pNewChild);
3826                     ok(hr == E_FAIL, "ret %08x\n", hr );
3827                     ok(pNewChild == NULL, "pNewChild not NULL\n");
3828
3829                     hr = IXMLDOMElement_get_attributes(pElement, &pAttribs);
3830                     ok(hr == S_OK, "ret %08x\n", hr );
3831                     if ( hr == S_OK )
3832                     {
3833                         hr = IXMLDOMNamedNodeMap_setNamedItem(pAttribs, (IXMLDOMNode*)pAttribute, NULL );
3834                         ok(hr == S_OK, "ret %08x\n", hr );
3835
3836                         IXMLDOMNamedNodeMap_Release(pAttribs);
3837                     }
3838
3839                     hr = IXMLDOMAttribute_get_nodeName(pAttribute, &str);
3840                     ok(hr == S_OK, "ret %08x\n", hr );
3841                     ok( !lstrcmpW( str, szAttribute ), "incorrect attribute node Name\n");
3842                     SysFreeString(str);
3843
3844                     /* test nodeTypeString */
3845                     hr = IXMLDOMAttribute_get_nodeTypeString(pAttribute, &str);
3846                     ok(hr == S_OK, "ret %08x\n", hr );
3847                     ok( !lstrcmpW( str, _bstr_("attribute") ), "incorrect nodeTypeString string\n");
3848                     SysFreeString(str);
3849
3850                     /* test nodeName */
3851                     hr = IXMLDOMAttribute_get_nodeName(pAttribute, &str);
3852                     ok(hr == S_OK, "ret %08x\n", hr );
3853                     ok( !lstrcmpW( str, szAttribute ), "incorrect nodeName string\n");
3854                     SysFreeString(str);
3855
3856                     /* test name property */
3857                     hr = IXMLDOMAttribute_get_name(pAttribute, &str);
3858                     ok(hr == S_OK, "ret %08x\n", hr );
3859                     ok( !lstrcmpW( str, szAttribute ), "incorrect name string\n");
3860                     SysFreeString(str);
3861
3862                     hr = IXMLDOMAttribute_get_xml(pAttribute, &str);
3863                     ok(hr == S_OK, "ret %08x\n", hr );
3864                     ok( !lstrcmpW( str, szAttributeXML ), "incorrect attribute xml\n");
3865                     SysFreeString(str);
3866
3867                     hr = IXMLDOMAttribute_get_dataType(pAttribute, &v);
3868                     ok(hr == S_FALSE, "ret %08x\n", hr );
3869                     ok( V_VT(&v) == VT_NULL, "incorrect dataType type\n");
3870                     VariantClear(&v);
3871
3872                     IXMLDOMAttribute_Release(pAttribute);
3873
3874                     /* Check Element again with the Add Attribute*/
3875                     hr = IXMLDOMElement_get_xml(pElement, &str);
3876                     ok(hr == S_OK, "ret %08x\n", hr );
3877                     ok( !lstrcmpW( str, szElementXML2 ), "incorrect element xml\n");
3878                     SysFreeString(str);
3879                 }
3880
3881                 hr = IXMLDOMElement_put_text(pElement, _bstr_("TestingNode"));
3882                 ok(hr == S_OK, "ret %08x\n", hr );
3883
3884                 hr = IXMLDOMElement_get_xml(pElement, &str);
3885                 ok(hr == S_OK, "ret %08x\n", hr );
3886                 ok( !lstrcmpW( str, szElementXML3 ), "incorrect element xml\n");
3887                 SysFreeString(str);
3888
3889                 /* Test for reversible escaping */
3890                 str = SysAllocString( szStrangeChars );
3891                 hr = IXMLDOMElement_put_text(pElement, str);
3892                 ok(hr == S_OK, "ret %08x\n", hr );
3893                 SysFreeString( str );
3894
3895                 hr = IXMLDOMElement_get_xml(pElement, &str);
3896                 ok(hr == S_OK, "ret %08x\n", hr );
3897                 ok( !lstrcmpW( str, szElementXML4 ), "incorrect element xml\n");
3898                 SysFreeString(str);
3899
3900                 hr = IXMLDOMElement_get_text(pElement, &str);
3901                 ok(hr == S_OK, "ret %08x\n", hr );
3902                 ok( !lstrcmpW( str, szStrangeChars ), "incorrect element text\n");
3903                 SysFreeString(str);
3904
3905                 IXMLDOMElement_Release(pElement);
3906             }
3907
3908             /* CData Section */
3909             str = SysAllocString(szCData);
3910             hr = IXMLDOMDocument_createCDATASection(doc, str, NULL);
3911             ok(hr == E_INVALIDARG, "ret %08x\n", hr );
3912
3913             hr = IXMLDOMDocument_createCDATASection(doc, str, &pCDataSec);
3914             SysFreeString(str);
3915             ok(hr == S_OK, "ret %08x\n", hr );
3916             if(hr == S_OK)
3917             {
3918                 IXMLDOMNode *pNextChild = (IXMLDOMNode *)0x1;
3919                 VARIANT var;
3920
3921                 VariantInit(&var);
3922
3923                 hr = IXMLDOMCDATASection_QueryInterface(pCDataSec, &IID_IXMLDOMElement, (void**)&pElement);
3924                 ok(hr == E_NOINTERFACE, "ret %08x\n", hr);
3925
3926                 hr = IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pCDataSec, NULL);
3927                 ok(hr == S_OK, "ret %08x\n", hr );
3928
3929                 /* get Attribute Tests */
3930                 hr = IXMLDOMCDATASection_get_attributes(pCDataSec, NULL);
3931                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
3932
3933                 pAttribs = (IXMLDOMNamedNodeMap*)0x1;
3934                 hr = IXMLDOMCDATASection_get_attributes(pCDataSec, &pAttribs);
3935                 ok(hr == S_FALSE, "ret %08x\n", hr );
3936                 ok(pAttribs == NULL, "pAttribs != NULL\n");
3937
3938                 hr = IXMLDOMCDATASection_get_nodeName(pCDataSec, &str);
3939                 ok(hr == S_OK, "ret %08x\n", hr );
3940                 ok( !lstrcmpW( str, szCDataNodeText ), "incorrect cdata node Name\n");
3941                 SysFreeString(str);
3942
3943                 hr = IXMLDOMCDATASection_get_xml(pCDataSec, &str);
3944                 ok(hr == S_OK, "ret %08x\n", hr );
3945                 ok( !lstrcmpW( str, szCDataXML ), "incorrect cdata xml\n");
3946                 SysFreeString(str);
3947
3948                 /* test lastChild */
3949                 pNextChild = (IXMLDOMNode*)0x1;
3950                 hr = IXMLDOMCDATASection_get_lastChild(pCDataSec, &pNextChild);
3951                 ok(hr == S_FALSE, "ret %08x\n", hr );
3952                 ok(pNextChild == NULL, "pNextChild not NULL\n");
3953
3954                 /* test get_dataType */
3955                 hr = IXMLDOMCDATASection_get_dataType(pCDataSec, NULL);
3956                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
3957
3958                 hr = IXMLDOMCDATASection_get_dataType(pCDataSec, &v);
3959                 ok(hr == S_FALSE, "ret %08x\n", hr );
3960                 ok( V_VT(&v) == VT_NULL, "incorrect dataType type\n");
3961                 VariantClear(&v);
3962
3963                 /* test nodeTypeString */
3964                 hr = IXMLDOMCDATASection_get_nodeTypeString(pCDataSec, &str);
3965                 ok(hr == S_OK, "ret %08x\n", hr );
3966                 ok( !lstrcmpW( str, _bstr_("cdatasection") ), "incorrect nodeTypeString string\n");
3967                 SysFreeString(str);
3968
3969                 /* put data Tests */
3970                 hr = IXMLDOMCDATASection_put_data(pCDataSec, _bstr_("This &is a ; test <>\\"));
3971                 ok(hr == S_OK, "ret %08x\n", hr );
3972
3973                 /* Confirm XML text is good */
3974                 hr = IXMLDOMCDATASection_get_xml(pCDataSec, &str);
3975                 ok(hr == S_OK, "ret %08x\n", hr );
3976                 ok( !lstrcmpW( str, _bstr_("<![CDATA[This &is a ; test <>\\]]>") ), "incorrect xml string\n");
3977                 SysFreeString(str);
3978
3979                 /* Confirm we get the put_data Text back */
3980                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
3981                 ok(hr == S_OK, "ret %08x\n", hr );
3982                 ok( !lstrcmpW( str, _bstr_("This &is a ; test <>\\") ), "incorrect text string\n");
3983                 SysFreeString(str);
3984
3985                 /* test length property */
3986                 hr = IXMLDOMCDATASection_get_length(pCDataSec, &len);
3987                 ok(hr == S_OK, "ret %08x\n", hr );
3988                 ok(len == 21, "expected 21 got %d\n", len);
3989
3990                 /* test get nodeValue */
3991                 hr = IXMLDOMCDATASection_get_nodeValue(pCDataSec, &var);
3992                 ok(hr == S_OK, "ret %08x\n", hr );
3993                 ok(V_VT(&var) == VT_BSTR, "got vt %04x\n", V_VT(&var));
3994                 ok( !lstrcmpW( V_BSTR(&var), _bstr_("This &is a ; test <>\\") ), "incorrect text string\n");
3995                 VariantClear(&var);
3996
3997                 /* test get data */
3998                 hr = IXMLDOMCDATASection_get_data(pCDataSec, &str);
3999                 ok(hr == S_OK, "ret %08x\n", hr );
4000                 ok( !lstrcmpW( str, _bstr_("This &is a ; test <>\\") ), "incorrect text string\n");
4001                 SysFreeString(str);
4002
4003                 /* test substringData */
4004                 hr = IXMLDOMCDATASection_substringData(pCDataSec, 0, 4, NULL);
4005                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4006
4007                 /* test substringData - Invalid offset */
4008                 str = (BSTR)&szElement;
4009                 hr = IXMLDOMCDATASection_substringData(pCDataSec, -1, 4, &str);
4010                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4011                 ok( str == NULL, "incorrect string\n");
4012
4013                 /* test substringData - Invalid offset */
4014                 str = (BSTR)&szElement;
4015                 hr = IXMLDOMCDATASection_substringData(pCDataSec, 30, 0, &str);
4016                 ok(hr == S_FALSE, "ret %08x\n", hr );
4017                 ok( str == NULL, "incorrect string\n");
4018
4019                 /* test substringData - Invalid size */
4020                 str = (BSTR)&szElement;
4021                 hr = IXMLDOMCDATASection_substringData(pCDataSec, 0, -1, &str);
4022                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4023                 ok( str == NULL, "incorrect string\n");
4024
4025                 /* test substringData - Invalid size */
4026                 str = (BSTR)&szElement;
4027                 hr = IXMLDOMCDATASection_substringData(pCDataSec, 2, 0, &str);
4028                 ok(hr == S_FALSE, "ret %08x\n", hr );
4029                 ok( str == NULL, "incorrect string\n");
4030
4031                 /* test substringData - Start of string */
4032                 hr = IXMLDOMCDATASection_substringData(pCDataSec, 0, 4, &str);
4033                 ok(hr == S_OK, "ret %08x\n", hr );
4034                 ok( !lstrcmpW( str, _bstr_("This") ), "incorrect substringData string\n");
4035                 SysFreeString(str);
4036
4037                 /* test substringData - Middle of string */
4038                 hr = IXMLDOMCDATASection_substringData(pCDataSec, 13, 4, &str);
4039                 ok(hr == S_OK, "ret %08x\n", hr );
4040                 ok( !lstrcmpW( str, _bstr_("test") ), "incorrect substringData string\n");
4041                 SysFreeString(str);
4042
4043                 /* test substringData - End of string */
4044                 hr = IXMLDOMCDATASection_substringData(pCDataSec, 20, 4, &str);
4045                 ok(hr == S_OK, "ret %08x\n", hr );
4046                 ok( !lstrcmpW( str, _bstr_("\\") ), "incorrect substringData string\n");
4047                 SysFreeString(str);
4048
4049                 /* test appendData */
4050                 hr = IXMLDOMCDATASection_appendData(pCDataSec, NULL);
4051                 ok(hr == S_OK, "ret %08x\n", hr );
4052
4053                 hr = IXMLDOMCDATASection_appendData(pCDataSec, _bstr_(""));
4054                 ok(hr == S_OK, "ret %08x\n", hr );
4055
4056                 hr = IXMLDOMCDATASection_appendData(pCDataSec, _bstr_("Append"));
4057                 ok(hr == S_OK, "ret %08x\n", hr );
4058
4059                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
4060                 ok(hr == S_OK, "ret %08x\n", hr );
4061                 ok( !lstrcmpW( str, _bstr_("This &is a ; test <>\\Append") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4062                 SysFreeString(str);
4063
4064                 /* test insertData */
4065                 str = SysAllocStringLen(NULL, 0);
4066                 hr = IXMLDOMCDATASection_insertData(pCDataSec, -1, str);
4067                 ok(hr == S_OK, "ret %08x\n", hr );
4068
4069                 hr = IXMLDOMCDATASection_insertData(pCDataSec, -1, NULL);
4070                 ok(hr == S_OK, "ret %08x\n", hr );
4071
4072                 hr = IXMLDOMCDATASection_insertData(pCDataSec, 1000, str);
4073                 ok(hr == S_OK, "ret %08x\n", hr );
4074
4075                 hr = IXMLDOMCDATASection_insertData(pCDataSec, 1000, NULL);
4076                 ok(hr == S_OK, "ret %08x\n", hr );
4077
4078                 hr = IXMLDOMCDATASection_insertData(pCDataSec, 0, NULL);
4079                 ok(hr == S_OK, "ret %08x\n", hr );
4080
4081                 hr = IXMLDOMCDATASection_insertData(pCDataSec, 0, str);
4082                 ok(hr == S_OK, "ret %08x\n", hr );
4083                 SysFreeString(str);
4084
4085                 hr = IXMLDOMCDATASection_insertData(pCDataSec, -1, _bstr_("Inserting"));
4086                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4087
4088                 hr = IXMLDOMCDATASection_insertData(pCDataSec, 1000, _bstr_("Inserting"));
4089                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4090
4091                 hr = IXMLDOMCDATASection_insertData(pCDataSec, 0, _bstr_("Begin "));
4092                 ok(hr == S_OK, "ret %08x\n", hr );
4093
4094                 hr = IXMLDOMCDATASection_insertData(pCDataSec, 17, _bstr_("Middle"));
4095                 ok(hr == S_OK, "ret %08x\n", hr );
4096
4097                 hr = IXMLDOMCDATASection_insertData(pCDataSec, 39, _bstr_(" End"));
4098                 ok(hr == S_OK, "ret %08x\n", hr );
4099
4100                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
4101                 ok(hr == S_OK, "ret %08x\n", hr );
4102                 ok( !lstrcmpW( str, _bstr_("Begin This &is a Middle; test <>\\Append End") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4103                 SysFreeString(str);
4104
4105                 /* delete data */
4106                 /* invalid arguments */
4107                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, -1, 1);
4108                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4109
4110                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, 0, 0);
4111                 ok(hr == S_OK, "ret %08x\n", hr );
4112
4113                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, 0, -1);
4114                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4115
4116                 hr = IXMLDOMCDATASection_get_length(pCDataSec, &len);
4117                 ok(hr == S_OK, "ret %08x\n", hr );
4118                 ok(len == 43, "expected 43 got %d\n", len);
4119
4120                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, len, 1);
4121                 ok(hr == S_OK, "ret %08x\n", hr );
4122
4123                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, len+1, 1);
4124                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4125
4126                 /* delete from start */
4127                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, 0, 5);
4128                 ok(hr == S_OK, "ret %08x\n", hr );
4129
4130                 hr = IXMLDOMCDATASection_get_length(pCDataSec, &len);
4131                 ok(hr == S_OK, "ret %08x\n", hr );
4132                 ok(len == 38, "expected 38 got %d\n", len);
4133
4134                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
4135                 ok(hr == S_OK, "ret %08x\n", hr );
4136                 ok( !lstrcmpW( str, _bstr_(" This &is a Middle; test <>\\Append End") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4137                 SysFreeString(str);
4138
4139                 /* delete from end */
4140                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, 35, 3);
4141                 ok(hr == S_OK, "ret %08x\n", hr );
4142
4143                 hr = IXMLDOMCDATASection_get_length(pCDataSec, &len);
4144                 ok(hr == S_OK, "ret %08x\n", hr );
4145                 ok(len == 35, "expected 35 got %d\n", len);
4146
4147                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
4148                 ok(hr == S_OK, "ret %08x\n", hr );
4149                 ok( !lstrcmpW( str, _bstr_(" This &is a Middle; test <>\\Append ") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4150                 SysFreeString(str);
4151
4152                 /* delete from inside */
4153                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, 1, 33);
4154                 ok(hr == S_OK, "ret %08x\n", hr );
4155
4156                 hr = IXMLDOMCDATASection_get_length(pCDataSec, &len);
4157                 ok(hr == S_OK, "ret %08x\n", hr );
4158                 ok(len == 2, "expected 2 got %d\n", len);
4159
4160                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
4161                 ok(hr == S_OK, "ret %08x\n", hr );
4162                 ok( !lstrcmpW( str, _bstr_("  ") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4163                 SysFreeString(str);
4164
4165                 /* delete whole data ... */
4166                 hr = IXMLDOMCDATASection_get_length(pCDataSec, &len);
4167                 ok(hr == S_OK, "ret %08x\n", hr );
4168
4169                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, 0, len);
4170                 ok(hr == S_OK, "ret %08x\n", hr );
4171
4172                 /* ... and try again with empty string */
4173                 hr = IXMLDOMCDATASection_deleteData(pCDataSec, 0, len);
4174                 ok(hr == S_OK, "ret %08x\n", hr );
4175
4176                 /* ::replaceData() */
4177                 V_VT(&v) = VT_BSTR;
4178                 V_BSTR(&v) = SysAllocString(szstr1);
4179                 hr = IXMLDOMCDATASection_put_nodeValue(pCDataSec, v);
4180                 ok(hr == S_OK, "ret %08x\n", hr );
4181                 VariantClear(&v);
4182
4183                 hr = IXMLDOMCDATASection_replaceData(pCDataSec, 6, 0, NULL);
4184                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4185                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
4186                 ok(hr == S_OK, "ret %08x\n", hr );
4187                 ok( !lstrcmpW( str, _bstr_("str1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4188                 SysFreeString(str);
4189
4190                 hr = IXMLDOMCDATASection_replaceData(pCDataSec, 0, 0, NULL);
4191                 ok(hr == S_OK, "ret %08x\n", hr );
4192                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
4193                 ok(hr == S_OK, "ret %08x\n", hr );
4194                 ok( !lstrcmpW( str, _bstr_("str1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4195                 SysFreeString(str);
4196
4197                 /* NULL pointer means delete */
4198                 hr = IXMLDOMCDATASection_replaceData(pCDataSec, 0, 1, NULL);
4199                 ok(hr == S_OK, "ret %08x\n", hr );
4200                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
4201                 ok(hr == S_OK, "ret %08x\n", hr );
4202                 ok( !lstrcmpW( str, _bstr_("tr1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4203                 SysFreeString(str);
4204
4205                 /* empty string means delete */
4206                 hr = IXMLDOMCDATASection_replaceData(pCDataSec, 0, 1, _bstr_(""));
4207                 ok(hr == S_OK, "ret %08x\n", hr );
4208                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
4209                 ok(hr == S_OK, "ret %08x\n", hr );
4210                 ok( !lstrcmpW( str, _bstr_("r1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4211                 SysFreeString(str);
4212
4213                 /* zero count means insert */
4214                 hr = IXMLDOMCDATASection_replaceData(pCDataSec, 0, 0, _bstr_("a"));
4215                 ok(hr == S_OK, "ret %08x\n", hr );
4216                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
4217                 ok(hr == S_OK, "ret %08x\n", hr );
4218                 ok( !lstrcmpW( str, _bstr_("ar1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4219                 SysFreeString(str);
4220
4221                 hr = IXMLDOMCDATASection_replaceData(pCDataSec, 0, 2, NULL);
4222                 ok(hr == S_OK, "ret %08x\n", hr );
4223
4224                 hr = IXMLDOMCDATASection_insertData(pCDataSec, 0, _bstr_("m"));
4225                 ok(hr == S_OK, "ret %08x\n", hr );
4226                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
4227                 ok(hr == S_OK, "ret %08x\n", hr );
4228                 ok( !lstrcmpW( str, _bstr_("m1") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4229                 SysFreeString(str);
4230
4231                 /* nonempty string, count greater than its length */
4232                 hr = IXMLDOMCDATASection_replaceData(pCDataSec, 0, 2, _bstr_("a1.2"));
4233                 ok(hr == S_OK, "ret %08x\n", hr );
4234                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
4235                 ok(hr == S_OK, "ret %08x\n", hr );
4236                 ok( !lstrcmpW( str, _bstr_("a1.2") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4237                 SysFreeString(str);
4238
4239                 /* nonempty string, count less than its length */
4240                 hr = IXMLDOMCDATASection_replaceData(pCDataSec, 0, 1, _bstr_("wine"));
4241                 ok(hr == S_OK, "ret %08x\n", hr );
4242                 hr = IXMLDOMCDATASection_get_text(pCDataSec, &str);
4243                 ok(hr == S_OK, "ret %08x\n", hr );
4244                 ok( !lstrcmpW( str, _bstr_("wine1.2") ), "incorrect get_text string, got '%s'\n", wine_dbgstr_w(str) );
4245                 SysFreeString(str);
4246
4247                 IXMLDOMCDATASection_Release(pCDataSec);
4248             }
4249
4250             /* Document Fragments */
4251             hr = IXMLDOMDocument_createDocumentFragment(doc, NULL);
4252             ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4253
4254             hr = IXMLDOMDocument_createDocumentFragment(doc, &pDocFrag);
4255             ok(hr == S_OK, "ret %08x\n", hr );
4256             if(hr == S_OK)
4257             {
4258                 IXMLDOMNode *node;
4259
4260                 hr = IXMLDOMDocumentFragment_get_parentNode(pDocFrag, NULL);
4261                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4262
4263                 node = (IXMLDOMNode *)0x1;
4264                 hr = IXMLDOMDocumentFragment_get_parentNode(pDocFrag, &node);
4265                 ok(hr == S_FALSE, "ret %08x\n", hr );
4266                 ok(node == NULL, "expected NULL, got %p\n", node);
4267
4268                 hr = IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pDocFrag, NULL);
4269                 ok(hr == S_OK, "ret %08x\n", hr );
4270
4271                 /* get Attribute Tests */
4272                 hr = IXMLDOMDocumentFragment_get_attributes(pDocFrag, NULL);
4273                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4274
4275                 pAttribs = (IXMLDOMNamedNodeMap*)0x1;
4276                 hr = IXMLDOMDocumentFragment_get_attributes(pDocFrag, &pAttribs);
4277                 ok(hr == S_FALSE, "ret %08x\n", hr );
4278                 ok(pAttribs == NULL, "pAttribs != NULL\n");
4279
4280                 hr = IXMLDOMDocumentFragment_get_nodeName(pDocFrag, &str);
4281                 ok(hr == S_OK, "ret %08x\n", hr );
4282                 ok( !lstrcmpW( str, szDocFragmentText ), "incorrect docfragment node Name\n");
4283                 SysFreeString(str);
4284
4285                 /* test next Sibling*/
4286                 hr = IXMLDOMDocumentFragment_get_nextSibling(pDocFrag, NULL);
4287                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4288
4289                 node = (IXMLDOMNode *)0x1;
4290                 hr = IXMLDOMDocumentFragment_get_nextSibling(pDocFrag, &node);
4291                 ok(hr == S_FALSE, "ret %08x\n", hr );
4292                 ok(node == NULL, "next sibling not NULL\n");
4293
4294                 /* test Previous Sibling*/
4295                 hr = IXMLDOMDocumentFragment_get_previousSibling(pDocFrag, NULL);
4296                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4297
4298                 node = (IXMLDOMNode *)0x1;
4299                 hr = IXMLDOMDocumentFragment_get_previousSibling(pDocFrag, &node);
4300                 ok(hr == S_FALSE, "ret %08x\n", hr );
4301                 ok(node == NULL, "previous sibling not NULL\n");
4302
4303                 /* test get_dataType */
4304                 hr = IXMLDOMDocumentFragment_get_dataType(pDocFrag, NULL);
4305                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4306
4307                 hr = IXMLDOMDocumentFragment_get_dataType(pDocFrag, &v);
4308                 ok(hr == S_FALSE, "ret %08x\n", hr );
4309                 ok( V_VT(&v) == VT_NULL, "incorrect dataType type\n");
4310                 VariantClear(&v);
4311
4312                 /* test nodeTypeString */
4313                 hr = IXMLDOMDocumentFragment_get_nodeTypeString(pDocFrag, &str);
4314                 ok(hr == S_OK, "ret %08x\n", hr );
4315                 ok( !lstrcmpW( str, _bstr_("documentfragment") ), "incorrect nodeTypeString string\n");
4316                 SysFreeString(str);
4317
4318                 IXMLDOMDocumentFragment_Release(pDocFrag);
4319             }
4320
4321             /* Entity References */
4322             hr = IXMLDOMDocument_createEntityReference(doc, NULL, &pEntityRef);
4323             ok(hr == E_FAIL, "ret %08x\n", hr );
4324             hr = IXMLDOMDocument_createEntityReference(doc, _bstr_(""), &pEntityRef);
4325             ok(hr == E_FAIL, "ret %08x\n", hr );
4326
4327             str = SysAllocString(szEntityRef);
4328             hr = IXMLDOMDocument_createEntityReference(doc, str, NULL);
4329             ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4330
4331             hr = IXMLDOMDocument_createEntityReference(doc, str, &pEntityRef);
4332             SysFreeString(str);
4333             ok(hr == S_OK, "ret %08x\n", hr );
4334             if(hr == S_OK)
4335             {
4336                 hr = IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pEntityRef, NULL);
4337                 ok(hr == S_OK, "ret %08x\n", hr );
4338
4339                 /* get Attribute Tests */
4340                 hr = IXMLDOMEntityReference_get_attributes(pEntityRef, NULL);
4341                 ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4342
4343                 pAttribs = (IXMLDOMNamedNodeMap*)0x1;
4344                 hr = IXMLDOMEntityReference_get_attributes(pEntityRef, &pAttribs);
4345                 ok(hr == S_FALSE, "ret %08x\n", hr );
4346                 ok(pAttribs == NULL, "pAttribs != NULL\n");
4347
4348                 /* test dataType */
4349                 hr = IXMLDOMEntityReference_get_dataType(pEntityRef, &v);
4350                 ok(hr == S_FALSE, "ret %08x\n", hr );
4351                 ok( V_VT(&v) == VT_NULL, "incorrect dataType type\n");
4352                 VariantClear(&v);
4353
4354                 /* test nodeTypeString */
4355                 hr = IXMLDOMEntityReference_get_nodeTypeString(pEntityRef, &str);
4356                 ok(hr == S_OK, "ret %08x\n", hr );
4357                 ok( !lstrcmpW( str, _bstr_("entityreference") ), "incorrect nodeTypeString string\n");
4358                 SysFreeString(str);
4359
4360                 /* test get_xml*/
4361                 hr = IXMLDOMEntityReference_get_xml(pEntityRef, &str);
4362                 ok(hr == S_OK, "ret %08x\n", hr );
4363                 ok( !lstrcmpW( str, szEntityRefXML ), "incorrect xml string\n");
4364                 SysFreeString(str);
4365
4366                 IXMLDOMEntityReference_Release(pEntityRef);
4367             }
4368
4369             IXMLDOMElement_Release( pRoot );
4370         }
4371     }
4372
4373     IXMLDOMDocument_Release(doc);
4374
4375     free_bstrs();
4376 }
4377
4378 static void test_nodeTypeTests( void )
4379 {
4380     IXMLDOMDocument *doc = NULL;
4381     IXMLDOMElement *pRoot;
4382     IXMLDOMElement *pElement;
4383     HRESULT hr;
4384
4385     doc = create_document(&IID_IXMLDOMDocument);
4386     if (!doc) return;
4387
4388     hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing"), NULL);
4389     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4390
4391     hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing"), &pRoot);
4392     ok(hr == S_OK, "ret %08x\n", hr );
4393     if(hr == S_OK)
4394     {
4395         hr = IXMLDOMDocument_appendChild(doc, (IXMLDOMNode*)pRoot, NULL);
4396         ok(hr == S_OK, "ret %08x\n", hr );
4397         if(hr == S_OK)
4398         {
4399             hr = IXMLDOMElement_put_dataType(pRoot, NULL);
4400             ok(hr == E_INVALIDARG, "ret %08x\n", hr );
4401
4402             /* Invalid Value */
4403             hr = IXMLDOMElement_put_dataType(pRoot, _bstr_("abcdefg") );
4404             ok(hr == E_FAIL, "ret %08x\n", hr );
4405
4406             /* NOTE:
4407              *   The name passed into put_dataType is case-insensitive. So many of the names
4408              *     have been changed to reflect this.
4409              */
4410             /* Boolean */
4411             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_Boolean"), &pElement);
4412             ok(hr == S_OK, "ret %08x\n", hr );
4413             if(hr == S_OK)
4414             {
4415                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
4416
4417                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("Boolean") );
4418                 ok(hr == S_OK, "ret %08x\n", hr );
4419
4420                 IXMLDOMElement_Release(pElement);
4421             }
4422
4423             /* String */
4424             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_String"), &pElement);
4425             ok(hr == S_OK, "ret %08x\n", hr );
4426             if(hr == S_OK)
4427             {
4428                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
4429
4430                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("String") );
4431                 ok(hr == S_OK, "ret %08x\n", hr );
4432
4433                 IXMLDOMElement_Release(pElement);
4434             }
4435
4436             /* Number */
4437             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_Number"), &pElement);
4438             ok(hr == S_OK, "ret %08x\n", hr );
4439             if(hr == S_OK)
4440             {
4441                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
4442
4443                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("number") );
4444                 ok(hr == S_OK, "ret %08x\n", hr );
4445
4446                 IXMLDOMElement_Release(pElement);
4447             }
4448
4449             /* Int */
4450             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_Int"), &pElement);
4451             ok(hr == S_OK, "ret %08x\n", hr );
4452             if(hr == S_OK)
4453             {
4454                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
4455
4456                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("InT") );
4457                 ok(hr == S_OK, "ret %08x\n", hr );
4458
4459                 IXMLDOMElement_Release(pElement);
4460             }
4461
4462             /* Fixed */
4463             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_Fixed"), &pElement);
4464             ok(hr == S_OK, "ret %08x\n", hr );
4465             if(hr == S_OK)
4466             {
4467                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
4468
4469                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("fixed.14.4") );
4470                 ok(hr == S_OK, "ret %08x\n", hr );
4471
4472                 IXMLDOMElement_Release(pElement);
4473             }
4474
4475             /* DateTime */
4476             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_DateTime"), &pElement);
4477             ok(hr == S_OK, "ret %08x\n", hr );
4478             if(hr == S_OK)
4479             {
4480                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
4481
4482                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("DateTime") );
4483                 ok(hr == S_OK, "ret %08x\n", hr );
4484
4485                 IXMLDOMElement_Release(pElement);
4486             }
4487
4488             /* DateTime TZ */
4489             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_DateTime_tz"), &pElement);
4490             ok(hr == S_OK, "ret %08x\n", hr );
4491             if(hr == S_OK)
4492             {
4493                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
4494
4495                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("DateTime.tz") );
4496                 ok(hr == S_OK, "ret %08x\n", hr );
4497
4498                 IXMLDOMElement_Release(pElement);
4499             }
4500
4501             /* Date */
4502             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_Date"), &pElement);
4503             ok(hr == S_OK, "ret %08x\n", hr );
4504             if(hr == S_OK)
4505             {
4506                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
4507
4508                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("Date") );
4509                 ok(hr == S_OK, "ret %08x\n", hr );
4510
4511                 IXMLDOMElement_Release(pElement);
4512             }
4513
4514             /* Time */
4515             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_Time"), &pElement);
4516             ok(hr == S_OK, "ret %08x\n", hr );
4517             if(hr == S_OK)
4518             {
4519                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
4520
4521                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("Time") );
4522                 ok(hr == S_OK, "ret %08x\n", hr );
4523
4524                 IXMLDOMElement_Release(pElement);
4525             }
4526
4527             /* Time.tz */
4528             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_Time_TZ"), &pElement);
4529             ok(hr == S_OK, "ret %08x\n", hr );
4530             if(hr == S_OK)
4531             {
4532                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
4533
4534                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("Time.tz") );
4535                 ok(hr == S_OK, "ret %08x\n", hr );
4536
4537                 IXMLDOMElement_Release(pElement);
4538             }
4539
4540             /* I1 */
4541             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_I1"), &pElement);
4542             ok(hr == S_OK, "ret %08x\n", hr );
4543             if(hr == S_OK)
4544             {
4545                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
4546
4547                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("I1") );
4548                 ok(hr == S_OK, "ret %08x\n", hr );
4549
4550                 IXMLDOMElement_Release(pElement);
4551             }
4552
4553             /* I2 */
4554             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_I2"), &pElement);
4555             ok(hr == S_OK, "ret %08x\n", hr );
4556             if(hr == S_OK)
4557             {
4558                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
4559
4560                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("I2") );
4561                 ok(hr == S_OK, "ret %08x\n", hr );
4562
4563                 IXMLDOMElement_Release(pElement);
4564             }
4565
4566             /* I4 */
4567             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_I4"), &pElement);
4568             ok(hr == S_OK, "ret %08x\n", hr );
4569             if(hr == S_OK)
4570             {
4571                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
4572
4573                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("I4") );
4574                 ok(hr == S_OK, "ret %08x\n", hr );
4575
4576                 IXMLDOMElement_Release(pElement);
4577             }
4578
4579             /* UI1 */
4580             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_UI1"), &pElement);
4581             ok(hr == S_OK, "ret %08x\n", hr );
4582             if(hr == S_OK)
4583             {
4584                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
4585
4586                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("UI1") );
4587                 ok(hr == S_OK, "ret %08x\n", hr );
4588
4589                 IXMLDOMElement_Release(pElement);
4590             }
4591
4592             /* UI2 */
4593             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_UI2"), &pElement);
4594             ok(hr == S_OK, "ret %08x\n", hr );
4595             if(hr == S_OK)
4596             {
4597                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
4598
4599                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("UI2") );
4600                 ok(hr == S_OK, "ret %08x\n", hr );
4601
4602                 IXMLDOMElement_Release(pElement);
4603             }
4604
4605             /* UI4 */
4606             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_UI4"), &pElement);
4607             ok(hr == S_OK, "ret %08x\n", hr );
4608             if(hr == S_OK)
4609             {
4610                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
4611
4612                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("UI4") );
4613                 ok(hr == S_OK, "ret %08x\n", hr );
4614
4615                 IXMLDOMElement_Release(pElement);
4616             }
4617
4618             /* r4 */
4619             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_r4"), &pElement);
4620             ok(hr == S_OK, "ret %08x\n", hr );
4621             if(hr == S_OK)
4622             {
4623                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
4624
4625                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("r4") );
4626                 ok(hr == S_OK, "ret %08x\n", hr );
4627
4628                 IXMLDOMElement_Release(pElement);
4629             }
4630
4631             /* r8 */
4632             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_r8"), &pElement);
4633             ok(hr == S_OK, "ret %08x\n", hr );
4634             if(hr == S_OK)
4635             {
4636                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
4637
4638                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("r8") );
4639                 ok(hr == S_OK, "ret %08x\n", hr );
4640
4641                 IXMLDOMElement_Release(pElement);
4642             }
4643
4644             /* float */
4645             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_float"), &pElement);
4646             ok(hr == S_OK, "ret %08x\n", hr );
4647             if(hr == S_OK)
4648             {
4649                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
4650
4651                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("float") );
4652                 ok(hr == S_OK, "ret %08x\n", hr );
4653
4654                 IXMLDOMElement_Release(pElement);
4655             }
4656
4657             /* uuid */
4658             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_uuid"), &pElement);
4659             ok(hr == S_OK, "ret %08x\n", hr );
4660             if(hr == S_OK)
4661             {
4662                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
4663
4664                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("UuId") );
4665                 ok(hr == S_OK, "ret %08x\n", hr );
4666
4667                 IXMLDOMElement_Release(pElement);
4668             }
4669
4670             /* bin.hex */
4671             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_bin_hex"), &pElement);
4672             ok(hr == S_OK, "ret %08x\n", hr );
4673             if(hr == S_OK)
4674             {
4675                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
4676
4677                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("bin.hex") );
4678                 ok(hr == S_OK, "ret %08x\n", hr );
4679
4680                 IXMLDOMElement_Release(pElement);
4681             }
4682
4683             /* bin.base64 */
4684             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_bin_base64"), &pElement);
4685             ok(hr == S_OK, "ret %08x\n", hr );
4686             if(hr == S_OK)
4687             {
4688                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
4689
4690                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("bin.base64") );
4691                 ok(hr == S_OK, "ret %08x\n", hr );
4692
4693                 IXMLDOMElement_Release(pElement);
4694             }
4695
4696             /* Check changing types */
4697             hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing_Change"), &pElement);
4698             ok(hr == S_OK, "ret %08x\n", hr );
4699             if(hr == S_OK)
4700             {
4701                 IXMLDOMElement_appendChild(pRoot, (IXMLDOMNode*)pElement, NULL);
4702
4703                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("DateTime.tz") );
4704                 ok(hr == S_OK, "ret %08x\n", hr );
4705
4706                 hr = IXMLDOMElement_put_dataType(pElement, _bstr_("string") );
4707                 ok(hr == S_OK, "ret %08x\n", hr );
4708
4709                 IXMLDOMElement_Release(pElement);
4710             }
4711
4712             IXMLDOMElement_Release(pRoot);
4713         }
4714     }
4715
4716     IXMLDOMDocument_Release(doc);
4717
4718     free_bstrs();
4719 }
4720
4721 static void test_DocumentSaveToDocument(void)
4722 {
4723     IXMLDOMDocument *doc, *doc2;
4724     IXMLDOMElement *pRoot;
4725     HRESULT hr;
4726
4727     doc = create_document(&IID_IXMLDOMDocument);
4728     if (!doc) return;
4729
4730     doc2 = create_document(&IID_IXMLDOMDocument);
4731     if (!doc2)
4732     {
4733         IXMLDOMDocument_Release(doc);
4734         return;
4735     }
4736
4737     hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing"), &pRoot);
4738     ok(hr == S_OK, "ret %08x\n", hr );
4739     if(hr == S_OK)
4740     {
4741         hr = IXMLDOMDocument_appendChild(doc, (IXMLDOMNode*)pRoot, NULL);
4742         ok(hr == S_OK, "ret %08x\n", hr );
4743         if(hr == S_OK)
4744         {
4745             VARIANT vDoc;
4746             BSTR sOrig;
4747             BSTR sNew;
4748
4749             V_VT(&vDoc) = VT_UNKNOWN;
4750             V_UNKNOWN(&vDoc) = (IUnknown*)doc2;
4751
4752             hr = IXMLDOMDocument_save(doc, vDoc);
4753             ok(hr == S_OK, "ret %08x\n", hr );
4754
4755             hr = IXMLDOMDocument_get_xml(doc, &sOrig);
4756             ok(hr == S_OK, "ret %08x\n", hr );
4757
4758             hr = IXMLDOMDocument_get_xml(doc2, &sNew);
4759             ok(hr == S_OK, "ret %08x\n", hr );
4760
4761             ok( !lstrcmpW( sOrig, sNew ), "New document is not the same as origial\n");
4762
4763             SysFreeString(sOrig);
4764             SysFreeString(sNew);
4765         }
4766     }
4767
4768     IXMLDOMDocument_Release(doc2);
4769     IXMLDOMDocument_Release(doc);
4770 }
4771
4772 static void test_DocumentSaveToFile(void)
4773 {
4774     IXMLDOMDocument *doc;
4775     IXMLDOMElement *pRoot;
4776     HANDLE file;
4777     char buffer[100];
4778     DWORD read = 0;
4779     HRESULT hr;
4780
4781     doc = create_document(&IID_IXMLDOMDocument);
4782     if (!doc) return;
4783
4784     hr = IXMLDOMDocument_createElement(doc, _bstr_("Testing"), &pRoot);
4785     ok(hr == S_OK, "ret %08x\n", hr );
4786     if(hr == S_OK)
4787     {
4788         hr = IXMLDOMDocument_appendChild(doc, (IXMLDOMNode*)pRoot, NULL);
4789         ok(hr == S_OK, "ret %08x\n", hr );
4790         if(hr == S_OK)
4791         {
4792             VARIANT vFile;
4793
4794             V_VT(&vFile) = VT_BSTR;
4795             V_BSTR(&vFile) = _bstr_("test.xml");
4796
4797             hr = IXMLDOMDocument_save(doc, vFile);
4798             ok(hr == S_OK, "ret %08x\n", hr );
4799         }
4800     }
4801
4802     IXMLDOMElement_Release(pRoot);
4803     IXMLDOMDocument_Release(doc);
4804
4805     file = CreateFile("test.xml", GENERIC_READ, 0, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL );
4806     ok(file != INVALID_HANDLE_VALUE, "Could not open file: %u\n", GetLastError());
4807     if(file == INVALID_HANDLE_VALUE)
4808         return;
4809
4810     ReadFile(file, buffer, sizeof(buffer), &read, NULL);
4811     ok(read != 0, "could not read file\n");
4812     ok(buffer[0] != '<' || buffer[1] != '?', "File contains processing instruction\n");
4813
4814     CloseHandle(file);
4815     DeleteFile("test.xml");
4816 }
4817
4818 static void test_testTransforms(void)
4819 {
4820     IXMLDOMDocument *doc, *docSS;
4821     IXMLDOMNode *pNode;
4822     VARIANT_BOOL bSucc;
4823
4824     HRESULT hr;
4825
4826     doc = create_document(&IID_IXMLDOMDocument);
4827     if (!doc) return;
4828
4829     docSS = create_document(&IID_IXMLDOMDocument);
4830     if (!docSS)
4831     {
4832         IXMLDOMDocument_Release(doc);
4833         return;
4834     }
4835
4836     hr = IXMLDOMDocument_loadXML(doc, _bstr_(szTransformXML), &bSucc);
4837     ok(hr == S_OK, "ret %08x\n", hr );
4838
4839     hr = IXMLDOMDocument_loadXML(docSS, _bstr_(szTransformSSXML), &bSucc);
4840     ok(hr == S_OK, "ret %08x\n", hr );
4841
4842     hr = IXMLDOMDocument_QueryInterface(docSS, &IID_IXMLDOMNode, (void**)&pNode );
4843     ok(hr == S_OK, "ret %08x\n", hr );
4844     if(hr == S_OK)
4845     {
4846         BSTR bOut;
4847
4848         hr = IXMLDOMDocument_transformNode(doc, pNode, &bOut);
4849         ok(hr == S_OK, "ret %08x\n", hr );
4850         ok( compareIgnoreReturns( bOut, _bstr_(szTransformOutput)), "Stylesheet output not correct\n");
4851         SysFreeString(bOut);
4852
4853         IXMLDOMNode_Release(pNode);
4854     }
4855
4856     IXMLDOMDocument_Release(docSS);
4857     IXMLDOMDocument_Release(doc);
4858
4859     free_bstrs();
4860 }
4861
4862 static void test_Namespaces(void)
4863 {
4864     IXMLDOMDocument *doc;
4865     IXMLDOMNode *pNode;
4866     IXMLDOMNode *pNode2 = NULL;
4867     VARIANT_BOOL bSucc;
4868     HRESULT hr;
4869     BSTR str;
4870     static  const CHAR szNamespacesXML[] =
4871 "<?xml version=\"1.0\"?>\n"
4872 "<root xmlns:WEB='http://www.winehq.org'>\n"
4873 "<WEB:Site version=\"1.0\" />\n"
4874 "</root>";
4875
4876     doc = create_document(&IID_IXMLDOMDocument);
4877     if (!doc) return;
4878
4879     hr = IXMLDOMDocument_loadXML(doc, _bstr_(szNamespacesXML), &bSucc);
4880     ok(hr == S_OK, "ret %08x\n", hr );
4881     ok(bSucc == VARIANT_TRUE, "Expected VARIANT_TRUE got VARIANT_FALSE\n");
4882
4883     hr = IXMLDOMDocument_selectSingleNode(doc, _bstr_("root"), &pNode );
4884     ok(hr == S_OK, "ret %08x\n", hr );
4885     if(hr == S_OK)
4886     {
4887         hr = IXMLDOMNode_get_firstChild( pNode, &pNode2 );
4888         ok( hr == S_OK, "ret %08x\n", hr );
4889         ok( pNode2 != NULL, "pNode2 == NULL\n");
4890
4891         /* Test get_prefix */
4892         hr = IXMLDOMNode_get_prefix(pNode2, NULL);
4893         ok( hr == E_INVALIDARG, "ret %08x\n", hr );
4894         /* NOTE: Need to test that arg2 gets cleared on Error. */
4895
4896         hr = IXMLDOMNode_get_prefix(pNode2, &str);
4897         ok( hr == S_OK, "ret %08x\n", hr );
4898         ok( !lstrcmpW( str, _bstr_("WEB")), "incorrect prefix string\n");
4899         SysFreeString(str);
4900
4901         /* Test get_namespaceURI */
4902         hr = IXMLDOMNode_get_namespaceURI(pNode2, NULL);
4903         ok( hr == E_INVALIDARG, "ret %08x\n", hr );
4904         /* NOTE: Need to test that arg2 gets cleared on Error. */
4905
4906         hr = IXMLDOMNode_get_namespaceURI(pNode2, &str);
4907         ok( hr == S_OK, "ret %08x\n", hr );
4908         ok( !lstrcmpW( str, _bstr_("http://www.winehq.org")), "incorrect namespaceURI string\n");
4909         SysFreeString(str);
4910
4911         IXMLDOMNode_Release(pNode2);
4912         IXMLDOMNode_Release(pNode);
4913     }
4914
4915     IXMLDOMDocument_Release(doc);
4916
4917     free_bstrs();
4918 }
4919
4920 static void test_FormattingXML(void)
4921 {
4922     IXMLDOMDocument *doc;
4923     IXMLDOMElement *pElement;
4924     VARIANT_BOOL bSucc;
4925     HRESULT hr;
4926     BSTR str;
4927     static const CHAR szLinefeedXML[] = "<?xml version=\"1.0\"?>\n<Root>\n\t<Sub val=\"A\" />\n</Root>";
4928     static const CHAR szLinefeedRootXML[] = "<Root>\r\n\t<Sub val=\"A\"/>\r\n</Root>";
4929
4930     doc = create_document(&IID_IXMLDOMDocument);
4931     if (!doc) return;
4932
4933     hr = IXMLDOMDocument_loadXML(doc, _bstr_(szLinefeedXML), &bSucc);
4934     ok(hr == S_OK, "ret %08x\n", hr );
4935     ok(bSucc == VARIANT_TRUE, "Expected VARIANT_TRUE got VARIANT_FALSE\n");
4936
4937     if(bSucc == VARIANT_TRUE)
4938     {
4939         hr = IXMLDOMDocument_get_documentElement(doc, &pElement);
4940         ok(hr == S_OK, "ret %08x\n", hr );
4941         if(hr == S_OK)
4942         {
4943             hr = IXMLDOMElement_get_xml(pElement, &str);
4944             ok(hr == S_OK, "ret %08x\n", hr );
4945             ok( !lstrcmpW( str, _bstr_(szLinefeedRootXML) ), "incorrect element xml\n");
4946             SysFreeString(str);
4947
4948             IXMLDOMElement_Release(pElement);
4949         }
4950     }
4951
4952     IXMLDOMDocument_Release(doc);
4953
4954     free_bstrs();
4955 }
4956
4957 typedef struct _nodetypedvalue_t {
4958     const char *name;
4959     VARTYPE type;
4960     const char *value; /* value in string format */
4961 } nodetypedvalue_t;
4962
4963 static const nodetypedvalue_t get_nodetypedvalue[] = {
4964     { "root/string",    VT_BSTR, "Wine" },
4965     { "root/string2",   VT_BSTR, "String" },
4966     { "root/number",    VT_BSTR, "12.44" },
4967     { "root/number2",   VT_BSTR, "-3.71e3" },
4968     { "root/int",       VT_I4,   "-13" },
4969     { "root/fixed",     VT_CY,   "7322.9371" },
4970     { "root/bool",      VT_BOOL, "-1" },
4971     { "root/datetime",  VT_DATE, "40135.14" },
4972     { "root/datetimetz",VT_DATE, "37813.59" },
4973     { "root/date",      VT_DATE, "665413" },
4974     { "root/time",      VT_DATE, "0.5813889" },
4975     { "root/timetz",    VT_DATE, "1.112512" },
4976     { "root/i1",        VT_I1,   "-13" },
4977     { "root/i2",        VT_I2,   "31915" },
4978     { "root/i4",        VT_I4,   "-312232" },
4979     { "root/ui1",       VT_UI1,  "123" },
4980     { "root/ui2",       VT_UI2,  "48282" },
4981     { "root/ui4",       VT_UI4,  "949281" },
4982     { "root/r4",        VT_R4,   "213124" },
4983     { "root/r8",        VT_R8,   "0.412" },
4984     { "root/float",     VT_R8,   "41221.421" },
4985     { "root/uuid",      VT_BSTR, "333C7BC4-460F-11D0-BC04-0080C7055a83" },
4986     { 0 }
4987 };
4988
4989 static void test_nodeTypedValue(void)
4990 {
4991     const nodetypedvalue_t *entry = get_nodetypedvalue;
4992     IXMLDOMDocument *doc;
4993     IXMLDOMNode *node;
4994     VARIANT_BOOL b;
4995     VARIANT value;
4996     HRESULT hr;
4997
4998     doc = create_document(&IID_IXMLDOMDocument);
4999     if (!doc) return;
5000
5001     b = VARIANT_FALSE;
5002     hr = IXMLDOMDocument_loadXML(doc, _bstr_(szTypeValueXML), &b);
5003     ok(hr == S_OK, "ret %08x\n", hr );
5004     ok(b == VARIANT_TRUE, "got %d\n", b);
5005
5006     hr = IXMLDOMDocument_get_nodeValue(doc, NULL);
5007     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5008
5009     V_VT(&value) = VT_BSTR;
5010     V_BSTR(&value) = NULL;
5011     hr = IXMLDOMDocument_get_nodeValue(doc, &value);
5012     ok(hr == S_FALSE, "ret %08x\n", hr );
5013     ok(V_VT(&value) == VT_NULL, "expect VT_NULL got %d\n", V_VT(&value));
5014
5015     hr = IXMLDOMDocument_get_nodeTypedValue(doc, NULL);
5016     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5017
5018     hr = IXMLDOMDocument_get_nodeTypedValue(doc, &value);
5019     ok(hr == S_FALSE, "ret %08x\n", hr );
5020
5021     hr = IXMLDOMDocument_selectSingleNode(doc, _bstr_("root/string"), &node);
5022     ok(hr == S_OK, "ret %08x\n", hr );
5023
5024     V_VT(&value) = VT_BSTR;
5025     V_BSTR(&value) = NULL;
5026     hr = IXMLDOMNode_get_nodeValue(node, &value);
5027     ok(hr == S_FALSE, "ret %08x\n", hr );
5028     ok(V_VT(&value) == VT_NULL, "expect VT_NULL got %d\n", V_VT(&value));
5029
5030     hr = IXMLDOMNode_get_nodeTypedValue(node, NULL);
5031     ok(hr == E_INVALIDARG, "ret %08x\n", hr );
5032
5033     IXMLDOMNode_Release(node);
5034
5035     hr = IXMLDOMDocument_selectSingleNode(doc, _bstr_("root/binhex"), &node);
5036     ok(hr == S_OK, "ret %08x\n", hr );
5037     {
5038         BYTE bytes[] = {0xff,0xfc,0xa0,0x12,0x00,0x3c};
5039
5040         hr = IXMLDOMNode_get_nodeTypedValue(node, &value);
5041         ok(hr == S_OK, "ret %08x\n", hr );
5042         ok(V_VT(&value) == (VT_ARRAY|VT_UI1), "incorrect type\n");
5043         ok(V_ARRAY(&value)->rgsabound[0].cElements == 6, "incorrect array size\n");
5044         if(V_ARRAY(&value)->rgsabound[0].cElements == 6)
5045             ok(!memcmp(bytes, V_ARRAY(&value)->pvData, sizeof(bytes)), "incorrect value\n");
5046         VariantClear(&value);
5047         IXMLDOMNode_Release(node);
5048     }
5049
5050     hr = IXMLDOMDocument_selectSingleNode(doc, _bstr_("root/binbase64"), &node);
5051     ok(hr == S_OK, "ret %08x\n", hr );
5052     {
5053         BYTE bytes[] = {0x62,0x61,0x73,0x65,0x36,0x34,0x20,0x74,0x65,0x73,0x74};
5054
5055         hr = IXMLDOMNode_get_nodeTypedValue(node, &value);
5056         ok(hr == S_OK, "ret %08x\n", hr );
5057         ok(V_VT(&value) == (VT_ARRAY|VT_UI1), "incorrect type\n");
5058         ok(V_ARRAY(&value)->rgsabound[0].cElements == 11, "incorrect array size\n");
5059         if(V_ARRAY(&value)->rgsabound[0].cElements == 11)
5060             ok(!memcmp(bytes, V_ARRAY(&value)->pvData, sizeof(bytes)), "incorrect value\n");
5061         VariantClear(&value);
5062         IXMLDOMNode_Release(node);
5063     }
5064
5065     while (entry->name)
5066     {
5067         hr = IXMLDOMDocument_selectSingleNode(doc, _bstr_(entry->name), &node);
5068         ok(hr == S_OK, "ret %08x\n", hr );
5069
5070         hr = IXMLDOMNode_get_nodeTypedValue(node, &value);
5071         ok(hr == S_OK, "ret %08x\n", hr );
5072         ok(V_VT(&value) == entry->type, "incorrect type, expected %d, got %d\n", entry->type, V_VT(&value));
5073
5074         if (entry->type != VT_BSTR)
5075         {
5076            if (entry->type == VT_DATE)
5077            {
5078                hr = VariantChangeType(&value, &value, 0, VT_R4);
5079                ok(hr == S_OK, "ret %08x\n", hr );
5080                hr = VariantChangeType(&value, &value, 0, VT_BSTR);
5081                ok(hr == S_OK, "ret %08x\n", hr );
5082                ok(lstrcmpW( V_BSTR(&value), _bstr_(entry->value)) == 0,
5083                    "expected %s, got %s\n", entry->value, wine_dbgstr_w(V_BSTR(&value)));
5084            }
5085            else
5086            {
5087                hr = VariantChangeType(&value, &value, 0, VT_BSTR);
5088                ok(hr == S_OK, "ret %08x\n", hr );
5089                ok(lstrcmpW( V_BSTR(&value), _bstr_(entry->value)) == 0,
5090                    "expected %s, got %s\n", entry->value, wine_dbgstr_w(V_BSTR(&value)));
5091            }
5092         }
5093         else
5094            ok(lstrcmpW( V_BSTR(&value), _bstr_(entry->value)) == 0,
5095                "expected %s, got %s\n", entry->value, wine_dbgstr_w(V_BSTR(&value)));
5096
5097         VariantClear( &value );
5098         IXMLDOMNode_Release(node);
5099
5100         entry++;
5101     }
5102
5103     IXMLDOMDocument_Release(doc);
5104     free_bstrs();
5105 }
5106
5107 static void test_TransformWithLoadingLocalFile(void)
5108 {
5109     IXMLDOMDocument *doc;
5110     IXMLDOMDocument *xsl;
5111     IXMLDOMNode *pNode;
5112     VARIANT_BOOL bSucc;
5113     HRESULT hr;
5114     HANDLE file;
5115     DWORD dwWritten;
5116     char lpPathBuffer[MAX_PATH];
5117     int i;
5118
5119     /* Create a Temp File. */
5120     GetTempPathA(MAX_PATH, lpPathBuffer);
5121     strcat(lpPathBuffer, "customers.xml" );
5122
5123     file = CreateFile(lpPathBuffer, GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL );
5124     ok(file != INVALID_HANDLE_VALUE, "Could not create file: %u\n", GetLastError());
5125     if(file == INVALID_HANDLE_VALUE)
5126         return;
5127
5128     WriteFile(file, szBasicTransformXML, strlen(szBasicTransformXML), &dwWritten, NULL);
5129     CloseHandle(file);
5130
5131     /* Correct path to not include a escape character. */
5132     for(i=0; i < strlen(lpPathBuffer); i++)
5133     {
5134         if(lpPathBuffer[i] == '\\')
5135             lpPathBuffer[i] = '/';
5136     }
5137
5138     doc = create_document(&IID_IXMLDOMDocument);
5139     if (!doc) return;
5140
5141     xsl = create_document(&IID_IXMLDOMDocument);
5142     if (!xsl)
5143     {
5144         IXMLDOMDocument2_Release(doc);
5145         return;
5146     }
5147
5148     hr = IXMLDOMDocument_loadXML(doc, _bstr_(szTypeValueXML), &bSucc);
5149     ok(hr == S_OK, "ret %08x\n", hr );
5150     ok(bSucc == VARIANT_TRUE, "Expected VARIANT_TRUE got VARIANT_FALSE\n");
5151     if(bSucc == VARIANT_TRUE)
5152     {
5153         BSTR sXSL;
5154         BSTR sPart1 = _bstr_(szBasicTransformSSXMLPart1);
5155         BSTR sPart2 = _bstr_(szBasicTransformSSXMLPart2);
5156         BSTR sFileName = _bstr_(lpPathBuffer);
5157         int nLegnth = lstrlenW(sPart1) + lstrlenW(sPart2) + lstrlenW(sFileName) + 1;
5158
5159         sXSL = SysAllocStringLen(NULL, nLegnth);
5160         lstrcpyW(sXSL, sPart1);
5161         lstrcatW(sXSL, sFileName);
5162         lstrcatW(sXSL, sPart2);
5163
5164         hr = IXMLDOMDocument_loadXML(xsl, sXSL, &bSucc);
5165         ok(hr == S_OK, "ret %08x\n", hr );
5166         ok(bSucc == VARIANT_TRUE, "Expected VARIANT_TRUE got VARIANT_FALSE\n");
5167         if(bSucc == VARIANT_TRUE)
5168         {
5169             BSTR sResult;
5170
5171             hr = IXMLDOMDocument_QueryInterface(xsl, &IID_IXMLDOMNode, (void**)&pNode );
5172             ok(hr == S_OK, "ret %08x\n", hr );
5173             if(hr == S_OK)
5174             {
5175                 /* This will load the temp file via the XSL */
5176                 hr = IXMLDOMDocument_transformNode(doc, pNode, &sResult);
5177                 ok(hr == S_OK, "ret %08x\n", hr );
5178                 if(hr == S_OK)
5179                 {
5180                     ok( compareIgnoreReturns( sResult, _bstr_(szBasicTransformOutput)), "Stylesheet output not correct\n");
5181                     SysFreeString(sResult);
5182                 }
5183
5184                 IXMLDOMNode_Release(pNode);
5185             }
5186         }
5187
5188         SysFreeString(sXSL);
5189     }
5190
5191     IXMLDOMDocument_Release(doc);
5192     IXMLDOMDocument_Release(xsl);
5193
5194     DeleteFile(lpPathBuffer);
5195     free_bstrs();
5196 }
5197
5198 static void test_put_nodeValue(void)
5199 {
5200     IXMLDOMDocument *doc;
5201     IXMLDOMEntityReference *entityref;
5202     IXMLDOMNode *node;
5203     HRESULT hr;
5204     VARIANT data, type;
5205
5206     doc = create_document(&IID_IXMLDOMDocument);
5207     if (!doc) return;
5208
5209     /* test for unsupported types */
5210     /* NODE_DOCUMENT */
5211     hr = IXMLDOMDocument_QueryInterface(doc, &IID_IXMLDOMNode, (void**)&node);
5212     ok(hr == S_OK, "ret %08x\n", hr );
5213     V_VT(&data) = VT_BSTR;
5214     V_BSTR(&data) = _bstr_("one two three");
5215     hr = IXMLDOMNode_put_nodeValue(node, data);
5216     ok(hr == E_FAIL, "ret %08x\n", hr );
5217     IXMLDOMNode_Release(node);
5218
5219     /* NODE_DOCUMENT_FRAGMENT */
5220     V_VT(&type) = VT_I1;
5221     V_I1(&type) = NODE_DOCUMENT_FRAGMENT;
5222     hr = IXMLDOMDocument_createNode(doc, type, _bstr_("test"), NULL, &node);
5223     ok(hr == S_OK, "ret %08x\n", hr );
5224     V_VT(&data) = VT_BSTR;
5225     V_BSTR(&data) = _bstr_("one two three");
5226     hr = IXMLDOMNode_put_nodeValue(node, data);
5227     ok(hr == E_FAIL, "ret %08x\n", hr );
5228     IXMLDOMNode_Release(node);
5229
5230     /* NODE_ELEMENT */
5231     V_VT(&type) = VT_I1;
5232     V_I1(&type) = NODE_ELEMENT;
5233     hr = IXMLDOMDocument_createNode(doc, type, _bstr_("test"), NULL, &node);
5234     ok(hr == S_OK, "ret %08x\n", hr );
5235     V_VT(&data) = VT_BSTR;
5236     V_BSTR(&data) = _bstr_("one two three");
5237     hr = IXMLDOMNode_put_nodeValue(node, data);
5238     ok(hr == E_FAIL, "ret %08x\n", hr );
5239     IXMLDOMNode_Release(node);
5240
5241     /* NODE_ENTITY_REFERENCE */
5242     hr = IXMLDOMDocument_createEntityReference(doc, _bstr_("ref"), &entityref);
5243     ok(hr == S_OK, "ret %08x\n", hr );
5244
5245     V_VT(&data) = VT_BSTR;
5246     V_BSTR(&data) = _bstr_("one two three");
5247     hr = IXMLDOMEntityReference_put_nodeValue(entityref, data);
5248     ok(hr == E_FAIL, "ret %08x\n", hr );
5249
5250     hr = IXMLDOMEntityReference_QueryInterface(entityref, &IID_IXMLDOMNode, (void**)&node);
5251     ok(hr == S_OK, "ret %08x\n", hr );
5252     V_VT(&data) = VT_BSTR;
5253     V_BSTR(&data) = _bstr_("one two three");
5254     hr = IXMLDOMNode_put_nodeValue(node, data);
5255     ok(hr == E_FAIL, "ret %08x\n", hr );
5256     IXMLDOMNode_Release(node);
5257     IXMLDOMEntityReference_Release(entityref);
5258
5259     free_bstrs();
5260
5261     IXMLDOMDocument_Release(doc);
5262 }
5263
5264 static void test_IObjectSafety_set(IObjectSafety *safety, HRESULT result, HRESULT result2, DWORD set, DWORD mask, DWORD expected, DWORD expected2)
5265 {
5266     DWORD enabled, supported;
5267     HRESULT hr;
5268
5269     hr = IObjectSafety_SetInterfaceSafetyOptions(safety, NULL, set, mask);
5270     if (result == result2)
5271         ok(hr == result, "SetInterfaceSafetyOptions: expected %08x, returned %08x\n", result, hr );
5272     else
5273         ok(broken(hr == result) || hr == result2,
5274            "SetInterfaceSafetyOptions: expected %08x, got %08x\n", result2, hr );
5275
5276     supported = enabled = 0xCAFECAFE;
5277     hr = IObjectSafety_GetInterfaceSafetyOptions(safety, NULL, &supported, &enabled);
5278     ok(hr == S_OK, "ret %08x\n", hr );
5279     if (expected == expected2)
5280         ok(enabled == expected, "Expected %08x, got %08x\n", expected, enabled);
5281     else
5282         ok(broken(enabled == expected) || enabled == expected2,
5283            "Expected %08x, got %08x\n", expected2, enabled);
5284
5285     /* reset the safety options */
5286
5287     hr = IObjectSafety_SetInterfaceSafetyOptions(safety, NULL,
5288             INTERFACESAFE_FOR_UNTRUSTED_CALLER|INTERFACESAFE_FOR_UNTRUSTED_DATA|INTERFACE_USES_SECURITY_MANAGER,
5289             0);
5290     ok(hr == S_OK, "ret %08x\n", hr );
5291
5292     hr = IObjectSafety_GetInterfaceSafetyOptions(safety, NULL, &supported, &enabled);
5293     ok(hr == S_OK, "ret %08x\n", hr );
5294     ok(enabled == 0, "Expected 0, got %08x\n", enabled);
5295 }
5296
5297 static void test_document_IObjectSafety(void)
5298 {
5299     IXMLDOMDocument *doc;
5300     IObjectSafety *safety;
5301     DWORD enabled = 0, supported = 0;
5302     HRESULT hr;
5303
5304     doc = create_document(&IID_IXMLDOMDocument);
5305     if (!doc) return;
5306
5307     hr = IXMLDOMDocument_QueryInterface(doc, &IID_IObjectSafety, (void**)&safety);
5308     ok(hr == S_OK, "ret %08x\n", hr );
5309
5310     /* get */
5311     hr = IObjectSafety_GetInterfaceSafetyOptions(safety, NULL, NULL, &enabled);
5312     ok(hr == E_POINTER, "ret %08x\n", hr );
5313     hr = IObjectSafety_GetInterfaceSafetyOptions(safety, NULL, &supported, NULL);
5314     ok(hr == E_POINTER, "ret %08x\n", hr );
5315
5316     hr = IObjectSafety_GetInterfaceSafetyOptions(safety, NULL, &supported, &enabled);
5317     ok(hr == S_OK, "ret %08x\n", hr );
5318     ok(broken(supported == (INTERFACESAFE_FOR_UNTRUSTED_CALLER | INTERFACESAFE_FOR_UNTRUSTED_DATA)) ||
5319        supported == (INTERFACESAFE_FOR_UNTRUSTED_CALLER | INTERFACESAFE_FOR_UNTRUSTED_DATA | INTERFACE_USES_SECURITY_MANAGER) /* msxml3 SP8+ */,
5320         "Expected (INTERFACESAFE_FOR_UNTRUSTED_CALLER | INTERFACESAFE_FOR_UNTRUSTED_DATA | INTERFACE_USES_SECURITY_MANAGER), "
5321              "got %08x\n", supported);
5322     ok(enabled == 0, "Expected 0, got %08x\n", enabled);
5323
5324     /* set -- individual flags */
5325
5326     test_IObjectSafety_set(safety, S_OK, S_OK,
5327         INTERFACESAFE_FOR_UNTRUSTED_CALLER, INTERFACESAFE_FOR_UNTRUSTED_CALLER,
5328         INTERFACESAFE_FOR_UNTRUSTED_CALLER, INTERFACESAFE_FOR_UNTRUSTED_CALLER);
5329
5330     test_IObjectSafety_set(safety, S_OK, S_OK,
5331         INTERFACESAFE_FOR_UNTRUSTED_DATA, INTERFACESAFE_FOR_UNTRUSTED_DATA,
5332         INTERFACESAFE_FOR_UNTRUSTED_DATA, INTERFACESAFE_FOR_UNTRUSTED_DATA);
5333
5334     test_IObjectSafety_set(safety, S_OK, S_OK,
5335         INTERFACE_USES_SECURITY_MANAGER, INTERFACE_USES_SECURITY_MANAGER,
5336         0, INTERFACE_USES_SECURITY_MANAGER /* msxml3 SP8+ */);
5337
5338     /* set INTERFACE_USES_DISPEX  */
5339
5340     test_IObjectSafety_set(safety, S_OK, E_FAIL /* msxml3 SP8+ */,
5341         INTERFACE_USES_DISPEX, INTERFACE_USES_DISPEX,
5342         0, 0);
5343
5344     test_IObjectSafety_set(safety, S_OK, E_FAIL /* msxml3 SP8+ */,
5345         INTERFACE_USES_DISPEX, 0,
5346         0, 0);
5347
5348     test_IObjectSafety_set(safety, S_OK, S_OK /* msxml3 SP8+ */,
5349         0, INTERFACE_USES_DISPEX,
5350         0, 0);
5351
5352     /* set option masking */
5353
5354     test_IObjectSafety_set(safety, S_OK, S_OK,
5355         INTERFACESAFE_FOR_UNTRUSTED_CALLER|INTERFACESAFE_FOR_UNTRUSTED_DATA,
5356         INTERFACESAFE_FOR_UNTRUSTED_CALLER,
5357         INTERFACESAFE_FOR_UNTRUSTED_CALLER,
5358         INTERFACESAFE_FOR_UNTRUSTED_CALLER);
5359
5360     test_IObjectSafety_set(safety, S_OK, S_OK,
5361         INTERFACESAFE_FOR_UNTRUSTED_CALLER|INTERFACESAFE_FOR_UNTRUSTED_DATA,
5362         INTERFACESAFE_FOR_UNTRUSTED_DATA,
5363         INTERFACESAFE_FOR_UNTRUSTED_DATA,
5364         INTERFACESAFE_FOR_UNTRUSTED_DATA);
5365
5366     test_IObjectSafety_set(safety, S_OK, S_OK,
5367         INTERFACESAFE_FOR_UNTRUSTED_CALLER|INTERFACESAFE_FOR_UNTRUSTED_DATA,
5368         INTERFACE_USES_SECURITY_MANAGER,
5369         0,
5370         0);
5371
5372     /* set -- inheriting previous settings */
5373
5374     hr = IObjectSafety_SetInterfaceSafetyOptions(safety, NULL,
5375                                                          INTERFACESAFE_FOR_UNTRUSTED_CALLER,
5376                                                          INTERFACESAFE_FOR_UNTRUSTED_CALLER);
5377     ok(hr == S_OK, "ret %08x\n", hr );
5378     hr = IObjectSafety_GetInterfaceSafetyOptions(safety, NULL, &supported, &enabled);
5379     ok(hr == S_OK, "ret %08x\n", hr );
5380     todo_wine
5381     ok(broken(enabled == INTERFACESAFE_FOR_UNTRUSTED_CALLER) ||
5382        enabled == (INTERFACESAFE_FOR_UNTRUSTED_CALLER | INTERFACE_USES_SECURITY_MANAGER) /* msxml3 SP8+ */,
5383          "Expected (INTERFACESAFE_FOR_UNTRUSTED_CALLER | INTERFACE_USES_SECURITY_MANAGER), "
5384          "got %08x\n", enabled);
5385
5386     hr = IObjectSafety_SetInterfaceSafetyOptions(safety, NULL,
5387                                                          INTERFACESAFE_FOR_UNTRUSTED_DATA,
5388                                                          INTERFACESAFE_FOR_UNTRUSTED_DATA);
5389     ok(hr == S_OK, "ret %08x\n", hr );
5390     hr = IObjectSafety_GetInterfaceSafetyOptions(safety, NULL, &supported, &enabled);
5391     ok(hr == S_OK, "ret %08x\n", hr );
5392     todo_wine
5393     ok(broken(enabled == INTERFACESAFE_FOR_UNTRUSTED_DATA) ||
5394        enabled == (INTERFACESAFE_FOR_UNTRUSTED_CALLER | INTERFACESAFE_FOR_UNTRUSTED_DATA) /* msxml3 SP8+ */,
5395         "Expected (INTERFACESAFE_FOR_UNTRUSTED_CALLER | INTERFACESAFE_FOR_UNTRUSTED_DATA), "
5396         "got %08x\n", enabled);
5397
5398     IObjectSafety_Release(safety);
5399
5400     IXMLDOMDocument_Release(doc);
5401 }
5402
5403 typedef struct _property_test_t {
5404     const GUID *guid;
5405     const char *clsid;
5406     const char *property;
5407     const char *value;
5408 } property_test_t;
5409
5410 static const property_test_t properties_test_data[] = {
5411     { &CLSID_DOMDocument,  "CLSID_DOMDocument" , "SelectionLanguage", "XSLPattern" },
5412     { &CLSID_DOMDocument2,  "CLSID_DOMDocument2" , "SelectionLanguage", "XSLPattern" },
5413     { &CLSID_DOMDocument30, "CLSID_DOMDocument30", "SelectionLanguage", "XSLPattern" },
5414     { &CLSID_DOMDocument40, "CLSID_DOMDocument40", "SelectionLanguage", "XPath" },
5415     { &CLSID_DOMDocument60, "CLSID_DOMDocument60", "SelectionLanguage", "XPath" },
5416     { 0 }
5417 };
5418
5419 static void test_default_properties(void)
5420 {
5421     const property_test_t *entry = properties_test_data;
5422     IXMLDOMDocument2 *doc;
5423     VARIANT var;
5424     HRESULT hr;
5425
5426     while (entry->guid)
5427     {
5428         hr = CoCreateInstance(entry->guid, NULL, CLSCTX_INPROC_SERVER, &IID_IXMLDOMDocument2, (void**)&doc);
5429         if (hr != S_OK)
5430         {
5431             win_skip("can't create %s instance\n", entry->clsid);
5432             entry++;
5433             continue;
5434         }
5435
5436         hr = IXMLDOMDocument2_getProperty(doc, _bstr_(entry->property), &var);
5437         ok(hr == S_OK, "got 0x%08x\n", hr);
5438         ok(lstrcmpW(V_BSTR(&var), _bstr_(entry->value)) == 0, "expected %s, for %s\n",
5439            entry->value, entry->clsid);
5440         VariantClear(&var);
5441
5442         IXMLDOMDocument2_Release(doc);
5443
5444         entry++;
5445     }
5446 }
5447
5448 static void test_XSLPattern(void)
5449 {
5450     IXMLDOMDocument2 *doc;
5451     IXMLDOMNodeList *list;
5452     VARIANT_BOOL b;
5453     LONG len;
5454
5455     doc = create_document(&IID_IXMLDOMDocument2);
5456     if (!doc) return;
5457
5458     ole_check(IXMLDOMDocument2_loadXML(doc, _bstr_(szExampleXML), &b));
5459     ok(b == VARIANT_TRUE, "failed to load XML string\n");
5460
5461     /* switch to XSLPattern */
5462     ole_check(IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionLanguage"), _variantbstr_("XSLPattern")));
5463
5464     /* XPath doesn't select elements with non-null default namespace with unqualified selectors, XSLPattern does */
5465     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("//elem/c"), &list));
5466     len = 0;
5467     ole_check(IXMLDOMNodeList_get_length(list, &len));
5468     /* should select <elem><c> and <elem xmlns='...'><c> but not <elem><foo:c> */
5469     todo_wine ok(len == 3, "expected 3 entries in list, got %d\n", len);
5470     IXMLDOMNodeList_Release(list);
5471
5472     /* for XSLPattern start index is 0, for XPath it's 1 */
5473     ole_check(IXMLDOMDocument2_selectNodes(doc, _bstr_("root//elem[0]"), &list));
5474     len = 0;
5475     ole_check(IXMLDOMNodeList_get_length(list, &len));
5476     todo_wine ok(len != 0, "expected filled list\n");
5477     if (len)
5478         todo_wine expect_list_and_release(list, "E1.E2.D1");
5479
5480     IXMLDOMDocument2_Release(doc);
5481     free_bstrs();
5482 }
5483
5484 static void test_splitText(void)
5485 {
5486     IXMLDOMCDATASection *cdata;
5487     IXMLDOMElement *root;
5488     IXMLDOMDocument *doc;
5489     IXMLDOMText *text, *text2;
5490     IXMLDOMNode *node;
5491     VARIANT var;
5492     VARIANT_BOOL success;
5493     LONG length;
5494     HRESULT hr;
5495
5496     doc = create_document(&IID_IXMLDOMDocument);
5497     if (!doc) return;
5498
5499     hr = IXMLDOMDocument_loadXML(doc, _bstr_("<root></root>"), &success);
5500     ok(hr == S_OK, "got 0x%08x\n", hr);
5501
5502     hr = IXMLDOMDocument_get_documentElement(doc, &root);
5503     ok(hr == S_OK, "got 0x%08x\n", hr);
5504
5505     hr = IXMLDOMDocument_createCDATASection(doc, _bstr_("beautiful plumage"), &cdata);
5506     ok(hr == S_OK, "got 0x%08x\n", hr);
5507
5508     V_VT(&var) = VT_EMPTY;
5509     hr = IXMLDOMElement_appendChild(root, (IXMLDOMNode*)cdata, NULL);
5510     ok(hr == S_OK, "got 0x%08x\n", hr);
5511
5512     length = 0;
5513     hr = IXMLDOMCDATASection_get_length(cdata, &length);
5514     ok(hr == S_OK, "got 0x%08x\n", hr);
5515     ok(length > 0, "got %d\n", length);
5516
5517     hr = IXMLDOMCDATASection_splitText(cdata, 0, NULL);
5518     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
5519
5520     text = (void*)0xdeadbeef;
5521     /* negative offset */
5522     hr = IXMLDOMCDATASection_splitText(cdata, -1, &text);
5523     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
5524     ok(text == (void*)0xdeadbeef, "got %p\n", text);
5525
5526     text = (void*)0xdeadbeef;
5527     /* offset outside data */
5528     hr = IXMLDOMCDATASection_splitText(cdata, length + 1, &text);
5529     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
5530     ok(text == 0, "got %p\n", text);
5531
5532     text = (void*)0xdeadbeef;
5533     /* offset outside data */
5534     hr = IXMLDOMCDATASection_splitText(cdata, length, &text);
5535     ok(hr == S_FALSE, "got 0x%08x\n", hr);
5536     ok(text == 0, "got %p\n", text);
5537
5538     /* no empty node created */
5539     node = (void*)0xdeadbeef;
5540     hr = IXMLDOMCDATASection_get_nextSibling(cdata, &node);
5541     ok(hr == S_FALSE, "got 0x%08x\n", hr);
5542     ok(node == 0, "got %p\n", text);
5543
5544     hr = IXMLDOMCDATASection_splitText(cdata, 10, &text);
5545     ok(hr == S_OK, "got 0x%08x\n", hr);
5546
5547     length = 0;
5548     hr = IXMLDOMText_get_length(text, &length);
5549     ok(hr == S_OK, "got 0x%08x\n", hr);
5550     ok(length == 7, "got %d\n", length);
5551
5552     hr = IXMLDOMCDATASection_get_nextSibling(cdata, &node);
5553     ok(hr == S_OK, "got 0x%08x\n", hr);
5554     IXMLDOMNode_Release(node);
5555
5556     /* split new text node */
5557     hr = IXMLDOMText_get_length(text, &length);
5558     ok(hr == S_OK, "got 0x%08x\n", hr);
5559
5560     node = (void*)0xdeadbeef;
5561     hr = IXMLDOMText_get_nextSibling(text, &node);
5562     ok(hr == S_FALSE, "got 0x%08x\n", hr);
5563     ok(node == 0, "got %p\n", text);
5564
5565     hr = IXMLDOMText_splitText(text, 0, NULL);
5566     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
5567
5568     text2 = (void*)0xdeadbeef;
5569     /* negative offset */
5570     hr = IXMLDOMText_splitText(text, -1, &text2);
5571     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
5572     ok(text2 == (void*)0xdeadbeef, "got %p\n", text2);
5573
5574     text2 = (void*)0xdeadbeef;
5575     /* offset outside data */
5576     hr = IXMLDOMText_splitText(text, length + 1, &text2);
5577     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
5578     ok(text2 == 0, "got %p\n", text2);
5579
5580     text2 = (void*)0xdeadbeef;
5581     /* offset outside data */
5582     hr = IXMLDOMText_splitText(text, length, &text2);
5583     ok(hr == S_FALSE, "got 0x%08x\n", hr);
5584     ok(text2 == 0, "got %p\n", text);
5585
5586     text2 = 0;
5587     hr = IXMLDOMText_splitText(text, 4, &text2);
5588     todo_wine ok(hr == S_OK, "got 0x%08x\n", hr);
5589     if (text2) IXMLDOMText_Release(text2);
5590
5591     node = 0;
5592     hr = IXMLDOMText_get_nextSibling(text, &node);
5593     todo_wine ok(hr == S_OK, "got 0x%08x\n", hr);
5594     if (node) IXMLDOMNode_Release(node);
5595
5596     IXMLDOMText_Release(text);
5597     IXMLDOMElement_Release(root);
5598     IXMLDOMCDATASection_Release(cdata);
5599     free_bstrs();
5600 }
5601
5602 static void test_getQualifiedItem(void)
5603 {
5604     IXMLDOMDocument *doc;
5605     IXMLDOMElement *element;
5606     IXMLDOMNode *pr_node, *node;
5607     IXMLDOMNodeList *root_list;
5608     IXMLDOMNamedNodeMap *map;
5609     VARIANT_BOOL b;
5610     BSTR str;
5611     LONG len;
5612     HRESULT hr;
5613
5614     doc = create_document(&IID_IXMLDOMDocument);
5615     if (!doc) return;
5616
5617     str = SysAllocString( szComplete4 );
5618     hr = IXMLDOMDocument_loadXML( doc, str, &b );
5619     ok( hr == S_OK, "loadXML failed\n");
5620     ok( b == VARIANT_TRUE, "failed to load XML string\n");
5621     SysFreeString( str );
5622
5623     hr = IXMLDOMDocument_get_documentElement(doc, &element);
5624     ok( hr == S_OK, "ret %08x\n", hr);
5625
5626     hr = IXMLDOMElement_get_childNodes(element, &root_list);
5627     ok( hr == S_OK, "ret %08x\n", hr);
5628
5629     hr = IXMLDOMNodeList_get_item(root_list, 1, &pr_node);
5630     ok( hr == S_OK, "ret %08x\n", hr);
5631     IXMLDOMNodeList_Release(root_list);
5632
5633     hr = IXMLDOMNode_get_attributes(pr_node, &map);
5634     ok( hr == S_OK, "ret %08x\n", hr);
5635     IXMLDOMNode_Release(pr_node);
5636
5637     hr = IXMLDOMNamedNodeMap_get_length(map, &len);
5638     ok( hr == S_OK, "ret %08x\n", hr);
5639     ok( len == 3, "length %d\n", len);
5640
5641     hr = IXMLDOMNamedNodeMap_getQualifiedItem(map, NULL, NULL, NULL);
5642     ok( hr == E_INVALIDARG, "ret %08x\n", hr);
5643
5644     node = (void*)0xdeadbeef;
5645     hr = IXMLDOMNamedNodeMap_getQualifiedItem(map, NULL, NULL, &node);
5646     ok( hr == E_INVALIDARG, "ret %08x\n", hr);
5647     ok( node == (void*)0xdeadbeef, "got %p\n", node);
5648
5649     hr = IXMLDOMNamedNodeMap_getQualifiedItem(map, _bstr_("id"), NULL, NULL);
5650     ok( hr == E_INVALIDARG, "ret %08x\n", hr);
5651
5652     hr = IXMLDOMNamedNodeMap_getQualifiedItem(map, _bstr_("id"), NULL, &node);
5653     ok( hr == S_OK, "ret %08x\n", hr);
5654     IXMLDOMNode_Release(node);
5655
5656     IXMLDOMNamedNodeMap_Release( map );
5657     IXMLDOMElement_Release( element );
5658     IXMLDOMDocument_Release( doc );
5659     free_bstrs();
5660 }
5661
5662 static void test_removeQualifiedItem(void)
5663 {
5664     IXMLDOMDocument *doc;
5665     IXMLDOMElement *element;
5666     IXMLDOMNode *pr_node, *node;
5667     IXMLDOMNodeList *root_list;
5668     IXMLDOMNamedNodeMap *map;
5669     VARIANT_BOOL b;
5670     BSTR str;
5671     LONG len;
5672     HRESULT hr;
5673
5674     doc = create_document(&IID_IXMLDOMDocument);
5675     if (!doc) return;
5676
5677     str = SysAllocString( szComplete4 );
5678     hr = IXMLDOMDocument_loadXML( doc, str, &b );
5679     ok( hr == S_OK, "loadXML failed\n");
5680     ok( b == VARIANT_TRUE, "failed to load XML string\n");
5681     SysFreeString( str );
5682
5683     hr = IXMLDOMDocument_get_documentElement(doc, &element);
5684     ok( hr == S_OK, "ret %08x\n", hr);
5685
5686     hr = IXMLDOMElement_get_childNodes(element, &root_list);
5687     ok( hr == S_OK, "ret %08x\n", hr);
5688
5689     hr = IXMLDOMNodeList_get_item(root_list, 1, &pr_node);
5690     ok( hr == S_OK, "ret %08x\n", hr);
5691     IXMLDOMNodeList_Release(root_list);
5692
5693     hr = IXMLDOMNode_get_attributes(pr_node, &map);
5694     ok( hr == S_OK, "ret %08x\n", hr);
5695     IXMLDOMNode_Release(pr_node);
5696
5697     hr = IXMLDOMNamedNodeMap_get_length(map, &len);
5698     ok( hr == S_OK, "ret %08x\n", hr);
5699     ok( len == 3, "length %d\n", len);
5700
5701     hr = IXMLDOMNamedNodeMap_removeQualifiedItem(map, NULL, NULL, NULL);
5702     ok( hr == E_INVALIDARG, "ret %08x\n", hr);
5703
5704     node = (void*)0xdeadbeef;
5705     hr = IXMLDOMNamedNodeMap_removeQualifiedItem(map, NULL, NULL, &node);
5706     ok( hr == E_INVALIDARG, "ret %08x\n", hr);
5707     ok( node == (void*)0xdeadbeef, "got %p\n", node);
5708
5709     /* out pointer is optional */
5710     hr = IXMLDOMNamedNodeMap_removeQualifiedItem(map, _bstr_("id"), NULL, NULL);
5711     ok( hr == S_OK, "ret %08x\n", hr);
5712
5713     /* already removed */
5714     hr = IXMLDOMNamedNodeMap_removeQualifiedItem(map, _bstr_("id"), NULL, NULL);
5715     ok( hr == S_FALSE, "ret %08x\n", hr);
5716
5717     hr = IXMLDOMNamedNodeMap_removeQualifiedItem(map, _bstr_("vr"), NULL, &node);
5718     ok( hr == S_OK, "ret %08x\n", hr);
5719     IXMLDOMNode_Release(node);
5720
5721     IXMLDOMNamedNodeMap_Release( map );
5722     IXMLDOMElement_Release( element );
5723     IXMLDOMDocument_Release( doc );
5724     free_bstrs();
5725 }
5726
5727 static void test_get_ownerDocument(void)
5728 {
5729     IXMLDOMDocument *doc1, *doc2, *doc3;
5730     IXMLDOMDocument2 *doc, *doc_owner;
5731     IXMLDOMNode *node;
5732     VARIANT_BOOL b;
5733     VARIANT var;
5734     HRESULT hr;
5735     BSTR str;
5736
5737     doc = create_document(&IID_IXMLDOMDocument2);
5738     if (!doc) return;
5739
5740     str = SysAllocString( szComplete4 );
5741     hr = IXMLDOMDocument_loadXML( doc, str, &b );
5742     ok( hr == S_OK, "loadXML failed\n");
5743     ok( b == VARIANT_TRUE, "failed to load XML string\n");
5744     SysFreeString( str );
5745
5746     hr = IXMLDOMDocument2_getProperty(doc, _bstr_("SelectionLanguage"), &var);
5747     ok( hr == S_OK, "got 0x%08x\n", hr);
5748     ok( lstrcmpW(V_BSTR(&var), _bstr_("XSLPattern")) == 0, "expected XSLPattern\n");
5749     VariantClear(&var);
5750
5751     /* set to XPath and check that new instances use it */
5752     V_VT(&var) = VT_BSTR;
5753     V_BSTR(&var) = _bstr_("XPath");
5754     hr = IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionLanguage"), var);
5755     ok( hr == S_OK, "got 0x%08x\n", hr);
5756
5757     V_VT(&var) = VT_BSTR;
5758     V_BSTR(&var) = _bstr_("xmlns:wi=\'www.winehq.org\'");
5759     hr = IXMLDOMDocument2_setProperty(doc, _bstr_("SelectionNamespaces"), var);
5760     todo_wine ok( hr == S_OK, "got 0x%08x\n", hr);
5761
5762     hr = IXMLDOMDocument2_get_firstChild(doc, &node);
5763     ok( hr == S_OK, "got 0x%08x\n", hr);
5764
5765     hr = IXMLDOMNode_get_ownerDocument(node, &doc1);
5766     ok( hr == S_OK, "got 0x%08x\n", hr);
5767
5768     VariantClear(&var);
5769     hr = IXMLDOMDocument_QueryInterface(doc1, &IID_IXMLDOMDocument2, (void**)&doc_owner);
5770     ok( hr == S_OK, "got 0x%08x\n", hr);
5771     ok( doc_owner != doc, "got %p, doc %p\n", doc_owner, doc);
5772     hr = IXMLDOMDocument2_getProperty(doc_owner, _bstr_("SelectionNamespaces"), &var);
5773     todo_wine ok( hr == S_OK, "got 0x%08x\n", hr);
5774     todo_wine ok( lstrcmpW(V_BSTR(&var), _bstr_("xmlns:wi=\'www.winehq.org\'")) == 0, "expected previously set value\n");
5775     VariantClear(&var);
5776
5777     hr = IXMLDOMDocument2_getProperty(doc_owner, _bstr_("SelectionLanguage"), &var);
5778     ok( hr == S_OK, "got 0x%08x\n", hr);
5779     ok( lstrcmpW(V_BSTR(&var), _bstr_("XPath")) == 0, "expected XPath\n");
5780     VariantClear(&var);
5781     IXMLDOMDocument2_Release(doc_owner);
5782
5783     hr = IXMLDOMNode_get_ownerDocument(node, &doc2);
5784     ok( hr == S_OK, "got 0x%08x\n", hr);
5785     IXMLDOMNode_Release(node);
5786
5787     ok(doc1 != doc2, "got %p, expected %p. original %p\n", doc2, doc1, doc);
5788
5789     /* reload */
5790     str = SysAllocString( szComplete4 );
5791     hr = IXMLDOMDocument2_loadXML( doc, str, &b );
5792     ok( hr == S_OK, "got 0x%08x\n", hr);
5793     ok( b == VARIANT_TRUE, "failed to load XML string\n");
5794     SysFreeString( str );
5795
5796     /* property retained even after reload */
5797     VariantClear(&var);
5798     hr = IXMLDOMDocument2_getProperty(doc, _bstr_("SelectionNamespaces"), &var);
5799     todo_wine ok( hr == S_OK, "got 0x%08x\n", hr);
5800     todo_wine ok( lstrcmpW(V_BSTR(&var), _bstr_("xmlns:wi=\'www.winehq.org\'")) == 0, "expected previously set value\n");
5801     VariantClear(&var);
5802
5803     hr = IXMLDOMDocument2_get_firstChild(doc, &node);
5804     ok( hr == S_OK, "got 0x%08x\n", hr);
5805
5806     hr = IXMLDOMNode_get_ownerDocument(node, &doc3);
5807     ok( hr == S_OK, "got 0x%08x\n", hr);
5808     IXMLDOMNode_Release(node);
5809
5810     hr = IXMLDOMDocument_QueryInterface(doc3, &IID_IXMLDOMDocument2, (void**)&doc_owner);
5811     ok( hr == S_OK, "got 0x%08x\n", hr);
5812     ok(doc3 != doc1 && doc3 != doc2 && doc_owner != doc, "got %p, (%p, %p, %p)\n", doc3, doc, doc1, doc2);
5813
5814     IXMLDOMDocument_Release(doc1);
5815     IXMLDOMDocument_Release(doc2);
5816     IXMLDOMDocument_Release(doc3);
5817     IXMLDOMDocument2_Release(doc);
5818     free_bstrs();
5819 }
5820
5821 static void test_setAttributeNode(void)
5822 {
5823     IXMLDOMDocument *doc, *doc2;
5824     IXMLDOMElement *elem;
5825     IXMLDOMAttribute *attr, *attr2, *ret_attr;
5826     VARIANT_BOOL b;
5827     HRESULT hr;
5828     BSTR str;
5829
5830     doc = create_document(&IID_IXMLDOMDocument);
5831     if (!doc) return;
5832
5833     str = SysAllocString( szComplete4 );
5834     hr = IXMLDOMDocument2_loadXML( doc, str, &b );
5835     ok( hr == S_OK, "loadXML failed\n");
5836     ok( b == VARIANT_TRUE, "failed to load XML string\n");
5837     SysFreeString( str );
5838
5839     hr = IXMLDOMDocument_get_documentElement(doc, &elem);
5840     ok( hr == S_OK, "got 0x%08x\n", hr);
5841
5842     ret_attr = (void*)0xdeadbeef;
5843     hr = IXMLDOMElement_setAttributeNode(elem, NULL, &ret_attr);
5844     ok( hr == E_INVALIDARG, "got 0x%08x\n", hr);
5845     ok( ret_attr == (void*)0xdeadbeef, "got %p\n", ret_attr);
5846
5847     hr = IXMLDOMDocument_createAttribute(doc, _bstr_("attr"), &attr);
5848     ok( hr == S_OK, "got 0x%08x\n", hr);
5849
5850     ret_attr = (void*)0xdeadbeef;
5851     hr = IXMLDOMElement_setAttributeNode(elem, attr, &ret_attr);
5852     todo_wine ok( hr == S_OK, "got 0x%08x\n", hr);
5853     todo_wine ok( ret_attr == NULL, "got %p\n", ret_attr);
5854
5855     attr2 = NULL;
5856     hr = IXMLDOMElement_getAttributeNode(elem, _bstr_("attr"), &attr2);
5857     todo_wine ok( hr == S_OK, "got 0x%08x\n", hr);
5858     if (attr2) IXMLDOMAttribute_Release(attr2);
5859
5860     /* try to add it another time */
5861     ret_attr = (void*)0xdeadbeef;
5862     hr = IXMLDOMElement_setAttributeNode(elem, attr, &ret_attr);
5863     todo_wine ok( hr == E_FAIL, "got 0x%08x\n", hr);
5864     ok( ret_attr == (void*)0xdeadbeef, "got %p\n", ret_attr);
5865
5866     IXMLDOMElement_Release(elem);
5867
5868     /* add attribute already attached to another document */
5869     doc2 = create_document(&IID_IXMLDOMDocument);
5870
5871     str = SysAllocString( szComplete4 );
5872     hr = IXMLDOMDocument_loadXML( doc2, str, &b );
5873     ok( hr == S_OK, "loadXML failed\n");
5874     ok( b == VARIANT_TRUE, "failed to load XML string\n");
5875     SysFreeString( str );
5876
5877     hr = IXMLDOMDocument_get_documentElement(doc2, &elem);
5878     ok( hr == S_OK, "got 0x%08x\n", hr);
5879     hr = IXMLDOMElement_setAttributeNode(elem, attr, NULL);
5880     todo_wine ok( hr == E_FAIL, "got 0x%08x\n", hr);
5881     IXMLDOMElement_Release(elem);
5882
5883     IXMLDOMDocument_Release(doc2);
5884
5885     IXMLDOMAttribute_Release(attr);
5886     IXMLDOMDocument_Release(doc);
5887     free_bstrs();
5888 }
5889
5890 static void test_put_dataType(void)
5891 {
5892     IXMLDOMCDATASection *cdata;
5893     IXMLDOMDocument *doc;
5894     VARIANT_BOOL b;
5895     HRESULT hr;
5896     BSTR str;
5897
5898     doc = create_document(&IID_IXMLDOMDocument);
5899     if (!doc) return;
5900
5901     str = SysAllocString( szComplete4 );
5902     hr = IXMLDOMDocument_loadXML( doc, str, &b );
5903     ok( hr == S_OK, "loadXML failed\n");
5904     ok( b == VARIANT_TRUE, "failed to load XML string\n");
5905     SysFreeString( str );
5906
5907     hr = IXMLDOMDocument_createCDATASection(doc, _bstr_("test"), &cdata);
5908     ok( hr == S_OK, "got 0x%08x\n", hr);
5909     hr = IXMLDOMCDATASection_put_dataType(cdata, _bstr_("number"));
5910     ok( hr == E_FAIL, "got 0x%08x\n", hr);
5911     hr = IXMLDOMCDATASection_put_dataType(cdata, _bstr_("string"));
5912     ok( hr == E_FAIL, "got 0x%08x\n", hr);
5913     IXMLDOMCDATASection_Release(cdata);
5914
5915     IXMLDOMDocument_Release(doc);
5916     free_bstrs();
5917 }
5918
5919 static void test_createNode(void)
5920 {
5921     IXMLDOMDocument *doc;
5922     IXMLDOMElement *elem;
5923     IXMLDOMNode *node;
5924     VARIANT v, var;
5925     BSTR prefix, str;
5926     HRESULT hr;
5927
5928     doc = create_document(&IID_IXMLDOMDocument);
5929     if (!doc) return;
5930
5931     /* NODE_ELEMENT nodes */
5932     /* 1. specified namespace */
5933     V_VT(&v) = VT_I4;
5934     V_I4(&v) = NODE_ELEMENT;
5935
5936     hr = IXMLDOMDocument_createNode(doc, v, _bstr_("ns1:test"), _bstr_("http://winehq.org"), &node);
5937     ok( hr == S_OK, "got 0x%08x\n", hr);
5938     prefix = NULL;
5939     hr = IXMLDOMNode_get_prefix(node, &prefix);
5940     ok( hr == S_OK, "got 0x%08x\n", hr);
5941     ok(lstrcmpW(prefix, _bstr_("ns1")) == 0, "wrong prefix\n");
5942     SysFreeString(prefix);
5943     IXMLDOMNode_Release(node);
5944
5945     /* 2. default namespace */
5946     hr = IXMLDOMDocument_createNode(doc, v, _bstr_("test"), _bstr_("http://winehq.org/default"), &node);
5947     ok( hr == S_OK, "got 0x%08x\n", hr);
5948     prefix = (void*)0xdeadbeef;
5949     hr = IXMLDOMNode_get_prefix(node, &prefix);
5950     ok( hr == S_FALSE, "got 0x%08x\n", hr);
5951     ok(prefix == 0, "expected empty prefix, got %p\n", prefix);
5952
5953     hr = IXMLDOMNode_QueryInterface(node, &IID_IXMLDOMElement, (void**)&elem);
5954     ok( hr == S_OK, "got 0x%08x\n", hr);
5955
5956     V_VT(&var) = VT_BSTR;
5957     hr = IXMLDOMElement_getAttribute(elem, _bstr_("xmlns"), &var);
5958     ok( hr == S_FALSE, "got 0x%08x\n", hr);
5959     ok( V_VT(&var) == VT_NULL, "got %d\n", V_VT(&var));
5960
5961     str = NULL;
5962     hr = IXMLDOMElement_get_namespaceURI(elem, &str);
5963     ok( hr == S_OK, "got 0x%08x\n", hr);
5964     ok( lstrcmpW(str, _bstr_("http://winehq.org/default")) == 0, "expected default namespace\n");
5965     SysFreeString(str);
5966
5967     IXMLDOMElement_Release(elem);
5968     IXMLDOMNode_Release(node);
5969
5970     IXMLDOMDocument_Release(doc);
5971     free_bstrs();
5972 }
5973
5974 static void test_get_prefix(void)
5975 {
5976     IXMLDOMDocument *doc;
5977     IXMLDOMElement *element;
5978     HRESULT hr;
5979     BSTR str;
5980
5981     doc = create_document(&IID_IXMLDOMDocument);
5982     if (!doc) return;
5983
5984     /* no prefix */
5985     hr = IXMLDOMDocument_createElement(doc, _bstr_("elem"), &element);
5986     ok( hr == S_OK, "got 0x%08x\n", hr);
5987
5988     hr = IXMLDOMElement_get_prefix(element, NULL);
5989     ok( hr == E_INVALIDARG, "got 0x%08x\n", hr);
5990
5991     str = (void*)0xdeadbeef;
5992     hr = IXMLDOMElement_get_prefix(element, &str);
5993     ok( hr == S_FALSE, "got 0x%08x\n", hr);
5994     ok( str == 0, "got %p\n", str);
5995
5996     IXMLDOMElement_Release(element);
5997
5998     /* with prefix */
5999     hr = IXMLDOMDocument_createElement(doc, _bstr_("a:elem"), &element);
6000     ok( hr == S_OK, "got 0x%08x\n", hr);
6001
6002     str = (void*)0xdeadbeef;
6003     hr = IXMLDOMElement_get_prefix(element, &str);
6004     ok( hr == S_OK, "got 0x%08x\n", hr);
6005     ok( lstrcmpW(str, _bstr_("a")) == 0, "expected prefix \"a\"\n");
6006     SysFreeString(str);
6007
6008     str = (void*)0xdeadbeef;
6009     hr = IXMLDOMElement_get_namespaceURI(element, &str);
6010     ok( hr == S_FALSE, "got 0x%08x\n", hr);
6011     ok( str == 0, "got %p\n", str);
6012
6013     IXMLDOMElement_Release(element);
6014
6015     IXMLDOMDocument_Release(doc);
6016     free_bstrs();
6017 }
6018
6019 static void test_selectSingleNode(void)
6020 {
6021     IXMLDOMDocument *doc;
6022     IXMLDOMNodeList *list;
6023     IXMLDOMNode *node;
6024     VARIANT_BOOL b;
6025     HRESULT hr;
6026     LONG len;
6027     BSTR str;
6028
6029     doc = create_document(&IID_IXMLDOMDocument);
6030     if (!doc) return;
6031
6032     hr = IXMLDOMDocument_selectSingleNode(doc, NULL, NULL);
6033     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
6034
6035     hr = IXMLDOMDocument_selectNodes(doc, NULL, NULL);
6036     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
6037
6038     str = SysAllocString( szComplete4 );
6039     hr = IXMLDOMDocument_loadXML( doc, str, &b );
6040     ok( hr == S_OK, "loadXML failed\n");
6041     ok( b == VARIANT_TRUE, "failed to load XML string\n");
6042     SysFreeString( str );
6043
6044     hr = IXMLDOMDocument_selectSingleNode(doc, NULL, NULL);
6045     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
6046
6047     hr = IXMLDOMDocument_selectNodes(doc, NULL, NULL);
6048     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
6049
6050     hr = IXMLDOMDocument_selectSingleNode(doc, _bstr_("lc"), NULL);
6051     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
6052
6053     hr = IXMLDOMDocument_selectNodes(doc, _bstr_("lc"), NULL);
6054     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
6055
6056     hr = IXMLDOMDocument_selectSingleNode(doc, _bstr_("lc"), &node);
6057     ok(hr == S_OK, "got 0x%08x\n", hr);
6058     IXMLDOMNode_Release(node);
6059
6060     hr = IXMLDOMDocument_selectNodes(doc, _bstr_("lc"), &list);
6061     ok(hr == S_OK, "got 0x%08x\n", hr);
6062     IXMLDOMNodeList_Release(list);
6063
6064     list = (void*)0xdeadbeef;
6065     hr = IXMLDOMDocument_selectNodes(doc, NULL, &list);
6066     ok(hr == E_INVALIDARG, "got 0x%08x\n", hr);
6067     ok(list == (void*)0xdeadbeef, "got %p\n", list);
6068
6069     node = (void*)0xdeadbeef;
6070     hr = IXMLDOMDocument_selectSingleNode(doc, _bstr_("nonexistent"), &node);
6071     ok(hr == S_FALSE, "got 0x%08x\n", hr);
6072     ok(node == 0, "got %p\n", node);
6073
6074     list = (void*)0xdeadbeef;
6075     hr = IXMLDOMDocument_selectNodes(doc, _bstr_("nonexistent"), &list);
6076     ok(hr == S_OK, "got 0x%08x\n", hr);
6077     len = 1;
6078     hr = IXMLDOMNodeList_get_length(list, &len);
6079     ok(hr == S_OK, "got 0x%08x\n", hr);
6080     ok(len == 0, "got %d\n", len);
6081     IXMLDOMNodeList_Release(list);
6082
6083     IXMLDOMDocument_Release(doc);
6084     free_bstrs();
6085 }
6086
6087 START_TEST(domdoc)
6088 {
6089     IXMLDOMDocument *doc;
6090     HRESULT hr;
6091
6092     hr = CoInitialize( NULL );
6093     ok( hr == S_OK, "failed to init com\n");
6094     if (hr != S_OK)
6095         return;
6096
6097     test_XMLHTTP();
6098
6099     hr = CoCreateInstance( &CLSID_DOMDocument, NULL, CLSCTX_INPROC_SERVER, &IID_IXMLDOMDocument, (void**)&doc );
6100     if (hr != S_OK)
6101     {
6102         win_skip("IXMLDOMDocument is not available (0x%08x)\n", hr);
6103         return;
6104     }
6105
6106     IXMLDOMDocument_Release(doc);
6107
6108     test_domdoc();
6109     test_persiststreaminit();
6110     test_domnode();
6111     test_refs();
6112     test_create();
6113     test_getElementsByTagName();
6114     test_get_text();
6115     test_get_childNodes();
6116     test_get_firstChild();
6117     test_get_lastChild();
6118     test_removeChild();
6119     test_replaceChild();
6120     test_removeNamedItem();
6121     test_IXMLDOMDocument2();
6122     test_XPath();
6123     test_XSLPattern();
6124     test_cloneNode();
6125     test_xmlTypes();
6126     test_nodeTypeTests();
6127     test_DocumentSaveToDocument();
6128     test_DocumentSaveToFile();
6129     test_testTransforms();
6130     test_Namespaces();
6131     test_FormattingXML();
6132     test_nodeTypedValue();
6133     test_TransformWithLoadingLocalFile();
6134     test_put_nodeValue();
6135     test_document_IObjectSafety();
6136     test_splitText();
6137     test_getQualifiedItem();
6138     test_removeQualifiedItem();
6139     test_get_ownerDocument();
6140     test_setAttributeNode();
6141     test_put_dataType();
6142     test_createNode();
6143     test_get_prefix();
6144     test_default_properties();
6145     test_selectSingleNode();
6146
6147     CoUninitialize();
6148 }