визуальные HippoMocks — макет COM-интерфейса в C ++?

Последняя версия 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?

1

Решение

У меня была такая же проблема, и я решил ее.
Актуальная версия гиппопотамов теперь опубликована на 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
}
2

Другие решения

Других решений пока нет …

По вопросам рекламы [email protected]