import datetime
import json
import os
import subprocess
from time import sleep, time

import allure
import requests
from openai import OpenAI
from pytest_assume.plugin import assume
from utils.config_utils import _is_bf16_supported_by_device, get_cuda_prefix_by_workerid, get_workerid
from utils.get_run_config import get_command_with_extra
from utils.restful_return_check import assert_chat_completions_batch_return
from utils.rule_condition_assert import assert_result

from lmdeploy.serve.openai.api_client import APIClient

MASTER_ADDR = os.getenv('MASTER_ADDR', 'localhost')
BASE_HTTP_URL = f'http://{MASTER_ADDR}'
DEFAULT_PORT = 23333
PROXY_PORT = 8000


def start_restful_api(config, param, model, model_path, backend_type, worker_id):
    log_path = config.get('log_path')

    cuda_prefix = param['cuda_prefix']
    tp_num = param['tp_num']

    parallel_config = param.get('parallel_config', {})

    if 'extra' in param.keys():
        extra = param['extra']
    else:
        extra = ''

    # temp remove testcase because of issue 3434
    if ('InternVL3' in model or 'InternVL2_5' in model or 'MiniCPM-V-2_6' in model):
        if 'turbomind' in backend_type and extra is not None and 'cuda-ipc' in extra and tp_num > 1:
            return 0, 'skip'

    if 'modelscope' in param.keys():
        modelscope = param['modelscope']
        if modelscope:
            os.environ['LMDEPLOY_USE_MODELSCOPE'] = 'True'
            model_path = model

    if cuda_prefix is None:
        cuda_prefix = get_cuda_prefix_by_workerid(worker_id, parallel_config=tp_num)

    if tp_num > 1 and 'gw' in worker_id:
        os.environ['MASTER_PORT'] = str(int(worker_id.replace('gw', '')) + 29500)

    worker_num = get_workerid(worker_id)
    if worker_num is None:
        port = DEFAULT_PORT
    else:
        port = DEFAULT_PORT + worker_num

    cmd = get_command_with_extra('lmdeploy serve api_server ' + model_path + ' --server-port ' + str(port) +
                                 ' --allow-terminate-by-client',
                                 config,
                                 model,
                                 need_tp=True,
                                 cuda_prefix=cuda_prefix,
                                 extra=extra)

    device = os.environ.get('DEVICE', '')
    if device:
        cmd += f' --device {device} '

    if parallel_config:
        if 'dp' in parallel_config:
            dp = parallel_config['dp']
            cmd += f' --dp {dp}'
        if 'ep' in parallel_config:
            ep = parallel_config['ep']
            cmd += f' --ep {ep}'
        if 'cp' in parallel_config:
            cp = parallel_config['cp']
            cmd += f' --cp {cp}'

    if backend_type == 'turbomind':
        if ('w4' in model or '4bits' in model or 'awq' in model.lower()):
            cmd += ' --model-format awq'
        elif 'gptq' in model.lower():
            cmd += ' --model-format gptq'
    if backend_type == 'pytorch':
        cmd += ' --backend pytorch'
        if not _is_bf16_supported_by_device():
            cmd += ' --dtype float16'
    if 'quant_policy' in param.keys() and param['quant_policy'] is not None:
        quant_policy = param['quant_policy']
        cmd += f' --quant-policy {quant_policy}'

    if not _is_bf16_supported_by_device():
        cmd += ' --cache-max-entry-count 0.5'
    if str(config.get('env_tag')) == '3090' or str(config.get('env_tag')) == '5080':
        cmd += ' --cache-max-entry-count 0.5'

    timestamp = datetime.datetime.now().strftime('%Y%m%d_%H%M%S')
    start_log = os.path.join(log_path, 'start_restful_' + model.split('/')[1] + worker_id + '_' + timestamp + '.log')

    print('reproduce command restful: ' + cmd)

    file = open(start_log, 'w')

    startRes = subprocess.Popen([cmd],
                                stdout=file,
                                stderr=file,
                                shell=True,
                                text=True,
                                encoding='utf-8',
                                errors='replace',
                                start_new_session=True)
    pid = startRes.pid

    http_url = BASE_HTTP_URL + ':' + str(port)
    start_time = int(time())
    start_timeout = 300
    if not _is_bf16_supported_by_device() or tp_num >= 4:
        start_timeout = 720

    sleep(5)
    for i in range(start_timeout):
        sleep(1)
        end_time = int(time())
        total_time = end_time - start_time
        result = health_check(http_url)
        if result or total_time >= start_timeout:
            break
        try:
            # Check if process is still running
            return_code = startRes.wait(timeout=1)  # Small timeout to check status
            if return_code != 0:
                with open(start_log, 'r') as f:
                    content = f.read()
                    print(content)
                return 0, startRes
        except subprocess.TimeoutExpired:
            continue
    file.close()
    allure.attach.file(start_log, attachment_type=allure.attachment_type.TEXT)
    return pid, startRes


