mirror of
https://git.jami.net/savoirfairelinux/jami-daemon.git
synced 2025-08-12 22:09:25 +08:00

Load actors from YAML description for unit tests. Change-Id: Id1840e2bd7244abf902c9be95589b82503c72840
295 lines
9.8 KiB
C++
295 lines
9.8 KiB
C++
/*
|
|
* Copyright (C)2020-2021 Savoir-faire Linux Inc.
|
|
* Author: Sébastien Blin <sebastien.blin@savoirfairelinux.com>
|
|
*
|
|
* 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 <string>
|
|
|
|
#include "manager.h"
|
|
#include "jamidht/connectionmanager.h"
|
|
#include "jamidht/jamiaccount.h"
|
|
#include "../../test_runner.h"
|
|
#include "dring.h"
|
|
#include "data_transfer.h"
|
|
#include "dring/datatransfer_interface.h"
|
|
#include "account_const.h"
|
|
#include "common.h"
|
|
|
|
using namespace DRing::Account;
|
|
|
|
namespace jami {
|
|
namespace test {
|
|
|
|
class FileTransferTest : public CppUnit::TestFixture
|
|
{
|
|
public:
|
|
FileTransferTest()
|
|
{
|
|
// Init daemon
|
|
DRing::init(DRing::InitFlag(DRing::DRING_FLAG_DEBUG | DRing::DRING_FLAG_CONSOLE_LOG));
|
|
if (not Manager::instance().initialized)
|
|
CPPUNIT_ASSERT(DRing::start("dring-sample.yml"));
|
|
}
|
|
~FileTransferTest() { DRing::fini(); }
|
|
static std::string name() { return "Call"; }
|
|
bool compare(const std::string& fileA, const std::string& fileB) const;
|
|
void setUp();
|
|
void tearDown();
|
|
|
|
std::string aliceId;
|
|
std::string bobId;
|
|
|
|
private:
|
|
void testCachedFileTransfer();
|
|
void testMultipleFileTransfer();
|
|
|
|
CPPUNIT_TEST_SUITE(FileTransferTest);
|
|
CPPUNIT_TEST(testCachedFileTransfer);
|
|
CPPUNIT_TEST(testMultipleFileTransfer);
|
|
CPPUNIT_TEST_SUITE_END();
|
|
};
|
|
|
|
CPPUNIT_TEST_SUITE_NAMED_REGISTRATION(FileTransferTest, FileTransferTest::name());
|
|
|
|
bool
|
|
FileTransferTest::compare(const std::string& fileA, const std::string& fileB) const
|
|
{
|
|
std::ifstream f1(fileA, std::ifstream::binary | std::ifstream::ate);
|
|
std::ifstream f2(fileB, std::ifstream::binary | std::ifstream::ate);
|
|
|
|
if (f1.fail() || f2.fail() || f1.tellg() != f2.tellg()) {
|
|
return false;
|
|
}
|
|
|
|
f1.seekg(0, std::ifstream::beg);
|
|
f2.seekg(0, std::ifstream::beg);
|
|
return std::equal(std::istreambuf_iterator<char>(f1.rdbuf()),
|
|
std::istreambuf_iterator<char>(),
|
|
std::istreambuf_iterator<char>(f2.rdbuf()));
|
|
}
|
|
|
|
void
|
|
FileTransferTest::setUp()
|
|
{
|
|
auto actors = load_actors_and_wait_for_announcement("actors/alice-bob.yml");
|
|
aliceId = actors["alice"];
|
|
bobId = actors["bob"];
|
|
}
|
|
|
|
void
|
|
FileTransferTest::tearDown()
|
|
{
|
|
JAMI_INFO("Remove created accounts...");
|
|
|
|
std::map<std::string, std::shared_ptr<DRing::CallbackWrapperBase>> confHandlers;
|
|
std::mutex mtx;
|
|
std::unique_lock<std::mutex> lk {mtx};
|
|
std::condition_variable cv;
|
|
auto currentAccSize = Manager::instance().getAccountList().size();
|
|
confHandlers.insert(
|
|
DRing::exportable_callback<DRing::ConfigurationSignal::AccountsChanged>([&]() {
|
|
if (Manager::instance().getAccountList().size() <= currentAccSize - 2) {
|
|
cv.notify_one();
|
|
}
|
|
}));
|
|
DRing::registerSignalHandlers(confHandlers);
|
|
|
|
Manager::instance().removeAccount(aliceId, true);
|
|
Manager::instance().removeAccount(bobId, true);
|
|
// Because cppunit is not linked with dbus, just poll if removed
|
|
cv.wait_for(lk, std::chrono::seconds(30));
|
|
|
|
DRing::unregisterSignalHandlers();
|
|
}
|
|
|
|
void
|
|
FileTransferTest::testCachedFileTransfer()
|
|
{
|
|
auto aliceAccount = Manager::instance().getAccount<JamiAccount>(aliceId);
|
|
auto bobAccount = Manager::instance().getAccount<JamiAccount>(bobId);
|
|
auto bobUri = bobAccount->getUsername();
|
|
auto bobDeviceId = bobAccount->currentDeviceId();
|
|
auto aliceUri = aliceAccount->getUsername();
|
|
|
|
std::mutex mtx;
|
|
std::unique_lock<std::mutex> lk {mtx};
|
|
std::condition_variable cv;
|
|
std::condition_variable cv2;
|
|
std::map<std::string, std::shared_ptr<DRing::CallbackWrapperBase>> confHandlers;
|
|
bool transferWaiting = false, transferFinished = false;
|
|
DRing::DataTransferId finalId;
|
|
// Watch signals
|
|
confHandlers.insert(DRing::exportable_callback<DRing::DataTransferSignal::DataTransferEvent>(
|
|
[&](const long unsigned int& id, int code) {
|
|
if (code == static_cast<int>(DRing::DataTransferEventCode::wait_host_acceptance)) {
|
|
transferWaiting = true;
|
|
finalId = id;
|
|
cv.notify_one();
|
|
} else if (code == static_cast<int>(DRing::DataTransferEventCode::finished)) {
|
|
transferFinished = true;
|
|
finalId = id;
|
|
cv.notify_one();
|
|
}
|
|
}));
|
|
DRing::registerSignalHandlers(confHandlers);
|
|
|
|
// Create file to send
|
|
std::ofstream sendFile("SEND");
|
|
CPPUNIT_ASSERT(sendFile.is_open());
|
|
sendFile << std::string(64000, 'A');
|
|
sendFile.close();
|
|
|
|
// Send File
|
|
DRing::DataTransferInfo info;
|
|
uint64_t id;
|
|
info.accountId = aliceAccount->getAccountID();
|
|
info.peer = bobUri;
|
|
info.path = "SEND";
|
|
info.displayName = "SEND";
|
|
info.bytesProgress = 0;
|
|
CPPUNIT_ASSERT(DRing::sendFile(info, id) == DRing::DataTransferError::success);
|
|
|
|
cv.wait_for(lk, std::chrono::seconds(30));
|
|
CPPUNIT_ASSERT(transferWaiting);
|
|
|
|
auto rcv_path = "RECV";
|
|
CPPUNIT_ASSERT(DRing::acceptFileTransfer(finalId, rcv_path, 0)
|
|
== DRing::DataTransferError::success);
|
|
|
|
// Wait 2 times, both sides will got a finished status
|
|
cv.wait_for(lk, std::chrono::seconds(30));
|
|
cv.wait_for(lk, std::chrono::seconds(30));
|
|
CPPUNIT_ASSERT(transferFinished);
|
|
|
|
CPPUNIT_ASSERT(compare(info.path, rcv_path));
|
|
|
|
// TODO FIX ME. The ICE take some time to stop and it doesn't seems to like
|
|
// when stopping the daemon and removing the accounts to soon.
|
|
std::remove("SEND");
|
|
std::remove("RECV");
|
|
JAMI_INFO("Waiting....");
|
|
std::this_thread::sleep_for(std::chrono::seconds(3));
|
|
}
|
|
|
|
void
|
|
FileTransferTest::testMultipleFileTransfer()
|
|
{
|
|
auto aliceAccount = Manager::instance().getAccount<JamiAccount>(aliceId);
|
|
auto bobAccount = Manager::instance().getAccount<JamiAccount>(bobId);
|
|
auto bobUri = bobAccount->getUsername();
|
|
auto bobDeviceId = bobAccount->currentDeviceId();
|
|
auto aliceUri = aliceAccount->getUsername();
|
|
|
|
std::mutex mtx;
|
|
std::unique_lock<std::mutex> lk {mtx};
|
|
std::condition_variable cv;
|
|
std::condition_variable cv2;
|
|
std::map<std::string, std::shared_ptr<DRing::CallbackWrapperBase>> confHandlers;
|
|
bool transferWaiting = false, transferFinished = false;
|
|
DRing::DataTransferId finalId;
|
|
// Watch signals
|
|
confHandlers.insert(DRing::exportable_callback<DRing::DataTransferSignal::DataTransferEvent>(
|
|
[&](const long unsigned int& id, int code) {
|
|
if (code == static_cast<int>(DRing::DataTransferEventCode::wait_host_acceptance)) {
|
|
transferWaiting = true;
|
|
finalId = id;
|
|
cv.notify_one();
|
|
} else if (code == static_cast<int>(DRing::DataTransferEventCode::finished)) {
|
|
transferFinished = true;
|
|
finalId = id;
|
|
cv.notify_one();
|
|
}
|
|
}));
|
|
DRing::registerSignalHandlers(confHandlers);
|
|
|
|
// Create file to send
|
|
std::ofstream sendFile("SEND");
|
|
CPPUNIT_ASSERT(sendFile.is_open());
|
|
sendFile << std::string(64000, 'A');
|
|
sendFile.close();
|
|
std::ofstream sendFile2("SEND2");
|
|
CPPUNIT_ASSERT(sendFile2.is_open());
|
|
sendFile2 << std::string(64000, 'B');
|
|
sendFile2.close();
|
|
|
|
// Send first File
|
|
DRing::DataTransferInfo info;
|
|
uint64_t id;
|
|
info.accountId = aliceAccount->getAccountID();
|
|
info.peer = bobUri;
|
|
info.path = "SEND";
|
|
info.displayName = "SEND";
|
|
info.bytesProgress = 0;
|
|
CPPUNIT_ASSERT(DRing::sendFile(info, id) == DRing::DataTransferError::success);
|
|
|
|
cv.wait_for(lk, std::chrono::seconds(30));
|
|
CPPUNIT_ASSERT(transferWaiting);
|
|
transferWaiting = false;
|
|
|
|
auto rcv_path = "RECV";
|
|
CPPUNIT_ASSERT(DRing::acceptFileTransfer(finalId, rcv_path, 0)
|
|
== DRing::DataTransferError::success);
|
|
|
|
// Wait 2 times, both sides will got a finished status
|
|
cv.wait_for(lk, std::chrono::seconds(30));
|
|
cv.wait_for(lk, std::chrono::seconds(30));
|
|
CPPUNIT_ASSERT(transferFinished);
|
|
|
|
CPPUNIT_ASSERT(compare(info.path, rcv_path));
|
|
|
|
// Send File
|
|
DRing::DataTransferInfo info2;
|
|
info2.accountId = aliceAccount->getAccountID();
|
|
info2.peer = bobUri;
|
|
info2.path = "SEND2";
|
|
info2.displayName = "SEND2";
|
|
info2.bytesProgress = 0;
|
|
CPPUNIT_ASSERT(DRing::sendFile(info2, id) == DRing::DataTransferError::success);
|
|
|
|
cv.wait_for(lk, std::chrono::seconds(30));
|
|
CPPUNIT_ASSERT(transferWaiting);
|
|
|
|
rcv_path = "RECV2";
|
|
CPPUNIT_ASSERT(DRing::acceptFileTransfer(finalId, rcv_path, 0)
|
|
== DRing::DataTransferError::success);
|
|
|
|
// Wait 2 times, both sides will got a finished status
|
|
cv.wait_for(lk, std::chrono::seconds(30));
|
|
cv.wait_for(lk, std::chrono::seconds(30));
|
|
CPPUNIT_ASSERT(transferFinished);
|
|
|
|
CPPUNIT_ASSERT(compare(info2.path, rcv_path));
|
|
|
|
// TODO FIX ME. The ICE take some time to stop and it doesn't seems to like
|
|
// when stopping the daemon and removing the accounts to soon.
|
|
std::remove("SEND");
|
|
std::remove("SEND2");
|
|
std::remove("RECV");
|
|
std::remove("RECV2");
|
|
JAMI_INFO("Waiting....");
|
|
std::this_thread::sleep_for(std::chrono::seconds(3));
|
|
}
|
|
|
|
} // namespace test
|
|
} // namespace jami
|
|
|
|
RING_TEST_RUNNER(jami::test::FileTransferTest::name())
|