forked from iloveicedgreentea/GPM-to-Spotify
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmain.py
executable file
·160 lines (131 loc) · 6.16 KB
/
main.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
#!/usr/bin/env python3
from gmusicapi import Mobileclient
from spotipy import Spotify, util, client
from dotenv import load_dotenv
from os import getenv
import logging
import json
from sys import exit
# env
def load_env():
load_dotenv()
logging.basicConfig(level=logging.INFO, format="%(levelname)s:%(lineno)d -- %(message)s")
#################
# spotify
#################
class Spotify_client:
def __init__(self, *args, **kwargs):
client_id = getenv('SPOTIFY_client_id')
client_secret = getenv('SPOTIFY_client_secret')
self.username = getenv('SPOTIFY_username')
redirect_uri = getenv('SPOTIFY_redirect_uri')
scope = 'playlist-modify-private'
# login
logging.info("Starting Spotify client")
token = util.prompt_for_user_token(self.username,scope,client_id=client_id,client_secret=client_secret,redirect_uri=redirect_uri)
self.sp_client = Spotify(auth=token)
def create_playlist(self, playlist_name):
logging.debug(f'playlist_name: {playlist_name}')
return self.sp_client.user_playlist_create(user=self.username, name=playlist_name, public=False)
def search_track(self, search_string):
logging.debug(f'search_string: {search_string}')
return self.sp_client.search(search_string, limit=1)
def add_to_playlist(self, playlist_id, track_ids):
logging.debug(f'playlist_id: {playlist_id}, track_ids: {track_ids}')
return self.sp_client.user_playlist_add_tracks(self.username, playlist_id, track_ids)
#################
# GPM
#################
class GPM_client:
def __init__(self):
logging.info("Starting GPM client")
self.gpm_client = Mobileclient()
# login
while not self.gpm_client.is_authenticated():
logging.info("Logging you in...")
if not self.gpm_client.oauth_login(device_id=Mobileclient.FROM_MAC_ADDRESS, oauth_credentials='./.gpmtoken'):
logging.debug("No previous credentials - performing Oauth")
self.gpm_client.perform_oauth(open_browser=True, storage_filepath='./.gpmtoken')
#################
# Common
#################
def main():
'''
Playlists data structure is a list of dicts - each playlist, "name" field for playlist name and "tracks" list for all tracks
Tracks is a list, each with a track ID
We can grab the name field to know what playlists to create and the track ID to convert/search
'''
# load env
logging.debug('Loading env')
load_env()
# initialize GPM
logging.debug('loading GPM')
gpm = GPM_client()
# initialize spotify
logging.debug('loading Spotify')
spot = Spotify_client()
# Get a full dump of all songs in library
library = gpm.gpm_client.get_all_songs()
# Get a full dump of all playlists as a massive list of dicts
logging.info('Getting all GPM playlists')
full_playlist_list = gpm.gpm_client.get_all_user_playlist_contents()
for playlist in full_playlist_list:
# create the new playlists GPM->Spotify
logging.info('Making new playlists...')
name = playlist.get('name')
new_playlist_id = spot.create_playlist(name).get('id')
logging.info(f'Playlist created: {name} -- ID: {new_playlist_id}')
# search and add the track into new album
logging.info(f'Adding tracks to {name}')
for track in playlist.get('tracks'):
artist = None
title = None
try:
track_data = track.get('track')
if not track_data:
track_data = [item for item in library if item['id'] == track['trackId']][0]
artist = track_data.get('artist')
title = track_data.get('title')
logging.info(f'Searching {artist} {title}')
# search by artist and title
search_result = spot.search_track(f'{artist} {title}')
logging.debug(search_result)
#todo: This search relies on the naming structure being similar. Can be improved with regex
track_uri = search_result.get('tracks').get('items')[0].get('uri')
logging.debug(track_uri)
# add to new playlist
logging.info(f'Adding {track_uri} to {name}')
#todo: speed could be improved with async
playlist_add = spot.add_to_playlist(new_playlist_id, [track_uri])
logging.info('Song added')
logging.debug(playlist_add)
# thrown when 'tracks' is missing
except AttributeError:
logging.info("This track does not have metadata - probably uploaded. Skipping")
try:
with open('./errored-tracks.log', 'a') as file:
file.writelines(f'playlist: {name}\n\n')
except:
logging.info("Error writing playlist to errored list. It most likely has special characters.")
pass
# cheap way to fix this, almost certainly means the track doesn't exist on spotify
except IndexError:
logging.info("Index out of range: This track may not exist or was not found. Skipping")
try:
with open('./errored-tracks.log', 'a') as file:
file.writelines(f'playlist: {name}\nartist: {artist}\ntitle: {title}\n\n')
except:
logging.info("Error writing track to errored list. It most likely has special characters.")
pass
# cheap fix for random 500 errors
except client.SpotifyException:
logging.info("500 error - failed track written to log. It might have been added anyway, check later")
try:
with open('./errored-tracks.log', 'a') as file:
file.writelines(f'playlist: {name}\nartist: {artist}\ntitle: {title}\n\n')
except:
logging.info("Error writing track to errored list. It most likely has special characters.")
pass
if __name__ == "__main__":
main()
print("Finished\nPlease check errored-tracks.log for missing tracks")