Added --skip-previous implementation
This commit is contained in:
parent
e829b39683
commit
9d3441ffd7
5 changed files with 96 additions and 12 deletions
|
@ -146,6 +146,7 @@ class Selection:
|
||||||
class App:
|
class App:
|
||||||
def __init__(self, args: Namespace):
|
def __init__(self, args: Namespace):
|
||||||
self.__config = Config(args)
|
self.__config = Config(args)
|
||||||
|
self.__existing = {}
|
||||||
Logger(self.__config)
|
Logger(self.__config)
|
||||||
|
|
||||||
# Create session
|
# Create session
|
||||||
|
@ -180,6 +181,11 @@ class App:
|
||||||
Logger.log(LogChannel.ERRORS, str(e))
|
Logger.log(LogChannel.ERRORS, str(e))
|
||||||
exit(1)
|
exit(1)
|
||||||
if len(collections) > 0:
|
if len(collections) > 0:
|
||||||
|
self.scan(
|
||||||
|
collections,
|
||||||
|
self.__config.skip_previous,
|
||||||
|
self.__config.skip_duplicates,
|
||||||
|
)
|
||||||
self.download_all(collections)
|
self.download_all(collections)
|
||||||
else:
|
else:
|
||||||
Logger.log(LogChannel.WARNINGS, "there is nothing to do")
|
Logger.log(LogChannel.WARNINGS, "there is nothing to do")
|
||||||
|
@ -240,6 +246,21 @@ class App:
|
||||||
raise ParseError(f'Unsupported content type "{id_type}"')
|
raise ParseError(f'Unsupported content type "{id_type}"')
|
||||||
return collections
|
return collections
|
||||||
|
|
||||||
|
def scan(
|
||||||
|
self,
|
||||||
|
collections: list[Collection],
|
||||||
|
skip_previous: bool,
|
||||||
|
skip_duplicate: bool,
|
||||||
|
):
|
||||||
|
if skip_previous:
|
||||||
|
for collection in collections:
|
||||||
|
existing = collection.get_existing(
|
||||||
|
self.__config.audio_format.value.ext
|
||||||
|
)
|
||||||
|
self.__existing.update(existing)
|
||||||
|
if skip_duplicate:
|
||||||
|
pass
|
||||||
|
|
||||||
def download_all(self, collections: list[Collection]) -> None:
|
def download_all(self, collections: list[Collection]) -> None:
|
||||||
count = 0
|
count = 0
|
||||||
total = sum(len(c.playables) for c in collections)
|
total = sum(len(c.playables) for c in collections)
|
||||||
|
@ -247,6 +268,13 @@ class App:
|
||||||
for playable in collection.playables:
|
for playable in collection.playables:
|
||||||
count += 1
|
count += 1
|
||||||
|
|
||||||
|
if playable.existing:
|
||||||
|
Logger.log(
|
||||||
|
LogChannel.SKIPS,
|
||||||
|
f'Skipping "{self.__existing[playable.id]}": Previously downloaded',
|
||||||
|
)
|
||||||
|
continue
|
||||||
|
|
||||||
# Get track data
|
# Get track data
|
||||||
if playable.type == PlayableType.TRACK:
|
if playable.type == PlayableType.TRACK:
|
||||||
with Loader("Fetching track..."):
|
with Loader("Fetching track..."):
|
||||||
|
@ -257,7 +285,7 @@ class App:
|
||||||
except RuntimeError as err:
|
except RuntimeError as err:
|
||||||
Logger.log(
|
Logger.log(
|
||||||
LogChannel.SKIPS,
|
LogChannel.SKIPS,
|
||||||
f'Skipping song id = {playable.id}: {err}',
|
f'Skipping track id = {playable.id}: {err}',
|
||||||
)
|
)
|
||||||
continue
|
continue
|
||||||
elif playable.type == PlayableType.EPISODE:
|
elif playable.type == PlayableType.EPISODE:
|
||||||
|
|
|
@ -1,3 +1,6 @@
|
||||||
|
from pathlib import Path
|
||||||
|
from glob import iglob
|
||||||
|
|
||||||
from librespot.metadata import (
|
from librespot.metadata import (
|
||||||
AlbumId,
|
AlbumId,
|
||||||
ArtistId,
|
ArtistId,
|
||||||
|
@ -7,17 +10,55 @@ from librespot.metadata import (
|
||||||
|
|
||||||
from zotify import ApiClient
|
from zotify import ApiClient
|
||||||
from zotify.config import Config
|
from zotify.config import Config
|
||||||
from zotify.utils import MetadataEntry, PlayableData, PlayableType, bytes_to_base62
|
from zotify.file import LocalFile
|
||||||
|
from zotify.utils import (
|
||||||
|
MetadataEntry,
|
||||||
|
PlayableData,
|
||||||
|
PlayableType,
|
||||||
|
bytes_to_base62,
|
||||||
|
fix_filename,
|
||||||
|
)
|
||||||
|
|
||||||
class Collection:
|
class Collection:
|
||||||
def __init__(self, b62_id: str, api: ApiClient, config: Config = Config()):
|
def __init__(self):
|
||||||
self.playables: list[PlayableData] = []
|
self.playables: list[PlayableData] = []
|
||||||
|
|
||||||
|
def get_existing(self, ext: str) -> dict[str, str]:
|
||||||
|
existing: dict[str, str] = {}
|
||||||
|
|
||||||
|
meta_tags = ["album_artist", "album", "podcast", "playlist"]
|
||||||
|
library = Path(self.playables[0].library)
|
||||||
|
output = self.playables[0].output_template
|
||||||
|
metadata = self.playables[0].metadata
|
||||||
|
id_type = self.playables[0].type
|
||||||
|
|
||||||
|
for meta in metadata:
|
||||||
|
if meta.name in meta_tags:
|
||||||
|
output = output.replace(
|
||||||
|
"{" + meta.name + "}", fix_filename(meta.string)
|
||||||
|
)
|
||||||
|
|
||||||
|
collection_path = library.joinpath(output).expanduser()
|
||||||
|
if collection_path.parent.exists():
|
||||||
|
file_path = "*.{}".format(ext)
|
||||||
|
scan_path = str(collection_path.parent.joinpath(file_path))
|
||||||
|
|
||||||
|
# Check contents of path
|
||||||
|
for file in iglob(scan_path):
|
||||||
|
f_path = Path(file)
|
||||||
|
f = LocalFile(f_path)
|
||||||
|
existing[f.get_metadata("key")] = f_path.stem
|
||||||
|
|
||||||
|
for playable in self.playables:
|
||||||
|
if playable.id in existing.keys():
|
||||||
|
playable.existing = True
|
||||||
|
|
||||||
|
return existing
|
||||||
|
|
||||||
|
|
||||||
class Album(Collection):
|
class Album(Collection):
|
||||||
def __init__(self, b62_id: str, api: ApiClient, config: Config = Config()):
|
def __init__(self, b62_id: str, api: ApiClient, config: Config = Config()):
|
||||||
super().__init__(b62_id, api, config)
|
super().__init__()
|
||||||
album = api.get_metadata_4_album(AlbumId.from_base62(b62_id))
|
album = api.get_metadata_4_album(AlbumId.from_base62(b62_id))
|
||||||
for disc in album.disc:
|
for disc in album.disc:
|
||||||
for track in disc.track:
|
for track in disc.track:
|
||||||
|
@ -35,7 +76,7 @@ class Album(Collection):
|
||||||
|
|
||||||
class Artist(Collection):
|
class Artist(Collection):
|
||||||
def __init__(self, b62_id: str, api: ApiClient, config: Config = Config()):
|
def __init__(self, b62_id: str, api: ApiClient, config: Config = Config()):
|
||||||
super().__init__(b62_id, api, config)
|
super().__init__()
|
||||||
artist = api.get_metadata_4_artist(ArtistId.from_base62(b62_id))
|
artist = api.get_metadata_4_artist(ArtistId.from_base62(b62_id))
|
||||||
for album_group in (
|
for album_group in (
|
||||||
artist.album_group
|
artist.album_group
|
||||||
|
@ -60,7 +101,7 @@ class Artist(Collection):
|
||||||
|
|
||||||
class Show(Collection):
|
class Show(Collection):
|
||||||
def __init__(self, b62_id: str, api: ApiClient, config: Config = Config()):
|
def __init__(self, b62_id: str, api: ApiClient, config: Config = Config()):
|
||||||
super().__init__(b62_id, api, config)
|
super().__init__()
|
||||||
show = api.get_metadata_4_show(ShowId.from_base62(b62_id))
|
show = api.get_metadata_4_show(ShowId.from_base62(b62_id))
|
||||||
for episode in show.episode:
|
for episode in show.episode:
|
||||||
metadata = [MetadataEntry("key", bytes_to_base62(episode.gid))]
|
metadata = [MetadataEntry("key", bytes_to_base62(episode.gid))]
|
||||||
|
@ -77,7 +118,7 @@ class Show(Collection):
|
||||||
|
|
||||||
class Playlist(Collection):
|
class Playlist(Collection):
|
||||||
def __init__(self, b62_id: str, api: ApiClient, config: Config = Config()):
|
def __init__(self, b62_id: str, api: ApiClient, config: Config = Config()):
|
||||||
super().__init__(b62_id, api, config)
|
super().__init__()
|
||||||
playlist = api.get_playlist(PlaylistId(b62_id))
|
playlist = api.get_playlist(PlaylistId(b62_id))
|
||||||
for i in range(len(playlist.contents.items)):
|
for i in range(len(playlist.contents.items)):
|
||||||
item = playlist.contents.items[i]
|
item = playlist.contents.items[i]
|
||||||
|
@ -124,7 +165,7 @@ class Playlist(Collection):
|
||||||
|
|
||||||
class Track(Collection):
|
class Track(Collection):
|
||||||
def __init__(self, b62_id: str, api: ApiClient, config: Config = Config()):
|
def __init__(self, b62_id: str, api: ApiClient, config: Config = Config()):
|
||||||
super().__init__(b62_id, api, config)
|
super().__init__()
|
||||||
metadata = [MetadataEntry("key", b62_id)]
|
metadata = [MetadataEntry("key", b62_id)]
|
||||||
self.playables.append(
|
self.playables.append(
|
||||||
PlayableData(
|
PlayableData(
|
||||||
|
@ -139,7 +180,7 @@ class Track(Collection):
|
||||||
|
|
||||||
class Episode(Collection):
|
class Episode(Collection):
|
||||||
def __init__(self, b62_id: str, api: ApiClient, config: Config = Config()):
|
def __init__(self, b62_id: str, api: ApiClient, config: Config = Config()):
|
||||||
super().__init__(b62_id, api, config)
|
super().__init__()
|
||||||
metadata = [MetadataEntry("key", b62_id)]
|
metadata = [MetadataEntry("key", b62_id)]
|
||||||
self.playables.append(
|
self.playables.append(
|
||||||
PlayableData(
|
PlayableData(
|
||||||
|
|
|
@ -117,3 +117,13 @@ class LocalFile:
|
||||||
f.save()
|
f.save()
|
||||||
except OggVorbisHeaderError:
|
except OggVorbisHeaderError:
|
||||||
pass # Thrown when using untranscoded file, nothing breaks.
|
pass # Thrown when using untranscoded file, nothing breaks.
|
||||||
|
|
||||||
|
def get_metadata(self, tag: str) -> str:
|
||||||
|
"""
|
||||||
|
Gets metadata from file
|
||||||
|
Args:
|
||||||
|
tag: metadata tag to be retrieved
|
||||||
|
"""
|
||||||
|
f = load_file(self.__path)
|
||||||
|
|
||||||
|
return f[tag].value
|
|
@ -89,6 +89,8 @@ class Playable:
|
||||||
file_path = library.joinpath(output).expanduser()
|
file_path = library.joinpath(output).expanduser()
|
||||||
file_path = Path(f'{file_path}.{ext}')
|
file_path = Path(f'{file_path}.{ext}')
|
||||||
if file_path.exists() and not replace:
|
if file_path.exists() and not replace:
|
||||||
|
f = LocalFile(file_path)
|
||||||
|
f.write_metadata(self.metadata)
|
||||||
raise FileExistsError("File already downloaded")
|
raise FileExistsError("File already downloaded")
|
||||||
else:
|
else:
|
||||||
file_path.parent.mkdir(parents=True, exist_ok=True)
|
file_path.parent.mkdir(parents=True, exist_ok=True)
|
||||||
|
|
|
@ -3,6 +3,7 @@ from enum import Enum, IntEnum
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
from re import IGNORECASE, sub
|
from re import IGNORECASE, sub
|
||||||
from typing import Any, NamedTuple
|
from typing import Any, NamedTuple
|
||||||
|
from dataclasses import dataclass, field
|
||||||
|
|
||||||
from librespot.audio.decoders import AudioQuality
|
from librespot.audio.decoders import AudioQuality
|
||||||
from librespot.util import Base62
|
from librespot.util import Base62
|
||||||
|
@ -110,12 +111,14 @@ class PlayableType(Enum):
|
||||||
EPISODE = "episode"
|
EPISODE = "episode"
|
||||||
|
|
||||||
|
|
||||||
class PlayableData(NamedTuple):
|
@dataclass
|
||||||
|
class PlayableData():
|
||||||
type: PlayableType
|
type: PlayableType
|
||||||
id: str
|
id: str
|
||||||
library: Path
|
library: Path
|
||||||
output_template: str
|
output_template: str
|
||||||
metadata: list[MetadataEntry] = []
|
metadata: list[MetadataEntry] = field(default_factory=list)
|
||||||
|
existing: bool = False
|
||||||
|
|
||||||
|
|
||||||
class OptionalOrFalse(Action):
|
class OptionalOrFalse(Action):
|
||||||
|
|
Loading…
Add table
Reference in a new issue