Files
jami-daemon/test/unitTest/swarm/swarm_conversation.cpp
Adrien Béraud e2503d3a4f update license
Change-Id: I8b9e25f99da5945684e265d3923b74065c8ef8eb
2024-11-11 22:42:19 -05:00

220 lines
6.9 KiB
C++
Raw Blame History

This file contains invisible Unicode characters

This file contains invisible Unicode characters that are indistinguishable to humans but may be processed differently by a computer. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/*
* Copyright (C) 2004-2024 Savoir-faire Linux Inc.
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program 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 General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <https://www.gnu.org/licenses/>.
*/
#include <cppunit/TestAssert.h>
#include <cppunit/TestFixture.h>
#include <cppunit/extensions/HelperMacros.h>
#include <condition_variable>
#include <filesystem>
#include <string>
#include "jami.h"
#include "../common.h"
#include "jamidht/swarm/swarm_manager.h"
#include <dhtnet/multiplexed_socket.h>
#include <opendht/thread_pool.h>
#include "../../test_runner.h"
#include "account_const.h"
#include "conversation/conversationcommon.h"
#include "manager.h"
using namespace dht;
using NodeId = dht::PkId;
using namespace std::literals::chrono_literals;
namespace jami {
namespace test {
struct ConvData
{
std::string id {};
bool requestReceived {false};
bool needsHost {false};
bool conferenceChanged {false};
bool conferenceRemoved {false};
std::string hostState {};
std::vector<std::map<std::string, std::string>> messages {};
};
class SwarmConversationTest : public CppUnit::TestFixture
{
public:
~SwarmConversationTest();
static std::string name() { return "SwarmConversationTest"; }
void setUp();
std::map<std::string, ConvData> accountMap;
std::vector<std::string> accountIds;
std::mutex mtx;
std::unique_lock<std::mutex> lk {mtx};
std::condition_variable cv;
private:
void connectSignals();
std::map<std::string, std::shared_ptr<libjami::CallbackWrapperBase>> confHandlers;
void testSendMessage();
CPPUNIT_TEST_SUITE(SwarmConversationTest);
CPPUNIT_TEST(testSendMessage);
CPPUNIT_TEST_SUITE_END();
};
CPPUNIT_TEST_SUITE_NAMED_REGISTRATION(SwarmConversationTest, SwarmConversationTest::name());
void
SwarmConversationTest::setUp()
{
libjami::init(
libjami::InitFlag(libjami::LIBJAMI_FLAG_DEBUG | libjami::LIBJAMI_FLAG_CONSOLE_LOG));
if (not Manager::instance().initialized)
CPPUNIT_ASSERT(libjami::start("jami-sample.yml"));
auto actors = load_actors("actors/account_list.yml");
for (const auto& act : actors) {
auto id = act.second;
accountIds.emplace_back(id);
std::cout << act.second << std::endl;
accountMap.insert({id, {}});
}
wait_for_announcement_of(accountIds, 60s);
}
SwarmConversationTest::~SwarmConversationTest()
{
wait_for_removal_of(accountIds);
libjami::fini();
}
void
SwarmConversationTest::connectSignals()
{
std::map<std::string, std::shared_ptr<libjami::CallbackWrapperBase>> confHandlers;
confHandlers.insert(libjami::exportable_callback<libjami::ConversationSignal::ConversationReady>(
[&](const std::string& accountId, const std::string& conversationId) {
for (const auto& accId : accountIds) {
if (accountId == accId) {
accountMap[accId].id = conversationId;
}
}
cv.notify_one();
}));
confHandlers.insert(libjami::exportable_callback<libjami::ConversationSignal::MessageReceived>(
[&](const std::string& accountId,
const std::string& conversationId,
std::map<std::string, std::string> message) {
for (const auto& accId : accountIds) {
if (accountId == accId && accountMap[accId].id == conversationId) {
accountMap[accId].messages.emplace_back(message);
}
}
cv.notify_one();
}));
confHandlers.insert(
libjami::exportable_callback<libjami::ConversationSignal::ConversationRequestReceived>(
[&](const std::string& accountId,
const std::string&,
std::map<std::string, std::string>) {
for (const auto& accId : accountIds) {
if (accountId == accId) {
accountMap[accId].requestReceived = true;
}
}
cv.notify_one();
}));
libjami::registerSignalHandlers(confHandlers);
}
void
SwarmConversationTest::testSendMessage()
{
std::map<std::string, std::shared_ptr<JamiAccount>> jamiAccounts;
for (const auto& accId : accountIds) {
jamiAccounts.insert({accId, Manager::instance().getAccount<JamiAccount>(accId)});
std::cout << "created account for: " << accId << std::endl;
}
std::mutex mtx;
std::unique_lock lk {mtx};
std::condition_variable cv;
connectSignals();
auto aliceId = jamiAccounts.begin()->first;
auto convId = libjami::startConversation(aliceId);
std::cout << "started conversation: " << convId << std::endl;
for (auto it = std::next(jamiAccounts.begin()); it != jamiAccounts.end(); ++it) {
auto userUri = it->second->getUsername();
std::cout << "adding member: " << userUri << std::endl;
libjami::addConversationMember(aliceId, convId, userUri);
CPPUNIT_ASSERT_EQUAL_MESSAGE("ERROR", true, cv.wait_for(lk, 40s, [&]() {
return accountMap[it->first].requestReceived == true;
}));
libjami::acceptConversationRequest(it->first, convId);
std::this_thread::sleep_for(std::chrono::seconds(10));
}
std::cout << "waiting for conversation ready" << std::endl;
for (size_t i = 1; i < accountIds.size(); i++) {
CPPUNIT_ASSERT(
cv.wait_for(lk, 40s, [&]() { return accountMap[accountIds.at(i)].id == convId; }));
}
std::cout << "messages size " << accountMap[accountIds.at(0)].messages.size() << std::endl;
CPPUNIT_ASSERT(
cv.wait_for(lk, 70s, [&]() { return accountMap[accountIds.at(0)].messages.size() >= 2; }));
libjami::sendMessage(aliceId, convId, "hi"s, "");
for (size_t i = 1; i < accountIds.size(); i++) {
std::cout << "COUNTER: " << i << " messages size "
<< accountMap[accountIds.at(i)].messages.size() << std::endl;
if (accountMap[accountIds.at(i)].messages.size() >= 105) {
for (const auto& msg : accountMap[accountIds.at(i)].messages) {
std::cout << "Message id: " << msg.at("id") << " type: " << msg.at("type")
<< std::endl;
}
}
CPPUNIT_ASSERT(cv.wait_for(lk, 40s, [&]() {
return accountMap[accountIds.at(i)].messages.size() >= 1;
}));
}
libjami::unregisterSignalHandlers();
}
} // namespace test
} // namespace jami
RING_TEST_RUNNER(jami::test::SwarmConversationTest::name())