Assert that str matches regex in pytest

Once in a while every Pythonista faces a need to test that some string value matches a regex pattern. When it comes to it we have no option but to use re module directly.

import re

def test_something_very_useful():
    value = get_some_string()
    assert re.match("\d+", value)

While it works perfectly fine in case of a lone string, it's not always convenient to do the same when a string is a part of more complex data structure (e.g. dict) because you need to extract the string, check it and only then get back to check rest attributes.

import re

def test_something_even_more_useful():
    mapping = get_some_structure()

    # inconvenient and boring! :(
    assert "value" in mapping
    value = mapping.pop("value")
    assert re.match("\d+", value)

    # rest assertion
    assert mapping == {
        "foo": 1,
        "bar": 2,
    }

Fortunately, it's pretty easy to write a convenient helper that would check a string against some pattern when comparing complex data structures.

def test_something_even_more_useful():
    mapping = get_some_structure()
    assert mapping == {
        "foo": 1,
        "bar": 2,
        "value": pytest_regex("\d+"),
    }

Looks better, huh? :) The trick can be achieved by implementing a class wrapper that implements both __eq__ and __repr__ dunder methods. The first method is used to implement matching while the other – to return the value that will be shown in error message in case of errors.

import re

class pytest_regex:
    """Assert that a given string meets some expectations."""

    def __init__(self, pattern, flags=0):
        self._regex = re.compile(pattern, flags)

    def __eq__(self, actual):
        return bool(self._regex.match(actual))

    def __repr__(self):
        return self._regex.pattern

Well, actually, it's not by any means tied to pytest and could be used even in non-test production code, though I found this rather queer.