import random
import pandas as pd
import trueskill
from itertools import combinations


def get_history(SCORING, players, df):
    env = trueskill.TrueSkill(
        mu=25, sigma=25 / 3, beta=25 / 6, tau=0, draw_probability=0.1
    )

    players = set(players)
    player_set = {player: env.Rating() for player in players}

    # Track mu and sigma for each player at each step
    history = {player: {"mu": [], "sigma": []} for player in players}

    for round_num in range(50):
        # Filter the DataFrame for the current round number
        round_df = df[df["round_number"] == round_num + 1]

        if round_df.empty:
            continue  # Skip if no data for this round

        # Get all unique pairs of players
        player_pairs = list(combinations(players, 2))

        # Iterate through each question in the current round
        for idx, row in round_df.iterrows():
            random.shuffle(
                player_pairs
            )  # Shuffle pairs to ensure randomness in processing order

            if SCORING == "all-v-all":
                # For all-v-all, we need to get all players' scores

                scores = {
                    player: row[player]
                    for player in players
                    if not pd.isna(row[player])
                }
                if len(scores) < 2:
                    continue
                sorted_scores_items = sorted(
                    scores.items(), key=lambda x: x[1], reverse=True
                )

                trueskill_ranks = []
                current_rank = 0  # TrueSkill ranks are 0-indexed for the best players
                processed_players = set()

                # Iterate through the sorted players
                for player_name, score in sorted_scores_items:
                    if player_name in processed_players:
                        continue  # Skip if already processed as part of a tie group

                    # Find all players with the exact same score
                    tied_players = []
                    for p_name, p_score in sorted_scores_items:
                        if p_score == score:
                            tied_players.append(p_name)

                            # Assign the current rank to this group
                            trueskill_ranks.append(current_rank)

                            processed_players.add(p_name)

                    # Increment rank for the next distinct score group
                    current_rank += 1

                output = trueskill.rate(
                    [(player_set[i[0]],) for i in sorted_scores_items],
                    ranks=trueskill_ranks,
                )  # lower score is better
                for idx, (player, rating) in enumerate(sorted_scores_items):
                    new_rating = output[idx][0]
                    player_set[player] = new_rating
                    history[player]["mu"].append(new_rating.mu)
                    history[player]["sigma"].append(new_rating.sigma)

            else:
                # get the scores of each player in each pair
                pairwise_scores = {}
                for player1, player2 in player_pairs:
                    score1 = row[player1]
                    score2 = row[player2]

                    r1 = player_set[player1]
                    r2 = player_set[player2]

                    if pd.isna(score1) or pd.isna(score2):
                        continue

                    if SCORING == "absolute":
                        if score1 >= 0.55 and score2 < 0.55:
                            new_r1, new_r2 = env.rate_1vs1(r1, r2)
                        elif score1 < 0.55 and score2 >= 0.55:
                            new_r2, new_r1 = env.rate_1vs1(r2, r1)
                        else:
                            new_r1, new_r2 = env.rate_1vs1(r1, r2, drawn=True)

                    elif SCORING == "relative":
                        if score1 - score2 > 0.05:
                            new_r1, new_r2 = env.rate_1vs1(r1, r2)
                        elif score1 - score2 < -0.05:
                            new_r2, new_r1 = env.rate_1vs1(r2, r1)
                        else:
                            new_r1, new_r2 = env.rate_1vs1(r1, r2, drawn=True)

                    player_set[player1] = new_r1
                    player_set[player2] = new_r2

                    history[player1]["mu"].append(player_set[player1].mu)
                    history[player1]["sigma"].append(player_set[player1].sigma)
                    history[player2]["mu"].append(player_set[player2].mu)
                    history[player2]["sigma"].append(player_set[player2].sigma)

    return history
