Skip to content
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

Spirc: Replace Mecury with Dealer #1356

Merged
merged 140 commits into from
Dec 10, 2024

Conversation

photovoltex
Copy link
Member

@photovoltex photovoltex commented Sep 29, 2024

Initial PR Description

As suggested from @roderickvd a draft PR so i can have some community feedback and give now and then updates on the progress.

The replacement is in a working state, so the very basic do work again. There is still a lot of cleanup and re-implementing to do.

But so far what is working:

  • transferring the playback so that we are the player
    • funnily enough, transferring back isn't working yet
  • play, pause, resume, prev, next, seek
  • just general communication of our state to other clients

Previously Known Problems:

  • on the initial connection librespot doesn't appear, but if for example the volume is slightly updated, we should appear

    this might be a web-player specific problem, in the client the dealer just pops up

    • this seems to be to some degree fine? if the devices are reopend it seems to be always there, it just isn't dynamically updated.
  • if the session or spirc/dealer dies, we lose our complete ConnectState
    • it would be nice if that is preserved so we could start from that again
  • sometimes other client's forget that we are the current player
    • might be that that is already resolved, but i think i still saw that behavior not so long ago
    • update: seems to be resolved, was probably just an early development problem

What i still have to do:

  • handling the remaining request commands
    • queue handling (set_queue, add_to_queue)
    • repeat handling (set_repeating_track, set_repeating_context)
      • i didn't see these in the wild yet, so it might that they do not exists
    • context handling (update_context)
    • options handling (set_options)
      • should be related to shuffle and repeat
  • transferring the state to a different player again (stop playback if we are not active anymore)
  • resolving the context from a cold state
    • currently it works really well if the playback starts from a different device like the web-player
    • but from a cold start it's still a bit weird and not fully sorted out
    • we don't copy the player state from a running instance anymore,
    • so that we actually use the the cold state resolving/handling/transfering
  • shuffling has to be re-implemented
    • we can set restrictions for the clients getting our state
    • so for autoplay we should ignore shuffeling and just disable repeat and shuffle
  • autoplay has to be fixed
  • i added the option for track repeat, but it's not handled yet
  • i would like to remove the remaining mercury calls
    • but for that i will have to see if the dealer sendes equivalent messages/requests

I tried to keep my code clean, but if there are points that i could improve and such, i would like to here them^^

Currently Known Issues:

  • podcasts/episodes don't work
  • starting a track out of "recent search" from mobile and then trying to switch back to mobile
  • on mobile it can occur that librespot is seen as still playing with the current account, even tho it isn't playing
  • shufflling a context with multiple pages does only shuffle the first page (because initially only that page is loaded)
    • this is a problem for example, when shuffling an artist context

Fixes #19

  • the dealer does support this via the field repeating_track and set_options command

Fixes #434

  • a context is now resolved completely

Fixes #442

  • a context is now resolved completely

Fixes #701

  • the dealer does support this action via the command set_queue

Fixes #765

  • resolve context does provide the top 10 tracks and all albums in order from newest to oldest

Fixes #861

  • the next tracks are completely reworked

Fixes #1205

Fixes #1353

  • I encountered this issue during development, but after fixing everything around the uid's it seems to be fully functional now

Thx again to all people testing the branch, helps a lot :D

@photovoltex
Copy link
Member Author

oh the branch might not compile because there are still some todo!() here and there which result in unreachable statements that clippy, understandably, doesn't like.

@roderickvd
Copy link
Member

Super job! I know you're not there yet, so let's all contribute here. Stuff like this almost makes me sign up for a Spotify plan again 😉

I'm gonna have a more thorough look later. Wanted to ping back on your comment over at #1349:

The hm://pusher/v1/connections/ message is send after connecting to set up some connect stuff initially. Can be safely ignored as far as i know. It just falls under one of the many unknown subscriptions category that are logged.

If you look at librespot-java, you'll see that it listens to this message to extract the Spotify-Connection-Id from the headers:

https://github.com/librespot-org/librespot-java/blob/d0ff31b4f476590c9c17401aac01a6762e4ba908/player/src/main/java/xyz/gianlu/librespot/player/state/DeviceStateHandler.java#L162

I think if you follow the code flow, updating the connection ID then triggers the ready event in the deviceStateListener, which in turn gets the volume. And that seems to gel with you saying that:

on the initial connection librespot doesn't appear, but if for example the volume is slightly updated, we should appear

