Spaces:
Sleeping
Sleeping
Commit
·
b487b99
1
Parent(s):
cd0d86b
update: modified a bit to display the user input in console, also added a version with checkpointer
Browse files- modules/nodes.py +1 -0
- paintrek-chat-v1.ipynb +304 -0
- paintrek-chat-v2.ipynb +88 -29
modules/nodes.py
CHANGED
@@ -87,6 +87,7 @@ def human_node(state: DataState) -> DataState:
|
|
87 |
print("Model:", last_msg.content)
|
88 |
|
89 |
user_input = input("User: ")
|
|
|
90 |
|
91 |
# If it looks like the user is trying to quit, flag the conversation
|
92 |
# as over.
|
|
|
87 |
print("Model:", last_msg.content)
|
88 |
|
89 |
user_input = input("User: ")
|
90 |
+
print("User:", user_input)
|
91 |
|
92 |
# If it looks like the user is trying to quit, flag the conversation
|
93 |
# as over.
|
paintrek-chat-v1.ipynb
ADDED
@@ -0,0 +1,304 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
{
|
2 |
+
"cells": [
|
3 |
+
{
|
4 |
+
"cell_type": "code",
|
5 |
+
"execution_count": null,
|
6 |
+
"metadata": {
|
7 |
+
"execution": {
|
8 |
+
"iopub.execute_input": "2025-01-29T20:09:11.440091Z",
|
9 |
+
"iopub.status.busy": "2025-01-29T20:09:11.439766Z",
|
10 |
+
"iopub.status.idle": "2025-01-29T20:09:11.751153Z",
|
11 |
+
"shell.execute_reply": "2025-01-29T20:09:11.750263Z",
|
12 |
+
"shell.execute_reply.started": "2025-01-29T20:09:11.440060Z"
|
13 |
+
},
|
14 |
+
"id": "xaiioUQni_ga",
|
15 |
+
"trusted": true
|
16 |
+
},
|
17 |
+
"outputs": [],
|
18 |
+
"source": [
|
19 |
+
"from modules.data_class import DataState\n",
|
20 |
+
"from modules.tools import data_node\n",
|
21 |
+
"from modules.nodes import chatbot_with_tools, human_node, maybe_exit_human_node, maybe_route_to_tools\n",
|
22 |
+
"\n",
|
23 |
+
"from langgraph.graph import StateGraph, START, END\n",
|
24 |
+
"\n",
|
25 |
+
"from IPython.display import Image, display\n",
|
26 |
+
"from pprint import pprint\n",
|
27 |
+
"from typing import Literal\n",
|
28 |
+
"\n",
|
29 |
+
"from langgraph.prebuilt import ToolNode\n",
|
30 |
+
"\n",
|
31 |
+
"from collections.abc import Iterable\n",
|
32 |
+
"from IPython.display import display, clear_output\n",
|
33 |
+
"import sys"
|
34 |
+
]
|
35 |
+
},
|
36 |
+
{
|
37 |
+
"cell_type": "code",
|
38 |
+
"execution_count": null,
|
39 |
+
"metadata": {
|
40 |
+
"execution": {
|
41 |
+
"iopub.execute_input": "2025-01-29T20:09:11.906458Z",
|
42 |
+
"iopub.status.busy": "2025-01-29T20:09:11.905241Z",
|
43 |
+
"iopub.status.idle": "2025-01-29T20:09:11.994921Z",
|
44 |
+
"shell.execute_reply": "2025-01-29T20:09:11.993761Z",
|
45 |
+
"shell.execute_reply.started": "2025-01-29T20:09:11.906419Z"
|
46 |
+
},
|
47 |
+
"id": "9rqkQzlZxrzp",
|
48 |
+
"trusted": true
|
49 |
+
},
|
50 |
+
"outputs": [
|
51 |
+
{
|
52 |
+
"data": {
|
53 |
+
"image/png": "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",
|
54 |
+
"text/plain": [
|
55 |
+
"<IPython.core.display.Image object>"
|
56 |
+
]
|
57 |
+
},
|
58 |
+
"execution_count": 2,
|
59 |
+
"metadata": {},
|
60 |
+
"output_type": "execute_result"
|
61 |
+
}
|
62 |
+
],
|
63 |
+
"source": [
|
64 |
+
"graph_builder = StateGraph(DataState)\n",
|
65 |
+
"\n",
|
66 |
+
"# Nodes\n",
|
67 |
+
"graph_builder.add_node(\"chatbot_healthassistant\", chatbot_with_tools)\n",
|
68 |
+
"graph_builder.add_node(\"patient\", human_node)\n",
|
69 |
+
"graph_builder.add_node(\"documenting\", data_node)\n",
|
70 |
+
"\n",
|
71 |
+
"# Chatbot -> {ordering, tools, human, END}\n",
|
72 |
+
"graph_builder.add_conditional_edges(\"chatbot_healthassistant\", maybe_route_to_tools)\n",
|
73 |
+
"# Human -> {chatbot, END}\n",
|
74 |
+
"graph_builder.add_conditional_edges(\"patient\", maybe_exit_human_node)\n",
|
75 |
+
"# TestCase_Paintrek\n",
|
76 |
+
"# Tools (both kinds) always route back to chat afterwards.\n",
|
77 |
+
"graph_builder.add_edge(\"documenting\", \"chatbot_healthassistant\")\n",
|
78 |
+
"\n",
|
79 |
+
"graph_builder.add_edge(START, \"chatbot_healthassistant\")\n",
|
80 |
+
"graph_with_order_tools = graph_builder.compile()\n",
|
81 |
+
"\n",
|
82 |
+
"Image(graph_with_order_tools.get_graph().draw_mermaid_png())"
|
83 |
+
]
|
84 |
+
},
|
85 |
+
{
|
86 |
+
"cell_type": "code",
|
87 |
+
"execution_count": null,
|
88 |
+
"metadata": {
|
89 |
+
"execution": {
|
90 |
+
"iopub.execute_input": "2025-01-29T20:09:38.185616Z",
|
91 |
+
"iopub.status.busy": "2025-01-29T20:09:38.185131Z",
|
92 |
+
"iopub.status.idle": "2025-01-29T20:10:08.474591Z",
|
93 |
+
"shell.execute_reply": "2025-01-29T20:10:08.472926Z",
|
94 |
+
"shell.execute_reply.started": "2025-01-29T20:09:38.185577Z"
|
95 |
+
},
|
96 |
+
"id": "NCRSgaBUfIHF",
|
97 |
+
"trusted": true
|
98 |
+
},
|
99 |
+
"outputs": [
|
100 |
+
{
|
101 |
+
"name": "stdout",
|
102 |
+
"output_type": "stream",
|
103 |
+
"text": [
|
104 |
+
"Executing the chatbot graph...\n",
|
105 |
+
"Model: Welcome to the Paintrek world. I am a health assistant, an interactive clinical recording system. I will ask you questions about your pain and related symptoms and record your responses. I will then store this information securely. At any time, you can type `q` to quit.\n",
|
106 |
+
"User: Let's start\n",
|
107 |
+
"Model: First, I'll need some basic information for identification. What is your full name, date of birth, gender, and contact number? Also, please provide the name and contact information for an emergency contact.\n",
|
108 |
+
"User: I am Frank, DOB 1986-01-01, male, phone number is 12345, emergency name is Zoe, number is 67890.\n",
|
109 |
+
"Model: Thank you. Now, can you describe your main symptom and how long you've been experiencing it?\n",
|
110 |
+
"User: I had a headache, and it started two days ago.\n",
|
111 |
+
"Model: Okay. Let's move on to a pain assessment. To help me understand your headache, please tell me:\n",
|
112 |
+
"\n",
|
113 |
+
"1. Where is the pain located on your head? (e.g., forehead, temple, back of head)\n",
|
114 |
+
"2. On which side of your head is the pain located? (left, right, or both)\n",
|
115 |
+
"3. On a scale of 0 to 10, with 0 being no pain and 10 being the worst pain imaginable, how intense is your headache right now?\n",
|
116 |
+
"4. Can you describe the characteristics of your pain? (e.g., sharp, dull, throbbing, aching)\n",
|
117 |
+
"5. When did the headache begin?\n",
|
118 |
+
"6. Does the pain radiate to any other areas? If so, where?\n",
|
119 |
+
"7. Are there any triggers that seem to make the headache worse?\n",
|
120 |
+
"8. Do you have any other symptoms associated with the headache? (e.g., nausea, vomiting, sensitivity to light or sound)\n",
|
121 |
+
"\n",
|
122 |
+
"Please answer as accurately as possible. Remember, there are no wrong answers.\n",
|
123 |
+
"User: My pain is located on the right head, and the scale is 2, the pain is sharp, the pain started 2 days ago, and the pain is not radiating. Cold will trigger the pain, and no other symptoms. \n",
|
124 |
+
"Model: Excellent. Now, let's gather some information about your medical history. Do you have any existing medical conditions, allergies, current medications, or a history of surgeries? Please include dates where applicable.\n",
|
125 |
+
"User: I had sinus, diagnosed in 2001-01-01, I did not have any surgery. I am taking ibprofin, no allergies.\n",
|
126 |
+
"Model: We're making good progress. Next, I need some information about your family medical history, social history, and a brief systems review. Could you tell me about any significant medical conditions in your family? What is your occupation, and are there any lifestyle factors like smoking, alcohol, or drug use that I should be aware of? Also, describe your living situation and support system. Finally, have you experienced any recent changes in your weight, sleep patterns, gastrointestinal or urinary function, or any constitutional symptoms like fever, chills, or night sweats?\n",
|
127 |
+
"User: My mom had a lung cancer at 45.\n",
|
128 |
+
"Model: Thank you. Could you please answer the rest of the questions regarding your social history and systems review? This information is crucial for a complete picture of your health.\n",
|
129 |
+
"User: I am a data scientist, I am a viggie. I don't use tobbaco or alcohol. My stress level is midium. \n",
|
130 |
+
"Model: Okay. And what about your sleep patterns, gastrointestinal and urinary function, and any constitutional symptoms?\n",
|
131 |
+
"User: I sleep well, and no other symptoms.\n",
|
132 |
+
"Model: To complete the intake, let's briefly discuss your pain management history and functional assessment. Are you currently taking any pain medication for your headache? Have you consulted any specialists for your headache or explored any alternative therapies? How would you rate the effectiveness of any treatments you've tried? Finally, how has this headache affected your quality of life, your ability to perform daily activities, and your overall mood?\n",
|
133 |
+
"User: No, I am not taking any other medication, and I haven't tried any alternative therapies. \n",
|
134 |
+
"Model: And how has this headache impacted your daily life, activities, and mood?\n",
|
135 |
+
"User: I am okay on my mood, and my quality of life is also good. I want to fix my headache.\n",
|
136 |
+
"Model: Finally, let's talk about your treatment goals and expectations. What are your hopes for resolving this headache? Are there any alternative treatments you'd consider?\n",
|
137 |
+
"User: I want to fix my headache.\n",
|
138 |
+
"Model: Before we finalize this record, let me summarize the information we've gathered. Please confirm if everything is accurate:\n",
|
139 |
+
"\n",
|
140 |
+
"Patient Name: Frank\n",
|
141 |
+
"DOB: 1986-01-01\n",
|
142 |
+
"Gender: Male\n",
|
143 |
+
"Contact: 12345\n",
|
144 |
+
"Emergency Contact: Zoe, 67890\n",
|
145 |
+
"\n",
|
146 |
+
"Main Symptom: Headache\n",
|
147 |
+
"Symptom Duration: Two days\n",
|
148 |
+
"\n",
|
149 |
+
"Pain Location: Right head\n",
|
150 |
+
"Pain Intensity: 2/10\n",
|
151 |
+
"Pain Characteristics: Sharp\n",
|
152 |
+
"Pain Onset: Two days ago\n",
|
153 |
+
"Radiation: No\n",
|
154 |
+
"Triggers: Cold\n",
|
155 |
+
"Associated Symptoms: None\n",
|
156 |
+
"\n",
|
157 |
+
"Medical History: Sinus (diagnosed 2001-01-01), taking ibuprofen, no allergies, no surgeries.\n",
|
158 |
+
"\n",
|
159 |
+
"Family History: Mother had lung cancer at age 45.\n",
|
160 |
+
"\n",
|
161 |
+
"Social History: Data scientist, no smoking, alcohol, or drug use. Living situation and support system unknown. Moderate stress levels.\n",
|
162 |
+
"\n",
|
163 |
+
"Systems Review: Good sleep, normal gastrointestinal and urinary function, no fever, chills, or night sweats, no recent weight changes.\n",
|
164 |
+
"\n",
|
165 |
+
"Pain Management History: No current pain medication, no specialist consultations, no alternative therapies.\n",
|
166 |
+
"\n",
|
167 |
+
"Functional Assessment: Good quality of life, no activity limitations, okay mood.\n",
|
168 |
+
"\n",
|
169 |
+
"Treatment Goals: Resolve headache. No alternative treatment considerations.\n",
|
170 |
+
"\n",
|
171 |
+
"\n",
|
172 |
+
"Is all of this information correct? Please let me know if any corrections are needed.\n",
|
173 |
+
"User: Yes\n",
|
174 |
+
"Your input data:\n",
|
175 |
+
"{'ID': {'name': 'Frank', 'DOB': '1986-01-01', 'gender': 'male', 'contact': '12345', 'emergency_contact': 'Zoe, 67890'}, 'symptom': {'main_symptom': 'headache', 'symptom_length': 'two days'}, 'pain': {'pain_location': 'right head', 'pain_side': 'right', 'pain_intensity': 2.0, 'pain_description': 'sharp', 'start_time': '2 days ago', 'radiation': False, 'triggers': 'cold', 'symptom': 'headache'}, 'medical_hist': {'medical_condition': 'sinus', 'first_time': '2001-01-01', 'surgery_history': 'None', 'medication': 'ibprofin', 'allergy': 'None'}, 'family_hist': {'family_history': 'Mom had lung cancer at 45'}, 'social_hist': {'occupation': 'data scientist', 'smoke': False, 'alcohol': False, 'drug': False, 'support_system': 'unknown', 'living_condition': 'unknown'}, 'review_system': {'weight_change': 'None', 'fever': False, 'chill': False, 'night_sweats': False, 'sleep': 'well', 'gastrointestinal': 'normal', 'urinary': 'normal'}, 'pain_manage': {'pain_medication': 'None', 'specialist': False, 'other_therapy': 'None', 'effectiveness': 'unknown'}, 'functional': {'life_quality': 'good', 'limit_activity': 'None', 'mood': 'okay'}, 'plan': {'goal': 'fix headache', 'expectation': 'fix headache', 'alternative_treatment': 'None'}}\n",
|
176 |
+
"Saving the data!\n",
|
177 |
+
"{'ID': {'name': 'Frank', 'DOB': '1986-01-01', 'gender': 'male', 'contact': '12345', 'emergency_contact': 'Zoe, 67890'}, 'symptom': {'main_symptom': 'headache', 'symptom_length': 'two days'}, 'pain': {'pain_location': 'right head', 'pain_side': 'right', 'pain_intensity': 2.0, 'pain_description': 'sharp', 'start_time': '2 days ago', 'radiation': False, 'triggers': 'cold', 'symptom': 'headache'}, 'medical_hist': {'medical_condition': 'sinus', 'first_time': '2001-01-01', 'surgery_history': 'None', 'medication': 'ibprofin', 'allergy': 'None'}, 'family_hist': {'family_history': 'Mom had lung cancer at 45'}, 'social_hist': {'occupation': 'data scientist', 'smoke': False, 'alcohol': False, 'drug': False, 'support_system': 'unknown', 'living_condition': 'unknown'}, 'review_system': {'weight_change': 'None', 'fever': False, 'chill': False, 'night_sweats': False, 'sleep': 'well', 'gastrointestinal': 'normal', 'urinary': 'normal'}, 'pain_manage': {'pain_medication': 'None', 'specialist': False, 'other_therapy': 'None', 'effectiveness': 'unknown'}, 'functional': {'life_quality': 'good', 'limit_activity': 'None', 'mood': 'okay'}, 'plan': {'goal': 'fix headache', 'expectation': 'fix headache', 'alternative_treatment': 'None'}}\n"
|
178 |
+
]
|
179 |
+
}
|
180 |
+
],
|
181 |
+
"source": [
|
182 |
+
"# The default recursion limit for traversing nodes is 25 - setting it higher means you can try a more complex order with multiple steps and round-trips.\n",
|
183 |
+
"config = {\"recursion_limit\": 1000}\n",
|
184 |
+
"\n",
|
185 |
+
"# Uncomment this line to execute the graph:\n",
|
186 |
+
"# Clear output before running new states\n",
|
187 |
+
"clear_output(wait=True)\n",
|
188 |
+
"\n",
|
189 |
+
"# Ensure messages print immediately\n",
|
190 |
+
"print(\"Executing the chatbot graph...\", flush=True)\n",
|
191 |
+
"state = graph_with_order_tools.invoke({\"messages\": []}, config)\n",
|
192 |
+
"# display(state) # Ensures state is shown in Jupyter\n",
|
193 |
+
"# sys.stdout.flush()\n",
|
194 |
+
"\n",
|
195 |
+
"# pprint(state)"
|
196 |
+
]
|
197 |
+
},
|
198 |
+
{
|
199 |
+
"cell_type": "code",
|
200 |
+
"execution_count": 4,
|
201 |
+
"metadata": {
|
202 |
+
"trusted": true
|
203 |
+
},
|
204 |
+
"outputs": [
|
205 |
+
{
|
206 |
+
"data": {
|
207 |
+
"text/plain": [
|
208 |
+
"{'ID': {'name': 'Frank',\n",
|
209 |
+
" 'DOB': '1986-01-01',\n",
|
210 |
+
" 'gender': 'male',\n",
|
211 |
+
" 'contact': '12345',\n",
|
212 |
+
" 'emergency_contact': 'Zoe, 67890'},\n",
|
213 |
+
" 'symptom': {'main_symptom': 'headache', 'symptom_length': 'two days'},\n",
|
214 |
+
" 'pain': {'pain_location': 'right head',\n",
|
215 |
+
" 'pain_side': 'right',\n",
|
216 |
+
" 'pain_intensity': 2.0,\n",
|
217 |
+
" 'pain_description': 'sharp',\n",
|
218 |
+
" 'start_time': '2 days ago',\n",
|
219 |
+
" 'radiation': False,\n",
|
220 |
+
" 'triggers': 'cold',\n",
|
221 |
+
" 'symptom': 'headache'},\n",
|
222 |
+
" 'medical_hist': {'medical_condition': 'sinus',\n",
|
223 |
+
" 'first_time': '2001-01-01',\n",
|
224 |
+
" 'surgery_history': 'None',\n",
|
225 |
+
" 'medication': 'ibprofin',\n",
|
226 |
+
" 'allergy': 'None'},\n",
|
227 |
+
" 'family_hist': {'family_history': 'Mom had lung cancer at 45'},\n",
|
228 |
+
" 'social_hist': {'occupation': 'data scientist',\n",
|
229 |
+
" 'smoke': False,\n",
|
230 |
+
" 'alcohol': False,\n",
|
231 |
+
" 'drug': False,\n",
|
232 |
+
" 'support_system': 'unknown',\n",
|
233 |
+
" 'living_condition': 'unknown'},\n",
|
234 |
+
" 'review_system': {'weight_change': 'None',\n",
|
235 |
+
" 'fever': False,\n",
|
236 |
+
" 'chill': False,\n",
|
237 |
+
" 'night_sweats': False,\n",
|
238 |
+
" 'sleep': 'well',\n",
|
239 |
+
" 'gastrointestinal': 'normal',\n",
|
240 |
+
" 'urinary': 'normal'},\n",
|
241 |
+
" 'pain_manage': {'pain_medication': 'None',\n",
|
242 |
+
" 'specialist': False,\n",
|
243 |
+
" 'other_therapy': 'None',\n",
|
244 |
+
" 'effectiveness': 'unknown'},\n",
|
245 |
+
" 'functional': {'life_quality': 'good',\n",
|
246 |
+
" 'limit_activity': 'None',\n",
|
247 |
+
" 'mood': 'okay'},\n",
|
248 |
+
" 'plan': {'goal': 'fix headache',\n",
|
249 |
+
" 'expectation': 'fix headache',\n",
|
250 |
+
" 'alternative_treatment': 'None'}}"
|
251 |
+
]
|
252 |
+
},
|
253 |
+
"execution_count": 4,
|
254 |
+
"metadata": {},
|
255 |
+
"output_type": "execute_result"
|
256 |
+
}
|
257 |
+
],
|
258 |
+
"source": [
|
259 |
+
"state[\"data\"]"
|
260 |
+
]
|
261 |
+
},
|
262 |
+
{
|
263 |
+
"cell_type": "code",
|
264 |
+
"execution_count": null,
|
265 |
+
"metadata": {},
|
266 |
+
"outputs": [],
|
267 |
+
"source": []
|
268 |
+
}
|
269 |
+
],
|
270 |
+
"metadata": {
|
271 |
+
"colab": {
|
272 |
+
"name": "day-3-building-an-agent-with-langgraph.ipynb",
|
273 |
+
"toc_visible": true
|
274 |
+
},
|
275 |
+
"kaggle": {
|
276 |
+
"accelerator": "none",
|
277 |
+
"dataSources": [],
|
278 |
+
"dockerImageVersionId": 30786,
|
279 |
+
"isGpuEnabled": false,
|
280 |
+
"isInternetEnabled": true,
|
281 |
+
"language": "python",
|
282 |
+
"sourceType": "notebook"
|
283 |
+
},
|
284 |
+
"kernelspec": {
|
285 |
+
"display_name": "paintrek",
|
286 |
+
"language": "python",
|
287 |
+
"name": "python3"
|
288 |
+
},
|
289 |
+
"language_info": {
|
290 |
+
"codemirror_mode": {
|
291 |
+
"name": "ipython",
|
292 |
+
"version": 3
|
293 |
+
},
|
294 |
+
"file_extension": ".py",
|
295 |
+
"mimetype": "text/x-python",
|
296 |
+
"name": "python",
|
297 |
+
"nbconvert_exporter": "python",
|
298 |
+
"pygments_lexer": "ipython3",
|
299 |
+
"version": "3.10.13"
|
300 |
+
}
|
301 |
+
},
|
302 |
+
"nbformat": 4,
|
303 |
+
"nbformat_minor": 4
|
304 |
+
}
|
paintrek-chat-v2.ipynb
CHANGED
@@ -2,7 +2,7 @@
|
|
2 |
"cells": [
|
3 |
{
|
4 |
"cell_type": "code",
|
5 |
-
"execution_count":
|
6 |
"metadata": {
|
7 |
"execution": {
|
8 |
"iopub.execute_input": "2025-01-29T20:09:11.440091Z",
|
@@ -21,6 +21,7 @@
|
|
21 |
"from modules.nodes import chatbot_with_tools, human_node, maybe_exit_human_node, maybe_route_to_tools\n",
|
22 |
"\n",
|
23 |
"from langgraph.graph import StateGraph, START, END\n",
|
|
|
24 |
"\n",
|
25 |
"from IPython.display import Image, display\n",
|
26 |
"from pprint import pprint\n",
|
@@ -35,7 +36,7 @@
|
|
35 |
},
|
36 |
{
|
37 |
"cell_type": "code",
|
38 |
-
"execution_count":
|
39 |
"metadata": {
|
40 |
"execution": {
|
41 |
"iopub.execute_input": "2025-01-29T20:09:11.906458Z",
|
@@ -62,6 +63,7 @@
|
|
62 |
],
|
63 |
"source": [
|
64 |
"graph_builder = StateGraph(DataState)\n",
|
|
|
65 |
"\n",
|
66 |
"# Nodes\n",
|
67 |
"graph_builder.add_node(\"chatbot_healthassistant\", chatbot_with_tools)\n",
|
@@ -77,14 +79,14 @@
|
|
77 |
"graph_builder.add_edge(\"documenting\", \"chatbot_healthassistant\")\n",
|
78 |
"\n",
|
79 |
"graph_builder.add_edge(START, \"chatbot_healthassistant\")\n",
|
80 |
-
"graph_with_order_tools = graph_builder.compile()\n",
|
81 |
"\n",
|
82 |
"Image(graph_with_order_tools.get_graph().draw_mermaid_png())"
|
83 |
]
|
84 |
},
|
85 |
{
|
86 |
"cell_type": "code",
|
87 |
-
"execution_count":
|
88 |
"metadata": {
|
89 |
"execution": {
|
90 |
"iopub.execute_input": "2025-01-29T20:09:38.185616Z",
|
@@ -103,30 +105,85 @@
|
|
103 |
"text": [
|
104 |
"Executing the chatbot graph...\n",
|
105 |
"Model: Welcome to the Paintrek world. I am a health assistant, an interactive clinical recording system. I will ask you questions about your pain and related symptoms and record your responses. I will then store this information securely. At any time, you can type `q` to quit.\n",
|
106 |
-
"
|
107 |
-
"Model:
|
108 |
-
"
|
109 |
-
"Model:
|
110 |
-
"
|
111 |
-
"Model:
|
112 |
-
"
|
113 |
-
"
|
114 |
-
"
|
115 |
-
"
|
116 |
-
"
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
117 |
"\n",
|
118 |
-
"Is this information correct
|
|
|
119 |
"Your input data:\n",
|
120 |
-
"{'ID': {'name': 'Frank', 'DOB': '1986-01-01', 'gender': 'male', 'contact': '12345', 'emergency_contact': 'Zoe, 67890'}, 'symptom': {'main_symptom': 'headache', 'symptom_length': 'two days'}, 'pain': {'pain_location': 'right head', 'pain_side': 'right', 'pain_intensity': 2.0, 'pain_description': 'sharp', 'start_time': '2 days ago', 'radiation': False, 'triggers': 'cold', 'symptom': '
|
121 |
"Saving the data!\n",
|
122 |
-
"{'ID': {'name': 'Frank', 'DOB': '1986-01-01', 'gender': 'male', 'contact': '12345', 'emergency_contact': 'Zoe, 67890'}, 'symptom': {'main_symptom': 'headache', 'symptom_length': 'two days'}, 'pain': {'pain_location': 'right head', 'pain_side': 'right', 'pain_intensity': 2.0, 'pain_description': 'sharp', 'start_time': '2 days ago', 'radiation': False, 'triggers': 'cold', 'symptom': '
|
123 |
]
|
124 |
}
|
125 |
],
|
126 |
"source": [
|
127 |
-
"#
|
128 |
-
"# means you can try a more complex order with multiple steps and round-trips.\n",
|
129 |
-
"config = {\"recursion_limit\": 1000}\n",
|
130 |
"\n",
|
131 |
"# Uncomment this line to execute the graph:\n",
|
132 |
"# Clear output before running new states\n",
|
@@ -164,34 +221,36 @@
|
|
164 |
" 'start_time': '2 days ago',\n",
|
165 |
" 'radiation': False,\n",
|
166 |
" 'triggers': 'cold',\n",
|
167 |
-
" 'symptom': '
|
168 |
" 'medical_hist': {'medical_condition': 'sinus',\n",
|
169 |
" 'first_time': '2001-01-01',\n",
|
170 |
" 'surgery_history': 'None',\n",
|
171 |
" 'medication': 'ibprofin',\n",
|
172 |
" 'allergy': 'None'},\n",
|
173 |
-
" 'family_hist': {'family_history': '
|
174 |
" 'social_hist': {'occupation': 'data scientist',\n",
|
175 |
" 'smoke': False,\n",
|
176 |
" 'alcohol': False,\n",
|
177 |
" 'drug': False,\n",
|
178 |
-
" 'support_system': '
|
179 |
-
" 'living_condition': '
|
180 |
" 'review_system': {'weight_change': 'None',\n",
|
181 |
" 'fever': False,\n",
|
182 |
" 'chill': False,\n",
|
183 |
" 'night_sweats': False,\n",
|
184 |
" 'sleep': 'well',\n",
|
185 |
-
" 'gastrointestinal': '
|
186 |
-
" 'urinary': '
|
187 |
" 'pain_manage': {'pain_medication': 'None',\n",
|
188 |
" 'specialist': False,\n",
|
189 |
" 'other_therapy': 'None',\n",
|
190 |
-
" 'effectiveness':
|
191 |
" 'functional': {'life_quality': 'good',\n",
|
192 |
" 'limit_activity': 'None',\n",
|
193 |
" 'mood': 'okay'},\n",
|
194 |
-
" 'plan': {'goal': '
|
|
|
|
|
195 |
]
|
196 |
},
|
197 |
"execution_count": 4,
|
|
|
2 |
"cells": [
|
3 |
{
|
4 |
"cell_type": "code",
|
5 |
+
"execution_count": null,
|
6 |
"metadata": {
|
7 |
"execution": {
|
8 |
"iopub.execute_input": "2025-01-29T20:09:11.440091Z",
|
|
|
21 |
"from modules.nodes import chatbot_with_tools, human_node, maybe_exit_human_node, maybe_route_to_tools\n",
|
22 |
"\n",
|
23 |
"from langgraph.graph import StateGraph, START, END\n",
|
24 |
+
"from langgraph.checkpoint.memory import MemorySaver\n",
|
25 |
"\n",
|
26 |
"from IPython.display import Image, display\n",
|
27 |
"from pprint import pprint\n",
|
|
|
36 |
},
|
37 |
{
|
38 |
"cell_type": "code",
|
39 |
+
"execution_count": null,
|
40 |
"metadata": {
|
41 |
"execution": {
|
42 |
"iopub.execute_input": "2025-01-29T20:09:11.906458Z",
|
|
|
63 |
],
|
64 |
"source": [
|
65 |
"graph_builder = StateGraph(DataState)\n",
|
66 |
+
"memory = MemorySaver()\n",
|
67 |
"\n",
|
68 |
"# Nodes\n",
|
69 |
"graph_builder.add_node(\"chatbot_healthassistant\", chatbot_with_tools)\n",
|
|
|
79 |
"graph_builder.add_edge(\"documenting\", \"chatbot_healthassistant\")\n",
|
80 |
"\n",
|
81 |
"graph_builder.add_edge(START, \"chatbot_healthassistant\")\n",
|
82 |
+
"graph_with_order_tools = graph_builder.compile(checkpointer=memory)\n",
|
83 |
"\n",
|
84 |
"Image(graph_with_order_tools.get_graph().draw_mermaid_png())"
|
85 |
]
|
86 |
},
|
87 |
{
|
88 |
"cell_type": "code",
|
89 |
+
"execution_count": null,
|
90 |
"metadata": {
|
91 |
"execution": {
|
92 |
"iopub.execute_input": "2025-01-29T20:09:38.185616Z",
|
|
|
105 |
"text": [
|
106 |
"Executing the chatbot graph...\n",
|
107 |
"Model: Welcome to the Paintrek world. I am a health assistant, an interactive clinical recording system. I will ask you questions about your pain and related symptoms and record your responses. I will then store this information securely. At any time, you can type `q` to quit.\n",
|
108 |
+
"User: Let's start\n",
|
109 |
+
"Model: First, I'll need some basic information for identification. What is your full name, date of birth, gender, and contact number? Also, please provide the name and contact information for an emergency contact.\n",
|
110 |
+
"User: I am Frank, DOB 1986-01-01, male, phone number is 12345, emergency name is Zoe, number is 67890.\n",
|
111 |
+
"Model: Thank you. Now, can you describe your main symptom and how long you've been experiencing it?\n",
|
112 |
+
"User: I had a headache, and it started two days ago.\n",
|
113 |
+
"Model: Okay. Let's move on to a pain assessment. To help me understand your headache, please tell me:\n",
|
114 |
+
"\n",
|
115 |
+
"1. Where is the pain located on your head? (e.g., forehead, temple, back of head)\n",
|
116 |
+
"2. On which side of your head is the pain located? (left, right, or both)\n",
|
117 |
+
"3. On a scale of 0 to 10, with 0 being no pain and 10 being the worst pain imaginable, how intense is your headache right now?\n",
|
118 |
+
"4. Can you describe the characteristics of your pain? (e.g., sharp, dull, throbbing, aching)\n",
|
119 |
+
"5. When did the headache begin?\n",
|
120 |
+
"6. Does the pain radiate to any other areas? If so, where?\n",
|
121 |
+
"7. Are there any triggers that seem to make the headache worse?\n",
|
122 |
+
"8. Do you have any other symptoms associated with the headache? (e.g., nausea, vomiting, sensitivity to light or sound)\n",
|
123 |
+
"\n",
|
124 |
+
"Please answer as accurately as possible. Remember, there are no wrong answers.\n",
|
125 |
+
"User: My pain is located on the right head, and the scale is 2, the pain is sharp, the pain started 2 days ago, and the pain is not radiating. Cold will trigger the pain, and no other symptoms. \n",
|
126 |
+
"Model: Excellent. Now, let's gather some information about your medical history. Do you have any existing medical conditions, allergies, current medications, or a history of surgeries? Please include dates where applicable.\n",
|
127 |
+
"User: I had sinus, diagnosed in 2001-01-01, I did not have any surgery. I am taking ibprofin, no allergies.\n",
|
128 |
+
"Model: We're making good progress. Next, I need some information about your family medical history, social history, and a brief systems review. Could you tell me about any significant medical conditions in your family? What is your occupation, and are there any lifestyle factors like smoking, alcohol, or drug use that I should be aware of? Also, describe your living situation and support system. Finally, have you experienced any recent changes in your weight, sleep patterns, gastrointestinal or urinary function, or any constitutional symptoms like fever, chills, or night sweats?\n",
|
129 |
+
"User: My mom had a lung cancer at 45.\n",
|
130 |
+
"Model: Thank you. Could you please answer the rest of the questions regarding your social history and systems review? This information is crucial for a complete picture of your health.\n",
|
131 |
+
"User: I am a data scientist, I am a viggie. I don't use tobbaco or alcohol. My stress level is midium. \n",
|
132 |
+
"Model: Okay. And what about your sleep patterns, gastrointestinal and urinary function, and any constitutional symptoms?\n",
|
133 |
+
"User: I sleep well, and no other symptoms.\n",
|
134 |
+
"Model: To complete the intake, let's briefly discuss your pain management history and functional assessment. Are you currently taking any pain medication for your headache? Have you consulted any specialists for your headache or explored any alternative therapies? How would you rate the effectiveness of any treatments you've tried? Finally, how has this headache affected your quality of life, your ability to perform daily activities, and your overall mood?\n",
|
135 |
+
"User: No, I am not taking any other medication, and I haven't tried any alternative therapies. \n",
|
136 |
+
"Model: And how has this headache impacted your daily life, activities, and mood?\n",
|
137 |
+
"User: I am okay on my mood, and my quality of life is also good. I want to fix my headache.\n",
|
138 |
+
"Model: Finally, let's talk about your treatment goals and expectations. What are your hopes for resolving this headache? Are there any alternative treatments you'd consider?\n",
|
139 |
+
"User: I want to fix my headache.\n",
|
140 |
+
"Model: Before we finalize this record, let me summarize the information we've gathered. Please confirm if everything is accurate:\n",
|
141 |
+
"\n",
|
142 |
+
"Patient Name: Frank\n",
|
143 |
+
"DOB: 1986-01-01\n",
|
144 |
+
"Gender: Male\n",
|
145 |
+
"Contact: 12345\n",
|
146 |
+
"Emergency Contact: Zoe, 67890\n",
|
147 |
+
"\n",
|
148 |
+
"Main Symptom: Headache\n",
|
149 |
+
"Symptom Duration: Two days\n",
|
150 |
+
"\n",
|
151 |
+
"Pain Location: Right head\n",
|
152 |
+
"Pain Intensity: 2/10\n",
|
153 |
+
"Pain Characteristics: Sharp\n",
|
154 |
+
"Pain Onset: Two days ago\n",
|
155 |
+
"Radiation: No\n",
|
156 |
+
"Triggers: Cold\n",
|
157 |
+
"Associated Symptoms: None\n",
|
158 |
+
"\n",
|
159 |
+
"Medical History: Sinus (diagnosed 2001-01-01), taking ibuprofen, no allergies, no surgeries.\n",
|
160 |
+
"\n",
|
161 |
+
"Family History: Mother had lung cancer at age 45.\n",
|
162 |
+
"\n",
|
163 |
+
"Social History: Data scientist, no smoking, alcohol, or drug use. Living situation and support system unknown. Moderate stress levels.\n",
|
164 |
+
"\n",
|
165 |
+
"Systems Review: Good sleep, normal gastrointestinal and urinary function, no fever, chills, or night sweats, no recent weight changes.\n",
|
166 |
+
"\n",
|
167 |
+
"Pain Management History: No current pain medication, no specialist consultations, no alternative therapies.\n",
|
168 |
+
"\n",
|
169 |
+
"Functional Assessment: Good quality of life, no activity limitations, okay mood.\n",
|
170 |
+
"\n",
|
171 |
+
"Treatment Goals: Resolve headache. No alternative treatment considerations.\n",
|
172 |
+
"\n",
|
173 |
"\n",
|
174 |
+
"Is all of this information correct? Please let me know if any corrections are needed.\n",
|
175 |
+
"User: Yes\n",
|
176 |
"Your input data:\n",
|
177 |
+
"{'ID': {'name': 'Frank', 'DOB': '1986-01-01', 'gender': 'male', 'contact': '12345', 'emergency_contact': 'Zoe, 67890'}, 'symptom': {'main_symptom': 'headache', 'symptom_length': 'two days'}, 'pain': {'pain_location': 'right head', 'pain_side': 'right', 'pain_intensity': 2.0, 'pain_description': 'sharp', 'start_time': '2 days ago', 'radiation': False, 'triggers': 'cold', 'symptom': 'headache'}, 'medical_hist': {'medical_condition': 'sinus', 'first_time': '2001-01-01', 'surgery_history': 'None', 'medication': 'ibprofin', 'allergy': 'None'}, 'family_hist': {'family_history': 'Mom had lung cancer at 45'}, 'social_hist': {'occupation': 'data scientist', 'smoke': False, 'alcohol': False, 'drug': False, 'support_system': 'unknown', 'living_condition': 'unknown'}, 'review_system': {'weight_change': 'None', 'fever': False, 'chill': False, 'night_sweats': False, 'sleep': 'well', 'gastrointestinal': 'normal', 'urinary': 'normal'}, 'pain_manage': {'pain_medication': 'None', 'specialist': False, 'other_therapy': 'None', 'effectiveness': 'unknown'}, 'functional': {'life_quality': 'good', 'limit_activity': 'None', 'mood': 'okay'}, 'plan': {'goal': 'fix headache', 'expectation': 'fix headache', 'alternative_treatment': 'None'}}\n",
|
178 |
"Saving the data!\n",
|
179 |
+
"{'ID': {'name': 'Frank', 'DOB': '1986-01-01', 'gender': 'male', 'contact': '12345', 'emergency_contact': 'Zoe, 67890'}, 'symptom': {'main_symptom': 'headache', 'symptom_length': 'two days'}, 'pain': {'pain_location': 'right head', 'pain_side': 'right', 'pain_intensity': 2.0, 'pain_description': 'sharp', 'start_time': '2 days ago', 'radiation': False, 'triggers': 'cold', 'symptom': 'headache'}, 'medical_hist': {'medical_condition': 'sinus', 'first_time': '2001-01-01', 'surgery_history': 'None', 'medication': 'ibprofin', 'allergy': 'None'}, 'family_hist': {'family_history': 'Mom had lung cancer at 45'}, 'social_hist': {'occupation': 'data scientist', 'smoke': False, 'alcohol': False, 'drug': False, 'support_system': 'unknown', 'living_condition': 'unknown'}, 'review_system': {'weight_change': 'None', 'fever': False, 'chill': False, 'night_sweats': False, 'sleep': 'well', 'gastrointestinal': 'normal', 'urinary': 'normal'}, 'pain_manage': {'pain_medication': 'None', 'specialist': False, 'other_therapy': 'None', 'effectiveness': 'unknown'}, 'functional': {'life_quality': 'good', 'limit_activity': 'None', 'mood': 'okay'}, 'plan': {'goal': 'fix headache', 'expectation': 'fix headache', 'alternative_treatment': 'None'}}\n"
|
180 |
]
|
181 |
}
|
182 |
],
|
183 |
"source": [
|
184 |
+
"# This is for the checkpointer to have a thread id to remember for each user, and I don't know if letter mix with number will work here \n",
|
185 |
+
"# The default recursion limit for traversing nodes is 25 - setting it higher means you can try a more complex order with multiple steps and round-trips.\n",
|
186 |
+
"config = {\"configurable\": {\"thread_id\": \"1\"}, \"recursion_limit\": 1000}\n",
|
187 |
"\n",
|
188 |
"# Uncomment this line to execute the graph:\n",
|
189 |
"# Clear output before running new states\n",
|
|
|
221 |
" 'start_time': '2 days ago',\n",
|
222 |
" 'radiation': False,\n",
|
223 |
" 'triggers': 'cold',\n",
|
224 |
+
" 'symptom': 'headache'},\n",
|
225 |
" 'medical_hist': {'medical_condition': 'sinus',\n",
|
226 |
" 'first_time': '2001-01-01',\n",
|
227 |
" 'surgery_history': 'None',\n",
|
228 |
" 'medication': 'ibprofin',\n",
|
229 |
" 'allergy': 'None'},\n",
|
230 |
+
" 'family_hist': {'family_history': 'Mom had lung cancer at 45'},\n",
|
231 |
" 'social_hist': {'occupation': 'data scientist',\n",
|
232 |
" 'smoke': False,\n",
|
233 |
" 'alcohol': False,\n",
|
234 |
" 'drug': False,\n",
|
235 |
+
" 'support_system': 'unknown',\n",
|
236 |
+
" 'living_condition': 'unknown'},\n",
|
237 |
" 'review_system': {'weight_change': 'None',\n",
|
238 |
" 'fever': False,\n",
|
239 |
" 'chill': False,\n",
|
240 |
" 'night_sweats': False,\n",
|
241 |
" 'sleep': 'well',\n",
|
242 |
+
" 'gastrointestinal': 'normal',\n",
|
243 |
+
" 'urinary': 'normal'},\n",
|
244 |
" 'pain_manage': {'pain_medication': 'None',\n",
|
245 |
" 'specialist': False,\n",
|
246 |
" 'other_therapy': 'None',\n",
|
247 |
+
" 'effectiveness': 'unknown'},\n",
|
248 |
" 'functional': {'life_quality': 'good',\n",
|
249 |
" 'limit_activity': 'None',\n",
|
250 |
" 'mood': 'okay'},\n",
|
251 |
+
" 'plan': {'goal': 'fix headache',\n",
|
252 |
+
" 'expectation': 'fix headache',\n",
|
253 |
+
" 'alternative_treatment': 'None'}}"
|
254 |
]
|
255 |
},
|
256 |
"execution_count": 4,
|