mirror of
https://git.jami.net/savoirfairelinux/jami-daemon.git
synced 2025-08-12 22:09:25 +08:00
220 lines
6.9 KiB
C++
220 lines
6.9 KiB
C++
/*
|
||
* 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())
|