Omkar31415
commited on
Commit
·
6b6215c
1
Parent(s):
fd55db0
HuggingFace Deployment Done
Browse files- .github/workflows/main.yaml +20 -0
- Examples.txt +28 -0
- app.py +77 -0
- apt.txt +1 -0
- mental_health.py +1253 -0
- physical_health.py +1253 -0
- requirements.txt +12 -0
.github/workflows/main.yaml
ADDED
@@ -0,0 +1,20 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
name: Sync to Hugging Face hub
|
2 |
+
on:
|
3 |
+
push:
|
4 |
+
branches: [main]
|
5 |
+
|
6 |
+
# to run this workflow manually from the Actions tab
|
7 |
+
workflow_dispatch:
|
8 |
+
|
9 |
+
jobs:
|
10 |
+
sync-to-hub:
|
11 |
+
runs-on: ubuntu-latest
|
12 |
+
steps:
|
13 |
+
- uses: actions/checkout@v3
|
14 |
+
with:
|
15 |
+
fetch-depth: 0
|
16 |
+
lfs: true
|
17 |
+
- name: Push to hub
|
18 |
+
env:
|
19 |
+
HF_TOKEN: ${{ secrets.HF_TOKEN }}
|
20 |
+
run: git push https://OmkarDattaSowri:$HF_TOKEN@huggingface.co/spaces/OmkarDattaSowri/HealthCareDiagnosisLLM main
|
Examples.txt
ADDED
@@ -0,0 +1,28 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
# Physical Health
|
2 |
+
Example 1: Respiratory Symptoms
|
3 |
+
I've had a persistent dry cough for about 2 weeks now. It gets worse at night and early morning. I also have a mild fever (around 99.5°F) and feel fatigued throughout the day. Recently, I've noticed some shortness of breath when climbing stairs. No chest pain, but I feel a tightness sometimes when breathing deeply. No previous respiratory conditions.
|
4 |
+
|
5 |
+
Example 2: Digestive Issues
|
6 |
+
For the past 5 days, I've experienced severe abdominal pain in the lower right quadrant. The pain is sharp and worsens after eating. I've had nausea and vomited twice yesterday. No diarrhea, but haven't had a bowel movement in 3 days. No blood in vomit. Lost appetite completely. No fever. History of IBS but this feels different and more intense.
|
7 |
+
|
8 |
+
Example 3: Headache/Neurological
|
9 |
+
I've been experiencing intense headaches for the past week, mainly on the right side of my head. The pain is throbbing and sometimes feels like pressure behind my right eye. Light sensitivity and occasional nausea accompany the headaches. They last about 4-6 hours and over-the-counter pain medication barely helps. I've also noticed some blurred vision in my right eye just before the headaches start. No history of migraines, but my mother has them.
|
10 |
+
|
11 |
+
Example 4: Joint Pain
|
12 |
+
My knee joints, especially the right one, have been increasingly painful over the last 3 months. The pain is worse in the morning with stiffness that lasts about an hour after waking up. There's some swelling, and the area feels warm to touch. Walking up and down stairs is becoming difficult. I'm 52 years old and have worked in construction most of my life. No recent injuries, but I was diagnosed with mild hypertension last year and take lisinopril daily.
|
13 |
+
|
14 |
+
Example 5: Skin Condition
|
15 |
+
I've developed a red, itchy rash on my hands and wrists that started about 10 days ago. The rash has small bumps that sometimes weep clear fluid when scratched. The itching gets worse at night and the affected areas are now starting to show some scaling and cracking. I recently started using a new dish soap, but otherwise no changes to my routine or diet. No known allergies and I haven't taken any new medications.
|
16 |
+
|
17 |
+
Example 6: Combination of Symptoms (More Complex)
|
18 |
+
'm experiencing three main issues that might be related: extreme fatigue even after 9 hours of sleep, unexplained weight loss (about 15 pounds in the last 2 months without trying), and frequent urination, including getting up 3-4 times at night. I'm constantly thirsty, and my mouth feels dry most of the time. I've also noticed that small cuts seem to take longer to heal than usual. I'm 47 years old with a family history of diabetes, but I've never been diagnosed with any chronic conditions. My job is sedentary, and I admit my diet has included a lot of processed foods lately.
|
19 |
+
|
20 |
+
|
21 |
+
# Mental Health
|
22 |
+
I've been feeling really overwhelmed lately with work and my personal life. I can't seem to get a handle on things.
|
23 |
+
|
24 |
+
I don't see the point in anything anymore. I'm so tired all the time and nothing brings me joy.
|
25 |
+
|
26 |
+
I'm actually doing better today than last week. I still feel anxious about my presentation tomorrow, but I'm managing it better than I used to.
|
27 |
+
|
28 |
+
I'm sad and I don't know what I'll do anymore. my dog just died
|
app.py
ADDED
@@ -0,0 +1,77 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import streamlit as st
|
2 |
+
from physical_health import run_physical_health
|
3 |
+
from mental_health import run_mental_health
|
4 |
+
|
5 |
+
# Set up page config for a consistent header and favicon
|
6 |
+
st.set_page_config(
|
7 |
+
page_title="AI Health Diagnostic Assistant",
|
8 |
+
page_icon="❤️",
|
9 |
+
layout="centered",
|
10 |
+
initial_sidebar_state="auto",
|
11 |
+
)
|
12 |
+
|
13 |
+
# Add some custom CSS for styling
|
14 |
+
st.markdown(
|
15 |
+
"""
|
16 |
+
<style>
|
17 |
+
/* Center the title and adjust margins */
|
18 |
+
.main > div:first-child {
|
19 |
+
text-align: center;
|
20 |
+
}
|
21 |
+
/* Custom button style (uses st.markdown to override) */
|
22 |
+
div.stButton > button {
|
23 |
+
background-color: #4CAF50;
|
24 |
+
color: white;
|
25 |
+
padding: 10px 24px;
|
26 |
+
border: none;
|
27 |
+
border-radius: 8px;
|
28 |
+
font-size: 16px;
|
29 |
+
margin: 10px;
|
30 |
+
transition: background-color 0.3s;
|
31 |
+
}
|
32 |
+
div.stButton > button:hover {
|
33 |
+
background-color: #45a049;
|
34 |
+
}
|
35 |
+
/* Page background style */
|
36 |
+
.reportview-container {
|
37 |
+
background: #f9f9f9;
|
38 |
+
}
|
39 |
+
</style>
|
40 |
+
""",
|
41 |
+
unsafe_allow_html=True,
|
42 |
+
)
|
43 |
+
|
44 |
+
# Initialize navigation state
|
45 |
+
if 'page' not in st.session_state:
|
46 |
+
st.session_state.page = "main"
|
47 |
+
|
48 |
+
# Main page UI
|
49 |
+
if st.session_state.page == "main":
|
50 |
+
st.title("AI Health Diagnostic Assistant 🤖")
|
51 |
+
st.markdown("### Your personalized health companion")
|
52 |
+
st.markdown("#### Welcome! Please select a diagnostic tool below:")
|
53 |
+
|
54 |
+
def main():
|
55 |
+
# Main menu page
|
56 |
+
if st.session_state.page == "main":
|
57 |
+
# Two-column layout for buttons
|
58 |
+
col1, col2 = st.columns(2)
|
59 |
+
with col1:
|
60 |
+
if st.button("Mental Health Assessment"):
|
61 |
+
st.session_state.page = "mental"
|
62 |
+
st.rerun()
|
63 |
+
with col2:
|
64 |
+
if st.button("Physical Health Assessment"):
|
65 |
+
st.session_state.page = "physical"
|
66 |
+
st.rerun()
|
67 |
+
|
68 |
+
# Mental Health interface
|
69 |
+
elif st.session_state.page == "mental":
|
70 |
+
run_mental_health()
|
71 |
+
|
72 |
+
# Physical Health interface
|
73 |
+
elif st.session_state.page == "physical":
|
74 |
+
run_physical_health()
|
75 |
+
|
76 |
+
if __name__ == "__main__":
|
77 |
+
main()
|
apt.txt
ADDED
@@ -0,0 +1 @@
|
|
|
|
|
1 |
+
ffmpeg
|
mental_health.py
ADDED
@@ -0,0 +1,1253 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import streamlit as st
|
2 |
+
import os
|
3 |
+
import json
|
4 |
+
import requests
|
5 |
+
import nltk
|
6 |
+
from nltk.sentiment import SentimentIntensityAnalyzer
|
7 |
+
from transformers import BlenderbotTokenizer, BlenderbotForConditionalGeneration
|
8 |
+
from dotenv import load_dotenv
|
9 |
+
import time
|
10 |
+
|
11 |
+
# Load environment variables
|
12 |
+
load_dotenv()
|
13 |
+
groq_api_key = os.getenv("GROQ_API_KEY")
|
14 |
+
|
15 |
+
def run_mental_health():
|
16 |
+
# Initialize NLTK resources if not already downloaded
|
17 |
+
try:
|
18 |
+
nltk.data.find('vader_lexicon')
|
19 |
+
except LookupError:
|
20 |
+
nltk.download('vader_lexicon')
|
21 |
+
|
22 |
+
# Initialize session state variables
|
23 |
+
if 'conversation' not in st.session_state:
|
24 |
+
st.session_state.conversation = []
|
25 |
+
if 'therapy_state' not in st.session_state:
|
26 |
+
st.session_state.therapy_state = "initial" # States: initial, assessment, therapy, summary
|
27 |
+
if 'mental_health_scores' not in st.session_state:
|
28 |
+
st.session_state.mental_health_scores = {
|
29 |
+
"anxiety": 0,
|
30 |
+
"depression": 0,
|
31 |
+
"stress": 0,
|
32 |
+
"loneliness": 0,
|
33 |
+
"grief": 0,
|
34 |
+
"relationship_issues": 0,
|
35 |
+
"self_esteem": 0,
|
36 |
+
"trauma": 0
|
37 |
+
}
|
38 |
+
if 'assessment_progress' not in st.session_state:
|
39 |
+
st.session_state.assessment_progress = 0
|
40 |
+
if 'current_question' not in st.session_state:
|
41 |
+
st.session_state.current_question = None
|
42 |
+
if 'current_options' not in st.session_state:
|
43 |
+
st.session_state.current_options = None
|
44 |
+
if 'llm_service' not in st.session_state:
|
45 |
+
st.session_state.llm_service = "llama3" # Default LLM service: llama3 or blenderbot
|
46 |
+
if 'user_submitted' not in st.session_state:
|
47 |
+
st.session_state.user_submitted = False
|
48 |
+
if 'severity_rating' not in st.session_state:
|
49 |
+
st.session_state.severity_rating = None
|
50 |
+
if 'sentiment_analysis' not in st.session_state:
|
51 |
+
st.session_state.sentiment_analysis = []
|
52 |
+
if 'other_selected' not in st.session_state:
|
53 |
+
st.session_state.other_selected = False
|
54 |
+
if 'response_delay' not in st.session_state:
|
55 |
+
st.session_state.response_delay = False
|
56 |
+
|
57 |
+
# Load Blenderbot model and tokenizer
|
58 |
+
@st.cache_resource
|
59 |
+
def load_blenderbot_model():
|
60 |
+
tokenizer = BlenderbotTokenizer.from_pretrained("facebook/blenderbot-400M-distill")
|
61 |
+
model = BlenderbotForConditionalGeneration.from_pretrained("facebook/blenderbot-400M-distill")
|
62 |
+
return tokenizer, model
|
63 |
+
|
64 |
+
# Load sentiment analyzer
|
65 |
+
@st.cache_resource
|
66 |
+
def load_sentiment_analyzer():
|
67 |
+
return SentimentIntensityAnalyzer()
|
68 |
+
|
69 |
+
# Use Groq API with LLaMA3 model
|
70 |
+
def use_llama3_api(prompt, max_tokens=1000):
|
71 |
+
headers = {
|
72 |
+
"Authorization": f"Bearer {groq_api_key}",
|
73 |
+
"Content-Type": "application/json"
|
74 |
+
}
|
75 |
+
|
76 |
+
data = {
|
77 |
+
"model": "llama3-70b-8192", # Using LLaMA3 70B model
|
78 |
+
"messages": [{"role": "user", "content": prompt}],
|
79 |
+
"max_tokens": max_tokens,
|
80 |
+
"temperature": 0.7
|
81 |
+
}
|
82 |
+
|
83 |
+
try:
|
84 |
+
response = requests.post(
|
85 |
+
"https://api.groq.com/openai/v1/chat/completions",
|
86 |
+
headers=headers,
|
87 |
+
json=data,
|
88 |
+
timeout=10 # Add timeout
|
89 |
+
)
|
90 |
+
|
91 |
+
if response.status_code == 200:
|
92 |
+
return response.json()["choices"][0]["message"]["content"]
|
93 |
+
else:
|
94 |
+
st.error(f"Error from Groq API: {response.text}")
|
95 |
+
return "I'm having trouble connecting. Let's continue our conversation more simply."
|
96 |
+
except Exception as e:
|
97 |
+
st.error(f"Exception when calling Groq API: {str(e)}")
|
98 |
+
return "I encountered an error when trying to respond. Let me try a simpler approach."
|
99 |
+
|
100 |
+
# Use Blenderbot for simple responses
|
101 |
+
def use_blenderbot(input_text, tokenizer, model):
|
102 |
+
try:
|
103 |
+
inputs = tokenizer([input_text], return_tensors="pt")
|
104 |
+
reply_ids = model.generate(**inputs, max_length=100)
|
105 |
+
response = tokenizer.batch_decode(reply_ids, skip_special_tokens=True)[0]
|
106 |
+
return response
|
107 |
+
except Exception as e:
|
108 |
+
st.error(f"Error from Blenderbot: {str(e)}")
|
109 |
+
return "I'm having trouble understanding that. Could you rephrase or tell me more?"
|
110 |
+
|
111 |
+
# Callback to handle option selection
|
112 |
+
def handle_option_select():
|
113 |
+
selected_option = st.session_state.selected_option
|
114 |
+
if selected_option:
|
115 |
+
# Add selected option to conversation
|
116 |
+
if selected_option == "Other":
|
117 |
+
st.session_state.other_selected = True
|
118 |
+
else:
|
119 |
+
# Record the response in conversation
|
120 |
+
st.session_state.conversation.append(selected_option)
|
121 |
+
|
122 |
+
# Update mental health scores based on response
|
123 |
+
update_mental_health_scores(selected_option)
|
124 |
+
|
125 |
+
st.session_state.user_submitted = True
|
126 |
+
st.session_state.other_selected = False
|
127 |
+
|
128 |
+
# Don't generate options here, let the main loop handle it
|
129 |
+
|
130 |
+
# Callback to handle custom input submission
|
131 |
+
def handle_custom_submit():
|
132 |
+
custom_input = st.session_state.custom_input
|
133 |
+
if custom_input:
|
134 |
+
# Add custom input to conversation
|
135 |
+
st.session_state.conversation.append(custom_input)
|
136 |
+
|
137 |
+
# Update mental health scores based on custom input
|
138 |
+
update_mental_health_scores(custom_input)
|
139 |
+
|
140 |
+
st.session_state.user_submitted = True
|
141 |
+
st.session_state.custom_input = "" # Clear the input field
|
142 |
+
st.session_state.other_selected = False
|
143 |
+
|
144 |
+
# Function to update mental health scores based on user input
|
145 |
+
def update_mental_health_scores(user_input):
|
146 |
+
# Load sentiment analyzer
|
147 |
+
sia = load_sentiment_analyzer()
|
148 |
+
sentiment = sia.polarity_scores(user_input)
|
149 |
+
|
150 |
+
# Store sentiment analysis for tracking mood over time
|
151 |
+
st.session_state.sentiment_analysis.append({
|
152 |
+
"text": user_input,
|
153 |
+
"sentiment": sentiment,
|
154 |
+
"timestamp": time.time()
|
155 |
+
})
|
156 |
+
|
157 |
+
# Keywords related to different mental health issues
|
158 |
+
mental_health_keywords = {
|
159 |
+
"anxiety": ["anxious", "nervous", "worry", "panic", "fear", "stress", "tense", "overwhelm", "anxiousness", "uneasy"],
|
160 |
+
"depression": ["sad", "depress", "hopeless", "meaningless", "empty", "tired", "exhausted", "unmotivated", "worthless", "guilt"],
|
161 |
+
"stress": ["stress", "pressure", "overwhelm", "burden", "strain", "tension", "burnout", "overworked", "deadline", "rush"],
|
162 |
+
"loneliness": ["lonely", "alone", "isolate", "disconnect", "abandoned", "reject", "outcast", "friendless", "solitary", "unloved"],
|
163 |
+
"grief": ["grief", "loss", "death", "miss", "mourn", "gone", "passed away", "bereavement", "widow", "funeral"],
|
164 |
+
"relationship_issues": ["relationship", "partner", "marriage", "divorce", "argument", "fight", "breakup", "separation", "trust", "jealous"],
|
165 |
+
"self_esteem": ["confidence", "worth", "value", "failure", "ugly", "stupid", "incompetent", "loser", "undeserving", "inadequate"],
|
166 |
+
"trauma": ["trauma", "abuse", "assault", "accident", "violence", "nightmare", "flashback", "ptsd", "terrify", "horrific"]
|
167 |
+
}
|
168 |
+
|
169 |
+
# Check user input against keywords
|
170 |
+
user_input_lower = user_input.lower()
|
171 |
+
|
172 |
+
for issue, keywords in mental_health_keywords.items():
|
173 |
+
for keyword in keywords:
|
174 |
+
if keyword in user_input_lower:
|
175 |
+
# Increase score based on sentiment - more negative sentiment means higher score
|
176 |
+
increase = 1 + (1 - sentiment["compound"]) * 0.5
|
177 |
+
if sentiment["compound"] < -0.2: # If negative sentiment
|
178 |
+
increase *= 1.5
|
179 |
+
|
180 |
+
st.session_state.mental_health_scores[issue] += increase
|
181 |
+
|
182 |
+
# LLM-based assessment for more complex understanding
|
183 |
+
if len(user_input) > 15: # Only for substantial responses
|
184 |
+
try:
|
185 |
+
assess_prompt = f"""
|
186 |
+
Analyze this statement for signs of mental health issues. Give a rating from 0-5
|
187 |
+
(0 = not present, 5 = severe) for each of these categories:
|
188 |
+
- Anxiety
|
189 |
+
- Depression
|
190 |
+
- Stress
|
191 |
+
- Loneliness
|
192 |
+
- Grief
|
193 |
+
- Relationship issues
|
194 |
+
- Self-esteem issues
|
195 |
+
- Trauma indicators
|
196 |
+
|
197 |
+
Return ONLY the numerical ratings in JSON format like this:
|
198 |
+
{{
|
199 |
+
"anxiety": X,
|
200 |
+
"depression": X,
|
201 |
+
"stress": X,
|
202 |
+
"loneliness": X,
|
203 |
+
"grief": X,
|
204 |
+
"relationship_issues": X,
|
205 |
+
"self_esteem": X,
|
206 |
+
"trauma": X
|
207 |
+
}}
|
208 |
+
|
209 |
+
The statement: "{user_input}"
|
210 |
+
"""
|
211 |
+
|
212 |
+
llm_assessment = use_llama3_api(assess_prompt, max_tokens=100)
|
213 |
+
|
214 |
+
# Extract the JSON part
|
215 |
+
try:
|
216 |
+
json_start = llm_assessment.find('{')
|
217 |
+
json_end = llm_assessment.rfind('}') + 1
|
218 |
+
if json_start >= 0 and json_end > json_start:
|
219 |
+
json_str = llm_assessment[json_start:json_end]
|
220 |
+
llm_scores = json.loads(json_str)
|
221 |
+
|
222 |
+
# Update our scores with LLM insights (give them less weight than keyword matches)
|
223 |
+
for issue, score in llm_scores.items():
|
224 |
+
if issue in st.session_state.mental_health_scores:
|
225 |
+
# Add 0.4 points for every point in LLM rating
|
226 |
+
st.session_state.mental_health_scores[issue] += score * 0.4
|
227 |
+
except:
|
228 |
+
# If JSON parsing fails, continue without LLM assessment
|
229 |
+
pass
|
230 |
+
except:
|
231 |
+
# If LLM call fails, continue with just keyword matching
|
232 |
+
pass
|
233 |
+
|
234 |
+
# Generate next therapy question and relevant options
|
235 |
+
def generate_next_question_with_options(conversation_history, mental_health_scores):
|
236 |
+
# Create a prompt for determining the next question with options
|
237 |
+
scores_summary = ", ".join([f"{issue}: {score:.1f}" for issue, score in mental_health_scores.items()])
|
238 |
+
|
239 |
+
previous_convo = ""
|
240 |
+
if conversation_history:
|
241 |
+
previous_convo = "\nPrevious conversation: " + " ".join([f"{'User: ' if i%2==1 else 'Therapist: '}{msg}" for i, msg in enumerate(conversation_history)])
|
242 |
+
|
243 |
+
prompt = f"""Act as a supportive mental health therapist. You're having a conversation with someone seeking help.
|
244 |
+
Current mental health indicators: {scores_summary}
|
245 |
+
{previous_convo}
|
246 |
+
|
247 |
+
Based on this information, what's the most important therapeutic question to ask next?
|
248 |
+
Also provide 5 likely response options the person might give.
|
249 |
+
|
250 |
+
Format your response as a JSON object like this:
|
251 |
+
{{
|
252 |
+
"question": "Your therapeutic question here?",
|
253 |
+
"options": [
|
254 |
+
"Possible response 1",
|
255 |
+
"Possible response 2",
|
256 |
+
"Possible response 3",
|
257 |
+
"Possible response 4",
|
258 |
+
"Possible response 5"
|
259 |
+
]
|
260 |
+
}}
|
261 |
+
|
262 |
+
Ensure your question is empathetic, supportive, and helps explore the person's feelings or situation further.
|
263 |
+
Make the options specific and relevant to potential mental health concerns."""
|
264 |
+
|
265 |
+
try:
|
266 |
+
response = use_llama3_api(prompt, max_tokens=500)
|
267 |
+
|
268 |
+
# Parse the JSON response
|
269 |
+
try:
|
270 |
+
# Find the JSON object within the response
|
271 |
+
json_start = response.find('{')
|
272 |
+
json_end = response.rfind('}') + 1
|
273 |
+
if json_start >= 0 and json_end > json_start:
|
274 |
+
json_str = response[json_start:json_end]
|
275 |
+
result = json.loads(json_str)
|
276 |
+
|
277 |
+
# Ensure proper format
|
278 |
+
if "question" in result and "options" in result:
|
279 |
+
# Always add "Other" option
|
280 |
+
if "Other" not in result["options"]:
|
281 |
+
result["options"].append("Other")
|
282 |
+
return result["question"], result["options"]
|
283 |
+
|
284 |
+
except json.JSONDecodeError:
|
285 |
+
# Default question if JSON parsing fails
|
286 |
+
return "How are you feeling right now?", [
|
287 |
+
"I'm feeling anxious",
|
288 |
+
"I'm feeling sad",
|
289 |
+
"I'm feeling overwhelmed",
|
290 |
+
"I'm feeling okay",
|
291 |
+
"I don't know how I feel",
|
292 |
+
"Other"
|
293 |
+
]
|
294 |
+
|
295 |
+
except Exception as e:
|
296 |
+
st.error(f"Error generating question: {str(e)}")
|
297 |
+
|
298 |
+
# Fallback if everything else fails
|
299 |
+
return "Would you like to tell me more about what's on your mind?", [
|
300 |
+
"Yes, I need to talk",
|
301 |
+
"I'm not sure where to start",
|
302 |
+
"I don't think it will help",
|
303 |
+
"I'm feeling too overwhelmed",
|
304 |
+
"I'd rather listen to advice",
|
305 |
+
"Other"
|
306 |
+
]
|
307 |
+
|
308 |
+
# Generate therapeutic response based on user input
|
309 |
+
def generate_therapeutic_response(user_input, conversation_history, mental_health_scores):
|
310 |
+
# If using Blenderbot for simple conversation
|
311 |
+
if st.session_state.llm_service == "blenderbot":
|
312 |
+
tokenizer, model = load_blenderbot_model()
|
313 |
+
return use_blenderbot(user_input, tokenizer, model)
|
314 |
+
|
315 |
+
# If using LLaMA3 for more sophisticated responses
|
316 |
+
scores_summary = ", ".join([f"{issue}: {score:.1f}" for issue, score in mental_health_scores.items()])
|
317 |
+
|
318 |
+
# Get previous 5 exchanges to maintain context without making prompt too long
|
319 |
+
recent_convo = ""
|
320 |
+
if len(conversation_history) > 0:
|
321 |
+
# Get up to last 10 messages (5 exchanges)
|
322 |
+
recent_messages = conversation_history[-10:] if len(conversation_history) >= 10 else conversation_history
|
323 |
+
recent_convo = "\n".join([f"{'User: ' if i%2==1 else 'Therapist: '}{msg}" for i, msg in enumerate(recent_messages)])
|
324 |
+
|
325 |
+
# Determine highest scoring mental health issues
|
326 |
+
top_issues = sorted(mental_health_scores.items(), key=lambda x: x[1], reverse=True)[:3]
|
327 |
+
top_issues_str = ", ".join([f"{issue}" for issue, score in top_issues if score > 1])
|
328 |
+
focus_areas = f"Potential areas of focus: {top_issues_str}" if top_issues_str else "No clear mental health concerns identified yet."
|
329 |
+
|
330 |
+
prompt = f"""Act as an empathetic, supportive mental health therapist using person-centered therapy approaches.
|
331 |
+
You're having a conversation with someone seeking help.
|
332 |
+
|
333 |
+
Current mental health indicators: {scores_summary}
|
334 |
+
{focus_areas}
|
335 |
+
|
336 |
+
Recent conversation:
|
337 |
+
{recent_convo}
|
338 |
+
|
339 |
+
User's most recent message: "{user_input}"
|
340 |
+
|
341 |
+
Provide a thoughtful, validating response that:
|
342 |
+
1. Shows you understand their feelings
|
343 |
+
2. Offers support without judgment
|
344 |
+
3. Asks open-ended questions to explore their concerns deeper
|
345 |
+
4. Avoids giving simplistic advice or dismissing feelings
|
346 |
+
5. Uses techniques like reflective listening and validation
|
347 |
+
|
348 |
+
Keep your response conversational, warm and natural - like a supportive friend would talk.
|
349 |
+
Limit your response to 3-4 sentences to maintain engagement."""
|
350 |
+
|
351 |
+
try:
|
352 |
+
return use_llama3_api(prompt, max_tokens=250)
|
353 |
+
except Exception as e:
|
354 |
+
st.error(f"Error generating response: {str(e)}")
|
355 |
+
return "I'm here to listen. Would you like to tell me more about what you're experiencing?"
|
356 |
+
|
357 |
+
# Determine if assessment is complete
|
358 |
+
def assessment_complete(mental_health_scores, conversation_length):
|
359 |
+
# Check if we have enough information to provide a summary
|
360 |
+
# Criteria: At least 5 exchanges and some significant scores
|
361 |
+
|
362 |
+
# Count significant issues (score > 2)
|
363 |
+
significant_issues = sum(1 for score in mental_health_scores.values() if score > 2)
|
364 |
+
|
365 |
+
# Complete assessment if we have:
|
366 |
+
# - At least 5 conversation exchanges AND some significant issues identified
|
367 |
+
# - OR at least 10 exchanges (regardless of issues identified)
|
368 |
+
return (conversation_length >= 10 and significant_issues >= 1) or conversation_length >= 20
|
369 |
+
|
370 |
+
# Generate mental health assessment summary
|
371 |
+
def generate_assessment_summary(conversation_history, mental_health_scores):
|
372 |
+
# Only include scores that are significant
|
373 |
+
significant_scores = {issue: score for issue, score in mental_health_scores.items() if score > 1}
|
374 |
+
|
375 |
+
# Sort by score (highest first)
|
376 |
+
sorted_scores = sorted(significant_scores.items(), key=lambda x: x[1], reverse=True)
|
377 |
+
|
378 |
+
# Create a text representation of the scores
|
379 |
+
scores_text = ""
|
380 |
+
for issue, score in sorted_scores:
|
381 |
+
# Convert numeric score to severity level
|
382 |
+
severity = "mild"
|
383 |
+
if score > 5:
|
384 |
+
severity = "moderate"
|
385 |
+
if score > 8:
|
386 |
+
severity = "significant"
|
387 |
+
if score > 12:
|
388 |
+
severity = "severe"
|
389 |
+
|
390 |
+
formatted_issue = issue.replace("_", " ").title()
|
391 |
+
scores_text += f"- {formatted_issue}: {severity} (score: {score:.1f})\n"
|
392 |
+
|
393 |
+
if not scores_text:
|
394 |
+
scores_text = "- No significant mental health concerns detected\n"
|
395 |
+
|
396 |
+
# Selected excerpts from conversation
|
397 |
+
sentiment_data = st.session_state.sentiment_analysis
|
398 |
+
|
399 |
+
# Get the most negative and most positive statements
|
400 |
+
if sentiment_data:
|
401 |
+
most_negative = min(sentiment_data, key=lambda x: x["sentiment"]["compound"])
|
402 |
+
most_positive = max(sentiment_data, key=lambda x: x["sentiment"]["compound"])
|
403 |
+
|
404 |
+
significant_statements = f"""
|
405 |
+
Most concerning statement: "{most_negative['text']}"
|
406 |
+
Most positive statement: "{most_positive['text']}"
|
407 |
+
"""
|
408 |
+
else:
|
409 |
+
significant_statements = "No significant statements analyzed."
|
410 |
+
|
411 |
+
# Create prompt for generating the assessment
|
412 |
+
prompt = f"""As a mental health professional, create a supportive therapeutic assessment summary
|
413 |
+
based on the following information from a conversation with a client:
|
414 |
+
|
415 |
+
Mental health indicators:
|
416 |
+
{scores_text}
|
417 |
+
|
418 |
+
{significant_statements}
|
419 |
+
|
420 |
+
Create a compassionate assessment summary that includes:
|
421 |
+
1. The primary mental health concerns identified (if any)
|
422 |
+
2. Supportive validation of the person's experiences
|
423 |
+
3. General self-care recommendations
|
424 |
+
4. When professional help would be recommended
|
425 |
+
5. A hopeful message about the possibility of improvement
|
426 |
+
|
427 |
+
Your assessment should be non-judgmental, respectful, and empowering. Focus on the person's
|
428 |
+
strengths as well as challenges. Make it clear this is NOT a clinical diagnosis."""
|
429 |
+
|
430 |
+
try:
|
431 |
+
assessment = use_llama3_api(prompt, max_tokens=500)
|
432 |
+
|
433 |
+
# Determine overall severity rating
|
434 |
+
severity = "Low"
|
435 |
+
highest_score = max(mental_health_scores.values()) if mental_health_scores else 0
|
436 |
+
if highest_score > 8:
|
437 |
+
severity = "High"
|
438 |
+
elif highest_score > 4:
|
439 |
+
severity = "Moderate"
|
440 |
+
|
441 |
+
# Add disclaimer
|
442 |
+
assessment += "\n\n[Note: This is an AI-generated assessment for educational purposes only and should not replace professional mental health advice.]"
|
443 |
+
|
444 |
+
return assessment, severity
|
445 |
+
except Exception as e:
|
446 |
+
st.error(f"Error generating assessment: {str(e)}")
|
447 |
+
return "Unable to generate a complete assessment at this time. Please consider speaking with a mental health professional for personalized support.", "Unknown"
|
448 |
+
|
449 |
+
# Generate resources based on mental health concerns
|
450 |
+
def generate_resources(mental_health_scores):
|
451 |
+
# Identify top 3 concerns
|
452 |
+
top_concerns = sorted(mental_health_scores.items(), key=lambda x: x[1], reverse=True)[:3]
|
453 |
+
top_concerns = [concern for concern, score in top_concerns if score > 1]
|
454 |
+
|
455 |
+
if not top_concerns:
|
456 |
+
top_concerns = ["general_wellbeing"]
|
457 |
+
|
458 |
+
concerns_text = ", ".join(top_concerns)
|
459 |
+
|
460 |
+
prompt = f"""Create a list of helpful resources for someone dealing with these mental health concerns: {concerns_text}.
|
461 |
+
|
462 |
+
Include:
|
463 |
+
1. Three self-help techniques they can try immediately
|
464 |
+
2. Three types of professionals who might help with these concerns
|
465 |
+
3. Two reputable organizations or hotlines that provide support
|
466 |
+
4. Two recommended books or workbooks that address these concerns
|
467 |
+
|
468 |
+
Format your response with clear headings and brief explanations. Focus on practical, evidence-based resources."""
|
469 |
+
|
470 |
+
try:
|
471 |
+
return use_llama3_api(prompt, max_tokens=400)
|
472 |
+
except:
|
473 |
+
# Fallback resources if API fails
|
474 |
+
return """
|
475 |
+
## Helpful Resources
|
476 |
+
|
477 |
+
### Self-help Techniques
|
478 |
+
- Practice deep breathing exercises (4-7-8 method)
|
479 |
+
- Journal about your thoughts and feelings
|
480 |
+
- Engage in regular physical activity
|
481 |
+
|
482 |
+
### Professional Support
|
483 |
+
- Licensed therapists or counselors
|
484 |
+
- Psychiatrists (for medication evaluation)
|
485 |
+
- Support groups for your specific concerns
|
486 |
+
|
487 |
+
### Support Organizations
|
488 |
+
- Crisis Text Line: Text HOME to 741741
|
489 |
+
- National Alliance on Mental Health (NAMI): 1-800-950-NAMI (6264)
|
490 |
+
|
491 |
+
### Recommended Reading
|
492 |
+
- "Feeling Good" by David Burns
|
493 |
+
- "The Anxiety and Phobia Workbook" by Edmund Bourne
|
494 |
+
|
495 |
+
Remember that seeking help is a sign of strength, not weakness.
|
496 |
+
"""
|
497 |
+
|
498 |
+
def display_interactive_summary(mental_health_scores, assessment, resources):
|
499 |
+
"""Display an interactive, visually appealing summary of the therapy session"""
|
500 |
+
|
501 |
+
st.markdown("## Your Wellness Summary")
|
502 |
+
|
503 |
+
# Create tabs for different sections of the report
|
504 |
+
summary_tabs = st.tabs(["Overview", "Insights", "Recommendations", "Resources"])
|
505 |
+
|
506 |
+
with summary_tabs[0]: # Overview tab
|
507 |
+
st.markdown("### How You're Feeling")
|
508 |
+
|
509 |
+
# Create two columns for layout
|
510 |
+
col1, col2 = st.columns([3, 2])
|
511 |
+
|
512 |
+
with col1:
|
513 |
+
# Get top concerns and sort scores for visualization
|
514 |
+
sorted_scores = sorted(mental_health_scores.items(), key=lambda x: x[1], reverse=True)
|
515 |
+
concerns = [item[0].replace("_", " ").title() for item in sorted_scores if item[1] > 1]
|
516 |
+
scores = [item[1] for item in sorted_scores if item[1] > 1]
|
517 |
+
|
518 |
+
# If we have concerns to display
|
519 |
+
if concerns:
|
520 |
+
# Create color scale
|
521 |
+
colors = []
|
522 |
+
for score in scores:
|
523 |
+
if score > 8:
|
524 |
+
colors.append("#FF4B4B") # Red for high scores
|
525 |
+
elif score > 4:
|
526 |
+
colors.append("#FFA64B") # Orange for medium scores
|
527 |
+
else:
|
528 |
+
colors.append("#4B9AFF") # Blue for low scores
|
529 |
+
|
530 |
+
# Create horizontal bar chart
|
531 |
+
chart_data = {
|
532 |
+
"concern": concerns,
|
533 |
+
"score": scores
|
534 |
+
}
|
535 |
+
|
536 |
+
# Use Altair for better visualization if available
|
537 |
+
try:
|
538 |
+
import altair as alt
|
539 |
+
import pandas as pd
|
540 |
+
|
541 |
+
chart_df = pd.DataFrame(chart_data)
|
542 |
+
|
543 |
+
chart = alt.Chart(chart_df).mark_bar().encode(
|
544 |
+
x='score',
|
545 |
+
y=alt.Y('concern', sort='-x'),
|
546 |
+
color=alt.Color('score', scale=alt.Scale(domain=[1, 5, 10], range=['#4B9AFF', '#FFA64B', '#FF4B4B'])),
|
547 |
+
tooltip=['concern', 'score']
|
548 |
+
).properties(
|
549 |
+
title='Mental Health Indicators',
|
550 |
+
height=min(250, len(concerns) * 40)
|
551 |
+
)
|
552 |
+
|
553 |
+
st.altair_chart(chart, use_container_width=True)
|
554 |
+
except:
|
555 |
+
# Fallback to simple bar chart if Altair isn't available
|
556 |
+
st.bar_chart(chart_data, x="concern", y="score", use_container_width=True)
|
557 |
+
else:
|
558 |
+
st.info("No significant mental health concerns were detected.")
|
559 |
+
|
560 |
+
with col2:
|
561 |
+
# Overall wellness status
|
562 |
+
highest_score = max(mental_health_scores.values()) if mental_health_scores else 0
|
563 |
+
|
564 |
+
if highest_score > 8:
|
565 |
+
wellness_status = "Needs Attention"
|
566 |
+
status_color = "#FF4B4B"
|
567 |
+
elif highest_score > 4:
|
568 |
+
wellness_status = "Moderate Concern"
|
569 |
+
status_color = "#FFA64B"
|
570 |
+
else:
|
571 |
+
wellness_status = "Doing Well"
|
572 |
+
status_color = "#4CAF50"
|
573 |
+
|
574 |
+
st.markdown(f"""
|
575 |
+
<div style="padding: 20px; border-radius: 10px; background-color: {status_color}20;
|
576 |
+
border: 1px solid {status_color}; text-align: center; margin-bottom: 20px;">
|
577 |
+
<h3 style="color: {status_color};">Current Status</h3>
|
578 |
+
<h2 style="color: {status_color};">{wellness_status}</h2>
|
579 |
+
</div>
|
580 |
+
""", unsafe_allow_html=True)
|
581 |
+
|
582 |
+
# Quick mood check
|
583 |
+
st.markdown("### How are you feeling right now?")
|
584 |
+
mood = st.select_slider(
|
585 |
+
"My current mood is:",
|
586 |
+
options=["Very Low", "Low", "Neutral", "Good", "Great"],
|
587 |
+
value="Neutral"
|
588 |
+
)
|
589 |
+
|
590 |
+
# Encouragement based on mood
|
591 |
+
if mood in ["Very Low", "Low"]:
|
592 |
+
st.markdown("🌱 It's okay to not be okay. Small steps lead to big changes.")
|
593 |
+
elif mood == "Neutral":
|
594 |
+
st.markdown("✨ You're doing better than you think. Keep going!")
|
595 |
+
else:
|
596 |
+
st.markdown("🌟 That's wonderful! Celebrate your positive moments.")
|
597 |
+
|
598 |
+
with summary_tabs[1]: # Insights tab
|
599 |
+
st.markdown("### Understanding Your Experience")
|
600 |
+
|
601 |
+
# Extract key points from assessment
|
602 |
+
if assessment:
|
603 |
+
# Find primary concerns
|
604 |
+
import re
|
605 |
+
|
606 |
+
# Extract concerns and validation text with improved pattern matching
|
607 |
+
concerns_match = re.search(r"Primary Mental Health Concerns:(.*?)(?:Validation|General Self-Care)", assessment, re.DOTALL)
|
608 |
+
validation_match = re.search(r"Validation of Your Experiences:(.*?)(?:General Self-Care|When Professional)", assessment, re.DOTALL)
|
609 |
+
|
610 |
+
# Clean function to remove asterisks and replace placeholders
|
611 |
+
def clean_text(text):
|
612 |
+
# Remove asterisks
|
613 |
+
text = text.replace("**", "")
|
614 |
+
# Replace placeholders
|
615 |
+
text = text.replace("[Client]", "friend").replace("Dear [Client],", "")
|
616 |
+
text = text.replace("[Your Name]", "Your Well-Wisher")
|
617 |
+
return text.strip()
|
618 |
+
|
619 |
+
if concerns_match:
|
620 |
+
st.markdown("#### Key Insights")
|
621 |
+
concerns_text = clean_text(concerns_match.group(1).strip())
|
622 |
+
st.info(concerns_text)
|
623 |
+
|
624 |
+
if validation_match:
|
625 |
+
st.markdown("#### Reflections")
|
626 |
+
validation_text = clean_text(validation_match.group(1).strip())
|
627 |
+
st.success(validation_text)
|
628 |
+
|
629 |
+
# Allow user to see full assessment if desired
|
630 |
+
with st.expander("See Complete Analysis"):
|
631 |
+
# Clean the full assessment text before displaying
|
632 |
+
cleaned_assessment = clean_text(assessment)
|
633 |
+
# Replace the formal greeting and signature
|
634 |
+
cleaned_assessment = cleaned_assessment.replace("Dear friend,", "")
|
635 |
+
cleaned_assessment = re.sub(r"Sincerely,.*$", "- Your Well-Wisher", cleaned_assessment)
|
636 |
+
st.write(cleaned_assessment)
|
637 |
+
else:
|
638 |
+
st.warning("We couldn't generate a detailed assessment. Please speak with a mental health professional for personalized insights.")
|
639 |
+
|
640 |
+
with summary_tabs[2]: # Recommendations tab
|
641 |
+
st.markdown("### Suggested Next Steps")
|
642 |
+
|
643 |
+
# Create priority recommendations based on top concerns
|
644 |
+
top_concerns = sorted(mental_health_scores.items(), key=lambda x: x[1], reverse=True)[:2]
|
645 |
+
top_concerns = [concern for concern, score in top_concerns if score > 1]
|
646 |
+
|
647 |
+
if not top_concerns:
|
648 |
+
top_concerns = ["general_wellbeing"]
|
649 |
+
|
650 |
+
# Extract recommendations from resources if available
|
651 |
+
recommendations = []
|
652 |
+
self_help_match = re.search(r"Self-Help Techniques(.*?)(?:Professionals Who Can Help|Support)", resources, re.DOTALL) if resources else None
|
653 |
+
|
654 |
+
if self_help_match:
|
655 |
+
techniques = re.findall(r"([\w\s]+):", self_help_match.group(1))
|
656 |
+
recommendations = [t.strip() for t in techniques if t.strip()]
|
657 |
+
|
658 |
+
# Fallback recommendations if none found
|
659 |
+
if not recommendations:
|
660 |
+
recommendations = [
|
661 |
+
"Practice deep breathing exercises",
|
662 |
+
"Connect with supportive friends or family",
|
663 |
+
"Engage in physical activity",
|
664 |
+
"Practice mindfulness meditation"
|
665 |
+
]
|
666 |
+
|
667 |
+
# Display actionable recommendations
|
668 |
+
for i, rec in enumerate(recommendations[:3]):
|
669 |
+
col1, col2 = st.columns([1, 20])
|
670 |
+
with col1:
|
671 |
+
if st.checkbox("", key=f"rec_{i}", value=False):
|
672 |
+
pass
|
673 |
+
with col2:
|
674 |
+
st.markdown(f"**{rec}**")
|
675 |
+
|
676 |
+
# Add custom action
|
677 |
+
st.markdown("#### Add Your Own Action")
|
678 |
+
custom_action = st.text_input("What's one small step you can take today?")
|
679 |
+
if custom_action:
|
680 |
+
st.success(f"Great! Remember to try: {custom_action}")
|
681 |
+
|
682 |
+
# Professional support recommendation based on severity
|
683 |
+
highest_score = max(mental_health_scores.values()) if mental_health_scores else 0
|
684 |
+
|
685 |
+
st.markdown("#### Professional Support")
|
686 |
+
if highest_score > 8:
|
687 |
+
st.warning("Based on our conversation, speaking with a mental health professional could be beneficial.")
|
688 |
+
elif highest_score > 4:
|
689 |
+
st.info("Consider reaching out to a mental health professional if you continue to experience these feelings.")
|
690 |
+
else:
|
691 |
+
st.success("Continue practicing self-care. Reach out to a professional if you notice your symptoms worsening.")
|
692 |
+
|
693 |
+
with summary_tabs[3]: # Resources tab
|
694 |
+
st.markdown("### Helpful Resources")
|
695 |
+
|
696 |
+
# Create toggles for different types of resources
|
697 |
+
resource_types = ["Crisis Support", "Professional Help", "Self-Help Books", "Mobile Apps", "Support Groups"]
|
698 |
+
|
699 |
+
selected_resource = st.radio("What type of resources are you looking for?", resource_types)
|
700 |
+
|
701 |
+
# Emergency resources always visible
|
702 |
+
if selected_resource == "Crisis Support":
|
703 |
+
st.markdown("""
|
704 |
+
#### Immediate Support
|
705 |
+
- **Crisis Text Line**: Text HOME to 741741 (24/7 support)
|
706 |
+
- **National Suicide Prevention Lifeline**: 988 or 1-800-273-8255
|
707 |
+
- **Emergency Services**: Call 911 if you're in immediate danger
|
708 |
+
""")
|
709 |
+
|
710 |
+
elif selected_resource == "Professional Help":
|
711 |
+
st.markdown("""
|
712 |
+
#### Finding a Therapist
|
713 |
+
- **Psychology Today**: Search for therapists in your area
|
714 |
+
- **BetterHelp**: Online therapy platform
|
715 |
+
- **Your insurance provider**: Many insurance plans cover mental health services
|
716 |
+
""")
|
717 |
+
|
718 |
+
st.markdown("#### Types of Mental Health Professionals")
|
719 |
+
professionals = {
|
720 |
+
"Therapist/Counselor": "Provides talk therapy and emotional support",
|
721 |
+
"Psychiatrist": "Can prescribe medication and provide treatment",
|
722 |
+
"Psychologist": "Specializes in psychological testing and therapy"
|
723 |
+
}
|
724 |
+
|
725 |
+
for prof, desc in professionals.items():
|
726 |
+
st.markdown(f"**{prof}**: {desc}")
|
727 |
+
|
728 |
+
elif selected_resource == "Self-Help Books":
|
729 |
+
# Extract book recommendations if available
|
730 |
+
books = []
|
731 |
+
books_match = re.search(r"Recommended (Books|Reading)(.*?)(?:\[Note|\Z)", resources, re.DOTALL) if resources else None
|
732 |
+
|
733 |
+
if books_match:
|
734 |
+
book_text = books_match.group(2)
|
735 |
+
books = re.findall(r'"([^"]+)"', book_text)
|
736 |
+
|
737 |
+
# Fallback books if none found
|
738 |
+
if not books:
|
739 |
+
books = [
|
740 |
+
"Feeling Good by David Burns",
|
741 |
+
"The Anxiety and Phobia Workbook by Edmund Bourne",
|
742 |
+
"Man's Search for Meaning by Viktor Frankl"
|
743 |
+
]
|
744 |
+
|
745 |
+
for book in books:
|
746 |
+
st.markdown(f"- **{book}**")
|
747 |
+
|
748 |
+
elif selected_resource == "Mobile Apps":
|
749 |
+
st.markdown("""
|
750 |
+
#### Helpful Mobile Apps
|
751 |
+
- **Headspace**: Guided meditation and mindfulness exercises
|
752 |
+
- **Calm**: Sleep, meditation and relaxation aid
|
753 |
+
- **Woebot**: AI chatbot for mental health support
|
754 |
+
- **Daylio**: Mood tracking journal
|
755 |
+
- **Breathe2Relax**: Guided breathing exercises
|
756 |
+
""")
|
757 |
+
|
758 |
+
elif selected_resource == "Support Groups":
|
759 |
+
st.markdown("""
|
760 |
+
#### Finding Support Groups
|
761 |
+
- **NAMI**: National Alliance on Mental Illness offers support groups
|
762 |
+
- **Mental Health America**: Provides peer support group resources
|
763 |
+
- **Support Group Central**: Online support groups for various needs
|
764 |
+
|
765 |
+
Remember that connecting with others who understand your experience can be incredibly healing.
|
766 |
+
""")
|
767 |
+
|
768 |
+
# Option to download resources as PDF
|
769 |
+
st.markdown("### Save These Resources")
|
770 |
+
if st.button("Prepare Resources PDF"):
|
771 |
+
st.success("Your personalized resource list has been prepared!")
|
772 |
+
st.markdown("""
|
773 |
+
**Note:** In a full implementation, this would generate a downloadable PDF with
|
774 |
+
all relevant resources customized to the user's needs.
|
775 |
+
""")
|
776 |
+
|
777 |
+
# Final encouragement message
|
778 |
+
st.markdown("---")
|
779 |
+
st.markdown("""
|
780 |
+
### Remember
|
781 |
+
Your mental health journey is unique. Small steps forward still move you in the right direction.
|
782 |
+
Each day is a new opportunity to prioritize your wellbeing.
|
783 |
+
""")
|
784 |
+
|
785 |
+
# Disclaimer
|
786 |
+
st.caption("This summary is for educational purposes only and should not replace professional mental health advice.")
|
787 |
+
|
788 |
+
# Options for next steps
|
789 |
+
st.markdown("### What would you like to do next?")
|
790 |
+
next_steps = st.columns(3)
|
791 |
+
|
792 |
+
with next_steps[0]:
|
793 |
+
if st.button("Continue Talking"):
|
794 |
+
return "continue"
|
795 |
+
|
796 |
+
with next_steps[1]:
|
797 |
+
if st.button("Start New Session"):
|
798 |
+
return "new"
|
799 |
+
|
800 |
+
with next_steps[2]:
|
801 |
+
if st.button("End Session"):
|
802 |
+
return "end"
|
803 |
+
|
804 |
+
return None
|
805 |
+
|
806 |
+
st.title("AI Therapy Assistant")
|
807 |
+
st.markdown("_This is a prototype for educational purposes only and should not be used as a replacement for professional mental health services._")
|
808 |
+
|
809 |
+
def submit_initial_response():
|
810 |
+
if st.session_state.initial_response:
|
811 |
+
# Record the conversation - AI message first, then user
|
812 |
+
st.session_state.conversation.append(initial_greeting)
|
813 |
+
st.session_state.conversation.append(st.session_state.initial_response)
|
814 |
+
|
815 |
+
# Update mental health scores based on initial response
|
816 |
+
update_mental_health_scores(st.session_state.initial_response)
|
817 |
+
|
818 |
+
# Generate first therapeutic response
|
819 |
+
first_response = generate_therapeutic_response(
|
820 |
+
st.session_state.initial_response,
|
821 |
+
st.session_state.conversation,
|
822 |
+
st.session_state.mental_health_scores
|
823 |
+
)
|
824 |
+
|
825 |
+
# Add to conversation
|
826 |
+
st.session_state.conversation.append(first_response)
|
827 |
+
|
828 |
+
# Change state to therapy
|
829 |
+
st.session_state.therapy_state = "therapy"
|
830 |
+
|
831 |
+
# Prepare next question with options
|
832 |
+
question, options = generate_next_question_with_options(
|
833 |
+
st.session_state.conversation,
|
834 |
+
st.session_state.mental_health_scores
|
835 |
+
)
|
836 |
+
|
837 |
+
st.session_state.current_options = options
|
838 |
+
st.session_state.user_submitted = False # Change to false since we don't want auto response
|
839 |
+
|
840 |
+
def handle_custom_submit():
|
841 |
+
if st.session_state.custom_input:
|
842 |
+
# Record the response in conversation
|
843 |
+
st.session_state.conversation.append(st.session_state.custom_input)
|
844 |
+
|
845 |
+
# Update mental health scores
|
846 |
+
update_mental_health_scores(st.session_state.custom_input)
|
847 |
+
|
848 |
+
# Reset other_selected flag
|
849 |
+
st.session_state.other_selected = False
|
850 |
+
|
851 |
+
# Set user_submitted flag
|
852 |
+
st.session_state.user_submitted = True
|
853 |
+
|
854 |
+
# Clear input - avoids duplicate submissions
|
855 |
+
st.session_state.custom_input = ""
|
856 |
+
|
857 |
+
# Side panel for controls
|
858 |
+
with st.sidebar:
|
859 |
+
st.header("Controls")
|
860 |
+
|
861 |
+
# LLM Service Selection
|
862 |
+
st.subheader("AI Model")
|
863 |
+
llm_option = st.radio(
|
864 |
+
"Select AI Model",
|
865 |
+
["LLaMA3-70B (Advanced)", "Blenderbot (Simple Conversation)"],
|
866 |
+
index=0 if st.session_state.llm_service == "llama3" else 1
|
867 |
+
)
|
868 |
+
|
869 |
+
# Update LLM service based on selection
|
870 |
+
if (llm_option == "LLaMA3-70B (Advanced)" and st.session_state.llm_service != "llama3") or \
|
871 |
+
(llm_option == "Blenderbot (Simple Conversation)" and st.session_state.llm_service != "blenderbot"):
|
872 |
+
# Store the new service selection
|
873 |
+
st.session_state.llm_service = "llama3" if llm_option == "LLaMA3-70B (Advanced)" else "blenderbot"
|
874 |
+
|
875 |
+
# Reset conversation state
|
876 |
+
st.session_state.conversation = []
|
877 |
+
st.session_state.therapy_state = "initial"
|
878 |
+
st.session_state.mental_health_scores = {
|
879 |
+
"anxiety": 0,
|
880 |
+
"depression": 0,
|
881 |
+
"stress": 0,
|
882 |
+
"loneliness": 0,
|
883 |
+
"grief": 0,
|
884 |
+
"relationship_issues": 0,
|
885 |
+
"self_esteem": 0,
|
886 |
+
"trauma": 0
|
887 |
+
}
|
888 |
+
st.session_state.assessment_progress = 0
|
889 |
+
st.session_state.current_question = None
|
890 |
+
st.session_state.current_options = None
|
891 |
+
st.session_state.user_submitted = False
|
892 |
+
st.session_state.severity_rating = None
|
893 |
+
st.session_state.sentiment_analysis = []
|
894 |
+
st.session_state.other_selected = False
|
895 |
+
|
896 |
+
# Show notification
|
897 |
+
st.success(f"Switched to {llm_option}. Starting new conversation.")
|
898 |
+
st.rerun()
|
899 |
+
|
900 |
+
# Display mental health scores
|
901 |
+
if st.session_state.mental_health_scores:
|
902 |
+
st.subheader("Mental Health Indicators")
|
903 |
+
for issue, score in sorted(st.session_state.mental_health_scores.items(), key=lambda x: x[1], reverse=True):
|
904 |
+
# Only show scores with some significance
|
905 |
+
if score > 0.5:
|
906 |
+
# Format the issue name for display
|
907 |
+
display_name = issue.replace("_", " ").title()
|
908 |
+
# Create color gradient based on score
|
909 |
+
color_intensity = min(score / 15, 1.0) # Max at 15
|
910 |
+
color = f"rgba(255, {int(255*(1-color_intensity))}, {int(255*(1-color_intensity))}, 0.8)"
|
911 |
+
st.markdown(
|
912 |
+
f"""<div style="background-color: {color}; padding: 5px; border-radius: 5px;">
|
913 |
+
{display_name}: {score:.1f}</div>""",
|
914 |
+
unsafe_allow_html=True
|
915 |
+
)
|
916 |
+
|
917 |
+
# Settings section
|
918 |
+
st.subheader("Settings")
|
919 |
+
delay_option = st.checkbox("Simulate therapist typing delay", value=st.session_state.response_delay)
|
920 |
+
if delay_option != st.session_state.response_delay:
|
921 |
+
st.session_state.response_delay = delay_option
|
922 |
+
|
923 |
+
# Reset conversation button
|
924 |
+
st.subheader("Session")
|
925 |
+
if st.button("Start New Conversation"):
|
926 |
+
st.session_state.conversation = []
|
927 |
+
st.session_state.therapy_state = "initial"
|
928 |
+
st.session_state.mental_health_scores = {
|
929 |
+
"anxiety": 0,
|
930 |
+
"depression": 0,
|
931 |
+
"stress": 0,
|
932 |
+
"loneliness": 0,
|
933 |
+
"grief": 0,
|
934 |
+
"relationship_issues": 0,
|
935 |
+
"self_esteem": 0,
|
936 |
+
"trauma": 0
|
937 |
+
}
|
938 |
+
st.session_state.assessment_progress = 0
|
939 |
+
st.session_state.current_question = None
|
940 |
+
st.session_state.current_options = None
|
941 |
+
st.session_state.user_submitted = False
|
942 |
+
st.session_state.severity_rating = None
|
943 |
+
st.session_state.sentiment_analysis = []
|
944 |
+
st.session_state.other_selected = False
|
945 |
+
st.rerun()
|
946 |
+
|
947 |
+
# Main chat interface
|
948 |
+
st.header("Therapeutic Conversation")
|
949 |
+
|
950 |
+
# Display selected model
|
951 |
+
st.caption(f"Using {'LLaMA3-70B' if st.session_state.llm_service == 'llama3' else 'Blenderbot'} for conversation")
|
952 |
+
|
953 |
+
# Chat container for better styling
|
954 |
+
chat_container = st.container()
|
955 |
+
|
956 |
+
# Display conversation history
|
957 |
+
with chat_container:
|
958 |
+
if st.session_state.conversation:
|
959 |
+
for i, message in enumerate(st.session_state.conversation):
|
960 |
+
if i % 2 == 1: # User messages (odd indices)
|
961 |
+
with st.chat_message("user"):
|
962 |
+
st.write(message)
|
963 |
+
else: # AI messages (even indices)
|
964 |
+
with st.chat_message("assistant", avatar="🧠"):
|
965 |
+
st.write(message)
|
966 |
+
else:
|
967 |
+
st.write("No conversation history yet.")
|
968 |
+
|
969 |
+
# Check for end session state first - this is a new state we'll add
|
970 |
+
if hasattr(st.session_state, 'therapy_state') and st.session_state.therapy_state == "ended":
|
971 |
+
# When the session is ended, we don't need to display any additional UI elements
|
972 |
+
# The thank you message should already be in the conversation history
|
973 |
+
pass
|
974 |
+
|
975 |
+
# Initial greeting - only show if conversation is empty
|
976 |
+
elif st.session_state.therapy_state == "initial":
|
977 |
+
initial_greeting = "Hello, I'm here to provide a safe space for you to talk. How are you feeling today?"
|
978 |
+
|
979 |
+
with chat_container:
|
980 |
+
with st.chat_message("assistant", avatar="🧠"):
|
981 |
+
st.write(initial_greeting)
|
982 |
+
|
983 |
+
# Add an input field for initial response
|
984 |
+
with st.form(key="initial_response_form"):
|
985 |
+
st.text_area("Share how you're feeling:", key="initial_response", height=100)
|
986 |
+
st.form_submit_button("Send", on_click=submit_initial_response)
|
987 |
+
|
988 |
+
# Therapeutic conversation phase
|
989 |
+
elif st.session_state.therapy_state == "therapy":
|
990 |
+
# Check if we should provide an assessment
|
991 |
+
if assessment_complete(st.session_state.mental_health_scores, len(st.session_state.conversation)):
|
992 |
+
with st.spinner("Preparing assessment..."):
|
993 |
+
assessment, severity = generate_assessment_summary(
|
994 |
+
st.session_state.conversation,
|
995 |
+
st.session_state.mental_health_scores
|
996 |
+
)
|
997 |
+
|
998 |
+
# Generate helpful resources
|
999 |
+
resources = generate_resources(st.session_state.mental_health_scores)
|
1000 |
+
|
1001 |
+
# Store results
|
1002 |
+
st.session_state.assessment = assessment
|
1003 |
+
st.session_state.resources = resources
|
1004 |
+
st.session_state.severity_rating = severity
|
1005 |
+
st.session_state.therapy_state = "summary"
|
1006 |
+
|
1007 |
+
# Add transition message to conversation
|
1008 |
+
transition_message = "I've had a chance to reflect on our conversation. Would you like to see a summary of what I'm hearing from you, along with some resources that might be helpful?"
|
1009 |
+
st.session_state.conversation.append(transition_message)
|
1010 |
+
|
1011 |
+
# Rerun to show the new state
|
1012 |
+
st.rerun()
|
1013 |
+
|
1014 |
+
# Process user input and generate response
|
1015 |
+
if st.session_state.user_submitted:
|
1016 |
+
# Generate therapeutic response
|
1017 |
+
last_user_message = st.session_state.conversation[-1]
|
1018 |
+
|
1019 |
+
with st.spinner("Thinking..."):
|
1020 |
+
# Apply optional delay to simulate typing
|
1021 |
+
if st.session_state.response_delay:
|
1022 |
+
time.sleep(1.5)
|
1023 |
+
|
1024 |
+
# Generate response
|
1025 |
+
response = generate_therapeutic_response(
|
1026 |
+
last_user_message,
|
1027 |
+
st.session_state.conversation,
|
1028 |
+
st.session_state.mental_health_scores
|
1029 |
+
)
|
1030 |
+
|
1031 |
+
# Add to conversation
|
1032 |
+
st.session_state.conversation.append(response)
|
1033 |
+
|
1034 |
+
# Prepare next question with options
|
1035 |
+
question, options = generate_next_question_with_options(
|
1036 |
+
st.session_state.conversation,
|
1037 |
+
st.session_state.mental_health_scores
|
1038 |
+
)
|
1039 |
+
|
1040 |
+
st.session_state.current_options = options
|
1041 |
+
|
1042 |
+
# Reset user_submitted flag
|
1043 |
+
st.session_state.user_submitted = False
|
1044 |
+
|
1045 |
+
# Just rerun to refresh the page with the new conversation state
|
1046 |
+
st.rerun()
|
1047 |
+
|
1048 |
+
# Show free-form input or options for user
|
1049 |
+
if not st.session_state.user_submitted and not st.session_state.other_selected:
|
1050 |
+
# First, show options if we have them
|
1051 |
+
if st.session_state.current_options:
|
1052 |
+
options_container = st.container()
|
1053 |
+
with options_container:
|
1054 |
+
st.radio(
|
1055 |
+
"Quick responses:",
|
1056 |
+
st.session_state.current_options,
|
1057 |
+
key="selected_option"
|
1058 |
+
)
|
1059 |
+
|
1060 |
+
cols = st.columns([1, 1])
|
1061 |
+
with cols[0]:
|
1062 |
+
if st.button("Send Quick Response"):
|
1063 |
+
if "selected_option" in st.session_state and st.session_state.selected_option:
|
1064 |
+
if st.session_state.selected_option == "Other":
|
1065 |
+
st.session_state.other_selected = True
|
1066 |
+
else:
|
1067 |
+
# Record the response in conversation
|
1068 |
+
st.session_state.conversation.append(st.session_state.selected_option)
|
1069 |
+
|
1070 |
+
# Update mental health scores
|
1071 |
+
update_mental_health_scores(st.session_state.selected_option)
|
1072 |
+
|
1073 |
+
st.session_state.user_submitted = True
|
1074 |
+
st.rerun()
|
1075 |
+
|
1076 |
+
with cols[1]:
|
1077 |
+
if st.button("I'd prefer to type my response"):
|
1078 |
+
st.session_state.other_selected = True
|
1079 |
+
st.rerun()
|
1080 |
+
|
1081 |
+
# Show free-form text input if "Other" is selected or user prefers typing
|
1082 |
+
if st.session_state.other_selected:
|
1083 |
+
with st.form(key="custom_response_form"):
|
1084 |
+
st.text_area("Your response:", key="custom_input", height=100)
|
1085 |
+
st.form_submit_button("Send", on_click=handle_custom_submit)
|
1086 |
+
|
1087 |
+
# Summary and resources phase
|
1088 |
+
elif st.session_state.therapy_state == "summary":
|
1089 |
+
# Define summary choice submission handler
|
1090 |
+
def submit_summary_choice():
|
1091 |
+
if st.session_state.summary_choice:
|
1092 |
+
# Add user response to conversation
|
1093 |
+
st.session_state.conversation.append(st.session_state.summary_choice)
|
1094 |
+
|
1095 |
+
# If user wants to see summary, show it
|
1096 |
+
if "yes" in st.session_state.summary_choice.lower():
|
1097 |
+
# Set flag to display interactive summary instead of text summary
|
1098 |
+
st.session_state.show_interactive_summary = True
|
1099 |
+
else:
|
1100 |
+
# Continue conversation
|
1101 |
+
st.session_state.conversation.append("That's completely fine. We can continue our conversation. What would you like to talk about next?")
|
1102 |
+
st.session_state.therapy_state = "therapy"
|
1103 |
+
|
1104 |
+
# Add a respond to transition message form if not already responded
|
1105 |
+
if len(st.session_state.conversation) % 2 == 1: # Odd number means waiting for user response
|
1106 |
+
with st.form(key="summary_choice_form"):
|
1107 |
+
st.radio(
|
1108 |
+
"Would you like to see a summary and helpful resources?",
|
1109 |
+
["Yes, I'd like to see the summary", "No, I'd prefer to continue talking"],
|
1110 |
+
key="summary_choice"
|
1111 |
+
)
|
1112 |
+
st.form_submit_button("Send", on_click=submit_summary_choice)
|
1113 |
+
|
1114 |
+
# If user has seen summary and we're waiting for their next message
|
1115 |
+
elif len(st.session_state.conversation) % 2 == 0 and len(st.session_state.conversation) >= 2:
|
1116 |
+
# Display interactive summary if flag is set
|
1117 |
+
if hasattr(st.session_state, 'show_interactive_summary') and st.session_state.show_interactive_summary:
|
1118 |
+
# Use the display_interactive_summary function directly here
|
1119 |
+
# instead of adding assessment to conversation first
|
1120 |
+
next_action = display_interactive_summary(
|
1121 |
+
st.session_state.mental_health_scores,
|
1122 |
+
st.session_state.assessment,
|
1123 |
+
st.session_state.resources
|
1124 |
+
)
|
1125 |
+
|
1126 |
+
# Handle the return value from the interactive summary
|
1127 |
+
if next_action == "continue":
|
1128 |
+
# Don't use rerun here - update session state only
|
1129 |
+
st.session_state.therapy_state = "therapy"
|
1130 |
+
st.session_state.conversation.append("Let's continue our conversation. What's on your mind?")
|
1131 |
+
st.session_state.show_interactive_summary = False
|
1132 |
+
elif next_action == "new":
|
1133 |
+
# Reset for new conversation
|
1134 |
+
st.session_state.conversation = []
|
1135 |
+
st.session_state.therapy_state = "initial"
|
1136 |
+
st.session_state.mental_health_scores = {
|
1137 |
+
"anxiety": 0,
|
1138 |
+
"depression": 0,
|
1139 |
+
"stress": 0,
|
1140 |
+
"loneliness": 0,
|
1141 |
+
"grief": 0,
|
1142 |
+
"relationship_issues": 0,
|
1143 |
+
"self_esteem": 0,
|
1144 |
+
"trauma": 0
|
1145 |
+
}
|
1146 |
+
st.session_state.assessment_progress = 0
|
1147 |
+
st.session_state.current_question = None
|
1148 |
+
st.session_state.current_options = None
|
1149 |
+
st.session_state.user_submitted = False
|
1150 |
+
st.session_state.severity_rating = None
|
1151 |
+
st.session_state.sentiment_analysis = []
|
1152 |
+
st.session_state.other_selected = False
|
1153 |
+
st.session_state.show_interactive_summary = False
|
1154 |
+
elif next_action == "end":
|
1155 |
+
# Add ending message and set state to "ended"
|
1156 |
+
st.session_state.conversation.append("Thank you for talking with me today. Remember that this is just a simulation for educational purposes. If you're experiencing mental health challenges, please consider reaching out to a professional. Take care of yourself.")
|
1157 |
+
st.session_state.therapy_state = "ended" # New state for ended sessions
|
1158 |
+
st.session_state.show_interactive_summary = False
|
1159 |
+
st.rerun() # Rerun to refresh UI
|
1160 |
+
else:
|
1161 |
+
# This block is for after the interactive summary has been shown
|
1162 |
+
# or if the user chose to see text summary instead
|
1163 |
+
|
1164 |
+
# First check if we need to add the text assessment to conversation
|
1165 |
+
# (only do this if interactive summary has been shown and closed)
|
1166 |
+
if hasattr(st.session_state, 'show_interactive_summary') and not st.session_state.show_interactive_summary and not any(st.session_state.assessment in msg for msg in st.session_state.conversation):
|
1167 |
+
st.session_state.conversation.append(st.session_state.assessment + "\n\n" + st.session_state.resources)
|
1168 |
+
|
1169 |
+
# After showing summary, offer options
|
1170 |
+
final_options = [
|
1171 |
+
"I'd like to continue our conversation",
|
1172 |
+
"I found this helpful, thank you",
|
1173 |
+
"I'd like to start a new conversation",
|
1174 |
+
"I'd like to learn more about specific resources",
|
1175 |
+
"Show me an interactive summary" # Added option for interactive summary
|
1176 |
+
]
|
1177 |
+
|
1178 |
+
def handle_final_choice():
|
1179 |
+
if st.session_state.final_choice:
|
1180 |
+
# Add user choice to conversation
|
1181 |
+
st.session_state.conversation.append(st.session_state.final_choice)
|
1182 |
+
|
1183 |
+
if "continue" in st.session_state.final_choice.lower():
|
1184 |
+
# Return to therapy state
|
1185 |
+
st.session_state.therapy_state = "therapy"
|
1186 |
+
st.session_state.conversation.append("I'm here to continue our conversation. What's on your mind?")
|
1187 |
+
elif "new" in st.session_state.final_choice.lower():
|
1188 |
+
# Reset for new conversation
|
1189 |
+
st.session_state.conversation = []
|
1190 |
+
st.session_state.therapy_state = "initial"
|
1191 |
+
st.session_state.mental_health_scores = {
|
1192 |
+
"anxiety": 0,
|
1193 |
+
"depression": 0,
|
1194 |
+
"stress": 0,
|
1195 |
+
"loneliness": 0,
|
1196 |
+
"grief": 0,
|
1197 |
+
"relationship_issues": 0,
|
1198 |
+
"self_esteem": 0,
|
1199 |
+
"trauma": 0
|
1200 |
+
}
|
1201 |
+
st.session_state.assessment_progress = 0
|
1202 |
+
st.session_state.current_question = None
|
1203 |
+
st.session_state.current_options = None
|
1204 |
+
st.session_state.user_submitted = False
|
1205 |
+
st.session_state.severity_rating = None
|
1206 |
+
st.session_state.sentiment_analysis = []
|
1207 |
+
st.session_state.other_selected = False
|
1208 |
+
st.rerun()
|
1209 |
+
|
1210 |
+
elif "resources" in st.session_state.final_choice.lower():
|
1211 |
+
# Generate more specific resources
|
1212 |
+
with st.spinner("Finding more specific resources..."):
|
1213 |
+
detailed_resources = use_llama3_api(
|
1214 |
+
"Provide a detailed list of mental health resources including specific apps, websites, hotlines, and books. Include resources for both immediate crisis and long-term support.",
|
1215 |
+
max_tokens=600
|
1216 |
+
)
|
1217 |
+
st.session_state.conversation.append("Here are some more detailed resources that might be helpful:\n\n" + detailed_resources)
|
1218 |
+
elif "interactive summary" in st.session_state.final_choice.lower():
|
1219 |
+
# Show interactive summary
|
1220 |
+
st.session_state.show_interactive_summary = True
|
1221 |
+
elif 'end' in st.session_state.final_choice.lower() or "thank" in st.session_state.final_choice.lower():
|
1222 |
+
# Add ending message and set state to "ended"
|
1223 |
+
st.session_state.conversation.append("Thank you for talking with me today. Remember that this is just a simulation for educational purposes. If you're experiencing mental health challenges, please consider reaching out to a professional. Take care of yourself.")
|
1224 |
+
st.session_state.therapy_state = "ended" # New state for ended sessions
|
1225 |
+
st.rerun() # Rerun to refresh UI
|
1226 |
+
|
1227 |
+
with st.form(key="final_choice_form"):
|
1228 |
+
st.radio(
|
1229 |
+
"What would you like to do next?",
|
1230 |
+
final_options,
|
1231 |
+
key="final_choice"
|
1232 |
+
)
|
1233 |
+
st.form_submit_button("Send", on_click=handle_final_choice)
|
1234 |
+
|
1235 |
+
# Add End Session button
|
1236 |
+
if st.form_submit_button("End Session"):
|
1237 |
+
# Add ending message and set state to "ended"
|
1238 |
+
st.session_state.conversation.append("Thank you for talking with me today. Remember that this is just a simulation for educational purposes. If you're experiencing mental health challenges, please consider reaching out to a professional. Take care of yourself.")
|
1239 |
+
st.session_state.therapy_state = "ended" # New state for ended sessions
|
1240 |
+
st.rerun() # Rerun to refresh UI
|
1241 |
+
|
1242 |
+
# Footer with disclaimer
|
1243 |
+
st.markdown("---")
|
1244 |
+
st.caption("**IMPORTANT DISCLAIMER:** This is an educational prototype only and should not be used for actual mental health support. The AI models used have limitations and may not provide accurate or appropriate responses. If you're experiencing mental health issues, please contact a qualified healthcare professional or a crisis service such as the National Suicide Prevention Lifeline at 988 or 1-800-273-8255.")
|
1245 |
+
|
1246 |
+
# Back button
|
1247 |
+
if st.button("Back to Main Menu"):
|
1248 |
+
st.session_state.page = "main"
|
1249 |
+
st.rerun()
|
1250 |
+
|
1251 |
+
# Run the application
|
1252 |
+
if __name__ == "__main__":
|
1253 |
+
run_mental_health()
|
physical_health.py
ADDED
@@ -0,0 +1,1253 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import streamlit as st
|
2 |
+
import tempfile
|
3 |
+
import os
|
4 |
+
import librosa
|
5 |
+
from transformers import pipeline
|
6 |
+
from dotenv import load_dotenv
|
7 |
+
import json
|
8 |
+
import requests
|
9 |
+
import anthropic
|
10 |
+
|
11 |
+
# Load environment variables
|
12 |
+
load_dotenv()
|
13 |
+
hf_token = os.getenv("HUGGINGFACE_API_KEY")
|
14 |
+
groq_api_key = os.getenv("GROQ_API_KEY")
|
15 |
+
anthropic_api_key = os.getenv("ANTHROPIC_API_KEY")
|
16 |
+
|
17 |
+
def run_physical_health():
|
18 |
+
# Initialize session state variables
|
19 |
+
if 'conversation' not in st.session_state:
|
20 |
+
st.session_state.conversation = []
|
21 |
+
if 'diagnostic_state' not in st.session_state:
|
22 |
+
st.session_state.diagnostic_state = "initial" # States: initial, gathering, requesting_audio, diagnosing, complete
|
23 |
+
if 'gathered_symptoms' not in st.session_state:
|
24 |
+
st.session_state.gathered_symptoms = []
|
25 |
+
if 'audio_analysis' not in st.session_state:
|
26 |
+
st.session_state.audio_analysis = None
|
27 |
+
if 'current_question' not in st.session_state:
|
28 |
+
st.session_state.current_question = None
|
29 |
+
if 'current_options' not in st.session_state:
|
30 |
+
st.session_state.current_options = None
|
31 |
+
if 'llm_service' not in st.session_state:
|
32 |
+
st.session_state.llm_service = "groq" # Default LLM service
|
33 |
+
if 'user_submitted' not in st.session_state:
|
34 |
+
st.session_state.user_submitted = False
|
35 |
+
if 'severity_rating' not in st.session_state:
|
36 |
+
st.session_state.severity_rating = None
|
37 |
+
if 'audio_processed' not in st.session_state:
|
38 |
+
st.session_state.audio_processed = False
|
39 |
+
if 'waiting_for_next_question' not in st.session_state:
|
40 |
+
st.session_state.waiting_for_next_question = False
|
41 |
+
if 'condition_evidence' not in st.session_state:
|
42 |
+
st.session_state.condition_evidence = {}
|
43 |
+
if 'other_selected' not in st.session_state:
|
44 |
+
st.session_state.other_selected = False
|
45 |
+
|
46 |
+
# Callback to handle option selection
|
47 |
+
def handle_option_select():
|
48 |
+
selected_option = st.session_state.selected_option
|
49 |
+
if selected_option:
|
50 |
+
# Add selected option to conversation
|
51 |
+
if selected_option == "Other":
|
52 |
+
st.session_state.other_selected = True
|
53 |
+
else:
|
54 |
+
# Record the response in conversation
|
55 |
+
st.session_state.conversation.append(selected_option)
|
56 |
+
st.session_state.gathered_symptoms.append(selected_option)
|
57 |
+
|
58 |
+
# Update evidence for conditions
|
59 |
+
update_condition_evidence(selected_option)
|
60 |
+
|
61 |
+
st.session_state.user_submitted = True
|
62 |
+
st.session_state.other_selected = False
|
63 |
+
|
64 |
+
# Callback to handle custom input submission
|
65 |
+
def handle_custom_submit():
|
66 |
+
custom_input = st.session_state.custom_input
|
67 |
+
if custom_input:
|
68 |
+
# Add custom input to conversation
|
69 |
+
st.session_state.conversation.append(custom_input)
|
70 |
+
st.session_state.gathered_symptoms.append(custom_input)
|
71 |
+
st.session_state.user_submitted = True
|
72 |
+
st.session_state.custom_input = "" # Clear the input field
|
73 |
+
st.session_state.other_selected = False
|
74 |
+
|
75 |
+
# Function to update evidence scores for different conditions
|
76 |
+
def update_condition_evidence(selected_option):
|
77 |
+
# This serves as a starting point but won't limit the conditions
|
78 |
+
if not hasattr(st.session_state, 'condition_symptoms'):
|
79 |
+
st.session_state.condition_symptoms = {
|
80 |
+
"Upper Respiratory Infection": [
|
81 |
+
"runny nose", "congestion", "sore throat", "sneezing",
|
82 |
+
"mild fever", "cough", "headache", "nasal", "sinus"
|
83 |
+
],
|
84 |
+
"Bronchitis": [
|
85 |
+
"persistent cough", "chest tightness", "shortness of breath",
|
86 |
+
"wheezing", "fatigue", "yellow", "green", "sputum", "phlegm"
|
87 |
+
],
|
88 |
+
"Pneumonia": [
|
89 |
+
"high fever", "severe cough", "difficulty breathing", "chest pain",
|
90 |
+
"rapid breathing", "rust colored", "blood", "sputum", "chills"
|
91 |
+
],
|
92 |
+
"Asthma": [
|
93 |
+
"wheezing", "shortness of breath", "chest tightness",
|
94 |
+
"coughing", "difficulty sleeping", "allergies", "exercise"
|
95 |
+
],
|
96 |
+
"GERD": [
|
97 |
+
"heartburn", "regurgitation", "chest pain", "sour taste",
|
98 |
+
"difficulty swallowing", "night cough", "hoarseness", "throat clearing"
|
99 |
+
],
|
100 |
+
"Allergies": [
|
101 |
+
"sneezing", "itchy eyes", "runny nose", "congestion",
|
102 |
+
"itchy throat", "seasonal", "pet", "food", "rash"
|
103 |
+
],
|
104 |
+
"Osteoarthritis": [
|
105 |
+
"joint pain", "stiffness", "swelling", "reduced mobility", "morning stiffness",
|
106 |
+
"knee", "hip", "joint", "age", "older", "construction", "physical labor",
|
107 |
+
"creaking", "grinding", "warmth", "stairs", "walking"
|
108 |
+
],
|
109 |
+
"Rheumatoid Arthritis": [
|
110 |
+
"joint pain", "symmetrical", "multiple joints", "morning stiffness",
|
111 |
+
"fatigue", "fever", "swelling", "warmth", "autoimmune"
|
112 |
+
],
|
113 |
+
"Gout": [
|
114 |
+
"sudden pain", "intense pain", "big toe", "red", "swollen", "hot",
|
115 |
+
"tender", "joint", "alcohol", "meat", "seafood"
|
116 |
+
],
|
117 |
+
"Meniscus Tear": [
|
118 |
+
"knee pain", "swelling", "popping", "locking", "giving way",
|
119 |
+
"inability to straighten", "twist", "injury", "sports"
|
120 |
+
],
|
121 |
+
"Tendinitis": [
|
122 |
+
"pain", "tenderness", "mild swelling", "warm", "movement pain",
|
123 |
+
"repetitive motion", "overuse", "tendon", "wrist", "elbow", "shoulder", "knee", "heel"
|
124 |
+
]
|
125 |
+
}
|
126 |
+
|
127 |
+
# Initialize condition evidence if not already done
|
128 |
+
for condition in st.session_state.condition_symptoms:
|
129 |
+
if condition not in st.session_state.condition_evidence:
|
130 |
+
st.session_state.condition_evidence[condition] = 0
|
131 |
+
|
132 |
+
option_lower = selected_option.lower()
|
133 |
+
|
134 |
+
# Track if any matches are found
|
135 |
+
matched = False
|
136 |
+
|
137 |
+
# Check against existing conditions
|
138 |
+
for condition, symptoms in st.session_state.condition_symptoms.items():
|
139 |
+
for symptom in symptoms:
|
140 |
+
if symptom in option_lower:
|
141 |
+
st.session_state.condition_evidence[condition] += 1
|
142 |
+
matched = True
|
143 |
+
|
144 |
+
# Now check for new conditions mentioned directly in the text
|
145 |
+
# List of common medical condition indicators
|
146 |
+
condition_indicators = ["disease", "syndrome", "disorder", "itis", "infection", "condition", "illness"]
|
147 |
+
|
148 |
+
# Check if text contains a likely medical condition name
|
149 |
+
potential_new_conditions = []
|
150 |
+
words = option_lower.replace(",", " ").replace(".", " ").split()
|
151 |
+
|
152 |
+
# Look for condition patterns
|
153 |
+
for i, word in enumerate(words):
|
154 |
+
# Check for disease indicators
|
155 |
+
is_condition = any(indicator in word for indicator in condition_indicators)
|
156 |
+
|
157 |
+
# Check for capitalized words that might be condition names
|
158 |
+
capitalized = i > 0 and words[i][0].isupper() and not words[i-1].endswith(".")
|
159 |
+
|
160 |
+
if is_condition or capitalized:
|
161 |
+
# Extract the potential condition name (include surrounding words for context)
|
162 |
+
start_idx = max(0, i-2)
|
163 |
+
end_idx = min(len(words), i+3)
|
164 |
+
potential_condition = " ".join(words[start_idx:end_idx])
|
165 |
+
potential_new_conditions.append(potential_condition)
|
166 |
+
|
167 |
+
# Also use LLM to extract any mentions of medical conditions
|
168 |
+
if len(selected_option) > 15: # Only for substantial text
|
169 |
+
try:
|
170 |
+
# Use more focused LLM prompt to extract conditions
|
171 |
+
extract_prompt = f"""
|
172 |
+
Extract any specific medical conditions or diseases mentioned in this text.
|
173 |
+
Return ONLY the condition names separated by commas, or "none" if no specific
|
174 |
+
conditions are mentioned: "{selected_option}"
|
175 |
+
"""
|
176 |
+
|
177 |
+
if st.session_state.llm_service == "groq":
|
178 |
+
extracted_conditions = use_groq_api(extract_prompt, max_tokens=50)
|
179 |
+
else:
|
180 |
+
extracted_conditions = use_claude_api(extract_prompt, max_tokens=50)
|
181 |
+
|
182 |
+
# Add these conditions to our potential list
|
183 |
+
if extracted_conditions and "none" not in extracted_conditions.lower():
|
184 |
+
for cond in extracted_conditions.split(","):
|
185 |
+
clean_cond = cond.strip()
|
186 |
+
if clean_cond:
|
187 |
+
potential_new_conditions.append(clean_cond)
|
188 |
+
except:
|
189 |
+
# If there's an error with the API call, continue without it
|
190 |
+
pass
|
191 |
+
|
192 |
+
# Process the potential conditions
|
193 |
+
for potential_condition in set(potential_new_conditions): # Use set to remove duplicates
|
194 |
+
# Clean up the condition name
|
195 |
+
clean_condition = potential_condition.strip()
|
196 |
+
if len(clean_condition) > 3: # Avoid very short terms
|
197 |
+
# Add as a new condition if not already present
|
198 |
+
condition_key = clean_condition.title() # Capitalize for consistency
|
199 |
+
|
200 |
+
if condition_key not in st.session_state.condition_evidence:
|
201 |
+
# Add to evidence with initial score
|
202 |
+
st.session_state.condition_evidence[condition_key] = 1
|
203 |
+
# Create an empty symptom list for this new condition
|
204 |
+
st.session_state.condition_symptoms[condition_key] = []
|
205 |
+
matched = True
|
206 |
+
else:
|
207 |
+
# Increment existing condition score
|
208 |
+
st.session_state.condition_evidence[condition_key] += 1
|
209 |
+
matched = True
|
210 |
+
|
211 |
+
# If no specific condition matched but we have symptoms, try another pass with general matching
|
212 |
+
if not matched and len(selected_option) > 10:
|
213 |
+
for condition, symptoms in st.session_state.condition_symptoms.items():
|
214 |
+
for symptom in symptoms:
|
215 |
+
for word in symptom.split():
|
216 |
+
if len(word) > 3 and word in option_lower:
|
217 |
+
st.session_state.condition_evidence[condition] += 0.5
|
218 |
+
matched = True
|
219 |
+
break
|
220 |
+
if matched:
|
221 |
+
break
|
222 |
+
|
223 |
+
# Check if audio would be helpful based on symptoms
|
224 |
+
def would_audio_help(condition_evidence):
|
225 |
+
# Determine if audio would be helpful based on condition evidence
|
226 |
+
respiratory_conditions = ["Upper Respiratory Infection", "Bronchitis", "Pneumonia", "Asthma"]
|
227 |
+
orthopedic_conditions = ["Osteoarthritis", "Rheumatoid Arthritis", "Gout", "Meniscus Tear", "Tendinitis"]
|
228 |
+
|
229 |
+
# Calculate total evidence for respiratory vs. orthopedic conditions
|
230 |
+
respiratory_evidence = sum(st.session_state.condition_evidence.get(condition, 0) for condition in respiratory_conditions)
|
231 |
+
orthopedic_evidence = sum(st.session_state.condition_evidence.get(condition, 0) for condition in orthopedic_conditions)
|
232 |
+
|
233 |
+
# If there's significantly more evidence for orthopedic issues, audio won't help
|
234 |
+
if orthopedic_evidence > respiratory_evidence + 2:
|
235 |
+
return False
|
236 |
+
|
237 |
+
# If there's any significant evidence for respiratory issues, audio may help
|
238 |
+
if respiratory_evidence > 1:
|
239 |
+
return True
|
240 |
+
|
241 |
+
# Default - if we're not sure what the condition is, audio might help
|
242 |
+
return respiratory_evidence > 0
|
243 |
+
|
244 |
+
# Load audio classifier model
|
245 |
+
@st.cache_resource
|
246 |
+
def load_audio_classifier():
|
247 |
+
# Audio classifier for cough/breathing analysis
|
248 |
+
audio_classifier = pipeline(
|
249 |
+
"audio-classification",
|
250 |
+
model="MIT/ast-finetuned-audioset-10-10-0.4593",
|
251 |
+
token=hf_token
|
252 |
+
)
|
253 |
+
|
254 |
+
return audio_classifier
|
255 |
+
|
256 |
+
# Use Groq API
|
257 |
+
def use_groq_api(prompt, max_tokens=500):
|
258 |
+
headers = {
|
259 |
+
"Authorization": f"Bearer {groq_api_key}",
|
260 |
+
"Content-Type": "application/json"
|
261 |
+
}
|
262 |
+
|
263 |
+
data = {
|
264 |
+
"model": "llama3-70b-8192", # Using LLaMA3 70B model
|
265 |
+
"messages": [{"role": "user", "content": prompt}],
|
266 |
+
"max_tokens": max_tokens,
|
267 |
+
"temperature": 0.3
|
268 |
+
}
|
269 |
+
|
270 |
+
response = requests.post(
|
271 |
+
"https://api.groq.com/openai/v1/chat/completions",
|
272 |
+
headers=headers,
|
273 |
+
json=data
|
274 |
+
)
|
275 |
+
|
276 |
+
if response.status_code == 200:
|
277 |
+
return response.json()["choices"][0]["message"]["content"]
|
278 |
+
else:
|
279 |
+
st.error(f"Error from Groq API: {response.text}")
|
280 |
+
return "Error communicating with the diagnostic model."
|
281 |
+
|
282 |
+
# Use Claude API function
|
283 |
+
def use_claude_api(prompt, max_tokens=1000):
|
284 |
+
client = anthropic.Client(api_key=anthropic_api_key)
|
285 |
+
|
286 |
+
try:
|
287 |
+
response = client.messages.create(
|
288 |
+
model="claude-3-opus-20240229",
|
289 |
+
max_tokens=max_tokens,
|
290 |
+
temperature=0.3,
|
291 |
+
system="You are a medical expert providing diagnostic assistance. Focus on identifying potential conditions based on symptoms and providing evidence-based recommendations.",
|
292 |
+
messages=[
|
293 |
+
{"role": "user", "content": prompt}
|
294 |
+
]
|
295 |
+
)
|
296 |
+
return response.content[0].text
|
297 |
+
except Exception as e:
|
298 |
+
st.error(f"Error from Claude API: {str(e)}")
|
299 |
+
return "Error communicating with the Claude model."
|
300 |
+
|
301 |
+
# Function to analyze audio
|
302 |
+
def analyze_audio(audio_file, audio_classifier):
|
303 |
+
with tempfile.NamedTemporaryFile(delete=False, suffix=".wav") as tmp:
|
304 |
+
tmp.write(audio_file.getvalue())
|
305 |
+
tmp_path = tmp.name
|
306 |
+
|
307 |
+
try:
|
308 |
+
# Process audio
|
309 |
+
audio, sr = librosa.load(tmp_path, sr=16000)
|
310 |
+
result = audio_classifier(tmp_path)
|
311 |
+
|
312 |
+
# Map audio classifications to potential medical implications
|
313 |
+
medical_context = interpret_audio_results(result)
|
314 |
+
|
315 |
+
os.unlink(tmp_path)
|
316 |
+
return result, medical_context
|
317 |
+
except Exception as e:
|
318 |
+
if os.path.exists(tmp_path):
|
319 |
+
os.unlink(tmp_path)
|
320 |
+
st.error(f"Error analyzing audio: {str(e)}")
|
321 |
+
return None, None
|
322 |
+
|
323 |
+
def interpret_audio_results(audio_results):
|
324 |
+
"""Convert audio classification results to medically relevant information"""
|
325 |
+
medical_interpretations = {
|
326 |
+
"Speech": "Voice patterns may indicate respiratory or neurological conditions.",
|
327 |
+
"Cough": "Cough patterns can indicate respiratory conditions like bronchitis, pneumonia, or COVID-19.",
|
328 |
+
"Wheeze": "Wheezing may indicate asthma, COPD, or bronchitis.",
|
329 |
+
"Breathing": "Breathing patterns may indicate respiratory distress or conditions.",
|
330 |
+
"Snoring": "May indicate sleep apnea or nasal obstruction.",
|
331 |
+
"Sneeze": "May indicate allergies or upper respiratory infections.",
|
332 |
+
"Throat clearing": "May indicate postnasal drip, GERD, or throat irritation.",
|
333 |
+
"Gasping": "May indicate severe respiratory distress or sleep apnea."
|
334 |
+
}
|
335 |
+
|
336 |
+
interpretations = []
|
337 |
+
for result in audio_results[:3]: # Focus on top 3 classifications
|
338 |
+
label = result['label']
|
339 |
+
confidence = result['score']
|
340 |
+
|
341 |
+
# Find relevant medical context for any matching keywords
|
342 |
+
for key, interpretation in medical_interpretations.items():
|
343 |
+
if key.lower() in label.lower():
|
344 |
+
interpretations.append(f"{label} (Confidence: {confidence:.2f}): {interpretation}")
|
345 |
+
break
|
346 |
+
else:
|
347 |
+
# If no specific medical interpretation, provide a generic one
|
348 |
+
interpretations.append(f"{label} (Confidence: {confidence:.2f}): May provide context for diagnosis.")
|
349 |
+
|
350 |
+
return interpretations
|
351 |
+
|
352 |
+
# Generate next question and relevant options based on conversation history
|
353 |
+
def generate_next_question_with_options(conversation_history, gathered_symptoms, audio_context=None):
|
354 |
+
# Combine all information for context
|
355 |
+
audio_info = ""
|
356 |
+
if audio_context:
|
357 |
+
audio_info = "\nAudio analysis detected: " + ", ".join(audio_context)
|
358 |
+
|
359 |
+
symptoms_summary = "\n".join(gathered_symptoms) if gathered_symptoms else "No symptoms reported yet."
|
360 |
+
|
361 |
+
# Create condition evidence summary
|
362 |
+
evidence_summary = ""
|
363 |
+
if st.session_state.condition_evidence:
|
364 |
+
evidence_list = []
|
365 |
+
for condition, score in st.session_state.condition_evidence.items():
|
366 |
+
evidence_list.append(f"{condition}: {score}")
|
367 |
+
evidence_summary = "Condition evidence scores: " + ", ".join(evidence_list)
|
368 |
+
|
369 |
+
# Create a prompt for determining the next question with options
|
370 |
+
prompt = f"""You are an expert medical diagnostic assistant gathering information from a patient.
|
371 |
+
Current patient information:
|
372 |
+
{symptoms_summary}
|
373 |
+
{audio_info}
|
374 |
+
{evidence_summary}
|
375 |
+
|
376 |
+
Previous conversation:
|
377 |
+
{' '.join([f"{'Patient: ' if i%2==0 else 'Doctor: '}{msg}" for i, msg in enumerate(conversation_history)])}
|
378 |
+
|
379 |
+
Based on the information gathered so far, what is the most important follow-up question to ask the patient?
|
380 |
+
Also provide 4-5 most likely answer options based on potential conditions.
|
381 |
+
|
382 |
+
Format your response as a JSON object with the following structure:
|
383 |
+
{{
|
384 |
+
"question": "Your follow-up question here?",
|
385 |
+
"options": [
|
386 |
+
"Option 1 (specific symptom or detail)",
|
387 |
+
"Option 2 (specific symptom or detail)",
|
388 |
+
"Option 3 (specific symptom or detail)",
|
389 |
+
"Option 4 (specific symptom or detail)",
|
390 |
+
"Option 5 (specific symptom or detail)"
|
391 |
+
]
|
392 |
+
}}
|
393 |
+
|
394 |
+
Ensure options are specific, clinically relevant to the likely conditions, and help distinguish between possible diagnoses."""
|
395 |
+
|
396 |
+
# Get next question using selected API
|
397 |
+
try:
|
398 |
+
if st.session_state.llm_service == "groq":
|
399 |
+
response = use_groq_api(prompt, max_tokens=500)
|
400 |
+
elif st.session_state.llm_service == "claude": # New condition for Claude
|
401 |
+
response = use_claude_api(prompt, max_tokens=500)
|
402 |
+
|
403 |
+
# Parse the JSON response
|
404 |
+
try:
|
405 |
+
# Find the JSON object within the response
|
406 |
+
json_start = response.find('{')
|
407 |
+
json_end = response.rfind('}') + 1
|
408 |
+
if json_start >= 0 and json_end > json_start:
|
409 |
+
json_str = response[json_start:json_end]
|
410 |
+
result = json.loads(json_str)
|
411 |
+
|
412 |
+
# Ensure proper format
|
413 |
+
if "question" in result and "options" in result:
|
414 |
+
# Always add "Other" option
|
415 |
+
if "Other" not in result["options"]:
|
416 |
+
result["options"].append("Other")
|
417 |
+
return result["question"], result["options"]
|
418 |
+
|
419 |
+
# Fallback if JSON parsing fails
|
420 |
+
return "Can you provide more details about your symptoms?", [
|
421 |
+
"Symptoms are getting worse",
|
422 |
+
"Symptoms are about the same",
|
423 |
+
"Symptoms are improving",
|
424 |
+
"New symptoms have appeared",
|
425 |
+
"Other"
|
426 |
+
]
|
427 |
+
except json.JSONDecodeError:
|
428 |
+
# Fallback for JSON parsing errors
|
429 |
+
return "Can you tell me more about your symptoms?", [
|
430 |
+
"Symptoms are mild",
|
431 |
+
"Symptoms are moderate",
|
432 |
+
"Symptoms are severe",
|
433 |
+
"Symptoms come and go",
|
434 |
+
"Other"
|
435 |
+
]
|
436 |
+
except Exception as e:
|
437 |
+
st.error(f"Error generating question: {str(e)}")
|
438 |
+
return "How would you describe your symptoms?", [
|
439 |
+
"Getting better",
|
440 |
+
"Getting worse",
|
441 |
+
"Staying the same",
|
442 |
+
"Fluctuating throughout the day",
|
443 |
+
"Other"
|
444 |
+
]
|
445 |
+
|
446 |
+
# Check if more information is needed
|
447 |
+
def needs_more_information(conversation_history, gathered_symptoms, condition_evidence):
|
448 |
+
# Create a prompt to determine if we need more information
|
449 |
+
evidence_summary = ""
|
450 |
+
if condition_evidence:
|
451 |
+
evidence_list = []
|
452 |
+
for condition, score in condition_evidence.items():
|
453 |
+
evidence_list.append(f"{condition}: {score}")
|
454 |
+
evidence_summary = "Condition evidence scores: " + ", ".join(evidence_list)
|
455 |
+
|
456 |
+
prompt = f"""You are an expert medical diagnostic assistant gathering information from a patient.
|
457 |
+
Current patient information:
|
458 |
+
{' '.join(gathered_symptoms)}
|
459 |
+
{evidence_summary}
|
460 |
+
|
461 |
+
Previous conversation:
|
462 |
+
{' '.join([f"{'Patient: ' if i%2==0 else 'Doctor: '}{msg}" for i, msg in enumerate(conversation_history)])}
|
463 |
+
|
464 |
+
Based on the information gathered so far, is there enough information to make a preliminary diagnosis?
|
465 |
+
Answer with only YES or NO."""
|
466 |
+
|
467 |
+
try:
|
468 |
+
if st.session_state.llm_service == "groq":
|
469 |
+
result = use_groq_api(prompt, max_tokens=10)
|
470 |
+
else:
|
471 |
+
result = use_claude_api(prompt, max_tokens=10)
|
472 |
+
|
473 |
+
# Clean up the response to get just YES or NO
|
474 |
+
result = result.strip().upper()
|
475 |
+
if "NO" in result:
|
476 |
+
return True # Need more information
|
477 |
+
else:
|
478 |
+
return False # Have enough information
|
479 |
+
except Exception as e:
|
480 |
+
st.error(f"Error checking information sufficiency: {str(e)}")
|
481 |
+
return True # Default to needing more information
|
482 |
+
|
483 |
+
# Generate audio request prompt
|
484 |
+
def generate_audio_request():
|
485 |
+
return "To better understand your condition, it would be helpful to analyze your cough or breathing sounds. Could you please upload an audio recording using the upload button in the sidebar? Alternatively, you can skip this step and continue with text-based questions."
|
486 |
+
|
487 |
+
# Add a flag to track if user has declined audio upload
|
488 |
+
if 'audio_declined' not in st.session_state:
|
489 |
+
st.session_state.audio_declined = False
|
490 |
+
|
491 |
+
def extract_conditions_from_diagnosis(diagnosis_text):
|
492 |
+
"""Extract condition names from diagnosis to update our condition evidence"""
|
493 |
+
if not diagnosis_text:
|
494 |
+
return []
|
495 |
+
|
496 |
+
# Create a prompt to extract the conditions
|
497 |
+
prompt = f"""Extract only the medical conditions (diseases, syndromes, disorders)
|
498 |
+
mentioned in this diagnosis. DO NOT include symptoms, signs, or explanatory text.
|
499 |
+
Return ONLY a comma-separated list of legitimate medical condition names without any
|
500 |
+
prefacing text or explanation:
|
501 |
+
|
502 |
+
{diagnosis_text}"""
|
503 |
+
|
504 |
+
try:
|
505 |
+
if st.session_state.llm_service == "groq":
|
506 |
+
result = use_groq_api(prompt, max_tokens=100)
|
507 |
+
else:
|
508 |
+
result = use_claude_api(prompt, max_tokens=100)
|
509 |
+
|
510 |
+
# Clean up the result to remove any explanatory text
|
511 |
+
# Look for the first comma-separated list in the result
|
512 |
+
import re
|
513 |
+
|
514 |
+
# Remove any explanatory phrases or headers
|
515 |
+
cleaned_result = re.sub(r'^.*?([\w\s]+(?:,\s*[\w\s]+)+).*$', r'\1', result, flags=re.DOTALL)
|
516 |
+
|
517 |
+
if cleaned_result == result and "," not in result:
|
518 |
+
# If regex didn't match, try to extract any medical condition looking phrases
|
519 |
+
condition_indicators = ["disease", "syndrome", "infection", "itis", "disorder"]
|
520 |
+
potential_conditions = []
|
521 |
+
|
522 |
+
for line in result.split('\n'):
|
523 |
+
line = line.strip()
|
524 |
+
# Skip lines that are likely explanatory
|
525 |
+
if line.startswith("Here") or line.startswith("These") or ":" in line:
|
526 |
+
continue
|
527 |
+
|
528 |
+
# Check for condition indicators
|
529 |
+
if any(indicator in line.lower() for indicator in condition_indicators):
|
530 |
+
potential_conditions.append(line)
|
531 |
+
# Check for capitalized phrases that might be condition names
|
532 |
+
elif len(line) > 0 and line[0].isupper() and len(line.split()) <= 4:
|
533 |
+
potential_conditions.append(line)
|
534 |
+
|
535 |
+
if potential_conditions:
|
536 |
+
cleaned_result = ", ".join(potential_conditions)
|
537 |
+
else:
|
538 |
+
cleaned_result = result
|
539 |
+
|
540 |
+
# Convert to list of condition names and filter out non-conditions
|
541 |
+
conditions = []
|
542 |
+
common_symptoms = [
|
543 |
+
"pain", "ache", "fever", "cough", "sneeze", "wheeze", "breath",
|
544 |
+
"breathing", "shortness", "fatigue", "tired", "dizzy", "nausea",
|
545 |
+
"vomit", "diarrhea", "constipation", "rash", "itch", "swelling",
|
546 |
+
"tightness", "pressure", "discomfort"
|
547 |
+
]
|
548 |
+
|
549 |
+
for cond in cleaned_result.split(","):
|
550 |
+
cond = cond.strip()
|
551 |
+
# Skip empty strings or very short terms
|
552 |
+
if not cond or len(cond) < 4:
|
553 |
+
continue
|
554 |
+
|
555 |
+
# Skip terms that are clearly symptoms, not conditions
|
556 |
+
if any(symptom in cond.lower() for symptom in common_symptoms):
|
557 |
+
continue
|
558 |
+
|
559 |
+
# Only add if it looks like a condition
|
560 |
+
if len(cond) >= 4:
|
561 |
+
conditions.append(cond)
|
562 |
+
|
563 |
+
return conditions
|
564 |
+
except Exception as e:
|
565 |
+
st.error(f"Error extracting conditions: {str(e)}")
|
566 |
+
# Fallback to simpler approach
|
567 |
+
common_indicators = ["disease", "syndrome", "infection", "itis", "disorder"]
|
568 |
+
words = diagnosis_text.split()
|
569 |
+
conditions = []
|
570 |
+
|
571 |
+
for i, word in enumerate(words):
|
572 |
+
if any(indicator in word.lower() for indicator in common_indicators):
|
573 |
+
start_idx = max(0, i-2)
|
574 |
+
end_idx = min(len(words), i+1)
|
575 |
+
potential_condition = " ".join(words[start_idx:end_idx])
|
576 |
+
conditions.append(potential_condition)
|
577 |
+
|
578 |
+
return conditions
|
579 |
+
|
580 |
+
# Generate diagnosis based on all gathered information using selected API
|
581 |
+
def generate_diagnosis(conversation_history, gathered_symptoms, condition_evidence, audio_context=None):
|
582 |
+
# Combine all information
|
583 |
+
audio_info = ""
|
584 |
+
if audio_context:
|
585 |
+
audio_info = "\nAudio analysis detected: " + ", ".join(audio_context)
|
586 |
+
|
587 |
+
symptoms_summary = "\n".join(gathered_symptoms) if gathered_symptoms else "Limited symptom information available."
|
588 |
+
|
589 |
+
# Create condition evidence summary with all conditions found
|
590 |
+
evidence_summary = ""
|
591 |
+
if condition_evidence:
|
592 |
+
evidence_list = []
|
593 |
+
for condition, score in sorted(condition_evidence.items(), key=lambda x: x[1], reverse=True):
|
594 |
+
evidence_list.append(f"{condition}: {score}")
|
595 |
+
evidence_summary = "Condition evidence scores: " + ", ".join(evidence_list)
|
596 |
+
|
597 |
+
# Create prompt with all potential conditions
|
598 |
+
prompt = f"""Act as an expert medical diagnostic assistant. Based on the following patient information, provide:
|
599 |
+
1. Potential diagnoses with likelihood assessment (high, moderate, or low probability)
|
600 |
+
2. Overall severity rating (High, Moderate, Low) based on the most likely diagnosis
|
601 |
+
3. Recommended next steps based on severity:
|
602 |
+
- If HIGH severity: Urgently recommend medical attention and specify which specialists to see
|
603 |
+
- If MODERATE severity: Recommend medical consultation and provide management tips until seen
|
604 |
+
- If LOW severity: Provide self-care tips and when to seek medical attention if symptoms worsen
|
605 |
+
4. When the patient should seek immediate medical attention
|
606 |
+
|
607 |
+
Patient information:
|
608 |
+
{symptoms_summary}
|
609 |
+
{audio_info}
|
610 |
+
{evidence_summary}
|
611 |
+
|
612 |
+
Conversation history:
|
613 |
+
{' '.join([f"{'Patient: ' if i%2==0 else 'Doctor: '}{msg}" for i, msg in enumerate(conversation_history)])}
|
614 |
+
|
615 |
+
Consider ALL possible relevant conditions, including those not mentioned in the evidence scores.
|
616 |
+
Provide a comprehensive but concise diagnostic assessment and recommendations, clearly indicating the SEVERITY RATING (High, Moderate, or Low) at the beginning of your response:"""
|
617 |
+
|
618 |
+
# Rest of the function remains the same
|
619 |
+
try:
|
620 |
+
if st.session_state.llm_service == "groq":
|
621 |
+
diagnosis = use_groq_api(prompt, max_tokens=500)
|
622 |
+
else: # Claude 3 Opus
|
623 |
+
diagnosis = use_claude_api(prompt, max_tokens=500)
|
624 |
+
|
625 |
+
# Extract severity rating
|
626 |
+
severity = None
|
627 |
+
if "SEVERITY RATING: HIGH" in diagnosis.upper():
|
628 |
+
severity = "High"
|
629 |
+
elif "SEVERITY RATING: MODERATE" in diagnosis.upper():
|
630 |
+
severity = "Moderate"
|
631 |
+
elif "SEVERITY RATING: LOW" in diagnosis.upper():
|
632 |
+
severity = "Low"
|
633 |
+
|
634 |
+
# Add disclaimer
|
635 |
+
diagnosis += "\n\n[Note: This is an AI-generated assessment for testing purposes only and should not replace professional medical advice.]"
|
636 |
+
|
637 |
+
return diagnosis, severity
|
638 |
+
except Exception as e:
|
639 |
+
st.error(f"Error generating diagnosis: {str(e)}")
|
640 |
+
return "Error during diagnostic assessment. Please try again.", None
|
641 |
+
|
642 |
+
def display_interactive_diagnosis(diagnosis_text, severity_rating, condition_evidence):
|
643 |
+
"""Display an interactive, visually appealing summary of the diagnostic assessment using only Streamlit."""
|
644 |
+
|
645 |
+
st.markdown("## Your Diagnostic Assessment")
|
646 |
+
|
647 |
+
# Create tabs for different sections of the report
|
648 |
+
diagnosis_tabs = st.tabs(["Overview", "Conditions", "Recommendations", "Action Steps"])
|
649 |
+
|
650 |
+
with diagnosis_tabs[0]: # Overview tab
|
651 |
+
st.markdown("### Assessment Summary")
|
652 |
+
|
653 |
+
col1, col2 = st.columns([3, 2])
|
654 |
+
|
655 |
+
with col1:
|
656 |
+
st.markdown("#### Key Findings")
|
657 |
+
# Simplify diagnosis_text into a summary
|
658 |
+
summary_length = 100
|
659 |
+
if len(diagnosis_text) > summary_length:
|
660 |
+
summary = diagnosis_text[:summary_length].rsplit(' ', 1)[0] + "..."
|
661 |
+
else:
|
662 |
+
summary = diagnosis_text
|
663 |
+
st.info(f"Summary: {summary}")
|
664 |
+
|
665 |
+
st.markdown("#### Self-Care Recommendations")
|
666 |
+
# Fetch self-care recommendations from LLM
|
667 |
+
rec_prompt = f"Based on this diagnosis: '{diagnosis_text}', provide 3 concise self-care recommendations."
|
668 |
+
if st.session_state.llm_service == "groq":
|
669 |
+
rec_response = use_groq_api(rec_prompt, max_tokens=150)
|
670 |
+
else:
|
671 |
+
rec_response = use_claude_api(rec_prompt, max_tokens=150)
|
672 |
+
recommendations = rec_response.split("\n")[:3]
|
673 |
+
for rec in recommendations:
|
674 |
+
if rec.strip():
|
675 |
+
st.success(f"• {rec.strip()}")
|
676 |
+
|
677 |
+
with col2:
|
678 |
+
# Severity status display (kept simple as per your logic)
|
679 |
+
if severity_rating == "High":
|
680 |
+
action_needed = "Immediate Medical Attention"
|
681 |
+
elif severity_rating == "Moderate":
|
682 |
+
action_needed = "Medical Consultation Recommended"
|
683 |
+
else:
|
684 |
+
action_needed = "Follow Self-Care Guidelines"
|
685 |
+
|
686 |
+
st.markdown(f"### Severity Level: {severity_rating}")
|
687 |
+
st.write(action_needed)
|
688 |
+
|
689 |
+
st.markdown("### How are you feeling now?")
|
690 |
+
current_feeling = st.select_slider(
|
691 |
+
"My current symptoms are:",
|
692 |
+
options=["Much Worse", "Worse", "Same", "Better", "Much Better"],
|
693 |
+
value="Same"
|
694 |
+
)
|
695 |
+
|
696 |
+
if current_feeling in ["Much Worse", "Worse"]:
|
697 |
+
st.write("🚨 Consider seeking immediate medical attention if symptoms are worsening.")
|
698 |
+
elif current_feeling == "Same":
|
699 |
+
st.write("👨⚕️ Follow the recommended care steps and monitor your symptoms.")
|
700 |
+
else:
|
701 |
+
st.write("✅ Great! Continue following recommendations for full recovery.")
|
702 |
+
|
703 |
+
with diagnosis_tabs[1]: # Conditions tab
|
704 |
+
st.markdown("### Potential Conditions")
|
705 |
+
|
706 |
+
if condition_evidence:
|
707 |
+
sorted_conditions = sorted(
|
708 |
+
condition_evidence.items(),
|
709 |
+
key=lambda x: x[1],
|
710 |
+
reverse=True
|
711 |
+
)[:5]
|
712 |
+
for condition, score in sorted_conditions:
|
713 |
+
if score > 0:
|
714 |
+
percentage = min(score / 5 * 100, 100)
|
715 |
+
st.write(f"**{condition}**: Probability Score: {score}")
|
716 |
+
st.progress(percentage / 100)
|
717 |
+
|
718 |
+
with st.expander("Learn More About These Conditions"):
|
719 |
+
if condition_evidence:
|
720 |
+
for condition in condition_evidence.keys():
|
721 |
+
# Fetch condition description from LLM
|
722 |
+
desc_prompt = f"Provide a brief description (1-2 sentences) of the medical condition '{condition}'."
|
723 |
+
if st.session_state.llm_service == "groq":
|
724 |
+
description = use_groq_api(desc_prompt, max_tokens=100)
|
725 |
+
else:
|
726 |
+
description = use_claude_api(desc_prompt, max_tokens=100)
|
727 |
+
st.write(f"**{condition}**: {description.strip()}")
|
728 |
+
else:
|
729 |
+
st.write("No specific conditions identified yet.")
|
730 |
+
|
731 |
+
with diagnosis_tabs[2]: # Recommendations tab
|
732 |
+
st.markdown("### Care Recommendations")
|
733 |
+
|
734 |
+
st.markdown("#### Warning Signs - Seek Medical Help If:")
|
735 |
+
# Fetch warning signs from LLM
|
736 |
+
warn_prompt = f"Based on this diagnosis: '{diagnosis_text}' and severity '{severity_rating}', list 3 warning signs indicating the need for immediate medical help."
|
737 |
+
if st.session_state.llm_service == "groq":
|
738 |
+
warn_response = use_groq_api(warn_prompt, max_tokens=150)
|
739 |
+
else:
|
740 |
+
warn_response = use_claude_api(warn_prompt, max_tokens=150)
|
741 |
+
warnings = warn_response.split("\n")[:3]
|
742 |
+
for warning in warnings:
|
743 |
+
if warning.strip():
|
744 |
+
st.write(f"⚠️ {warning.strip()}")
|
745 |
+
|
746 |
+
st.markdown("#### Medications to Consider")
|
747 |
+
# Fetch medications from LLM
|
748 |
+
med_prompt = f"Based on this diagnosis: '{diagnosis_text}', suggest 3 medications or treatment options."
|
749 |
+
if st.session_state.llm_service == "groq":
|
750 |
+
med_response = use_groq_api(med_prompt, max_tokens=150)
|
751 |
+
else:
|
752 |
+
med_response = use_claude_api(med_prompt, max_tokens=150)
|
753 |
+
medications = med_response.split("\n")[:3]
|
754 |
+
col1, col2 = st.columns(2)
|
755 |
+
for i, med in enumerate(medications):
|
756 |
+
if med.strip():
|
757 |
+
with col1 if i % 2 == 0 else col2:
|
758 |
+
st.write(f"💊 {med.strip()}")
|
759 |
+
|
760 |
+
st.markdown("#### Home Care Tips")
|
761 |
+
# Fetch home care tips from LLM
|
762 |
+
care_prompt = f"Based on this diagnosis: '{diagnosis_text}', provide 3 home care tips."
|
763 |
+
if st.session_state.llm_service == "groq":
|
764 |
+
care_response = use_groq_api(care_prompt, max_tokens=150)
|
765 |
+
else:
|
766 |
+
care_response = use_claude_api(care_prompt, max_tokens=150)
|
767 |
+
home_care = care_response.split("\n")[:3]
|
768 |
+
for tip in home_care:
|
769 |
+
if tip.strip():
|
770 |
+
st.write(f"✅ {tip.strip()}")
|
771 |
+
|
772 |
+
with diagnosis_tabs[3]: # Action Steps tab
|
773 |
+
st.markdown("### Next Steps")
|
774 |
+
|
775 |
+
if severity_rating in ["High", "Moderate"]:
|
776 |
+
st.markdown("#### Medical Consultation")
|
777 |
+
# Fetch specialists from LLM
|
778 |
+
spec_prompt = f"Based on this diagnosis: '{diagnosis_text}' and conditions: {list(condition_evidence.keys())}, suggest 3 types of medical specialists to consult."
|
779 |
+
if st.session_state.llm_service == "groq":
|
780 |
+
spec_response = use_groq_api(spec_prompt, max_tokens=150)
|
781 |
+
else:
|
782 |
+
spec_response = use_claude_api(spec_prompt, max_tokens=150)
|
783 |
+
specialists = spec_response.split("\n")[:3]
|
784 |
+
col1, col2 = st.columns(2)
|
785 |
+
for i, spec in enumerate(specialists):
|
786 |
+
if spec.strip():
|
787 |
+
with col1 if i % 2 == 0 else col2:
|
788 |
+
st.write(f"👨⚕️ {spec.strip()}")
|
789 |
+
|
790 |
+
st.markdown("#### Schedule Consultation")
|
791 |
+
col1, col2 = st.columns(2)
|
792 |
+
with col1:
|
793 |
+
if st.button("Find Doctors Near Me", key="find_doctors"):
|
794 |
+
st.info("This would connect to a directory of medical providers.")
|
795 |
+
with col2:
|
796 |
+
if st.button("Virtual Consultation Options", key="virtual_consult"):
|
797 |
+
st.info("This would connect to telemedicine services.")
|
798 |
+
|
799 |
+
st.markdown("#### Symptom Monitoring")
|
800 |
+
with st.expander("Add Symptom Entry"):
|
801 |
+
with st.form("symptom_tracker"):
|
802 |
+
st.date_input("Date", value=None)
|
803 |
+
st.slider("Temperature (°F)", min_value=96.0, max_value=104.0, value=98.6, step=0.1)
|
804 |
+
st.slider("Cough Severity", min_value=0, max_value=10, value=5)
|
805 |
+
st.slider("Overall Feeling", min_value=0, max_value=10, value=5)
|
806 |
+
st.form_submit_button("Save Symptom Entry")
|
807 |
+
|
808 |
+
st.markdown("---")
|
809 |
+
st.write("### Important Note")
|
810 |
+
st.write("Monitor your symptoms closely and seek medical attention if they worsen.")
|
811 |
+
st.caption("This assessment is for testing and educational purposes only.")
|
812 |
+
|
813 |
+
st.markdown("### What would you like to do next?")
|
814 |
+
next_steps = st.columns(3)
|
815 |
+
|
816 |
+
with next_steps[0]:
|
817 |
+
if st.button("Print Assessment", key="print_assessment"):
|
818 |
+
# Generate PDF content dynamically
|
819 |
+
pdf_content = "Your Diagnostic Assessment\n\n"
|
820 |
+
pdf_content += f"Severity Level: {severity_rating}\nAction Needed: {action_needed}\n\n"
|
821 |
+
pdf_content += f"Summary: {summary}\n\nConditions:\n"
|
822 |
+
if condition_evidence:
|
823 |
+
sorted_conditions = sorted(condition_evidence.items(), key=lambda x: x[1], reverse=True)[:5]
|
824 |
+
for condition, score in sorted_conditions:
|
825 |
+
desc_prompt = f"Provide a brief description of '{condition}'."
|
826 |
+
if st.session_state.llm_service == "groq":
|
827 |
+
description = use_groq_api(desc_prompt, max_tokens=100)
|
828 |
+
else:
|
829 |
+
description = use_claude_api(desc_prompt, max_tokens=100)
|
830 |
+
pdf_content += f"- {condition} (Score: {score}): {description.strip()}\n"
|
831 |
+
pdf_content += "\nCare Recommendations:\n"
|
832 |
+
pdf_content += "Warning Signs:\n" + "\n".join([f"- {w.strip()}" for w in warnings if w.strip()]) + "\n"
|
833 |
+
pdf_content += "Medications:\n" + "\n".join([f"- {m.strip()}" for m in medications if m.strip()]) + "\n"
|
834 |
+
pdf_content += "Home Care Tips:\n" + "\n".join([f"- {t.strip()}" for t in home_care if t.strip()]) + "\n"
|
835 |
+
pdf_content += "\nFull Diagnosis:\n" + diagnosis_text
|
836 |
+
|
837 |
+
# Convert to bytes for download
|
838 |
+
pdf_bytes = pdf_content.encode('utf-8')
|
839 |
+
st.download_button(
|
840 |
+
label="Download Assessment",
|
841 |
+
data=pdf_bytes,
|
842 |
+
file_name="diagnostic_assessment.txt", # Using .txt due to Streamlit-only constraint
|
843 |
+
mime="text/plain",
|
844 |
+
key="download_assessment"
|
845 |
+
)
|
846 |
+
st.success("Assessment ready for download above.")
|
847 |
+
|
848 |
+
with next_steps[1]:
|
849 |
+
if st.button("Share with Doctor", key="share_doctor"):
|
850 |
+
st.success("In a full implementation, this would prepare the assessment for sharing.")
|
851 |
+
|
852 |
+
with next_steps[2]:
|
853 |
+
if st.button("Start New Consultation", key="start_new"):
|
854 |
+
return "new"
|
855 |
+
|
856 |
+
return None
|
857 |
+
|
858 |
+
# Main application
|
859 |
+
st.title("AI Diagnostic Assistant")
|
860 |
+
st.markdown("_This is a prototype for testing purposes only and should not be used for actual medical diagnosis._")
|
861 |
+
|
862 |
+
# Side panel for audio upload and conversation controls
|
863 |
+
with st.sidebar:
|
864 |
+
st.header("Controls")
|
865 |
+
|
866 |
+
# LLM Service Selection
|
867 |
+
st.subheader("LLM Service")
|
868 |
+
llm_option = st.radio(
|
869 |
+
"Select LLM Service",
|
870 |
+
["Groq (LLaMA3-70B)", "Anthropic Claude-3 Opus"],
|
871 |
+
index=0 if st.session_state.llm_service == "groq" else 1
|
872 |
+
)
|
873 |
+
|
874 |
+
# Update LLM service based on selection and reset conversation if service changes
|
875 |
+
if (llm_option == "Groq (LLaMA3-70B)" and st.session_state.llm_service != "groq") or \
|
876 |
+
(llm_option == "Anthropic Claude-3 Opus" and st.session_state.llm_service != "claude"):
|
877 |
+
# Store the new service selection
|
878 |
+
st.session_state.llm_service = "groq" if llm_option == "Groq (LLaMA3-70B)" else "claude"
|
879 |
+
|
880 |
+
# Reset conversation state
|
881 |
+
st.session_state.conversation = []
|
882 |
+
st.session_state.diagnostic_state = "initial"
|
883 |
+
st.session_state.gathered_symptoms = []
|
884 |
+
st.session_state.audio_analysis = None
|
885 |
+
st.session_state.current_question = None
|
886 |
+
st.session_state.current_options = None
|
887 |
+
st.session_state.user_submitted = False
|
888 |
+
st.session_state.severity_rating = None
|
889 |
+
st.session_state.audio_processed = False
|
890 |
+
st.session_state.waiting_for_next_question = False
|
891 |
+
st.session_state.condition_evidence = {}
|
892 |
+
st.session_state.other_selected = False
|
893 |
+
st.session_state.audio_declined = False # Reset audio declined flag
|
894 |
+
|
895 |
+
# Show notification
|
896 |
+
st.success(f"Switched to {llm_option}. Starting new consultation.")
|
897 |
+
st.rerun()
|
898 |
+
|
899 |
+
# Audio upload
|
900 |
+
st.subheader("Audio Analysis")
|
901 |
+
audio_file = st.file_uploader("Upload audio (cough, breathing, etc.)", type=["wav", "mp3"])
|
902 |
+
|
903 |
+
# Handle audio analysis
|
904 |
+
if audio_file and not st.session_state.audio_analysis:
|
905 |
+
if st.button("Analyze Audio"):
|
906 |
+
with st.spinner("Analyzing audio sample..."):
|
907 |
+
try:
|
908 |
+
audio_classifier = load_audio_classifier()
|
909 |
+
audio_results, audio_context = analyze_audio(audio_file, audio_classifier)
|
910 |
+
if audio_results:
|
911 |
+
st.session_state.audio_analysis = audio_context
|
912 |
+
st.session_state.audio_processed = True
|
913 |
+
# If we were in the requesting_audio state, set flag to generate next question
|
914 |
+
if st.session_state.diagnostic_state == "requesting_audio":
|
915 |
+
st.session_state.waiting_for_next_question = True
|
916 |
+
st.success("Audio analysis complete!")
|
917 |
+
except Exception as e:
|
918 |
+
st.error(f"Error analyzing audio: {str(e)}")
|
919 |
+
|
920 |
+
# Replace the current condition evidence display in the sidebar with this code
|
921 |
+
if st.session_state.condition_evidence:
|
922 |
+
st.subheader("Condition Evidence")
|
923 |
+
|
924 |
+
# Filter out items that are not likely to be real medical conditions
|
925 |
+
non_conditions = ["here is", "here are", "these are", "following", "specific", "mentioned"]
|
926 |
+
filtered_evidence = {
|
927 |
+
condition: score for condition, score in st.session_state.condition_evidence.items()
|
928 |
+
if not any(nc in condition.lower() for nc in non_conditions) and len(condition) > 3
|
929 |
+
}
|
930 |
+
|
931 |
+
# Create a sorted list of all conditions based on evidence score
|
932 |
+
sorted_conditions = sorted(
|
933 |
+
filtered_evidence.items(),
|
934 |
+
key=lambda x: x[1],
|
935 |
+
reverse=True
|
936 |
+
)
|
937 |
+
|
938 |
+
# Display all conditions with their evidence scores in a cleaner format
|
939 |
+
for condition, score in sorted_conditions:
|
940 |
+
# Only show conditions with positive scores
|
941 |
+
if score > 0:
|
942 |
+
# Calculate percentage (max score assumed to be 5 for full bar)
|
943 |
+
percentage = min(score / 5 * 100, 100)
|
944 |
+
|
945 |
+
# Create colored bars based on evidence strength
|
946 |
+
if score >= 3:
|
947 |
+
bar_color = "rgba(0, 204, 102, 0.8)" # Green for strong evidence
|
948 |
+
elif score >= 1.5:
|
949 |
+
bar_color = "rgba(255, 153, 51, 0.8)" # Orange for moderate evidence
|
950 |
+
else:
|
951 |
+
bar_color = "rgba(160, 160, 160, 0.8)" # Gray for weak evidence
|
952 |
+
|
953 |
+
# Display condition with score and bar
|
954 |
+
st.markdown(
|
955 |
+
f"""
|
956 |
+
<div style="margin-bottom: 5px;">
|
957 |
+
<span style="font-size: 0.9em;">{condition}: {score}</span>
|
958 |
+
<div style="background-color: #f0f0f0; height: 10px; border-radius: 5px; margin-top: 2px;">
|
959 |
+
<div style="width: {percentage}%; background-color: {bar_color}; height: 10px; border-radius: 5px;"></div>
|
960 |
+
</div>
|
961 |
+
</div>
|
962 |
+
""",
|
963 |
+
unsafe_allow_html=True
|
964 |
+
)
|
965 |
+
|
966 |
+
# Reset conversation button
|
967 |
+
st.subheader("Session")
|
968 |
+
if st.button("Start New Consultation"):
|
969 |
+
st.session_state.conversation = []
|
970 |
+
st.session_state.diagnostic_state = "initial"
|
971 |
+
st.session_state.gathered_symptoms = []
|
972 |
+
st.session_state.audio_analysis = None
|
973 |
+
st.session_state.current_question = None
|
974 |
+
st.session_state.current_options = None
|
975 |
+
st.session_state.user_submitted = False
|
976 |
+
st.session_state.severity_rating = None
|
977 |
+
st.session_state.audio_processed = False
|
978 |
+
st.session_state.waiting_for_next_question = False
|
979 |
+
st.session_state.condition_evidence = {}
|
980 |
+
st.session_state.other_selected = False
|
981 |
+
st.session_state.audio_declined = False # Reset audio declined flag
|
982 |
+
st.rerun()
|
983 |
+
|
984 |
+
# Main chat interface
|
985 |
+
st.header("Medical Consultation")
|
986 |
+
|
987 |
+
# Display selected LLM service
|
988 |
+
st.caption(f"Using {'Groq (LLaMA3-70B)' if st.session_state.llm_service == 'groq' else 'Claude 3 Opus'} for diagnosis")
|
989 |
+
|
990 |
+
# Display conversation history
|
991 |
+
for i, message in enumerate(st.session_state.conversation):
|
992 |
+
if i % 2 == 0: # Assistant messages
|
993 |
+
st.markdown(f"**Medical Assistant:** {message}")
|
994 |
+
else: # User messages
|
995 |
+
st.markdown(f"**You:** {message}")
|
996 |
+
|
997 |
+
# Handle audio processing and next question generation after analysis
|
998 |
+
if st.session_state.audio_processed and st.session_state.waiting_for_next_question:
|
999 |
+
st.session_state.diagnostic_state = "gathering" # Resume gathering state
|
1000 |
+
|
1001 |
+
# Generate next question based on all info including audio
|
1002 |
+
with st.spinner("Processing audio analysis and preparing next question..."):
|
1003 |
+
question, options = generate_next_question_with_options(
|
1004 |
+
st.session_state.conversation,
|
1005 |
+
st.session_state.gathered_symptoms,
|
1006 |
+
st.session_state.audio_analysis
|
1007 |
+
)
|
1008 |
+
|
1009 |
+
# Add the generated question to conversation only if it's not already there
|
1010 |
+
if len(st.session_state.conversation) == 0 or question != st.session_state.conversation[-1]:
|
1011 |
+
# Make sure this is an assistant message (should be at an even index)
|
1012 |
+
if len(st.session_state.conversation) % 2 == 0:
|
1013 |
+
st.session_state.conversation.append(question)
|
1014 |
+
|
1015 |
+
st.session_state.current_question = question
|
1016 |
+
st.session_state.current_options = options
|
1017 |
+
|
1018 |
+
# Reset flags
|
1019 |
+
st.session_state.waiting_for_next_question = False
|
1020 |
+
st.session_state.audio_processed = False
|
1021 |
+
|
1022 |
+
# Rerun to display updated conversation
|
1023 |
+
st.rerun()
|
1024 |
+
|
1025 |
+
# Initial greeting - only show if conversation is empty
|
1026 |
+
if st.session_state.diagnostic_state == "initial":
|
1027 |
+
initial_greeting = "Hello, I'm your medical assistant for today. Could you please tell me what symptoms you're experiencing?"
|
1028 |
+
st.markdown(f"**Medical Assistant:** {initial_greeting}")
|
1029 |
+
|
1030 |
+
# Add a callback for the initial symptoms form submission
|
1031 |
+
def submit_initial_symptoms():
|
1032 |
+
if st.session_state.initial_symptoms:
|
1033 |
+
# Record the user's symptoms in the conversation
|
1034 |
+
st.session_state.conversation.append(initial_greeting)
|
1035 |
+
st.session_state.conversation.append(st.session_state.initial_symptoms)
|
1036 |
+
st.session_state.gathered_symptoms.append(st.session_state.initial_symptoms)
|
1037 |
+
|
1038 |
+
# Update condition evidence based on initial symptoms
|
1039 |
+
update_condition_evidence(st.session_state.initial_symptoms)
|
1040 |
+
|
1041 |
+
# Change state to indicate we need to generate the next question
|
1042 |
+
st.session_state.diagnostic_state = "generate_next_question"
|
1043 |
+
|
1044 |
+
# Add an input field for initial symptoms
|
1045 |
+
with st.form(key="initial_symptoms_form"):
|
1046 |
+
st.text_area("Please describe your symptoms:", key="initial_symptoms")
|
1047 |
+
st.form_submit_button("Submit", on_click=submit_initial_symptoms)
|
1048 |
+
|
1049 |
+
# Add a new state to handle generating the next question after initial symptoms
|
1050 |
+
elif st.session_state.diagnostic_state == "generate_next_question":
|
1051 |
+
with st.spinner("Analyzing your symptoms..."):
|
1052 |
+
question, options = generate_next_question_with_options(
|
1053 |
+
st.session_state.conversation,
|
1054 |
+
st.session_state.gathered_symptoms,
|
1055 |
+
st.session_state.audio_analysis
|
1056 |
+
)
|
1057 |
+
|
1058 |
+
st.session_state.current_question = question
|
1059 |
+
st.session_state.current_options = options
|
1060 |
+
st.session_state.conversation.append(question)
|
1061 |
+
st.session_state.diagnostic_state = "gathering"
|
1062 |
+
|
1063 |
+
# Force a rerun to refresh the page with the new conversation state
|
1064 |
+
st.rerun()
|
1065 |
+
|
1066 |
+
# Current assistant question (if in gathering phase)
|
1067 |
+
elif st.session_state.diagnostic_state == "gathering" and st.session_state.current_question:
|
1068 |
+
# Only show current question if it's not already in conversation history
|
1069 |
+
if len(st.session_state.conversation) == 0 or st.session_state.current_question != st.session_state.conversation[-1]:
|
1070 |
+
# Ensure this is an assistant message (should be at an odd index in conversation)
|
1071 |
+
if len(st.session_state.conversation) % 2 == 0:
|
1072 |
+
st.markdown(f"**Medical Assistant:** {st.session_state.current_question}")
|
1073 |
+
|
1074 |
+
# Audio request state
|
1075 |
+
elif st.session_state.diagnostic_state == "requesting_audio" and not st.session_state.audio_analysis:
|
1076 |
+
# Only add the audio request to conversation once
|
1077 |
+
if len(st.session_state.conversation) == 0 or st.session_state.current_question != st.session_state.conversation[-1]:
|
1078 |
+
# Make sure this is an assistant message (should be at an even index in conversation)
|
1079 |
+
if len(st.session_state.conversation) % 2 == 0:
|
1080 |
+
st.session_state.conversation.append(st.session_state.current_question)
|
1081 |
+
|
1082 |
+
# Always display the request
|
1083 |
+
st.markdown(f"**Medical Assistant:** {st.session_state.current_question}")
|
1084 |
+
|
1085 |
+
# Add buttons to skip or continue with audio upload
|
1086 |
+
col1, col2 = st.columns(2)
|
1087 |
+
with col1:
|
1088 |
+
if st.button("Skip Audio Upload"):
|
1089 |
+
# Mark audio as declined
|
1090 |
+
st.session_state.audio_declined = True
|
1091 |
+
|
1092 |
+
# Add user response to conversation
|
1093 |
+
st.session_state.conversation.append("I'd like to skip the audio upload and continue with text-based questions.")
|
1094 |
+
|
1095 |
+
# Generate the next question and go back to gathering state
|
1096 |
+
st.session_state.diagnostic_state = "gathering"
|
1097 |
+
|
1098 |
+
# Generate next question
|
1099 |
+
with st.spinner("Preparing next question..."):
|
1100 |
+
question, options = generate_next_question_with_options(
|
1101 |
+
st.session_state.conversation,
|
1102 |
+
st.session_state.gathered_symptoms,
|
1103 |
+
None # No audio analysis
|
1104 |
+
)
|
1105 |
+
|
1106 |
+
st.session_state.current_question = question
|
1107 |
+
st.session_state.current_options = options
|
1108 |
+
st.session_state.conversation.append(question)
|
1109 |
+
|
1110 |
+
st.rerun()
|
1111 |
+
|
1112 |
+
with col2:
|
1113 |
+
st.info("Please upload an audio sample using the sidebar upload control")
|
1114 |
+
|
1115 |
+
# Display final diagnosis (if complete)
|
1116 |
+
elif st.session_state.diagnostic_state == "complete":
|
1117 |
+
if 'diagnosis' in st.session_state:
|
1118 |
+
action = display_interactive_diagnosis(
|
1119 |
+
st.session_state.diagnosis,
|
1120 |
+
st.session_state.severity_rating,
|
1121 |
+
st.session_state.condition_evidence
|
1122 |
+
)
|
1123 |
+
# Handle the "Start New Consultation" action
|
1124 |
+
if action == "new":
|
1125 |
+
# Reset session state for a new consultation
|
1126 |
+
st.session_state.conversation = []
|
1127 |
+
st.session_state.diagnostic_state = "initial"
|
1128 |
+
st.session_state.gathered_symptoms = []
|
1129 |
+
st.session_state.audio_analysis = None
|
1130 |
+
st.session_state.current_question = None
|
1131 |
+
st.session_state.current_options = None
|
1132 |
+
st.session_state.user_submitted = False
|
1133 |
+
st.session_state.severity_rating = None
|
1134 |
+
st.session_state.audio_processed = False
|
1135 |
+
st.session_state.waiting_for_next_question = False
|
1136 |
+
st.session_state.condition_evidence = {}
|
1137 |
+
st.session_state.other_selected = False
|
1138 |
+
st.session_state.audio_declined = False
|
1139 |
+
st.rerun() # Rerun the app to reflect the reset state
|
1140 |
+
|
1141 |
+
# Show options for user selection if we have current options and are in gathering state
|
1142 |
+
if st.session_state.diagnostic_state == "gathering" and st.session_state.current_options and not st.session_state.user_submitted and not st.session_state.other_selected:
|
1143 |
+
# Create a container for the radio buttons to ensure proper rendering
|
1144 |
+
options_container = st.container()
|
1145 |
+
with options_container:
|
1146 |
+
# Initialize the radio without an on_change callback
|
1147 |
+
st.radio(
|
1148 |
+
"Select your response:",
|
1149 |
+
st.session_state.current_options,
|
1150 |
+
key="selected_option"
|
1151 |
+
)
|
1152 |
+
# Add a submit button to confirm selection
|
1153 |
+
if st.button("Confirm Selection"):
|
1154 |
+
if "selected_option" in st.session_state and st.session_state.selected_option:
|
1155 |
+
if st.session_state.selected_option == "Other":
|
1156 |
+
st.session_state.other_selected = True
|
1157 |
+
else:
|
1158 |
+
# Record the response in conversation
|
1159 |
+
st.session_state.conversation.append(st.session_state.selected_option)
|
1160 |
+
st.session_state.gathered_symptoms.append(st.session_state.selected_option)
|
1161 |
+
|
1162 |
+
# Update evidence for conditions
|
1163 |
+
update_condition_evidence(st.session_state.selected_option)
|
1164 |
+
|
1165 |
+
st.session_state.user_submitted = True
|
1166 |
+
st.rerun()
|
1167 |
+
|
1168 |
+
# Show custom input field if "Other" is selected
|
1169 |
+
if st.session_state.other_selected:
|
1170 |
+
with st.form(key="custom_input_form"):
|
1171 |
+
st.text_area("Please describe your symptoms in detail:", key="custom_input")
|
1172 |
+
submit_button = st.form_submit_button("Submit", on_click=handle_custom_submit)
|
1173 |
+
|
1174 |
+
# Process form submission after page rerun
|
1175 |
+
if st.session_state.user_submitted and st.session_state.diagnostic_state == "gathering":
|
1176 |
+
# After several questions, check if we need audio data for better diagnosis
|
1177 |
+
# But only request audio if user hasn't already declined and we don't already have audio
|
1178 |
+
if len(st.session_state.gathered_symptoms) >= 2 and not st.session_state.audio_analysis and not st.session_state.audio_declined and needs_more_information(
|
1179 |
+
st.session_state.conversation,
|
1180 |
+
st.session_state.gathered_symptoms,
|
1181 |
+
st.session_state.condition_evidence
|
1182 |
+
) and would_audio_help(st.session_state.condition_evidence):
|
1183 |
+
# Request audio sample
|
1184 |
+
audio_request = generate_audio_request()
|
1185 |
+
st.session_state.current_question = audio_request
|
1186 |
+
st.session_state.conversation.append(audio_request)
|
1187 |
+
st.session_state.diagnostic_state = "requesting_audio"
|
1188 |
+
# Determine next action based on conversation length
|
1189 |
+
elif len(st.session_state.conversation) >= 9 or not needs_more_information(
|
1190 |
+
st.session_state.conversation,
|
1191 |
+
st.session_state.gathered_symptoms,
|
1192 |
+
st.session_state.condition_evidence
|
1193 |
+
):
|
1194 |
+
# Generate diagnosis after enough information gathered
|
1195 |
+
with st.spinner("Analyzing your symptoms..."):
|
1196 |
+
diagnosis, severity = generate_diagnosis(
|
1197 |
+
st.session_state.conversation,
|
1198 |
+
st.session_state.gathered_symptoms,
|
1199 |
+
st.session_state.condition_evidence,
|
1200 |
+
st.session_state.audio_analysis
|
1201 |
+
)
|
1202 |
+
|
1203 |
+
# Extract conditions from the diagnosis and update evidence
|
1204 |
+
new_conditions = extract_conditions_from_diagnosis(diagnosis)
|
1205 |
+
for condition in new_conditions:
|
1206 |
+
if condition not in st.session_state.condition_evidence:
|
1207 |
+
# Add new conditions from the diagnosis
|
1208 |
+
st.session_state.condition_evidence[condition] = 2 # Give it a reasonable starting score
|
1209 |
+
else:
|
1210 |
+
# Increase evidence for conditions mentioned in diagnosis
|
1211 |
+
st.session_state.condition_evidence[condition] += 1
|
1212 |
+
|
1213 |
+
st.session_state.diagnosis = diagnosis
|
1214 |
+
st.session_state.severity_rating = severity
|
1215 |
+
st.session_state.conversation.append("Based on your symptoms, I've prepared a diagnostic assessment.")
|
1216 |
+
st.session_state.diagnostic_state = "complete"
|
1217 |
+
else:
|
1218 |
+
# Generate next question with options
|
1219 |
+
with st.spinner("Thinking..."):
|
1220 |
+
question, options = generate_next_question_with_options(
|
1221 |
+
st.session_state.conversation,
|
1222 |
+
st.session_state.gathered_symptoms,
|
1223 |
+
st.session_state.audio_analysis
|
1224 |
+
)
|
1225 |
+
|
1226 |
+
st.session_state.current_question = question
|
1227 |
+
st.session_state.current_options = options
|
1228 |
+
st.session_state.conversation.append(question)
|
1229 |
+
|
1230 |
+
# Reset user_submitted flag to prepare for next input
|
1231 |
+
st.session_state.user_submitted = False
|
1232 |
+
|
1233 |
+
# Refresh the page to show the updated conversation
|
1234 |
+
st.rerun()
|
1235 |
+
|
1236 |
+
# Display audio analysis results if available
|
1237 |
+
if st.session_state.audio_analysis:
|
1238 |
+
with st.expander("Audio Analysis Results"):
|
1239 |
+
st.write("The following was detected in your audio sample:")
|
1240 |
+
for context in st.session_state.audio_analysis:
|
1241 |
+
st.write(f"- {context}")
|
1242 |
+
|
1243 |
+
# Large disclaimer at bottom of app
|
1244 |
+
st.markdown("---")
|
1245 |
+
st.warning("⚠️ **IMPORTANT DISCLAIMER**: This application is for testing and educational purposes only. It does not provide actual medical diagnosis or treatment recommendations. Always consult with qualified healthcare professionals for medical advice.")
|
1246 |
+
|
1247 |
+
# Back button to return to main menu
|
1248 |
+
if st.button("Back to Main Menu"):
|
1249 |
+
st.session_state.page = "main"
|
1250 |
+
st.rerun()
|
1251 |
+
|
1252 |
+
if __name__ == "__main__":
|
1253 |
+
run_physical_health()
|
requirements.txt
ADDED
@@ -0,0 +1,12 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
streamlit
|
2 |
+
transformers
|
3 |
+
torch
|
4 |
+
torchaudio
|
5 |
+
soundfile
|
6 |
+
librosa
|
7 |
+
python-dotenv
|
8 |
+
anthropic
|
9 |
+
nltk
|
10 |
+
|
11 |
+
# run this in terminal
|
12 |
+
# conda install -c conda-forge ffmpeg
|