# Copyright 2023-2024 SGLang Team
# Copyright 2025 Bytedance Ltd. and/or its affiliates
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
#     http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
from unittest.mock import AsyncMock, MagicMock, patch

import pytest
from omegaconf import DictConfig


@patch.dict(
    "sys.modules",
    {
        "verl.workers.rollout.sglang_rollout.sglang_rollout": MagicMock(SGLangRollout=MagicMock()),
        "verl.workers.rollout.chat_scheduler": MagicMock(ChatCompletionScheduler=MagicMock()),
        "fastapi": MagicMock(FastAPI=MagicMock()),
        "uvicorn": MagicMock(FastAPI=MagicMock()),
        "starlette.requests": MagicMock(Request=MagicMock()),
        "starlette.responses": MagicMock(JSONResponse=MagicMock()),
    },
)
class TestAsyncSglangServer:
    @pytest.fixture
    def server_config(self):
        return DictConfig({"actor_rollout_ref": {"rollout": {"tensor_model_parallel_size": 2}}})

    @pytest.mark.asyncio
    @patch("verl.workers.rollout.sglang_rollout.async_sglang_server.ray.util.list_named_actors")
    @patch("verl.workers.rollout.async_server.AsyncServerBase._start_fastapi_server", new_callable=AsyncMock)
    @pytest.mark.filterwarnings("ignore:Ray state API is no longer experimental:DeprecationWarning")
    async def test_init_engine(self, mock_start_fastapi_server, mock_list_actors, server_config):
        from verl.workers.rollout.sglang_rollout.async_sglang_server import AsyncSGLangServer

        ActualClassToInstantiate = AsyncSGLangServer
        if hasattr(AsyncSGLangServer, "__ray_metadata__") and hasattr(
            AsyncSGLangServer.__ray_metadata__, "modified_class"
        ):
            ActualClassToInstantiate = AsyncSGLangServer.__ray_metadata__.modified_class

        def mock_get_actor_side_effect(name, namespace=None):
            # Create a new mock actor for each call
            actor_mock = MagicMock()

            # Support .name attribute access
            actor_mock.name = name  # Use 'name' here

            # Support ['name'] item access by mocking __getitem__
            def getitem_mock(key):
                if key == "name":
                    return name  # Use 'name' here
                # For other keys, return a new MagicMock to mimic default behavior or raise KeyError
                # Returning a MagicMock is consistent with the original error's cause for unmocked keys
                return MagicMock(name=f"mock.__getitem__('{key}')")

            actor_mock.__getitem__.side_effect = getitem_mock

            return actor_mock

        # Verify instance.workers is correctly populated
        with patch(
            "verl.workers.rollout.sglang_rollout.async_sglang_server.ray.get_actor",
            side_effect=mock_get_actor_side_effect,
        ):
            # nnodes: 2
            # n_gpus_per_node: 4
            # tensor_model_parallel_size: 2
            # DP_size: 4
            mock_list_actors.return_value = [
                {"name": "test_xxxx", "namespace": "test"},
                {"name": "test_prefixWorkerDict_1:0", "namespace": "test"},
                {"name": "test_prefixWorkerDict_1:1", "namespace": "test"},
                {"name": "test_prefixWorkerDict_0:0", "namespace": "test"},
                {"name": "test_prefixWorkerDict_0:1", "namespace": "test"},
                {"name": "test_prefixWorkerDict_1:2", "namespace": "test"},
                {"name": "test_prefixWorkerDict_1:3", "namespace": "test"},
                {"name": "test_prefixWorkerDict_0:2", "namespace": "test"},
                {"name": "test_prefixWorkerDict_0:3", "namespace": "test"},
            ]

            # Instance 1
            instance = ActualClassToInstantiate(server_config, 4, 0, "test_prefix")
            await instance.init_engine()

            assert len(instance.workers) == 2
            assert instance.master_worker["name"] == "test_prefixWorkerDict_0:0"
            assert instance.workers[0].name == "test_prefixWorkerDict_0:0"
            assert instance.workers[1].name == "test_prefixWorkerDict_0:1"

            # Instance 2
            instance = ActualClassToInstantiate(server_config, 4, 1, "test_prefix")
            await instance.init_engine()

            assert len(instance.workers) == 2
            assert instance.master_worker["name"] == "test_prefixWorkerDict_0:2"
            assert instance.workers[0].name == "test_prefixWorkerDict_0:2"
            assert instance.workers[1].name == "test_prefixWorkerDict_0:3"

            # Instance 3
            instance = ActualClassToInstantiate(server_config, 4, 3, "test_prefix")
            await instance.init_engine()

            assert len(instance.workers) == 2
            assert instance.master_worker["name"] == "test_prefixWorkerDict_1:2"
            assert instance.workers[0].name == "test_prefixWorkerDict_1:2"
            assert instance.workers[1].name == "test_prefixWorkerDict_1:3"

            # nnodes: 4
            # n_gpus_per_node: 8
            # tensor_model_parallel_size: 8
            # DP_size: 4
            mock_list_actors.return_value = [
                {"name": "test_prefixWorkerDict_0:0", "namespace": "test"},
                {"name": "test_prefixWorkerDict_0:1", "namespace": "test"},
                {"name": "test_prefixWorkerDict_0:2", "namespace": "test"},
                {"name": "test_prefixWorkerDict_0:3", "namespace": "test"},
                {"name": "test_prefixWorkerDict_0:4", "namespace": "test"},
                {"name": "test_prefixWorkerDict_0:5", "namespace": "test"},
                {"name": "test_prefixWorkerDict_0:6", "namespace": "test"},
                {"name": "test_prefixWorkerDict_0:7", "namespace": "test"},
                {"name": "test_prefixWorkerDict_1:0", "namespace": "test"},
                {"name": "test_prefixWorkerDict_1:1", "namespace": "test"},
                {"name": "test_prefixWorkerDict_1:2", "namespace": "test"},
                {"name": "test_prefixWorkerDict_1:3", "namespace": "test"},
                {"name": "test_prefixWorkerDict_1:4", "namespace": "test"},
                {"name": "test_prefixWorkerDict_1:5", "namespace": "test"},
                {"name": "test_prefixWorkerDict_1:6", "namespace": "test"},
                {"name": "test_prefixWorkerDict_1:7", "namespace": "test"},
                {"name": "test_prefixWorkerDict_2:0", "namespace": "test"},
                {"name": "test_prefixWorkerDict_2:1", "namespace": "test"},
                {"name": "test_prefixWorkerDict_2:2", "namespace": "test"},
                {"name": "test_prefixWorkerDict_2:3", "namespace": "test"},
                {"name": "test_prefixWorkerDict_2:4", "namespace": "test"},
                {"name": "test_prefixWorkerDict_2:5", "namespace": "test"},
                {"name": "test_prefixWorkerDict_2:6", "namespace": "test"},
                {"name": "test_prefixWorkerDict_2:7", "namespace": "test"},
                {"name": "test_prefixWorkerDict_3:0", "namespace": "test"},
                {"name": "test_prefixWorkerDict_3:1", "namespace": "test"},
                {"name": "test_prefixWorkerDict_3:2", "namespace": "test"},
                {"name": "test_prefixWorkerDict_3:3", "namespace": "test"},
                {"name": "test_prefixWorkerDict_3:4", "namespace": "test"},
                {"name": "test_prefixWorkerDict_3:5", "namespace": "test"},
                {"name": "test_prefixWorkerDict_3:6", "namespace": "test"},
                {"name": "test_prefixWorkerDict_3:7", "namespace": "test"},
            ]

            server_config.actor_rollout_ref.rollout.tensor_model_parallel_size = 8
            # Instance 1
            instance = ActualClassToInstantiate(server_config, 4, 0, "test_prefix")
            await instance.init_engine()

            assert len(instance.workers) == 8
            assert instance.master_worker["name"] == "test_prefixWorkerDict_0:0"
            assert instance.workers[0].name == "test_prefixWorkerDict_0:0"
            assert instance.workers[7].name == "test_prefixWorkerDict_0:7"

            # Instance 2
            instance = ActualClassToInstantiate(server_config, 4, 1, "test_prefix")
            await instance.init_engine()

            assert len(instance.workers) == 8
            assert instance.master_worker["name"] == "test_prefixWorkerDict_1:0"
            assert instance.workers[0].name == "test_prefixWorkerDict_1:0"
            assert instance.workers[7].name == "test_prefixWorkerDict_1:7"

            # Instance 3
            instance = ActualClassToInstantiate(server_config, 4, 3, "test_prefix")
            await instance.init_engine()

            assert len(instance.workers) == 8
            assert instance.master_worker["name"] == "test_prefixWorkerDict_3:0"
            assert instance.workers[0].name == "test_prefixWorkerDict_3:0"
            assert instance.workers[7].name == "test_prefixWorkerDict_3:7"
