lightning/tests/tests_app/storage/test_orchestrator.py

78 lines
3.2 KiB
Python

from unittest.mock import MagicMock
from lightning_app.storage.orchestrator import StorageOrchestrator
from lightning_app.storage.path import GetRequest, GetResponse
from lightning_app.testing.helpers import MockQueue
from lightning_app.utilities.enum import WorkStageStatus
def test_orchestrator():
"""Simulate orchestration when Work B requests a file from Work A."""
request_queues = {"work_a": MockQueue(), "work_b": MockQueue()}
response_queues = {"work_a": MockQueue(), "work_b": MockQueue()}
copy_request_queues = {"work_a": MockQueue(), "work_b": MockQueue()}
copy_response_queues = {"work_a": MockQueue(), "work_b": MockQueue()}
app = MagicMock()
work = MagicMock()
work.status.stage = WorkStageStatus.RUNNING
app.get_component_by_name = MagicMock(return_value=work)
orchestrator = StorageOrchestrator(
app,
request_queues=request_queues,
response_queues=response_queues,
copy_request_queues=copy_request_queues,
copy_response_queues=copy_response_queues,
)
# test idle behavior when queues are empty
orchestrator.run_once("work_a")
orchestrator.run_once("work_b")
assert not orchestrator.waiting_for_response
# simulate Work B sending a request for a file in Work A
request = GetRequest(source="work_a", path="/a/b/c.txt", hash="", destination="", name="")
request_queues["work_b"].put(request)
orchestrator.run_once("work_a")
assert not orchestrator.waiting_for_response
orchestrator.run_once("work_b")
# orchestrator is now waiting for a response for copier in Work A
assert "work_b" in orchestrator.waiting_for_response
assert not request_queues["work_a"]
assert request in copy_request_queues["work_a"]
assert request.destination == "work_b"
# simulate loop while waiting for new elements in the queues
orchestrator.run_once("work_a")
orchestrator.run_once("work_b")
# simulate copier A confirms that the file is available on the shared volume
response = GetResponse(source="work_a", path="/a/b/c.txt", hash="", destination="work_b", name="")
copy_request_queues["work_a"].get()
copy_response_queues["work_a"].put(response)
# orchestrator processes confirmation and confirms to the pending request from Work B
orchestrator.run_once("work_a")
assert not copy_response_queues["work_a"]
assert response in response_queues["work_b"]
assert not orchestrator.waiting_for_response
orchestrator.run_once("work_b")
# simulate loop while waiting for new elements in the queues
orchestrator.run_once("work_a")
orchestrator.run_once("work_b")
assert not orchestrator.waiting_for_response
# simulate Work B receiving the confirmation that the file was copied
response = response_queues["work_b"].get()
assert response.source == "work_a"
assert response.destination == "work_b"
assert response.exception is None
# all queues should be empty
assert all(not queue for queue in request_queues.values())
assert all(not queue for queue in response_queues.values())
assert all(not queue for queue in copy_request_queues.values())
assert all(not queue for queue in copy_response_queues.values())