def stop_restful_api(pid, startRes, param):
    if pid > 0:
        startRes.terminate()
    if 'modelscope' in param.keys():
        modelscope = param['modelscope']
        if modelscope:
            del os.environ['LMDEPLOY_USE_MODELSCOPE']
    if 'MASTER_PORT' in os.environ:
        del os.environ['MASTER_PORT']


def terminate_restful_api(worker_id, param):
    worker_num = get_workerid(worker_id)
    if worker_num is None:
        port = DEFAULT_PORT
    else:
        port = DEFAULT_PORT + worker_num
    http_url = BASE_HTTP_URL + ':' + str(port)

    response = None
    request_error = None
    try:
        response = requests.get(f'{http_url}/terminate')
    except requests.exceptions.RequestException as exc:
        request_error = exc
    finally:
        if 'modelscope' in param.keys():
            modelscope = param['modelscope']
            if modelscope:
                del os.environ['LMDEPLOY_USE_MODELSCOPE']
        if 'MASTER_PORT' in os.environ:
            del os.environ['MASTER_PORT']
    if request_error is not None:
        assert False, f'terminate request failed: {request_error}'
    assert response is not None and response.status_code == 200, f'terminate with {response}'


def run_all_step(config, cases_info, worker_id: str = '', port: int = DEFAULT_PORT):
    http_url = BASE_HTTP_URL + ':' + str(port)
    model = get_model(http_url)

    if model is None:
        assert False, 'server not start correctly'
    for case in cases_info.keys():
        if ('coder' in model.lower() or 'codellama' in model.lower()) and 'code' not in case:
            continue

        case_info = cases_info.get(case)

        with allure.step(case + ' restful_test - openai chat'):
            restful_result, restful_log, msg = open_chat_test(config, case, case_info, model, http_url, worker_id)
            allure.attach.file(restful_log, attachment_type=allure.attachment_type.TEXT)
        with assume:
            assert restful_result, msg


def open_chat_test(config, case, case_info, model, url, worker_id: str = ''):
    log_path = config.get('log_path')

    restful_log = os.path.join(log_path, 'restful_' + model + worker_id + '_' + case + '.log')

    file = open(restful_log, 'w')

    result = True

    client = OpenAI(api_key='YOUR_API_KEY', base_url=f'{url}/v1')
    model_name = client.models.list().data[0].id

    messages = []
    msg = ''
    for prompt_detail in case_info:
        if not result:
            break
        prompt = list(prompt_detail.keys())[0]
        messages.append({'role': 'user', 'content': prompt})
        file.writelines('prompt:' + prompt + '\n')

        response = client.chat.completions.create(model=model_name,
                                                  messages=messages,
                                                  temperature=0.01,
                                                  top_p=0.8,
                                                  max_completion_tokens=1024)

        output_content = response.choices[0].message.content
        file.writelines('output:' + output_content + '\n')
        messages.append({'role': 'assistant', 'content': output_content})

        case_result, reason = assert_result(output_content, prompt_detail.values(), model_name)
        file.writelines('result:' + str(case_result) + ',reason:' + reason + '\n')
        if not case_result:
            msg += reason
        result = result & case_result
    file.close()
    return result, restful_log, msg


def health_check(url):
    try:
        api_client = APIClient(url)
        model_name = api_client.available_models[0]
        messages = []
        messages.append({'role': 'user', 'content': '你好'})
        for output in api_client.chat_completions_v1(model=model_name, messages=messages, top_k=1):
            if output.get('code') is not None and output.get('code') != 0:
                return False
            return True
    except Exception:
        return False


def get_model(url):
    try:
        api_client = APIClient(url)
        model_name = api_client.available_models[0]
        return model_name.split('/')[-1]
    except Exception:
        return None


