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 # method to input (user playlists) and (track when using manual mode) def input_link(links): 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!') # take input correctly for both python2 & 3 def user_input(string=''): if sys.version_info > (3, 0): return input(string) else: return raw_input(string) # remove first song from .txt def trim_song(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() # check if input song is spotify link def is_spotify(raw_song): if (len(raw_song) == 22 and raw_song.replace(" ", "%20") == raw_song) or (raw_song.find('spotify') > -1): return True else: return False # generate filename of the song to be downloaded def generate_filename(title): # 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) # please respect these credentials :) def generate_token(): credentials = oauth2.SpotifyClientCredentials( client_id='4fe3fecfe5334023a1472516cc99d805', client_secret='0f02b7c483c04257984695007a4a8d5c') token = credentials.get_access_token() return token def generate_search_url(song): # urllib2.quote() encodes URL with special characters url = "https://www.youtube.com/results?sp=EgIQAQ%253D%253D&q={0}".format( quote(song)) return url # fix encoding issues in python2 def fix_encoding(query): if sys.version_info < (3, 0): query = query.encode('utf-8') return query def fix_decoding(query): 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()