// This file Copyright © 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. #include #include // EINVAL #include // size_t #include // uint16_t #include #include #include #include #include #include "libtransmission/transmission.h" #include "libtransmission/error.h" #include "libtransmission/session.h" #include "libtransmission/torrent-metainfo.h" #include "libtransmission/torrent.h" #include "libtransmission/tr-assert.h" #include "libtransmission/utils.h" using namespace std::literals; namespace { struct optional_args { std::optional paused; std::optional peer_limit; std::string download_dir; }; } // namespace /** Opaque class used when instantiating torrents. * @ingroup tr_ctor */ struct tr_ctor { tr_session const* const session; std::optional delete_source; tr_torrent_metainfo metainfo = {}; tr_priority_t priority = TR_PRI_NORMAL; tr_torrent::labels_t labels{}; std::array optional_args{}; std::string incomplete_dir; std::string torrent_filename; std::vector wanted; std::vector unwanted; std::vector low; std::vector normal; std::vector high; std::vector contents; tr_torrent::VerifyDoneCallback verify_done_callback_; explicit tr_ctor(tr_session const* session_in) : session{ session_in } { } }; // --- void tr_ctorSetVerifyDoneCallback(tr_ctor* ctor, tr_torrent::VerifyDoneCallback&& callback) { ctor->verify_done_callback_ = std::move(callback); } tr_torrent::VerifyDoneCallback tr_ctorStealVerifyDoneCallback(tr_ctor* ctor) { auto tmp = tr_torrent::VerifyDoneCallback{}; std::swap(ctor->verify_done_callback_, tmp); return tmp; } // --- bool tr_ctorSetMetainfoFromFile(tr_ctor* ctor, std::string_view filename, tr_error* error) { if (std::empty(filename)) { if (error != nullptr) { error->set(EINVAL, "no filename specified"sv); } return false; } if (!tr_file_read(filename, ctor->contents, error)) { return false; } ctor->torrent_filename = filename; auto const contents_sv = std::string_view{ std::data(ctor->contents), std::size(ctor->contents) }; return ctor->metainfo.parse_benc(contents_sv, error); } bool tr_ctorSetMetainfoFromFile(tr_ctor* ctor, char const* filename, tr_error* error) { return tr_ctorSetMetainfoFromFile(ctor, std::string_view{ filename != nullptr ? filename : "" }, error); } bool tr_ctorSetMetainfo(tr_ctor* ctor, char const* metainfo, size_t len, tr_error* error) { ctor->torrent_filename.clear(); ctor->contents.assign(metainfo, metainfo + len); auto const contents_sv = std::string_view{ std::data(ctor->contents), std::size(ctor->contents) }; return ctor->metainfo.parse_benc(contents_sv, error); } bool tr_ctorSetMetainfoFromMagnetLink(tr_ctor* ctor, std::string_view magnet_link, tr_error* error) { ctor->torrent_filename.clear(); ctor->metainfo = {}; return ctor->metainfo.parseMagnet(magnet_link, error); } bool tr_ctorSetMetainfoFromMagnetLink(tr_ctor* ctor, char const* magnet_link, tr_error* error) { return tr_ctorSetMetainfoFromMagnetLink(ctor, std::string_view{ magnet_link != nullptr ? magnet_link : "" }, error); } char const* tr_ctorGetSourceFile(tr_ctor const* ctor) { return ctor->torrent_filename.c_str(); } bool tr_ctorSaveContents(tr_ctor const* ctor, std::string_view filename, tr_error* error) { TR_ASSERT(ctor != nullptr); TR_ASSERT(!std::empty(filename)); if (std::empty(ctor->contents)) { if (error != nullptr) { error->set(EINVAL, "torrent ctor has no contents to save"sv); } return false; } return tr_file_save(filename, ctor->contents, error); } // --- void tr_ctorSetFilePriorities(tr_ctor* ctor, tr_file_index_t const* files, tr_file_index_t file_count, tr_priority_t priority) { switch (priority) { case TR_PRI_LOW: ctor->low.assign(files, files + file_count); break; case TR_PRI_HIGH: ctor->high.assign(files, files + file_count); break; default: // TR_PRI_NORMAL ctor->normal.assign(files, files + file_count); break; } } void tr_ctorInitTorrentPriorities(tr_ctor const* ctor, tr_torrent* tor) { tor->set_file_priorities(std::data(ctor->low), std::size(ctor->low), TR_PRI_LOW); tor->set_file_priorities(std::data(ctor->normal), std::size(ctor->normal), TR_PRI_NORMAL); tor->set_file_priorities(std::data(ctor->high), std::size(ctor->high), TR_PRI_HIGH); } void tr_ctorSetFilesWanted(tr_ctor* ctor, tr_file_index_t const* files, tr_file_index_t file_count, bool wanted) { auto& indices = wanted ? ctor->wanted : ctor->unwanted; indices.assign(files, files + file_count); } void tr_ctorInitTorrentWanted(tr_ctor const* ctor, tr_torrent* tor) { tor->init_files_wanted(std::data(ctor->unwanted), std::size(ctor->unwanted), false); tor->init_files_wanted(std::data(ctor->wanted), std::size(ctor->wanted), true); } // --- void tr_ctorSetDeleteSource(tr_ctor* ctor, bool delete_source) { ctor->delete_source = delete_source; } bool tr_ctorGetDeleteSource(tr_ctor const* ctor, bool* setme) { auto const& delete_source = ctor->delete_source; if (!delete_source) { return false; } if (setme != nullptr) { *setme = *delete_source; } return true; } // --- void tr_ctorSetPaused(tr_ctor* ctor, tr_ctorMode mode, bool paused) { TR_ASSERT(ctor != nullptr); TR_ASSERT(mode == TR_FALLBACK || mode == TR_FORCE); ctor->optional_args[mode].paused = paused; } void tr_ctorSetPeerLimit(tr_ctor* ctor, tr_ctorMode mode, uint16_t peer_limit) { TR_ASSERT(ctor != nullptr); TR_ASSERT(mode == TR_FALLBACK || mode == TR_FORCE); ctor->optional_args[mode].peer_limit = peer_limit; } void tr_ctorSetDownloadDir(tr_ctor* ctor, tr_ctorMode mode, char const* directory) { TR_ASSERT(ctor != nullptr); TR_ASSERT(mode == TR_FALLBACK || mode == TR_FORCE); ctor->optional_args[mode].download_dir.assign(directory != nullptr ? directory : ""); } void tr_ctorSetIncompleteDir(tr_ctor* ctor, char const* directory) { ctor->incomplete_dir.assign(directory != nullptr ? directory : ""); } bool tr_ctorGetPeerLimit(tr_ctor const* ctor, tr_ctorMode mode, uint16_t* setme) { auto const& peer_limit = ctor->optional_args[mode].peer_limit; if (!peer_limit) { return false; } if (setme != nullptr) { *setme = *peer_limit; } return true; } bool tr_ctorGetPaused(tr_ctor const* ctor, tr_ctorMode mode, bool* setme) { auto const& paused = ctor->optional_args[mode].paused; if (!paused) { return false; } if (setme != nullptr) { *setme = *paused; } return true; } bool tr_ctorGetDownloadDir(tr_ctor const* ctor, tr_ctorMode mode, char const** setme) { auto const& str = ctor->optional_args[mode].download_dir; if (std::empty(str)) { return false; } if (setme != nullptr) { *setme = str.c_str(); } return true; } bool tr_ctorGetIncompleteDir(tr_ctor const* ctor, char const** setme) { auto const& str = ctor->incomplete_dir; if (std::empty(str)) { return false; } if (setme != nullptr) { *setme = str.c_str(); } return true; } tr_torrent_metainfo tr_ctorStealMetainfo(tr_ctor* ctor) { auto metainfo = tr_torrent_metainfo{}; std::swap(ctor->metainfo, metainfo); return metainfo; } tr_torrent_metainfo const* tr_ctorGetMetainfo(tr_ctor const* ctor) { return !std::empty(ctor->metainfo.info_hash_string()) ? &ctor->metainfo : nullptr; } tr_session* tr_ctorGetSession(tr_ctor const* ctor) { return const_cast(ctor->session); } // --- void tr_ctorSetBandwidthPriority(tr_ctor* ctor, tr_priority_t priority) { if (priority != TR_PRI_LOW && priority != TR_PRI_NORMAL && priority != TR_PRI_HIGH) { return; } ctor->priority = priority; } tr_priority_t tr_ctorGetBandwidthPriority(tr_ctor const* ctor) { return ctor->priority; } // --- void tr_ctorSetLabels(tr_ctor* ctor, tr_torrent::labels_t&& labels) { ctor->labels = std::move(labels); } tr_torrent::labels_t const& tr_ctorGetLabels(tr_ctor const* ctor) { return ctor->labels; } // --- tr_ctor* tr_ctorNew(tr_session const* session) { auto* const ctor = new tr_ctor{ session }; tr_ctorSetDeleteSource(ctor, session->shouldDeleteSource()); tr_ctorSetPaused(ctor, TR_FALLBACK, session->shouldPauseAddedTorrents()); tr_ctorSetPeerLimit(ctor, TR_FALLBACK, session->peerLimitPerTorrent()); tr_ctorSetDownloadDir(ctor, TR_FALLBACK, tr_sessionGetDownloadDir(session)); return ctor; } void tr_ctorFree(tr_ctor* ctor) { delete ctor; }