import argparse import sys import os from slugify import slugify import spotipy.oauth2 as oauth2 try: from urllib2 import quote except ImportError: from urllib.request import quote def input_link(links): """Let the user input a number.""" while True: try: the_chosen_one = int(user_input('>> Choose your number: ')) if 1 <= the_chosen_one <= len(links): return links[the_chosen_one - 1] elif the_chosen_one == 0: return None else: print('Choose a valid number!') except ValueError: print('Choose a valid number!') def user_input(string=''): """Take input correctly for both Python 2 & 3.""" if sys.version_info > (3, 0): return input(string) else: return raw_input(string) def trim_song(file): """Remove the first song from file.""" with open(file, 'r') as file_in: data = file_in.read().splitlines(True) with open(file, 'w') as file_out: file_out.writelines(data[1:]) def get_arguments(): parser = argparse.ArgumentParser( description='Download and convert songs from Spotify, Youtube etc.', formatter_class=argparse.ArgumentDefaultsHelpFormatter) group = parser.add_mutually_exclusive_group(required=True) group.add_argument( '-s', '--song', help='download song by spotify link or name') group.add_argument( '-l', '--list', help='download songs from a file') group.add_argument( '-u', '--username', help="load user's playlists into .txt") parser.add_argument( '-m', '--manual', default=False, help='choose the song to download manually', action='store_true') parser.add_argument( '-nm', '--no-metadata', default=False, help='do not embed metadata in songs', action='store_true') parser.add_argument( '-a', '--avconv', default=False, help='Use avconv for conversion otherwise set defaults to ffmpeg', action='store_true') parser.add_argument( '-v', '--verbose', default=False, help='show debug output', action='store_true') parser.add_argument( '-i', '--input_ext', default='.m4a', help='prefered input format .m4a or .webm (Opus)') parser.add_argument( '-o', '--output_ext', default='.mp3', help='prefered output extension .mp3 or .m4a (AAC)') return parser.parse_args() def is_spotify(raw_song): """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 else: return False def generate_filename(title): """Generate filename of the song to be downloaded.""" # IMO python2 sucks dealing with unicode title = fix_encoding(title) title = fix_decoding(title) title = title.replace(' ', '_') # slugify removes any special characters filename = slugify(title, ok='-_()[]{}', lower=False) return fix_encoding(filename) def generate_token(): """Generate the token. Please respect these credentials :)""" credentials = oauth2.SpotifyClientCredentials( client_id='4fe3fecfe5334023a1472516cc99d805', client_secret='0f02b7c483c04257984695007a4a8d5c') token = credentials.get_access_token() return token def generate_search_url(song): """Generate YouTube search URL for the given song.""" # urllib2.quote() encodes URL with special characters url = u"https://www.youtube.com/results?sp=EgIQAQ%253D%253D&q={0}".format( quote(song)) return url def fix_encoding(query): """Fix encoding issues in Python 2.""" if sys.version_info < (3, 0): query = query.encode('utf-8') return query def fix_decoding(query): """Fix decoding issues in Python 2.""" if sys.version_info < (3, 0): query = query.decode('utf-8') return query def filter_path(path): os.chdir(sys.path[0]) if not os.path.exists(path): os.makedirs(path) for temp in os.listdir(path): if temp.endswith('.temp'): os.remove('{0}/{1}'.format(path, temp)) def grace_quit(): print('\n\nExiting.') sys.exit()