Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Suspend coroutine until condition is true

I have a use case where I need to connect and disconnect from a class that acts as a service. Actions can be performed on the service only when the service is connected. Clients are notified when the service connects or disconnects by a callback:

class Service {

    constructor(callback: ConnectionCallback) { ... }

    fun connect() {
        // Call callback.onConnected() some time after this method returns.
    }

    fun disconnect() {
        // Call callback.onConnectionSuspended() some time after this method returns.
    }

    fun isConnected(): Boolean { ... }

    fun performAction(actionName: String, callback: ActionCallback) {
        // Perform a given action on the service, failing with a fatal exception if called when the service is not connected.
    }

    interface ConnectionCallback {
        fun onConnected() // May be called multiple times
        fun onConnectionSuspended() // May be called multiple times
        fun onConnectionFailed()
    }
}

I'd like to write a wrapper for that Service class (that I don't control) using Kotlin Coroutines. Here is a skeleton of ServiceWrapper:

class ServiceWrapper {
    private val service = Service(object : ConnectionCallback { ... })

    fun connect() {
        service.connect()
    }

    fun disconnect() {
        service.disconnect()
    }

    suspend fun performActionWhenConnected(actionName: String): ActionResult {
        suspendUntilConnected()

        return suspendCoroutine { continuation ->
            service.performAction(actionName, object : ActionCallback() {
                override fun onSuccess(result: ActionResult) {
                    continuation.resume(result)
                }

                override fun onError() {
                    continuation.resumeWithException(RuntimeException())
                }
            }
        }
    }
}

How can I implement this suspendUntilConnected() behavior using Coroutines ? Thanks in advance.

like image 817
Thibault Seisel Avatar asked Dec 23 '18 18:12

Thibault Seisel


2 Answers

Here's how you can implement it:

class ServiceWrapper {
    @Volatile
    private var deferredUntilConnected = CompletableDeferred<Unit>()

    private val service = Service(object : ConnectionCallback {
        override fun onConnected() {
            deferredUntilConnected.complete(Unit)
        }

        override fun onConnectionSuspended() {
            deferredUntilConnected = CompletableDeferred()
        }
    })

    private suspend fun suspendUntilConnected() = deferredUntilConnected.await()

    ...
}

A general note: just because the service got connected at a certain point doesn't guarantee it will still be connected by the time you use it.

like image 184
Marko Topolnik Avatar answered Oct 22 '22 08:10

Marko Topolnik


Another approach with StateFlow

  1. Define some "state" first
enum class ServiceState {
    CONNECTED, SUSPENDED, FAILED
}
  1. Convert the callback-based code to flow
val connectionState = MutableStateFlow(ServiceState.FAILED)

private val service = Service(object : ConnectionCallback {
    override fun onConnected() {
        connectionState.value = ServiceState.CONNECTED
    }

    override fun onConnectionSuspended() {
        connectionState.value = ServiceState.SUSPENDED
    }

    override fun onConnectionFailed() {
        connectionState.value = ServiceState.FAILED
    }
})
  1. Copy and paste these utils code
class ConditionalAwait<T>(
    private val stateFlow: StateFlow<T>,
    private val condition: (T) -> Boolean
) {
    suspend fun await(): T {
        val nowValue = stateFlow.value
        return if (condition(nowValue)) {
            nowValue
        } else {
            stateFlow.first { condition(it) }
        }
    }
}

suspend fun <T> StateFlow<T>.conditionalAwait(condition: (T) -> Boolean): T =
    ConditionalAwait(this, condition).await()
  1. USE IT~
suspend fun performActionWhenConnected() {
    connectionState.conditionalAwait { it == ServiceState.CONNECTED }
    // other actions when service is connected
}

  1. Advanced usage
suspend fun performActionWhenConnected() {
    val state = connectionState.conditionalAwait { 
        it == ServiceState.CONNECTED || it == ServiceState.FAILED
    } // keep suspended when Service.SUSPENDED

    if (state is ServiceState.CONNECTED) {
        // other actions when service is connected
    } else {
        // error handling
    }
}
like image 1
Owen Chen Avatar answered Oct 22 '22 09:10

Owen Chen