mirror of
https://gitlab.com/RemixDev/deemix-py.git
synced 2025-01-09 08:25:15 +00:00
f530a4e89f
Removed saveDownloadQueue and tagsLanguage from lib settings Revert embedded cover change Fixed bitrate fallback check Use overwriteFile setting when downloading embedded covers Fixed bitrate fallback not working Fixed some issues to make the lib work Implemented spotify plugin back Better handling of albums upcs Fixed queue item not cancelling correctly Code parity with deemix-js Code cleanup with pylint Even more rework on the library More work on the library (WIP) Total rework of the library (WIP) Some rework done on types Added start queue function Made nextitem work on a thread Removed dz as first parameter Started queuemanager refactoring Removed eventlet Co-authored-by: RemixDev <RemixDev64@gmail.com> Reviewed-on: https://git.freezer.life/RemixDev/deemix-py/pulls/4 Co-Authored-By: RemixDev <remixdev@noreply.localhost> Co-Committed-By: RemixDev <remixdev@noreply.localhost>
244 lines
11 KiB
Python
244 lines
11 KiB
Python
import re
|
|
from os.path import sep as pathSep
|
|
from pathlib import Path
|
|
from unicodedata import normalize
|
|
from deezer import TrackFormats
|
|
|
|
bitrateLabels = {
|
|
TrackFormats.MP4_RA3: "360 HQ",
|
|
TrackFormats.MP4_RA2: "360 MQ",
|
|
TrackFormats.MP4_RA1: "360 LQ",
|
|
TrackFormats.FLAC : "FLAC",
|
|
TrackFormats.MP3_320: "320",
|
|
TrackFormats.MP3_128: "128",
|
|
TrackFormats.DEFAULT: "128",
|
|
TrackFormats.LOCAL : "MP3"
|
|
}
|
|
|
|
def fixName(txt, char='_'):
|
|
txt = str(txt)
|
|
txt = re.sub(r'[\0\/\\:*?"<>|]', char, txt)
|
|
txt = normalize("NFC", txt)
|
|
return txt
|
|
|
|
def fixLongName(name):
|
|
def fixEndOfData(bString):
|
|
try:
|
|
bString.decode()
|
|
return True
|
|
except Exception:
|
|
return False
|
|
if pathSep in name:
|
|
sepName = name.split(pathSep)
|
|
name = ""
|
|
for txt in sepName:
|
|
txt = fixLongName(txt)
|
|
name += txt + pathSep
|
|
name = name[:-1]
|
|
else:
|
|
name = name.encode('utf-8')[:200]
|
|
while not fixEndOfData(name):
|
|
name = name[:-1]
|
|
name = name.decode()
|
|
return name
|
|
|
|
|
|
def antiDot(string):
|
|
while string[-1:] == "." or string[-1:] == " " or string[-1:] == "\n":
|
|
string = string[:-1]
|
|
if len(string) < 1:
|
|
string = "dot"
|
|
return string
|
|
|
|
|
|
def pad(num, max_val, settings):
|
|
if int(settings['paddingSize']) == 0:
|
|
paddingSize = len(str(max_val))
|
|
else:
|
|
paddingSize = len(str(10 ** (int(settings['paddingSize']) - 1)))
|
|
if paddingSize == 1:
|
|
paddingSize = 2
|
|
if settings['padTracks']:
|
|
return str(num).zfill(paddingSize)
|
|
return str(num)
|
|
|
|
def generatePath(track, downloadObject, settings):
|
|
filenameTemplate = "%artist% - %title%"
|
|
singleTrack = False
|
|
if downloadObject.type == "track":
|
|
if settings['createSingleFolder']:
|
|
filenameTemplate = settings['albumTracknameTemplate']
|
|
else:
|
|
filenameTemplate = settings['tracknameTemplate']
|
|
singleTrack = True
|
|
elif downloadObject.type == "album":
|
|
filenameTemplate = settings['albumTracknameTemplate']
|
|
else:
|
|
filenameTemplate = settings['playlistTracknameTemplate']
|
|
|
|
filename = generateTrackName(filenameTemplate, track, settings)
|
|
|
|
filepath = Path(settings['downloadLocation'] or '.')
|
|
artistPath = None
|
|
coverPath = None
|
|
extrasPath = None
|
|
|
|
if settings['createPlaylistFolder'] and track.playlist and not settings['tags']['savePlaylistAsCompilation']:
|
|
filepath = filepath / generatePlaylistName(settings['playlistNameTemplate'], track.playlist, settings)
|
|
|
|
if track.playlist and not settings['tags']['savePlaylistAsCompilation']:
|
|
extrasPath = filepath
|
|
|
|
if (
|
|
(settings['createArtistFolder'] and not track.playlist) or
|
|
(settings['createArtistFolder'] and track.playlist and settings['tags']['savePlaylistAsCompilation']) or
|
|
(settings['createArtistFolder'] and track.playlist and settings['createStructurePlaylist'])
|
|
):
|
|
filepath = filepath / generateArtistName(settings['artistNameTemplate'], track.album.mainArtist, settings, rootArtist=track.album.rootArtist)
|
|
artistPath = filepath
|
|
|
|
if (settings['createAlbumFolder'] and
|
|
(not singleTrack or (singleTrack and settings['createSingleFolder'])) and
|
|
(not track.playlist or
|
|
(track.playlist and settings['tags']['savePlaylistAsCompilation']) or
|
|
(track.playlist and settings['createStructurePlaylist'])
|
|
)
|
|
):
|
|
filepath = filepath / generateAlbumName(settings['albumNameTemplate'], track.album, settings, track.playlist)
|
|
coverPath = filepath
|
|
|
|
if not extrasPath: extrasPath = filepath
|
|
|
|
if (
|
|
int(track.album.discTotal) > 1 and (
|
|
(settings['createAlbumFolder'] and settings['createCDFolder']) and
|
|
(not singleTrack or (singleTrack and settings['createSingleFolder'])) and
|
|
(not track.playlist or
|
|
(track.playlist and settings['tags']['savePlaylistAsCompilation']) or
|
|
(track.playlist and settings['createStructurePlaylist'])
|
|
)
|
|
)):
|
|
filepath = filepath / f'CD{track.discNumber}'
|
|
|
|
# Remove subfolders from filename and add it to filepath
|
|
if pathSep in filename:
|
|
tempPath = filename[:filename.rfind(pathSep)]
|
|
filepath = filepath / tempPath
|
|
filename = filename[filename.rfind(pathSep) + len(pathSep):]
|
|
|
|
return (filename, filepath, artistPath, coverPath, extrasPath)
|
|
|
|
|
|
def generateTrackName(filename, track, settings):
|
|
c = settings['illegalCharacterReplacer']
|
|
filename = filename.replace("%title%", fixName(track.title, c))
|
|
filename = filename.replace("%artist%", fixName(track.mainArtist.name, c))
|
|
filename = filename.replace("%artists%", fixName(", ".join(track.artists), c))
|
|
filename = filename.replace("%allartists%", fixName(track.artistsString, c))
|
|
filename = filename.replace("%mainartists%", fixName(track.mainArtistsString, c))
|
|
if track.featArtistsString:
|
|
filename = filename.replace("%featartists%", fixName('('+track.featArtistsString+')', c))
|
|
else:
|
|
filename = filename.replace("%featartists%", '')
|
|
filename = filename.replace("%album%", fixName(track.album.title, c))
|
|
filename = filename.replace("%albumartist%", fixName(track.album.mainArtist.name, c))
|
|
filename = filename.replace("%tracknumber%", pad(track.trackNumber, track.album.trackTotal, settings))
|
|
filename = filename.replace("%tracktotal%", str(track.album.trackTotal))
|
|
filename = filename.replace("%discnumber%", str(track.discNumber))
|
|
filename = filename.replace("%disctotal%", str(track.album.discTotal))
|
|
if len(track.album.genre) > 0:
|
|
filename = filename.replace("%genre%", fixName(track.album.genre[0], c))
|
|
else:
|
|
filename = filename.replace("%genre%", "Unknown")
|
|
filename = filename.replace("%year%", str(track.date.year))
|
|
filename = filename.replace("%date%", track.dateString)
|
|
filename = filename.replace("%bpm%", str(track.bpm))
|
|
filename = filename.replace("%label%", fixName(track.album.label, c))
|
|
filename = filename.replace("%isrc%", track.ISRC)
|
|
filename = filename.replace("%upc%", track.album.barcode)
|
|
filename = filename.replace("%explicit%", "(Explicit)" if track.explicit else "")
|
|
|
|
filename = filename.replace("%track_id%", str(track.id))
|
|
filename = filename.replace("%album_id%", str(track.album.id))
|
|
filename = filename.replace("%artist_id%", str(track.mainArtist.id))
|
|
if track.playlist:
|
|
filename = filename.replace("%playlist_id%", str(track.playlist.playlistID))
|
|
filename = filename.replace("%position%", pad(track.position, track.playlist.trackTotal, settings))
|
|
else:
|
|
filename = filename.replace("%playlist_id%", '')
|
|
filename = filename.replace("%position%", pad(track.position, track.album.trackTotal, settings))
|
|
filename = filename.replace('\\', pathSep).replace('/', pathSep)
|
|
return antiDot(fixLongName(filename))
|
|
|
|
|
|
def generateAlbumName(foldername, album, settings, playlist=None):
|
|
c = settings['illegalCharacterReplacer']
|
|
if playlist and settings['tags']['savePlaylistAsCompilation']:
|
|
foldername = foldername.replace("%album_id%", "pl_" + str(playlist.playlistID))
|
|
foldername = foldername.replace("%genre%", "Compile")
|
|
else:
|
|
foldername = foldername.replace("%album_id%", str(album.id))
|
|
if len(album.genre) > 0:
|
|
foldername = foldername.replace("%genre%", fixName(album.genre[0], c))
|
|
else:
|
|
foldername = foldername.replace("%genre%", "Unknown")
|
|
foldername = foldername.replace("%album%", fixName(album.title, c))
|
|
foldername = foldername.replace("%artist%", fixName(album.mainArtist.name, c))
|
|
foldername = foldername.replace("%artist_id%", str(album.mainArtist.id))
|
|
if album.rootArtist:
|
|
foldername = foldername.replace("%root_artist%", fixName(album.rootArtist.name, c))
|
|
foldername = foldername.replace("%root_artist_id%", str(album.rootArtist.id))
|
|
else:
|
|
foldername = foldername.replace("%root_artist%", fixName(album.mainArtist.name, c))
|
|
foldername = foldername.replace("%root_artist_id%", str(album.mainArtist.id))
|
|
foldername = foldername.replace("%tracktotal%", str(album.trackTotal))
|
|
foldername = foldername.replace("%disctotal%", str(album.discTotal))
|
|
foldername = foldername.replace("%type%", fixName(album.recordType.capitalize(), c))
|
|
foldername = foldername.replace("%upc%", album.barcode)
|
|
foldername = foldername.replace("%explicit%", "(Explicit)" if album.explicit else "")
|
|
foldername = foldername.replace("%label%", fixName(album.label, c))
|
|
foldername = foldername.replace("%year%", str(album.date.year))
|
|
foldername = foldername.replace("%date%", album.dateString)
|
|
foldername = foldername.replace("%bitrate%", bitrateLabels[int(album.bitrate)])
|
|
|
|
foldername = foldername.replace('\\', pathSep).replace('/', pathSep)
|
|
return antiDot(fixLongName(foldername))
|
|
|
|
|
|
def generateArtistName(foldername, artist, settings, rootArtist=None):
|
|
c = settings['illegalCharacterReplacer']
|
|
foldername = foldername.replace("%artist%", fixName(artist.name, c))
|
|
foldername = foldername.replace("%artist_id%", str(artist.id))
|
|
if rootArtist:
|
|
foldername = foldername.replace("%root_artist%", fixName(rootArtist.name, c))
|
|
foldername = foldername.replace("%root_artist_id%", str(rootArtist.id))
|
|
else:
|
|
foldername = foldername.replace("%root_artist%", fixName(artist.name, c))
|
|
foldername = foldername.replace("%root_artist_id%", str(artist.id))
|
|
foldername = foldername.replace('\\', pathSep).replace('/', pathSep)
|
|
return antiDot(fixLongName(foldername))
|
|
|
|
|
|
def generatePlaylistName(foldername, playlist, settings):
|
|
c = settings['illegalCharacterReplacer']
|
|
foldername = foldername.replace("%playlist%", fixName(playlist.title, c))
|
|
foldername = foldername.replace("%playlist_id%", fixName(playlist.playlistID, c))
|
|
foldername = foldername.replace("%owner%", fixName(playlist.owner['name'], c))
|
|
foldername = foldername.replace("%owner_id%", str(playlist.owner['id']))
|
|
foldername = foldername.replace("%year%", str(playlist.date.year))
|
|
foldername = foldername.replace("%date%", str(playlist.dateString))
|
|
foldername = foldername.replace("%explicit%", "(Explicit)" if playlist.explicit else "")
|
|
foldername = foldername.replace('\\', pathSep).replace('/', pathSep)
|
|
return antiDot(fixLongName(foldername))
|
|
|
|
def generateDownloadObjectName(foldername, queueItem, settings):
|
|
c = settings['illegalCharacterReplacer']
|
|
foldername = foldername.replace("%title%", fixName(queueItem.title, c))
|
|
foldername = foldername.replace("%artist%", fixName(queueItem.artist, c))
|
|
foldername = foldername.replace("%size%", str(queueItem.size))
|
|
foldername = foldername.replace("%type%", fixName(queueItem.type, c))
|
|
foldername = foldername.replace("%id%", fixName(queueItem.id, c))
|
|
foldername = foldername.replace("%bitrate%", bitrateLabels[int(queueItem.bitrate)])
|
|
foldername = foldername.replace('\\', pathSep).replace('/', pathSep).replace(pathSep, c)
|
|
return antiDot(fixLongName(foldername))
|