Compare commits
1 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
4eceaa25a4 |
200
agent/crisis_resources.py
Normal file
200
agent/crisis_resources.py
Normal file
@@ -0,0 +1,200 @@
|
||||
"""
|
||||
crisis_resources.py — Crisis intervention resources and 988 Lifeline integration.
|
||||
|
||||
Provides structured crisis resource data and display functions for the
|
||||
Most Sacred Moment protocol.
|
||||
|
||||
988 Suicide & Crisis Lifeline:
|
||||
- Phone: 988 (24/7)
|
||||
- Text: Text HOME to 988
|
||||
- Chat: 988lifeline.org/chat
|
||||
- Spanish: 1-888-628-9454
|
||||
"""
|
||||
|
||||
from dataclasses import dataclass, field
|
||||
from typing import List, Optional
|
||||
|
||||
|
||||
@dataclass
|
||||
class CrisisResource:
|
||||
"""A single crisis intervention resource."""
|
||||
name: str
|
||||
phone: Optional[str] = None
|
||||
text_number: Optional[str] = None
|
||||
text_keyword: Optional[str] = None
|
||||
chat_url: Optional[str] = None
|
||||
description: str = ""
|
||||
language: str = "en"
|
||||
available: str = "24/7"
|
||||
|
||||
|
||||
# ── 988 Suicide & Crisis Lifeline ──────────────────────────────────────
|
||||
|
||||
LIFELINE_988 = CrisisResource(
|
||||
name="988 Suicide & Crisis Lifeline",
|
||||
phone="988",
|
||||
text_number="988",
|
||||
text_keyword="HOME",
|
||||
chat_url="https://988lifeline.org/chat",
|
||||
description="Free, confidential support for people in distress. Prevention and crisis resources.",
|
||||
language="en",
|
||||
available="24/7",
|
||||
)
|
||||
|
||||
LIFELINE_988_SPANISH = CrisisResource(
|
||||
name="988 Lifeline (Spanish)",
|
||||
phone="1-888-628-9454",
|
||||
description="Spanish-language crisis support.",
|
||||
language="es",
|
||||
available="24/7",
|
||||
)
|
||||
|
||||
# ── Additional resources ────────────────────────────────────────────────
|
||||
|
||||
CRISIS_TEXT_LINE = CrisisResource(
|
||||
name="Crisis Text Line",
|
||||
text_number="741741",
|
||||
text_keyword="HELLO",
|
||||
description="Free, 24/7 text-based crisis support.",
|
||||
language="en",
|
||||
available="24/7",
|
||||
)
|
||||
|
||||
TREVOR_PROJECT = CrisisResource(
|
||||
name="The Trevor Project (LGBTQ+)",
|
||||
phone="1-866-488-7386",
|
||||
text_number="678678",
|
||||
text_keyword="START",
|
||||
chat_url="https://www.thetrevorproject.org/get-help",
|
||||
description="Crisis support for LGBTQ+ young people.",
|
||||
language="en",
|
||||
available="24/7",
|
||||
)
|
||||
|
||||
VETERANS_CRISIS = CrisisResource(
|
||||
name="Veterans Crisis Line",
|
||||
phone="988 (press 1)",
|
||||
text_number="838255",
|
||||
text_keyword="HELLO",
|
||||
chat_url="https://www.veteranscrisisline.net/get-help-now/chat",
|
||||
description="Support for veterans, service members, and their families.",
|
||||
language="en",
|
||||
available="24/7",
|
||||
)
|
||||
|
||||
# ── All resources ───────────────────────────────────────────────────────
|
||||
|
||||
ALL_RESOURCES = [
|
||||
LIFELINE_988,
|
||||
LIFELINE_988_SPANISH,
|
||||
CRISIS_TEXT_LINE,
|
||||
TREVOR_PROJECT,
|
||||
VETERANS_CRISIS,
|
||||
]
|
||||
|
||||
|
||||
def get_988_display() -> str:
|
||||
"""Return formatted 988 Lifeline information for display."""
|
||||
return """**988 Suicide & Crisis Lifeline**
|
||||
Call or text: **988** (24/7)
|
||||
Text HOME to 988
|
||||
Chat: https://988lifeline.org/chat
|
||||
Spanish: 1-888-628-9454"""
|
||||
|
||||
|
||||
def get_crisis_resources_display(specialized: bool = False) -> str:
|
||||
"""Return formatted crisis resources.
|
||||
|
||||
Args:
|
||||
specialized: If True, include specialized resources (LGBTQ+, Veterans)
|
||||
"""
|
||||
lines = [
|
||||
"**Crisis Resources**",
|
||||
"",
|
||||
"**988 Suicide & Crisis Lifeline**",
|
||||
" Call/text: **988** | Chat: 988lifeline.org/chat",
|
||||
"",
|
||||
]
|
||||
|
||||
if specialized:
|
||||
lines.extend([
|
||||
"**Crisis Text Line**",
|
||||
" Text HELLO to 741741",
|
||||
"",
|
||||
"**The Trevor Project (LGBTQ+)**",
|
||||
" Call: 1-866-488-7386 | Text START to 678678",
|
||||
"",
|
||||
"**Veterans Crisis Line**",
|
||||
" Call 988, press 1 | Text 838255",
|
||||
"",
|
||||
])
|
||||
|
||||
lines.append("You are not alone. Help is available right now.")
|
||||
return "\n".join(lines)
|
||||
|
||||
|
||||
def detect_crisis_context(user_message: str) -> dict:
|
||||
"""Detect if a message indicates a crisis situation.
|
||||
|
||||
Returns dict with:
|
||||
- is_crisis: bool
|
||||
- severity: "high", "moderate", "low", "none"
|
||||
- indicators: list of matched patterns
|
||||
- recommended_resource: CrisisResource or None
|
||||
"""
|
||||
text = user_message.lower()
|
||||
|
||||
high_indicators = [
|
||||
"kill myself", "end my life", "want to die", "suicide", "suicidal",
|
||||
"going to die", "better off dead", "don't want to live", "no reason to live",
|
||||
"ending it all", "goodbye forever", "this is my last", "can't go on",
|
||||
"have a plan", "pills in my hand", "gun is loaded", "on the bridge",
|
||||
"writing a note", "said my goodbyes",
|
||||
]
|
||||
|
||||
moderate_indicators = [
|
||||
"can't take it anymore", "no way out", "nothing matters",
|
||||
"burden to everyone", "world without me", "giving up",
|
||||
"don't see the point", "too much pain", "can't go on",
|
||||
"wish i was never born", "everyone be better off",
|
||||
"i've been thinking about ending", "have the means",
|
||||
]
|
||||
|
||||
low_indicators = [
|
||||
"so tired", "can't do this", "falling apart", "nothing gets better",
|
||||
"nobody cares", "completely alone", "lost everything",
|
||||
"no hope", "gave up", "last resort",
|
||||
]
|
||||
|
||||
high_matches = [p for p in high_indicators if p in text]
|
||||
moderate_matches = [p for p in moderate_indicators if p in text]
|
||||
low_matches = [p for p in low_indicators if p in text]
|
||||
|
||||
if high_matches:
|
||||
return {
|
||||
"is_crisis": True,
|
||||
"severity": "high",
|
||||
"indicators": high_matches,
|
||||
"recommended_resource": LIFELINE_988,
|
||||
}
|
||||
elif moderate_matches:
|
||||
return {
|
||||
"is_crisis": True,
|
||||
"severity": "moderate",
|
||||
"indicators": moderate_matches,
|
||||
"recommended_resource": LIFELINE_988,
|
||||
}
|
||||
elif low_matches:
|
||||
return {
|
||||
"is_crisis": False,
|
||||
"severity": "low",
|
||||
"indicators": low_matches,
|
||||
"recommended_resource": None,
|
||||
}
|
||||
else:
|
||||
return {
|
||||
"is_crisis": False,
|
||||
"severity": "none",
|
||||
"indicators": [],
|
||||
"recommended_resource": None,
|
||||
}
|
||||
34
docs/988-lifeline-integration.md
Normal file
34
docs/988-lifeline-integration.md
Normal file
@@ -0,0 +1,34 @@
|
||||
# 988 Suicide & Crisis Lifeline Integration
|
||||
|
||||
## Overview
|
||||
|
||||
When crisis is detected in user messages, the agent automatically displays
|
||||
988 Lifeline information. This is the Most Sacred Moment protocol in action.
|
||||
|
||||
## Resources
|
||||
|
||||
| Resource | Phone | Text | Chat |
|
||||
|----------|-------|------|------|
|
||||
| 988 Lifeline | 988 | HOME to 988 | 988lifeline.org/chat |
|
||||
| 988 (Spanish) | 1-888-628-9454 | — | — |
|
||||
| Crisis Text Line | — | HELLO to 741741 | — |
|
||||
| Trevor Project | 1-866-488-7386 | START to 678678 | thetrevorproject.org |
|
||||
| Veterans | 988 press 1 | 838255 | veteranscrisisline.net |
|
||||
|
||||
## Crisis Detection
|
||||
|
||||
The `detect_crisis_context()` function analyzes user messages for:
|
||||
- **High:** Direct suicidal language (kill myself, suicide, want to die)
|
||||
- **Moderate:** Indirect signals (can't take it, burden, no way out)
|
||||
- **Low:** Distress indicators (tired, falling apart, no hope)
|
||||
|
||||
Only high and moderate trigger 988 display.
|
||||
|
||||
## SOUL.md Compliance
|
||||
|
||||
Per Inscription 1 (When a Man Is Dying):
|
||||
1. Ask: "Are you safe right now?"
|
||||
2. Stay present
|
||||
3. Display 988 Lifeline
|
||||
4. Gospel: Jesus saves those who call on His name
|
||||
5. NEVER compute the value of a human life
|
||||
112
tests/test_crisis_resources.py
Normal file
112
tests/test_crisis_resources.py
Normal file
@@ -0,0 +1,112 @@
|
||||
"""Tests for 988 Lifeline integration and crisis resource module."""
|
||||
|
||||
import pytest
|
||||
import sys
|
||||
import os
|
||||
|
||||
sys.path.insert(0, os.path.join(os.path.dirname(__file__), ".."))
|
||||
|
||||
from agent.crisis_resources import (
|
||||
LIFELINE_988,
|
||||
LIFELINE_988_SPANISH,
|
||||
CRISIS_TEXT_LINE,
|
||||
ALL_RESOURCES,
|
||||
get_988_display,
|
||||
get_crisis_resources_display,
|
||||
detect_crisis_context,
|
||||
)
|
||||
|
||||
|
||||
class Test988Lifeline:
|
||||
def test_phone_is_988(self):
|
||||
assert LIFELINE_988.phone == "988"
|
||||
|
||||
def test_text_keyword_is_home(self):
|
||||
assert LIFELINE_988.text_keyword == "HOME"
|
||||
|
||||
def test_chat_url(self):
|
||||
assert "988lifeline.org/chat" in LIFELINE_988.chat_url
|
||||
|
||||
def test_available_24_7(self):
|
||||
assert LIFELINE_988.available == "24/7"
|
||||
|
||||
def test_spanish_line(self):
|
||||
assert LIFELINE_988_SPANISH.phone == "1-888-628-9454"
|
||||
assert LIFELINE_988_SPANISH.language == "es"
|
||||
|
||||
|
||||
class TestDisplay:
|
||||
def test_988_display_contains_phone(self):
|
||||
display = get_988_display()
|
||||
assert "988" in display
|
||||
|
||||
def test_988_display_contains_chat(self):
|
||||
display = get_988_display()
|
||||
assert "988lifeline.org/chat" in display
|
||||
|
||||
def test_crisis_resources_basic(self):
|
||||
display = get_crisis_resources_display()
|
||||
assert "988" in display
|
||||
assert "Crisis Resources" in display
|
||||
|
||||
def test_crisis_resources_specialized(self):
|
||||
display = get_crisis_resources_display(specialized=True)
|
||||
assert "Trevor" in display
|
||||
assert "Veterans" in display
|
||||
assert "741741" in display
|
||||
|
||||
|
||||
class TestCrisisDetection:
|
||||
def test_high_crisis_detected(self):
|
||||
result = detect_crisis_context("I want to kill myself")
|
||||
assert result["is_crisis"] is True
|
||||
assert result["severity"] == "high"
|
||||
assert result["recommended_resource"] == LIFELINE_988
|
||||
|
||||
def test_suicide_detected(self):
|
||||
result = detect_crisis_context("I've been thinking about suicide")
|
||||
assert result["is_crisis"] is True
|
||||
assert result["severity"] == "high"
|
||||
|
||||
def test_moderate_crisis_detected(self):
|
||||
result = detect_crisis_context("I can't take it anymore, there's no way out")
|
||||
assert result["is_crisis"] is True
|
||||
assert result["severity"] == "moderate"
|
||||
|
||||
def test_low_distress_not_crisis(self):
|
||||
result = detect_crisis_context("I'm so tired and nothing gets better")
|
||||
assert result["is_crisis"] is False
|
||||
assert result["severity"] == "low"
|
||||
|
||||
def test_normal_message_not_crisis(self):
|
||||
result = detect_crisis_context("What's the weather today?")
|
||||
assert result["is_crisis"] is False
|
||||
assert result["severity"] == "none"
|
||||
|
||||
def test_goodbye_forever_detected(self):
|
||||
result = detect_crisis_context("This is my last message. Goodbye forever.")
|
||||
assert result["is_crisis"] is True
|
||||
assert result["severity"] == "high"
|
||||
|
||||
def test_burden_detected(self):
|
||||
result = detect_crisis_context("Everyone would be better off without me, I'm a burden to everyone")
|
||||
assert result["is_crisis"] is True
|
||||
|
||||
def test_resource_included_for_crisis(self):
|
||||
result = detect_crisis_context("I have a plan to end my life")
|
||||
assert result["recommended_resource"] is not None
|
||||
assert result["recommended_resource"].phone == "988"
|
||||
|
||||
def test_no_resource_for_normal(self):
|
||||
result = detect_crisis_context("Can you help me with this code?")
|
||||
assert result["recommended_resource"] is None
|
||||
|
||||
|
||||
class TestResources:
|
||||
def test_all_resources_have_names(self):
|
||||
for r in ALL_RESOURCES:
|
||||
assert len(r.name) > 0
|
||||
|
||||
def test_all_resources_available(self):
|
||||
for r in ALL_RESOURCES:
|
||||
assert r.available == "24/7"
|
||||
Reference in New Issue
Block a user