Skip to main content

Detect test environment - pytest, doctest, unittest, or regular execution

Project description

lib_detect_testenv

CI CodeQL License: MIT Open in Codespaces PyPI PyPI - Downloads Code Style: Ruff codecov Maintainability Known Vulnerabilities security: bandit

Detect test environments: pytest, doctest, and setup.py test

This library provides utility functions to detect whether your code is running in a test environment. It supports detection of pytest, doctest (including PyCharm's docrunner), and setup.py test execution contexts.

Table of Contents

Installation

Recommended: Install via UV

UV is an ultrafast Python package installer written in Rust (10–20× faster than pip/poetry):

# Install uv
pip install --upgrade uv

# Create and activate a virtual environment (optional but recommended)
uv venv

# macOS/Linux
source .venv/bin/activate
# Windows (PowerShell)
.venv\Scripts\Activate.ps1

# Install from PyPI
uv pip install lib_detect_testenv

Alternative: Install via pip

# Upgrade pip first
python -m pip install --upgrade pip

# Install from PyPI
python -m pip install --upgrade lib_detect_testenv

# Or install with test dependencies
python -m pip install --upgrade lib_detect_testenv[test]

For more installation methods (pipx, source builds, etc.), see INSTALL.md.

Python Version Requirements

  • Python 3.9+ is required
  • Tested on Linux, macOS, and Windows
  • CI runs on CPython 3.9+ and PyPy

Usage

Command Line Interface (CLI)

The package provides a CLI for shell scripting and CI/CD integration. All commands use exit codes for programmatic usage.

Exit Codes

All detection commands use a 3-level exit code system:

Code Meaning Use Case
0 Test environment detected Success - test env found
1 Test environment not detected Normal negative result (not an error)
2 Error occurred Command failed (bad arguments, exception)

This design follows Unix conventions and makes shell scripting intuitive:

# Check if in test environment (exit 0 = yes, exit 1 = no)
if lib_detect_testenv check --quiet; then
    echo "Running in test mode"
else
    echo "Running in production"
fi

# Handle errors separately
lib_detect_testenv check --quiet
case $? in
    0) echo "Test environment detected" ;;
    1) echo "No test environment" ;;
    2) echo "Error occurred" ;;
esac

Available Commands

# Test Detection Commands (exit codes: 0=detected, 1=not detected, 2=error)
lib_detect_testenv check                    # Check any test environment
lib_detect_testenv check --quiet            # Silent mode, use exit code only
lib_detect_testenv check --arg-string "pytest test.py"  # Check custom string

lib_detect_testenv pytest                   # Check for pytest
lib_detect_testenv doctest                  # Check for doctest
lib_detect_testenv setup                    # Check for setup.py
lib_detect_testenv setup --test-only        # Check for "setup.py test" specifically

# Utility Commands
lib_detect_testenv info                     # Show package information
lib_detect_testenv hello                    # Demo: print greeting
lib_detect_testenv fail                     # Demo: trigger intentional failure

# Help
lib_detect_testenv --help                   # Show all commands
lib_detect_testenv check --help             # Command-specific help

CLI Examples

# Use in shell scripts
if lib_detect_testenv check --quiet; then
    echo "Running in test environment"
fi

# Use in CI/CD pipelines
lib_detect_testenv pytest --quiet && echo "pytest detected"

# Check with custom arguments
lib_detect_testenv pytest --arg-string "/pytest/__main__.py"

Python API

The library exports the following functions from lib_detect_testenv:

from lib_detect_testenv import (
    is_testenv_active,      # Detect any test environment
    is_doctest_active,      # Detect doctest/docrunner
    is_pytest_active,       # Detect pytest
    is_setup_active,        # Detect setup.py
    is_setup_test_active,   # Detect setup.py test
    is_doctest_in_arg_string,  # Check if doctest in arg string
    add_path_to_syspath,    # Add path to sys.path
    PathLikeOrString,       # Type alias for paths
)

Quick Start

The most common use case is detecting whether any test environment is active:

from lib_detect_testenv import is_testenv_active

if is_testenv_active():
    print("Running in test environment")
else:
    print("Running in production")

Function Reference

is_testenv_active(arg_string: Optional[str] = None) -> bool

Returns True if any test environment is detected (pytest, doctest, or setup.py test).

Parameters:

  • arg_string (optional): If None, uses str(sys.argv())

Returns:

  • True if test environment is detected

Example:

from lib_detect_testenv import is_testenv_active

# Auto-detect from sys.argv
if is_testenv_active():
    print("Test environment detected")

# Or pass custom arg string
if is_testenv_active(arg_string="pytest test.py"):
    print("pytest detected")

is_doctest_active(arg_string: Optional[str] = None) -> bool

Returns True if PyCharm's docrunner.py or doctest.py is detected.

Parameters:

  • arg_string (optional): If None, uses str(sys.argv())

Returns:

  • True if doctest/docrunner is detected

Example:

