mirror of
https://github.com/Kistler-Group/sdbus-cpp.git
synced 2026-01-25 15:12:20 +01:00
233 lines
8.8 KiB
C++
233 lines
8.8 KiB
C++
/**
|
|
* (C) 2016 - 2021 KISTLER INSTRUMENTE AG, Winterthur, Switzerland
|
|
* (C) 2016 - 2026 Stanislav Angelovic <stanislav.angelovic@protonmail.com>
|
|
*
|
|
* @file TestProxy.cpp
|
|
*
|
|
* Created on: May 23, 2020
|
|
* Project: sdbus-c++
|
|
* Description: High-level D-Bus IPC C++ library based on sd-bus
|
|
*
|
|
* This file is part of sdbus-c++.
|
|
*
|
|
* sdbus-c++ 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.
|
|
*
|
|
* sdbus-c++ 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 sdbus-c++. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "TestProxy.h"
|
|
#include <sdbus-c++/sdbus-c++.h>
|
|
#include "Defs.h"
|
|
#include <string>
|
|
#include <memory>
|
|
#include <map>
|
|
#include <cstdint>
|
|
#include <optional>
|
|
#include <functional>
|
|
#include <future>
|
|
#include <chrono>
|
|
#include <atomic>
|
|
#include <utility>
|
|
#include <vector>
|
|
|
|
namespace sdbus::test {
|
|
|
|
TestProxy::TestProxy(ServiceName destination, ObjectPath objectPath)
|
|
: ProxyInterfaces(std::move(destination), std::move(objectPath))
|
|
{
|
|
getProxy().uponSignal("signalWithoutRegistration").onInterface(sdbus::test::INTERFACE_NAME).call([this](const sdbus::Struct<std::string, sdbus::Struct<sdbus::Signature>>& strct){ this->onSignalWithoutRegistration(strct); });
|
|
|
|
registerProxy();
|
|
}
|
|
|
|
TestProxy::TestProxy(ServiceName destination, ObjectPath objectPath, dont_run_event_loop_thread_t)
|
|
: ProxyInterfaces(std::move(destination), std::move(objectPath), dont_run_event_loop_thread)
|
|
{
|
|
// It doesn't make sense to register any signals here since proxy upon a D-Bus connection with no event loop thread
|
|
// will not receive any incoming messages except replies to synchronous D-Bus calls.
|
|
}
|
|
|
|
TestProxy::TestProxy(sdbus::IConnection& connection, ServiceName destination, ObjectPath objectPath)
|
|
: ProxyInterfaces(connection, std::move(destination), std::move(objectPath))
|
|
{
|
|
getProxy().uponSignal("signalWithoutRegistration").onInterface(sdbus::test::INTERFACE_NAME).call([this](const sdbus::Struct<std::string, sdbus::Struct<sdbus::Signature>>& strct){ this->onSignalWithoutRegistration(strct); });
|
|
|
|
registerProxy();
|
|
}
|
|
|
|
TestProxy::~TestProxy()
|
|
{
|
|
unregisterProxy();
|
|
}
|
|
|
|
void TestProxy::onSimpleSignal()
|
|
{
|
|
m_signalMsg = std::make_unique<sdbus::Message>(getProxy().getCurrentlyProcessedMessage());
|
|
m_signalName = m_signalMsg->getMemberName();
|
|
|
|
m_gotSimpleSignal = true;
|
|
}
|
|
|
|
void TestProxy::onSignalWithMap(const std::map<int32_t, std::string>& aMap)
|
|
{
|
|
m_mapFromSignal = aMap;
|
|
m_gotSignalWithMap = true;
|
|
}
|
|
|
|
void TestProxy::onSignalWithVariant(const sdbus::Variant& aVariant)
|
|
{
|
|
m_variantFromSignal = aVariant.get<double>();
|
|
m_gotSignalWithVariant = true;
|
|
}
|
|
|
|
void TestProxy::onSignalWithoutRegistration(const sdbus::Struct<std::string, sdbus::Struct<sdbus::Signature>>& strct)
|
|
{
|
|
// Static cast to std::string is a workaround for gcc 11.4 false positive warning (which later gcc versions nor Clang emit)
|
|
m_signatureFromSignal[std::get<0>(strct)] = static_cast<std::string>(std::get<0>(std::get<1>(strct)));
|
|
m_gotSignalWithSignature = true;
|
|
}
|
|
|
|
void TestProxy::onDoOperationReply(uint32_t returnValue, std::optional<sdbus::Error> error) const
|
|
{
|
|
if (m_DoOperationClientSideAsyncReplyHandler)
|
|
m_DoOperationClientSideAsyncReplyHandler(returnValue, std::move(error));
|
|
}
|
|
|
|
void TestProxy::onPropertiesChanged( const sdbus::InterfaceName& interfaceName
|
|
, const std::map<PropertyName, sdbus::Variant>& changedProperties
|
|
, const std::vector<PropertyName>& invalidatedProperties )
|
|
{
|
|
if (m_onPropertiesChangedHandler)
|
|
m_onPropertiesChangedHandler(interfaceName, changedProperties, invalidatedProperties);
|
|
}
|
|
|
|
void TestProxy::installDoOperationClientSideAsyncReplyHandler(std::function<void(uint32_t res, std::optional<sdbus::Error> err)> handler)
|
|
{
|
|
m_DoOperationClientSideAsyncReplyHandler = std::move(handler);
|
|
}
|
|
|
|
uint32_t TestProxy::doOperationWithTimeout(const std::chrono::microseconds &timeout, uint32_t param)
|
|
{
|
|
using namespace std::chrono_literals;
|
|
uint32_t result = 0;
|
|
getProxy().callMethod("doOperation").onInterface(sdbus::test::INTERFACE_NAME).withTimeout(timeout).withArguments(param).storeResultsTo(result);
|
|
return result;
|
|
}
|
|
|
|
MethodReply TestProxy::doOperationOnBasicAPILevel(uint32_t param)
|
|
{
|
|
auto methodCall = getProxy().createMethodCall(test::INTERFACE_NAME, MethodName{"doOperation"});
|
|
methodCall << param;
|
|
|
|
m_methodCallMsg = std::make_unique<MethodCall>(methodCall);
|
|
|
|
return getProxy().callMethod(methodCall);
|
|
}
|
|
|
|
sdbus::PendingAsyncCall TestProxy::doOperationClientSideAsync(uint32_t param)
|
|
{
|
|
return getProxy().callMethodAsync("doOperation")
|
|
.onInterface(sdbus::test::INTERFACE_NAME)
|
|
.withArguments(param)
|
|
.uponReplyInvoke([this](std::optional<sdbus::Error> error, uint32_t returnValue)
|
|
{
|
|
this->onDoOperationReply(returnValue, std::move(error));
|
|
});
|
|
}
|
|
|
|
Slot TestProxy::doOperationClientSideAsync(uint32_t param, sdbus::return_slot_t)
|
|
{
|
|
return getProxy().callMethodAsync("doOperation")
|
|
.onInterface(sdbus::test::INTERFACE_NAME)
|
|
.withArguments(param)
|
|
.uponReplyInvoke([this](std::optional<sdbus::Error> error, uint32_t returnValue)
|
|
{
|
|
this->onDoOperationReply(returnValue, std::move(error));
|
|
}, sdbus::return_slot);
|
|
}
|
|
|
|
std::future<uint32_t> TestProxy::doOperationClientSideAsync(uint32_t param, with_future_t)
|
|
{
|
|
return getProxy().callMethodAsync("doOperation")
|
|
.onInterface(sdbus::test::INTERFACE_NAME)
|
|
.withArguments(param)
|
|
.getResultAsFuture<uint32_t>();
|
|
}
|
|
|
|
std::future<MethodReply> TestProxy::doOperationClientSideAsyncOnBasicAPILevel(uint32_t param)
|
|
{
|
|
auto methodCall = getProxy().createMethodCall(sdbus::test::INTERFACE_NAME, sdbus::MethodName{"doOperation"});
|
|
methodCall << param;
|
|
|
|
return getProxy().callMethodAsync(methodCall, sdbus::with_future);
|
|
}
|
|
|
|
std::future<std::map<int32_t, std::string>> TestProxy::doOperationWithLargeDataClientSideAsync(const std::map<int32_t, std::string>& largeParam, with_future_t)
|
|
{
|
|
return getProxy().callMethodAsync("doOperationWithLargeData")
|
|
.onInterface(sdbus::test::INTERFACE_NAME)
|
|
.withArguments(largeParam)
|
|
.getResultAsFuture<std::map<int32_t, std::string>>();
|
|
}
|
|
|
|
void TestProxy::doErroneousOperationClientSideAsync()
|
|
{
|
|
getProxy().callMethodAsync("throwError")
|
|
.onInterface(sdbus::test::INTERFACE_NAME)
|
|
.uponReplyInvoke([this](std::optional<sdbus::Error> error)
|
|
{
|
|
this->onDoOperationReply(0, std::move(error));
|
|
});
|
|
}
|
|
|
|
std::future<void> TestProxy::doErroneousOperationClientSideAsync(with_future_t)
|
|
{
|
|
return getProxy().callMethodAsync("throwError")
|
|
.onInterface(sdbus::test::INTERFACE_NAME)
|
|
.getResultAsFuture<>();
|
|
}
|
|
|
|
void TestProxy::doOperationClientSideAsyncWithTimeout(const std::chrono::microseconds &timeout, uint32_t param)
|
|
{
|
|
using namespace std::chrono_literals;
|
|
getProxy().callMethodAsync("doOperation")
|
|
.onInterface(sdbus::test::INTERFACE_NAME)
|
|
.withTimeout(timeout)
|
|
.withArguments(param)
|
|
.uponReplyInvoke([this](std::optional<sdbus::Error> error, uint32_t returnValue)
|
|
{
|
|
this->onDoOperationReply(returnValue, std::move(error));
|
|
});
|
|
}
|
|
|
|
int32_t TestProxy::callNonexistentMethod()
|
|
{
|
|
int32_t result = 0;
|
|
getProxy().callMethod("callNonexistentMethod").onInterface(sdbus::test::INTERFACE_NAME).storeResultsTo(result);
|
|
return result;
|
|
}
|
|
|
|
int32_t TestProxy::callMethodOnNonexistentInterface()
|
|
{
|
|
sdbus::InterfaceName const nonexistentInterfaceName{"sdbuscpp.interface.that.does.not.exist"};
|
|
int32_t result = 0;
|
|
getProxy().callMethod("someMethod").onInterface(nonexistentInterfaceName).storeResultsTo(result);
|
|
return result;
|
|
}
|
|
|
|
void TestProxy::setStateProperty(const std::string& value)
|
|
{
|
|
getProxy().setProperty("state").onInterface(sdbus::test::INTERFACE_NAME).toValue(value);
|
|
}
|
|
|
|
} // namespace sdbus::test
|