Source code for transmission_rpc.utils

# Copyright (c) 2018-2021 Trim21 <i@trim21.me>
# Copyright (c) 2008-2014 Erik Svensson <erik.public@gmail.com>
# Licensed under the MIT license.
import base64
import pathlib
import datetime
import warnings
from typing import Any, Dict, List, Tuple, Union, BinaryIO, Callable, Optional
from urllib.parse import urlparse

from transmission_rpc import constants
from transmission_rpc.constants import Type

UNITS = ["B", "KiB", "MiB", "GiB", "TiB", "PiB", "EiB"]


[docs]def format_size(size: int) -> Tuple[float, str]: """ Format byte size into IEC prefixes, B, KiB, MiB ... """ s = float(size) i = 0 while s >= 1024.0 and i < len(UNITS): i += 1 s /= 1024.0 return s, UNITS[i]
[docs]def format_speed(size: int) -> Tuple[float, str]: """ Format bytes per second speed into IEC prefixes, B/s, KiB/s, MiB/s ... """ (s, unit) = format_size(size) return s, unit + "/s"
[docs]def format_timedelta(delta: datetime.timedelta) -> str: """ Format datetime.timedelta into <days> <hours>:<minutes>:<seconds>. """ minutes, seconds = divmod(delta.seconds, 60) hours, minutes = divmod(minutes, 60) return f"{delta.days:d} {hours:02d}:{minutes:02d}:{seconds:02d}"
[docs]def rpc_bool(arg: Any) -> int: """ Convert between Python boolean and Transmission RPC boolean. """ if isinstance(arg, str): try: arg = bool(int(arg)) except ValueError: arg = arg.lower() in ["true", "yes"] return 1 if bool(arg) else 0
TR_TYPE_MAP: Dict[str, Callable] = { Type.number: int, Type.string: str, Type.double: float, Type.boolean: rpc_bool, Type.array: list, Type.object: dict, }
[docs]def make_python_name(name: str) -> str: """ Convert Transmission RPC name to python compatible name. """ return name.replace("-", "_")
[docs]def make_rpc_name(name: str) -> str: """ Convert python compatible name to Transmission RPC name. """ return name.replace("_", "-")
[docs]def get_torrent_arguments(rpc_version: int) -> List[str]: """ Get torrent arguments for method in specified Transmission RPC version. """ accessible = [] for argument, info in constants.TORRENT_GET_ARGS.items(): valid_version = True if rpc_version < info.added_version: valid_version = False if info.removed_version is not None and info.removed_version <= rpc_version: valid_version = False if valid_version: accessible.append(argument) return accessible
def _try_read_torrent(torrent: Union[BinaryIO, str, bytes, pathlib.Path]) -> Optional[str]: # pylint: disable=R0911 """ if torrent should be encoded with base64, return a non-None value. """ # torrent is a str, may be a url if isinstance(torrent, str): parsed_uri = urlparse(torrent) # torrent starts with file, read from local disk and encode it to base64 url. if parsed_uri.scheme in ["https", "http", "magnet"]: return None if parsed_uri.scheme in ["file"]: warnings.warn("support for `file://` URL is deprecated.", DeprecationWarning) filepath = torrent # uri decoded different on linux / windows ? if len(parsed_uri.path) > 0: filepath = parsed_uri.path elif len(parsed_uri.netloc) > 0: filepath = parsed_uri.netloc with open(filepath, "rb") as torrent_file: return base64.b64encode(torrent_file.read()).decode("utf-8") # maybe it's base64 encoded file content try: # check if this is base64 data base64.b64decode(torrent.encode("utf-8"), validate=True) warnings.warn( "support for base64 encoded torrent content str is deprecated.", DeprecationWarning, ) return torrent except (TypeError, ValueError): pass elif isinstance(torrent, pathlib.Path): return base64.b64encode(torrent.read_bytes()).decode("utf-8") elif isinstance(torrent, bytes): return base64.b64encode(torrent).decode("utf-8") # maybe a file, try read content and encode it. elif hasattr(torrent, "read"): return base64.b64encode(torrent.read()).decode("utf-8") return None