mirror of
https://github.com/spotipy-dev/spotipy.git
synced 2026-06-19 01:03:53 +00:00
Changed cache_handler.py to utilize Python's Context Management Protocol (#991)
This commit is contained in:
parent
d319c6e09f
commit
1dbbbf65ec
1
.github/workflows/integration_tests.yml
vendored
1
.github/workflows/integration_tests.yml
vendored
@ -4,7 +4,6 @@ on: [push, pull_request]
|
|||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
build:
|
build:
|
||||||
|
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
env:
|
env:
|
||||||
SPOTIPY_CLIENT_ID: ${{ secrets.SPOTIPY_CLIENT_ID }}
|
SPOTIPY_CLIENT_ID: ${{ secrets.SPOTIPY_CLIENT_ID }}
|
||||||
|
|||||||
1
.github/workflows/unit_tests.yml
vendored
1
.github/workflows/unit_tests.yml
vendored
@ -4,7 +4,6 @@ on: [push, pull_request]
|
|||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
build:
|
build:
|
||||||
|
|
||||||
runs-on: ubuntu-20.04
|
runs-on: ubuntu-20.04
|
||||||
strategy:
|
strategy:
|
||||||
matrix:
|
matrix:
|
||||||
|
|||||||
23
CHANGELOG.md
23
CHANGELOG.md
@ -6,22 +6,31 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/),
|
|||||||
and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
|
and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
|
||||||
|
|
||||||
## Unreleased
|
## Unreleased
|
||||||
|
|
||||||
Add your changes below.
|
Add your changes below.
|
||||||
|
|
||||||
### Added
|
### Added
|
||||||
|
|
||||||
- Added examples for audiobooks, shows and episodes methods to examples directory
|
- Added examples for audiobooks, shows and episodes methods to examples directory
|
||||||
- Use newer string formatters (https://pyformat.info)
|
- Use newer string formatters (https://pyformat.info)
|
||||||
- Marked `recommendation_genre_seeds` as deprecated
|
- Marked `recommendation_genre_seeds` as deprecated
|
||||||
|
|
||||||
### Fixed
|
### Fixed
|
||||||
|
|
||||||
- Fixed scripts in examples directory that didn't run correctly
|
- Fixed scripts in examples directory that didn't run correctly
|
||||||
- Updated documentation for `Client.current_user_top_artists` to indicate maximum number of artists limit
|
- Updated documentation for `Client.current_user_top_artists` to indicate maximum number of artists limit
|
||||||
|
|
||||||
|
### Changed
|
||||||
|
|
||||||
|
- Updated get_cached_token and save_token_to_cache methods to utilize Python's Context Management Protocol
|
||||||
|
- Added except clause to get_cached_token method to handle json decode errors
|
||||||
|
|
||||||
### Removed
|
### Removed
|
||||||
|
|
||||||
## [2.25.0] - 2025-03-01
|
## [2.25.0] - 2025-03-01
|
||||||
|
|
||||||
### Added
|
### Added
|
||||||
|
|
||||||
- Added unit tests for queue functions
|
- Added unit tests for queue functions
|
||||||
- Added detailed function docstrings to 'util.py', including descriptions and special sections that lists arguments, returns, and raises.
|
- Added detailed function docstrings to 'util.py', including descriptions and special sections that lists arguments, returns, and raises.
|
||||||
- Updated order of instructions for Python and pip package manager installation in TUTORIAL.md
|
- Updated order of instructions for Python and pip package manager installation in TUTORIAL.md
|
||||||
@ -44,44 +53,57 @@ Add your changes below.
|
|||||||
### Changed
|
### Changed
|
||||||
|
|
||||||
- Split test and lint workflows
|
- Split test and lint workflows
|
||||||
|
- Updated get_cached_token and save_token_to_cache methods to utilize Python's Context Management Protocol
|
||||||
|
- Added except clause to get_cached_token method to handle json decode errors
|
||||||
|
|
||||||
### Fixed
|
### Fixed
|
||||||
|
|
||||||
- Audiobook integration tests
|
- Audiobook integration tests
|
||||||
- Edited docstrings for certain functions in client.py for functions that are no longer in use and have been replaced.
|
- Edited docstrings for certain functions in client.py for functions that are no longer in use and have been replaced.
|
||||||
- `current_user_unfollow_playlist()` now supports playlist IDs, URLs, and URIs rather than previously where it only supported playlist IDs.
|
- `current_user_unfollow_playlist()` now supports playlist IDs, URLs, and URIs rather than previously where it only supported playlist IDs.
|
||||||
|
|
||||||
### Removed
|
### Removed
|
||||||
|
|
||||||
- `mock` no longer listed as a test dependency. Only built-in `unittest.mock` is actually used.
|
- `mock` no longer listed as a test dependency. Only built-in `unittest.mock` is actually used.
|
||||||
|
|
||||||
## [2.24.0] - 2024-05-30
|
## [2.24.0] - 2024-05-30
|
||||||
|
|
||||||
### Added
|
### Added
|
||||||
|
|
||||||
- Added `MemcacheCacheHandler`, a cache handler that stores the token info using pymemcache.
|
- Added `MemcacheCacheHandler`, a cache handler that stores the token info using pymemcache.
|
||||||
- Added support for audiobook endpoints: `get_audiobook`, `get_audiobooks`, and `get_audiobook_chapters`.
|
- Added support for audiobook endpoints: `get_audiobook`, `get_audiobooks`, and `get_audiobook_chapters`.
|
||||||
- Added integration tests for audiobook endpoints.
|
- Added integration tests for audiobook endpoints.
|
||||||
- Added `update` field to `current_user_follow_playlist`.
|
- Added `update` field to `current_user_follow_playlist`.
|
||||||
|
|
||||||
### Changed
|
### Changed
|
||||||
|
|
||||||
|
- Updated get_cached_token and save_token_to_cache methods to utilize Python's Context Management Protocol
|
||||||
|
- Added except clause to get_cached_token method to handle json decode errors
|
||||||
|
|
||||||
- Fixed error obfuscation when Spotify class is being inherited and an error is raised in the Child's `__init__`
|
- Fixed error obfuscation when Spotify class is being inherited and an error is raised in the Child's `__init__`
|
||||||
- Replaced `artist_albums(album_type=...)` with `artist_albums(include_groups=...)` due to an API change.
|
- Replaced `artist_albums(album_type=...)` with `artist_albums(include_groups=...)` due to an API change.
|
||||||
- Updated `_regex_spotify_url` to ignore `/intl-<countrycode>` in Spotify links
|
- Updated `_regex_spotify_url` to ignore `/intl-<countrycode>` in Spotify links
|
||||||
- Improved README, docs and examples
|
- Improved README, docs and examples
|
||||||
|
|
||||||
### Fixed
|
### Fixed
|
||||||
|
|
||||||
- Readthedocs build
|
- Readthedocs build
|
||||||
- Split `test_current_user_save_and_usave_tracks` unit test
|
- Split `test_current_user_save_and_usave_tracks` unit test
|
||||||
|
|
||||||
### Removed
|
### Removed
|
||||||
|
|
||||||
- Drop support for EOL Python 3.7
|
- Drop support for EOL Python 3.7
|
||||||
|
|
||||||
## [2.23.0] - 2023-04-07
|
## [2.23.0] - 2023-04-07
|
||||||
|
|
||||||
### Added
|
### Added
|
||||||
|
|
||||||
- Added optional `encoder_cls` argument to `CacheFileHandler`, which overwrite default encoder for token before writing to disk
|
- Added optional `encoder_cls` argument to `CacheFileHandler`, which overwrite default encoder for token before writing to disk
|
||||||
- Integration tests for searching multiple types in multiple markets (non-user endpoints)
|
- Integration tests for searching multiple types in multiple markets (non-user endpoints)
|
||||||
- Publish to PyPI action
|
- Publish to PyPI action
|
||||||
|
|
||||||
### Fixed
|
### Fixed
|
||||||
|
|
||||||
- Fixed the regex for matching playlist URIs with the format spotify:user:USERNAME:playlist:PLAYLISTID.
|
- Fixed the regex for matching playlist URIs with the format spotify:user:USERNAME:playlist:PLAYLISTID.
|
||||||
- `search_markets` now factors the counts of all types in the `total` rather than just the first type ([#534](https://github.com/spotipy-dev/spotipy/issues/534))
|
- `search_markets` now factors the counts of all types in the `total` rather than just the first type ([#534](https://github.com/spotipy-dev/spotipy/issues/534))
|
||||||
|
|
||||||
@ -466,6 +488,7 @@ Add your changes below.
|
|||||||
### Changed
|
### Changed
|
||||||
|
|
||||||
- Made instructions in the CONTRIBUTING.md file more clear such that it is easier to onboard and there are no conflicts with TUTORIAL.md
|
- Made instructions in the CONTRIBUTING.md file more clear such that it is easier to onboard and there are no conflicts with TUTORIAL.md
|
||||||
|
|
||||||
## [2.5.0] - 2020-01-11
|
## [2.5.0] - 2020-01-11
|
||||||
|
|
||||||
Added follow and player endpoints
|
Added follow and player endpoints
|
||||||
|
|||||||
1
examples
1
examples
@ -1 +0,0 @@
|
|||||||
Subproject commit c610a79705ef4aa55e4d61572a012f77b6f7245d
|
|
||||||
@ -41,7 +41,6 @@ class CacheHandler():
|
|||||||
Save a token_info dictionary object to the cache and return None.
|
Save a token_info dictionary object to the cache and return None.
|
||||||
"""
|
"""
|
||||||
raise NotImplementedError()
|
raise NotImplementedError()
|
||||||
return None
|
|
||||||
|
|
||||||
|
|
||||||
class CacheFileHandler(CacheHandler):
|
class CacheFileHandler(CacheHandler):
|
||||||
@ -77,24 +76,24 @@ class CacheFileHandler(CacheHandler):
|
|||||||
token_info = None
|
token_info = None
|
||||||
|
|
||||||
try:
|
try:
|
||||||
f = open(self.cache_path)
|
with open(self.cache_path, encoding='utf-8') as f:
|
||||||
token_info_string = f.read()
|
token_info_string = f.read()
|
||||||
f.close()
|
|
||||||
token_info = json.loads(token_info_string)
|
token_info = json.loads(token_info_string)
|
||||||
|
|
||||||
except OSError as error:
|
except OSError as error:
|
||||||
if error.errno == errno.ENOENT:
|
if error.errno == errno.ENOENT:
|
||||||
logger.debug(f"cache does not exist at: {self.cache_path}")
|
logger.debug(f"cache does not exist at: {self.cache_path}")
|
||||||
else:
|
else:
|
||||||
logger.warning(f"Couldn't read cache at: {self.cache_path}")
|
logger.warning("Couldn't read cache at: %s", self.cache_path)
|
||||||
|
except json.JSONDecodeError:
|
||||||
|
logger.warning("Couldn't decode JSON from cache at: %s", self.cache_path)
|
||||||
|
|
||||||
return token_info
|
return token_info
|
||||||
|
|
||||||
def save_token_to_cache(self, token_info):
|
def save_token_to_cache(self, token_info):
|
||||||
try:
|
try:
|
||||||
f = open(self.cache_path, "w")
|
with open(self.cache_path, "w", encoding='utf-8') as f:
|
||||||
f.write(json.dumps(token_info, cls=self.encoder_cls))
|
f.write(json.dumps(token_info, cls=self.encoder_cls))
|
||||||
f.close()
|
|
||||||
except OSError:
|
except OSError:
|
||||||
logger.warning(f"Couldn't write token to cache at: {self.cache_path}")
|
logger.warning(f"Couldn't write token to cache at: {self.cache_path}")
|
||||||
|
|
||||||
@ -214,6 +213,7 @@ class RedisCacheHandler(CacheHandler):
|
|||||||
class MemcacheCacheHandler(CacheHandler):
|
class MemcacheCacheHandler(CacheHandler):
|
||||||
"""A Cache handler that stores the token info in Memcache using the pymemcache client
|
"""A Cache handler that stores the token info in Memcache using the pymemcache client
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def __init__(self, memcache, key=None) -> None:
|
def __init__(self, memcache, key=None) -> None:
|
||||||
"""
|
"""
|
||||||
Parameters:
|
Parameters:
|
||||||
|
|||||||
@ -153,6 +153,7 @@ class Retry(urllib3.Retry):
|
|||||||
"""
|
"""
|
||||||
Custom class for printing a warning when a rate/request limit is reached.
|
Custom class for printing a warning when a rate/request limit is reached.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def increment(
|
def increment(
|
||||||
self,
|
self,
|
||||||
method: str | None = None,
|
method: str | None = None,
|
||||||
|
|||||||
@ -52,18 +52,21 @@ class OAuthCacheTest(unittest.TestCase):
|
|||||||
@patch('spotipy.cache_handler.open', create=True)
|
@patch('spotipy.cache_handler.open', create=True)
|
||||||
def test_gets_from_cache_path(self, opener,
|
def test_gets_from_cache_path(self, opener,
|
||||||
is_token_expired, refresh_access_token):
|
is_token_expired, refresh_access_token):
|
||||||
|
"""Test that the token is retrieved from the cache path."""
|
||||||
scope = "playlist-modify-private"
|
scope = "playlist-modify-private"
|
||||||
path = ".cache-username"
|
path = ".cache-username"
|
||||||
tok = _make_fake_token(1, 1, scope)
|
tok = _make_fake_token(1, 1, scope)
|
||||||
|
token_file = _token_file(json.dumps(tok, ensure_ascii=False))
|
||||||
opener.return_value = _token_file(json.dumps(tok, ensure_ascii=False))
|
opener.return_value = token_file
|
||||||
|
opener.return_value.__enter__ = mock.Mock(return_value=token_file)
|
||||||
|
opener.return_value.__exit__ = mock.Mock(return_value=False)
|
||||||
is_token_expired.return_value = False
|
is_token_expired.return_value = False
|
||||||
|
|
||||||
spot = _make_oauth(scope, path)
|
spot = _make_oauth(scope, path)
|
||||||
cached_tok = spot.validate_token(spot.cache_handler.get_cached_token())
|
cached_tok = spot.validate_token(spot.cache_handler.get_cached_token())
|
||||||
cached_tok_legacy = spot.get_cached_token()
|
cached_tok_legacy = spot.get_cached_token()
|
||||||
|
|
||||||
opener.assert_called_with(path)
|
opener.assert_called_with(path, encoding='utf-8')
|
||||||
self.assertIsNotNone(cached_tok)
|
self.assertIsNotNone(cached_tok)
|
||||||
self.assertIsNotNone(cached_tok_legacy)
|
self.assertIsNotNone(cached_tok_legacy)
|
||||||
self.assertEqual(refresh_access_token.call_count, 0)
|
self.assertEqual(refresh_access_token.call_count, 0)
|
||||||
@ -73,13 +76,15 @@ class OAuthCacheTest(unittest.TestCase):
|
|||||||
@patch('spotipy.cache_handler.open', create=True)
|
@patch('spotipy.cache_handler.open', create=True)
|
||||||
def test_expired_token_refreshes(self, opener,
|
def test_expired_token_refreshes(self, opener,
|
||||||
is_token_expired, refresh_access_token):
|
is_token_expired, refresh_access_token):
|
||||||
|
"""Test that an expired token is refreshed."""
|
||||||
scope = "playlist-modify-private"
|
scope = "playlist-modify-private"
|
||||||
path = ".cache-username"
|
path = ".cache-username"
|
||||||
expired_tok = _make_fake_token(0, None, scope)
|
expired_tok = _make_fake_token(0, None, scope)
|
||||||
fresh_tok = _make_fake_token(1, 1, scope)
|
fresh_tok = _make_fake_token(1, 1, scope)
|
||||||
|
|
||||||
token_file = _token_file(json.dumps(expired_tok, ensure_ascii=False))
|
token_file = _token_file(json.dumps(expired_tok, ensure_ascii=False))
|
||||||
opener.return_value = token_file
|
opener.return_value.__enter__ = mock.Mock(return_value=token_file)
|
||||||
|
opener.return_value.__exit__ = mock.Mock(return_value=False)
|
||||||
refresh_access_token.return_value = fresh_tok
|
refresh_access_token.return_value = fresh_tok
|
||||||
|
|
||||||
spot = _make_oauth(scope, path)
|
spot = _make_oauth(scope, path)
|
||||||
@ -87,7 +92,7 @@ class OAuthCacheTest(unittest.TestCase):
|
|||||||
|
|
||||||
is_token_expired.assert_called_with(expired_tok)
|
is_token_expired.assert_called_with(expired_tok)
|
||||||
refresh_access_token.assert_called_with(expired_tok['refresh_token'])
|
refresh_access_token.assert_called_with(expired_tok['refresh_token'])
|
||||||
opener.assert_any_call(path)
|
opener.assert_any_call(path, encoding='utf-8')
|
||||||
|
|
||||||
@patch.multiple(SpotifyOAuth,
|
@patch.multiple(SpotifyOAuth,
|
||||||
is_token_expired=DEFAULT, refresh_access_token=DEFAULT)
|
is_token_expired=DEFAULT, refresh_access_token=DEFAULT)
|
||||||
@ -99,29 +104,35 @@ class OAuthCacheTest(unittest.TestCase):
|
|||||||
path = ".cache-username"
|
path = ".cache-username"
|
||||||
tok = _make_fake_token(1, 1, token_scope)
|
tok = _make_fake_token(1, 1, token_scope)
|
||||||
|
|
||||||
opener.return_value = _token_file(json.dumps(tok, ensure_ascii=False))
|
token_file = _token_file(json.dumps(tok, ensure_ascii=False))
|
||||||
|
opener.return_value = token_file
|
||||||
|
opener.return_value.__enter__ = mock.Mock(return_value=token_file)
|
||||||
|
opener.return_value.__exit__ = mock.Mock(return_value=False)
|
||||||
is_token_expired.return_value = False
|
is_token_expired.return_value = False
|
||||||
|
|
||||||
spot = _make_oauth(requested_scope, path)
|
spot = _make_oauth(requested_scope, path)
|
||||||
cached_tok = spot.validate_token(spot.cache_handler.get_cached_token())
|
cached_tok = spot.validate_token(spot.cache_handler.get_cached_token())
|
||||||
|
|
||||||
opener.assert_called_with(path)
|
opener.assert_called_with(path, encoding='utf-8')
|
||||||
self.assertIsNone(cached_tok)
|
self.assertIsNone(cached_tok)
|
||||||
self.assertEqual(refresh_access_token.call_count, 0)
|
self.assertEqual(refresh_access_token.call_count, 0)
|
||||||
|
|
||||||
@patch('spotipy.cache_handler.open', create=True)
|
@patch('spotipy.cache_handler.open', create=True)
|
||||||
def test_saves_to_cache_path(self, opener):
|
def test_saves_to_cache_path(self, opener):
|
||||||
|
"""Test that the token is saved to the cache path."""
|
||||||
scope = "playlist-modify-private"
|
scope = "playlist-modify-private"
|
||||||
path = ".cache-username"
|
path = ".cache-username"
|
||||||
tok = _make_fake_token(1, 1, scope)
|
tok = _make_fake_token(1, 1, scope)
|
||||||
|
|
||||||
fi = _fake_file()
|
fi = _fake_file()
|
||||||
opener.return_value = fi
|
opener.return_value = fi
|
||||||
|
opener.return_value.__enter__ = mock.Mock(return_value=fi)
|
||||||
|
opener.return_value.__exit__ = mock.Mock(return_value=False)
|
||||||
|
|
||||||
spot = SpotifyOAuth("CLID", "CLISEC", "REDIR", "STATE", scope, path)
|
spot = SpotifyOAuth("CLID", "CLISEC", "REDIR", "STATE", scope, path)
|
||||||
spot.cache_handler.save_token_to_cache(tok)
|
spot.cache_handler.save_token_to_cache(tok)
|
||||||
|
|
||||||
opener.assert_called_with(path, 'w')
|
opener.assert_called_with(path, 'w', encoding='utf-8')
|
||||||
self.assertTrue(fi.write.called)
|
self.assertTrue(fi.write.called)
|
||||||
|
|
||||||
@patch('spotipy.cache_handler.open', create=True)
|
@patch('spotipy.cache_handler.open', create=True)
|
||||||
@ -132,11 +143,13 @@ class OAuthCacheTest(unittest.TestCase):
|
|||||||
|
|
||||||
fi = _fake_file()
|
fi = _fake_file()
|
||||||
opener.return_value = fi
|
opener.return_value = fi
|
||||||
|
opener.return_value.__enter__ = mock.Mock(return_value=fi)
|
||||||
|
opener.return_value.__exit__ = mock.Mock(return_value=False)
|
||||||
|
|
||||||
spot = SpotifyOAuth("CLID", "CLISEC", "REDIR", "STATE", scope, path)
|
spot = SpotifyOAuth("CLID", "CLISEC", "REDIR", "STATE", scope, path)
|
||||||
spot._save_token_info(tok)
|
spot._save_token_info(tok)
|
||||||
|
|
||||||
opener.assert_called_with(path, 'w')
|
opener.assert_called_with(path, 'w', encoding='utf-8')
|
||||||
self.assertTrue(fi.write.called)
|
self.assertTrue(fi.write.called)
|
||||||
|
|
||||||
def test_cache_handler(self):
|
def test_cache_handler(self):
|
||||||
@ -252,32 +265,38 @@ class ImplicitGrantCacheTest(unittest.TestCase):
|
|||||||
path = ".cache-username"
|
path = ".cache-username"
|
||||||
tok = _make_fake_token(1, 1, scope)
|
tok = _make_fake_token(1, 1, scope)
|
||||||
|
|
||||||
opener.return_value = _token_file(json.dumps(tok, ensure_ascii=False))
|
token_file = _token_file(json.dumps(tok, ensure_ascii=False))
|
||||||
|
opener.return_value = token_file
|
||||||
|
opener.return_value.__enter__ = mock.Mock(return_value=token_file)
|
||||||
|
opener.return_value.__exit__ = mock.Mock(return_value=False)
|
||||||
is_token_expired.return_value = False
|
is_token_expired.return_value = False
|
||||||
|
|
||||||
spot = _make_implicitgrantauth(scope, path)
|
spot = _make_implicitgrantauth(scope, path)
|
||||||
cached_tok = spot.cache_handler.get_cached_token()
|
cached_tok = spot.cache_handler.get_cached_token()
|
||||||
cached_tok_legacy = spot.get_cached_token()
|
cached_tok_legacy = spot.get_cached_token()
|
||||||
|
|
||||||
opener.assert_called_with(path)
|
opener.assert_called_with(path, encoding='utf-8')
|
||||||
self.assertIsNotNone(cached_tok)
|
self.assertIsNotNone(cached_tok)
|
||||||
self.assertIsNotNone(cached_tok_legacy)
|
self.assertIsNotNone(cached_tok_legacy)
|
||||||
|
|
||||||
@patch.object(SpotifyImplicitGrant, "is_token_expired", DEFAULT)
|
@patch.object(SpotifyImplicitGrant, "is_token_expired", DEFAULT)
|
||||||
@patch('spotipy.cache_handler.open', create=True)
|
@patch('spotipy.cache_handler.open', create=True)
|
||||||
def test_expired_token_returns_none(self, opener, is_token_expired):
|
def test_expired_token_returns_none(self, opener, is_token_expired):
|
||||||
|
"""Test that an expired token returns None."""
|
||||||
scope = "playlist-modify-private"
|
scope = "playlist-modify-private"
|
||||||
path = ".cache-username"
|
path = ".cache-username"
|
||||||
expired_tok = _make_fake_token(0, None, scope)
|
expired_tok = _make_fake_token(0, None, scope)
|
||||||
|
|
||||||
token_file = _token_file(json.dumps(expired_tok, ensure_ascii=False))
|
token_file = _token_file(json.dumps(expired_tok, ensure_ascii=False))
|
||||||
opener.return_value = token_file
|
opener.return_value = token_file
|
||||||
|
opener.return_value.__enter__ = mock.Mock(return_value=token_file)
|
||||||
|
opener.return_value.__exit__ = mock.Mock(return_value=False)
|
||||||
|
|
||||||
spot = _make_implicitgrantauth(scope, path)
|
spot = _make_implicitgrantauth(scope, path)
|
||||||
cached_tok = spot.validate_token(spot.cache_handler.get_cached_token())
|
cached_tok = spot.validate_token(spot.cache_handler.get_cached_token())
|
||||||
|
|
||||||
is_token_expired.assert_called_with(expired_tok)
|
is_token_expired.assert_called_with(expired_tok)
|
||||||
opener.assert_any_call(path)
|
opener.assert_any_call(path, encoding='utf-8')
|
||||||
self.assertIsNone(cached_tok)
|
self.assertIsNone(cached_tok)
|
||||||
|
|
||||||
@patch.object(SpotifyImplicitGrant, "is_token_expired", DEFAULT)
|
@patch.object(SpotifyImplicitGrant, "is_token_expired", DEFAULT)
|
||||||
@ -288,13 +307,16 @@ class ImplicitGrantCacheTest(unittest.TestCase):
|
|||||||
path = ".cache-username"
|
path = ".cache-username"
|
||||||
tok = _make_fake_token(1, 1, token_scope)
|
tok = _make_fake_token(1, 1, token_scope)
|
||||||
|
|
||||||
opener.return_value = _token_file(json.dumps(tok, ensure_ascii=False))
|
token_file = _token_file(json.dumps(tok, ensure_ascii=False))
|
||||||
|
opener.return_value = token_file
|
||||||
|
opener.return_value.__enter__ = mock.Mock(return_value=token_file)
|
||||||
|
opener.return_value.__exit__ = mock.Mock(return_value=False)
|
||||||
is_token_expired.return_value = False
|
is_token_expired.return_value = False
|
||||||
|
|
||||||
spot = _make_implicitgrantauth(requested_scope, path)
|
spot = _make_implicitgrantauth(requested_scope, path)
|
||||||
cached_tok = spot.validate_token(spot.cache_handler.get_cached_token())
|
cached_tok = spot.validate_token(spot.cache_handler.get_cached_token())
|
||||||
|
|
||||||
opener.assert_called_with(path)
|
opener.assert_called_with(path, encoding='utf-8')
|
||||||
self.assertIsNone(cached_tok)
|
self.assertIsNone(cached_tok)
|
||||||
|
|
||||||
@patch('spotipy.cache_handler.open', create=True)
|
@patch('spotipy.cache_handler.open', create=True)
|
||||||
@ -306,10 +328,12 @@ class ImplicitGrantCacheTest(unittest.TestCase):
|
|||||||
fi = _fake_file()
|
fi = _fake_file()
|
||||||
opener.return_value = fi
|
opener.return_value = fi
|
||||||
|
|
||||||
|
opener.return_value.__enter__ = mock.Mock(return_value=fi)
|
||||||
|
opener.return_value.__exit__ = mock.Mock(return_value=False)
|
||||||
spot = SpotifyImplicitGrant("CLID", "REDIR", "STATE", scope, path)
|
spot = SpotifyImplicitGrant("CLID", "REDIR", "STATE", scope, path)
|
||||||
spot.cache_handler.save_token_to_cache(tok)
|
spot.cache_handler.save_token_to_cache(tok)
|
||||||
|
|
||||||
opener.assert_called_with(path, 'w')
|
opener.assert_called_with(path, 'w', encoding='utf-8')
|
||||||
self.assertTrue(fi.write.called)
|
self.assertTrue(fi.write.called)
|
||||||
|
|
||||||
@patch('spotipy.cache_handler.open', create=True)
|
@patch('spotipy.cache_handler.open', create=True)
|
||||||
@ -320,11 +344,13 @@ class ImplicitGrantCacheTest(unittest.TestCase):
|
|||||||
|
|
||||||
fi = _fake_file()
|
fi = _fake_file()
|
||||||
opener.return_value = fi
|
opener.return_value = fi
|
||||||
|
opener.return_value.__enter__ = mock.Mock(return_value=fi)
|
||||||
|
opener.return_value.__exit__ = mock.Mock(return_value=False)
|
||||||
|
|
||||||
spot = SpotifyImplicitGrant("CLID", "REDIR", "STATE", scope, path)
|
spot = SpotifyImplicitGrant("CLID", "REDIR", "STATE", scope, path)
|
||||||
spot._save_token_info(tok)
|
spot._save_token_info(tok)
|
||||||
|
|
||||||
opener.assert_called_with(path, 'w')
|
opener.assert_called_with(path, 'w', encoding='utf-8')
|
||||||
self.assertTrue(fi.write.called)
|
self.assertTrue(fi.write.called)
|
||||||
|
|
||||||
|
|
||||||
@ -389,14 +415,17 @@ class SpotifyPKCECacheTest(unittest.TestCase):
|
|||||||
path = ".cache-username"
|
path = ".cache-username"
|
||||||
tok = _make_fake_token(1, 1, scope)
|
tok = _make_fake_token(1, 1, scope)
|
||||||
|
|
||||||
opener.return_value = _token_file(json.dumps(tok, ensure_ascii=False))
|
token_file = _token_file(json.dumps(tok, ensure_ascii=False))
|
||||||
|
opener.return_value = token_file
|
||||||
|
opener.return_value.__enter__ = mock.Mock(return_value=token_file)
|
||||||
|
opener.return_value.__exit__ = mock.Mock(return_value=False)
|
||||||
is_token_expired.return_value = False
|
is_token_expired.return_value = False
|
||||||
|
|
||||||
spot = _make_pkceauth(scope, path)
|
spot = _make_pkceauth(scope, path)
|
||||||
cached_tok = spot.cache_handler.get_cached_token()
|
cached_tok = spot.cache_handler.get_cached_token()
|
||||||
cached_tok_legacy = spot.get_cached_token()
|
cached_tok_legacy = spot.get_cached_token()
|
||||||
|
|
||||||
opener.assert_called_with(path)
|
opener.assert_called_with(path, encoding='utf-8')
|
||||||
self.assertIsNotNone(cached_tok)
|
self.assertIsNotNone(cached_tok)
|
||||||
self.assertIsNotNone(cached_tok_legacy)
|
self.assertIsNotNone(cached_tok_legacy)
|
||||||
self.assertEqual(refresh_access_token.call_count, 0)
|
self.assertEqual(refresh_access_token.call_count, 0)
|
||||||
@ -412,7 +441,8 @@ class SpotifyPKCECacheTest(unittest.TestCase):
|
|||||||
fresh_tok = _make_fake_token(1, 1, scope)
|
fresh_tok = _make_fake_token(1, 1, scope)
|
||||||
|
|
||||||
token_file = _token_file(json.dumps(expired_tok, ensure_ascii=False))
|
token_file = _token_file(json.dumps(expired_tok, ensure_ascii=False))
|
||||||
opener.return_value = token_file
|
opener.return_value.__enter__ = mock.Mock(return_value=token_file)
|
||||||
|
opener.return_value.__exit__ = mock.Mock(return_value=False)
|
||||||
refresh_access_token.return_value = fresh_tok
|
refresh_access_token.return_value = fresh_tok
|
||||||
|
|
||||||
spot = _make_pkceauth(scope, path)
|
spot = _make_pkceauth(scope, path)
|
||||||
@ -420,7 +450,7 @@ class SpotifyPKCECacheTest(unittest.TestCase):
|
|||||||
|
|
||||||
is_token_expired.assert_called_with(expired_tok)
|
is_token_expired.assert_called_with(expired_tok)
|
||||||
refresh_access_token.assert_called_with(expired_tok['refresh_token'])
|
refresh_access_token.assert_called_with(expired_tok['refresh_token'])
|
||||||
opener.assert_any_call(path)
|
opener.assert_any_call(path, encoding='utf-8')
|
||||||
|
|
||||||
@patch.multiple(SpotifyPKCE,
|
@patch.multiple(SpotifyPKCE,
|
||||||
is_token_expired=DEFAULT, refresh_access_token=DEFAULT)
|
is_token_expired=DEFAULT, refresh_access_token=DEFAULT)
|
||||||
@ -432,13 +462,16 @@ class SpotifyPKCECacheTest(unittest.TestCase):
|
|||||||
path = ".cache-username"
|
path = ".cache-username"
|
||||||
tok = _make_fake_token(1, 1, token_scope)
|
tok = _make_fake_token(1, 1, token_scope)
|
||||||
|
|
||||||
opener.return_value = _token_file(json.dumps(tok, ensure_ascii=False))
|
token_file = _token_file(json.dumps(tok, ensure_ascii=False))
|
||||||
|
opener.return_value = token_file
|
||||||
|
opener.return_value.__enter__ = mock.Mock(return_value=token_file)
|
||||||
|
opener.return_value.__exit__ = mock.Mock(return_value=False)
|
||||||
is_token_expired.return_value = False
|
is_token_expired.return_value = False
|
||||||
|
|
||||||
spot = _make_pkceauth(requested_scope, path)
|
spot = _make_pkceauth(requested_scope, path)
|
||||||
cached_tok = spot.validate_token(spot.cache_handler.get_cached_token())
|
cached_tok = spot.validate_token(spot.cache_handler.get_cached_token())
|
||||||
|
|
||||||
opener.assert_called_with(path)
|
opener.assert_called_with(path, encoding='utf-8')
|
||||||
self.assertIsNone(cached_tok)
|
self.assertIsNone(cached_tok)
|
||||||
self.assertEqual(refresh_access_token.call_count, 0)
|
self.assertEqual(refresh_access_token.call_count, 0)
|
||||||
|
|
||||||
@ -450,11 +483,12 @@ class SpotifyPKCECacheTest(unittest.TestCase):
|
|||||||
|
|
||||||
fi = _fake_file()
|
fi = _fake_file()
|
||||||
opener.return_value = fi
|
opener.return_value = fi
|
||||||
|
opener.return_value.__enter__ = mock.Mock(return_value=fi)
|
||||||
|
opener.return_value.__exit__ = mock.Mock(return_value=False)
|
||||||
spot = SpotifyPKCE("CLID", "REDIR", "STATE", scope, path)
|
spot = SpotifyPKCE("CLID", "REDIR", "STATE", scope, path)
|
||||||
spot.cache_handler.save_token_to_cache(tok)
|
spot.cache_handler.save_token_to_cache(tok)
|
||||||
|
|
||||||
opener.assert_called_with(path, 'w')
|
opener.assert_called_with(path, 'w', encoding='utf-8')
|
||||||
self.assertTrue(fi.write.called)
|
self.assertTrue(fi.write.called)
|
||||||
|
|
||||||
@patch('spotipy.cache_handler.open', create=True)
|
@patch('spotipy.cache_handler.open', create=True)
|
||||||
@ -465,11 +499,13 @@ class SpotifyPKCECacheTest(unittest.TestCase):
|
|||||||
|
|
||||||
fi = _fake_file()
|
fi = _fake_file()
|
||||||
opener.return_value = fi
|
opener.return_value = fi
|
||||||
|
opener.return_value.__enter__ = mock.Mock(return_value=fi)
|
||||||
|
opener.return_value.__exit__ = mock.Mock(return_value=False)
|
||||||
|
|
||||||
spot = SpotifyPKCE("CLID", "REDIR", "STATE", scope, path)
|
spot = SpotifyPKCE("CLID", "REDIR", "STATE", scope, path)
|
||||||
spot._save_token_info(tok)
|
spot._save_token_info(tok)
|
||||||
|
|
||||||
opener.assert_called_with(path, 'w')
|
opener.assert_called_with(path, 'w', encoding='utf-8')
|
||||||
self.assertTrue(fi.write.called)
|
self.assertTrue(fi.write.called)
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user