def test_logprobs(worker_id: str = None):
    worker_num = get_workerid(worker_id)
    if worker_num is None:
        port = DEFAULT_PORT
    else:
        port = DEFAULT_PORT + worker_num
    http_url = BASE_HTTP_URL + ':' + str(port)
    api_client = APIClient(http_url)
    model_name = api_client.available_models[0]
    for output in api_client.chat_completions_v1(model=model_name,
                                                 messages='Hi, pls intro yourself',
                                                 max_tokens=5,
                                                 temperature=0.01,
                                                 logprobs=True,
                                                 top_logprobs=10):
        continue
    print(output)
    assert_chat_completions_batch_return(output, model_name, check_logprobs=True, logprobs_num=10)
    assert output.get('choices')[0].get('finish_reason') == 'length'
    assert output.get('usage').get('completion_tokens') == 6 or output.get('usage').get('completion_tokens') == 5


PIC = 'https://raw.githubusercontent.com/open-mmlab/mmdeploy/main/tests/data/tiger.jpeg'  # noqa E501
PIC2 = 'https://raw.githubusercontent.com/open-mmlab/mmdeploy/main/demo/resources/human-pose.jpg'  # noqa E501


def run_vl_testcase(config, port: int = DEFAULT_PORT):
    http_url = BASE_HTTP_URL + ':' + str(port)
    log_path = config.get('log_path')

    model = get_model(http_url)
    if model is None:
        assert False, 'server not start correctly'

    client = OpenAI(api_key='YOUR_API_KEY', base_url=http_url + '/v1')
    model_name = client.models.list().data[0].id

    restful_log = os.path.join(log_path, 'restful_vl_' + model_name.split('/')[-1] + str(port) + '.log')
    file = open(restful_log, 'w')

    prompt_messages = [{
        'role':
        'user',
        'content': [{
            'type': 'text',
            'text': 'Describe the image please',
        }, {
            'type': 'image_url',
            'image_url': {
                'url': PIC,
            },
        }, {
            'type': 'image_url',
            'image_url': {
                'url': PIC2,
            },
        }],
    }]

    response = client.chat.completions.create(model=model_name, messages=prompt_messages, temperature=0.8, top_p=0.8)
    file.writelines(str(response).lower() + '\n')

    api_client = APIClient(http_url)
    model_name = api_client.available_models[0]
    for item in api_client.chat_completions_v1(model=model_name, messages=prompt_messages):
        continue
    file.writelines(str(item) + '\n')
    file.close()

    allure.attach.file(restful_log, attachment_type=allure.attachment_type.TEXT)

    assert 'tiger' in str(response).lower() or '虎' in str(response).lower() or 'ski' in str(
        response).lower() or '滑雪' in str(response).lower(), response
    assert 'tiger' in str(item).lower() or '虎' in str(item).lower() or 'ski' in str(item).lower() or '滑雪' in str(
        item).lower(), item


def run_reasoning_case(config, port: int = DEFAULT_PORT):
    http_url = BASE_HTTP_URL + ':' + str(port)
    log_path = config.get('log_path')

    model = get_model(http_url)

    if model is None:
        assert False, 'server not start correctly'

    restful_log = os.path.join(log_path, 'restful_reasoning_' + model + str(port) + '.log')
    file = open(restful_log, 'w')

    client = OpenAI(api_key='YOUR_API_KEY', base_url=http_url + '/v1')
    model_name = client.models.list().data[0].id

    with allure.step('step1 - stream'):
        messages = [{'role': 'user', 'content': '9.11 and 9.8, which is greater?'}]
        response = client.chat.completions.create(model=model_name, messages=messages, temperature=0.01, stream=True)
        outputList = []
        final_content = ''
        final_reasoning_content = ''
        for stream_response in response:
            if stream_response.choices[0].delta.content is not None:
                final_content += stream_response.choices[0].delta.content
            if stream_response.choices[0].delta.reasoning_content is not None:
                final_reasoning_content += stream_response.choices[0].delta.reasoning_content
            outputList.append(stream_response)
        file.writelines(str(outputList) + '\n')
        with assume:
            assert '9.11' in final_reasoning_content and '9.11' in final_content and len(outputList) > 1, str(
                outputList)

    with allure.step('step2 - batch'):
        response = client.chat.completions.create(model=model_name, messages=messages, temperature=0.01, stream=False)
        print(response)
        reasoning_content = response.choices[0].message.reasoning_content
        content = response.choices[0].message.content
        file.writelines(str(outputList) + '\n')
        with assume:
            assert '9.11' in reasoning_content and '9.11' in content and len(outputList) > 1, str(outputList)

    file.close()
    allure.attach.file(restful_log, attachment_type=allure.attachment_type.TEXT)


