mirror of
https://github.com/Wojtek242/qobuz-dl.git
synced 2024-11-25 20:45:25 +01:00
Add the option to disable quality fallback (close #35)
Also stop the version number heresy
This commit is contained in:
parent
c53181b885
commit
1bf8bb16de
@ -87,6 +87,7 @@ def main():
|
|||||||
arguments.embed_art,
|
arguments.embed_art,
|
||||||
ignore_singles_eps=arguments.albums_only,
|
ignore_singles_eps=arguments.albums_only,
|
||||||
no_m3u_for_playlists=arguments.no_m3u,
|
no_m3u_for_playlists=arguments.no_m3u,
|
||||||
|
quality_fallback=not arguments.no_fallback,
|
||||||
)
|
)
|
||||||
qobuz.initialize_client(email, password, app_id, secrets)
|
qobuz.initialize_client(email, password, app_id, secrets)
|
||||||
|
|
||||||
|
@ -56,9 +56,6 @@ def dl_args(subparsers):
|
|||||||
|
|
||||||
|
|
||||||
def add_common_arg(custom_parser, default_folder, default_quality):
|
def add_common_arg(custom_parser, default_folder, default_quality):
|
||||||
custom_parser.add_argument(
|
|
||||||
"-e", "--embed-art", action="store_true", help="embed cover art into files"
|
|
||||||
)
|
|
||||||
custom_parser.add_argument(
|
custom_parser.add_argument(
|
||||||
"-d",
|
"-d",
|
||||||
"--directory",
|
"--directory",
|
||||||
@ -86,6 +83,14 @@ def add_common_arg(custom_parser, default_folder, default_quality):
|
|||||||
action="store_true",
|
action="store_true",
|
||||||
help="don't create .m3u files when downloading playlists",
|
help="don't create .m3u files when downloading playlists",
|
||||||
)
|
)
|
||||||
|
custom_parser.add_argument(
|
||||||
|
"--no-fallback",
|
||||||
|
action="store_true",
|
||||||
|
help="disable quality fallback (skip releases not available in set quality)",
|
||||||
|
)
|
||||||
|
custom_parser.add_argument(
|
||||||
|
"-e", "--embed-art", action="store_true", help="embed cover art into files"
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
def qobuz_dl_args(
|
def qobuz_dl_args(
|
||||||
|
@ -53,7 +53,7 @@ class QobuzDL:
|
|||||||
interactive_limit=20,
|
interactive_limit=20,
|
||||||
ignore_singles_eps=False,
|
ignore_singles_eps=False,
|
||||||
no_m3u_for_playlists=False,
|
no_m3u_for_playlists=False,
|
||||||
raise_request_exceptions=False,
|
quality_fallback=True,
|
||||||
):
|
):
|
||||||
self.directory = self.create_dir(directory)
|
self.directory = self.create_dir(directory)
|
||||||
self.quality = quality
|
self.quality = quality
|
||||||
@ -63,6 +63,7 @@ class QobuzDL:
|
|||||||
self.interactive_limit = interactive_limit
|
self.interactive_limit = interactive_limit
|
||||||
self.ignore_singles_eps = ignore_singles_eps
|
self.ignore_singles_eps = ignore_singles_eps
|
||||||
self.no_m3u_for_playlists = no_m3u_for_playlists
|
self.no_m3u_for_playlists = no_m3u_for_playlists
|
||||||
|
self.quality_fallback = quality_fallback
|
||||||
|
|
||||||
def initialize_client(self, email, pwd, app_id, secrets):
|
def initialize_client(self, email, pwd, app_id, secrets):
|
||||||
self.client = qopy.Client(email, pwd, app_id, secrets)
|
self.client = qopy.Client(email, pwd, app_id, secrets)
|
||||||
@ -96,6 +97,7 @@ class QobuzDL:
|
|||||||
album,
|
album,
|
||||||
self.embed_art,
|
self.embed_art,
|
||||||
self.ignore_singles_eps,
|
self.ignore_singles_eps,
|
||||||
|
self.quality_fallback,
|
||||||
)
|
)
|
||||||
|
|
||||||
def handle_url(self, url):
|
def handle_url(self, url):
|
||||||
|
@ -6,6 +6,8 @@ from tqdm import tqdm
|
|||||||
|
|
||||||
import qobuz_dl.metadata as metadata
|
import qobuz_dl.metadata as metadata
|
||||||
|
|
||||||
|
QL_DOWNGRADE = "FormatRestrictedByFormatAvailability"
|
||||||
|
|
||||||
|
|
||||||
def tqdm_download(url, fname, track_name):
|
def tqdm_download(url, fname, track_name):
|
||||||
r = requests.get(url, allow_redirects=True, stream=True)
|
r = requests.get(url, allow_redirects=True, stream=True)
|
||||||
@ -31,23 +33,34 @@ def get_description(u, mt, multiple=None):
|
|||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
def get_format(client, item_dict, quality, is_track_id=False):
|
def get_format(client, item_dict, quality, is_track_id=False, track_url_dict=None):
|
||||||
|
quality_met = True
|
||||||
if int(quality) == 5:
|
if int(quality) == 5:
|
||||||
return "MP3"
|
return "MP3", quality_met
|
||||||
track_dict = item_dict
|
track_dict = item_dict
|
||||||
if not is_track_id:
|
if not is_track_id:
|
||||||
track_dict = item_dict["tracks"]["items"][0]
|
track_dict = item_dict["tracks"]["items"][0]
|
||||||
|
|
||||||
try:
|
try:
|
||||||
new_track_dict = client.get_track_url(track_dict["id"], quality)
|
new_track_dict = (
|
||||||
|
client.get_track_url(track_dict["id"], quality)
|
||||||
|
if not track_url_dict
|
||||||
|
else track_url_dict
|
||||||
|
)
|
||||||
|
restrictions = new_track_dict.get("restrictions")
|
||||||
|
if isinstance(restrictions, list):
|
||||||
|
if any(
|
||||||
|
restriction.get("code") == QL_DOWNGRADE for restriction in restrictions
|
||||||
|
):
|
||||||
|
quality_met = False
|
||||||
if (
|
if (
|
||||||
new_track_dict["bit_depth"] == 16
|
new_track_dict["bit_depth"] == 16
|
||||||
and new_track_dict["sampling_rate"] == 44.1
|
and new_track_dict["sampling_rate"] == 44.1
|
||||||
):
|
):
|
||||||
return "FLAC"
|
return "FLAC", quality_met
|
||||||
return "Hi-Res"
|
return "Hi-Res", quality_met
|
||||||
except (KeyError, requests.exceptions.HTTPError):
|
except (KeyError, requests.exceptions.HTTPError):
|
||||||
return "Unknown"
|
return "Unknown", quality_met
|
||||||
|
|
||||||
|
|
||||||
def get_title(item_dict):
|
def get_title(item_dict):
|
||||||
@ -151,7 +164,14 @@ def download_and_tag(
|
|||||||
|
|
||||||
|
|
||||||
def download_id_by_type(
|
def download_id_by_type(
|
||||||
client, item_id, path, quality, album=False, embed_art=False, albums_only=False
|
client,
|
||||||
|
item_id,
|
||||||
|
path,
|
||||||
|
quality,
|
||||||
|
album=False,
|
||||||
|
embed_art=False,
|
||||||
|
albums_only=False,
|
||||||
|
downgrade_quality=True,
|
||||||
):
|
):
|
||||||
"""
|
"""
|
||||||
Download and get metadata by ID and type (album or track)
|
Download and get metadata by ID and type (album or track)
|
||||||
@ -163,6 +183,7 @@ def download_id_by_type(
|
|||||||
:param bool album: album type or not
|
:param bool album: album type or not
|
||||||
:param embed_art album: Embed cover art into files
|
:param embed_art album: Embed cover art into files
|
||||||
:param bool albums_only: Ignore Singles, EPs and VA releases
|
:param bool albums_only: Ignore Singles, EPs and VA releases
|
||||||
|
:param bool downgrade: Skip releases not available in set quality
|
||||||
"""
|
"""
|
||||||
count = 0
|
count = 0
|
||||||
|
|
||||||
@ -177,12 +198,17 @@ def download_id_by_type(
|
|||||||
return
|
return
|
||||||
|
|
||||||
album_title = get_title(meta)
|
album_title = get_title(meta)
|
||||||
|
album_format, quality_met = get_format(client, meta, quality)
|
||||||
|
if not downgrade_quality and not quality_met:
|
||||||
|
print("Skipping release as doesn't met quality requirement")
|
||||||
|
return
|
||||||
|
|
||||||
print("\nDownloading: {}\n".format(album_title))
|
print("\nDownloading: {}\n".format(album_title))
|
||||||
dirT = (
|
dirT = (
|
||||||
meta["artist"]["name"],
|
meta["artist"]["name"],
|
||||||
album_title,
|
album_title,
|
||||||
meta["release_date_original"].split("-")[0],
|
meta["release_date_original"].split("-")[0],
|
||||||
get_format(client, meta, quality),
|
album_format,
|
||||||
)
|
)
|
||||||
sanitized_title = sanitize_filename("{} - {} [{}] [{}]".format(*dirT))
|
sanitized_title = sanitize_filename("{} - {} [{}] [{}]".format(*dirT))
|
||||||
dirn = os.path.join(path, sanitized_title)
|
dirn = os.path.join(path, sanitized_title)
|
||||||
@ -191,8 +217,8 @@ def download_id_by_type(
|
|||||||
if "goodies" in meta:
|
if "goodies" in meta:
|
||||||
try:
|
try:
|
||||||
get_extra(meta["goodies"][0]["url"], dirn, "booklet.pdf")
|
get_extra(meta["goodies"][0]["url"], dirn, "booklet.pdf")
|
||||||
except Exception as e:
|
except: # noqa
|
||||||
print("Error: " + e)
|
pass
|
||||||
media_numbers = [track["media_number"] for track in meta["tracks"]["items"]]
|
media_numbers = [track["media_number"] for track in meta["tracks"]["items"]]
|
||||||
is_multiple = True if len([*{*media_numbers}]) > 1 else False
|
is_multiple = True if len([*{*media_numbers}]) > 1 else False
|
||||||
for i in meta["tracks"]["items"]:
|
for i in meta["tracks"]["items"]:
|
||||||
@ -228,11 +254,15 @@ def download_id_by_type(
|
|||||||
meta = client.get_track_meta(item_id)
|
meta = client.get_track_meta(item_id)
|
||||||
track_title = get_title(meta)
|
track_title = get_title(meta)
|
||||||
print("\nDownloading: {}\n".format(track_title))
|
print("\nDownloading: {}\n".format(track_title))
|
||||||
|
track_format, quality_met = get_format(client, meta, quality, True, parse)
|
||||||
|
if not downgrade_quality and not quality_met:
|
||||||
|
print("Skipping track as doesn't met quality requirement")
|
||||||
|
return
|
||||||
dirT = (
|
dirT = (
|
||||||
meta["album"]["artist"]["name"],
|
meta["album"]["artist"]["name"],
|
||||||
track_title,
|
track_title,
|
||||||
meta["album"]["release_date_original"].split("-")[0],
|
meta["album"]["release_date_original"].split("-")[0],
|
||||||
get_format(client, meta, quality, True),
|
track_format,
|
||||||
)
|
)
|
||||||
sanitized_title = sanitize_filename("{} - {} [{}] [{}]".format(*dirT))
|
sanitized_title = sanitize_filename("{} - {} [{}] [{}]".format(*dirT))
|
||||||
dirn = os.path.join(path, sanitized_title)
|
dirn = os.path.join(path, sanitized_title)
|
||||||
|
2
setup.py
2
setup.py
@ -13,7 +13,7 @@ requirements = read_file("requirements.txt").strip().split()
|
|||||||
|
|
||||||
setup(
|
setup(
|
||||||
name=pkg_name,
|
name=pkg_name,
|
||||||
version="0.5.6",
|
version="0.6.0",
|
||||||
author="Vitiko",
|
author="Vitiko",
|
||||||
author_email="vhnz98@gmail.com",
|
author_email="vhnz98@gmail.com",
|
||||||
description="The complete Lossless and Hi-Res music downloader for Qobuz",
|
description="The complete Lossless and Hi-Res music downloader for Qobuz",
|
||||||
|
Loading…
Reference in New Issue
Block a user