| name | python |
| description | Apply when writing Python code. Type hints, error handling, mutable defaults, async patterns, and packaging conventions. |
| license | MIT |
| version | 1.1.0 |
| tokens_target | 2900 |
| triggers | python code, type hints, python packaging |
| loads_after | code-quality |
| supersedes |
Sub-Skill: Python Best Practices
Purpose: Prevents the Python-specific mistakes LLMs make on autopilot — mutable defaults, bare excepts, missing guards, and subtle performance traps that pass review but break in production.
Rule classification
- MUST — load-bearing. Violating causes bugs, security issues, or invisible failures. Never break.
- SHOULD — default behavior. Deviation needs a documented reason in the code or PR.
- AVOID — usually wrong; documented exception inline where needed.
Where these rules don't strictly apply: test fixtures, generated code, throwaway scripts, REPL exploration, and tutorial snippets may legitimately differ. The rules below apply to production code paths and reusable libraries.
Rules
Type Hints
SHOULD: Annotate function signatures with types.
def process(data)tells callers nothing. Usedef process(data: list[str]) -> dict[str, int]:. Return typeNoneshould be explicit too. Exception: lambdas and short generator helpers in private scope.SHOULD: Use built-in lowercase generics for Python 3.9+.
list[str],dict[str, int],tuple[int, ...]rather thanList,Dict,Tuplefromtyping.MUST: Use
Optional[X]orX | Nonefor nullable parameters, never a bare default ofNonewithout a type annotation.def find(id: int) -> User | None:notdef find(id):.AVOID:
Anyas a shortcut. If the type is genuinely unknown, document why with a comment.Anysilences the type checker and hides bugs. Exception: third-party libraries without stubs and explicit dynamic-data boundaries (e.g. JSON decode at the API edge).
Error Handling
MUST: Never use bare
except:. It catchesSystemExit,KeyboardInterrupt, andGeneratorExit. Always catchexcept Exception:at minimum, or a specific exception class.# Wrong try: risky() except: pass # Correct try: risky() except ValueError as e: logger.warning("Invalid value: %s", e)MUST: Never silently swallow exceptions with
pass. At minimum log the error. Silent failures produce ghost bugs that are impossible to trace.SHOULD: Raise with context when re-raising. Use
raise NewError("msg") from original_errorto preserve the traceback chain, notraise NewError("msg")alone.
Common Pitfalls
MUST: Never use mutable default arguments. Python evaluates defaults once at function definition, not per call. The list or dict is shared across all calls.
# Wrong — items accumulates across calls def append_item(val, items=[]): items.append(val) return items # Correct def append_item(val, items=None): if items is None: items = [] items.append(val) return itemsMUST: Guard script entry points with
if __name__ == "__main__":. Without it, importing the module executes top-level code, breaking tests and imports.MUST: Use
withfor file handles, sockets, and locks. Never open a file without a context manager.f = open(...)withoutwithleaks handles on exceptions.# Wrong f = open("data.txt") data = f.read() f.close() # Correct with open("data.txt") as f: data = f.read()AVOID: String concatenation in loops. Each
+=on a string creates a new object. Collect into a list and call"".join(parts)at the end.# Wrong — O(n^2) memory result = "" for word in words: result += word + " " # Correct result = " ".join(words)SHOULD: Use f-strings for string interpolation in Python 3.6+. Avoid
%formatting or"Hello " + name. F-strings are faster, safer, and readable.# Avoid msg = "User %s has %d items" % (name, count) # Prefer msg = f"User {name} has {count} items"AVOID:
dict()constructor when a literal suffices.{}is faster and more idiomatic.dict(key=value)is only justified when keys are dynamic or come from variables.
Performance
SHOULD: Use list comprehensions or generator expressions instead of
map/filterwithlambda. Comprehensions are more readable and equally fast. Use generators when the full list is not needed at once.# Avoid result = list(map(lambda x: x * 2, items)) # Prefer result = [x * 2 for x in items] # For large data, use a generator total = sum(x * 2 for x in items)SHOULD: Use a
setfor membership lookups, not alist.x in listis O(n).x in setis O(1). Convert once, query many times.# Wrong for repeated lookups valid_ids = [1, 2, 3, ...] if user_id in valid_ids: # O(n) every call # Correct valid_ids = {1, 2, 3, ...} if user_id in valid_ids: # O(1)
Testing
SHOULD: Name test functions to describe the scenario, not just the function under test.
test_process_returns_empty_dict_on_empty_inputnottest_process.MUST: Never use
assertstatements in production code for validation.assertis stripped withpython -O. Use explicitifchecks withraise ValueError(...)for runtime validation.MUST: Use
pytest.raisesas a context manager to assert exceptions, never wrap in try/except inside a test. A bare try/except can mask a missing exception.# Wrong def test_bad_input(): try: process(None) except ValueError: pass # test passes even if no exception is raised # Correct def test_bad_input(): with pytest.raises(ValueError, match="input cannot be None"): process(None)SHOULD: Use
@pytest.mark.parametrizeto test the same logic with multiple inputs. Write one parameterized test instead of duplicating test functions for each case.SHOULD: Place shared fixtures in
conftest.pyat the nearest common ancestor directory. Do not scatter fixtures across individual test files. Pytest auto-discoversconftest.pyat every level.MUST: Never return mutable objects directly from a pytest fixture. Each test must receive a fresh instance. Use factory fixtures that return a callable creating fresh objects. Reference: ERR-2026-014.
AVOID: Fixtures with side effects (network, DB) without explicit scope. Use
scope="session"orscope="module"for expensive shared resources with proper teardown viayield.SHOULD: Use the
tmp_pathfixture for filesystem tests instead of manual temporary directories. It auto-cleans and is process-safe.
Packaging
SHOULD: Include
__init__.pyin every package directory. Without it, Python 3 treats the directory as a namespace package, which breaks relative imports and tool discovery in many environments. Exception: explicit namespace packages (PEP 420) where the omission is documented intent.AVOID: Imports from
__init__.pyinside the same package's submodules. This creates circular imports. Keep__init__.pyas a re-export surface only, not a logic file.SHOULD: Use
pyproject.tomlas the single source of project metadata. Avoidsetup.pyandsetup.cfgfor new projects. Modern tooling (pip, build, hatch, uv) all readpyproject.tomlnatively.MUST: Pin exact versions in lock files but use ranges in
pyproject.tomldependencies. Lock files (uv.lock,poetry.lock) ensure reproducibility; flexible ranges in project metadata allow resolver to find compatible versions.SHOULD: Define CLI entry points in
[project.scripts]instead of relying onpython -mpatterns. Entry points generate proper executables and integrate with system PATH.
Why This Sub-Skill Earns Stars
These rules target the exact failure modes that appear in LLM-generated Python:
- Mutable defaults and missing
__main__guards are invisible bugs that only surface at runtime. - Bare
except:and silentpassblocks make debugging take 10x longer. - String concatenation in loops and list membership checks are performance traps that scale badly.
- Missing type annotations and
Anyshortcuts defeat the entire value of static analysis. - Skipping
withfor file handles causes resource leaks under load.
None of these are caught by syntax checkers. All of them appear in production incidents. The MUST/SHOULD/AVOID classification means the security/correctness rules are strict and the stylistic rules respect context.