Skip to content

Commit

Permalink
Integrated torrent file tree into Download/TorrentDef (#7694)
Browse files Browse the repository at this point in the history
  • Loading branch information
qstokkink authored Nov 21, 2023
1 parent 4140aff commit da5911d
Show file tree
Hide file tree
Showing 7 changed files with 515 additions and 35 deletions.
11 changes: 10 additions & 1 deletion src/tribler/core/components/libtorrent/conftest.py
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@
from tribler.core.components.libtorrent.download_manager.download import Download
from tribler.core.components.libtorrent.download_manager.download_config import DownloadConfig
from tribler.core.components.libtorrent.torrentdef import TorrentDef
from tribler.core.tests.tools.common import TESTS_DATA_DIR
from tribler.core.tests.tools.common import TESTS_DATA_DIR, TORRENT_UBUNTU_FILE, TORRENT_VIDEO_FILE
from tribler.core.utilities.unicode import hexlify


Expand Down Expand Up @@ -83,3 +83,12 @@ def mock_lt_status():
lt_status.pieces = []
lt_status.finished_time = 10
return lt_status


@pytest.fixture
def dual_movie_tdef() -> TorrentDef:
tdef = TorrentDef()
tdef.add_content(TORRENT_VIDEO_FILE)
tdef.add_content(TORRENT_UBUNTU_FILE)
tdef.save()
return tdef
Original file line number Diff line number Diff line change
Expand Up @@ -3,13 +3,16 @@
Author(s): Arno Bakker, Egbert Bouman
"""
from __future__ import annotations

import asyncio
import base64
import itertools
import logging
from asyncio import CancelledError, Future, iscoroutine, sleep, wait_for
from asyncio import CancelledError, Future, iscoroutine, sleep, wait_for, get_running_loop
from collections import defaultdict
from contextlib import suppress
from enum import Enum
from typing import Any, Awaitable, Callable, Dict, List, Optional, Tuple

from bitarray import bitarray
Expand All @@ -21,6 +24,7 @@
from tribler.core.components.libtorrent.download_manager.download_state import DownloadState
from tribler.core.components.libtorrent.download_manager.stream import Stream
from tribler.core.components.libtorrent.settings import DownloadDefaultsSettings
from tribler.core.components.libtorrent.torrent_file_tree import TorrentFileTree
from tribler.core.components.libtorrent.torrentdef import TorrentDef, TorrentDefNoMetainfo
from tribler.core.components.libtorrent.utils.libtorrent_helper import libtorrent as lt
from tribler.core.components.libtorrent.utils.torrent_utils import check_handle, get_info_from_handle, require_handle
Expand All @@ -37,6 +41,15 @@
Getter = Callable[[Any], Any]


class IllegalFileIndex(Enum):
"""
Error codes for Download.get_file_index(). These are used by the GUI to render directories.
"""
collapsed_dir = -1
expanded_dir = -2
unloaded = -3


class Download(TaskManager):
""" Download subclass that represents a libtorrent download."""

Expand Down Expand Up @@ -379,6 +392,9 @@ def on_metadata_received_alert(self, alert: lt.metadata_received_alert):

try:
self.tdef = TorrentDef.load_from_dict(metadata)
with suppress(RuntimeError):
# Try to load the torrent info in the background if we have a loop.
get_running_loop().run_in_executor(None, self.tdef.load_torrent_info)
except ValueError as ve:
self._logger.exception(ve)
return
Expand Down Expand Up @@ -794,3 +810,75 @@ def set_piece_deadline(self, piece, deadline, flags=0):
@check_handle([])
def get_file_priorities(self):
return self.handle.file_priorities()

def file_piece_range(self, file_path: Path) -> list[int]:
"""
Get the piece range of a given file, specified by the path.
Calling this method with anything but a file path will return an empty list.
"""
file_index = self.get_file_index(file_path)
if file_index < 0:
return []

start_piece = self.tdef.torrent_info.map_file(file_index, 0, 1).piece
# Note: next_piece contains the next piece that is NOT part of this file.
if file_index < self.tdef.torrent_info.num_files() - 1:
next_piece = self.tdef.torrent_info.map_file(file_index + 1, 0, 1).piece
else:
# There is no next file so the nex piece is the last piece index + 1 (num_pieces()).
next_piece = self.tdef.torrent_info.num_pieces()

return list(range(start_piece, next_piece))

@check_handle(0.0)
def get_file_completion(self, path: Path) -> float:
"""
Calculate the completion of a given file or directory.
"""
total = 0
have = 0
for piece_index in self.file_piece_range(path):
have += self.handle.have_piece(piece_index)
total += 1
if total == 0:
return 1.0
return have/total

def get_file_length(self, path: Path) -> int:
"""
Get the length of a file or directory in bytes. Returns 0 if the given path does not point to an existing path.
"""
result = self.tdef.torrent_file_tree.find(path)
if result is not None:
return result.size
return 0

def get_file_index(self, path: Path) -> int:
"""
Get the index of a file or directory in a torrent. Note that directories do not have real file indices.
Special cases ("error codes"):
- ``-1`` (IllegalFileIndex.collapsed_dir): the given path is not a file but a collapsed directory.
- ``-2`` (IllegalFileIndex.expanded_dir): the given path is not a file but an expanded directory.
- ``-3`` (IllegalFileIndex.unloaded): the data structure is not loaded or the path is not found.
"""
result = self.tdef.torrent_file_tree.find(path)
if isinstance(result, TorrentFileTree.File):
return self.tdef.torrent_file_tree.find(path).index
if isinstance(result, TorrentFileTree.Directory):
return (IllegalFileIndex.collapsed_dir.value if result.collapsed
else IllegalFileIndex.expanded_dir.value)
return IllegalFileIndex.unloaded.value

def is_file_selected(self, file_path: Path) -> bool:
"""
Check if the given file path is selected.
Calling this method with anything but a file path will return False.
"""
result = self.tdef.torrent_file_tree.find(file_path)
if isinstance(result, TorrentFileTree.File):
return result.selected
return False
184 changes: 177 additions & 7 deletions src/tribler/core/components/libtorrent/tests/test_download.py
Original file line number Diff line number Diff line change
Expand Up @@ -8,14 +8,14 @@
from ipv8.util import succeed
from libtorrent import bencode

from tribler.core.components.libtorrent.download_manager.download import Download
from tribler.core.components.libtorrent.download_manager.download import Download, IllegalFileIndex
from tribler.core.components.libtorrent.download_manager.download_config import DownloadConfig
from tribler.core.components.libtorrent.torrentdef import TorrentDefNoMetainfo
from tribler.core.components.libtorrent.torrentdef import TorrentDef, TorrentDefNoMetainfo
from tribler.core.components.libtorrent.utils.torrent_utils import get_info_from_handle
from tribler.core.components.reporter.exception_handler import NoCrashException
from tribler.core.exceptions import SaveResumeDataError
from tribler.core.tests.tools.base_test import MockObject
from tribler.core.tests.tools.common import TESTS_DATA_DIR
from tribler.core.tests.tools.common import TESTS_DATA_DIR, TORRENT_UBUNTU_FILE, TORRENT_VIDEO_FILE
from tribler.core.utilities.unicode import hexlify
from tribler.core.utilities.utilities import bdecode_compat

Expand Down Expand Up @@ -308,15 +308,15 @@ def mocked_checkpoint():
test_download.on_metadata_received_alert(None)


@patch('tribler.core.components.libtorrent.download_manager.download.get_info_from_handle', Mock())
@patch('tribler.core.components.libtorrent.download_manager.download.bdecode_compat', Mock())
def test_on_metadata_received_alert_unicode_error(test_download):
def test_on_metadata_received_alert_unicode_error(test_download, dual_movie_tdef):
""" Test the the case the field 'url' is not unicode compatible. In this case no exceptions should be raised.
See: https://github.com/Tribler/tribler/issues/7223
"""
test_download.tdef = dual_movie_tdef
tracker = {'url': Mock(encode=Mock(side_effect=UnicodeDecodeError('', b'', 0, 0, '')))}
test_download.handle = MagicMock(trackers=Mock(return_value=[tracker]))
test_download.handle = MagicMock(trackers=Mock(return_value=[tracker]),
torrent_file=lambda: dual_movie_tdef.torrent_info)

test_download.on_metadata_received_alert(MagicMock())

Expand Down Expand Up @@ -506,3 +506,173 @@ async def test_shutdown(test_download: Download):

assert not test_download.futures
assert test_download.stream.close.called


def test_file_piece_range_flat(test_download: Download) -> None:
"""
Test if the piece range of a single-file torrent is correctly determined.
"""
total_pieces = test_download.tdef.torrent_info.num_pieces()

piece_range = test_download.file_piece_range(Path("video.avi"))

assert piece_range == list(range(total_pieces))


def test_file_piece_range_wide(dual_movie_tdef: TorrentDef) -> None:
"""
Test if the piece range of a two-file torrent is correctly determined.
The torrent is no longer flat after adding content! Data is now in the "data" directory.
"""
download = Download(dual_movie_tdef, checkpoint_disabled=True)

piece_range_video = download.file_piece_range(Path("data") / TORRENT_VIDEO_FILE.name)
piece_range_ubuntu = download.file_piece_range(Path("data") / TORRENT_UBUNTU_FILE.name)
last_piece = piece_range_video[-1] + 1

assert 0 < last_piece < download.tdef.torrent_info.num_pieces()
assert piece_range_video == list(range(0, last_piece))
assert piece_range_ubuntu == list(range(last_piece, download.tdef.torrent_info.num_pieces()))


def test_file_piece_range_nonexistent(test_download: Download) -> None:
"""
Test if the piece range of a single-file torrent is correctly determined.
"""
piece_range = test_download.file_piece_range(Path("I don't exist"))

assert piece_range == []


def test_file_completion_full(test_download: Download) -> None:
"""
Test if a complete file shows 1.0 completion.
"""
test_download.handle = MagicMock(have_piece=Mock(return_value=True))

assert 1.0 == test_download.get_file_completion(Path("video.avi"))


def test_file_completion_nonexistent(test_download: Download) -> None:
"""
Test if an unknown path (does not exist in a torrent) shows 1.0 completion.
"""
test_download.handle = MagicMock(have_piece=Mock(return_value=True))

assert 1.0 == test_download.get_file_completion(Path("I don't exist"))


def test_file_completion_directory(dual_movie_tdef: TorrentDef) -> None:
"""
Test if a directory (does not exist in a torrent) shows 1.0 completion.
"""
download = Download(dual_movie_tdef, checkpoint_disabled=True)
download.handle = MagicMock(have_piece=Mock(return_value=True))

assert 1.0 == download.get_file_completion(Path("data"))


def test_file_completion_nohandle(test_download: Download) -> None:
"""
Test if a file shows 0.0 completion if the torrent handle is not valid.
"""
test_download.handle = MagicMock(is_valid=Mock(return_value=False))

assert 0.0 == test_download.get_file_completion(Path("video.avi"))


def test_file_completion_partial(test_download: Download) -> None:
"""
Test if a file shows 0.0 completion if the torrent handle is not valid.
"""
total_pieces = test_download.tdef.torrent_info.num_pieces()
expected = (total_pieces // 2) / total_pieces

def fake_has_piece(piece_index: int) -> bool:
return piece_index > total_pieces / 2 # total_pieces // 2 will return True
test_download.handle = MagicMock(have_piece=fake_has_piece)

result = test_download.get_file_completion(Path("video.avi"))

assert round(expected, 4) == round(result, 4) # Round to make sure we don't get float rounding errors


def test_file_length(test_download: Download) -> None:
"""
Test if we can get the length of a file.
"""
assert 1942100 == test_download.get_file_length(Path("video.avi"))


def test_file_length_two(dual_movie_tdef: TorrentDef) -> None:
"""
Test if we can get the length of a file in a two-file torrent.
"""
download = Download(dual_movie_tdef, checkpoint_disabled=True)

assert 291888 == download.get_file_length(Path("data") / TORRENT_VIDEO_FILE.name)
assert 44258 == download.get_file_length(Path("data") / TORRENT_UBUNTU_FILE.name)


def test_file_length_nonexistent(test_download: Download) -> None:
"""
Test if the length of a non-existent file is 0.
"""
assert 0 == test_download.get_file_length(Path("I don't exist"))


def test_file_index_unloaded(test_download: Download) -> None:
"""
Test if a non-existent path leads to the special unloaded index.
"""
assert IllegalFileIndex.unloaded.value == test_download.get_file_index(Path("I don't exist"))


def test_file_index_directory_collapsed(dual_movie_tdef: TorrentDef) -> None:
"""
Test if a collapsed-dir path leads to the special collapsed dir index.
"""
download = Download(dual_movie_tdef, checkpoint_disabled=True)

assert IllegalFileIndex.collapsed_dir.value == download.get_file_index(Path("data"))


def test_file_index_directory_expanded(dual_movie_tdef: TorrentDef) -> None:
"""
Test if a expanded-dir path leads to the special expanded dir index.
"""
download = Download(dual_movie_tdef, checkpoint_disabled=True)
download.tdef.torrent_file_tree.expand(Path("data"))

assert IllegalFileIndex.expanded_dir.value == download.get_file_index(Path("data"))


def test_file_index_file(test_download: Download) -> None:
"""
Test if we can get the index of a file.
"""
assert 0 == test_download.get_file_index(Path("video.avi"))


def test_file_selected_nonexistent(test_download: Download) -> None:
"""
Test if a non-existent file does not register as selected.
"""
assert not test_download.is_file_selected(Path("I don't exist"))


def test_file_selected_realfile(test_download: Download) -> None:
"""
Test if a file starts off as selected.
"""
assert test_download.is_file_selected(Path("video.avi"))


def test_file_selected_directory(dual_movie_tdef: TorrentDef) -> None:
"""
Test if a directory does not register as selected.
"""
download = Download(dual_movie_tdef, checkpoint_disabled=True)

assert not download.is_file_selected(Path("data"))
Loading

0 comments on commit da5911d

Please sign in to comment.