-
-
Notifications
You must be signed in to change notification settings - Fork 899
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
add docs about testing #1171
Comments
I don't think this code snippet is sufficient restore the functionality most people used from sqlalchemy 1. If any of your code or tests rely on rollback inside the test, then this will just rollback you entire session. Basically any call to It's also painful because you can't override the behaviour via events either. For example trying to use retval=True and stop a commit or modify the behaviour doesn't work. |
It sounds like you're talking about a difference in behavior between SQLAlchemy 1.4 and 2.0. That's not what this is addressing, this is addressing how to isolate each test so that it doesn't modify the database (it's also not trying to be comprehensive to every potential way SQLAlchemy can be used). If there's something that doesn't work in SQLAlchemy 2 that you think should still work, that sounds like something you should report to SQLAlchemy. |
This is a fair response. And nor should flask_sqlalchemy/you have to handle changes in behaviour in sqlalchemy2. I should also be including But I still don't believe your snippet is sufficient for the majority of people trying to test their application. For example if you include |
But yes I am speaking about sqlalchemy 2. Sorry that was probably not clear. Your documentation is sufficient for sqlalchemy 1. |
This snippet works as written for SQLAlchemy 1.4 and 2.0. Commits in views are persisted during each test, but are not present in subsequent tests. |
You are correct. I wrote up the above with simple test cases. I overlook/expected By calling In any case, I retract my comments. It works, is simple, and is possible to put in a test fixture. Which the solution I have been working with is not so elegant. Cheers! |
I used the suggested code in a fixture, but after 20 tests, an error is thrown:
It look like the connections are not closed properly. Edit: David edited my code snipet, so i don't dare to rewrite it, but the error was on my side: i made the mistake of creating the app in the same fixture, so for each test, a new app was instancied, moving the app creation in a session scoped fixture fixed the problem. Thank you |
If anyone else was like me and had issues migrating their test suite to Flask-SQLAlchemy 3.1 / SQLAlchemy 2.0, I created a minimum working example which I used to figure out why my errors were happening. My main issue was ensuring tests could run in rollback-able transactions. https://github.com/gmassman/fsa-rollback-per-test-example To be clear, Flask-SQLAlchemy works perfectly well with pytest. However, it doesn't work well with some of the older patterns that might have been promoted in the past (e.g. creating a |
In order to get this working in code that test views with But doing so results in some warnings:
@pytest.fixture(autouse=True, scope="function")
def database(app):
# https://github.com/pallets-eco/flask-sqlalchemy/issues/1171
with app.app_context():
engines = db.engines
engine_cleanup = []
for key, engine in engines.items():
connection = engine.connect()
transaction = connection.begin_nested()
engines[key] = connection
engine_cleanup.append((key, engine, connection, transaction))
yield db
for key, engine, connection, transaction in engine_cleanup:
with warnings.catch_warnings():
warnings.simplefilter("ignore")
transaction.rollback()
connection.close()
engines[key] = engine (dumb example) but it's pretty common you might have some code that does something like: @index.put("/user")
def update_email():
user = db.session.get(User, request.json["id"])
user.email = request.json["email"]
if "@" not in user.email:
db.session.rollback()
return "error", 400
db.session.commit()
return "ok" def test_change_email(client):
user = User(email="[email protected]")
db.session.add(user)
db.session.commit()
# causes a warning
resp = client.put("/user", json={"id": user.id, "email": "haha"})
assert resp.status_code == 400
assert resp.text == "error"
assert user.email == "[email protected]"
resp = client.put("/user", json={"id": user.id, "email": "[email protected]"})
assert resp.status_code == 200
assert resp.text == "ok"
assert user.email == "[email protected]" |
This comment was marked as outdated.
This comment was marked as outdated.
@gmassman I'm hiding your reply because it has an even worse bug that's immediately apparent: you're pushing an app context then yielding, so the context is active during the entire test session. This is explicitly warned against in multiple places, you must not keep an active context around an individual test or the session. Only push a context for exactly as long as needed to do a specific piece of work. Perhaps your code does fix some shortcoming, but it's otherwise incorrect. You might be better off explaining what specifically you needed to add to the previous suggested code and why. |
Got it, thanks for the heads up! I wasn't aware this was a discouraged pattern. It's been part of our conftest.py for many years at this point, so I'm happy to remove it and handle app_context the right way. I'm sure with this removal I can get the previous code snippets to work. |
The fixtures that push transactions use a lot of objects that are either contextmanagers themselves or fit closing I wonder if using a exitstack wouldn't be of help |
Mainly, show how to isolated the database per test, so that changes are rolled back without having to re-create the database each time. The pytest-flask-sqlalchemy package provided this, but stopped working when Flask-SQAlchemy 3 changed how engines and the session were handled. After investigation, it seems like there's a much simpler way to do this.
I'll at least document the following code, but it's probably better to include it as a patch function in a
flask_sqlalchemy.test
module. Could possibly add pytest fixtures as well that used it.Unlike the linked package, this is all that's needed and supports the multiple binds feature.
The text was updated successfully, but these errors were encountered: