2023-02-25 21:16:32 +00:00
|
|
|
from pathlib import Path
|
2023-04-04 18:58:22 +00:00
|
|
|
from typing import List, Tuple
|
|
|
|
|
2023-03-31 07:47:03 +00:00
|
|
|
import requests
|
2023-04-04 15:20:27 +00:00
|
|
|
from tqdm import tqdm
|
2023-02-25 21:16:32 +00:00
|
|
|
|
|
|
|
from .parents import DatabaseObject
|
2023-04-04 18:58:22 +00:00
|
|
|
from ..utils import shared
|
2023-02-25 21:16:32 +00:00
|
|
|
|
|
|
|
|
|
|
|
class Target(DatabaseObject):
|
|
|
|
"""
|
|
|
|
create somehow like that
|
|
|
|
```python
|
|
|
|
# I know path is pointless, and I will change that (don't worry about backwards compatibility there)
|
|
|
|
Target(file="song.mp3", path="~/Music/genre/artist/album")
|
|
|
|
```
|
|
|
|
"""
|
|
|
|
|
2023-03-18 11:36:53 +00:00
|
|
|
SIMPLE_ATTRIBUTES = {
|
|
|
|
"_file": None,
|
|
|
|
"_path": None
|
|
|
|
}
|
2023-03-13 13:33:17 +00:00
|
|
|
COLLECTION_ATTRIBUTES = tuple()
|
2023-03-09 19:44:57 +00:00
|
|
|
|
2023-02-25 21:16:32 +00:00
|
|
|
def __init__(
|
|
|
|
self,
|
|
|
|
file: str = None,
|
|
|
|
path: str = None,
|
|
|
|
dynamic: bool = False,
|
|
|
|
relative_to_music_dir: bool = False
|
|
|
|
) -> None:
|
2023-03-30 12:39:28 +00:00
|
|
|
super().__init__(dynamic=dynamic)
|
2023-02-25 21:16:32 +00:00
|
|
|
self._file: Path = Path(file)
|
2023-03-30 12:39:28 +00:00
|
|
|
self._path: Path = Path(shared.MUSIC_DIR, path) if relative_to_music_dir else Path(path)
|
2023-02-25 21:16:32 +00:00
|
|
|
|
|
|
|
self.is_relative_to_music_dir: bool = relative_to_music_dir
|
|
|
|
|
2023-03-30 12:39:28 +00:00
|
|
|
def __repr__(self) -> str:
|
|
|
|
return str(self.file_path)
|
|
|
|
|
2023-02-25 21:16:32 +00:00
|
|
|
@property
|
|
|
|
def file_path(self) -> Path:
|
|
|
|
return Path(self._path, self._file)
|
2023-03-09 19:44:57 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def indexing_values(self) -> List[Tuple[str, object]]:
|
|
|
|
return [('filepath', self.file_path)]
|
2023-04-04 15:54:05 +00:00
|
|
|
|
2023-03-30 12:59:53 +00:00
|
|
|
@property
|
|
|
|
def exists(self) -> bool:
|
2023-03-30 14:10:48 +00:00
|
|
|
return self.file_path.is_file()
|
2023-04-04 15:54:05 +00:00
|
|
|
|
2023-03-30 12:59:53 +00:00
|
|
|
def create_path(self):
|
|
|
|
self._path.mkdir(parents=True, exist_ok=True)
|
2023-04-04 15:54:05 +00:00
|
|
|
|
2023-03-30 13:28:23 +00:00
|
|
|
def copy_content(self, copy_to: "Target"):
|
|
|
|
if not self.exists:
|
|
|
|
return
|
2023-04-04 15:54:05 +00:00
|
|
|
|
2023-03-30 13:28:23 +00:00
|
|
|
with open(self.file_path, "rb") as read_from:
|
|
|
|
copy_to.create_path()
|
2023-03-31 07:54:31 +00:00
|
|
|
with open(copy_to.file_path, "wb") as write_to:
|
2023-03-30 13:28:23 +00:00
|
|
|
write_to.write(read_from.read())
|
2023-03-31 07:47:03 +00:00
|
|
|
|
2023-04-04 15:59:08 +00:00
|
|
|
def stream_into(self, r: requests.Response, desc: str = None) -> bool:
|
2023-04-03 10:14:58 +00:00
|
|
|
if r is None:
|
|
|
|
return False
|
2023-04-04 15:54:05 +00:00
|
|
|
|
2023-03-31 07:47:03 +00:00
|
|
|
self.create_path()
|
2023-04-04 15:54:05 +00:00
|
|
|
|
2023-03-31 07:47:03 +00:00
|
|
|
total_size = int(r.headers.get('content-length'))
|
2023-04-04 15:54:05 +00:00
|
|
|
|
|
|
|
with open(self.file_path, 'wb') as f:
|
2023-04-03 10:14:58 +00:00
|
|
|
try:
|
2023-04-04 15:20:27 +00:00
|
|
|
"""
|
|
|
|
https://en.wikipedia.org/wiki/Kilobyte
|
|
|
|
> The internationally recommended unit symbol for the kilobyte is kB.
|
|
|
|
"""
|
2023-04-04 15:59:08 +00:00
|
|
|
with tqdm(total=total_size, unit='B', unit_scale=True, unit_divisor=1024, desc=desc) as t:
|
|
|
|
|
|
|
|
for chunk in r.iter_content(chunk_size=shared.CHUNK_SIZE):
|
2023-04-04 15:54:05 +00:00
|
|
|
size = f.write(chunk)
|
|
|
|
t.update(size)
|
2023-04-04 18:58:22 +00:00
|
|
|
return True
|
2023-04-04 15:54:05 +00:00
|
|
|
|
2023-04-03 10:14:58 +00:00
|
|
|
except requests.exceptions.Timeout:
|
|
|
|
shared.DOWNLOAD_LOGGER.error("Stream timed out.")
|
|
|
|
return False
|