- impl shuffle again
- extracted fill up of next tracks in own method
- moved queue revision update into next track fill up
- removed unused method `set_playing_track_index`
- added option to specify index when resetting the playback context
- reshuffle after repeat context
@roderickvd
Copy link
Member

Awesome so let's get it in dev. Would you consider squashing it?

@PocketMiner82
Copy link

PocketMiner82 commented Dec 6, 2024

Hmmm... I think I found a new issue, happened to me 3 times today (but for some reason never before): I was listening to the liked songs playlists for some time and suddenly the music stops playing in the middle of a song. the following errors got logged:

Dec 06 10:39:52 rpi-3a librespot[1607]: [2024-12-06T09:39:52Z ERROR librespot_playback::player] Skipping to next track, unable to get next packet for track <SpotifyId("spotify:track:4GewRLEiI4BDTQsoH8BRVV")>: SymphoniaDecoder("Deadline expired before operation could complete { wait timeout exceeded }")
Dec 06 10:40:00 rpi-3a librespot[1607]: [2024-12-06T09:40:00Z ERROR librespot_playback::player] Skipping to next track, unable to get next packet for track <SpotifyId("spotify:track:4GewRLEiI4BDTQsoH8BRVV")>: SymphoniaDecoder("Deadline expired before operation could complete { wait timeout exceeded }")
Dec 06 10:40:08 rpi-3a librespot[1607]: [2024-12-06T09:40:08Z ERROR librespot_playback::player] Skipping to next track, unable to get next packet for track <SpotifyId("spotify:track:4GewRLEiI4BDTQsoH8BRVV")>: SymphoniaDecoder("Deadline expired before operation could complete { wait timeout exceeded }")
Dec 06 10:40:16 rpi-3a librespot[1607]: [2024-12-06T09:40:16Z ERROR librespot_playback::player] Skipping to next track, unable to get next packet for track <SpotifyId("spotify:track:4GewRLEiI4BDTQsoH8BRVV")>: SymphoniaDecoder("Deadline expired before operation could complete { wait timeout exceeded }")
Dec 06 10:40:24 rpi-3a librespot[1607]: [2024-12-06T09:40:24Z ERROR librespot_playback::player] Skipping to next track, unable to get next packet for track <SpotifyId("spotify:track:4GewRLEiI4BDTQsoH8BRVV")>: SymphoniaDecoder("Deadline expired before operation could complete { wait timeout exceeded }")
Dec 06 10:40:32 rpi-3a librespot[1607]: [2024-12-06T09:40:32Z ERROR librespot_playback::player] Skipping to next track, unable to get next packet for track <SpotifyId("spotify:track:4GewRLEiI4BDTQsoH8BRVV")>: SymphoniaDecoder("Deadline expired before operation could complete { wait timeout exceeded }")
Dec 06 10:40:40 rpi-3a librespot[1607]: [2024-12-06T09:40:40Z ERROR librespot_playback::player] Skipping to next track, unable to get next packet for track <SpotifyId("spotify:track:4GewRLEiI4BDTQsoH8BRVV")>: SymphoniaDecoder("Deadline expired before operation could complete { wait timeout exceeded }")
Dec 06 10:40:48 rpi-3a librespot[1607]: [2024-12-06T09:40:48Z ERROR librespot_playback::player] Skipping to next track, unable to get next packet for track <SpotifyId("spotify:track:4GewRLEiI4BDTQsoH8BRVV")>: SymphoniaDecoder("Deadline expired before operation could complete { wait timeout exceeded }")
Dec 06 10:40:56 rpi-3a librespot[1607]: [2024-12-06T09:40:56Z ERROR librespot_playback::player] Skipping to next track, unable to get next packet for track <SpotifyId("spotify:track:4GewRLEiI4BDTQsoH8BRVV")>: SymphoniaDecoder("Deadline expired before operation could complete { wait timeout exceeded }")
Dec 06 10:41:04 rpi-3a librespot[1607]: [2024-12-06T09:41:04Z ERROR librespot_playback::player] Skipping to next track, unable to get next packet for track <SpotifyId("spotify:track:4GewRLEiI4BDTQsoH8BRVV")>: SymphoniaDecoder("Deadline expired before operation could complete { wait timeout exceeded }")

Note that the wifi of this device (Raspberry Pi 3 A+) is a bit unstable for me, sometimes it doesn't respond for ~15 seconds... Maybe that's why the wait timeout exceeded. I thought I'll report it regardless, maybe some timeout can be increased 🤷

