2023-04-27 09:43:12 +00:00
|
|
|
import contextlib
|
2022-06-30 20:43:04 +00:00
|
|
|
import os
|
|
|
|
import shutil
|
2022-12-19 23:12:55 +00:00
|
|
|
import signal
|
2022-10-28 13:26:08 +00:00
|
|
|
import threading
|
2022-06-30 20:43:04 +00:00
|
|
|
from datetime import datetime
|
|
|
|
from pathlib import Path
|
2022-12-19 23:12:55 +00:00
|
|
|
from threading import Thread
|
2022-06-30 20:43:04 +00:00
|
|
|
|
|
|
|
import psutil
|
|
|
|
import py
|
|
|
|
import pytest
|
2023-02-03 15:03:25 +00:00
|
|
|
from lightning.app.core import constants
|
2023-02-01 11:07:00 +00:00
|
|
|
from lightning.app.utilities.app_helpers import _collect_child_process_pids
|
|
|
|
from lightning.app.utilities.component import _set_context
|
|
|
|
from lightning.app.utilities.packaging import cloud_compute
|
|
|
|
from lightning.app.utilities.packaging.app_config import _APP_CONFIG_FILENAME
|
|
|
|
from lightning.app.utilities.state import AppState
|
2022-06-30 20:43:04 +00:00
|
|
|
|
2022-11-09 20:46:31 +00:00
|
|
|
os.environ["LIGHTNING_DISPATCHED"] = "1"
|
|
|
|
|
2022-12-19 23:12:55 +00:00
|
|
|
original_method = Thread._wait_for_tstate_lock
|
|
|
|
|
|
|
|
|
|
|
|
def fn(self, *args, timeout=None, **kwargs):
|
|
|
|
original_method(self, *args, timeout=1, **kwargs)
|
|
|
|
|
|
|
|
|
|
|
|
Thread._wait_for_tstate_lock = fn
|
|
|
|
|
2022-06-30 20:43:04 +00:00
|
|
|
|
|
|
|
def pytest_sessionfinish(session, exitstatus):
|
|
|
|
"""Pytest hook that get called after whole test run finished, right before returning the exit status to the
|
|
|
|
system."""
|
|
|
|
# kill all the processes and threads created by parent
|
|
|
|
# TODO this isn't great. We should have each tests doing it's own cleanup
|
|
|
|
current_process = psutil.Process()
|
|
|
|
for child in current_process.children(recursive=True):
|
2023-04-27 09:43:12 +00:00
|
|
|
with contextlib.suppress(psutil.NoSuchProcess):
|
2022-11-10 11:21:51 +00:00
|
|
|
params = child.as_dict() or {}
|
|
|
|
cmd_lines = params.get("cmdline", [])
|
|
|
|
# we shouldn't kill the resource tracker from multiprocessing. If we do,
|
|
|
|
# `atexit` will throw as it uses resource tracker to try to clean up
|
|
|
|
if cmd_lines and "resource_tracker" in cmd_lines[-1]:
|
|
|
|
continue
|
|
|
|
child.kill()
|
2022-06-30 20:43:04 +00:00
|
|
|
|
2022-10-28 13:26:08 +00:00
|
|
|
main_thread = threading.current_thread()
|
|
|
|
for t in threading.enumerate():
|
|
|
|
if t is not main_thread:
|
|
|
|
t.join(0)
|
|
|
|
|
2022-12-19 23:12:55 +00:00
|
|
|
for child_pid in _collect_child_process_pids(os.getpid()):
|
|
|
|
os.kill(child_pid, signal.SIGTERM)
|
|
|
|
|
2022-06-30 20:43:04 +00:00
|
|
|
|
2023-05-04 15:50:39 +00:00
|
|
|
@pytest.fixture(autouse=True)
|
2022-06-30 20:43:04 +00:00
|
|
|
def cleanup():
|
2023-02-01 11:07:00 +00:00
|
|
|
from lightning.app.utilities.app_helpers import _LightningAppRef
|
2022-06-30 20:43:04 +00:00
|
|
|
|
|
|
|
yield
|
|
|
|
_LightningAppRef._app_instance = None
|
|
|
|
shutil.rmtree("./storage", ignore_errors=True)
|
2023-09-28 13:37:37 +00:00
|
|
|
shutil.rmtree("./.storage", ignore_errors=True)
|
2022-06-30 20:43:04 +00:00
|
|
|
shutil.rmtree("./.shared", ignore_errors=True)
|
|
|
|
if os.path.isfile(_APP_CONFIG_FILENAME):
|
|
|
|
os.remove(_APP_CONFIG_FILENAME)
|
|
|
|
_set_context(None)
|
|
|
|
|
|
|
|
|
2023-05-04 15:50:39 +00:00
|
|
|
@pytest.fixture(autouse=True)
|
2022-06-30 20:43:04 +00:00
|
|
|
def clear_app_state_state_variables():
|
|
|
|
"""Resets global variables in order to prevent interference between tests."""
|
|
|
|
yield
|
2023-02-01 11:07:00 +00:00
|
|
|
import lightning.app.utilities.state
|
2022-06-30 20:43:04 +00:00
|
|
|
|
2023-02-01 11:07:00 +00:00
|
|
|
lightning.app.utilities.state._STATE = None
|
|
|
|
lightning.app.utilities.state._LAST_STATE = None
|
2022-06-30 20:43:04 +00:00
|
|
|
AppState._MY_AFFILIATION = ()
|
2022-10-20 14:18:06 +00:00
|
|
|
if hasattr(cloud_compute, "_CLOUD_COMPUTE_STORE"):
|
|
|
|
cloud_compute._CLOUD_COMPUTE_STORE.clear()
|
2022-06-30 20:43:04 +00:00
|
|
|
|
|
|
|
|
2023-05-04 15:50:39 +00:00
|
|
|
@pytest.fixture()
|
2022-06-30 20:43:04 +00:00
|
|
|
def another_tmpdir(tmp_path: Path) -> py.path.local:
|
|
|
|
random_dir = datetime.now().strftime("%m-%d-%Y-%H-%M-%S")
|
|
|
|
tmp_path = os.path.join(tmp_path, random_dir)
|
|
|
|
return py.path.local(tmp_path)
|
2022-11-24 14:09:25 +00:00
|
|
|
|
|
|
|
|
2023-05-04 15:50:39 +00:00
|
|
|
@pytest.fixture()
|
2022-11-24 14:09:25 +00:00
|
|
|
def caplog(caplog):
|
|
|
|
"""Workaround for https://github.com/pytest-dev/pytest/issues/3697.
|
|
|
|
|
|
|
|
Setting ``filterwarnings`` with pytest breaks ``caplog`` when ``not logger.propagate``.
|
2023-08-09 14:44:20 +00:00
|
|
|
|
2022-11-24 14:09:25 +00:00
|
|
|
"""
|
|
|
|
import logging
|
|
|
|
|
|
|
|
root_logger = logging.getLogger()
|
|
|
|
root_propagate = root_logger.propagate
|
|
|
|
root_logger.propagate = True
|
|
|
|
|
|
|
|
propagation_dict = {
|
|
|
|
name: logging.getLogger(name).propagate
|
|
|
|
for name in logging.root.manager.loggerDict
|
2023-02-01 11:07:00 +00:00
|
|
|
if name.startswith("lightning.app")
|
2022-11-24 14:09:25 +00:00
|
|
|
}
|
2023-04-24 21:57:08 +00:00
|
|
|
for name in propagation_dict:
|
2022-11-24 14:09:25 +00:00
|
|
|
logging.getLogger(name).propagate = True
|
|
|
|
|
|
|
|
yield caplog
|
|
|
|
|
|
|
|
root_logger.propagate = root_propagate
|
|
|
|
for name, propagate in propagation_dict.items():
|
|
|
|
logging.getLogger(name).propagate = propagate
|
2023-02-03 15:03:25 +00:00
|
|
|
|
|
|
|
|
2023-05-04 15:50:39 +00:00
|
|
|
@pytest.fixture()
|
2023-02-03 15:03:25 +00:00
|
|
|
def patch_constants(request):
|
2023-08-09 14:44:20 +00:00
|
|
|
"""This fixture can be used with indirect parametrization to patch values in `lightning.app.core.constants` for the
|
|
|
|
duration of a test.
|
2023-02-03 15:03:25 +00:00
|
|
|
|
|
|
|
Example::
|
|
|
|
|
|
|
|
@pytest.mark.parametrize("patch_constants", [{"LIGHTNING_CLOUDSPACE_HOST": "any"}], indirect=True)
|
|
|
|
def test_my_stuff(patch_constants):
|
|
|
|
...
|
2023-08-09 14:44:20 +00:00
|
|
|
|
2023-02-03 15:03:25 +00:00
|
|
|
"""
|
|
|
|
# Set constants
|
|
|
|
old_constants = {}
|
|
|
|
for constant, value in request.param.items():
|
|
|
|
old_constants[constant] = getattr(constants, constant)
|
|
|
|
setattr(constants, constant, value)
|
|
|
|
|
|
|
|
yield
|
|
|
|
|
|
|
|
for constant, value in old_constants.items():
|
|
|
|
setattr(constants, constant, value)
|