mirror of
https://github.com/transmission/transmission.git
synced 2026-05-04 15:49:10 +01:00
05aef3e787
* refactor: change `leftUntilDone` to `left_until_done` * refactor: change `magnetLink` to `magnet_link` * refactor: change `manualAnnounceTime` to `manual_announce_time` * refactor: change `maxConnectedPeers` to `max_connected_peers` * refactor: change `metadataPercentComplete` to `metadata_percent_complete` * refactor: change `peersConnected` to `peers_connected` * refactor: change `peersFrom` to `peers_from` * refactor: change `peersGettingFromUs` to `peers_getting_from_us` * refactor: change `peersSendingToUs` to `peers_sending_to_us` * refactor: change `percentComplete` to `percent_complete` * refactor: change `percentDone` to `percent_done` * refactor: change `pieceCount` to `piece_count` * refactor: use quark when possible * refactor: change `pieceSize` to `piece_size` * refactor: change `primary-mime-type` to `primary_mime_type` * refactor: change `rateDownload` to `rate_download` * refactor: change `rateUpload` to `rate_upload` * refactor: change `recheckProgress` to `recheck_progress` * refactor: change `secondsDownloading` to `seconds_downloading` * refactor: change `secondsSeeding` to `seconds_seeding` * refactor: change `sizeWhenDone` to `size_when_done` * refactor: change `startDate` to `start_date` * refactor: change `trackerStats` to `tracker_stats` * refactor: change `totalSize` to `total_size` * refactor: change `torrentFile` to `torrent_file` * refactor: change `uploadedEver` to `uploaded_ever` * refactor: change `uploadRatio` to `upload_ratio` * refactor: change `webseedsSendingToUs` to `webseeds_sending_to_us` * refactor: change `bytesCompleted` to `bytes_completed` * refactor: change `clientName` to `client_name` * refactor: change `clientIsChoked` to `client_is_choked` * refactor: change `clientIsInterested` to `client_is_interested` * refactor: change `flagStr` to `flag_str` * refactor: change `isDownloadingFrom` to `is_downloading_from` * refactor: change `isEncrypted` to `is_encrypted` * refactor: change `isIncoming` to `is_incoming` * refactor: change `isUploadingTo` to `is_uploading_to` * refactor: change `isUTP` to `is_utp` * refactor: change `peerIsChoked` to `peer_is_choked` * refactor: change `peerIsInterested` to `peer_is_interested` * refactor: change `rateToClient` to `rate_to_client` * refactor: change `rateToPeer` to `rate_to_peer` * refactor: change `fromCache` to `from_cache` * refactor: change `fromDht` to `from_dht` * refactor: change `fromIncoming` to `from_incoming` * refactor: change `fromLpd` to `from_lpd` * refactor: change `fromLtep` to `from_ltep` * refactor: change `fromPex` to `from_pex` * refactor: change `fromTracker` to `from_tracker` * refactor: change `announceState` to `announce_state` * refactor: change `downloadCount` to `download_count` * refactor: change `hasAnnounced` to `has_announced` * refactor: change `hasScraped` to `has_scraped` * refactor: change `isBackup` to `is_backup` * refactor: change `lastAnnouncePeerCount` to `last_announce_peer_count` * refactor: change `lastAnnounceResult` to `last_announce_result` * refactor: change `lastAnnounceStartTime` to `last_announce_start_time` * refactor: change `lastAnnounceSucceeded` to `last_announce_succeeded` * refactor: change `lastAnnounceTime` to `last_announce_time` * refactor: change `lastAnnounceTimedOut` to `last_announce_timed_out` * refactor: change `lastScrapeResult` to `last_scrape_result` * refactor: change `lastScrapeStartTime` to `last_scrape_start_time` * refactor: change `lastScrapeSucceeded` to `last_scrape_succeeded` * refactor: change `lastScrapeTime` to `last_scrape_time` * refactor: change `lastScrapeTimedOut` to `last_scrape_timed_out` * refactor: change `leecherCount` to `leecher_count` * refactor: change `nextAnnounceTime` to `next_announce_time` * refactor: change `nextScrapeTime` to `next_scrape_time` * refactor: change `scrapeState` to `scrape_state` * refactor: change `seederCount` to `seeder_count` * refactor: change `torrent-added` to `torrent_added` * refactor: change `torrent-duplicate` to `torrent_duplicate` * refactor: change `torrent-remove` to `torrent_remove` * refactor: change `delete-local-data` to `delete_local_data` * refactor: change `torrent-rename-path` to `torrent_rename_path` * refactor: change `alt-speed-down` to `alt_speed_down` * refactor: convert `pref_toggle_entries` to quark array * refactor: change `alt-speed-enabled` to `alt_speed_enabled` * refactor: change `compact-view` to `compact_view` * refactor: change `sort-reversed` to `sort_reversed` * refactor: change `show-filterbar` to `show_filterbar` * refactor: change `show-statusbar` to `show_statusbar` * refactor: change `show-toolbar` to `show_toolbar` * refactor: change `alt-speed-time-begin` to `alt_speed_time_begin` * refactor: change `alt-speed-time-day` to `alt_speed_time_day` * refactor: change `alt-speed-time-end` to `alt_speed_time_end` * refactor: change `alt-speed-up` to `alt_speed_up` * refactor: change `alt-speed-time-enabled` to `alt_speed_time_enabled` * refactor: change `blocklist-enabled` to `blocklist_enabled` * refactor: change `blocklist-size` to `blocklist_size` * refactor: change `blocklist-url` to `blocklist_url` * refactor: change `cache-size-mb` to `cache_size_mb` * refactor: change `config-dir` to `config_dir` * refactor: change `default-trackers` to `default_trackers` * refactor: change `dht-enabled` to `dht_enabled` * refactor: change `download-dir-free-space` to `download_dir_free_space` * refactor: change `download-queue-enabled` to `download_queue_enabled` * refactor: change `download-queue-size` to `download_queue_size` * refactor: change `idle-seeding-limit-enabled` to `idle_seeding_limit_enabled` * refactor: change `idle-seeding-limit` to `idle_seeding_limit` * refactor: change `incomplete-dir-enabled` to `incomplete_dir_enabled` * refactor: change `incomplete-dir` to `incomplete_dir` * refactor: change `lpd-enabled` to `lpd_enabled` * refactor: change `peer-limit-global` to `peer_limit_global` * refactor: change `peer-limit-per-torrent` to `peer_limit_per_torrent` * refactor: change `peer-port-random-on-start` to `peer_port_random_on_start` * refactor: change `peer-port` to `peer_port` * refactor: change `pex-enabled` to `pex_enabled` * refactor: change `port-forwarding-enabled` to `port_forwarding_enabled` * refactor: change `queue-stalled-enabled` to `queue_stalled_enabled` * refactor: change `queue-stalled-minutes` to `queue_stalled_minutes` * refactor: change `rename-partial-files` to `rename_partial_files` * refactor: change `rpc-version-minimum` to `rpc_version_minimum` * refactor: change `rpc-version-semver` to `rpc_version_semver` * refactor: change `rpc-version` to `rpc_version` * refactor: change `script-torrent-added-enabled` to `script_torrent_added_enabled` * refactor: change `script-torrent-added-filename` to `script_torrent_added_filename` * refactor: change `script-torrent-done-enabled` to `script_torrent_done_enabled` * refactor: change `script-torrent-done-filename` to `script_torrent_done_filename` * refactor: change `script-torrent-done-seeding-enabled` to `script_torrent_done_seeding_enabled` * refactor: change `script-torrent-done-seeding-filename` to `script_torrent_done_seeding_filename` * refactor: change `seed-queue-enabled` to `seed_queue_enabled` * refactor: change `seed-queue-size` to `seed_queue_size` * refactor: change `seedRatioLimited` to `seed_ratio_limited` * refactor: change `session-id` to `session_id` * refactor: change `speed-limit-down-enabled` to `speed_limit_down_enabled` * refactor: change `speed-limit-down` to `speed_limit_down` * refactor: change `speed-limit-up-enabled` to `speed_limit_up_enabled` * refactor: change `speed-limit-up` to `speed_limit_up` * refactor: change `start-added-torrents` to `start_added_torrents` * refactor: change `trash-original-torrent-files` to `trash_original_torrent_files` * refactor: change `utp-enabled` to `utp_enabled` * refactor: change `tcp-enabled` to `tcp_enabled` * docs: add missing docs for RPC `tcp_enabled` * refactor: change `speed-units` to `speed_units` * refactor: change `speed-bytes` to `speed_bytes` * refactor: change `size-units` to `size_units` * refactor: change `size-bytes` to `size_bytes` * refactor: change `memory-units` to `memory_units` * refactor: change `memory-bytes` to `memory_bytes` * refactor: change `session-set` to `session_set` * refactor: change `session-get` to `session_get` * refactor: change `session-stats` to `session_stats` * refactor: change `activeTorrentCount` to `active_torrent_count` * refactor: change `downloadSpeed` to `download_speed` * refactor: change `pausedTorrentCount` to `paused_torrent_count` * refactor: change `torrentCount` to `torrent_count` * refactor: change `uploadSpeed` to `upload_speed` * refactor: change `cumulative-stats` to `cumulative_stats` * refactor: change `current-stats` to `current_stats` * refactor: change `uploadedBytes` and `uploaded-bytes` to `uploaded_bytes` * refactor: change `downloadedBytes` and `downloaded-bytes` to `downloaded_bytes` * refactor: change `filesAdded` and `files-added` to `files_added` * refactor: change `sessionCount` and `session-count` to `session_count` * refactor: change `secondsActive` and `seconds-active` to `seconds_active` * refactor: change `blocklist-update` to `blocklist_update` * refactor: change `port-test` to `port_test` * refactor: change `session-close` to `session_close` * refactor: change `queue-move-top` to `queue_move_top` * refactor: change `queue-move-up` to `queue_move_up` * refactor: change `queue-move-down` to `queue_move_down` * refactor: change `queue-move-bottom` to `queue_move_bottom` * refactor: change `free-space` to `free_space` * refactor: change `group-set` to `group_set` * refactor: change `group-get` to `group_get` * refactor: change `announce-ip` to `announce_ip` * refactor: change `announce-ip-enabled` to `announce_ip_enabled` * refactor: change `upload-slots-per-torrent` to `upload_slots_per_torrent` * refactor: change `trash-can-enabled` to `trash_can_enabled` * refactor: change `watch-dir-enabled` to `watch_dir_enabled` * refactor: change `watch-dir-force-generic` to `watch_dir_force_generic` * refactor: change `watch-dir` to `watch_dir` * refactor: change `message-level` to `message_level` * refactor: change `scrape-paused-torrents-enabled` to `scrape_paused_torrents_enabled` * refactor: change `torrent-added-verify-mode` to `torrent_added_verify_mode` * refactor: change `sleep-per-seconds-during-verify` to `sleep_per_seconds_during_verify` * refactor: change `bind-address-ipv4` to `bind_address_ipv4` * refactor: change `bind-address-ipv6` to `bind_address_ipv6` * refactor: change `peer-congestion-algorithm` to `peer_congestion_algorithm` * refactor: change `peer-socket-tos` to `peer_socket_tos` * refactor: change `peer-port-random-high` to `peer_port_random_high` * refactor: change `peer-port-random-low` to `peer_port_random_low` * refactor: change `anti-brute-force-enabled` to `anti_brute_force_enabled` * refactor: change `rpc-authentication-required` to `rpc_authentication_required` * refactor: change `rpc-bind-address` to `rpc_bind_address` * refactor: change `rpc-enabled` to `rpc_enabled` * refactor: change `rpc-host-whitelist` to `rpc_host_whitelist` * refactor: change `rpc-host-whitelist-enabled` to `rpc_host_whitelist_enabled` * refactor: change `rpc-password` to `rpc_password` * refactor: change `rpc-port` to `rpc_port` * refactor: change `rpc-socket-mode` to `rpc_socket_mode` * refactor: change `rpc-url` to `rpc_url` * refactor: change `rpc-username` to `rpc_username` * refactor: change `rpc-whitelist` to `rpc_whitelist` * refactor: change `rpc-whitelist-enabled` to `rpc_whitelist_enabled` * refactor: change `ratio-limit-enabled` to `ratio_limit_enabled` * refactor: change `ratio-limit` to `ratio_limit` * refactor: change `show-options-window` to `show_options_window` * refactor: change `open-dialog-dir` to `open_dialog_dir` * refactor: change `inhibit-desktop-hibernation` to `inhibit_desktop_hibernation` * refactor: change `show-notification-area-icon` to `show_notification_area_icon` * refactor: change `start-minimized` to `start_minimized` * refactor: change `torrent-added-notification-enabled` to `torrent_added_notification_enabled` * refactor: change `anti-brute-force-threshold` to `anti_brute_force_threshold` * refactor: change `torrent-complete-notification-enabled` to `torrent_complete_notification_enabled` * refactor: change `prompt-before-exit` to `prompt_before_exit` * refactor: change `sort-mode` to `sort_mode` * refactor: change `statusbar-stats` to `statusbar_stats` * refactor: change `show-extra-peer-details` to `show_extra_peer_details` * refactor: change `show-backup-trackers` to `show_backup_trackers` * refactor: change `blocklist-date` to `blocklist_date` * refactor: change `blocklist-updates-enabled` to `blocklist_updates_enabled` * refactor: change `main-window-layout-order` to `main_window_layout_order` * refactor: change `main-window-height` to `main_window_height` * refactor: change `main-window-width` to `main_window_width` * refactor: change `main-window-x` to `main_window_x` * refactor: change `main-window-y` to `main_window_y` * refactor: change `filter-mode` to `filter_mode` * refactor: change `filter-trackers` to `filter_trackers` * refactor: change `filter-text` to `filter_text` * refactor: change `remote-session-enabled` to `remote_session_enabled` * refactor: change `remote-session-host` to `remote_session_host` * refactor: change `remote-session-https` to `remote_session_https` * refactor: change `remote-session-password` to `remote_session_password` * refactor: change `remote-session-port` to `remote_session_port` * refactor: change `remote-session-requres-authentication` to `remote_session_requires_authentication` * refactor: change `remote-session-username` to `remote_session_username` * refactor: change `torrent-complete-sound-command` to `torrent_complete_sound_command` * refactor: change `torrent-complete-sound-enabled` to `torrent_complete_sound_enabled` * refactor: change `user-has-given-informed-consent` to `user_has_given_informed_consent` * refactor: change `read-clipboard` to `read_clipboard` * refactor: change `details-window-height` to `details_window_height` * refactor: change `details-window-width` to `details_window_width` * refactor: change `main-window-is-maximized` to `main_window_is_maximized` * refactor: change `port-is-open` to `port_is_open` * refactor: change `show-tracker-scrapes` to `show_tracker_scrapes` * refactor: change `max-peers` to `max_peers` * refactor: change `peers2-6` to `peers2_6` * refactor: change `seeding-time-seconds` to `seeding_time_seconds` * refactor: change `downloading-time-seconds` to `downloading_time_seconds` * refactor: change `ratio-mode` to `ratio_mode` * refactor: change `idle-limit` to `idle_limit` * refactor: change `idle-mode` to `idle_mode` * refactor: change `speed-Bps` to `speed_Bps` * refactor: change `use-global-speed-limit` to `use_global_speed_limit` * refactor: change `use-speed-limit` to `use_speed_limit` * chore: remove TODO comment * docs: add upgrade instructions to `5.0.0` * chore: bump rpc semver major version * chore: housekeeping
509 lines
19 KiB
C++
509 lines
19 KiB
C++
// This file Copyright (C) 2021-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.
|
|
|
|
#include <algorithm>
|
|
#include <array>
|
|
#include <cstddef> // size_t
|
|
#include <cstdint>
|
|
#include <set>
|
|
|
|
#include <libtransmission/transmission.h>
|
|
|
|
#include <libtransmission/bitfield.h>
|
|
#include <libtransmission/block-info.h>
|
|
#include <libtransmission/crypto-utils.h> // for tr_rand_obj()
|
|
#include <libtransmission/completion.h>
|
|
|
|
#include "gtest/gtest.h"
|
|
|
|
using CompletionTest = ::testing::Test;
|
|
|
|
namespace
|
|
{
|
|
|
|
struct TestTorrent
|
|
{
|
|
std::set<tr_piece_index_t> dnd_pieces;
|
|
|
|
[[nodiscard]] tr_completion makeCompletion(tr_block_info const& block_info) const
|
|
{
|
|
return { [this](tr_piece_index_t const piece) { return dnd_pieces.count(piece) == 0; }, &block_info };
|
|
}
|
|
};
|
|
|
|
auto constexpr BlockSize = uint64_t{ 16 } * 1024U;
|
|
|
|
} // namespace
|
|
|
|
TEST_F(CompletionTest, MagnetLink)
|
|
{
|
|
auto torrent = TestTorrent{};
|
|
auto block_info = tr_block_info{};
|
|
auto completion = torrent.makeCompletion(block_info);
|
|
|
|
EXPECT_FALSE(completion.has_all());
|
|
EXPECT_TRUE(completion.has_none());
|
|
EXPECT_FALSE(completion.has_blocks({ 0, 1 }));
|
|
EXPECT_FALSE(completion.has_blocks({ 0, 1000 }));
|
|
EXPECT_FALSE(completion.has_piece(0));
|
|
EXPECT_DOUBLE_EQ(0.0, completion.percent_done());
|
|
EXPECT_DOUBLE_EQ(0.0, completion.percent_complete());
|
|
EXPECT_EQ(TR_LEECH, completion.status());
|
|
EXPECT_EQ(0, completion.has_total());
|
|
EXPECT_EQ(0, completion.has_valid());
|
|
EXPECT_EQ(0, completion.left_until_done());
|
|
EXPECT_EQ(0, completion.size_when_done());
|
|
}
|
|
|
|
TEST_F(CompletionTest, setBlocks)
|
|
{
|
|
auto constexpr TotalSize = uint64_t{ BlockSize * 64 * 50000 }; // 50GB
|
|
auto constexpr PieceSize = uint64_t{ BlockSize * 64 };
|
|
|
|
auto torrent = TestTorrent{};
|
|
auto const block_info = tr_block_info{ TotalSize, PieceSize };
|
|
auto completion = torrent.makeCompletion(block_info);
|
|
EXPECT_FALSE(completion.blocks().has_all());
|
|
EXPECT_FALSE(completion.has_all());
|
|
EXPECT_EQ(0, completion.has_total());
|
|
|
|
auto bitfield = tr_bitfield{ block_info.block_count() };
|
|
bitfield.set_has_all();
|
|
|
|
// test that the bitfield did get replaced
|
|
completion.set_blocks(bitfield);
|
|
EXPECT_TRUE(completion.blocks().has_all());
|
|
EXPECT_TRUE(completion.has_all());
|
|
EXPECT_EQ(block_info.total_size(), completion.has_total());
|
|
}
|
|
|
|
TEST_F(CompletionTest, hasBlock)
|
|
{
|
|
auto torrent = TestTorrent{};
|
|
auto constexpr TotalSize = uint64_t{ BlockSize * 4096 };
|
|
auto constexpr PieceSize = uint64_t{ BlockSize * 64 };
|
|
auto const block_info = tr_block_info{ TotalSize, PieceSize };
|
|
auto completion = torrent.makeCompletion(block_info);
|
|
|
|
EXPECT_FALSE(completion.has_block(0));
|
|
EXPECT_FALSE(completion.has_block(1));
|
|
|
|
completion.add_block(0);
|
|
EXPECT_TRUE(completion.has_block(0));
|
|
EXPECT_FALSE(completion.has_block(1));
|
|
|
|
completion.add_piece(0);
|
|
EXPECT_TRUE(completion.has_block(0));
|
|
EXPECT_TRUE(completion.has_block(1));
|
|
}
|
|
|
|
TEST_F(CompletionTest, hasBlocks)
|
|
{
|
|
auto torrent = TestTorrent{};
|
|
auto constexpr TotalSize = uint64_t{ BlockSize * 4096 };
|
|
auto constexpr PieceSize = uint64_t{ BlockSize * 64 };
|
|
auto const block_info = tr_block_info{ TotalSize, PieceSize };
|
|
|
|
auto completion = torrent.makeCompletion(block_info);
|
|
EXPECT_FALSE(completion.has_blocks({ 0, 1 }));
|
|
EXPECT_FALSE(completion.has_blocks({ 0, 2 }));
|
|
|
|
completion.add_block(0);
|
|
EXPECT_TRUE(completion.has_blocks({ 0, 1 }));
|
|
EXPECT_FALSE(completion.has_blocks({ 0, 2 }));
|
|
}
|
|
|
|
TEST_F(CompletionTest, hasNone)
|
|
{
|
|
auto torrent = TestTorrent{};
|
|
auto constexpr TotalSize = uint64_t{ BlockSize * 4096 };
|
|
auto constexpr PieceSize = uint64_t{ BlockSize * 64 };
|
|
auto const block_info = tr_block_info{ TotalSize, PieceSize };
|
|
|
|
auto completion = torrent.makeCompletion(block_info);
|
|
EXPECT_TRUE(completion.has_none());
|
|
|
|
completion.add_block(0);
|
|
EXPECT_FALSE(completion.has_none());
|
|
}
|
|
|
|
TEST_F(CompletionTest, hasPiece)
|
|
{
|
|
auto torrent = TestTorrent{};
|
|
auto constexpr TotalSize = uint64_t{ BlockSize * 4096 };
|
|
auto constexpr PieceSize = uint64_t{ BlockSize * 64 };
|
|
auto const block_info = tr_block_info{ TotalSize, PieceSize };
|
|
|
|
// check that the initial state does not have it
|
|
auto completion = torrent.makeCompletion(block_info);
|
|
EXPECT_FALSE(completion.has_piece(0));
|
|
EXPECT_FALSE(completion.has_piece(1));
|
|
EXPECT_EQ(0, completion.has_valid());
|
|
|
|
// check that adding a piece means we have it
|
|
completion.add_piece(0);
|
|
EXPECT_TRUE(completion.has_piece(0));
|
|
EXPECT_FALSE(completion.has_piece(1));
|
|
EXPECT_EQ(PieceSize, completion.has_valid());
|
|
|
|
// check that removing a piece means we don't have it
|
|
completion.remove_piece(0);
|
|
EXPECT_FALSE(completion.has_piece(0));
|
|
EXPECT_FALSE(completion.has_piece(1));
|
|
EXPECT_EQ(0, completion.has_valid());
|
|
|
|
// check that adding all the blocks in a piece means we have it
|
|
for (tr_block_index_t i = 1, n = block_info.piece_loc(1).block; i < n; ++i)
|
|
{
|
|
completion.add_block(i);
|
|
}
|
|
EXPECT_FALSE(completion.has_piece(0));
|
|
EXPECT_EQ(0, completion.has_valid());
|
|
completion.add_block(0);
|
|
EXPECT_TRUE(completion.has_piece(0));
|
|
EXPECT_EQ(PieceSize, completion.has_valid());
|
|
}
|
|
|
|
TEST_F(CompletionTest, percentCompleteAndDone)
|
|
{
|
|
auto torrent = TestTorrent{};
|
|
auto constexpr TotalSize = uint64_t{ BlockSize * 4096 };
|
|
auto constexpr PieceSize = uint64_t{ BlockSize * 64 };
|
|
auto const block_info = tr_block_info{ TotalSize, PieceSize };
|
|
|
|
// check that in blank-slate initial state, isDone() is false
|
|
auto completion = torrent.makeCompletion(block_info);
|
|
EXPECT_DOUBLE_EQ(0.0, completion.percent_complete());
|
|
EXPECT_DOUBLE_EQ(0.0, completion.percent_done());
|
|
|
|
// add half the pieces
|
|
for (size_t i = 0; i < 32; ++i)
|
|
{
|
|
completion.add_piece(i);
|
|
}
|
|
EXPECT_DOUBLE_EQ(0.5, completion.percent_complete());
|
|
EXPECT_DOUBLE_EQ(0.5, completion.percent_done());
|
|
|
|
// but marking some of the pieces we have as unwanted
|
|
// should not change percent_done
|
|
for (size_t i = 0; i < 16; ++i)
|
|
{
|
|
torrent.dnd_pieces.insert(i);
|
|
}
|
|
completion.invalidate_size_when_done();
|
|
EXPECT_DOUBLE_EQ(0.5, completion.percent_complete());
|
|
EXPECT_DOUBLE_EQ(0.5, completion.percent_done());
|
|
|
|
// but marking some of the pieces we DON'T have as unwanted
|
|
// SHOULD change percent_done
|
|
for (size_t i = 32; i < 48; ++i)
|
|
{
|
|
torrent.dnd_pieces.insert(i);
|
|
}
|
|
completion.invalidate_size_when_done();
|
|
EXPECT_DOUBLE_EQ(0.5, completion.percent_complete());
|
|
EXPECT_DOUBLE_EQ(2.0 / 3.0, completion.percent_done());
|
|
}
|
|
|
|
TEST_F(CompletionTest, hasTotalAndValid)
|
|
{
|
|
auto torrent = TestTorrent{};
|
|
auto constexpr TotalSize = uint64_t{ BlockSize * 4096 } + 1;
|
|
auto constexpr PieceSize = uint64_t{ BlockSize * 64 };
|
|
auto const block_info = tr_block_info{ TotalSize, PieceSize };
|
|
|
|
// check that the initial blank-slate state has nothing
|
|
auto completion = torrent.makeCompletion(block_info);
|
|
EXPECT_EQ(0, completion.has_total());
|
|
EXPECT_EQ(completion.has_valid(), completion.has_total());
|
|
|
|
// check that adding the final piece adjusts by block_info.final_piece_size
|
|
completion.set_has_piece(block_info.piece_count() - 1, true);
|
|
EXPECT_EQ(block_info.piece_size(block_info.piece_count() - 1), completion.has_total());
|
|
EXPECT_EQ(completion.has_valid(), completion.has_total());
|
|
|
|
// check that adding a non-final piece adjusts by block_info.pieceSize()
|
|
completion.set_has_piece(0, true);
|
|
EXPECT_EQ(block_info.piece_size(block_info.piece_count() - 1) + block_info.piece_size(), completion.has_total());
|
|
EXPECT_EQ(completion.has_valid(), completion.has_total());
|
|
|
|
// check that removing the final piece adjusts by block_info.final_piece_size
|
|
completion.set_has_piece(block_info.piece_count() - 1, false);
|
|
EXPECT_EQ(block_info.piece_size(), completion.has_valid());
|
|
EXPECT_EQ(completion.has_valid(), completion.has_total());
|
|
|
|
// check that removing a non-final piece adjusts by block_info.pieceSize()
|
|
completion.set_has_piece(0, false);
|
|
EXPECT_EQ(0, completion.has_valid());
|
|
EXPECT_EQ(completion.has_valid(), completion.has_total());
|
|
|
|
// check that adding an incomplete piece adjusts hasTotal but not hasValid
|
|
completion.add_block(0);
|
|
EXPECT_EQ(0, completion.has_valid());
|
|
EXPECT_EQ(BlockSize, completion.has_total());
|
|
}
|
|
|
|
TEST_F(CompletionTest, leftUntilDone)
|
|
{
|
|
auto torrent = TestTorrent{};
|
|
auto constexpr TotalSize = uint64_t{ BlockSize * 4096 } + 1;
|
|
auto constexpr PieceSize = uint64_t{ BlockSize * 64 };
|
|
auto const block_info = tr_block_info{ TotalSize, PieceSize };
|
|
|
|
// check that the initial blank-slate state has nothing
|
|
auto completion = torrent.makeCompletion(block_info);
|
|
EXPECT_EQ(block_info.total_size(), completion.left_until_done());
|
|
|
|
// check that adding the final piece adjusts by block_info.final_piece_size
|
|
completion.add_piece(block_info.piece_count() - 1);
|
|
EXPECT_EQ(block_info.total_size() - block_info.piece_size(block_info.piece_count() - 1), completion.left_until_done());
|
|
|
|
// check that adding a non-final piece adjusts by block_info.pieceSize()
|
|
completion.add_piece(0);
|
|
EXPECT_EQ(
|
|
block_info.total_size() - block_info.piece_size(block_info.piece_count() - 1) - block_info.piece_size(),
|
|
completion.left_until_done());
|
|
|
|
// check that removing the final piece adjusts by block_info.final_piece_size
|
|
completion.remove_piece(block_info.piece_count() - 1);
|
|
EXPECT_EQ(block_info.total_size() - block_info.piece_size(), completion.left_until_done());
|
|
|
|
// check that dnd-flagging a piece we already have affects nothing
|
|
torrent.dnd_pieces.insert(0);
|
|
completion.invalidate_size_when_done();
|
|
EXPECT_EQ(block_info.total_size() - block_info.piece_size(), completion.left_until_done());
|
|
torrent.dnd_pieces.clear();
|
|
completion.invalidate_size_when_done();
|
|
|
|
// check that dnd-flagging a piece we DON'T already have adjusts by block_info.pieceSize()
|
|
torrent.dnd_pieces.insert(1);
|
|
completion.invalidate_size_when_done();
|
|
EXPECT_EQ(block_info.total_size() - (block_info.piece_size() * uint64_t{ 2U }), completion.left_until_done());
|
|
torrent.dnd_pieces.clear();
|
|
completion.invalidate_size_when_done();
|
|
|
|
// check that removing a non-final piece adjusts by block_info.pieceSize()
|
|
completion.remove_piece(0);
|
|
EXPECT_EQ(block_info.total_size(), completion.left_until_done());
|
|
|
|
// check that adding a block adjusts by block_info.block_size
|
|
completion.add_block(0);
|
|
EXPECT_EQ(block_info.total_size() - tr_block_info::BlockSize, completion.left_until_done());
|
|
}
|
|
|
|
TEST_F(CompletionTest, sizeWhenDone)
|
|
{
|
|
auto torrent = TestTorrent{};
|
|
auto constexpr TotalSize = uint64_t{ BlockSize * 64 * 50000 }; // 50GB
|
|
auto constexpr PieceSize = uint64_t{ BlockSize * 64 };
|
|
auto const block_info = tr_block_info{ TotalSize, PieceSize };
|
|
|
|
// check that adding or removing blocks or pieces does not affect size_when_done
|
|
auto completion = torrent.makeCompletion(block_info);
|
|
EXPECT_EQ(block_info.total_size(), completion.size_when_done());
|
|
completion.add_block(0);
|
|
EXPECT_EQ(block_info.total_size(), completion.size_when_done());
|
|
completion.add_piece(0);
|
|
EXPECT_EQ(block_info.total_size(), completion.size_when_done());
|
|
completion.remove_piece(0);
|
|
EXPECT_EQ(block_info.total_size(), completion.size_when_done());
|
|
|
|
// check that flagging complete pieces as dnd does not affect size_when_done
|
|
for (size_t i = 0; i < 32; ++i)
|
|
{
|
|
completion.add_piece(i);
|
|
torrent.dnd_pieces.insert(i);
|
|
}
|
|
completion.invalidate_size_when_done();
|
|
EXPECT_EQ(block_info.total_size(), completion.size_when_done());
|
|
|
|
// check that flagging missing pieces as dnd does not affect size_when_done
|
|
for (size_t i = 32; i < 48; ++i)
|
|
{
|
|
torrent.dnd_pieces.insert(i);
|
|
}
|
|
completion.invalidate_size_when_done();
|
|
EXPECT_EQ(block_info.total_size() - (uint64_t{ 16U } * block_info.piece_size()), completion.size_when_done());
|
|
}
|
|
|
|
TEST_F(CompletionTest, createPieceBitfield)
|
|
{
|
|
auto torrent = TestTorrent{};
|
|
auto constexpr TotalSize = uint64_t{ BlockSize * 4096 } + 1;
|
|
auto constexpr PieceSize = uint64_t{ BlockSize * 64 };
|
|
auto const block_info = tr_block_info{ TotalSize, PieceSize };
|
|
|
|
// make a completion object that has a random assortment of pieces
|
|
auto completion = torrent.makeCompletion(block_info);
|
|
auto buf = tr_rand_obj<std::array<char, 65>>();
|
|
ASSERT_EQ(std::size(buf), block_info.piece_count());
|
|
for (uint64_t i = 0; i < block_info.piece_count(); ++i)
|
|
{
|
|
if ((buf[i] % 2) != 0)
|
|
{
|
|
completion.add_piece(i);
|
|
}
|
|
}
|
|
|
|
// serialize it to a raw bitfield, read it back into a bitfield,
|
|
// and test that the new bitfield matches
|
|
auto const pieces_raw_bitfield = completion.create_piece_bitfield();
|
|
tr_bitfield pieces{ size_t{ block_info.piece_count() } };
|
|
pieces.set_raw(std::data(pieces_raw_bitfield), std::size(pieces_raw_bitfield));
|
|
for (uint64_t i = 0; i < block_info.piece_count(); ++i)
|
|
{
|
|
EXPECT_EQ(completion.has_piece(i), pieces.test(i));
|
|
}
|
|
}
|
|
|
|
TEST_F(CompletionTest, setHasPiece)
|
|
{
|
|
}
|
|
|
|
TEST_F(CompletionTest, countMissingBytesInPiece)
|
|
{
|
|
auto torrent = TestTorrent{};
|
|
auto constexpr TotalSize = uint64_t{ BlockSize * 4096 } + 1;
|
|
auto constexpr PieceSize = uint64_t{ BlockSize * 64 };
|
|
auto const block_info = tr_block_info{ TotalSize, PieceSize };
|
|
auto completion = torrent.makeCompletion(block_info);
|
|
|
|
EXPECT_EQ(block_info.piece_size(0), completion.count_missing_bytes_in_piece(0));
|
|
completion.add_block(0);
|
|
EXPECT_EQ(block_info.piece_size(0) - tr_block_info::BlockSize, completion.count_missing_bytes_in_piece(0));
|
|
completion.add_piece(0);
|
|
EXPECT_EQ(0U, completion.count_missing_bytes_in_piece(0));
|
|
|
|
auto const final_piece = block_info.piece_count() - 1;
|
|
auto const final_block = block_info.block_count() - 1;
|
|
EXPECT_EQ(block_info.piece_size(final_piece), completion.count_missing_bytes_in_piece(final_piece));
|
|
completion.add_block(final_block);
|
|
EXPECT_EQ(1U, block_info.piece_size(block_info.piece_count() - 1));
|
|
EXPECT_TRUE(completion.has_piece(final_piece));
|
|
EXPECT_EQ(0U, completion.count_missing_bytes_in_piece(final_piece));
|
|
}
|
|
|
|
TEST_F(CompletionTest, amountDone)
|
|
{
|
|
auto torrent = TestTorrent{};
|
|
auto constexpr TotalSize = uint64_t{ BlockSize * 4096 } + 1;
|
|
auto constexpr PieceSize = uint64_t{ BlockSize * 64 };
|
|
auto const block_info = tr_block_info{ TotalSize, PieceSize };
|
|
auto completion = torrent.makeCompletion(block_info);
|
|
|
|
// make bins s.t. each bin is a single piece
|
|
auto bins = std::array<float, TotalSize / PieceSize>{};
|
|
|
|
for (tr_piece_index_t piece = 0; piece < block_info.piece_count(); ++piece)
|
|
{
|
|
completion.remove_piece(piece);
|
|
}
|
|
completion.amount_done(std::data(bins), std::size(bins));
|
|
std::for_each(std::begin(bins), std::end(bins), [](float bin) { EXPECT_DOUBLE_EQ(0.0, bin); });
|
|
|
|
// one block
|
|
completion.add_block(0);
|
|
completion.amount_done(std::data(bins), std::size(bins));
|
|
EXPECT_DOUBLE_EQ(0.0, bins[1]);
|
|
|
|
// one piece
|
|
completion.add_piece(0);
|
|
completion.amount_done(std::data(bins), std::size(bins));
|
|
EXPECT_DOUBLE_EQ(1.0, bins[0]);
|
|
EXPECT_DOUBLE_EQ(0.0, bins[1]);
|
|
|
|
// all pieces
|
|
for (tr_piece_index_t piece = 0; piece < block_info.piece_count(); ++piece)
|
|
{
|
|
completion.add_piece(piece);
|
|
}
|
|
completion.amount_done(std::data(bins), std::size(bins));
|
|
std::for_each(std::begin(bins), std::end(bins), [](float bin) { EXPECT_DOUBLE_EQ(1.0, bin); });
|
|
|
|
// don't do anything if fed bad input
|
|
auto const backup = bins;
|
|
completion.amount_done(std::data(bins), 0);
|
|
EXPECT_EQ(backup, bins);
|
|
}
|
|
|
|
TEST_F(CompletionTest, countHasBytesInSpan)
|
|
{
|
|
// set up a fake torrent
|
|
auto torrent = TestTorrent{};
|
|
auto constexpr TotalSize = uint64_t{ BlockSize * 4096 } + 1;
|
|
auto constexpr PieceSize = uint64_t{ BlockSize * 64 };
|
|
auto const block_info = tr_block_info{ TotalSize, PieceSize };
|
|
auto completion = torrent.makeCompletion(block_info);
|
|
|
|
// torrent is complete
|
|
auto blocks = tr_bitfield{ block_info.block_count() };
|
|
blocks.set_has_all();
|
|
completion.set_blocks(blocks);
|
|
|
|
EXPECT_EQ(TotalSize, completion.count_has_bytes_in_span({ 0, TotalSize }));
|
|
EXPECT_EQ(TotalSize, completion.count_has_bytes_in_span({ 0, TotalSize + 1 }));
|
|
// test span that's entirely in a single block
|
|
EXPECT_EQ(1, completion.count_has_bytes_in_span({ 16, 17 }));
|
|
EXPECT_EQ(16, completion.count_has_bytes_in_span({ 16, 32 }));
|
|
// test edge cases on block boundary
|
|
EXPECT_EQ(1, completion.count_has_bytes_in_span({ BlockSize - 1, BlockSize }));
|
|
EXPECT_EQ(1, completion.count_has_bytes_in_span({ BlockSize, BlockSize + 1 }));
|
|
EXPECT_EQ(2, completion.count_has_bytes_in_span({ BlockSize - 1, BlockSize + 1 }));
|
|
// test edge cases on piece boundary
|
|
EXPECT_EQ(1, completion.count_has_bytes_in_span({ PieceSize - 1, PieceSize }));
|
|
EXPECT_EQ(1, completion.count_has_bytes_in_span({ PieceSize, PieceSize + 1 }));
|
|
EXPECT_EQ(2, completion.count_has_bytes_in_span({ PieceSize - 1, PieceSize + 1 }));
|
|
|
|
// test span that has a middle block
|
|
EXPECT_EQ(BlockSize * 3, completion.count_has_bytes_in_span({ 0, BlockSize * 3 }));
|
|
EXPECT_EQ(BlockSize * 2, completion.count_has_bytes_in_span({ BlockSize / 2, (BlockSize * 2) + (BlockSize / 2) }));
|
|
|
|
// test span where first block is missing
|
|
blocks.unset(0);
|
|
completion.set_blocks(blocks);
|
|
EXPECT_EQ(BlockSize * 2, completion.count_has_bytes_in_span({ 0, BlockSize * 3 }));
|
|
EXPECT_EQ(BlockSize * 1.5, completion.count_has_bytes_in_span({ BlockSize / 2, (BlockSize * 2) + (BlockSize / 2) }));
|
|
// test span where final block is missing
|
|
blocks.set_has_all();
|
|
blocks.unset(2);
|
|
completion.set_blocks(blocks);
|
|
EXPECT_EQ(BlockSize * 2, completion.count_has_bytes_in_span({ 0, BlockSize * 3 }));
|
|
EXPECT_EQ(BlockSize * 1.5, completion.count_has_bytes_in_span({ BlockSize / 2, (BlockSize * 2) + (BlockSize / 2) }));
|
|
}
|
|
|
|
TEST_F(CompletionTest, wantNone)
|
|
{
|
|
auto torrent = TestTorrent{};
|
|
auto constexpr TotalSize = uint64_t{ BlockSize * 4096 };
|
|
auto constexpr PieceSize = uint64_t{ BlockSize * 64 };
|
|
auto const block_info = tr_block_info{ TotalSize, PieceSize };
|
|
auto completion = torrent.makeCompletion(block_info);
|
|
|
|
// we have some data
|
|
completion.add_block(0);
|
|
|
|
// and want nothing
|
|
for (tr_piece_index_t i = 0, n = block_info.block_count(); i < n; ++i)
|
|
{
|
|
torrent.dnd_pieces.insert(i);
|
|
}
|
|
completion.invalidate_size_when_done();
|
|
|
|
EXPECT_LE(completion.has_total(), completion.size_when_done());
|
|
EXPECT_EQ(completion.has_total(), block_info.BlockSize);
|
|
EXPECT_EQ(completion.size_when_done(), block_info.BlockSize);
|
|
EXPECT_LE(completion.left_until_done(), completion.size_when_done());
|
|
EXPECT_EQ(completion.left_until_done(), 0);
|
|
|
|
// but we magically get a block anyway
|
|
completion.add_block(1);
|
|
|
|
EXPECT_LE(completion.has_total(), completion.size_when_done());
|
|
EXPECT_EQ(completion.has_total(), 2 * block_info.BlockSize);
|
|
EXPECT_EQ(completion.size_when_done(), 2 * block_info.BlockSize);
|
|
EXPECT_LE(completion.left_until_done(), completion.size_when_done());
|
|
EXPECT_EQ(completion.left_until_done(), 0);
|
|
}
|