Последняя версия Hippo Mocks (в своем репозитории Git), похоже, добавил поддержку интерфейсов COM. Я пытался издеваться над объектом подключения ADO; что потребовало некоторой настройки Hippo Mocks для правильной сборки (кажется, что COM-версия кода не была обновлена для изменений в остальных Hippo Mocks). У меня это сейчас строится, но следующий тест не пройден:
MockRepository mocks;
auto pConn = mocks.Mock<ADONS::_Connection>();
mocks.OnCall(pConn, ADONS::_Connection::AddRef).Return(1);
ADONS::_ConnectionPtr conn = pConn;
Самое первое, что делает умный указатель, это AddRef
интерфейс. Мой макет не должен заботиться о подсчете ссылок, поэтому я добавляю ожидание вызова, которое просто возвращает 1
, Однако, как только AddRef
вызывается, HippoMocks::NotImplementedException
брошен
Кто-нибудь имел успех с насмешкой над интерфейсом COM с помощью Hippo Mocks?
У меня была такая же проблема, и я решил ее.
Актуальная версия гиппопотамов теперь опубликована на github:
https://github.com/dascandy/hippomocks
Большое спасибо за предоставленные ссылки, которые помогли найти идею для исправления.
ОБНОВЛЕНИЕ, Детали о моей реализации и добавленной поддержке COM.
Сначала я заставил это работать, что демонстрирует следующий тест
class ICom
{
public:
virtual ~ICom() {}
virtual long __stdcall A(void) = 0;
virtual long __stdcall B(int) = 0;
virtual long __stdcall C(int, int) = 0;
...
};TEST(checkStdCallBase)
{
MockRepository mocks;
ICom* ic = mocks.Mock<ICom>();
mocks.ExpectCall(ic, ICom::A)
.Return(1);
long actual = ic->A();
EQUALS(1, actual);
}
Чтобы заставить его работать, мне пришлось исправить несколько мест в hippomocks.h, наиболее важном из которых является метод virtual_function_index. Коррекция обеспечивает правильное вычисление адреса для вызова на интерфейсе.
Во-вторых, я добавил несколько общих помощников настройки для COM-объектов, обеспечивающих стандартное поведение для AddRef, Release и QueryInterface.
Тесты показывают, как его использовать:
MIDL_INTERFACE("4745C05E-23E6-4c6d-B9F2-E483359A8B89")
COMInterface1 : public IUnknown
{
public:
virtual HRESULT STDMETHODCALLTYPE getTObjectCount(
/* [out] */ unsigned long *pCount) = 0;
};
typedef GUID ESTypeID;
MIDL_INTERFACE("356D44D9-980A-4149-A586-C5CB8B191437")
COMInterface2 : public IUnknown
{
public:
virtual HRESULT STDMETHODCALLTYPE getMappablePackages(
/* [out] */ long *pSize,
/* [size_is][size_is][out] */ ESTypeID **pIdList) = 0;
};
TEST(CheckThat_AddCommExpectations_Stubs_QueryInterface_AddRef_Release)
{
MockRepository mocks;
COMInterface1* deviceMock = mocks.Mock<COMInterface1>();
AddComExpectations(mocks, deviceMock);
{
CComPtr<IUnknown> pUnk = deviceMock;
CComQIPtr<COMInterface1> pDevice = pUnk;
CHECK(pDevice == pUnk);
IUnknown* p = NULL;
pDevice->QueryInterface(__uuidof(IUnknown), (void**)&p);
CHECK(p == deviceMock);
}
}
TEST(CheckThat_ConnectComInterfaces_Stubs_QueryInterface_ToEachOther)
{
MockRepository mocks;
COMInterface1* deviceMock = mocks.Mock<COMInterface1>();
COMInterface2* devMappingMock = mocks.Mock<COMInterface2>();
ConnectComInterfaces(mocks, deviceMock, devMappingMock);
{
//Com objects can reach each other
CComQIPtr<COMInterface2> pDevMapping = deviceMock;
CHECK(pDevMapping != NULL);
CHECK(pDevMapping == devMappingMock);
CComQIPtr<COMInterface1> pDevNavigate = devMappingMock;
CHECK(pDevNavigate != NULL);
CHECK(pDevNavigate == deviceMock);
}
}
Вспомогательные методы AddComExpectations и ConnectComInterfaces предоставляются в отдельном заголовке «comsupport.h». Заголовок является надстройкой для Hippomocks:
template <typename T>
void AddComExpectations(HM_NS MockRepository& mocks, T* m)
{
mocks.OnCall(m, T::AddRef)
.Return(1);
mocks.OnCall(m, T::Release)
.Return(1);
mocks.OnCall(m, T::QueryInterface)
.With(__uuidof(T), Out((void**)m))
.Return(S_OK);
mocks.OnCall(m, T::QueryInterface)
.With(__uuidof(IUnknown), Out((void**)m))
.Return(S_OK);
}
template <typename T1, typename T2>
void ConnectComInterfaces(HM_NS MockRepository& mocks, T1* m1, T2* m2)
{
//from T1 to T2
mocks.OnCall(m1, T1::QueryInterface)
.With(__uuidof(T2), Out((void**)m2))
.Return(S_OK);
//from T2 to T1
mocks.OnCall(m2, T2::QueryInterface)
.With(__uuidof(T1), Out((void**)m1))
.Return(S_OK);
AddComExpectations(mocks, m1);
AddComExpectations(mocks, m2);
//no support for interface hierarchies
//no Base IUnknown -> do it yourself if you really need that special case
}
Других решений пока нет …