def test_internlm_multiple_round_prompt(client, model):

    def add(a: int, b: int):
        return a + b

    def mul(a: int, b: int):
        return a * b

    tools = [{
        'type': 'function',
        'function': {
            'name': 'add',
            'description': 'Compute the sum of two numbers',
            'parameters': {
                'type': 'object',
                'properties': {
                    'a': {
                        'type': 'int',
                        'description': 'A number',
                    },
                    'b': {
                        'type': 'int',
                        'description': 'A number',
                    },
                },
                'required': ['a', 'b'],
            },
        }
    }, {
        'type': 'function',
        'function': {
            'name': 'mul',
            'description': 'Calculate the product of two numbers',
            'parameters': {
                'type': 'object',
                'properties': {
                    'a': {
                        'type': 'int',
                        'description': 'A number',
                    },
                    'b': {
                        'type': 'int',
                        'description': 'A number',
                    },
                },
                'required': ['a', 'b'],
            },
        }
    }]
    messages = [{'role': 'user', 'content': 'Compute (3+5)*2'}]

    response = client.chat.completions.create(model=model,
                                              messages=messages,
                                              temperature=0.01,
                                              stream=False,
                                              tools=tools)
    print(response)
    response_list = [response]
    func1_name = response.choices[0].message.tool_calls[0].function.name
    func1_args = response.choices[0].message.tool_calls[0].function.arguments
    func1_out = eval(f'{func1_name}(**{func1_args})')
    with assume:
        assert response.choices[0].finish_reason == 'tool_calls'
    with assume:
        assert func1_name == 'add'
    with assume:
        assert func1_args == '{"a": 3, "b": 5}'
    with assume:
        assert func1_out == 8
    with assume:
        assert response.choices[0].message.tool_calls[0].type == 'function'

    messages.append({'role': 'assistant', 'content': response.choices[0].message.content})
    messages.append({'role': 'environment', 'content': f'3+5={func1_out}', 'name': 'plugin'})
    response = client.chat.completions.create(model=model,
                                              messages=messages,
                                              temperature=0.8,
                                              top_p=0.8,
                                              stream=False,
                                              tools=tools)
    print(response)
    response_list.append(response)
    func2_name = response.choices[0].message.tool_calls[0].function.name
    func2_args = response.choices[0].message.tool_calls[0].function.arguments
    func2_out = eval(f'{func2_name}(**{func2_args})')
    with assume:
        assert response.choices[0].finish_reason == 'tool_calls'
    with assume:
        assert func2_name == 'mul'
    with assume:
        assert func2_args == '{"a": 8, "b": 2}'
    with assume:
        assert func2_out == 16
    with assume:
        assert response.choices[0].message.tool_calls[0].type == 'function'

    return response_list


