msxml3: Support setting namespaces feature to default value.
[wine/multimedia.git] / dlls / msxml3 / tests / saxreader.c
blob6493776d69399c2c7ac094fc599700fa1fa1ce32
1 /*
2 * XML test
4 * Copyright 2008 Piotr Caban
5 * Copyright 2011 Thomas Mullaly
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.
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.
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
22 #define COBJMACROS
23 #define CONST_VTABLE
25 #include <stdio.h>
26 #include <assert.h>
28 #include "windows.h"
29 #include "ole2.h"
30 #include "msxml2.h"
31 #include "ocidl.h"
33 #include "wine/test.h"
35 #define EXPECT_HR(hr,hr_exp) \
36 ok(hr == hr_exp, "got 0x%08x, expected 0x%08x\n", hr, hr_exp)
38 #define EXPECT_REF(obj,ref) _expect_ref((IUnknown*)obj, ref, __LINE__)
39 static void _expect_ref(IUnknown* obj, ULONG ref, int line)
41 ULONG rc = IUnknown_AddRef(obj);
42 IUnknown_Release(obj);
43 ok_(__FILE__,line)(rc-1 == ref, "expected refcount %d, got %d\n", ref, rc-1);
46 static BSTR alloc_str_from_narrow(const char *str)
48 int len = MultiByteToWideChar(CP_ACP, 0, str, -1, NULL, 0);
49 BSTR ret = SysAllocStringLen(NULL, len - 1); /* NUL character added automatically */
50 MultiByteToWideChar(CP_ACP, 0, str, -1, ret, len);
51 return ret;
54 static BSTR alloced_bstrs[512];
55 static int alloced_bstrs_count;
57 static BSTR _bstr_(const char *str)
59 assert(alloced_bstrs_count < sizeof(alloced_bstrs)/sizeof(alloced_bstrs[0]));
60 alloced_bstrs[alloced_bstrs_count] = alloc_str_from_narrow(str);
61 return alloced_bstrs[alloced_bstrs_count++];
64 static void free_bstrs(void)
66 int i;
67 for (i = 0; i < alloced_bstrs_count; i++)
68 SysFreeString(alloced_bstrs[i]);
69 alloced_bstrs_count = 0;
72 typedef enum _CH {
73 CH_ENDTEST,
74 CH_PUTDOCUMENTLOCATOR,
75 CH_STARTDOCUMENT,
76 CH_ENDDOCUMENT,
77 CH_STARTPREFIXMAPPING,
78 CH_ENDPREFIXMAPPING,
79 CH_STARTELEMENT,
80 CH_ENDELEMENT,
81 CH_CHARACTERS,
82 CH_IGNORABLEWHITESPACE,
83 CH_PROCESSINGINSTRUCTION,
84 CH_SKIPPEDENTITY
85 } CH;
87 static const WCHAR szSimpleXML[] = {
88 '<','?','x','m','l',' ','v','e','r','s','i','o','n','=','\"','1','.','0','\"',' ','?','>','\n',
89 '<','B','a','n','k','A','c','c','o','u','n','t','>','\n',
90 ' ',' ',' ','<','N','u','m','b','e','r','>','1','2','3','4','<','/','N','u','m','b','e','r','>','\n',
91 ' ',' ',' ','<','N','a','m','e','>','C','a','p','t','a','i','n',' ','A','h','a','b','<','/','N','a','m','e','>','\n',
92 '<','/','B','a','n','k','A','c','c','o','u','n','t','>','\n','\0'
95 static const WCHAR szCarriageRetTest[] = {
96 '<','?','x','m','l',' ','v','e','r','s','i','o','n','=','"','1','.','0','"','?','>','\r','\n',
97 '<','B','a','n','k','A','c','c','o','u','n','t','>','\r','\n',
98 '\t','<','N','u','m','b','e','r','>','1','2','3','4','<','/','N','u','m','b','e','r','>','\r','\n',
99 '\t','<','N','a','m','e','>','C','a','p','t','a','i','n',' ','A','h','a','b','<','/','N','a','m','e','>','\r','\n',
100 '<','/','B','a','n','k','A','c','c','o','u','n','t','>','\0'
103 static const WCHAR szUtf16XML[] = {
104 '<','?','x','m','l',' ','v','e','r','s','i','o','n','=','"','1','.','0','"',' ',
105 'e','n','c','o','d','i','n','g','=','"','U','T','F','-','1','6','"',' ',
106 's','t','a','n','d','a','l','o','n','e','=','"','n','o','"','?','>','\r','\n'
109 static const CHAR szUtf16BOM[] = {0xff, 0xfe};
111 static const CHAR szUtf8XML[] =
112 "<?xml version=\"1.0\" encoding=\"UTF-8\" standalone=\"no\"?>\r\n";
114 static const CHAR szTestXML[] =
115 "<?xml version=\"1.0\" ?>\n"
116 "<BankAccount>\n"
117 " <Number>1234</Number>\n"
118 " <Name>Captain Ahab</Name>\n"
119 "</BankAccount>\n";
121 typedef struct _contenthandlercheck {
122 CH id;
123 int line;
124 int column;
125 const char *arg1;
126 const char *arg2;
127 const char *arg3;
128 } content_handler_test;
130 static content_handler_test contentHandlerTest1[] = {
131 { CH_PUTDOCUMENTLOCATOR, 0, 0 },
132 { CH_STARTDOCUMENT, 0, 0 },
133 { CH_STARTELEMENT, 2, 14, "", "BankAccount", "BankAccount" },
134 { CH_CHARACTERS, 2, 14, "\n " },
135 { CH_STARTELEMENT, 3, 12, "", "Number", "Number" },
136 { CH_CHARACTERS, 3, 12, "1234" },
137 { CH_ENDELEMENT, 3, 18, "", "Number", "Number" },
138 { CH_CHARACTERS, 3, 25, "\n " },
139 { CH_STARTELEMENT, 4, 10, "", "Name", "Name" },
140 { CH_CHARACTERS, 4, 10, "Captain Ahab" },
141 { CH_ENDELEMENT, 4, 24, "", "Name", "Name" },
142 { CH_CHARACTERS, 4, 29, "\n" },
143 { CH_ENDELEMENT, 5, 3, "", "BankAccount", "BankAccount" },
144 { CH_ENDDOCUMENT, 0, 0 },
145 { CH_ENDTEST }
148 static content_handler_test contentHandlerTest2[] = {
149 { CH_PUTDOCUMENTLOCATOR, 0, 0 },
150 { CH_STARTDOCUMENT, 0, 0 },
151 { CH_STARTELEMENT, 2, 14, "", "BankAccount", "BankAccount" },
152 { CH_CHARACTERS, 2, 14, "\n" },
153 { CH_CHARACTERS, 2, 16, "\t" },
154 { CH_STARTELEMENT, 3, 10, "", "Number", "Number" },
155 { CH_CHARACTERS, 3, 10, "1234" },
156 { CH_ENDELEMENT, 3, 16, "", "Number", "Number" },
157 { CH_CHARACTERS, 3, 23, "\n" },
158 { CH_CHARACTERS, 3, 25, "\t" },
159 { CH_STARTELEMENT, 4, 8, "", "Name", "Name" },
160 { CH_CHARACTERS, 4, 8, "Captain Ahab" },
161 { CH_ENDELEMENT, 4, 22, "", "Name", "Name" },
162 { CH_CHARACTERS, 4, 27, "\n" },
163 { CH_ENDELEMENT, 5, 3, "", "BankAccount", "BankAccount" },
164 { CH_ENDDOCUMENT, 0, 0 },
165 { CH_ENDTEST }
168 static content_handler_test *expectCall;
169 static ISAXLocator *locator;
171 static void test_saxstr(unsigned line, const WCHAR *szStr, int nStr, const char *szTest)
173 WCHAR buf[1024];
174 int len;
176 if(!szTest) {
177 ok_(__FILE__,line) (szStr == NULL, "szStr != NULL\n");
178 ok_(__FILE__,line) (nStr == 0, "nStr = %d, expected 0\n", nStr);
179 return;
182 len = strlen(szTest);
183 ok_(__FILE__,line) (len == nStr, "nStr = %d, expected %d (%s)\n", nStr, len, szTest);
184 if(len != nStr)
185 return;
187 MultiByteToWideChar(CP_ACP, 0, szTest, -1, buf, sizeof(buf)/sizeof(WCHAR));
188 ok_(__FILE__,line) (!memcmp(szStr, buf, len*sizeof(WCHAR)), "unexpected szStr %s, expected %s\n",
189 wine_dbgstr_wn(szStr, nStr), szTest);
192 static BOOL test_expect_call(CH id)
194 ok(expectCall->id == id, "unexpected call %d, expected %d\n", id, expectCall->id);
195 return expectCall->id == id;
198 static void test_locator(unsigned line, int loc_line, int loc_column)
200 int rcolumn, rline;
201 ISAXLocator_getLineNumber(locator, &rline);
202 ISAXLocator_getColumnNumber(locator, &rcolumn);
204 ok_(__FILE__,line) (rline == loc_line,
205 "unexpected line %d, expected %d\n", rline, loc_line);
206 ok_(__FILE__,line) (rcolumn == loc_column,
207 "unexpected column %d, expected %d\n", rcolumn, loc_column);
210 static HRESULT WINAPI contentHandler_QueryInterface(
211 ISAXContentHandler* iface,
212 REFIID riid,
213 void **ppvObject)
215 *ppvObject = NULL;
217 if(IsEqualGUID(riid, &IID_IUnknown) || IsEqualGUID(riid, &IID_ISAXContentHandler))
219 *ppvObject = iface;
221 else
223 return E_NOINTERFACE;
226 return S_OK;
229 static ULONG WINAPI contentHandler_AddRef(
230 ISAXContentHandler* iface)
232 return 2;
235 static ULONG WINAPI contentHandler_Release(
236 ISAXContentHandler* iface)
238 return 1;
241 static HRESULT WINAPI contentHandler_putDocumentLocator(
242 ISAXContentHandler* iface,
243 ISAXLocator *pLocator)
245 if(!test_expect_call(CH_PUTDOCUMENTLOCATOR))
246 return E_FAIL;
248 locator = pLocator;
249 test_locator(__LINE__, expectCall->line, expectCall->column);
251 expectCall++;
252 return S_OK;
255 static HRESULT WINAPI contentHandler_startDocument(
256 ISAXContentHandler* iface)
258 if(!test_expect_call(CH_STARTDOCUMENT))
259 return E_FAIL;
261 test_locator(__LINE__, expectCall->line, expectCall->column);
263 expectCall++;
264 return S_OK;
267 static HRESULT WINAPI contentHandler_endDocument(
268 ISAXContentHandler* iface)
270 if(!test_expect_call(CH_ENDDOCUMENT))
271 return E_FAIL;
273 test_locator(__LINE__, expectCall->line, expectCall->column);
275 expectCall++;
276 return S_OK;
279 static HRESULT WINAPI contentHandler_startPrefixMapping(
280 ISAXContentHandler* iface,
281 const WCHAR *pPrefix,
282 int nPrefix,
283 const WCHAR *pUri,
284 int nUri)
286 if(!test_expect_call(CH_ENDDOCUMENT))
287 return E_FAIL;
289 test_saxstr(__LINE__, pPrefix, nPrefix, expectCall->arg1);
290 test_saxstr(__LINE__, pUri, nUri, expectCall->arg2);
291 test_locator(__LINE__, expectCall->line, expectCall->column);
293 expectCall++;
294 return S_OK;
297 static HRESULT WINAPI contentHandler_endPrefixMapping(
298 ISAXContentHandler* iface,
299 const WCHAR *pPrefix,
300 int nPrefix)
302 if(!test_expect_call(CH_ENDPREFIXMAPPING))
303 return E_FAIL;
305 test_saxstr(__LINE__, pPrefix, nPrefix, expectCall->arg1);
306 test_locator(__LINE__, expectCall->line, expectCall->column);
308 expectCall++;
309 return S_OK;
312 static HRESULT WINAPI contentHandler_startElement(
313 ISAXContentHandler* iface,
314 const WCHAR *pNamespaceUri,
315 int nNamespaceUri,
316 const WCHAR *pLocalName,
317 int nLocalName,
318 const WCHAR *pQName,
319 int nQName,
320 ISAXAttributes *pAttr)
322 if(!test_expect_call(CH_STARTELEMENT))
323 return E_FAIL;
325 test_saxstr(__LINE__, pNamespaceUri, nNamespaceUri, expectCall->arg1);
326 test_saxstr(__LINE__, pLocalName, nLocalName, expectCall->arg2);
327 test_saxstr(__LINE__, pQName, nQName, expectCall->arg3);
328 test_locator(__LINE__, expectCall->line, expectCall->column);
330 expectCall++;
331 return S_OK;
334 static HRESULT WINAPI contentHandler_endElement(
335 ISAXContentHandler* iface,
336 const WCHAR *pNamespaceUri,
337 int nNamespaceUri,
338 const WCHAR *pLocalName,
339 int nLocalName,
340 const WCHAR *pQName,
341 int nQName)
343 if(!test_expect_call(CH_ENDELEMENT))
344 return E_FAIL;
346 test_saxstr(__LINE__, pNamespaceUri, nNamespaceUri, expectCall->arg1);
347 test_saxstr(__LINE__, pLocalName, nLocalName, expectCall->arg2);
348 test_saxstr(__LINE__, pQName, nQName, expectCall->arg3);
349 test_locator(__LINE__, expectCall->line, expectCall->column);
351 expectCall++;
352 return S_OK;
355 static HRESULT WINAPI contentHandler_characters(
356 ISAXContentHandler* iface,
357 const WCHAR *pChars,
358 int nChars)
360 if(!test_expect_call(CH_CHARACTERS))
361 return E_FAIL;
363 test_saxstr(__LINE__, pChars, nChars, expectCall->arg1);
364 test_locator(__LINE__, expectCall->line, expectCall->column);
366 expectCall++;
367 return S_OK;
370 static HRESULT WINAPI contentHandler_ignorableWhitespace(
371 ISAXContentHandler* iface,
372 const WCHAR *pChars,
373 int nChars)
375 if(!test_expect_call(CH_IGNORABLEWHITESPACE))
376 return E_FAIL;
378 test_saxstr(__LINE__, pChars, nChars, expectCall->arg1);
379 test_locator(__LINE__, expectCall->line, expectCall->column);
381 expectCall++;
382 return S_OK;
385 static HRESULT WINAPI contentHandler_processingInstruction(
386 ISAXContentHandler* iface,
387 const WCHAR *pTarget,
388 int nTarget,
389 const WCHAR *pData,
390 int nData)
392 if(!test_expect_call(CH_PROCESSINGINSTRUCTION))
393 return E_FAIL;
395 test_saxstr(__LINE__, pTarget, nTarget, expectCall->arg1);
396 test_saxstr(__LINE__, pData, nData, expectCall->arg2);
397 test_locator(__LINE__, expectCall->line, expectCall->column);
399 expectCall++;
400 return S_OK;
403 static HRESULT WINAPI contentHandler_skippedEntity(
404 ISAXContentHandler* iface,
405 const WCHAR *pName,
406 int nName)
408 if(!test_expect_call(CH_SKIPPEDENTITY))
409 return E_FAIL;
411 test_saxstr(__LINE__, pName, nName, expectCall->arg1);
412 test_locator(__LINE__, expectCall->line, expectCall->column);
414 expectCall++;
415 return S_OK;
419 static const ISAXContentHandlerVtbl contentHandlerVtbl =
421 contentHandler_QueryInterface,
422 contentHandler_AddRef,
423 contentHandler_Release,
424 contentHandler_putDocumentLocator,
425 contentHandler_startDocument,
426 contentHandler_endDocument,
427 contentHandler_startPrefixMapping,
428 contentHandler_endPrefixMapping,
429 contentHandler_startElement,
430 contentHandler_endElement,
431 contentHandler_characters,
432 contentHandler_ignorableWhitespace,
433 contentHandler_processingInstruction,
434 contentHandler_skippedEntity
437 static ISAXContentHandler contentHandler = { &contentHandlerVtbl };
439 static HRESULT WINAPI isaxerrorHandler_QueryInterface(
440 ISAXErrorHandler* iface,
441 REFIID riid,
442 void **ppvObject)
444 *ppvObject = NULL;
446 if(IsEqualGUID(riid, &IID_IUnknown) || IsEqualGUID(riid, &IID_ISAXErrorHandler))
448 *ppvObject = iface;
450 else
452 return E_NOINTERFACE;
455 return S_OK;
458 static ULONG WINAPI isaxerrorHandler_AddRef(
459 ISAXErrorHandler* iface)
461 return 2;
464 static ULONG WINAPI isaxerrorHandler_Release(
465 ISAXErrorHandler* iface)
467 return 1;
470 static HRESULT WINAPI isaxerrorHandler_error(
471 ISAXErrorHandler* iface,
472 ISAXLocator *pLocator,
473 const WCHAR *pErrorMessage,
474 HRESULT hrErrorCode)
476 return S_OK;
479 static HRESULT WINAPI isaxerrorHandler_fatalError(
480 ISAXErrorHandler* iface,
481 ISAXLocator *pLocator,
482 const WCHAR *pErrorMessage,
483 HRESULT hrErrorCode)
485 return S_OK;
488 static HRESULT WINAPI isaxerrorHanddler_ignorableWarning(
489 ISAXErrorHandler* iface,
490 ISAXLocator *pLocator,
491 const WCHAR *pErrorMessage,
492 HRESULT hrErrorCode)
494 return S_OK;
497 static const ISAXErrorHandlerVtbl errorHandlerVtbl =
499 isaxerrorHandler_QueryInterface,
500 isaxerrorHandler_AddRef,
501 isaxerrorHandler_Release,
502 isaxerrorHandler_error,
503 isaxerrorHandler_fatalError,
504 isaxerrorHanddler_ignorableWarning
507 static ISAXErrorHandler errorHandler = { &errorHandlerVtbl };
509 static HRESULT WINAPI isaxattributes_QueryInterface(
510 ISAXAttributes* iface,
511 REFIID riid,
512 void **ppvObject)
514 *ppvObject = NULL;
516 if(IsEqualGUID(riid, &IID_IUnknown) || IsEqualGUID(riid, &IID_ISAXAttributes))
518 *ppvObject = iface;
520 else
522 return E_NOINTERFACE;
525 return S_OK;
528 static ULONG WINAPI isaxattributes_AddRef(ISAXAttributes* iface)
530 return 2;
533 static ULONG WINAPI isaxattributes_Release(ISAXAttributes* iface)
535 return 1;
538 static HRESULT WINAPI isaxattributes_getLength(ISAXAttributes* iface, int *length)
540 *length = 2;
541 return S_OK;
544 static HRESULT WINAPI isaxattributes_getURI(
545 ISAXAttributes* iface,
546 int nIndex,
547 const WCHAR **pUrl,
548 int *pUriSize)
550 ok(0, "unexpected call\n");
551 return E_NOTIMPL;
554 static HRESULT WINAPI isaxattributes_getLocalName(
555 ISAXAttributes* iface,
556 int nIndex,
557 const WCHAR **pLocalName,
558 int *pLocalNameLength)
560 ok(0, "unexpected call\n");
561 return E_NOTIMPL;
564 static HRESULT WINAPI isaxattributes_getQName(
565 ISAXAttributes* iface,
566 int nIndex,
567 const WCHAR **pQName,
568 int *pQNameLength)
570 static const WCHAR attr1W[] = {'a',':','a','t','t','r','1','j','u','n','k',0};
571 static const WCHAR attr2W[] = {'a','t','t','r','2','j','u','n','k',0};
573 ok(nIndex == 0 || nIndex == 1, "invalid index received %d\n", nIndex);
575 *pQName = (nIndex == 0) ? attr1W : attr2W;
576 *pQNameLength = (nIndex == 0) ? 7 : 5;
578 return S_OK;
581 static HRESULT WINAPI isaxattributes_getName(
582 ISAXAttributes* iface,
583 int nIndex,
584 const WCHAR **pUri,
585 int * pUriLength,
586 const WCHAR ** pLocalName,
587 int * pLocalNameSize,
588 const WCHAR ** pQName,
589 int * pQNameLength)
591 ok(0, "unexpected call\n");
592 return E_NOTIMPL;
595 static HRESULT WINAPI isaxattributes_getIndexFromName(
596 ISAXAttributes* iface,
597 const WCHAR * pUri,
598 int cUriLength,
599 const WCHAR * pLocalName,
600 int cocalNameLength,
601 int * index)
603 ok(0, "unexpected call\n");
604 return E_NOTIMPL;
607 static HRESULT WINAPI isaxattributes_getIndexFromQName(
608 ISAXAttributes* iface,
609 const WCHAR * pQName,
610 int nQNameLength,
611 int * index)
613 ok(0, "unexpected call\n");
614 return E_NOTIMPL;
617 static HRESULT WINAPI isaxattributes_getType(
618 ISAXAttributes* iface,
619 int nIndex,
620 const WCHAR ** pType,
621 int * pTypeLength)
623 ok(0, "unexpected call\n");
624 return E_NOTIMPL;
627 static HRESULT WINAPI isaxattributes_getTypeFromName(
628 ISAXAttributes* iface,
629 const WCHAR * pUri,
630 int nUri,
631 const WCHAR * pLocalName,
632 int nLocalName,
633 const WCHAR ** pType,
634 int * nType)
636 ok(0, "unexpected call\n");
637 return E_NOTIMPL;
640 static HRESULT WINAPI isaxattributes_getTypeFromQName(
641 ISAXAttributes* iface,
642 const WCHAR * pQName,
643 int nQName,
644 const WCHAR ** pType,
645 int * nType)
647 ok(0, "unexpected call\n");
648 return E_NOTIMPL;
651 static HRESULT WINAPI isaxattributes_getValue(
652 ISAXAttributes* iface,
653 int nIndex,
654 const WCHAR ** pValue,
655 int * nValue)
657 static const WCHAR attrval1W[] = {'a','1','j','u','n','k',0};
658 static const WCHAR attrval2W[] = {'a','2','j','u','n','k',0};
660 ok(nIndex == 0 || nIndex == 1, "invalid index received %d\n", nIndex);
662 *pValue = (nIndex == 0) ? attrval1W : attrval2W;
663 *nValue = 2;
665 return S_OK;
668 static HRESULT WINAPI isaxattributes_getValueFromName(
669 ISAXAttributes* iface,
670 const WCHAR * pUri,
671 int nUri,
672 const WCHAR * pLocalName,
673 int nLocalName,
674 const WCHAR ** pValue,
675 int * nValue)
677 ok(0, "unexpected call\n");
678 return E_NOTIMPL;
681 static HRESULT WINAPI isaxattributes_getValueFromQName(
682 ISAXAttributes* iface,
683 const WCHAR * pQName,
684 int nQName,
685 const WCHAR ** pValue,
686 int * nValue)
688 ok(0, "unexpected call\n");
689 return E_NOTIMPL;
692 static const ISAXAttributesVtbl SAXAttributesVtbl =
694 isaxattributes_QueryInterface,
695 isaxattributes_AddRef,
696 isaxattributes_Release,
697 isaxattributes_getLength,
698 isaxattributes_getURI,
699 isaxattributes_getLocalName,
700 isaxattributes_getQName,
701 isaxattributes_getName,
702 isaxattributes_getIndexFromName,
703 isaxattributes_getIndexFromQName,
704 isaxattributes_getType,
705 isaxattributes_getTypeFromName,
706 isaxattributes_getTypeFromQName,
707 isaxattributes_getValue,
708 isaxattributes_getValueFromName,
709 isaxattributes_getValueFromQName
712 static ISAXAttributes saxattributes = { &SAXAttributesVtbl };
714 static int handler_addrefcalled;
716 static HRESULT WINAPI isaxlexical_QueryInterface(ISAXLexicalHandler* iface, REFIID riid, void **ppvObject)
718 *ppvObject = NULL;
720 if(IsEqualGUID(riid, &IID_IUnknown) ||
721 IsEqualGUID(riid, &IID_ISAXLexicalHandler))
723 *ppvObject = iface;
725 else
727 return E_NOINTERFACE;
730 return S_OK;
733 static ULONG WINAPI isaxlexical_AddRef(ISAXLexicalHandler* iface)
735 handler_addrefcalled++;
736 return 2;
739 static ULONG WINAPI isaxlexical_Release(ISAXLexicalHandler* iface)
741 return 1;
744 static HRESULT WINAPI isaxlexical_startDTD(ISAXLexicalHandler* iface,
745 const WCHAR * pName, int nName, const WCHAR * pPublicId,
746 int nPublicId, const WCHAR * pSystemId, int nSystemId)
748 ok(0, "call not expected\n");
749 return E_NOTIMPL;
752 static HRESULT WINAPI isaxlexical_endDTD(ISAXLexicalHandler* iface)
754 ok(0, "call not expected\n");
755 return E_NOTIMPL;
758 static HRESULT WINAPI isaxlexical_startEntity(ISAXLexicalHandler *iface,
759 const WCHAR * pName, int nName)
761 ok(0, "call not expected\n");
762 return E_NOTIMPL;
765 static HRESULT WINAPI isaxlexical_endEntity(ISAXLexicalHandler *iface,
766 const WCHAR * pName, int nName)
768 ok(0, "call not expected\n");
769 return E_NOTIMPL;
772 static HRESULT WINAPI isaxlexical_startCDATA(ISAXLexicalHandler *iface)
774 ok(0, "call not expected\n");
775 return E_NOTIMPL;
778 static HRESULT WINAPI isaxlexical_endCDATA(ISAXLexicalHandler *iface)
780 ok(0, "call not expected\n");
781 return E_NOTIMPL;
784 static HRESULT WINAPI isaxlexical_comment(ISAXLexicalHandler *iface,
785 const WCHAR * pChars, int nChars)
787 ok(0, "call not expected\n");
788 return E_NOTIMPL;
791 static const ISAXLexicalHandlerVtbl SAXLexicalHandlerVtbl =
793 isaxlexical_QueryInterface,
794 isaxlexical_AddRef,
795 isaxlexical_Release,
796 isaxlexical_startDTD,
797 isaxlexical_endDTD,
798 isaxlexical_startEntity,
799 isaxlexical_endEntity,
800 isaxlexical_startCDATA,
801 isaxlexical_endCDATA,
802 isaxlexical_comment
805 static ISAXLexicalHandler saxlexicalhandler = { &SAXLexicalHandlerVtbl };
807 static HRESULT WINAPI isaxdecl_QueryInterface(ISAXDeclHandler* iface, REFIID riid, void **ppvObject)
809 *ppvObject = NULL;
811 if(IsEqualGUID(riid, &IID_IUnknown) ||
812 IsEqualGUID(riid, &IID_ISAXDeclHandler))
814 *ppvObject = iface;
816 else
818 return E_NOINTERFACE;
821 return S_OK;
824 static ULONG WINAPI isaxdecl_AddRef(ISAXDeclHandler* iface)
826 handler_addrefcalled++;
827 return 2;
830 static ULONG WINAPI isaxdecl_Release(ISAXDeclHandler* iface)
832 return 1;
835 static HRESULT WINAPI isaxdecl_elementDecl(ISAXDeclHandler* iface,
836 const WCHAR * pName, int nName, const WCHAR * pModel, int nModel)
838 ok(0, "call not expected\n");
839 return E_NOTIMPL;
842 static HRESULT WINAPI isaxdecl_attributeDecl(ISAXDeclHandler* iface,
843 const WCHAR * pElementName, int nElementName, const WCHAR * pAttributeName,
844 int nAttributeName, const WCHAR * pType, int nType, const WCHAR * pValueDefault,
845 int nValueDefault, const WCHAR * pValue, int nValue)
847 ok(0, "call not expected\n");
848 return E_NOTIMPL;
851 static HRESULT WINAPI isaxdecl_internalEntityDecl(ISAXDeclHandler* iface,
852 const WCHAR * pName, int nName, const WCHAR * pValue, int nValue)
854 ok(0, "call not expected\n");
855 return E_NOTIMPL;
858 static HRESULT WINAPI isaxdecl_externalEntityDecl(ISAXDeclHandler* iface,
859 const WCHAR * pName, int nName, const WCHAR * pPublicId, int nPublicId,
860 const WCHAR * pSystemId, int nSystemId)
862 ok(0, "call not expected\n");
863 return E_NOTIMPL;
866 static const ISAXDeclHandlerVtbl SAXDeclHandlerVtbl =
868 isaxdecl_QueryInterface,
869 isaxdecl_AddRef,
870 isaxdecl_Release,
871 isaxdecl_elementDecl,
872 isaxdecl_attributeDecl,
873 isaxdecl_internalEntityDecl,
874 isaxdecl_externalEntityDecl
877 static ISAXDeclHandler saxdeclhandler = { &SAXDeclHandlerVtbl };
879 typedef struct mxwriter_write_test_t {
880 BOOL last;
881 const BYTE *data;
882 DWORD cb;
883 BOOL null_written;
884 BOOL fail_write;
885 } mxwriter_write_test;
887 typedef struct mxwriter_stream_test_t {
888 VARIANT_BOOL bom;
889 const char *encoding;
890 mxwriter_write_test expected_writes[4];
891 } mxwriter_stream_test;
893 static const mxwriter_write_test *current_write_test;
894 static DWORD current_stream_test_index;
896 static HRESULT WINAPI istream_QueryInterface(IStream *iface, REFIID riid, void **ppvObject)
898 *ppvObject = NULL;
900 if(IsEqualGUID(riid, &IID_IStream) || IsEqualGUID(riid, &IID_IUnknown))
901 *ppvObject = iface;
902 else
903 return E_NOINTERFACE;
905 return S_OK;
908 static ULONG WINAPI istream_AddRef(IStream *iface)
910 return 2;
913 static ULONG WINAPI istream_Release(IStream *iface)
915 return 1;
918 static HRESULT WINAPI istream_Read(IStream *iface, void *pv, ULONG cb, ULONG *pcbRead)
920 ok(0, "unexpected call\n");
921 return E_NOTIMPL;
924 static HRESULT WINAPI istream_Write(IStream *iface, const void *pv, ULONG cb, ULONG *pcbWritten)
926 BOOL fail = FALSE;
928 ok(pv != NULL, "pv == NULL\n");
930 if(current_write_test->last) {
931 ok(0, "Too many Write calls made on test %d\n", current_stream_test_index);
932 return E_FAIL;
935 fail = current_write_test->fail_write;
937 ok(current_write_test->cb == cb, "Expected %d, but got %d on test %d\n",
938 current_write_test->cb, cb, current_stream_test_index);
940 if(!pcbWritten)
941 ok(current_write_test->null_written, "pcbWritten was NULL on test %d\n", current_stream_test_index);
942 else
943 ok(!memcmp(current_write_test->data, pv, cb), "Unexpected data on test %d\n", current_stream_test_index);
945 ++current_write_test;
947 if(pcbWritten)
948 *pcbWritten = cb;
950 return fail ? E_FAIL : S_OK;
953 static HRESULT WINAPI istream_Seek(IStream *iface, LARGE_INTEGER dlibMove, DWORD dwOrigin,
954 ULARGE_INTEGER *plibNewPosition)
956 ok(0, "unexpected call\n");
957 return E_NOTIMPL;
960 static HRESULT WINAPI istream_SetSize(IStream *iface, ULARGE_INTEGER libNewSize)
962 ok(0, "unexpected call\n");
963 return E_NOTIMPL;
966 static HRESULT WINAPI istream_CopyTo(IStream *iface, IStream *pstm, ULARGE_INTEGER cb,
967 ULARGE_INTEGER *pcbRead, ULARGE_INTEGER *plibWritten)
969 ok(0, "unexpected call\n");
970 return E_NOTIMPL;
973 static HRESULT WINAPI istream_Commit(IStream *iface, DWORD grfCommitFlags)
975 ok(0, "unexpected call\n");
976 return E_NOTIMPL;
979 static HRESULT WINAPI istream_Revert(IStream *iface)
981 ok(0, "unexpected call\n");
982 return E_NOTIMPL;
985 static HRESULT WINAPI istream_LockRegion(IStream *iface, ULARGE_INTEGER libOffset,
986 ULARGE_INTEGER cb, DWORD dwLockType)
988 ok(0, "unexpected call\n");
989 return E_NOTIMPL;
992 static HRESULT WINAPI istream_UnlockRegion(IStream *iface, ULARGE_INTEGER libOffset,
993 ULARGE_INTEGER cb, DWORD dwLockType)
995 ok(0, "unexpected call\n");
996 return E_NOTIMPL;
999 static HRESULT WINAPI istream_Stat(IStream *iface, STATSTG *pstatstg, DWORD grfStatFlag)
1001 ok(0, "unexpected call\n");
1002 return E_NOTIMPL;
1005 static HRESULT WINAPI istream_Clone(IStream *iface, IStream **ppstm)
1007 ok(0, "unexpected call\n");
1008 return E_NOTIMPL;
1011 static const IStreamVtbl StreamVtbl = {
1012 istream_QueryInterface,
1013 istream_AddRef,
1014 istream_Release,
1015 istream_Read,
1016 istream_Write,
1017 istream_Seek,
1018 istream_SetSize,
1019 istream_CopyTo,
1020 istream_Commit,
1021 istream_Revert,
1022 istream_LockRegion,
1023 istream_UnlockRegion,
1024 istream_Stat,
1025 istream_Clone
1028 static IStream mxstream = { &StreamVtbl };
1030 static void test_saxreader(void)
1032 HRESULT hr;
1033 ISAXXMLReader *reader = NULL;
1034 VARIANT var;
1035 ISAXContentHandler *lpContentHandler;
1036 ISAXErrorHandler *lpErrorHandler;
1037 SAFEARRAY *pSA;
1038 SAFEARRAYBOUND SADim[1];
1039 char *pSAData = NULL;
1040 IStream *iStream;
1041 ULARGE_INTEGER liSize;
1042 LARGE_INTEGER liPos;
1043 ULONG bytesWritten;
1044 HANDLE file;
1045 static const CHAR testXmlA[] = "test.xml";
1046 static const WCHAR testXmlW[] = {'t','e','s','t','.','x','m','l',0};
1047 IXMLDOMDocument *domDocument;
1048 BSTR bstrData;
1049 VARIANT_BOOL vBool;
1051 hr = CoCreateInstance(&CLSID_SAXXMLReader, NULL, CLSCTX_INPROC_SERVER,
1052 &IID_ISAXXMLReader, (LPVOID*)&reader);
1053 ok(hr == S_OK, "Expected S_OK, got %08x\n", hr);
1055 hr = ISAXXMLReader_getContentHandler(reader, NULL);
1056 ok(hr == E_POINTER, "Expected E_POINTER, got %08x\n", hr);
1058 hr = ISAXXMLReader_getErrorHandler(reader, NULL);
1059 ok(hr == E_POINTER, "Expected E_POINTER, got %08x\n", hr);
1061 hr = ISAXXMLReader_getContentHandler(reader, &lpContentHandler);
1062 ok(hr == S_OK, "Expected S_OK, got %08x\n", hr);
1063 ok(lpContentHandler == NULL, "Expected %p, got %p\n", NULL, lpContentHandler);
1065 hr = ISAXXMLReader_getErrorHandler(reader, &lpErrorHandler);
1066 ok(hr == S_OK, "Expected S_OK, got %08x\n", hr);
1067 ok(lpErrorHandler == NULL, "Expected %p, got %p\n", NULL, lpErrorHandler);
1069 hr = ISAXXMLReader_putContentHandler(reader, NULL);
1070 ok(hr == S_OK, "Expected S_OK, got %08x\n", hr);
1072 hr = ISAXXMLReader_putContentHandler(reader, &contentHandler);
1073 ok(hr == S_OK, "Expected S_OK, got %08x\n", hr);
1075 hr = ISAXXMLReader_putErrorHandler(reader, &errorHandler);
1076 ok(hr == S_OK, "Expected S_OK, got %08x\n", hr);
1078 hr = ISAXXMLReader_getContentHandler(reader, &lpContentHandler);
1079 ok(hr == S_OK, "Expected S_OK, got %08x\n", hr);
1080 ok(lpContentHandler == &contentHandler, "Expected %p, got %p\n", &contentHandler, lpContentHandler);
1082 V_VT(&var) = VT_BSTR;
1083 V_BSTR(&var) = SysAllocString(szSimpleXML);
1085 expectCall = contentHandlerTest1;
1086 hr = ISAXXMLReader_parse(reader, var);
1087 ok(hr == S_OK, "Expected S_OK, got %08x\n", hr);
1088 test_expect_call(CH_ENDTEST);
1090 VariantClear(&var);
1092 SADim[0].lLbound= 0;
1093 SADim[0].cElements= sizeof(szTestXML)-1;
1094 pSA = SafeArrayCreate(VT_UI1, 1, SADim);
1095 SafeArrayAccessData(pSA, (void**)&pSAData);
1096 memcpy(pSAData, szTestXML, sizeof(szTestXML)-1);
1097 SafeArrayUnaccessData(pSA);
1098 V_VT(&var) = VT_ARRAY|VT_UI1;
1099 V_ARRAY(&var) = pSA;
1101 expectCall = contentHandlerTest1;
1102 hr = ISAXXMLReader_parse(reader, var);
1103 ok(hr == S_OK, "Expected S_OK, got %08x\n", hr);
1104 test_expect_call(CH_ENDTEST);
1106 SafeArrayDestroy(pSA);
1108 CreateStreamOnHGlobal(NULL, TRUE, &iStream);
1109 liSize.QuadPart = strlen(szTestXML);
1110 IStream_SetSize(iStream, liSize);
1111 IStream_Write(iStream, szTestXML, strlen(szTestXML), &bytesWritten);
1112 liPos.QuadPart = 0;
1113 IStream_Seek(iStream, liPos, STREAM_SEEK_SET, NULL);
1114 V_VT(&var) = VT_UNKNOWN|VT_DISPATCH;
1115 V_UNKNOWN(&var) = (IUnknown*)iStream;
1117 expectCall = contentHandlerTest1;
1118 hr = ISAXXMLReader_parse(reader, var);
1119 ok(hr == S_OK, "Expected S_OK, got %08x\n", hr);
1120 test_expect_call(CH_ENDTEST);
1122 IStream_Release(iStream);
1124 V_VT(&var) = VT_BSTR;
1125 V_BSTR(&var) = SysAllocString(szCarriageRetTest);
1127 expectCall = contentHandlerTest2;
1128 hr = ISAXXMLReader_parse(reader, var);
1129 ok(hr == S_OK, "Expected S_OK, got %08x\n", hr);
1130 test_expect_call(CH_ENDTEST);
1132 VariantClear(&var);
1134 file = CreateFileA(testXmlA, GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL);
1135 ok(file != INVALID_HANDLE_VALUE, "Could not create file: %u\n", GetLastError());
1136 WriteFile(file, szTestXML, sizeof(szTestXML)-1, &bytesWritten, NULL);
1137 CloseHandle(file);
1139 expectCall = contentHandlerTest1;
1140 hr = ISAXXMLReader_parseURL(reader, testXmlW);
1141 ok(hr == S_OK, "Expected S_OK, got %08x\n", hr);
1142 test_expect_call(CH_ENDTEST);
1144 DeleteFileA(testXmlA);
1146 hr = CoCreateInstance(&CLSID_DOMDocument, NULL, CLSCTX_INPROC_SERVER,
1147 &IID_IXMLDOMDocument, (LPVOID*)&domDocument);
1148 if(FAILED(hr))
1150 skip("Failed to create DOMDocument instance\n");
1151 return;
1153 bstrData = SysAllocString(szSimpleXML);
1154 hr = IXMLDOMDocument_loadXML(domDocument, bstrData, &vBool);
1155 ok(hr == S_OK, "Expected S_OK, got %08x\n", hr);
1156 V_VT(&var) = VT_UNKNOWN;
1157 V_UNKNOWN(&var) = (IUnknown*)domDocument;
1159 expectCall = contentHandlerTest2;
1160 hr = ISAXXMLReader_parse(reader, var);
1161 ok(hr == S_OK, "Expected S_OK, got %08x\n", hr);
1162 test_expect_call(CH_ENDTEST);
1163 IXMLDOMDocument_Release(domDocument);
1165 ISAXXMLReader_Release(reader);
1166 SysFreeString(bstrData);
1169 struct saxreader_props_test_t
1171 const char *prop_name;
1172 IUnknown *iface;
1175 static const struct saxreader_props_test_t props_test_data[] = {
1176 { "http://xml.org/sax/properties/lexical-handler", (IUnknown*)&saxlexicalhandler },
1177 { "http://xml.org/sax/properties/declaration-handler", (IUnknown*)&saxdeclhandler },
1178 { 0 }
1181 static void test_saxreader_properties(void)
1183 const struct saxreader_props_test_t *ptr = props_test_data;
1184 ISAXXMLReader *reader;
1185 HRESULT hr;
1187 hr = CoCreateInstance(&CLSID_SAXXMLReader, NULL, CLSCTX_INPROC_SERVER,
1188 &IID_ISAXXMLReader, (void**)&reader);
1189 EXPECT_HR(hr, S_OK);
1191 hr = ISAXXMLReader_getProperty(reader, _bstr_("http://xml.org/sax/properties/lexical-handler"), NULL);
1192 EXPECT_HR(hr, E_POINTER);
1194 while (ptr->prop_name)
1196 VARIANT v;
1198 V_VT(&v) = VT_EMPTY;
1199 V_UNKNOWN(&v) = (IUnknown*)0xdeadbeef;
1200 hr = ISAXXMLReader_getProperty(reader, _bstr_(ptr->prop_name), &v);
1201 EXPECT_HR(hr, S_OK);
1202 ok(V_VT(&v) == VT_UNKNOWN, "got %d\n", V_VT(&v));
1203 ok(V_UNKNOWN(&v) == NULL, "got %p\n", V_UNKNOWN(&v));
1205 V_VT(&v) = VT_UNKNOWN;
1206 V_UNKNOWN(&v) = ptr->iface;
1207 hr = ISAXXMLReader_putProperty(reader, _bstr_(ptr->prop_name), v);
1208 EXPECT_HR(hr, S_OK);
1210 V_VT(&v) = VT_EMPTY;
1211 V_UNKNOWN(&v) = (IUnknown*)0xdeadbeef;
1212 handler_addrefcalled = 0;
1213 hr = ISAXXMLReader_getProperty(reader, _bstr_(ptr->prop_name), &v);
1214 EXPECT_HR(hr, S_OK);
1215 ok(V_VT(&v) == VT_UNKNOWN, "got %d\n", V_VT(&v));
1216 ok(V_UNKNOWN(&v) == ptr->iface, "got %p\n", V_UNKNOWN(&v));
1217 ok(handler_addrefcalled == 1, "AddRef called %d times\n", handler_addrefcalled);
1218 VariantClear(&v);
1220 V_VT(&v) = VT_EMPTY;
1221 V_UNKNOWN(&v) = (IUnknown*)0xdeadbeef;
1222 hr = ISAXXMLReader_putProperty(reader, _bstr_(ptr->prop_name), v);
1223 EXPECT_HR(hr, S_OK);
1225 V_VT(&v) = VT_EMPTY;
1226 V_UNKNOWN(&v) = (IUnknown*)0xdeadbeef;
1227 hr = ISAXXMLReader_getProperty(reader, _bstr_(ptr->prop_name), &v);
1228 EXPECT_HR(hr, S_OK);
1229 ok(V_VT(&v) == VT_UNKNOWN, "got %d\n", V_VT(&v));
1230 ok(V_UNKNOWN(&v) == NULL, "got %p\n", V_UNKNOWN(&v));
1232 V_VT(&v) = VT_UNKNOWN;
1233 V_UNKNOWN(&v) = ptr->iface;
1234 hr = ISAXXMLReader_putProperty(reader, _bstr_(ptr->prop_name), v);
1235 EXPECT_HR(hr, S_OK);
1237 /* only VT_EMPTY seems to be valid to reset property */
1238 V_VT(&v) = VT_I4;
1239 V_UNKNOWN(&v) = (IUnknown*)0xdeadbeef;
1240 hr = ISAXXMLReader_putProperty(reader, _bstr_(ptr->prop_name), v);
1241 EXPECT_HR(hr, E_INVALIDARG);
1243 V_VT(&v) = VT_EMPTY;
1244 V_UNKNOWN(&v) = (IUnknown*)0xdeadbeef;
1245 hr = ISAXXMLReader_getProperty(reader, _bstr_(ptr->prop_name), &v);
1246 EXPECT_HR(hr, S_OK);
1247 ok(V_VT(&v) == VT_UNKNOWN, "got %d\n", V_VT(&v));
1248 ok(V_UNKNOWN(&v) == ptr->iface, "got %p\n", V_UNKNOWN(&v));
1249 VariantClear(&v);
1251 V_VT(&v) = VT_UNKNOWN;
1252 V_UNKNOWN(&v) = NULL;
1253 hr = ISAXXMLReader_putProperty(reader, _bstr_(ptr->prop_name), v);
1254 EXPECT_HR(hr, S_OK);
1256 V_VT(&v) = VT_EMPTY;
1257 V_UNKNOWN(&v) = (IUnknown*)0xdeadbeef;
1258 hr = ISAXXMLReader_getProperty(reader, _bstr_(ptr->prop_name), &v);
1259 EXPECT_HR(hr, S_OK);
1260 ok(V_VT(&v) == VT_UNKNOWN, "got %d\n", V_VT(&v));
1261 ok(V_UNKNOWN(&v) == NULL, "got %p\n", V_UNKNOWN(&v));
1263 ptr++;
1266 ISAXXMLReader_Release(reader);
1267 free_bstrs();
1270 struct feature_ns_entry_t {
1271 const GUID *guid;
1272 const char *clsid;
1273 VARIANT_BOOL value;
1276 static const struct feature_ns_entry_t feature_ns_entry_data[] = {
1277 { &CLSID_SAXXMLReader, "CLSID_SAXXMLReader", VARIANT_TRUE },
1278 { &CLSID_SAXXMLReader30, "CLSID_SAXXMLReader30", VARIANT_TRUE },
1279 { &CLSID_SAXXMLReader40, "CLSID_SAXXMLReader40", VARIANT_TRUE },
1280 { &CLSID_SAXXMLReader60, "CLSID_SAXXMLReader60", VARIANT_TRUE },
1281 { 0 }
1284 static void test_saxreader_features(void)
1286 const struct feature_ns_entry_t *entry = feature_ns_entry_data;
1287 ISAXXMLReader *reader;
1289 while (entry->guid)
1291 VARIANT_BOOL value;
1292 HRESULT hr;
1294 hr = CoCreateInstance(entry->guid, NULL, CLSCTX_INPROC_SERVER, &IID_ISAXXMLReader, (void**)&reader);
1295 if (hr != S_OK)
1297 win_skip("can't create %s instance\n", entry->clsid);
1298 entry++;
1299 continue;
1302 value = 0xc;
1303 hr = ISAXXMLReader_getFeature(reader, _bstr_("http://xml.org/sax/features/namespaces"), &value);
1304 EXPECT_HR(hr, S_OK);
1306 ok(entry->value == value, "%s: got wrong default value %x, expected %x\n", entry->clsid, value, entry->value);
1308 ISAXXMLReader_Release(reader);
1310 entry++;
1314 /* UTF-8 data with UTF-8 BOM and UTF-16 in prolog */
1315 static const CHAR UTF8BOMTest[] =
1316 "\xEF\xBB\xBF<?xml version = \"1.0\" encoding = \"UTF-16\"?>\n"
1317 "<a></a>\n";
1319 struct enc_test_entry_t {
1320 const GUID *guid;
1321 const char *clsid;
1322 const char *data;
1323 HRESULT hr;
1324 int todo;
1327 static const struct enc_test_entry_t encoding_test_data[] = {
1328 { &CLSID_SAXXMLReader, "CLSID_SAXXMLReader", UTF8BOMTest, 0xc00ce56f, 1 },
1329 { &CLSID_SAXXMLReader30, "CLSID_SAXXMLReader30", UTF8BOMTest, 0xc00ce56f, 1 },
1330 { &CLSID_SAXXMLReader40, "CLSID_SAXXMLReader40", UTF8BOMTest, S_OK, 0 },
1331 { &CLSID_SAXXMLReader60, "CLSID_SAXXMLReader60", UTF8BOMTest, S_OK, 0 },
1332 { 0 }
1335 static void test_encoding(void)
1337 const struct enc_test_entry_t *entry = encoding_test_data;
1338 static const WCHAR testXmlW[] = {'t','e','s','t','.','x','m','l',0};
1339 static const CHAR testXmlA[] = "test.xml";
1340 ISAXXMLReader *reader;
1341 DWORD written;
1342 HANDLE file;
1343 HRESULT hr;
1345 while (entry->guid)
1347 hr = CoCreateInstance(entry->guid, NULL, CLSCTX_INPROC_SERVER, &IID_ISAXXMLReader, (void**)&reader);
1348 if (hr != S_OK)
1350 win_skip("can't create %s instance\n", entry->clsid);
1351 entry++;
1352 continue;
1355 file = CreateFileA(testXmlA, GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL);
1356 ok(file != INVALID_HANDLE_VALUE, "Could not create file: %u\n", GetLastError());
1357 WriteFile(file, UTF8BOMTest, sizeof(UTF8BOMTest)-1, &written, NULL);
1358 CloseHandle(file);
1360 hr = ISAXXMLReader_parseURL(reader, testXmlW);
1361 if (entry->todo)
1362 todo_wine ok(hr == entry->hr, "Expected 0x%08x, got 0x%08x. CLSID %s\n", entry->hr, hr, entry->clsid);
1363 else
1364 ok(hr == entry->hr, "Expected 0x%08x, got 0x%08x. CLSID %s\n", entry->hr, hr, entry->clsid);
1366 DeleteFileA(testXmlA);
1367 ISAXXMLReader_Release(reader);
1369 entry++;
1373 static void test_mxwriter_contenthandler(void)
1375 ISAXContentHandler *handler;
1376 IMXWriter *writer, *writer2;
1377 HRESULT hr;
1379 hr = CoCreateInstance(&CLSID_MXXMLWriter, NULL, CLSCTX_INPROC_SERVER,
1380 &IID_IMXWriter, (void**)&writer);
1381 ok(hr == S_OK, "Expected S_OK, got %08x\n", hr);
1383 EXPECT_REF(writer, 1);
1385 hr = IMXWriter_QueryInterface(writer, &IID_ISAXContentHandler, (void**)&handler);
1386 ok(hr == S_OK, "Expected S_OK, got %08x\n", hr);
1387 EXPECT_REF(writer, 2);
1388 EXPECT_REF(handler, 2);
1390 hr = ISAXContentHandler_QueryInterface(handler, &IID_IMXWriter, (void**)&writer2);
1391 ok(hr == S_OK, "Expected S_OK, got %08x\n", hr);
1392 ok(writer2 == writer, "got %p, expected %p\n", writer2, writer);
1393 EXPECT_REF(writer, 3);
1394 EXPECT_REF(writer2, 3);
1395 IMXWriter_Release(writer2);
1397 ISAXContentHandler_Release(handler);
1398 IMXWriter_Release(writer);
1401 struct msxmlsupported_data_t
1403 const GUID *clsid;
1404 const char *name;
1405 BOOL supported;
1408 static struct msxmlsupported_data_t msxmlsupported_data[] =
1410 { &CLSID_MXXMLWriter, "MXXMLWriter" },
1411 { &CLSID_MXXMLWriter30, "MXXMLWriter30" },
1412 { &CLSID_MXXMLWriter40, "MXXMLWriter40" },
1413 { &CLSID_MXXMLWriter60, "MXXMLWriter60" },
1414 { NULL }
1417 static BOOL is_mxwriter_supported(const GUID *clsid, const struct msxmlsupported_data_t *table)
1419 while (table->clsid)
1421 if (table->clsid == clsid) return table->supported;
1422 table++;
1424 return FALSE;
1427 struct mxwriter_props_t
1429 const GUID *clsid;
1430 VARIANT_BOOL bom;
1431 VARIANT_BOOL disable_escape;
1432 VARIANT_BOOL indent;
1433 VARIANT_BOOL omitdecl;
1434 VARIANT_BOOL standalone;
1435 const char *encoding;
1438 static const struct mxwriter_props_t mxwriter_default_props[] =
1440 { &CLSID_MXXMLWriter, VARIANT_TRUE, VARIANT_FALSE, VARIANT_FALSE, VARIANT_FALSE, VARIANT_FALSE, "UTF-16" },
1441 { &CLSID_MXXMLWriter30, VARIANT_TRUE, VARIANT_FALSE, VARIANT_FALSE, VARIANT_FALSE, VARIANT_FALSE, "UTF-16" },
1442 { &CLSID_MXXMLWriter40, VARIANT_TRUE, VARIANT_FALSE, VARIANT_FALSE, VARIANT_FALSE, VARIANT_FALSE, "UTF-16" },
1443 { &CLSID_MXXMLWriter60, VARIANT_TRUE, VARIANT_FALSE, VARIANT_FALSE, VARIANT_FALSE, VARIANT_FALSE, "UTF-16" },
1444 { NULL }
1447 static void test_mxwriter_default_properties(const struct mxwriter_props_t *table)
1449 int i = 0;
1451 while (table->clsid)
1453 IMXWriter *writer;
1454 VARIANT_BOOL b;
1455 BSTR encoding;
1456 HRESULT hr;
1458 if (!is_mxwriter_supported(table->clsid, msxmlsupported_data))
1460 table++;
1461 i++;
1462 continue;
1465 hr = CoCreateInstance(table->clsid, NULL, CLSCTX_INPROC_SERVER,
1466 &IID_IMXWriter, (void**)&writer);
1467 EXPECT_HR(hr, S_OK);
1469 b = !table->bom;
1470 hr = IMXWriter_get_byteOrderMark(writer, &b);
1471 EXPECT_HR(hr, S_OK);
1472 ok(table->bom == b, "test %d: got BOM %d, expected %d\n", i, b, table->bom);
1474 b = !table->disable_escape;
1475 hr = IMXWriter_get_disableOutputEscaping(writer, &b);
1476 EXPECT_HR(hr, S_OK);
1477 ok(table->disable_escape == b, "test %d: got disable escape %d, expected %d\n", i, b,
1478 table->disable_escape);
1480 b = !table->indent;
1481 hr = IMXWriter_get_indent(writer, &b);
1482 EXPECT_HR(hr, S_OK);
1483 ok(table->indent == b, "test %d: got indent %d, expected %d\n", i, b, table->indent);
1485 b = !table->omitdecl;
1486 hr = IMXWriter_get_omitXMLDeclaration(writer, &b);
1487 EXPECT_HR(hr, S_OK);
1488 ok(table->omitdecl == b, "test %d: got omitdecl %d, expected %d\n", i, b, table->omitdecl);
1490 b = !table->standalone;
1491 hr = IMXWriter_get_standalone(writer, &b);
1492 EXPECT_HR(hr, S_OK);
1493 ok(table->standalone == b, "test %d: got standalone %d, expected %d\n", i, b, table->standalone);
1495 hr = IMXWriter_get_encoding(writer, &encoding);
1496 EXPECT_HR(hr, S_OK);
1497 ok(!lstrcmpW(encoding, _bstr_(table->encoding)), "test %d: got encoding %s, expected %s\n",
1498 i, wine_dbgstr_w(encoding), table->encoding);
1499 SysFreeString(encoding);
1501 IMXWriter_Release(writer);
1503 table++;
1504 i++;
1508 static void test_mxwriter_properties(void)
1510 static const WCHAR utf16W[] = {'U','T','F','-','1','6',0};
1511 static const WCHAR emptyW[] = {0};
1512 static const WCHAR testW[] = {'t','e','s','t',0};
1513 IMXWriter *writer;
1514 VARIANT_BOOL b;
1515 HRESULT hr;
1516 BSTR str, str2;
1518 test_mxwriter_default_properties(mxwriter_default_props);
1520 hr = CoCreateInstance(&CLSID_MXXMLWriter, NULL, CLSCTX_INPROC_SERVER,
1521 &IID_IMXWriter, (void**)&writer);
1522 ok(hr == S_OK, "Expected S_OK, got %08x\n", hr);
1524 hr = IMXWriter_get_disableOutputEscaping(writer, NULL);
1525 ok(hr == E_POINTER, "got %08x\n", hr);
1527 hr = IMXWriter_get_byteOrderMark(writer, NULL);
1528 ok(hr == E_POINTER, "got %08x\n", hr);
1530 hr = IMXWriter_get_indent(writer, NULL);
1531 ok(hr == E_POINTER, "got %08x\n", hr);
1533 hr = IMXWriter_get_omitXMLDeclaration(writer, NULL);
1534 ok(hr == E_POINTER, "got %08x\n", hr);
1536 hr = IMXWriter_get_standalone(writer, NULL);
1537 ok(hr == E_POINTER, "got %08x\n", hr);
1539 /* set and check */
1540 hr = IMXWriter_put_standalone(writer, VARIANT_TRUE);
1541 ok(hr == S_OK, "got %08x\n", hr);
1543 b = VARIANT_FALSE;
1544 hr = IMXWriter_get_standalone(writer, &b);
1545 ok(hr == S_OK, "got %08x\n", hr);
1546 ok(b == VARIANT_TRUE, "got %d\n", b);
1548 hr = IMXWriter_get_encoding(writer, NULL);
1549 ok(hr == E_POINTER, "got %08x\n", hr);
1551 /* UTF-16 is a default setting apparently */
1552 str = (void*)0xdeadbeef;
1553 hr = IMXWriter_get_encoding(writer, &str);
1554 ok(hr == S_OK, "got %08x\n", hr);
1555 ok(lstrcmpW(str, utf16W) == 0, "expected empty string, got %s\n", wine_dbgstr_w(str));
1557 str2 = (void*)0xdeadbeef;
1558 hr = IMXWriter_get_encoding(writer, &str2);
1559 ok(hr == S_OK, "got %08x\n", hr);
1560 ok(str != str2, "expected newly allocated, got same %p\n", str);
1562 SysFreeString(str2);
1563 SysFreeString(str);
1565 /* put empty string */
1566 str = SysAllocString(emptyW);
1567 hr = IMXWriter_put_encoding(writer, str);
1568 ok(hr == E_INVALIDARG, "got %08x\n", hr);
1569 SysFreeString(str);
1571 str = (void*)0xdeadbeef;
1572 hr = IMXWriter_get_encoding(writer, &str);
1573 ok(hr == S_OK, "got %08x\n", hr);
1574 ok(!lstrcmpW(str, emptyW) == 0, "expected empty string, got %s\n", wine_dbgstr_w(str));
1575 SysFreeString(str);
1577 /* invalid encoding name */
1578 str = SysAllocString(testW);
1579 hr = IMXWriter_put_encoding(writer, str);
1580 ok(hr == E_INVALIDARG, "got %08x\n", hr);
1581 SysFreeString(str);
1583 hr = IMXWriter_get_version(writer, NULL);
1584 ok(hr == E_POINTER, "got %08x\n", hr);
1585 /* default version is 'surprisingly' 1.0 */
1586 hr = IMXWriter_get_version(writer, &str);
1587 ok(hr == S_OK, "got %08x\n", hr);
1588 ok(!lstrcmpW(str, _bstr_("1.0")), "got %s\n", wine_dbgstr_w(str));
1589 SysFreeString(str);
1591 /* store version string as is */
1592 hr = IMXWriter_put_version(writer, NULL);
1593 ok(hr == E_INVALIDARG, "got %08x\n", hr);
1595 hr = IMXWriter_put_version(writer, _bstr_("1.0"));
1596 ok(hr == S_OK, "got %08x\n", hr);
1598 hr = IMXWriter_put_version(writer, _bstr_(""));
1599 ok(hr == S_OK, "got %08x\n", hr);
1600 hr = IMXWriter_get_version(writer, &str);
1601 ok(hr == S_OK, "got %08x\n", hr);
1602 ok(!lstrcmpW(str, _bstr_("")), "got %s\n", wine_dbgstr_w(str));
1603 SysFreeString(str);
1605 hr = IMXWriter_put_version(writer, _bstr_("a.b"));
1606 ok(hr == S_OK, "got %08x\n", hr);
1607 hr = IMXWriter_get_version(writer, &str);
1608 ok(hr == S_OK, "got %08x\n", hr);
1609 ok(!lstrcmpW(str, _bstr_("a.b")), "got %s\n", wine_dbgstr_w(str));
1610 SysFreeString(str);
1612 hr = IMXWriter_put_version(writer, _bstr_("2.0"));
1613 ok(hr == S_OK, "got %08x\n", hr);
1614 hr = IMXWriter_get_version(writer, &str);
1615 ok(hr == S_OK, "got %08x\n", hr);
1616 ok(!lstrcmpW(str, _bstr_("2.0")), "got %s\n", wine_dbgstr_w(str));
1617 SysFreeString(str);
1619 IMXWriter_Release(writer);
1620 free_bstrs();
1623 static void test_mxwriter_flush(void)
1625 ISAXContentHandler *content;
1626 IMXWriter *writer;
1627 LARGE_INTEGER pos;
1628 ULARGE_INTEGER pos2;
1629 IStream *stream;
1630 VARIANT dest;
1631 HRESULT hr;
1633 hr = CoCreateInstance(&CLSID_MXXMLWriter, NULL, CLSCTX_INPROC_SERVER,
1634 &IID_IMXWriter, (void**)&writer);
1635 ok(hr == S_OK, "Expected S_OK, got %08x\n", hr);
1637 hr = CreateStreamOnHGlobal(NULL, TRUE, &stream);
1638 ok(hr == S_OK, "Expected S_OK, got %08x\n", hr);
1639 EXPECT_REF(stream, 1);
1641 /* detach when nothing was attached */
1642 V_VT(&dest) = VT_EMPTY;
1643 hr = IMXWriter_put_output(writer, dest);
1644 ok(hr == S_OK, "Expected S_OK, got %08x\n", hr);
1646 /* attach stream */
1647 V_VT(&dest) = VT_UNKNOWN;
1648 V_UNKNOWN(&dest) = (IUnknown*)stream;
1649 hr = IMXWriter_put_output(writer, dest);
1650 ok(hr == S_OK, "Expected S_OK, got %08x\n", hr);
1651 todo_wine EXPECT_REF(stream, 3);
1653 /* detach setting VT_EMPTY destination */
1654 V_VT(&dest) = VT_EMPTY;
1655 hr = IMXWriter_put_output(writer, dest);
1656 ok(hr == S_OK, "Expected S_OK, got %08x\n", hr);
1657 EXPECT_REF(stream, 1);
1659 V_VT(&dest) = VT_UNKNOWN;
1660 V_UNKNOWN(&dest) = (IUnknown*)stream;
1661 hr = IMXWriter_put_output(writer, dest);
1662 ok(hr == S_OK, "Expected S_OK, got %08x\n", hr);
1664 /* flush() doesn't detach a stream */
1665 hr = IMXWriter_flush(writer);
1666 ok(hr == S_OK, "Expected S_OK, got %08x\n", hr);
1667 todo_wine EXPECT_REF(stream, 3);
1669 pos.QuadPart = 0;
1670 hr = IStream_Seek(stream, pos, STREAM_SEEK_CUR, &pos2);
1671 ok(hr == S_OK, "Expected S_OK, got %08x\n", hr);
1672 ok(pos2.QuadPart == 0, "expected stream beginning\n");
1674 hr = IMXWriter_QueryInterface(writer, &IID_ISAXContentHandler, (void**)&content);
1675 ok(hr == S_OK, "got %08x\n", hr);
1677 hr = ISAXContentHandler_startDocument(content);
1678 ok(hr == S_OK, "got %08x\n", hr);
1680 pos.QuadPart = 0;
1681 hr = IStream_Seek(stream, pos, STREAM_SEEK_CUR, &pos2);
1682 ok(hr == S_OK, "Expected S_OK, got %08x\n", hr);
1683 ok(pos2.QuadPart != 0, "expected stream beginning\n");
1685 /* already started */
1686 hr = ISAXContentHandler_startDocument(content);
1687 ok(hr == S_OK, "got %08x\n", hr);
1689 hr = ISAXContentHandler_endDocument(content);
1690 ok(hr == S_OK, "got %08x\n", hr);
1692 /* flushed on endDocument() */
1693 pos.QuadPart = 0;
1694 hr = IStream_Seek(stream, pos, STREAM_SEEK_CUR, &pos2);
1695 ok(hr == S_OK, "Expected S_OK, got %08x\n", hr);
1696 ok(pos2.QuadPart != 0, "expected stream position moved\n");
1698 ISAXContentHandler_Release(content);
1699 IStream_Release(stream);
1700 IMXWriter_Release(writer);
1703 static void test_mxwriter_startenddocument(void)
1705 ISAXContentHandler *content;
1706 IMXWriter *writer;
1707 VARIANT dest;
1708 HRESULT hr;
1710 hr = CoCreateInstance(&CLSID_MXXMLWriter, NULL, CLSCTX_INPROC_SERVER,
1711 &IID_IMXWriter, (void**)&writer);
1712 ok(hr == S_OK, "Expected S_OK, got %08x\n", hr);
1714 hr = IMXWriter_QueryInterface(writer, &IID_ISAXContentHandler, (void**)&content);
1715 ok(hr == S_OK, "got %08x\n", hr);
1717 hr = ISAXContentHandler_startDocument(content);
1718 ok(hr == S_OK, "got %08x\n", hr);
1720 hr = ISAXContentHandler_endDocument(content);
1721 ok(hr == S_OK, "got %08x\n", hr);
1723 V_VT(&dest) = VT_EMPTY;
1724 hr = IMXWriter_get_output(writer, &dest);
1725 ok(hr == S_OK, "got %08x\n", hr);
1726 ok(V_VT(&dest) == VT_BSTR, "got %d\n", V_VT(&dest));
1727 ok(!lstrcmpW(_bstr_("<?xml version=\"1.0\" encoding=\"UTF-16\" standalone=\"no\"?>\r\n"), V_BSTR(&dest)),
1728 "got wrong content %s\n", wine_dbgstr_w(V_BSTR(&dest)));
1729 VariantClear(&dest);
1731 /* now try another startDocument */
1732 hr = ISAXContentHandler_startDocument(content);
1733 ok(hr == S_OK, "got %08x\n", hr);
1734 /* and get duplicated prolog */
1735 V_VT(&dest) = VT_EMPTY;
1736 hr = IMXWriter_get_output(writer, &dest);
1737 ok(hr == S_OK, "got %08x\n", hr);
1738 ok(V_VT(&dest) == VT_BSTR, "got %d\n", V_VT(&dest));
1739 ok(!lstrcmpW(_bstr_("<?xml version=\"1.0\" encoding=\"UTF-16\" standalone=\"no\"?>\r\n"
1740 "<?xml version=\"1.0\" encoding=\"UTF-16\" standalone=\"no\"?>\r\n"), V_BSTR(&dest)),
1741 "got wrong content %s\n", wine_dbgstr_w(V_BSTR(&dest)));
1742 VariantClear(&dest);
1744 ISAXContentHandler_Release(content);
1745 IMXWriter_Release(writer);
1747 /* now with omitted declaration */
1748 hr = CoCreateInstance(&CLSID_MXXMLWriter, NULL, CLSCTX_INPROC_SERVER,
1749 &IID_IMXWriter, (void**)&writer);
1750 ok(hr == S_OK, "Expected S_OK, got %08x\n", hr);
1752 hr = IMXWriter_QueryInterface(writer, &IID_ISAXContentHandler, (void**)&content);
1753 ok(hr == S_OK, "got %08x\n", hr);
1755 hr = IMXWriter_put_omitXMLDeclaration(writer, VARIANT_TRUE);
1756 ok(hr == S_OK, "got %08x\n", hr);
1758 hr = ISAXContentHandler_startDocument(content);
1759 ok(hr == S_OK, "got %08x\n", hr);
1761 hr = ISAXContentHandler_endDocument(content);
1762 ok(hr == S_OK, "got %08x\n", hr);
1764 V_VT(&dest) = VT_EMPTY;
1765 hr = IMXWriter_get_output(writer, &dest);
1766 ok(hr == S_OK, "got %08x\n", hr);
1767 ok(V_VT(&dest) == VT_BSTR, "got %d\n", V_VT(&dest));
1768 ok(!lstrcmpW(_bstr_(""), V_BSTR(&dest)), "got wrong content %s\n", wine_dbgstr_w(V_BSTR(&dest)));
1769 VariantClear(&dest);
1771 ISAXContentHandler_Release(content);
1772 IMXWriter_Release(writer);
1774 free_bstrs();
1777 enum startendtype
1779 StartElement,
1780 EndElement,
1781 StartEndElement
1784 struct writer_startendelement_t {
1785 const GUID *clsid;
1786 enum startendtype type;
1787 const char *uri;
1788 const char *local_name;
1789 const char *qname;
1790 const char *output;
1791 HRESULT hr;
1792 ISAXAttributes *attr;
1795 static const struct writer_startendelement_t writer_startendelement[] = {
1796 /* 0 */
1797 { &CLSID_MXXMLWriter, StartElement, NULL, NULL, NULL, NULL, E_INVALIDARG },
1798 { &CLSID_MXXMLWriter30, StartElement, NULL, NULL, NULL, NULL, E_INVALIDARG },
1799 { &CLSID_MXXMLWriter40, StartElement, NULL, NULL, NULL, NULL, E_INVALIDARG },
1800 { &CLSID_MXXMLWriter60, StartElement, NULL, NULL, NULL, "<>", S_OK },
1801 { &CLSID_MXXMLWriter, StartElement, "uri", NULL, NULL, NULL, E_INVALIDARG },
1802 /* 5 */
1803 { &CLSID_MXXMLWriter30, StartElement, "uri", NULL, NULL, NULL, E_INVALIDARG },
1804 { &CLSID_MXXMLWriter40, StartElement, "uri", NULL, NULL, NULL, E_INVALIDARG },
1805 { &CLSID_MXXMLWriter60, StartElement, "uri", NULL, NULL, "<>", S_OK },
1806 { &CLSID_MXXMLWriter, StartElement, NULL, "local", NULL, NULL, E_INVALIDARG },
1807 { &CLSID_MXXMLWriter30, StartElement, NULL, "local", NULL, NULL, E_INVALIDARG },
1808 /* 10 */
1809 { &CLSID_MXXMLWriter40, StartElement, NULL, "local", NULL, NULL, E_INVALIDARG },
1810 { &CLSID_MXXMLWriter60, StartElement, NULL, "local", NULL, "<>", S_OK },
1811 { &CLSID_MXXMLWriter, StartElement, NULL, NULL, "qname", NULL, E_INVALIDARG },
1812 { &CLSID_MXXMLWriter30, StartElement, NULL, NULL, "qname", NULL, E_INVALIDARG },
1813 { &CLSID_MXXMLWriter40, StartElement, NULL, NULL, "qname", NULL, E_INVALIDARG },
1814 /* 15 */
1815 { &CLSID_MXXMLWriter60, StartElement, NULL, NULL, "qname", "<qname>", S_OK },
1816 { &CLSID_MXXMLWriter, StartElement, "uri", "local", "qname", "<qname>", S_OK },
1817 { &CLSID_MXXMLWriter30, StartElement, "uri", "local", "qname", "<qname>", S_OK },
1818 { &CLSID_MXXMLWriter40, StartElement, "uri", "local", "qname", "<qname>", S_OK },
1819 { &CLSID_MXXMLWriter60, StartElement, "uri", "local", "qname", "<qname>", S_OK },
1820 /* 20 */
1821 { &CLSID_MXXMLWriter, StartElement, "uri", "local", NULL, NULL, E_INVALIDARG },
1822 { &CLSID_MXXMLWriter30, StartElement, "uri", "local", NULL, NULL, E_INVALIDARG },
1823 { &CLSID_MXXMLWriter40, StartElement, "uri", "local", NULL, NULL, E_INVALIDARG },
1824 { &CLSID_MXXMLWriter60, StartElement, "uri", "local", NULL, "<>", S_OK },
1825 { &CLSID_MXXMLWriter, StartElement, "uri", "local", "uri:local", "<uri:local>", S_OK },
1826 /* 25 */
1827 { &CLSID_MXXMLWriter30, StartElement, "uri", "local", "uri:local", "<uri:local>", S_OK },
1828 { &CLSID_MXXMLWriter40, StartElement, "uri", "local", "uri:local", "<uri:local>", S_OK },
1829 { &CLSID_MXXMLWriter60, StartElement, "uri", "local", "uri:local", "<uri:local>", S_OK },
1830 { &CLSID_MXXMLWriter, StartElement, "uri", "local", "uri:local2", "<uri:local2>", S_OK },
1831 { &CLSID_MXXMLWriter30, StartElement, "uri", "local", "uri:local2", "<uri:local2>", S_OK },
1832 /* 30 */
1833 { &CLSID_MXXMLWriter40, StartElement, "uri", "local", "uri:local2", "<uri:local2>", S_OK },
1834 { &CLSID_MXXMLWriter60, StartElement, "uri", "local", "uri:local2", "<uri:local2>", S_OK },
1835 /* endElement tests */
1836 { &CLSID_MXXMLWriter, EndElement, NULL, NULL, NULL, NULL, E_INVALIDARG },
1837 { &CLSID_MXXMLWriter30, EndElement, NULL, NULL, NULL, NULL, E_INVALIDARG },
1838 { &CLSID_MXXMLWriter40, EndElement, NULL, NULL, NULL, NULL, E_INVALIDARG },
1839 /* 35 */
1840 { &CLSID_MXXMLWriter60, EndElement, NULL, NULL, NULL, "</>", S_OK },
1841 { &CLSID_MXXMLWriter, EndElement, "uri", NULL, NULL, NULL, E_INVALIDARG },
1842 { &CLSID_MXXMLWriter30, EndElement, "uri", NULL, NULL, NULL, E_INVALIDARG },
1843 { &CLSID_MXXMLWriter40, EndElement, "uri", NULL, NULL, NULL, E_INVALIDARG },
1844 { &CLSID_MXXMLWriter60, EndElement, "uri", NULL, NULL, "</>", S_OK },
1845 /* 40 */
1846 { &CLSID_MXXMLWriter, EndElement, NULL, "local", NULL, NULL, E_INVALIDARG },
1847 { &CLSID_MXXMLWriter30, EndElement, NULL, "local", NULL, NULL, E_INVALIDARG },
1848 { &CLSID_MXXMLWriter40, EndElement, NULL, "local", NULL, NULL, E_INVALIDARG },
1849 { &CLSID_MXXMLWriter60, EndElement, NULL, "local", NULL, "</>", S_OK },
1850 { &CLSID_MXXMLWriter, EndElement, NULL, NULL, "qname", NULL, E_INVALIDARG },
1851 /* 45 */
1852 { &CLSID_MXXMLWriter30, EndElement, NULL, NULL, "qname", NULL, E_INVALIDARG },
1853 { &CLSID_MXXMLWriter40, EndElement, NULL, NULL, "qname", NULL, E_INVALIDARG },
1854 { &CLSID_MXXMLWriter60, EndElement, NULL, NULL, "qname", "</qname>", S_OK },
1855 { &CLSID_MXXMLWriter, EndElement, "uri", "local", "qname", "</qname>", S_OK },
1856 { &CLSID_MXXMLWriter30, EndElement, "uri", "local", "qname", "</qname>", S_OK },
1857 /* 50 */
1858 { &CLSID_MXXMLWriter40, EndElement, "uri", "local", "qname", "</qname>", S_OK },
1859 { &CLSID_MXXMLWriter60, EndElement, "uri", "local", "qname", "</qname>", S_OK },
1860 { &CLSID_MXXMLWriter, EndElement, "uri", "local", NULL, NULL, E_INVALIDARG },
1861 { &CLSID_MXXMLWriter30, EndElement, "uri", "local", NULL, NULL, E_INVALIDARG },
1862 { &CLSID_MXXMLWriter40, EndElement, "uri", "local", NULL, NULL, E_INVALIDARG },
1863 /* 55 */
1864 { &CLSID_MXXMLWriter60, EndElement, "uri", "local", NULL, "</>", S_OK },
1865 { &CLSID_MXXMLWriter, EndElement, "uri", "local", "uri:local", "</uri:local>", S_OK },
1866 { &CLSID_MXXMLWriter30, EndElement, "uri", "local", "uri:local", "</uri:local>", S_OK },
1867 { &CLSID_MXXMLWriter40, EndElement, "uri", "local", "uri:local", "</uri:local>", S_OK },
1868 { &CLSID_MXXMLWriter60, EndElement, "uri", "local", "uri:local", "</uri:local>", S_OK },
1869 /* 60 */
1870 { &CLSID_MXXMLWriter, EndElement, "uri", "local", "uri:local2", "</uri:local2>", S_OK },
1871 { &CLSID_MXXMLWriter30, EndElement, "uri", "local", "uri:local2", "</uri:local2>", S_OK },
1872 { &CLSID_MXXMLWriter40, EndElement, "uri", "local", "uri:local2", "</uri:local2>", S_OK },
1873 { &CLSID_MXXMLWriter60, EndElement, "uri", "local", "uri:local2", "</uri:local2>", S_OK },
1875 /* with attributes */
1876 { &CLSID_MXXMLWriter, StartElement, "uri", "local", "uri:local", "<uri:local a:attr1=\"a1\" attr2=\"a2\">", S_OK, &saxattributes },
1877 /* 65 */
1878 { &CLSID_MXXMLWriter30, StartElement, "uri", "local", "uri:local", "<uri:local a:attr1=\"a1\" attr2=\"a2\">", S_OK, &saxattributes },
1879 { &CLSID_MXXMLWriter40, StartElement, "uri", "local", "uri:local", "<uri:local a:attr1=\"a1\" attr2=\"a2\">", S_OK, &saxattributes },
1880 { &CLSID_MXXMLWriter60, StartElement, "uri", "local", "uri:local", "<uri:local a:attr1=\"a1\" attr2=\"a2\">", S_OK, &saxattributes },
1881 /* empty elements */
1882 { &CLSID_MXXMLWriter, StartEndElement, "uri", "local", "uri:local", "<uri:local a:attr1=\"a1\" attr2=\"a2\"/>", S_OK, &saxattributes },
1883 { &CLSID_MXXMLWriter30, StartEndElement, "uri", "local", "uri:local", "<uri:local a:attr1=\"a1\" attr2=\"a2\"/>", S_OK, &saxattributes },
1884 /* 70 */
1885 { &CLSID_MXXMLWriter40, StartEndElement, "uri", "local", "uri:local", "<uri:local a:attr1=\"a1\" attr2=\"a2\"/>", S_OK, &saxattributes },
1886 { &CLSID_MXXMLWriter60, StartEndElement, "uri", "local", "uri:local", "<uri:local a:attr1=\"a1\" attr2=\"a2\"/>", S_OK, &saxattributes },
1887 { &CLSID_MXXMLWriter, StartEndElement, "", "", "", "</>", S_OK },
1888 { &CLSID_MXXMLWriter30, StartEndElement, "", "", "", "</>", S_OK },
1889 { &CLSID_MXXMLWriter40, StartEndElement, "", "", "", "</>", S_OK },
1890 /* 75 */
1891 { &CLSID_MXXMLWriter60, StartEndElement, "", "", "", "</>", S_OK },
1892 { NULL }
1895 static void get_supported_mxwriter_data(struct msxmlsupported_data_t *table)
1897 while (table->clsid)
1899 IMXWriter *writer;
1900 HRESULT hr;
1902 hr = CoCreateInstance(table->clsid, NULL, CLSCTX_INPROC_SERVER,
1903 &IID_IMXWriter, (void**)&writer);
1904 if (hr == S_OK) IMXWriter_Release(writer);
1906 table->supported = hr == S_OK;
1907 if (hr != S_OK) win_skip("class %s not supported\n", table->name);
1909 table++;
1913 static void test_mxwriter_startendelement_batch(const struct writer_startendelement_t *table)
1915 int i = 0;
1917 while (table->clsid)
1919 ISAXContentHandler *content;
1920 IMXWriter *writer;
1921 HRESULT hr;
1923 if (!is_mxwriter_supported(table->clsid, msxmlsupported_data))
1925 table++;
1926 i++;
1927 continue;
1930 hr = CoCreateInstance(table->clsid, NULL, CLSCTX_INPROC_SERVER,
1931 &IID_IMXWriter, (void**)&writer);
1932 EXPECT_HR(hr, S_OK);
1934 hr = IMXWriter_QueryInterface(writer, &IID_ISAXContentHandler, (void**)&content);
1935 EXPECT_HR(hr, S_OK);
1937 hr = IMXWriter_put_omitXMLDeclaration(writer, VARIANT_TRUE);
1938 EXPECT_HR(hr, S_OK);
1940 hr = ISAXContentHandler_startDocument(content);
1941 EXPECT_HR(hr, S_OK);
1943 if (table->type == StartElement)
1945 hr = ISAXContentHandler_startElement(content, _bstr_(table->uri), lstrlen(table->uri),
1946 _bstr_(table->local_name), lstrlen(table->local_name), _bstr_(table->qname), lstrlen(table->qname), table->attr);
1947 ok(hr == table->hr, "test %d: got 0x%08x, expected 0x%08x\n", i, hr, table->hr);
1949 else if (table->type == EndElement)
1951 hr = ISAXContentHandler_endElement(content, _bstr_(table->uri), lstrlen(table->uri),
1952 _bstr_(table->local_name), lstrlen(table->local_name), _bstr_(table->qname), lstrlen(table->qname));
1953 ok(hr == table->hr, "test %d: got 0x%08x, expected 0x%08x\n", i, hr, table->hr);
1955 else
1957 hr = ISAXContentHandler_startElement(content, _bstr_(table->uri), lstrlen(table->uri),
1958 _bstr_(table->local_name), lstrlen(table->local_name), _bstr_(table->qname), lstrlen(table->qname), table->attr);
1959 ok(hr == table->hr, "test %d: got 0x%08x, expected 0x%08x\n", i, hr, table->hr);
1960 hr = ISAXContentHandler_endElement(content, _bstr_(table->uri), lstrlen(table->uri),
1961 _bstr_(table->local_name), lstrlen(table->local_name), _bstr_(table->qname), lstrlen(table->qname));
1962 ok(hr == table->hr, "test %d: got 0x%08x, expected 0x%08x\n", i, hr, table->hr);
1965 /* test output */
1966 if (hr == S_OK)
1968 VARIANT dest;
1970 V_VT(&dest) = VT_EMPTY;
1971 hr = IMXWriter_get_output(writer, &dest);
1972 EXPECT_HR(hr, S_OK);
1973 ok(V_VT(&dest) == VT_BSTR, "got %d\n", V_VT(&dest));
1974 ok(!lstrcmpW(_bstr_(table->output), V_BSTR(&dest)),
1975 "test %d: got wrong content %s, expected %s\n", i, wine_dbgstr_w(V_BSTR(&dest)), table->output);
1976 VariantClear(&dest);
1979 ISAXContentHandler_Release(content);
1980 IMXWriter_Release(writer);
1982 table++;
1983 i++;
1986 free_bstrs();
1989 static void test_mxwriter_startendelement(void)
1991 ISAXContentHandler *content;
1992 IMXWriter *writer;
1993 VARIANT dest;
1994 HRESULT hr;
1996 test_mxwriter_startendelement_batch(writer_startendelement);
1998 hr = CoCreateInstance(&CLSID_MXXMLWriter, NULL, CLSCTX_INPROC_SERVER,
1999 &IID_IMXWriter, (void**)&writer);
2000 ok(hr == S_OK, "Expected S_OK, got %08x\n", hr);
2002 hr = IMXWriter_QueryInterface(writer, &IID_ISAXContentHandler, (void**)&content);
2003 ok(hr == S_OK, "got %08x\n", hr);
2005 hr = IMXWriter_put_omitXMLDeclaration(writer, VARIANT_TRUE);
2006 ok(hr == S_OK, "got %08x\n", hr);
2008 hr = ISAXContentHandler_startDocument(content);
2009 ok(hr == S_OK, "got %08x\n", hr);
2011 /* all string pointers should be not null */
2012 hr = ISAXContentHandler_startElement(content, _bstr_(""), 0, _bstr_("b"), 1, _bstr_(""), 0, NULL);
2013 ok(hr == S_OK, "got %08x\n", hr);
2015 V_VT(&dest) = VT_EMPTY;
2016 hr = IMXWriter_get_output(writer, &dest);
2017 ok(hr == S_OK, "got %08x\n", hr);
2018 ok(V_VT(&dest) == VT_BSTR, "got %d\n", V_VT(&dest));
2019 ok(!lstrcmpW(_bstr_("<>"), V_BSTR(&dest)), "got wrong content %s\n", wine_dbgstr_w(V_BSTR(&dest)));
2020 VariantClear(&dest);
2022 hr = ISAXContentHandler_startElement(content, _bstr_(""), 0, _bstr_(""), 0, _bstr_("b"), 1, NULL);
2023 ok(hr == S_OK, "got %08x\n", hr);
2025 V_VT(&dest) = VT_EMPTY;
2026 hr = IMXWriter_get_output(writer, &dest);
2027 ok(hr == S_OK, "got %08x\n", hr);
2028 ok(V_VT(&dest) == VT_BSTR, "got %d\n", V_VT(&dest));
2029 ok(!lstrcmpW(_bstr_("<><b>"), V_BSTR(&dest)), "got wrong content %s\n", wine_dbgstr_w(V_BSTR(&dest)));
2030 VariantClear(&dest);
2032 hr = ISAXContentHandler_endElement(content, NULL, 0, NULL, 0, _bstr_("a:b"), 3);
2033 EXPECT_HR(hr, E_INVALIDARG);
2035 hr = ISAXContentHandler_endElement(content, NULL, 0, _bstr_("b"), 1, _bstr_("a:b"), 3);
2036 EXPECT_HR(hr, E_INVALIDARG);
2038 /* only local name is an error too */
2039 hr = ISAXContentHandler_endElement(content, NULL, 0, _bstr_("b"), 1, NULL, 0);
2040 EXPECT_HR(hr, E_INVALIDARG);
2042 hr = ISAXContentHandler_endElement(content, _bstr_(""), 0, _bstr_(""), 0, _bstr_("b"), 1);
2043 EXPECT_HR(hr, S_OK);
2045 V_VT(&dest) = VT_EMPTY;
2046 hr = IMXWriter_get_output(writer, &dest);
2047 EXPECT_HR(hr, S_OK);
2048 ok(V_VT(&dest) == VT_BSTR, "got %d\n", V_VT(&dest));
2049 ok(!lstrcmpW(_bstr_("<><b></b>"), V_BSTR(&dest)), "got wrong content %s\n", wine_dbgstr_w(V_BSTR(&dest)));
2050 VariantClear(&dest);
2052 hr = ISAXContentHandler_endDocument(content);
2053 EXPECT_HR(hr, S_OK);
2055 V_VT(&dest) = VT_EMPTY;
2056 hr = IMXWriter_put_output(writer, dest);
2057 EXPECT_HR(hr, S_OK);
2059 hr = ISAXContentHandler_startDocument(content);
2060 EXPECT_HR(hr, S_OK);
2062 hr = ISAXContentHandler_startElement(content, _bstr_(""), 0, _bstr_(""), 0, _bstr_("abcdef"), 3, NULL);
2063 EXPECT_HR(hr, S_OK);
2065 V_VT(&dest) = VT_EMPTY;
2066 hr = IMXWriter_get_output(writer, &dest);
2067 EXPECT_HR(hr, S_OK);
2068 ok(V_VT(&dest) == VT_BSTR, "got %d\n", V_VT(&dest));
2069 ok(!lstrcmpW(_bstr_("<abc>"), V_BSTR(&dest)), "got wrong content %s\n", wine_dbgstr_w(V_BSTR(&dest)));
2070 VariantClear(&dest);
2072 ISAXContentHandler_Release(content);
2073 IMXWriter_Release(writer);
2075 free_bstrs();
2078 static void test_mxwriter_characters(void)
2080 static const WCHAR chardataW[] = {'T','E','S','T','C','H','A','R','D','A','T','A',' ','.',0};
2081 ISAXContentHandler *content;
2082 IMXWriter *writer;
2083 VARIANT dest;
2084 HRESULT hr;
2086 hr = CoCreateInstance(&CLSID_MXXMLWriter, NULL, CLSCTX_INPROC_SERVER,
2087 &IID_IMXWriter, (void**)&writer);
2088 EXPECT_HR(hr, S_OK);
2090 hr = IMXWriter_QueryInterface(writer, &IID_ISAXContentHandler, (void**)&content);
2091 EXPECT_HR(hr, S_OK);
2093 hr = IMXWriter_put_omitXMLDeclaration(writer, VARIANT_TRUE);
2094 EXPECT_HR(hr, S_OK);
2096 hr = ISAXContentHandler_startDocument(content);
2097 EXPECT_HR(hr, S_OK);
2099 hr = ISAXContentHandler_characters(content, NULL, 0);
2100 EXPECT_HR(hr, E_INVALIDARG);
2102 hr = ISAXContentHandler_characters(content, chardataW, 0);
2103 EXPECT_HR(hr, S_OK);
2105 hr = ISAXContentHandler_characters(content, chardataW, sizeof(chardataW)/sizeof(WCHAR) - 1);
2106 EXPECT_HR(hr, S_OK);
2108 V_VT(&dest) = VT_EMPTY;
2109 hr = IMXWriter_get_output(writer, &dest);
2110 EXPECT_HR(hr, S_OK);
2111 ok(V_VT(&dest) == VT_BSTR, "got %d\n", V_VT(&dest));
2112 ok(!lstrcmpW(_bstr_("TESTCHARDATA ."), V_BSTR(&dest)), "got wrong content %s\n", wine_dbgstr_w(V_BSTR(&dest)));
2113 VariantClear(&dest);
2115 hr = ISAXContentHandler_endDocument(content);
2116 EXPECT_HR(hr, S_OK);
2118 ISAXContentHandler_Release(content);
2119 IMXWriter_Release(writer);
2121 /* try empty characters data to see if element is closed */
2122 hr = CoCreateInstance(&CLSID_MXXMLWriter, NULL, CLSCTX_INPROC_SERVER,
2123 &IID_IMXWriter, (void**)&writer);
2124 EXPECT_HR(hr, S_OK);
2126 hr = IMXWriter_QueryInterface(writer, &IID_ISAXContentHandler, (void**)&content);
2127 EXPECT_HR(hr, S_OK);
2129 hr = IMXWriter_put_omitXMLDeclaration(writer, VARIANT_TRUE);
2130 EXPECT_HR(hr, S_OK);
2132 hr = ISAXContentHandler_startDocument(content);
2133 EXPECT_HR(hr, S_OK);
2135 hr = ISAXContentHandler_startElement(content, _bstr_(""), 0, _bstr_(""), 0, _bstr_("a"), 1, NULL);
2136 EXPECT_HR(hr, S_OK);
2138 hr = ISAXContentHandler_characters(content, chardataW, 0);
2139 EXPECT_HR(hr, S_OK);
2141 hr = ISAXContentHandler_endElement(content, _bstr_(""), 0, _bstr_(""), 0, _bstr_("a"), 1);
2142 EXPECT_HR(hr, S_OK);
2144 V_VT(&dest) = VT_EMPTY;
2145 hr = IMXWriter_get_output(writer, &dest);
2146 EXPECT_HR(hr, S_OK);
2147 ok(V_VT(&dest) == VT_BSTR, "got %d\n", V_VT(&dest));
2148 ok(!lstrcmpW(_bstr_("<a></a>"), V_BSTR(&dest)), "got wrong content %s\n", wine_dbgstr_w(V_BSTR(&dest)));
2149 VariantClear(&dest);
2151 ISAXContentHandler_Release(content);
2152 IMXWriter_Release(writer);
2154 free_bstrs();
2157 static const mxwriter_stream_test mxwriter_stream_tests[] = {
2159 VARIANT_TRUE,"UTF-16",
2161 {FALSE,(const BYTE*)szUtf16BOM,sizeof(szUtf16BOM),TRUE},
2162 {FALSE,(const BYTE*)szUtf16XML,sizeof(szUtf16XML)},
2163 {TRUE}
2167 VARIANT_FALSE,"UTF-16",
2169 {FALSE,(const BYTE*)szUtf16XML,sizeof(szUtf16XML)},
2170 {TRUE}
2174 VARIANT_TRUE,"UTF-8",
2176 {FALSE,(const BYTE*)szUtf8XML,sizeof(szUtf8XML)-1},
2177 /* For some reason Windows makes an empty write call when UTF-8 encoding is used
2178 * and the writer is released.
2180 {FALSE,NULL,0},
2181 {TRUE}
2185 VARIANT_TRUE,"UTF-16",
2187 {FALSE,(const BYTE*)szUtf16BOM,sizeof(szUtf16BOM),TRUE},
2188 {FALSE,(const BYTE*)szUtf16XML,sizeof(szUtf16XML)},
2189 {TRUE}
2193 VARIANT_TRUE,"UTF-16",
2195 {FALSE,(const BYTE*)szUtf16BOM,sizeof(szUtf16BOM),TRUE,TRUE},
2196 {FALSE,(const BYTE*)szUtf16XML,sizeof(szUtf16XML)},
2197 {TRUE}
2202 static void test_mxwriter_stream(void)
2204 IMXWriter *writer;
2205 ISAXContentHandler *content;
2206 HRESULT hr;
2207 VARIANT dest;
2208 IStream *stream;
2209 LARGE_INTEGER pos;
2210 ULARGE_INTEGER pos2;
2211 DWORD test_count = sizeof(mxwriter_stream_tests)/sizeof(mxwriter_stream_tests[0]);
2213 for(current_stream_test_index = 0; current_stream_test_index < test_count; ++current_stream_test_index) {
2214 const mxwriter_stream_test *test = mxwriter_stream_tests+current_stream_test_index;
2216 hr = CoCreateInstance(&CLSID_MXXMLWriter, NULL, CLSCTX_INPROC_SERVER,
2217 &IID_IMXWriter, (void**)&writer);
2218 ok(hr == S_OK, "CoCreateInstance failed: %08x\n", hr);
2220 hr = IMXWriter_QueryInterface(writer, &IID_ISAXContentHandler, (void**)&content);
2221 ok(hr == S_OK, "QueryInterface(ISAXContentHandler) failed: %08x\n", hr);
2223 hr = IMXWriter_put_encoding(writer, _bstr_(test->encoding));
2224 ok(hr == S_OK, "put_encoding failed with %08x on test %d\n", hr, current_stream_test_index);
2226 V_VT(&dest) = VT_UNKNOWN;
2227 V_UNKNOWN(&dest) = (IUnknown*)&mxstream;
2228 hr = IMXWriter_put_output(writer, dest);
2229 ok(hr == S_OK, "put_output failed with %08x on test %d\n", hr, current_stream_test_index);
2230 VariantClear(&dest);
2232 hr = IMXWriter_put_byteOrderMark(writer, test->bom);
2233 ok(hr == S_OK, "put_byteOrderMark failed with %08x on test %d\n", hr, current_stream_test_index);
2235 current_write_test = test->expected_writes;
2237 hr = ISAXContentHandler_startDocument(content);
2238 ok(hr == S_OK, "startDocument failed with %08x on test %d\n", hr, current_stream_test_index);
2240 hr = ISAXContentHandler_endDocument(content);
2241 ok(hr == S_OK, "endDocument failed with %08x on test %d\n", hr, current_stream_test_index);
2243 ISAXContentHandler_Release(content);
2244 IMXWriter_Release(writer);
2246 ok(current_write_test->last, "The last %d write calls on test %d were missed\n",
2247 (int)(current_write_test-test->expected_writes), current_stream_test_index);
2250 hr = CoCreateInstance(&CLSID_MXXMLWriter, NULL, CLSCTX_INPROC_SERVER,
2251 &IID_IMXWriter, (void**)&writer);
2252 ok(hr == S_OK, "CoCreateInstance failed: %08x\n", hr);
2254 hr = CreateStreamOnHGlobal(NULL, TRUE, &stream);
2255 ok(hr == S_OK, "CreateStreamOnHGlobal failed: %08x\n", hr);
2257 hr = IMXWriter_QueryInterface(writer, &IID_ISAXContentHandler, (void**)&content);
2258 ok(hr == S_OK, "QueryInterface(ISAXContentHandler) failed: %08x\n", hr);
2260 hr = IMXWriter_put_encoding(writer, _bstr_("UTF-8"));
2261 ok(hr == S_OK, "put_encoding failed: %08x\n", hr);
2263 V_VT(&dest) = VT_UNKNOWN;
2264 V_UNKNOWN(&dest) = (IUnknown*)stream;
2265 hr = IMXWriter_put_output(writer, dest);
2266 ok(hr == S_OK, "put_output failed: %08x\n", hr);
2268 hr = ISAXContentHandler_startDocument(content);
2269 ok(hr == S_OK, "startDocument failed: %08x\n", hr);
2271 /* Setting output of the mxwriter causes the current output to be flushed,
2272 * and the writer to start over.
2274 V_VT(&dest) = VT_EMPTY;
2275 hr = IMXWriter_put_output(writer, dest);
2276 ok(hr == S_OK, "put_output failed: %08x\n", hr);
2278 pos.QuadPart = 0;
2279 hr = IStream_Seek(stream, pos, STREAM_SEEK_CUR, &pos2);
2280 ok(hr == S_OK, "Seek failed: %08x\n", hr);
2281 ok(pos2.QuadPart != 0, "expected stream position moved\n");
2283 hr = ISAXContentHandler_startDocument(content);
2284 ok(hr == S_OK, "startDocument failed: %08x\n", hr);
2286 hr = ISAXContentHandler_endDocument(content);
2287 ok(hr == S_OK, "endDocument failed: %08x\n", hr);
2289 V_VT(&dest) = VT_EMPTY;
2290 hr = IMXWriter_get_output(writer, &dest);
2291 ok(hr == S_OK, "get_output failed: %08x\n", hr);
2292 ok(V_VT(&dest) == VT_BSTR, "Expected VT_BSTR, got %d\n", V_VT(&dest));
2293 ok(!lstrcmpW(_bstr_("<?xml version=\"1.0\" encoding=\"UTF-16\" standalone=\"no\"?>\r\n"), V_BSTR(&dest)),
2294 "Got wrong content: %s\n", wine_dbgstr_w(V_BSTR(&dest)));
2295 VariantClear(&dest);
2297 ISAXContentHandler_Release(content);
2298 IMXWriter_Release(writer);
2300 free_bstrs();
2303 static void test_mxwriter_encoding(void)
2305 IMXWriter *writer;
2306 ISAXContentHandler *content;
2307 HRESULT hr;
2308 VARIANT dest;
2310 hr = CoCreateInstance(&CLSID_MXXMLWriter, NULL, CLSCTX_INPROC_SERVER,
2311 &IID_IMXWriter, (void**)&writer);
2312 ok(hr == S_OK, "CoCreateInstance failed: %08x\n", hr);
2314 hr = IMXWriter_QueryInterface(writer, &IID_ISAXContentHandler, (void**)&content);
2315 ok(hr == S_OK, "QueryInterface(ISAXContentHandler) failed: %08x\n", hr);
2317 hr = IMXWriter_put_encoding(writer, _bstr_("UTF-8"));
2318 ok(hr == S_OK, "put_encoding failed: %08x\n", hr);
2320 hr = ISAXContentHandler_startDocument(content);
2321 ok(hr == S_OK, "startDocument failed: %08x\n", hr);
2323 hr = ISAXContentHandler_endDocument(content);
2324 ok(hr == S_OK, "endDocument failed: %08x\n", hr);
2326 /* The content is always re-encoded to UTF-16 when the output is
2327 * retrieved as a BSTR.
2329 V_VT(&dest) = VT_EMPTY;
2330 hr = IMXWriter_get_output(writer, &dest);
2331 todo_wine ok(hr == S_OK, "get_output failed: %08x\n", hr);
2332 todo_wine ok(V_VT(&dest) == VT_BSTR, "Expected VT_BSTR, got %d\n", V_VT(&dest));
2333 if (V_VT(&dest) == VT_BSTR) todo_wine ok(!lstrcmpW(_bstr_("<?xml version=\"1.0\" encoding=\"UTF-16\" standalone=\"no\"?>\r\n"), V_BSTR(&dest)),
2334 "got wrong content: %s\n", wine_dbgstr_w(V_BSTR(&dest)));
2335 VariantClear(&dest);
2337 ISAXContentHandler_Release(content);
2338 IMXWriter_Release(writer);
2340 free_bstrs();
2343 START_TEST(saxreader)
2345 ISAXXMLReader *reader;
2346 HRESULT hr;
2348 hr = CoInitialize(NULL);
2349 ok(hr == S_OK, "failed to init com\n");
2351 hr = CoCreateInstance(&CLSID_SAXXMLReader, NULL, CLSCTX_INPROC_SERVER,
2352 &IID_ISAXXMLReader, (void**)&reader);
2354 if(FAILED(hr))
2356 skip("Failed to create SAXXMLReader instance\n");
2357 CoUninitialize();
2358 return;
2360 ISAXXMLReader_Release(reader);
2362 test_saxreader();
2363 test_saxreader_properties();
2364 test_saxreader_features();
2365 test_encoding();
2367 /* MXXMLWriter tests */
2368 get_supported_mxwriter_data(msxmlsupported_data);
2369 if (is_mxwriter_supported(&CLSID_MXXMLWriter, msxmlsupported_data))
2371 test_mxwriter_contenthandler();
2372 test_mxwriter_startenddocument();
2373 test_mxwriter_startendelement();
2374 test_mxwriter_characters();
2375 test_mxwriter_properties();
2376 test_mxwriter_flush();
2377 test_mxwriter_stream();
2378 test_mxwriter_encoding();
2380 else
2381 win_skip("MXXMLWriter not supported\n");
2383 CoUninitialize();