main.com.squareup.workflow1.testing.RenderTester.kt Maven / Gradle / Ivy
package com.squareup.workflow1.testing
import com.squareup.workflow1.ActionApplied
import com.squareup.workflow1.SessionWorkflow
import com.squareup.workflow1.StatefulWorkflow
import com.squareup.workflow1.Workflow
import com.squareup.workflow1.WorkflowAction
import com.squareup.workflow1.WorkflowExperimentalApi
import com.squareup.workflow1.WorkflowIdentifier
import com.squareup.workflow1.WorkflowOutput
import com.squareup.workflow1.identifier
import com.squareup.workflow1.testing.RenderTester.ChildWorkflowMatch
import com.squareup.workflow1.workflowIdentifier
import kotlinx.coroutines.CoroutineScope
import kotlin.reflect.KClass
import kotlin.reflect.KType
import kotlin.reflect.KTypeProjection
/**
* Create a [RenderTester] to unit test an individual render pass of this workflow, using the
* workflow's [initial state][StatefulWorkflow.initialState].
*
* See [RenderTester] for usage documentation.
*/
@OptIn(WorkflowExperimentalApi::class) // Opt-in is only for the argument check.
@Suppress("UNCHECKED_CAST")
public fun Workflow.testRender(
props: PropsT
): RenderTester {
val statefulWorkflow = asStatefulWorkflow() as StatefulWorkflow
return statefulWorkflow.testRender(
props = props,
initialState = run {
require(this !is SessionWorkflow) {
"Called testRender on a SessionWorkflow without a CoroutineScope. Use the version that passes a CoroutineScope."
}
statefulWorkflow.initialState(props, null)
}
) as RenderTester
}
/**
* Create a [RenderTester] to unit test an individual render pass of this [SessionWorkflow],
* using the workflow's [initial state][StatefulWorkflow.initialState], in the [workflowScope].
*
* See [RenderTester] for usage documentation.
*/
@OptIn(WorkflowExperimentalApi::class)
@Suppress("UNCHECKED_CAST")
public fun SessionWorkflow.testRender(
props: PropsT,
workflowScope: CoroutineScope
): RenderTester {
val sessionWorkflow: SessionWorkflow =
asStatefulWorkflow() as SessionWorkflow
return sessionWorkflow.testRender(
props = props,
initialState = sessionWorkflow.initialState(props, null, workflowScope)
) as RenderTester
}
/**
* Create a [RenderTester] to unit test an individual render pass of this workflow.
*
* See [RenderTester] for usage documentation.
*/
public fun
StatefulWorkflow.testRender(
props: PropsT,
initialState: StateT
): RenderTester =
RealRenderTester(this, props, initialState)
/**
* The props must be specified, the initial state may be specified, and then all child workflows
* and workers that are expected to run, and any outputs from them, must be specified with
* [expectWorkflow] and (optionally) [expectWorker] and [expectSideEffect] calls.
* If one needs to verify all workers explicitly, perhaps to verify that a worker is *not* run,
* then use [requireExplicitWorkerExpectations]. Likewise [requireExplicitSideEffectExpectations]
* for side effects.
* Then call [render] and perform any assertions on the rendering. An event may also be sent to the
* rendering if no workflows or workers emitted an output. Lastly, the [RenderTestResult] returned
* by `render` may be used to assert on the [WorkflowAction]s processed to handle events or outputs
* by calling [verifyAction][RenderTestResult.verifyAction] or
* [verifyActionResult][RenderTestResult.verifyActionResult].
*
* - All workflows that are rendered/ran by this workflow must be specified.
* - Workers are optionally specified. Specified workers must run. Unexpected workers on a render
* pass do not cause a test failure unless [requireExplicitWorkerExpectations] is true.
* Side effects are optionally specified. Specified side effects must run. Unexpected side effects
* on a render pass do not cause a test failure unless [requireExplicitSideEffectExpectations] is
* true.
* - It is an error if more than one workflow or worker specifies an output.
* - It is a test failure if any workflows or workers that were expected were not ran.
* - It is a test failure if the workflow tried to run any workflows that were not expected.
* - It is a test failure if no workflow or worker emitted an output, no rendering event was
* invoked, and any of the action verification methods on [RenderTestResult] is called.
*
* ## Examples
*
* ### Worker output
*
* The following example tests a render pass that runs one worker, `SubmitLoginWorker`, which
* is configured to have "emitted" an output, and one workflow, `ChildWorkflow`, which expects a
* props containing "[email protected]" and returning a `ChildRendering` as its rendering.
*
* It checks that the rendering properties are expected and that the output handler for the
* `SubmitLoginWorker` returned the `CompleteLogin` action.
*
* ```
* workflow
* .testRender(
* props = MyProps(…),
* initialState = MyState(…)
* )
* .expectWorker(
* workerClass = SubmitLoginWorker::class
* key = "signin",
* output = WorkflowOutput(LoginResponse(success = true))
* )
* .expectWorkflow(
* workflowType = ChildWorkflow::class,
* key = "child",
* assertProps = { assertThat(it.email).isEqualTo("[email protected]") },
* rendering = ChildRendering("message")
* )
* .render { rendering ->
* assertThat(rendering.text).isEqualTo("foo")
* }
* .verifyAction { action ->
* assertThat(action).isEqualTo(Action.CompleteLogin(success = true))
* }
* ```
*
* ### Rendering event
*
* This is similar to the example above, but will test an event sent to the rendering instead.
*
* ```
* workflow
* .testRender(
* props = MyProps(…),
* initialState = MyState(…)
* )
* .expectWorker(
* matchesWhen = { it is SubmitLoginWorker },
* key = "signin"
* )
* .expectWorkflow(
* workflowType = ChildWorkflow::class,
* key = "child",
* assertProps = { assertThat(it.email).isEqualTo("[email protected]") },
* rendering = ChildRendering("message")
* )
* .render { rendering ->
* rendering.onCancelClicked()
* assertThat(rendering.text).isEqualTo("foo")
* }
* .verifyAction { action ->
* assertThat(action).isEqualTo(Action.CancelLogin)
* }
* ```
*
* ### Verify action result
*
* This test verifies the action _result_ instead of the action itself. This technique is useful
* if the [WorkflowAction] is anonymous or inline.
*
* ```
* val currentState = …
* val previousState = …
*
* workflow
* .testRender(
* props = MyProps(…),
* initialState = currentState
* )
* .render { rendering ->
* rendering.onCancelClicked()
* }
* .verifyActionResult { newState, output ->
* // Check that the workflow navigated back correctly.
* assertThat(newState).isEqualTo(previousState)
*
* // Check that the workflow didn't emit any output from the button click.
* assertThat(output).isNull()
* }
* ```
*
* ### Too many outputs
*
* This is an example of what **not** to do – this test will error out because a worker is emitting
* and output _and_ a rendering event is sent.
*
* ```
* workflow
* .testRender(
* props = MyProps(…),
* initialState = MyState(…)
* )
* .expectWorker(
* matchesWhen = { it is SubmitLoginWorker },
* key = "signin",
* output = WorkflowOutput(LoginResponse(success = true))
* )
* .expectWorkflow(
* workflowType = ChildWorkflow::class,
* key = "child",
* assertProps = { assertThat(it.email).isEqualTo("[email protected]") },
* rendering = ChildRendering("message")
* )
* .render { rendering ->
* // This will throw and fail the test because the SubmitLoginWorker is also configured to emit
* // an output.
* rendering.onCancelClicked()
* ```
*/
public abstract class RenderTester {
/**
* Specifies that this render pass is expected to render a particular child workflow.
*
* @param description String that will be used to describe this expectation in error messages.
* The description is required since no human-readable description can be derived from the
* predicate alone.
* @param exactMatch If true, then the test will fail if any other matching expectations are also
* exact matches, and the expectation will only be allowed to match a single child workflow.
* If false, the match will only be used if no other expectations return exclusive matches (in
* which case the first match will be used), and the expectation may match multiple children.
* @param matcher A function that determines whether a given [RenderChildInvocation] matches this
* expectation by returning a [ChildWorkflowMatch]. If the expectation matches, the function
* must include the rendering and optional output for the child workflow.
*/
internal abstract fun expectWorkflow(
description: String,
exactMatch: Boolean = true,
matcher: (RenderChildInvocation) -> ChildWorkflowMatch
): RenderTester
/**
* Specifies that this render pass is expected to run a side effect with a key that satisfies
* [matcher]. This expectation is strict, and will fail if multiple side effects match.
*
* @param description String that will be used to describe this expectation in error messages.
* The description is required since no human-readable description can be derived from the
* predicate alone.
* @param exactMatch If true, then the test will fail if any other matching expectations are also
* exact matches, and the expectation will only be allowed to match a single side effect.
* If false, the match will only be used if no other expectations return exclusive matches (in
* which case the first match will be used), and the expectation may match multiple side effects.
* @param matcher A function that is passed the key value from
* [runningSideEffect][com.squareup.workflow1.BaseRenderContext.runningSideEffect] and return
* true if this key is expected.
*/
public abstract fun expectSideEffect(
description: String,
exactMatch: Boolean = true,
matcher: (key: String) -> Boolean
): RenderTester
/**
* Execute the workflow's `render` method and run [block] to perform assertions on and send events
* to the resulting rendering.
*
* All workflows rendered/ran by the workflow must be specified before calling this
* method. Workers are optionally specified.
*
* @param block Passed the result of the render pass to perform assertions on.
* If no child workflow or worker was configured to emit an output, may also invoke one of the
* rendering's event handlers. It is an error to invoke an event handler if a child emitted an
* output.
* @return A [RenderTestResult] that can be used to verify the [WorkflowAction] that was used to
* handle a workflow or worker output or a rendering event.
*/
public abstract fun render(
block: (rendering: RenderingT) -> Unit = {}
): RenderTestResult
public abstract fun requireExplicitWorkerExpectations():
RenderTester
public abstract fun requireExplicitSideEffectExpectations():
RenderTester
/**
* Describes a call to
* [RenderContext.renderChild][com.squareup.workflow1.BaseRenderContext.renderChild].
*
* ## Output and rendering types
*
* The testing library will attempt to determine the output and rendering types by using
* reflection to determine the type arguments that the concrete workflow class passes to the
* [Workflow] interface. This is subject to the limitations of Kotlin's reflection. Notably, there
* is a compiler bug ([KT-17103](https://youtrack.jetbrains.com/issue/KT-17103)) that prevents
* reflecting on these types when the workflow is an anonymous class that was created by an inline
* function with reified types, such as `Workflow.stateful` and `Workflow.stateless`.
*
* @param workflow The child workflow that is being rendered.
* @param props The props value passed to `renderChild`.
* @param outputType The [KType] of the workflow's `OutputT`.
* @param renderingType The [KType] of the workflow's `RenderingT`.
* @param renderKey The string key passed to `renderChild`.
*/
public class RenderChildInvocation(
public val workflow: Workflow<*, *, *>,
public val props: Any?,
public val outputType: KTypeProjection,
public val renderingType: KTypeProjection,
public val renderKey: String
)
public sealed class ChildWorkflowMatch {
/**
* Indicates that the child workflow did not match the predicate and must match a different
* expectation. The test will fail if all expectations return this value.
*/
public object NotMatched : ChildWorkflowMatch()
/**
* Indicates that the workflow matches the predicate.
*
* @param childRendering The value to return as the child's rendering.
* @param output If non-null, [ActionApplied.output] will be "emitted" when this workflow is
* rendered. The [WorkflowAction] used to handle this output can be verified using methods on
* [RenderTestResult].
*/
public class Matched(
public val childRendering: Any?,
public val output: WorkflowOutput? = null
) : ChildWorkflowMatch()
}
}
/**
* Specifies that this render pass is expected to render a particular child workflow.
*
* Workflow identifiers are compared taking the type hierarchy into account. When a workflow is
* rendered, it will match any expectation that specifies the type of that workflow, or any of
* its supertypes. This means that if you have a workflow that is split into an interface and a
* concrete class, your render tests can pass the class of the interface to this method instead of
* the actual class that implements it.
*
* ## Expecting impostor workflows
*
* If the workflow-under-test renders an
* [ImpostorWorkflow][com.squareup.workflow1.ImpostorWorkflow], the match will not be performed
* using the impostor type, but rather the
* [real identifier][WorkflowIdentifier.getRealIdentifierType] of the impostor's
* [WorkflowIdentifier]. This will be the last identifier in the chain of impostor workflows'
* [realIdentifier][com.squareup.workflow1.ImpostorWorkflow.realIdentifier]s.
*
* A workflow that is wrapped multiple times by various operators will be matched on the upstream
* workflow, so for example the following expectation would succeed:
*
* ```
* val workflow = Workflow.stateless<…> {
* renderChild(
* childWorkflow.mapRendering { … }
* .mapOutput { … }
* )
* }
*
* workflow.testRender(…)
* .expectWorkflow(childWorkflow::class, …)
* ```
*
* @param identifier The [WorkflowIdentifier] of the expected workflow. May identify any supertype
* of the actual rendered workflow, e.g. if the workflow type is an interface and the
* workflow-under-test injects a fake.
* @param rendering The rendering to return from
* [renderChild][com.squareup.workflow1.BaseRenderContext.renderChild] when this workflow is
* rendered.
* @param key The key passed to [renderChild][com.squareup.workflow1.BaseRenderContext.renderChild]
* when rendering this workflow.
* @param assertProps A function that performs assertions on the props passed to
* [renderChild][com.squareup.workflow1.BaseRenderContext.renderChild].
* @param description Optional string that will be used to describe this expectation in error
* messages.
*/
@Suppress("NOTHING_TO_INLINE")
public inline fun
RenderTester.expectWorkflow(
identifier: WorkflowIdentifier,
rendering: ChildRenderingT,
key: String = "",
description: String = "",
noinline assertProps: (props: Any?) -> Unit = {}
): RenderTester =
expectWorkflow(identifier, rendering, null as WorkflowOutput<*>?, key, description, assertProps)
/**
* Specifies that this render pass is expected to render a particular child workflow.
*
* Workflow identifiers are compared taking the type hierarchy into account. When a workflow is
* rendered, it will match any expectation that specifies the type of that workflow, or any of
* its supertypes. This means that if you have a workflow that is split into an interface and a
* concrete class, your render tests can pass the class of the interface to this method instead of
* the actual class that implements it.
*
* ## Expecting impostor workflows
*
* If the workflow-under-test renders an
* [ImpostorWorkflow][com.squareup.workflow1.ImpostorWorkflow], the match will not be performed
* using the impostor type, but rather the
* [real identifier][WorkflowIdentifier.getRealIdentifierType] of the impostor's
* [WorkflowIdentifier]. This will be the last identifier in the chain of impostor workflows'
* [realIdentifier][com.squareup.workflow1.ImpostorWorkflow.realIdentifier]s.
*
* A workflow that is wrapped multiple times by various operators will be matched on the upstream
* workflow, so for example the following expectation would succeed:
*
* ```
* val workflow = Workflow.stateless<…> {
* renderChild(
* childWorkflow.mapRendering { … }
* .mapOutput { … }
* )
* }
*
* workflow.testRender(…)
* .expectWorkflow(childWorkflow::class, …)
* ```
*
* @param identifier The [WorkflowIdentifier] of the expected workflow. May identify any supertype
* of the actual rendered workflow, e.g. if the workflow type is an interface and the
* workflow-under-test injects a fake.
* @param rendering The rendering to return from
* [renderChild][com.squareup.workflow1.BaseRenderContext.renderChild] when this workflow is
* rendered.
* @param key The key passed to [renderChild][com.squareup.workflow1.BaseRenderContext.renderChild]
* when rendering this workflow.
* @param assertProps A function that performs assertions on the props passed to
* [renderChild][com.squareup.workflow1.BaseRenderContext.renderChild].
* @param output If non-null, [WorkflowOutput.value] will be "emitted" when this workflow is
* rendered. The [WorkflowAction] used to handle this output can be verified using methods on
* [RenderTestResult].
* @param description Optional string that will be used to describe this expectation in error
* messages.
*/
public fun
RenderTester.expectWorkflow(
identifier: WorkflowIdentifier,
rendering: ChildRenderingT,
output: WorkflowOutput?,
key: String = "",
description: String = "",
assertProps: (props: Any?) -> Unit = {}
): RenderTester = expectWorkflow(
exactMatch = true,
description = description.ifBlank {
"workflow " +
"identifier=$identifier, " +
"key=$key, " +
"rendering=$rendering, " +
"output=$output"
}
) {
if (it.workflow.identifier.realTypeMatchesExpectation(identifier) &&
it.renderKey == key
) {
assertProps(it.props)
ChildWorkflowMatch.Matched(rendering, output)
} else {
ChildWorkflowMatch.NotMatched
}
}
/**
* Specifies that this render pass is expected to render a particular child workflow.
*
* Workflow identifiers are compared taking the type hierarchy into account. When a workflow is
* rendered, it will match any expectation that specifies the type of that workflow, or any of
* its supertypes. This means that if you have a workflow that is split into an interface and a
* concrete class, your render tests can pass the class of the interface to this method instead of
* the actual class that implements it.
*
* ## Expecting impostor workflows
*
* If the workflow-under-test renders an
* [ImpostorWorkflow][com.squareup.workflow1.ImpostorWorkflow], the match will not be performed
* using the impostor type, but rather the
* [real identifier][WorkflowIdentifier.getRealIdentifierType] of the impostor's
* [WorkflowIdentifier]. This will be the last identifier in the chain of impostor workflows'
* [realIdentifier][com.squareup.workflow1.ImpostorWorkflow.realIdentifier]s.
*
* A workflow that is wrapped multiple times by various operators will be matched on the upstream
* workflow, so for example the following expectation would succeed:
*
* ```
* val workflow = Workflow.stateless<…> {
* renderChild(childWorkflow.mapRendering { … })
* }
*
* workflow.testRender(…)
* .expectWorkflow(childWorkflow::class, …)
* ```
*
* @param workflowType The [KClass] of the expected workflow. May also be any of the supertypes
* of the expected workflow, e.g. if the workflow type is an interface and the workflow-under-test
* injects a fake.
* @param rendering The rendering to return from
* [renderChild][com.squareup.workflow1.BaseRenderContext.renderChild] when this workflow is
* rendered.
* @param key The key passed to [renderChild][com.squareup.workflow1.BaseRenderContext.renderChild]
* when rendering this workflow.
* @param assertProps A function that performs assertions on the props passed to
* [renderChild][com.squareup.workflow1.BaseRenderContext.renderChild].
* @param output If non-null, [WorkflowOutput.value] will be "emitted" when this workflow is
* rendered. The [WorkflowAction] used to handle this output can be verified using methods on
* [RenderTestResult].
* @param description Optional string that will be used to describe this expectation in error
* messages.
*/
public inline fun
RenderTester.expectWorkflow(
workflowType: KClass>,
rendering: ChildRenderingT,
key: String = "",
crossinline assertProps: (props: ChildPropsT) -> Unit = {},
output: WorkflowOutput? = null,
description: String = ""
): RenderTester =
expectWorkflow(
workflowType.workflowIdentifier,
rendering,
key = key,
output = output,
description = description,
assertProps = {
@Suppress("UNCHECKED_CAST")
assertProps(it as ChildPropsT)
}
)
/**
* Specifies that this render pass is expected to run a particular side effect.
*
* @param key The key passed to
* [runningSideEffect][com.squareup.workflow1.BaseRenderContext.runningSideEffect] when rendering
* this workflow.
*/
public fun
RenderTester.expectSideEffect(key: String):
RenderTester =
expectSideEffect("side effect with key \"$key\"", exactMatch = true) { it == key }
© 2015 - 2025 Weber Informatics LLC | Privacy Policy