Skip to content

Commit

Permalink
Add auxiliary API for Proxy
Browse files Browse the repository at this point in the history
  • Loading branch information
Vitalii Arteev committed Oct 21, 2020
1 parent 63e4de6 commit 0b53b18
Show file tree
Hide file tree
Showing 5 changed files with 224 additions and 10 deletions.
1 change: 1 addition & 0 deletions src/ipc/ipc-common/CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,7 @@ facelift_add_library(FaceliftIPCCommonLib
IPCServiceAdapterBase.h
NewIPCServiceAdapterBase.h
IPCAttachedPropertyFactory.h
observer.h
HEADERS_NO_MOC
AppendDBUSSignatureFunction.h
ipc-serialization.h
Expand Down
6 changes: 0 additions & 6 deletions src/ipc/ipc-common/IPCProxyBase.h
Original file line number Diff line number Diff line change
Expand Up @@ -35,12 +35,6 @@
#include "IPCProxyBinderBase.h"


#if defined(FaceliftIPCCommonLib_LIBRARY)
# define FaceliftIPCCommonLib_EXPORT Q_DECL_EXPORT
#else
# define FaceliftIPCCommonLib_EXPORT Q_DECL_IMPORT
#endif

namespace facelift {

template<typename AdapterType>
Expand Down
116 changes: 116 additions & 0 deletions src/ipc/ipc-common/observer.h
Original file line number Diff line number Diff line change
@@ -0,0 +1,116 @@
/**********************************************************************
**
** Copyright (C) 2020 Luxoft Sweden AB
**
** This file is part of the FaceLift project
**
** Permission is hereby granted, freIPCServiceAdapterBasee of charge, to any person
** obtaining a copy of this software and associated documentation files
** (the "Software"), to deal in the Software without restriction,
** including without limitation the rights to use, copy, modify, merge,
** publish, distribute, sublicense, and/or sell copies of the Software,
** and to permit persons to whom the Software is furnished to do so,
** subject to the following conditions:
**
** The above copyright notice and this permission notice shall be
** included in all copies or substantial portions of the Software.
**
** THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
** EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
** MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
** NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
** BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
** ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
** CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
** SOFTWARE.
**
** SPDX-License-Identifier: MIT
**
**********************************************************************/

#pragma once

#include <memory>
#include <QVector>
#include <QObject>

namespace facelift {

class IObserver : public QObject
{
Q_OBJECT
public:
virtual void IsReadyObserver(const std::shared_ptr<QMetaObject::Connection> &connection) = 0;
};

class IsReadyObserver: public QObject
{
Q_OBJECT
QVector< QPointer<IObserver> > m_observers{};

public:
IsReadyObserver() {}

template<typename T, typename F>
void watch(T const *object, F signal)
{
QObject::connect(object, std::move(signal), this, &IsReadyObserver::readyChanged);
}

// Set observers
void setObservers(const QVector< QPointer<IObserver> > &observers) {
m_observers.clear();
for (auto observer: observers) {
Q_ASSERT (observer != nullptr);
m_observers.push_back(observer);
auto connection = std::make_shared<QMetaObject::Connection>();
*connection = QObject::connect(this, &IsReadyObserver::readyChanged, observer, [observer, connection](){
observer->IsReadyObserver( connection );
});
}
}

// Get observers
const QVector< QPointer<IObserver> > &getObservers() const {
return m_observers;
}

Q_SIGNAL void readyChanged();

void onReadyChanged() {
emit readyChanged();
}
};

// Single-time observer which will unregister itself when done
template<typename T>
class SingleTimeObserver : public IObserver
{
T m_function;

public:
explicit SingleTimeObserver(T function) : m_function{function} {}
~SingleTimeObserver() = default;

void IsReadyObserver(const std::shared_ptr<QMetaObject::Connection> &connection) override {
m_function();
QObject::disconnect(*connection);
}
};

// Standard observer which will work for each signal
template<typename T>
class StandartObserver : public IObserver
{
T m_function;

public:
explicit StandartObserver(T function) : m_function{function} {}
~StandartObserver() = default;

void IsReadyObserver(const std::shared_ptr<QMetaObject::Connection> &) override {
m_function();
}
};

}
24 changes: 20 additions & 4 deletions tests/unittest/CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -6,18 +6,34 @@ if(${GTEST_FOUND})
add_library(GTest::GMock UNKNOWN IMPORTED)
set_target_properties(GTest::GMock PROPERTIES IMPORTED_LOCATION ${GMOCK_LIBRARY})
else()
message(WARNING "Google test/mock not found.")
message(ERROR "Google test/mock not found.")
endif()

find_package(Qt5 COMPONENTS Test REQUIRED)
if(${QT5TEST_NOTFOUND})
message(ERROR "Required package Qt5Test not found.")
endif()
find_package(Threads REQUIRED)
set(FACELIFT_GTEST_LIBRARIES ${GTEST_BOTH_LIBRARIES} GTest::GMock Threads::Threads)
if(${THREADS_NOTFOUND})
message(ERROR "Required package Threads not found.")
endif()
set(FACELIFT_GTEST_LIBRARIES ${GTEST_BOTH_LIBRARIES} GTest::GMock Qt5::Test Threads::Threads)
include_directories(${GTEST_INCLUDE_DIRS})

facelift_add_test(UnitTests
SOURCES FaceliftUtilsTest.cpp
LINK_LIBRARIES ${FACELIFT_GTEST_LIBRARIES} FaceliftCommonLib)
LINK_LIBRARIES
${FACELIFT_GTEST_LIBRARIES}
FaceliftCommonLib
)

facelift_add_test(UnitTestsObserver
SOURCES FaceliftObserverTest.cpp
LINK_LIBRARIES
${FACELIFT_GTEST_LIBRARIES}
FaceliftIPCCommonLib
)
else()
message(WARNING "Required package google test not found!")
message(ERROR "Required package google test not found!")
endif()

87 changes: 87 additions & 0 deletions tests/unittest/FaceliftObserverTest.cpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,87 @@
#include <gtest/gtest.h>
#include "IPCProxyBase.h"
#include "InterfaceBase.h"
#include <QSignalSpy>
#include "observer.h"

namespace {

using namespace facelift;

class Counter
{
public:
Counter() { m_value = 0; }

void incValue() {
++m_value;
}
int getValue() const {
return m_value;
}
private:
int m_value;
};

class IPCProxyBaseTest : public ::testing::Test
{
public:
// Objects containing executable functions
Counter c1;
Counter c2;
// Installing objects/function to be executed in observers
// Observer to execute the task type "callOnReady"
StandartObserver < std::function<void()> > obs1 { std::bind(&Counter::incValue, &c1) };
// Observer to execute the task type "callOnceReady"
SingleTimeObserver< std::function<void()> > obs2 { std::bind(&Counter::incValue, &c2) };
// The "Proxy" is the source of the signal of interest
IPCProxyBase<InterfaceBase> proxyBase{nullptr};
// The "IsReadyObserver" is the watcher of the signal of interest in "Proxy" and contain observers to execute: "callOnReady", "callOnceReady"
IsReadyObserver readyObserver{};

~IPCProxyBaseTest() {}
};

TEST_F(IPCProxyBaseTest, testObservers)
{
// Setting up an "IsReadyObserver" to watch out for signal in "Proxy"
readyObserver.watch( &proxyBase, &IPCProxyBase<InterfaceBase>::readyChanged );
// Set observers to "IsReadyObserver"
const auto expected = QVector< QPointer<IObserver> >{ &obs1, &obs2 };
readyObserver.setObservers(expected);

// Check values before calling a signals
ASSERT_EQ(c1.getValue(), 0); // for StandartObserver ("callOnReady")
ASSERT_EQ(c2.getValue(), 0); // for SingleTimeObserver ("callOnceReady")

// Set signal tracking for "IsReadyObserver"
QSignalSpy spyReadyObserver(&readyObserver, &IsReadyObserver::readyChanged );
ASSERT_EQ( spyReadyObserver.isValid(), true);
spyReadyObserver.clear();

// Set signal tracking for "Proxy"
QSignalSpy spyProxyBase(&proxyBase, &IPCProxyBase<InterfaceBase>::readyChanged );
ASSERT_EQ( spyProxyBase.isValid(), true);
spyProxyBase.clear();

// Check that the signals on "IsReadyObserver" and "Proxy" are absent
ASSERT_EQ(spyReadyObserver.count(), 0);
ASSERT_EQ(spyProxyBase.count(), 0);
// Generate a some signals on the "Proxy" which is tracking on "IsReadyObserver"
proxyBase.readyChanged();
proxyBase.readyChanged();
proxyBase.readyChanged();

// Check the number of received signals on "Proxy"
ASSERT_EQ(spyProxyBase.count(), 3);
// Check the number of received signals on "IsReadyObserver"
ASSERT_EQ(spyReadyObserver.count(), 3);
// No one signal must be lost or skipped
ASSERT_EQ(spyProxyBase.count(), spyReadyObserver.count());

// Check values after signals call
ASSERT_EQ(c1.getValue(), 3); // for StandartObserver ("callOnReady")
ASSERT_EQ(c2.getValue(), 1); // for SingleTimeObserver ("callOnceReady")
}

} // end namespace

0 comments on commit 0b53b18

Please sign in to comment.