2
2
3
3
import asyncio
4
4
import contextlib
5
+ import logging
5
6
import signal
6
7
import sys
7
8
from typing import Callable , ContextManager , Generator
8
9
10
+ import httpx
9
11
import pytest
10
12
13
+ from tests .utils import run_server
14
+ from uvicorn ._types import ASGIReceiveCallable , ASGISendCallable , Scope
11
15
from uvicorn .config import Config
16
+ from uvicorn .protocols .http .h11_impl import H11Protocol
17
+ from uvicorn .protocols .http .httptools_impl import HttpToolsProtocol
12
18
from uvicorn .server import Server
13
19
20
+ pytestmark = pytest .mark .anyio
21
+
14
22
15
23
# asyncio does NOT allow raising in signal handlers, so to detect
16
24
# raised signals raised a mutable `witness` receives the signal
@@ -37,6 +45,12 @@ async def dummy_app(scope, receive, send): # pragma: py-win32
37
45
pass
38
46
39
47
48
+ async def app (scope : Scope , receive : ASGIReceiveCallable , send : ASGISendCallable ) -> None :
49
+ assert scope ["type" ] == "http"
50
+ await send ({"type" : "http.response.start" , "status" : 200 , "headers" : []})
51
+ await send ({"type" : "http.response.body" , "body" : b"" , "more_body" : False })
52
+
53
+
40
54
if sys .platform == "win32" : # pragma: py-not-win32
41
55
signals = [signal .SIGBREAK ]
42
56
signal_captures = [capture_signal_sync ]
@@ -45,7 +59,6 @@ async def dummy_app(scope, receive, send): # pragma: py-win32
45
59
signal_captures = [capture_signal_sync , capture_signal_async ]
46
60
47
61
48
- @pytest .mark .anyio
49
62
@pytest .mark .parametrize ("exception_signal" , signals )
50
63
@pytest .mark .parametrize ("capture_signal" , signal_captures )
51
64
async def test_server_interrupt (
@@ -65,3 +78,16 @@ async def interrupt_running(srv: Server):
65
78
assert witness
66
79
# set by the server's graceful exit handler
67
80
assert server .should_exit
81
+
82
+
83
+ async def test_request_than_limit_max_requests_warn_log (
84
+ unused_tcp_port : int , http_protocol_cls : type [H11Protocol | HttpToolsProtocol ], caplog : pytest .LogCaptureFixture
85
+ ):
86
+ caplog .set_level (logging .WARNING , logger = "uvicorn.error" )
87
+ config = Config (app = app , limit_max_requests = 1 , port = unused_tcp_port , http = http_protocol_cls )
88
+ async with run_server (config ):
89
+ async with httpx .AsyncClient () as client :
90
+ tasks = [client .get (f"http://127.0.0.1:{ unused_tcp_port } " ) for _ in range (2 )]
91
+ responses = await asyncio .gather (* tasks )
92
+ assert len (responses ) == 2
93
+ assert "Maximum request limit of 1 exceeded. Terminating process." in caplog .text
0 commit comments