LCOV - code coverage report
Current view: top level - test/unitTest/syncHistory - syncHistory.cpp (source / functions) Hit Total Coverage
Test: jami-coverage-filtered.info Lines: 591 622 95.0 %
Date: 2024-11-15 09:04:49 Functions: 92 94 97.9 %

          Line data    Source code
       1             : /*
       2             :  *  Copyright (C) 2004-2024 Savoir-faire Linux Inc.
       3             :  *
       4             :  *  This program is free software: you can redistribute it and/or modify
       5             :  *  it under the terms of the GNU General Public License as published by
       6             :  *  the Free Software Foundation, either version 3 of the License, or
       7             :  *  (at your option) any later version.
       8             :  *
       9             :  *  This program is distributed in the hope that it will be useful,
      10             :  *  but WITHOUT ANY WARRANTY; without even the implied warranty of
      11             :  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
      12             :  *  GNU General Public License for more details.
      13             :  *
      14             :  *  You should have received a copy of the GNU General Public License
      15             :  *  along with this program. If not, see <https://www.gnu.org/licenses/>.
      16             :  */
      17             : 
      18             : #include "fileutils.h"
      19             : #include "manager.h"
      20             : #include "jamidht/jamiaccount.h"
      21             : #include "../../test_runner.h"
      22             : #include "jami.h"
      23             : #include "account_const.h"
      24             : #include "common.h"
      25             : 
      26             : #include <dhtnet/connectionmanager.h>
      27             : #include <dhtnet/multiplexed_socket.h>
      28             : 
      29             : #include <cppunit/TestAssert.h>
      30             : #include <cppunit/TestFixture.h>
      31             : #include <cppunit/extensions/HelperMacros.h>
      32             : 
      33             : #include <chrono>
      34             : #include <condition_variable>
      35             : #include <filesystem>
      36             : 
      37             : using namespace libjami::Account;
      38             : using namespace std::literals::chrono_literals;
      39             : 
      40             : namespace jami {
      41             : namespace test {
      42             : 
      43             : struct UserData {
      44             :     std::string conversationId;
      45             :     bool removed {false};
      46             :     bool requestReceived {false};
      47             :     bool requestRemoved {false};
      48             :     bool errorDetected {false};
      49             :     bool registered {false};
      50             :     bool stopped {false};
      51             :     bool deviceAnnounced {false};
      52             :     bool sending {false};
      53             :     bool sent {false};
      54             :     std::string profilePath;
      55             :     std::string payloadTrustRequest;
      56             :     std::vector<libjami::SwarmMessage> messages;
      57             :     std::vector<libjami::SwarmMessage> messagesLoaded;
      58             :     std::vector<libjami::SwarmMessage> messagesUpdated;
      59             :     std::map<std::string, int> members;
      60             : };
      61             : 
      62             : class SyncHistoryTest : public CppUnit::TestFixture
      63             : {
      64             : public:
      65          14 :     SyncHistoryTest()
      66          14 :     {
      67             :         // Init daemon
      68          14 :         libjami::init(libjami::InitFlag(libjami::LIBJAMI_FLAG_DEBUG | libjami::LIBJAMI_FLAG_CONSOLE_LOG));
      69          14 :         if (not Manager::instance().initialized)
      70           1 :             CPPUNIT_ASSERT(libjami::start("jami-sample.yml"));
      71          14 :     }
      72          28 :     ~SyncHistoryTest() { libjami::fini(); }
      73           2 :     static std::string name() { return "SyncHistory"; }
      74             :     void setUp();
      75             :     void tearDown();
      76             : 
      77             :     std::string aliceId;
      78             :     UserData aliceData;
      79             :     std::string bobId;
      80             :     UserData bobData;
      81             :     std::string alice2Id;
      82             :     UserData alice2Data;
      83             : 
      84             :     std::mutex mtx;
      85             :     std::unique_lock<std::mutex> lk {mtx};
      86             :     std::condition_variable cv;
      87             :     void connectSignals();
      88             : 
      89             : private:
      90             :     void testCreateConversationThenSync();
      91             :     void testCreateConversationWithOnlineDevice();
      92             :     void testCreateConversationWithMessagesThenAddDevice();
      93             :     void testCreateMultipleConversationThenAddDevice();
      94             :     void testReceivesInviteThenAddDevice();
      95             :     void testRemoveConversationOnAllDevices();
      96             :     void testSyncCreateAccountExportDeleteReimportOldBackup();
      97             :     void testSyncCreateAccountExportDeleteReimportWithConvId();
      98             :     void testSyncCreateAccountExportDeleteReimportWithConvReq();
      99             :     void testSyncOneToOne();
     100             :     void testConversationRequestRemoved();
     101             :     void testProfileReceivedMultiDevice();
     102             :     void testLastInteractionAfterClone();
     103             :     void testLastInteractionAfterSomeMessages();
     104             : 
     105           2 :     CPPUNIT_TEST_SUITE(SyncHistoryTest);
     106           1 :     CPPUNIT_TEST(testCreateConversationThenSync);
     107           1 :     CPPUNIT_TEST(testCreateConversationWithOnlineDevice);
     108           1 :     CPPUNIT_TEST(testCreateConversationWithMessagesThenAddDevice);
     109           1 :     CPPUNIT_TEST(testCreateMultipleConversationThenAddDevice);
     110           1 :     CPPUNIT_TEST(testReceivesInviteThenAddDevice);
     111           1 :     CPPUNIT_TEST(testRemoveConversationOnAllDevices);
     112           1 :     CPPUNIT_TEST(testSyncCreateAccountExportDeleteReimportOldBackup);
     113           1 :     CPPUNIT_TEST(testSyncCreateAccountExportDeleteReimportWithConvId);
     114           1 :     CPPUNIT_TEST(testSyncCreateAccountExportDeleteReimportWithConvReq);
     115           1 :     CPPUNIT_TEST(testSyncOneToOne);
     116           1 :     CPPUNIT_TEST(testConversationRequestRemoved);
     117           1 :     CPPUNIT_TEST(testProfileReceivedMultiDevice);
     118           1 :     CPPUNIT_TEST(testLastInteractionAfterClone);
     119           1 :     CPPUNIT_TEST(testLastInteractionAfterSomeMessages);
     120           4 :     CPPUNIT_TEST_SUITE_END();
     121             : };
     122             : 
     123             : CPPUNIT_TEST_SUITE_NAMED_REGISTRATION(SyncHistoryTest, SyncHistoryTest::name());
     124             : 
     125             : void
     126          14 : SyncHistoryTest::setUp()
     127             : {
     128          28 :     auto actors = load_actors_and_wait_for_announcement("actors/alice-bob.yml");
     129          14 :     aliceId = actors["alice"];
     130          14 :     bobId = actors["bob"];
     131          14 :     alice2Id = "";
     132          14 :     aliceData = {};
     133          14 :     bobData = {};
     134          14 :     alice2Data = {};
     135          14 : }
     136             : 
     137             : void
     138          14 : SyncHistoryTest::tearDown()
     139             : {
     140          28 :     auto aliceArchive = std::filesystem::current_path().string() + "/alice.gz";
     141          14 :     std::remove(aliceArchive.c_str());
     142          14 :     if (alice2Id.empty()) {
     143           0 :         wait_for_removal_of({aliceId, bobId});
     144             :     } else {
     145          56 :         wait_for_removal_of({aliceId, bobId, alice2Id});
     146             :     }
     147          14 : }
     148             : 
     149             : void
     150          13 : SyncHistoryTest::connectSignals()
     151             : {
     152          13 :     std::map<std::string, std::shared_ptr<libjami::CallbackWrapperBase>> confHandlers;
     153          13 :     confHandlers.insert(
     154          26 :         libjami::exportable_callback<libjami::ConfigurationSignal::VolatileDetailsChanged>(
     155          59 :             [&](const std::string& accountId, const std::map<std::string, std::string>&) {
     156          59 :                 if (accountId == aliceId) {
     157           3 :                     auto aliceAccount = Manager::instance().getAccount<JamiAccount>(aliceId);
     158           3 :                     auto details = aliceAccount->getVolatileAccountDetails();
     159           6 :                     auto daemonStatus = details[libjami::Account::ConfProperties::Registration::STATUS];
     160           3 :                     if (daemonStatus == "REGISTERED") {
     161           0 :                         aliceData.registered = true;
     162           3 :                     } else if (daemonStatus == "UNREGISTERED") {
     163           3 :                         aliceData.stopped = true;
     164             :                     }
     165           6 :                     auto deviceAnnounced = details[libjami::Account::VolatileProperties::DEVICE_ANNOUNCED];
     166           3 :                     aliceData.deviceAnnounced = deviceAnnounced == "true";
     167          59 :                 } else if (accountId == bobId) {
     168           0 :                     auto bobAccount = Manager::instance().getAccount<JamiAccount>(bobId);
     169           0 :                     auto details = bobAccount->getVolatileAccountDetails();
     170           0 :                     auto daemonStatus = details[libjami::Account::ConfProperties::Registration::STATUS];
     171           0 :                     if (daemonStatus == "REGISTERED") {
     172           0 :                         bobData.registered = true;
     173           0 :                     } else if (daemonStatus == "UNREGISTERED") {
     174           0 :                         bobData.stopped = true;
     175             :                     }
     176           0 :                     auto deviceAnnounced = details[libjami::Account::VolatileProperties::DEVICE_ANNOUNCED];
     177           0 :                     bobData.deviceAnnounced = deviceAnnounced == "true";
     178          56 :                 } else if (accountId == alice2Id) {
     179          43 :                     auto alice2Account = Manager::instance().getAccount<JamiAccount>(alice2Id);
     180          43 :                     auto details = alice2Account->getVolatileAccountDetails();
     181          86 :                     auto daemonStatus = details[libjami::Account::ConfProperties::Registration::STATUS];
     182          43 :                     if (daemonStatus == "REGISTERED") {
     183          17 :                         alice2Data.registered = true;
     184          26 :                     } else if (daemonStatus == "UNREGISTERED") {
     185          13 :                         alice2Data.stopped = true;
     186             :                     }
     187          86 :                     auto deviceAnnounced = details[libjami::Account::VolatileProperties::DEVICE_ANNOUNCED];
     188          43 :                     alice2Data.deviceAnnounced = deviceAnnounced == "true";
     189          43 :                 }
     190          59 :                 cv.notify_one();
     191          59 :             }));
     192          13 :     confHandlers.insert(libjami::exportable_callback<libjami::ConversationSignal::ConversationReady>(
     193          29 :         [&](const std::string& accountId, const std::string& conversationId) {
     194          29 :             if (accountId == aliceId) {
     195          10 :                 aliceData.conversationId = conversationId;
     196          19 :             } else if (accountId == bobId) {
     197           8 :                 bobData.conversationId = conversationId;
     198          11 :             } else if (accountId == alice2Id) {
     199          11 :                 alice2Data.conversationId = conversationId;
     200             :             }
     201          29 :             cv.notify_one();
     202          29 :         }));
     203          13 :     confHandlers.insert(libjami::exportable_callback<libjami::ConfigurationSignal::ProfileReceived>(
     204           9 :         [&](const std::string& accountId, const std::string& peerId, const std::string& path) {
     205           9 :             if (accountId == bobId)
     206           5 :                 bobData.profilePath = path;
     207           4 :             else if (accountId == aliceId)
     208           1 :                 aliceData.profilePath = path;
     209           3 :             else if (accountId == alice2Id)
     210           3 :                 alice2Data.profilePath = path;
     211           9 :             cv.notify_one();
     212           9 :         }));
     213          13 :     confHandlers.insert(libjami::exportable_callback<libjami::ConversationSignal::ConversationMemberEvent>(
     214          12 :         [&](const std::string& accountId, const std::string& conversationId, const auto& member, auto status) {
     215          12 :             if (accountId == aliceId) {
     216           7 :                 aliceData.members[member] = status;
     217           5 :             } else if (accountId == bobId) {
     218           4 :                 bobData.members[member] = status;
     219           1 :             } else if (accountId == alice2Id) {
     220           1 :                 alice2Data.members[member] = status;
     221             :             }
     222          12 :             cv.notify_one();
     223          12 :         }));
     224          13 :     confHandlers.insert(
     225          26 :         libjami::exportable_callback<libjami::ConfigurationSignal::AccountMessageStatusChanged>(
     226         144 :             [&](const std::string& accountId,
     227             :                 const std::string& /*conversationId*/,
     228             :                 const std::string& /*peer*/,
     229             :                 const std::string& /*msgId*/,
     230             :                 int status) {
     231         144 :                 if (accountId == aliceId) {
     232          43 :                     if (status == 2)
     233           6 :                         aliceData.sending = true;
     234          43 :                     if (status == 3)
     235          10 :                         aliceData.sent = true;
     236         101 :                 } else if (accountId == alice2Id) {
     237          10 :                     if (status == 2)
     238           5 :                         alice2Data.sending = true;
     239          10 :                     if (status == 3)
     240           1 :                         alice2Data.sent = true;
     241          91 :                 } else if (accountId == bobId) {
     242          91 :                     if (status == 2)
     243          28 :                         bobData.sending = true;
     244          91 :                     if (status == 3)
     245           7 :                         bobData.sent = true;
     246             :                 }
     247         144 :                 cv.notify_one();
     248         144 :             }));
     249          13 :     confHandlers.insert(
     250          26 :         libjami::exportable_callback<libjami::ConfigurationSignal::IncomingTrustRequest>(
     251           3 :             [&](const std::string& account_id,
     252             :                 const std::string& /*from*/,
     253             :                 const std::string& /*conversationId*/,
     254             :                 const std::vector<uint8_t>& payload,
     255             :                 time_t /*received*/) {
     256           3 :                 auto payloadStr = std::string(payload.data(), payload.data() + payload.size());
     257           3 :                 if (account_id == aliceId)
     258           0 :                     aliceData.payloadTrustRequest = payloadStr;
     259           3 :                 else if (account_id == bobId)
     260           3 :                     bobData.payloadTrustRequest = payloadStr;
     261           3 :                 cv.notify_one();
     262           3 :             }));
     263          13 :     confHandlers.insert(
     264          26 :         libjami::exportable_callback<libjami::ConversationSignal::ConversationRequestReceived>(
     265          10 :             [&](const std::string& accountId,
     266             :                 const std::string& /* conversationId */,
     267             :                 std::map<std::string, std::string> /*metadatas*/) {
     268          10 :                 if (accountId == aliceId) {
     269           3 :                     aliceData.requestReceived = true;
     270           7 :                 } else if (accountId == bobId) {
     271           5 :                     bobData.requestReceived = true;
     272           2 :                 } else if (accountId == alice2Id) {
     273           2 :                     alice2Data.requestReceived = true;
     274             :                 }
     275          10 :                 cv.notify_one();
     276          10 :             }));
     277          13 :     confHandlers.insert(
     278          26 :         libjami::exportable_callback<libjami::ConversationSignal::ConversationRequestDeclined>(
     279           2 :             [&](const std::string& accountId, const std::string&) {
     280           2 :                 if (accountId == bobId) {
     281           0 :                     bobData.requestRemoved = true;
     282           2 :                 } else if (accountId == aliceId) {
     283           1 :                     aliceData.requestRemoved = true;
     284           1 :                 } else if (accountId == alice2Id) {
     285           1 :                     alice2Data.requestRemoved = true;
     286             :                 }
     287           2 :                 cv.notify_one();
     288           2 :             }));
     289          13 :     confHandlers.insert(libjami::exportable_callback<libjami::ConversationSignal::SwarmMessageReceived>(
     290          35 :         [&](const std::string& accountId,
     291             :             const std::string& /* conversationId */,
     292             :             libjami::SwarmMessage message) {
     293          35 :             if (accountId == aliceId) {
     294          16 :                 aliceData.messages.emplace_back(message);
     295          19 :             } else if (accountId == bobId) {
     296          13 :                 bobData.messages.emplace_back(message);
     297           6 :             } else if (accountId == alice2Id) {
     298           6 :                 alice2Data.messages.emplace_back(message);
     299             :             }
     300          35 :             cv.notify_one();
     301          35 :         }));
     302          13 :     confHandlers.insert(libjami::exportable_callback<libjami::ConversationSignal::SwarmLoaded>(
     303           1 :         [&](uint32_t, const std::string& accountId,
     304             :             const std::string& /* conversationId */,
     305             :             std::vector<libjami::SwarmMessage> messages) {
     306           1 :             if (accountId == aliceId) {
     307           0 :                 aliceData.messagesLoaded.insert(aliceData.messagesLoaded.end(), messages.begin(), messages.end());
     308           1 :             } else if (accountId == alice2Id) {
     309           1 :                 alice2Data.messagesLoaded.insert(alice2Data.messagesLoaded.end(), messages.begin(), messages.end());
     310           0 :             } else if (accountId == bobId) {
     311           0 :                 bobData.messagesLoaded.insert(bobData.messagesLoaded.end(), messages.begin(), messages.end());
     312             :             }
     313           1 :             cv.notify_one();
     314           1 :         }));
     315          13 :     confHandlers.insert(libjami::exportable_callback<libjami::ConversationSignal::SwarmMessageUpdated>(
     316           0 :         [&](const std::string& accountId,
     317             :             const std::string& /* conversationId */,
     318             :             libjami::SwarmMessage message) {
     319           0 :             if (accountId == aliceId) {
     320           0 :                 aliceData.messagesUpdated.emplace_back(message);
     321           0 :             } else if (accountId == bobId) {
     322           0 :                 bobData.messagesUpdated.emplace_back(message);
     323             :             }
     324           0 :             cv.notify_one();
     325           0 :         }));
     326          13 :     confHandlers.insert(
     327          26 :         libjami::exportable_callback<libjami::ConversationSignal::OnConversationError>(
     328           0 :             [&](const std::string& accountId,
     329             :                 const std::string& /* conversationId */,
     330             :                 int /*code*/,
     331             :                 const std::string& /* what */) {
     332           0 :                 if (accountId == aliceId)
     333           0 :                     aliceData.errorDetected = true;
     334           0 :                 else if (accountId == bobId)
     335           0 :                     bobData.errorDetected = true;
     336           0 :                 cv.notify_one();
     337           0 :             }));
     338          13 :     confHandlers.insert(
     339          26 :         libjami::exportable_callback<libjami::ConversationSignal::ConversationRemoved>(
     340           2 :             [&](const std::string& accountId, const std::string&) {
     341           2 :                 if (accountId == aliceId)
     342           1 :                     aliceData.removed = true;
     343           1 :                 else if (accountId == bobId)
     344           0 :                     bobData.removed = true;
     345           1 :                 else if (accountId == alice2Id)
     346           1 :                     alice2Data.removed = true;
     347           2 :                 cv.notify_one();
     348           2 :             }));
     349          13 :     libjami::registerSignalHandlers(confHandlers);
     350          13 : }
     351             : 
     352             : void
     353           1 : SyncHistoryTest::testCreateConversationThenSync()
     354             : {
     355           1 :     connectSignals();
     356             : 
     357           1 :     auto aliceAccount = Manager::instance().getAccount<JamiAccount>(aliceId);
     358             :     // Start conversation
     359           1 :     auto convId = libjami::startConversation(aliceId);
     360             : 
     361             :     // Now create alice2
     362           2 :     auto aliceArchive = std::filesystem::current_path().string() + "/alice.gz";
     363           1 :     aliceAccount->exportArchive(aliceArchive);
     364           2 :     std::map<std::string, std::string> details = libjami::getAccountTemplate("RING");
     365           1 :     details[ConfProperties::TYPE] = "RING";
     366           1 :     details[ConfProperties::DISPLAYNAME] = "ALICE2";
     367           1 :     details[ConfProperties::ALIAS] = "ALICE2";
     368           1 :     details[ConfProperties::UPNP_ENABLED] = "true";
     369           1 :     details[ConfProperties::ARCHIVE_PASSWORD] = "";
     370           1 :     details[ConfProperties::ARCHIVE_PIN] = "";
     371           1 :     details[ConfProperties::ARCHIVE_PATH] = aliceArchive;
     372             : 
     373           1 :     alice2Id = Manager::instance().addAccount(details);
     374           6 :     CPPUNIT_ASSERT(cv.wait_for(lk, 30s, [&] { return !alice2Data.conversationId.empty(); }));
     375           1 : }
     376             : 
     377             : void
     378           1 : SyncHistoryTest::testCreateConversationWithOnlineDevice()
     379             : {
     380           1 :     connectSignals();
     381           1 :     auto aliceAccount = Manager::instance().getAccount<JamiAccount>(aliceId);
     382             : 
     383             :     // Now create alice2
     384           2 :     auto aliceArchive = std::filesystem::current_path().string() + "/alice.gz";
     385           1 :     aliceAccount->exportArchive(aliceArchive);
     386           2 :     std::map<std::string, std::string> details = libjami::getAccountTemplate("RING");
     387           1 :     details[ConfProperties::TYPE] = "RING";
     388           1 :     details[ConfProperties::DISPLAYNAME] = "ALICE2";
     389           1 :     details[ConfProperties::ALIAS] = "ALICE2";
     390           1 :     details[ConfProperties::UPNP_ENABLED] = "true";
     391           1 :     details[ConfProperties::ARCHIVE_PASSWORD] = "";
     392           1 :     details[ConfProperties::ARCHIVE_PIN] = "";
     393           1 :     details[ConfProperties::ARCHIVE_PATH] = aliceArchive;
     394           1 :     auto convId = libjami::startConversation(aliceId);
     395           1 :     alice2Id = Manager::instance().addAccount(details);
     396           6 :     CPPUNIT_ASSERT(cv.wait_for(lk, 30s, [&] { return !alice2Data.conversationId.empty(); }));
     397           1 : }
     398             : 
     399             : void
     400           1 : SyncHistoryTest::testCreateConversationWithMessagesThenAddDevice()
     401             : {
     402           1 :     connectSignals();
     403           1 :     auto aliceAccount = Manager::instance().getAccount<JamiAccount>(aliceId);
     404           1 :     auto convId = libjami::startConversation(aliceId);
     405             : 
     406             :     // Start conversation
     407           1 :     auto aliceMsgSize = aliceData.messages.size();
     408           1 :     libjami::sendMessage(aliceId, convId, std::string("Message 1"), "");
     409           3 :     CPPUNIT_ASSERT(cv.wait_for(lk, 10s, [&] { return aliceMsgSize + 1 == aliceData.messages.size(); }));
     410           1 :     libjami::sendMessage(aliceId, convId, std::string("Message 2"), "");
     411           3 :     CPPUNIT_ASSERT(cv.wait_for(lk, 10s, [&] { return aliceMsgSize + 2 == aliceData.messages.size(); }));
     412           1 :     libjami::sendMessage(aliceId, convId, std::string("Message 3"), "");
     413           3 :     CPPUNIT_ASSERT(cv.wait_for(lk, 10s, [&] { return aliceMsgSize + 3 == aliceData.messages.size(); }));
     414             : 
     415             :     // Now create alice2
     416           2 :     auto aliceArchive = std::filesystem::current_path().string() + "/alice.gz";
     417           1 :     aliceAccount->exportArchive(aliceArchive);
     418           2 :     std::map<std::string, std::string> details = libjami::getAccountTemplate("RING");
     419           1 :     details[ConfProperties::TYPE] = "RING";
     420           1 :     details[ConfProperties::DISPLAYNAME] = "ALICE2";
     421           1 :     details[ConfProperties::ALIAS] = "ALICE2";
     422           1 :     details[ConfProperties::UPNP_ENABLED] = "true";
     423           1 :     details[ConfProperties::ARCHIVE_PASSWORD] = "";
     424           1 :     details[ConfProperties::ARCHIVE_PIN] = "";
     425           1 :     details[ConfProperties::ARCHIVE_PATH] = aliceArchive;
     426           1 :     alice2Id = Manager::instance().addAccount(details);
     427           6 :     CPPUNIT_ASSERT(cv.wait_for(lk, 30s, [&] { return !alice2Data.conversationId.empty(); }));
     428             : 
     429           1 :     libjami::loadConversation(alice2Id, convId, "", 0);
     430           3 :     CPPUNIT_ASSERT(cv.wait_for(lk, 30s, [&] { return alice2Data.messagesLoaded.size() == 4; }));
     431             : 
     432             :     // Check messages
     433           1 :     CPPUNIT_ASSERT(alice2Data.messagesLoaded[0].body["body"] == "Message 3");
     434           1 :     CPPUNIT_ASSERT(alice2Data.messagesLoaded[1].body["body"] == "Message 2");
     435           1 :     CPPUNIT_ASSERT(alice2Data.messagesLoaded[2].body["body"] == "Message 1");
     436           1 : }
     437             : 
     438             : void
     439           1 : SyncHistoryTest::testCreateMultipleConversationThenAddDevice()
     440             : {
     441           1 :     auto aliceAccount = Manager::instance().getAccount<JamiAccount>(aliceId);
     442             :     // Start conversation
     443           1 :     auto convId = libjami::startConversation(aliceId);
     444           1 :     libjami::sendMessage(aliceId, convId, std::string("Message 1"), "");
     445           1 :     libjami::sendMessage(aliceId, convId, std::string("Message 2"), "");
     446           1 :     libjami::sendMessage(aliceId, convId, std::string("Message 3"), "");
     447           1 :     std::this_thread::sleep_for(1s);
     448           1 :     auto convId2 = libjami::startConversation(aliceId);
     449           1 :     libjami::sendMessage(aliceId, convId2, std::string("Message 1"), "");
     450           1 :     libjami::sendMessage(aliceId, convId2, std::string("Message 2"), "");
     451           1 :     libjami::sendMessage(aliceId, convId2, std::string("Message 3"), "");
     452           1 :     std::this_thread::sleep_for(1s);
     453           1 :     auto convId3 = libjami::startConversation(aliceId);
     454           1 :     libjami::sendMessage(aliceId, convId3, std::string("Message 1"), "");
     455           1 :     libjami::sendMessage(aliceId, convId3, std::string("Message 2"), "");
     456           1 :     libjami::sendMessage(aliceId, convId3, std::string("Message 3"), "");
     457           1 :     std::this_thread::sleep_for(1s);
     458           1 :     auto convId4 = libjami::startConversation(aliceId);
     459           1 :     libjami::sendMessage(aliceId, convId4, std::string("Message 1"), "");
     460           1 :     libjami::sendMessage(aliceId, convId4, std::string("Message 2"), "");
     461           1 :     libjami::sendMessage(aliceId, convId4, std::string("Message 3"), "");
     462             : 
     463             :     // Now create alice2
     464           2 :     auto aliceArchive = std::filesystem::current_path().string() + "/alice.gz";
     465           1 :     aliceAccount->exportArchive(aliceArchive);
     466           2 :     std::map<std::string, std::string> details = libjami::getAccountTemplate("RING");
     467           1 :     details[ConfProperties::TYPE] = "RING";
     468           1 :     details[ConfProperties::DISPLAYNAME] = "ALICE2";
     469           1 :     details[ConfProperties::ALIAS] = "ALICE2";
     470           1 :     details[ConfProperties::UPNP_ENABLED] = "true";
     471           1 :     details[ConfProperties::ARCHIVE_PASSWORD] = "";
     472           1 :     details[ConfProperties::ARCHIVE_PIN] = "";
     473           1 :     details[ConfProperties::ARCHIVE_PATH] = aliceArchive;
     474           1 :     alice2Id = Manager::instance().addAccount(details);
     475             : 
     476           1 :     std::mutex mtx;
     477           1 :     std::unique_lock lk {mtx};
     478           1 :     std::condition_variable cv;
     479           1 :     std::map<std::string, std::shared_ptr<libjami::CallbackWrapperBase>> confHandlers;
     480           1 :     std::atomic_int conversationReady = 0;
     481           1 :     confHandlers.insert(libjami::exportable_callback<libjami::ConversationSignal::ConversationReady>(
     482           4 :         [&](const std::string& accountId, const std::string&) {
     483           4 :             if (accountId == alice2Id) {
     484           4 :                 conversationReady += 1;
     485           4 :                 cv.notify_one();
     486             :             }
     487           4 :         }));
     488           1 :     libjami::registerSignalHandlers(confHandlers);
     489           1 :     confHandlers.clear();
     490             : 
     491             :     // Check if conversation is ready
     492           6 :     CPPUNIT_ASSERT(cv.wait_for(lk, 60s, [&]() { return conversationReady == 4; }));
     493           1 : }
     494             : 
     495             : void
     496           1 : SyncHistoryTest::testReceivesInviteThenAddDevice()
     497             : {
     498           1 :     connectSignals();
     499           1 :     auto aliceAccount = Manager::instance().getAccount<JamiAccount>(aliceId);
     500             : 
     501             :     // Export alice
     502           2 :     auto aliceArchive = std::filesystem::current_path().string() + "/alice.gz";
     503           1 :     aliceAccount->exportArchive(aliceArchive);
     504           1 :     auto bobAccount = Manager::instance().getAccount<JamiAccount>(bobId);
     505           1 :     auto uri = aliceAccount->getUsername();
     506             : 
     507             :     // Start conversation for Alice
     508           1 :     auto convId = libjami::startConversation(bobId);
     509             : 
     510           1 :     libjami::addConversationMember(bobId, convId, uri);
     511          10 :     CPPUNIT_ASSERT(cv.wait_for(lk, 30s, [&] { return aliceData.requestReceived; }));
     512             : 
     513             :     // Now create alice2
     514           2 :     std::map<std::string, std::string> details = libjami::getAccountTemplate("RING");
     515           1 :     details[ConfProperties::TYPE] = "RING";
     516           1 :     details[ConfProperties::DISPLAYNAME] = "ALICE2";
     517           1 :     details[ConfProperties::ALIAS] = "ALICE2";
     518           1 :     details[ConfProperties::UPNP_ENABLED] = "true";
     519           1 :     details[ConfProperties::ARCHIVE_PASSWORD] = "";
     520           1 :     details[ConfProperties::ARCHIVE_PIN] = "";
     521           1 :     details[ConfProperties::ARCHIVE_PATH] = aliceArchive;
     522           1 :     alice2Id = Manager::instance().addAccount(details);
     523             : 
     524          18 :     CPPUNIT_ASSERT(cv.wait_for(lk, 30s, [&] { return alice2Data.requestReceived; }));
     525           1 : }
     526             : 
     527             : void
     528           1 : SyncHistoryTest::testRemoveConversationOnAllDevices()
     529             : {
     530           1 :     connectSignals();
     531           1 :     auto aliceAccount = Manager::instance().getAccount<JamiAccount>(aliceId);
     532             : 
     533             :     // Now create alice2
     534           2 :     auto aliceArchive = std::filesystem::current_path().string() + "/alice.gz";
     535           1 :     aliceAccount->exportArchive(aliceArchive);
     536           2 :     std::map<std::string, std::string> details = libjami::getAccountTemplate("RING");
     537           1 :     details[ConfProperties::TYPE] = "RING";
     538           1 :     details[ConfProperties::DISPLAYNAME] = "ALICE2";
     539           1 :     details[ConfProperties::ALIAS] = "ALICE2";
     540           1 :     details[ConfProperties::UPNP_ENABLED] = "true";
     541           1 :     details[ConfProperties::ARCHIVE_PASSWORD] = "";
     542           1 :     details[ConfProperties::ARCHIVE_PIN] = "";
     543           1 :     details[ConfProperties::ARCHIVE_PATH] = aliceArchive;
     544           1 :     auto convId = libjami::startConversation(aliceId);
     545             : 
     546           1 :     alice2Id = Manager::instance().addAccount(details);
     547           6 :     CPPUNIT_ASSERT(cv.wait_for(lk, 60s, [&] { return !alice2Data.conversationId.empty(); }));
     548           1 :     libjami::removeConversation(aliceId, aliceData.conversationId);
     549           3 :     CPPUNIT_ASSERT(cv.wait_for(lk, 30s, [&] { return alice2Data.removed; }));
     550           1 : }
     551             : 
     552             : void
     553           1 : SyncHistoryTest::testSyncCreateAccountExportDeleteReimportOldBackup()
     554             : {
     555           1 :     connectSignals();
     556           1 :     auto aliceAccount = Manager::instance().getAccount<JamiAccount>(aliceId);
     557           1 :     auto bobAccount = Manager::instance().getAccount<JamiAccount>(bobId);
     558           1 :     auto bobUri = bobAccount->getUsername();
     559             : 
     560             :     // Backup alice before start conversation, worst scenario for invites
     561           2 :     auto aliceArchive = std::filesystem::current_path().string() + "/alice.gz";
     562           1 :     aliceAccount->exportArchive(aliceArchive);
     563             : 
     564             :     // Start conversation
     565           1 :     auto convId = libjami::startConversation(aliceId);
     566             : 
     567           1 :     libjami::addConversationMember(aliceId, convId, bobUri);
     568           8 :     CPPUNIT_ASSERT(cv.wait_for(lk, 30s, [&]() { return bobData.requestReceived; }));
     569           1 :     auto aliceMsgSize = aliceData.messages.size();
     570           1 :     libjami::acceptConversationRequest(bobId, convId);
     571           5 :     CPPUNIT_ASSERT(cv.wait_for(lk, 30s, [&]() { return aliceMsgSize + 1 == aliceData.messages.size() && !bobData.conversationId.empty(); }));
     572             : 
     573             :     // disable account (same as removed)
     574           1 :     Manager::instance().sendRegister(aliceId, false);
     575           1 :     std::this_thread::sleep_for(5s);
     576             : 
     577           2 :     std::map<std::string, std::string> details = libjami::getAccountTemplate("RING");
     578           1 :     details[ConfProperties::TYPE] = "RING";
     579           1 :     details[ConfProperties::DISPLAYNAME] = "ALICE2";
     580           1 :     details[ConfProperties::ALIAS] = "ALICE2";
     581           1 :     details[ConfProperties::UPNP_ENABLED] = "true";
     582           1 :     details[ConfProperties::ARCHIVE_PASSWORD] = "";
     583           1 :     details[ConfProperties::ARCHIVE_PIN] = "";
     584           1 :     details[ConfProperties::ARCHIVE_PATH] = aliceArchive;
     585           1 :     alice2Id = Manager::instance().addAccount(details);
     586           6 :     CPPUNIT_ASSERT(cv.wait_for(lk, 30s, [&] { return alice2Data.deviceAnnounced; }));
     587             : 
     588             :     // This will trigger a conversation request. Cause alice2 is unable to know of first conversation
     589           1 :     libjami::sendMessage(bobId, convId, std::string("hi"), "");
     590          10 :     CPPUNIT_ASSERT(cv.wait_for(lk, 30s, [&] { return alice2Data.requestReceived; }));
     591             : 
     592           1 :     libjami::acceptConversationRequest(alice2Id, convId);
     593           4 :     CPPUNIT_ASSERT(cv.wait_for(lk, 30s, [&] { return !alice2Data.conversationId.empty(); }));
     594             : 
     595           1 :     auto bobMsgSize = bobData.messages.size();
     596           1 :     libjami::sendMessage(alice2Id, convId, std::string("hi"), "");
     597           5 :     CPPUNIT_ASSERT(cv.wait_for(lk, 30s, [&]() { return bobMsgSize + 1 == bobData.messages.size(); }));
     598           1 : }
     599             : 
     600             : void
     601           1 : SyncHistoryTest::testSyncCreateAccountExportDeleteReimportWithConvId()
     602             : {
     603           1 :     connectSignals();
     604           1 :     auto aliceAccount = Manager::instance().getAccount<JamiAccount>(aliceId);
     605           1 :     auto bobAccount = Manager::instance().getAccount<JamiAccount>(bobId);
     606           1 :     auto aliceUri = aliceAccount->getUsername();
     607           1 :     auto bobUri = bobAccount->getUsername();
     608             : 
     609             :     // Start conversation
     610           1 :     auto convId = libjami::startConversation(aliceId);
     611             : 
     612           1 :     libjami::addConversationMember(aliceId, convId, bobUri);
     613          10 :     CPPUNIT_ASSERT(cv.wait_for(lk, 30s, [&]() { return bobData.requestReceived; }));
     614             : 
     615           1 :     auto aliceMsgSize = aliceData.messages.size();
     616           1 :     libjami::acceptConversationRequest(bobId, convId);
     617           5 :     CPPUNIT_ASSERT(cv.wait_for(lk, 30s, [&]() { return aliceMsgSize + 1 == aliceData.messages.size(); }));
     618             : 
     619             :     // We need to track presence to know when to sync
     620           1 :     bobAccount->trackBuddyPresence(aliceUri, true);
     621             : 
     622             :     // Backup alice after startConversation with member
     623           2 :     auto aliceArchive = std::filesystem::current_path().string() + "/alice.gz";
     624           1 :     aliceAccount->exportArchive(aliceArchive);
     625             : 
     626             :     // disable account (same as removed)
     627           1 :     Manager::instance().sendRegister(aliceId, false);
     628           1 :     std::this_thread::sleep_for(5s);
     629             : 
     630           2 :     std::map<std::string, std::string> details = libjami::getAccountTemplate("RING");
     631           1 :     details[ConfProperties::TYPE] = "RING";
     632           1 :     details[ConfProperties::DISPLAYNAME] = "ALICE2";
     633           1 :     details[ConfProperties::ALIAS] = "ALICE2";
     634           1 :     details[ConfProperties::UPNP_ENABLED] = "true";
     635           1 :     details[ConfProperties::ARCHIVE_PASSWORD] = "";
     636           1 :     details[ConfProperties::ARCHIVE_PIN] = "";
     637           1 :     details[ConfProperties::ARCHIVE_PATH] = aliceArchive;
     638           1 :     alice2Id = Manager::instance().addAccount(details);
     639             :     // Should retrieve conversation, no need for action as the convInfos is in the archive
     640          10 :     CPPUNIT_ASSERT(cv.wait_for(lk, 30s, [&] { return !alice2Data.conversationId.empty(); }));
     641             : 
     642           1 :     auto bobMsgSize = bobData.messages.size();
     643           1 :     libjami::sendMessage(alice2Id, convId, std::string("hi"), "");
     644           5 :     cv.wait_for(lk, 30s, [&]() { return bobMsgSize + 1 == bobData.messages.size(); });
     645           1 : }
     646             : 
     647             : void
     648           1 : SyncHistoryTest::testSyncCreateAccountExportDeleteReimportWithConvReq()
     649             : {
     650           1 :     connectSignals();
     651           1 :     auto aliceAccount = Manager::instance().getAccount<JamiAccount>(aliceId);
     652           1 :     auto aliceUri = aliceAccount->getUsername();
     653             : 
     654             :     // Start conversation
     655           1 :     auto convId = libjami::startConversation(bobId);
     656             : 
     657           1 :     libjami::addConversationMember(bobId, convId, aliceUri);
     658           9 :     CPPUNIT_ASSERT(cv.wait_for(lk, 30s, [&]() { return aliceData.requestReceived; }));
     659             : 
     660             :     // Backup alice after startConversation with member
     661           2 :     auto aliceArchive = std::filesystem::current_path().string() + "/alice.gz";
     662           1 :     aliceAccount->exportArchive(aliceArchive);
     663             : 
     664             :     // disable account (same as removed)
     665           1 :     Manager::instance().sendRegister(aliceId, false);
     666           1 :     std::this_thread::sleep_for(5s);
     667             : 
     668           2 :     std::map<std::string, std::string> details = libjami::getAccountTemplate("RING");
     669           1 :     details[ConfProperties::TYPE] = "RING";
     670           1 :     details[ConfProperties::DISPLAYNAME] = "ALICE2";
     671           1 :     details[ConfProperties::ALIAS] = "ALICE2";
     672           1 :     details[ConfProperties::UPNP_ENABLED] = "true";
     673           1 :     details[ConfProperties::ARCHIVE_PASSWORD] = "";
     674           1 :     details[ConfProperties::ARCHIVE_PIN] = "";
     675           1 :     details[ConfProperties::ARCHIVE_PATH] = aliceArchive;
     676           1 :     alice2Id = Manager::instance().addAccount(details);
     677           6 :     CPPUNIT_ASSERT(cv.wait_for(lk, 30s, [&] { return alice2Data.deviceAnnounced; }));
     678             : 
     679             :     // Should get the same request as before.
     680           1 :     auto bobMsgSize = bobData.messages.size();
     681           1 :     libjami::acceptConversationRequest(alice2Id, convId);
     682           8 :     CPPUNIT_ASSERT(
     683             :         cv.wait_for(lk, 30s, [&]() { return bobMsgSize + 1 == bobData.messages.size(); }));
     684           1 : }
     685             : 
     686             : void
     687           1 : SyncHistoryTest::testSyncOneToOne()
     688             : {
     689           1 :     connectSignals();
     690           1 :     auto aliceAccount = Manager::instance().getAccount<JamiAccount>(aliceId);
     691           1 :     auto bobAccount = Manager::instance().getAccount<JamiAccount>(bobId);
     692             : 
     693           1 :     aliceAccount->addContact(bobAccount->getUsername());
     694             :     // Now create alice2
     695           2 :     auto aliceArchive = std::filesystem::current_path().string() + "/alice.gz";
     696           1 :     aliceAccount->exportArchive(aliceArchive);
     697           2 :     std::map<std::string, std::string> details = libjami::getAccountTemplate("RING");
     698           1 :     details[ConfProperties::TYPE] = "RING";
     699           1 :     details[ConfProperties::DISPLAYNAME] = "ALICE2";
     700           1 :     details[ConfProperties::ALIAS] = "ALICE2";
     701           1 :     details[ConfProperties::UPNP_ENABLED] = "true";
     702           1 :     details[ConfProperties::ARCHIVE_PASSWORD] = "";
     703           1 :     details[ConfProperties::ARCHIVE_PIN] = "";
     704           1 :     details[ConfProperties::ARCHIVE_PATH] = aliceArchive;
     705             : 
     706           1 :     alice2Id = Manager::instance().addAccount(details);
     707           6 :     CPPUNIT_ASSERT(cv.wait_for(lk, 30s, [&] { return !alice2Data.conversationId.empty(); }));
     708           1 : }
     709             : 
     710             : void
     711           1 : SyncHistoryTest::testConversationRequestRemoved()
     712             : {
     713           1 :     connectSignals();
     714           1 :     auto aliceAccount = Manager::instance().getAccount<JamiAccount>(aliceId);
     715           1 :     auto uri = aliceAccount->getUsername();
     716             : 
     717             :     // Export alice
     718           2 :     auto aliceArchive = std::filesystem::current_path().string() + "/alice.gz";
     719           1 :     aliceAccount->exportArchive(aliceArchive);
     720             : 
     721           1 :     auto bobAccount = Manager::instance().getAccount<JamiAccount>(bobId);
     722             : 
     723             :     // Start conversation for Alice
     724           1 :     auto convId = libjami::startConversation(bobId);
     725             : 
     726             :     // Check that alice receives the request
     727           1 :     libjami::addConversationMember(bobId, convId, uri);
     728          10 :     CPPUNIT_ASSERT(cv.wait_for(lk, 30s, [&] { return aliceData.requestReceived; }));
     729             : 
     730             :     // Now create alice2
     731           2 :     std::map<std::string, std::string> details = libjami::getAccountTemplate("RING");
     732           1 :     details[ConfProperties::TYPE] = "RING";
     733           1 :     details[ConfProperties::DISPLAYNAME] = "ALICE2";
     734           1 :     details[ConfProperties::ALIAS] = "ALICE2";
     735           1 :     details[ConfProperties::UPNP_ENABLED] = "true";
     736           1 :     details[ConfProperties::ARCHIVE_PASSWORD] = "";
     737           1 :     details[ConfProperties::ARCHIVE_PIN] = "";
     738           1 :     details[ConfProperties::ARCHIVE_PATH] = aliceArchive;
     739           1 :     alice2Id = Manager::instance().addAccount(details);
     740             : 
     741           2 :     CPPUNIT_ASSERT(cv.wait_for(lk, 30s, [&] { return aliceData.requestReceived; }));
     742             :     // Now decline trust request, this should trigger ConversationRequestDeclined both sides for Alice
     743           1 :     libjami::declineConversationRequest(aliceId, convId);
     744             : 
     745           6 :     CPPUNIT_ASSERT(cv.wait_for(lk, 30s, [&] { return aliceData.requestRemoved && alice2Data.requestRemoved; }));
     746           1 : }
     747             : 
     748             : void
     749           1 : SyncHistoryTest::testProfileReceivedMultiDevice()
     750             : {
     751           1 :     connectSignals();
     752           1 :     auto aliceAccount = Manager::instance().getAccount<JamiAccount>(aliceId);
     753           1 :     auto bobAccount = Manager::instance().getAccount<JamiAccount>(bobId);
     754           1 :     auto aliceUri = aliceAccount->getUsername();
     755           1 :     auto bobUri = bobAccount->getUsername();
     756             : 
     757             :     // Export alice
     758           2 :     auto aliceArchive = std::filesystem::current_path().string() + "/alice.gz";
     759           1 :     aliceAccount->exportArchive(aliceArchive);
     760             : 
     761             :     // Set VCards
     762           1 :     std::string vcard = "BEGIN:VCARD\n\
     763             : VERSION:2.1\n\
     764             : FN:TITLE\n\
     765             : DESCRIPTION:DESC\n\
     766             : END:VCARD";
     767           2 :     auto alicePath = fileutils::get_data_dir() / aliceId
     768           2 :                      / "profile.vcf";
     769           2 :     auto bobPath = fileutils::get_data_dir() / bobId
     770           2 :                    / "profile.vcf";
     771             :     // Save VCard
     772           1 :     auto p = std::filesystem::path(alicePath);
     773           1 :     dhtnet::fileutils::recursive_mkdir(p.parent_path());
     774           1 :     std::ofstream aliceFile(alicePath);
     775           1 :     if (aliceFile.is_open()) {
     776           1 :         aliceFile << vcard;
     777           1 :         aliceFile.close();
     778             :     }
     779           1 :     p = std::filesystem::path(bobPath);
     780           1 :     dhtnet::fileutils::recursive_mkdir(p.parent_path());
     781           1 :     std::ofstream bobFile(bobPath);
     782           1 :     if (bobFile.is_open()) {
     783           1 :         bobFile << vcard;
     784           1 :         bobFile.close();
     785             :     }
     786             : 
     787           1 :     aliceAccount->addContact(bobUri);
     788           1 :     aliceAccount->sendTrustRequest(bobUri, {});
     789           3 :     CPPUNIT_ASSERT(cv.wait_for(lk, 30s, [&]() { return bobData.requestReceived; }));
     790             : 
     791           1 :     CPPUNIT_ASSERT(bobAccount->acceptTrustRequest(aliceUri));
     792           5 :     CPPUNIT_ASSERT(cv.wait_for(lk, 30s, [&]() {
     793             :         return !bobData.profilePath.empty() && !aliceData.profilePath.empty() && !bobData.conversationId.empty();
     794             :     }));
     795           1 :     CPPUNIT_ASSERT(std::filesystem::is_regular_file(bobData.profilePath));
     796             : 
     797             :     // Now create alice2
     798           2 :     std::map<std::string, std::string> details = libjami::getAccountTemplate("RING");
     799           1 :     details[ConfProperties::TYPE] = "RING";
     800           1 :     details[ConfProperties::DISPLAYNAME] = "ALICE2";
     801           1 :     details[ConfProperties::ALIAS] = "ALICE2";
     802           1 :     details[ConfProperties::UPNP_ENABLED] = "true";
     803           1 :     details[ConfProperties::ARCHIVE_PASSWORD] = "";
     804           1 :     details[ConfProperties::ARCHIVE_PIN] = "";
     805           1 :     details[ConfProperties::ARCHIVE_PATH] = aliceArchive;
     806           1 :     bobData.profilePath = {};
     807           1 :     alice2Data.profilePath = {};
     808           1 :     alice2Id = Manager::instance().addAccount(details);
     809             : 
     810          12 :     CPPUNIT_ASSERT(cv.wait_for(lk, 60s, [&] {
     811             :         return alice2Data.deviceAnnounced && !bobData.profilePath.empty() && !alice2Data.profilePath.empty(); }));
     812           1 : }
     813             : 
     814             : void
     815           1 : SyncHistoryTest::testLastInteractionAfterClone()
     816             : {
     817           1 :     connectSignals();
     818           1 :     auto aliceAccount = Manager::instance().getAccount<JamiAccount>(aliceId);
     819           1 :     auto aliceUri = aliceAccount->getUsername();
     820           1 :     auto bobAccount = Manager::instance().getAccount<JamiAccount>(bobId);
     821           1 :     auto bobUri = bobAccount->getUsername();
     822             : 
     823           1 :     aliceAccount->addContact(bobUri);
     824           1 :     aliceAccount->sendTrustRequest(bobUri, {});
     825           3 :     CPPUNIT_ASSERT(cv.wait_for(lk, 30s, [&]() { return bobData.requestReceived; }));
     826             : 
     827           1 :     auto aliceMsgSize = aliceData.messages.size();
     828           1 :     CPPUNIT_ASSERT(bobAccount->acceptTrustRequest(aliceUri));
     829           5 :     CPPUNIT_ASSERT(cv.wait_for(lk, 30s, [&]() { return aliceMsgSize + 1 == aliceData.messages.size(); }));
     830             : 
     831             :     // Start conversation
     832           1 :     libjami::sendMessage(bobId, aliceData.conversationId, std::string("Message 1"), "");
     833           5 :     CPPUNIT_ASSERT(cv.wait_for(lk, 10s, [&] { return aliceMsgSize + 2 == aliceData.messages.size(); }));
     834           1 :     libjami::sendMessage(bobId, aliceData.conversationId, std::string("Message 2"), "");
     835           5 :     CPPUNIT_ASSERT(cv.wait_for(lk, 10s, [&] { return aliceMsgSize + 3 == aliceData.messages.size(); }));
     836           1 :     libjami::sendMessage(bobId, aliceData.conversationId, std::string("Message 3"), "");
     837           5 :     CPPUNIT_ASSERT(cv.wait_for(lk, 10s, [&] { return aliceMsgSize + 4 == aliceData.messages.size(); }));
     838             : 
     839           1 :     auto msgId = aliceData.messages.rbegin()->id;
     840           1 :     libjami::setMessageDisplayed(aliceId, "swarm:" + aliceData.conversationId, msgId, 3);
     841           3 :     CPPUNIT_ASSERT(cv.wait_for(lk, 10s, [&] { return aliceData.sent; }));
     842             : 
     843             :     // Now create alice2
     844           2 :     auto aliceArchive = std::filesystem::current_path().string() + "/alice.gz";
     845           1 :     aliceAccount->exportArchive(aliceArchive);
     846           2 :     std::map<std::string, std::string> details = libjami::getAccountTemplate("RING");
     847           1 :     details[ConfProperties::TYPE] = "RING";
     848           1 :     details[ConfProperties::DISPLAYNAME] = "ALICE2";
     849           1 :     details[ConfProperties::ALIAS] = "ALICE2";
     850           1 :     details[ConfProperties::UPNP_ENABLED] = "true";
     851           1 :     details[ConfProperties::ARCHIVE_PASSWORD] = "";
     852           1 :     details[ConfProperties::ARCHIVE_PIN] = "";
     853           1 :     details[ConfProperties::ARCHIVE_PATH] = aliceArchive;
     854           1 :     alice2Id = Manager::instance().addAccount(details);
     855             : 
     856             :     // Check if conversation is ready
     857           6 :     CPPUNIT_ASSERT(cv.wait_for(lk, 60s, [&]() { return !alice2Data.conversationId.empty(); }));
     858             :     // Check that last displayed is synched
     859           1 :     auto membersInfos = libjami::getConversationMembers(alice2Id, alice2Data.conversationId);
     860           2 :     CPPUNIT_ASSERT(std::find_if(membersInfos.begin(),
     861             :                                 membersInfos.end(),
     862             :                                 [&](auto infos) {
     863             :                                     return infos["uri"] == aliceUri
     864             :                                            && infos["lastDisplayed"] == msgId;
     865             :                                 })
     866             :                    != membersInfos.end());
     867           1 : }
     868             : 
     869             : void
     870           1 : SyncHistoryTest::testLastInteractionAfterSomeMessages()
     871             : {
     872           1 :     connectSignals();
     873           1 :     auto aliceAccount = Manager::instance().getAccount<JamiAccount>(aliceId);
     874           1 :     auto aliceUri = aliceAccount->getUsername();
     875           1 :     auto bobAccount = Manager::instance().getAccount<JamiAccount>(bobId);
     876           1 :     auto bobUri = bobAccount->getUsername();
     877             : 
     878             :     // Creates alice2
     879           2 :     auto aliceArchive = std::filesystem::current_path().string() + "/alice.gz";
     880           1 :     aliceAccount->exportArchive(aliceArchive);
     881           2 :     std::map<std::string, std::string> details = libjami::getAccountTemplate("RING");
     882           1 :     details[ConfProperties::TYPE] = "RING";
     883           1 :     details[ConfProperties::DISPLAYNAME] = "ALICE2";
     884           1 :     details[ConfProperties::ALIAS] = "ALICE2";
     885           1 :     details[ConfProperties::UPNP_ENABLED] = "true";
     886           1 :     details[ConfProperties::ARCHIVE_PASSWORD] = "";
     887           1 :     details[ConfProperties::ARCHIVE_PIN] = "";
     888           1 :     details[ConfProperties::ARCHIVE_PATH] = aliceArchive;
     889           1 :     alice2Id = Manager::instance().addAccount(details);
     890           6 :     CPPUNIT_ASSERT(cv.wait_for(lk, 30s, [&]() {return alice2Data.deviceAnnounced; }));
     891           9 :     auto getMessageFromBody = [](const auto& data, const auto& body) -> std::string {
     892           9 :         auto it = std::find_if(data.messages.begin(), data.messages.end(), [&](auto& msg) {
     893           6 :             return msg.body.find("body") != msg.body.end() && msg.body.at("body") == body; });
     894           9 :         if (it != data.messages.end()) {
     895           6 :             return it->id;
     896             :         }
     897           3 :         return {};
     898             :     };
     899          18 :     auto getMessage = [](const auto& data, const auto& mid) -> bool {
     900          63 :         return std::find_if(data.messages.begin(), data.messages.end(), [&](auto& msg) { return msg.id == mid; }) != data.messages.end();
     901             :     };
     902             : 
     903           1 :     aliceAccount->addContact(bobUri);
     904           1 :     aliceAccount->sendTrustRequest(bobUri, {});
     905           5 :     CPPUNIT_ASSERT(cv.wait_for(lk, 30s, [&]() { return bobData.requestReceived && !alice2Data.conversationId.empty(); }));
     906             : 
     907           1 :     CPPUNIT_ASSERT(bobAccount->acceptTrustRequest(aliceUri));
     908           6 :     CPPUNIT_ASSERT(
     909             :         cv.wait_for(lk, 30s, [&]() { return aliceData.members[bobUri] == 1 && alice2Data.members[bobUri] == 1; }));
     910             : 
     911             :     // Start conversation
     912           1 :     bobData.messages.clear();
     913           1 :     libjami::sendMessage(bobId, aliceData.conversationId, std::string("Message 1"), "");
     914           3 :     CPPUNIT_ASSERT(cv.wait_for(lk, 30s, [&]() { return !getMessageFromBody(bobData, "Message 1").empty(); }));
     915           1 :     auto msgId = getMessageFromBody(bobData, "Message 1");
     916           5 :     CPPUNIT_ASSERT(cv.wait_for(lk, 20s, [&] { return getMessage(aliceData, msgId) && getMessage(alice2Data, msgId); }));
     917             : 
     918           1 :     bobData.messages.clear();
     919           1 :     libjami::sendMessage(bobId, aliceData.conversationId, std::string("Message 2"), "");
     920           3 :     CPPUNIT_ASSERT(cv.wait_for(lk, 30s, [&]() { return !getMessageFromBody(bobData, "Message 2").empty(); }));
     921           1 :     msgId = getMessageFromBody(bobData, "Message 2");
     922           5 :     CPPUNIT_ASSERT(cv.wait_for(lk, 10s, [&] { return getMessage(aliceData, msgId) && getMessage(alice2Data, msgId); }));
     923             : 
     924           1 :     bobData.messages.clear();
     925           1 :     libjami::sendMessage(bobId, aliceData.conversationId, std::string("Message 3"), "");
     926           3 :     CPPUNIT_ASSERT(cv.wait_for(lk, 30s, [&]() { return !getMessageFromBody(bobData, "Message 3").empty(); }));
     927           1 :     msgId = getMessageFromBody(bobData, "Message 3");
     928           5 :     CPPUNIT_ASSERT(cv.wait_for(lk, 10s, [&] { return getMessage(aliceData, msgId) && getMessage(alice2Data, msgId); }));
     929             : 
     930           1 :     libjami::setMessageDisplayed(aliceId, "swarm:" + aliceData.conversationId, msgId, 3);
     931           4 :     CPPUNIT_ASSERT(cv.wait_for(lk, 20s, [&] {
     932             :         return aliceData.sent && alice2Data.sent; }));
     933             : 
     934           1 :     auto membersInfos = libjami::getConversationMembers(alice2Id, alice2Data.conversationId);
     935           2 :     CPPUNIT_ASSERT(std::find_if(membersInfos.begin(),
     936             :                                 membersInfos.end(),
     937             :                                 [&](auto infos) {
     938             :                                     return infos["uri"] == aliceUri
     939             :                                            && infos["lastDisplayed"] == msgId;
     940             :                                 })
     941             :                    != membersInfos.end());
     942           1 : }
     943             : 
     944             : } // namespace test
     945             : } // namespace jami
     946             : 
     947           1 : RING_TEST_RUNNER(jami::test::SyncHistoryTest::name())

Generated by: LCOV version 1.14