Skip to content
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

TestClient.request does not honor timeout #1108

Open
2 tasks done
falkben opened this issue Dec 8, 2020 · 7 comments
Open
2 tasks done

TestClient.request does not honor timeout #1108

falkben opened this issue Dec 8, 2020 · 7 comments
Labels
good first issue Good for beginners testclient TestClient-related

Comments

@falkben
Copy link
Contributor

falkben commented Dec 8, 2020

Checklist

  • The bug is reproducible against the latest release and/or master.
  • There are no similar issues or pull requests to fix it yet.

Describe the bug

timeout parameter exposed in testclient.request has no effect. Looking through the code, it seems as though we use the same interface as requests package but only a subset of those features are supported by testclient

To reproduce

import time

from starlette.applications import Starlette
from starlette.responses import JSONResponse
from starlette.testclient import TestClient

mock_service = Starlette()


@mock_service.route("/slow_response")
def slow_response(request):
    time.sleep(0.01)
    return JSONResponse({"mock": "slow example"})


def test_timeout():
    """ no timeout exception raised -- this test passes when it should fail """
    client = TestClient(mock_service, raise_server_exceptions=True)
    response = client.get("/slow_response", timeout=0.001)
    assert response.json() == {"mock": "slow example"}

Expected behavior

A TimeoutException should be raised if the timeout specified in request is shorter than the time the request takes.

Actual behavior

No timeout exceptions are raised when setting timeout.

Additional context

In CI/CD context, testing our own endpoints which, during integration testing, may timeout, is very helpful. If this is unsupported, it might be nice to have a list of differences between the interface of requests/testclient (this might include #1102).

I did look a bit how the requests package implemented their timeout exceptions, and to be completely honest I got a bit lost going down the rabbit hole of requests -> urllib3 -> http.client. Since we're using a different adapter for the testclient, it might be some work to reimplement all the way down the timeout exceptions in the same way.

I did notice that the async_asgi_testclient works with a timeout. Of course, it requires an async endpoint. But their implementation might still be useful?

Found that httpx had a similar issue with timeout having no effect when mounting an app with its client.

@JayH5 JayH5 added the testclient TestClient-related label Feb 4, 2021
@tomchristie
Copy link
Member

Closing as per #1109

timeouts == flaky tests

@tomchristie
Copy link
Member

Okay - that comment is a bit brief. But see #1109. Frameworks such as Flask and Django have always had this constraint. It's something that we can live with as well.

@logi
Copy link

logi commented Sep 6, 2022

I have a case where a "timeout" would be useful in testing how the server handles disconnecting clients. So a "timeout" where the client times out immediately after sending the request would be useful and should not be flaky since no actual timing is involved.

@Kludex
Copy link
Member

Kludex commented Sep 6, 2022

I have a case where a "timeout" would be useful in testing how the server handles disconnecting clients. So a "timeout" where the client times out immediately after sending the request would be useful and should not be flaky since no actual timing is involved.

This may help: 25a52fe

@falkben
Copy link
Contributor Author

falkben commented Sep 6, 2022

I was just looking at this again last week. I never could figure out how to get around the BlockingPortal. We run our tests in a synchronous context (not async as you linked to above) but that example @Kludex looks quite helpful.

Anyways, I sort of came to the conclusion that we should remove our individual test timeouts from our service unit tests as they were a sort of anti-pattern. Many of our service unit tests also serve as integration tests, but we probably need to separate those out better.

@dbmikus
Copy link

dbmikus commented Jan 4, 2025

I understand that this is closed, but I want to echo that I had this same use case #1108 (comment):

I have a case where a "timeout" would be useful in testing how the server handles disconnecting clients. So a "timeout" where the client times out immediately after sending the request would be useful and should not be flaky since no actual timing is involved.

I don't think that timeouts de-facto mean that tests are flaky.

If the Starlette maintainers still wish to exclude timeout behavior for some reason (I understand the maintainers put in the effort so they get to be the boss 😁), I think it would be best to deprecate the timeout field on the test client. It is misleading to have it there, but not change behavior based on it (for example, I wasted an hour trying to debug what was going on in my tests because of this)

@Kludex
Copy link
Member

Kludex commented Jan 4, 2025

Since the timeout is not doing anything, let's add a deprecation warning explaining it doesn't work, and remove the parameter in some time.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
good first issue Good for beginners testclient TestClient-related
Projects
None yet
Development

Successfully merging a pull request may close this issue.

6 participants