2021-04-18 18:55:51 +00:00
|
|
|
"""Test Home Assistant executor util."""
|
|
|
|
|
|
|
|
import concurrent.futures
|
|
|
|
import time
|
|
|
|
from unittest.mock import patch
|
|
|
|
|
|
|
|
import pytest
|
|
|
|
|
|
|
|
from homeassistant.util import executor
|
|
|
|
from homeassistant.util.executor import InterruptibleThreadPoolExecutor
|
|
|
|
|
|
|
|
|
2023-02-20 10:42:56 +00:00
|
|
|
async def test_executor_shutdown_can_interrupt_threads(
|
|
|
|
caplog: pytest.LogCaptureFixture,
|
|
|
|
) -> None:
|
2021-04-18 18:55:51 +00:00
|
|
|
"""Test that the executor shutdown can interrupt threads."""
|
|
|
|
|
|
|
|
iexecutor = InterruptibleThreadPoolExecutor()
|
|
|
|
|
|
|
|
def _loop_sleep_in_executor():
|
|
|
|
while True:
|
|
|
|
time.sleep(0.1)
|
|
|
|
|
2024-03-14 09:22:20 +00:00
|
|
|
sleep_futures = [iexecutor.submit(_loop_sleep_in_executor) for _ in range(100)]
|
2021-04-18 18:55:51 +00:00
|
|
|
|
2021-07-07 07:23:24 +00:00
|
|
|
iexecutor.shutdown()
|
2021-04-18 18:55:51 +00:00
|
|
|
|
|
|
|
for future in sleep_futures:
|
|
|
|
with pytest.raises((concurrent.futures.CancelledError, SystemExit)):
|
|
|
|
future.result()
|
|
|
|
|
|
|
|
assert "is still running at shutdown" in caplog.text
|
|
|
|
assert "time.sleep(0.1)" in caplog.text
|
|
|
|
|
|
|
|
|
2023-02-20 10:42:56 +00:00
|
|
|
async def test_executor_shutdown_only_logs_max_attempts(
|
|
|
|
caplog: pytest.LogCaptureFixture,
|
|
|
|
) -> None:
|
2021-04-18 18:55:51 +00:00
|
|
|
"""Test that the executor shutdown will only log max attempts."""
|
|
|
|
|
|
|
|
iexecutor = InterruptibleThreadPoolExecutor()
|
|
|
|
|
|
|
|
def _loop_sleep_in_executor():
|
|
|
|
time.sleep(0.2)
|
|
|
|
|
|
|
|
iexecutor.submit(_loop_sleep_in_executor)
|
|
|
|
|
|
|
|
with patch.object(executor, "EXECUTOR_SHUTDOWN_TIMEOUT", 0.3):
|
2021-07-07 07:23:24 +00:00
|
|
|
iexecutor.shutdown()
|
2021-04-18 18:55:51 +00:00
|
|
|
|
|
|
|
assert "time.sleep(0.2)" in caplog.text
|
2023-02-16 17:38:32 +00:00
|
|
|
assert "is still running at shutdown" in caplog.text
|
2021-07-07 07:23:24 +00:00
|
|
|
iexecutor.shutdown()
|
2021-04-18 18:55:51 +00:00
|
|
|
|
|
|
|
|
2023-02-20 10:42:56 +00:00
|
|
|
async def test_executor_shutdown_does_not_log_shutdown_on_first_attempt(
|
|
|
|
caplog: pytest.LogCaptureFixture,
|
|
|
|
) -> None:
|
2021-04-18 18:55:51 +00:00
|
|
|
"""Test that the executor shutdown does not log on first attempt."""
|
|
|
|
|
|
|
|
iexecutor = InterruptibleThreadPoolExecutor()
|
|
|
|
|
|
|
|
def _do_nothing():
|
|
|
|
return
|
|
|
|
|
|
|
|
for _ in range(5):
|
|
|
|
iexecutor.submit(_do_nothing)
|
|
|
|
|
2021-07-07 07:23:24 +00:00
|
|
|
iexecutor.shutdown()
|
2021-04-18 18:55:51 +00:00
|
|
|
|
|
|
|
assert "is still running at shutdown" not in caplog.text
|
|
|
|
|
|
|
|
|
2023-02-20 10:42:56 +00:00
|
|
|
async def test_overall_timeout_reached(caplog: pytest.LogCaptureFixture) -> None:
|
2021-04-18 18:55:51 +00:00
|
|
|
"""Test that shutdown moves on when the overall timeout is reached."""
|
|
|
|
|
|
|
|
def _loop_sleep_in_executor():
|
|
|
|
time.sleep(1)
|
|
|
|
|
|
|
|
with patch.object(executor, "EXECUTOR_SHUTDOWN_TIMEOUT", 0.5):
|
2023-11-03 01:00:34 +00:00
|
|
|
iexecutor = InterruptibleThreadPoolExecutor()
|
|
|
|
for _ in range(6):
|
|
|
|
iexecutor.submit(_loop_sleep_in_executor)
|
|
|
|
start = time.monotonic()
|
2021-07-07 07:23:24 +00:00
|
|
|
iexecutor.shutdown()
|
2023-11-03 01:00:34 +00:00
|
|
|
finish = time.monotonic()
|
2021-04-18 18:55:51 +00:00
|
|
|
|
2024-05-24 06:22:29 +00:00
|
|
|
# Ideally execution time (finish - start) should be < 1.2 sec.
|
2023-12-01 06:05:26 +00:00
|
|
|
# CI tests might not run in an ideal environment and timing might
|
|
|
|
# not be accurate, so we let this test pass
|
|
|
|
# if the duration is below 3 seconds.
|
|
|
|
assert finish - start < 3.0
|
2021-04-18 18:55:51 +00:00
|
|
|
|
2021-07-07 07:23:24 +00:00
|
|
|
iexecutor.shutdown()
|