from .utils import transform_tensor_to_list


class FedAVGTrainer(object):

    def __init__(self, client_index, train_data_local_dict, train_data_local_num_dict, test_data_local_dict,
                 train_data_num, device, args, model_trainer):
        self.trainer = model_trainer

        self.client_index = client_index
        self.train_data_local_dict = train_data_local_dict
        self.train_data_local_num_dict = train_data_local_num_dict
        self.test_data_local_dict = test_data_local_dict
        self.all_train_data_num = train_data_num
        self.train_local = None
        self.local_sample_number = None
        self.test_local = None

        self.device = device
        self.args = args

    def update_model(self, weights):
        self.trainer.set_model_params(weights)

    def update_dataset(self, client_index):
        self.client_index = client_index
        self.train_local = self.train_data_local_dict[client_index]
        self.local_sample_number = self.train_data_local_num_dict[client_index]
        self.test_local = self.test_data_local_dict[client_index]

    def train(self, round_idx = None):
        self.args.round_idx = round_idx
        self.trainer.train(self.train_local, self.client_index, self.device, self.args)

        weights = self.trainer.get_model_params()
        averaged_loss = self.trainer.get_averaged_loss()

        # transform Tensor to list
        if self.args.is_mobile == 1:
            weights = transform_tensor_to_list(weights)
        return weights, self.local_sample_number, averaged_loss, self.client_index

    def test_personalized_model_on_local_data(self, round_idx):
        if self.args.per_optimizer == "FedAvg":
            return 0.0
        elif self.args.per_optimizer == "perFedAvg":
            # for MAML-based method, one more step fine-tuning is required
            self.trainer.finetune_personal_model(self.train_local, self.device)
            # Test it to get local model validation accuracy
            metrics = self.trainer.test(self.test_local, self.device, self.args)
            train_tot_correct, train_num_sample, train_loss = metrics['test_correct'], \
                                                              metrics['test_total'], metrics['test_loss']
            return train_tot_correct/train_num_sample
        elif self.args.per_optimizer == "Ditto":
            metrics = self.trainer.test(self.test_local, self.device, self.args)
            train_tot_correct, train_num_sample, train_loss = metrics['test_correct'], \
                                                              metrics['test_total'], metrics['test_loss']
            return train_tot_correct/train_num_sample

    def test(self):
        # train data
        train_metrics = self.trainer.test(self.train_local, self.device, self.args)
        train_tot_correct, train_num_sample, train_loss = train_metrics['test_correct'], \
                                                          train_metrics['test_total'], train_metrics['test_loss']

        # test data
        test_metrics = self.trainer.test(self.test_local, self.device, self.args)
        test_tot_correct, test_num_sample, test_loss = test_metrics['test_correct'], \
                                                          test_metrics['test_total'], test_metrics['test_loss']

        return train_tot_correct, train_loss, train_num_sample, test_tot_correct, test_loss, test_num_sample