-
Notifications
You must be signed in to change notification settings - Fork 502
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
java.lang.IllegalStateException: Future was expected to be done #917
Comments
class AudioPlayer(private val service: AudioPlaybackService) : SimpleBasePlayer(service.mainLooper) {
private val state = object : PlayerState(this) {}
override fun getState(): State {
return State.Builder()
.setAvailableCommands(state.availableCommands)
// Here will crash:
.setPlayWhenReady(state.playWhenReady, state.playWhenReadyChangeReason)
.setPlaybackState(state.playbackState)
// Here will crash:
.setPlaylist(
service.audioItemTree.getChildren(":songs", 0, Int.MAX_VALUE).map {
MediaItemData.Builder(it.mediaId)
.setMediaItem(it)
.setDurationUs(it.clippingConfiguration.endPositionMs * 1000)
.setIsSeekable(true)
.build()
}
)
.setCurrentMediaItemIndex(0)
.build()
}
init {
state.availableCommands = Player.Commands.Builder().addAllCommands().build()
}
override fun handleSetPlayWhenReady(playWhenReady: Boolean): ListenableFuture<*> = launchImmediateFuture {
state.playWhenReady = playWhenReady
state.playWhenReadyChangeReason = Player.PLAY_WHEN_READY_CHANGE_REASON_USER_REQUEST
state.playbackState = Player.STATE_READY
}
override fun handlePrepare(): ListenableFuture<*> = launchImmediateFuture {
state.playbackState = Player.STATE_BUFFERING
}
override fun handleStop(): ListenableFuture<*> = launchImmediateFuture {
state.playWhenReady = false
state.playbackState = Player.STATE_IDLE
}
override fun handleRelease(): ListenableFuture<*> = launchImmediateFuture {
state.release()
}
}
abstract class PlayerState(private val player: Player) {
var availableCommands = Player.Commands.EMPTY
var playWhenReady = false
var playWhenReadyChangeReason: @Player.PlayWhenReadyChangeReason Int =
Player.PLAY_WHEN_READY_CHANGE_REASON_USER_REQUEST
var playbackState: @Player.State Int = Player.STATE_IDLE
private val listener = object : Player.Listener {
override fun onAvailableCommandsChanged(availableCommands: Player.Commands) {
this@PlayerState.availableCommands = availableCommands
}
override fun onPlayWhenReadyChanged(playWhenReady: Boolean, reason: Int) {
this@PlayerState.playWhenReady = playWhenReady
this@PlayerState.playWhenReadyChangeReason = reason
}
override fun onPlaybackStateChanged(playbackState: Int) {
this@PlayerState.playbackState = playbackState
}
}
init {
player.addListener(listener)
}
fun release() {
player.removeListener(listener)
}
}
|
Thanks for reporting! From what I see I think you are using threads and the API in a valid way. My hypothesis is that the player has a state that triggers a code path early that is currently not anticipated by the library. I think Media3 should be able to handle an arbitrary valid player state when the session is built with a I'm trying to repro with your setup. The exception seems to be triggered when a bitmap has completed loading and is racing with the media notification controller being connected to the session. It's a bit puzzling that this bitmap completes loading before the media notification controller has connected in which case the Am I right in assuming that you are using an |
No, I am using |
When the media notification controller is requested for a session with `getConnectedControllerForSession` and the `Future` is not null but not yet completed, the `Future` was returned either way. This was reported as creating a race condition between the notification being requested for update the very first time, and the media notification controller having completed connecting to the session. Returning null from `getConnectedControllerForSession` when the `Future` is available but not yet done fixes the problem. This is safe because for the case when a notification update is dropped, the media notification controller will trigger the update as soon as the connection completes. Issue: #917 #minor-release PiperOrigin-RevId: 595699929
When the media notification controller is requested for a session with `getConnectedControllerForSession` and the `Future` is not null but not yet completed, the `Future` was returned either way. This was reported as creating a race condition between the notification being requested for update the very first time, and the media notification controller having completed connecting to the session. Returning null from `getConnectedControllerForSession` when the `Future` is available but not yet done fixes the problem. This is safe because for the case when a notification update is dropped, the media notification controller will trigger the update as soon as the connection completes. Issue: #917 #minor-release PiperOrigin-RevId: 595699929 (cherry picked from commit 5c50b27)
Version
Media3 1.2.0
More version details
Using a thread looper rather than main thread in the custom player, it got crashed.
Then delay few milliseconds at the end of service
onCreate
, it rarely crashed. But this is not a solution.Is it an intented behavior or I mis-used the threading? I don't want player run on main thread, because it blocks UI sometimes.
Devices that reproduce the issue
Pixel 4 XL (Android 13)
Devices that do not reproduce the issue
No response
Reproducible in the demo app?
Not tested
Reproduction steps
Expected result
It won't crash at starup
Actual result
Media
None
Bug Report
adb bugreport
to android-media-github@google.com after filing this issue.The text was updated successfully, but these errors were encountered: