4 * Copyright 2010, 2012-2013 Nikolay Sivov
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
21 #define WIN32_NO_STATUS
23 #define COM_NO_WINDOWS_H
37 #include <wine/test.h>
39 DEFINE_GUID(IID_IXmlReaderInput
, 0x0b3ccc9b, 0x9214, 0x428b, 0xa2, 0xae, 0xef, 0x3a, 0xa8, 0x71, 0xaf, 0xda);
41 static WCHAR
*a2w(const char *str
)
43 int len
= MultiByteToWideChar(CP_ACP
, 0, str
, -1, NULL
, 0);
44 WCHAR
*ret
= HeapAlloc(GetProcessHeap(), 0, len
*sizeof(WCHAR
));
45 MultiByteToWideChar(CP_ACP
, 0, str
, -1, ret
, len
);
49 static void free_str(WCHAR
*str
)
51 HeapFree(GetProcessHeap(), 0, str
);
54 static const char xmldecl_full
[] = "\xef\xbb\xbf<?xml version=\"1.0\" encoding=\"UTF-8\" standalone=\"yes\"?>\n";
55 static const char xmldecl_short
[] = "<?xml version=\"1.0\"?><RegistrationInfo/>";
57 static IStream
*create_stream_on_data(const char *data
, int size
)
59 IStream
*stream
= NULL
;
64 hglobal
= GlobalAlloc(GHND
, size
);
65 ptr
= GlobalLock(hglobal
);
67 memcpy(ptr
, data
, size
);
69 hr
= CreateStreamOnHGlobal(hglobal
, TRUE
, &stream
);
70 ok(hr
== S_OK
, "Expected S_OK, got %08x\n", hr
);
71 ok(stream
!= NULL
, "Expected non-NULL stream\n");
73 GlobalUnlock(hglobal
);
78 static void ok_pos_(IXmlReader
*reader
, int line
, int pos
, int line_broken
,
79 int pos_broken
, BOOL todo
, int _line_
)
85 hr
= IXmlReader_GetLineNumber(reader
, &l
);
86 ok_(__FILE__
, _line_
)(hr
== S_OK
, "Expected S_OK, got %08x\n", hr
);
87 hr
= IXmlReader_GetLinePosition(reader
, &p
);
88 ok_(__FILE__
, _line_
)(hr
== S_OK
, "Expected S_OK, got %08x\n", hr
);
90 if (line_broken
== -1 && pos_broken
== -1)
93 broken_state
= broken((line_broken
== -1 ? line
: line_broken
) == l
&&
94 (pos_broken
== -1 ? pos
: pos_broken
) == p
);
97 ok_(__FILE__
, _line_
)((l
== line
&& pos
== p
) || broken_state
,
98 "Expected (%d,%d), got (%d,%d)\n", line
, pos
, l
, p
);
100 #define ok_pos(reader, l, p, l_brk, p_brk, todo) ok_pos_(reader, l, p, l_brk, p_brk, todo, __LINE__)
102 typedef struct input_iids_t
{
107 static const IID
*setinput_full
[] = {
108 &IID_IXmlReaderInput
,
110 &IID_ISequentialStream
,
114 /* this applies to early xmllite versions */
115 static const IID
*setinput_full_old
[] = {
116 &IID_IXmlReaderInput
,
117 &IID_ISequentialStream
,
122 /* after ::SetInput(IXmlReaderInput*) */
123 static const IID
*setinput_readerinput
[] = {
125 &IID_ISequentialStream
,
129 static const IID
*empty_seq
[] = {
133 static input_iids_t input_iids
;
135 static void ok_iids_(const input_iids_t
*iids
, const IID
**expected
, const IID
**exp_broken
, BOOL todo
, int line
)
139 while (expected
[i
++]) size
++;
142 ok_(__FILE__
, line
)(iids
->count
== size
, "Sequence size mismatch (%d), got (%d)\n", size
, iids
->count
);
144 if (iids
->count
!= size
) return;
146 for (i
= 0; i
< size
; i
++) {
147 ok_(__FILE__
, line
)(IsEqualGUID(&iids
->iids
[i
], expected
[i
]) ||
148 (exp_broken
? broken(IsEqualGUID(&iids
->iids
[i
], exp_broken
[i
])) : FALSE
),
149 "Wrong IID(%d), got %s\n", i
, wine_dbgstr_guid(&iids
->iids
[i
]));
152 #define ok_iids(got, exp, brk, todo) ok_iids_(got, exp, brk, todo, __LINE__)
154 static const char *state_to_str(XmlReadState state
)
156 static const char* state_names
[] = {
157 "XmlReadState_Initial",
158 "XmlReadState_Interactive",
159 "XmlReadState_Error",
160 "XmlReadState_EndOfFile",
161 "XmlReadState_Closed"
164 static const char unknown
[] = "unknown";
168 case XmlReadState_Initial
:
169 case XmlReadState_Interactive
:
170 case XmlReadState_Error
:
171 case XmlReadState_EndOfFile
:
172 case XmlReadState_Closed
:
173 return state_names
[state
];
179 static const char *type_to_str(XmlNodeType type
)
181 static const char* type_names
[] = {
183 "XmlNodeType_Element",
184 "XmlNodeType_Attribute",
188 "XmlNodeType_ProcessingInstruction",
189 "XmlNodeType_Comment",
191 "XmlNodeType_DocumentType",
193 "XmlNodeType_Whitespace",
195 "XmlNodeType_EndElement",
197 "XmlNodeType_XmlDeclaration"
200 static const char unknown
[] = "unknown";
204 case XmlNodeType_None
:
205 case XmlNodeType_Element
:
206 case XmlNodeType_Attribute
:
207 case XmlNodeType_Text
:
208 case XmlNodeType_CDATA
:
209 case XmlNodeType_ProcessingInstruction
:
210 case XmlNodeType_Comment
:
211 case XmlNodeType_DocumentType
:
212 case XmlNodeType_Whitespace
:
213 case XmlNodeType_EndElement
:
214 case XmlNodeType_XmlDeclaration
:
215 return type_names
[type
];
221 static void test_read_state_(IXmlReader
*reader
, XmlReadState expected
,
222 XmlReadState exp_broken
, BOOL todo
, int line
)
228 state
= -1; /* invalid value */
229 hr
= IXmlReader_GetProperty(reader
, XmlReaderProperty_ReadState
, &state
);
230 ok_(__FILE__
, line
)(hr
== S_OK
, "Expected S_OK, got %08x\n", hr
);
232 if (exp_broken
== -1)
233 broken_state
= FALSE
;
235 broken_state
= broken(exp_broken
== state
);
238 ok_(__FILE__
, line
)(state
== expected
|| broken_state
, "Expected (%s), got (%s)\n",
239 state_to_str(expected
), state_to_str(state
));
242 #define test_read_state(reader, exp, brk, todo) test_read_state_(reader, exp, brk, todo, __LINE__)
244 typedef struct _testinput
246 IUnknown IUnknown_iface
;
250 static inline testinput
*impl_from_IUnknown(IUnknown
*iface
)
252 return CONTAINING_RECORD(iface
, testinput
, IUnknown_iface
);
255 static HRESULT WINAPI
testinput_QueryInterface(IUnknown
*iface
, REFIID riid
, void** ppvObj
)
257 if (IsEqualGUID( riid
, &IID_IUnknown
))
260 IUnknown_AddRef(iface
);
264 input_iids
.iids
[input_iids
.count
++] = *riid
;
268 return E_NOINTERFACE
;
271 static ULONG WINAPI
testinput_AddRef(IUnknown
*iface
)
273 testinput
*This
= impl_from_IUnknown(iface
);
274 return InterlockedIncrement(&This
->ref
);
277 static ULONG WINAPI
testinput_Release(IUnknown
*iface
)
279 testinput
*This
= impl_from_IUnknown(iface
);
282 ref
= InterlockedDecrement(&This
->ref
);
285 HeapFree(GetProcessHeap(), 0, This
);
291 static const struct IUnknownVtbl testinput_vtbl
=
293 testinput_QueryInterface
,
298 static HRESULT
testinput_createinstance(void **ppObj
)
302 input
= HeapAlloc(GetProcessHeap(), 0, sizeof (*input
));
303 if(!input
) return E_OUTOFMEMORY
;
305 input
->IUnknown_iface
.lpVtbl
= &testinput_vtbl
;
308 *ppObj
= &input
->IUnknown_iface
;
313 static HRESULT WINAPI
teststream_QueryInterface(ISequentialStream
*iface
, REFIID riid
, void **obj
)
315 if (IsEqualIID(riid
, &IID_IUnknown
) || IsEqualIID(riid
, &IID_ISequentialStream
))
322 return E_NOINTERFACE
;
325 static ULONG WINAPI
teststream_AddRef(ISequentialStream
*iface
)
330 static ULONG WINAPI
teststream_Release(ISequentialStream
*iface
)
335 static int stream_readcall
;
337 static HRESULT WINAPI
teststream_Read(ISequentialStream
*iface
, void *pv
, ULONG cb
, ULONG
*pread
)
339 static const char xml
[] = "<!-- comment -->";
341 if (stream_readcall
++)
347 *pread
= sizeof(xml
) / 2;
348 memcpy(pv
, xml
, *pread
);
352 static HRESULT WINAPI
teststream_Write(ISequentialStream
*iface
, const void *pv
, ULONG cb
, ULONG
*written
)
354 ok(0, "unexpected call\n");
358 static const ISequentialStreamVtbl teststreamvtbl
=
360 teststream_QueryInterface
,
367 static HRESULT WINAPI
resolver_QI(IXmlResolver
*iface
, REFIID riid
, void **obj
)
369 ok(0, "unexpected call, riid %s\n", wine_dbgstr_guid(riid
));
371 if (IsEqualIID(riid
, &IID_IXmlResolver
) || IsEqualIID(riid
, &IID_IUnknown
))
374 IXmlResolver_AddRef(iface
);
379 return E_NOINTERFACE
;
382 static ULONG WINAPI
resolver_AddRef(IXmlResolver
*iface
)
387 static ULONG WINAPI
resolver_Release(IXmlResolver
*iface
)
392 static HRESULT WINAPI
resolver_ResolveUri(IXmlResolver
*iface
, const WCHAR
*base_uri
,
393 const WCHAR
*public_id
, const WCHAR
*system_id
, IUnknown
**input
)
395 ok(0, "unexpected call\n");
399 static const IXmlResolverVtbl resolvervtbl
=
407 static IXmlResolver testresolver
= { &resolvervtbl
};
409 static void test_reader_create(void)
411 IXmlResolver
*resolver
;
416 XmlNodeType nodetype
;
421 CreateXmlReader(&IID_IXmlReader
, NULL
, NULL
);
422 CreateXmlReader(NULL
, (void**)&reader
, NULL
);
425 hr
= CreateXmlReader(&IID_IXmlReader
, (void**)&reader
, NULL
);
426 ok(hr
== S_OK
, "Expected S_OK, got %08x\n", hr
);
428 test_read_state(reader
, XmlReadState_Closed
, -1, FALSE
);
430 nodetype
= XmlNodeType_Element
;
431 hr
= IXmlReader_GetNodeType(reader
, &nodetype
);
432 ok(hr
== S_FALSE
, "got %08x\n", hr
);
433 ok(nodetype
== XmlNodeType_None
, "got %d\n", nodetype
);
435 resolver
= (void*)0xdeadbeef;
436 hr
= IXmlReader_GetProperty(reader
, XmlReaderProperty_XmlResolver
, (LONG_PTR
*)&resolver
);
437 ok(hr
== S_OK
, "got 0x%08x\n", hr
);
438 ok(resolver
== NULL
, "got %p\n", resolver
);
440 hr
= IXmlReader_SetProperty(reader
, XmlReaderProperty_XmlResolver
, 0);
441 ok(hr
== S_OK
, "got 0x%08x\n", hr
);
443 hr
= IXmlReader_SetProperty(reader
, XmlReaderProperty_XmlResolver
, (LONG_PTR
)&testresolver
);
444 ok(hr
== S_OK
, "got 0x%08x\n", hr
);
447 hr
= IXmlReader_GetProperty(reader
, XmlReaderProperty_XmlResolver
, (LONG_PTR
*)&resolver
);
448 ok(hr
== S_OK
, "got 0x%08x\n", hr
);
449 ok(resolver
== &testresolver
, "got %p\n", resolver
);
450 IXmlResolver_Release(resolver
);
452 hr
= IXmlReader_SetProperty(reader
, XmlReaderProperty_XmlResolver
, 0);
453 ok(hr
== S_OK
, "got 0x%08x\n", hr
);
456 hr
= IXmlReader_GetProperty(reader
, XmlReaderProperty_DtdProcessing
, (LONG_PTR
*)&dtd
);
457 ok(hr
== S_OK
, "Expected S_OK, got %08x\n", hr
);
458 ok(dtd
== DtdProcessing_Prohibit
, "got %d\n", dtd
);
461 hr
= IXmlReader_SetProperty(reader
, XmlReaderProperty_DtdProcessing
, dtd
);
462 ok(hr
== E_INVALIDARG
, "Expected E_INVALIDARG, got %08x\n", hr
);
464 hr
= IXmlReader_SetProperty(reader
, XmlReaderProperty_DtdProcessing
, -1);
465 ok(hr
== E_INVALIDARG
, "Expected E_INVALIDARG, got %08x\n", hr
);
467 /* Null input pointer, releases previous input */
468 hr
= IXmlReader_SetInput(reader
, NULL
);
469 ok(hr
== S_OK
, "Expected S_OK, got %08x\n", hr
);
471 test_read_state(reader
, XmlReadState_Initial
, XmlReadState_Closed
, FALSE
);
473 /* test input interface selection sequence */
474 hr
= testinput_createinstance((void**)&input
);
475 ok(hr
== S_OK
, "Expected S_OK, got %08x\n", hr
);
479 input_iids
.count
= 0;
480 hr
= IXmlReader_SetInput(reader
, input
);
481 ok(hr
== E_NOINTERFACE
, "Expected E_NOINTERFACE, got %08x\n", hr
);
482 ok_iids(&input_iids
, setinput_full
, setinput_full_old
, FALSE
);
483 IUnknown_Release(input
);
485 IXmlReader_Release(reader
);
488 static void test_readerinput(void)
490 IXmlReaderInput
*reader_input
;
491 IXmlReader
*reader
, *reader2
;
492 IUnknown
*obj
, *input
;
493 IStream
*stream
, *stream2
;
494 XmlNodeType nodetype
;
498 hr
= CreateXmlReaderInputWithEncodingName(NULL
, NULL
, NULL
, FALSE
, NULL
, NULL
);
499 ok(hr
== E_INVALIDARG
, "Expected E_INVALIDARG, got %08x\n", hr
);
500 hr
= CreateXmlReaderInputWithEncodingName(NULL
, NULL
, NULL
, FALSE
, NULL
, &reader_input
);
501 ok(hr
== E_INVALIDARG
, "Expected E_INVALIDARG, got %08x\n", hr
);
503 hr
= CreateStreamOnHGlobal(NULL
, TRUE
, &stream
);
504 ok(hr
== S_OK
, "Expected S_OK, got %08x\n", hr
);
506 ref
= IStream_AddRef(stream
);
507 ok(ref
== 2, "Expected 2, got %d\n", ref
);
508 IStream_Release(stream
);
509 hr
= CreateXmlReaderInputWithEncodingName((IUnknown
*)stream
, NULL
, NULL
, FALSE
, NULL
, &reader_input
);
510 ok(hr
== S_OK
, "Expected S_OK, got %08x\n", hr
);
512 hr
= IUnknown_QueryInterface(reader_input
, &IID_IStream
, (void**)&stream2
);
513 ok(hr
== E_NOINTERFACE
, "Expected S_OK, got %08x\n", hr
);
515 hr
= IUnknown_QueryInterface(reader_input
, &IID_ISequentialStream
, (void**)&stream2
);
516 ok(hr
== E_NOINTERFACE
, "Expected S_OK, got %08x\n", hr
);
518 /* IXmlReaderInput grabs a stream reference */
519 ref
= IStream_AddRef(stream
);
520 ok(ref
== 3, "Expected 3, got %d\n", ref
);
521 IStream_Release(stream
);
523 /* try ::SetInput() with valid IXmlReaderInput */
524 hr
= CreateXmlReader(&IID_IXmlReader
, (void**)&reader
, NULL
);
525 ok(hr
== S_OK
, "Expected S_OK, got %08x\n", hr
);
527 ref
= IUnknown_AddRef(reader_input
);
528 ok(ref
== 2, "Expected 2, got %d\n", ref
);
529 IUnknown_Release(reader_input
);
531 hr
= IXmlReader_SetInput(reader
, reader_input
);
532 ok(hr
== S_OK
, "Expected S_OK, got %08x\n", hr
);
534 test_read_state(reader
, XmlReadState_Initial
, -1, FALSE
);
536 nodetype
= XmlNodeType_Element
;
537 hr
= IXmlReader_GetNodeType(reader
, &nodetype
);
538 ok(hr
== S_OK
, "got %08x\n", hr
);
539 ok(nodetype
== XmlNodeType_None
, "got %d\n", nodetype
);
541 /* IXmlReader grabs a IXmlReaderInput reference */
542 ref
= IUnknown_AddRef(reader_input
);
543 ok(ref
== 3, "Expected 3, got %d\n", ref
);
544 IUnknown_Release(reader_input
);
546 ref
= IStream_AddRef(stream
);
547 ok(ref
== 4, "Expected 4, got %d\n", ref
);
548 IStream_Release(stream
);
550 /* reset input and check state */
551 hr
= IXmlReader_SetInput(reader
, NULL
);
552 ok(hr
== S_OK
, "Expected S_OK, got %08x\n", hr
);
554 test_read_state(reader
, XmlReadState_Initial
, XmlReadState_Closed
, FALSE
);
556 IXmlReader_Release(reader
);
558 ref
= IStream_AddRef(stream
);
559 ok(ref
== 3, "Expected 3, got %d\n", ref
);
560 IStream_Release(stream
);
562 ref
= IUnknown_AddRef(reader_input
);
563 ok(ref
== 2, "Expected 2, got %d\n", ref
);
564 IUnknown_Release(reader_input
);
566 /* IID_IXmlReaderInput */
567 /* it returns a kind of private undocumented vtable incompatible with IUnknown,
568 so it's not a COM interface actually.
569 Such query will be used only to check if input is really IXmlReaderInput */
570 obj
= (IUnknown
*)0xdeadbeef;
571 hr
= IUnknown_QueryInterface(reader_input
, &IID_IXmlReaderInput
, (void**)&obj
);
572 ok(hr
== S_OK
, "Expected S_OK, got %08x\n", hr
);
573 ref
= IUnknown_AddRef(reader_input
);
574 ok(ref
== 3, "Expected 3, got %d\n", ref
);
575 IUnknown_Release(reader_input
);
577 IUnknown_Release(reader_input
);
578 IUnknown_Release(reader_input
);
579 IStream_Release(stream
);
581 /* test input interface selection sequence */
583 hr
= testinput_createinstance((void**)&input
);
584 ok(hr
== S_OK
, "Expected S_OK, got %08x\n", hr
);
586 input_iids
.count
= 0;
587 ref
= IUnknown_AddRef(input
);
588 ok(ref
== 2, "Expected 2, got %d\n", ref
);
589 IUnknown_Release(input
);
590 hr
= CreateXmlReaderInputWithEncodingName(input
, NULL
, NULL
, FALSE
, NULL
, &reader_input
);
591 ok(hr
== S_OK
, "Expected S_OK, got %08x\n", hr
);
592 ok_iids(&input_iids
, empty_seq
, NULL
, FALSE
);
593 /* IXmlReaderInput stores stream interface as IUnknown */
594 ref
= IUnknown_AddRef(input
);
595 ok(ref
== 3, "Expected 3, got %d\n", ref
);
596 IUnknown_Release(input
);
598 hr
= CreateXmlReader(&IID_IXmlReader
, (LPVOID
*)&reader
, NULL
);
599 ok(hr
== S_OK
, "Expected S_OK, got %08x\n", hr
);
601 input_iids
.count
= 0;
602 ref
= IUnknown_AddRef(reader_input
);
603 ok(ref
== 2, "Expected 2, got %d\n", ref
);
604 IUnknown_Release(reader_input
);
605 ref
= IUnknown_AddRef(input
);
606 ok(ref
== 3, "Expected 3, got %d\n", ref
);
607 IUnknown_Release(input
);
608 hr
= IXmlReader_SetInput(reader
, reader_input
);
609 ok(hr
== E_NOINTERFACE
, "Expected E_NOINTERFACE, got %08x\n", hr
);
610 ok_iids(&input_iids
, setinput_readerinput
, NULL
, FALSE
);
612 test_read_state(reader
, XmlReadState_Closed
, -1, FALSE
);
614 ref
= IUnknown_AddRef(input
);
615 ok(ref
== 3, "Expected 3, got %d\n", ref
);
616 IUnknown_Release(input
);
618 ref
= IUnknown_AddRef(reader_input
);
619 ok(ref
== 3 || broken(ref
== 2) /* versions 1.0.x and 1.1.x - XP, Vista */,
620 "Expected 3, got %d\n", ref
);
621 IUnknown_Release(reader_input
);
622 /* repeat another time, no check or caching here */
623 input_iids
.count
= 0;
624 hr
= IXmlReader_SetInput(reader
, reader_input
);
625 ok(hr
== E_NOINTERFACE
, "Expected E_NOINTERFACE, got %08x\n", hr
);
626 ok_iids(&input_iids
, setinput_readerinput
, NULL
, FALSE
);
629 hr
= CreateXmlReader(&IID_IXmlReader
, (LPVOID
*)&reader2
, NULL
);
630 ok(hr
== S_OK
, "Expected S_OK, got %08x\n", hr
);
632 /* resolving from IXmlReaderInput to IStream/ISequentialStream is done at
633 ::SetInput() level, each time it's called */
634 input_iids
.count
= 0;
635 hr
= IXmlReader_SetInput(reader2
, reader_input
);
636 ok(hr
== E_NOINTERFACE
, "Expected E_NOINTERFACE, got %08x\n", hr
);
637 ok_iids(&input_iids
, setinput_readerinput
, NULL
, FALSE
);
639 IXmlReader_Release(reader2
);
640 IXmlReader_Release(reader
);
642 IUnknown_Release(reader_input
);
643 IUnknown_Release(input
);
646 static void test_reader_state(void)
649 XmlNodeType nodetype
;
652 hr
= CreateXmlReader(&IID_IXmlReader
, (void**)&reader
, NULL
);
653 ok(hr
== S_OK
, "Expected S_OK, got %08x\n", hr
);
655 /* invalid arguments */
656 hr
= IXmlReader_GetProperty(reader
, XmlReaderProperty_ReadState
, NULL
);
657 ok(hr
== E_INVALIDARG
, "Expected E_INVALIDARG, got %08x\n", hr
);
659 /* attempt to read on closed reader */
660 test_read_state(reader
, XmlReadState_Closed
, -1, FALSE
);
663 /* newer versions crash here, probably cause no input was set */
664 hr
= IXmlReader_Read(reader
, &nodetype
);
665 ok(hr
== S_FALSE
, "got %08x\n", hr
);
667 IXmlReader_Release(reader
);
670 static void test_read_xmldeclaration(void)
672 static const WCHAR xmlW
[] = {'x','m','l',0};
673 static const WCHAR RegistrationInfoW
[] = {'R','e','g','i','s','t','r','a','t','i','o','n','I','n','f','o',0};
680 { {'v','e','r','s','i','o','n',0}, {'1','.','0',0} },
681 { {'e','n','c','o','d','i','n','g',0}, {'U','T','F','-','8',0} },
682 { {'s','t','a','n','d','a','l','o','n','e',0}, {'y','e','s',0} }
688 UINT count
= 0, len
, i
;
692 hr
= CreateXmlReader(&IID_IXmlReader
, (LPVOID
*)&reader
, NULL
);
693 ok(hr
== S_OK
, "Expected S_OK, got %08x\n", hr
);
695 /* position methods with Null args */
696 hr
= IXmlReader_GetLineNumber(reader
, NULL
);
697 ok(hr
== E_INVALIDARG
, "Expected E_INVALIDARG, got %08x\n", hr
);
699 hr
= IXmlReader_GetLinePosition(reader
, NULL
);
700 ok(hr
== E_INVALIDARG
, "Expected E_INVALIDARG, got %08x\n", hr
);
702 stream
= create_stream_on_data(xmldecl_full
, sizeof(xmldecl_full
));
704 hr
= IXmlReader_SetInput(reader
, (IUnknown
*)stream
);
705 ok(hr
== S_OK
, "Expected S_OK, got %08x\n", hr
);
707 hr
= IXmlReader_GetAttributeCount(reader
, &count
);
708 ok(hr
== S_OK
, "got %08x\n", hr
);
709 ok(count
== 0, "got %d\n", count
);
711 /* try to move without attributes */
712 hr
= IXmlReader_MoveToElement(reader
);
713 ok(hr
== S_FALSE
, "got %08x\n", hr
);
715 hr
= IXmlReader_MoveToNextAttribute(reader
);
716 ok(hr
== S_FALSE
, "got %08x\n", hr
);
718 hr
= IXmlReader_MoveToFirstAttribute(reader
);
719 ok(hr
== S_FALSE
, "got %08x\n", hr
);
721 ok_pos(reader
, 0, 0, -1, -1, FALSE
);
724 hr
= IXmlReader_Read(reader
, &type
);
725 ok(hr
== S_OK
, "Expected S_OK, got %08x\n", hr
);
726 ok(type
== XmlNodeType_XmlDeclaration
,
727 "Expected XmlNodeType_XmlDeclaration, got %s\n", type_to_str(type
));
728 /* new version 1.2.x and 1.3.x properly update position for <?xml ?> */
729 ok_pos(reader
, 1, 3, -1, 55, TRUE
);
730 test_read_state(reader
, XmlReadState_Interactive
, -1, FALSE
);
732 hr
= IXmlReader_GetValue(reader
, &val
, NULL
);
733 ok(hr
== S_OK
, "got %08x\n", hr
);
734 ok(*val
== 0, "got %s\n", wine_dbgstr_w(val
));
736 /* check attributes */
737 hr
= IXmlReader_MoveToNextAttribute(reader
);
738 ok(hr
== S_OK
, "got %08x\n", hr
);
740 type
= XmlNodeType_None
;
741 hr
= IXmlReader_GetNodeType(reader
, &type
);
742 ok(hr
== S_OK
, "got %08x\n", hr
);
743 ok(type
== XmlNodeType_Attribute
, "got %d\n", type
);
745 ok_pos(reader
, 1, 7, -1, 55, TRUE
);
747 /* try to move from last attribute */
748 hr
= IXmlReader_MoveToNextAttribute(reader
);
749 ok(hr
== S_OK
, "got %08x\n", hr
);
750 hr
= IXmlReader_MoveToNextAttribute(reader
);
751 ok(hr
== S_OK
, "got %08x\n", hr
);
752 hr
= IXmlReader_MoveToNextAttribute(reader
);
753 ok(hr
== S_FALSE
, "got %08x\n", hr
);
755 type
= XmlNodeType_None
;
756 hr
= IXmlReader_GetNodeType(reader
, &type
);
757 ok(hr
== S_OK
, "got %08x\n", hr
);
758 ok(type
== XmlNodeType_Attribute
, "got %d\n", type
);
760 hr
= IXmlReader_MoveToFirstAttribute(reader
);
761 ok(hr
== S_OK
, "got %08x\n", hr
);
762 ok_pos(reader
, 1, 7, -1, 55, TRUE
);
764 hr
= IXmlReader_GetAttributeCount(reader
, NULL
);
765 ok(hr
== E_INVALIDARG
, "got %08x\n", hr
);
767 hr
= IXmlReader_GetAttributeCount(reader
, &count
);
768 ok(hr
== S_OK
, "got %08x\n", hr
);
769 ok(count
== 3, "Expected 3, got %d\n", count
);
771 for (i
= 0; i
< count
; i
++)
774 hr
= IXmlReader_GetLocalName(reader
, &val
, &len
);
775 ok(hr
== S_OK
, "got %08x\n", hr
);
776 ok(len
== lstrlenW(name_val
[i
].name
), "expected %u, got %u\n", lstrlenW(name_val
[i
].name
), len
);
777 ok(!lstrcmpW(name_val
[i
].name
, val
), "expected %s, got %s\n", wine_dbgstr_w(name_val
[i
].name
), wine_dbgstr_w(val
));
780 hr
= IXmlReader_GetValue(reader
, &val
, &len
);
781 ok(hr
== S_OK
, "got %08x\n", hr
);
782 ok(len
== lstrlenW(name_val
[i
].val
), "expected %u, got %u\n", lstrlenW(name_val
[i
].val
), len
);
783 ok(!lstrcmpW(name_val
[i
].val
, val
), "expected %s, got %s\n", wine_dbgstr_w(name_val
[i
].val
), wine_dbgstr_w(val
));
785 hr
= IXmlReader_MoveToNextAttribute(reader
);
786 ok(hr
== ((i
< count
- 1) ? S_OK
: S_FALSE
), "got %08x\n", hr
);
789 hr
= IXmlReader_GetDepth(reader
, &count
);
790 ok(hr
== S_OK
, "Expected S_OK, got %08x\n", hr
);
791 ok(count
== 1, "Expected 1, got %d\n", count
);
793 hr
= IXmlReader_MoveToElement(reader
);
794 ok(hr
== S_OK
, "got %08x\n", hr
);
796 type
= XmlNodeType_None
;
797 hr
= IXmlReader_GetNodeType(reader
, &type
);
798 ok(hr
== S_OK
, "got %08x\n", hr
);
799 ok(type
== XmlNodeType_XmlDeclaration
, "got %d\n", type
);
801 type
= XmlNodeType_XmlDeclaration
;
802 hr
= IXmlReader_Read(reader
, &type
);
803 /* newer versions return syntax error here cause document is incomplete,
804 it makes more sense than invalid char error */
806 ok(hr
== WC_E_SYNTAX
|| broken(hr
== WC_E_XMLCHARACTER
), "got 0x%08x\n", hr
);
807 ok(type
== XmlNodeType_None
, "got %d\n", type
);
809 IStream_Release(stream
);
811 /* test short variant */
812 stream
= create_stream_on_data(xmldecl_short
, sizeof(xmldecl_short
));
814 hr
= IXmlReader_SetInput(reader
, (IUnknown
*)stream
);
815 ok(hr
== S_OK
, "expected S_OK, got %08x\n", hr
);
818 hr
= IXmlReader_Read(reader
, &type
);
819 ok(hr
== S_OK
, "expected S_OK, got %08x\n", hr
);
820 ok(type
== XmlNodeType_XmlDeclaration
, "expected XmlDeclaration, got %s\n", type_to_str(type
));
821 ok_pos(reader
, 1, 3, 1, 21, TRUE
);
822 test_read_state(reader
, XmlReadState_Interactive
, -1, TRUE
);
824 hr
= IXmlReader_GetAttributeCount(reader
, &count
);
825 ok(hr
== S_OK
, "expected S_OK, got %08x\n", hr
);
826 ok(count
== 1, "expected 1, got %d\n", count
);
828 ret
= IXmlReader_IsEmptyElement(reader
);
829 ok(!ret
, "element should not be empty\n");
831 hr
= IXmlReader_GetValue(reader
, &val
, NULL
);
832 ok(hr
== S_OK
, "expected S_OK, got %08x\n", hr
);
833 ok(*val
== 0, "got %s\n", wine_dbgstr_w(val
));
835 hr
= IXmlReader_GetLocalName(reader
, &val
, NULL
);
836 ok(hr
== S_OK
, "expected S_OK, got %08x\n", hr
);
838 ok(!lstrcmpW(val
, xmlW
), "got %s\n", wine_dbgstr_w(val
));
840 /* check attributes */
841 hr
= IXmlReader_MoveToNextAttribute(reader
);
842 ok(hr
== S_OK
, "expected S_OK, got %08x\n", hr
);
845 hr
= IXmlReader_GetNodeType(reader
, &type
);
846 ok(hr
== S_OK
, "expected S_OK, got %08x\n", hr
);
847 ok(type
== XmlNodeType_Attribute
, "got %d\n", type
);
848 ok_pos(reader
, 1, 7, 1, 21, TRUE
);
850 /* try to move from last attribute */
851 hr
= IXmlReader_MoveToNextAttribute(reader
);
852 ok(hr
== S_FALSE
, "expected S_FALSE, got %08x\n", hr
);
855 hr
= IXmlReader_Read(reader
, &type
);
856 ok(hr
== S_OK
, "expected S_OK, got %08x\n", hr
);
857 ok(type
== XmlNodeType_Element
, "expected Element, got %s\n", type_to_str(type
));
858 ok_pos(reader
, 1, 23, 1, 40, TRUE
);
859 test_read_state(reader
, XmlReadState_Interactive
, -1, TRUE
);
861 hr
= IXmlReader_GetAttributeCount(reader
, &count
);
862 ok(hr
== S_OK
, "expected S_OK, got %08x\n", hr
);
863 ok(count
== 0, "expected 0, got %d\n", count
);
865 ret
= IXmlReader_IsEmptyElement(reader
);
866 ok(ret
, "element should be empty\n");
868 hr
= IXmlReader_GetValue(reader
, &val
, NULL
);
869 ok(hr
== S_OK
, "expected S_OK, got %08x\n", hr
);
871 ok(*val
== 0, "got %s\n", wine_dbgstr_w(val
));
873 hr
= IXmlReader_GetLocalName(reader
, &val
, NULL
);
874 ok(hr
== S_OK
, "expected S_OK, got %08x\n", hr
);
875 ok(!lstrcmpW(val
, RegistrationInfoW
), "got %s\n", wine_dbgstr_w(val
));
878 hr
= IXmlReader_Read(reader
, &type
);
880 ok(hr
== WC_E_SYNTAX
|| hr
== WC_E_XMLCHARACTER
/* XP */, "expected WC_E_SYNTAX, got %08x\n", hr
);
882 ok(type
== XmlNodeType_None
, "expected None, got %s\n", type_to_str(type
));
883 ok_pos(reader
, 1, 41, -1, -1, TRUE
);
884 test_read_state(reader
, XmlReadState_Error
, -1, TRUE
);
886 IStream_Release(stream
);
887 IXmlReader_Release(reader
);
895 HRESULT hr_broken
; /* this is set to older version results */
899 static struct test_entry comment_tests
[] = {
900 { "<!-- comment -->", "", " comment ", S_OK
},
901 { "<!-- - comment-->", "", " - comment", S_OK
},
902 { "<!-- -- comment-->", NULL
, NULL
, WC_E_COMMENT
, WC_E_GREATERTHAN
},
903 { "<!-- -- comment--->", NULL
, NULL
, WC_E_COMMENT
, WC_E_GREATERTHAN
},
907 static void test_read_comment(void)
909 struct test_entry
*test
= comment_tests
;
913 hr
= CreateXmlReader(&IID_IXmlReader
, (void**)&reader
, NULL
);
914 ok(hr
== S_OK
, "S_OK, got %08x\n", hr
);
921 stream
= create_stream_on_data(test
->xml
, strlen(test
->xml
)+1);
922 hr
= IXmlReader_SetInput(reader
, (IUnknown
*)stream
);
923 ok(hr
== S_OK
, "got %08x\n", hr
);
925 type
= XmlNodeType_None
;
926 hr
= IXmlReader_Read(reader
, &type
);
928 ok(hr
== test
->hr
|| broken(hr
== test
->hr_broken
), "got %08x for %s\n", hr
, test
->xml
);
930 ok(hr
== test
->hr
, "got %08x for %s\n", hr
, test
->xml
);
937 ok(type
== XmlNodeType_Comment
, "got %d for %s\n", type
, test
->xml
);
941 hr
= IXmlReader_GetLocalName(reader
, &str
, &len
);
942 ok(hr
== S_OK
, "got 0x%08x\n", hr
);
943 ok(len
== strlen(test
->name
), "got %u\n", len
);
944 str_exp
= a2w(test
->name
);
945 ok(!lstrcmpW(str
, str_exp
), "got %s\n", wine_dbgstr_w(str
));
950 hr
= IXmlReader_GetQualifiedName(reader
, &str
, &len
);
951 ok(hr
== S_OK
, "got 0x%08x\n", hr
);
952 ok(len
== strlen(test
->name
), "got %u\n", len
);
953 str_exp
= a2w(test
->name
);
954 ok(!lstrcmpW(str
, str_exp
), "got %s\n", wine_dbgstr_w(str
));
960 hr
= IXmlReader_GetValue(reader
, &str
, &len
);
961 ok(hr
== S_OK
, "got 0x%08x\n", hr
);
962 ok(len
== strlen(test
->value
), "got %u\n", len
);
963 str_exp
= a2w(test
->value
);
964 ok(!lstrcmpW(str
, str_exp
), "got %s\n", wine_dbgstr_w(str
));
968 IStream_Release(stream
);
972 IXmlReader_Release(reader
);
975 static struct test_entry pi_tests
[] = {
976 { "<?pi?>", "pi", "", S_OK
},
977 { "<?pi ?>", "pi", "", S_OK
},
978 { "<?pi ?>", "pi", "", S_OK
},
979 { "<?pi pi data?>", "pi", "pi data", S_OK
},
980 { "<?pi pi data ?>", "pi", "pi data ", S_OK
},
981 { "<?pi data ?>", "pi", "data ", S_OK
},
982 { "<?pi:pi?>", NULL
, NULL
, NC_E_NAMECOLON
, WC_E_NAMECHARACTER
},
983 { "<?:pi ?>", NULL
, NULL
, WC_E_PI
, WC_E_NAMECHARACTER
},
984 { "<?-pi ?>", NULL
, NULL
, WC_E_PI
, WC_E_NAMECHARACTER
},
985 { "<?xml-stylesheet ?>", "xml-stylesheet", "", S_OK
},
989 static void test_read_pi(void)
991 struct test_entry
*test
= pi_tests
;
995 hr
= CreateXmlReader(&IID_IXmlReader
, (void**)&reader
, NULL
);
996 ok(hr
== S_OK
, "S_OK, got %08x\n", hr
);
1003 stream
= create_stream_on_data(test
->xml
, strlen(test
->xml
)+1);
1004 hr
= IXmlReader_SetInput(reader
, (IUnknown
*)stream
);
1005 ok(hr
== S_OK
, "got %08x\n", hr
);
1007 type
= XmlNodeType_None
;
1008 hr
= IXmlReader_Read(reader
, &type
);
1009 if (test
->hr_broken
)
1010 ok(hr
== test
->hr
|| broken(hr
== test
->hr_broken
), "got %08x for %s\n", hr
, test
->xml
);
1012 ok(hr
== test
->hr
, "got %08x for %s\n", hr
, test
->xml
);
1019 ok(type
== XmlNodeType_ProcessingInstruction
, "got %d for %s\n", type
, test
->xml
);
1023 hr
= IXmlReader_GetLocalName(reader
, &str
, &len
);
1024 ok(hr
== S_OK
, "got 0x%08x\n", hr
);
1025 ok(len
== strlen(test
->name
), "got %u\n", len
);
1026 str_exp
= a2w(test
->name
);
1027 ok(!lstrcmpW(str
, str_exp
), "got %s\n", wine_dbgstr_w(str
));
1032 hr
= IXmlReader_GetQualifiedName(reader
, &str
, &len
);
1033 ok(hr
== S_OK
, "got 0x%08x\n", hr
);
1034 ok(len
== strlen(test
->name
), "got %u\n", len
);
1035 str_exp
= a2w(test
->name
);
1036 ok(!lstrcmpW(str
, str_exp
), "got %s\n", wine_dbgstr_w(str
));
1040 len
= !strlen(test
->value
);
1042 hr
= IXmlReader_GetValue(reader
, &str
, &len
);
1043 ok(hr
== S_OK
, "got 0x%08x\n", hr
);
1044 ok(len
== strlen(test
->value
), "got %u\n", len
);
1045 str_exp
= a2w(test
->value
);
1046 ok(!lstrcmpW(str
, str_exp
), "got %s\n", wine_dbgstr_w(str
));
1050 IStream_Release(stream
);
1054 IXmlReader_Release(reader
);
1059 XmlNodeType types
[20];
1062 static const char misc_test_xml
[] =
1080 static struct nodes_test misc_test
= {
1083 XmlNodeType_Comment
,
1084 XmlNodeType_Comment
,
1085 XmlNodeType_ProcessingInstruction
,
1086 XmlNodeType_Comment
,
1087 XmlNodeType_Whitespace
,
1088 XmlNodeType_Comment
,
1089 XmlNodeType_Element
,
1090 XmlNodeType_Whitespace
,
1091 XmlNodeType_Element
,
1093 XmlNodeType_Comment
,
1095 XmlNodeType_ProcessingInstruction
,
1096 XmlNodeType_Whitespace
,
1097 XmlNodeType_EndElement
,
1102 static void test_read_full(void)
1104 struct nodes_test
*test
= &misc_test
;
1111 hr
= CreateXmlReader(&IID_IXmlReader
, (void**)&reader
, NULL
);
1112 ok(hr
== S_OK
, "S_OK, got %08x\n", hr
);
1114 stream
= create_stream_on_data(test
->xml
, strlen(test
->xml
)+1);
1115 hr
= IXmlReader_SetInput(reader
, (IUnknown
*)stream
);
1116 ok(hr
== S_OK
, "got %08x\n", hr
);
1119 type
= XmlNodeType_None
;
1120 hr
= IXmlReader_Read(reader
, &type
);
1123 ok(test
->types
[i
] != XmlNodeType_None
, "%d: unexpected end of test data\n", i
);
1124 if (test
->types
[i
] == XmlNodeType_None
) break;
1125 ok(type
== test
->types
[i
], "%d: got wrong type %d, expected %d\n", i
, type
, test
->types
[i
]);
1126 if (type
== XmlNodeType_Whitespace
)
1131 hr
= IXmlReader_GetValue(reader
, &ptr
, &len
);
1132 ok(hr
== S_OK
, "%d: GetValue failed 0x%08x\n", i
, hr
);
1133 ok(len
> 0, "%d: wrong value length %d\n", i
, len
);
1135 hr
= IXmlReader_Read(reader
, &type
);
1138 ok(test
->types
[i
] == XmlNodeType_None
, "incomplete sequence, got %d\n", test
->types
[i
]);
1140 IStream_Release(stream
);
1141 IXmlReader_Release(reader
);
1144 static const char test_dtd
[] =
1145 "<!DOCTYPE testdtd SYSTEM \"externalid uri\" >"
1148 static void test_read_dtd(void)
1150 static const WCHAR sysvalW
[] = {'e','x','t','e','r','n','a','l','i','d',' ','u','r','i',0};
1151 static const WCHAR dtdnameW
[] = {'t','e','s','t','d','t','d',0};
1152 static const WCHAR sysW
[] = {'S','Y','S','T','E','M',0};
1160 hr
= CreateXmlReader(&IID_IXmlReader
, (void**)&reader
, NULL
);
1161 ok(hr
== S_OK
, "S_OK, got %08x\n", hr
);
1163 hr
= IXmlReader_SetProperty(reader
, XmlReaderProperty_DtdProcessing
, DtdProcessing_Parse
);
1164 ok(hr
== S_OK
, "got 0x%8x\n", hr
);
1166 stream
= create_stream_on_data(test_dtd
, sizeof(test_dtd
));
1167 hr
= IXmlReader_SetInput(reader
, (IUnknown
*)stream
);
1168 ok(hr
== S_OK
, "got %08x\n", hr
);
1170 type
= XmlNodeType_None
;
1171 hr
= IXmlReader_Read(reader
, &type
);
1172 ok(hr
== S_OK
, "got 0x%8x\n", hr
);
1173 ok(type
== XmlNodeType_DocumentType
, "got type %d\n", type
);
1176 hr
= IXmlReader_GetAttributeCount(reader
, &count
);
1177 ok(hr
== S_OK
, "got %08x\n", hr
);
1178 ok(count
== 1, "got %d\n", count
);
1180 hr
= IXmlReader_MoveToFirstAttribute(reader
);
1181 ok(hr
== S_OK
, "got %08x\n", hr
);
1183 type
= XmlNodeType_None
;
1184 hr
= IXmlReader_GetNodeType(reader
, &type
);
1185 ok(hr
== S_OK
, "got %08x\n", hr
);
1186 ok(type
== XmlNodeType_Attribute
, "got %d\n", type
);
1190 hr
= IXmlReader_GetLocalName(reader
, &str
, &len
);
1191 ok(hr
== S_OK
, "got 0x%08x\n", hr
);
1192 ok(len
== lstrlenW(sysW
), "got %u\n", len
);
1193 ok(!lstrcmpW(str
, sysW
), "got %s\n", wine_dbgstr_w(str
));
1197 hr
= IXmlReader_GetValue(reader
, &str
, &len
);
1198 ok(hr
== S_OK
, "got 0x%08x\n", hr
);
1199 ok(len
== lstrlenW(sysvalW
), "got %u\n", len
);
1200 ok(!lstrcmpW(str
, sysvalW
), "got %s\n", wine_dbgstr_w(str
));
1202 hr
= IXmlReader_MoveToElement(reader
);
1203 ok(hr
== S_OK
, "got 0x%08x\n", hr
);
1207 hr
= IXmlReader_GetLocalName(reader
, &str
, &len
);
1208 ok(hr
== S_OK
, "got 0x%08x\n", hr
);
1210 ok(len
== lstrlenW(dtdnameW
), "got %u\n", len
);
1211 ok(!lstrcmpW(str
, dtdnameW
), "got %s\n", wine_dbgstr_w(str
));
1215 hr
= IXmlReader_GetQualifiedName(reader
, &str
, &len
);
1216 ok(hr
== S_OK
, "got 0x%08x\n", hr
);
1217 ok(len
== lstrlenW(dtdnameW
), "got %u\n", len
);
1218 ok(!lstrcmpW(str
, dtdnameW
), "got %s\n", wine_dbgstr_w(str
));
1220 type
= XmlNodeType_None
;
1221 hr
= IXmlReader_Read(reader
, &type
);
1222 ok(hr
== S_OK
, "got 0x%8x\n", hr
);
1223 ok(type
== XmlNodeType_Comment
, "got type %d\n", type
);
1225 IStream_Release(stream
);
1226 IXmlReader_Release(reader
);
1229 static struct test_entry element_tests
[] = {
1230 { "<a/>", "a", "", S_OK
},
1231 { "<a />", "a", "", S_OK
},
1232 { "<a:b/>", "a:b", "", NC_E_UNDECLAREDPREFIX
},
1233 { "<:a/>", NULL
, NULL
, NC_E_QNAMECHARACTER
},
1234 { "< a/>", NULL
, NULL
, NC_E_QNAMECHARACTER
},
1235 { "<a>", "a", "", S_OK
},
1236 { "<a >", "a", "", S_OK
},
1237 { "<a \r \t\n>", "a", "", S_OK
},
1238 { "</a>", NULL
, NULL
, NC_E_QNAMECHARACTER
},
1242 static void test_read_element(void)
1244 struct test_entry
*test
= element_tests
;
1245 static const char stag
[] = "<a><b></b></a>";
1246 static const char mismatch
[] = "<a></b>";
1253 hr
= CreateXmlReader(&IID_IXmlReader
, (void**)&reader
, NULL
);
1254 ok(hr
== S_OK
, "S_OK, got %08x\n", hr
);
1258 stream
= create_stream_on_data(test
->xml
, strlen(test
->xml
)+1);
1259 hr
= IXmlReader_SetInput(reader
, (IUnknown
*)stream
);
1260 ok(hr
== S_OK
, "got %08x\n", hr
);
1262 type
= XmlNodeType_None
;
1263 hr
= IXmlReader_Read(reader
, &type
);
1264 if (test
->hr_broken
)
1265 ok(hr
== test
->hr
|| broken(hr
== test
->hr_broken
), "got %08x for %s\n", hr
, test
->xml
);
1267 ok(hr
== test
->hr
, "got %08x for %s\n", hr
, test
->xml
);
1274 ok(type
== XmlNodeType_Element
, "got %d for %s\n", type
, test
->xml
);
1278 hr
= IXmlReader_GetQualifiedName(reader
, &str
, &len
);
1279 ok(hr
== S_OK
, "got 0x%08x\n", hr
);
1280 ok(len
== strlen(test
->name
), "got %u\n", len
);
1281 str_exp
= a2w(test
->name
);
1282 ok(!lstrcmpW(str
, str_exp
), "got %s\n", wine_dbgstr_w(str
));
1288 hr
= IXmlReader_GetValue(reader
, &str
, &len
);
1289 ok(hr
== S_OK
, "got 0x%08x\n", hr
);
1290 ok(len
== 0, "got %u\n", len
);
1291 ok(*str
== 0, "got %s\n", wine_dbgstr_w(str
));
1294 IStream_Release(stream
);
1298 /* test reader depth increment */
1299 stream
= create_stream_on_data(stag
, sizeof(stag
));
1300 hr
= IXmlReader_SetInput(reader
, (IUnknown
*)stream
);
1301 ok(hr
== S_OK
, "got %08x\n", hr
);
1304 hr
= IXmlReader_GetDepth(reader
, &depth
);
1305 ok(hr
== S_OK
, "got %08x\n", hr
);
1306 ok(depth
== 0, "got %d\n", depth
);
1308 type
= XmlNodeType_None
;
1309 hr
= IXmlReader_Read(reader
, &type
);
1310 ok(hr
== S_OK
, "got %08x\n", hr
);
1311 ok(type
== XmlNodeType_Element
, "got %d\n", type
);
1314 hr
= IXmlReader_GetDepth(reader
, &depth
);
1315 ok(hr
== S_OK
, "got %08x\n", hr
);
1316 ok(depth
== 0, "got %d\n", depth
);
1318 type
= XmlNodeType_None
;
1319 hr
= IXmlReader_Read(reader
, &type
);
1320 ok(hr
== S_OK
, "got %08x\n", hr
);
1321 ok(type
== XmlNodeType_Element
, "got %d\n", type
);
1324 hr
= IXmlReader_GetDepth(reader
, &depth
);
1325 ok(hr
== S_OK
, "got %08x\n", hr
);
1326 ok(depth
== 1, "got %d\n", depth
);
1328 /* read end tag for inner element */
1329 type
= XmlNodeType_None
;
1330 hr
= IXmlReader_Read(reader
, &type
);
1331 ok(hr
== S_OK
, "got %08x\n", hr
);
1332 ok(type
== XmlNodeType_EndElement
, "got %d\n", type
);
1335 hr
= IXmlReader_GetDepth(reader
, &depth
);
1336 ok(hr
== S_OK
, "got %08x\n", hr
);
1338 ok(depth
== 2, "got %d\n", depth
);
1340 /* read end tag for container element */
1341 type
= XmlNodeType_None
;
1342 hr
= IXmlReader_Read(reader
, &type
);
1343 ok(hr
== S_OK
, "got %08x\n", hr
);
1344 ok(type
== XmlNodeType_EndElement
, "got %d\n", type
);
1347 hr
= IXmlReader_GetDepth(reader
, &depth
);
1348 ok(hr
== S_OK
, "got %08x\n", hr
);
1349 ok(depth
== 1, "got %d\n", depth
);
1351 IStream_Release(stream
);
1353 /* start/end tag mismatch */
1354 stream
= create_stream_on_data(mismatch
, sizeof(mismatch
));
1355 hr
= IXmlReader_SetInput(reader
, (IUnknown
*)stream
);
1356 ok(hr
== S_OK
, "got %08x\n", hr
);
1358 type
= XmlNodeType_None
;
1359 hr
= IXmlReader_Read(reader
, &type
);
1360 ok(hr
== S_OK
, "got %08x\n", hr
);
1361 ok(type
== XmlNodeType_Element
, "got %d\n", type
);
1363 type
= XmlNodeType_Element
;
1364 hr
= IXmlReader_Read(reader
, &type
);
1365 ok(hr
== WC_E_ELEMENTMATCH
, "got %08x\n", hr
);
1367 ok(type
== XmlNodeType_None
, "got %d\n", type
);
1369 IStream_Release(stream
);
1371 IXmlReader_Release(reader
);
1374 static ISequentialStream teststream
= { &teststreamvtbl
};
1376 static void test_read_pending(void)
1384 hr
= CreateXmlReader(&IID_IXmlReader
, (void**)&reader
, NULL
);
1385 ok(hr
== S_OK
, "S_OK, got 0x%08x\n", hr
);
1387 hr
= IXmlReader_SetInput(reader
, (IUnknown
*)&teststream
);
1388 ok(hr
== S_OK
, "got 0x%08x\n", hr
);
1390 /* first read call returns incomplete node, second attempt fails with E_PENDING */
1391 stream_readcall
= 0;
1392 type
= XmlNodeType_Element
;
1393 hr
= IXmlReader_Read(reader
, &type
);
1394 ok(hr
== S_OK
|| broken(hr
== E_PENDING
), "got 0x%08x\n", hr
);
1395 /* newer versions are happy when it's enough data to detect node type,
1396 older versions keep reading until it fails to read more */
1397 ok(stream_readcall
== 1 || broken(stream_readcall
> 1), "got %d\n", stream_readcall
);
1398 ok(type
== XmlNodeType_Comment
|| broken(type
== XmlNodeType_None
), "got %d\n", type
);
1400 /* newer versions' GetValue() makes an attempt to read more */
1401 c
= stream_readcall
;
1402 value
= (void*)0xdeadbeef;
1403 hr
= IXmlReader_GetValue(reader
, &value
, NULL
);
1404 ok(hr
== E_PENDING
, "got 0x%08x\n", hr
);
1405 ok(value
== NULL
|| broken(value
== (void*)0xdeadbeef) /* Win8 sets it to NULL */, "got %p\n", value
);
1406 ok(c
< stream_readcall
|| broken(c
== stream_readcall
), "got %d, expected %d\n", stream_readcall
, c
+1);
1408 IXmlReader_Release(reader
);
1411 static void test_readvaluechunk(void)
1413 static const char testA
[] = "<!-- comment1 -->";
1422 hr
= CreateXmlReader(&IID_IXmlReader
, (void**)&reader
, NULL
);
1423 ok(hr
== S_OK
, "S_OK, got %08x\n", hr
);
1425 stream
= create_stream_on_data(testA
, sizeof(testA
));
1426 hr
= IXmlReader_SetInput(reader
, (IUnknown
*)stream
);
1427 ok(hr
== S_OK
, "got %08x\n", hr
);
1429 hr
= IXmlReader_Read(reader
, &type
);
1430 ok(hr
== S_OK
, "got %08x\n", hr
);
1434 hr
= IXmlReader_ReadValueChunk(reader
, &b
, 1, &c
);
1435 ok(hr
== S_OK
, "got %08x\n", hr
);
1436 ok(c
== 1, "got %u\n", c
);
1437 ok(b
== ' ', "got %x\n", b
);
1439 /* portion read as chunk is skipped from resulting node value */
1441 hr
= IXmlReader_GetValue(reader
, &value
, NULL
);
1442 ok(hr
== S_OK
, "got %08x\n", hr
);
1443 ok(value
[0] == 'c', "got %s\n", wine_dbgstr_w(value
));
1445 /* once value is returned/allocated it's not possible to read by chunk */
1448 hr
= IXmlReader_ReadValueChunk(reader
, &b
, 1, &c
);
1449 ok(hr
== S_FALSE
, "got %08x\n", hr
);
1450 ok(c
== 0, "got %u\n", c
);
1451 ok(b
== 0, "got %x\n", b
);
1454 hr
= IXmlReader_GetValue(reader
, &value
, NULL
);
1455 ok(hr
== S_OK
, "got %08x\n", hr
);
1456 ok(value
[0] == 'c', "got %s\n", wine_dbgstr_w(value
));
1458 IXmlReader_Release(reader
);
1459 IStream_Release(stream
);
1462 static struct test_entry cdata_tests
[] = {
1463 { "<a><![CDATA[ ]]data ]]></a>", "", " ]]data ", S_OK
},
1464 { "<a><![CDATA[<![CDATA[ data ]]]]></a>", "", "<![CDATA[ data ]]", S_OK
},
1465 { "<a><![CDATA[\n \r\n \n\n ]]></a>", "", "\n \n \n\n ", S_OK
, S_OK
, TRUE
},
1466 { "<a><![CDATA[\r \r\r\n \n\n ]]></a>", "", "\n \n\n \n\n ", S_OK
, S_OK
, TRUE
},
1467 { "<a><![CDATA[\r\r \n\r \r \n\n ]]></a>", "", "\n\n \n\n \n \n\n ", S_OK
},
1471 static void test_read_cdata(void)
1473 struct test_entry
*test
= cdata_tests
;
1477 hr
= CreateXmlReader(&IID_IXmlReader
, (void**)&reader
, NULL
);
1478 ok(hr
== S_OK
, "S_OK, got %08x\n", hr
);
1485 stream
= create_stream_on_data(test
->xml
, strlen(test
->xml
)+1);
1486 hr
= IXmlReader_SetInput(reader
, (IUnknown
*)stream
);
1487 ok(hr
== S_OK
, "got %08x\n", hr
);
1489 type
= XmlNodeType_None
;
1490 hr
= IXmlReader_Read(reader
, &type
);
1492 /* read one more to get to CDATA */
1493 if (type
== XmlNodeType_Element
)
1495 type
= XmlNodeType_None
;
1496 hr
= IXmlReader_Read(reader
, &type
);
1499 if (test
->hr_broken
)
1500 ok(hr
== test
->hr
|| broken(hr
== test
->hr_broken
), "got %08x for %s\n", hr
, test
->xml
);
1502 ok(hr
== test
->hr
, "got %08x for %s\n", hr
, test
->xml
);
1509 ok(type
== XmlNodeType_CDATA
, "got %d for %s\n", type
, test
->xml
);
1511 str_exp
= a2w(test
->name
);
1515 hr
= IXmlReader_GetLocalName(reader
, &str
, &len
);
1516 ok(hr
== S_OK
, "got 0x%08x\n", hr
);
1517 ok(len
== strlen(test
->name
), "got %u\n", len
);
1518 ok(!lstrcmpW(str
, str_exp
), "got %s\n", wine_dbgstr_w(str
));
1521 hr
= IXmlReader_GetLocalName(reader
, &str
, NULL
);
1522 ok(hr
== S_OK
, "got 0x%08x\n", hr
);
1523 ok(!lstrcmpW(str
, str_exp
), "got %s\n", wine_dbgstr_w(str
));
1529 hr
= IXmlReader_GetQualifiedName(reader
, &str
, &len
);
1530 ok(hr
== S_OK
, "got 0x%08x\n", hr
);
1531 ok(len
== strlen(test
->name
), "got %u\n", len
);
1532 str_exp
= a2w(test
->name
);
1533 ok(!lstrcmpW(str
, str_exp
), "got %s\n", wine_dbgstr_w(str
));
1539 hr
= IXmlReader_GetValue(reader
, &str
, &len
);
1540 ok(hr
== S_OK
, "got 0x%08x\n", hr
);
1542 str_exp
= a2w(test
->value
);
1543 todo_wine_if (test
->todo
)
1545 ok(len
== strlen(test
->value
), "got %u\n", len
);
1546 ok(!lstrcmpW(str
, str_exp
), "got %s\n", wine_dbgstr_w(str
));
1551 IStream_Release(stream
);
1555 IXmlReader_Release(reader
);
1558 static struct test_entry text_tests
[] = {
1559 { "<a>simple text</a>", "", "simple text", S_OK
},
1560 { "<a>text ]]> text</a>", "", "", WC_E_CDSECTEND
},
1564 static void test_read_text(void)
1566 struct test_entry
*test
= text_tests
;
1570 hr
= CreateXmlReader(&IID_IXmlReader
, (void**)&reader
, NULL
);
1571 ok(hr
== S_OK
, "S_OK, got %08x\n", hr
);
1578 stream
= create_stream_on_data(test
->xml
, strlen(test
->xml
)+1);
1579 hr
= IXmlReader_SetInput(reader
, (IUnknown
*)stream
);
1580 ok(hr
== S_OK
, "got %08x\n", hr
);
1582 type
= XmlNodeType_None
;
1583 hr
= IXmlReader_Read(reader
, &type
);
1585 /* read one more to get to CDATA */
1586 if (type
== XmlNodeType_Element
)
1588 type
= XmlNodeType_None
;
1589 hr
= IXmlReader_Read(reader
, &type
);
1592 if (test
->hr_broken
)
1593 ok(hr
== test
->hr
|| broken(hr
== test
->hr_broken
), "got %08x for %s\n", hr
, test
->xml
);
1595 ok(hr
== test
->hr
, "got %08x for %s\n", hr
, test
->xml
);
1602 ok(type
== XmlNodeType_Text
, "got %d for %s\n", type
, test
->xml
);
1604 str_exp
= a2w(test
->name
);
1608 hr
= IXmlReader_GetLocalName(reader
, &str
, &len
);
1609 ok(hr
== S_OK
, "got 0x%08x\n", hr
);
1610 ok(len
== strlen(test
->name
), "got %u\n", len
);
1611 ok(!lstrcmpW(str
, str_exp
), "got %s\n", wine_dbgstr_w(str
));
1614 hr
= IXmlReader_GetLocalName(reader
, &str
, NULL
);
1615 ok(hr
== S_OK
, "got 0x%08x\n", hr
);
1616 ok(!lstrcmpW(str
, str_exp
), "got %s\n", wine_dbgstr_w(str
));
1622 hr
= IXmlReader_GetQualifiedName(reader
, &str
, &len
);
1623 ok(hr
== S_OK
, "got 0x%08x\n", hr
);
1624 ok(len
== strlen(test
->name
), "got %u\n", len
);
1625 str_exp
= a2w(test
->name
);
1626 ok(!lstrcmpW(str
, str_exp
), "got %s\n", wine_dbgstr_w(str
));
1632 hr
= IXmlReader_GetValue(reader
, &str
, &len
);
1633 ok(hr
== S_OK
, "got 0x%08x\n", hr
);
1635 str_exp
= a2w(test
->value
);
1636 todo_wine_if (test
->todo
)
1638 ok(len
== strlen(test
->value
), "got %u\n", len
);
1639 ok(!lstrcmpW(str
, str_exp
), "got %s\n", wine_dbgstr_w(str
));
1644 IStream_Release(stream
);
1648 IXmlReader_Release(reader
);
1651 struct test_entry_empty
{
1656 static struct test_entry_empty empty_element_tests
[] = {
1657 { "<a></a>", FALSE
},
1662 static void test_isemptyelement(void)
1664 struct test_entry_empty
*test
= empty_element_tests
;
1668 hr
= CreateXmlReader(&IID_IXmlReader
, (void**)&reader
, NULL
);
1669 ok(hr
== S_OK
, "S_OK, got %08x\n", hr
);
1677 stream
= create_stream_on_data(test
->xml
, strlen(test
->xml
)+1);
1678 hr
= IXmlReader_SetInput(reader
, (IUnknown
*)stream
);
1679 ok(hr
== S_OK
, "got %08x\n", hr
);
1681 type
= XmlNodeType_None
;
1682 hr
= IXmlReader_Read(reader
, &type
);
1683 ok(hr
== S_OK
, "got 0x%08x\n", hr
);
1684 ok(type
== XmlNodeType_Element
, "got %d\n", type
);
1686 ret
= IXmlReader_IsEmptyElement(reader
);
1687 ok(ret
== test
->empty
, "got %d, expected %d. xml=%s\n", ret
, test
->empty
, test
->xml
);
1689 IStream_Release(stream
);
1693 IXmlReader_Release(reader
);
1696 static struct test_entry attributes_tests
[] = {
1697 { "<a attr1=\"attrvalue\"/>", "attr1", "attrvalue", S_OK
},
1698 { "<a attr1=\"a\'\'ttrvalue\"/>", "attr1", "a\'\'ttrvalue", S_OK
},
1699 { "<a attr1=\'a\"ttrvalue\'/>", "attr1", "a\"ttrvalue", S_OK
},
1700 { "<a attr1=\' \'/>", "attr1", " ", S_OK
},
1701 { "<a attr1=\" \"/>", "attr1", " ", S_OK
},
1702 { "<a attr1=\"\r\n \r \n \t\n\r\"/>", "attr1", " ", S_OK
},
1703 { "<a attr1=\" val \"/>", "attr1", " val ", S_OK
},
1704 { "<a attr1=\"\r\n\tval\n\"/>", "attr1", " val ", S_OK
},
1705 { "<a attr1=\"val \"/>", "attr1", "val ", S_OK
},
1706 { "<a attr1=\"val \"/>", "attr1", "val ", S_OK
},
1707 { "<a attr1=\"<>&'"\"/>", "attr1", "<>&\'\"", S_OK
},
1708 { "<a attr1=\"&entname;\"/>", NULL
, NULL
, WC_E_UNDECLAREDENTITY
},
1709 { "<a attr1=\"val\"/>", NULL
, NULL
, WC_E_XMLCHARACTER
},
1710 { "<a attr1=\"val &#a;\"/>", NULL
, NULL
, WC_E_DIGIT
, WC_E_SEMICOLON
},
1711 { "<a attr1=\"val a;\"/>", NULL
, NULL
, WC_E_SEMICOLON
},
1712 { "<a attr1=\"val g;\"/>", NULL
, NULL
, WC_E_SEMICOLON
},
1713 { "<a attr1=\"val &#xg;\"/>", NULL
, NULL
, WC_E_HEXDIGIT
, WC_E_SEMICOLON
},
1714 { "<a attr1=attrvalue/>", NULL
, NULL
, WC_E_QUOTE
},
1715 { "<a attr1=\"attr<value\"/>", NULL
, NULL
, WC_E_LESSTHAN
},
1716 { "<a attr1=\"&entname\"/>", NULL
, NULL
, WC_E_SEMICOLON
},
1720 static void test_read_attribute(void)
1722 struct test_entry
*test
= attributes_tests
;
1726 hr
= CreateXmlReader(&IID_IXmlReader
, (void**)&reader
, NULL
);
1727 ok(hr
== S_OK
, "S_OK, got %08x\n", hr
);
1734 stream
= create_stream_on_data(test
->xml
, strlen(test
->xml
)+1);
1735 hr
= IXmlReader_SetInput(reader
, (IUnknown
*)stream
);
1736 ok(hr
== S_OK
, "got %08x\n", hr
);
1738 type
= XmlNodeType_None
;
1739 hr
= IXmlReader_Read(reader
, &type
);
1741 if (test
->hr_broken
)
1742 ok(hr
== test
->hr
|| broken(hr
== test
->hr_broken
), "got %08x for %s\n", hr
, test
->xml
);
1744 ok(hr
== test
->hr
, "got %08x for %s\n", hr
, test
->xml
);
1751 ok(type
== XmlNodeType_Element
, "got %d for %s\n", type
, test
->xml
);
1753 hr
= IXmlReader_MoveToFirstAttribute(reader
);
1754 ok(hr
== S_OK
, "got 0x%08x\n", hr
);
1758 hr
= IXmlReader_GetLocalName(reader
, &str
, &len
);
1759 ok(hr
== S_OK
, "got 0x%08x\n", hr
);
1760 ok(len
== strlen(test
->name
), "got %u\n", len
);
1761 str_exp
= a2w(test
->name
);
1762 ok(!lstrcmpW(str
, str_exp
), "got %s\n", wine_dbgstr_w(str
));
1767 hr
= IXmlReader_GetQualifiedName(reader
, &str
, &len
);
1768 ok(hr
== S_OK
, "got 0x%08x\n", hr
);
1770 ok(len
== strlen(test
->name
), "got %u\n", len
);
1771 str_exp
= a2w(test
->name
);
1772 ok(!lstrcmpW(str
, str_exp
), "got %s\n", wine_dbgstr_w(str
));
1778 hr
= IXmlReader_GetValue(reader
, &str
, &len
);
1779 ok(hr
== S_OK
, "got 0x%08x\n", hr
);
1780 ok(len
== strlen(test
->value
), "got %u\n", len
);
1781 str_exp
= a2w(test
->value
);
1782 ok(!lstrcmpW(str
, str_exp
), "got %s\n", wine_dbgstr_w(str
));
1786 IStream_Release(stream
);
1790 IXmlReader_Release(reader
);
1793 static void test_reader_properties(void)
1798 hr
= CreateXmlReader(&IID_IXmlReader
, (void**)&reader
, NULL
);
1799 ok(hr
== S_OK
, "S_OK, got %08x\n", hr
);
1801 hr
= IXmlReader_SetProperty(reader
, XmlReaderProperty_MultiLanguage
, 0);
1802 ok(hr
== S_OK
, "SetProperty failed: %08x\n", hr
);
1804 hr
= IXmlReader_SetProperty(reader
, XmlReaderProperty_MaxElementDepth
, 0);
1805 ok(hr
== S_OK
, "SetProperty failed: %08x\n", hr
);
1807 IXmlReader_Release(reader
);
1812 test_reader_create();
1814 test_reader_state();
1815 test_read_attribute();
1817 test_read_comment();
1820 test_read_element();
1821 test_isemptyelement();
1824 test_read_pending();
1825 test_readvaluechunk();
1826 test_read_xmldeclaration();
1827 test_reader_properties();