def test_qwen_multiple_round_prompt(client, model):

    def get_current_temperature(location: str, unit: str = 'celsius'):
        """Get current temperature at a location.

        Args:
            location: The location to get the temperature for, in the format "City, State, Country".
            unit: The unit to return the temperature in. Defaults to "celsius". (choices: ["celsius", "fahrenheit"])

        Returns:
            the temperature, the location, and the unit in a dict
        """
        return {
            'temperature': 26.1,
            'location': location,
            'unit': unit,
        }

    def get_temperature_date(location: str, date: str, unit: str = 'celsius'):
        """Get temperature at a location and date.

        Args:
            location: The location to get the temperature for, in the format 'City, State, Country'.
            date: The date to get the temperature for, in the format 'Year-Month-Day'.
            unit: The unit to return the temperature in. Defaults to 'celsius'. (choices: ['celsius', 'fahrenheit'])

        Returns:
            the temperature, the location, the date and the unit in a dict
        """
        return {
            'temperature': 25.9,
            'location': location,
            'date': date,
            'unit': unit,
        }

    def get_function_by_name(name):
        if name == 'get_current_temperature':
            return get_current_temperature
        if name == 'get_temperature_date':
            return get_temperature_date

    tools = [{
        'type': 'function',
        'function': {
            'name': 'get_current_temperature',
            'description': 'Get current temperature at a location.',
            'parameters': {
                'type': 'object',
                'properties': {
                    'location': {
                        'type': 'string',
                        'description':
                        'The location to get the temperature for, in the format \'City, State, Country\'.'
                    },
                    'unit': {
                        'type': 'string',
                        'enum': ['celsius', 'fahrenheit'],
                        'description': 'The unit to return the temperature in. Defaults to \'celsius\'.'
                    }
                },
                'required': ['location']
            }
        }
    }, {
        'type': 'function',
        'function': {
            'name': 'get_temperature_date',
            'description': 'Get temperature at a location and date.',
            'parameters': {
                'type': 'object',
                'properties': {
                    'location': {
                        'type': 'string',
                        'description':
                        'The location to get the temperature for, in the format \'City, State, Country\'.'
                    },
                    'date': {
                        'type': 'string',
                        'description': 'The date to get the temperature for, in the format \'Year-Month-Day\'.'
                    },
                    'unit': {
                        'type': 'string',
                        'enum': ['celsius', 'fahrenheit'],
                        'description': 'The unit to return the temperature in. Defaults to \'celsius\'.'
                    }
                },
                'required': ['location', 'date']
            }
        }
    }]
    messages = [{
        'role': 'user',
        'content': 'Today is 2024-11-14, What\'s the temperature in San Francisco now? How about tomorrow?'
    }]

    response = client.chat.completions.create(model=model,
                                              messages=messages,
                                              temperature=0.8,
                                              top_p=0.8,
                                              stream=False,
                                              tools=tools)
    print(response)
    response_list = [response]
    func1_name = response.choices[0].message.tool_calls[0].function.name
    func1_args = response.choices[0].message.tool_calls[0].function.arguments
    func2_name = response.choices[0].message.tool_calls[1].function.name
    func2_args = response.choices[0].message.tool_calls[1].function.arguments
    with assume:
        assert response.choices[0].finish_reason == 'tool_calls'
        assert func1_name == 'get_current_temperature'
        assert func1_args == '{"location": "San Francisco, CA, USA"}' \
            or func1_args == '{"location": "San Francisco, California, USA", "unit": "celsius"}'
        assert func2_name == 'get_temperature_date'
        assert func2_args == '{"location": "San Francisco, CA, USA", "date": "2024-11-15"}' \
            or func2_args == '{"location": "San Francisco, California, USA", "date": "2024-11-15", "unit": "celsius"}'
        assert response.choices[0].message.tool_calls[0].type == 'function'

    messages.append(response.choices[0].message)

    for tool_call in response.choices[0].message.tool_calls:
        tool_call_args = json.loads(tool_call.function.arguments)
        tool_call_result = get_function_by_name(tool_call.function.name)(**tool_call_args)
        messages.append({
            'role': 'tool',
            'name': tool_call.function.name,
            'content': tool_call_result,
            'tool_call_id': tool_call.id
        })

    response = client.chat.completions.create(model=model,
                                              messages=messages,
                                              temperature=0.8,
                                              top_p=0.8,
                                              stream=False,
                                              tools=tools)
    print(response)
    response_list.append(response)
    with assume:
        assert response.choices[0].finish_reason == 'stop'
        assert '26.1' in response.choices[0].message.content

    return response_list