from lib_detect_testenv import is_doctest_active

# Examples
assert is_doctest_active(arg_string="") == False
assert is_doctest_active(arg_string="docrunner.py") == True
assert is_doctest_active(arg_string="doctest.py") == True

is_pytest_active(arg_string: Optional[str] = None) -> bool

Returns True if pytest is detected.

Parameters:

  • arg_string (optional): If None, uses str(sys.argv())

Returns:

  • True if pytest is detected

Example:

from lib_detect_testenv import is_pytest_active

# Examples
assert is_pytest_active(arg_string="pytest.py") == True
assert is_pytest_active(arg_string="/pytest/__main__.py") == True
assert is_pytest_active(arg_string="python -m pytest") == True

is_setup_active(arg_string: Optional[str] = None) -> bool

Returns True if setup.py is detected.

Parameters:

  • arg_string (optional): If None, uses str(sys.argv())

Returns:

  • True if setup.py is detected

Example:

from lib_detect_testenv import is_setup_active

# Examples
assert is_setup_active(arg_string="") == False
assert is_setup_active(arg_string="setup.py") == True

is_setup_test_active(arg_string: Optional[str] = None) -> bool

Returns True if "setup.py test" is detected.

Parameters:

  • arg_string (optional): If None, uses str(sys.argv())

Returns:

  • True if "setup.py test" is detected

Example:

from lib_detect_testenv import is_setup_test_active

# Examples
assert is_setup_test_active('') == False
assert is_setup_test_active('setup.py') == False
assert is_setup_test_active('setup.py test') == True

is_doctest_in_arg_string(arg_string: str) -> bool

Checks if docrunner.py is present in the argument string.

Parameters:

  • arg_string: String to check

Returns:

  • True if docrunner.py is found

Example:

from lib_detect_testenv import is_doctest_in_arg_string

# Examples
assert is_doctest_in_arg_string('test') == False
assert is_doctest_in_arg_string('test/docrunner.py::::test') == True

add_path_to_syspath(path_to_append: PathLikeOrString) -> None

Adds a path to sys.path if not already present. If the path is a file, its parent directory will be added instead.

Parameters:

  • path_to_append: Path to append (string or PathLike object). Will be resolved to absolute path. If it's a file, the parent directory is added.

Returns:

  • None

Example:

import pathlib
from lib_detect_testenv import add_path_to_syspath

# Add a directory
add_path_to_syspath(pathlib.Path(__file__).parent)

# Add a file (parent directory will be added)
add_path_to_syspath(pathlib.Path(__file__))

# Adding the same path twice won't create duplicates
add_path_to_syspath("/some/path")
add_path_to_syspath("/some/path")  # No duplicate added

Common Pattern: Auto-add Package to sys.path in Tests

A useful pattern is to automatically add your package directory to sys.path when running tests. This is helpful for local testing without installing the package:

# __init__.py
from lib_detect_testenv import is_testenv_active, add_path_to_syspath
import pathlib

if is_testenv_active():
    # Add parent directory to sys.path for local testing
    add_path_to_syspath(pathlib.Path(__file__).parent)

Development

For development setup, testing, and contribution guidelines, see:

Quick Development Setup

# Clone the repository
git clone https://github.com/bitranox/lib_detect_testenv.git
cd lib_detect_testenv

# Run tests (sets up environment, runs tests with coverage)
make test

# Or use the Python scripts directly
python -m scripts.test

Further Documentation

Acknowledgements

Special thanks to "Uncle Bob" Robert C. Martin for his invaluable books on clean code and clean architecture.

License

This software is licensed under the MIT License.

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

lib_detect_testenv-3.0.1.tar.gz (84.0 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

lib_detect_testenv-3.0.1-py3-none-any.whl (14.8 kB view details)

Uploaded Python 3

File details

Details for the file lib_detect_testenv-3.0.1.tar.gz.

File metadata

  • Download URL: lib_detect_testenv-3.0.1.tar.gz
  • Upload date:
  • Size: 84.0 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for lib_detect_testenv-3.0.1.tar.gz
Algorithm Hash digest
SHA256 c23d5f1c8aed4ce030d977f6848ca52544b487212063d542e1ce3ec00f52ca6a
MD5 9e7a9686f365bcc595625cd5dfa0c0ee
BLAKE2b-256 4bc5d92d043d2e900ac5086314b993916cb1b4940c1de1ca98bb25abafbe00ab

See more details on using hashes here.

File details

Details for the file lib_detect_testenv-3.0.1-py3-none-any.whl.

File metadata

File hashes

Hashes for lib_detect_testenv-3.0.1-py3-none-any.whl
Algorithm Hash digest
SHA256 514bb01e730e0cb5fc5bdd9aeeed798a696348a89dfc399ff748a68f905ff9b2
MD5 00993a10dc94d97b2b672ed21cf11d6c
BLAKE2b-256 a269fcd3bd58782176dac80ad69a10976dccb5dada5b14a5279cb83380d63849

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page