File size: 9,492 Bytes
0adb6ea
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
from datasets import load_dataset, Dataset
import os 
import json 
from datasets import load_dataset
from datasets.utils.logging import disable_progress_bar # type: ignore
from ui_constants import column_names, all_task_types
import random 
disable_progress_bar()
import math 
from sotopia_space.constants import MODEL_INFO

id_to_data = None 
model_len_info = None 


def make_clickable_model(model_name):
    global MODEL_INFO
    if model_name in MODEL_INFO:
        if MODEL_INFO[model_name]["hf_model_id"].startswith("http"):
            link = MODEL_INFO[model_name]["hf_model_id"]
            return f'πŸ”’ <a target="_blank" href="{link}" style="color: var(--link-text-color); text-decoration: underline;text-decoration-style: dotted;">{MODEL_INFO[model_name]["pretty_name"]}</a>'
        else:
            link = f"https://huggingface.co/{MODEL_INFO[model_name]['hf_model_id']}"
            return f'πŸ”₯ <a target="_blank" href="{link}" style="color: var(--link-text-color); text-decoration: underline;text-decoration-style: dotted;">{MODEL_INFO[model_name]["pretty_name"]}</a>'
    else:
        return model_name
    

def styled_error(error):
    return f"<p style='color: red; font-size: 20px; text-align: center;'>{error}</p>"

def styled_warning(warn):
    return f"<p style='color: orange; font-size: 20px; text-align: center;'>{warn}</p>"

def styled_message(message):
    return f"<p style='color: green; font-size: 20px; text-align: center;'>{message}</p>"
        

def estimated_win_rate(elo_a, elo_b, LP=0):
    """
    Calculate the estimated win rate for player A against player B using their Elo ratings.
    :param elo_a: Elo rating of player A
    :param elo_b: Elo rating of player B
    :return: Estimated win rate for player A
    """
    exponent = (elo_b - elo_a)*(10**LP) / 400
    probability_a_wins = 1 / (1 + 10 ** exponent)
    return (1-probability_a_wins)*100



# Formats the columns
def formatter(x):
    if type(x) is str:
        x = x
    else: 
        x = round(x, 1)
    return x


def add_winrates(current_df, LP=0):
    df = current_df.copy()
    elo_column = "Task-Avg Elo" 

    # Correct way to filter the DataFrame and get the Elo rating for "gpt-4-0125-preview"
    model_a_elo = df[df["Model"].str.contains("gpt-4")][elo_column].iloc[0]

    # Correct way to filter the DataFrame and get the Elo rating for "gpt-3.5-turbo-0125"
    model_b_elo = df[df["Model"].str.contains("gpt-3.5")][elo_column].iloc[0]

    
    # Calculate the win rate of "gpt-4-0125-preview" against all models
    df['Win% vs GPT-4'] = df[elo_column].apply(lambda x: estimated_win_rate(model_a_elo, x, LP=LP)).apply(formatter)    
    df['Win% vs GPT-3.5T'] = df[elo_column].apply(lambda x: estimated_win_rate(model_b_elo, x, LP=LP)).apply(formatter)    
    # apply the formatter for the two new columns 
    cols = list(df.columns)
    cols.remove("# battles"); cols.append("# battles")
    cols.remove("Length"); cols.append("Length")
    df = df[cols]
    return df

def add_winrates_tasks(current_df, ref="gpt-4", LP=0):
    new_df = current_df.copy()
    for t in all_task_types:
        column = column_names[t]
        model_a_elo = current_df[current_df["Model"].str.contains(ref)][column].iloc[0]
        new_df[column] = current_df[column].apply(lambda x: estimated_win_rate(model_a_elo, x, LP=LP)).apply(formatter)
    return new_df
        

def post_processing(df, model_len_info):
    if model_len_info:
        df["Length"] = df["model name "].apply(lambda x: model_len_info[x]["avg_len"])

    for col in df.columns:
        if col == "model name ":
            df[col] = df[col].apply(lambda x: x.replace(x, make_clickable_model(x)))
        else:
            df[col] = df[col].apply(formatter) # For numerical values 
    df.rename(columns=column_names, inplace=True)
    df.sort_values(by="Task-Avg Elo", inplace=True, ascending=False)
    # put the "Overall Elo" and "Task-Avg Elo" column to the front
    # add the length info
    df = df[["Model", "Task-Avg Elo"] + [col for col in df.columns if col not in ["Model", "Task-Avg Elo"]]]
    return df

