Skip to main content

Type Hinting

Here are some examples and basics on typing. More information on the why and how to run type checking can be found in our style guide This is just a summary and some common examples, but more examples on Type Hinting in Python can be found in the documentation for Python and mypy also has a very helpful Cheatsheet

Basics

You can type any variable at or before its definition by placing a : followed by the type after it, eg. potato: Potato,

from idaho import Potato

potato: Potato = Potato() # Just an example, the type hint here is not required.

Any built-in type or class can be used as a type annotation as well as some specialized classes classed Generics

You can type the return value with a -> followed by the type between the end of the method and the :, eg def return_potato() -> Potato: .... If the function does not return anything

def my_method(db: Database) -> None:
self.db: Database = db

If there are no return statements or values, methods will still transparently return None behind the scenes and thus should be typed with -> None. However, if a method always raise an exception you can use the special NoReturn type which indicates the code after will be inaccessible.

from typing import NoReturn

def fail() -> NoReturn:
raise Exception('This will never return anything')

def do_nothing() -> None:
pass # This method does nothing, but still "returns" None as it does not raise an exception.

class MyClass:
def __init__(self) -> None: # __init__ returns None
...

Typing attributes

class MyClass:
default_number: int = 1
will_be_set_later: str = None # type: ignore[assignment]

def __init__(self, timeout: int = 15) -> None:
self.timeout = timeout # No need to type this as it is inferred from above
self.names: list[str] = [] # when creating collections, it's important to type future contents
self.setup()

def setup(self) -> None:
self.will_be_set_later = 'hello'

Multiple types (Unions)

If multiple types are accepted they can be specified using the generic Union, which can also be more succinctly expressed using the pipe symbol | as of python 3.10. Note: this can still be used on older version of Python in annotation as long as you use lazy annotations, and it is not assigned to a variable.

from __future__ import annotations

from typing import Union

alphanumeric_list: list[int | str] = ['a', 1]
alphanumeric_list: list[Union[int, str]] = ['a', 1]

Type classes vs instances

Using a class as the type, it indicates that the variable is an instance of said class. If you instead wish to say that the variable is the class itself, for example for factories which receive the class and return an instance, you can use the type (or Type < 3.9) generic. If the class is defined in the same file, use lazy annotations or put the class name in quotes.

from __future__ import annotations

class MyClass:
@classmethod
def from_dict(cls: type[MyClass], data: dict[str, str]) -> MyClass: # or 'MyClass'
return cls(**data)

def my_factory(factory_class: type[MyClass]) -> MyClass:
return factory_class.from_dict({'test': 'hello'})

Typing Async methods

Although async methods and functions technically return coroutines, when using annotations, you specify the actual return value of the function as if it were not async.

from typing import Awaitable

async def get_user_ids() -> list[int]:
return [1, 2, 3, 4, 5]

user_ids: list[int] = await get_user_ids()
user_ids_coroutine: Awaitable[list[int]] = get_user_ids()

Type Stubs

Type stubs are stub files (.pyi) that contain the type definitions for libraries that are not themselves typed directly. They can be installed through pip for type checking. They are less and less required as more and more libraries are using types, but were used a lot when both Python 2 & 3 support was required. ex: - types-requests - types-pytz. The main repo that collects all of these is called typeshed, but stub files can also be included in projects, even if that is no longer needed as the code can be directly typed.

Lazy Annotations / ìf TYPE_CHECKING:

You can import annotations from __future__ to delay the evaluation of annotations. Thus, they are only evaluated by type checkers or if you specifically inspect that type in your code.

If you have annotations it’s always good to use this import since:

  • It prevents wasted processing of annotations at runtime
  • It allows you to use classes that are defined in the same file, or currently being defined, as type annotations
  • It allows you to use features and syntax of later versions of Python if your type checker is run with them. E.g. You can use the union operator from Python 3.10 in code that runs in 3.9 at runtime so long as your type checking is done with 3.10 or later and you use the lazy evaluation.

You can also use if TYPE_CHECKING: to encompass code that is only necessary for type checking or that you don’t want to run at runtime.

This allows you to, for example:

  • Create Self, NotRequired or other types/options for typing (if these aren’t supported in the current version)
  • Do some unfortunate workaround for typing in older language versions
  • Import things from typing that don’t exist in older versions of Python like StringLiteral
  • Import things that could create problems if imported at runtime (e.g. avoid circular import errors).

In general, use TYPE_CHECKING only if necessary. Typing-only imports should be treated as any other import.

Example:

# Copyright ...

from __future__ import annotations

import threading
from collections.abc import Callable, Collection
from typing import TYPE_CHECKING, TypedDict

from wazo_auth_client.client import AuthClient # only needed for type checking

if TYPE_CHECKING:
from typing import NotRequired # Only exists in 3.11+

Callback = Callable[[Collection[str]], None]
class CallbackDict(TypedDict):
method: Callback
details: bool
extra: NotRequired[str]


class TokenRenewer:
def __int__(self, auth_client: AuthClient) -> None:
self._auth_client = auth_client
self._callback_lock = threading.Lock()
self._callbacks: list[CallbackDict] = []

