# -*- coding: utf-8 -*-
"""
Created on Tue Mar 18 11:41:34 2025

@author: baran
"""

# -*- coding: utf-8 -*-
"""
Created on Thu Feb 13 21:46:41 2025

@author: baran
"""

# -*- coding: utf-8 -*-

import numpy as np
from optimal_rand_tele import opt_eval
from final_rand_med import final_eval
from prompt_maker import input_maker
from sum_call import get_summary
import pickle
# from sum_call import get_summary

# def get_optimal_super_arm_reward(deployments,prompt,task):
    
#     #call Azure API to get optimal reward by trying out all combinations

#     return opt_eval(deployments, prompt,task)

# def get_reward(deployment,cat,prompt,task,all_rewards_sum,all_rewards_diag):
    
#     #call Azure API to get final reward
    
#     return final_eval(deployment, cat, prompt,task,all_rewards_sum,all_rewards_diag)
    
def get_optimal_super_arm_reward(deployments,prompt,task):
    
    #call Azure API to get optimal reward by trying out all combinations

    return opt_eval(deployments, prompt,task)

def get_reward(deployment,cat,prompt,task,all_rewards_sum,all_rewards_diag):
    
    #call Azure API to get final reward
    
    return final_eval(deployment, cat, prompt,task,all_rewards_sum,all_rewards_diag)
    

def get_regret(deployments,prompt,task,selected,avg_array,t,all_rewards_sum,all_rewards_diag,labels,dataset):
    #return get_optimal_super_arm_reward(deployments,prompt,task)-reward
    return opt_eval(deployments, prompt,task,selected,avg_array,t,all_rewards_sum,all_rewards_diag,labels,dataset)
# def sigmoid(x):
#     return np.where(x >= 0,
#                     1 / (1 + np.exp(-x)),
#                     np.exp(x) / (1 + np.exp(x)))

# def oracle(budget, g_list):
#     return np.argsort(g_list)[-budget:]

with open('diagnoses_100.pkl', 'rb') as file: 

    # Call load method to deserialze 
    diagnoses = pickle.load(file)

with open('input_reports_100.pkl', 'rb') as file: 

    # Call load method to deserialze 
    input_reports = pickle.load(file)

new_labels = diagnoses
        
    
    
    
import scipy as sp
import torch
import torch.nn as nn
import torch.optim as optim
from backpack import backpack, extend
from backpack.extensions import BatchGrad
import argparse
from embed_tele import get_context
class Network(nn.Module):
    def __init__(self, dim = 100, hidden_size=100):
        super(Network, self).__init__()
        #nn.Module.__init__(self)
        #self.fc1 = nn.Linear(dim, hidden_size)
        #self.activate = nn.ReLU()
        #self.fc2 = nn.Linear(hidden_size, 1)
        self.model = nn.Sequential(nn.Linear(dim, hidden_size),nn.ReLU(),nn.Linear(hidden_size, 1))
        #self.model = nn.Sequential(nn.Linear(dim, dim),nn.ReLU(),nn.Linear(dim, hidden_size),nn.ReLU(),nn.Linear(hidden_size, 1))
    #def forward(self, x):

    #    return self.fc2(self.activate(self.fc1(x)))


