This is the 4th time in a row that I’ve wasted time figuring out how to mock out a function
during testing that calls the chained methods of a datetime.datetime object in the
function body. So I thought I’d document it here. Consider this function:
# src.py
from __future__ import annotations
import datetime
def get_utcnow_isoformat() -> str:
"""Get UTCnow as an isoformat compliant string."""
return datetime.datetime.utcnow().isoformat()
How’d you test it? Mocking out datetime.datetime is tricky because of its immutable
nature. Third-party libraries like freezegun1 make it easier to mock and test functions
like the one above. However, it’s not too difficult to cover this simple case without any
additional dependencies. Here’s one way to achieve the goal:
# src.py
from __future__ import annotations
import datetime
from unittest.mock import patch
import pytest
def get_utcnow_isoformat() -> str:
"""Get UTCnow as an isoformat compliant string."""
return datetime.datetime.utcnow().isoformat()
@pytest.fixture
def mock_datetime():
with patch("datetime.datetime") as m:
# This is where the magic happens!
m.utcnow.return_value.isoformat.return_value = (
"2022-03-15T23:11:12.432048"
)
yield m
def test_get_utcnow_isoformat(mock_datetime):
frozen_date = "2022-03-15T23:11:12.432048"
assert get_utcnow_isoformat() == frozen_date
Here, the mock_datetime fixture function makes the output of the chained calls on the
datetime object deterministic. Then I used it in the test_get_utcnow_isoformat function to
get a frozen output every time the function get_utcnow_isoformat gets called. If you run
the above snippet with Python, it’ll pass.
======test session starts ======
platform linux -- Python 3.10.2, pytest-7.0.1, pluggy-1.0.0
rootdir: /home/rednafi/canvas/personal/reflections
plugins: anyio-3.5.0
collected 1 item
src.py . [100%]
====== 1 passed in 0.01s ======
Recent posts
- Revisiting interface segregation in Go
- Avoiding collisions in Go context keys
- Organizing Go tests
- Subtest grouping in Go
- Let the domain guide your application structure
- Test state, not interactions
- Early return and goroutine leak
- Lifecycle management in Go tests
- Gateway pattern for external service calls
- Flags for discoverable test config in Go