def run_tools_case(config, port: int = DEFAULT_PORT):
    http_url = BASE_HTTP_URL + ':' + str(port)
    log_path = config.get('log_path')

    model = get_model(http_url)

    if model is None:
        assert False, 'server not start correctly'

    restful_log = os.path.join(log_path, 'restful_reasoning_' + model + str(port) + '.log')
    file = open(restful_log, 'w')

    client = OpenAI(api_key='YOUR_API_KEY', base_url=http_url + '/v1')
    model_name = client.models.list().data[0].id

    with allure.step('step1 - one_round_prompt'):
        tools = [{
            'type': 'function',
            'function': {
                'name': 'get_current_weather',
                'description': 'Get the current weather in a given location',
                'parameters': {
                    'type': 'object',
                    'properties': {
                        'location': {
                            'type': 'string',
                            'description': 'The city and state, e.g. San Francisco, CA',
                        },
                        'unit': {
                            'type': 'string',
                            'enum': ['celsius', 'fahrenheit']
                        },
                    },
                    'required': ['location'],
                },
            }
        }]
        messages = [{'role': 'user', 'content': 'What\'s the weather like in Boston today?'}]
        response = client.chat.completions.create(model=model_name,
                                                  messages=messages,
                                                  temperature=0.01,
                                                  stream=False,
                                                  tools=tools)
        print(response)
        with assume:
            assert response.choices[0].finish_reason == 'tool_calls'
        with assume:
            assert response.choices[0].message.tool_calls[0].function.name == 'get_current_weather'
        with assume:
            assert 'Boston' in response.choices[0].message.tool_calls[0].function.arguments
        with assume:
            assert response.choices[0].message.tool_calls[0].type == 'function'
        file.writelines(str(response) + '\n')

    with allure.step('step2 - search prompt'):
        tools = [{
            'type': 'function',
            'function': {
                'name': 'search',
                'description': 'BING search API',
                'parameters': {
                    'type': 'object',
                    'properties': {
                        'query': {
                            'type': 'string',
                            'description': 'list of search query strings'
                        }
                    },
                    'required': ['location']
                }
            }
        }]
        messages = [{'role': 'user', 'content': '搜索最近的人工智能发展趋势'}]
        response = client.chat.completions.create(model=model_name,
                                                  messages=messages,
                                                  temperature=0.01,
                                                  stream=False,
                                                  tools=tools)
        print(response)
        with assume:
            assert response.choices[0].finish_reason == 'tool_calls'
        with assume:
            assert response.choices[0].message.tool_calls[0].function.name == 'search'
        with assume:
            assert '人工智能' in response.choices[0].message.tool_calls[0].function.arguments
        with assume:
            assert response.choices[0].message.tool_calls[0].type == 'function'
        file.writelines(str(response) + '\n')

    with allure.step('step3 - multiple_round_prompt'):
        if 'intern' in model.lower():
            response_list = test_internlm_multiple_round_prompt(client, model_name)
        if 'qwen' in model.lower():
            response_list = test_qwen_multiple_round_prompt(client, model_name)

        file.writelines(str(response_list) + '\n')

    file.close()
    allure.attach.file(restful_log, attachment_type=allure.attachment_type.TEXT)


def proxy_health_check(url):
    """Check if proxy server is healthy."""
    try:
        # For proxy server, we check if it responds to the /v1/models endpoint
        import requests
        response = requests.get(f'{url}/v1/models', timeout=5)
        if response.status_code == 200:
            return True
        return False
    except Exception:
        return False


def start_proxy_server(config, worker_id):
    """Start the proxy server for testing with enhanced error handling and
    logging."""
    log_path = config.get('eval_log_path')
    if log_path is None:
        log_path = '/nvme/qa_test_models/evaluation_report'
    os.makedirs(log_path, exist_ok=True)

    timestamp = datetime.datetime.now().strftime('%Y%m%d_%H%M%S')
    proxy_log = os.path.join(log_path, f'proxy_server_{worker_id}_{timestamp}.log')

    worker_num = get_workerid(worker_id)
    if worker_num is None:
        port = PROXY_PORT
    else:
        port = PROXY_PORT + worker_num

    proxy_url = f'http://127.0.0.1:{port}'  # noqa: E231, E261
    try:
        response = requests.get(f'{proxy_url}/nodes/status', timeout=5)
        if response.status_code == 200:
            print(f'Terminating existing nodes on proxy {proxy_url}')
            requests.get(f'{proxy_url}/nodes/terminate_all', timeout=10)
            sleep(5)
    except requests.exceptions.RequestException:
        pass

    cmd = (f'lmdeploy serve proxy --server-name 127.0.0.1 --server-port {port} '
           f'--routing-strategy min_expected_latency --serving-strategy Hybrid')

    print(f'Starting proxy server with command: {cmd}')
    print(f'Proxy log will be saved to: {proxy_log}')

    proxy_file = open(proxy_log, 'w')
    proxy_process = subprocess.Popen([cmd],
                                     stdout=proxy_file,
                                     stderr=proxy_file,
                                     shell=True,
                                     text=True,
                                     encoding='utf-8')
    pid = proxy_process.pid

    start_time = int(time())
    timeout = 300

    sleep(5)
    for i in range(timeout):
        sleep(1)
        if proxy_health_check(f'http://127.0.0.1:{port}'):  # noqa: E231, E261
            break

        try:
            # Check if process is still running
            return_code = proxy_process.wait(timeout=1)  # Small timeout to check status
            if return_code != 0:
                with open(proxy_log, 'r') as f:
                    content = f.read()
                    print(content)
                return 0, proxy_process
        except subprocess.TimeoutExpired:
            continue

        end_time = int(time())
        total_time = end_time - start_time
        if total_time >= timeout:
            break

    proxy_file.close()
    allure.attach.file(proxy_log, attachment_type=allure.attachment_type.TEXT)

    print(f'Proxy server started successfully with PID: {pid}')
    return pid, proxy_process
