from environment_real_only_auxiliary import (
    Warning_from_too_many_auxiliary_exam,
    LLM_Caller_for_One_Thread,
)
from .interface import BaseDoctorAgent

##########################################################################################################################################
# 辅检阶段，需要详细的鉴别诊断分析
prompt_long_diagnosis = """你是一名医学专家，你的任务是根据提供的患者病历，推理出top-5鉴别诊断列表。

**核心要求：**
- **诊断排序**：诊断列表按可能性由高到低排列。
- **诊断完整性**：每个诊断都应是完整的，应具体明确（例如：使用“右下叶肺炎”而非“肺炎”；“冠心病不稳定型心绞痛”而非“心脏病”）；可包含主要疾病和相关的并发症/合并症（例如：2型糖尿病 合并 社区获得性肺炎）。
- **诊断竞争性**：列表中的各项诊断应该是相互竞争的备选方案（即鉴别诊断）。**不要将一个统一病理过程的不同方面拆分成独立的条目**（如将“社区获得性肺炎”和“发热”分别列为两个诊断）。
- **聚焦诊断**：你的回答应专注于诊断推理过程和最终的诊断列表。**严禁**提供任何治疗方案、用药建议或健康指导，也不要包含病人的检查结果等信息。
- **诊断个数**：允许鉴别诊断个数不足5个。


以下是病人的信息：
<病历>
{}
</病历>


输出格式：
逐步的分析...
<answer>
诊断1
诊断2
...
</answer>

输出示例：
...（逐步的分析）
<answer>
结核性脑膜炎/脑膜脑炎，伴有社区获得性肺炎
鼻窦旁脓肿，并发结核性全身感染
</answer>


现在请先给出逐步的分析，然后在<answer>标签内输出若干个相互竞争、完整的诊断方案，不要给出其他无内容。
"""

# 判断是继续检查，还是直接诊断
prompt_exam_or_diagnosis = """你是一名医学专家，负责根据 患者现有的病历信息 和 初步鉴别诊断列表，为临床医生建议下一步是否进行辅助检查，还是直接给出最终诊断。


直接出具最终诊断的条件（满足其一即可）：
   **唯一诊断且证据确凿**: 鉴别诊断列表中仅有一个疾病，该疾病能完全解释患者的核心症状，并且已有足够的证据支持（例如“金标准”检查结果）
   **诊断层级、因果清晰**: 列表中的一个诊断是根本病因（首要诊断），而其他诊断均可被明确解释为该诊断的**直接诱因、并发症或附属表现**，而非独立的竞争性诊断。同时，首要诊断已有足够的证据支持
   **证据优势原则**: 某一个诊断的证据链条完整、逻辑严密，与病历中的所有信息（包括症状、体征、已有检查）完美契合、无任何矛盾，并且已有足够证据支持；而列表中的其他诊断均存在无法解释当前病历信息的关键性疑点

注意，只有诊断结果有足够证据支持时，才可以给出最终诊断；否则，必须继续辅助检查。


如果选择给出最终诊断，还需要根据病历对诊断进行扩充，最终输出两句话(核心诊断+完整诊断)，包括（仅在病历包含相关信息时描述，**严禁编造**）：
*   **a. 病因:** 明确指出导致疾病的直接病原体，以及根本的、促成的危险因素（例如，生活习惯、既往病史等）
*   **b. 核心疾病:** 陈述核心诊断名称
*   **c. 主要并发症:** 列出由核心疾病直接引发的最重要的并发症和伴随疾病
*   **d. 关键病理生理状态:** 描述这些并发症导致的身体功能异常状态


输出格式要求：
- 最终结果只包含是否继续检查或最终的完整诊断：**严禁**提供任何治疗方案、用药建议或健康指导，也不要输出病人的检查结果等无矛盾信息
- 如果选择继续辅助检查，输出 `继续辅助检查`
- 如果选择给出最终诊断，输出核心诊断+完整诊断，形如 `您的诊断结果为：xxx(核心诊断)\n您的完整诊断如下：xxx(完整诊断)`
- 先给出逐步的分析，然后最终结果放在<answer>标签中

输出示例1
逐步的分析...
<answer>
继续辅助检查
</answer>

输出示例2
...（逐步的分析）
<answer>
您的诊断结果为：颅内静脉窦血栓
您的完整诊断如下：由终末期肝病（肝硬化失代偿期）继发的凝血功能障碍和高氨血症所引发的颅内静脉窦血栓；并继发颅内高压、脑缺血性损伤及肝性脑病。
</answer>



患者现有的病历信息：
<病历>
{}
</病历>


初步鉴别诊断列表：
<诊断>
{}
</诊断>


请你逐步分析，判断医生下一步应该继续辅助检查，还是给出最终诊断。
"""

