mirror of
https://github.com/Kistler-Group/sdbus-cpp.git
synced 2026-01-25 23:22:18 +01:00
This PR makes things around connection factories a little more consistent and more intuitive:
* createConnection() has been removed. One shall call more expressive createSystemConnection() instead to get a connection to the system bus.
* createDefaultBusConnection() has been renamed to createBusConnection(), so as not to be confused with libsystemd's default_bus, which is a different thing (a reusable thread-local bus).
Proxies still by default call createBusConnection() to get a connection when the connection is not provided explicitly by the caller, but now createBusConnection() does a different thing, so now the proxies connect to either session bus or system bus depending on the context (as opposed to always to system bus like before).
181 lines
6.3 KiB
C++
181 lines
6.3 KiB
C++
/**
|
|
* (C) 2016 - 2021 KISTLER INSTRUMENTE AG, Winterthur, Switzerland
|
|
* (C) 2016 - 2022 Stanislav Angelovic <stanislav.angelovic@protonmail.com>
|
|
*
|
|
* @file DBusGeneralTests.cpp
|
|
*
|
|
* Created on: Jan 2, 2017
|
|
* 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 "TestAdaptor.h"
|
|
#include "TestProxy.h"
|
|
#include "TestFixture.h"
|
|
#include "sdbus-c++/sdbus-c++.h"
|
|
|
|
#include <gtest/gtest.h>
|
|
#include <gmock/gmock.h>
|
|
#include <string>
|
|
#include <thread>
|
|
#include <tuple>
|
|
#include <chrono>
|
|
#include <fstream>
|
|
#include <future>
|
|
#include <unistd.h>
|
|
#include <variant>
|
|
|
|
using ::testing::ElementsAre;
|
|
using ::testing::Eq;
|
|
using namespace std::chrono_literals;
|
|
using namespace sdbus::test;
|
|
|
|
using ADirectConnection = TestFixtureWithDirectConnection;
|
|
|
|
/*-------------------------------------*/
|
|
/* -- TEST CASES -- */
|
|
/*-------------------------------------*/
|
|
|
|
TEST(AdaptorAndProxy, CanBeConstructedSuccesfully)
|
|
{
|
|
auto connection = sdbus::createBusConnection();
|
|
connection->requestName(BUS_NAME);
|
|
|
|
ASSERT_NO_THROW(TestAdaptor adaptor(*connection, OBJECT_PATH));
|
|
ASSERT_NO_THROW(TestProxy proxy(BUS_NAME, OBJECT_PATH));
|
|
}
|
|
|
|
TEST(AProxy, SupportsMoveSemantics)
|
|
{
|
|
static_assert(std::is_move_constructible_v<DummyTestProxy>);
|
|
static_assert(std::is_move_assignable_v<DummyTestProxy>);
|
|
}
|
|
|
|
TEST(AnAdaptor, SupportsMoveSemantics)
|
|
{
|
|
static_assert(std::is_move_constructible_v<DummyTestAdaptor>);
|
|
static_assert(std::is_move_assignable_v<DummyTestAdaptor>);
|
|
}
|
|
|
|
TYPED_TEST(AConnection, WillCallCallbackHandlerForIncomingMessageMatchingMatchRule)
|
|
{
|
|
auto matchRule = "sender='" + BUS_NAME + "',path='" + OBJECT_PATH + "'";
|
|
std::atomic<bool> matchingMessageReceived{false};
|
|
auto slot = this->s_proxyConnection->addMatch(matchRule, [&](sdbus::Message msg)
|
|
{
|
|
if(msg.getPath() == OBJECT_PATH)
|
|
matchingMessageReceived = true;
|
|
});
|
|
|
|
this->m_adaptor->emitSimpleSignal();
|
|
|
|
ASSERT_TRUE(waitUntil(matchingMessageReceived));
|
|
}
|
|
|
|
TYPED_TEST(AConnection, CanInstallMatchRuleAsynchronously)
|
|
{
|
|
auto matchRule = "sender='" + BUS_NAME + "',path='" + OBJECT_PATH + "'";
|
|
std::atomic<bool> matchingMessageReceived{false};
|
|
std::atomic<bool> matchRuleInstalled{false};
|
|
auto slot = this->s_proxyConnection->addMatchAsync( matchRule
|
|
, [&](sdbus::Message msg)
|
|
{
|
|
if(msg.getPath() == OBJECT_PATH)
|
|
matchingMessageReceived = true;
|
|
}
|
|
, [&](sdbus::Message /*msg*/)
|
|
{
|
|
matchRuleInstalled = true;
|
|
} );
|
|
|
|
EXPECT_TRUE(waitUntil(matchRuleInstalled));
|
|
|
|
this->m_adaptor->emitSimpleSignal();
|
|
|
|
ASSERT_TRUE(waitUntil(matchingMessageReceived));
|
|
}
|
|
|
|
TYPED_TEST(AConnection, WillUnsubscribeMatchRuleWhenClientDestroysTheAssociatedSlot)
|
|
{
|
|
auto matchRule = "sender='" + BUS_NAME + "',path='" + OBJECT_PATH + "'";
|
|
std::atomic<bool> matchingMessageReceived{false};
|
|
auto slot = this->s_proxyConnection->addMatch(matchRule, [&](sdbus::Message msg)
|
|
{
|
|
if(msg.getPath() == OBJECT_PATH)
|
|
matchingMessageReceived = true;
|
|
});
|
|
slot.reset();
|
|
|
|
this->m_adaptor->emitSimpleSignal();
|
|
|
|
ASSERT_FALSE(waitUntil(matchingMessageReceived, 1s));
|
|
}
|
|
|
|
TYPED_TEST(AConnection, CanAddFloatingMatchRule)
|
|
{
|
|
auto matchRule = "sender='" + BUS_NAME + "',path='" + OBJECT_PATH + "'";
|
|
std::atomic<bool> matchingMessageReceived{false};
|
|
auto con = sdbus::createSystemBusConnection();
|
|
con->enterEventLoopAsync();
|
|
auto callback = [&](sdbus::Message msg)
|
|
{
|
|
if(msg.getPath() == OBJECT_PATH)
|
|
matchingMessageReceived = true;
|
|
};
|
|
con->addMatch(matchRule, std::move(callback), sdbus::floating_slot);
|
|
this->m_adaptor->emitSimpleSignal();
|
|
[[maybe_unused]] auto gotMessage = waitUntil(matchingMessageReceived, 2s);
|
|
assert(gotMessage);
|
|
matchingMessageReceived = false;
|
|
|
|
con.reset();
|
|
this->m_adaptor->emitSimpleSignal();
|
|
|
|
ASSERT_FALSE(waitUntil(matchingMessageReceived, 1s));
|
|
}
|
|
|
|
TYPED_TEST(AConnection, WillNotPassToMatchCallbackMessagesThatDoNotMatchTheRule)
|
|
{
|
|
auto matchRule = "type='signal',interface='" + INTERFACE_NAME + "',member='simpleSignal'";
|
|
std::atomic<size_t> numberOfMatchingMessages{};
|
|
auto slot = this->s_proxyConnection->addMatch(matchRule, [&](sdbus::Message msg)
|
|
{
|
|
if(msg.getMemberName() == "simpleSignal")
|
|
numberOfMatchingMessages++;
|
|
});
|
|
auto adaptor2 = std::make_unique<TestAdaptor>(*this->s_adaptorConnection, OBJECT_PATH_2);
|
|
|
|
this->m_adaptor->emitSignalWithMap({});
|
|
adaptor2->emitSimpleSignal();
|
|
this->m_adaptor->emitSimpleSignal();
|
|
|
|
ASSERT_TRUE(waitUntil([&](){ return numberOfMatchingMessages == 2; }));
|
|
ASSERT_FALSE(waitUntil([&](){ return numberOfMatchingMessages > 2; }, 1s));
|
|
}
|
|
|
|
// A simple direct connection test similar in nature to https://github.com/systemd/systemd/blob/main/src/libsystemd/sd-bus/test-bus-server.c
|
|
TEST_F(ADirectConnection, CanBeUsedBetweenClientAndServer)
|
|
{
|
|
auto val = m_proxy->sumArrayItems({1, 7}, {2, 3, 4});
|
|
m_adaptor->emitSimpleSignal();
|
|
|
|
// Make sure method call passes and emitted signal is received
|
|
ASSERT_THAT(val, Eq(1 + 7 + 2 + 3 + 4));
|
|
ASSERT_TRUE(waitUntil(m_proxy->m_gotSimpleSignal));
|
|
}
|