Spaces:
Sleeping
Sleeping
import os | |
import streamlit as st | |
import re | |
from datetime import datetime | |
from groq import Groq | |
from langchain_community.vectorstores import FAISS | |
from langchain_text_splitters import RecursiveCharacterTextSplitter | |
from langchain_huggingface import HuggingFaceEmbeddings | |
import pandas as pd | |
# Load environment variables | |
try: | |
from dotenv import load_dotenv | |
load_dotenv() | |
except ImportError: | |
pass # dotenv not installed, likely running on Hugging Face Spaces | |
# Function to get the API key | |
def get_api_key(): | |
api_key = os.environ.get("GROQ_API_KEY") | |
if not api_key: | |
api_key = st.secrets.get("GROQ_API_KEY") | |
if not api_key: | |
st.error( | |
"GROQ_API_KEY is not set. Please set it in your environment or Streamlit secrets.") | |
st.stop() | |
return api_key | |
def parse_transcript(content): | |
parsed_segments = [] | |
current_speaker = "" | |
current_company = "" | |
current_timestamp = "" | |
# Split the content into paragraphs | |
paragraphs = re.split(r'\n\s*\n', content) | |
for paragraph in paragraphs: | |
paragraph = paragraph.strip() | |
if not paragraph: | |
continue | |
# Check if the paragraph contains speaker information | |
speaker_match = re.match(r'(.*?),\s*(.*?)\((.*?)\):', paragraph) | |
if speaker_match: | |
current_speaker, current_company, current_timestamp = speaker_match.groups() | |
text = paragraph.split('\n', 1)[1] if '\n' in paragraph else '' | |
# Check if the paragraph contains only a timestamp | |
elif re.match(r'\((.*?)\):', paragraph): | |
current_timestamp = re.match(r'\((.*?)\):', paragraph).group(1) | |
text = re.sub(r'^\(.*?\):\s*', '', paragraph).strip() | |
# If it's not a speaker line or timestamp line, it's just text | |
else: | |
text = paragraph | |
# Add the segment | |
if text: | |
add_segment(parsed_segments, current_speaker, | |
current_company, current_timestamp, text) | |
return parsed_segments | |
def add_segment(parsed_segments, speaker, company, timestamp, text): | |
segment = { | |
"speaker": speaker, | |
"company": company, | |
"timestamp": timestamp, | |
"text": text | |
} | |
parsed_segments.append(segment) | |
print_segment(speaker, company, timestamp, text) | |
def print_segment(speaker, company, timestamp, text): | |
print(f"Speaker: {speaker}") | |
print(f"Company: {company}") | |
print(f"Timestamp: {timestamp}") | |
print(f"Text: {text[:100]}...") # Print first 100 characters of text | |
print("-" * 50) | |
def create_searchable_segments(parsed_segments): | |
searchable_segments = [] | |
for segment in parsed_segments: | |
searchable_text = ( | |
f"{segment['speaker']},{segment['company']}," | |
f"{segment['timestamp']}:: {segment['text']}" | |
) | |
searchable_segments.append(searchable_text) | |
return searchable_segments | |
# Load and parse the transcript | |
def load_transcript(content): | |
global vectorstore | |
# Parse the transcript | |
parsed_transcript = parse_transcript(content) | |
searchable_segments = create_searchable_segments(parsed_transcript) | |
# Create text splitter and split the searchable segments | |
text_splitter = RecursiveCharacterTextSplitter( | |
chunk_size=1000, chunk_overlap=200) | |
splits = text_splitter.create_documents(searchable_segments) | |
# Create vector store with HuggingFaceEmbeddings | |
embeddings = HuggingFaceEmbeddings() | |
vectorstore = FAISS.from_documents(documents=splits, embedding=embeddings) | |
def search_transcript(query, k=30): | |
# Perform similarity search | |
docs = vectorstore.similarity_search(query=query, k=k) | |
# Format results | |
results = [] | |
for doc in docs: | |
content = doc.page_content | |
match = re.match(r'(.*?),(.*?),(.*?)::\s*(.*)', content, re.DOTALL) | |
if match: | |
speaker, company, timestamp, text = match.groups() | |
results.append({ | |
"speaker": speaker.strip(), | |
"company": company.strip(), | |
"timestamp": timestamp.strip(), | |
"text": text.strip() | |
}) | |
return results | |
# Groq client setup | |
client = Groq(api_key=get_api_key()) | |
def generate_response(query, search_results): | |
# Prepare the prompt with search results | |
prompt = f"""You are a friendly assistant. Your job is to answer the user's question based on the transcript excerpts provided below: | |
Transcript excerpts: | |
{search_results} | |
Question: {query} | |
Please provide a concise and relevant answer based on the information in the transcript excerpts. If the information is not directly related to the question, say so and provide the most relevant information available.""" | |
completion = client.chat.completions.create( | |
model="llama3-8b-8192", | |
messages=[ | |
{ | |
"role": "user", | |
"content": prompt | |
} | |
], | |
temperature=0.5, | |
max_tokens=3000, | |
top_p=1, | |
stream=False, | |
stop=None, | |
) | |
return completion.choices[0].message.content | |
# Streamlit app | |
def main(): | |
st.title("Transcript Search and Q&A") | |
st.caption("This site takes a TelecomTV video transcript and allows a chat session with it. If no transcript is provided it defaults to this one: https://www.telecomtv.com/content/dsp-leaders-forum/enabling-the-autonomous-network-with-ai-50536/") | |
# File upload | |
uploaded_file = st.file_uploader("Upload a transcript file", type="txt") | |
if uploaded_file is None: | |
file_name = "Enabling the autonomous network with AI" | |
with open("example-transcript.txt", 'r') as file: | |
content = file.read() | |
else: | |
content = uploaded_file.getvalue().decode("utf-8") | |
file_name = uploaded_file.name | |
# Read and process the uploaded file | |
load_transcript(content) | |
st.subheader(f"Chat with {file_name}") | |
# User input | |
user_query = st.text_input( | |
"Enter your question:", placeholder="e.g.What are people speaking about? or List all people speaking") | |
# Add a slider for selecting the number of results | |
num_results = st.slider("Number of relevant transcript excerpts to show:", | |
min_value=1, max_value=50, value=30, step=1) | |
if user_query: | |
search_results = search_transcript(user_query, k=num_results) | |
formatted_results = ( | |
"\n\n".join([f"{result['speaker']} {result['company']} ({result['timestamp']}): " | |
"{result['text']}" for result in search_results]) | |
) | |
response = generate_response(user_query, formatted_results) | |
st.subheader("Assistant's response:") | |
st.write(response) | |
st.subheader("Relevant transcript excerpts:") | |
# Create a DataFrame from the search results | |
df = pd.DataFrame(search_results) | |
# Rename columns for better readability | |
df.columns = ['Speaker', 'Company', 'Timestamp', 'Quote'] | |
# Display the DataFrame as a table | |
st.table(df) | |
if __name__ == "__main__": | |
main() | |