# 基于病历+鉴别诊断，生成下一步辅助检查
prompt_recommend_auxiliary_exams_based_on_diagnosis_list = """你是一名医学专家，负责根据 患者现有的病历信息 和 初步鉴别诊断列表，为临床医生提供下一步的辅助检查建议。

### 决策核心原则
请遵循临床医学的“阶梯式检查”策略：
1. **必要性优先**：只推荐能显著改变临床决策或鉴别诊断方向的检查。
2. **不重复检查**：一般不要重复推荐已做过的检查项目。
3. **精准度与成本控制**：
   - 首选无创、低成本、高敏感度的筛查项目。
   - 昂贵或侵入性检查（如穿刺、PET-CT、基因检测）仅在低成本检查无法确诊，且对区分核心鉴别诊断至关重要时才推荐。
4. **名称具体化**：检查项目的名称需要足够具体，例如CT/MRI等检查项目必须注明部位及方式（如：头部CT平扫、腹部增强CT、冠状动脉CTA）。

### 检查目标
推荐的检查组合应致力于实现以下目标之一：
- **确诊**：获取“金标准”证据。
- **排除**：通过关键阴性结果，安全地排除高危的鉴别诊断。
- **鉴别**：有效区分列表中最相似的两个竞争性诊断。


### 输出格式要求
- 只输出推荐的检查项目：你的回答应专注于推荐的检查项目。**严禁**提供任何治疗方案、用药建议或健康指导，也不要输出病人的检查结果等无矛盾信息；不要输出分析过程
- 检查项目需要足够具体和确定，严禁使用‘建议考虑’、‘或者’等模糊表述。输出列表应为一份确定的、必须执行的临床医嘱清单。
- 每一行输出一个具体的检查项目，**禁止把多个项目放在同一行中**（例如，禁止形如“肝硬度检测或血清肝纤维化标志物检测”、“全血细胞计数+凝血功能+肝功能”等表述）
- 内容放在<answer>标签中


输出格式示例：在<answer>标签内第一行固定为“请求进行以下辅助检查：”
逐步的分析...
<answer>
请求进行以下辅助检查：
- 血清肝纤维化标志物检测
- 全血细胞计数
- 大便潜血试验
</answer>


患者现有的病历信息：
<病历>
{}
</病历>


初步鉴别诊断列表：
<诊断>
{}
</诊断>

请你根据病历和诊断，先给出逐步的分析，然后按格式在<answer>标签内推荐若干项辅助检查。
"""


class DoctorAgent_Diagnosis_Decision_Exam_0103(BaseDoctorAgent):
    """
    先鉴别诊断，再判断是否继续，最后推荐检查。
    """

    def _get_next_action(self, patient_answer: str, current_summary: str) -> str:
        # 根据病历总结，鉴别诊断
        diagnosis_str = self.LLM_caller.query_model_and_extract_label(
            model_str=self.model_name,
            prompt=prompt_long_diagnosis.format(current_summary),
            system_prompt=None,
            role="Doctor: long differential diagnosis",
            ensure_label="answer",
        )
        # 用于之后提取top-k
        self.diagnosis_str = diagnosis_str
        
        decision = self.LLM_caller.query_model_and_extract_label(
            model_str=self.model_name,
            prompt=prompt_exam_or_diagnosis.format(current_summary, diagnosis_str),
            system_prompt=None,
            role="Doctor: decision exam_or_diagnosis",
            ensure_label="answer",
        )
        if decision != "继续辅助检查":
            return decision

        if patient_answer == Warning_from_too_many_auxiliary_exam:
            return f"您的诊断结果为：{diagnosis_str.strip().splitlines()[0]}"

        # 根据病历总结+鉴别诊断,做检查
        answer = self.LLM_caller.query_model_and_extract_label(
            model_str=self.model_name,
            prompt=prompt_recommend_auxiliary_exams_based_on_diagnosis_list.format(
                current_summary, diagnosis_str
            ),
            system_prompt=None,
            role="Doctor: auxiliary exam",
            ensure_label="answer",
        )
        return answer

    def get_top3_diagnosis(self) -> list:
        diagnosis_list = self.diagnosis_str.strip().splitlines()[:3]
        if len(diagnosis_list) < 3:
            diagnosis_list += [None] * (3 - len(diagnosis_list))
        return diagnosis_list