class NeuralUCBDiag:
    def __init__(self, style, dim, lamdba=1, nu=1, hidden=100):
        self.device = 'cuda' if torch.cuda.is_available() else 'cpu'
        if self.device =='cuda':
            self.func = extend(Network(dim, hidden_size=hidden).model.cuda())
        else:
            self.func = Network(dim, hidden_size=hidden)
        self.context_list = []
        self.reward = []
        self.lamdba = lamdba
        #for p in self.func.parameters():
          #print(p.size())
        self.total_param = sum(p.numel() for p in self.func.parameters() if p.requires_grad)
        self.device = 'cuda' if torch.cuda.is_available() else 'cpu'
        if self.device =='cuda':
            self.U = lamdba * torch.ones((self.total_param,)).cuda()
            self.U_random = lamdba * torch.ones((self.total_param,)).cuda()
        else:
            self.U = lamdba * torch.ones((self.total_param,))
            self.U_random = lamdba * torch.ones((self.total_param,))
        self.nu = nu
        self.num_rounds = 100
        self.style = style
        
        if self.device =='cuda':
            self.loss_func = extend(nn.MSELoss().cuda())
        else:
            self.loss_func = extend(nn.MSELoss())
        self.len = 0

    def update_params(self,g_list):
        #print(g_list.size())
        for g in g_list:
            self.U += g * g
        return 0
    
    # def selection(self,context):
    #     tensor = torch.from_numpy(np.array(context)).float().cuda()
    #     mu = self.func(tensor)
    #     g_list = []
    #     #sum_mu = torch.sum(mu)
    #     #with backpack(BatchGrad()):
    #     #    sum_mu.backward()
    #     sampled = []
    #     for fx in mu:
    #         self.func.zero_grad()
    #         fx.backward(retain_graph=True)
    #         g = torch.cat([p.grad.flatten().detach() for p in self.func.parameters()])
    #         g_list.append(g)
    #         sigma2 = self.lamdba * self.nu * g * g / self.U
    #         sigma = torch.sqrt(torch.sum(sigma2))
    #         sample_r = fx.item() + 0.1* sigma.item()   
    #         sampled.append(sample_r)
    #     arm = np.argmax(sampled)
    #     print(arm)
    #     print(sampled)
    #     self.U += g_list[arm] * g_list[arm]
    #     return arm
    #         # g_list = torch.cat([p.grad_batch.flatten(start_dim=1).detach() for p in self.func.parameters()], dim=1)
    #         # sigma = torch.sqrt(torch.sum(self.lamdba * self.nu * g_list * g_list / self.U, dim=1))
    #         # if self.style == 'ts':
    #         #     sample_r = torch.normal(mu.view(-1), 0.01*sigma.view(-1))
    #         # elif self.style == 'ucb':
    #         #     sample_r = mu.view(-1) + 0.1*sigma.view(-1)
    
    
    def selection(self,context):
        tensor = torch.from_numpy(np.array(context)).float().cuda()
        mu = self.func(tensor)
        #g_list = []
        #sum_mu = torch.sum(mu)
        #with backpack(BatchGrad()):
        #    sum_mu.backward()
        #sampled = []
        # for fx in mu:
        self.func.zero_grad()
        mu.backward(retain_graph=True)
        g = torch.cat([p.grad.flatten().detach() for p in self.func.parameters()])
        #g_list.append(g)
        sigma2 = self.lamdba * self.nu * g * g / self.U
        sigma = torch.sqrt(torch.sum(sigma2))
        #sample_r = 0.1*mu.item() +5* sigma.item() 
        sample_r = torch.normal(mu.view(-1), 0.01*sigma.view(-1))
        #sample_r = mu.item()
        #sampled.append(sample_r)
        #arm = np.argmax(sampled)
        #print(arm)
        #print(sampled)
        self.U += g * g
        return sample_r
            # g_list = torch.cat([p.grad_batch.flatten(start_dim=1).detach() for p in self.func.parameters()], dim=1)
            # sigma = torch.sqrt(torch.sum(self.lamdba * self.nu * g_list * g_list / self.U, dim=1))
            # if self.style == 'ts':
            #     sample_r = torch.normal(mu.view(-1), 0.01*sigma.view(-1))
            # elif self.style == 'ucb':
            #     sample_r = mu.view(-1) + 0.1*sigma.view(-1)
            
    # def selection(self,context):
    #     tensor = torch.from_numpy(np.array(context)).float().cuda()
    #     # tensor = torch.from_numpy(np.transpose(context)).float()
    #     #for arm in available_arms:
    #     #    print(arm.unique_id)

    #     mu = self.func(tensor)
    #     sum_mu = torch.sum(mu)

    #     with backpack(BatchGrad()):
    #         sum_mu.backward()
    #     g_list = torch.cat([p.grad_batch.flatten(start_dim=1).detach() for p in self.func.parameters()], dim=1)
    #     sigma = torch.sqrt(torch.sum(self.lamdba * self.nu * g_list * g_list / self.U, dim=1))
    #     if self.style == 'ts':
    #         sample_r = torch.normal(mu.view(-1), 0.01*sigma.view(-1))
    #     elif self.style == 'ucb':
    #         sample_r = mu.view(-1) + 0.1*sigma.view(-1)
    #     self.U += g * g
    #     return sample_r
    
    def train(self, context, reward):
        self.context_list.append(torch.from_numpy(context.reshape(1, -1)).float())
        self.reward.append(reward)
        optimizer = optim.SGD(self.func.parameters(), lr=1e-4, weight_decay=self.lamdba)
        length = len(self.reward)
        index = np.arange(length)
        np.random.shuffle(index)
        cnt = 0
        tot_loss = 0
        while True:
            batch_loss = 0
            for idx in index:
                c = self.context_list[idx]
                r = self.reward[idx]
                optimizer.zero_grad()
                if self.device =='cuda':
                    delta = self.func(c.cuda()) - torch.tensor(float(r))
                else:
                    delta = self.func(c) - torch.tensor(float(r))
                loss = delta * delta
                loss.backward()
                optimizer.step()
                batch_loss += loss.item()
                tot_loss += loss.item()
                cnt += 1
                if cnt >= 5:
                    return tot_loss / 5
            if batch_loss / length <= 1e-3:
                return batch_loss / length

