From c905ad10464b97b612a07c954466969124b8564b Mon Sep 17 00:00:00 2001 From: Ray Ryan Date: Mon, 4 Nov 2024 16:47:27 -0800 Subject: [PATCH] Introduces `SavedStateHandle.removeWorkflowState()` If you're juggling multiple workflow runtimes per `Activity` (if you're the kind of person who needs to call `Job.cancel` on the `Job` returned from `WorkflowLayout.take`) then you might also need to throw away the state captured by an AndroidX `SavedStateHandler` passed to the Android flavor of `renderWorkflowIn()` to prevent memory leaks. --- workflow-ui/core-android/api/core-android.api | 1 + workflow-ui/core-android/build.gradle.kts | 3 + .../ui/AndroidRenderWorkflowInTest.kt | 90 +++++++++++++++++++ .../workflow1/ui/AndroidRenderWorkflow.kt | 14 ++- 4 files changed, 107 insertions(+), 1 deletion(-) create mode 100644 workflow-ui/core-android/src/androidTest/java/com/squareup/workflow1/ui/AndroidRenderWorkflowInTest.kt diff --git a/workflow-ui/core-android/api/core-android.api b/workflow-ui/core-android/api/core-android.api index dcd11bfbd..d0218530c 100644 --- a/workflow-ui/core-android/api/core-android.api +++ b/workflow-ui/core-android/api/core-android.api @@ -1,4 +1,5 @@ public final class com/squareup/workflow1/ui/AndroidRenderWorkflowKt { + public static final fun removeWorkflowState (Landroidx/lifecycle/SavedStateHandle;)V public static final fun renderWorkflowIn (Lcom/squareup/workflow1/Workflow;Lkotlinx/coroutines/CoroutineScope;Landroidx/lifecycle/SavedStateHandle;Ljava/util/List;Ljava/util/Set;Lkotlin/jvm/functions/Function2;)Lkotlinx/coroutines/flow/StateFlow; public static final fun renderWorkflowIn (Lcom/squareup/workflow1/Workflow;Lkotlinx/coroutines/CoroutineScope;Ljava/lang/Object;Landroidx/lifecycle/SavedStateHandle;Ljava/util/List;Ljava/util/Set;Lkotlin/jvm/functions/Function2;)Lkotlinx/coroutines/flow/StateFlow; public static final fun renderWorkflowIn (Lcom/squareup/workflow1/Workflow;Lkotlinx/coroutines/CoroutineScope;Lkotlinx/coroutines/flow/StateFlow;Landroidx/lifecycle/SavedStateHandle;Ljava/util/List;Ljava/util/Set;Lkotlin/jvm/functions/Function2;)Lkotlinx/coroutines/flow/StateFlow; diff --git a/workflow-ui/core-android/build.gradle.kts b/workflow-ui/core-android/build.gradle.kts index 9f634d9f6..89adc8b26 100644 --- a/workflow-ui/core-android/build.gradle.kts +++ b/workflow-ui/core-android/build.gradle.kts @@ -11,7 +11,10 @@ android { } dependencies { + androidTestImplementation(libs.androidx.activity.ktx) androidTestImplementation(libs.androidx.appcompat) + androidTestImplementation(libs.androidx.lifecycle.viewmodel.ktx) + androidTestImplementation(libs.androidx.lifecycle.viewmodel.savedstate) androidTestImplementation(libs.truth) api(libs.androidx.lifecycle.common) diff --git a/workflow-ui/core-android/src/androidTest/java/com/squareup/workflow1/ui/AndroidRenderWorkflowInTest.kt b/workflow-ui/core-android/src/androidTest/java/com/squareup/workflow1/ui/AndroidRenderWorkflowInTest.kt new file mode 100644 index 000000000..8a493351d --- /dev/null +++ b/workflow-ui/core-android/src/androidTest/java/com/squareup/workflow1/ui/AndroidRenderWorkflowInTest.kt @@ -0,0 +1,90 @@ +package com.squareup.workflow1.ui + +import android.widget.FrameLayout +import androidx.activity.ComponentActivity +import androidx.activity.viewModels +import androidx.lifecycle.Lifecycle.State.CREATED +import androidx.lifecycle.SavedStateHandle +import androidx.lifecycle.ViewModel +import androidx.lifecycle.viewModelScope +import androidx.test.ext.junit.rules.ActivityScenarioRule +import com.google.common.truth.Truth.assertThat +import com.squareup.workflow1.StatelessWorkflow +import com.squareup.workflow1.ui.internal.test.IdlingDispatcherRule +import kotlinx.coroutines.Job +import kotlinx.coroutines.flow.StateFlow +import leakcanary.DetectLeaksAfterTestSuccess +import org.junit.Rule +import org.junit.Test +import org.junit.rules.RuleChain + +@OptIn(WorkflowUiExperimentalApi::class) +internal class AndroidRenderWorkflowInTest { + @get:Rule val scenarioRule = ActivityScenarioRule(ComponentActivity::class.java) + private val scenario get() = scenarioRule.scenario + + @get:Rule val rules: RuleChain = RuleChain.outerRule(DetectLeaksAfterTestSuccess()) + .around(scenarioRule) + .around(IdlingDispatcherRule) + + @Test fun removeWorkflowStateDoesWhatItSaysOnTheTin() { + var job: Job? = null + + // Activity.onCreate(), the take() call won't start pulling yet. + scenario.onActivity { activity -> + val model: SomeViewModel by activity.viewModels() + val renderings: StateFlow = renderWorkflowIn( + workflow = SomeWorkflow, + scope = model.viewModelScope, + savedStateHandle = model.savedStateHandle + ) + + val layout = WorkflowLayout(activity) + activity.setContentView(layout) + + assertThat(model.savedStateHandle.contains(KEY)).isFalse() + + job = layout.take(activity.lifecycle, renderings) + assertThat(model.savedStateHandle.contains(KEY)).isFalse() + } + + // Exit onCreate() and move to CREATED status. take() starts to draw + // and the renderWorkflowIn() call above starts pushing TreeSnapshots + // (lazy serialization functions) to the SavedStateHandle. + scenario.moveToState(CREATED) + scenario.onActivity { activity -> + val model: SomeViewModel by activity.viewModels() + assertThat(model.savedStateHandle.contains(KEY)).isTrue() + + // The Job returned from take() is canceled. There is still a + // TreeSnapshot and whatever pointer it captured in the SavedStateHandle. + job?.cancel() + assertThat(model.savedStateHandle.contains(KEY)).isTrue() + + // We can remove it. + model.savedStateHandle.removeWorkflowState() + assertThat(model.savedStateHandle.contains(KEY)).isFalse() + } + } + + object SomeScreen : AndroidScreen { + override val viewFactory: ScreenViewFactory = + ScreenViewFactory.fromCode { _, initialEnvironment, context, _ -> + ScreenViewHolder( + initialEnvironment, + FrameLayout(context) + ) { _, _ -> } + } + } + + object SomeWorkflow : StatelessWorkflow() { + override fun render( + renderProps: Unit, + context: RenderContext + ): Screen { + return SomeScreen + } + } + + class SomeViewModel(val savedStateHandle: SavedStateHandle) : ViewModel() +} diff --git a/workflow-ui/core-android/src/main/java/com/squareup/workflow1/ui/AndroidRenderWorkflow.kt b/workflow-ui/core-android/src/main/java/com/squareup/workflow1/ui/AndroidRenderWorkflow.kt index feba25380..2ba4defea 100644 --- a/workflow-ui/core-android/src/main/java/com/squareup/workflow1/ui/AndroidRenderWorkflow.kt +++ b/workflow-ui/core-android/src/main/java/com/squareup/workflow1/ui/AndroidRenderWorkflow.kt @@ -1,5 +1,6 @@ package com.squareup.workflow1.ui +import androidx.annotation.VisibleForTesting import androidx.lifecycle.SavedStateHandle import com.squareup.workflow1.RuntimeConfig import com.squareup.workflow1.RuntimeConfigOptions @@ -290,4 +291,15 @@ public fun renderWorkflowIn( .stateIn(scope, Eagerly, renderingsAndSnapshots.value.rendering) } -private const val KEY = "com.squareup.workflow1.ui.renderWorkflowIn-snapshot" +/** + * Removes state added to the `savedStateHandle` argument of the Android-specific + * overload of [renderWorkflowIn]. For use in obscure cases like swapping between + * different Workflow runtimes in an app. Most apps will not use this function. + */ +@WorkflowUiExperimentalApi +public fun SavedStateHandle.removeWorkflowState() { + remove(KEY) +} + +@VisibleForTesting +internal const val KEY = "com.squareup.workflow1.ui.renderWorkflowIn-snapshot"