A mypy plugin that adds support for checking pure functions.
Project description
mypy-pure
Enforce functional purity in Python with static type checking.
A mypy plugin that helps you write safer, more predictable code by detecting side effects in functions marked as @pure.
Why mypy-pure?
Pure functions are:
- Easier to test - No mocks needed, same inputs always give same outputs
- Easier to reason about - No hidden state changes or side effects
- Easier to refactor - Can be moved, renamed, or reordered safely
- Easier to parallelize - No race conditions or shared state issues
- Easier to cache - Results can be memoized safely
But enforcing purity manually is error-prone. mypy-pure catches impure code at type-check time, before it reaches production.
What is a Pure Function?
A pure function:
- Always returns the same output for the same inputs (deterministic)
- Has no side effects (no I/O, no mutations, no external state changes)
# ✅ Pure - deterministic, no side effects
def add(x: int, y: int) -> int:
return x + y
# ❌ Impure - side effect (I/O)
def add_and_log(x: int, y: int) -> int:
print(f"Adding {x} + {y}") # Side effect!
return x + y
Installation
pip install mypy-pure
Enable the plugin in your mypy.ini or pyproject.toml:
[mypy]
plugins = mypy_pure.plugin
Quick Start
Mark functions as pure with the @pure decorator:
from mypy_pure import pure
@pure
def calculate_total(prices: list[float], tax_rate: float) -> float:
subtotal = sum(prices)
return subtotal * (1 + tax_rate)
Run mypy to check for purity violations:
mypy your_code.py
Examples
✅ Valid Pure Functions
from mypy_pure import pure
@pure
def fibonacci(n: int) -> int:
"""Pure recursive function."""
if n <= 1:
return n
return fibonacci(n - 1) + fibonacci(n - 2)
@pure
def process_data(items: list[dict]) -> list[str]:
"""Pure data transformation."""
return [item['name'].upper() for item in items if item['active']]
@pure
def calculate_discount(price: float, discount_pct: float) -> float:
"""Pure business logic."""
return price * (1 - discount_pct / 100)
❌ Detected Impurity Violations
from mypy_pure import pure
import os
@pure
def read_config() -> dict:
# Error: Function 'read_config' is annotated as pure but calls impure functions.
with open('config.json') as f: # I/O is impure!
return json.load(f)
@pure
def delete_temp_files(directory: str) -> None:
# Error: Function 'delete_temp_files' is annotated as pure but calls impure functions.
for file in os.listdir(directory):
os.remove(file) # File system modification is impure!
@pure
def log_and_calculate(x: int, y: int) -> int:
# Error: Function 'log_and_calculate' is annotated as pure but calls impure functions.
print(f"Calculating {x} + {y}") # Logging is impure!
return x + y
More examples
More examples can be found in the mypy-pure-examples repository.
Advanced Usage
Configuration Options
mypy-pure supports two configuration options in the [mypy-pure] section of your mypy.ini:
1. impure_functions (Blacklist)
Add custom impure functions that should be flagged as side-effecting:
[mypy-pure]
impure_functions = my_module.send_email, analytics.track_event, cache.set
Use cases:
- Your own functions that have side effects
- Third-party library functions not in the built-in blacklist
- Project-specific impure operations
Example:
# my_module.py
def send_email(to: str, subject: str) -> None:
# Sends an email (side effect)
...
# main.py
from mypy_pure import pure
from my_module import send_email
@pure
def process_user(user: dict) -> dict:
send_email(user['email'], 'Welcome') # ❌ Error: calls impure function
return user
2. pure_functions (Whitelist)
Mark third-party library functions as pure, overriding the default assumption:
[mypy-pure]
pure_functions = requests.utils.quote, pandas.DataFrame.copy, my_lib.helper
Use cases:
- Pure utility functions from third-party libraries
- Functions you've verified have no side effects
- Overriding false positives
Example:
from mypy_pure import pure
import requests.utils
@pure
def sanitize_url(url: str) -> str:
# OK because requests.utils.quote is in pure_functions config
return requests.utils.quote(url)
Combining Both
You can use both options together:
[mypy-pure]
# Blacklist your impure functions
impure_functions = my_module.send_email, my_module.log_event
# Whitelist pure third-party functions
pure_functions = requests.utils.quote, requests.utils.unquote
Priority: pure_functions (whitelist) takes precedence over impure_functions (blacklist).
Library Authors: Auto-Discovery with __mypy_pure__
If you're a library author, you can declare your pure functions using the __mypy_pure__ module-level list. This enables zero-configuration purity checking for your users.
Declaring Pure Functions
Add a __mypy_pure__ list to your module:
# my_library.py
__mypy_pure__ = [
'pure_helper',
'utils.calculate',
'ClassName.method_name',
]
def pure_helper(x: int) -> int:
"""A pure utility function."""
return x * 2
class utils:
@staticmethod
def calculate(a: int, b: int) -> int:
"""A pure calculation."""
return a + b
def impure_logger(msg: str) -> None:
"""Not in __mypy_pure__, so treated as impure."""
print(msg)
User Experience
Users of your library automatically benefit without any configuration:
from mypy_pure import pure
import my_library
@pure
def process(x: int) -> int:
# ✅ OK - pure_helper is in __mypy_pure__
return my_library.pure_helper(x)
@pure
def log_process(x: int) -> int:
# ❌ Error: Function 'log_process' is impure because it calls 'my_library.impure_logger'
my_library.impure_logger(f"Processing {x}")
return x
Benefits
- Zero configuration for library users
- Self-documenting API - pure functions are explicitly declared
- Compile-time guarantees - purity violations caught during type checking
- Better IDE support - users see which functions are safe to use in pure contexts
Cross-Module References
Reference functions from other modules using fully qualified names:
[mypy-pure]
impure_functions = external_lib.impure_function
Supported Function Types
mypy-pure works with all Python function and method types:
- ✅ Regular functions
- ✅ Instance methods
- ✅ Class methods (
@classmethod) - ✅ Static methods (
@staticmethod) - ✅ Async functions (
async def) - ✅ Async methods
- ✅ Property methods (
@property) - ✅ Nested/inner functions
Built-in Impurity Detection
mypy-pure includes a comprehensive blacklist of 200+ impure functions from Python's standard library:
- File I/O:
open(),pathlib.Path.write_text(), etc. - System operations:
os.remove(),subprocess.run(), etc. - Network:
socket.socket(),urllib.request.urlopen(), etc. - Logging:
logging.info(),print(), etc. - State modification:
random.seed(),sys.exit(), etc. - Databases:
sqlite3.connect(), etc. - And many more...
Limitations
mypy-pure performs static analysis and has some limitations:
What it CAN detect:
- ✅ Direct calls to known impure functions
- ✅ Indirect calls through pure functions calling impure functions
- ✅ Deeply nested impure calls
What it CANNOT detect:
- ❌ Mutations of mutable arguments (e.g.,
list.append()) - ❌ Global variable modifications
- ❌ Impure functions not in the blacklist
- ❌ Dynamic function calls (e.g.,
getattr(),eval()) - ❌ Side effects in third-party libraries (unless configured)
Recommendation: Use mypy-pure as a helpful guard rail, not a guarantee of purity. Combine it with code reviews and testing.
Real-World Use Cases
Data Processing Pipelines
@pure
def transform_user_data(raw_data: dict) -> dict:
"""Pure transformation - easy to test and parallelize."""
return {
'id': raw_data['user_id'],
'name': raw_data['full_name'].title(),
'age': calculate_age(raw_data['birth_date']),
}
Business Logic
@pure
def calculate_shipping_cost(
weight_kg: float,
distance_km: float,
is_express: bool
) -> float:
"""Pure business logic - deterministic and testable."""
base_cost = weight_kg * 0.5 + distance_km * 0.1
return base_cost * 1.5 if is_express else base_cost
Configuration Processing
@pure
def merge_configs(default: dict, user: dict) -> dict:
"""Pure config merging - no file I/O."""
return {**default, **user}
Contributing
Contributions are welcome! Here's how you can help:
- Expand the blacklist - Add more impure functions from stdlib or popular libraries
- Report bugs - Open an issue if you find incorrect behavior
- Suggest features - Ideas for improving purity detection
- Improve documentation - Help make the docs clearer
See CONTRIBUTING.md for details.
Development
# Clone the repository
git clone https://github.com/diegojromerolopez/mypy-pure.git
cd mypy-pure
# Install dependencies
pip install -e ".[dev]"
# Run tests
pytest
# Run mypy
mypy mypy_pure
License
MIT License - see LICENSE for details.
Acknowledgments
This project was created with the assistance of AI tools (ChatGPT and Antigravity/Gemini) and manually reviewed and refined.
Related Projects
- mypy - Optional static typing for Python
- returns - Make your functions return something meaningful
- deal - Design by contract for Python
Made with ❤️ for the Python community
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
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file mypy_pure-0.2.1.tar.gz.
File metadata
- Download URL: mypy_pure-0.2.1.tar.gz
- Upload date:
- Size: 21.9 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
66c34af6791a0ddab9d458badbeb95b0189558bcbb5fd7c9ae3b7a48c057dd32
|
|
| MD5 |
a19312120c56db1575cf8162c631d13f
|
|
| BLAKE2b-256 |
4f11666f2e99127eaea7967d1350f41e59ef4fb83f6be91f885d5363c47007eb
|
Provenance
The following attestation bundles were made for mypy_pure-0.2.1.tar.gz:
Publisher:
publish_on_pypi.yml on diegojromerolopez/mypy-pure
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
mypy_pure-0.2.1.tar.gz -
Subject digest:
66c34af6791a0ddab9d458badbeb95b0189558bcbb5fd7c9ae3b7a48c057dd32 - Sigstore transparency entry: 737350940
- Sigstore integration time:
-
Permalink:
diegojromerolopez/mypy-pure@88f70e7957e5344c7f207b1b513494618437a791 -
Branch / Tag:
refs/tags/0.2.1 - Owner: https://github.com/diegojromerolopez
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish_on_pypi.yml@88f70e7957e5344c7f207b1b513494618437a791 -
Trigger Event:
release
-
Statement type:
File details
Details for the file mypy_pure-0.2.1-py3-none-any.whl.
File metadata
- Download URL: mypy_pure-0.2.1-py3-none-any.whl
- Upload date:
- Size: 27.4 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
0f06d5cc0932c3fae02bb7fb3082556095f3d9c31734e258a4d0d88850031707
|
|
| MD5 |
6122264f12903fdf598d60edfaef944c
|
|
| BLAKE2b-256 |
099d66e3d6498e66fd206cf10c3d8dd4b8faa3b9e4cc6183b6d37df49926f69f
|
Provenance
The following attestation bundles were made for mypy_pure-0.2.1-py3-none-any.whl:
Publisher:
publish_on_pypi.yml on diegojromerolopez/mypy-pure
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
mypy_pure-0.2.1-py3-none-any.whl -
Subject digest:
0f06d5cc0932c3fae02bb7fb3082556095f3d9c31734e258a4d0d88850031707 - Sigstore transparency entry: 737350944
- Sigstore integration time:
-
Permalink:
diegojromerolopez/mypy-pure@88f70e7957e5344c7f207b1b513494618437a791 -
Branch / Tag:
refs/tags/0.2.1 - Owner: https://github.com/diegojromerolopez
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish_on_pypi.yml@88f70e7957e5344c7f207b1b513494618437a791 -
Trigger Event:
release
-
Statement type: