mirror of
https://github.com/transmission/transmission.git
synced 2026-04-18 07:56:33 +01:00
* build: overhaul tr_add_external_auto_library * build: use package-provided CMake config file libevent * build: use package-provided CMake config file miniupnpc * build: update libutp find module * build: make LIBNAME an optional parameter * build: use package-provided CMake config file libdeflate * build: update libb64 find module * build: update libnatpmp find module * build: update libpsl find module * build: support system fast_float library * chore: reformat long brew commands * build: support system fmt library * build: support system rapidjson library * build: support system small library * build: support system library utf8cpp * build: support system library WideInteger * build: support system library gtest * fix: incorrectly labeled test suites * build: remove unused parameters from tr_add_external_auto_library * build: update crc32c cmake script * fix: dht system library * fix: add libutp interface target * code review: move TrGtest.cmake * code review: move tr_get_fmt_version into Findfmt.cmake * code review: use option() for gtest * code review: move find_package(PkgConfig) out of loop * build: delete FindCrc32c.cmake Impossible to parse package version from distributed source files. * code review: Finddht.cmake * build: delete FindFastFloat.cmake Impossible to parse package version from distributed source files. * code review: Findfmt.cmake * code review: Findlibb64.cmake * code review: Findlibdeflate.cmake * code review: Findlibevent.cmake * code review: Findlibnatpmp.cmake * code review: Findlibpsl.cmake * code review: Findlibutp.cmake * code review: Findlibminiupnpc.cmake * code review: FindRapidJSON.cmake * build: delete FindSmall.cmake Impossible to parse package version from distributed source files. * build: only accept cmake config package for utf8cpp Impossible to parse package version from distributed source files. * build: delete FindWideInteger.cmake Impossible to parse package version from distributed source files. * build: add `USE_SYSTEM_DEFAULT` * ci: drop Fedora 40 and adopt Fedora 43 * ci: try to silence system header warnings * ci: use `cmake --install` * Revert "build: only accept cmake config package for utf8cpp" This reverts commit2158d631fd. * build: harden utf8cpp find module * chore: bump wide-integer Pick upbf9398f9daandbcc726a30f* refactor: gtest should be included with angled brackets Now that gtest is built as a system library, it should be included with angled brackets instead of quotes. * code review: fixup libutp variables before `find_package_handle_standard_args` * code review: define `WIDE_INTEGER_HAS_LIMB_TYPE_UINT64` only for targets depending on WideInteger * chore: bump wide-integer Pickup4b2258acacso that wide-integer tests won't run in Transmission project.
522 lines
16 KiB
C++
522 lines
16 KiB
C++
// This file Copyright (C) 2013-2022 Mnemosyne LLC.
|
|
// It may be used under GPLv2 (SPDX: GPL-2.0-only), GPLv3 (SPDX: GPL-3.0-only),
|
|
// or any future license endorsed by Mnemosyne LLC.
|
|
// License text can be found in the licenses/ folder.
|
|
|
|
#pragma once
|
|
|
|
#include <chrono>
|
|
#include <condition_variable>
|
|
#include <cstdlib> // getenv()
|
|
#include <cstring> // strlen()
|
|
#include <iostream>
|
|
#include <memory>
|
|
#include <mutex> // std::once_flag()
|
|
#include <string>
|
|
#include <string_view>
|
|
#include <thread>
|
|
|
|
#include <event2/event.h>
|
|
|
|
#include <gtest/gtest.h>
|
|
|
|
#include <libtransmission/crypto-utils.h> // tr_base64_decode()
|
|
#include <libtransmission/error.h>
|
|
#include <libtransmission/file.h> // tr_sys_file_*()
|
|
#include <libtransmission/quark.h>
|
|
#include <libtransmission/torrent-ctor.h>
|
|
#include <libtransmission/torrent.h>
|
|
#include <libtransmission/utils.h>
|
|
#include <libtransmission/variant.h>
|
|
|
|
using namespace std::literals;
|
|
|
|
inline std::ostream& operator<<(std::ostream& os, tr_error const& err)
|
|
{
|
|
os << err.message() << ' ' << err.code();
|
|
return os;
|
|
}
|
|
|
|
namespace libtransmission::test
|
|
{
|
|
|
|
using file_func_t = std::function<void(char const* filename)>;
|
|
|
|
static void depthFirstWalk(char const* path, file_func_t const& func)
|
|
{
|
|
if (auto const info = tr_sys_path_get_info(path); info && info->isFolder())
|
|
{
|
|
if (auto const odir = tr_sys_dir_open(path); odir != TR_BAD_SYS_DIR)
|
|
{
|
|
for (;;)
|
|
{
|
|
char const* const name = tr_sys_dir_read_name(odir);
|
|
if (name == nullptr)
|
|
{
|
|
break;
|
|
}
|
|
|
|
if ("."sv != name && ".."sv != name)
|
|
{
|
|
auto const child = fmt::format("{:s}/{:s}"sv, path, name);
|
|
depthFirstWalk(child.c_str(), func);
|
|
}
|
|
}
|
|
|
|
tr_sys_dir_close(odir);
|
|
}
|
|
}
|
|
|
|
func(path);
|
|
}
|
|
|
|
inline bool waitFor(std::function<bool()> const& test, std::chrono::milliseconds msec)
|
|
{
|
|
auto const deadline = std::chrono::steady_clock::now() + msec;
|
|
|
|
for (;;)
|
|
{
|
|
if (test())
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if (std::chrono::steady_clock::now() > deadline)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
std::this_thread::sleep_for(10ms);
|
|
}
|
|
}
|
|
|
|
inline bool waitFor(std::function<bool()> const& test, int msec)
|
|
{
|
|
return waitFor(test, std::chrono::milliseconds{ msec });
|
|
}
|
|
|
|
inline bool waitFor(
|
|
struct event_base* evb,
|
|
std::function<bool()> const& test,
|
|
std::chrono::milliseconds msec = std::chrono::seconds{ 5 })
|
|
{
|
|
auto const deadline = std::chrono::steady_clock::now() + msec;
|
|
|
|
for (;;)
|
|
{
|
|
if (test())
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if (std::chrono::steady_clock::now() > deadline)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
event_base_loop(evb, EVLOOP_ONCE | EVLOOP_NONBLOCK);
|
|
}
|
|
}
|
|
|
|
class TransmissionTest : public ::testing::Test
|
|
{
|
|
protected:
|
|
static void SetUpTestSuite()
|
|
{
|
|
tr_lib_init();
|
|
}
|
|
};
|
|
|
|
class Sandbox
|
|
{
|
|
public:
|
|
Sandbox()
|
|
: parent_dir_{ getDefaultParentDir() }
|
|
, sandbox_dir_{ createSandbox(parent_dir_, "transmission-test-XXXXXX") }
|
|
{
|
|
}
|
|
|
|
~Sandbox()
|
|
{
|
|
rimraf(sandbox_dir_);
|
|
}
|
|
|
|
Sandbox(Sandbox const&) = delete;
|
|
Sandbox(Sandbox&&) = delete;
|
|
Sandbox& operator=(Sandbox const&) = delete;
|
|
Sandbox& operator=(Sandbox&&) = delete;
|
|
|
|
[[nodiscard]] constexpr std::string const& path() const
|
|
{
|
|
return sandbox_dir_;
|
|
}
|
|
|
|
static std::string createSandbox(std::string const& parent_dir, std::string const& tmpl)
|
|
{
|
|
auto path = fmt::format(FMT_STRING("{:s}/{:s}"sv), tr_sys_path_resolve(parent_dir), tmpl);
|
|
tr_sys_dir_create_temp(std::data(path));
|
|
tr_sys_path_native_separators(std::data(path));
|
|
return path;
|
|
}
|
|
|
|
protected:
|
|
static std::string getDefaultParentDir()
|
|
{
|
|
if (auto* const path = getenv("TMPDIR"); path != nullptr)
|
|
{
|
|
return path;
|
|
}
|
|
|
|
auto error = tr_error{};
|
|
return tr_sys_dir_get_current(&error);
|
|
}
|
|
|
|
static void rimraf(std::string const& path, bool verbose = false)
|
|
{
|
|
auto remove = [verbose](char const* filename)
|
|
{
|
|
if (verbose)
|
|
{
|
|
std::cerr << "cleanup: removing '" << filename << "'\n";
|
|
}
|
|
|
|
tr_sys_path_remove(filename);
|
|
};
|
|
|
|
depthFirstWalk(path.c_str(), remove);
|
|
}
|
|
|
|
private:
|
|
std::string const parent_dir_;
|
|
std::string const sandbox_dir_;
|
|
};
|
|
|
|
class SandboxedTest : public TransmissionTest
|
|
{
|
|
protected:
|
|
[[nodiscard]] std::string sandboxDir() const
|
|
{
|
|
return sandbox_.path();
|
|
}
|
|
|
|
[[nodiscard]] static auto currentTestName()
|
|
{
|
|
auto const* i = ::testing::UnitTest::GetInstance()->current_test_info();
|
|
auto child = std::string(i->test_suite_name());
|
|
child += '_';
|
|
child += i->name();
|
|
return child;
|
|
}
|
|
|
|
static void buildParentDir(std::string_view path)
|
|
{
|
|
auto const tmperr = errno;
|
|
|
|
auto dir = tr_pathbuf{ path };
|
|
dir.popdir();
|
|
if (auto const info = tr_sys_path_get_info(path); !info)
|
|
{
|
|
auto error = tr_error{};
|
|
tr_sys_dir_create(dir, TR_SYS_DIR_CREATE_PARENTS, 0700, &error);
|
|
EXPECT_FALSE(error) << "path[" << path << "] dir[" << dir << "] " << error;
|
|
}
|
|
|
|
errno = tmperr;
|
|
}
|
|
|
|
static void blockingFileWrite(tr_sys_file_t fd, void const* data, size_t data_len, tr_error* error = nullptr)
|
|
{
|
|
auto local_error = tr_error{};
|
|
if (error == nullptr)
|
|
{
|
|
error = &local_error;
|
|
}
|
|
|
|
uint64_t n_left = data_len;
|
|
auto const* left = static_cast<uint8_t const*>(data);
|
|
while (n_left > 0)
|
|
{
|
|
uint64_t n = {};
|
|
if (!tr_sys_file_write(fd, left, n_left, &n, error))
|
|
{
|
|
fmt::print(stderr, "Error writing file: '{:s}'\n", error->message());
|
|
break;
|
|
}
|
|
|
|
left += n;
|
|
n_left -= n;
|
|
}
|
|
}
|
|
|
|
static void createTmpfileWithContents(char* tmpl, void const* payload, size_t n)
|
|
{
|
|
auto const tmperr = errno;
|
|
|
|
buildParentDir(tmpl);
|
|
|
|
auto error = tr_error{};
|
|
auto const fd = tr_sys_file_open_temp(tmpl, &error);
|
|
blockingFileWrite(fd, payload, n, &error);
|
|
tr_sys_file_close(fd, &error);
|
|
if (error)
|
|
{
|
|
fmt::print(
|
|
"Couldn't create '{path}': {error} ({error_code})\n",
|
|
fmt::arg("path", tmpl),
|
|
fmt::arg("error", error.message()),
|
|
fmt::arg("error_code", error.code()));
|
|
}
|
|
sync();
|
|
|
|
errno = tmperr;
|
|
}
|
|
|
|
static void createFileWithContents(std::string_view path, void const* payload, size_t n)
|
|
{
|
|
auto const tmperr = errno;
|
|
|
|
buildParentDir(path);
|
|
|
|
auto const fd = tr_sys_file_open(
|
|
tr_pathbuf{ path },
|
|
TR_SYS_FILE_WRITE | TR_SYS_FILE_CREATE | TR_SYS_FILE_TRUNCATE,
|
|
0600,
|
|
nullptr);
|
|
blockingFileWrite(fd, payload, n);
|
|
tr_sys_file_close(fd);
|
|
sync();
|
|
|
|
errno = tmperr;
|
|
}
|
|
|
|
static void createFileWithContents(std::string_view path, std::string_view payload)
|
|
{
|
|
createFileWithContents(path, std::data(payload), std::size(payload));
|
|
}
|
|
|
|
static void createFileWithContents(std::string_view path, void const* payload)
|
|
{
|
|
createFileWithContents(path, payload, strlen(static_cast<char const*>(payload)));
|
|
}
|
|
|
|
bool verbose_ = false;
|
|
|
|
static void sync()
|
|
{
|
|
#ifndef _WIN32
|
|
::sync();
|
|
#endif
|
|
}
|
|
|
|
private:
|
|
Sandbox sandbox_;
|
|
};
|
|
|
|
class SessionTest : public SandboxedTest
|
|
{
|
|
private:
|
|
std::shared_ptr<tr_variant> settings_;
|
|
|
|
tr_session* sessionInit(tr_variant& settings)
|
|
{
|
|
auto* const settings_map = settings.get_if<tr_variant::Map>();
|
|
EXPECT_NE(settings_map, nullptr);
|
|
|
|
// download dir
|
|
auto key = TR_KEY_download_dir;
|
|
auto val = settings_map->value_if<std::string_view>(key).value_or("Downloads"sv);
|
|
auto const download_dir = tr_pathbuf{ sandboxDir(), '/', val };
|
|
tr_sys_dir_create(download_dir, TR_SYS_DIR_CREATE_PARENTS, 0700);
|
|
(*settings_map)[key] = download_dir.sv();
|
|
|
|
// incomplete dir
|
|
key = TR_KEY_incomplete_dir;
|
|
val = settings_map->value_if<std::string_view>(key).value_or("Incomplete"sv);
|
|
auto const incomplete_dir = tr_pathbuf{ sandboxDir(), '/', val };
|
|
(*settings_map)[key] = incomplete_dir.sv();
|
|
|
|
// blocklists
|
|
tr_sys_dir_create(tr_pathbuf{ sandboxDir(), "/blocklists" }, TR_SYS_DIR_CREATE_PARENTS, 0700);
|
|
|
|
// fill in any missing settings
|
|
settings_map->try_emplace(TR_KEY_port_forwarding_enabled, false);
|
|
settings_map->try_emplace(TR_KEY_dht_enabled, false);
|
|
settings_map->try_emplace(TR_KEY_message_level, verbose_ ? TR_LOG_DEBUG : TR_LOG_ERROR);
|
|
|
|
return tr_sessionInit(sandboxDir(), !verbose_, settings);
|
|
}
|
|
|
|
static void sessionClose(tr_session* session)
|
|
{
|
|
tr_sessionClose(session);
|
|
tr_logFreeQueue(tr_logGetQueue());
|
|
}
|
|
|
|
protected:
|
|
enum class ZeroTorrentState : uint8_t
|
|
{
|
|
NoFiles,
|
|
Partial,
|
|
Complete
|
|
};
|
|
|
|
[[nodiscard]] tr_torrent* createTorrentAndWaitForVerifyDone(tr_ctor* ctor)
|
|
{
|
|
auto verified_lock = std::unique_lock(verified_mutex_);
|
|
auto const n_previously_verified = std::size(verified_);
|
|
|
|
ctor->set_verify_done_callback(
|
|
[this](tr_torrent* const tor)
|
|
{
|
|
auto lambda_verified_lock = std::lock_guard{ verified_mutex_ };
|
|
verified_.emplace_back(tor);
|
|
verified_cv_.notify_one();
|
|
});
|
|
|
|
auto* const tor = tr_torrentNew(ctor, nullptr);
|
|
auto const stop_waiting = [this, tor, n_previously_verified]()
|
|
{
|
|
return std::size(verified_) > n_previously_verified && verified_.back() == tor;
|
|
};
|
|
|
|
EXPECT_NE(nullptr, tor);
|
|
verified_cv_.wait_for(verified_lock, 20s, stop_waiting);
|
|
return tor;
|
|
}
|
|
|
|
[[nodiscard]] tr_torrent* zeroTorrentInit(ZeroTorrentState state)
|
|
{
|
|
// 1048576 files-filled-with-zeroes/1048576
|
|
// 4096 files-filled-with-zeroes/4096
|
|
// 512 files-filled-with-zeroes/512
|
|
static auto constexpr BencBase64 =
|
|
"ZDg6YW5ub3VuY2UzMTpodHRwOi8vd3d3LmV4YW1wbGUuY29tL2Fubm91bmNlMTA6Y3JlYXRlZCBi"
|
|
"eTI1OlRyYW5zbWlzc2lvbi8yLjYxICgxMzQwNykxMzpjcmVhdGlvbiBkYXRlaTEzNTg3MDQwNzVl"
|
|
"ODplbmNvZGluZzU6VVRGLTg0OmluZm9kNTpmaWxlc2xkNjpsZW5ndGhpMTA0ODU3NmU0OnBhdGhs"
|
|
"NzoxMDQ4NTc2ZWVkNjpsZW5ndGhpNDA5NmU0OnBhdGhsNDo0MDk2ZWVkNjpsZW5ndGhpNTEyZTQ6"
|
|
"cGF0aGwzOjUxMmVlZTQ6bmFtZTI0OmZpbGVzLWZpbGxlZC13aXRoLXplcm9lczEyOnBpZWNlIGxl"
|
|
"bmd0aGkzMjc2OGU2OnBpZWNlczY2MDpRiEMYSbRhMVL9e9umo/8KT9ZCS1GIQxhJtGExUv1726aj"
|
|
"/wpP1kJLUYhDGEm0YTFS/XvbpqP/Ck/WQktRiEMYSbRhMVL9e9umo/8KT9ZCS1GIQxhJtGExUv17"
|
|
"26aj/wpP1kJLUYhDGEm0YTFS/XvbpqP/Ck/WQktRiEMYSbRhMVL9e9umo/8KT9ZCS1GIQxhJtGEx"
|
|
"Uv1726aj/wpP1kJLUYhDGEm0YTFS/XvbpqP/Ck/WQktRiEMYSbRhMVL9e9umo/8KT9ZCS1GIQxhJ"
|
|
"tGExUv1726aj/wpP1kJLUYhDGEm0YTFS/XvbpqP/Ck/WQktRiEMYSbRhMVL9e9umo/8KT9ZCS1GI"
|
|
"QxhJtGExUv1726aj/wpP1kJLUYhDGEm0YTFS/XvbpqP/Ck/WQktRiEMYSbRhMVL9e9umo/8KT9ZC"
|
|
"S1GIQxhJtGExUv1726aj/wpP1kJLUYhDGEm0YTFS/XvbpqP/Ck/WQktRiEMYSbRhMVL9e9umo/8K"
|
|
"T9ZCS1GIQxhJtGExUv1726aj/wpP1kJLUYhDGEm0YTFS/XvbpqP/Ck/WQktRiEMYSbRhMVL9e9um"
|
|
"o/8KT9ZCS1GIQxhJtGExUv1726aj/wpP1kJLUYhDGEm0YTFS/XvbpqP/Ck/WQktRiEMYSbRhMVL9"
|
|
"e9umo/8KT9ZCS1GIQxhJtGExUv1726aj/wpP1kJLUYhDGEm0YTFS/XvbpqP/Ck/WQktRiEMYSbRh"
|
|
"MVL9e9umo/8KT9ZCS1GIQxhJtGExUv1726aj/wpP1kJLUYhDGEm0YTFS/XvbpqP/Ck/WQktRiEMY"
|
|
"SbRhMVL9e9umo/8KT9ZCS1GIQxhJtGExUv1726aj/wpP1kJLOlf5A+Tz30nMBVuNM2hpV3wg/103"
|
|
"OnByaXZhdGVpMGVlZQ==";
|
|
|
|
// create the torrent ctor
|
|
auto const benc = tr_base64_decode(BencBase64);
|
|
EXPECT_LT(0U, std::size(benc));
|
|
auto* ctor = tr_ctorNew(session_);
|
|
auto error = tr_error{};
|
|
EXPECT_TRUE(tr_ctorSetMetainfo(ctor, std::data(benc), std::size(benc), &error));
|
|
EXPECT_FALSE(error) << error;
|
|
tr_ctorSetPaused(ctor, TR_FORCE, true);
|
|
|
|
// maybe create the files
|
|
if (state != ZeroTorrentState::NoFiles)
|
|
{
|
|
auto const* const metainfo = tr_ctorGetMetainfo(ctor);
|
|
for (tr_file_index_t i = 0, n = metainfo->file_count(); i < n; ++i)
|
|
{
|
|
auto const base = state == ZeroTorrentState::Partial && tr_sessionIsIncompleteDirEnabled(session_) ?
|
|
tr_sessionGetIncompleteDir(session_) :
|
|
tr_sessionGetDownloadDir(session_);
|
|
auto const& subpath = metainfo->file_subpath(i);
|
|
auto const partial = state == ZeroTorrentState::Partial && i == 0;
|
|
auto const suffix = std::string_view{ partial ? ".part" : "" };
|
|
auto const filename = tr_pathbuf{ base, '/', subpath, suffix };
|
|
|
|
auto dirname = tr_pathbuf{ filename.sv() };
|
|
dirname.popdir();
|
|
tr_sys_dir_create(dirname, TR_SYS_DIR_CREATE_PARENTS, 0700);
|
|
|
|
auto fd = tr_sys_file_open(filename, TR_SYS_FILE_WRITE | TR_SYS_FILE_CREATE | TR_SYS_FILE_TRUNCATE, 0600);
|
|
auto const file_size = metainfo->file_size(i);
|
|
for (uint64_t j = 0; j < file_size; ++j)
|
|
{
|
|
auto const ch = partial && j < metainfo->piece_size() ? '\1' : '\0';
|
|
tr_sys_file_write(fd, &ch, 1, nullptr);
|
|
}
|
|
|
|
tr_sys_file_close(fd);
|
|
sync();
|
|
}
|
|
}
|
|
|
|
auto* const tor = createTorrentAndWaitForVerifyDone(ctor);
|
|
tr_ctorFree(ctor);
|
|
return tor;
|
|
}
|
|
|
|
[[nodiscard]] tr_torrent* zeroTorrentMagnetInit()
|
|
{
|
|
static auto constexpr V1Hash = "fa5794674a18241bec985ddc3390e3cb171345e4";
|
|
|
|
auto ctor = tr_ctorNew(session_);
|
|
ctor->set_metainfo_from_magnet_link(V1Hash);
|
|
tr_ctorSetPaused(ctor, TR_FORCE, true);
|
|
|
|
auto* const tor = tr_torrentNew(ctor, nullptr);
|
|
EXPECT_NE(nullptr, tor);
|
|
tr_ctorFree(ctor);
|
|
return tor;
|
|
}
|
|
|
|
void blockingTorrentVerify(tr_torrent* tor)
|
|
{
|
|
EXPECT_NE(nullptr, tor->session);
|
|
EXPECT_FALSE(tor->session->am_in_session_thread());
|
|
|
|
auto verified_lock = std::unique_lock(verified_mutex_);
|
|
|
|
auto const n_previously_verified = std::size(verified_);
|
|
auto const stop_waiting = [this, tor, n_previously_verified]()
|
|
{
|
|
return std::size(verified_) > n_previously_verified && verified_.back() == tor;
|
|
};
|
|
tr_torrentVerify(tor);
|
|
verified_cv_.wait_for(verified_lock, 20s, stop_waiting);
|
|
}
|
|
|
|
tr_session* session_ = nullptr;
|
|
|
|
tr_variant* settings()
|
|
{
|
|
if (!settings_)
|
|
{
|
|
settings_ = std::make_shared<tr_variant>(tr_variant::make_map(10U));
|
|
}
|
|
|
|
return settings_.get();
|
|
}
|
|
|
|
void SetUp() override
|
|
{
|
|
SandboxedTest::SetUp();
|
|
|
|
session_ = sessionInit(*settings());
|
|
}
|
|
|
|
void TearDown() override
|
|
{
|
|
sessionClose(session_);
|
|
session_ = nullptr;
|
|
settings_.reset();
|
|
|
|
SandboxedTest::TearDown();
|
|
}
|
|
|
|
private:
|
|
std::mutex verified_mutex_;
|
|
std::condition_variable verified_cv_;
|
|
std::vector<tr_torrent*> verified_;
|
|
};
|
|
|
|
} // namespace libtransmission::test
|