question-mark
Stuck on an issue?

Lightrun Answers was designed to reduce the constant googling that comes with debugging 3rd party libraries. It collects links to all the places you might be looking at while hunting down a tough bug.

And, if you’re still stuck at the end, we’re happy to hop on a call to see how we can help out.

Tortoise.get_connection in pytest test raises a KeyError

See original GitHub issue

Describe the bug My project uses some raw SQL. For this, I use Tortoise.get_connection() to get a DB connection. This fails when I run unit tests with pytest with a KeyError.

To Reproduce test_connection.py:

from tortoise import Tortoise
async def test_get_connection():
    c = Tortoise.get_connection('default')

$ pytest test_connection.py

Expected behavior The test should succeed without an exception. I also tried ‘models’ and ‘test’ for the connection name.

Additional context This seems to be caused by tortoise.contrib.test.initializer() which contains this code:

_CONNECTIONS = Tortoise._connections.copy()
_CONN_MAP = current_transaction_map.copy()
Tortoise.apps = {}
Tortoise._connections = {} 

Issue Analytics

  • State:open
  • Created 3 years ago
  • Comments:5 (2 by maintainers)

github_iconTop GitHub Comments

1reaction
s3riuscommented, Oct 22, 2021

I’ve found a workaround. This is my create_db fixture.

import pytest
from tortoise.contrib.test import finalizer, initializer
from tortoise.transactions import current_transaction_map


@pytest.fixture(autouse=True)
@pytest.mark.asyncio
def create_db() -> Generator[None, None, None]:
    """
    Fixture that initializes database.

    :yield: Nothing.
    """
    db_url = os.environ.get("PROJECT_TEST_DB", "sqlite://:memory:")
    initializer(MODELS_MODULES, db_url=db_url, app_label="models")
    current_transaction_map["default"] = current_transaction_map["models"]

    yield

    finalizer()

And the function looks like this:

from tortoise.transactions import atomic

@atomic("default")
async def grow(message: Message) -> None:
  ... # code

1reaction
DrJackilDcommented, Oct 4, 2020

@janpascal I have the same problems with pytest (and to be honest I don’t want to switch to other test framework or unittest class-based style), but I found a solution to use a connection name in transaction.atomic decorator. The general difference between my and your setup is that I use module names during Tortoise initialize (in conftest and the application)

Like this:

initializer(["app.db.models"], db_url=str(settings.DB_URL), app_label="models", loop=event_loop)

In that case Tortoise will create current_transaction_map[name] link in tortoise.Tortoise._init_connections and you can get it with @transaction.atomic("models").

Hope this will help.

UPD: No, it’s not, now I’ve got a problem in the application 😅

UPD2 I made this ugly hack to get what I want. I’m using TESTING in my settings to understand is this a test environment, so I’ve added this to the decorator:

@transaction.atomic("models" if settings.TESTING else None)

Looks awful, but it works.

I believe the reason that it works is here:

# tortoise.Tortoise._init_connections
...
await connection.create_connection(with_db=True)
cls._connections[name] = connection
current_transaction_map[name] = ContextVar(name, default=connection)

The we clear Tortoise._connections in tortoise.contrib.test.initializer, but current_transaction_map left untouched.

Read more comments on GitHub >

github_iconTop Results From Across the Web

Tortoise-orm raise KeyError in High Concurrency
I use Tortoise-orm in FastAPI, Mysql connection pool settings is {"minsize":8, "maxsize":16} 。 # models.py class Device(models.
Read more >
How to write and report assertions in tests - Pytest
pytest allows you to use the standard Python assert for verifying expectations and values in Python tests. For example, you can write the...
Read more >
Changelog — Tortoise ORM v0.17.3 Documentation
.clone() will raise a ParamsError if tortoise can't generate a primary key. ... Check whether charset name is valid for the MySQL connection...
Read more >
KeyError: 'TEST' during running pytest - M220P - MongoDB
Hi, Really need some help for running pytest here. ... raise KeyError(key) ... (mflix) linxz-MAC:tests linxz$ pytest -m connection.
Read more >
How to Check if an Exception Is Raised (or Not) With pytest
Solution: Enclose your code in a try/except block and and if the code raises, you can catch it and print a nice message....
Read more >

github_iconTop Related Medium Post

No results found

github_iconTop Related StackOverflow Question

No results found

github_iconTroubleshoot Live Code

Lightrun enables developers to add logs, metrics and snapshots to live code - no restarts or redeploys required.
Start Free

github_iconTop Related Reddit Thread

No results found

github_iconTop Related Hackernoon Post

No results found

github_iconTop Related Tweet

No results found

github_iconTop Related Dev.to Post

No results found

github_iconTop Related Hashnode Post

No results found