| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708 | import osfrom abc import ABC, abstractmethodfrom typing import List, Dict, Any, Optional, Union, Tupleimport pandas as pdimport plotly.graph_objsfrom vanna.base import VannaBase# 导入配置参数from app_config import REWRITE_QUESTION_ENABLED, DISPLAY_RESULT_THINKINGclass BaseLLMChat(VannaBase, ABC):    """自定义LLM聊天基类,包含公共方法"""        def __init__(self, config=None):        VannaBase.__init__(self, config=config)        # 存储LLM解释性文本        self.last_llm_explanation = None                print("传入的 config 参数如下:")        for key, value in self.config.items():            print(f"  {key}: {value}")                # 默认参数        self.temperature = 0.7                if "temperature" in config:            print(f"temperature is changed to: {config['temperature']}")            self.temperature = config["temperature"]                # 加载错误SQL提示配置        self.enable_error_sql_prompt = self._load_error_sql_prompt_config()    def _load_error_sql_prompt_config(self) -> bool:        """从app_config.py加载错误SQL提示配置"""        try:            import app_config            enable_error_sql = getattr(app_config, 'ENABLE_ERROR_SQL_PROMPT', False)            print(f"[DEBUG] 错误SQL提示配置: ENABLE_ERROR_SQL_PROMPT = {enable_error_sql}")            return enable_error_sql        except (ImportError, AttributeError) as e:            print(f"[WARNING] 无法加载错误SQL提示配置: {e},使用默认值 False")            return False    def system_message(self, message: str) -> dict:        """创建系统消息格式"""        print(f"system_content: {message}")        return {"role": "system", "content": message}    def user_message(self, message: str) -> dict:        """创建用户消息格式"""        print(f"\nuser_content: {message}")        return {"role": "user", "content": message}    def assistant_message(self, message: str) -> dict:        """创建助手消息格式"""        print(f"assistant_content: {message}")        return {"role": "assistant", "content": message}    def get_sql_prompt(self, initial_prompt: str, question: str, question_sql_list: list, ddl_list: list, doc_list: list, **kwargs):        """        基于VannaBase源码实现,在第7点添加中文别名指令        """        print(f"[DEBUG] 开始生成SQL提示词,问题: {question}")                if initial_prompt is None:            initial_prompt = f"You are a {self.dialect} expert. " + \            "Please help to generate a SQL query to answer the question. Your response should ONLY be based on the given context and follow the response guidelines and format instructions."        # 提取DDL内容(适配新的字典格式)        ddl_content_list = []        if ddl_list:            for item in ddl_list:                if isinstance(item, dict) and "content" in item:                    ddl_content_list.append(item["content"])                elif isinstance(item, str):                    ddl_content_list.append(item)                initial_prompt = self.add_ddl_to_prompt(            initial_prompt, ddl_content_list, max_tokens=self.max_tokens        )        # 提取文档内容(适配新的字典格式)        doc_content_list = []        if doc_list:            for item in doc_list:                if isinstance(item, dict) and "content" in item:                    doc_content_list.append(item["content"])                elif isinstance(item, str):                    doc_content_list.append(item)                if self.static_documentation != "":            doc_content_list.append(self.static_documentation)        initial_prompt = self.add_documentation_to_prompt(            initial_prompt, doc_content_list, max_tokens=self.max_tokens        )        # 新增:添加错误SQL示例作为负面示例(放在Response Guidelines之前)        if self.enable_error_sql_prompt:            try:                error_sql_list = self.get_related_error_sql(question, **kwargs)                if error_sql_list:                    print(f"[DEBUG] 找到 {len(error_sql_list)} 个相关的错误SQL示例")                                        # 构建格式化的负面提示内容                    negative_prompt_content = "===Negative Examples\n"                    negative_prompt_content += "下面是错误的SQL示例,请分析这些错误SQL的问题所在,并在生成新SQL时避免类似错误:\n\n"                                        for i, error_example in enumerate(error_sql_list, 1):                        if "question" in error_example and "sql" in error_example:                            similarity = error_example.get('similarity', 'N/A')                            print(f"[DEBUG] 错误SQL示例 {i}: 相似度={similarity}")                            negative_prompt_content += f"问题: {error_example['question']}\n"                            negative_prompt_content += f"错误的SQL: {error_example['sql']}\n\n"                                        # 将负面提示添加到初始提示中                    initial_prompt += negative_prompt_content                else:                    print("[DEBUG] 未找到相关的错误SQL示例")            except Exception as e:                print(f"[WARNING] 获取错误SQL示例失败: {e}")        initial_prompt += (            "===Response Guidelines \n"            "1. If the provided context is sufficient, please generate a valid SQL query without any explanations for the question. \n"            "2. If the provided context is almost sufficient but requires knowledge of a specific string in a particular column, please generate an intermediate SQL query to find the distinct strings in that column. Prepend the query with a comment saying intermediate_sql \n"            "3. If the provided context is insufficient, please explain why it can't be generated. \n"            "4. Please use the most relevant table(s). \n"            "5. If the question has been asked and answered before, please repeat the answer exactly as it was given before. \n"            f"6. Ensure that the output SQL is {self.dialect}-compliant and executable, and free of syntax errors. \n"            "7. 在生成 SQL 查询时,如果出现 ORDER BY 子句,请遵循以下规则:\n"            "   - 对所有的排序字段(如聚合字段 SUM()、普通列等),请在 ORDER BY 中显式添加 NULLS LAST。\n"            "   - 不论是否使用 LIMIT,只要排序字段存在,都必须添加 NULLS LAST,以防止 NULL 排在结果顶部。\n"            "   - 示例参考:\n"            "     - ORDER BY total DESC NULLS LAST\n"            "     - ORDER BY zf_order DESC NULLS LAST\n"            "     - ORDER BY SUM(c.customer_count) DESC NULLS LAST \n"            "8. 【重要】请在SQL查询中为所有SELECT的列都使用中文别名:\n"            "   - 每个列都必须使用 AS 中文别名 的格式,没有例外\n"            "   - 包括原始字段名也要添加中文别名,例如:SELECT gender AS 性别, card_category AS 卡片类型\n"            "   - 计算字段也要有中文别名,例如:SELECT COUNT(*) AS 持卡人数\n"            "   - 中文别名要准确反映字段的业务含义"        )        message_log = [self.system_message(initial_prompt)]        for example in question_sql_list:            if example is None:                print("example is None")            else:                if example is not None and "question" in example and "sql" in example:                    message_log.append(self.user_message(example["question"]))                    message_log.append(self.assistant_message(example["sql"]))        message_log.append(self.user_message(question))                return message_log    def generate_plotly_code(self, question: str = None, sql: str = None, df_metadata: str = None, **kwargs) -> str:        """        重写父类方法,添加明确的中文图表指令        """        # 构建更智能的中文图表指令,根据问题和数据内容生成有意义的标签        chinese_chart_instructions = (            "使用中文创建图表,要求:\n"            "1. 根据用户问题和数据内容,为图表生成有意义的中文标题\n"            "2. 根据数据列的实际含义,为X轴和Y轴生成准确的中文标签\n"            "3. 如果有图例,确保图例标签使用中文\n"            "4. 所有文本(包括标题、轴标签、图例、数据标签等)都必须使用中文\n"            "5. 标题应该简洁明了地概括图表要展示的内容\n"            "6. 轴标签应该准确反映对应数据列的业务含义\n"            "7. 选择最适合数据特点的图表类型(柱状图、折线图、饼图等)"        )        # 构建父类方法要求的message_log        system_msg_parts = []        if question:            system_msg_parts.append(                f"用户问题:'{question}'"            )            system_msg_parts.append(                f"以下是回答用户问题的pandas DataFrame数据:"            )        else:            system_msg_parts.append("以下是一个pandas DataFrame数据:")        if sql:            system_msg_parts.append(f"数据来源SQL查询:\n{sql}")        system_msg_parts.append(f"DataFrame结构信息:\n{df_metadata}")        system_msg = "\n\n".join(system_msg_parts)        # 构建更详细的用户消息,强调中文标签的重要性        user_msg = (            "请为这个DataFrame生成Python Plotly可视化代码。要求:\n\n"            "1. 假设数据存储在名为'df'的pandas DataFrame中\n"            "2. 如果DataFrame只有一个值,使用Indicator图表\n"            "3. 只返回Python代码,不要任何解释\n"            "4. 代码必须可以直接运行\n\n"            f"{chinese_chart_instructions}\n\n"            "特别注意:\n"            "- 不要使用'图表标题'、'X轴标签'、'Y轴标签'这样的通用标签\n"            "- 要根据实际数据内容和用户问题生成具体、有意义的中文标签\n"            "- 例如:如果是性别统计,X轴可能是'性别',Y轴可能是'人数'或'占比'\n"            "- 标题应该概括图表的主要内容,如'男女持卡比例分布'\n\n"            "数据标签和悬停信息要求:\n"            "- 不要使用%{text}这样的占位符变量\n"            "- 使用具体的数据值和中文单位,例如:text=df['列名'].astype(str) + '人'\n"            "- 悬停信息要清晰易懂,使用中文描述\n"            "- 确保所有显示的文本都是实际的数据值,不是变量占位符"        )        message_log = [            self.system_message(system_msg),            self.user_message(user_msg),        ]        # 调用submit_prompt方法,并清理结果        plotly_code = self.submit_prompt(message_log, **kwargs)                # 根据 DISPLAY_RESULT_THINKING 参数处理thinking内容        if not DISPLAY_RESULT_THINKING:            original_code = plotly_code            plotly_code = self._remove_thinking_content(plotly_code)            print(f"[DEBUG] generate_plotly_code隐藏thinking内容 - 原始长度: {len(original_code)}, 处理后长度: {len(plotly_code)}")        return self._sanitize_plotly_code(self._extract_python_code(plotly_code))    def _extract_python_code(self, response: str) -> str:        """从LLM响应中提取Python代码"""        if not response:            return ""                # 查找代码块        import re                # 匹配 ```python 或 ``` 代码块        code_pattern = r'```(?:python)?\s*(.*?)```'        matches = re.findall(code_pattern, response, re.DOTALL)                if matches:            return matches[0].strip()                # 如果没有找到代码块,返回原始响应        return response.strip()    def _sanitize_plotly_code(self, code: str) -> str:        """清理和验证Plotly代码"""        if not code:            return ""                # 基本的代码清理        lines = code.split('\n')        cleaned_lines = []                for line in lines:            # 移除空行和注释行            line = line.strip()            if line and not line.startswith('#'):                cleaned_lines.append(line)                return '\n'.join(cleaned_lines)    def should_generate_chart(self, df) -> bool:        """        判断是否应该生成图表        对于Flask应用,这个方法决定了前端是否显示图表生成按钮        """        if df is None or df.empty:            print(f"[DEBUG] should_generate_chart: df为空,返回False")            return False                # 如果数据有多行或多列,通常适合生成图表        result = len(df) > 1 or len(df.columns) > 1        print(f"[DEBUG] should_generate_chart: df.shape={df.shape}, 返回{result}")                if result:            return True                return False    def generate_sql(self, question: str, **kwargs) -> str:        """        重写父类的 generate_sql 方法,增加异常处理和解释性文本保存        """        try:            # 清空上次的解释性文本            self.last_llm_explanation = None                        print(f"[DEBUG] 尝试为问题生成SQL: {question}")            # 调用父类的 generate_sql            sql = super().generate_sql(question, **kwargs)                        if not sql or sql.strip() == "":                print(f"[WARNING] 生成的SQL为空")                explanation = "无法生成SQL查询,可能是问题描述不够清晰或缺少必要的数据表信息。"                # 根据 DISPLAY_RESULT_THINKING 参数处理thinking内容                if not DISPLAY_RESULT_THINKING:                    explanation = self._remove_thinking_content(explanation)                self.last_llm_explanation = explanation                return None                        # 替换 "\_" 为 "_",解决特殊字符转义问题            sql = sql.replace("\\_", "_")                        # 检查返回内容是否为有效SQL或错误信息            sql_lower = sql.lower().strip()                        # 检查是否包含错误提示信息            error_indicators = [                "insufficient context", "无法生成", "sorry", "cannot", "不能",                "no relevant", "no suitable", "unable to", "无法", "抱歉",                "i don't have", "i cannot", "没有相关", "找不到", "不存在"            ]                        for indicator in error_indicators:                if indicator in sql_lower:                    print(f"[WARNING] LLM返回错误信息而非SQL: {sql}")                    # 保存LLM的解释性文本,并根据配置处理thinking内容                    explanation = sql                    if not DISPLAY_RESULT_THINKING:                        explanation = self._remove_thinking_content(explanation)                        print(f"[DEBUG] 隐藏thinking内容 - SQL生成解释性文本")                    self.last_llm_explanation = explanation                    return None                        # 简单检查是否像SQL语句(至少包含一些SQL关键词)            sql_keywords = ["select", "insert", "update", "delete", "with", "from", "where"]            if not any(keyword in sql_lower for keyword in sql_keywords):                print(f"[WARNING] 返回内容不像有效SQL: {sql}")                # 保存LLM的解释性文本,并根据配置处理thinking内容                explanation = sql                if not DISPLAY_RESULT_THINKING:                    explanation = self._remove_thinking_content(explanation)                    print(f"[DEBUG] 隐藏thinking内容 - SQL生成非有效SQL内容")                self.last_llm_explanation = explanation                return None                            print(f"[SUCCESS] 成功生成SQL:\n {sql}")            # 清空解释性文本            self.last_llm_explanation = None            return sql                    except Exception as e:            print(f"[ERROR] SQL生成过程中出现异常: {str(e)}")            print(f"[ERROR] 异常类型: {type(e).__name__}")            # 导入traceback以获取详细错误信息            import traceback            print(f"[ERROR] 详细错误信息: {traceback.format_exc()}")            explanation = f"SQL生成过程中出现异常: {str(e)}"            # 根据 DISPLAY_RESULT_THINKING 参数处理thinking内容            if not DISPLAY_RESULT_THINKING:                explanation = self._remove_thinking_content(explanation)            self.last_llm_explanation = explanation            return None    def generate_question(self, sql: str, **kwargs) -> str:        """根据SQL生成中文问题"""        prompt = [            self.system_message(                "请你根据下方SQL语句推测用户的业务提问,只返回清晰的自然语言问题,不要包含任何解释或SQL内容,也不要出现表名,问题要使用中文,并以问号结尾。"            ),            self.user_message(sql)        ]        response = self.submit_prompt(prompt, **kwargs)                # 根据 DISPLAY_RESULT_THINKING 参数处理thinking内容        if not DISPLAY_RESULT_THINKING:            original_response = response            response = self._remove_thinking_content(response)            print(f"[DEBUG] generate_question隐藏thinking内容 - 原始长度: {len(original_response)}, 处理后长度: {len(response)}")                return response    # def chat_with_llm(self, question: str, **kwargs) -> str:    #     """    #     直接与LLM对话,不涉及SQL生成    #     """    #     try:    #         prompt = [    #             self.system_message(    #                 "你是一个友好的AI助手。如果用户询问的是数据库相关问题,请建议他们重新表述问题以便进行SQL查询。对于其他问题,请尽力提供有帮助的回答。"    #             ),    #             self.user_message(question)    #         ]    #         response = self.submit_prompt(prompt, **kwargs)    #         return response    #     except Exception as e:    #         print(f"[ERROR] LLM对话失败: {str(e)}")    #         return f"抱歉,我暂时无法回答您的问题。请稍后再试。"    def chat_with_llm(self, question: str, system_prompt: str = None, **kwargs) -> str:        """        直接与LLM对话,不涉及SQL生成                Args:            question: 用户问题            system_prompt: 自定义系统提示词,如果为None则使用默认提示词            **kwargs: 其他传递给submit_prompt的参数                    Returns:            LLM的响应文本        """        try:            # 如果没有提供自定义系统提示词,使用默认的            if system_prompt is None:                system_prompt = (                    "你是一个友好的AI助手,请用中文回答用户的问题。"                )                        prompt = [                self.system_message(system_prompt),                self.user_message(question)            ]                        response = self.submit_prompt(prompt, **kwargs)                        # 根据 DISPLAY_RESULT_THINKING 参数处理thinking内容            if not DISPLAY_RESULT_THINKING:                original_response = response                response = self._remove_thinking_content(response)                print(f"[DEBUG] chat_with_llm隐藏thinking内容 - 原始长度: {len(original_response)}, 处理后长度: {len(response)}")                        return response                    except Exception as e:            print(f"[ERROR] LLM对话失败: {str(e)}")            return f"抱歉,我暂时无法回答您的问题。请稍后再试。"    def generate_rewritten_question(self, last_question: str, new_question: str, **kwargs) -> str:        """        重写问题合并方法,通过配置参数控制是否启用合并功能                Args:            last_question (str): 上一个问题            new_question (str): 新问题            **kwargs: 其他参数                    Returns:            str: 如果启用合并且问题相关则返回合并后的问题,否则返回新问题        """        # 如果未启用合并功能或没有上一个问题,直接返回新问题        if not REWRITE_QUESTION_ENABLED or last_question is None:            print(f"[DEBUG] 问题合并功能{'未启用' if not REWRITE_QUESTION_ENABLED else '上一个问题为空'},直接返回新问题")            return new_question                print(f"[DEBUG] 启用问题合并功能,尝试合并问题")        print(f"[DEBUG] 上一个问题: {last_question}")        print(f"[DEBUG] 新问题: {new_question}")                try:            prompt = [                self.system_message(                    "你的目标是将一系列相关的问题合并成一个单一的问题。如果第二个问题与第一个问题无关且完全独立,则返回第二个问题。"                    "只返回新的合并问题,不要添加任何额外的解释。该问题理论上应该能够用一个SQL语句来回答。"                    "请用中文回答。"                ),                self.user_message(f"第一个问题: {last_question}\n第二个问题: {new_question}")            ]                        rewritten_question = self.submit_prompt(prompt=prompt, **kwargs)                        # 根据 DISPLAY_RESULT_THINKING 参数处理thinking内容            if not DISPLAY_RESULT_THINKING:                original_question = rewritten_question                rewritten_question = self._remove_thinking_content(rewritten_question)                print(f"[DEBUG] generate_rewritten_question隐藏thinking内容 - 原始长度: {len(original_question)}, 处理后长度: {len(rewritten_question)}")                        print(f"[DEBUG] 合并后的问题: {rewritten_question}")            return rewritten_question                    except Exception as e:            print(f"[ERROR] 问题合并失败: {str(e)}")            # 如果合并失败,返回新问题            return new_question    def generate_summary(self, question: str, df, **kwargs) -> str:        """        覆盖父类的 generate_summary 方法,添加中文思考和回答指令                Args:            question (str): 用户提出的问题            df: 查询结果的 DataFrame            **kwargs: 其他参数                    Returns:            str: 数据摘要        """        try:            # 导入 pandas 用于 DataFrame 处理            import pandas as pd                        # 确保 df 是 pandas DataFrame            if not isinstance(df, pd.DataFrame):                print(f"[WARNING] df 不是 pandas DataFrame,类型: {type(df)}")                return "无法生成摘要:数据格式不正确"                        if df.empty:                return "查询结果为空,无数据可供摘要。"                        print(f"[DEBUG] 生成摘要 - 问题: {question}")            print(f"[DEBUG] DataFrame 形状: {df.shape}")                        # 构建包含中文指令的系统消息            system_content = (                f"你是一个专业的数据分析助手。用户提出了问题:'{question}'\n\n"                f"以下是查询结果的 pandas DataFrame 数据:\n{df.to_markdown()}\n\n"                "请用中文进行思考和分析,并用中文回答。"            )                        # 构建用户消息,强调中文思考和回答            user_content = (                "请基于用户提出的问题,简要总结这些数据。要求:\n"                             "1. 只进行简要总结,不要添加额外的解释\n"                "2. 如果数据中有数字,请保留适当的精度\n"                        )                        message_log = [                self.system_message(system_content),                self.user_message(user_content)            ]                        summary = self.submit_prompt(message_log, **kwargs)                        # 检查是否需要隐藏 thinking 内容            display_thinking = kwargs.get("display_result_thinking", DISPLAY_RESULT_THINKING)                        if not display_thinking:                # 移除 <think></think> 标签及其内容                original_summary = summary                summary = self._remove_thinking_content(summary)                print(f"[DEBUG] 隐藏thinking内容 - 原始长度: {len(original_summary)}, 处理后长度: {len(summary)}")                        print(f"[DEBUG] 生成的摘要: {summary[:100]}...")            return summary                    except Exception as e:            print(f"[ERROR] 生成摘要失败: {str(e)}")            import traceback            print(f"[ERROR] 详细错误信息: {traceback.format_exc()}")            return f"生成摘要时出现错误:{str(e)}"    def _remove_thinking_content(self, text: str) -> str:        """        移除文本中的 <think></think> 标签及其内容                Args:            text (str): 包含可能的 thinking 标签的文本                    Returns:            str: 移除 thinking 内容后的文本        """        if not text:            return text                import re                # 移除 <think>...</think> 标签及其内容(支持多行)        # 使用 re.DOTALL 标志使 . 匹配包括换行符在内的任何字符        cleaned_text = re.sub(r'<think>.*?</think>\s*', '', text, flags=re.DOTALL | re.IGNORECASE)                # 移除可能的多余空行        cleaned_text = re.sub(r'\n\s*\n\s*\n', '\n\n', cleaned_text)                # 去除开头和结尾的空白字符        cleaned_text = cleaned_text.strip()                return cleaned_text        def ask(        self,        question: Union[str, None] = None,        print_results: bool = True,        auto_train: bool = True,        visualize: bool = True,        allow_llm_to_see_data: bool = False,    ) -> Union[        Tuple[            Union[str, None],            Union[pd.DataFrame, None],            Union[plotly.graph_objs.Figure, None],        ],        None,    ]:        """        重载父类的ask方法,处理LLM解释性文本        当generate_sql无法生成SQL时,保存解释性文本供API层使用        """        if question is None:            question = input("Enter a question: ")        # 清空上次的解释性文本        self.last_llm_explanation = None        try:            sql = self.generate_sql(question=question, allow_llm_to_see_data=allow_llm_to_see_data)        except Exception as e:            print(e)            self.last_llm_explanation = str(e)            if print_results:                return None            else:                return None, None, None        # 如果SQL为空,说明有解释性文本,按照正常流程返回None        # API层会检查 last_llm_explanation 来获取解释        if sql is None:            print(f"[INFO] 无法生成SQL,解释: {self.last_llm_explanation}")            if print_results:                return None            else:                return None, None, None        # 以下是正常的SQL执行流程(保持VannaBase原有逻辑)        if print_results:            print(sql)        if self.run_sql_is_set is False:            print("If you want to run the SQL query, connect to a database first.")            if print_results:                return None            else:                return sql, None, None        try:            df = self.run_sql(sql)                        if df is None:                print("The SQL query returned no results.")                if print_results:                    return None                else:                    return sql, None, None            if print_results:                # 显示结果表格                if len(df) > 10:                    print(df.head(10).to_string())                    print(f"... ({len(df)} rows)")                else:                    print(df.to_string())            # 如果启用了自动训练,添加问题-SQL对到训练集            if auto_train:                try:                    self.add_question_sql(question=question, sql=sql)                except Exception as e:                    print(f"Could not add question and sql to training data: {e}")            if visualize:                try:                    # 检查是否应该生成图表                    if self.should_generate_chart(df):                        plotly_code = self.generate_plotly_code(                            question=question,                             sql=sql,                             df=df,                            chart_instructions=""                        )                        if plotly_code is not None and plotly_code.strip() != "":                            fig = self.get_plotly_figure(                                plotly_code=plotly_code,                                 df=df,                                 dark_mode=False                            )                            if fig is not None:                                if print_results:                                    print("Chart generated (use fig.show() to display)")                                return sql, df, fig                            else:                                print("Could not generate chart")                                return sql, df, None                        else:                            print("No chart generated")                            return sql, df, None                    else:                        print("Not generating chart for this data")                        return sql, df, None                except Exception as e:                    print(f"Couldn't generate chart: {e}")                    return sql, df, None            else:                return sql, df, None        except Exception as e:            print("Couldn't run sql: ", e)            if print_results:                return None            else:                return sql, None, None    @abstractmethod    def submit_prompt(self, prompt, **kwargs) -> str:        """        子类必须实现的核心提交方法                Args:            prompt: 消息列表            **kwargs: 其他参数                    Returns:            str: LLM的响应        """        pass 
 |