From 9987f18109029d7516003690fcf70482a48e4599 Mon Sep 17 00:00:00 2001 From: Aleksei Tiurin Date: Sat, 30 Nov 2024 16:39:57 +0200 Subject: [PATCH] improve readme --- .../commonTest/kotlin/UltronTestFlowTest.kt | 2 +- .../commonTest/kotlin/UltronTestFlowTest2.kt | 2 +- docs/docs/common/ultrontest.md | 61 ++++++++++++++----- .../tests/testlifecycle/UltronTestFlowTest.kt | 2 +- .../testlifecycle/UltronTestFlowTest2.kt | 2 +- .../atiurin/ultron/core/test/UltronTest.kt | 4 +- .../test/context/DefaultUltronTestContext.kt | 2 +- .../DefaultUltronTestContextProvider.kt | 2 +- 8 files changed, 55 insertions(+), 22 deletions(-) diff --git a/composeApp/src/commonTest/kotlin/UltronTestFlowTest.kt b/composeApp/src/commonTest/kotlin/UltronTestFlowTest.kt index c23ee3d..bba2037 100644 --- a/composeApp/src/commonTest/kotlin/UltronTestFlowTest.kt +++ b/composeApp/src/commonTest/kotlin/UltronTestFlowTest.kt @@ -16,7 +16,7 @@ class UltronTestFlowTest : UltronTest() { } @OptIn(ExperimentalUltronApi::class) - override val beforeFirstTests = { + override val beforeFirstTest = { beforeAllTestCounter = order UltronLog.info("Before Class") } diff --git a/composeApp/src/commonTest/kotlin/UltronTestFlowTest2.kt b/composeApp/src/commonTest/kotlin/UltronTestFlowTest2.kt index 56622a0..36afd43 100644 --- a/composeApp/src/commonTest/kotlin/UltronTestFlowTest2.kt +++ b/composeApp/src/commonTest/kotlin/UltronTestFlowTest2.kt @@ -9,7 +9,7 @@ class UltronTestFlowTest2 : UltronTest() { var beforeAllTestCounter = 0 @OptIn(ExperimentalUltronApi::class) - override val beforeFirstTests = { + override val beforeFirstTest = { beforeAllTestCounter = order order++ UltronLog.info("Before Class") diff --git a/docs/docs/common/ultrontest.md b/docs/docs/common/ultrontest.md index dda8962..08ebae8 100644 --- a/docs/docs/common/ultrontest.md +++ b/docs/docs/common/ultrontest.md @@ -4,15 +4,13 @@ sidebar_position: 2 # UltronTest -Using `UltronTest` as a Base Class for Tests - `UltronTest` is a powerful base class provided by the Ultron framework that enables the definition of common preconditions and postconditions for tests. By extending this class, you can streamline test setup and teardown, ensuring consistent execution across your test suite. ## Features of `UltronTest` - **Pre-Test Actions:** Define actions to be executed before each test. - **Post-Test Actions:** Define actions to be executed after each test. -- **Lifecycle Management:** Execute code once before all tests in a class using `beforeFirstTests`. +- **Lifecycle Management:** Execute code once before all tests in a class using `beforeFirstTest`. - **Customizable Test Execution:** Suppress pre-test or post-test actions when needed. ### Example @@ -23,7 +21,7 @@ Here is an example of using `UltronTest`: class SampleUltronFlowTest : UltronTest() { @OptIn(ExperimentalUltronApi::class) - override val beforeFirstTests = { + override val beforeFirstTest = { UltronLog.info("Before Class") } @@ -36,8 +34,8 @@ class SampleUltronFlowTest : UltronTest() { } /** - * An order of methods execution is follow: - * beforeFirstTests, beforeTest, before, go, after, afterTest + * The order of method execution is as follows:: + * beforeFirstTest, beforeTest, before, go, after, afterTest */ @Test fun someTest1() = test { @@ -52,9 +50,9 @@ class SampleUltronFlowTest : UltronTest() { /** * An order of methods execution is follow: before, go, after - * `beforeFirstTests` - not executed, since it was executed before `someTest1` - * `beforeTest` - not executed, as it was suppressed - * `afterTest` - not executed, as it was suppressed + * `beforeFirstTest` - Not executed, as it is only run once and was already executed before `someTest1`. + * `beforeTest` - Not executed because it was suppressed using `suppressCommonBefore`. + * `afterTest` - Not executed because it was suppressed using `suppressCommonAfter`. */ @Test fun someTest2() = test( @@ -72,7 +70,7 @@ class SampleUltronFlowTest : UltronTest() { /** * An order of methods execution is follow: beforeTest, test, afterTest - * `beforeFirstTests` - not executed, since it was executed before `someTest1` + * `beforeFirstTest` - Not executed, since it was executed before `someTest1` */ @Test fun someTest3() = test { @@ -83,13 +81,48 @@ class SampleUltronFlowTest : UltronTest() { ### Key Methods -- **`beforeFirstTests`**: Code executed once before all tests in a class. +- **`beforeFirstTest`**: Code executed once before all tests in a class. - **`beforeTest`**: Code executed before each test. - **`afterTest`**: Code executed after each test. - **`test`**: Executes a test with options to suppress pre-test or post-test actions. +### Key Features of the `test` Method + +- **Test Context Recreation:** + The `test` method automatically recreates the `UltronTestContext` for each test execution, ensuring a clean and isolated state for the test context. + +- **Soft Assertion Reset:** + Any exceptions captured during `softAssertions` in the previous test are cleared at the start of each new `test` execution, maintaining a clean state. + +- **Lifecycle Management:** + It invokes `beforeTest` and `afterTest` methods around your test logic unless explicitly suppressed. --- +### Purpose of `before`, `go`, and `after` +- **`before`:** Defines preconditions or setup actions that must be performed before the main test logic is executed. +These actions might include preparing data, navigating to a specific screen, or setting up the environment. + ```kotlin + before { + UltronLog.info("Setting up preconditions for TestMethod 2") + } + ``` + +- **`go`:** Encapsulates the core logic or actions of the test. This is where the actual operations being tested are performed, such as interacting with UI elements or executing specific functionality. + ```kotlin + go { + UltronLog.info("Executing the main logic of TestMethod 2") + } + ``` + +- **`after`:** Block is used for postconditions or cleanup actions that need to occur after the main test logic has executed. This might include verifying results, resetting the environment, or clearing resources. + ```kotlin + after { + UltronLog.info("Cleaning up after TestMethod 2") + } + ``` + +These methods help clearly separate test phases, making tests easier to read and maintain. + ## Using `softAssertion` for Flexible Error Handling The `softAssertion` mechanism in Ultron allows tests to catch and verify multiple exceptions during their execution without failing immediately. This feature is particularly useful for validating multiple conditions within a single test. @@ -108,8 +141,8 @@ class SampleTest : UltronTest() { } ``` -The `softAssertion` is not generally depends on `UltronTest`. -You can use it outside `UltronTest` base class. But you have to clear exceptions between tests +The `softAssertion` mechanism does not inherently depend on `UltronTest`. +You can use `softAssertion` independently of the `UltronTest` base class. However, in such cases, you must manually clear exceptions between tests to ensure they do not persist across test executions. ```kotlin class SampleTest { @Test @@ -131,7 +164,7 @@ This approach ensures granular control over how exceptions are handled and repor --- -## Benefits +## Benefits of `UltronTest` usage - Simplifies test setup and teardown with consistent preconditions and postconditions. - Enhances error handling by allowing multiple assertions within a single test. diff --git a/sample-app/src/androidTest/java/com/atiurin/sampleapp/tests/testlifecycle/UltronTestFlowTest.kt b/sample-app/src/androidTest/java/com/atiurin/sampleapp/tests/testlifecycle/UltronTestFlowTest.kt index 239b565..09c2734 100644 --- a/sample-app/src/androidTest/java/com/atiurin/sampleapp/tests/testlifecycle/UltronTestFlowTest.kt +++ b/sample-app/src/androidTest/java/com/atiurin/sampleapp/tests/testlifecycle/UltronTestFlowTest.kt @@ -16,7 +16,7 @@ class UltronTestFlowTest : BaseTest() { } @OptIn(ExperimentalUltronApi::class) - override val beforeFirstTests = { + override val beforeFirstTest = { beforeAllTestCounter = order UltronLog.info("Before Class") } diff --git a/sample-app/src/androidTest/java/com/atiurin/sampleapp/tests/testlifecycle/UltronTestFlowTest2.kt b/sample-app/src/androidTest/java/com/atiurin/sampleapp/tests/testlifecycle/UltronTestFlowTest2.kt index ad38b1d..c91910f 100644 --- a/sample-app/src/androidTest/java/com/atiurin/sampleapp/tests/testlifecycle/UltronTestFlowTest2.kt +++ b/sample-app/src/androidTest/java/com/atiurin/sampleapp/tests/testlifecycle/UltronTestFlowTest2.kt @@ -9,7 +9,7 @@ class UltronTestFlowTest2 : BaseTest() { var order = 0 var beforeAllTestCounter = 0 @OptIn(ExperimentalUltronApi::class) - override val beforeFirstTests = { + override val beforeFirstTest = { beforeAllTestCounter = order order++ UltronLog.info("Before Class") diff --git a/ultron-common/src/commonMain/kotlin/com/atiurin/ultron/core/test/UltronTest.kt b/ultron-common/src/commonMain/kotlin/com/atiurin/ultron/core/test/UltronTest.kt index b66628f..1700417 100644 --- a/ultron-common/src/commonMain/kotlin/com/atiurin/ultron/core/test/UltronTest.kt +++ b/ultron-common/src/commonMain/kotlin/com/atiurin/ultron/core/test/UltronTest.kt @@ -34,7 +34,7 @@ open class UltronTest( * Can be overridden in subclasses. */ @ExperimentalUltronApi - open val beforeFirstTests: () -> Unit = {} + open val beforeFirstTest: () -> Unit = {} /** * Function to be executed before each test. @@ -64,7 +64,7 @@ open class UltronTest( TestMethod(testContextProvider.provide()).apply { // Ensure `beforeAllTests` is executed only once per class if (beforeAllTestsExecutionMap[className] != true) { - beforeFirstTests() + beforeFirstTest() beforeAllTestsExecutionMap[className] = true } diff --git a/ultron-common/src/commonMain/kotlin/com/atiurin/ultron/core/test/context/DefaultUltronTestContext.kt b/ultron-common/src/commonMain/kotlin/com/atiurin/ultron/core/test/context/DefaultUltronTestContext.kt index ba670d2..7f60e24 100644 --- a/ultron-common/src/commonMain/kotlin/com/atiurin/ultron/core/test/context/DefaultUltronTestContext.kt +++ b/ultron-common/src/commonMain/kotlin/com/atiurin/ultron/core/test/context/DefaultUltronTestContext.kt @@ -3,7 +3,7 @@ package com.atiurin.ultron.core.test.context import com.atiurin.ultron.core.common.resultanalyzer.DefaultSoftAssertionOperationResultAnalyzer import com.atiurin.ultron.core.common.resultanalyzer.OperationResultAnalyzer -class DefaultUltronTestContext : UltronTestContext { +open class DefaultUltronTestContext : UltronTestContext { override var softAssertion: Boolean = false override val softAnalyzer = DefaultSoftAssertionOperationResultAnalyzer() diff --git a/ultron-common/src/commonMain/kotlin/com/atiurin/ultron/core/test/context/DefaultUltronTestContextProvider.kt b/ultron-common/src/commonMain/kotlin/com/atiurin/ultron/core/test/context/DefaultUltronTestContextProvider.kt index 9b70e9d..a600e69 100644 --- a/ultron-common/src/commonMain/kotlin/com/atiurin/ultron/core/test/context/DefaultUltronTestContextProvider.kt +++ b/ultron-common/src/commonMain/kotlin/com/atiurin/ultron/core/test/context/DefaultUltronTestContextProvider.kt @@ -1,6 +1,6 @@ package com.atiurin.ultron.core.test.context -class DefaultUltronTestContextProvider : UltronTestContextProvider { +open class DefaultUltronTestContextProvider : UltronTestContextProvider { override fun provide(): UltronTestContext { return DefaultUltronTestContext() }