Pytest-django: Регистрация типа базы данных после создания тестовых баз данных
У меня есть поле модели django, которое регистрирует тип базы данных:
class FieldMeta(type):
def __init__(cls, name, bases, clsdict):
super().__init__(name, bases, clsdict)
cls.register_type()
def register_type(cls):
db_type = cls().db_type(connection)
cursor = connection.cursor()
try:
cursor.execute(
"CREATE TYPE foo AS (first integer, second integer);"
)
_logger.warn("creating type")
except ProgrammingError:
pass
cls.python_type = register_composite(
db_type, connection.cursor().cursor, globally=True
).type
_logger.warn("registering composite")
def adapt_composite(composite):
return AsIs(...get sql...)
register_adapter(Foo, adapt_composite)
class FooField(models.Field, metaclass=FieldMeta):
def db_type(self, connection):
return "foo"
Это отлично работает в таких ситуациях:
- База данных уже знает о типе
foo
- База данных не знает о типе
foo
Проблема в том, что pytest-django, похоже, запускает эти методы init перед созданием тестовой базы данных, что означает, что тестовая база данных не имеет требуемого типа.
Я могу исправить это на отдельных тестах, вызывая FooField.register_type()
в методе setUpClass
теста, но это не идеально, так как он должен присутствовать всегда.
Я попробовал переопределить приспособление django_db_setup
, чтобы оно выглядело следующим образом:
@pytest.fixture(scope="session")
def django_db_setup(
request,
django_test_environment: None,
django_db_blocker,
django_db_use_migrations: bool,
django_db_keepdb: bool,
django_db_createdb: bool,
django_db_modify_db_settings: None,
) -> None:
"""Top level fixture to ensure test databases are available"""
from django.test.utils import setup_databases, teardown_databases
setup_databases_args = {}
if django_db_keepdb and not django_db_createdb:
setup_databases_args["keepdb"] = True
with django_db_blocker.unblock():
db_cfg = setup_databases(
verbosity=request.config.option.verbose,
interactive=False,
**setup_databases_args
)
FooField.register_type() # <----------------------------
def teardown_database() -> None:
with django_db_blocker.unblock():
try:
teardown_databases(db_cfg, verbosity=request.config.option.verbose)
except Exception as exc:
request.node.warn(
pytest.PytestWarning(
"Error when trying to teardown test databases: %r" % exc
)
)
if not django_db_keepdb:
request.addfinalizer(teardown_database)
Но это не дало никакого эффекта. Я также попробовал поставить точку останова вместо этой строки (она никогда не вызывалась) и поставить точку останова перед with django_db_blocker.unblock():
(которая была вызвана, так что conftest.py
определенно вызывается).
В общем, вопрос в том, как заставить pytest-django запустить этот метод послесоздания тестовой базы данных?