@photovoltex
Copy link
Member Author

Awesome so let's get it in dev. Would you consider squashing it?

Sure thing. But did anyone look over the code or do we just merge it? I could understand why, the changes are a bit too drastic/large to review. But just wanted to hear what the stance in regards to that is^^

@roderickvd
Copy link
Member

In an ideal world I'd like to run through the code, but I don't have a lot of time for it in the coming weeks. So I thought that it's better to get it in and we can still place code comments on GitHub after the merge.

@photovoltex
Copy link
Member Author

Sounds reasonable. Feel free to merge whenever fits best :)

@photovoltex
Copy link
Member Author

photovoltex commented Dec 6, 2024

Hmmm... I think I found a new issue, happened to me 3 times today (but for some reason never before): I was listening to the liked songs playlists for some time and suddenly the music stops playing in the middle of a song. the following errors got logged:

Dec 06 10:39:52 rpi-3a librespot[1607]: [2024-12-06T09:39:52Z ERROR librespot_playback::player] Skipping to next track, unable to get next packet for track <SpotifyId("spotify:track:4GewRLEiI4BDTQsoH8BRVV")>: SymphoniaDecoder("Deadline expired before operation could complete { wait timeout exceeded }")
Dec 06 10:40:00 rpi-3a librespot[1607]: [2024-12-06T09:40:00Z ERROR librespot_playback::player] Skipping to next track, unable to get next packet for track <SpotifyId("spotify:track:4GewRLEiI4BDTQsoH8BRVV")>: SymphoniaDecoder("Deadline expired before operation could complete { wait timeout exceeded }")
Dec 06 10:40:08 rpi-3a librespot[1607]: [2024-12-06T09:40:08Z ERROR librespot_playback::player] Skipping to next track, unable to get next packet for track <SpotifyId("spotify:track:4GewRLEiI4BDTQsoH8BRVV")>: SymphoniaDecoder("Deadline expired before operation could complete { wait timeout exceeded }")
Dec 06 10:40:16 rpi-3a librespot[1607]: [2024-12-06T09:40:16Z ERROR librespot_playback::player] Skipping to next track, unable to get next packet for track <SpotifyId("spotify:track:4GewRLEiI4BDTQsoH8BRVV")>: SymphoniaDecoder("Deadline expired before operation could complete { wait timeout exceeded }")
Dec 06 10:40:24 rpi-3a librespot[1607]: [2024-12-06T09:40:24Z ERROR librespot_playback::player] Skipping to next track, unable to get next packet for track <SpotifyId("spotify:track:4GewRLEiI4BDTQsoH8BRVV")>: SymphoniaDecoder("Deadline expired before operation could complete { wait timeout exceeded }")
Dec 06 10:40:32 rpi-3a librespot[1607]: [2024-12-06T09:40:32Z ERROR librespot_playback::player] Skipping to next track, unable to get next packet for track <SpotifyId("spotify:track:4GewRLEiI4BDTQsoH8BRVV")>: SymphoniaDecoder("Deadline expired before operation could complete { wait timeout exceeded }")
Dec 06 10:40:40 rpi-3a librespot[1607]: [2024-12-06T09:40:40Z ERROR librespot_playback::player] Skipping to next track, unable to get next packet for track <SpotifyId("spotify:track:4GewRLEiI4BDTQsoH8BRVV")>: SymphoniaDecoder("Deadline expired before operation could complete { wait timeout exceeded }")
Dec 06 10:40:48 rpi-3a librespot[1607]: [2024-12-06T09:40:48Z ERROR librespot_playback::player] Skipping to next track, unable to get next packet for track <SpotifyId("spotify:track:4GewRLEiI4BDTQsoH8BRVV")>: SymphoniaDecoder("Deadline expired before operation could complete { wait timeout exceeded }")
Dec 06 10:40:56 rpi-3a librespot[1607]: [2024-12-06T09:40:56Z ERROR librespot_playback::player] Skipping to next track, unable to get next packet for track <SpotifyId("spotify:track:4GewRLEiI4BDTQsoH8BRVV")>: SymphoniaDecoder("Deadline expired before operation could complete { wait timeout exceeded }")
Dec 06 10:41:04 rpi-3a librespot[1607]: [2024-12-06T09:41:04Z ERROR librespot_playback::player] Skipping to next track, unable to get next packet for track <SpotifyId("spotify:track:4GewRLEiI4BDTQsoH8BRVV")>: SymphoniaDecoder("Deadline expired before operation could complete { wait timeout exceeded }")

