/* OLEDB Database tests * * Copyright 2012 Alistair Leslie-Hughes * * 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 #include #define COBJMACROS #define CONST_VTABLE #define DBINITCONSTANTS #include "windef.h" #include "winbase.h" #include "ole2.h" #include "msdadc.h" #include "msdasc.h" #include "msdaguid.h" #include "initguid.h" #include "oledb.h" #include "oledberr.h" #include "msdasql.h" #include "wine/test.h" DEFINE_GUID(CSLID_MSDAER, 0xc8b522cf,0x5cf3,0x11ce,0xad,0xe5,0x00,0xaa,0x00,0x44,0x77,0x3d); static WCHAR initstring_default[] = {'D','a','t','a',' ','S','o','u','r','c','e','=','d','u','m','m','y',';',0}; #define EXPECT_REF(obj,ref) _expect_ref((IUnknown*)obj, ref, __LINE__) static void _expect_ref(IUnknown* obj, ULONG ref, int line) { ULONG rc; IUnknown_AddRef(obj); rc = IUnknown_Release(obj); ok_(__FILE__, line)(rc == ref, "expected refcount %d, got %d\n", ref, rc); } static void test_GetDataSource(WCHAR *initstring) { IDataInitialize *datainit = NULL; IDBInitialize *dbinit = NULL; HRESULT hr; trace("Data Source: %s\n", wine_dbgstr_w(initstring)); hr = CoCreateInstance(&CLSID_MSDAINITIALIZE, NULL, CLSCTX_INPROC_SERVER, &IID_IDataInitialize,(void**)&datainit); ok(hr == S_OK, "got %08x\n", hr); EXPECT_REF(datainit, 1); /* a failure to create data source here may indicate provider is simply not present */ hr = IDataInitialize_GetDataSource(datainit, NULL, CLSCTX_INPROC_SERVER, initstring, &IID_IDBInitialize, (IUnknown**)&dbinit); if(SUCCEEDED(hr)) { IDBProperties *props = NULL; EXPECT_REF(datainit, 1); EXPECT_REF(dbinit, 1); hr = IDBInitialize_QueryInterface(dbinit, &IID_IDBProperties, (void**)&props); ok(hr == S_OK, "got %08x\n", hr); if(SUCCEEDED(hr)) { ULONG cnt; DBPROPINFOSET *pInfoset; OLECHAR *ary; EXPECT_REF(dbinit, 2); EXPECT_REF(props, 2); hr = IDBProperties_GetPropertyInfo(props, 0, NULL, &cnt, &pInfoset, &ary); todo_wine ok(hr == S_OK, "got %08x\n", hr); if(hr == S_OK) { ULONG i; for(i =0; i < pInfoset->cPropertyInfos; i++) { trace("(0x%04x) '%s' %d\n", pInfoset->rgPropertyInfos[i].dwPropertyID, wine_dbgstr_w(pInfoset->rgPropertyInfos[i].pwszDescription), pInfoset->rgPropertyInfos[i].vtType); } CoTaskMemFree(pInfoset); CoTaskMemFree(ary); } IDBProperties_Release(props); } EXPECT_REF(dbinit, 1); IDBInitialize_Release(dbinit); } EXPECT_REF(datainit, 1); IDataInitialize_Release(datainit); } /* IDBProperties stub */ static HRESULT WINAPI dbprops_QI(IDBProperties *iface, REFIID riid, void **obj) { if (IsEqualIID(riid, &IID_IDBProperties) || IsEqualIID(riid, &IID_IUnknown)) { *obj = iface; IDBProperties_AddRef(iface); return S_OK; } *obj = NULL; return E_NOINTERFACE; } static ULONG WINAPI dbprops_AddRef(IDBProperties *iface) { return 2; } static ULONG WINAPI dbprops_Release(IDBProperties *iface) { return 1; } static HRESULT WINAPI dbprops_GetProperties(IDBProperties *iface, ULONG cPropertyIDSets, const DBPROPIDSET rgPropertyIDSets[], ULONG *pcPropertySets, DBPROPSET **prgPropertySets) { ok(0, "unexpected call\n"); return E_NOTIMPL; } static HRESULT WINAPI dbprops_GetPropertyInfo(IDBProperties *iface, ULONG cPropertyIDSets, const DBPROPIDSET rgPropertyIDSets[], ULONG *pcPropertyInfoSets, DBPROPINFOSET **prgPropertyInfoSets, OLECHAR **ppDescBuffer) { ok(0, "unexpected call\n"); return E_NOTIMPL; } static HRESULT WINAPI dbprops_SetProperties(IDBProperties *iface, ULONG set_count, DBPROPSET propsets[]) { ok(set_count == 1, "got %u\n", set_count); ok(IsEqualIID(&propsets->guidPropertySet, &DBPROPSET_DBINIT), "set guid %s\n", wine_dbgstr_guid(&propsets->guidPropertySet)); ok(propsets->cProperties == 2, "got propcount %u\n", propsets->cProperties); if (propsets->cProperties == 2) { ok(propsets->rgProperties[0].dwPropertyID == DBPROP_INIT_DATASOURCE, "got propid[0] %u\n", propsets->rgProperties[0].dwPropertyID); ok(propsets->rgProperties[0].dwOptions == DBPROPOPTIONS_REQUIRED, "got options[0] %u\n", propsets->rgProperties[0].dwOptions); ok(propsets->rgProperties[0].dwStatus == 0, "got status[0] %u\n", propsets->rgProperties[0].dwStatus); ok(V_VT(&propsets->rgProperties[0].vValue) == VT_BSTR, "got vartype[0] %u\n", V_VT(&propsets->rgProperties[0].vValue)); ok(propsets->rgProperties[1].dwPropertyID == DBPROP_INIT_PROVIDERSTRING, "got propid[1] %u\n", propsets->rgProperties[1].dwPropertyID); ok(propsets->rgProperties[1].dwOptions == DBPROPOPTIONS_REQUIRED, "got options[1] %u\n", propsets->rgProperties[1].dwOptions); ok(propsets->rgProperties[1].dwStatus == 0, "got status[1] %u\n", propsets->rgProperties[1].dwStatus); ok(V_VT(&propsets->rgProperties[1].vValue) == VT_BSTR, "got vartype[1] %u\n", V_VT(&propsets->rgProperties[1].vValue)); } return S_OK; } static const IDBPropertiesVtbl dbpropsvtbl = { dbprops_QI, dbprops_AddRef, dbprops_Release, dbprops_GetProperties, dbprops_GetPropertyInfo, dbprops_SetProperties }; static IDBProperties dbprops = { &dbpropsvtbl }; /* IPersist stub */ static HRESULT WINAPI dbpersist_QI(IPersist *iface, REFIID riid, void **obj) { if (IsEqualIID(riid, &IID_IPersist) || IsEqualIID(riid, &IID_IUnknown)) { *obj = iface; IPersist_AddRef(iface); return S_OK; } *obj = NULL; return E_NOINTERFACE; } static ULONG WINAPI dbpersist_AddRef(IPersist *iface) { return 2; } static ULONG WINAPI dbpersist_Release(IPersist *iface) { return 1; } static HRESULT WINAPI dbpersist_GetClassID(IPersist *iface, CLSID *clsid) { static const WCHAR msdasqlW[] = {'M','S','D','A','S','Q','L',0}; return CLSIDFromProgID(msdasqlW, clsid); } static const IPersistVtbl dbpersistvtbl = { dbpersist_QI, dbpersist_AddRef, dbpersist_Release, dbpersist_GetClassID }; static IPersist dbpersist = { &dbpersistvtbl }; /* IDBInitialize stub */ static HRESULT WINAPI dbinit_QI(IDBInitialize *iface, REFIID riid, void **obj) { if (IsEqualIID(riid, &IID_IDBInitialize) || IsEqualIID(riid, &IID_IUnknown)) { *obj = iface; IDBInitialize_AddRef(iface); return S_OK; } else if (IsEqualIID(riid, &IID_IPersist)) { *obj = &dbpersist; return S_OK; } else if (IsEqualIID(riid, &IID_IDBProperties)) { *obj = &dbprops; return S_OK; } *obj = NULL; return E_NOINTERFACE; } static ULONG WINAPI dbinit_AddRef(IDBInitialize *iface) { return 2; } static ULONG WINAPI dbinit_Release(IDBInitialize *iface) { return 1; } static HRESULT WINAPI dbinit_Initialize(IDBInitialize *iface) { ok(0, "unexpected call\n"); return E_NOTIMPL; } static HRESULT WINAPI dbinit_Uninitialize(IDBInitialize *iface) { ok(0, "unexpected call\n"); return E_NOTIMPL; } static const IDBInitializeVtbl dbinitvtbl = { dbinit_QI, dbinit_AddRef, dbinit_Release, dbinit_Initialize, dbinit_Uninitialize }; static IDBInitialize dbinittest = { &dbinitvtbl }; static void test_GetDataSource2(WCHAR *initstring) { IDataInitialize *datainit = NULL; IDBInitialize *dbinit = NULL; HRESULT hr; hr = CoCreateInstance(&CLSID_MSDAINITIALIZE, NULL, CLSCTX_INPROC_SERVER, &IID_IDataInitialize,(void**)&datainit); ok(hr == S_OK, "got %08x\n", hr); dbinit = &dbinittest; hr = IDataInitialize_GetDataSource(datainit, NULL, CLSCTX_INPROC_SERVER, initstring, &IID_IDBInitialize, (IUnknown**)&dbinit); ok(hr == S_OK, "got %08x\n", hr); IDataInitialize_Release(datainit); } static void test_database(void) { static WCHAR initstring_jet[] = {'P','r','o','v','i','d','e','r','=','M','i','c','r','o','s','o','f','t','.', 'J','e','t','.','O','L','E','D','B','.','4','.','0',';', 'D','a','t','a',' ','S','o','u','r','c','e','=','d','u','m','m','y',';', 'P','e','r','s','i','s','t',' ','S','e','c','u','r','i','t','y',' ','I','n','f','o','=','F','a','l','s','e',';',0}; static WCHAR initstring_lower[] = {'d','a','t','a',' ','s','o','u','r','c','e','=','d','u','m','m','y',';',0}; static WCHAR customprop[] = {'d','a','t','a',' ','s','o','u','r','c','e','=','d','u','m','m','y',';', 'c','u','s','t','o','m','p','r','o','p','=','1','2','3','.','4',';',0}; static WCHAR extended_prop[] = {'d','a','t','a',' ','s','o','u','r','c','e','=','d','u','m','m','y',';', 'E','x','t','e','n','d','e','d',' ','P','r','o','p','e','r','t','i','e','s','=','\"','D','R','I','V','E','R','=','A', ' ','W','i','n','e',' ','O','D','B','C',' ','d','r','i','v','e','r',';','U','I','D','=','w','i','n','e',';','\"',';',0}; static WCHAR extended_prop2[] = {'d','a','t','a',' ','s','o','u','r','c','e','=','\'','d','u','m','m','y','\'',';', 'c','u','s','t','o','m','p','r','o','p','=','\'','1','2','3','.','4','\'',';',0}; IDataInitialize *datainit = NULL; HRESULT hr; hr = CoCreateInstance(&CLSID_MSDAINITIALIZE, NULL, CLSCTX_INPROC_SERVER, &IID_IDataInitialize, (void**)&datainit); if (FAILED(hr)) { win_skip("Unable to load oledb library\n"); return; } IDataInitialize_Release(datainit); test_GetDataSource(NULL); test_GetDataSource(initstring_jet); test_GetDataSource(initstring_default); test_GetDataSource(initstring_lower); test_GetDataSource2(customprop); test_GetDataSource2(extended_prop); test_GetDataSource2(extended_prop2); } static void free_dispparams(DISPPARAMS *params) { unsigned int i; for (i = 0; i < params->cArgs && params->rgvarg; i++) VariantClear(¶ms->rgvarg[i]); CoTaskMemFree(params->rgvarg); CoTaskMemFree(params->rgdispidNamedArgs); } static void test_errorinfo(void) { ICreateErrorInfo *createerror; ERRORINFO info, info2, info3; IErrorInfo *errorinfo, *errorinfo2; IErrorRecords *errrecs; IUnknown *unk = NULL, *unk2; DISPPARAMS dispparams; DISPID dispid; DWORD context; ULONG cnt = 0; VARIANT arg; HRESULT hr; GUID guid; BSTR str; hr = CoCreateInstance(&CSLID_MSDAER, NULL, CLSCTX_INPROC_SERVER, &IID_IUnknown, (void**)&unk); ok(hr == S_OK, "got %08x\n", hr); hr = IUnknown_QueryInterface(unk, &IID_IErrorInfo, (void**)&errorinfo); ok(hr == S_OK, "got %08x\n", hr); hr = IErrorInfo_GetGUID(errorinfo, NULL); ok(hr == E_INVALIDARG, "got %08x\n", hr); hr = IErrorInfo_GetSource(errorinfo, NULL); ok(hr == E_INVALIDARG, "got %08x\n", hr); hr = IErrorInfo_GetDescription(errorinfo, NULL); ok(hr == E_INVALIDARG, "got %08x\n", hr); hr = IErrorInfo_GetHelpFile(errorinfo, NULL); ok(hr == E_INVALIDARG, "got %08x\n", hr); hr = IErrorInfo_GetHelpContext(errorinfo, NULL); ok(hr == E_INVALIDARG, "got %08x\n", hr); memset(&guid, 0xac, sizeof(guid)); hr = IErrorInfo_GetGUID(errorinfo, &guid); ok(hr == S_OK, "got %08x\n", hr); ok(IsEqualGUID(&guid, &GUID_NULL), "got wrong guid\n"); str = (BSTR)0x1; hr = IErrorInfo_GetSource(errorinfo, &str); ok(hr == E_FAIL, "got %08x\n", hr); ok(str == NULL, "got %s\n", wine_dbgstr_w(str)); str = (BSTR)0x1; hr = IErrorInfo_GetDescription(errorinfo, &str); ok(hr == E_FAIL, "got %08x\n", hr); ok(str == NULL, "got %s\n", wine_dbgstr_w(str)); str = (BSTR)0x1; hr = IErrorInfo_GetHelpFile(errorinfo, &str); ok(hr == E_FAIL, "got %08x\n", hr); ok(str == NULL, "got %s\n", wine_dbgstr_w(str)); context = 1; hr = IErrorInfo_GetHelpContext(errorinfo, &context); ok(hr == E_FAIL, "got %08x\n", hr); ok(context == 0, "got %d\n", context); IErrorInfo_Release(errorinfo); hr = IErrorInfo_QueryInterface(errorinfo, &IID_ICreateErrorInfo, (void**)&createerror); ok(hr == E_NOINTERFACE, "got %08x\n", hr); hr = IUnknown_QueryInterface(unk, &IID_IErrorRecords, (void**)&errrecs); ok(hr == S_OK, "got %08x\n", hr); hr = IErrorRecords_GetRecordCount(errrecs, &cnt); ok(hr == S_OK, "got %08x\n", hr); ok(cnt == 0, "Got unexpected record count %u\n", cnt); hr = IErrorRecords_GetBasicErrorInfo(errrecs, 0, &info3); ok(hr == DB_E_BADRECORDNUM, "got %08x\n", hr); hr = IErrorRecords_GetCustomErrorObject(errrecs, 0, &IID_IUnknown, &unk2); ok(hr == DB_E_BADRECORDNUM, "got %08x\n", hr); hr = IErrorRecords_GetErrorInfo(errrecs, 0, 0, &errorinfo2); ok(hr == DB_E_BADRECORDNUM, "got %08x\n", hr); hr = IErrorRecords_GetErrorParameters(errrecs, 0, &dispparams); ok(hr == DB_E_BADRECORDNUM, "got %08x\n", hr); memset(&info, 0, sizeof(ERRORINFO)); info.dwMinor = 1; memset(&info2, 0, sizeof(ERRORINFO)); info2.dwMinor = 2; memset(&info3, 0, sizeof(ERRORINFO)); hr = IErrorRecords_AddErrorRecord(errrecs, NULL, 268435456, NULL, NULL, 0); ok(hr == E_INVALIDARG, "got %08x\n", hr); hr = IErrorRecords_AddErrorRecord(errrecs, &info, 1, NULL, NULL, 0); ok(hr == S_OK, "got %08x\n", hr); hr = IErrorRecords_GetRecordCount(errrecs, &cnt); ok(hr == S_OK, "got %08x\n", hr); ok(cnt == 1, "expected 1 got %d\n", cnt); /* Record does not contain custom error object. */ unk2 = (void*)0xdeadbeef; hr = IErrorRecords_GetCustomErrorObject(errrecs, 0, &IID_IUnknown, &unk2); ok(hr == S_OK, "got %08x\n", hr); ok(unk2 == NULL, "Got custom object %p.\n", unk2); hr = IErrorRecords_AddErrorRecord(errrecs, &info2, 2, NULL, NULL, 0); ok(hr == S_OK, "got %08x\n", hr); hr = IErrorRecords_GetRecordCount(errrecs, &cnt); ok(hr == S_OK, "got %08x\n", hr); ok(cnt == 2, "expected 2 got %d\n", cnt); hr = IErrorRecords_GetBasicErrorInfo(errrecs, 0, NULL); ok(hr == E_INVALIDARG, "got %08x\n", hr); hr = IErrorRecords_GetBasicErrorInfo(errrecs, 100, &info3); ok(hr == DB_E_BADRECORDNUM, "got %08x\n", hr); hr = IErrorRecords_GetBasicErrorInfo(errrecs, 0, &info3); ok(hr == S_OK, "got %08x\n", hr); ok(info3.dwMinor == 2, "expected 2 got %d\n", info3.dwMinor); hr = IErrorRecords_GetErrorParameters(errrecs, 0, NULL); ok(hr == E_INVALIDARG, "got %08x\n", hr); memset(&dispparams, 0xcc, sizeof(dispparams)); hr = IErrorRecords_GetErrorParameters(errrecs, 0, &dispparams); ok(hr == S_OK, "got %08x\n", hr); ok(dispparams.rgvarg == NULL, "Got arguments %p\n", dispparams.rgvarg); ok(dispparams.rgdispidNamedArgs == NULL, "Got named arguments %p\n", dispparams.rgdispidNamedArgs); ok(dispparams.cArgs == 0, "Got argument count %u\n", dispparams.cArgs); ok(dispparams.cNamedArgs == 0, "Got named argument count %u\n", dispparams.cNamedArgs); V_VT(&arg) = VT_BSTR; V_BSTR(&arg) = SysAllocStringLen(NULL, 0); dispid = 0x123; dispparams.rgvarg = &arg; dispparams.cArgs = 1; dispparams.rgdispidNamedArgs = &dispid; dispparams.cNamedArgs = 1; hr = IErrorRecords_AddErrorRecord(errrecs, &info2, 0, &dispparams, NULL, 0); ok(hr == S_OK, "got %08x\n", hr); memset(&dispparams, 0, sizeof(dispparams)); hr = IErrorRecords_GetErrorParameters(errrecs, 0, &dispparams); ok(hr == S_OK, "got %08x\n", hr); ok(V_VT(&dispparams.rgvarg[0]) == VT_BSTR, "Got arg type %d\n", V_VT(&dispparams.rgvarg[0])); ok(V_BSTR(&dispparams.rgvarg[0]) != V_BSTR(&arg), "Got arg bstr %d\n", V_VT(&dispparams.rgvarg[0])); ok(dispparams.rgdispidNamedArgs[0] == 0x123, "Got named argument %d\n", dispparams.rgdispidNamedArgs[0]); ok(dispparams.cArgs == 1, "Got argument count %u\n", dispparams.cArgs); ok(dispparams.cNamedArgs == 1, "Got named argument count %u\n", dispparams.cNamedArgs); free_dispparams(&dispparams); VariantClear(&arg); IErrorRecords_Release(errrecs); IUnknown_Release(unk); } static void test_initializationstring(void) { static const WCHAR initstring_msdasql[] = {'P','r','o','v','i','d','e','r','=','M','S','D','A','S','Q','L','.','1',';', 'D','a','t','a',' ','S','o','u','r','c','e','=','d','u','m','m','y', 0}; static const WCHAR initstring_msdasql2[] = {'p','R','o','V','i','D','e','R','=','M','S','D','A','S','Q','L','.','1',';', 'D','a','t','a',' ','S','o','u','r','c','e','=','d','u','m','m','y', 0}; static const WCHAR initstring_sqloledb[] = {'P','r','o','v','i','d','e','r','=','S','Q','L','O','L','E','D','B','.','1',';', 'D','a','t','a',' ','S','o','u','r','c','e','=','d','u','m','m','y', 0}; static const WCHAR initstring_mode[] = {'P','r','o','v','i','d','e','r','=','M','S','D','A','S','Q','L','.','1',';', 'D','a','t','a',' ','S','o','u','r','c','e','=','d','u','m','m','y',';', 'M','o','d','e','=','i','n','v','a','l','i','d',0}; static const WCHAR initstring_mode2[] = {'P','r','o','v','i','d','e','r','=','M','S','D','A','S','Q','L','.','1',';', 'D','a','t','a',' ','S','o','u','r','c','e','=','d','u','m','m','y',';', 'M','o','d','e','=','W','r','i','t','e','R','e','a','d',0}; static const WCHAR initstring_mode3[] = {'P','r','o','v','i','d','e','r','=','M','S','D','A','S','Q','L','.','1',';', 'D','a','t','a',' ','S','o','u','r','c','e','=','d','u','m','m','y',';', 'M','o','d','e','=','R','e','a','d','W','R','I','T','E',0}; IDataInitialize *datainit = NULL; IDBInitialize *dbinit; HRESULT hr; WCHAR *initstring = NULL; hr = CoCreateInstance(&CLSID_MSDAINITIALIZE, NULL, CLSCTX_INPROC_SERVER, &IID_IDataInitialize,(void**)&datainit); ok(hr == S_OK, "got %08x\n", hr); if(SUCCEEDED(hr)) { EXPECT_REF(datainit, 1); dbinit = NULL; hr = IDataInitialize_GetDataSource(datainit, NULL, CLSCTX_INPROC_SERVER, initstring_default, &IID_IDBInitialize, (IUnknown**)&dbinit); if(SUCCEEDED(hr)) { EXPECT_REF(datainit, 1); EXPECT_REF(dbinit, 1); hr = IDataInitialize_GetInitializationString(datainit, (IUnknown*)dbinit, 0, &initstring); ok(hr == S_OK, "got %08x\n", hr); if(hr == S_OK) { trace("Init String: %s\n", wine_dbgstr_w(initstring)); todo_wine ok(!lstrcmpW(initstring_msdasql, initstring) || !lstrcmpW(initstring_sqloledb, initstring), "got %s\n", wine_dbgstr_w(initstring)); CoTaskMemFree(initstring); } IDBInitialize_Release(dbinit); /* mixed casing string */ dbinit = NULL; hr = IDataInitialize_GetDataSource(datainit, NULL, CLSCTX_INPROC_SERVER, (WCHAR*)initstring_msdasql2, &IID_IDBInitialize, (IUnknown**)&dbinit); ok(hr == S_OK, "got 0x%08x\n", hr); IDBInitialize_Release(dbinit); /* Invalid Mode value */ dbinit = NULL; hr = IDataInitialize_GetDataSource(datainit, NULL, CLSCTX_INPROC_SERVER, (WCHAR *)initstring_mode, &IID_IDBInitialize, (IUnknown **)&dbinit); ok(FAILED(hr), "got 0x%08x\n", hr); dbinit = NULL; hr = IDataInitialize_GetDataSource(datainit, NULL, CLSCTX_INPROC_SERVER, (WCHAR *)initstring_mode2, &IID_IDBInitialize, (IUnknown **)&dbinit); ok(FAILED(hr), "got 0x%08x\n", hr); dbinit = NULL; hr = IDataInitialize_GetDataSource(datainit, NULL, CLSCTX_INPROC_SERVER, (WCHAR *)initstring_mode3, &IID_IDBInitialize, (IUnknown **)&dbinit); ok(hr == S_OK, "got 0x%08x\n", hr); IDBInitialize_Release(dbinit); } else ok(dbinit == NULL, "got %p\n", dbinit); IDataInitialize_Release(datainit); } } static void test_rowposition(void) { IEnumConnectionPoints *enum_points; IConnectionPointContainer *cpc; IConnectionPoint *cp; IRowPosition *rowpos; HRESULT hr; IID iid; hr = CoCreateInstance(&CLSID_OLEDB_ROWPOSITIONLIBRARY, NULL, CLSCTX_INPROC_SERVER, &IID_IRowPosition, (void**)&rowpos); ok(hr == S_OK, "got %08x\n", hr); hr = IRowPosition_QueryInterface(rowpos, &IID_IConnectionPointContainer, (void**)&cpc); ok(hr == S_OK, "got 0x%08x\n", hr); hr = IConnectionPointContainer_EnumConnectionPoints(cpc, &enum_points); todo_wine ok(hr == S_OK, "got 0x%08x\n", hr); if (hr == S_OK) { hr = IEnumConnectionPoints_Next(enum_points, 1, &cp, NULL); ok(hr == S_OK, "got 0x%08x\n", hr); hr = IConnectionPoint_GetConnectionInterface(cp, &iid); ok(hr == S_OK, "got 0x%08x\n", hr); ok(IsEqualIID(&iid, &IID_IRowPositionChange), "got %s\n", wine_dbgstr_guid(&iid)); IConnectionPoint_Release(cp); hr = IEnumConnectionPoints_Next(enum_points, 1, &cp, NULL); ok(hr == S_FALSE, "got 0x%08x\n", hr); IEnumConnectionPoints_Release(enum_points); } IConnectionPointContainer_Release(cpc); IRowPosition_Release(rowpos); } typedef struct { IRowset IRowset_iface; IChapteredRowset IChapteredRowset_iface; } test_rset_t; static test_rset_t test_rset; static HRESULT WINAPI rset_QI(IRowset *iface, REFIID riid, void **obj) { if (IsEqualIID(riid, &IID_IUnknown) || IsEqualIID(riid, &IID_IRowset)) { *obj = &test_rset.IRowset_iface; return S_OK; } else if (IsEqualIID(riid, &IID_IChapteredRowset)) { *obj = &test_rset.IChapteredRowset_iface; return S_OK; } ok(0, "unexpected riid %s\n", wine_dbgstr_guid(riid)); return E_NOINTERFACE; } static ULONG WINAPI rset_AddRef(IRowset *iface) { return 2; } static ULONG WINAPI rset_Release(IRowset *iface) { return 1; } static HRESULT WINAPI rset_AddRefRows(IRowset *iface, DBCOUNTITEM cRows, const HROW rghRows[], DBREFCOUNT rgRefCounts[], DBROWSTATUS rgRowStatus[]) { trace("AddRefRows: %ld\n", rghRows[0]); return S_OK; } static HRESULT WINAPI rset_GetData(IRowset *iface, HROW hRow, HACCESSOR hAccessor, void *pData) { ok(0, "unexpected call\n"); return E_NOTIMPL; } static HRESULT WINAPI rset_GetNextRows(IRowset *iface, HCHAPTER hReserved, DBROWOFFSET lRowsOffset, DBROWCOUNT cRows, DBCOUNTITEM *pcRowObtained, HROW **prghRows) { ok(0, "unexpected call\n"); return E_NOTIMPL; } static HRESULT WINAPI rset_ReleaseRows(IRowset *iface, DBCOUNTITEM cRows, const HROW rghRows[], DBROWOPTIONS rgRowOptions[], DBREFCOUNT rgRefCounts[], DBROWSTATUS rgRowStatus[]) { return S_OK; } static HRESULT WINAPI rset_RestartPosition(IRowset *iface, HCHAPTER hReserved) { ok(0, "unexpected call\n"); return E_NOTIMPL; } static const IRowsetVtbl rset_vtbl = { rset_QI, rset_AddRef, rset_Release, rset_AddRefRows, rset_GetData, rset_GetNextRows, rset_ReleaseRows, rset_RestartPosition }; static HRESULT WINAPI chrset_QI(IChapteredRowset *iface, REFIID riid, void **obj) { return IRowset_QueryInterface(&test_rset.IRowset_iface, riid, obj); } static ULONG WINAPI chrset_AddRef(IChapteredRowset *iface) { return IRowset_AddRef(&test_rset.IRowset_iface); } static ULONG WINAPI chrset_Release(IChapteredRowset *iface) { return IRowset_Release(&test_rset.IRowset_iface); } static HRESULT WINAPI chrset_AddRefChapter(IChapteredRowset *iface, HCHAPTER chapter, DBREFCOUNT *refcount) { return S_OK; } static HRESULT WINAPI chrset_ReleaseChapter(IChapteredRowset *iface, HCHAPTER chapter, DBREFCOUNT *refcount) { return S_OK; } static const IChapteredRowsetVtbl chrset_vtbl = { chrset_QI, chrset_AddRef, chrset_Release, chrset_AddRefChapter, chrset_ReleaseChapter }; static void init_test_rset(void) { test_rset.IRowset_iface.lpVtbl = &rset_vtbl; test_rset.IChapteredRowset_iface.lpVtbl = &chrset_vtbl; } static void test_rowpos_initialize(void) { IRowPosition *rowpos; HRESULT hr; hr = CoCreateInstance(&CLSID_OLEDB_ROWPOSITIONLIBRARY, NULL, CLSCTX_INPROC_SERVER, &IID_IRowPosition, (void**)&rowpos); ok(hr == S_OK, "got %08x\n", hr); init_test_rset(); hr = IRowPosition_Initialize(rowpos, (IUnknown*)&test_rset.IRowset_iface); ok(hr == S_OK, "got %08x\n", hr); IRowPosition_Release(rowpos); } static HRESULT WINAPI onchange_QI(IRowPositionChange *iface, REFIID riid, void **obj) { if (IsEqualIID(riid, &IID_IUnknown) || IsEqualIID(riid, &IID_IRowPositionChange)) { *obj = iface; return S_OK; } return E_NOINTERFACE; } static ULONG WINAPI onchange_AddRef(IRowPositionChange *iface) { return 2; } static ULONG WINAPI onchange_Release(IRowPositionChange *iface) { return 1; } static HRESULT WINAPI onchange_OnRowPositionChange(IRowPositionChange *iface, DBREASON reason, DBEVENTPHASE phase, BOOL cant_deny) { trace("%d %d %d\n", reason, phase, cant_deny); return S_OK; } static const IRowPositionChangeVtbl onchange_vtbl = { onchange_QI, onchange_AddRef, onchange_Release, onchange_OnRowPositionChange }; static IRowPositionChange onchangesink = { &onchange_vtbl }; static void init_onchange_sink(IRowPosition *rowpos) { IConnectionPointContainer *cpc; IConnectionPoint *cp; DWORD cookie; HRESULT hr; hr = IRowPosition_QueryInterface(rowpos, &IID_IConnectionPointContainer, (void**)&cpc); ok(hr == S_OK, "got %08x\n", hr); hr = IConnectionPointContainer_FindConnectionPoint(cpc, &IID_IRowPositionChange, &cp); ok(hr == S_OK, "got %08x\n", hr); hr = IConnectionPoint_Advise(cp, (IUnknown*)&onchangesink, &cookie); ok(hr == S_OK, "got %08x\n", hr); IConnectionPoint_Release(cp); IConnectionPointContainer_Release(cpc); } static void test_rowpos_clearrowposition(void) { DBPOSITIONFLAGS flags; IRowPosition *rowpos; HCHAPTER chapter; IUnknown *unk; HRESULT hr; HROW row; hr = CoCreateInstance(&CLSID_OLEDB_ROWPOSITIONLIBRARY, NULL, CLSCTX_INPROC_SERVER, &IID_IRowPosition, (void**)&rowpos); ok(hr == S_OK, "got %08x\n", hr); hr = IRowPosition_ClearRowPosition(rowpos); ok(hr == E_UNEXPECTED, "got %08x\n", hr); hr = IRowPosition_GetRowset(rowpos, &IID_IStream, &unk); ok(hr == E_UNEXPECTED, "got %08x\n", hr); chapter = 1; row = 1; flags = DBPOSITION_OK; hr = IRowPosition_GetRowPosition(rowpos, &chapter, &row, &flags); ok(hr == E_UNEXPECTED, "got %08x\n", hr); ok(chapter == DB_NULL_HCHAPTER, "got %ld\n", chapter); ok(row == DB_NULL_HROW, "got %ld\n", row); ok(flags == DBPOSITION_NOROW, "got %d\n", flags); init_test_rset(); hr = IRowPosition_Initialize(rowpos, (IUnknown*)&test_rset.IRowset_iface); ok(hr == S_OK, "got %08x\n", hr); chapter = 1; row = 1; flags = DBPOSITION_OK; hr = IRowPosition_GetRowPosition(rowpos, &chapter, &row, &flags); ok(hr == S_OK, "got %08x\n", hr); ok(chapter == DB_NULL_HCHAPTER, "got %ld\n", chapter); ok(row == DB_NULL_HROW, "got %ld\n", row); ok(flags == DBPOSITION_NOROW, "got %d\n", flags); hr = IRowPosition_GetRowset(rowpos, &IID_IRowset, &unk); ok(hr == S_OK, "got %08x\n", hr); init_onchange_sink(rowpos); hr = IRowPosition_ClearRowPosition(rowpos); ok(hr == S_OK, "got %08x\n", hr); chapter = 1; row = 1; flags = DBPOSITION_OK; hr = IRowPosition_GetRowPosition(rowpos, &chapter, &row, &flags); ok(hr == S_OK, "got %08x\n", hr); ok(chapter == DB_NULL_HCHAPTER, "got %ld\n", chapter); ok(row == DB_NULL_HROW, "got %ld\n", row); ok(flags == DBPOSITION_NOROW, "got %d\n", flags); IRowPosition_Release(rowpos); } static void test_rowpos_setrowposition(void) { IRowPosition *rowpos; HRESULT hr; hr = CoCreateInstance(&CLSID_OLEDB_ROWPOSITIONLIBRARY, NULL, CLSCTX_INPROC_SERVER, &IID_IRowPosition, (void**)&rowpos); ok(hr == S_OK, "got %08x\n", hr); init_test_rset(); hr = IRowPosition_Initialize(rowpos, (IUnknown*)&test_rset.IRowset_iface); ok(hr == S_OK, "got %08x\n", hr); hr = IRowPosition_ClearRowPosition(rowpos); ok(hr == S_OK, "got %08x\n", hr); init_onchange_sink(rowpos); hr = IRowPosition_SetRowPosition(rowpos, 0x123, 0x456, DBPOSITION_OK); ok(hr == S_OK, "got %08x\n", hr); IRowPosition_Release(rowpos); } static void test_dslocator(void) { IDataSourceLocator *dslocator = NULL; HRESULT hr; hr = CoCreateInstance(&CLSID_DataLinks, NULL, CLSCTX_INPROC_SERVER, &IID_IDataSourceLocator,(void**)&dslocator); ok(hr == S_OK, "got %08x\n", hr); if(SUCCEEDED(hr)) { IDataInitialize *datainit, *datainit2; IRunnableObject *runable; IProvideClassInfo *info; IMarshal *marshal; IRpcOptions *opts; COMPATIBLE_LONG hwnd = 0; if (0) /* Crashes under Window 7 */ hr = IDataSourceLocator_get_hWnd(dslocator, NULL); hr = IDataSourceLocator_get_hWnd(dslocator, &hwnd); ok(hr == S_OK, "got %08x\n", hr); ok(hwnd == 0, "got %p\n", (HWND)hwnd); hwnd = 0xDEADBEEF; hr = IDataSourceLocator_get_hWnd(dslocator, &hwnd); ok(hr == S_OK, "got %08x\n", hr); ok(hwnd == 0, "got %p\n", (HWND)hwnd); hwnd = 0xDEADBEEF; hr = IDataSourceLocator_put_hWnd(dslocator, hwnd); ok(hr == S_OK, "got %08x\n", hr); hwnd = 0; hr = IDataSourceLocator_get_hWnd(dslocator, &hwnd); ok(hr == S_OK, "got %08x\n", hr); ok(hwnd == 0xDEADBEEF, "got %p\n", (HWND)hwnd); hwnd = 0; hr = IDataSourceLocator_put_hWnd(dslocator, hwnd); ok(hr == S_OK, "got %08x\n", hr); hwnd = 0xDEADBEEF; hr = IDataSourceLocator_get_hWnd(dslocator, &hwnd); ok(hr == S_OK, "got %08x\n", hr); ok(hwnd == 0, "got %p\n", (HWND)hwnd); hr = IDataSourceLocator_QueryInterface(dslocator, &IID_IDataInitialize, (void **)&datainit); ok(hr == S_OK, "got %08x\n", hr); hr = IDataSourceLocator_QueryInterface(dslocator, &IID_IDataInitialize, (void **)&datainit2); ok(hr == S_OK, "got %08x\n", hr); ok(datainit == datainit2, "Got %p, previous %p\n", datainit, datainit2); hr = IDataSourceLocator_QueryInterface(dslocator, &IID_IRunnableObject, (void **)&runable); ok(hr == E_NOINTERFACE, "got %08x\n", hr); hr = IDataSourceLocator_QueryInterface(dslocator, &IID_IMarshal, (void **)&marshal); ok(hr == E_NOINTERFACE, "got %08x\n", hr); hr = IDataSourceLocator_QueryInterface(dslocator, &IID_IProvideClassInfo, (void **)&info); ok(hr == E_NOINTERFACE, "got %08x\n", hr); hr = IDataSourceLocator_QueryInterface(dslocator, &IID_IRpcOptions, (void **)&opts); ok(hr == E_NOINTERFACE, "got %08x\n", hr); if (winetest_interactive) { IDispatch *disp = NULL; hr = IDataSourceLocator_PromptNew(dslocator, NULL); ok(hr == E_INVALIDARG, "got %08x\n", hr); hr = IDataSourceLocator_PromptNew(dslocator, &disp); if (hr == S_OK) { ok(disp != NULL, "got %08x\n", hr); IDispatch_Release(disp); } else { ok(hr == S_FALSE, "got %08x\n", hr); ok(!disp, "got %08x\n", hr); } } IDataInitialize_Release(datainit2); IDataInitialize_Release(datainit); IDataSourceLocator_Release(dslocator); } } static void test_odbc_provider(void) { HRESULT hr; IDBProperties *props; DBPROPIDSET propidset; ULONG infocount; DBPROPINFOSET *propinfoset; WCHAR *desc; DBPROPID properties[14] = { DBPROP_AUTH_PASSWORD, DBPROP_AUTH_PERSIST_SENSITIVE_AUTHINFO, DBPROP_AUTH_USERID, DBPROP_INIT_DATASOURCE, DBPROP_INIT_HWND, DBPROP_INIT_LOCATION, DBPROP_INIT_MODE, DBPROP_INIT_PROMPT, DBPROP_INIT_TIMEOUT, DBPROP_INIT_PROVIDERSTRING, DBPROP_INIT_LCID, DBPROP_INIT_CATALOG, DBPROP_INIT_OLEDBSERVICES, DBPROP_INIT_GENERALTIMEOUT }; hr = CoCreateInstance( &CLSID_MSDASQL, NULL, CLSCTX_ALL, &IID_IDBProperties, (void **)&props); todo_wine ok(hr == S_OK, "Failed to create object 0x%08x\n", hr); if (FAILED(hr)) { return; } propidset.rgPropertyIDs = NULL; propidset.cPropertyIDs = 0; propidset.guidPropertySet = DBPROPSET_DBINITALL; infocount = 0; hr = IDBProperties_GetPropertyInfo(props, 1, &propidset, &infocount, &propinfoset, &desc); ok(hr == S_OK, "got 0x%08x\n", hr); if (hr == S_OK) { ULONG i; DBPROPIDSET propidlist; ULONG propcnt; DBPROPSET *propset; ok(IsEqualGUID(&propinfoset->guidPropertySet, &DBPROPSET_DBINIT), "got %s\n", debugstr_guid(&propinfoset->guidPropertySet)); ok(propinfoset->cPropertyInfos == 14, "got %d\n", propinfoset->cPropertyInfos); propidlist.guidPropertySet = DBPROPSET_DBINIT; propidlist.cPropertyIDs = propinfoset->cPropertyInfos; propidlist.rgPropertyIDs = CoTaskMemAlloc(propinfoset->cPropertyInfos * sizeof(DBPROP)); for (i = 0; i < propinfoset->cPropertyInfos; i++) { ok(properties[i] == propinfoset->rgPropertyInfos[i].dwPropertyID, "%d, got %d\n", i, propinfoset->rgPropertyInfos[i].dwPropertyID); ok(propinfoset->rgPropertyInfos[i].vtType != VT_EMPTY, "%d, got %d\n", i, propinfoset->rgPropertyInfos[i].vtType); propidlist.rgPropertyIDs[i] = propinfoset->rgPropertyInfos[i].dwPropertyID; } for (i = 0; i < propinfoset->cPropertyInfos; i++) VariantClear(&propinfoset->rgPropertyInfos[i].vValues); CoTaskMemFree(propinfoset->rgPropertyInfos); CoTaskMemFree(propinfoset); hr = IDBProperties_GetProperties(props, 1, &propidlist, &propcnt, &propset); ok(hr == S_OK, "got 0x%08x\n", hr); ok(propidlist.cPropertyIDs == 14, "got %d\n", propinfoset->cPropertyInfos); for (i = 0; i < propidlist.cPropertyIDs; i++) { ok(properties[i] == propidlist.rgPropertyIDs[i], "%d, got %d\n", i, propidlist.rgPropertyIDs[i]); propidlist.rgPropertyIDs[i] = propinfoset->rgPropertyInfos[i].dwPropertyID; } CoTaskMemFree(propidlist.rgPropertyIDs); CoTaskMemFree(propset); } IDBProperties_Release(props); } static void test_odbc_enumerator(void) { HRESULT hr; ISourcesRowset *source; IRowset *rowset; hr = CoCreateInstance( &CLSID_MSDASQL_ENUMERATOR, NULL, CLSCTX_ALL, &IID_ISourcesRowset, (void **)&source); todo_wine ok(hr == S_OK, "Failed to create object 0x%08x\n", hr); if (FAILED(hr)) { return; } hr = ISourcesRowset_GetSourcesRowset(source, NULL, &IID_IRowset, 0, 0, (IUnknown**)&rowset); todo_wine ok(hr == S_OK, "Failed to create object 0x%08x\n", hr); if (hr == S_OK) { IAccessor *accessor; IRowsetInfo *info; hr = IRowset_QueryInterface(rowset, &IID_IAccessor, (void **)&accessor); ok(hr == S_OK, "got %08x\n", hr); if (hr == S_OK) IAccessor_Release(accessor); hr = IRowset_QueryInterface(rowset, &IID_IRowsetInfo, (void **)&info); ok(hr == S_OK, "got %08x\n", hr); if (hr == S_OK) IRowsetInfo_Release(info); IRowset_Release(rowset); } ISourcesRowset_Release(source); } START_TEST(database) { OleInitialize(NULL); test_database(); test_errorinfo(); test_initializationstring(); test_dslocator(); test_odbc_provider(); test_odbc_enumerator(); /* row position */ test_rowposition(); test_rowpos_initialize(); test_rowpos_clearrowposition(); test_rowpos_setrowposition(); OleUninitialize(); }