Bird
Raised Fist0
PyTesttesting~8 mins

Test independence in PyTest - Framework Patterns

Choose your learning style10 modes available

Start learning this pattern below

Jump into concepts and practice - no test required

or
Recommended
Test this pattern10 questions across easy, medium, and hard to know if this pattern is strong
Framework Mode - Test independence
Folder Structure
project-root/
├── tests/
│   ├── test_login.py
│   ├── test_cart.py
│   ├── test_checkout.py
│   └── conftest.py
├── src/
│   └── app_code.py
├── utils/
│   └── helpers.py
├── pytest.ini
└── requirements.txt
    
Test Framework Layers
  • Tests: Individual test files in tests/ folder. Each test is independent and does not rely on others.
  • Fixtures: Setup and teardown code in conftest.py to prepare test data or environment fresh for each test.
  • Utilities: Helper functions in utils/helpers.py to avoid repeating code inside tests.
  • Application Code: The main code under test in src/app_code.py.
  • Configuration: pytest.ini for pytest settings and options.
Configuration Patterns

Use pytest.ini to configure pytest options like markers and test paths.

Use conftest.py to define fixtures that provide fresh setup for each test, ensuring no shared state.

Example fixture for fresh user data:

import pytest

@pytest.fixture
def new_user():
    return {"username": "testuser", "password": "pass123"}
    

Each test uses fixtures to get clean data and environment, avoiding dependencies between tests.

Test Reporting and CI/CD Integration

Use pytest built-in reporting with --junitxml=report.xml to generate XML reports.

Integrate with CI/CD tools (like GitHub Actions, Jenkins) to run tests on every code push.

Test independence ensures that failures are isolated and easy to diagnose in reports.

Best Practices for Test Independence
  • Isolate Tests: Each test should run alone without relying on others.
  • Use Fixtures: Setup fresh data and environment for each test using pytest fixtures.
  • No Shared State: Avoid global variables or shared data that can cause side effects.
  • Clear Setup and Teardown: Clean up after tests to avoid leftover data affecting others.
  • Idempotent Tests: Running tests multiple times should produce the same result.
Self Check

Where in this framework structure would you add a fixture that provides a fresh database connection for each test?

Key Result
Use pytest fixtures to ensure each test runs independently with fresh setup and no shared state.

Practice

(1/5)
1. Why is test independence important in pytest?
easy
A. It groups tests to run in a specific order.
B. It allows tests to share variables for faster execution.
C. It ensures tests do not affect each other and run reliably alone.
D. It makes tests run only when previous tests pass.

Solution

  1. Step 1: Understand test independence concept

    Test independence means each test runs alone without relying on others.
  2. Step 2: Identify why independence matters

    This prevents tests from failing due to side effects or order, making results reliable.
  3. Final Answer:

    It ensures tests do not affect each other and run reliably alone. -> Option C
  4. Quick Check:

    Test independence = tests run alone [OK]
Hint: Tests should run alone without relying on others [OK]
Common Mistakes:
  • Thinking tests must share data to be efficient
  • Believing tests run only if previous tests pass
  • Assuming test order controls correctness
2. Which pytest feature helps keep tests independent by running setup code before each test?
easy
A. Using setup_method or setup_function
B. Using yield_fixture to share data
C. Using @pytest.mark.parametrize
D. Using pytest.skip() to skip tests

Solution

  1. Step 1: Identify setup features in pytest

    Pytest runs setup code before each test using setup_method or setup_function.
  2. Step 2: Understand their role in test independence

    Setup prepares fresh state for each test, avoiding shared state and keeping tests independent.
  3. Final Answer:

    Using setup_method or setup_function -> Option A
  4. Quick Check:

    Setup before each test = setup_method/setup_function [OK]
Hint: Setup code before each test keeps tests independent [OK]
Common Mistakes:
  • Confusing parameterize with setup
  • Using yield_fixture to share state incorrectly
  • Skipping tests does not setup state
3. Given the code below, what will be the output when running both tests?
counter = 0

def test_first():
    global counter
    counter += 1
    assert counter == 1

def test_second():
    global counter
    counter += 1
    assert counter == 1
medium
A. First test passes, second test fails
B. Both tests pass
C. First test fails, second test passes
D. Both tests fail

Solution

  1. Step 1: Analyze test_first behavior

    Initially, counter=0. test_first increments to 1 and asserts counter == 1, so it passes.
  2. Step 2: Analyze test_second behavior

    counter is now 1 from previous test. test_second increments to 2 and asserts counter == 1, which fails.
  3. Final Answer:

    First test passes, second test fails -> Option A
  4. Quick Check:

    Shared state causes second test failure [OK]
Hint: Shared global state breaks test independence [OK]
Common Mistakes:
  • Assuming counter resets automatically
  • Thinking both tests run with fresh state
  • Ignoring global variable effects
4. Identify the problem in this pytest code that breaks test independence:
shared_list = []

def test_add_one():
    shared_list.append(1)
    assert len(shared_list) == 1

def test_add_two():
    shared_list.append(2)
    assert len(shared_list) == 1
medium
A. shared_list should be a global variable inside tests
B. Assertions are incorrect; they should check for length 2
C. Tests run in parallel causing race conditions
D. shared_list is not cleared between tests causing length to grow

Solution

  1. Step 1: Understand shared_list usage

    shared_list is defined outside tests and not reset, so it keeps growing with each test.
  2. Step 2: Identify why independence breaks

    Because shared_list is not cleared, test_add_two sees length 2, but asserts length 1, causing failure.
  3. Final Answer:

    shared_list is not cleared between tests causing length to grow -> Option D
  4. Quick Check:

    Shared mutable state without reset breaks independence [OK]
Hint: Reset shared data between tests to keep independence [OK]
Common Mistakes:
  • Assuming pytest resets global variables automatically
  • Changing assertions instead of fixing state
  • Confusing parallel runs with shared state issues
5. You want to ensure two pytest tests that modify a database table run independently. Which approach best maintains test independence?
hard
A. Run tests in a fixed order so changes apply sequentially
B. Use a fixture to create and rollback a transaction for each test
C. Share a global database connection and commit changes after all tests
D. Skip tests that depend on database state

Solution

  1. Step 1: Understand database state isolation

    To keep tests independent, each test should not affect others' database state.
  2. Step 2: Choose best isolation method

    Using a fixture that creates a transaction and rolls back after each test resets database state, ensuring independence.
  3. Final Answer:

    Use a fixture to create and rollback a transaction for each test -> Option B
  4. Quick Check:

    Transaction rollback per test = independence [OK]
Hint: Rollback DB changes after each test to isolate state [OK]
Common Mistakes:
  • Relying on test order for correctness
  • Sharing global DB connection without isolation
  • Skipping tests instead of fixing independence