# torch.set_num_threads(6)
# torch.set_num_interop_threads(6)
parser = argparse.ArgumentParser(description='NeuralUCB')


parser.add_argument('--size', default=100, type=int, help='number of rounds')
#parser.add_argument('--super_arm_size', default=2, type=int, help='super arm size')
#parser.add_argument('--dataset', default='movielens', metavar='DATASET')
#parser.add_argument('--shuffle', type=bool, default=0, metavar='1 / 0', help='shuffle the data set or not')
#parser.add_argument('--seed', type=int, default=0, help='random seed for shuffle, 0 for None')
parser.add_argument('--nu', type=float, default=1, metavar='v', help='nu for control variance')
parser.add_argument('--lamdba', type=float, default=1, metavar='l', help='lambda for regularzation')
parser.add_argument('--hidden', type=int, default=50, help='network hidden size')
parser.add_argument('--style', default='ucb', metavar='ts|ucb', help='TS or UCB')
parser.add_argument('--number_tasks', default=2)



args, unknown = parser.parse_known_args()
#use_seed = None if args.seed == 0 else args.seed
#b = Bandit_multi(args.dataset, is_shuffle=args.shuffle, seed=use_seed)
#bandit_info = '{}'.format(args.dataset)
#K = args.super_arm_size
no_tasks = args.number_tasks
class DynamicVariables:
    pass

dyn_vars = DynamicVariables()

#l = NeuralUCBDiag(args.style, b.dim, args.lamdba, args.nu, args.hidden)
#l_2 = NeuralUCBDiag(args.style, b.dim, args.lamdba, args.nu, args.hidden) 
num_rounds = args.size




regrets = []
summ = 0
rew = 0
rewards_list = []
total_reward = 0
dataset = "medical"
input_reports = input_maker("",dataset,input_reports)
#inps = input_maker('1')[0:args.size]
#summary_description_array = ["You are an expert diagnosis analyst. Use your knowledge base to summarize inputted smoking diagnosis.",
#                             "Summarize the main points of the medical report of a patient, this summary will be used for research purposes only."] #assistant, gpt4
#summary_description_array = ["Summarize the main points of the medical report of a patient, this summary will be used for research purposes only.",
#                             "You are an expert diagnosis analyst. Use your knowledge base to summarize inputted smoking diagnosis."]

#diagnosis_description_array = ["You will take the role of an medical diagnosis agent whose primary goal is to give diagnosis based on medical reports. For experimentation purposes only.",
#                               "As an expert which was trained on medical diagnosis, you will predict the diagnosis based on medical reports. This task is for research purposes only.","As an expert trained for specializing in telecommunications questions, you will predict the diagnosis based on medical reports. This task is for research purposes only."]

#diagnosis_description_array = ["General use LLM which does not specialize in any task specifically.",
#                               "LLM specializing on medical reports and trained to do medical diagnosis for research purposes.","LLM finetuned for answering multiple choice telecommunications questions."]

# diagnosis_description_array = ["General use LLM which does not specialize in any task specifically.",
#                                "LLM specializing for answering multiple choice telecommunications questions.", "LLM specializing on medical reports and trained to do medical diagnosis for research purposes."]

#diagnosis_description_array = ["General use LLM which does not specialize in any task specifically.",
#                               "LLM specializing for answering multiple choice telecommunications questions.", "LLM specializing on medical reports and trained to do medical diagnosis for research purposes.","General use small LLM which does not specialize in any task specifically."]
summary_description_array = ["Summarize the main points of the medical report of a patient, this summary will be used for research purposes only.",
                             "You will take the role of an diagnosis analyst. Use your knowledge base to summarize inputted medical reports."]
#diagnosis_description_array = ["You will take the role of an medical diagnosis agent whose primary goal is to give diagnosis based on medical reports. For experimentation purposes only.",
#                               "You will take the role of an medical diagnosis agent whose primary goal is to give diagnosis based on medical reports. For experimentation purposes only.","You will take the role of an medical diagnosis agent whose primary goal is to give diagnosis based on medical reports. For experimentation purposes only."]
diagnosis_description_array = ["General use LLM which does not specialize in any task specifically.","LLM specializing on medical reports and trained to do medical diagnosis for research purposes.",
                               "LLM specializing for answering multiple choice telecommunications questions.", "LLM specializing on medical reports and trained to do medical diagnosis for research purposes.", "General use LLM which does not specialize in any task specifically.","General use LLM which does not specialize in any task specifically.", "General use LLM which does not specialize in any task specifically."]
