Initial commit: avc-phone-ai codebase + CLAUDE.md

This commit is contained in:
tocmo0nlord
2026-06-23 22:38:22 +00:00
parent 4bf72b9616
commit c3c719b77e
16 changed files with 1491 additions and 0 deletions

155
practice.py Normal file
View File

@@ -0,0 +1,155 @@
"""Advanced Vision Care practice facts + the phone agent's tools.
Facts sourced from advancedvisioncareflorida.com (8 locations across Broward,
Miami-Dade, Palm Beach). NOTE: the website does NOT publish office hours, so we do
NOT assert hours — the agent must offer to have staff confirm them instead of
inventing them. Fill HOURS in if/when you have them.
"""
import json
import os
import re
from datetime import datetime, timezone
from loguru import logger
# ─────────────────────────────────────────────────────────────────────────────
# Real facts from advancedvisioncareflorida.com
# ─────────────────────────────────────────────────────────────────────────────
LOCATIONS = [
# Broward County
{"city": "Hollywood / Fort Lauderdale", "address": "2873 Stirling Rd, Fort Lauderdale, FL 33312", "phone": "(954) 983-4969"},
{"city": "Tamarac", "address": "5865 N University Dr, Tamarac, FL 33321", "phone": "(954) 720-2720"},
{"city": "Pembroke Pines", "address": "246 S Flamingo Rd, Pembroke Pines, FL 33027", "phone": "(954) 443-1230"},
{"city": "Lauderdale Lakes", "address": "3682 W Oakland Park Blvd, Lauderdale Lakes, FL 33311", "phone": "(954) 730-8087"},
# Miami-Dade County
{"city": "Hialeah", "address": "1770 W 32nd Pl, Hialeah, FL 33012", "phone": "(305) 885-4477"},
{"city": "Kendall", "address": "11605 N Kendall Dr, Miami, FL 33176", "phone": "(305) 982-8927"},
{"city": "Miami Gardens", "address": "4771 NW 183rd St, Miami Gardens, FL 33055", "phone": "(305) 390-2467"},
# Palm Beach County
{"city": "Boca Raton", "address": "21673 State Road 7, Boca Raton, FL 33428", "phone": "(561) 470-2310"},
]
PRACTICE_FACTS = {
"name": "Advanced Vision Care",
"locations": LOCATIONS,
"insurance": [
"CarePlus", "Doctors Health", "Florida Blue Medicare", "Optum", "Spectera",
"Sunshine Health", "VSP", "WellCare",
],
"services": (
"routine and medical eye exams, contact lens exams, pediatric eye exams, "
"and LASIK consultations"
),
# Website does not publish hours — leave None so the agent won't invent them.
"hours": None,
}
REQUESTS_LOG = os.path.join(os.path.dirname(os.path.abspath(__file__)), "appointment_requests.jsonl")
# Expand street abbreviations so the TTS speaks "North Kendall Drive", not "N … D-R".
_ABBREV = {
"NW": "Northwest", "NE": "Northeast", "SW": "Southwest", "SE": "Southeast",
"N": "North", "S": "South", "E": "East", "W": "West",
"Dr": "Drive", "Rd": "Road", "Blvd": "Boulevard", "St": "Street",
"Ave": "Avenue", "Pl": "Place", "Ln": "Lane", "Ct": "Court", "Hwy": "Highway",
"FL": "Florida",
}
def _spoken_address(addr: str) -> str:
"""Expand directional + street-type abbreviations for natural speech."""
return re.sub(
r"\b(" + "|".join(re.escape(k) for k in _ABBREV) + r")\b",
lambda m: _ABBREV[m.group(1)],
addr,
)
def practice_summary() -> str:
"""Compact facts block for the system prompt."""
f = PRACTICE_FACTS
loc_lines = "\n".join(f" - {l['city']}: {_spoken_address(l['address'])}{l['phone']}" for l in f["locations"])
hours = f["hours"] or (
"NOT published — do not state specific hours; offer to have the office confirm."
)
return (
f"Practice name: {f['name']}\n"
f"Locations ({len(f['locations'])} offices across South Florida):\n{loc_lines}\n"
f"Insurance accepted (these EXACT plans only): {', '.join(f['insurance'])}.\n"
f"Services: {f['services']}\n"
f"Hours: {hours}\n"
)
def _find_location(name: str):
"""Loose match a caller's city/location text to a known office."""
if not name:
return None
n = name.lower()
for l in LOCATIONS:
if n in l["city"].lower() or l["city"].lower() in n:
return l
return None
# ─── Tools (used when ENABLE_TOOLS=true and the model supports tool-calling) ──
def persist_appointment(record: dict) -> str:
"""Write an appointment request to Odoo (a crm.lead) if configured, else to the
JSONL fallback so a request is never lost. Returns where it landed. Used by both
the post-call extraction and the (optional) in-call tool."""
record.setdefault("ts", datetime.now(timezone.utc).isoformat())
if os.environ.get("ODOO_USER") and os.environ.get("ODOO_API_KEY"):
try:
from odoo_client import create_appointment_request
model, rec_id = create_appointment_request(
patient_name=record.get("patient_name"),
callback_number=record.get("callback_number"),
reason=f"[{record.get('location') or 'location TBD'}] {record.get('reason') or ''}".strip(),
preferred_time=record.get("preferred_time"),
call_sid=record.get("call_sid"),
)
logger.info(f"Appointment -> Odoo {model} id={rec_id}: {record.get('patient_name')}")
return f"odoo:{model}:{rec_id}"
except Exception as e:
logger.warning(f"Odoo write failed ({e!r}); falling back to local log")
record["odoo_error"] = repr(e)
with open(REQUESTS_LOG, "a") as fh:
fh.write(json.dumps(record) + "\n")
logger.info(f"Appointment -> JSONL: {record.get('patient_name')}")
return "jsonl"
async def record_appointment_request(params):
"""In-call tool path (only used when ENABLE_TOOLS=true). Wraps persist_appointment."""
args = params.arguments or {}
persist_appointment({
"call_sid": getattr(params, "call_sid", None),
"patient_name": args.get("patient_name"),
"callback_number": args.get("callback_number"),
"location": args.get("location"),
"reason": args.get("reason"),
"preferred_time": args.get("preferred_time"),
"source": "in_call_tool",
})
await params.result_callback(
{"status": "captured", "message": "Got it — our staff will call you back to confirm the time."}
)
async def get_practice_info(params):
"""Return practice facts (optionally narrowed to one location) for accurate answers."""
args = params.arguments or {}
loc = _find_location(args.get("location", ""))
result = {
"name": PRACTICE_FACTS["name"],
"insurance": PRACTICE_FACTS["insurance"],
"services": PRACTICE_FACTS["services"],
"hours": "not published — offer to have the office confirm",
}
result["location"] = loc if loc else PRACTICE_FACTS["locations"]
await params.result_callback(result)