/* * Schema test * * Copyright 2007 Huw Davies * Copyright 2010 Adam Martinson for CodeWeavers * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA */ #include <stdio.h> #include <assert.h> #define COBJMACROS #include "initguid.h" #include "windows.h" #include "ole2.h" #include "msxml2.h" #undef CLSID_DOMDocument #include "msxml2did.h" #include "dispex.h" #include "cguid.h" #include "wine/test.h" #define check_interface(a, b, c) check_interface_(__LINE__, a, b, c) static void check_interface_(unsigned int line, void *iface_ptr, REFIID iid, BOOL supported) { IUnknown *iface = iface_ptr; HRESULT hr, expected_hr; IUnknown *unk; expected_hr = supported ? S_OK : E_NOINTERFACE; hr = IUnknown_QueryInterface(iface, iid, (void **)&unk); ok_(__FILE__, line)(hr == expected_hr, "Got hr %#lx, expected %#lx.\n", hr, expected_hr); if (SUCCEEDED(hr)) IUnknown_Release(unk); } static const WCHAR xdr_schema1_uri[] = L"x-schema:test1.xdr"; static const WCHAR xdr_schema1_xml[] = L"<?xml version='1.0'?>" "<Schema xmlns='urn:schemas-microsoft-com:xml-data'" " xmlns:dt='urn:schemas-microsoft-com:datatypes'" " name='test1.xdr'>" " <ElementType name='x' dt:type='boolean'/>" " <ElementType name='y'>" " <datatype dt:type='int'/>" " </ElementType>" " <ElementType name='z'/>" " <ElementType name='root' content='eltOnly' model='open' order='seq'>" " <element type='x'/>" " <element type='y'/>" " <element type='z'/>" " </ElementType>" "</Schema>"; static const WCHAR xdr_schema2_uri[] = L"x-schema:test2.xdr"; static const WCHAR xdr_schema2_xml[] = L"<?xml version='1.0'?>" "<Schema xmlns='urn:schemas-microsoft-com:xml-data'" " xmlns:dt='urn:schemas-microsoft-com:datatypes'" " name='test2.xdr'>" " <ElementType name='x' dt:type='bin.base64'/>" " <ElementType name='y' dt:type='uuid'/>" " <ElementType name='z'/>" " <ElementType name='root' content='eltOnly' model='closed' order='one'>" " <element type='x'/>" " <element type='y'/>" " <element type='z'/>" " </ElementType>" "</Schema>"; static const WCHAR xdr_schema3_uri[] = L"x-schema:test3.xdr"; static const WCHAR xdr_schema3_xml[] = L"<?xml version='1.0'?>" "<Schema xmlns='urn:schemas-microsoft-com:xml-data'" " xmlns:dt='urn:schemas-microsoft-com:datatypes'" " name='test3.xdr'>" " <ElementType name='root' content='textOnly' model='open'>" " <AttributeType name='x' dt:type='int'/>" " <AttributeType name='y' dt:type='enumeration' dt:values='a b c'/>" " <AttributeType name='z' dt:type='uuid'/>" " <attribute type='x'/>" " <attribute type='y'/>" " <attribute type='z'/>" " </ElementType>" "</Schema>"; static const WCHAR xsd_schema1_uri[] = L"x-schema:test1.xsd"; static const WCHAR xsd_schema1_xml[] = L"<?xml version='1.0'?>" "<schema xmlns='http://www.w3.org/2001/XMLSchema'" " targetNamespace='x-schema:test1.xsd'>" " <element name='root'>" " <complexType>" " <sequence maxOccurs='unbounded'>" " <any/>" " </sequence>" " </complexType>" " </element>" "</schema>"; static const WCHAR xsd_schema2_uri[] = L"x-schema:test2.xsd"; static const WCHAR xsd_schema2_xml[] = L"<?xml version='1.0'?>" "<schema xmlns='http://www.w3.org/2001/XMLSchema'" " targetNamespace='x-schema:test2.xsd'>" " <element name='root'>" " <complexType>" " <sequence maxOccurs='unbounded'>" " <any/>" " </sequence>" " </complexType>" " </element>" "</schema>"; static const WCHAR xsd_schema3_uri[] = L"x-schema:test3.xsd"; static const WCHAR xsd_schema3_xml[] = L"<?xml version='1.0'?>" "<schema xmlns='http://www.w3.org/2001/XMLSchema'" " targetNamespace='x-schema:test3.xsd'>" " <element name='root'>" " <complexType>" " <sequence maxOccurs='unbounded'>" " <any/>" " </sequence>" " </complexType>" " </element>" "</schema>"; static const WCHAR szDatatypeXDR[] = L"<Schema xmlns='urn:schemas-microsoft-com:xml-data'\n" " xmlns:dt='urn:schemas-microsoft-com:datatypes'>\n" " <ElementType name='base64Data' content='textOnly' dt:type='bin.base64'/>\n" " <ElementType name='hexData' content='textOnly' dt:type='bin.hex'/>\n" " <ElementType name='boolData' content='textOnly' dt:type='boolean'/>\n" " <ElementType name='charData' content='textOnly' dt:type='char'/>\n" " <ElementType name='dateData' content='textOnly' dt:type='date'/>\n" " <ElementType name='dateTimeData' content='textOnly' dt:type='dateTime'/>\n" " <ElementType name='dateTimeTzData' content='textOnly' dt:type='dateTime.tz'/>\n" " <ElementType name='entityData' content='textOnly' dt:type='entity'/>\n" " <ElementType name='entitiesData' content='textOnly' dt:type='entities'/>\n" " <ElementType name='fixedData' content='textOnly' dt:type='fixed.14.4'/>\n" " <ElementType name='floatData' content='textOnly' dt:type='float'/>\n" " <ElementType name='i1Data' content='textOnly' dt:type='i1'/>\n" " <ElementType name='i2Data' content='textOnly' dt:type='i2'/>\n" " <ElementType name='i4Data' content='textOnly' dt:type='i4'/>\n" " <ElementType name='i8Data' content='textOnly' dt:type='i8'/>\n" " <ElementType name='intData' content='textOnly' dt:type='int'/>\n" " <ElementType name='nmtokData' content='textOnly' dt:type='nmtoken'/>\n" " <ElementType name='nmtoksData' content='textOnly' dt:type='nmtokens'/>\n" " <ElementType name='numData' content='textOnly' dt:type='number'/>\n" " <ElementType name='r4Data' content='textOnly' dt:type='r4'/>\n" " <ElementType name='r8Data' content='textOnly' dt:type='r8'/>\n" " <ElementType name='stringData' content='textOnly' dt:type='string'/>\n" " <ElementType name='timeData' content='textOnly' dt:type='time'/>\n" " <ElementType name='timeTzData' content='textOnly' dt:type='time.tz'/>\n" " <ElementType name='u1Data' content='textOnly' dt:type='ui1'/>\n" " <ElementType name='u2Data' content='textOnly' dt:type='ui2'/>\n" " <ElementType name='u4Data' content='textOnly' dt:type='ui4'/>\n" " <ElementType name='u8Data' content='textOnly' dt:type='ui8'/>\n" " <ElementType name='uriData' content='textOnly' dt:type='uri'/>\n" " <ElementType name='uuidData' content='textOnly' dt:type='uuid'/>\n" "\n" " <ElementType name='Name' content='textOnly' dt:type='nmtoken'/>\n" " <ElementType name='Value' content='eltOnly' order='many'>\n" " <element type='base64Data'/>\n" " <element type='hexData'/>\n" " <element type='boolData'/>\n" " <element type='charData'/>\n" " <element type='dateData'/>\n" " <element type='dateTimeData'/>\n" " <element type='dateTimeTzData'/>\n" " <element type='entityData'/>\n" " <element type='entitiesData'/>\n" " <element type='fixedData'/>\n" " <element type='floatData'/>\n" " <element type='i1Data'/>\n" " <element type='i2Data'/>\n" " <element type='i4Data'/>\n" " <element type='i8Data'/>\n" " <element type='intData'/>\n" " <element type='nmtokData'/>\n" " <element type='nmtoksData'/>\n" " <element type='numData'/>\n" " <element type='r4Data'/>\n" " <element type='r8Data'/>\n" " <element type='stringData'/>\n" " <element type='timeData'/>\n" " <element type='timeTzData'/>\n" " <element type='u1Data'/>\n" " <element type='u2Data'/>\n" " <element type='u4Data'/>\n" " <element type='u8Data'/>\n" " <element type='uriData'/>\n" " <element type='uuidData'/>\n" " </ElementType>\n" " <ElementType name='Property' content='eltOnly' order='seq'>\n" " <element type='Name'/>\n" " <element type='Value'/>\n" " </ElementType>\n" " <ElementType name='Properties' content='eltOnly'>\n" " <element type='Property' minOccurs='0' maxOccurs='*'/>\n" " </ElementType>\n" "</Schema>"; static const WCHAR szDatatypeXML[] = L"<?xml version='1.0'?>\n" "<Properties xmlns='urn:x-schema:datatype-test-xdr'>\n" " <Property>\n" " <Name>testBase64</Name>\n" " <Value>\n" " <base64Data>+HugeNumber+</base64Data>\n" " </Value>\n" " </Property>\n" " <Property>\n" " <Name>testHex</Name>\n" " <Value>\n" " <hexData>deadbeef</hexData>\n" " </Value>\n" " </Property>\n" " <Property>\n" " <Name>testBool</Name>\n" " <Value>\n" " <boolData>1</boolData>\n" " </Value>\n" " </Property>\n" " <Property>\n" " <Name>testChar</Name>\n" " <Value>\n" " <charData>u</charData>\n" " </Value>\n" " </Property>\n" " <Property>\n" " <Name>testDate</Name>\n" " <Value>\n" " <dateData>1998-02-01</dateData>\n" " </Value>\n" " </Property>\n" " <Property>\n" " <Name>testDateTime</Name>\n" " <Value>\n" " <dateTimeData>1998-02-01T12:34:56</dateTimeData>\n" " </Value>\n" " </Property>\n" " <Property>\n" " <Name>testDateTimeTz</Name>\n" " <Value>\n" " <dateTimeTzData>1998-02-01T12:34:56-06:00</dateTimeTzData>\n" " </Value>\n" " </Property>\n" " <Property>\n" " <Name>testFixed</Name>\n" " <Value>\n" " <fixedData>3.1416</fixedData>\n" " </Value>\n" " </Property>\n" " <Property>\n" " <Name>testFloat</Name>\n" " <Value>\n" " <floatData>3.14159</floatData>\n" " </Value>\n" " </Property>\n" " <Property>\n" " <Name>testI1</Name>\n" " <Value>\n" " <i1Data>42</i1Data>\n" " </Value>\n" " </Property>\n" " <Property>\n" " <Name>testI2</Name>\n" " <Value>\n" " <i2Data>420</i2Data>\n" " </Value>\n" " </Property>\n" " <Property>\n" " <Name>testI4</Name>\n" " <Value>\n" " <i4Data>-420000000</i4Data>\n" " </Value>\n" " </Property>\n" " <Property>\n" " <Name>testI8</Name>\n" " <Value>\n" " <i8Data>-4200000000</i8Data>\n" " </Value>\n" " </Property>\n" " <Property>\n" " <Name>testInt</Name>\n" " <Value>\n" " <intData>42</intData>\n" " </Value>\n" " </Property>\n" " <Property>\n" " <Name>testNmtoken</Name>\n" " <Value>\n" " <nmtokData>tok1</nmtokData>\n" " </Value>\n" " </Property>\n" " <Property>\n" " <Name>testNmtokens</Name>\n" " <Value>\n" " <nmtoksData>tok1 tok2 tok3</nmtoksData>\n" " </Value>\n" " </Property>\n" " <Property>\n" " <Name>testNumber</Name>\n" " <Value>\n" " <numData>3.14159</numData>\n" " </Value>\n" " </Property>\n" " <Property>\n" " <Name>testR4</Name>\n" " <Value>\n" " <r4Data>3.14159265</r4Data>\n" " </Value>\n" " </Property>\n" " <Property>\n" " <Name>testR8</Name>\n" " <Value>\n" " <r8Data>3.14159265358979323846</r8Data>\n" " </Value>\n" " </Property>\n" " <Property>\n" " <Name>testString</Name>\n" " <Value>\n" " <stringData>foo bar</stringData>\n" " </Value>\n" " </Property>\n" " <Property>\n" " <Name>testTime</Name>\n" " <Value>\n" " <timeData>12:34:56</timeData>\n" " </Value>\n" " </Property>\n" " <Property>\n" " <Name>testTimeTz</Name>\n" " <Value>\n" " <timeTzData>12:34:56-06:00</timeTzData>\n" " </Value>\n" " </Property>\n" " <Property>\n" " <Name>testU1</Name>\n" " <Value>\n" " <u1Data>255</u1Data>\n" " </Value>\n" " </Property>\n" " <Property>\n" " <Name>testU2</Name>\n" " <Value>\n" " <u2Data>65535</u2Data>\n" " </Value>\n" " </Property>\n" " <Property>\n" " <Name>testU4</Name>\n" " <Value>\n" " <u4Data>4294967295</u4Data>\n" " </Value>\n" " </Property>\n" " <Property>\n" " <Name>testU8</Name>\n" " <Value>\n" " <u8Data>18446744073709551615</u8Data>\n" " </Value>\n" " </Property>\n" " <Property>\n" " <Name>testURI</Name>\n" " <Value>\n" " <uriData>urn:schemas-microsoft-com:datatypes</uriData>\n" " </Value>\n" " </Property>\n" " <Property>\n" " <Name>testUUID</Name>\n" " <Value>\n" " <uuidData>2933BF81-7B36-11D2-B20E-00C04F983E60</uuidData>\n" " </Value>\n" " </Property>\n" "</Properties>"; static const WCHAR szOpenSeqXDR[] = L"<Schema xmlns='urn:schemas-microsoft-com:xml-data'>\n" " <ElementType name='w' content='empty' model='closed'/>\n" " <ElementType name='x' content='empty' model='closed'/>\n" " <ElementType name='y' content='empty' model='closed'/>\n" " <ElementType name='z' content='empty' model='closed'/>\n" " <ElementType name='test' content='eltOnly' model='open' order='seq'>\n" " <element type='x'/>\n" " <group order='seq'>\n" " <element type='x'/>\n" " <element type='y'/>\n" " <element type='z'/>\n" " </group>\n" " <element type='z'/>\n" " </ElementType>\n" "</Schema>"; static const WCHAR szOpenSeqXML1[] = L"<test><x/><x/><y/><z/><z/></test>"; static const WCHAR szOpenSeqXML2[] = L"<test><x/><x/><y/><z/><z/><w/></test>"; static const WCHAR szOpenSeqXML3[] = L"<test><w/><x/><x/><y/><z/><z/></test>"; static const WCHAR szOpenSeqXML4[] = L"<test><x/><x/><y/><z/><z/><v/></test>"; static ULONG get_refcount(void *iface) { IUnknown_AddRef((IUnknown *)iface); return IUnknown_Release((IUnknown *)iface); } #define _expect64(expr, str, base, TYPE, CONV) { \ TYPE v1 = expr; \ TYPE v2 = CONV(str, NULL, base); \ ok(v1 == v2, #expr "returned %s, expected %s\n", \ wine_dbgstr_longlong(v1), wine_dbgstr_longlong(v2)); \ } #define expect_int64(expr, x, base) _expect64(expr, #x, base, LONG64, strtoll) #define expect_uint64(expr, x, base) _expect64(expr, #x, base, ULONG64, strtoull) static BSTR alloced_bstrs[256]; static int alloced_bstrs_count; static BSTR _bstr_(const WCHAR *str) { assert(alloced_bstrs_count < ARRAY_SIZE(alloced_bstrs)); alloced_bstrs[alloced_bstrs_count] = SysAllocString(str); return alloced_bstrs[alloced_bstrs_count++]; } static void free_bstrs(void) { int i; for (i = 0; i < alloced_bstrs_count; i++) SysFreeString(alloced_bstrs[i]); alloced_bstrs_count = 0; } static VARIANT _variantdoc_(void* doc) { VARIANT v; V_VT(&v) = VT_DISPATCH; V_DISPATCH(&v) = (IDispatch*)doc; return v; } static void* _create_object(const GUID *clsid, const char *name, const IID *iid, int line) { void *obj = NULL; HRESULT hr; hr = CoCreateInstance(clsid, NULL, CLSCTX_INPROC_SERVER, iid, &obj); if (hr != S_OK) win_skip_(__FILE__,line)("failed to create %s instance: hr %#lx.\n", name, hr); return obj; } #define _create(cls) cls, #cls #define create_document(iid) _create_object(&_create(CLSID_DOMDocument), iid, __LINE__) #define create_document_version(v, iid) _create_object(&_create(CLSID_DOMDocument ## v), iid, __LINE__) #define create_cache(iid) _create_object(&_create(CLSID_XMLSchemaCache), iid, __LINE__) #define create_cache_version(v, iid) _create_object(&_create(CLSID_XMLSchemaCache ## v), iid, __LINE__) static void test_schema_refs(void) { static const WCHAR xdr_schema_xml[] = L"<Schema xmlns=\"urn:schemas-microsoft-com:xml-data\"\nxmlns:dt=\"urn:schemas-microsoft-com:datatypes\">\n</Schema>\n"; IXMLDOMDocument2 *doc; IXMLDOMNode *node; IXMLDOMSchemaCollection *cache; LONG len, refcount; VARIANT v; VARIANT_BOOL b; BSTR str; HRESULT hr; doc = create_document(&IID_IXMLDOMDocument2); if (!doc) return; cache = create_cache(&IID_IXMLDOMSchemaCollection); if(!cache) { IXMLDOMDocument2_Release(doc); return; } VariantInit(&v); str = SysAllocString(xdr_schema_xml); hr = IXMLDOMDocument2_loadXML(doc, str, &b); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(b == VARIANT_TRUE, "b %04x\n", b); SysFreeString(str); node = (void*)0xdeadbeef; hr = IXMLDOMSchemaCollection_get(cache, NULL, &node); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(node == NULL, "%p\n", node); /* NULL uri pointer, still adds a document */ hr = IXMLDOMSchemaCollection_add(cache, NULL, _variantdoc_(doc)); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); len = -1; hr = IXMLDOMSchemaCollection_get_length(cache, &len); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(len == 1, "Unexpected length %ld.\n", len); /* read back - empty valid BSTR */ str = NULL; hr = IXMLDOMSchemaCollection_get_namespaceURI(cache, 0, &str); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(str && *str == 0, "got %p\n", str); SysFreeString(str); node = NULL; hr = IXMLDOMSchemaCollection_get(cache, NULL, &node); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(node != NULL, "%p\n", node); IXMLDOMNode_Release(node); node = NULL; str = SysAllocString(L""); hr = IXMLDOMSchemaCollection_get(cache, str, &node); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(node != NULL, "%p\n", node); IXMLDOMNode_Release(node); SysFreeString(str); /* remove with NULL uri */ hr = IXMLDOMSchemaCollection_remove(cache, NULL); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); len = -1; hr = IXMLDOMSchemaCollection_get_length(cache, &len); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(len == 0, "Unexpected length %ld.\n", len); /* same, but with VT_UNKNOWN type */ V_VT(&v) = VT_UNKNOWN; V_UNKNOWN(&v) = (IUnknown*)doc; hr = IXMLDOMSchemaCollection_add(cache, NULL, v); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); len = -1; hr = IXMLDOMSchemaCollection_get_length(cache, &len); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(len == 1, "Unexpected length %ld.\n", len); hr = IXMLDOMSchemaCollection_remove(cache, NULL); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); len = -1; hr = IXMLDOMSchemaCollection_get_length(cache, &len); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(len == 0, "Unexpected length %ld.\n", len); str = SysAllocString(L"x-schema:test.xml"); hr = IXMLDOMSchemaCollection_add(cache, str, _variantdoc_(doc)); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); /* IXMLDOMSchemaCollection_add doesn't add a ref on doc */ refcount = get_refcount(doc); ok(refcount == 1, "Unexpected refcount %ld.\n", refcount); SysFreeString(str); V_VT(&v) = VT_INT; hr = IXMLDOMDocument2_get_schemas(doc, &v); ok(hr == S_FALSE, "Unexpected hr %#lx.\n", hr); ok(V_VT(&v) == VT_NULL, "vt %x\n", V_VT(&v)); refcount = IXMLDOMSchemaCollection_AddRef(cache); ok(refcount == 2, "Unexpected refcount %ld.\n", refcount); V_VT(&v) = VT_DISPATCH; V_DISPATCH(&v) = (IDispatch*)cache; /* check that putref_schemas takes a ref */ hr = IXMLDOMDocument2_putref_schemas(doc, v); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); refcount = get_refcount(cache); ok(refcount == 3, "Unexpected refcount %ld.\n", refcount); VariantClear(&v); /* refs now 2 */ V_VT(&v) = VT_INT; /* check that get_schemas adds a ref */ hr = IXMLDOMDocument2_get_schemas(doc, &v); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(V_VT(&v) == VT_DISPATCH, "vt %x\n", V_VT(&v)); refcount = get_refcount(cache); ok(refcount == 3, "Unexpected refcount %ld.\n", refcount); /* get_schemas doesn't release a ref if passed VT_DISPATCH - ie it doesn't call VariantClear() */ hr = IXMLDOMDocument2_get_schemas(doc, &v); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(V_VT(&v) == VT_DISPATCH, "vt %x\n", V_VT(&v)); refcount = get_refcount(cache); ok(refcount == 4, "Unexpected refcount %ld.\n", refcount); /* release the two refs returned by get_schemas */ refcount = IXMLDOMSchemaCollection_Release(cache); ok(refcount == 3, "Unexpected refcount %ld.\n", refcount); refcount = IXMLDOMSchemaCollection_Release(cache); ok(refcount == 2, "Unexpected refcount %ld.\n", refcount); /* check that taking another ref on the document doesn't change the schema's ref count */ refcount = IXMLDOMDocument2_AddRef(doc); ok(refcount == 2, "Unexpected refcount %ld.\n", refcount); refcount = get_refcount(cache); ok(refcount == 2, "Unexpected refcount %ld.\n", refcount); refcount = IXMLDOMDocument2_Release(doc); ok(refcount == 1, "Unexpected refcount %ld.\n", refcount); /* call putref_schema with some odd variants */ V_VT(&v) = VT_INT; hr = IXMLDOMDocument2_putref_schemas(doc, v); ok(hr == E_FAIL, "Unexpected hr %#lx.\n", hr); refcount = get_refcount(cache); ok(refcount == 2, "Unexpected refcount %ld.\n", refcount); /* calling with VT_EMPTY releases the schema */ V_VT(&v) = VT_EMPTY; hr = IXMLDOMDocument2_putref_schemas(doc, v); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); refcount = get_refcount(cache); ok(refcount == 1, "Unexpected refcount %ld.\n", refcount); /* try setting with VT_UNKNOWN */ refcount = IXMLDOMSchemaCollection_AddRef(cache); ok(refcount == 2, "Unexpected refcount %ld.\n", refcount); V_VT(&v) = VT_UNKNOWN; V_UNKNOWN(&v) = (IUnknown*)cache; hr = IXMLDOMDocument2_putref_schemas(doc, v); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); refcount = get_refcount(cache); ok(refcount == 3, "Unexpected refcount %ld.\n", refcount); VariantClear(&v); /* refs now 2 */ /* calling with VT_NULL releases the schema */ V_VT(&v) = VT_NULL; hr = IXMLDOMDocument2_putref_schemas(doc, v); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); refcount = get_refcount(cache); ok(refcount == 1, "Unexpected refcount %ld.\n", refcount); /* refs now 1 */ /* set again */ refcount = IXMLDOMSchemaCollection_AddRef(cache); ok(refcount == 2, "Unexpected refcount %ld.\n", refcount); V_VT(&v) = VT_UNKNOWN; V_UNKNOWN(&v) = (IUnknown*)cache; hr = IXMLDOMDocument2_putref_schemas(doc, v); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); refcount = get_refcount(cache); ok(refcount == 3, "Unexpected refcount %ld.\n", refcount); VariantClear(&v); /* refs now 2 */ /* release the final ref on the doc which should release its ref on the schema */ refcount = IXMLDOMDocument2_Release(doc); ok(!refcount, "Unexpected refcount %ld.\n", refcount); refcount = get_refcount(cache); ok(refcount == 1, "Unexpected refcount %ld.\n", refcount); refcount = IXMLDOMSchemaCollection_Release(cache); ok(!refcount, "Unexpected refcount %ld.\n", refcount); } static void test_collection_refs(void) { IXMLDOMDocument2 *schema1, *schema2, *schema3; IXMLDOMSchemaCollection *cache1, *cache2, *cache3; LONG refcount, length; VARIANT_BOOL b; HRESULT hr; schema1 = create_document(&IID_IXMLDOMDocument2); ok(schema1 != NULL, "Failed to create a document.\n"); cache1 = create_cache(&IID_IXMLDOMSchemaCollection); ok(cache1 != NULL, "Failed to create schema collection.\n"); if (!schema1 || !cache1) { if (schema1) IXMLDOMDocument2_Release(schema1); if (cache1) IXMLDOMSchemaCollection_Release(cache1); return; } schema2 = create_document(&IID_IXMLDOMDocument2); schema3 = create_document(&IID_IXMLDOMDocument2); cache2 = create_cache(&IID_IXMLDOMSchemaCollection); cache3 = create_cache(&IID_IXMLDOMSchemaCollection); hr = IXMLDOMDocument2_loadXML(schema1, _bstr_(xdr_schema1_xml), &b); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(b == VARIANT_TRUE, "failed to load XML\n"); hr = IXMLDOMDocument2_loadXML(schema2, _bstr_(xdr_schema2_xml), &b); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(b == VARIANT_TRUE, "failed to load XML\n"); hr = IXMLDOMDocument2_loadXML(schema3, _bstr_(xdr_schema3_xml), &b); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(b == VARIANT_TRUE, "failed to load XML\n"); hr = IXMLDOMSchemaCollection_add(cache1, _bstr_(xdr_schema1_uri), _variantdoc_(schema1)); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); hr = IXMLDOMSchemaCollection_add(cache2, _bstr_(xdr_schema2_uri), _variantdoc_(schema2)); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); hr = IXMLDOMSchemaCollection_add(cache3, _bstr_(xdr_schema3_uri), _variantdoc_(schema3)); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); refcount = IXMLDOMDocument2_Release(schema1); ok(!refcount, "Unexpected refcount %ld.\n", refcount); refcount = IXMLDOMDocument2_Release(schema2); ok(!refcount, "Unexpected refcount %ld.\n", refcount); refcount = IXMLDOMDocument2_Release(schema3); ok(!refcount, "Unexpected refcount %ld.\n", refcount); schema1 = NULL; schema2 = NULL; schema3 = NULL; /* releasing the original doc does not affect the schema cache */ hr = IXMLDOMSchemaCollection_get(cache1, _bstr_(xdr_schema1_uri), (IXMLDOMNode**)&schema1); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); hr = IXMLDOMSchemaCollection_get(cache2, _bstr_(xdr_schema2_uri), (IXMLDOMNode**)&schema2); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); hr = IXMLDOMSchemaCollection_get(cache3, _bstr_(xdr_schema3_uri), (IXMLDOMNode**)&schema3); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); /* we get a read-only domdoc interface, created just for us */ if (schema1) { refcount = get_refcount(schema1); ok(refcount == 1, "Unexpected refcount %ld.\n", refcount); } if (schema2) { refcount = get_refcount(schema2); ok(refcount == 1, "Unexpected refcount %ld.\n", refcount); } if (schema3) { refcount = get_refcount(schema3); ok(refcount == 1, "Unexpected refcount %ld.\n", refcount); } hr = IXMLDOMSchemaCollection_addCollection(cache1, NULL); ok(hr == E_POINTER, "Unexpected hr %#lx.\n", hr); hr = IXMLDOMSchemaCollection_addCollection(cache2, cache1); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); hr = IXMLDOMSchemaCollection_addCollection(cache3, cache2); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); length = -1; hr = IXMLDOMSchemaCollection_get_length(cache1, &length); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(length == 1, "Unexpected length %ld.\n", length); length = -1; hr = IXMLDOMSchemaCollection_get_length(cache2, &length); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(length == 2, "Unexpected length %ld.\n", length); length = -1; hr = IXMLDOMSchemaCollection_get_length(cache3, &length); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(length == 3, "Unexpected length %ld.\n", length); /* merging collections does not affect the ref count */ refcount = get_refcount(cache1); ok(refcount == 1, "Unexpected refcount %ld.\n", refcount); refcount = get_refcount(cache2); ok(refcount == 1, "Unexpected refcount %ld.\n", refcount); refcount = get_refcount(cache3); ok(refcount == 1, "Unexpected refcount %ld.\n", refcount); /* nor does it affect the domdoc instances */ if (schema1) { refcount = IXMLDOMDocument2_Release(schema1); ok(!refcount, "Unexpected refcount %ld.\n", refcount); } if (schema2) { refcount = IXMLDOMDocument2_Release(schema2); ok(!refcount, "Unexpected refcount %ld.\n", refcount); } if (schema3) { refcount = IXMLDOMDocument2_Release(schema3); ok(!refcount, "Unexpected refcount %ld.\n", refcount); } schema1 = NULL; schema2 = NULL; schema3 = NULL; /* releasing the domdoc instances doesn't change the cache */ hr = IXMLDOMSchemaCollection_get(cache1, _bstr_(xdr_schema1_uri), (IXMLDOMNode**)&schema1); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); hr = IXMLDOMSchemaCollection_get(cache2, _bstr_(xdr_schema2_uri), (IXMLDOMNode**)&schema2); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); hr = IXMLDOMSchemaCollection_get(cache3, _bstr_(xdr_schema3_uri), (IXMLDOMNode**)&schema3); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); /* we can just get them again */ if (schema1) { refcount = get_refcount(schema1); ok(refcount == 1, "Unexpected refcount %ld.\n", refcount); } if (schema2) { refcount = get_refcount(schema2); ok(refcount == 1, "Unexpected refcount %ld.\n", refcount); } if (schema3) { refcount = get_refcount(schema3); ok(refcount == 1, "Unexpected refcount %ld.\n", refcount); } /* releasing the caches does not affect the domdoc instances */ refcount = IXMLDOMSchemaCollection_Release(cache1); ok(!refcount, "Unexpected refcount %ld.\n", refcount); refcount = IXMLDOMSchemaCollection_Release(cache2); ok(!refcount, "Unexpected refcount %ld.\n", refcount); refcount = IXMLDOMSchemaCollection_Release(cache3); ok(!refcount, "Unexpected refcount %ld.\n", refcount); /* they're just for us */ if (schema1) { refcount = IXMLDOMDocument2_Release(schema1); ok(!refcount, "Unexpected refcount %ld.\n", refcount); } if (schema2) { refcount = IXMLDOMDocument2_Release(schema2); ok(!refcount, "Unexpected refcount %ld.\n", refcount); } if (schema3) { refcount = IXMLDOMDocument2_Release(schema3); ok(!refcount, "Unexpected refcount %ld.\n", refcount); } free_bstrs(); } static void test_length(void) { IXMLDOMDocument2 *schema1, *schema2, *schema3; IXMLDOMSchemaCollection *cache; VARIANT_BOOL b; HRESULT hr; VARIANT v; LONG length; schema1 = create_document(&IID_IXMLDOMDocument2); schema2 = create_document(&IID_IXMLDOMDocument2); schema3 = create_document(&IID_IXMLDOMDocument2); cache = create_cache(&IID_IXMLDOMSchemaCollection); if (!schema1 || !schema2 || !schema3 || !cache) { if (schema1) IXMLDOMDocument2_Release(schema1); if (schema2) IXMLDOMDocument2_Release(schema2); if (schema3) IXMLDOMDocument2_Release(schema3); if (cache) IXMLDOMSchemaCollection_Release(cache); return; } VariantInit(&v); hr = IXMLDOMDocument2_loadXML(schema1, _bstr_(xdr_schema1_xml), &b); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(b == VARIANT_TRUE, "failed to load XML\n"); hr = IXMLDOMDocument2_loadXML(schema2, _bstr_(xdr_schema2_xml), &b); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(b == VARIANT_TRUE, "failed to load XML\n"); hr = IXMLDOMDocument2_loadXML(schema3, _bstr_(xdr_schema3_xml), &b); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(b == VARIANT_TRUE, "failed to load XML\n"); hr = IXMLDOMSchemaCollection_get_length(cache, NULL); ok(hr == E_POINTER, "Unexpected hr %#lx.\n", hr); /* MSDN lies; removing a nonexistent entry produces no error */ hr = IXMLDOMSchemaCollection_remove(cache, NULL); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); hr = IXMLDOMSchemaCollection_remove(cache, _bstr_(xdr_schema1_uri)); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); length = -1; hr = IXMLDOMSchemaCollection_get_length(cache, &length); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(length == 0, "Unexpected length %ld.\n", length); hr = IXMLDOMSchemaCollection_add(cache, _bstr_(xdr_schema1_uri), _variantdoc_(schema1)); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); length = -1; hr = IXMLDOMSchemaCollection_get_length(cache, &length); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(length == 1, "Unexpected length %ld.\n", length); hr = IXMLDOMSchemaCollection_add(cache, _bstr_(xdr_schema2_uri), _variantdoc_(schema2)); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); length = -1; hr = IXMLDOMSchemaCollection_get_length(cache, &length); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(length == 2, "Unexpected length %ld.\n", length); hr = IXMLDOMSchemaCollection_add(cache, _bstr_(xdr_schema3_uri), _variantdoc_(schema3)); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); length = -1; hr = IXMLDOMSchemaCollection_get_length(cache, &length); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(length == 3, "Unexpected length %ld.\n", length); /* adding with VT_NULL is the same as removing */ V_VT(&v) = VT_NULL; hr = IXMLDOMSchemaCollection_add(cache, _bstr_(xdr_schema1_uri), v); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); length = -1; hr = IXMLDOMSchemaCollection_get_length(cache, &length); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(length == 2, "Unexpected length %ld.\n", length); hr = IXMLDOMSchemaCollection_remove(cache, _bstr_(xdr_schema2_uri)); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); length = -1; hr = IXMLDOMSchemaCollection_get_length(cache, &length); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(length == 1, "Unexpected length %ld.\n", length); hr = IXMLDOMSchemaCollection_remove(cache, _bstr_(xdr_schema3_uri)); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); length = -1; hr = IXMLDOMSchemaCollection_get_length(cache, &length); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(length == 0, "Unexpected length %ld.\n", length); IXMLDOMDocument2_Release(schema1); IXMLDOMDocument2_Release(schema2); IXMLDOMDocument2_Release(schema3); IXMLDOMSchemaCollection_Release(cache); free_bstrs(); } static void test_collection_content(void) { IXMLDOMDocument2 *schema1, *schema2, *schema3, *schema4, *schema5; BSTR content[5] = {NULL, NULL, NULL, NULL, NULL}; IXMLDOMSchemaCollection *cache1, *cache2; VARIANT_BOOL b; LONG length; HRESULT hr; BSTR bstr; int i, j; schema1 = create_document_version(30, &IID_IXMLDOMDocument2); schema2 = create_document_version(30, &IID_IXMLDOMDocument2); schema3 = create_document_version(30, &IID_IXMLDOMDocument2); cache1 = create_cache_version(30, &IID_IXMLDOMSchemaCollection); cache2 = create_cache_version(40, &IID_IXMLDOMSchemaCollection); if (!schema1 || !schema2 || !schema3 || !cache1) { if (schema1) IXMLDOMDocument2_Release(schema1); if (schema2) IXMLDOMDocument2_Release(schema2); if (schema3) IXMLDOMDocument2_Release(schema3); if (cache1) IXMLDOMSchemaCollection_Release(cache1); return; } hr = IXMLDOMDocument2_loadXML(schema1, _bstr_(xdr_schema1_xml), &b); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(b == VARIANT_TRUE, "failed to load XML\n"); hr = IXMLDOMDocument2_loadXML(schema2, _bstr_(xdr_schema2_xml), &b); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(b == VARIANT_TRUE, "failed to load XML\n"); hr = IXMLDOMDocument2_loadXML(schema3, _bstr_(xdr_schema3_xml), &b); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(b == VARIANT_TRUE, "failed to load XML\n"); hr = IXMLDOMSchemaCollection_add(cache1, _bstr_(xdr_schema1_uri), _variantdoc_(schema1)); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); hr = IXMLDOMSchemaCollection_add(cache1, _bstr_(xdr_schema2_uri), _variantdoc_(schema2)); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); hr = IXMLDOMSchemaCollection_add(cache1, _bstr_(xdr_schema3_uri), _variantdoc_(schema3)); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); length = -1; hr = IXMLDOMSchemaCollection_get_length(cache1, &length); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(length == 3, "Unexpected length %ld.\n", length); IXMLDOMDocument2_Release(schema1); IXMLDOMDocument2_Release(schema2); IXMLDOMDocument2_Release(schema3); if (cache2) { schema1 = create_document_version(40, &IID_IXMLDOMDocument2); schema2 = create_document_version(40, &IID_IXMLDOMDocument2); schema3 = create_document_version(40, &IID_IXMLDOMDocument2); schema4 = create_document_version(40, &IID_IXMLDOMDocument2); schema5 = create_document_version(40, &IID_IXMLDOMDocument2); hr = IXMLDOMDocument2_loadXML(schema1, _bstr_(xdr_schema1_xml), &b); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(b == VARIANT_TRUE, "failed to load XML\n"); hr = IXMLDOMDocument2_loadXML(schema2, _bstr_(xdr_schema2_xml), &b); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(b == VARIANT_TRUE, "failed to load XML\n"); hr = IXMLDOMDocument2_loadXML(schema3, _bstr_(xsd_schema1_xml), &b); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(b == VARIANT_TRUE, "failed to load XML\n"); hr = IXMLDOMDocument2_loadXML(schema4, _bstr_(xsd_schema2_xml), &b); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(b == VARIANT_TRUE, "failed to load XML\n"); hr = IXMLDOMDocument2_loadXML(schema5, _bstr_(xsd_schema3_xml), &b); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(b == VARIANT_TRUE, "failed to load XML\n"); /* combining XDR and XSD schemas in the same cache is fine */ hr = IXMLDOMSchemaCollection_add(cache2, _bstr_(xdr_schema1_uri), _variantdoc_(schema1)); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); hr = IXMLDOMSchemaCollection_add(cache2, _bstr_(xdr_schema2_uri), _variantdoc_(schema2)); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); hr = IXMLDOMSchemaCollection_add(cache2, _bstr_(xsd_schema1_uri), _variantdoc_(schema3)); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); hr = IXMLDOMSchemaCollection_add(cache2, _bstr_(xsd_schema2_uri), _variantdoc_(schema4)); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); hr = IXMLDOMSchemaCollection_add(cache2, _bstr_(xsd_schema3_uri), _variantdoc_(schema5)); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); length = -1; hr = IXMLDOMSchemaCollection_get_length(cache2, &length); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(length == 5, "Unexpected length %ld.\n", length); IXMLDOMDocument2_Release(schema1); IXMLDOMDocument2_Release(schema2); IXMLDOMDocument2_Release(schema3); IXMLDOMDocument2_Release(schema4); IXMLDOMDocument2_Release(schema5); } bstr = (void*)0xdeadbeef; /* error if index is out of range */ hr = IXMLDOMSchemaCollection_get_namespaceURI(cache1, 3, &bstr); ok(hr == E_FAIL, "Unexpected hr %#lx.\n", hr); ok(bstr == (void*)0xdeadbeef, "got %p\n", bstr); /* error if return pointer is NULL */ hr = IXMLDOMSchemaCollection_get_namespaceURI(cache1, 0, NULL); ok(hr == E_POINTER, "Unexpected hr %#lx.\n", hr); /* pointer is checked first */ hr = IXMLDOMSchemaCollection_get_namespaceURI(cache1, 3, NULL); ok(hr == E_POINTER, "Unexpected hr %#lx.\n", hr); schema1 = NULL; /* no error if ns uri does not exist */ hr = IXMLDOMSchemaCollection_get(cache1, _bstr_(xsd_schema1_uri), (IXMLDOMNode**)&schema1); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(!schema1, "expected NULL\n"); /* a NULL bstr corresponds to no-uri ns */ hr = IXMLDOMSchemaCollection_get(cache1, NULL, (IXMLDOMNode**)&schema1); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(!schema1, "expected NULL\n"); /* error if return pointer is NULL */ hr = IXMLDOMSchemaCollection_get(cache1, _bstr_(xdr_schema1_uri), NULL); ok(hr == E_POINTER, "Unexpected hr %#lx.\n", hr); for (i = 0; i < 3; ++i) { bstr = NULL; hr = IXMLDOMSchemaCollection_get_namespaceURI(cache1, i, &bstr); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(bstr != NULL && *bstr, "expected non-empty string\n"); content[i] = bstr; for (j = 0; j < i; ++j) ok(wcscmp(content[j], bstr), "got duplicate entry\n"); } for (i = 0; i < 3; ++i) { SysFreeString(content[i]); content[i] = NULL; } if (cache2) { for (i = 0; i < 5; ++i) { bstr = NULL; hr = IXMLDOMSchemaCollection_get_namespaceURI(cache2, i, &bstr); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(bstr != NULL && *bstr, "expected non-empty string\n"); for (j = 0; j < i; ++j) ok(wcscmp(content[j], bstr), "got duplicate entry\n"); content[i] = bstr; } for (i = 0; i < 5; ++i) { SysFreeString(content[i]); content[i] = NULL; } } IXMLDOMSchemaCollection_Release(cache1); if (cache2) IXMLDOMSchemaCollection_Release(cache2); free_bstrs(); } static HRESULT validate_regex_document(IXMLDOMDocument2 *doc, IXMLDOMDocument2 *schema, IXMLDOMSchemaCollection* cache, const WCHAR *regex, const WCHAR *input) { static const WCHAR regex_doc[] = L"" "<?xml version='1.0'?>" "<root xmlns='urn:test'>%s</root>"; static const WCHAR regex_schema[] = L"<?xml version='1.0'?>" "<schema xmlns='http://www.w3.org/2001/XMLSchema'" " targetNamespace='urn:test'>" " <element name='root'>" " <simpleType>" " <restriction base='string'>" " <pattern value='%s'/>" " </restriction>" " </simpleType>" " </element>" "</schema>"; WCHAR buffer[1024]; IXMLDOMParseError* err; VARIANT v; VARIANT_BOOL b; BSTR namespace; BSTR bstr; HRESULT hr; VariantInit(&v); swprintf(buffer, ARRAY_SIZE(buffer), regex_doc, input); bstr = SysAllocString(buffer); hr = IXMLDOMDocument2_loadXML(doc, bstr, &b); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(b == VARIANT_TRUE, "failed to load XML\n"); SysFreeString(bstr); swprintf(buffer, ARRAY_SIZE(buffer), regex_schema, regex); bstr = SysAllocString(buffer); hr = IXMLDOMDocument2_loadXML(schema, bstr, &b); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(b == VARIANT_TRUE, "failed to load XML\n"); SysFreeString(bstr); /* add the schema to the cache */ V_VT(&v) = VT_DISPATCH; V_DISPATCH(&v) = NULL; hr = IXMLDOMDocument2_QueryInterface(schema, &IID_IDispatch, (void**)&V_DISPATCH(&v)); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(V_DISPATCH(&v) != NULL, "failed to get IDispatch interface\n"); namespace = SysAllocString(L"urn:test"); hr = IXMLDOMSchemaCollection_add(cache, namespace, v); SysFreeString(namespace); VariantClear(&v); if (FAILED(hr)) return hr; /* associate the cache to the doc */ V_VT(&v) = VT_DISPATCH; V_DISPATCH(&v) = NULL; hr = IXMLDOMSchemaCollection_QueryInterface(cache, &IID_IDispatch, (void**)&V_DISPATCH(&v)); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(V_DISPATCH(&v) != NULL, "failed to get IDispatch interface\n"); hr = IXMLDOMDocument2_putref_schemas(doc, v); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); VariantClear(&v); /* validate the doc * only declared elements in the declared order * this is fine */ err = NULL; bstr = NULL; hr = IXMLDOMDocument2_validate(doc, &err); ok(err != NULL, "domdoc_validate() should always set err\n"); if (IXMLDOMParseError_get_reason(err, &bstr) != S_FALSE) trace("got error: %s\n", wine_dbgstr_w(bstr)); SysFreeString(bstr); IXMLDOMParseError_Release(err); return hr; } static void test_regex(void) { struct regex_test { const WCHAR *regex; const WCHAR *input; }; struct regex_test tests[] = { { L"\\!", L"!" }, { L"\\\"", L"\"" }, { L"\\#", L"#" }, { L"\\$", L"$" }, { L"\\%", L"%" }, { L"\\,", L"," }, { L"\\/", L"/" }, { L"\\:", L":" }, { L"\\;", L";" }, { L"\\=", L"=" }, { L"\\>", L">" }, { L"\\@", L"@" }, { L"\\`", L"`" }, { L"\\~", L"~" }, { L"\\uCAFE", L"\xCAFE" }, /* non-BMP character in surrogate pairs: */ { L"\\uD83D\\uDE00", L"\xD83D\xDE00" }, /* "x{,2}" is non-standard and only works on libxml2 <= v2.9.10 */ { L"x{0,2}", L"x" } }; int i; for (i = 0; i < ARRAY_SIZE(tests); i++) { IXMLDOMDocument2 *doc40, *doc60; IXMLDOMDocument2 *schema40, *schema60; IXMLDOMSchemaCollection *cache40, *cache60; doc40 = create_document_version(40, &IID_IXMLDOMDocument2); doc60 = create_document_version(60, &IID_IXMLDOMDocument2); schema40 = create_document_version(40, &IID_IXMLDOMDocument2); schema60 = create_document_version(60, &IID_IXMLDOMDocument2); cache40 = create_cache_version(40, &IID_IXMLDOMSchemaCollection); cache60 = create_cache_version(60, &IID_IXMLDOMSchemaCollection); if (doc60 && schema60 && cache60) { HRESULT hr = validate_regex_document(doc60, schema60, cache60, tests[i].regex, tests[i].input); ok(hr == S_OK, "got %#lx for version 60 regex %s input %s\n", hr, wine_dbgstr_w(tests[i].regex), wine_dbgstr_w(tests[i].input)); if (doc40 && schema40 && cache40) { hr = validate_regex_document(doc40, schema40, cache40, tests[i].regex, tests[i].input); ok(hr == S_OK, "got %#lx version 40 regex %s input %s\n", hr, wine_dbgstr_w(tests[i].regex), wine_dbgstr_w(tests[i].input)); } } else ok(0, "out of memory\n"); if (doc40) IXMLDOMDocument2_Release(doc40); if (doc60) IXMLDOMDocument2_Release(doc60); if (schema40) IXMLDOMDocument2_Release(schema40); if (schema60) IXMLDOMDocument2_Release(schema60); if (cache40) IXMLDOMSchemaCollection_Release(cache40); if (cache60) IXMLDOMSchemaCollection_Release(cache60); } } static void test_XDR_schemas(void) { IXMLDOMDocument2 *doc, *schema; IXMLDOMSchemaCollection* cache; IXMLDOMParseError* err; VARIANT_BOOL b; HRESULT hr; VARIANT v; BSTR bstr; doc = create_document(&IID_IXMLDOMDocument2); schema = create_document(&IID_IXMLDOMDocument2); cache = create_cache(&IID_IXMLDOMSchemaCollection); if (!doc || !schema || !cache) { if (doc) IXMLDOMDocument2_Release(doc); if (schema) IXMLDOMDocument2_Release(schema); if (cache) IXMLDOMSchemaCollection_Release(cache); return; } VariantInit(&v); hr = IXMLDOMDocument2_loadXML(doc, _bstr_(szOpenSeqXML1), &b); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(b == VARIANT_TRUE, "failed to load XML string\n"); hr = IXMLDOMDocument2_loadXML(schema, _bstr_(szOpenSeqXDR), &b); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(b == VARIANT_TRUE, "failed to load XML string\n"); /* load the schema */ V_VT(&v) = VT_DISPATCH; V_DISPATCH(&v) = NULL; hr = IXMLDOMDocument2_QueryInterface(schema, &IID_IDispatch, (void**)&V_DISPATCH(&v)); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(V_DISPATCH(&v) != NULL, "failed to get IDispatch interface\n"); hr = IXMLDOMSchemaCollection_add(cache, _bstr_(L""), v); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); VariantClear(&v); /* associate the cache to the doc */ V_VT(&v) = VT_DISPATCH; V_DISPATCH(&v) = NULL; hr = IXMLDOMSchemaCollection_QueryInterface(cache, &IID_IDispatch, (void**)&V_DISPATCH(&v)); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(V_DISPATCH(&v) != NULL, "failed to get IDispatch interface\n"); hr = IXMLDOMDocument2_putref_schemas(doc, v); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); VariantClear(&v); /* validate the doc * only declared elements in the declared order * this is fine */ err = NULL; bstr = NULL; hr = IXMLDOMDocument2_validate(doc, &err); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(err != NULL, "domdoc_validate() should always set err\n"); hr = IXMLDOMParseError_get_reason(err, &bstr); ok(hr == S_FALSE, "Unexpected hr %#lx.\n", hr); ok(IXMLDOMParseError_get_reason(err, &bstr) == S_FALSE, "got error: %s\n", wine_dbgstr_w(bstr)); SysFreeString(bstr); IXMLDOMParseError_Release(err); /* load the next doc */ IXMLDOMDocument2_Release(doc); doc = create_document(&IID_IXMLDOMDocument2); hr = IXMLDOMDocument2_loadXML(doc, _bstr_(szOpenSeqXML2), &b); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(b == VARIANT_TRUE, "failed to load XML string\n"); /* associate the cache to the doc */ V_VT(&v) = VT_DISPATCH; V_DISPATCH(&v) = NULL; hr = IXMLDOMSchemaCollection_QueryInterface(cache, &IID_IDispatch, (void**)&V_DISPATCH(&v)); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(V_DISPATCH(&v) != NULL, "failed to get IDispatch interface\n"); hr = IXMLDOMDocument2_putref_schemas(doc, v); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); VariantClear(&v); /* validate the doc * declared elements in the declared order, with an extra declared element at the end * this is fine */ err = NULL; bstr = NULL; hr = IXMLDOMDocument2_validate(doc, &err); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(err != NULL, "domdoc_validate() should always set err\n"); hr = IXMLDOMParseError_get_reason(err, &bstr); ok(hr == S_FALSE, "Unexpected hr %#lx.\n", hr); ok(IXMLDOMParseError_get_reason(err, &bstr) == S_FALSE, "got error: %s\n", wine_dbgstr_w(bstr)); SysFreeString(bstr); IXMLDOMParseError_Release(err); /* load the next doc */ IXMLDOMDocument2_Release(doc); doc = create_document(&IID_IXMLDOMDocument2); hr = IXMLDOMDocument2_loadXML(doc, _bstr_(szOpenSeqXML3), &b); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(b == VARIANT_TRUE, "failed to load XML string\n"); /* associate the cache to the doc */ V_VT(&v) = VT_DISPATCH; V_DISPATCH(&v) = NULL; hr = IXMLDOMSchemaCollection_QueryInterface(cache, &IID_IDispatch, (void**)&V_DISPATCH(&v)); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(V_DISPATCH(&v) != NULL, "failed to get IDispatch interface\n"); hr = IXMLDOMDocument2_putref_schemas(doc, v); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); VariantClear(&v); /* validate the doc * fails, extra elements are only allowed at the end */ err = NULL; bstr = NULL; hr = IXMLDOMDocument2_validate(doc, &err); ok(hr == S_FALSE, "Unexpected hr %#lx.\n", hr); ok(err != NULL, "domdoc_validate() should always set err\n"); todo_wine ok(IXMLDOMParseError_get_reason(err, &bstr) == S_OK, "got error: %s\n", wine_dbgstr_w(bstr)); SysFreeString(bstr); IXMLDOMParseError_Release(err); /* load the next doc */ IXMLDOMDocument2_Release(doc); doc = create_document(&IID_IXMLDOMDocument2); hr = IXMLDOMDocument2_loadXML(doc, _bstr_(szOpenSeqXML4), &b); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(b == VARIANT_TRUE, "failed to load XML string\n"); /* associate the cache to the doc */ V_VT(&v) = VT_DISPATCH; V_DISPATCH(&v) = NULL; hr = IXMLDOMSchemaCollection_QueryInterface(cache, &IID_IDispatch, (void**)&V_DISPATCH(&v)); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(V_DISPATCH(&v) != NULL, "failed to get IDispatch interface\n"); hr = IXMLDOMDocument2_putref_schemas(doc, v); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); VariantClear(&v); /* validate the doc * fails, undeclared elements are not allowed */ err = NULL; bstr = NULL; hr = IXMLDOMDocument2_validate(doc, &err); ok(hr == S_FALSE, "Unexpected hr %#lx.\n", hr); ok(err != NULL, "domdoc_validate() should always set err\n"); todo_wine ok(IXMLDOMParseError_get_reason(err, &bstr) == S_OK, "got error: %s\n", wine_dbgstr_w(bstr)); SysFreeString(bstr); IXMLDOMParseError_Release(err); IXMLDOMDocument2_Release(doc); IXMLDOMDocument2_Release(schema); IXMLDOMSchemaCollection_Release(cache); free_bstrs(); } typedef struct { const WCHAR *query; enum VARENUM type_schema; const WCHAR *typename; BOOL todo; } xdr_datatypes; static const xdr_datatypes xdr_datatypes_data[] = { { L"//Property[Name!text()='testBase64']/Value/base64Data", VT_ARRAY|VT_UI1, L"bin.base64" }, { L"//Property[Name!text()='testHex']/Value/hexData", VT_ARRAY|VT_UI1, L"bin.hex" }, { L"//Property[Name!text()='testBool']/Value/boolData", VT_BOOL, L"boolean" }, { L"//Property[Name!text()='testChar']/Value/charData", VT_I4, L"char", TRUE }, { L"//Property[Name!text()='testDate']/Value/dateData", VT_DATE, L"date" }, { L"//Property[Name!text()='testDateTime']/Value/dateTimeData", VT_DATE, L"dateTime" }, { L"//Property[Name!text()='testDateTimeTz']/Value/dateTimeTzData", VT_DATE, L"dateTime.tz" }, { L"//Property[Name!text()='testFixed']/Value/fixedData", VT_CY, L"fixed.14.4" }, { L"//Property[Name!text()='testFloat']/Value/floatData", VT_R8, L"float" }, { L"//Property[Name!text()='testI1']/Value/i1Data", VT_I1, L"i1" }, { L"//Property[Name!text()='testI2']/Value/i2Data", VT_I2, L"i2" }, { L"//Property[Name!text()='testI4']/Value/i4Data", VT_I4, L"i4" }, { L"//Property[Name!text()='testI8']/Value/i8Data", VT_NULL, L"i8", TRUE }, { L"//Property[Name!text()='testInt']/Value/intData", VT_I4, L"int" }, { L"//Property[Name!text()='testNmtoken']/Value/nmtokData", VT_BSTR, NULL }, { L"//Property[Name!text()='testNmtokens']/Value/nmtoksData", VT_BSTR, NULL }, { L"//Property[Name!text()='testNumber']/Value/numData", VT_BSTR, L"number" }, { L"//Property[Name!text()='testR4']/Value/r4Data", VT_R4, L"r4" }, { L"//Property[Name!text()='testR8']/Value/r8Data", VT_R8, L"r8" }, { L"//Property[Name!text()='testString']/Value/stringData", VT_BSTR, NULL }, { L"//Property[Name!text()='testTime']/Value/timeData", VT_DATE, L"time" }, { L"//Property[Name!text()='testTimeTz']/Value/timeTzData", VT_DATE, L"time.tz" }, { L"//Property[Name!text()='testU1']/Value/u1Data", VT_UI1, L"ui1" }, { L"//Property[Name!text()='testU2']/Value/u2Data", VT_UI2, L"ui2" }, { L"//Property[Name!text()='testU4']/Value/u4Data", VT_UI4, L"ui4" }, { L"//Property[Name!text()='testU8']/Value/u8Data", VT_NULL, L"ui8", TRUE }, { L"//Property[Name!text()='testURI']/Value/uriData", VT_BSTR, L"uri" }, { L"//Property[Name!text()='testUUID']/Value/uuidData", VT_BSTR, L"uuid" }, { NULL } }; static void test_XDR_datatypes(void) { IXMLDOMDocument2 *doc, *schema, *doc2; IXMLDOMSchemaCollection* cache; const xdr_datatypes *ptr; IXMLDOMParseError* err; VARIANT_BOOL b; HRESULT hr; VARIANT v; BSTR bstr; LONG l; VariantInit(&v); doc = create_document(&IID_IXMLDOMDocument2); doc2 = create_document(&IID_IXMLDOMDocument2); schema = create_document(&IID_IXMLDOMDocument2); cache = create_cache(&IID_IXMLDOMSchemaCollection); if (!doc || !doc2 || !schema || !cache) { if (doc) IXMLDOMDocument2_Release(doc); if (doc2) IXMLDOMDocument2_Release(doc2); if (schema) IXMLDOMDocument2_Release(schema); if (cache) IXMLDOMSchemaCollection_Release(cache); return; } hr = IXMLDOMDocument2_loadXML(doc, _bstr_(szDatatypeXML), &b); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(b == VARIANT_TRUE, "failed to load XML string\n"); hr = IXMLDOMDocument2_loadXML(doc2, _bstr_(szDatatypeXML), &b); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(b == VARIANT_TRUE, "failed to load XML string\n"); hr = IXMLDOMDocument2_loadXML(schema, _bstr_(szDatatypeXDR), &b); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(b == VARIANT_TRUE, "failed to load XML string\n"); err = NULL; hr = IXMLDOMDocument2_validate(doc, &err); ok(hr == S_FALSE, "Unexpected hr %#lx.\n", hr); ok(err != NULL, "domdoc_validate() should always set err\n"); hr = IXMLDOMParseError_get_errorCode(err, &l); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(l == E_XML_NODTD, "Unexpected error code %#lx.\n", l); IXMLDOMParseError_Release(err); err = NULL; hr = IXMLDOMDocument2_validate(doc2, &err); ok(hr == S_FALSE, "Unexpected hr %#lx.\n", hr); ok(err != NULL, "domdoc_validate() should always set err\n"); hr = IXMLDOMParseError_get_errorCode(err, &l); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(l == E_XML_NODTD, "Unexpected error code %#lx.\n", l); IXMLDOMParseError_Release(err); /* now load the schema */ V_VT(&v) = VT_DISPATCH; V_DISPATCH(&v) = NULL; hr = IXMLDOMDocument2_QueryInterface(schema, &IID_IDispatch, (void**)&V_DISPATCH(&v)); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(V_DISPATCH(&v) != NULL, "failed to get IDispatch interface\n"); hr = IXMLDOMSchemaCollection_add(cache, _bstr_(L"urn:x-schema:datatype-test-xdr"), v); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); VariantClear(&v); /* associate the cache to the doc */ V_VT(&v) = VT_DISPATCH; V_DISPATCH(&v) = NULL; hr = IXMLDOMSchemaCollection_QueryInterface(cache, &IID_IDispatch, (void**)&V_DISPATCH(&v)); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(V_DISPATCH(&v) != NULL, "failed to get IDispatch interface\n"); hr = IXMLDOMDocument2_putref_schemas(doc2, v); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); VariantClear(&v); /* validate the doc */ err = NULL; l = 0; bstr = NULL; hr = IXMLDOMDocument2_validate(doc2, &err); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(err != NULL, "domdoc_validate() should always set err\n"); hr = IXMLDOMParseError_get_errorCode(err, &l); ok(hr == S_FALSE, "Unexpected hr %#lx.\n", hr); hr = IXMLDOMParseError_get_reason(err, &bstr); ok(hr == S_FALSE, "Unexpected hr %#lx.\n", hr); ok(l == 0, "Unexpected value %lx : %s\n", l, wine_dbgstr_w(bstr)); SysFreeString(bstr); IXMLDOMParseError_Release(err); ptr = xdr_datatypes_data; while (ptr->query) { IXMLDOMNode* node = NULL; VARIANT type; /* check data types without the schema */ hr = IXMLDOMDocument2_selectSingleNode(doc, _bstr_(ptr->query), &node); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(node != NULL, "expected node\n"); V_VT(&type) = VT_EMPTY; V_BSTR(&type) = (void*)-1; hr = IXMLDOMNode_get_dataType(node, &type); ok(hr == S_FALSE, "Unexpected hr %#lx.\n", hr); ok(V_VT(&type) == VT_NULL, "got type %i\n", V_VT(&type)); /* when returning VT_NULL, the pointer is set to NULL */ ok(V_BSTR(&type) == NULL, "got %p\n", V_BSTR(&type)); VariantClear(&type); hr = IXMLDOMNode_get_nodeTypedValue(node, &type); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(V_VT(&type) == VT_BSTR, "got variant type %i\n", V_VT(&v)); VariantClear(&type); IXMLDOMNode_Release(node); /* check the data with schema */ node = NULL; hr = IXMLDOMDocument2_selectSingleNode(doc2, _bstr_(ptr->query), &node); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(node != NULL, "expected node\n"); V_VT(&type) = VT_EMPTY; hr = IXMLDOMNode_get_dataType(node, &type); if (ptr->typename) { ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(V_VT(&type) == VT_BSTR, "got type %i\n", V_VT(&type)); ok(!lstrcmpW(V_BSTR(&type), _bstr_(ptr->typename)), "got %s\n", wine_dbgstr_w(V_BSTR(&type))); } else { ok(hr == S_FALSE, "Unexpected hr %#lx.\n", hr); ok(V_VT(&type) == VT_NULL, "%s: got type %i\n", wine_dbgstr_w(ptr->query), V_VT(&type)); } VariantClear(&type); VariantClear(&v); hr = IXMLDOMNode_get_nodeTypedValue(node, &v); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); todo_wine_if(ptr->todo) ok(V_VT(&v) == ptr->type_schema, "%s: got variant type %i\n", wine_dbgstr_w(ptr->query), V_VT(&v)); switch (ptr->type_schema) { case VT_BOOL: ok(V_BOOL(&v) == VARIANT_TRUE, "got %x\n", V_BOOL(&v)); break; case VT_I1: ok(V_I1(&v) == 42, "got %i\n", V_I1(&v)); break; case VT_I2: ok(V_I2(&v) == 420, "got %i\n", V_I2(&v)); break; case VT_I4: if (!wcscmp(ptr->typename, L"int")) ok(V_I4(&v) == 42, "got %ld\n", V_I4(&v)); else if (!wcscmp(ptr->typename, L"char")) todo_wine ok(V_I4(&v) == 'u', "got %lx\n", V_I4(&v)); else ok(V_I4(&v) == -420000000, "got %ld\n", V_I4(&v)); break; case VT_I8: expect_int64(V_I8(&v), -4200000000, 10); break; case VT_R4: ok(V_R4(&v) == (float)3.14159265, "got %f\n", V_R4(&v)); break; case VT_R8: if (!wcscmp(ptr->typename, L"float")) ok(V_R8(&v) == 3.14159, "got %f\n", V_R8(&v)); else todo_wine ok(V_R8(&v) == 3.14159265358979323846, "got %.20f\n", V_R8(&v)); break; case VT_UI1: ok(V_UI1(&v) == 0xFF, "got %02x\n", V_UI1(&v)); break; case VT_UI2: ok(V_UI2(&v) == 0xFFFF, "got %04x\n", V_UI2(&v)); break; case VT_UI4: ok(V_UI4(&v) == 0xFFFFFFFF, "got %#lx\n", V_UI4(&v)); break; case VT_UI8: expect_uint64(V_UI8(&v), 0xFFFFFFFFFFFFFFFF, 16); break; default: ; } VariantClear(&v); IXMLDOMNode_Release(node); ptr++; } IXMLDOMDocument2_Release(schema); IXMLDOMDocument2_Release(doc); IXMLDOMDocument2_Release(doc2); IXMLDOMSchemaCollection_Release(cache); free_bstrs(); } static void test_validate_on_load(void) { IXMLDOMSchemaCollection2 *cache; VARIANT_BOOL b; HRESULT hr; cache = create_cache_version(40, &IID_IXMLDOMSchemaCollection2); if (!cache) return; hr = IXMLDOMSchemaCollection2_get_validateOnLoad(cache, NULL); ok(hr == E_POINTER, "Unexpected hr %#lx.\n", hr); b = VARIANT_FALSE; hr = IXMLDOMSchemaCollection2_get_validateOnLoad(cache, &b); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(b == VARIANT_TRUE, "got %d\n", b); IXMLDOMSchemaCollection2_Release(cache); } static void test_obj_dispex(IUnknown *obj) { DISPID dispid = DISPID_SAX_XMLREADER_GETFEATURE; IDispatchEx *dispex; IUnknown *unk; DWORD props; UINT ticnt; HRESULT hr; BSTR name; hr = IUnknown_QueryInterface(obj, &IID_IDispatchEx, (void**)&dispex); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); if (FAILED(hr)) return; ticnt = 0; hr = IDispatchEx_GetTypeInfoCount(dispex, &ticnt); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(ticnt == 1, "ticnt=%u\n", ticnt); name = SysAllocString(L"*"); hr = IDispatchEx_DeleteMemberByName(dispex, name, fdexNameCaseSensitive); ok(hr == E_NOTIMPL, "Unexpected hr %#lx.\n", hr); SysFreeString(name); hr = IDispatchEx_DeleteMemberByDispID(dispex, dispid); ok(hr == E_NOTIMPL, "Unexpected hr %#lx.\n", hr); props = 0; hr = IDispatchEx_GetMemberProperties(dispex, dispid, grfdexPropCanAll, &props); ok(hr == E_NOTIMPL, "Unexpected hr %#lx.\n", hr); ok(props == 0, "expected 0 got %ld\n", props); hr = IDispatchEx_GetMemberName(dispex, dispid, &name); ok(hr == E_NOTIMPL, "Unexpected hr %#lx.\n", hr); if (SUCCEEDED(hr)) SysFreeString(name); hr = IDispatchEx_GetNextDispID(dispex, fdexEnumDefault, DISPID_XMLDOM_SCHEMACOLLECTION_ADD, &dispid); ok(hr == E_NOTIMPL, "Unexpected hr %#lx.\n", hr); unk = (IUnknown*)0xdeadbeef; hr = IDispatchEx_GetNameSpaceParent(dispex, &unk); ok(hr == E_NOTIMPL, "Unexpected hr %#lx.\n", hr); ok(unk == (IUnknown*)0xdeadbeef, "got %p\n", unk); name = SysAllocString(L"testprop"); hr = IDispatchEx_GetDispID(dispex, name, fdexNameEnsure, &dispid); ok(hr == DISP_E_UNKNOWNNAME, "Unexpected hr %#lx.\n", hr); SysFreeString(name); IDispatchEx_Release(dispex); } static void test_dispex(void) { IXMLDOMSchemaCollection *cache; IDispatchEx *dispex; IUnknown *unk; HRESULT hr; DISPPARAMS dispparams; VARIANT arg, ret; cache = create_cache(&IID_IXMLDOMSchemaCollection); if (!cache) return; hr = IXMLDOMSchemaCollection_QueryInterface(cache, &IID_IUnknown, (void**)&unk); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); test_obj_dispex(unk); IUnknown_Release(unk); hr = IXMLDOMSchemaCollection_QueryInterface(cache, &IID_IDispatchEx, (void**)&dispex); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); V_VT(&arg) = VT_I4; V_I4(&arg) = 0; dispparams.cArgs = 1; dispparams.cNamedArgs = 0; dispparams.rgdispidNamedArgs = NULL; dispparams.rgvarg = &arg; V_VT(&ret) = VT_EMPTY; V_DISPATCH(&ret) = (void*)0x1; hr = IDispatchEx_Invoke(dispex, DISPID_VALUE, &IID_NULL, 0, DISPATCH_METHOD, &dispparams, &ret, NULL, NULL); ok(hr == DISP_E_MEMBERNOTFOUND, "Unexpected hr %#lx.\n", hr); ok(V_VT(&ret) == VT_EMPTY, "got %d\n", V_VT(&ret)); ok(V_DISPATCH(&ret) == (void*)0x1, "got %p\n", V_DISPATCH(&ret)); IDispatchEx_Release(dispex); IXMLDOMSchemaCollection_Release(cache); cache = create_cache_version(60, &IID_IXMLDOMSchemaCollection); if (cache) { test_obj_dispex((IUnknown*)cache); IXMLDOMSchemaCollection_Release(cache); } } static void test_get(void) { IXMLDOMSchemaCollection2 *cache; IXMLDOMNode *node; HRESULT hr; cache = create_cache_version(60, &IID_IXMLDOMSchemaCollection2); if (!cache) return; hr = IXMLDOMSchemaCollection2_get(cache, NULL, NULL); ok(hr == E_NOTIMPL || hr == E_POINTER /* win8 */, "Unexpected hr %#lx.\n", hr); hr = IXMLDOMSchemaCollection2_get(cache, _bstr_(L"uri"), &node); ok(hr == E_NOTIMPL, "Unexpected hr %#lx.\n", hr); IXMLDOMSchemaCollection2_Release(cache); cache = create_cache_version(40, &IID_IXMLDOMSchemaCollection2); if (!cache) return; hr = IXMLDOMSchemaCollection2_get(cache, NULL, NULL); ok(hr == E_POINTER, "Unexpected hr %#lx.\n", hr); hr = IXMLDOMSchemaCollection2_get(cache, _bstr_(L"uri"), &node); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); IXMLDOMSchemaCollection2_Release(cache); free_bstrs(); } static void test_remove(void) { IXMLDOMSchemaCollection2 *cache; IXMLDOMDocument *doc; VARIANT_BOOL b; HRESULT hr; VARIANT v; LONG len; cache = create_cache_version(60, &IID_IXMLDOMSchemaCollection2); if (!cache) return; doc = create_document_version(60, &IID_IXMLDOMDocument); ok(doc != NULL, "got %p\n", doc); hr = IXMLDOMDocument_loadXML(doc, _bstr_(xsd_schema1_xml), &b); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); V_VT(&v) = VT_DISPATCH; V_DISPATCH(&v) = (IDispatch*)doc; hr = IXMLDOMSchemaCollection2_add(cache, _bstr_(xsd_schema1_uri), v); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); len = -1; hr = IXMLDOMSchemaCollection2_get_length(cache, &len); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(len == 1, "Unexpected length %ld.\n", len); /* ::remove() is a stub for version 6 */ hr = IXMLDOMSchemaCollection2_remove(cache, NULL); ok(hr == E_NOTIMPL, "Unexpected hr %#lx.\n", hr); hr = IXMLDOMSchemaCollection2_remove(cache, _bstr_(L"invaliduri")); ok(hr == E_NOTIMPL, "Unexpected hr %#lx.\n", hr); hr = IXMLDOMSchemaCollection2_remove(cache, _bstr_(xsd_schema1_uri)); ok(hr == E_NOTIMPL, "Unexpected hr %#lx.\n", hr); len = -1; hr = IXMLDOMSchemaCollection2_get_length(cache, &len); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(len == 1, "Unexpected length %ld.\n", len); IXMLDOMDocument_Release(doc); IXMLDOMSchemaCollection2_Release(cache); free_bstrs(); /* ::remove() works for version 4 */ cache = create_cache_version(40, &IID_IXMLDOMSchemaCollection2); if (!cache) return; doc = create_document_version(40, &IID_IXMLDOMDocument); ok(doc != NULL, "got %p\n", doc); hr = IXMLDOMDocument_loadXML(doc, _bstr_(xsd_schema1_xml), &b); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); V_VT(&v) = VT_DISPATCH; V_DISPATCH(&v) = (IDispatch*)doc; hr = IXMLDOMSchemaCollection2_add(cache, _bstr_(xsd_schema1_uri), v); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); len = -1; hr = IXMLDOMSchemaCollection2_get_length(cache, &len); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(len == 1, "Unexpected length %ld.\n", len); hr = IXMLDOMSchemaCollection2_remove(cache, NULL); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); hr = IXMLDOMSchemaCollection2_remove(cache, _bstr_(L"invaliduri")); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); len = -1; hr = IXMLDOMSchemaCollection2_get_length(cache, &len); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(len == 1, "Unexpected length %ld.\n", len); hr = IXMLDOMSchemaCollection2_remove(cache, _bstr_(xsd_schema1_uri)); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); len = -1; hr = IXMLDOMSchemaCollection2_get_length(cache, &len); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(len == 0, "Unexpected length %ld.\n", len); IXMLDOMDocument_Release(doc); IXMLDOMSchemaCollection2_Release(cache); free_bstrs(); } static void test_ifaces(void) { IXMLDOMSchemaCollection2 *cache; IUnknown *unk; HRESULT hr; cache = create_cache_version(60, &IID_IXMLDOMSchemaCollection2); if (!cache) return; /* CLSID_XMLSchemaCache60 is returned as an interface (the same as IXMLDOMSchemaCollection2). */ hr = IXMLDOMSchemaCollection2_QueryInterface(cache, &CLSID_XMLSchemaCache60, (void**)&unk); ok(hr == S_OK, "Unexpected hr %#lx.\n", hr); ok(unk == (IUnknown*)cache, "unk != cache\n"); IUnknown_Release(unk); check_interface(cache, &IID_IXMLDOMSchemaCollection, TRUE); check_interface(cache, &IID_IXMLDOMSchemaCollection2, TRUE); check_interface(cache, &IID_IDispatch, TRUE); check_interface(cache, &IID_IDispatchEx, TRUE); IXMLDOMSchemaCollection2_Release(cache); } START_TEST(schema) { HRESULT r; r = CoInitialize( NULL ); ok( r == S_OK, "failed to init com\n"); test_schema_refs(); test_collection_refs(); test_length(); test_collection_content(); test_regex(); test_XDR_schemas(); test_XDR_datatypes(); test_validate_on_load(); test_dispex(); test_get(); test_remove(); test_ifaces(); CoUninitialize(); }