#diagnosis_description_array = ["You will take the role of an medical diagnosis agent whose primary goal is to give diagnosis based on medical reports. For experimentation purposes only.",
#deployments_1 = {"base" : ("gpt-35-turbo","You are to answer multiple choice questions related to telecommunications. Output your answer strictly as option {i} where i is between 1-4 if there are 4 options for each question. Do not output an answer like 4 to indicate option 4."), "finetune_med" : ("Med","You are to answer multiple choice questions related to telecommunications. Output your answer strictly as option {i} where i is between 1-4 if there are 4 options for each question. Do not output an answer like 4 to indicate option 4."), "finetune_tele" : ("Tele","You are to answer multiple choice questions related to telecommunications. Output your answer strictly as option {i} where i is between 1-4 if there are 4 options for each question. Do not output an answer like 4 to indicate option 4."),"small": ("SLM","You are to answer multiple choice questions related to telecommunications. Output your answer strictly as option {i} where i is between 1-4 as there are 4 options for each question. Do not output an answer like 4 to indicate option 4.")}
#deployments_1 = {"base" : ("gpt-35-turbo","You are an medical diagnosis agent whose primary goal is to give diagnosis based on medical reports. For experimentation purposes only."), "finetune_med" : ("Med","You are an medical diagnosis agent whose primary goal is to give diagnosis based on medical reports. For experimentation purposes only."),"finetune_tele" : ("Tele","You are an medical diagnosis agent whose primary goal is to give diagnosis based on medical reports. For experimentation purposes only.")}
#deployments_0 = {"base" : ("gpt-35-turbo","You are to summarize an inputted medical report for diagnosis purposes, this summary will be used for research purposes only."), "assistants" : ("Assistant","You are to summarize an inputted medical report for diagnosis purposes, this summary will be used for research purposes only.")}
deployments_0 = {"base" : ("gpt-35-turbo","You are to summarize an inputted medical report, this summary will be used for research purposes only."), "assistants" : ("Assistant","You are to summarize an inputted medical report, this summary will be used for research purposes only.")}
deployments_1 = {"base" : ("gpt-35-turbo","You will take the role of an medical agent whose primary goal is to give diagnosis based on medical reports for experimentation and research purposes only. You need to make inferences based on the provided report to make diagnosis predictions. Output at most 2 diagnoses. If you identify multiple diagnosis output them in a comma separated format like heart failure,colon cancer. Your given diagnoses for the patient must be one of the following: diabetes mellitus,huntington's disease,sepsis,encephalopathy,pulmonary embolism,pulmonary edema,tamponade,mitral stenosis,congestive heart failure,chronic obstructive pulmonary disease,abdominal thoracic aneurysm,neurosarcoidosis,renal failure,svc syndrome,urosepsis,acute myocardial infarction,acute coronary syndrome,carotid stenosis,aortic stenosis,coronary artery disease,osteoarthritis,aortic insufficiency,unstable angina/cath,hyperlipidemia,syncope,complete heart block,intravascular coagulation,septic shock,hepatic failure,pneumonia,pancreatitis,anemia,catheter tip infection,coma,urinary tract infection,wound infection,cerebral artery infarction,hyponatremia,cardiomyopathy,vertebral/basilar stenosis,hypoxia,sick sinus syndrome,pulmonary congestion,aseptic meningitis,neutropenia,cellulitis,cirrhosis of liver,liver failure,pericardial effusion,aortic valve dysfunction,venous thrombosis,respiratory failure,benzodiazepine overdose,vena cava obstruction,valvular heart disease,v-tach,aortic dissection,opiate intoxication"), 
                 "finetune_med" : ("Med","You will take the role of an medical agent whose primary goal is to give diagnosis based on medical reports for experimentation and research purposes only. You need to make inferences based on the provided report to make diagnosis predictions. Output at most 2 diagnoses. If you identify multiple diagnosis output them in a comma separated format like heart failure,colon cancer. Your given diagnoses for the patient must be one of the following: diabetes mellitus,huntington's disease,sepsis,encephalopathy,pulmonary embolism,pulmonary edema,tamponade,mitral stenosis,congestive heart failure,chronic obstructive pulmonary disease,abdominal thoracic aneurysm,neurosarcoidosis,renal failure,svc syndrome,urosepsis,acute myocardial infarction,acute coronary syndrome,carotid stenosis,aortic stenosis,coronary artery disease,,osteoarthritis,aortic insufficiency,unstable angina/cath,hyperlipidemia,syncope,complete heart block,intravascular coagulation,septic shock,hepatic failure,pneumonia,pancreatitis,anemia,catheter tip infection,coma,urinary tract infection,wound infection,cerebral artery infarction,hyponatremia,cardiomyopathy,vertebral/basilar stenosis,hypoxia,sick sinus syndrome,pulmonary congestion,aseptic meningitis,neutropenia,cellulitis,cirrhosis of liver,liver failure,pericardial effusion,aortic valve dysfunction,venous thrombosis,respiratory failure,benzodiazepine overdose,vena cava obstruction,valvular heart disease,v-tach,aortic dissection,opiate intoxication"),
                 "finetune_tele" : ("Tele","You will take the role of an medical agent whose primary goal is to give diagnosis based on medical reports for experimentation and research purposes only. You need to make inferences based on the provided report to make diagnosis predictions. Output at most 2 diagnoses. If you identify multiple diagnosis output them in a comma separated format like heart failure,colon cancer. Your given diagnoses for the patient must be one of the following: diabetes mellitus,huntington's disease,sepsis,encephalopathy,pulmonary embolism,pulmonary edema,tamponade,mitral stenosis,congestive heart failure,chronic obstructive pulmonary disease,abdominal thoracic aneurysm,neurosarcoidosis,renal failure,svc syndrome,urosepsis,acute myocardial infarction,acute coronary syndrome,carotid stenosis,aortic stenosis,coronary artery disease,,osteoarthritis,aortic insufficiency,unstable angina/cath,hyperlipidemia,syncope,complete heart block,intravascular coagulation,septic shock,hepatic failure,pneumonia,pancreatitis,anemia,catheter tip infection,coma,urinary tract infection,wound infection,cerebral artery infarction,hyponatremia,cardiomyopathy,vertebral/basilar stenosis,hypoxia,sick sinus syndrome,pulmonary congestion,aseptic meningitis,neutropenia,cellulitis,cirrhosis of liver,liver failure,pericardial effusion,aortic valve dysfunction,venous thrombosis,respiratory failure,benzodiazepine overdose,vena cava obstruction,valvular heart disease,v-tach,aortic dissection,opiate intoxication"), 
                 "finetune_med_new": ("Med_New","You will take the role of an medical agent whose primary goal is to give diagnosis based on medical reports for experimentation and research purposes only. You need to make inferences based on the provided report to make diagnosis predictions. Output at most 2 diagnoses. If you identify multiple diagnosis output them in a comma separated format like heart failure,colon cancer. Your given diagnoses for the patient must be one of the following: diabetes mellitus,huntington's disease,sepsis,encephalopathy,pulmonary embolism,pulmonary edema,tamponade,mitral stenosis,congestive heart failure,chronic obstructive pulmonary disease,abdominal thoracic aneurysm,neurosarcoidosis,renal failure,svc syndrome,urosepsis,acute myocardial infarction,acute coronary syndrome,carotid stenosis,aortic stenosis,coronary artery disease,osteoarthritis,aortic insufficiency,unstable angina/cath,hyperlipidemia,syncope,complete heart block,intravascular coagulation,septic shock,hepatic failure,pneumonia,pancreatitis,anemia,catheter tip infection,coma,urinary tract infection,wound infection,cerebral artery infarction,hyponatremia,cardiomyopathy,hypoxia,vertebral/basilar stenosis,sick sinus syndrome,pulmonary congestion,aseptic meningitis,neutropenia,cellulitis,cirrhosis of liver,liver failure,pericardial effusion,aortic valve dysfunction,venous thrombosis,respiratory failure,benzodiazepine overdose,vena cava obstruction,valvular heart disease,v-tach,aortic dissection,opiate intoxication"),
                 "small": ("SLM","You will take the role of an medical agent whose primary goal is to give diagnosis based on medical reports for experimentation and research purposes only. You need to make inferences based on the provided report to make diagnosis predictions. Output at most 2 diagnoses. If you identify multiple diagnosis output them in a comma separated format like heart failure,colon cancer. Your given diagnoses for the patient must be one of the following: diabetes mellitus,huntington's disease,sepsis,encephalopathy,pulmonary embolism,pulmonary edema,tamponade,mitral stenosis,congestive heart failure,chronic obstructive pulmonary disease,abdominal thoracic aneurysm,neurosarcoidosis,renal failure,svc syndrome,urosepsis,acute myocardial infarction,acute coronary syndrome,carotid stenosis,aortic stenosis,coronary artery disease,osteoarthritis,aortic insufficiency,unstable angina/cath,hyperlipidemia,syncope,complete heart block,intravascular coagulation,septic shock,hepatic failure,pneumonia,pancreatitis,anemia,catheter tip infection,coma,urinary tract infection,wound infection,cerebral artery infarction,hyponatremia,cardiomyopathy,hypoxia,vertebral/basilar stenosis,sick sinus syndrome,pulmonary congestion,aseptic meningitis,neutropenia,cellulitis,cirrhosis of liver,liver failure,pericardial effusion,aortic valve dysfunction,venous thrombosis,respiratory failure,benzodiazepine overdose,vena cava obstruction,valvular heart disease,v-tach,aortic dissection,opiate intoxication"),
                 "llama": ("llama","You will take the role of an medical agent whose primary goal is to give diagnosis based on medical reports for experimentation and research purposes only. You need to make inferences based on the provided report to make diagnosis predictions. Output at most 2 diagnoses. If you identify multiple diagnosis output them in a comma separated format like heart failure,colon cancer. Your given diagnoses for the patient must be one of the following: diabetes mellitus,huntington's disease,sepsis,encephalopathy,pulmonary embolism,pulmonary edema,tamponade,mitral stenosis,congestive heart failure,chronic obstructive pulmonary disease,abdominal thoracic aneurysm,neurosarcoidosis,renal failure,svc syndrome,urosepsis,acute myocardial infarction,acute coronary syndrome,carotid stenosis,aortic stenosis,coronary artery disease,osteoarthritis,aortic insufficiency,unstable angina/cath,hyperlipidemia,syncope,complete heart block,intravascular coagulation,septic shock,hepatic failure,pneumonia,pancreatitis,anemia,catheter tip infection,coma,urinary tract infection,wound infection,cerebral artery infarction,hyponatremia,cardiomyopathy,hypoxia,vertebral/basilar stenosis,sick sinus syndrome,pulmonary congestion,aseptic meningitis,neutropenia,cellulitis,cirrhosis of liver,liver failure,pericardial effusion,aortic valve dysfunction,venous thrombosis,respiratory failure,benzodiazepine overdose,vena cava obstruction,valvular heart disease,v-tach,aortic dissection,opiate intoxication"),
                 "phi": ("phi","You will take the role of an medical agent whose primary goal is to give diagnosis based on medical reports for experimentation and research purposes only. You need to make inferences based on the provided report to make diagnosis predictions. Output at most 2 diagnoses. If you identify multiple diagnosis output them in a comma separated format like heart failure,colon cancer. Your given diagnoses for the patient must be one of the following: diabetes mellitus,huntington's disease,sepsis,encephalopathy,pulmonary embolism,pulmonary edema,tamponade,mitral stenosis,congestive heart failure,chronic obstructive pulmonary disease,abdominal thoracic aneurysm,neurosarcoidosis,renal failure,svc syndrome,urosepsis,acute myocardial infarction,acute coronary syndrome,carotid stenosis,aortic stenosis,coronary artery disease,osteoarthritis,aortic insufficiency,unstable angina/cath,hyperlipidemia,syncope,complete heart block,intravascular coagulation,septic shock,hepatic failure,pneumonia,pancreatitis,anemia,catheter tip infection,coma,urinary tract infection,wound infection,cerebral artery infarction,hyponatremia,cardiomyopathy,hypoxia,vertebral/basilar stenosis,sick sinus syndrome,pulmonary congestion,aseptic meningitis,neutropenia,cellulitis,cirrhosis of liver,liver failure,pericardial effusion,aortic valve dysfunction,venous thrombosis,respiratory failure,benzodiazepine overdose,vena cava obstruction,valvular heart disease,v-tach,aortic dissection,opiate intoxication")}

