{
    "title": "Slack API Error Troubleshooting Guide",
    "turns": [{
        "turn": 1,
        "user": "from typing import Annotated, List, TypedDict, Optional\n\nfrom langgraph.graph.message import add\\_messages\n\nfrom langchain\\_community.document\\_transformers import LongContextReorder\n\nfrom langgraph.checkpoint.memory import MemorySaver\n\nfrom langchain\\_core.messages import AIMessage\n\nfrom langgraph.graph import StateGraph, START, END\n\nfrom langchain\\_core.runnables import RunnableConfig\n\nfrom dotenv import load\\_dotenv\n\nfrom langchain\\_openai import ChatOpenAI\n\nfrom mcp import ClientSession, StdioServerParameters\n\nfrom langchain\\_mcp\\_adapters.client import MultiServerMCPClient\n\nfrom openai import OpenAI\n\nimport asyncio\n\nimport json\n\nimport os\n\nimport re\n\nload\\_dotenv()\n\nclass ChatbotConfig:\n\n    \"\"\"Class to manage chatbot configuration\"\"\"\n\n    def \\_\\_init\\_\\_(self, config\\_file: str = \"chatbot\\_config.json\"):\n\n        self.config\\_file = config\\_file\n\n        self.config = self.load\\_config()\n\n    def load\\_config(self):\n\n        \"\"\"Load configuration file\"\"\"\n\n        default\\_config = {\n\n            \"model\": {\n\n                \"name\": \"gpt-4o\",\n\n                \"temperature\": 0.2\n\n            },\n\n            \"mcp\\_servers\": {\n\n                \"neo4j\\_retriever\": {\n\n                    \"command\": \"python\",  # Change to python on Windows\n\n                    \"args\": \\[\"mcp\\_neo4j\\_server.py\"\\],\n\n                    \"transport\": \"stdio\",\n\n                },\n\n                \"VectorDB\\_retriever\": {\n\n                    \"command\": \"python\",  # Change to python on Windows\n\n                    \"args\": \\[\"mcp\\_vectordb\\_server.py\"\\],\n\n                    \"transport\": \"stdio\",\n\n                }\n\n            },\n\n            \"slack\": {\n\n                \"enabled\": True,\n\n                \"token\": os.getenv(\"SLACK\\_BOT\\_TOKEN\"),\n\n                \"channel\": os.getenv(\"SLACK\\_CHANNEL\"),\n\n                \"user\\_mapping\": {\n\n                    \"YourName\": \"U093ELJBE3X\",  # Needs to be changed to actual Slack user ID\n\n                    #\"Name1\": \"U0987654321\",\n\n                    #\"Name2\": \"U1122334455\"\n\n                }\n\n            },\n\n            \"use\\_dummy\\_tools\": False,  # Use actual MCP server\n\n            \"memory\\_enabled\": True\n\n        }\n\n        if os.path.exists(self.config\\_file):\n\n            with open(self.config\\_file, 'r', encoding='utf-8') as f:\n\n                return {\\*\\*default\\_config, \\*\\*json.load(f)}\n\n        else:\n\n            # Create default config file\n\n            with open(self.config\\_file, 'w', encoding='utf-8') as f:\n\n                json.dump(default\\_config, f, indent=2, ensure\\_ascii=False)\n\n            return default\\_config\n\n# Load configuration\n\nconfig = ChatbotConfig()\n\n#Create client based on OpenAI API key\n\nmodel\\_client = OpenAI()\n\n#Define chatbot state\n\n#State object for use in LangGraph\n\n#Define data structure for data exchanged during chatbot flow\n\nclass ChatbotState(TypedDict):\n\n    question: Annotated\\[str, \"Question\"\\]   #User question\n\n    tools: Annotated\\[List, \"Tools\"\\] #Tools to use\n\n    neo4j\\_documents: Annotated\\[List, \"Neo4j\\_Documents\"\\] #   Results from Neo4j retriever\n\n    vector\\_documents : Annotated\\[List,\"Vector\\_Documents\"\\] #Results from VectorDB retriever\n\n    final\\_answer: Annotated\\[str, \"Final\\_Answer\"\\] #Final answer\n\n    messages: Annotated\\[List, add\\_messages\\] #Full conversation message log\n\n    slack\\_recipient: Annotated\\[Optional\\[str\\], \"Slack\\_Recipient\"\\] #Slack recipient\n\n    slack\\_message: Annotated\\[Optional\\[str\\], \"Slack\\_Message\"\\] #Slack message\n\n#Define memory storage\n\nmemory = MemorySaver() if config.config\\[\"memory\\_enabled\"\\] else None\n\n# Slack integration class\n\nclass SlackNotifier:\n\n    \"\"\"Class for sending Slack messages\"\"\"\n\n    def \\_\\_init\\_\\_(self, token: str, channel: str, user\\_mapping: dict):\n\n        self.token = token\n\n        self.channel = channel\n\n        self.user\\_mapping = user\\_mapping\n\n        self.enabled = bool(token)\n\n    async def send\\_message(self, recipient: str, message: str) -\\> str:\n\n        \"\"\"Send message via Slack\"\"\"\n\n        if not self.enabled:\n\n            return f\"[Slack Disabled] Message to {recipient}: {message}\"\n\n        try:\n\n            import requests\n\n            # Find Slack user ID\n\n            user\\_id = self.user\\_mapping.get(recipient)\n\n            if not user\\_id:\n\n                return f\"Error: Could not find user '{recipient}'. (Mapped users: {list(self.user\\_mapping.keys())})\"\n\n            print(f\"Calling Slack API...\")\n\n            print(f\"Token: {self.token\\[:10\\]}...\" if self.token else \"No token\")\n\n            print(f\"User ID: {user\\_id}\")\n\n            print(f\"Message: {message\\[:50\\]}...\")\n\n            # Method 1: Send message to channel as mention (simplest)\n\n            mention\\_message = f\"\\<@{user\\_id}\\> {message}\"\n\n            response = requests.post(\n\n                \"https://slack.com/api/chat.postMessage\",\n\n                headers={\"Authorization\": f\"Bearer {self.token}\"},\n\n                json={\n\n                    \"channel\": self.channel,  # Send to general channel\n\n                    \"text\": mention\\_message,\n\n                    \"username\": \"ThankYouBot\"\n\n                }\n\n            )\n\n            print(f\"Slack API response: {response.status\\_code}\")\n\n            print(f\"Response content: {response.text}\")\n\n            if response.status\\_code == 200:\n\n                result = response.json()\n\n                if result.get(\"ok\"):\n\n                    return f\"✅ Slack message sent!\\\\nRecipient: {recipient} (Channel: {self.channel})\\\\nMessage: {message\\[:100\\]}...\"\n\n                else:\n\n                    error\\_msg = result.get('error', 'Unknown error')\n\n                    if error\\_msg == \"missing\\_scope\":\n\n                        return f\"❌ Slack permission denied: Bot Token needs 'chat:write' permission.\"\n\n                    elif error\\_msg == \"channel\\_not\\_found\":\n\n                        return f\"❌ Channel not found: Channel '{self.channel}' does not exist.\"\n\n                    else:\n\n                        return f\"❌ Slack API error: {error\\_msg}\"\n\n            else:\n\n                return f\"❌ HTTP error: {response.status\\_code} - {response.text}\"\n\n        except Exception as e:\n\n            return f\"Failed to send Slack message: {e}\"\n\n# Initialize Slack notifier\n\nslack\\_notifier = SlackNotifier(\n\n    token=config.config\\[\"slack\"\\]\\[\"token\"\\],\n\n    channel=config.config\\[\"slack\"\\]\\[\"channel\"\\],\n\n    user\\_mapping=config.config\\[\"slack\"\\]\\[\"user\\_mapping\"\\]\n\n)\n\n# MCP client setup\n\nasync def setup\\_mcp\\_client():\n\n    \"\"\"Set up MCP client and get tools\"\"\"\n\n    try:\n\n        # Create MCP client\n\n        client = MultiServerMCPClient(\n\n            {\n\n                \"neo4j\\_retriever\": {\n\n                    \"command\": config.config\\[\"mcp\\_servers\"\\]\\[\"neo4j\\_retriever\"\\]\\[\"command\"\\],\n\n                    \"args\": config.config\\[\"mcp\\_servers\"\\]\\[\"neo4j\\_retriever\"\\]\\[\"args\"\\],\n\n                    \"transport\": config.config\\[\"mcp\\_servers\"\\]\\[\"neo4j\\_retriever\"\\]\\[\"transport\"\\],\n\n                },\n\n                \"VectorDB\\_retriever\": {\n\n                    \"command\": config.config\\[\"mcp\\_servers\"\\]\\[\"VectorDB\\_retriever\"\\]\\[\"command\"\\],\n\n                    \"args\": config.config\\[\"mcp\\_servers\"\\]\\[\"VectorDB\\_retriever\"\\]\\[\"args\"\\],\n\n                    \"transport\": config.config\\[\"mcp\\_servers\"\\]\\[\"VectorDB\\_retriever\"\\]\\[\"transport\"\\],\n\n                }\n\n            }\n\n        )\n\n        # Get tool info from MCP\n\n        tools = await client.get\\_tools()\n\n        # Map tool name to object\n\n        tools\\_dict = {tool.name: tool for tool in tools}\n\n        tools\\_dict\\[\"slack\\_sender\"\\] = SlackTool(slack\\_notifier)\n\n        print(f\"MCP client setup complete. Available tools: {list(tools\\_dict.keys())}\")\n\n        return client, tools, tools\\_dict\n\n    except Exception as e:\n\n        print(f\"Failed to set up MCP client: {e}\")\n\n        print(\"Switching to dummy tools.\")\n\n        return None, \\[\\], {}\n\n# Tool definitions - direct definition instead of MCP (for backup)\n\nbackup\\_tools = \\[\n\n    {\n\n        \"type\": \"function\",\n\n        \"function\": {\n\n            \"name\": \"neo4j\\_retriever\",\n\n            \"description\": \"Searches for patient-related information in the Neo4j database.\",\n\n            \"parameters\": {\n\n                \"type\": \"object\",\n\n                \"properties\": {\n\n                    \"query\": {\n\n                        \"type\": \"string\",\n\n                        \"description\": \"Question to search\"\n\n                    }\n\n                },\n\n                \"required\": \\[\"query\"\\]\n\n            }\n\n        }\n\n    },\n\n    {\n\n        \"type\": \"function\",\n\n        \"function\": {\n\n            \"name\": \"VectorDB\\_retriever\",\n\n            \"description\": \"Searches for medical knowledge in VectorDB.\",\n\n            \"parameters\": {\n\n                \"type\": \"object\",\n\n                \"properties\": {\n\n                    \"query\": {\n\n                        \"type\": \"string\",\n\n                        \"description\": \"Question to search\"\n\n                    }\n\n                },\n\n                \"required\": \\[\"query\"\\]\n\n            }\n\n        }\n\n    }\n\n\\]\n\n# Map tool name to object\n\nclass DummyTool:\n\n    \"\"\"Dummy tool - used until actual server connection\"\"\"\n\n    def \\_\\_init\\_\\_(self, name):\n\n        self.name = name\n\n    async def ainvoke(self, params):\n\n        return f\"Dummy result for {self.name} with query: {params.get('query', '')}\"\n\nclass SlackTool:\n\n    \"\"\"Slack message sending tool\"\"\"\n\n    def \\_\\_init\\_\\_(self, notifier: SlackNotifier):\n\n        self.notifier = notifier\n\n    async def ainvoke(self, params):\n\n        recipient = params.get(\"recipient\", \"\")\n\n        message = params.get(\"message\", \"\")\n\n        return await self.notifier.send\\_message(recipient, message)\n\n# Initialize global variables\n\nmcp\\_client = None\n\ntools = backup\\_tools\n\ntools\\_dict = {\n\n    \"neo4j\\_retriever\": DummyTool(\"neo4j\\_retriever\"),\n\n    \"VectorDB\\_retriever\": DummyTool(\"VectorDB\\_retriever\"),\n\n    \"slack\\_sender\": SlackTool(slack\\_notifier)\n\n}\n\nasync def initialize\\_mcp():\n\n    \"\"\"Initialize MCP\"\"\"\n\n    global mcp\\_client, tools, tools\\_dict\n\n    if not config.config\\[\"use\\_dummy\\_tools\"\\]:\n\n        mcp\\_client, mcp\\_tools, mcp\\_tools\\_dict = await setup\\_mcp\\_client()\n\n        if mcp\\_client and mcp\\_tools:\n\n            tools = mcp\\_tools\n\n            tools\\_dict = mcp\\_tools\\_dict\n\n            # Add Slack tool\n\n            tools\\_dict\\[\"slack\\_sender\"\\] = SlackTool(slack\\_notifier)\n\n            print(\"Using MCP tools\")\n\n        else:\n\n            print(\"MCP connection failed, using dummy tools\")\n\n            # Initialize with dummy tools\n\n            tools = backup\\_tools\n\n            tools\\_dict = {\n\n                \"neo4j\\_retriever\": DummyTool(\"neo4j\\_retriever\"),\n\n                \"VectorDB\\_retriever\": DummyTool(\"VectorDB\\_retriever\"),\n\n                \"slack\\_sender\": SlackTool(slack\\_notifier)\n\n            }\n\n    else:\n\n        print(\"Using dummy tools as per configuration\")\n\n        # Initialize with dummy tools\n\n        tools = backup\\_tools\n\n        tools\\_dict = {\n\n            \"neo4j\\_retriever\": DummyTool(\"neo4j\\_retriever\"),\n\n            \"VectorDB\\_retriever\": DummyTool(\"VectorDB\\_retriever\"),\n\n            \"slack\\_sender\": SlackTool(slack\\_notifier)\n\n        }\n\n    print(\"Available tools:\", list(tools\\_dict.keys()))\n\n    print(\"Using dummy tools:\", config.config\\[\"use\\_dummy\\_tools\"\\])\n\n#Prompt input.\n\nLLM\\_SYSTEM\\_PROMPTY = \"\"\"\n\n# INSTRUCTION\n\nYou are an expert AI specialized in medical data.\n\nWhen answering the user's question, refer to information from the following two sources:\n\n1. 🔎 Neo4j search results: Structured patient-related information (e.g., surgery history, test records, etc.)\n\n2. 📚 VectorDB search results: General medical knowledge (e.g., symptom descriptions, treatment guidelines, etc.)\n\n- If both results exist, integrate and distinguish each source, summarizing or merging duplicate content.\n- If only one result exists, answer based solely on that result, but do not mention the limitation of information and answer as sincerely as possible.\n- Even if the results are sparse or ambiguous, always try to provide a meaningful explanation.\n- Without unnecessary introductions, answer the question directly.\n\n# Neo4j CONTEXT\n\n{Neo4j}\n\n# Vector DB CONTEXT\n\n{VectorDB}\n\n# Question\n\n{question}\n\n\"\"\"\n\nmodel = ChatOpenAI(\n\n    temperature=config.config\\[\"model\"\\]\\[\"temperature\"\\],\n\n    model\\_name=config.config\\[\"model\"\\]\\[\"name\"\\]\n\n)\n\ndef extract\\_slack\\_info(question: str) -\\> tuple:\n\n    \"\"\"Extract Slack sending info from question\"\"\"\n\n    # Simple pattern: If both \"~to\" and \"deliver\" are present, send via Slack\n\n    recipient\\_pattern = r'(.+?)to'\n\n    delivery\\_pattern = r'deliver'\n\n    # Check if both recipient and delivery keywords are present\n\n    recipient\\_match = re.search(recipient\\_pattern, question)\n\n    delivery\\_match = re.search(delivery\\_pattern, question)\n\n    if recipient\\_match and delivery\\_match:\n\n        recipient = recipient\\_match.group(1).strip()\n\n        # Remove \"~to\" and \"deliver\" to extract medical question\n\n        # \"To YourName, search why anesthesia should be used carefully for 3-year-olds and deliver it\"\n\n        # → \"Search why anesthesia should be used carefully for 3-year-olds\"\n\n        clean\\_question = question.replace(f\"{recipient}to\", \"\").replace(\"deliver\", \"\").strip()\n\n        # Remove leading/trailing spaces and unnecessary particles\n\n        clean\\_question = re.sub(r'^\\\\s\\*\\[thisthat\\]\\\\s\\*', '', clean\\_question)  # Remove leading \"this/that\"\n\n        clean\\_question = re.sub(r'\\\\s\\*\\[을that\\]\\\\s\\*\\$', '', clean\\_question)  # Remove trailing \"this/that\"\n\n        clean\\_question = clean\\_question.strip()\n\n        print(f\"🔍 Slack send detected: recipient={recipient}, question={clean\\_question}\")\n\n        return recipient, clean\\_question\n\n    return None, question\n\nasync def decision\\_tools(state: ChatbotState):\n\n    question = state\\[\"question\"\\]\n\n    # Extract Slack sending info\n\n    recipient, clean\\_question = extract\\_slack\\_info(question)\n\n    # Tool selection logic\n\n    selected\\_tools = \\[\\]\n\n    # 1. If Slack send request exists, add slack\\_sender\n\n    if recipient:\n\n        selected\\_tools.append(\"slack\\_sender\")\n\n        print(f\"🔍 Slack send request detected: send to {recipient}\")\n\n        # Update original question to cleaned question\n\n        question = clean\\_question\n\n    # 2. Tool selection for medical question\n\n    if clean\\_question and clean\\_question.strip():\n\n        if config.config\\[\"use\\_dummy\\_tools\"\\]:\n\n            # When using dummy tools: select based on keywords\n\n            medical\\_keywords = \\[\n\n                'symptom', 'treatment', 'medication', 'surgery', 'test', 'diagnosis', 'disease', 'disorder', 'anesthesia',\\n\\n                'heart', 'brain', 'lung', 'liver', 'kidney', 'blood', 'infection', 'inflammation', 'pain',\\n\\n                'pediatric', 'adult', 'elderly', 'emergency', 'ICU', 'rehabilitation', 'prevention'\\n\\n            ]\\n\\n            patient_keywords = [\\n\\n                'patient', 'patient_info', 'patient_record', 'patient_history', 'patient_data', 'patient_condition',\\n\\n                'surgical_history', 'test_record', 'medical_record', 'hospitalization', 'discharge', 'prescribed_medication'\n\n            \\]\n\n            has\\_medical\\_content = any(keyword in clean\\_question for keyword in medical\\_keywords)\n\n            has\\_patient\\_content = any(keyword in clean\\_question for keyword in patient\\_keywords)\n\n            if has\\_medical\\_content:\n\n                selected\\_tools.append(\"VectorDB\\_retriever\")\n\n                print(f\"🔍 Medical question detected: VectorDB search\")\n\n                if has\\_patient\\_content:\n\n                    selected\\_tools.append(\"neo4j\\_retriever\")\n\n                    print(f\"🔍 Patient info detected: Add Neo4j search\")\n\n            else:\n\n                selected\\_tools.append(\"VectorDB\\_retriever\")\n\n                print(f\"🔍 General question detected: VectorDB search\")\n\n        else:\n\n            # When using MCP: LLM automatically selects tools\n\n            input\\_messages = \\[\n\n                {\"role\": \"system\", \"content\": \"Decide which tools to use to answer the user's question. You may call one or both.\"},\n\n                {\"role\": \"user\", \"content\": clean\\_question}\n\n            \\]\n\n            response = model\\_client.chat.completions.create(\n\n                model=config.config\\[\"model\"\\]\\[\"name\"\\],\n\n                messages=input\\_messages,\n\n                tools=tools\n\n            )\n\n            if response.choices\\[0\\].message.tool\\_calls:\n\n                for tool\\_call in response.choices\\[0\\].message.tool\\_calls:\n\n                    tool\\_name = tool\\_call.function.name\n\n                    if tool\\_name not in selected\\_tools:\n\n                        selected\\_tools.append(tool\\_name)\n\n                        print(f\"🤖 LLM automatically selected tool: {tool\\_name}\")\n\n    # 3. If no tool is selected, select VectorDB by default\n\n    if not selected\\_tools or (len(selected\\_tools) == 1 and selected\\_tools\\[0\\] == \"slack\\_sender\"):\n\n        selected\\_tools.append(\"VectorDB\\_retriever\")\n\n        print(f\"🔍 Default tool selected: VectorDB search\")\n\n    print(f\"📋 Selected tools: {selected\\_tools}\")\n\n    return ChatbotState(\n\n        tools=selected\\_tools,\n\n        slack\\_recipient=recipient,\n\n        question=clean\\_question if recipient else question\n\n    )\n\nasync def vector\\_db(state: ChatbotState):\n\n    question = state\\[\"question\"\\]\n\n    neo4j\\_documents = state.get(\"neo4j\\_documents\", \\[\\])\n\n    if \"VectorDB\\_retriever\" in state\\[\"tools\"\\]:\n\n        vectordb\\_tool = tools\\_dict.get(\"VectorDB\\_retriever\")\n\n        if vectordb\\_tool:\n\n            try:\n\n                # If patient info from Neo4j exists, include in search query\n\n                if neo4j\\_documents and neo4j\\_documents != \"Neo4j tool not found.\":\n\n                    enhanced\\_query = f\"Patient info: {neo4j\\_documents}\\\\n\\\\nQuestion: {question}\"\n\n                    print(f\"🔍 VectorDB search (with patient info): {enhanced\\_query\\[:100\\]}...\")\n\n                else:\n\n                    enhanced\\_query = question\n\n                    print(f\"🔍 VectorDB search: {question}\")\n\n                result = await vectordb\\_tool.ainvoke({\"query\": enhanced\\_query})\n\n            except Exception as e:\n\n                result = f\"VectorDB tool execution error: {e}\"\n\n        else:\n\n            result = \"VectorDB tool not found.\"\n\n    else:\n\n        result = None\n\n    return ChatbotState(vector\\_documents=result)\n\nasync def neo4j\\_db(state: ChatbotState):\n\n    question = state\\[\"question\"\\]\n\n    if \"neo4j\\_retriever\" in state\\[\"tools\"\\]:\n\n        neo4j\\_tool = tools\\_dict.get(\"neo4j\\_retriever\")\n\n        if neo4j\\_tool:\n\n            try:\n\n                result = await neo4j\\_tool.ainvoke({\"query\": question})\n\n            except Exception as e:\n\n                result = f\"Neo4j tool execution error: {e}\"\n\n        else:\n\n            result = \"Neo4j tool not found.\"\n\n    else:\n\n        result = None\n\n    return ChatbotState(neo4j\\_documents=result)\n\nasync def slack\\_send(state: ChatbotState):\n\n    \"\"\"Send Slack message\"\"\"\n\n    recipient = state.get(\"slack\\_recipient\")\n\n    final\\_answer = state.get(\"final\\_answer\", \"\")\n\n    if recipient and final\\_answer:\n\n        slack\\_tool = tools\\_dict.get(\"slack\\_sender\")\n\n        if slack\\_tool:\n\n            try:\n\n                result = await slack\\_tool.ainvoke({\n\n                    \"recipient\": recipient,\n\n                    \"message\": final\\_answer\n\n                })\n\n            except Exception as e:\n\n                result = f\"Slack send error: {e}\"\n\n        else:\n\n            result = \"Slack tool not found.\"\n\n    else:\n\n        result = None\n\n    return ChatbotState(slack\\_message=result)\n\nasync def merge\\_outputs(state:ChatbotState):\n\n    question = state\\['question'\\]\n\n    vector\\_documents = state\\['vector\\_documents'\\]\n\n    neo4j\\_documents = state\\['neo4j\\_documents'\\]\n\n    formatted = LLM\\_SYSTEM\\_PROMPTY.format(Neo4j = neo4j\\_documents,\n\n                                          VectorDB = vector\\_documents,\n\n                                          question = question)\n\n    response = model.invoke(formatted)\n\n    response\\_text = response.content if isinstance(response, AIMessage) else str(response)\n\n    return ChatbotState(final\\_answer=response\\_text,\n\n                        messages=\\[(\"user\", question), (\"assistant\", response\\_text)\\])\n\nclass MedicalChatbot:\n\n    \"\"\"Medical chatbot class\"\"\"\n\n    def \\_\\_init\\_\\_(self):\n\n        self.builder = StateGraph(ChatbotState)\n\n        self.setup\\_graph()\n\n        self.graph = self.builder.compile(checkpointer=memory)\n\n        self.config = RunnableConfig(configurable={\"thread\\_id\": 1})\n\n    def setup\\_graph(self):\n\n        \"\"\"Set up graph\"\"\"\n\n        self.builder.add\\_node(\"decision\\_tools\", decision\\_tools)\n\n        self.builder.add\\_node(\"vector\\_db\", vector\\_db)\n\n        self.builder.add\\_node(\"neo4j\\_db\", neo4j\\_db)\n\n        self.builder.add\\_node(\"merge\\_outputs\", merge\\_outputs)\n\n        self.builder.add\\_node(\"slack\\_send\", slack\\_send)\n\n        # Graph start point -> connect to first node\n\n        self.builder.add\\_edge(START, \"decision\\_tools\")\n\n        # Conditional branch: decide next node based on selected tools\n\n        def route\\_tools(state: ChatbotState):\n\n            tools = state.get(\"tools\", \\[\\])\n\n            # If patient info exists, run Neo4j first\n\n            if \"neo4j\\_retriever\" in tools:\n\n                return \"neo4j\\_db\"\n\n            elif \"VectorDB\\_retriever\" in tools:\n\n                return \"vector\\_db\"\n\n            else:\n\n                return \"merge\\_outputs\"  # If no tools, generate answer immediately\n\n        self.builder.add\\_conditional\\_edges(\n\n            \"decision\\_tools\",\n\n            route\\_tools,\n\n            {\n\n                \"neo4j\\_db\": \"neo4j\\_db\",\n\n                \"vector\\_db\": \"vector\\_db\",\n\n                \"merge\\_outputs\": \"merge\\_outputs\"\n\n            }\n\n        )\n\n        # After Neo4j -> go to VectorDB (search based on patient info)\n\n        def route\\_after\\_neo4j(state: ChatbotState):\n\n            if \"VectorDB\\_retriever\" in state.get(\"tools\", \\[\\]):\n\n                return \"vector\\_db\"\n\n            else:\n\n                return \"merge\\_outputs\"\n\n        self.builder.add\\_conditional\\_edges(\n\n            \"neo4j\\_db\",\n\n            route\\_after\\_neo4j,\n\n            {\n\n                \"vector\\_db\": \"vector\\_db\",\n\n                \"merge\\_outputs\": \"merge\\_outputs\"\n\n            }\n\n        )\n\n        # After VectorDB -> go to merge\\_outputs\n\n        self.builder.add\\_edge(\"vector\\_db\", \"merge\\_outputs\")\n\n        # After merge\\_outputs -> slack\\_send (conditional)\n\n        def route\\_after\\_merge(state: ChatbotState):\n\n            if \"slack\\_sender\" in state.get(\"tools\", \\[\\]):\n\n                return \"slack\\_send\"\n\n            return END\n\n        self.builder.add\\_conditional\\_edges(\n\n            \"merge\\_outputs\",\n\n            route\\_after\\_merge,\n\n            {\n\n                \"slack\\_send\": \"slack\\_send\",\n\n                END: END\n\n            }\n\n        )\n\n        # slack\\_send → END\n\n        self.builder.add\\_edge(\"slack\\_send\", END)\n\n    def create\\_initial\\_state(self, question: str):\n\n        \"\"\"Create initial state\"\"\"\n\n        return {\n\n            \"question\": question,\n\n            \"tools\": \\[\\],\n\n            \"neo4j\\_documents\": \\[\\],\n\n            \"vector\\_documents\": \\[\\],\n\n            \"final\\_answer\": \"\",\n\n            \"messages\": \\[\\],\n\n            \"slack\\_recipient\": None,\n\n            \"slack\\_message\": None,\n\n        }\n\n    async def ask(self, question: str, verbose: bool = False):\n\n        \"\"\"Answer question\"\"\"\n\n        initial\\_state = self.create\\_initial\\_state(question)\n\n        if verbose:\n\n            print(f\"Question: {question}\")\n\n            print(\"Processing...\")\n\n        # Method 1: Use invoke to get final state directly\n\n        try:\n\n            final\\_state = await self.graph.ainvoke(initial\\_state, config=self.config)\n\n            final\\_answer = final\\_state.get(\"final\\_answer\", \"\")\n\n            slack\\_message = final\\_state.get(\"slack\\_message\", \"\")\n\n            if verbose:\n\n                print(f\"Final state: {final\\_state}\")\n\n            # If Slack message exists, return together\n\n            if slack\\_message:\n\n                return f\"{final\\_answer}\\\\n\\\\n📱 Slack send result:\\\\n{slack\\_message}\"\n\n            return final\\_answer\n\n        except Exception as e:\n\n            if verbose:\n\n                print(f\"invoke failed, trying astream: {e}\")\n\n            # Method 2: Backup using astream\n\n            final\\_answer = None\n\n            slack\\_message = None\n\n            async for event in self.graph.astream(initial\\_state, config=self.config):\n\n                if verbose:\n\n                    print(f\"Event: {event}\")\n\n                if \"final\\_answer\" in event and event\\[\"final\\_answer\"\\]:\n\n                    final\\_answer = event\\[\"final\\_answer\"\\]\n\n                if \"slack\\_message\" in event and event\\[\"slack\\_message\"\\]:\n\n                    slack\\_message = event\\[\"slack\\_message\"\\]\n\n            if slack\\_message:\n\n                return f\"{final\\_answer}\\\\n\\\\n📱 Slack send result:\\\\n{slack\\_message}\"\n\n            return final\\_answer\n\nasync def interactive\\_chat():\n\n    \"\"\"Interactive chat interface\"\"\"\n\n    # Initialize MCP\n\n    await initialize\\_mcp()\n\n    chatbot = MedicalChatbot()\n\n    print(\"Welcome to the medical chatbot!\")\n\n    print(\"Enter your question (type 'quit' or 'exit' to end)\")\n\n    print(\"Slack send: Enter in the format 'Send this to [someone]'\")\n\n    print(\"-\" \\* 50)\n\n    while True:\n\n        try:\n\n            question = input(\"\\\\nQuestion: \").strip()\n\n            if question.lower() in \\['quit', 'exit', '종료'\\]:\n\n                print(\"Exiting chatbot.\")\n\n                break\n\n            if not question:\n\n                continue\n\n            # Process all questions via LangGraph (including Slack send)\n\n            print(\"Processing question...\")\n\n            answer = await chatbot.ask(question, verbose=True)\n\n            print(f\"\\\\nAnswer: {answer}\")\n\n            print(\"-\" \\* 50)\n\n        except KeyboardInterrupt:\n\n            print(\"\\\\n\\\\nExiting chatbot.\")\n\n            break\n\n        except Exception as e:\n\n            print(f\"An error occurred: {e}\")\n\nasync def single\\_question(question: str):\n\n    \"\"\"Process single question\"\"\"\n\n    # Initialize MCP\n\n    await initialize\\_mcp()\n\n    chatbot = MedicalChatbot()\n\n    answer = await chatbot.ask(question, verbose=True)\n\n    print(f\"\\\\nFinal answer: {answer}\")\n\n    return answer\n\n# Run functions\n\nasync def main():\n\n    \"\"\"Main run function\"\"\"\n\n    import argparse\n\n    parser = argparse.ArgumentParser(description=\"Medical chatbot\")\n\n    parser.add\\_argument(\"--question\", \"-q\", help=\"Question\")\n\n    parser.add\\_argument(\"--interactive\", \"-i\", action=\"store\\_true\", help=\"Interactive mode\")\n\n    args = parser.parse\\_args()\n\n    if args.question:\n\n        await single\\_question(args.question)\n\n    else:\n\n        # Start in interactive mode by default\n\n        await interactive\\_chat()\n\nif \\_\\_name\\_\\_ == \"\\_\\_main\\_\\_\":\n\n    asyncio.run(main())\n\nI'm getting a Slack API error in this code... Why is this happening?",
        "assistant": "Yes, I took a close look at the code you provided. The `SlackNotifier` class and the overall logic are very well written. If you're getting an error from the Slack API, it's much more likely to be a **configuration or permission issue** rather than a logical bug in your code.\n\nThere are many possible causes for API errors, but given your code and how Slack's API works, here are the most likely reasons and how to fix them:\n\n### ## 📄 Summary: Most Likely Causes\n\n1.  **Missing Slack app permissions**: Your bot token (`SLACK_BOT_TOKEN`) probably doesn't have the `chat:write` scope, which is required to send messages.\n2.  **Bot not in the channel**: The bot app hasn't been added as a member to the channel (`SLACK_CHANNEL`) you're trying to send messages to.\n3.  **Incorrect token or channel ID**: The Slack bot token or channel ID in your `.env` file is incorrect.\n\n* * *\n\n### ## 🛠️ Step-by-step Troubleshooting\n\nFollow these steps in order to resolve the issue:\n\n#### 1. Check Slack App Permissions (Most Important)\n\nTo use the Slack API, your bot app needs to be granted 'scopes' that allow it to perform certain actions.\n\n*   **Where to check:** \n    [Slack API site](https://api.slack.com/apps)\n     → Select your app → Left menu: **OAuth & Permissions**\n*   **What to check:** In the `Scopes` → `Bot Token Scopes` section, make sure **`chat:write`** is present.\n*   **How to fix:** If `chat:write` is missing, click **[Add an OAuth Scope]** and add `chat:write`. Then, at the top of the page, click the yellow banner **[reinstall your app to your workspace]** to apply the changes.\n\n#### 2. Invite the Bot to the Channel\n\nBots can only send messages to channels they are a member of.\n\n*   **How to check:** In the Slack channel you want to send messages to, check if your bot's name is in the member list.\n*   **How to fix:** In the channel, type:\n    ```\n    /invite @[Your Bot Name]\n    ```\n\n#### 3. Double-check `.env` Values\n\nMake sure the information in your `.env` file is correct.\n\n*   **`SLACK_BOT_TOKEN`**:\n    *   Make sure it's a **Bot User OAuth Token** starting with `xoxb-` (not a user token starting with `xoxp-`).\n    *   Make sure there are no leading or trailing spaces.\n*   **`SLACK_CHANNEL`**:\n    *   Make sure the channel ID is correct. It usually starts with `C` (e.g., `C0123456789`).\n    *   **How to find channel ID:** In the Slack desktop app, right-click the channel name → Copy link. The last part of the URL is the channel ID. Or check in the channel details.\n\n#### 4. Test the API Directly with `curl`\n\nYou can test the API directly from your terminal using `curl` to see if the problem is with your code or with your configuration.\n\nReplace `[YOUR_BOT_TOKEN]`, `[YOUR_CHANNEL_ID]`, and `[YOUR_USER_ID]` with your actual values and run this in your terminal:\n\n```\ncurl -X POST -H \"Authorization: Bearer [YOUR_BOT_TOKEN]\" \\\n-H \"Content-type: application/json; charset=utf-8\" \\\n--data '{\n    \"channel\": \"[YOUR_CHANNEL_ID]\",\n    \"text\": \"<@[YOUR_USER_ID]> Test message using curl.\"\n}' \\\nhttps://slack.com/api/chat.postMessage\n```\n\n*   **If this `curl` command succeeds:** Your Slack setup and token are fine. Double-check your Python code's environment variable loading (`load_dotenv()`) and so on.\n*   **If this `curl` command fails:** Then the problem is definitely with your token, permissions, or channel. Go back and check steps 1–3.\n\n### ## 💡 Code Improvement Suggestion\n\nYour error handling is already excellent. One more thing you could add for easier debugging is to handle the `not_in_channel` error, which occurs when the bot isn't a member of the channel.\n\n```"
    }, {
        "turn": 2,
        "user": "{\n\n    \"display\\_information\": {\n\n        \"name\": \"Thank You Pediatric Anesthesia Chatbot\",\n\n        \"description\": \"A Pediatric Anesthesia Chatbot Exclusively for Clinical Staff\",\n\n        \"background\\_color\": \"#0034cf\"\n\n    },\n\n    \"features\": {\n\n        \"bot\\_user\": {\n\n            \"display\\_name\": \"ThankYouBot\",\n\n            \"always\\_online\": true\n\n        }\n\n    },\n\n    \"oauth\\_config\": {\n\n        \"scopes\": {\n\n            \"bot\": \\[\n\n                \"chat:write\",\n\n                \"chat:write.public\",\n\n                \"im:history\",\n\n                \"im:write\",\n\n                \"users:read\",\n\n                \"channels:history\",\n\n                \"channels:read\",\n\n                \"chat:write.customize\",\n\n                \"reactions:write\",\n\n                \"users:read.email\"\n\n            \\]\n\n        }\n\n    },\n\n    \"settings\": {\n\n        \"org\\_deploy\\_enabled\": false,\n\n        \"socket\\_mode\\_enabled\": false,\n\n        \"token\\_rotation\\_enabled\": false\n\n    }\n\n}It looks like this...",
        "assistant": "Yes, I checked the Slack app settings (Manifest) you sent.\n\nTo get straight to the point, **your configuration looks very correct.** 👍\n\n### ## ✅ Configuration Analysis\n\n*   **`oauth_config`**: This is the most important part. Under `scopes`, your bot has the essential **`chat:write`** permission needed to send messages.\n*   **Additional permissions**: You also have other useful permissions like `chat:write.public`, `im:write`, etc., which are helpful for sending messages.\n\nSo, **your Slack app's permission (Scopes) settings are not the problem.**\n\n* * *\n\n### ## 🕵️‍♀️ So What Could Be the Cause?\n\nIf your settings are correct but you're still getting errors, the problem is likely elsewhere. Here’s what you should check next:\n\n1.  **Did you \"reinstall the app\"? (Most likely)**\n    *   After changing scopes, you **must reinstall the app to your workspace** for the new permissions to take effect. This is a very common thing to forget.\n    *   **How to fix:** \n        Go to [Slack API site](https://api.slack.com/apps)\n         → Select your app → **Basic Information** page → In the `Install your app` section, click **[Reinstall to Workspace]**.\n2.  **Is the bot invited to the channel?**\n    *   If your settings are fine, this is the next most likely issue. Even with `chat:write` permission, the bot **cannot send messages to a private channel unless it is a member**.\n    *   **How to fix:** In the Slack channel you want to send messages to, type `/invite @ThankYouBot` to invite the bot.\n3.  **Double-check your `.env` file values.**\n    *   **Token value:** If you reinstall the app, the token may change. Go to the `OAuth & Permissions` page, copy the latest `xoxb-` token, and paste it into your `.env` file.\n    *   **Channel ID:** Make sure the channel ID (starts with `C...`) is correct and has no extra spaces.\n\n**Summary: The first thing you should try is #1, \"reinstall the app.\"** Then check #2 and #3 in order, and your problem should be resolved."
    }, {
        "turn": 3,
        "user": "How do I load Excel files with different extensions?",
        "assistant": "Of course! The most efficient way to load multiple Excel files with different extensions (`.xlsx`, `.xls`, `.csv`, etc.) at once is to **use Python's `pandas` library**.\n\nHere's how you can automatically find all Excel-related files in a specific folder and load each file's data into a variable.\n\n* * *\n\n### ## 🐍 Solution with Python and Pandas\n\nThe code below finds all `.xlsx`, `.xls`, and `.csv` files in a folder called 'ExcelFiles', and stores their data in a dictionary called `excel_data`.\n\n**Preparation:**\n\n1.  Install Python\n2.  In your terminal (command prompt), run: `pip install pandas openpyxl xlrd`\n\n#### 📁 Full Example Code\n\n```\nimport pandas as pd\nimport os\nimport glob"
    }, {
        "turn": 4,
        "user": "Did you account for lowercase when handling the uppercase CSV?",
        "assistant": "Yes, great point! 👍\n\nYou're absolutely right. The code I first gave uses `.endswith('.csv')`, which **is case-sensitive**. So if your file extension is `.CSV` or `.Csv`, it won't recognize it as a CSV file.\n\n* * *\n\n### ## ⚙️ Solution: Use `lower()`\n\nAs you pointed out, it's much safer to **convert the file name to lowercase before checking the extension**.\n\n*   **Original code**\n    ```\n    if file_name.endswith('.csv'):\n    ```\n*   **Revised code**\n    ```\n    if file_name.lower().endswith('.csv'):\n    ```\n\nBy adding `.lower()`, whether your file is `Report.CSV` or `Data.cSv`, it will be converted to lowercase and checked as `.csv`, so nothing will be missed.\n\n* * *\n\n### ## ✅ Full Revised Code\n\nHere's the complete code with the case-insensitive fix:\n\n```\nimport pandas as pd\nimport os\nimport glob"
    }, {
        "turn": 5,
        "user": "There was an issue with the XLS file. Is this one okay?",
        "assistant": "That's a very important point. If you're getting errors only with `.xls` files, it's **almost certainly because the `xlrd` library is missing**.\n\nIn short, when Pandas reads Excel files, it uses different 'engines' under the hood:\n\n*   **`.xlsx` (modern Excel files)** → uses the `openpyxl` engine\n*   **`.xls` (old Excel files)** → uses the `xlrd` engine\n\nSo, to read `.xls` files, you must have the `xlrd` library installed.\n\n* * *\n\n### ## ⚙️ Solution: Install the `xlrd` Library\n\nOpen your terminal (command prompt or VSCode terminal) and run:\n\n```\npip install xlrd\n```\n\nYou may have previously run `pip install pandas openpyxl` but missed `xlrd`. Just install it with the command above, and in most cases, the problem will be solved right away.\n\n* * *\n\n### ## 💡 Even Safer: Specify the Engine in Code\n\nIf you've installed the library but still have issues, you can explicitly tell Pandas which engine to use for each file type.\n\n*   Check the file extension: if it's `.xls`, use `engine='xlrd'`; if it's `.xlsx`, use `engine='openpyxl'`.\n\n### ## ✅ Final Revised Code\n\nHere's the complete code that handles both the case-sensitivity and `.xls` engine issues:\n\n```\nimport pandas as pd\nimport os\nimport glob"
    }, {
        "turn": 6,
        "user": "It still gives an error even with xlrd.",
        "assistant": "If you're still getting errors even after installing `xlrd`, it's very likely that **the file itself is problematic** or in an unexpected format. This will require a bit more debugging.\n\nThe most important thing is to check **what the error message says**. Could you copy and paste the full error message here? That will be a crucial clue to finding the cause.\n\nBefore that, here are the most common causes and solutions for `.xls` file errors:\n\n* * *\n\n### ## 🕵️‍♀️ Main Causes and Solutions for `.xls` File Errors\n\n#### 1. File Corruption or Format Mismatch\n\nThis is the most common cause. The file extension is `.xls`, but the actual file is corrupted or in a different format.\n\n*   **How to check:** Try opening the `.xls` file **directly in Excel**.\n    *   **Case A:** If Excel can't open it or says \"The file is corrupted,\" then the file itself is the problem.\n    *   **Case B:** If Excel shows a warning like \"The file format and extension don't match. Do you want to open it anyway?\" then the file is probably not a real `.xls` file (it might be HTML or something else).\n*   **How to fix (most reliable!):**\n    1.  Open the problematic `.xls` file in Excel (click 'Yes' if you get a warning).\n    2.  Go to **File > Save As**.\n    3.  Choose **`Excel Workbook (*.xlsx)`** as the file type and **save as a new `.xlsx` file**.\n    4.  Try reading the new `.xlsx` file in your Python code.\n    This method can recover corrupted files or convert files in the wrong format to the modern Excel format, solving most issues.\n\n#### 2. Password-Protected Files\n\nIf the file is password-protected, `pandas` can't read it and will throw an error.\n\n*   **How to check:** Does Excel ask for a password when you open the file?\n*   **How to fix:** Open the file in Excel, remove the password, and save it again.\n\n#### 3. HTML Files Disguised as `.xls`\n\nSome web systems export 'Excel' files by actually creating an HTML table and saving it with a `.xls` extension.\n\n*   **How to check:** If the error message includes `XLRDError: Unsupported format, or corrupt file: Expected BOF record`, this is almost certainly the case.\n*   **How to fix:** Use the **fix in #1 above** (open in Excel and save as `.xlsx`).\n\n* * *\n\n### ## 🛠️ What to Do Now\n\n1.  **Copy and paste the full error message here.**\n2.  If that's difficult, try the **#1 fix above** (open the `.xls` file in Excel and save as `.xlsx`). This alone solves over 90% of such problems.\n\nIf you share the error message, I can help you pinpoint the exact cause!"
    }, {
        "turn": 7,
        "user": "So I renamed all the file extensions to .xlsx,,, but since the internal format is still CSV or XLS, they won't load properly. What should I do in this case?",
        "assistant": "Yes, that's a really confusing situation. This is a classic problem when the file extension and the actual file content don't match. It's like opening a box labeled 'apples' and finding 'pears' inside.\n\nThe computer sees the `.xlsx` extension and thinks, \"This must be a modern Excel file!\" But when it tries to open it, the content is actually CSV (plain text) or old XLS (a completely different structure), so it throws an error because \"this isn't what I expected!\"\n\n**To solve this, you need to ignore the file extension and read the file based on its actual content.**\n\n* * *\n\n### ## ⚙️ Solution: Read Files by Content, Not Extension\n\nDon't trust the file extension—write code that **tries all possible formats in order**. In other words, \"First, try to open as modern Excel (`.xlsx`), if that fails, try old Excel (`.xls`), and if that fails, try CSV.\" This way, your code will smartly handle files regardless of their extension.\n\nHere's the final code that implements this logic. With this, you can successfully load data no matter what the extension is, as long as the content is valid.\n\n* * *\n\n### ## ✅ Final Solution Code\n\n```\nimport pandas as pd\nimport os\nimport glob\n\ndef read_any_excel(file_path):\n    \"\"\"\n    Reads a file based on its actual content, not its extension.\n    Returns (DataFrame, file_type) on success, or (None, None) on failure.\n    \"\"\""
    }]
}