Note that the wifi of this device (Raspberry Pi 3 A+) is a bit unstable for me, sometimes it doesn't respond for ~15 seconds... Maybe that's why the wait timeout exceeded. I thought I'll report it regardless, maybe some timeout can be increased 🤷

That should probably be unrelated to these changes. But if they appear again we can open an issue^^

@YutongGu
Copy link

YutongGu commented Dec 6, 2024

Hmmm... I think I found a new issue, happened to me 3 times today (but for some reason never before): I was listening to the liked songs playlists for some time and suddenly the music stops playing in the middle of a song. the following errors got logged...

I can report that this issue occurred for me as well:

[2024-12-06T21:34:29Z INFO  librespot_playback::player] Loading <Guy For That (Feat. Luke Combs)> with Spotify URI <spotify:track:6StwwqB84sJeLr7tZDTxEX>
[2024-12-06T21:34:32Z ERROR librespot_core::audio_key] Audio key response timeout
Rate limited 1 logs
[2024-12-06T21:34:37Z ERROR librespot_core::session] Connection to server closed.
[2024-12-06T21:35:01Z WARN  librespot_core::dealer] Websocket peer does not respond.
[2024-12-06T21:35:06Z ERROR librespot_playback::player] Unable to read audio file: Symphonia Decoder Error: Deadline expired before operation could complete { wait timeout exceeded }
[2024-12-06T21:35:08Z WARN  librespot] Spirc shut down unexpectedly
[2024-12-06T21:35:08Z ERROR librespot_core::session] Transport endpoint is not connected (os error 107)
[2024-12-06T21:35:18Z INFO  librespot_core::session] Connecting to AP "ap-guc3.spotify.com:4070"
[2024-12-06T21:35:23Z INFO  librespot_core::session] Authenticated as '1219828745' !
Rate limited 1 logs
[2024-12-06T21:35:31Z INFO  librespot_core::spclient] Resolved "guc3-spclient.spotify.com:443" as spclient access point
[2024-12-06T21:35:39Z INFO  librespot_connect::spirc] active device is <> with session <0d5cfc65538b301a627f3c9d7dce54b2>
[2024-12-06T21:38:11Z WARN  librespot_connect::state::context] switching to active context had issues: Invalid state { context is not available. type: Default }
---- 2024-12-06 21:38:14.953302: Now starting Spotify ----

@photovoltex
Copy link
Member Author

Hui that's a lot of errors and warning in one spot. Seems all be related to sort of timeout or unresponsive network connection. Can both of you maybe try to reproduce these issues on the dev branch?

I will look into it if there might be some changes that unintentionally affect the player. But from the errors in the log it sounds more like network issues.

@photovoltex
Copy link
Member Author

I did also find a weird behavior from the context-resolver... if you request a playlist that contains tracks and episodes, only the tracks will be resolved. The response seems to not contain any info about the episodes... I will probably not solve it anymore in context of this PR, so might be moved out into a new issue afterwards.

@photovoltex
Copy link
Member Author

There is also an oversight with the current impl in relation to shuffle and a context with multiple pages. For example playing an artist with shuffle doesn't work yet as one would expect. That probably needs a revisit at some point in the future. Adjusting the current impl would probably add another 100+ lines to the current PR, which i would like to prevent for now^^

@photovoltex
Copy link
Member Author

Looked again at this comment (#1356 (comment)), and it might be a better solution in the end, when we play a shuffled context we usually don't need the default context. I had hopped we could just keep refs to the original tracks in a vec, but that collides with every state modification you still want to do.

That might be a nice next task to do after this initial state is merged and we iterate on improving performance and stability :D

@photovoltex
Copy link
Member Author

Continuing development here (https://github.com/photovoltex/librespot/tree/improve-dealer) so that this PR is only the basic integration (with some new features provided by the dealer)

@roderickvd
Copy link
Member

Drumroll for merging, coming up in 3... 2....

@roderickvd roderickvd merged commit 5839b36 into librespot-org:dev Dec 10, 2024
13 checks passed
@photovoltex
Copy link
Member Author

photovoltex commented Dec 10, 2024

Well, and now we wait for issues that we didn't spot during testing/development

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment