mirror of
				https://github.com/JonasunderscoreJones/turbo-octo-potato.git
				synced 2025-10-25 11:09:18 +02:00 
			
		
		
		
	
		
			
				
	
	
		
			155 lines
		
	
	
	
		
			6.3 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			155 lines
		
	
	
	
		
			6.3 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| import pylast, spotipy, sys, os, time
 | |
| from spotipy.oauth2 import SpotifyOAuth
 | |
| 
 | |
| # Define your Last.fm API credentials
 | |
| LASTFM_API_KEY = ""
 | |
| LASTFM_API_SECRET = ""
 | |
| LASTFM_USERNAME = "Jonas_Jones"
 | |
| LASTFM_PASSWORD_HASH = pylast.md5("")
 | |
| 
 | |
| # Define your Spotify API credentials
 | |
| SPOTIPY_CLIENT_ID = ""
 | |
| SPOTIPY_CLIENT_SECRET = ""
 | |
| SPOTIPY_REDIRECT_URI = ""
 | |
| 
 | |
| # Define your playlist IDs
 | |
| LIKEDSONGPLAYLIST_ID = ""
 | |
| 
 | |
| def progress_bar(current, total, last_time_stamp=time.time()):
 | |
|     this_timestamp = time.time()
 | |
|     width = os.get_terminal_size().columns
 | |
|     progress = round((current/total)*width)
 | |
|     total_num_len = len(str(total))
 | |
|     if width < 2* total_num_len + 15:
 | |
|         return f"{current}/{total}", this_timestamp
 | |
|     else:
 | |
|         current_spacer = " "*(total_num_len-len(str(current)))
 | |
|         eta = round((total - current)* (this_timestamp - last_time_stamp)/60)
 | |
|         percent = str(round(current/total*100))
 | |
|         percent = " "*(3-len(percent)) + percent
 | |
|         progress_bar_length = width - 2*total_num_len - 14 - len(str(eta)) - len(percent)
 | |
|         progress_bar_progress = round((current/total)*progress_bar_length)
 | |
|         progress_bar_spacer = " "*(progress_bar_length-progress_bar_progress)
 | |
|         return f"[{current_spacer}{current}/{total}|{percent}%|ETA: {eta}s|{'='*progress_bar_progress}>{progress_bar_spacer}]", this_timestamp
 | |
| 
 | |
| def verboseprint(message, end="\n"):
 | |
|     if VERBOSE_LOGGING:
 | |
|         print(message, end=end)
 | |
| 
 | |
| def handle_playlist_part_return(playlist_part, all_songs):
 | |
|     for item in playlist_part["items"]:
 | |
|         track_uri = item["track"]["uri"]
 | |
|         track_name = item["track"]["name"]
 | |
|         artist_name = item["track"]["artists"][0]["name"]
 | |
|         all_songs.append((track_uri, track_name, artist_name))
 | |
|         
 | |
|     return all_songs
 | |
| 
 | |
| def get_all_songs_from_playlist(playlist_id):
 | |
|     verboseprint("Fetching songs from the liked songs playlist...")
 | |
|     all_songs = []
 | |
|     limit = 100  # Adjust the limit based on your needs
 | |
|     offset = 0
 | |
|     last_time_stamp = time.time()
 | |
| 
 | |
|     while True:
 | |
|         playlist_part = sp.playlist_items(playlist_id, limit=limit, offset=offset)
 | |
|         if not playlist_part["items"]:
 | |
|             break
 | |
|         all_songs = handle_playlist_part_return(playlist_part, all_songs)
 | |
|         
 | |
|         progress_print, last_time_stamp = progress_bar(offset+limit, playlist_part["total"], last_time_stamp)
 | |
|         verboseprint(progress_print, end="\r")
 | |
| 
 | |
|         offset += limit
 | |
|     verboseprint("")
 | |
|     return all_songs
 | |
| 
 | |
| def get_all_liked_songs():
 | |
|     verboseprint("Fetching liked songs...")
 | |
|     verboseprint("This may take a while... (the API only allows for 50 songs per request for the liked songs)")
 | |
|     all_liked_songs = []
 | |
|     limit = 50  # Adjust the limit based on your needs
 | |
|     offset = 0
 | |
|     last_time_stamp = time.time()
 | |
| 
 | |
|     while True:
 | |
|         liked_songs_chunk = sp.current_user_saved_tracks(limit=limit, offset=offset)
 | |
|         if not liked_songs_chunk["items"]:
 | |
|             break
 | |
|         all_liked_songs = handle_playlist_part_return(liked_songs_chunk, all_liked_songs)
 | |
|         
 | |
|         progress_print, last_time_stamp = progress_bar(offset+limit, liked_songs_chunk["total"], last_time_stamp)
 | |