emb_size = 384
total_len = len(summary_description_array)+len(diagnosis_description_array)
for i in range(no_tasks):
#     # setattr(dyn_vars, f'l_{i}', NeuralUCBDiag(args.style, b.dim, args.lamdba, args.nu, args.hidden))
    setattr(dyn_vars, f'l_{i}', NeuralUCBDiag(args.style,emb_size, args.lamdba, args.nu, args.hidden))
#     setattr(dyn_vars, f'normalized_{i}', [])
#l = NeuralUCBDiag(args.style,emb_size, args.lamdba, args.nu, args.hidden)

#deployments_1 = {"base" : ("gpt-35-turbo","You are an medical diagnosis agent whose primary goal is to give diagnosis based on medical reports. For experimentation purposes only."), "finetune_med" : ("Med","You are an medical diagnosis agent whose primary goal is to give diagnosis based on medical reports. For experimentation purposes only."),"finetune_tele" : ("Tele","You are an medical diagnosis agent whose primary goal is to give diagnosis based on medical reports. For experimentation purposes only.")}
deploy = [deployments_0,deployments_1]
cat = ''
rewards = 0
#input_reports = input_maker()
#from gensim.models.doc2vec import Doc2Vec, TaggedDocument
input_reports = list(input_reports)
sum_len = len(summary_description_array)
documents = summary_description_array+diagnosis_description_array+input_reports
#documents = input_reports
from sentence_transformers import SentenceTransformer
inp_model = SentenceTransformer("paraphrase-MiniLM-L6-v2")
#tokenizer = AutoTokenizer.from_pretrained("pritamdeka/BioMedBERT-finetuned-biomedical")
#model = AutoModel.from_pretrained("pritamdeka/BioMedBERT-finetuned-biomedical")
#inp_model = SentenceTransformer("pritamdeka/BioMedBERT-finetuned-biomedical")