def subscribe_to_token_change(self, callback: Callback) -> None:
with self._callback_lock:
self._callbacks.append({'method': callback, 'details': False})

Typing from the future

If you would like to use typing feature have been added to the language, but do not exist in the version you are using, you have two options:

  1. You can use ìf TYPE_CHECKING and Lazy Annotations and run your linting with a more modern version of Python to run as mentioned above.

  2. You can use Typing Extensions which is a library that backports certain types to older versions of Python. So you can simply import them from their if they are available. However, if you use any imports from that library outside an if TYPE_CHECKING block, you must add it as a runtime dependency or else you will get an ImportError at runtime. Wazo includes a backport of version 4.4.0

    # from typing import Self # Only exists in Python 3.11+
    from typing_extensions import Self # use the backport

    class MyClass:
    @classmethod
    def setup(cls) -> Self:
    return cls()

Parametric typing & Generics

Generics are types that can be passed parameters to ensure a useful definition. For example, it’s good to know your method receives a list, but it’s not super useful if you don’t know what said list contains. So you can pass options to the list type to specify its contents (e.g. list[int | str]). You can create custom ones, but a lot of builtin ones exist e.g.:

Generic built-in collections

All the basic types that contain other types can be passed parameters to specify their contents (ie. list, tuple, dict, set).

import string

alphabet: list[str] = list(string.ascii_lowercase)
two_numbers: tuple[int, int] = (1, 2)
variable_numbers: tuple[int, ...] = (1, 2, 3, 4)
no_doubles: set[str] = {'a', 'b', 'a'}
scores: dict[str, float] = {'bob': 99, 'fred': 100}

Note: If you type a tuple it will expect exactly those types in that order. If you want to allow a variable number of items in a tuple, type the first entry and follow it with an ..., eg. tuple[int, ...]

Callables