def apply_length_penalty(original_df, ablation_df, length_penalty=0.2, mode='v1', LP_original_dfs=None):
    """
    Temporarily disable the length penalty feature
    if mode == 'v2' and LP_original_dfs is not None:
        L = f"{length_penalty:.1f}"
        return LP_original_dfs[L]
    original_df = original_df.copy()
    ablation_df = ablation_df.copy()
    # replace all values in original_df with the values as z = x - y * length_penalty where y is from ablation_df at the same row and column
    # except for the "Model" column and the "# battles" column 
    # do not assume the order of the rows are the same in both dataframes
    for i, row in original_df.iterrows():
        for col in original_df.columns:
            if col == "Model" or col == "# battles" or col == "Length":
                continue
            # assert that the model names are the same in both dataframes
            assert original_df.at[i, "Model"] == ablation_df[ablation_df["Model"] == row["Model"]]["Model"].values[0]
            original_df[col] = original_df[col].astype(float)
            if mode == "v1":
                original_df.at[i, col] = original_df.at[i, col] - ablation_df[ablation_df["Model"] == row["Model"]][col].values[0] * length_penalty
            elif mode == "v1.1":
                diff = original_df.at[i, col] - ablation_df[ablation_df["Model"] == row["Model"]][col].values[0] 
                original_df.at[i, col] = original_df.at[i, col] * (1-length_penalty) + diff*length_penalty
    # post_processing
    original_df = post_processing(original_df, model_len_info=None)
    """
    return original_df 

def load_benchdata():
    print("Loading sotopia data...")
    bench_data = load_dataset("cmu-lti/sotopia", split="test")
    return bench_data

def load_benchdata_dict():
    print("Loading sotopia data....")
    bench_data = load_dataset("cmu-lti/sotopia", data_files="sotopia_episodes_v1_hf.jsonl")['train']
    id_to_data = {}
    for item in bench_data:
        id_to_data[item["session_id"]] = item
    return id_to_data

def load_eval_results():
    print("Loading sotopia Evaluation data...")
    eval_results = load_dataset("WildEval/sotopia-Evaluation", "all", split="train")
    return eval_results

def load_infer_results(model_name):
    print(f"Loading sotopia Results for {model_name}...")
    infer_results = load_dataset("WildEval/sotopia-Results", model_name, split="train")
    return infer_results

def sample_an_eval_result(eval_results, model_list=[], tag_list=[]):
    global id_to_data          
    eval_results = list(eval_results)
    random.shuffle(eval_results)
    for eval_item in eval_results:  
        # print(json.dumps(eval_item, indent=2))
        # print(f"## Session ID: {eval_item['session_id']}")
        # eval_item["eval_id"]
        assignment = eval_item['assignment']
        model_1, model_2 = eval_item['model_1'], eval_item['model_2']
        model_A = model_1 if assignment['A'] == model_1 else model_2
        model_B = model_2 if assignment['B'] == model_2 else model_1
        if len(model_list) >= 2:
            if model_A not in model_list or model_B not in model_list:
                continue
        elif len(model_list) == 1:
            if model_A != model_list[0] and model_B != model_list[0]:
                continue
        else:
            pass 
        if tag_list:
            if set(tag_list).isdisjoint(set(eval_item['tags'])):
                continue
        winner = eval_item['winner']
        # print(f"## Model A: {model_A} | Model B: {model_B} | Winner: {winner}")
        task_type = eval_item['tags'][0] # primary task type
        chat_history = eval_item['history']
        last_query = eval_item['last_query']
        # print(f"## Task Type: {task_type}")
        # print(f"## Chat History: {chat_history}")
        # print(f"## Last Query -->  USER: {last_query}")

        model_A_output = eval_item['model_1_output'] if model_1 == model_A else eval_item['model_2_output']
        model_B_output = eval_item['model_2_output'] if model_2 == model_B else eval_item['model_1_output']

        if len(model_A_output.strip()) == 0 or len(model_B_output.strip()) == 0:
            continue

        conversation_input = id_to_data[eval_item['session_id']]["conversation_input"]
        # print(f"\n\n\n## Model A ({model_A}) Output ##\n{model_A_output}")
        # print(f"\n\n\n## Model B ({model_B}) Output ##\n{model_B_output}")

        # print(f"\n\n\n## Winner ##\n{winner}")
        # print(f"\n\n\n## GPT-4 Judgement ##\n{eval_item['parsed_result']}")

        result_dict = {
            "session_id": eval_item['session_id'],
            "model_A": model_A,
            "model_B": model_B,
            "winner": winner,
            "intent": id_to_data[eval_item['session_id']]["intent"],
            "task_type": task_type,
            "all_tags": eval_item['tags'],
            "chat_history": chat_history,
            "last_query": last_query,
            "conversation_input": conversation_input,
            "model_A_output": model_A_output,
            "model_B_output": model_B_output,
            "reason": eval_item['parsed_result']["reason"],
            "choice": eval_item['parsed_result']["choice"],
            "checklist": id_to_data[eval_item['session_id']]["checklist"],
        }
        break 
    return result_dict

#id_to_data = load_benchdata_dict()