Files
sdbus-cpp/tests/integrationtests/DBusGeneralTests.cpp

185 lines
6.5 KiB
C++
Raw Permalink Normal View History

2020-07-18 20:16:57 +02:00
/**
2022-07-05 17:08:35 +02:00
* (C) 2016 - 2021 KISTLER INSTRUMENTE AG, Winterthur, Switzerland
2024-04-16 22:48:34 +02:00
* (C) 2016 - 2024 Stanislav Angelovic <stanislav.angelovic@protonmail.com>
2020-07-18 20:16:57 +02:00
*
* @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"
2022-06-17 23:04:07 +02:00
#include "TestFixture.h"
2020-07-18 20:16:57 +02:00
#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>
2020-07-18 20:16:57 +02:00
2022-06-17 23:04:07 +02:00
using ::testing::ElementsAre;
using ::testing::Eq;
using namespace std::chrono_literals;
2020-07-18 20:16:57 +02:00
using namespace sdbus::test;
using namespace std::string_view_literals;
2020-07-18 20:16:57 +02:00
using ADirectConnection = TestFixtureWithDirectConnection;
2022-06-17 23:04:07 +02:00
2020-07-18 20:16:57 +02:00
/*-------------------------------------*/
/* -- TEST CASES -- */
/*-------------------------------------*/
TEST(AdaptorAndProxy, CanBeConstructedSuccessfully)
2020-07-18 20:16:57 +02:00
{
auto connection = sdbus::createBusConnection();
connection->requestName(SERVICE_NAME);
2020-07-18 20:16:57 +02:00
ASSERT_NO_THROW(TestAdaptor adaptor(*connection, OBJECT_PATH));
ASSERT_NO_THROW(TestProxy proxy(SERVICE_NAME, OBJECT_PATH));
connection->releaseName(SERVICE_NAME);
2020-07-18 20:16:57 +02:00
}
2022-06-17 23:04:07 +02:00
TEST(AProxy, DoesNotSupportMoveSemantics)
{
static_assert(!std::is_move_constructible_v<DummyTestProxy>);
static_assert(!std::is_move_assignable_v<DummyTestProxy>);
}
TEST(AnAdaptor, DoesNotSupportMoveSemantics)
{
static_assert(!std::is_move_constructible_v<DummyTestAdaptor>);
static_assert(!std::is_move_assignable_v<DummyTestAdaptor>);
}
2023-01-25 00:02:51 +01:00
TYPED_TEST(AConnection, WillCallCallbackHandlerForIncomingMessageMatchingMatchRule)
2022-06-17 23:04:07 +02:00
{
auto matchRule = "sender='" + SERVICE_NAME + "',path='" + OBJECT_PATH + "'";
2022-06-17 23:04:07 +02:00
std::atomic<bool> matchingMessageReceived{false};
auto slot = this->s_proxyConnection->addMatch(matchRule, [&](sdbus::Message msg)
2022-06-17 23:04:07 +02:00
{
if(msg.getPath() == OBJECT_PATH)
matchingMessageReceived = true;
}, sdbus::return_slot);
2022-06-17 23:04:07 +02:00
2023-01-25 00:02:51 +01:00
this->m_adaptor->emitSimpleSignal();
2022-06-17 23:04:07 +02:00
ASSERT_TRUE(waitUntil(matchingMessageReceived));
}
2023-01-25 00:02:51 +01:00
TYPED_TEST(AConnection, CanInstallMatchRuleAsynchronously)
{
auto matchRule = "sender='" + SERVICE_NAME + "',path='" + OBJECT_PATH + "'";
std::atomic<bool> matchingMessageReceived{false};
std::atomic<bool> matchRuleInstalled{false};
2023-01-25 00:02:51 +01:00
auto slot = this->s_proxyConnection->addMatchAsync( matchRule
, [&](sdbus::Message msg)
2023-01-25 00:02:51 +01:00
{
if(msg.getPath() == OBJECT_PATH)
matchingMessageReceived = true;
}
, [&](sdbus::Message /*msg*/)
2023-01-25 00:02:51 +01:00
{
matchRuleInstalled = true;
}
, sdbus::return_slot );
EXPECT_TRUE(waitUntil(matchRuleInstalled));
2023-01-25 00:02:51 +01:00
this->m_adaptor->emitSimpleSignal();
ASSERT_TRUE(waitUntil(matchingMessageReceived));
}
2023-01-25 00:02:51 +01:00
TYPED_TEST(AConnection, WillUnsubscribeMatchRuleWhenClientDestroysTheAssociatedSlot)
2022-06-17 23:04:07 +02:00
{
auto matchRule = "sender='" + SERVICE_NAME + "',path='" + OBJECT_PATH + "'";
2022-06-17 23:04:07 +02:00
std::atomic<bool> matchingMessageReceived{false};
auto slot = this->s_proxyConnection->addMatch(matchRule, [&](sdbus::Message msg)
2022-06-17 23:04:07 +02:00
{
if(msg.getPath() == OBJECT_PATH)
matchingMessageReceived = true;
}, sdbus::return_slot);
2022-06-17 23:04:07 +02:00
slot.reset();
2023-01-25 00:02:51 +01:00
this->m_adaptor->emitSimpleSignal();
2022-06-17 23:04:07 +02:00
2023-01-25 00:02:51 +01:00
ASSERT_FALSE(waitUntil(matchingMessageReceived, 1s));
2022-06-17 23:04:07 +02:00
}
2023-01-25 00:02:51 +01:00
TYPED_TEST(AConnection, CanAddFloatingMatchRule)
2022-06-17 23:04:07 +02:00
{
auto matchRule = "sender='" + SERVICE_NAME + "',path='" + OBJECT_PATH + "'";
2022-06-17 23:04:07 +02:00
std::atomic<bool> matchingMessageReceived{false};
auto con = sdbus::createBusConnection();
2022-06-17 23:04:07 +02:00
con->enterEventLoopAsync();
auto callback = [&](sdbus::Message msg)
2022-06-17 23:04:07 +02:00
{
if(msg.getPath() == OBJECT_PATH)
matchingMessageReceived = true;
};
con->addMatch(matchRule, std::move(callback));
2023-01-25 00:02:51 +01:00
this->m_adaptor->emitSimpleSignal();
2023-01-03 16:30:26 +01:00
[[maybe_unused]] auto gotMessage = waitUntil(matchingMessageReceived, 2s);
assert(gotMessage);
2022-06-17 23:04:07 +02:00
matchingMessageReceived = false;
con.reset();
2023-01-25 00:02:51 +01:00
this->m_adaptor->emitSimpleSignal();
2022-06-17 23:04:07 +02:00
2023-01-25 00:02:51 +01:00
ASSERT_FALSE(waitUntil(matchingMessageReceived, 1s));
2022-06-17 23:04:07 +02:00
}
2023-01-25 00:02:51 +01:00
TYPED_TEST(AConnection, WillNotPassToMatchCallbackMessagesThatDoNotMatchTheRule)
2022-06-17 23:04:07 +02:00
{
auto matchRule = "type='signal',interface='" + INTERFACE_NAME + "',member='simpleSignal'";
std::atomic<size_t> numberOfMatchingMessages{};
auto slot = this->s_proxyConnection->addMatch(matchRule, [&](sdbus::Message msg)
2022-06-17 23:04:07 +02:00
{
if(msg.getMemberName() == "simpleSignal"sv)
2022-06-17 23:04:07 +02:00
numberOfMatchingMessages++;
}, sdbus::return_slot);
2023-01-25 00:02:51 +01:00
auto adaptor2 = std::make_unique<TestAdaptor>(*this->s_adaptorConnection, OBJECT_PATH_2);
2022-06-17 23:04:07 +02:00
2023-01-25 00:02:51 +01:00
this->m_adaptor->emitSignalWithMap({});
2022-06-17 23:04:07 +02:00
adaptor2->emitSimpleSignal();
2023-01-25 00:02:51 +01:00
this->m_adaptor->emitSimpleSignal();
2022-06-17 23:04:07 +02:00
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));
}