|         verboseprint(progress_print, end="\r")
 | |
| 
 | |
|         offset += limit
 | |
|     verboseprint("")
 | |
|     return all_liked_songs
 | |
| 
 | |
| def is_track_in_playlist(playlist_song_list, track_uri):
 | |
|     playlist_tracks = playlist_song_list
 | |
|     for item in playlist_tracks:
 | |
|         if item[0] == track_uri:
 | |
|             return True
 | |
|     return False
 | |
| 
 | |
| def add_track_to_playlist(playlist_id, track_uri):
 | |
|     sp.playlist_add_items(playlist_id, [track_uri])
 | |
| 
 | |
| def remove_unliked_tracks(primary_playlist_id, liked_tracks):
 | |
|     verboseprint("Removing unliked tracks from the primary playlist...")
 | |
|     liked_track_uris = [track_uri for track_uri, _, _ in liked_tracks]
 | |
|     primary_playlist_tracks = get_all_songs_from_playlist(primary_playlist_id)
 | |
|     
 | |
|     #tracks_to_remove = [track_uri for track_uri in primary_playlist_tracks if track_uri not in liked_track_uris]
 | |
| 
 | |
|     tracks_to_remove = []
 | |
|     for track_uri, _, _ in primary_playlist_tracks:
 | |
|         if track_uri not in liked_track_uris:
 | |
|             tracks_to_remove.append(track_uri)
 | |
|     
 | |
|     if tracks_to_remove:
 | |
|         sp.playlist_remove_all_occurrences_of_items(primary_playlist_id, tracks_to_remove)
 | |
|         print(f"Removed {len(tracks_to_remove)} unliked track(s) from the primary playlist.")
 | |
|         print(f"Track URIs: {tracks_to_remove}")
 | |
| 
 | |
| if __name__ == "__main__":
 | |
| 
 | |
|     # Parse command-line arguments
 | |
|     VERBOSE_LOGGING = "-v" in sys.argv or "--verbose" in sys.argv
 | |
| 
 | |
|     verboseprint("Authenticating Spotify...")
 | |
| 
 | |
|     # Create a Spotipy instance with authentication
 | |
|     sp = spotipy.Spotify(auth_manager=SpotifyOAuth(client_id=SPOTIPY_CLIENT_ID, client_secret=SPOTIPY_CLIENT_SECRET, redirect_uri=SPOTIPY_REDIRECT_URI, scope="user-library-read playlist-modify-public playlist-modify-private"))
 | |
| 
 | |
|     verboseprint("Authenticating Last.fm...")
 | |
| 
 | |
|     # Set up Last.fm network
 | |
|     network = pylast.LastFMNetwork(api_key=LASTFM_API_KEY, api_secret=LASTFM_API_SECRET, username=LASTFM_USERNAME, password_hash=LASTFM_PASSWORD_HASH)
 | |
| 
 | |
|     # Main loop for syncing liked songs
 | |
|     liked_songs = get_all_liked_songs()
 | |
| 
 | |
|     #remove_unliked_tracks(LIKEDSONGPLAYLIST_ID, liked_songs)
 | |
| 
 | |
|     liked_songs_playlist_songs = get_all_songs_from_playlist(LIKEDSONGPLAYLIST_ID)
 | |
| 
 | |
|     print(f"Number of liked tracks: {len(liked_songs)}")
 | |
|     print(f"Number of playlist songs: {len(liked_songs_playlist_songs)}")
 | |
| 
 | |
|     tracknr = 0
 | |
|     for track_uri, track_name, artist_name in liked_songs:
 | |
|         tracknr += 1
 | |
|         # Love the track on Last.fm
 | |
|         track = sp.track(track_uri)
 | |
|         fm_track = network.get_track(artist_name, track_name)
 | |
|         fm_track.love()
 | |
|         fm_track.add_tags(("awesome", "favorite"))
 | |
|         
 | |
|         if not is_track_in_playlist(liked_songs_playlist_songs, track_uri):
 | |
|             add_track_to_playlist(LIKEDSONGPLAYLIST_ID, track_uri)
 | |
|             if VERBOSE_LOGGING:
 | |
|                 verboseprint("%-10s %15s" % (f"ETA:{round((((int(len(liked_songs))-tracknr)*0.75)/60))}min", f"[{tracknr}/{int(len(liked_songs))}|+]") + "%30.32s %s" % (track['artists'][0]['name'], track['name']))
 | |
|         else:
 | |
|             verboseprint("%-10s %13s" % (f"ETA:{round((((int(len(liked_songs))-tracknr)*0.`75)/60))}min", f"[{tracknr}/{int(len(liked_songs))}]") + "%32.32s %s" % (track['artists'][0]['name'], track['name']))
 |