2022-03-01 23:06:29 +00:00
|
|
|
// This file Copyright © 2022 Mnemosyne LLC.
|
2022-08-08 18:05:39 +00:00
|
|
|
// It may be used under GPLv2 (SPDX: GPL-2.0-only), GPLv3 (SPDX: GPL-3.0-only),
|
2022-03-01 23:06:29 +00:00
|
|
|
// or any future license endorsed by Mnemosyne LLC.
|
|
|
|
// License text can be found in the licenses/ folder.
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#ifndef __TRANSMISSION__
|
|
|
|
#error only libtransmission should #include this header.
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include <ctime>
|
|
|
|
#include <string_view>
|
2022-08-17 16:08:36 +00:00
|
|
|
#include <utility>
|
2022-03-01 23:06:29 +00:00
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
#include "transmission.h"
|
|
|
|
|
|
|
|
#include "torrent-metainfo.h"
|
|
|
|
|
|
|
|
struct tr_torrent;
|
|
|
|
struct tr_torrent_metainfo;
|
|
|
|
|
|
|
|
// A helper class to manage tracking sets of tr_torrent objects.
|
|
|
|
class tr_torrents
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
// returns a fast lookup id for `tor`
|
2022-06-17 15:43:04 +00:00
|
|
|
[[nodiscard]] tr_torrent_id_t add(tr_torrent* tor);
|
2022-03-01 23:06:29 +00:00
|
|
|
|
|
|
|
void remove(tr_torrent const* tor, time_t current_time);
|
|
|
|
|
|
|
|
// O(1)
|
2022-12-23 21:21:40 +00:00
|
|
|
[[nodiscard]] TR_CONSTEXPR20 tr_torrent* get(tr_torrent_id_t id)
|
|
|
|
{
|
|
|
|
auto const uid = static_cast<size_t>(id);
|
|
|
|
return uid >= std::size(by_id_) ? nullptr : by_id_.at(uid);
|
|
|
|
}
|
2022-03-01 23:06:29 +00:00
|
|
|
|
|
|
|
// O(log n)
|
|
|
|
[[nodiscard]] tr_torrent const* get(tr_sha1_digest_t const& hash) const;
|
|
|
|
[[nodiscard]] tr_torrent* get(tr_sha1_digest_t const& hash);
|
|
|
|
|
|
|
|
[[nodiscard]] tr_torrent const* get(tr_torrent_metainfo const& metainfo) const
|
|
|
|
{
|
|
|
|
return get(metainfo.infoHash());
|
|
|
|
}
|
|
|
|
|
|
|
|
[[nodiscard]] tr_torrent* get(tr_torrent_metainfo const& metainfo)
|
|
|
|
{
|
|
|
|
return get(metainfo.infoHash());
|
|
|
|
}
|
|
|
|
|
|
|
|
// These convenience functions use get(tr_sha1_digest_t const&)
|
|
|
|
// after parsing the magnet link to get the info hash. If you have
|
|
|
|
// the info hash already, use get() instead to avoid excess parsing.
|
|
|
|
[[nodiscard]] tr_torrent* get(std::string_view magnet_link);
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
[[nodiscard]] bool contains(T const& key) const
|
|
|
|
{
|
|
|
|
return get(key) != nullptr;
|
|
|
|
}
|
|
|
|
|
2022-06-17 15:43:04 +00:00
|
|
|
[[nodiscard]] std::vector<tr_torrent_id_t> removedSince(time_t) const;
|
2022-03-01 23:06:29 +00:00
|
|
|
|
2022-12-23 21:21:40 +00:00
|
|
|
[[nodiscard]] TR_CONSTEXPR20 auto cbegin() const noexcept
|
2022-03-01 23:06:29 +00:00
|
|
|
{
|
|
|
|
return std::cbegin(by_hash_);
|
|
|
|
}
|
2022-12-23 21:21:40 +00:00
|
|
|
[[nodiscard]] TR_CONSTEXPR20 auto begin() const noexcept
|
2022-03-01 23:06:29 +00:00
|
|
|
{
|
|
|
|
return cbegin();
|
|
|
|
}
|
2022-12-23 21:21:40 +00:00
|
|
|
[[nodiscard]] TR_CONSTEXPR20 auto begin() noexcept
|
2022-03-01 23:06:29 +00:00
|
|
|
{
|
|
|
|
return std::begin(by_hash_);
|
|
|
|
}
|
|
|
|
|
2022-12-23 21:21:40 +00:00
|
|
|
[[nodiscard]] TR_CONSTEXPR20 auto cend() const noexcept
|
2022-03-01 23:06:29 +00:00
|
|
|
{
|
|
|
|
return std::cend(by_hash_);
|
|
|
|
}
|
|
|
|
|
2022-12-23 21:21:40 +00:00
|
|
|
[[nodiscard]] TR_CONSTEXPR20 auto end() const noexcept
|
2022-03-01 23:06:29 +00:00
|
|
|
{
|
|
|
|
return cend();
|
|
|
|
}
|
|
|
|
|
2022-12-23 21:21:40 +00:00
|
|
|
[[nodiscard]] TR_CONSTEXPR20 auto end() noexcept
|
2022-03-01 23:06:29 +00:00
|
|
|
{
|
|
|
|
return std::end(by_hash_);
|
|
|
|
}
|
|
|
|
|
2022-12-23 21:21:40 +00:00
|
|
|
[[nodiscard]] TR_CONSTEXPR20 auto size() const noexcept
|
2022-03-01 23:06:29 +00:00
|
|
|
{
|
|
|
|
return std::size(by_hash_);
|
|
|
|
}
|
|
|
|
|
2022-12-23 21:21:40 +00:00
|
|
|
[[nodiscard]] TR_CONSTEXPR20 auto empty() const noexcept
|
2022-03-01 23:06:29 +00:00
|
|
|
{
|
|
|
|
return std::empty(by_hash_);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
std::vector<tr_torrent*> by_hash_;
|
|
|
|
|
2022-06-17 15:43:04 +00:00
|
|
|
// This is a lookup table where by_id_[id]->id() == id.
|
2022-03-01 23:06:29 +00:00
|
|
|
// There is a small tradeoff here -- lookup is O(1) at the cost
|
|
|
|
// of a wasted slot in the lookup table whenever a torrent is
|
|
|
|
// removed. This improves speed for all use cases at the cost of
|
|
|
|
// wasting a small amount of memory in uncommon use cases, e.g. a
|
2022-03-24 05:18:41 +00:00
|
|
|
// long-lived session where thousands of torrents are removed.
|
|
|
|
//
|
|
|
|
// Insert an empty pointer at by_id_[0] to ensure that the first
|
|
|
|
// added torrent doesn't get an ID of 0; ie, that every torrent has
|
|
|
|
// a positive ID number. This constraint isn't needed by libtransmission
|
|
|
|
// code but the ID is exported in the RPC API to 3rd party clients that
|
|
|
|
// may be testing for >0 as a validity check.
|
|
|
|
std::vector<tr_torrent*> by_id_{ nullptr };
|
2022-03-01 23:06:29 +00:00
|
|
|
|
2022-06-17 15:43:04 +00:00
|
|
|
std::vector<std::pair<tr_torrent_id_t, time_t>> removed_;
|
2022-03-01 23:06:29 +00:00
|
|
|
};
|