When passing around methods as callbacks you can use the built-in generic [Callable](https://docs.python.org/3/library/typing.html#typing.Callable). The first option is the arguments and the second is the return value. For more complex cases see Protocols

from collections.abc import Callable

def my_handler(name: str, timeout: int = 5) -> bool:
with contextlib.suppress(KeyError):
# do something
return True
return False

def process_handler(handler: Callable[[str, int], bool]) -> None:
handler('test')

process_hander(my_handler)

Literal values

To allow only a specific set of values you can use [Literal](https://docs.python.org/3/library/typing.html#typing.Literal). It is useful for when you expect specific strings like 'on' or 'off'. The standard library uses this for read/write modes for example.

from typing import Literal

the_answer: Literal[42] = 42 # No other int is accepted, but 42
mode: Literal['on', 'off'] = 'off' # Only accept these two strings

def open(filename: str | Path, mode: Literal['rb', 'r']) -> BytesIO | StringIO:
...

TypeVar

[TypeVar](https://docs.python.org/3/library/typing.html#typing.TypeVar) allows you to reference a type without knowing it. For example, a generic method that receives a list of items and returns the first. It doesn't matter what the list contains. Just that you will return the first item if it is not empty.

from typing import TypeVar

T = TypeVar('T')

def get_first(items: list[T]) -> T | None:
return items[0] if items else None

This allows for mypy and other type checkers to replace T based on what is passed to the function. For example if you pass a list of int it knows that it will return a single int or None

Protocols and structural typing

Protocols are used to implement essentially “static duck typing” 🦆. Where the actual type is not important so much as what it implements. Do you need a list or would anything that is iterable work?

Builtin types:

  • Iterable an object that implements __iter__ or __getitem__

  • Sequence an iterable that defines __len__ , __getitem__ and a few others. Eg.

    from collections.abc import Sequence

    def check_len(items: Sequence[int], limit: int) -> bool:
    return len(items) <= limit

Protocols

Protocols allow you type an object that has a specific signature. e.g. implements a given method or has a given property. You can also use them to handle more complex cases of callables.

from __future__ import annotations

from typing import Protocol

class ComplexCallable(Protocol):
def __call__(self, x: int = ..., /, timeout: int = 12) -> float:
...

Typing Dynamic patterns

Mixins

Mixins can be challenging to type if they have implicit dependency on other interfaces (i.e. expect the inheriting class to implement other methods used by the mixin implementation). Intersection types are not supported yet, but would allow specifying a self-type that combines the mixin’s interfaces.

Intersection types are equivalent to ad-hoc subclasses of the types in the intersection. To type a mixin properly, the implicit dependencies must be made explicit, for example through a separate protocol or ABC defining the required interface, which the mixin can subclass.

Self with inheritance

Self was introduced in 3.11, but is equivalent to Self = TypeVar("Self", bound="MyClass"). This can be important for class methods that return instances of themselves because simply using the class name with result in issues if the class is extended. In simple cases you can return the class itself, but Self is a safer option.

from typing import Self  # Self was added in 3.11

# For the same result on Python < 3.11 you can create a TypeVar bound to the class.
# Self = TypeVar("Self", bound="ParentClass")
# Or you can import from typing_extensions if you have it installed.

class ParentClass:
def return_self(self: Self) -> Self:
return self

class ChildClass(ParentClass):
def return_self(self: Self) -> Self:
# If we had used `-> ParentClass` this would then return the wrong type as it is now `ChildClass`
return super().return_self()

Type Aliases

You can easily create aliases for types by simply assigning them to variables. They can also be explicitly typed with TypeAlias, but it changes nothing functionally. This is no different than manually specifying the type each time, but can be more readable especially when types get complicated and it can avoid copy/paste errors and ensure the type is always the same.

from collections.abc import Callable
from typing import TypeAlias

HandleFunction = Callable[[FastAGI, DictCursor, list], None]
SetupFunction: TypeAlias = Callable[[DictCursor], None]

Extra specificity with New Types

[NewType](https://mypy.readthedocs.io/en/stable/more_types.html#newtypes) can be used to add increased specificity to a type allowing for a distinction between it and other values of the same base type. For example, if you have a functions that frequently use User IDs and you want to indicate that this is different from standard integers you can create a new UserID type and the type checker will indicate an error if it instead receives simply an int.

from typing import NewType

UserId = NewType('UserId', int)

def get_user(user_id: UserId) -> User:
return User(user_id)


get_user(42) # invalid
get_user(UserId(42)) # valid

This also enables quickly creating a thin abstraction over the underlying type, which helps reduce future refactoring. The actual implementation type can later be changed(for example to a custom class implementation) without breaking typing consistency everywhere and having to rewrite all interfaces.

Integrating into workflow

Config

You can configure mypy with the rest of the tool in the pyproject.toml file. Here is an example config entry for a fully typed or new project.

[tool.mypy]
python_version = "3.10"
show_error_codes = true
check_untyped_defs = true
warn_unused_configs = true
disallow_untyped_calls = true
disallow_untyped_defs = true
disallow_incomplete_defs = true
warn_unused_ignores = true
strict_equality = true
strict_concatenate = true
no_warn_no_return = true

[[tool.mypy.overrides]]
module = [
"*.tests.*",
"integration_tests.suite.*",
]
disallow_untyped_defs = false
disallow_untyped_calls = false

If you want to slowly add typing to an existing project you can start with a more relaxed config that will allow for some untyped or partially untyped functions.

[tool.mypy]
python_version = "3.10"
show_error_codes = true
check_untyped_defs = true
warn_unused_configs = true
ignore_missing_imports = true

If you have any settings that are unique to just one particular repo, you should add a comment separating them from the rest and indicate why the option was added in case it is no longer required in future.

Pre-commit

To Install and run

pip install pre-commit
# To run automatically as hook
pre-commit install
# To run manually
pre-commit run --all-files

Example config:

# See https://pre-commit.com for more information
repos:
- repo: https://github.com/PyCQA/flake8
rev: '6.0.0'
hooks:
- id: flake8
# Required to make flake8 read from pyproject.toml for now :(
additional_dependencies: ['flake8-pyproject']
- repo: https://github.com/psf/black
rev: 22.12.0
hooks:
- id: black
- repo: https://github.com/pre-commit/mirrors-mypy
rev: v0.991
hooks:
- id: mypy
language_version: '3.10'
additional_dependencies:
# Only include the stubs required for your project
- 'types-flask'
- 'types-psycopg2'
- 'types-pytz'
- 'types-pyyaml'
- 'types-requests'
- 'types-setuptools'
- 'types-werkzeug'

Tox

We can also use tox to run mypy with our other linters via pre-commit. This allows for a unified workflow, and it is run in our CI.

[testenv:linters]
basepython = python3.10
skip_install = true
deps = pre-commit
commands = pre-commit run --all-files

Editor integrations

PyCharm

PyCharm supports Type Hinting and code completion out of the box.

VSCode

VSCode supports type checking via the Pylance extension and can be enabled by adding this option to your settings.json file.

{"python.analysis.typeCheckingMode": "basic"}  # or "strict"

Also, the mypy plugin directly supports mypy as the type checker.

Jedi

VSCode, Vim, Emacs, Kate and many others can also make use of them via Jedi.

Special cases

Unknown type

If it is not possible to know what type a function can return, or if it can accept any value, you can use TypeVar to pass through the value or use Any which matches any type. Try to avoid Any as much as possible though and instead use TypeVar or Generics.

Ignoring types

In rare cases it might be required to tell type checkers to ignore a line of code.

test: int  # type: ignore
# https://mypy.readthedocs.io/en/stable/error_codes.html#error-codes
test: int = None # type: ignore[error-code]

For example, sometimes variables are set to None, but are initialized before anything is run, and thus are never really None. In these cases, you can ignore the type for assignment only.

from configparser import RawConfigParser

MY_CONFIG: RawConfigParser = None # type: ignore[assignment]

def is_debug() -> bool:
return MY_CONFIG['DEBUG'] is True # mypy knows this is not None


def init() -> None:
global MY_CONFIG
with open('config.ini') as f:
config = RawConfigParser()
config.read_file(f)
MY_CONFIG = config