From 31aeb6e69a8f2f416dd8dcad50ccecdaf9b846ad Mon Sep 17 00:00:00 2001 From: hugovk Date: Tue, 17 Oct 2017 21:45:20 +0300 Subject: [PATCH] Continue tests refactor --- tests/test_pylast.py | 2011 +++++++++++++++++++++--------------------- 1 file changed, 1010 insertions(+), 1001 deletions(-) diff --git a/tests/test_pylast.py b/tests/test_pylast.py index 3d8ad83..2342e02 100755 --- a/tests/test_pylast.py +++ b/tests/test_pylast.py @@ -56,6 +56,72 @@ class PyLastTestCase(unittest.TestCase): if value is None or len(value) == 0: pytest.skip("Last.fm API is broken.") + def helper_is_thing_hashable(self, thing): + # Arrange + things = set() + + # Act + things.add(thing) + + # Assert + self.assertIsNotNone(thing) + self.assertEqual(len(things), 1) + + def helper_validate_results(self, a, b, c): + # Assert + self.assertIsNotNone(a) + self.assertIsNotNone(b) + self.assertIsNotNone(c) + self.assertGreaterEqual(len(a), 0) + self.assertGreaterEqual(len(b), 0) + self.assertGreaterEqual(len(c), 0) + self.assertEqual(a, b) + self.assertEqual(b, c) + + def helper_validate_cacheable(self, thing, function_name): + # Arrange + # get thing.function_name() + func = getattr(thing, function_name, None) + + # Act + result1 = func(limit=1, cacheable=False) + result2 = func(limit=1, cacheable=True) + result3 = func(limit=1) + + # Assert + self.helper_validate_results(result1, result2, result3) + + def helper_at_least_one_thing_in_top_list(self, things, expected_type): + # Assert + self.assertGreater(len(things), 1) + self.assertIsInstance(things, list) + self.assertIsInstance(things[0], pylast.TopItem) + self.assertIsInstance(things[0].item, expected_type) + + def helper_only_one_thing_in_top_list(self, things, expected_type): + # Assert + self.assertEqual(len(things), 1) + self.assertIsInstance(things, list) + self.assertIsInstance(things[0], pylast.TopItem) + self.assertIsInstance(things[0].item, expected_type) + + def helper_only_one_thing_in_list(self, things, expected_type): + # Assert + self.assertEqual(len(things), 1) + self.assertIsInstance(things, list) + self.assertIsInstance(things[0], expected_type) + + def helper_two_different_things_in_top_list(self, things, expected_type): + # Assert + self.assertEqual(len(things), 2) + thing1 = things[0] + thing2 = things[1] + self.assertIsInstance(thing1, pylast.TopItem) + self.assertIsInstance(thing2, pylast.TopItem) + self.assertIsInstance(thing1.item, expected_type) + self.assertIsInstance(thing2.item, expected_type) + self.assertNotEqual(thing1, thing2) + class TestPyLastAlbum(PyLastTestCase): @@ -70,188 +136,6 @@ class TestPyLastAlbum(PyLastTestCase): self.assertGreater(len(tags), 0) self.assertIsInstance(tags[0], pylast.TopItem) - -class TestPyLastNetwork(PyLastTestCase): - - def test_scrobble(self): - # Arrange - artist = "Test Artist" - title = "test title" - timestamp = self.unix_timestamp() - lastfm_user = self.network.get_user(self.username) - - # Act - self.network.scrobble(artist=artist, title=title, timestamp=timestamp) - - # Assert - # limit=2 to ignore now-playing: - last_scrobble = lastfm_user.get_recent_tracks(limit=2)[0] - self.assertEqual(str(last_scrobble.track.artist), str(artist)) - self.assertEqual(str(last_scrobble.track.title), str(title)) - self.assertEqual(str(last_scrobble.timestamp), str(timestamp)) - - def test_update_now_playing(self): - # Arrange - artist = "Test Artist" - title = "test title" - album = "Test Album" - track_number = 1 - lastfm_user = self.network.get_user(self.username) - - # Act - self.network.update_now_playing( - artist=artist, title=title, album=album, track_number=track_number) - - # Assert - current_track = lastfm_user.get_now_playing() - self.assertIsNotNone(current_track) - self.assertEqual(str(current_track.title), "test title") - self.assertEqual(str(current_track.artist), "Test Artist") - - -class TestPyLastTrack(PyLastTestCase): - - def test_love(self): - # Arrange - artist = "Test Artist" - title = "test title" - track = self.network.get_track(artist, title) - lastfm_user = self.network.get_user(self.username) - - # Act - track.love() - - # Assert - loved = lastfm_user.get_loved_tracks(limit=1) - self.assertEqual(str(loved[0].track.artist), "Test Artist") - self.assertEqual(str(loved[0].track.title), "test title") - - def test_unlove(self): - # Arrange - artist = pylast.Artist("Test Artist", self.network) - title = "test title" - track = pylast.Track(artist, title, self.network) - lastfm_user = self.network.get_user(self.username) - track.love() - - # Act - track.unlove() - - # Assert - loved = lastfm_user.get_loved_tracks(limit=1) - if len(loved): # OK to be empty but if not: - self.assertNotEqual(str(loved.track.artist), "Test Artist") - self.assertNotEqual(str(loved.track.title), "test title") - - -class TestPyLastUser(PyLastTestCase): - - def test_get_user_registration(self): - # Arrange - username = "RJ" - user = self.network.get_user(username) - - # Act - registered = user.get_registered() - - # Assert - # Last.fm API broken? Should be yyyy-mm-dd not Unix timestamp - if int(registered): - pytest.skip("Last.fm API is broken.") - - # Just check date because of timezones - self.assertIn(u"2002-11-20 ", registered) - - def test_get_user_unixtime_registration(self): - # Arrange - username = "RJ" - user = self.network.get_user(username) - - # Act - unixtime_registered = user.get_unixtime_registered() - - # Assert - # Just check date because of timezones - self.assertEqual(unixtime_registered, u"1037793040") - - def test_get_genderless_user(self): - # Arrange - # Currently test_user has no gender set: - lastfm_user = self.network.get_user("test_user") - - # Act - gender = lastfm_user.get_gender() - - # Assert - self.assertIsNone(gender) - - def test_get_countryless_user(self): - # Arrange - # Currently test_user has no country set: - lastfm_user = self.network.get_user("test_user") - - # Act - country = lastfm_user.get_country() - - # Assert - self.assertIsNone(country) - - def test_user_equals_none(self): - # Arrange - lastfm_user = self.network.get_user(self.username) - - # Act - value = (lastfm_user is None) - - # Assert - self.assertFalse(value) - - def test_user_not_equal_to_none(self): - # Arrange - lastfm_user = self.network.get_user(self.username) - - # Act - value = (lastfm_user is not None) - - # Assert - self.assertTrue(value) - - def test_now_playing_user_with_no_scrobbles(self): - # Arrange - # Currently test-account has no scrobbles: - user = self.network.get_user('test-account') - - # Act - current_track = user.get_now_playing() - - # Assert - self.assertIsNone(current_track) - - def test_love_limits(self): - # Arrange - # Currently test-account has at least 23 loved tracks: - user = self.network.get_user("test-user") - - # Act/Assert - self.assertEqual(len(user.get_loved_tracks(limit=20)), 20) - self.assertLessEqual(len(user.get_loved_tracks(limit=100)), 100) - self.assertGreaterEqual(len(user.get_loved_tracks(limit=None)), 23) - self.assertGreaterEqual(len(user.get_loved_tracks(limit=0)), 23) - - -class TestPyLast(PyLastTestCase): - - def helper_is_thing_hashable(self, thing): - # Arrange - things = set() - - # Act - things.add(thing) - - # Assert - self.assertIsNotNone(thing) - self.assertEqual(len(things), 1) - def test_album_is_hashable(self): # Arrange album = self.network.get_album("Test Artist", "Test Album") @@ -259,96 +143,6 @@ class TestPyLast(PyLastTestCase): # Act/Assert self.helper_is_thing_hashable(album) - def test_artist_is_hashable(self): - # Arrange - test_artist = self.network.get_artist("Test Artist") - artist = test_artist.get_similar(limit=2)[0].item - self.assertIsInstance(artist, pylast.Artist) - - # Act/Assert - self.helper_is_thing_hashable(artist) - - def test_country_is_hashable(self): - # Arrange - country = self.network.get_country("Italy") - - # Act/Assert - self.helper_is_thing_hashable(country) - - def test_library_is_hashable(self): - # Arrange - library = pylast.Library(user=self.username, network=self.network) - - # Act/Assert - self.helper_is_thing_hashable(library) - - def test_tag_is_hashable(self): - # Arrange - tag = self.network.get_top_tags(limit=1)[0] - - # Act/Assert - self.helper_is_thing_hashable(tag) - - def test_track_is_hashable(self): - # Arrange - artist = self.network.get_artist("Test Artist") - track = artist.get_top_tracks()[0].item - self.assertIsInstance(track, pylast.Track) - - # Act/Assert - self.helper_is_thing_hashable(track) - - def test_user_is_hashable(self): - # Arrange - user = self.network.get_user(self.username) - - # Act/Assert - self.helper_is_thing_hashable(user) - - def test_invalid_xml(self): - # Arrange - # Currently causes PCDATA invalid Char value 25 - artist = "Blind Willie Johnson" - title = "It's nobody's fault but mine" - - # Act - search = self.network.search_for_track(artist, title) - total = search.get_total_result_count() - - # Assert - self.skip_if_lastfm_api_broken(total) - self.assertGreaterEqual(int(total), 0) - - def test_user_play_count_in_track_info(self): - # Arrange - artist = "Test Artist" - title = "test title" - track = pylast.Track( - artist=artist, title=title, - network=self.network, username=self.username) - - # Act - count = track.get_userplaycount() - - # Assert - self.assertGreaterEqual(count, 0) - - def test_user_loved_in_track_info(self): - # Arrange - artist = "Test Artist" - title = "test title" - track = pylast.Track( - artist=artist, title=title, - network=self.network, username=self.username) - - # Act - loved = track.get_userloved() - - # Assert - self.assertIsNotNone(loved) - self.assertIsInstance(loved, bool) - self.assertNotIsInstance(loved, str) - def test_album_in_recent_tracks(self): # Arrange lastfm_user = self.network.get_user(self.username) @@ -370,101 +164,6 @@ class TestPyLast(PyLastTestCase): # Assert self.assertTrue(hasattr(track, 'album')) - def test_enable_rate_limiting(self): - # Arrange - self.assertFalse(self.network.is_rate_limited()) - - # Act - self.network.enable_rate_limit() - then = time.time() - # Make some network call, limit not applied first time - self.network.get_user(self.username) - # Make a second network call, limiting should be applied - self.network.get_top_artists() - now = time.time() - - # Assert - self.assertTrue(self.network.is_rate_limited()) - self.assertGreaterEqual(now - then, 0.2) - - def test_disable_rate_limiting(self): - # Arrange - self.network.enable_rate_limit() - self.assertTrue(self.network.is_rate_limited()) - - # Act - self.network.disable_rate_limit() - # Make some network call, limit not applied first time - self.network.get_user(self.username) - # Make a second network call, limiting should be applied - self.network.get_top_artists() - - # Assert - self.assertFalse(self.network.is_rate_limited()) - - # Commented out because (a) it'll take a long time and (b) it strangely - # fails due Last.fm's complaining of hitting the rate limit, even when - # limited to one call per second. The ToS allows 5 calls per second. - # def test_get_all_scrobbles(self): - # # Arrange - # lastfm_user = self.network.get_user("RJ") - # self.network.enable_rate_limit() # this is going to be slow... - - # # Act - # tracks = lastfm_user.get_recent_tracks(limit=None) - - # # Assert - # self.assertGreaterEqual(len(tracks), 0) - - def test_pickle(self): - # Arrange - import pickle - lastfm_user = self.network.get_user(self.username) - filename = str(self.unix_timestamp()) + ".pkl" - - # Act - with open(filename, "wb") as f: - pickle.dump(lastfm_user, f) - with open(filename, "rb") as f: - loaded_user = pickle.load(f) - os.remove(filename) - - # Assert - self.assertEqual(lastfm_user, loaded_user) - - def test_bio_published_date(self): - # Arrange - artist = pylast.Artist("Test Artist", self.network) - - # Act - bio = artist.get_bio_published_date() - - # Assert - self.assertIsNotNone(bio) - self.assertGreaterEqual(len(bio), 1) - - def test_bio_content(self): - # Arrange - artist = pylast.Artist("Test Artist", self.network) - - # Act - bio = artist.get_bio_content(language="en") - - # Assert - self.assertIsNotNone(bio) - self.assertGreaterEqual(len(bio), 1) - - def test_bio_summary(self): - # Arrange - artist = pylast.Artist("Test Artist", self.network) - - # Act - bio = artist.get_bio_summary(language="en") - - # Assert - self.assertIsNotNone(bio) - self.assertGreaterEqual(len(bio), 1) - def test_album_wiki_content(self): # Arrange album = pylast.Album("Test Artist", "Test Album", self.network) @@ -498,189 +197,66 @@ class TestPyLast(PyLastTestCase): self.assertIsNotNone(wiki) self.assertGreaterEqual(len(wiki), 1) - def test_track_wiki_content(self): + def test_album_eq_none_is_false(self): # Arrange - track = pylast.Track("Test Artist", "test title", self.network) + album1 = None + album2 = pylast.Album("Test Artist", "Test Album", self.network) - # Act - wiki = track.get_wiki_content() + # Act / Assert + self.assertFalse(album1 == album2) - # Assert - self.assertIsNotNone(wiki) - self.assertGreaterEqual(len(wiki), 1) - - def test_track_wiki_summary(self): + def test_album_ne_none_is_true(self): # Arrange - track = pylast.Track("Test Artist", "test title", self.network) + album1 = None + album2 = pylast.Album("Test Artist", "Test Album", self.network) - # Act - wiki = track.get_wiki_summary() + # Act / Assert + self.assertTrue(album1 != album2) - # Assert - self.assertIsNotNone(wiki) - self.assertGreaterEqual(len(wiki), 1) - def test_lastfm_network_name(self): - # Act - name = str(self.network) +class TestPyLastArtist(PyLastTestCase): - # Assert - self.assertEqual(name, "Last.fm Network") - - def helper_validate_results(self, a, b, c): - # Assert - self.assertIsNotNone(a) - self.assertIsNotNone(b) - self.assertIsNotNone(c) - self.assertGreaterEqual(len(a), 0) - self.assertGreaterEqual(len(b), 0) - self.assertGreaterEqual(len(c), 0) - self.assertEqual(a, b) - self.assertEqual(b, c) - - def helper_validate_cacheable(self, thing, function_name): + def test_artist_is_hashable(self): # Arrange - # get thing.function_name() - func = getattr(thing, function_name, None) - - # Act - result1 = func(limit=1, cacheable=False) - result2 = func(limit=1, cacheable=True) - result3 = func(limit=1) - - # Assert - self.helper_validate_results(result1, result2, result3) - - def test_cacheable_library(self): - # Arrange - library = pylast.Library(self.username, self.network) + test_artist = self.network.get_artist("Test Artist") + artist = test_artist.get_similar(limit=2)[0].item + self.assertIsInstance(artist, pylast.Artist) # Act/Assert - self.helper_validate_cacheable(library, "get_artists") + self.helper_is_thing_hashable(artist) - def test_cacheable_user_artist_tracks(self): + def test_bio_published_date(self): # Arrange - lastfm_user = self.network.get_authenticated_user() + artist = pylast.Artist("Test Artist", self.network) # Act - result1 = lastfm_user.get_artist_tracks("Test Artist", cacheable=False) - result2 = lastfm_user.get_artist_tracks("Test Artist", cacheable=True) - result3 = lastfm_user.get_artist_tracks("Test Artist") + bio = artist.get_bio_published_date() # Assert - self.helper_validate_results(result1, result2, result3) + self.assertIsNotNone(bio) + self.assertGreaterEqual(len(bio), 1) - def test_cacheable_user(self): + def test_bio_content(self): # Arrange - lastfm_user = self.network.get_authenticated_user() - - # Act/Assert - self.helper_validate_cacheable(lastfm_user, "get_friends") - self.helper_validate_cacheable(lastfm_user, "get_loved_tracks") - self.helper_validate_cacheable(lastfm_user, "get_recent_tracks") - - def helper_dates_valid(self, dates): - # Assert - self.assertGreaterEqual(len(dates), 1) - self.assertIsInstance(dates[0], tuple) - (start, end) = dates[0] - self.assertLess(start, end) - - def test_geo_get_top_artists(self): - # Arrange - # Act - artists = self.network.get_geo_top_artists( - country="United Kingdom", limit=1) - - # Assert - self.assertEqual(len(artists), 1) - self.assertIsInstance(artists[0], pylast.TopItem) - self.assertIsInstance(artists[0].item, pylast.Artist) - - def test_geo_get_top_tracks(self): - # Arrange - # Act - tracks = self.network.get_geo_top_tracks( - country="United Kingdom", location="Manchester", limit=1) - - # Assert - self.assertEqual(len(tracks), 1) - self.assertIsInstance(tracks[0], pylast.TopItem) - self.assertIsInstance(tracks[0].item, pylast.Track) - - def helper_at_least_one_thing_in_top_list(self, things, expected_type): - # Assert - self.assertGreater(len(things), 1) - self.assertIsInstance(things, list) - self.assertIsInstance(things[0], pylast.TopItem) - self.assertIsInstance(things[0].item, expected_type) - - def helper_only_one_thing_in_top_list(self, things, expected_type): - # Assert - self.assertEqual(len(things), 1) - self.assertIsInstance(things, list) - self.assertIsInstance(things[0], pylast.TopItem) - self.assertIsInstance(things[0].item, expected_type) - - def helper_only_one_thing_in_list(self, things, expected_type): - # Assert - self.assertEqual(len(things), 1) - self.assertIsInstance(things, list) - self.assertIsInstance(things[0], expected_type) - - def helper_two_different_things_in_top_list(self, things, expected_type): - # Assert - self.assertEqual(len(things), 2) - thing1 = things[0] - thing2 = things[1] - self.assertIsInstance(thing1, pylast.TopItem) - self.assertIsInstance(thing2, pylast.TopItem) - self.assertIsInstance(thing1.item, expected_type) - self.assertIsInstance(thing2.item, expected_type) - self.assertNotEqual(thing1, thing2) - - def test_user_get_top_tags_with_limit(self): - # Arrange - user = self.network.get_user("RJ") + artist = pylast.Artist("Test Artist", self.network) # Act - tags = user.get_top_tags(limit=1) + bio = artist.get_bio_content(language="en") # Assert - self.skip_if_lastfm_api_broken(tags) - self.helper_only_one_thing_in_top_list(tags, pylast.Tag) + self.assertIsNotNone(bio) + self.assertGreaterEqual(len(bio), 1) - def test_network_get_top_artists_with_limit(self): + def test_bio_summary(self): # Arrange + artist = pylast.Artist("Test Artist", self.network) + # Act - artists = self.network.get_top_artists(limit=1) + bio = artist.get_bio_summary(language="en") # Assert - self.helper_only_one_thing_in_top_list(artists, pylast.Artist) - - def test_network_get_top_tags_with_limit(self): - # Arrange - # Act - tags = self.network.get_top_tags(limit=1) - - # Assert - self.helper_only_one_thing_in_top_list(tags, pylast.Tag) - - def test_network_get_top_tags_with_no_limit(self): - # Arrange - # Act - tags = self.network.get_top_tags() - - # Assert - self.helper_at_least_one_thing_in_top_list(tags, pylast.Tag) - - def test_network_get_top_tracks_with_limit(self): - # Arrange - # Act - tracks = self.network.get_top_tracks(limit=1) - - # Assert - self.helper_only_one_thing_in_top_list(tracks, pylast.Track) + self.assertIsNotNone(bio) + self.assertGreaterEqual(len(bio), 1) def test_artist_top_tracks(self): # Arrange @@ -704,310 +280,6 @@ class TestPyLast(PyLastTestCase): # Assert self.helper_two_different_things_in_top_list(things, pylast.Album) - def test_country_top_tracks(self): - # Arrange - country = self.network.get_country("Croatia") - - # Act - things = country.get_top_tracks(limit=2) - - # Assert - self.helper_two_different_things_in_top_list(things, pylast.Track) - - def test_country_network_top_tracks(self): - # Arrange - # Act - things = self.network.get_geo_top_tracks("Croatia", limit=2) - - # Assert - self.helper_two_different_things_in_top_list(things, pylast.Track) - - def test_tag_top_tracks(self): - # Arrange - tag = self.network.get_tag("blues") - - # Act - things = tag.get_top_tracks(limit=2) - - # Assert - self.helper_two_different_things_in_top_list(things, pylast.Track) - - def test_user_top_tracks(self): - # Arrange - lastfm_user = self.network.get_user(self.username) - - # Act - things = lastfm_user.get_top_tracks(limit=2) - - # Assert - self.helper_two_different_things_in_top_list(things, pylast.Track) - - def helper_assert_chart(self, chart, expected_type): - # Assert - self.assertIsNotNone(chart) - self.assertGreater(len(chart), 0) - self.assertIsInstance(chart[0], pylast.TopItem) - self.assertIsInstance(chart[0].item, expected_type) - - def helper_get_assert_charts(self, thing, date): - # Arrange - (from_date, to_date) = date - - # Act - artist_chart = thing.get_weekly_artist_charts(from_date, to_date) - if type(thing) is not pylast.Tag: - album_chart = thing.get_weekly_album_charts(from_date, to_date) - track_chart = thing.get_weekly_track_charts(from_date, to_date) - - # Assert - self.helper_assert_chart(artist_chart, pylast.Artist) - if type(thing) is not pylast.Tag: - self.helper_assert_chart(album_chart, pylast.Album) - self.helper_assert_chart(track_chart, pylast.Track) - - def test_user_charts(self): - # Arrange - lastfm_user = self.network.get_user("RJ") - dates = lastfm_user.get_weekly_chart_dates() - self.helper_dates_valid(dates) - - # Act/Assert - self.helper_get_assert_charts(lastfm_user, dates[0]) - - # Commented out to avoid spamming - # def test_share_spam(self): - # # Arrange - # users_to_spam = [TODO_ENTER_SPAMEES_HERE] - # spam_message = "Dig the krazee sound!" - # artist = self.network.get_top_artists(limit=1)[0].item - # track = artist.get_top_tracks(limit=1)[0].item - - # # Act - # artist.share(users_to_spam, spam_message) - # track.share(users_to_spam, spam_message) - - # Assert - # Check inbox for spam! - - # album/artist/track/user - - def test_album_data(self): - # Arrange - thing = self.network.get_album("Test Artist", "Test Album") - - # Act - stringed = str(thing) - repr = thing.__repr__() - title = thing.get_title() - name = thing.get_name() - playcount = thing.get_playcount() - url = thing.get_url() - - # Assert - self.assertEqual(stringed, "Test Artist - Test Album") - self.assertIn("pylast.Album('Test Artist', 'Test Album',", repr) - self.assertEqual(title, name) - self.assertIsInstance(playcount, int) - self.assertGreater(playcount, 1) - self.assertEqual( - "https://www.last.fm/music/test%2bartist/test%2balbum", url) - - def test_track_data(self): - # Arrange - thing = self.network.get_track("Test Artist", "test title") - - # Act - stringed = str(thing) - repr = thing.__repr__() - title = thing.get_title() - name = thing.get_name() - playcount = thing.get_playcount() - url = thing.get_url(pylast.DOMAIN_FRENCH) - - # Assert - self.assertEqual(stringed, "Test Artist - test title") - self.assertIn("pylast.Track('Test Artist', 'test title',", repr) - self.assertEqual(title, "test title") - self.assertEqual(title, name) - self.assertIsInstance(playcount, int) - self.assertGreater(playcount, 1) - self.assertEqual( - "https://www.last.fm/fr/music/test%2bartist/_/test%2btitle", url) - - def test_tag_top_artists(self): - # Arrange - tag = self.network.get_tag("blues") - - # Act - artists = tag.get_top_artists(limit=1) - - # Assert - self.helper_only_one_thing_in_top_list(artists, pylast.Artist) - - def test_country_top_artists(self): - # Arrange - country = self.network.get_country("Ukraine") - - # Act - artists = country.get_top_artists(limit=1) - - # Assert - self.helper_only_one_thing_in_top_list(artists, pylast.Artist) - - def test_user_top_artists(self): - # Arrange - lastfm_user = self.network.get_user(self.username) - - # Act - artists = lastfm_user.get_top_artists(limit=1) - - # Assert - self.helper_only_one_thing_in_top_list(artists, pylast.Artist) - - def test_tag_top_albums(self): - # Arrange - tag = self.network.get_tag("blues") - - # Act - albums = tag.get_top_albums(limit=1) - - # Assert - self.helper_only_one_thing_in_top_list(albums, pylast.Album) - - def test_user_top_albums(self): - # Arrange - user = self.network.get_user("RJ") - - # Act - albums = user.get_top_albums(limit=1) - - # Assert - self.helper_only_one_thing_in_top_list(albums, pylast.Album) - - def test_user_tagged_artists(self): - # Arrange - lastfm_user = self.network.get_user(self.username) - tags = ["artisttagola"] - artist = self.network.get_artist("Test Artist") - artist.add_tags(tags) - - # Act - artists = lastfm_user.get_tagged_artists('artisttagola', limit=1) - - # Assert - self.helper_only_one_thing_in_list(artists, pylast.Artist) - - def test_user_tagged_albums(self): - # Arrange - lastfm_user = self.network.get_user(self.username) - tags = ["albumtagola"] - album = self.network.get_album("Test Artist", "Test Album") - album.add_tags(tags) - - # Act - albums = lastfm_user.get_tagged_albums('albumtagola', limit=1) - - # Assert - self.helper_only_one_thing_in_list(albums, pylast.Album) - - def test_user_tagged_tracks(self): - # Arrange - lastfm_user = self.network.get_user(self.username) - tags = ["tracktagola"] - track = self.network.get_track("Test Artist", "test title") - track.add_tags(tags) - - # Act - tracks = lastfm_user.get_tagged_tracks('tracktagola', limit=1) - - # Assert - self.helper_only_one_thing_in_list(tracks, pylast.Track) - - def test_user_subscriber(self): - # Arrange - subscriber = self.network.get_user("RJ") - non_subscriber = self.network.get_user("Test User") - - # Act - subscriber_is_subscriber = subscriber.is_subscriber() - non_subscriber_is_subscriber = non_subscriber.is_subscriber() - - # Assert - self.assertTrue(subscriber_is_subscriber) - self.assertFalse(non_subscriber_is_subscriber) - - def test_user_get_image(self): - # Arrange - user = self.network.get_user("RJ") - - # Act - url = user.get_image() - - # Assert - self.assertTrue(url.startswith("https://")) - - def test_user_get_library(self): - # Arrange - user = self.network.get_user(self.username) - - # Act - library = user.get_library() - - # Assert - self.assertIsInstance(library, pylast.Library) - - def test_caching(self): - # Arrange - user = self.network.get_user("RJ") - - # Act - self.network.enable_caching() - tags1 = user.get_top_tags(limit=1, cacheable=True) - tags2 = user.get_top_tags(limit=1, cacheable=True) - - # Assert - self.assertTrue(self.network.is_caching_enabled()) - self.assertEqual(tags1, tags2) - self.network.disable_caching() - self.assertFalse(self.network.is_caching_enabled()) - - def test_album_mbid(self): - # Arrange - mbid = "a6a265bf-9f81-4055-8224-f7ac0aa6b937" - - # Act - album = self.network.get_album_by_mbid(mbid) - album_mbid = album.get_mbid() - - # Assert - self.assertIsInstance(album, pylast.Album) - self.assertEqual(album.title.lower(), "test") - self.assertEqual(album_mbid, mbid) - - def test_artist_mbid(self): - # Arrange - mbid = "7e84f845-ac16-41fe-9ff8-df12eb32af55" - - # Act - artist = self.network.get_artist_by_mbid(mbid) - - # Assert - self.assertIsInstance(artist, pylast.Artist) - self.assertEqual(artist.name, "MusicBrainz Test Artist") - - def test_track_mbid(self): - # Arrange - mbid = "ebc037b1-cc9c-44f2-a21f-83c219f0e1e0" - - # Act - track = self.network.get_track_by_mbid(mbid) - track_mbid = track.get_mbid() - - # Assert - self.assertIsInstance(track, pylast.Track) - self.assertEqual(track.title, "first") - self.assertEqual(track_mbid, mbid) - def test_artist_listener_count(self): # Arrange artist = self.network.get_artist("Test Artist") @@ -1120,6 +392,918 @@ class TestPyLast(PyLastTestCase): self.assertTrue(found1) self.assertTrue(found2) + def test_artists(self): + # Arrange + artist1 = self.network.get_artist("Radiohead") + artist2 = self.network.get_artist("Portishead") + + # Act + url = artist1.get_url() + mbid = artist1.get_mbid() + image = artist1.get_cover_image() + playcount = artist1.get_playcount() + streamable = artist1.is_streamable() + name = artist1.get_name(properly_capitalized=False) + name_cap = artist1.get_name(properly_capitalized=True) + + # Assert + self.assertIn("http", image) + self.assertGreater(playcount, 1) + self.assertTrue(artist1 != artist2) + self.assertEqual(name.lower(), name_cap.lower()) + self.assertEqual(url, "https://www.last.fm/music/radiohead") + self.assertEqual(mbid, "a74b1b7f-71a5-4011-9441-d0b5e4122711") + self.assertIsInstance(streamable, bool) + + def test_artist_eq_none_is_false(self): + # Arrange + artist1 = None + artist2 = pylast.Artist("Test Artist", self.network) + + # Act / Assert + self.assertFalse(artist1 == artist2) + + def test_artist_ne_none_is_true(self): + # Arrange + artist1 = None + artist2 = pylast.Artist("Test Artist", self.network) + + # Act / Assert + self.assertTrue(artist1 != artist2) + + def test_band_members(self): + # Arrange + artist = pylast.Artist("The Beatles", self.network) + + # Act + band_members = artist.get_band_members() + + # Assert + self.skip_if_lastfm_api_broken(band_members) + self.assertGreaterEqual(len(band_members), 4) + + def test_no_band_members(self): + # Arrange + artist = pylast.Artist("John Lennon", self.network) + + # Act + band_members = artist.get_band_members() + + # Assert + self.assertIsNone(band_members) + + def test_artist_get_correction(self): + # Arrange + artist = pylast.Artist("guns and roses", self.network) + + # Act + corrected_artist_name = artist.get_correction() + + # Assert + self.assertEqual(corrected_artist_name, "Guns N' Roses") + + +class TestPyLastCountry(PyLastTestCase): + + def test_country_is_hashable(self): + # Arrange + country = self.network.get_country("Italy") + + # Act/Assert + self.helper_is_thing_hashable(country) + + def test_countries(self): + # Arrange + country1 = pylast.Country("Italy", self.network) + country2 = pylast.Country("Finland", self.network) + + # Act + text = str(country1) + rep = repr(country1) + url = country1.get_url() + + # Assert + self.assertIn("Italy", rep) + self.assertIn("pylast.Country", rep) + self.assertEqual(text, "Italy") + self.assertTrue(country1 == country1) + self.assertTrue(country1 != country2) + self.assertEqual(url, "https://www.last.fm/place/italy") + + +class TestPyLastLibrary(PyLastTestCase): + + def test_library_is_hashable(self): + # Arrange + library = pylast.Library(user=self.username, network=self.network) + + # Act/Assert + self.helper_is_thing_hashable(library) + + def test_cacheable_library(self): + # Arrange + library = pylast.Library(self.username, self.network) + + # Act/Assert + self.helper_validate_cacheable(library, "get_artists") + + +class TestPyLastNetwork(PyLastTestCase): + + def test_scrobble(self): + # Arrange + artist = "Test Artist" + title = "test title" + timestamp = self.unix_timestamp() + lastfm_user = self.network.get_user(self.username) + + # Act + self.network.scrobble(artist=artist, title=title, timestamp=timestamp) + + # Assert + # limit=2 to ignore now-playing: + last_scrobble = lastfm_user.get_recent_tracks(limit=2)[0] + self.assertEqual(str(last_scrobble.track.artist), str(artist)) + self.assertEqual(str(last_scrobble.track.title), str(title)) + self.assertEqual(str(last_scrobble.timestamp), str(timestamp)) + + def test_update_now_playing(self): + # Arrange + artist = "Test Artist" + title = "test title" + album = "Test Album" + track_number = 1 + lastfm_user = self.network.get_user(self.username) + + # Act + self.network.update_now_playing( + artist=artist, title=title, album=album, track_number=track_number) + + # Assert + current_track = lastfm_user.get_now_playing() + self.assertIsNotNone(current_track) + self.assertEqual(str(current_track.title), "test title") + self.assertEqual(str(current_track.artist), "Test Artist") + + def test_invalid_xml(self): + # Arrange + # Currently causes PCDATA invalid Char value 25 + artist = "Blind Willie Johnson" + title = "It's nobody's fault but mine" + + # Act + search = self.network.search_for_track(artist, title) + total = search.get_total_result_count() + + # Assert + self.skip_if_lastfm_api_broken(total) + self.assertGreaterEqual(int(total), 0) + + def test_enable_rate_limiting(self): + # Arrange + self.assertFalse(self.network.is_rate_limited()) + + # Act + self.network.enable_rate_limit() + then = time.time() + # Make some network call, limit not applied first time + self.network.get_user(self.username) + # Make a second network call, limiting should be applied + self.network.get_top_artists() + now = time.time() + + # Assert + self.assertTrue(self.network.is_rate_limited()) + self.assertGreaterEqual(now - then, 0.2) + + def test_disable_rate_limiting(self): + # Arrange + self.network.enable_rate_limit() + self.assertTrue(self.network.is_rate_limited()) + + # Act + self.network.disable_rate_limit() + # Make some network call, limit not applied first time + self.network.get_user(self.username) + # Make a second network call, limiting should be applied + self.network.get_top_artists() + + # Assert + self.assertFalse(self.network.is_rate_limited()) + + def test_lastfm_network_name(self): + # Act + name = str(self.network) + + # Assert + self.assertEqual(name, "Last.fm Network") + + def test_geo_get_top_artists(self): + # Arrange + # Act + artists = self.network.get_geo_top_artists( + country="United Kingdom", limit=1) + + # Assert + self.assertEqual(len(artists), 1) + self.assertIsInstance(artists[0], pylast.TopItem) + self.assertIsInstance(artists[0].item, pylast.Artist) + + def test_geo_get_top_tracks(self): + # Arrange + # Act + tracks = self.network.get_geo_top_tracks( + country="United Kingdom", location="Manchester", limit=1) + + # Assert + self.assertEqual(len(tracks), 1) + self.assertIsInstance(tracks[0], pylast.TopItem) + self.assertIsInstance(tracks[0].item, pylast.Track) + + def test_network_get_top_artists_with_limit(self): + # Arrange + # Act + artists = self.network.get_top_artists(limit=1) + + # Assert + self.helper_only_one_thing_in_top_list(artists, pylast.Artist) + + def test_network_get_top_tags_with_limit(self): + # Arrange + # Act + tags = self.network.get_top_tags(limit=1) + + # Assert + self.helper_only_one_thing_in_top_list(tags, pylast.Tag) + + def test_network_get_top_tags_with_no_limit(self): + # Arrange + # Act + tags = self.network.get_top_tags() + + # Assert + self.helper_at_least_one_thing_in_top_list(tags, pylast.Tag) + + def test_network_get_top_tracks_with_limit(self): + # Arrange + # Act + tracks = self.network.get_top_tracks(limit=1) + + # Assert + self.helper_only_one_thing_in_top_list(tracks, pylast.Track) + + def test_country_top_tracks(self): + # Arrange + country = self.network.get_country("Croatia") + + # Act + things = country.get_top_tracks(limit=2) + + # Assert + self.helper_two_different_things_in_top_list(things, pylast.Track) + + def test_country_network_top_tracks(self): + # Arrange + # Act + things = self.network.get_geo_top_tracks("Croatia", limit=2) + + # Assert + self.helper_two_different_things_in_top_list(things, pylast.Track) + + def test_tag_top_tracks(self): + # Arrange + tag = self.network.get_tag("blues") + + # Act + things = tag.get_top_tracks(limit=2) + + # Assert + self.helper_two_different_things_in_top_list(things, pylast.Track) + + def test_album_data(self): + # Arrange + thing = self.network.get_album("Test Artist", "Test Album") + + # Act + stringed = str(thing) + repr = thing.__repr__() + title = thing.get_title() + name = thing.get_name() + playcount = thing.get_playcount() + url = thing.get_url() + + # Assert + self.assertEqual(stringed, "Test Artist - Test Album") + self.assertIn("pylast.Album('Test Artist', 'Test Album',", repr) + self.assertEqual(title, name) + self.assertIsInstance(playcount, int) + self.assertGreater(playcount, 1) + self.assertEqual( + "https://www.last.fm/music/test%2bartist/test%2balbum", url) + + def test_track_data(self): + # Arrange + thing = self.network.get_track("Test Artist", "test title") + + # Act + stringed = str(thing) + repr = thing.__repr__() + title = thing.get_title() + name = thing.get_name() + playcount = thing.get_playcount() + url = thing.get_url(pylast.DOMAIN_FRENCH) + + # Assert + self.assertEqual(stringed, "Test Artist - test title") + self.assertIn("pylast.Track('Test Artist', 'test title',", repr) + self.assertEqual(title, "test title") + self.assertEqual(title, name) + self.assertIsInstance(playcount, int) + self.assertGreater(playcount, 1) + self.assertEqual( + "https://www.last.fm/fr/music/test%2bartist/_/test%2btitle", url) + + def test_country_top_artists(self): + # Arrange + country = self.network.get_country("Ukraine") + + # Act + artists = country.get_top_artists(limit=1) + + # Assert + self.helper_only_one_thing_in_top_list(artists, pylast.Artist) + + def test_caching(self): + # Arrange + user = self.network.get_user("RJ") + + # Act + self.network.enable_caching() + tags1 = user.get_top_tags(limit=1, cacheable=True) + tags2 = user.get_top_tags(limit=1, cacheable=True) + + # Assert + self.assertTrue(self.network.is_caching_enabled()) + self.assertEqual(tags1, tags2) + self.network.disable_caching() + self.assertFalse(self.network.is_caching_enabled()) + + def test_album_mbid(self): + # Arrange + mbid = "a6a265bf-9f81-4055-8224-f7ac0aa6b937" + + # Act + album = self.network.get_album_by_mbid(mbid) + album_mbid = album.get_mbid() + + # Assert + self.assertIsInstance(album, pylast.Album) + self.assertEqual(album.title.lower(), "test") + self.assertEqual(album_mbid, mbid) + + def test_artist_mbid(self): + # Arrange + mbid = "7e84f845-ac16-41fe-9ff8-df12eb32af55" + + # Act + artist = self.network.get_artist_by_mbid(mbid) + + # Assert + self.assertIsInstance(artist, pylast.Artist) + self.assertEqual(artist.name, "MusicBrainz Test Artist") + + def test_track_mbid(self): + # Arrange + mbid = "ebc037b1-cc9c-44f2-a21f-83c219f0e1e0" + + # Act + track = self.network.get_track_by_mbid(mbid) + track_mbid = track.get_mbid() + + # Assert + self.assertIsInstance(track, pylast.Track) + self.assertEqual(track.title, "first") + self.assertEqual(track_mbid, mbid) + + def test_init_with_token(self): + # Arrange/Act + try: + pylast.LastFMNetwork( + api_key=self.__class__.secrets["api_key"], + api_secret=self.__class__.secrets["api_secret"], + token="invalid", + ) + except pylast.WSError as exc: + msg = str(exc) + + # Assert + self.assertEqual(msg, + "Unauthorized Token - This token has not been issued") + + +class TestPyLastTag(PyLastTestCase): + + def test_tag_is_hashable(self): + # Arrange + tag = self.network.get_top_tags(limit=1)[0] + + # Act/Assert + self.helper_is_thing_hashable(tag) + + def test_tag_top_artists(self): + # Arrange + tag = self.network.get_tag("blues") + + # Act + artists = tag.get_top_artists(limit=1) + + # Assert + self.helper_only_one_thing_in_top_list(artists, pylast.Artist) + + def test_tag_top_albums(self): + # Arrange + tag = self.network.get_tag("blues") + + # Act + albums = tag.get_top_albums(limit=1) + + # Assert + self.helper_only_one_thing_in_top_list(albums, pylast.Album) + + def test_tags(self): + # Arrange + tag1 = self.network.get_tag("blues") + tag2 = self.network.get_tag("rock") + + # Act + tag_repr = repr(tag1) + tag_str = str(tag1) + name = tag1.get_name(properly_capitalized=True) + url = tag1.get_url() + + # Assert + self.assertEqual("blues", tag_str) + self.assertIn("pylast.Tag", tag_repr) + self.assertIn("blues", tag_repr) + self.assertEqual("blues", name) + self.assertTrue(tag1 == tag1) + self.assertTrue(tag1 != tag2) + self.assertEqual(url, "https://www.last.fm/tag/blues") + + def test_tags_similar(self): + # Arrange + tag = self.network.get_tag("blues") + + # Act + similar = tag.get_similar() + + # Assert + self.skip_if_lastfm_api_broken(similar) + found = False + for tag in similar: + if tag.name == "delta blues": + found = True + break + self.assertTrue(found) + + +class TestPyLastTrack(PyLastTestCase): + + def test_love(self): + # Arrange + artist = "Test Artist" + title = "test title" + track = self.network.get_track(artist, title) + lastfm_user = self.network.get_user(self.username) + + # Act + track.love() + + # Assert + loved = lastfm_user.get_loved_tracks(limit=1) + self.assertEqual(str(loved[0].track.artist), "Test Artist") + self.assertEqual(str(loved[0].track.title), "test title") + + def test_unlove(self): + # Arrange + artist = pylast.Artist("Test Artist", self.network) + title = "test title" + track = pylast.Track(artist, title, self.network) + lastfm_user = self.network.get_user(self.username) + track.love() + + # Act + track.unlove() + + # Assert + loved = lastfm_user.get_loved_tracks(limit=1) + if len(loved): # OK to be empty but if not: + self.assertNotEqual(str(loved.track.artist), "Test Artist") + self.assertNotEqual(str(loved.track.title), "test title") + + def test_user_play_count_in_track_info(self): + # Arrange + artist = "Test Artist" + title = "test title" + track = pylast.Track( + artist=artist, title=title, + network=self.network, username=self.username) + + # Act + count = track.get_userplaycount() + + # Assert + self.assertGreaterEqual(count, 0) + + def test_user_loved_in_track_info(self): + # Arrange + artist = "Test Artist" + title = "test title" + track = pylast.Track( + artist=artist, title=title, + network=self.network, username=self.username) + + # Act + loved = track.get_userloved() + + # Assert + self.assertIsNotNone(loved) + self.assertIsInstance(loved, bool) + self.assertNotIsInstance(loved, str) + + def test_track_is_hashable(self): + # Arrange + artist = self.network.get_artist("Test Artist") + track = artist.get_top_tracks()[0].item + self.assertIsInstance(track, pylast.Track) + + # Act/Assert + self.helper_is_thing_hashable(track) + + def test_track_wiki_content(self): + # Arrange + track = pylast.Track("Test Artist", "test title", self.network) + + # Act + wiki = track.get_wiki_content() + + # Assert + self.assertIsNotNone(wiki) + self.assertGreaterEqual(len(wiki), 1) + + def test_track_wiki_summary(self): + # Arrange + track = pylast.Track("Test Artist", "test title", self.network) + + # Act + wiki = track.get_wiki_summary() + + # Assert + self.assertIsNotNone(wiki) + self.assertGreaterEqual(len(wiki), 1) + + +class TestPyLastUser(PyLastTestCase): + + def test_get_user_registration(self): + # Arrange + username = "RJ" + user = self.network.get_user(username) + + # Act + registered = user.get_registered() + + # Assert + # Last.fm API broken? Should be yyyy-mm-dd not Unix timestamp + if int(registered): + pytest.skip("Last.fm API is broken.") + + # Just check date because of timezones + self.assertIn(u"2002-11-20 ", registered) + + def test_get_user_unixtime_registration(self): + # Arrange + username = "RJ" + user = self.network.get_user(username) + + # Act + unixtime_registered = user.get_unixtime_registered() + + # Assert + # Just check date because of timezones + self.assertEqual(unixtime_registered, u"1037793040") + + def test_get_genderless_user(self): + # Arrange + # Currently test_user has no gender set: + lastfm_user = self.network.get_user("test_user") + + # Act + gender = lastfm_user.get_gender() + + # Assert + self.assertIsNone(gender) + + def test_get_countryless_user(self): + # Arrange + # Currently test_user has no country set: + lastfm_user = self.network.get_user("test_user") + + # Act + country = lastfm_user.get_country() + + # Assert + self.assertIsNone(country) + + def test_user_equals_none(self): + # Arrange + lastfm_user = self.network.get_user(self.username) + + # Act + value = (lastfm_user is None) + + # Assert + self.assertFalse(value) + + def test_user_not_equal_to_none(self): + # Arrange + lastfm_user = self.network.get_user(self.username) + + # Act + value = (lastfm_user is not None) + + # Assert + self.assertTrue(value) + + def test_now_playing_user_with_no_scrobbles(self): + # Arrange + # Currently test-account has no scrobbles: + user = self.network.get_user('test-account') + + # Act + current_track = user.get_now_playing() + + # Assert + self.assertIsNone(current_track) + + def test_love_limits(self): + # Arrange + # Currently test-account has at least 23 loved tracks: + user = self.network.get_user("test-user") + + # Act/Assert + self.assertEqual(len(user.get_loved_tracks(limit=20)), 20) + self.assertLessEqual(len(user.get_loved_tracks(limit=100)), 100) + self.assertGreaterEqual(len(user.get_loved_tracks(limit=None)), 23) + self.assertGreaterEqual(len(user.get_loved_tracks(limit=0)), 23) + + def test_user_is_hashable(self): + # Arrange + user = self.network.get_user(self.username) + + # Act/Assert + self.helper_is_thing_hashable(user) + + # Commented out because (a) it'll take a long time and (b) it strangely + # fails due Last.fm's complaining of hitting the rate limit, even when + # limited to one call per second. The ToS allows 5 calls per second. + # def test_get_all_scrobbles(self): + # # Arrange + # lastfm_user = self.network.get_user("RJ") + # self.network.enable_rate_limit() # this is going to be slow... + + # # Act + # tracks = lastfm_user.get_recent_tracks(limit=None) + + # # Assert + # self.assertGreaterEqual(len(tracks), 0) + + def test_pickle(self): + # Arrange + import pickle + lastfm_user = self.network.get_user(self.username) + filename = str(self.unix_timestamp()) + ".pkl" + + # Act + with open(filename, "wb") as f: + pickle.dump(lastfm_user, f) + with open(filename, "rb") as f: + loaded_user = pickle.load(f) + os.remove(filename) + + # Assert + self.assertEqual(lastfm_user, loaded_user) + + def test_cacheable_user_artist_tracks(self): + # Arrange + lastfm_user = self.network.get_authenticated_user() + + # Act + result1 = lastfm_user.get_artist_tracks("Test Artist", cacheable=False) + result2 = lastfm_user.get_artist_tracks("Test Artist", cacheable=True) + result3 = lastfm_user.get_artist_tracks("Test Artist") + + # Assert + self.helper_validate_results(result1, result2, result3) + + def test_cacheable_user(self): + # Arrange + lastfm_user = self.network.get_authenticated_user() + + # Act/Assert + self.helper_validate_cacheable(lastfm_user, "get_friends") + self.helper_validate_cacheable(lastfm_user, "get_loved_tracks") + self.helper_validate_cacheable(lastfm_user, "get_recent_tracks") + + def test_user_get_top_tags_with_limit(self): + # Arrange + user = self.network.get_user("RJ") + + # Act + tags = user.get_top_tags(limit=1) + + # Assert + self.skip_if_lastfm_api_broken(tags) + self.helper_only_one_thing_in_top_list(tags, pylast.Tag) + + def test_user_top_tracks(self): + # Arrange + lastfm_user = self.network.get_user(self.username) + + # Act + things = lastfm_user.get_top_tracks(limit=2) + + # Assert + self.helper_two_different_things_in_top_list(things, pylast.Track) + + def helper_assert_chart(self, chart, expected_type): + # Assert + self.assertIsNotNone(chart) + self.assertGreater(len(chart), 0) + self.assertIsInstance(chart[0], pylast.TopItem) + self.assertIsInstance(chart[0].item, expected_type) + + def helper_get_assert_charts(self, thing, date): + # Arrange + (from_date, to_date) = date + + # Act + artist_chart = thing.get_weekly_artist_charts(from_date, to_date) + if type(thing) is not pylast.Tag: + album_chart = thing.get_weekly_album_charts(from_date, to_date) + track_chart = thing.get_weekly_track_charts(from_date, to_date) + + # Assert + self.helper_assert_chart(artist_chart, pylast.Artist) + if type(thing) is not pylast.Tag: + self.helper_assert_chart(album_chart, pylast.Album) + self.helper_assert_chart(track_chart, pylast.Track) + + def helper_dates_valid(self, dates): + # Assert + self.assertGreaterEqual(len(dates), 1) + self.assertIsInstance(dates[0], tuple) + (start, end) = dates[0] + self.assertLess(start, end) + + def test_user_charts(self): + # Arrange + lastfm_user = self.network.get_user("RJ") + dates = lastfm_user.get_weekly_chart_dates() + self.helper_dates_valid(dates) + + # Act/Assert + self.helper_get_assert_charts(lastfm_user, dates[0]) + + # Commented out to avoid spamming + # def test_share_spam(self): + # # Arrange + # users_to_spam = [TODO_ENTER_SPAMEES_HERE] + # spam_message = "Dig the krazee sound!" + # artist = self.network.get_top_artists(limit=1)[0].item + # track = artist.get_top_tracks(limit=1)[0].item + + # # Act + # artist.share(users_to_spam, spam_message) + # track.share(users_to_spam, spam_message) + + # Assert + # Check inbox for spam! + + # album/artist/track/user + + def test_user_top_artists(self): + # Arrange + lastfm_user = self.network.get_user(self.username) + + # Act + artists = lastfm_user.get_top_artists(limit=1) + + # Assert + self.helper_only_one_thing_in_top_list(artists, pylast.Artist) + + def test_user_top_albums(self): + # Arrange + user = self.network.get_user("RJ") + + # Act + albums = user.get_top_albums(limit=1) + + # Assert + self.helper_only_one_thing_in_top_list(albums, pylast.Album) + + def test_user_tagged_artists(self): + # Arrange + lastfm_user = self.network.get_user(self.username) + tags = ["artisttagola"] + artist = self.network.get_artist("Test Artist") + artist.add_tags(tags) + + # Act + artists = lastfm_user.get_tagged_artists('artisttagola', limit=1) + + # Assert + self.helper_only_one_thing_in_list(artists, pylast.Artist) + + def test_user_tagged_albums(self): + # Arrange + lastfm_user = self.network.get_user(self.username) + tags = ["albumtagola"] + album = self.network.get_album("Test Artist", "Test Album") + album.add_tags(tags) + + # Act + albums = lastfm_user.get_tagged_albums('albumtagola', limit=1) + + # Assert + self.helper_only_one_thing_in_list(albums, pylast.Album) + + def test_user_tagged_tracks(self): + # Arrange + lastfm_user = self.network.get_user(self.username) + tags = ["tracktagola"] + track = self.network.get_track("Test Artist", "test title") + track.add_tags(tags) + + # Act + tracks = lastfm_user.get_tagged_tracks('tracktagola', limit=1) + + # Assert + self.helper_only_one_thing_in_list(tracks, pylast.Track) + + def test_user_subscriber(self): + # Arrange + subscriber = self.network.get_user("RJ") + non_subscriber = self.network.get_user("Test User") + + # Act + subscriber_is_subscriber = subscriber.is_subscriber() + non_subscriber_is_subscriber = non_subscriber.is_subscriber() + + # Assert + self.assertTrue(subscriber_is_subscriber) + self.assertFalse(non_subscriber_is_subscriber) + + def test_user_get_image(self): + # Arrange + user = self.network.get_user("RJ") + + # Act + url = user.get_image() + + # Assert + self.assertTrue(url.startswith("https://")) + + def test_user_get_library(self): + # Arrange + user = self.network.get_user(self.username) + + # Act + library = user.get_library() + + # Assert + self.assertIsInstance(library, pylast.Library) + + def test_get_recent_tracks_from_to(self): + # Arrange + lastfm_user = self.network.get_user("RJ") + + from datetime import datetime + start = datetime(2011, 7, 21, 15, 10) + end = datetime(2011, 7, 21, 15, 15) + import calendar + utc_start = calendar.timegm(start.utctimetuple()) + utc_end = calendar.timegm(end.utctimetuple()) + + # Act + tracks = lastfm_user.get_recent_tracks(time_from=utc_start, + time_to=utc_end) + + # Assert + self.assertEqual(len(tracks), 1) + self.assertEqual(str(tracks[0].track.artist), "Johnny Cash") + self.assertEqual(str(tracks[0].track.title), "Ring of Fire") + def test_tracks_notequal(self): # Arrange track1 = pylast.Track("Test Artist", "test title", self.network) @@ -1174,83 +1358,6 @@ class TestPyLast(PyLastTestCase): self.assertEqual(len(tracks), 4) self.assertTrue(url.startswith("https://www.last.fm/music/test")) - def test_tags(self): - # Arrange - tag1 = self.network.get_tag("blues") - tag2 = self.network.get_tag("rock") - - # Act - tag_repr = repr(tag1) - tag_str = str(tag1) - name = tag1.get_name(properly_capitalized=True) - url = tag1.get_url() - - # Assert - self.assertEqual("blues", tag_str) - self.assertIn("pylast.Tag", tag_repr) - self.assertIn("blues", tag_repr) - self.assertEqual("blues", name) - self.assertTrue(tag1 == tag1) - self.assertTrue(tag1 != tag2) - self.assertEqual(url, "https://www.last.fm/tag/blues") - - def test_tags_similar(self): - # Arrange - tag = self.network.get_tag("blues") - - # Act - similar = tag.get_similar() - - # Assert - self.skip_if_lastfm_api_broken(similar) - found = False - for tag in similar: - if tag.name == "delta blues": - found = True - break - self.assertTrue(found) - - def test_artists(self): - # Arrange - artist1 = self.network.get_artist("Radiohead") - artist2 = self.network.get_artist("Portishead") - - # Act - url = artist1.get_url() - mbid = artist1.get_mbid() - image = artist1.get_cover_image() - playcount = artist1.get_playcount() - streamable = artist1.is_streamable() - name = artist1.get_name(properly_capitalized=False) - name_cap = artist1.get_name(properly_capitalized=True) - - # Assert - self.assertIn("http", image) - self.assertGreater(playcount, 1) - self.assertTrue(artist1 != artist2) - self.assertEqual(name.lower(), name_cap.lower()) - self.assertEqual(url, "https://www.last.fm/music/radiohead") - self.assertEqual(mbid, "a74b1b7f-71a5-4011-9441-d0b5e4122711") - self.assertIsInstance(streamable, bool) - - def test_countries(self): - # Arrange - country1 = pylast.Country("Italy", self.network) - country2 = pylast.Country("Finland", self.network) - - # Act - text = str(country1) - rep = repr(country1) - url = country1.get_url() - - # Assert - self.assertIn("Italy", rep) - self.assertIn("pylast.Country", rep) - self.assertEqual(text, "Italy") - self.assertTrue(country1 == country1) - self.assertTrue(country1 != country2) - self.assertEqual(url, "https://www.last.fm/place/italy") - def test_track_eq_none_is_false(self): # Arrange track1 = None @@ -1267,89 +1374,6 @@ class TestPyLast(PyLastTestCase): # Act / Assert self.assertTrue(track1 != track2) - def test_artist_eq_none_is_false(self): - # Arrange - artist1 = None - artist2 = pylast.Artist("Test Artist", self.network) - - # Act / Assert - self.assertFalse(artist1 == artist2) - - def test_artist_ne_none_is_true(self): - # Arrange - artist1 = None - artist2 = pylast.Artist("Test Artist", self.network) - - # Act / Assert - self.assertTrue(artist1 != artist2) - - def test_album_eq_none_is_false(self): - # Arrange - album1 = None - album2 = pylast.Album("Test Artist", "Test Album", self.network) - - # Act / Assert - self.assertFalse(album1 == album2) - - def test_album_ne_none_is_true(self): - # Arrange - album1 = None - album2 = pylast.Album("Test Artist", "Test Album", self.network) - - # Act / Assert - self.assertTrue(album1 != album2) - - def test_band_members(self): - # Arrange - artist = pylast.Artist("The Beatles", self.network) - - # Act - band_members = artist.get_band_members() - - # Assert - self.skip_if_lastfm_api_broken(band_members) - self.assertGreaterEqual(len(band_members), 4) - - def test_no_band_members(self): - # Arrange - artist = pylast.Artist("John Lennon", self.network) - - # Act - band_members = artist.get_band_members() - - # Assert - self.assertIsNone(band_members) - - def test_get_recent_tracks_from_to(self): - # Arrange - lastfm_user = self.network.get_user("RJ") - - from datetime import datetime - start = datetime(2011, 7, 21, 15, 10) - end = datetime(2011, 7, 21, 15, 15) - import calendar - utc_start = calendar.timegm(start.utctimetuple()) - utc_end = calendar.timegm(end.utctimetuple()) - - # Act - tracks = lastfm_user.get_recent_tracks(time_from=utc_start, - time_to=utc_end) - - # Assert - self.assertEqual(len(tracks), 1) - self.assertEqual(str(tracks[0].track.artist), "Johnny Cash") - self.assertEqual(str(tracks[0].track.title), "Ring of Fire") - - def test_artist_get_correction(self): - # Arrange - artist = pylast.Artist("guns and roses", self.network) - - # Act - corrected_artist_name = artist.get_correction() - - # Assert - self.assertEqual(corrected_artist_name, "Guns N' Roses") - def test_track_get_correction(self): # Arrange track = pylast.Track("Guns N' Roses", "mrbrownstone", self.network) @@ -1370,21 +1394,6 @@ class TestPyLast(PyLastTestCase): # Assert self.assertEqual(mbid, None) - def test_init_with_token(self): - # Arrange/Act - try: - pylast.LastFMNetwork( - api_key=self.__class__.secrets["api_key"], - api_secret=self.__class__.secrets["api_secret"], - token="invalid", - ) - except pylast.WSError as exc: - msg = str(exc) - - # Assert - self.assertEqual(msg, - "Unauthorized Token - This token has not been issued") - @flaky(max_runs=5, min_passes=1) class TestPyLastWithLibreFm(unittest.TestCase):