Francis Reynders
03/16/2023, 5:03 PMonExitRescource.useresourceScopeonExitinitonExitlateinit varclass Test() {
  val scope = CoroutineScope(EmptyCoroutineContext)
  lateinit var resource: String
  init {
    scope.launch {
      resourceScope {
        resource = install({"test"}, { _,_ -> Unit })
      }
      awaitCancellation()
    }
  }simon.vergauwen
03/16/2023, 5:42 PMhowever I would like to have the resource available between object initialization andwithonExit
onExitResourceonExitTestTestTestsuspend fun ResourceScope.test(): Test {
  val resource: String = install({"test"}, { _,_ -> Unit })
  Test(resource)
}mainallocateResourcePair<A, suspend (ExitCase) -> Unit>suspend (ExitCase) -> UnitCoroutineScopecancelresourceawaitCancellationresourceScope { }Francis Reynders
03/16/2023, 5:44 PMonExitTestsimon.vergauwen
03/16/2023, 5:46 PMThe customhere has the same issue, which is leaking the resource ifCoroutineScopeis never called.cancel
CoroutineScopeResourceFrancis Reynders
03/16/2023, 6:06 PMallocateinitsimon.vergauwen
03/16/2023, 6:15 PMsuspendacquireallocatesuspend fun ResourceScope.test(): Test {
  val resource: String = install({"test"}, { _,_ -> Unit })
  Test(resource)
}suspend fun test(): Test {
  val (res, finalizer) =
    resource({"test"}, { _,_ -> Unit }).allocate()
  // call finalizer on `onExit` also suspend
  Test(resource, finalizer)
}fun test(): Test {
  val resource = resource({"test"}, { _,_ -> Unit })
  val (res, finalizer) = runBlocking { resource.allocate() }
  // call finalizer on `onExit` also suspend
  Test(resource, finalizer)
}useCloseableTestCloseableResourceResourceSuspendAppCloseableShutdownHookFileHandleFileHandleFrancis Reynders
03/16/2023, 6:21 PMπ just shows it's indeed not the best solution. I guess encapsulating side effects is not really FP style. My idea was to fully encapsulate it and capture all errors within the object, so the user would not be concerned with it. I'll need to rethink this. Thanks a lot for your feedback.is also aCoroutineScopeResource
simon.vergauwen
03/16/2023, 6:24 PM"test"suspend fun ResourceScope.test(): Test {fun Test.Companion.create(): Resource<Test>ResourceEitherTestResourceonExitsimon.vergauwen
03/16/2023, 6:28 PMclass ExternalRes {
  fun close(): Unit = Thread.sleep(100)
  fun doSomething(): Int = throw RuntimeException("Boom!")
}
class Test(private val res: ExternalRes) {
  // don't swallow but do something proper
  // like send log to metric server
  fun safe(): Int =
    catch({ res.doSomething() }) { -1 }
  companion object {
    fun create(): Resource<Test> = resource {
      val res = install({ ExternalRes() }) { res, _ -> res.close() }
      Test(res)
    } 
  }
}
suspend fun userCode() {
  Test.create().use { test ->
    println(test.doSomething()) // -1
  }
  resourceScope {
    val test = Test.create().bind()
    println(test.doSomething()) // -1
  }
}Francis Reynders
03/16/2023, 6:33 PMsimon.vergauwen
03/16/2023, 6:34 PM