import os
import json
import numpy as np
import pandas as pd
import math
import itertools
from scipy.stats import kendalltau

import matplotlib
import matplotlib.pyplot as plt
from matplotlib.pylab import savefig
import seaborn as sns
matplotlib.use('Agg')
sns.set_style("whitegrid")
plt.rcParams["font.family"] = 'DejaVu Sans'

from plot_utils import *

save_dir = 'graphs-separate-subtasks'
if not os.path.exists(save_dir):
    os.makedirs(save_dir)

num_source_models = 300
benchmarks = ['mmlu', 'mmlu-pro', 'bbh', 'gpqa']
techniques = ['Random (uniform)', 'Stratified sampling (confidence)', 'Diversity', 'Anchor Points', 'tinyBenchmarks']
all_fraction_sampled_points = [0.02, 0.04, 0.08, 0.16, 0.24, 0.32, 0.4]

f, axs = plt.subplots(len(benchmarks), len(techniques), figsize=(6.4*len(techniques), 3.6*len(benchmarks)))#, gridspec_kw={'height_ratios': [0.5, 0.5, 1]})
plt.subplots_adjust(wspace=0.25, hspace=0.15)    

for ax_row, benchmark in enumerate(benchmarks):

    cached_mdad_results_fn = f'{save_dir}/{benchmark}-mdad-results.csv'
    cached_estimation_results_fn = f'{save_dir}/{benchmark}-estimation-results.csv'

    if not os.path.exists(cached_mdad_results_fn) and not os.path.exists(cached_estimation_results_fn):
        print(f'Processed results files not generated for {benchmark}')
        continue
        
    df_mdad = pd.read_csv(cached_mdad_results_fn)
    full_estimation_df = pd.read_csv(cached_estimation_results_fn)

    # get difference between seen and unseen
    config_to_seen = {}
    config_to_unseen = {}
    for i, row in df_mdad.iterrows():
        config = (row['Technique'], row['Number of source models'], row['Fraction of sampled points'])
        m = row['MDAD']
        l = row['MDAD lower CI']
        u = row['MDAD upper CI']
        y1 = min(l, m-(u-m))
        y2 = max(m+(m-l), u)
        if row['Split'] == 'seen':
            config_to_seen[config] = (m, y1, y2)
        else:
            config_to_unseen[config] = (m, y1, y2)
    tidy_mdad_results = []
    for config in config_to_seen.keys():
        tidy_mdad_results.append({'Technique': config[0],
                                 'Number of source models': config[1],
                                 'Fraction of sampled points': config[2],
                                 'MDAD (seen split)': config_to_seen[config][0],
                                 'MDAD lower CI (seen split)': config_to_seen[config][1],
                                 'MDAD upper CI (seen split)': config_to_seen[config][2],
                                 'MDAD (unseen split)': config_to_unseen[config][0],
                                 'MDAD lower CI (unseen split)': config_to_unseen[config][1],
                                 'MDAD upper CI (unseen split)': config_to_unseen[config][2],
                                 'MDAD difference': config_to_unseen[config][0] - config_to_seen[config][0]})
    df_mdad = pd.DataFrame(tidy_mdad_results)

    all_fraction_sampled_points = [0.02, 0.04, 0.08, 0.16, 0.24, 0.32, 0.4]
    
    for ax_col, technique in enumerate(techniques):
        ax = axs[ax_row, ax_col]
        plot_df = df_mdad.loc[(df_mdad['Number of source models'] == num_source_models) \
                        & (df_mdad['Technique'] == technique)]
        sns.lineplot(ax=ax, data=plot_df, x='Fraction of sampled points', y=f"MDAD (seen split)", hue="Technique", palette=palette, marker='o', markersize=10, alpha=0.8, linewidth=4, zorder=10, clip_on=True)
        sns.lineplot(ax=ax, data=plot_df, x='Fraction of sampled points', y=f"MDAD (unseen split)", hue="Technique", palette=palette, linestyle='--', marker='s', markersize=10, alpha=0.8, linewidth=4, zorder=10, clip_on=True)
        for split in ['seen', 'unseen']:
            for t in techniques:
                technique_df = plot_df.loc[(plot_df['Technique'] == t)]
                data = sorted(zip(technique_df['Fraction of sampled points'].tolist(), technique_df[f'MDAD lower CI ({split} split)'].tolist(),  technique_df[f'MDAD upper CI ({split} split)'].tolist()))
                xs = [x for x, _, _, _ in data]
                y1s = [l for _, _, l, _ in data]
                y2s = [u for _, _, _, u in data]
                ax.fill_between(xs, y1s, y2s, color=palette[technique], alpha=0.2)
        ax.set_ylim((0, 0.5))
        ax.set_xlim((0, max(all_fraction_sampled_points) + 0.02))
        ax.set_xticks(all_fraction_sampled_points)
        ax.set_xticklabels(['2%', '4%', '8%', '16%', '24%', '32%', '40%'])
        yticks = ax.get_yticks()
        yticklabels = [f'{y * 100:.0f}' for y in yticks]
        ax.set_yticks(yticks)
        ax.set_yticklabels(yticklabels)    
        ax.set_xlabel('')
        ax.set_ylabel('')
        if ax_col == 0:
            ax.set_ylabel(f'MDAD', fontsize=14, labelpad=20, fontweight='bold')
            ax.text(-0.1, 0.5, '↓', fontsize=24, fontweight='bold', va='center', ha='center', transform=ax.transAxes)
            ax.text(-0.25, 0.5, benchmark_to_display_name[benchmark], transform=ax.transAxes, rotation_mode='anchor', va='center', ha='center', fontsize=24, fontweight='bold', rotation=90)
        legend = ax.get_legend()
        if legend is not None:
            legend.remove()
        if ax_row == 0:
            ax.set_title(technique, fontsize=18, fontweight='bold', pad=10)
        if ax_row == len(benchmarks) - 1:
            ax.set_xlabel('Percent of subtask selected for micro-benchmark', fontsize=10, fontweight='bold')
            
ls = [matplotlib.patches.Patch(facecolor=palette[t]) for t in techniques]
line1 = matplotlib.lines.Line2D([], [], color='black', linewidth=4,)
line2 = matplotlib.lines.Line2D([], [], color='black', linewidth=4, linestyle='--')
plt.legend([line1, line2], ['Micro-benchmark on full dataset used to construct it', 'Micro-benchmark on new draw of the task'], bbox_to_anchor=(0.5, 0.05), loc='upper center', bbox_transform=f.transFigure, borderaxespad=0., fontsize=20, ncols=2)#, prop={'weight':'bold'})
savefig(f'{save_dir}/figure-6_all-benchmarks.pdf', bbox_inches='tight')