Files
2025-06-09 20:11:35 +02:00

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.")