Added most common genre to all tracks (closes #34)
Resolved issue of there being multiple artists on a track and "manually created" an undefined genre for artists that don't have a genre.
This commit is contained in:
@@ -17,10 +17,5 @@
|
|||||||
<p class="browsehappy">You are using an <strong>outdated</strong> browser. Please <a href="#">upgrade your browser</a> to improve your experience.</p>
|
<p class="browsehappy">You are using an <strong>outdated</strong> browser. Please <a href="#">upgrade your browser</a> to improve your experience.</p>
|
||||||
<![endif]-->
|
<![endif]-->
|
||||||
<p>Logged in as {{ id }}</p>
|
<p>Logged in as {{ id }}</p>
|
||||||
<script src="{% static "spotifyvis/scripts/user_data.js" %}"></script>
|
|
||||||
<script>
|
|
||||||
sessionStorage.setItem('user_secret', "{{ user_secret }}");
|
|
||||||
getAudioFeatureData('instrumentalness', sessionStorage.getItem('user_secret'));
|
|
||||||
</script>
|
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
||||||
|
|||||||
@@ -12,10 +12,10 @@ import json
|
|||||||
# }}} imports #
|
# }}} imports #
|
||||||
|
|
||||||
USER_TRACKS_LIMIT = 50
|
USER_TRACKS_LIMIT = 50
|
||||||
# ARTIST_LIMIT = 50
|
ARTIST_LIMIT = 50
|
||||||
ARTIST_LIMIT = 25
|
FEATURES_LIMIT = 100
|
||||||
# FEATURES_LIMIT = 100
|
# ARTIST_LIMIT = 25
|
||||||
FEATURES_LIMIT = 25
|
# FEATURES_LIMIT = 25
|
||||||
|
|
||||||
# parse_library {{{ #
|
# parse_library {{{ #
|
||||||
|
|
||||||
@@ -37,7 +37,7 @@ def parse_library(headers, tracks, user):
|
|||||||
features_queue = []
|
features_queue = []
|
||||||
|
|
||||||
# iterate until hit requested num of tracks
|
# iterate until hit requested num of tracks
|
||||||
for _ in range(0, tracks, USER_TRACKS_LIMIT):
|
for i in range(0, tracks, USER_TRACKS_LIMIT):
|
||||||
payload['offset'] = str(offset)
|
payload['offset'] = str(offset)
|
||||||
saved_tracks_response = requests.get('https://api.spotify.com/v1/me/tracks',
|
saved_tracks_response = requests.get('https://api.spotify.com/v1/me/tracks',
|
||||||
headers=headers,
|
headers=headers,
|
||||||
@@ -80,6 +80,12 @@ def parse_library(headers, tracks, user):
|
|||||||
|
|
||||||
# }}} add audio features #
|
# }}} add audio features #
|
||||||
|
|
||||||
|
# temporary console logging
|
||||||
|
print("#{}-{}: {} - {}".format(offset + 1,
|
||||||
|
offset + USER_TRACKS_LIMIT,
|
||||||
|
track_obj.artists.first(),
|
||||||
|
track_obj.name))
|
||||||
|
|
||||||
# calculates num_songs with offset + songs retrieved
|
# calculates num_songs with offset + songs retrieved
|
||||||
offset += USER_TRACKS_LIMIT
|
offset += USER_TRACKS_LIMIT
|
||||||
|
|
||||||
@@ -111,11 +117,18 @@ def update_track_genres(user):
|
|||||||
"""
|
"""
|
||||||
user_tracks = Track.objects.filter(users__exact=user)
|
user_tracks = Track.objects.filter(users__exact=user)
|
||||||
for track in user_tracks:
|
for track in user_tracks:
|
||||||
track_artists = list(track.artists.all())
|
# just using this variable to save another call to db
|
||||||
if len(track_artists) == 1:
|
track_artists = track.artists.all()
|
||||||
track.genre = track_artists[0].genres.all().order_by('-num_songs').first()
|
# set genres to first artist's genres then find intersection with others
|
||||||
track.save()
|
shared_genres = track_artists.first().genres.all()
|
||||||
# print(track_artists, track.genre)
|
for artist in track_artists:
|
||||||
|
shared_genres.intersection(artist.genres.all())
|
||||||
|
|
||||||
|
most_common_genre = shared_genres.order_by('-num_songs').first()
|
||||||
|
track.genre = most_common_genre if most_common_genre is not None \
|
||||||
|
else "undefined"
|
||||||
|
track.save()
|
||||||
|
# print(track.name, track.genre)
|
||||||
|
|
||||||
# }}} update_track_genres #
|
# }}} update_track_genres #
|
||||||
|
|
||||||
@@ -186,23 +199,22 @@ def get_audio_features(headers, track_objs):
|
|||||||
|
|
||||||
# }}} get_audio_features #
|
# }}} get_audio_features #
|
||||||
|
|
||||||
# WIP: is this being removed to redo genre data?
|
def process_artist_genre(genre_name, artist_obj):
|
||||||
def get_top_genre(headers, top_artist_id):
|
"""Increase count for correspoding Genre object to genre_name and add that
|
||||||
"""Updates the top genre for a track by querying the Spotify API
|
Genre to artist_obj.
|
||||||
|
|
||||||
:headers: For making the API call.
|
:genre_name: Name of genre.
|
||||||
:top_artist: The first artist's (listed in the track) Spotify ID.
|
:artist_obj: Artist object to add Genre object to.
|
||||||
|
:returns: None
|
||||||
:returns: The first genre listed for the top_artist.
|
|
||||||
|
|
||||||
"""
|
"""
|
||||||
artist_response = requests.get('https://api.spotify.com/v1/artists/' +
|
genre_obj, created = Genre.objects.get_or_create(name=genre_name,
|
||||||
top_artist_id, headers=headers).json()
|
defaults={'num_songs':1})
|
||||||
# pprint.pprint(artist_response)
|
if not created:
|
||||||
if len(artist_response['genres']) > 0:
|
genre_obj.num_songs = F('num_songs') + 1
|
||||||
return artist_response['genres'][0]
|
genre_obj.save()
|
||||||
else:
|
artist_obj.genres.add(genre_obj)
|
||||||
return "undefined"
|
artist_obj.save()
|
||||||
|
|
||||||
# add_artist_genres {{{ #
|
# add_artist_genres {{{ #
|
||||||
|
|
||||||
@@ -223,14 +235,11 @@ def add_artist_genres(headers, artist_objs):
|
|||||||
headers=headers, params=params).json()['artists']
|
headers=headers, params=params).json()['artists']
|
||||||
# pprint.pprint(artists_response)
|
# pprint.pprint(artists_response)
|
||||||
for i in range(len(artist_objs)):
|
for i in range(len(artist_objs)):
|
||||||
for genre in artists_response[i]['genres']:
|
if len(artists_response[i]['genres']) == 0:
|
||||||
genre_obj, created = Genre.objects.get_or_create(name=genre,
|
process_artist_genre("undefined", artist_objs[i])
|
||||||
defaults={'num_songs':1})
|
else:
|
||||||
if not created:
|
for genre in artists_response[i]['genres']:
|
||||||
genre_obj.num_songs = F('num_songs') +1
|
process_artist_genre(genre, artist_objs[i])
|
||||||
genre_obj.save()
|
|
||||||
artist_objs[i].genres.add(genre_obj)
|
|
||||||
artist_objs[i].save()
|
|
||||||
|
|
||||||
# }}} add_artist_genres #
|
# }}} add_artist_genres #
|
||||||
|
|
||||||
@@ -255,6 +264,8 @@ def get_artists_in_genre(user, genre, max_songs):
|
|||||||
processed_artist_counts = {}
|
processed_artist_counts = {}
|
||||||
songs_added = 0
|
songs_added = 0
|
||||||
for artist in artist_counts:
|
for artist in artist_counts:
|
||||||
|
# hacky way to not have total count overflow due to there being multiple
|
||||||
|
# artists on a track
|
||||||
if songs_added + artist.num_songs <= max_songs:
|
if songs_added + artist.num_songs <= max_songs:
|
||||||
processed_artist_counts[artist.name] = artist.num_songs
|
processed_artist_counts[artist.name] = artist.num_songs
|
||||||
songs_added += artist.num_songs
|
songs_added += artist.num_songs
|
||||||
|
|||||||
@@ -19,7 +19,7 @@ from .models import User, Track, AudioFeatures, Artist
|
|||||||
# }}} imports #
|
# }}} imports #
|
||||||
|
|
||||||
TIME_FORMAT = '%Y-%m-%d-%H-%M-%S'
|
TIME_FORMAT = '%Y-%m-%d-%H-%M-%S'
|
||||||
TRACKS_TO_QUERY = 50
|
TRACKS_TO_QUERY = 200
|
||||||
|
|
||||||
# generate_random_string {{{ #
|
# generate_random_string {{{ #
|
||||||
|
|
||||||
@@ -237,6 +237,7 @@ def get_genre_data(request, user_secret):
|
|||||||
for genre_dict in genre_counts:
|
for genre_dict in genre_counts:
|
||||||
genre_dict['artists'] = get_artists_in_genre(user, genre_dict['genre'],
|
genre_dict['artists'] = get_artists_in_genre(user, genre_dict['genre'],
|
||||||
genre_dict['num_songs'])
|
genre_dict['num_songs'])
|
||||||
|
print("*** Genre Breakdown ***")
|
||||||
pprint.pprint(list(genre_counts))
|
pprint.pprint(list(genre_counts))
|
||||||
return JsonResponse(data=list(genre_counts), safe=False)
|
return JsonResponse(data=list(genre_counts), safe=False)
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user