{
    "title": "Slack Message Sending Multi-turn Implementation",
    "turns": [{
        "turn": 1,
        "user": "# JSON Response Handling\n\n        import json\n\n        try:\n\n            response_data = json.loads(result)\n\n            if response_data.get(\"action\") == \"send_slack_message\":\n\n                recipient = response_data.get(\"recipient\", \"\")\n\n                channel = response_data.get(\"channel\", \"\")\n\n                message = response_data.get(\"message\", \"\")\n\n                # If recipient name is missing\n\n                if not recipient or recipient == \"<recipient name>\":\n\n                    st.session_state[\"pending_slack\"] = True\n\n                    st.session_state[\"pending_message\"] = message\n\n                    st.session_state.sessions[st.session_state.current_session_index].append(\n\n                        {\"role\": \"assistant\", \"content\": \"Who should I send it to and where? (Please enter channel, direct, or name)\"}\n\n                    )\n\n                # If recipient exists but channel info is missing\n\n                elif not channel or channel == \"<Channel name or 'DM'>\":\n\n                    st.session_state[\"pending_slack\"] = True\n\n                    st.session_state[\"pending_recipient\"] = recipient\n\n                    st.session_state[\"pending_message\"] = message\n\n                    st.session_state.sessions[st.session_state.current_session_index].append(\n\n                        {\"role\": \"assistant\", \"content\": f\"Should I send it to {recipient} via a channel? Or as a direct message? (Please enter the Channel name or 'DM')\"}\n\n                    )\n\n                else:\n\n                    # If all info is complete, process Slack sending (add actual Slack sending logic here)\n\n                    try:\n\n                        # Actual Slack sending logic (example)\n\n                        # slack_result = send_slack_message(recipient, channel, message)\n\n                        st.session_state.sessions[st.session_state.current_session_index].append(\n\n                            {\"role\": \"assistant\", \"content\": f\"✅ Message sent to {recipient} via {channel}: {message}\"}\n\n                        )\n\n                    except Exception as e:\n\n                        # If Slack sending fails, ask a user-friendly question\n\n                        st.session_state[\"pending_slack\"] = True\n\n                        st.session_state[\"pending_message\"] = message\n\n                        st.session_state.sessions[st.session_state.current_session_index].append(\n\n                            {\"role\": \"assistant\", \"content\": f\"Sorry, I couldn't find {recipient}. Who should I send it to? (Available users: your name)\"}\n\n                        )\n\n            else:\n\n                # Handle general responses\n\n                st.session_state.sessions[st.session_state.current_session_index].append({\"role\": \"assistant\", \"content\": result})\n\n        except json.JSONDecodeError:\n\n            # If not JSON, handle as a general response\n\n            st.session_state.sessions[st.session_state.current_session_index].append({\"role\": \"assistant\", \"content\": result})\n\n        st.rerun()<< Can you check if this code is implemented as a multi-turn format, where if the user is missing, or if the user is present but the channel or DM info is missing, the bot asks in chat to get the desired value?",
        "assistant": "Yes, that's correct. The code you provided is **accurately implemented as a multi-turn conversation format, where if the user requests to send a Slack message but required information (recipient, channel) is missing, the bot asks follow-up questions to collect the missing info.**\n\nHere's how the code implements multi-turn conversation:\n\n### Code Analysis\n\n1.  **If recipient info is missing**\n    *   The condition `if not recipient or recipient == \"<recipient name>\":` detects when the `recipient` value in the LLM-generated JSON response is missing or is a placeholder.\n    *   By setting `st.session_state[\"pending_slack\"] = True`, it saves the state that a Slack sending task is \"in progress\" in `session_state`.\n    *   The bot then asks **\"Who should I send it to and where?\"** to the user, requesting the missing info. When the user enters the next message, the previously saved `pending_slack` state is checked to continue the Slack sending logic.\n2.  **If channel info is missing (but recipient exists)**\n    *   The condition `elif not channel or channel == \"<Channel name or 'DM'>\":` detects when `recipient` info exists but `channel` info is missing.\n    *   Similarly, it saves the state in `session_state` and also stores the `recipient` value for use in the next turn.\n    *   The bot then asks a more specific question using the known `recipient` info, like **\"Should I send it to {recipient} via a channel? Or as a direct message?\"**\n\n* * *\n\n### Conclusion 📝\n\nBy using `st.session_state` to maintain the conversation context and asking the user for missing information step by step, this is indeed a **proper multi-turn conversation implementation**. The user can provide all the necessary info through the conversation with the bot and eventually send the Slack message."
    }, {
        "turn": 2,
        "user": "But right now, it's not responding like that?",
        "assistant": "Yes, I checked the screenshot you uploaded. Currently, the chatbot isn't asking \"Who should I send it to?\" and is instead just outputting what looks like an internal system error message.\n\n**The logic itself in the code I analyzed earlier is fine.** However, what's happening now is **because the conditions for that code to run aren't being met.**\n\n### Cause of the Problem\n\nThe core issue is with the format of the `result` value being passed to the Python code.\n\n1.  **Expected behavior:**\n    *   User input: \"Send hello\"\n    *   The LLM (language model) sees this and realizes it needs to send a Slack message, but sees no recipient.\n    *   The LLM generates a response in **JSON format**:\n        ```\n        {\n          \"action\": \"send_slack_message\",\n          \"recipient\": \"<recipient name>\",\n          \"channel\": \"<Channel name or 'DM'>\",\n          \"message\": \"안녕\"\n        }\n        ```\n    *   The Python code receives this JSON, sees that `recipient` is empty, and then asks \"Who should I send it to?\"\n2.  **Actual behavior (based on screenshot):**\n    *   User input: \"Send hello\"\n    *   The LLM or a system in front of it **tries to send the Slack message but fails to find the user (recipient)**.\n    *   This failure result is passed to the Python code as **plain text (string)**, not JSON: `result = 'Slack send result: Error: Could not find user 'Z'. If the \"user question\" does not contain a '~' user, the recipient cannot be determined...'`\n    *   The Python code tries to run `json.loads(result)`, but since `result` isn't JSON, a `json.JSONDecodeError` is raised.\n    *   The `except` block runs, and the error message (`result`) is output as the chatbot's reply without any processing.\n    ```\n    try:\n        response_data = json.loads(result) # Error occurs here!"
    }, {
        "turn": 3,
        "user": "Then change it to work the way I want.",
        "assistant": "Understood. Even if the LLM (language model) doesn't behave as expected, I'll modify the current Python code so that your desired multi-turn conversation flow works.\n\nI'll update the code so that **when the LLM returns an inappropriate text error, the code detects this and forcibly switches to a flow where it asks the user for the recipient again.**\n\n* * *\n\n### ## 💡 Fundamental Solution: Edit the LLM Prompt\n\nFirst, the ideal solution is to edit the LLM's instructions (prompt). I recommend adding an explicit rule like below:\n\n> **Example LLM instruction:** \"If the user's request is missing `recipient` info, never attempt to send the message and fail. Instead, always generate a JSON response with `action: \"send_slack_message\"` and fill the `recipient` field with the placeholder `\"<recipient name>\"`.\"\n\nThis way, the LLM will always respond in the JSON format we expect, making the whole system stable.\n\n* * *\n\n### ## 💻 Immediate Code Fix\n\nAssuming you can't change the prompt right now, here's a code enhancement to handle exceptions.\n\nI've added a defensive logic to the existing code so that when a `json.JSONDecodeError` occurs, it **checks if the returned text is a specific error message like \"User not found\" (user not found)**.\n\n#### ### Updated Code\n\n```"
    }, {
        "turn": 4,
        "user": "That's good, but\n\nIn the except json.JSONDecodeError: part, you need to split it into two cases: when the user can't be found, and when the user is found but the DM or channel info is missing.",
        "assistant": "Yes, understood. That's a great point. It's much more user-friendly to update the code so that it asks more specific follow-up questions depending on the error message returned by the LLM.\n\nI'll expand the `except json.JSONDecodeError` block into an **`if/elif/else` structure** so it can handle those two scenarios separately.\n\n### Prerequisite\n\nFor this code to work as intended, the LLM needs to return **distinct, consistent error messages** for each case. For example:\n\n*   **User not found:** `... User not found ...`\n*   **Channel/DM unknown:** `... The user 'YourName' was found, but the channel information could not be determined ...`\n\n* * *\n\n### ## 💻 Code Update\n\nBelow is the revised code that splits the `except` block into two cases. **The keywords in the `elif` condition** should be adjusted to match the actual error messages returned by your LLM.\n\n```"
    }, {
        "turn": 5,
        "user": "Why can't it recognize when I say 'to so-and-so'?",
        "assistant": "Hi there. You're asking about the issue where the chatbot keeps repeating the same question.\n\nThe reason the chatbot keeps repeating itself, even when you answer \"to so-and-so,\" is because **it doesn't connect your reply to the previous conversation (\"Send hello\") and treats it as a completely new question.** In other words, the logic for managing conversation context and state is missing.\n\nCurrently, the code only handles 'responding to a question,' but **the part that processes the user's follow-up answer is missing.**\n\nTo fix this, you'll need to update the overall code flow.\n\n* * *\n\n### Cause of the Problem\n\n1.  **Missing state check logic:** When you type \"to so-and-so,\" the code doesn't first check if `st.session_state.get(\"pending_slack\")` is set. Instead, it just sends this input straight to the LLM, so the LLM, with no context, doesn't know what to do with \"to so-and-so\" and returns an error again.\n2.  **Lost context (message):** Even if you did check `pending_slack`, it's likely that the previous \"hello\" message wasn't saved in `session_state`. So even if the recipient is found, the message to send is forgotten.\n\n* * *\n\n### ## 💻 How to Fix the Overall Flow\n\nHere's a structure for the code to solve this problem. Think of it in **three steps**:\n\n1.  **(Step 1) State check:** For every user input, first check if you're in a `pending_slack` state.\n2.  **(Step 2) Handle follow-up answers:** If `pending_slack` is set, don't call the LLM. Instead, combine the new info with the existing info.\n3.  **(Step 3) Handle new questions:** If not in `pending_slack`, send the input to the LLM as before.\n\nTry applying the code below to your main project logic.\n\n```\nimport streamlit as st\nimport json"
    }, {
        "turn": 6,
        "user": "I applied the logic?",
        "assistant": "Okay, got it. It must be frustrating that the problem still isn't resolved even after applying the logic.\n\nTo pinpoint the exact cause, I need a bit more information. Let's check together where in the code the issue is happening.\n\n* * *\n\n### ## 🕵️ Information Needed to Diagnose the Cause\n\nTo solve the problem, please check the following. If you can copy and paste the content below, or share a screenshot, it will really help me find the cause.\n\n1.  **The full updated code:** Please show me the **entire code block** that includes the logic I suggested, from `if prompt := st.chat_input(...)` to `st.rerun()`. (There could be issues with code placement or variable names.)\n2.  **New run result (screenshot):** After updating the code, take a **screenshot of the full conversation screen** from the beginning. Show \"Send hello\" -> bot's reply -> \"to recipient\" -> bot's second reply, etc. (I need to see if it behaves the same as before or if a new issue has appeared.)\n3.  **Terminal error messages:** If there are any **red error messages** in the Streamlit terminal (the black command prompt window), please copy and share them. There could be unexpected errors like `KeyError`, `NameError`, etc.\n\n* * *\n\n### ## 🤔 Common Issues to Check\n\nWhile you're preparing that info, it might help to check for these common mistakes:\n\n*   **`session_state` initialization issues:** Make sure variables like `st.session_state.sessions` and `st.session_state.current_session_index` are properly initialized at the top of your code.\n*   **Variable scope issues:** Check that variables like `prompt` and `result` are accessible wherever they're needed, and are being passed correctly between functions.\n*   **`call_llm` function issues:** Make sure the `call_llm` function is including the full conversation history, not just the last input (`prompt`). For multi-turn conversations, it's important to send the entire previous context.\n\nIf you share the info above, I'll analyze it right away and help you find a solution."
    }, {
        "turn": 7,
        "user": "# --- Page Settings ---\n\nst.set_page_config(\n\n    page_title=\"Thank You Pediatric Anesthesia Chatbot\",\n\n    page_icon=\"🩺\",\n\n    layout=\"wide\"\n\n)\n\n# --- 1. Global Style and Font Settings ---\n\nst.markdown(\"\"\"\n\n<link href=\"https://fonts.googleapis.com/css2?family=Nanum+Gothic:wght@400;700&display=swap\" rel=\"stylesheet\">\n\n<style>\n\n    * {\n\n        font-family: 'Nanum Gothic', sans-serif !important;\n\n    }\n\n    body, .stApp {\n\n        background-color: #eaf4fb !important;\n\n    }\n\n    /* Apply font to all */\n\n    html, body, [class*=\"st-\"], button, input, textarea {\n\n        font-family: 'Nanum Gothic', sans-serif;\n\n    }\n\n    /* Main page title */\n\n    .main-title {\n\n        text-align: center;\n\n        color: #4F8BF9;\n\n        font-weight: 1000;\n\n    }\n\n    /* Main page subtitle */\n\n    .main-subtitle {\n\n        text-align: center;\n\n        color: #555555;\n\n        margin-bottom: 2rem;\n\n    }\n\n    /* Feature card style */\n\n    .feature-card {\n\n        background: #f7faff;\n\n        border-radius: 15px;\n\n        padding: 24px 20px;\n\n        box-shadow: 0 4px 12px rgba(0, 0, 0, 0.08);\n\n        transition: transform 0.2s ease-in-out, box-shadow 0.2s ease-in-out;\n\n        height: 100%;\n\n        display: flex;\n\n        flex-direction: column;\n\n        align-items: center;\n\n        text-align: center;\n\n    }\n\n    .feature-card:hover {\n\n        transform: translateY(-5px);\n\n        box-shadow: 0 8px 20px rgba(0, 0, 0, 0.12);\n\n    }\n\n    .feature-card h3 {\n\n        color: #4F8BF9;\n\n        margin-bottom: 12px;\n\n        font-size: 1.3em;\n\n    }\n\n    .feature-card p {\n\n        font-size: 1em;\n\n        color: #333;\n\n        line-height: 1.6;\n\n    }\n\n    .feature-icon {\n\n        font-size: 3em;\n\n        margin-bottom: 1rem;\n\n    }\n\n    /* Default button style */\n\n    .stButton>button {\n\n        background-color: #4F8BF9;\n\n        color: white;\n\n        font-size: 2rem;\n\n        font-weight: 900;\n\n        border-radius: 8px;\n\n        height: 3em;\n\n        width: 100%;\n\n        transition: all 0.2s ease-in-out;\n\n        border: none;\n\n        box-shadow: 0 4px 10px rgba(79, 139, 249, 0.3);\n\n    }\n\n    .stButton>button:hover {\n\n        background-color: #3c7de0;\n\n        transform: translateY(-2px);\n\n        box-shadow: 0 6px 14px rgba(79, 139, 249, 0.4);\n\n    }\n\n    .stButton>button:active {\n\n        background-color: #cccccc !important;\n\n        color: #222 !important;\n\n        transform: translateY(0);\n\n        box-shadow: none !important;\n\n    }\n\n    /* Footer style */\n\n    .footer {\n\n        text-align: center;\n\n        color: #aaa;\n\n        font-size: 0.9em;\n\n        margin-top: 3rem;\n\n    }\n\n    .start-chat-title {\n\n        text-align: center;\n\n        color: #222;\n\n        font-size: 1.1rem !important;\n\n        font-weight: 700;\n\n    }\n\n    .stButton>button {\n\n        background-color: #4F8BF9;\n\n        color: white;\n\n        font-size: 1.5rem;\n\n        font-weight: 900;\n\n        border-radius: 8px;\n\n        height: 3em;\n\n        width: 100%;\n\n        transition: all 0.2s ease-in-out;\n\n        border: none;\n\n        box-shadow: 0 4px 10px rgba(79, 139, 249, 0.3);\n\n    }\n\n    /* Reduce sidebar button width */\n\n    div[data-testid=\"stSidebar\"] .stButton > button {\n\n        width: 80% !important;\n\n        min-width: 0 !important;\n\n        max-width: 200px !important;\n\n        margin-left: auto;\n\n        margin-right: auto;\n\n        display: block;\n\n        height: 1.1em !important;\n\n        padding-top: 0.1em !important;\n\n        padding-bottom: 0.1em !important;\n\n        line-height: 0.6 !important;\n\n    }\n\n    /* Force sidebar button height/padding */\n\n    section[data-testid=\"stSidebar\"] button {\n\n        height: 2.2em !important;\n\n        padding-top: 0.4em !important;\n\n        padding-bottom: 0.4em !important;\n\n        line-height: 2.4 !important;\n\n        font-size: 1.1rem !important;\n\n        min-height: unset !important;\n\n        max-height: 2.6em !important;\n\n    }\n\n    /* Force sidebar session button text to one line, ellipsis if overflow */\n\n    div[data-testid=\"stSidebar\"] .stButton > button {\n\n        white-space: nowrap !important;\n\n        overflow: hidden !important;\n\n        text-overflow: ellipsis !important;\n\n        max-width: 220px !important;\n\n        min-width: 0 !important;\n\n        padding: 0.5em 1em !important;\n\n        height: auto !important;\n\n        line-height: 1.3 !important;\n\n    }\n\n    /* Center text input label */\n\n    div[data-testid=\"stTextInput\"] label {\n\n        width: 100%;\n\n        text-align: center !important;\n\n        display: block !important;\n\n        justify-content: center !important;\n\n        align-items: center !important;\n\n    }\n\n</style>\n\n\"\"\", unsafe_allow_html=True)\n\n# --- 2. Screen State Management ---\n\nif \"page\" not in st.session_state:\n\n    st.session_state[\"page\"] = \"main\"\n\ndef go_to_chatbot():\n\n    st.session_state[\"page\"] = \"chatbot\"\n\ndef go_to_main():\n\n    st.session_state[\"page\"] = \"main\"\n\n# --- 3. Page Rendering ---\n\nif st.session_state[\"page\"] == \"main\":\n\n    # --- Main Page ---\n\n    st.markdown(\"<h1 class='main-title'>🩺 Thank You Pediatric Anesthesia Chatbot</h1>\", unsafe_allow_html=True)\n\n    st.markdown(\"<p class='main-subtitle'>From clinical questions to patient data management, we help you with everything in pediatric anesthesia.</p>\", unsafe_allow_html=True)\n\n    st.write(\"\")\n\n    st.write(\"\")\n\n    st.subheader(\"✨ Main Features\")\n\n    st.write(\"\")\n\n    col1, col2 = st.columns(2, gap=\"medium\")\n\n    with col1:\n\n        st.markdown(\"\"\"\n\n        <div class=\"feature-card\">\n\n            <div class=\"feature-icon\">🤖</div>\n\n            <h3>RAG-based Clinical Q&A</h3>\n\n            <p>Get fast and accurate answers to all your pediatric anesthesia clinical questions, referencing the latest documents.</p>\n\n        </div>\n\n        \"\"\", unsafe_allow_html=True)\n\n    with col2:\n\n        st.markdown(\"\"\"\n\n        <div class=\"feature-card\">\n\n            <div class=\"feature-icon\">🗂️</div>\n\n            <h3>Patient Data Management</h3>\n\n            <p>Look up <strong>surgical info and clinical charts</strong> in real time by patient ID, and get tailored answers for efficient <strong>patient management</strong>.</p>\n\n        </div>\n\n        \"\"\", unsafe_allow_html=True)\n\n    st.write(\"\")\n\n    col3, col4 = st.columns(2, gap=\"medium\")\n\n    with col3:\n\n        st.markdown(\"\"\"\n\n        <div class=\"feature-card\">\n\n            <div class=\"feature-icon\">💬</div>\n\n            <h3>Slack Message Integration</h3>\n\n            <p>Instantly send important or shareable search results to colleagues via Slack for efficient collaboration.</p>\n\n        </div>\n\n        \"\"\", unsafe_allow_html=True)\n\n    with col4:\n\n        st.markdown(\"\"\"\n\n        <div class=\"feature-card\">\n\n            <div class=\"feature-icon\">💾</div>\n\n            <h3>Save Conversation</h3>\n\n            <p>You can download important chatbot conversations as a text file anytime for record-keeping and management.</p>\n\n        </div>\n\n        \"\"\", unsafe_allow_html=True)\n\n    st.divider()\n\n    st.markdown(\"\"\"\n\n    <div style='text-align:center;'>\n\n        <h3 class='start-chat-title'>👇 Enter your name to start the chatbot </h3>\n\n    </div>\n\n    \"\"\", unsafe_allow_html=True)\n\n    st.markdown(\"\"\"\n\n    <div style='display:flex; justify-content:center;'><div style='width:300px;'>\n\n    \"\"\", unsafe_allow_html=True)\n\n    col1, col2 = st.columns([1, 1])\n\n    with col1:\n\n        user_name = st.text_input(\"\", value=st.session_state.get(\"user_name\", \"\"), key=\"user_name_inline\", placeholder=\"Enter your name\")\n\n        st.session_state[\"user_name\"] = user_name\n\n    with col2:\n\n        st.write(\"\")  # Space to move button down\n\n        st.button(\"Start Chatbot\", on_click=go_to_chatbot, disabled=not bool(user_name.strip()), use_container_width=True)\n\n    st.image(\"image.png\", use_container_width=True)\n\n    st.markdown(\"<div class='footer'>© 2025 Thank You Pediatric Anesthesia. All Rights Reserved.</div>\", unsafe_allow_html=True)\n\nelse: # \"chatbot\" page\n\n    # --- Chatbot Page ---\n\n    user_name = st.session_state.get(\"user_name\", \"User\")\n\n    st.markdown(f\"<h4 style='color: #222; margin-bottom:1rem;'>Hello {user_name}!👋</h4>\", unsafe_allow_html=True)\n\n    st.markdown(\"<h2 style='color: #222;'>🩺 Thank You Pediatric Anesthesia Chatbot</h2>\", unsafe_allow_html=True)\n\n    col1, col2, col3 = st.columns(3)\n\n    box_style = \"background:{bg}; border-radius:12px; padding:14px; text-align:center; box-shadow:0 2px 8px #0001; height:120px; display:flex; flex-direction:column; justify-content:center; align-items:center; font-size:0.9rem;\"\n\n    icon_box_style = \"width:100%; display:flex; justify-content:center; align-items:center; height:36px; margin-bottom:0.5em;\"\n\n    with col1:\n\n        st.markdown(f\"\"\"<div style='{box_style.format(bg=\"#fef9e7\")}'><div style='{icon_box_style}'><span style='font-size:1.5rem;'>🆔</span></div><div><b>Try asking with a patient name or specific surgery history</b>.</div></div>\"\"\", unsafe_allow_html=True)\n\n    with col2:\n\n        st.markdown(f\"\"\"<div style='{box_style.format(bg=\"#eafaf1\")}'><div style='{icon_box_style}'><span style='font-size:1.5rem;'>📋</span></div><div>Answers are generated by looking up the patient's <b>clinical chart and surgery info</b> in real time.</div></div>\"\"\", unsafe_allow_html=True)\n\n    with col3:\n\n        st.markdown(f\"\"\"<div style='{box_style.format(bg=\"#fef9e7\")}'><div style='{icon_box_style}'><img src=\"https://a.slack-edge.com/80588/marketing/img/icons/icon_slack_hash_colored.png\" alt=\"Slack\" style=\"height:1.5rem;\"></div><div>Automatically send answers to colleagues via <b>Slack</b> as needed.</div></div>\"\"\", unsafe_allow_html=True)\n\n    st.divider()\n\n    if \"sessions\" not in st.session_state:\n\n        st.session_state.sessions = [[]]\n\n    if \"current_session_index\" not in st.session_state:\n\n        st.session_state.current_session_index = 0\n\n    def switch_session(session_index):\n\n        st.session_state.current_session_index = session_index\n\n    with st.sidebar:\n\n        st.markdown('<div style=\"text-align:center; font-size:2.5rem; margin-bottom:0.5em;\">🐘</div>', unsafe_allow_html=True)\n\n        if st.button(\"⬅️ Back to Main\"):\n\n            go_to_main()\n\n        st.markdown(\"---\")\n\n        if st.button(\"🆕 Start New Chat\"):\n\n            st.session_state.sessions.append([])\n\n            switch_session(len(st.session_state.sessions) - 1)\n\n        st.markdown(\"---\")\n\n        st.markdown(\"#### 💬 Chat Session History\")\n\n        if not any(st.session_state.sessions):\n\n            st.markdown(\"_No saved sessions yet._\")\n\n        else:\n\n            for i, session in enumerate(st.session_state.sessions):\n\n                if session:\n\n                    first_q = next((msg['content'] for msg in session if msg['role'] == 'user'), \"Session\")\n\n                    display_text = first_q[:10] + (\"...\" if len(first_q) > 10 else \"\")\n\n                    st.button(f\"📜 {display_text}\", key=f\"session_{i}\", on_click=switch_session, args=(i,))\n\n                else:\n\n                    st.button(f\"🆕 New Chat {i+1}\", key=f\"session_{i}\", on_click=switch_session, args=(i,))\n\n        st.markdown(\n\n            '''\n\n            <a href=\"https://app.slack.com/client/T093ELJBE2Z\" target=\"_blank\"\n\n                style=\"display:inline-block; background:#611f69; color:#fff; font-weight:700; padding:0.7em 2em; border-radius:8px; text-decoration:none; font-size:1.1rem; box-shadow:0 2px 8px #0002; margin-top:2em;\">\n\n                <img src=\"https://a.slack-edge.com/80588/marketing/img/icons/icon_slack_hash_colored.png\" alt=\"Slack\" style=\"height:1.3em; vertical-align:middle; margin-right:0.5em;\">\n\n                Go to Slack\n\n            </a>\n\n            ''', unsafe_allow_html=True\n\n        )\n\n    current_messages = st.session_state.sessions[st.session_state.current_session_index]\n\n    for message in current_messages:\n\n        if message[\"role\"] == \"user\":\n\n            st.markdown(f\"\"\"<div style='display:flex; justify-content:flex-end; margin-bottom:8px;'><div style='background:#fff; color:#222; border-radius:16px 16px 4px 16px; padding:12px 18px; max-width:70%; box-shadow:0 2px 8px #0001;'>{message[\"content\"]}</div></div>\"\"\", unsafe_allow_html=True)\n\n        else:\n\n            st.markdown(f\"\"\"<div style='display:flex; justify-content:flex-start; margin-bottom:8px;'><div style='background:#b3d8f6; color:#222; border-radius:16px 16px 16px 4px; padding:12px 18px; max-width:70%; box-shadow:0 2px 8px #0001;'>{message[\"content\"]}</div></div>\"\"\", unsafe_allow_html=True)\n\n    input_and_loading = st.empty()\n\n    if len(current_messages) > 0:\n\n        st.divider()\n\n        buffer = StringIO()\n\n        for m in current_messages:\n\n            role = \"User\" if m[\"role\"] == \"user\" else \"Chatbot\"\n\n            buffer.write(f\"{role}: {m['content']}\\n\\n\")\n\n        st.download_button(\n\n            label=\"📄 Download Current Conversation\",\n\n            data=buffer.getvalue(),\n\n            file_name=f\"chat_history_session_{st.session_state.current_session_index + 1}.txt\",\n\n            mime=\"text/plain\"\n\n        )\n\n    # 1. Define summary function (example: reuse single_question)\n\n    def summarize_history(history_text):\n\n        summary_prompt = f\"Summarize the following conversation in one or two sentences:\\n{history_text}\"\n\n        return asyncio.run(single_question(summary_prompt))\n\n    prompt = st.chat_input(\"Enter patient ID or your question...\")\n\n    if prompt:\n\n        st.session_state.sessions[st.session_state.current_session_index].append({\"role\": \"user\", \"content\": prompt})\n\n        with input_and_loading.container():\n\n            st.markdown(f\"\"\"\n\n            <div style='display:flex; justify-content:flex-end; margin-bottom:8px;'>\n\n                <div style='background:#fff; color:#222; border-radius:16px 16px 4px 16px; padding:12px 18px; max-width:70%; box-shadow:0 2px 8px #0001;'>\n\n                    {prompt}\n\n                </div>\n\n            </div>\n\n            \"\"\", unsafe_allow_html=True)\n\n            st.markdown(f\"\"\"\n\n            <div style='display:flex; justify-content:flex-start; margin-bottom:8px;'>\n\n                <div style='background:#b3d8f6; color:#222; border-radius:16px 16px 16px 4px; padding:12px 18px; max-width:70%; box-shadow:0 2px 8px #0001; font-style:italic;'>\n\n                    Generating answer...\n\n                </div>\n\n            </div>\n\n            \"\"\", unsafe_allow_html=True)\n\n        # 2. Conversation separation and summary\n\n        all_messages = current_messages + [{\"role\": \"user\", \"content\": prompt}]\n\n        if len(all_messages) > 3:\n\n            # Part to summarize\n\n            history_to_summarize = \"\\n\".join([\n\n                f\"{'User' if m['role']=='user' else 'Chatbot'}: {m['content']}\" for m in all_messages[:-3]\n\n            ])\n\n            summary = summarize_history(history_to_summarize)\n\n            recent_history = \"\\n\".join([\n\n                f\"{'User' if m['role']=='user' else 'Chatbot'}: {m['content']}\" for m in all_messages[-3:]\n\n            ])\n\n        else:\n\n            summary = \"\"\n\n            recent_history = \"\\n\".join([\n\n                f\"{'User' if m['role']=='user' else 'Chatbot'}: {m['content']}\" for m in all_messages\n\n            ])\n\n        user_question = prompt\n\n        # 3. Compose prompt and call LLM\n\n        multi_turn_prompt = f\"\"\"\\n\n\n        ### Previous Conversation Summary\n\n        <summary>\n\n        {summary}\n\n        </summary>\n\n        ### Recent Conversation\n\n        <recent_history>\n\n        {recent_history}\n\n        </recent_history>\n\n        ### Instructions\n\n        1. Refer to both the 'Previous Conversation Summary' and 'Recent Conversation' above and answer the 'User Question' naturally in Korean.\n\n        2. If the 'User Question' contains a clear request like \"send/deliver ~ (via message/slack) to ~\", you must output only the following JSON format instead of a general answer. Do not add any other explanation.\n\n        ```json\n\n        {{\n\n        \"action\": \"send_slack_message\",\n\n        \"recipient\": \"<recipient name>\",\n\n        \"channel\": \"<Channel name or 'DM'>\",\n\n        \"message\": \"Message sent by {user_name}: <content>\"\n\n        }}\n\n        ```\n\n        ### User Question\n\n        {user_question}\n\n        \"\"\"\n\n        result = asyncio.run(single_question(multi_turn_prompt))\n\n        # Clear input_and_loading container\n\n        input_and_loading.empty()\n\n        # JSON response handling\n\n        # JSON response handling\n\n        import json\n\n        try:\n\n            # (No change to existing try block code)\n\n            response_data = json.loads(result)\n\n            if response_data.get(\"action\") == \"send_slack_message\":\n\n                # ... (rest is the same)\n\n                recipient = response_data.get(\"recipient\", \"\")\n\n                channel = response_data.get(\"channel\", \"\")\n\n                message = response_data.get(\"message\", \"\")\n\n                if not recipient or recipient == \"<recipient name>\":\n\n                    st.session_state[\"pending_slack\"] = True\n\n                    st.session_state[\"pending_message\"] = message\n\n                    st.session_state.sessions[st.session_state.current_session_index].append(\n\n                        {\"role\": \"assistant\", \"content\": \"Who should I send it to and where? (Please enter channel, direct, or name)\"}\n\n                    )\n\n                elif not channel or channel == \"<Channel name or 'DM'>\":\n\n                    st.session_state[\"pending_slack\"] = True\n\n                    st.session_state[\"pending_recipient\"] = recipient\n\n                    st.session_state[\"pending_message\"] = message\n\n                    st.session_state.sessions[st.session_state.current_session_index].append(\n\n                        {\"role\": \"assistant\", \"content\": f\"Should I send it to {recipient} via a channel? Or as a direct message? (Please enter the Channel name or 'DM')\"}\n\n                    )\n\n                else:\n\n                    try:\n\n                        st.session_state.sessions[st.session_state.current_session_index].append(\n\n                            {\"role\": \"assistant\", \"content\": f\"✅ Message sent to {recipient} via {channel}: {message}\"}\n\n                        )\n\n                    except Exception as e:\n\n                        st.session_state[\"pending_slack\"] = True\n\n                        st.session_state[\"pending_message\"] = message\n\n                        st.session_state.sessions[st.session_state.current_session_index].append(\n\n                            {\"role\": \"assistant\", \"content\": f\"Sorry, I couldn't find {recipient}. Who should I send it to? (Available users: your name)\"}\n\n                        )\n\n            else:\n\n                st.session_state.sessions[st.session_state.current_session_index].append({\"role\": \"assistant\", \"content\": result})\n\n        except json.JSONDecodeError:\n\n            # 🟧 START: Modified Section 🟧\n\n            # If the response is not JSON, branch based on error content\n\n            # 1. If it's a 'user not found' error\n\n            if \"User not found\" in result:\n\n                st.session_state[\"pending_slack\"] = True\n\n                st.session_state.sessions[st.session_state.current_session_index].append(\n\n                    {\"role\": \"assistant\", \"content\": \"Couldn't find the recipient. Who should I send the message to? (e.g., Send hello to your name)\"}\n\n                )\n\n            # 2. If it's a 'channel/DM info missing' error (‼️ Adjust keywords to match actual LLM response)\n\n            elif \"Cannot determine channel information\" in result or \"Don't know where to send\" in result:\n\n                st.session_state[\"pending_slack\"] = True\n\n                # At this stage, the LLM has identified the recipient but doesn't know the channel/DM info\n\n                st.session_state.sessions[st.session_state.current_session_index].append(\n\n                    {\"role\": \"assistant\", \"content\": \"Recipient found, but I don't know where to send it. Should I send it to a channel or as a direct message (DM)?\"}\n\n                )\n\n            # 3. Otherwise, just output the text as is\n\n            else:\n\n                st.session_state.sessions[st.session_state.current_session_index].append({\"role\": \"assistant\", \"content\": result})\n\n            # 🟧 END: Modified Section 🟧\n\n        st.rerun()",
        "assistant": "Thank you for sharing your full code. I've identified the issue.\n\nAs expected, the current code **treats every user input as a new question, without any separate logic to handle follow-up answers (like \"to recipient\")**, so the conversation doesn't continue properly.\n\nThe fix is simple: For every user input, **add logic to first check if you're waiting for a Slack message follow-up (`pending_slack` state)**.\n\n* * *\n\n🔧 Code Fix\n---------\n\nBelow is a version where I've **only modified the problematic part and added the necessary logic**. Copy and paste this under your `prompt = st.chat_input(...)` section.\n\n```"
    }, {
        "turn": 8,
        "user": "pending_slack itself means you need Slack. So if you don't need this, you don't even need the recipient, right?",
        "assistant": "Yes, that's exactly right. You've understood it perfectly.\n\nThe `pending_slack` state is like a flag that says, \"I'm currently in the middle of sending a Slack message.\"\n\n*   **When `pending_slack` is on (True):** The chatbot thinks, \"Oh, I'm in the middle of sending a Slack message. The user's next input is probably an answer to fill in missing info (recipient, channel, message, etc.),\" and continues the conversation accordingly.\n*   **When `pending_slack` is off (False):** The chatbot is in its default state, waiting for a completely new question unrelated to Slack. In this state, of course, things like 'recipient' or 'channel' aren't needed at all.\n\nSo as you said, when `pending_slack` is not set, recipient info is also unnecessary. Using this state value is the key to making sure the chatbot doesn't mistake the user's answer for a new question, but instead correctly recognizes it as additional info for the previous question. That's the core of multi-turn conversation."
    }]
}