Spaces:
Sleeping
Sleeping
File size: 8,886 Bytes
2def93d 4060393 2060f4b 4060393 c645389 e39a43f 209dd92 e39a43f 2060f4b e39a43f 2060f4b c645389 e69869b 2060f4b c0332bf 4060393 2060f4b e39a43f 2060f4b e69869b 98f2c4f 691e8b4 5c03320 d4828d9 2060f4b d4828d9 a589322 53b56c5 a589322 53b56c5 a589322 53b56c5 a589322 209dd92 53b56c5 a589322 53b56c5 a589322 209dd92 a589322 209dd92 53b56c5 209dd92 53b56c5 209dd92 e69869b e39a43f db82163 d4828d9 53b56c5 d4828d9 db82163 e39a43f db82163 e39a43f 087e42a e39a43f 53b56c5 2060f4b e39a43f 53b56c5 e39a43f 53b56c5 e39a43f 087e42a 3c25af5 2060f4b e39a43f 2060f4b e39a43f 2060f4b e39a43f 2060f4b e39a43f 2060f4b e39a43f 2060f4b eb7d0ea 2060f4b 53b56c5 087e42a 2060f4b 53b56c5 2060f4b 53b56c5 2060f4b 087e42a 2060f4b 087e42a 2060f4b ab19d74 |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 |
import streamlit as st
from openai import OpenAI
from PIL import Image
import time
import os
import uuid
import firebase_admin
from firebase_admin import credentials, firestore
import re
# π Firebase setup
if not firebase_admin._apps:
cred = credentials.Certificate("firebase-service-account.json")
firebase_admin.initialize_app(cred)
db = firestore.client()
# π OpenAI setup
openai_key = os.getenv("openai_key")
assistant_id = os.getenv("ASSISTANT_ID")
client = OpenAI(api_key=openai_key)
# π Streamlit Config
st.set_page_config(page_title="Carfind.co.za AI Assistant", layout="wide")
# π― Session + User ID
if "user_id" not in st.session_state:
st.session_state["user_id"] = str(uuid.uuid4())
user_id = st.session_state["user_id"]
# πΌοΈ Branding + Styling
st.markdown("""
<style>
.block-container {padding-top: 1rem; padding-bottom: 0rem;}
header {visibility: hidden;}
.stChatMessage { max-width: 85%; border-radius: 12px; padding: 8px; margin-bottom: 10px; }
.stChatMessage[data-testid="stChatMessage-user"] { background: #f0f0f0; color: #000000; }
.stChatMessage[data-testid="stChatMessage-assistant"] { background: #D6E9FE; color: #000000; }
@keyframes bounceIn {
0% { transform: scale(0.7); opacity: 0; }
60% { transform: scale(1.1); opacity: 1; }
80% { transform: scale(0.95); }
100% { transform: scale(1); }
}
.carfind-logo {
animation: bounceIn 0.6s ease-out;
}
.car-spec-output {
font-family: "Segoe UI", sans-serif;
font-size: 10px;
background-color: #ffffff;
color: #003B6F;
padding: 18px;
border-radius: 10px;
box-shadow: 0 4px 12px rgba(0,0,0,0.05);
border-left: 4px solid #0071BC;
line-height: 1.5;
}
.spec-table {
width: 100%;
border-collapse: collapse;
margin-top: 12px;
margin-bottom: 16px;
font-size: 10px;
}
.spec-table th, .spec-table td {
border: 1px solid #ddd;
padding: 6px 8px;
text-align: left;
}
.spec-table th {
background-color: #0071BC;
color: white;
}
.spec-table td {
background-color: #f7f9fc;
}
.icon {
width: 14px;
vertical-align: middle;
margin-right: 5px;
}
</style>
""", unsafe_allow_html=True)
st.markdown("""
<div style='text-align: center; margin-top: 20px; margin-bottom: -10px;'>
<span style='display: inline-flex; align-items: center; gap: 8px;'>
<img src='https://www.carfind.co.za/images/Carfind-Icon.svg' width='28' class='carfind-logo'/>
<span style='font-size: 12px; color: gray;'>Powered by Carfind</span>
</span>
</div>
""", unsafe_allow_html=True)
def get_or_create_thread_id():
doc_ref = db.collection("users").document(user_id)
doc = doc_ref.get()
if doc.exists:
return doc.to_dict()["thread_id"]
else:
thread = client.beta.threads.create()
doc_ref.set({"thread_id": thread.id, "created_at": firestore.SERVER_TIMESTAMP})
return thread.id
def save_message(role, content):
db.collection("users").document(user_id).collection("messages").add({
"role": role,
"content": content,
"timestamp": firestore.SERVER_TIMESTAMP
})
def display_chat_history():
messages = db.collection("users").document(user_id).collection("messages").order_by("timestamp").stream()
assistant_icon_html = "<img src='https://www.carfind.co.za/images/Carfind-Icon.svg' width='20' style='vertical-align:middle;'/>"
for msg in list(messages)[::-1]:
data = msg.to_dict()
if data["role"] == "user":
st.markdown(f"<div class='stChatMessage' data-testid='stChatMessage-user'>π€ <strong>You:</strong> {data['content']}</div>", unsafe_allow_html=True)
else:
st.markdown(f"<div class='stChatMessage' data-testid='stChatMessage-assistant'>{assistant_icon_html} <strong>Carfind Assistant:</strong> {data['content']}</div>", unsafe_allow_html=True)
def format_car_response(raw_text):
cleaned = re.sub(r'\*\*(.*?)\*\*', r'<strong>\1</strong>', raw_text)
cleaned = cleaned.replace("###", "<h4>").replace("\n", "<br>")
lines = raw_text.split("\n")
table_rows = []
intro = ""
for line in lines:
if "|" in line and not set(line.strip()) <= {"|", "-"}:
parts = [part.strip() for part in line.split("|") if part.strip()]
if len(parts) == 2:
label, value = parts
if value.lower() not in ["n/a", "null", "[specify mileage]", "[specify year]", "", "-"]:
table_rows.append((label, value))
elif not intro and line.strip():
intro = line.strip()
html = f"<div style='margin-bottom: 10px; font-size: 12px; color: #003B6F;'>{intro}</div>"
if len(table_rows) >= 3:
html += "<table class='spec-table'><tr><th>Specification</th><th>Details</th></tr>"
for label, value in table_rows:
html += f"<tr><td>{label}</td><td>{value}</td></tr>"
html += "</table>"
else:
html += "<div style='color: grey;'>No detailed specifications available.</div>"
return html
# Tabs
tab1, tab2 = st.tabs(["AI Chat", "What car is that?"])
with tab1:
input_col, clear_col = st.columns([9, 1])
with input_col:
user_input = st.chat_input("Type your message here...")
with clear_col:
if st.button("ποΈ", key="clear-chat", help="Clear Chat"):
try:
user_doc_ref = db.collection("users").document(user_id)
for msg in user_doc_ref.collection("messages").stream():
msg.reference.delete()
user_doc_ref.delete()
st.session_state.clear()
st.rerun()
except Exception as e:
st.error(f"Failed to clear chat: {e}")
thread_id = get_or_create_thread_id()
display_chat_history()
if user_input:
client.beta.threads.messages.create(thread_id=thread_id, role="user", content=user_input)
save_message("user", user_input)
with st.spinner("Thinking and typing... π"):
run = client.beta.threads.runs.create(thread_id=thread_id, assistant_id=assistant_id)
while True:
run_status = client.beta.threads.runs.retrieve(thread_id=thread_id, run_id=run.id)
if run_status.status == "completed":
break
time.sleep(1)
messages_response = client.beta.threads.messages.list(thread_id=thread_id)
latest_response = sorted(messages_response.data, key=lambda x: x.created_at)[-1]
assistant_message = latest_response.content[0].text.value
save_message("assistant", assistant_message)
time.sleep(0.5)
st.rerun()
with tab2:
uploaded_image = st.file_uploader("Upload an image of a car and let Ai identify it for you", type=["jpg", "jpeg", "png"])
if uploaded_image:
col1, col2 = st.columns([1.2, 1.8])
with col1:
image = Image.open(uploaded_image)
st.image(image, caption="Uploaded Image", use_container_width=True)
with col2:
try:
image_thread = client.beta.threads.create()
file_response = client.files.create(file=uploaded_image, purpose="assistants")
client.beta.threads.messages.create(
thread_id=image_thread.id,
role="user",
content=[
{"type": "image_file", "image_file": {"file_id": file_response.id}},
{"type": "text", "text": "Please identify this car from the image and include a clean styled specifications as are available."}
]
)
run = client.beta.threads.runs.create(thread_id=image_thread.id, assistant_id=assistant_id)
with st.spinner("π Analyzing image and identifying the car..."):
while True:
run_status = client.beta.threads.runs.retrieve(thread_id=image_thread.id, run_id=run.id)
if run_status.status == "completed":
break
time.sleep(1)
messages = client.beta.threads.messages.list(thread_id=image_thread.id)
assistant_message = messages.data[0].content[0].text.value
formatted_html = format_car_response(assistant_message)
st.success("β
Identification Complete")
st.markdown(f"<div class='car-spec-output'>{formatted_html}</div>", unsafe_allow_html=True)
except Exception as e:
st.error(f"β Error during image analysis: {str(e)}") |