import gradio as gr from utils.data_loader import fetch_and_save_chess_data from utils.game_analysis import ( analyze_games, generate_monthly_report, calculate_average_and_median_games, analyze_streaks, analyze_sequences, format_duration ) from datetime import datetime import os import json import matplotlib.pyplot as plt import io from PIL import Image import plotly.graph_objects as go import pandas as pd import csv logged_in_user = None # Global state to store the logged-in user # Define your user credentials auth_users = [ ("Sacha", "SachaIsTheBest"), ("Florian", "FlorianIsTheBest"), ("Lucas", "Slevin"), ("Gael", "Kalel"), ("BlueNote", "MamaLinda") ] DATA_FOLDER = 'data/' LOG_FILE = 'user_logs.csv' # Initialize log file if it doesn't exist if not os.path.exists(LOG_FILE): with open(LOG_FILE, 'w', newline='') as log_file: writer = csv.writer(log_file) writer.writerow(["Username", "Timestamp", "Action", "Query"]) def log_user_action(username, action, query=""): """Log user actions to a CSV file.""" timestamp = datetime.now().strftime('%Y-%m-%d %H:%M:%S') with open(LOG_FILE, 'a', newline='') as log_file: writer = csv.writer(log_file) writer.writerow([username, timestamp, action, query]) def get_monthly_report(username, logged_in_user): """Fetch data for a given username, reuse existing data if available, and generate a monthly report.""" current_date = datetime.now().strftime('%Y-%m-%d') filename = os.path.join(DATA_FOLDER, f"{username}_{current_date}.json") log_user_action(logged_in_user, "Search", username) if os.path.exists(filename): print(f"Using existing data file: {filename}") with open(filename, 'r') as file: games = json.load(file) else: games = fetch_and_save_chess_data(username, filename) if not games: return "No data found for the specified username." games_sorted = sorted(games, key=lambda x: x.get('end_time')) # Perform monthly analysis games_per_month, stats_per_month, total_games, total_wins, total_losses, total_timeouts, total_months_played = analyze_games(games, username) report_df = generate_monthly_report(games_per_month, stats_per_month) # Calculate average and median games per day average_games, median_games = calculate_average_and_median_games(games) # Calculate streak probabilities win_prob, loss_prob = analyze_streaks(games_sorted, username) # Calculate sequence probabilities win_after_wl_prob, win_after_lw_prob = analyze_sequences(games_sorted, username) # Format the duration of months played formatted_duration = format_duration(total_months_played) # Prepare the text summary summary_text = ( f"Total duration played: {formatted_duration}\n" f"Total games played: {total_games}\n" f"Total wins: {total_wins}\n" f"Total losses: {total_losses}\n" f"Total timeouts: {total_timeouts}\n" f"Average games played per day: {average_games:.2f}\n" f"Median games played per day: {median_games}\n" f"Probability of winning the next game after a win in the same hour: {win_prob:.2f}%\n" f"Probability of losing the next game after a loss in the same hour: {loss_prob:.2f}%\n" f"Probability of winning the next game after a 'win-loss' sequence in the same hour: {win_after_wl_prob:.2f}%\n" f"Probability of winning the next game after a 'loss-win' sequence in the same hour: {win_after_lw_prob:.2f}%" ) stacked_bar_img = generate_stacked_bar_chart(report_df) return report_df, summary_text, stacked_bar_img def generate_stacked_bar_chart(report_df): """Generate an interactive stacked bar chart with wins, losses, and a line plot for timeouts using Plotly.""" # Extract data months = pd.to_datetime(report_df['Month']).dt.strftime('%b, %Y') wins = report_df['Wins'] losses = report_df['Losses'] timeouts = report_df['Timeout Rate (%)'] total_games = report_df['Games Played'] # Create the figure fig = go.Figure() # Add wins bars fig.add_trace(go.Bar( x=months, y=wins, name='Wins', marker=dict(color='#1f77b4'), hovertemplate='Wins: %{y}' )) # Add losses bars stacked on top of wins fig.add_trace(go.Bar( x=months, y=losses, name='Losses', marker=dict(color='#ff7f0e'), hovertemplate='Losses: %{y}' )) # Add timeouts as a line plot fig.add_trace(go.Scatter( x=months, y=timeouts, mode='lines+markers', name='Timeouts', line=dict(color='#da5bac', width=2), hovertemplate='Timeouts: %{y:.1f}%' )) # Add rotated annotations for total games on top of each bar for i, total in enumerate(total_games): fig.add_annotation( x=months[i], y=total + 5, text=str(int(total)), showarrow=False, font=dict(size=12, color='white'), align='center', textangle=-45 # Rotate the text label by -45 degrees ) # Update layout for stacked bars and hover information fig.update_layout( barmode='stack', title='Monthly Win/Loss Stacked Bar Chart with Total Games and Timeouts', xaxis_title='Month', yaxis_title='Count', legend_title='Legend', hovermode='x unified', template='plotly_dark' ) # Return the Plotly figure return fig logged_in_user_state = gr.State() # Authentication callback function def auth_callback(username, password): global logged_in_user valid_users = dict(auth_users) if username in valid_users and valid_users[username] == password: log_user_action(username, "Login") logged_in_user = username # Store the logged-in user globally return True return False def get_report_with_user(username): global logged_in_user return get_monthly_report(username, logged_in_user or "Unknown") # Custom layout using gr.Blocks with CSS with gr.Blocks(css="style.css", theme=gr.themes.Soft()) as app: gr.Markdown("# Chess Analysis App") username_input = gr.Textbox(label="Chess.com Username", placeholder="Enter Chess.com username") submit_button = gr.Button("Submit") # Define outputs output_data = gr.Dataframe(headers=["Month", "Games Played", "Wins", "Losses", "Win Rate (%)", "Loss Rate (%)", "Timeout Rate (%)"]) summary_text = gr.Textbox(label="Summary", interactive=False) # stacked_bar_img = gr.Image(label="Monthly Stacked Bar Chart with Timeouts") stacked_bar_img = gr.Plot(label="Monthly Stacked Bar Chart with Timeouts") # Link the click event submit_button.click( fn=get_report_with_user, inputs=[username_input], outputs=[output_data, summary_text, stacked_bar_img] ) app.launch(auth=auth_users, share=True)