# window = 20
# epoch = 40
# tagged_data = [TaggedDocument(words=doc.split(), tags=[str(i)]) for i, doc in enumerate(documents)]
# inp_model = Doc2Vec(vector_size=args.size,  # Dimensionality of the feature vectors
#             window=window,         # Context window size
#             min_count=2,      # Ignores words with total frequency lower than this
#             workers=4,        # Number of worker threads for training
#             epochs=epoch)        # Number of training epochs

# # Build vocabulary from tagged data
# inp_model.build_vocab(tagged_data)

# # Train the model
# inp_model.train(tagged_data, total_examples=inp_model.corpus_count, epochs=inp_model.epochs)



diag_len = len(diagnosis_description_array)
all_rewards_sum = []
all_rewards_diag = []
#gpt4, finetuned medical, finetuned telecom
#exploration = list(np.zeros(total_len))
plays_no = np.ones(total_len)
avg_array = {"gpt-35-turbo":0,"Med":0,"Tele":0,"Med_New":0,"SLM":0,"llama":0,"phi":0}
dataset = "medical"
#avg_array = {"gpt-35-turbo":0,"Med":0,"Tele":0,"SLM":0}
#i = 0
for t in range(num_rounds):
    for i in range(no_tasks):
    # for i in range(no_tasks):
        # if i==0:
            #context = np.zeros(len(summary_description_array))
        context = []
        if i==0:
            models = ["assistants","base"]
        #models = ["base","finetune_med","finetune_tele","small"]
            #models = ["base","finetune_med","finetune_tele","small","llama","phi"]
        
            #models = ["base","assistants"]
            prompt_to_model = input_reports[t]
            task = 'summary'
            values = []
            l = getattr(dyn_vars, f'l_{i}')
            for j in range(len(summary_description_array)):
                #context[j] = get_context(input_reports[t],summary_description_array[j])
                #l = getattr(dyn_vars, f'l_{j}')
                cont = get_context(documents,t,i,j,len(summary_description_array),len(diagnosis_description_array),len(input_reports),inp_model,dataset)
                context.append(cont)
                values.append(l.selection(cont))
            values_np = [v.cpu().detach().numpy() if isinstance(v, torch.Tensor) else v for v in values]
            #values = np.array(values)
            #exp = 1./plays_no
            #values_w_exp = np.array(values)+exp[0:sum_len]
            #arm = np.argmax(values_w_exp)
            #arm = np.argmax(values)
            arm = np.random.choice(np.where(np.array(values_np) == np.array(values_np).max())[0])
            plays_no[arm] += 1 
            arm_select = models[arm]
        
        else:
            models = ["base","finetune_med","finetune_tele","finetune_med_new","small","llama","phi"]
            values=[]
            task = 'diagnosis'
            prompt_to_model = get_summary(input_reports[t],cat)
            #print(get_summary(input_reports[t],cat))
            prompt_to_model = prompt_to_model.replace('\n','')
            print(prompt_to_model)
            #task = 'diagnosis'
            documents = documents+[prompt_to_model]
            l = getattr(dyn_vars, f'l_{i}')
            for j in range(len(diagnosis_description_array)):
                #context[j] = get_context(input_reports[t],summary_description_array[j])
                #l = getattr(dyn_vars, f'l_{j}')
                cont = get_context(documents,t,i,j,len(summary_description_array),len(diagnosis_description_array),len(input_reports),inp_model,dataset)
                context.append(cont)
                values.append(l.selection(cont))
            values_np = [v.cpu().detach().numpy() if isinstance(v, torch.Tensor) else v for v in values]
            #values = np.array(values)
            #exp = 1./plays_no
            #values_w_exp = np.array(values)+exp[0:sum_len]
            #arm = np.argmax(values_w_exp)
            #arm = np.argmax(values)
            arm = np.random.choice(np.where(np.array(values_np) == np.array(values_np).max())[0])
            plays_no[arm+sum_len] += 1 
            arm_select = models[arm]
    
        print(values)
        print(arm_select)
        #print(models[arm_select])
        if models[arm] == "finetune_med" or models[arm]=="finetune_tele" or models[arm]=="finetune_med_new":
            cat = "finetune"
            #print('11')
        else:    
            cat = models[arm]
        dep = deploy[i]
    
        selected = arm_select
        fin_prompt = prompt_to_model
        #print(fin_prompt)
        #print(selected)
        deployment = dep[selected]
        #print(opt_eval.__code__.co_varnames)
        #print(len(opt_eval.__code__.co_varnames))
    
        if i==no_tasks-1:
            reg,reward,avg_array,all_rewards_sum,all_rewards_diag = get_regret(deployments_1,fin_prompt,task,selected,avg_array,t,all_rewards_sum,all_rewards_diag,new_labels, dataset) #fill arguments here
            rewards += int(reward)
            rewards_list.append(rewards)
            #print(selected)
            print(reward)
            print(reg)
            print(plays_no)
            #print(all_rewards_diag)
            print("Done")
            summ+= reg
            regrets.append(summ)
            l = getattr(dyn_vars, f'l_{i}')
            #print(len(all_rewards_diag))
            new_rews = all_rewards_diag
            if t!= 0:
                new_rews = (new_rews-np.mean(all_rewards_diag))/np.std(all_rewards_diag)
            index = all_rewards_diag.index(reward)
            #print(f"all rewards diag: {all_rewards_diag}")
            #print(f"new_rews: {new_rews}")
            #print(f"reward feedback: {new_rews[index]}")
            #normalizeda.append()
            loss = l.train(context[arm], new_rews[index])
            #loss = l.train(context[arm], reward/100)

        else:
            #print('ay ay')
            #print(deployment)
            #print(cat)
            #print(task)
            #print(fin_prompt)
            reward,all_rewards_sum,all_rewards_diag = get_reward(deployment,cat,fin_prompt,task,all_rewards_sum,all_rewards_diag)
            #print('jsp')
            l = getattr(dyn_vars, f'l_{i}')
            #loss = l.train(context[arm], reward/100)
            #print(len(all_rewards_sum))
            # new_rews = all_rewards_sum
            # new_rews = (new_rews-np.mean(all_rewards_sum))/np.std(all_rewards_sum)
            # index = all_rewards_sum.index(reward)
            if t ==0:
                all_rewards_sum.append(reward+1)
                new_rews = all_rewards_sum
                #new_rews = (new_rews-np.mean(all_rewards_sum))/np.std(all_rewards_sum)
                index = all_rewards_sum.index(reward)
                #print(f"new_rews: {new_rews}")
                #print(f"reward feedback: {new_rews[index]}")
                loss = l.train(context[arm], new_rews[index]/100)
            #normalizeda.append()
            else:
                new_rews = all_rewards_sum
                new_rews = (new_rews-np.mean(all_rewards_sum))/np.std(all_rewards_sum)
                index = all_rewards_sum.index(reward)
                #print(f"new_rews: {new_rews}")
                #print(f"reward feedback: {new_rews[index]}")
                loss = l.train(context[arm], new_rews[index])
    
        
        # reg,reward,avg_array,all_rewards_sum,all_rewards_diag = get_regret(deployments_1,fin_prompt,task,selected,avg_array,t,all_rewards_sum,all_rewards_diag,new_labels,dataset)
        # #print('jsp')
        # rewards += int(reward)
        # rewards_list.append(rewards)
        # summ+= reg
        # regrets.append(summ)
        # print("reward:")
        # print(reward)
        # print("reg:")
        # print(reg)
        # print(plays_no)
        # print("Done")
        # #l = getattr(dyn_vars, f'l_{arm}')
    
        # if t ==0:
        #     if reward == 0:    
        #         all_rewards_diag.append(1)
        #     else:
        #         all_rewards_diag.append(0)
        #     new_rews = all_rewards_diag
        #     new_rews = (new_rews-np.mean(all_rewards_diag))/np.std(all_rewards_diag)
        #     index = 0
        #     loss = l.train(context[arm], new_rews[index])
        # #normalizeda.append()
        # else:
        #     new_rews = all_rewards_diag
        #     new_rews = (new_rews-np.mean(all_rewards_diag))/np.std(all_rewards_diag)
        #     index = all_rewards_diag.index(reward)
        #     loss = l.train(context[arm], new_rews[index])

    if (t+1) % 5 == 0:
        print('{}: {:.3f}, {:.3f}, {:.3f}'.format(t+1, summ, rewards,loss))

import pickle
with open("regrets_neucb_med_100_4.pkl", "wb") as file:
    pickle.dump(regrets, file)
with open("rewards_neucb_med_100_4.pkl", "wb") as file:
    pickle.dump(rewards_list, file)
with open("arm_plays_neucb_med_100_4.pkl", "wb") as file:
    pickle.dump(list(plays_no), file)
with open("avg_array_acc_llms_med_100_4.pkl", "wb") as file:
    pickle.dump(avg_array, file)