120f0262f7
* Fix Ctrl+C on Windows.
* Disable testing of a function N/A on Windows.
* Add test for coverage, avoid crash on missing _stopping.
* Initialise StreamingHTTPResponse.protocol = None
* Improved comments.
* Reduce amount of data in test_request_stream to avoid failures on Windows.
* The Windows test doesn't work on Windows :(
* Use port numbers more likely to be free than 8000.
* Disable the other signal tests on Windows as well.
* Windows doesn't properly support SO_REUSEADDR, so that's disabled in Python, and thus rebinding fails. For successful testing, reuse port instead.
* app.run argument handling: added server kwargs (alike create_server), added warning on extra kwargs, made auto_reload explicit argument. Another go at Windows tests
* Revert "app.run argument handling: added server kwargs (alike create_server), added warning on extra kwargs, made auto_reload explicit argument. Another go at Windows tests"
This reverts commit dc5d682448
.
* Use random test server port on most tests. Should avoid port/addr reuse issues.
* Another test to random port instead of 8000.
* Fix deprecation warnings about missing name on Sanic() in tests.
* Linter and typing
* Increase test coverage
* Rewrite test for ctrlc_windows_workaround
* py36 compat
* py36 compat
* py36 compat
* Don't rely on loop internals but add a stopping flag to app.
* App may be restarted.
* py36 compat
* Linter
* Add a constant for OS checking.
Co-authored-by: L. Kärkkäinen <tronic@users.noreply.github.com>
116 lines
2.9 KiB
Python
116 lines
2.9 KiB
Python
import asyncio
|
|
import os
|
|
import signal
|
|
|
|
from queue import Queue
|
|
from unittest.mock import MagicMock
|
|
|
|
import pytest
|
|
|
|
from sanic.compat import ctrlc_workaround_for_windows
|
|
from sanic.response import HTTPResponse
|
|
from sanic.testing import HOST, PORT
|
|
|
|
|
|
async def stop(app, loop):
|
|
await asyncio.sleep(0.1)
|
|
app.stop()
|
|
|
|
|
|
calledq = Queue()
|
|
|
|
|
|
def set_loop(app, loop):
|
|
global mock
|
|
mock = MagicMock()
|
|
if os.name == "nt":
|
|
signal.signal = mock
|
|
else:
|
|
loop.add_signal_handler = mock
|
|
|
|
|
|
def after(app, loop):
|
|
calledq.put(mock.called)
|
|
|
|
|
|
@pytest.mark.skipif(
|
|
os.name == "nt", reason="May hang CI on py38/windows"
|
|
)
|
|
def test_register_system_signals(app):
|
|
"""Test if sanic register system signals"""
|
|
|
|
@app.route("/hello")
|
|
async def hello_route(request):
|
|
return HTTPResponse()
|
|
|
|
app.listener("after_server_start")(stop)
|
|
app.listener("before_server_start")(set_loop)
|
|
app.listener("after_server_stop")(after)
|
|
|
|
app.run(HOST, PORT)
|
|
assert calledq.get() is True
|
|
|
|
|
|
@pytest.mark.skipif(
|
|
os.name == "nt", reason="May hang CI on py38/windows"
|
|
)
|
|
def test_dont_register_system_signals(app):
|
|
"""Test if sanic don't register system signals"""
|
|
|
|
@app.route("/hello")
|
|
async def hello_route(request):
|
|
return HTTPResponse()
|
|
|
|
app.listener("after_server_start")(stop)
|
|
app.listener("before_server_start")(set_loop)
|
|
app.listener("after_server_stop")(after)
|
|
|
|
app.run(HOST, PORT, register_sys_signals=False)
|
|
assert calledq.get() is False
|
|
|
|
|
|
@pytest.mark.skipif(
|
|
os.name == "nt", reason="windows cannot SIGINT processes"
|
|
)
|
|
def test_windows_workaround():
|
|
"""Test Windows workaround (on any other OS)"""
|
|
# At least some code coverage, even though this test doesn't work on
|
|
# Windows...
|
|
class MockApp:
|
|
def __init__(self):
|
|
self.is_stopping = False
|
|
|
|
def stop(self):
|
|
assert not self.is_stopping
|
|
self.is_stopping = True
|
|
|
|
def add_task(self, func):
|
|
loop = asyncio.get_event_loop()
|
|
self.stay_active_task = loop.create_task(func(self))
|
|
|
|
async def atest(stop_first):
|
|
app = MockApp()
|
|
ctrlc_workaround_for_windows(app)
|
|
await asyncio.sleep(0.05)
|
|
if stop_first:
|
|
app.stop()
|
|
await asyncio.sleep(0.2)
|
|
assert app.is_stopping == stop_first
|
|
# First Ctrl+C: should call app.stop() within 0.1 seconds
|
|
os.kill(os.getpid(), signal.SIGINT)
|
|
await asyncio.sleep(0.2)
|
|
assert app.is_stopping
|
|
assert app.stay_active_task.result() == None
|
|
# Second Ctrl+C should raise
|
|
with pytest.raises(KeyboardInterrupt):
|
|
os.kill(os.getpid(), signal.SIGINT)
|
|
return "OK"
|
|
|
|
# Run in our private loop
|
|
loop = asyncio.new_event_loop()
|
|
asyncio.set_event_loop(loop)
|
|
res = loop.run_until_complete(atest(False))
|
|
assert res == "OK"
|
|
res = loop.run_until_complete(atest(True))
|
|
assert res == "OK"
|