-
-
Notifications
You must be signed in to change notification settings - Fork 1.7k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
429/500/404 Errors and Rate Limiting Issues #2142
Comments
I am confused, since I thought downloads are from youtube these days, what is the spotify key used for? |
Getting the metadata from Spotify. We can't do that without authorization. |
thank you @xnetcat this worked perfectly to resolve the issue. |
i've been getting the same behaviour (sync hanging after i'm rotating 3 different spotify apps, but still doesn't (always) helps. any chance we could get some debug logs that could warn about the process being stuck and the reason ? (i guess if we don't get a console feedback after 30 minutes the issue is clear, but avoiding the infinite hanging would be nice) |
For anyone else having issues with the 429 Too Many Requests error: And now, this works for small albums, but when I try on a larger playlist (like my 300+ liked songs) it returns the 429 error. No matter, for if that's the case, it's probably better to split your liked songs playlist into several smaller public playlists, to then download sequentially. |
Fix doesn't work for me when searching for a song, only works if you enter album or song link. But still downloading very slowly compared to version 4.2.5 |
FWIW, using the |
Using a client id + client secret or user auth doesn't work for me when using Processing query: https://open.spotify.com/playlist/[redacted]
Max Retries reached
An error occurred
ResponseError: too many 429 error responses
The above exception was the direct cause of the following exception:
╭───────────────────── Traceback (most recent call last) ──────────────────────╮
│ /opt/homebrew/lib/python3.11/site-packages/requests/adapters.py:667 in send │
│ │
│ 664 │ │ │ timeout = TimeoutSauce(connect=timeout, read=timeout) │
│ 665 │ │ │
│ 666 │ │ try: │
│ ❱ 667 │ │ │ resp = conn.urlopen( │
│ 668 │ │ │ │ method=request.method, │
│ 669 │ │ │ │ url=url, │
│ 670 │ │ │ │ body=request.body, │
│ │
│ /opt/homebrew/lib/python3.11/site-packages/urllib3/connectionpool.py:948 in │
│ urlopen │
│ │
│ 945 │ │ │ response.drain_conn() │
│ 946 │ │ │ retries.sleep(response) │
│ 947 │ │ │ log.debug("Retry: %s", url) │
│ ❱ 948 │ │ │ return self.urlopen( │
│ 949 │ │ │ │ method, │
│ 950 │ │ │ │ url, │
│ 951 │ │ │ │ body, │
│ │
│ /opt/homebrew/lib/python3.11/site-packages/urllib3/connectionpool.py:948 in │
│ urlopen │
│ │
│ 945 │ │ │ response.drain_conn() │
│ 946 │ │ │ retries.sleep(response) │
│ 947 │ │ │ log.debug("Retry: %s", url) │
│ ❱ 948 │ │ │ return self.urlopen( │
│ 949 │ │ │ │ method, │
│ 950 │ │ │ │ url, │
│ 951 │ │ │ │ body, │
│ │
│ /opt/homebrew/lib/python3.11/site-packages/urllib3/connectionpool.py:948 in │
│ urlopen │
│ │
│ 945 │ │ │ response.drain_conn() │
│ 946 │ │ │ retries.sleep(response) │
│ 947 │ │ │ log.debug("Retry: %s", url) │
│ ❱ 948 │ │ │ return self.urlopen( │
│ 949 │ │ │ │ method, │
│ 950 │ │ │ │ url, │
│ 951 │ │ │ │ body, │
│ │
│ /opt/homebrew/lib/python3.11/site-packages/urllib3/connectionpool.py:938 in │
│ urlopen │
│ │
│ 935 │ │ has_retry_after = bool(response.headers.get("Retry-After")) │
│ 936 │ │ if retries.is_retry(method, response.status, has_retry_after) │
│ 937 │ │ │ try: │
│ ❱ 938 │ │ │ │ retries = retries.increment(method, url, response=res │
│ 939 │ │ │ except MaxRetryError: │
│ 940 │ │ │ │ if retries.raise_on_status: │
│ 941 │ │ │ │ │ response.drain_conn() │
│ │
│ /opt/homebrew/lib/python3.11/site-packages/urllib3/util/retry.py:515 in │
│ increment │
│ │
│ 512 │ │ │
│ 513 │ │ if new_retry.is_exhausted(): │
│ 514 │ │ │ reason = error or ResponseError(cause) │
│ ❱ 515 │ │ │ raise MaxRetryError(_pool, url, reason) from reason # typ │
│ ignore[arg-type] │
│ 516 │ │ │
│ 517 │ │ log.debug("Incremented Retry for (url='%s'): %r", url, new_ret │
│ 518 │
╰──────────────────────────────────────────────────────────────────────────────╯
MaxRetryError: HTTPSConnectionPool(host='api.spotify.com', port=443): Max
retries exceeded with url:
/v1/playlists/[redacted]?additional_types=track (Caused by
ResponseError('too many 429 error responses'))
During handling of the above exception, another exception occurred:
╭───────────────────── Traceback (most recent call last) ──────────────────────╮
│ /opt/homebrew/lib/python3.11/site-packages/spotipy/client.py:270 in │
│ _internal_call │
│ │
│ 267 │ │ │ │ │ method, url, args.get("params"), headers, args.g │
│ 268 │ │ │
│ 269 │ │ try: │
│ ❱ 270 │ │ │ response = self._session.request( │
│ 271 │ │ │ │ method, url, headers=headers, proxies=self.proxies, │
│ 272 │ │ │ │ timeout=self.requests_timeout, **args │
│ 273 │ │ │ ) │
│ │
│ /opt/homebrew/lib/python3.11/site-packages/requests/sessions.py:589 in │
│ request │
│ │
│ 586 │ │ │ "allow_redirects": allow_redirects, │
│ 587 │ │ } │
│ 588 │ │ send_kwargs.update(settings) │
│ ❱ 589 │ │ resp = self.send(prep, **send_kwargs) │
│ 590 │ │ │
│ 591 │ │ return resp │
│ 592 │
│ │
│ /opt/homebrew/lib/python3.11/site-packages/requests/sessions.py:703 in send │
│ │
│ 700 │ │ start = preferred_clock() │
│ 701 │ │ │
│ 702 │ │ # Send the request │
│ ❱ 703 │ │ r = adapter.send(request, **kwargs) │
│ 704 │ │ │
│ 705 │ │ # Total elapsed time of the request (approximately) │
│ 706 │ │ elapsed = preferred_clock() - start │
│ │
│ /opt/homebrew/lib/python3.11/site-packages/requests/adapters.py:691 in send │
│ │
│ 688 │ │ │ │ │ raise ConnectTimeout(e, request=request) │
│ 689 │ │ │ │
│ 690 │ │ │ if isinstance(e.reason, ResponseError): │
│ ❱ 691 │ │ │ │ raise RetryError(e, request=request) │
│ 692 │ │ │ │
│ 693 │ │ │ if isinstance(e.reason, _ProxyError): │
│ 694 │ │ │ │ raise ProxyError(e, request=request) │
╰──────────────────────────────────────────────────────────────────────────────╯
RetryError: HTTPSConnectionPool(host='api.spotify.com', port=443): Max retries
exceeded with url: /v1/playlists/[redacted]?additional_types=track
(Caused by ResponseError('too many 429 error responses'))
During handling of the above exception, another exception occurred:
╭───────────────────── Traceback (most recent call last) ──────────────────────╮
│ /opt/homebrew/lib/python3.11/site-packages/spotdl/console/entry_point.py:160 │
│ in entry_point │
│ │
│ 157 │ try: │
│ 158 │ │ # Pick the operation to perform │
│ 159 │ │ # based on the name and run it! │
│ ❱ 160 │ │ OPERATIONS[arguments.operation]( │
│ 161 │ │ │ query=arguments.query, │
│ 162 │ │ │ downloader=downloader, │
│ 163 │ │ ) │
│ │
│ /opt/homebrew/lib/python3.11/site-packages/spotdl/console/sync.py:109 in │
│ sync │
│ │
│ 106 │ │ │ raise ValueError("Sync file is not a valid sync file.") │
│ 107 │ │ │
│ 108 │ │ # Parse the query │
│ ❱ 109 │ │ songs_playlist = parse_query( │
│ 110 │ │ │ query=sync_data["query"], │
│ 111 │ │ │ threads=downloader.settings["threads"], │
│ 112 │ │ │ use_ytm_data=downloader.settings["ytm_data"], │
│ │
│ /opt/homebrew/lib/python3.11/site-packages/spotdl/utils/search.py:97 in │
│ parse_query │
│ │
│ 94 │ - List of song objects │
│ 95 │ """ │
│ 96 │ │
│ ❱ 97 │ songs: List[Song] = get_simple_songs( │
│ 98 │ │ query, │
│ 99 │ │ use_ytm_data=use_ytm_data, │
│ 100 │ │ playlist_numbering=playlist_numbering, │
│ │
│ /opt/homebrew/lib/python3.11/site-packages/spotdl/utils/search.py:260 in │
│ get_simple_songs │
│ │
│ 257 │ │ │ ) │
│ 258 │ │ │ songs.extend(full_lists) │
│ 259 │ │ elif "open.spotify.com" in request and "playlist" in request: │
│ ❱ 260 │ │ │ lists.append(Playlist.from_url(request, fetch_songs=False) │
│ 261 │ │ elif "open.spotify.com" in request and "album" in request: │
│ 262 │ │ │ lists.append(Album.from_url(request, fetch_songs=False)) │
│ 263 │ │ elif "open.spotify.com" in request and "artist" in request: │
│ │
│ /opt/homebrew/lib/python3.11/site-packages/spotdl/types/song.py:306 in │
│ from_url │
│ │
│ 303 │ │ - The SongList object. │
│ 304 │ │ """ │
│ 305 │ │ │
│ ❱ 306 │ │ metadata, songs = cls.get_metadata(url) │
│ 307 │ │ urls = [song.url for song in songs] │
│ 308 │ │ │
│ 309 │ │ if fetch_songs: │
│ │
│ /opt/homebrew/lib/python3.11/site-packages/spotdl/types/playlist.py:48 in │
│ get_metadata │
│ │
│ 45 │ │ │
│ 46 │ │ spotify_client = SpotifyClient() │
│ 47 │ │ │
│ ❱ 48 │ │ playlist = spotify_client.playlist(url) │
│ 49 │ │ if playlist is None: │
│ 50 │ │ │ raise PlaylistError("Invalid playlist URL.") │
│ 51 │
│ │
│ /opt/homebrew/lib/python3.11/site-packages/spotipy/client.py:666 in playlist │
│ │
│ 663 │ │ │ │ │ │ │ │ │ valid types are: track and episo │
│ 664 │ │ """ │
│ 665 │ │ plid = self._get_id("playlist", playlist_id) │
│ ❱ 666 │ │ return self._get( │
│ 667 │ │ │ f"playlists/{plid}", │
│ 668 │ │ │ fields=fields, │
│ 669 │ │ │ market=market, │
│ │
│ /opt/homebrew/lib/python3.11/site-packages/spotdl/utils/spotify.py:195 in │
│ _get │
│ │
│ 192 │ │ retries = self.max_retries # type: ignore # pylint: disable=E │
│ 193 │ │ while response is None: │
│ 194 │ │ │ try: │
│ ❱ 195 │ │ │ │ response = self._internal_call("GET", url, payload, kw │
│ 196 │ │ │ except (requests.exceptions.Timeout, requests.ConnectionEr │
│ 197 │ │ │ │ retries -= 1 │
│ 198 │ │ │ │ if retries <= 0: │
│ │
│ /opt/homebrew/lib/python3.11/site-packages/spotipy/client.py:311 in │
│ _internal_call │
│ │
│ 308 │ │ │ │ reason = retry_error.args[0].reason │
│ 309 │ │ │ except (IndexError, AttributeError): │
│ 310 │ │ │ │ reason = None │
│ ❱ 311 │ │ │ raise SpotifyException( │
│ 312 │ │ │ │ 429, │
│ 313 │ │ │ │ -1, │
│ 314 │ │ │ │ f"{request.path_url}:\n Max Retries", │
╰──────────────────────────────────────────────────────────────────────────────╯
SpotifyException: http status: 429, code:-1 -
/v1/playlists/[redacted]?additional_types=track:
Max Retries, reason: too many 429 error responses I have already tried creating a new sync file to no avail: SpotifyException: http status: 429, code:-1 -
/v1/playlists/[redacted]/tracks?offset=400&limit=100&additional_type
s=track:
Max Retries, reason: too many 429 error responses |
Just want to chip in and say I'm having the same issue as @Wallvon, no auth solution is fixing the sync command for myself |
I had the same issue as @Wallvon and @SudoZachCampbell, using my own Client ID and Secret, but it would still throw 429. What fixed it for me was to use your own Client ID and Secret, and delete |
Hi, thanks for the comment. After having done that I am now getting a different error: An error occurred
╭───────────────────── Traceback (most recent call last) ──────────────────────╮
│ /opt/homebrew/lib/python3.11/site-packages/spotdl/console/entry_point.py:160 │
│ in entry_point │
│ │
│ 157 │ try: │
│ 158 │ │ # Pick the operation to perform │
│ 159 │ │ # based on the name and run it! │
│ ❱ 160 │ │ OPERATIONS[arguments.operation]( │
│ 161 │ │ │ query=arguments.query, │
│ 162 │ │ │ downloader=downloader, │
│ 163 │ │ ) │
│ │
│ /opt/homebrew/lib/python3.11/site-packages/spotdl/console/sync.py:121 in │
│ sync │
│ │
│ 118 │ │ old_files = [] │
│ 119 │ │ for entry in sync_data["songs"]: │
│ 120 │ │ │ file_name = create_file_name( │
│ ❱ 121 │ │ │ │ Song.from_dict(entry), │
│ 122 │ │ │ │ downloader.settings["output"], │
│ 123 │ │ │ │ downloader.settings["format"], │
│ 124 │ │ │ │ downloader.settings["restrict"], │
│ │
│ /opt/homebrew/lib/python3.11/site-packages/spotdl/types/song.py:237 in │
│ from_dict │
│ │
│ 234 │ │ """ │
│ 235 │ │ │
│ 236 │ │ # Return product object │
│ ❱ 237 │ │ return cls(**data) │
│ 238 │ │
│ 239 │ @classmethod │
│ 240 │ def from_missing_data(cls, **kwargs) -> "Song": │
╰──────────────────────────────────────────────────────────────────────────────╯
TypeError: Song.__init__() missing 1 required positional argument: 'album_type' For further information: |
Once again I am matching @Wallvon, I am also seeing the same thing where it hangs on |
So it seems like the Spotify api is working correctly, have you tried to use yt-dlp to download something from YouTube? Recently, YouTube has also started flagging IPs and requiring you to be logged in. I got around this by using a proxy and passing it to spotdl through the config file. |
Yes, I have coincidentally used yt-dlp an hour ago on the same machine which worked fine. |
tried doing the deleting file, and also using own id and secret, but it still gives 429 and 404 errors, last time spotdl worked for me was Nov 14(havent used it since then till today) |
nvm, turns out it was a private playlist that caused the problem, just had to duplicate and make public |
If someone having the rate limit issue is willing/skilled to try building spotdl (or use docker) and test the changes from this PR, i'd be interested to have feedbacks : |
Hey everyone,
I've been seeing more 429/404/500 errors, and slow query processing lately. This is because of Spotify's rate limits on our shared project.
Quick Fix: Use Your Own Client ID and Secret
To avoid these issues, we recommend using your own client ID and secret from Spotify. Here’s how:
Web API
and Redirect URIs tohttp://127.0.0.1:9900/
(or whatever you want to use))Update Your Settings:
I’m working on a long-term fix to better manage our rate limits. Stay tuned for updates!
The text was updated successfully, but these errors were encountered: