mirror of
https://github.com/Shubhamsaboo/awesome-llm-apps.git
synced 2026-03-11 17:48:31 -05:00
139 lines
5.5 KiB
Python
139 lines
5.5 KiB
Python
from langchain.prompts import PromptTemplate
|
|
import pandas as pd
|
|
from langchain_core.runnables import RunnableParallel, RunnableLambda # Import necessary for LCEL
|
|
import random
|
|
import streamlit as st
|
|
import helpers.help_func as hf
|
|
from PIL import Image
|
|
|
|
|
|
|
|
# --- Load the dataset ---
|
|
csv_file_path = 'data/tarots.csv'
|
|
try:
|
|
# Read CSV file
|
|
df = pd.read_csv(csv_file_path, sep=';', encoding='latin1')
|
|
print(f"CSV dataset loaded successfully: {csv_file_path}. Number of rows: {len(df)}")
|
|
|
|
# Clean and normalize column names
|
|
df.columns = df.columns.str.strip().str.lower()
|
|
|
|
# Debug: Show column details
|
|
print("\nDetails after cleanup:")
|
|
for col in df.columns:
|
|
print(f"Column: '{col}' (length: {len(col)})")
|
|
|
|
# Define required columns (in lowercase)
|
|
required_columns = ['card', 'upright', 'reversed', 'symbolism']
|
|
|
|
# Verify all required columns are present
|
|
available_columns = set(df.columns)
|
|
missing_columns = [col for col in required_columns if col not in available_columns]
|
|
|
|
if missing_columns:
|
|
raise ValueError(
|
|
f"Missing columns in CSV file: {', '.join(missing_columns)}\n"
|
|
f"Available columns: {', '.join(available_columns)}"
|
|
)
|
|
|
|
# Create card meanings dictionary with cleaned data
|
|
card_meanings = {}
|
|
for _, row in df.iterrows():
|
|
card_name = row['card'].strip()
|
|
card_meanings[card_name] = {
|
|
'upright': str(row['upright']).strip() if pd.notna(row['upright']) else '',
|
|
'reversed': str(row['reversed']).strip() if pd.notna(row['reversed']) else '',
|
|
'symbolism': str(row['symbolism']).strip() if pd.notna(row['symbolism']) else ''
|
|
}
|
|
|
|
print(f"\nKnowledge base created with {len(card_meanings)} cards, meanings and symbolisms.")
|
|
|
|
except FileNotFoundError:
|
|
print(f"Error: CSV File not found: {csv_file_path}")
|
|
raise
|
|
except ValueError as e:
|
|
print(f"Validation Error: {str(e)}")
|
|
raise
|
|
except Exception as e:
|
|
print(f"Unexpected error: {str(e)}")
|
|
raise
|
|
|
|
|
|
|
|
# --- Define the Prompt Template ---
|
|
prompt_analysis = PromptTemplate.from_template("""
|
|
Analyze the following tarot cards, based on the meanings provided (also considering if they are reversed):
|
|
{card_details}
|
|
Pay attention to these aspects:
|
|
- Provide a detailed analysis of the meaning of each card (upright or reversed).
|
|
- Then offer a general interpretation of the answer based on the cards, linking it to the context: {context}.
|
|
- Be mystical and provide information on the interpretation related to the symbolism of the cards, based on the specific column: {symbolism}.
|
|
- At the end of the reading, always offer advice to improve or address the situation. Also, base it on your knowledge of psychology.
|
|
""")
|
|
print("\nPrompt Template 'prompt_analysis' defined.")
|
|
|
|
# --- Create the LangChain Chain ---
|
|
analyzer = (
|
|
RunnableParallel(
|
|
cards=lambda x: x['cards'],
|
|
context=lambda x: x['context']
|
|
)
|
|
| (lambda x: hf.prepare_prompt_input(x, card_meanings))
|
|
| prompt_analysis
|
|
| hf.llm
|
|
)
|
|
|
|
|
|
# --- Frontend Streamlit ---
|
|
st.set_page_config(
|
|
page_title="🔮 Interactive Tarot Reading",
|
|
page_icon="🃏",
|
|
layout="wide",
|
|
initial_sidebar_state="expanded"
|
|
)
|
|
st.title("🔮 Interactive Tarot Reading")
|
|
st.markdown("Welcome to your personalized tarot consultation!")
|
|
st.markdown("---")
|
|
|
|
num_cards = st.selectbox("🃏 Select the number of cards for your spread (3 for a more focused answer, 7 for a more general overview).)", [3, 5, 7])
|
|
context_question = st.text_area("✍️ Please enter your context or your question here. You can speak in natural language.", height=100)
|
|
|
|
if st.button("✨ Light your path: Draw and Analyze the Cards."):
|
|
if not context_question:
|
|
st.warning("For a more precise reading, please enter your context or question.")
|
|
else:
|
|
try:
|
|
card_names_in_dataset = df['card'].unique().tolist()
|
|
drawn_cards_list = hf.generate_random_draw(num_cards, card_names_in_dataset)
|
|
st.subheader("✨ Your Cards Revealed:")
|
|
st.markdown("---")
|
|
|
|
cols = st.columns(len(drawn_cards_list))
|
|
for i, card_info in enumerate(drawn_cards_list):
|
|
with cols[i]:
|
|
# The card_info['name'] from data/tarots.csv is now the direct image filename e.g., "00-thefool.jpg"
|
|
image_filename = card_info['name']
|
|
image_path = f"images/{image_filename}"
|
|
reversed_label = "(R)" if 'is_reversed' in card_info else ""
|
|
caption = f"{card_info['name']} {reversed_label}"
|
|
|
|
try:
|
|
img = Image.open(image_path)
|
|
if card_info.get('is_reversed', False):
|
|
img = img.rotate(180)
|
|
st.image(img, caption=caption, width=150)
|
|
except FileNotFoundError:
|
|
st.info(f"Symbol: {card_info['name']} {reversed_label} (Image not found at {image_path})")
|
|
|
|
st.markdown("---")
|
|
with st.spinner("🔮 Unveiling the meanings..."):
|
|
analysis_result = analyzer.invoke({"cards": drawn_cards_list, "context": context_question})
|
|
st.subheader("📜 The Interpretation:")
|
|
st.write(analysis_result.content)
|
|
|
|
except Exception as e:
|
|
st.error(f"An error has occurred: {e}")
|
|
st.error(f"Error details: {e}")
|
|
|
|
st.markdown("---")
|
|
st.info("Remember, the cards offer insights and reflections; your future is in your hands.") |