mirror of
https://github.com/KevinMidboe/spotify-downloader.git
synced 2025-12-08 20:39:08 +00:00
Add docstrings; Remove verbose comments; Fix errors introduced with cleanup
This comment will: - Transform docstrings above functions into docstrings - Remove some way too verbose comments - Apply some more recommendations from PEP8 forgotten last time - Fix some errors introduced with the first code cleanup Work left to do: - Add params to docstrings - Rename file variables
This commit is contained in:
@@ -4,11 +4,13 @@ import sys
|
|||||||
|
|
||||||
|
|
||||||
def song(input_song, output_song, avconv=False, verbose=False):
|
def song(input_song, output_song, avconv=False, verbose=False):
|
||||||
|
"""Do the audio format conversion."""
|
||||||
if not input_song == output_song:
|
if not input_song == output_song:
|
||||||
if sys.version_info < (3, 0):
|
if sys.version_info < (3, 0):
|
||||||
input_song = input_song.encode('utf-8')
|
input_song = input_song.encode('utf-8')
|
||||||
output_song = output_song.encode('utf-8')
|
output_song = output_song.encode('utf-8')
|
||||||
print('Converting ' + input_song + ' to ' + output_song.split('.')[-1])
|
print('Converting {0} to {1}'.format(
|
||||||
|
input_song, output_song.split('.')[-1]))
|
||||||
if avconv:
|
if avconv:
|
||||||
exit_code = convert_with_avconv(input_song, output_song, verbose)
|
exit_code = convert_with_avconv(input_song, output_song, verbose)
|
||||||
else:
|
else:
|
||||||
@@ -18,7 +20,7 @@ def song(input_song, output_song, avconv=False, verbose=False):
|
|||||||
|
|
||||||
|
|
||||||
def convert_with_avconv(input_song, output_song, verbose):
|
def convert_with_avconv(input_song, output_song, verbose):
|
||||||
# different path for windows
|
"""Convert the audio file using avconv."""
|
||||||
if os.name == 'nt':
|
if os.name == 'nt':
|
||||||
avconv_path = 'Scripts\\avconv.exe'
|
avconv_path = 'Scripts\\avconv.exe'
|
||||||
else:
|
else:
|
||||||
@@ -39,13 +41,16 @@ def convert_with_avconv(input_song, output_song, verbose):
|
|||||||
|
|
||||||
|
|
||||||
def convert_with_ffmpeg(input_song, output_song, verbose):
|
def convert_with_ffmpeg(input_song, output_song, verbose):
|
||||||
# What are the differences and similarities between ffmpeg, libav, and avconv?
|
"""Convert the audio file using FFMpeg.
|
||||||
# https://stackoverflow.com/questions/9477115
|
|
||||||
# ffmeg encoders high to lower quality
|
What are the differences and similarities between ffmpeg, libav, and avconv?
|
||||||
# libopus > libvorbis >= libfdk_aac > aac > libmp3lame
|
https://stackoverflow.com/questions/9477115
|
||||||
# libfdk_aac due to copyrights needs to be compiled by end user
|
ffmeg encoders high to lower quality
|
||||||
# on MacOS brew install ffmpeg --with-fdk-aac will do just that. Other OS?
|
libopus > libvorbis >= libfdk_aac > aac > libmp3lame
|
||||||
# https://trac.ffmpeg.org/wiki/Encode/AAC
|
libfdk_aac due to copyrights needs to be compiled by end user
|
||||||
|
on MacOS brew install ffmpeg --with-fdk-aac will do just that. Other OS?
|
||||||
|
https://trac.ffmpeg.org/wiki/Encode/AAC
|
||||||
|
"""
|
||||||
|
|
||||||
if os.name == "nt":
|
if os.name == "nt":
|
||||||
ffmpeg_pre = 'Scripts\\ffmpeg.exe '
|
ffmpeg_pre = 'Scripts\\ffmpeg.exe '
|
||||||
|
|||||||
@@ -10,8 +10,8 @@ except ImportError:
|
|||||||
import urllib.request as urllib2
|
import urllib.request as urllib2
|
||||||
|
|
||||||
|
|
||||||
# check if input file title matches with expected title
|
|
||||||
def compare(file, metadata):
|
def compare(file, metadata):
|
||||||
|
"""Check if the input file title matches the expected title."""
|
||||||
already_tagged = False
|
already_tagged = False
|
||||||
try:
|
try:
|
||||||
if file.endswith('.mp3'):
|
if file.endswith('.mp3'):
|
||||||
@@ -29,6 +29,7 @@ def compare(file, metadata):
|
|||||||
|
|
||||||
|
|
||||||
def embed(music_file, meta_tags):
|
def embed(music_file, meta_tags):
|
||||||
|
"""Embed metadata."""
|
||||||
if sys.version_info < (3, 0):
|
if sys.version_info < (3, 0):
|
||||||
music_file = music_file.encode('utf-8')
|
music_file = music_file.encode('utf-8')
|
||||||
if meta_tags is None:
|
if meta_tags is None:
|
||||||
@@ -46,6 +47,7 @@ def embed(music_file, meta_tags):
|
|||||||
|
|
||||||
|
|
||||||
def embed_mp3(music_file, meta_tags):
|
def embed_mp3(music_file, meta_tags):
|
||||||
|
"""Embed metadata to MP3 files."""
|
||||||
# EasyID3 is fun to use ;)
|
# EasyID3 is fun to use ;)
|
||||||
audiofile = EasyID3('Music/' + music_file)
|
audiofile = EasyID3('Music/' + music_file)
|
||||||
audiofile['artist'] = meta_tags['artists'][0]['name']
|
audiofile['artist'] = meta_tags['artists'][0]['name']
|
||||||
@@ -81,6 +83,7 @@ def embed_mp3(music_file, meta_tags):
|
|||||||
|
|
||||||
|
|
||||||
def embed_m4a(music_file, meta_tags):
|
def embed_m4a(music_file, meta_tags):
|
||||||
|
"""Embed metadata to M4A files."""
|
||||||
# Apple has specific tags - see mutagen docs -
|
# Apple has specific tags - see mutagen docs -
|
||||||
# http://mutagen.readthedocs.io/en/latest/api/mp4.html
|
# http://mutagen.readthedocs.io/en/latest/api/mp4.html
|
||||||
tags = {'album': '\xa9alb',
|
tags = {'album': '\xa9alb',
|
||||||
|
|||||||
20
core/misc.py
20
core/misc.py
@@ -10,8 +10,8 @@ except ImportError:
|
|||||||
from urllib.request import quote
|
from urllib.request import quote
|
||||||
|
|
||||||
|
|
||||||
# method to input (user playlists) and (track when using manual mode)
|
|
||||||
def input_link(links):
|
def input_link(links):
|
||||||
|
"""Let the user input a number."""
|
||||||
while True:
|
while True:
|
||||||
try:
|
try:
|
||||||
the_chosen_one = int(user_input('>> Choose your number: '))
|
the_chosen_one = int(user_input('>> Choose your number: '))
|
||||||
@@ -25,16 +25,16 @@ def input_link(links):
|
|||||||
print('Choose a valid number!')
|
print('Choose a valid number!')
|
||||||
|
|
||||||
|
|
||||||
# take input correctly for both python2 & 3
|
|
||||||
def user_input(string=''):
|
def user_input(string=''):
|
||||||
|
"""Take input correctly for both Python 2 & 3."""
|
||||||
if sys.version_info > (3, 0):
|
if sys.version_info > (3, 0):
|
||||||
return input(string)
|
return input(string)
|
||||||
else:
|
else:
|
||||||
return raw_input(string)
|
return raw_input(string)
|
||||||
|
|
||||||
|
|
||||||
# remove first song from .txt
|
|
||||||
def trim_song(file):
|
def trim_song(file):
|
||||||
|
"""Remove the first song from file."""
|
||||||
with open(file, 'r') as file_in:
|
with open(file, 'r') as file_in:
|
||||||
data = file_in.read().splitlines(True)
|
data = file_in.read().splitlines(True)
|
||||||
with open(file, 'w') as file_out:
|
with open(file, 'w') as file_out:
|
||||||
@@ -77,27 +77,29 @@ def get_arguments():
|
|||||||
return parser.parse_args()
|
return parser.parse_args()
|
||||||
|
|
||||||
|
|
||||||
# check if input song is spotify link
|
|
||||||
def is_spotify(raw_song):
|
def is_spotify(raw_song):
|
||||||
if (len(raw_song) == 22 and raw_song.replace(" ", "%20") == raw_song) or (raw_song.find('spotify') > -1):
|
"""Check if the input song is a Spotify link."""
|
||||||
|
if (len(raw_song) == 22 and raw_song.replace(" ", "%20") == raw_song) or \
|
||||||
|
(raw_song.find('spotify') > -1):
|
||||||
return True
|
return True
|
||||||
else:
|
else:
|
||||||
return False
|
return False
|
||||||
|
|
||||||
|
|
||||||
# generate filename of the song to be downloaded
|
|
||||||
def generate_filename(title):
|
def generate_filename(title):
|
||||||
|
"""Generate filename of the song to be downloaded."""
|
||||||
# IMO python2 sucks dealing with unicode
|
# IMO python2 sucks dealing with unicode
|
||||||
title = fix_encoding(title)
|
title = fix_encoding(title)
|
||||||
title = fix_decoding(title)
|
title = fix_decoding(title)
|
||||||
title = title.replace(' ', '_')
|
title = title.replace(' ', '_')
|
||||||
|
|
||||||
# slugify removes any special characters
|
# slugify removes any special characters
|
||||||
filename = slugify(title, ok='-_()[]{}', lower=False)
|
filename = slugify(title, ok='-_()[]{}', lower=False)
|
||||||
return fix_encoding(filename)
|
return fix_encoding(filename)
|
||||||
|
|
||||||
|
|
||||||
# please respect these credentials :)
|
|
||||||
def generate_token():
|
def generate_token():
|
||||||
|
"""Generate the token. Please respect these credentials :)"""
|
||||||
credentials = oauth2.SpotifyClientCredentials(
|
credentials = oauth2.SpotifyClientCredentials(
|
||||||
client_id='4fe3fecfe5334023a1472516cc99d805',
|
client_id='4fe3fecfe5334023a1472516cc99d805',
|
||||||
client_secret='0f02b7c483c04257984695007a4a8d5c')
|
client_secret='0f02b7c483c04257984695007a4a8d5c')
|
||||||
@@ -106,20 +108,22 @@ def generate_token():
|
|||||||
|
|
||||||
|
|
||||||
def generate_search_url(song):
|
def generate_search_url(song):
|
||||||
|
"""Generate YouTube search URL for the given song."""
|
||||||
# urllib2.quote() encodes URL with special characters
|
# urllib2.quote() encodes URL with special characters
|
||||||
url = "https://www.youtube.com/results?sp=EgIQAQ%253D%253D&q={0}".format(
|
url = "https://www.youtube.com/results?sp=EgIQAQ%253D%253D&q={0}".format(
|
||||||
quote(song))
|
quote(song))
|
||||||
return url
|
return url
|
||||||
|
|
||||||
|
|
||||||
# fix encoding issues in python2
|
|
||||||
def fix_encoding(query):
|
def fix_encoding(query):
|
||||||
|
"""Fix encoding issues in Python 2."""
|
||||||
if sys.version_info < (3, 0):
|
if sys.version_info < (3, 0):
|
||||||
query = query.encode('utf-8')
|
query = query.encode('utf-8')
|
||||||
return query
|
return query
|
||||||
|
|
||||||
|
|
||||||
def fix_decoding(query):
|
def fix_decoding(query):
|
||||||
|
"""Fix decoding issues in Python 2."""
|
||||||
if sys.version_info < (3, 0):
|
if sys.version_info < (3, 0):
|
||||||
query = query.decode('utf-8')
|
query = query.decode('utf-8')
|
||||||
return query
|
return query
|
||||||
|
|||||||
38
spotdl.py
38
spotdl.py
@@ -19,16 +19,16 @@ except ImportError:
|
|||||||
import urllib.request as urllib2
|
import urllib.request as urllib2
|
||||||
|
|
||||||
|
|
||||||
# "[artist] - [song]"
|
|
||||||
def generate_songname(raw_song):
|
def generate_songname(raw_song):
|
||||||
|
"""Generate a string of the format '[artist] - [song]' for the given song."""
|
||||||
if misc.is_spotify(raw_song):
|
if misc.is_spotify(raw_song):
|
||||||
tags = generate_metadata(raw_song)
|
tags = generate_metadata(raw_song)
|
||||||
raw_song = '{0} - {1}'.format(tags['artists'][0]['name'], tags['name'])
|
raw_song = '{0} - {1}'.format(tags['artists'][0]['name'], tags['name'])
|
||||||
return misc.fix_encoding(raw_song)
|
return misc.fix_encoding(raw_song)
|
||||||
|
|
||||||
|
|
||||||
# fetch song's metadata from spotify
|
|
||||||
def generate_metadata(raw_song):
|
def generate_metadata(raw_song):
|
||||||
|
"""Fetch a song's metadata from Spotify."""
|
||||||
if misc.is_spotify(raw_song):
|
if misc.is_spotify(raw_song):
|
||||||
# fetch track information directly if it is spotify link
|
# fetch track information directly if it is spotify link
|
||||||
meta_tags = spotify.track(raw_song)
|
meta_tags = spotify.track(raw_song)
|
||||||
@@ -60,9 +60,8 @@ def generate_metadata(raw_song):
|
|||||||
|
|
||||||
|
|
||||||
def generate_youtube_url(raw_song):
|
def generate_youtube_url(raw_song):
|
||||||
# decode spotify http link to "[artist] - [song]"
|
"""Search for the song on YouTube and generate an URL to its video."""
|
||||||
song = generate_songname(raw_song)
|
song = generate_songname(raw_song)
|
||||||
# generate direct search YouTube URL
|
|
||||||
search_url = misc.generate_search_url(song)
|
search_url = misc.generate_search_url(song)
|
||||||
item = urllib2.urlopen(search_url).read()
|
item = urllib2.urlopen(search_url).read()
|
||||||
# item = unicode(item, 'utf-8')
|
# item = unicode(item, 'utf-8')
|
||||||
@@ -97,23 +96,21 @@ def generate_youtube_url(raw_song):
|
|||||||
attrs={'class': 'yt-uix-tile-link'})[check]['href']
|
attrs={'class': 'yt-uix-tile-link'})[check]['href']
|
||||||
check += 1
|
check += 1
|
||||||
|
|
||||||
full_link = "youtube.com{0}'.format(result)
|
full_link = 'youtube.com{0}'.format(result)
|
||||||
return full_link
|
return full_link
|
||||||
|
|
||||||
|
|
||||||
# parse track from YouTube
|
|
||||||
def go_pafy(raw_song):
|
def go_pafy(raw_song):
|
||||||
# video link of the video to extract audio from
|
"""Parse track from YouTube."""
|
||||||
track_url = generate_youtube_url(raw_song)
|
track_url = generate_youtube_url(raw_song)
|
||||||
if track_url is None:
|
if track_url is None:
|
||||||
return None
|
return None
|
||||||
else:
|
else:
|
||||||
# parse the YouTube video
|
|
||||||
return pafy.new(track_url)
|
return pafy.new(track_url)
|
||||||
|
|
||||||
|
|
||||||
# title of the YouTube video
|
|
||||||
def get_youtube_title(content, number=None):
|
def get_youtube_title(content, number=None):
|
||||||
|
"""Get the YouTube video's title."""
|
||||||
title = misc.fix_encoding(content.title)
|
title = misc.fix_encoding(content.title)
|
||||||
if number is None:
|
if number is None:
|
||||||
return title
|
return title
|
||||||
@@ -121,14 +118,12 @@ def get_youtube_title(content, number=None):
|
|||||||
return '{0}. {1}'.format(number, title)
|
return '{0}. {1}'.format(number, title)
|
||||||
|
|
||||||
|
|
||||||
# fetch user playlists when using -u option
|
|
||||||
def feed_playlist(username):
|
def feed_playlist(username):
|
||||||
# fetch all user playlists
|
"""Fetch user playlists when using the -u option."""
|
||||||
playlists = spotify.user_playlists(username)
|
playlists = spotify.user_playlists(username)
|
||||||
links = []
|
links = []
|
||||||
check = 1
|
check = 1
|
||||||
|
|
||||||
# iterate over user playlists
|
|
||||||
while True:
|
while True:
|
||||||
for playlist in playlists['items']:
|
for playlist in playlists['items']:
|
||||||
# in rare cases, playlists may not be found, so playlists['next']
|
# in rare cases, playlists may not be found, so playlists['next']
|
||||||
@@ -145,13 +140,10 @@ def feed_playlist(username):
|
|||||||
break
|
break
|
||||||
|
|
||||||
print('')
|
print('')
|
||||||
# let user select playlist
|
|
||||||
playlist = misc.input_link(links)
|
playlist = misc.input_link(links)
|
||||||
# fetch detailed information for playlist
|
|
||||||
results = spotify.user_playlist(
|
results = spotify.user_playlist(
|
||||||
playlist['owner']['id'], playlist['id'], fields='tracks,next')
|
playlist['owner']['id'], playlist['id'], fields='tracks,next')
|
||||||
print('')
|
print('')
|
||||||
# slugify removes any special characters
|
|
||||||
file = '{0}.txt'.format(slugify(playlist['name'], ok='-_()[]{}'))
|
file = '{0}.txt'.format(slugify(playlist['name'], ok='-_()[]{}'))
|
||||||
print('Feeding {0} tracks to {1}'.format(playlist['tracks']['total'], file))
|
print('Feeding {0} tracks to {1}'.format(playlist['tracks']['total'], file))
|
||||||
|
|
||||||
@@ -174,11 +166,10 @@ def feed_playlist(username):
|
|||||||
|
|
||||||
|
|
||||||
def download_song(content):
|
def download_song(content):
|
||||||
|
"""Download the audio file from YouTube."""
|
||||||
if args.input_ext == '.webm':
|
if args.input_ext == '.webm':
|
||||||
# best available audio in .webm
|
|
||||||
link = content.getbestaudio(preftype='webm')
|
link = content.getbestaudio(preftype='webm')
|
||||||
elif args.input_ext == '.m4a':
|
elif args.input_ext == '.m4a':
|
||||||
# best available audio in .webm
|
|
||||||
link = content.getbestaudio(preftype='m4a')
|
link = content.getbestaudio(preftype='m4a')
|
||||||
else:
|
else:
|
||||||
return False
|
return False
|
||||||
@@ -187,14 +178,13 @@ def download_song(content):
|
|||||||
return False
|
return False
|
||||||
else:
|
else:
|
||||||
music_file = misc.generate_filename(content.title)
|
music_file = misc.generate_filename(content.title)
|
||||||
# download link
|
|
||||||
link.download(
|
link.download(
|
||||||
filepath='Music/{0}{1}'.format(music_file, args.input_ext))
|
filepath='Music/{0}{1}'.format(music_file, args.input_ext))
|
||||||
return True
|
return True
|
||||||
|
|
||||||
|
|
||||||
# check if input song already exists in Music folder
|
|
||||||
def check_exists(music_file, raw_song, islist=True):
|
def check_exists(music_file, raw_song, islist=True):
|
||||||
|
"""Check if the input song already exists in the 'Music' folder."""
|
||||||
files = os.listdir('Music')
|
files = os.listdir('Music')
|
||||||
for file in files:
|
for file in files:
|
||||||
if file.endswith('.temp'):
|
if file.endswith('.temp'):
|
||||||
@@ -229,8 +219,8 @@ def check_exists(music_file, raw_song, islist=True):
|
|||||||
return False
|
return False
|
||||||
|
|
||||||
|
|
||||||
# download songs from list
|
|
||||||
def grab_list(file):
|
def grab_list(file):
|
||||||
|
"""Download all songs from the list."""
|
||||||
with open(file, 'r') as listed:
|
with open(file, 'r') as listed:
|
||||||
lines = (listed.read()).splitlines()
|
lines = (listed.read()).splitlines()
|
||||||
# ignore blank lines in file (if any)
|
# ignore blank lines in file (if any)
|
||||||
@@ -270,9 +260,8 @@ def grab_list(file):
|
|||||||
number += 1
|
number += 1
|
||||||
|
|
||||||
|
|
||||||
# logic behind downloading some song
|
|
||||||
def grab_single(raw_song, number=None):
|
def grab_single(raw_song, number=None):
|
||||||
# check if song is being downloaded from list
|
"""Logic behind downloading a song."""
|
||||||
if number:
|
if number:
|
||||||
islist = True
|
islist = True
|
||||||
else:
|
else:
|
||||||
@@ -283,6 +272,7 @@ def grab_single(raw_song, number=None):
|
|||||||
# print '[number]. [artist] - [song]' if downloading from list
|
# print '[number]. [artist] - [song]' if downloading from list
|
||||||
# otherwise print '[artist] - [song]'
|
# otherwise print '[artist] - [song]'
|
||||||
print(get_youtube_title(content, number))
|
print(get_youtube_title(content, number))
|
||||||
|
|
||||||
# generate file name of the song to download
|
# generate file name of the song to download
|
||||||
music_file = misc.generate_filename(content.title)
|
music_file = misc.generate_filename(content.title)
|
||||||
music_file = misc.fix_decoding(music_file)
|
music_file = misc.fix_decoding(music_file)
|
||||||
@@ -291,9 +281,7 @@ def grab_single(raw_song, number=None):
|
|||||||
print('')
|
print('')
|
||||||
input_song = music_file + args.input_ext
|
input_song = music_file + args.input_ext
|
||||||
output_song = music_file + args.output_ext
|
output_song = music_file + args.output_ext
|
||||||
convert.song(input_song,
|
convert.song(input_song, output_song, avconv=args.avconv,
|
||||||
output_song,
|
|
||||||
avconv=args.avconv,
|
|
||||||
verbose=args.verbose)
|
verbose=args.verbose)
|
||||||
os.remove('Music/{0}'.format(file))
|
os.remove('Music/{0}'.format(file))
|
||||||
meta_tags = generate_metadata(raw_song)
|
meta_tags = generate_metadata(raw_song)
|
||||||
|
|||||||
@@ -22,7 +22,8 @@ def test_playlist():
|
|||||||
def test_tracks():
|
def test_tracks():
|
||||||
playlist = spotdl.spotify.user_playlists(username)['items'][0]
|
playlist = spotdl.spotify.user_playlists(username)['items'][0]
|
||||||
expect_lines = playlist['tracks']['total']
|
expect_lines = playlist['tracks']['total']
|
||||||
result = spotdl.spotify.user_playlist(playlist['owner']['id'], playlist['id'], fields='tracks,next')
|
result = spotdl.spotify.user_playlist(
|
||||||
|
playlist['owner']['id'], playlist['id'], fields='tracks,next')
|
||||||
tracks = result['tracks']
|
tracks = result['tracks']
|
||||||
|
|
||||||
with open('list.txt', 'a') as fout:
|
with open('list.txt', 'a') as fout:
|
||||||
|
|||||||
Reference in New Issue
Block a user