import datetime
import os
import time
import matplotlib.pyplot as plt
import torch
import torch.utils.data
from torch import nn
from torch.utils.tensorboard import SummaryWriter
from torchvision import transforms
import math
from torch.cuda import amp
import model, utils
from spikingjelly.clock_driven import functional
from spikingjelly.datasets import dvs128_gesture
from timm.models import create_model
from timm.data import Mixup
from timm.optim import create_optimizer
from timm.scheduler import create_scheduler
from timm.loss import LabelSmoothingCrossEntropy, SoftTargetCrossEntropy
import autoaugment
import random
import numpy as np
import csv
import gc

from Spik4lite import EnergyPenaltyTerm, GumbelTemperatureScheduler, SpikingConv2d

_seed_ = 2021
random.seed(2021)
root_path = os.path.abspath(__file__)

torch.manual_seed(_seed_)
torch.cuda.manual_seed_all(_seed_)
torch.backends.cudnn.deterministic = True
torch.backends.cudnn.benchmark = False
np.random.seed(_seed_)
writer = SummaryWriter("./")

def parse_args():
    import argparse
    parser = argparse.ArgumentParser(description='PyTorch Classification Training')

    parser.add_argument('--model', default='SEWResNet', help='model')
    parser.add_argument('--dataset', default='DVS128Gesture', help='dataset')
    parser.add_argument('--num-classes', type=int, default=11, metavar='N',
                        help='number of label classes (default: 1000)')
    parser.add_argument('--data-path', default='DVS128Gesture/', help='dataset')
    parser.add_argument('--device', default='cuda', help='device')
    parser.add_argument('-b', '--batch-size', default=16, type=int)
    parser.add_argument('-j', '--workers', default=4, type=int, metavar='N',
                        help='number of data loading workers (default: 4)')

    parser.add_argument('--print-freq', default=256, type=int, help='print frequency')
    parser.add_argument('--output-dir', default='spikingformer/DVS128Gesture/output/train', help='path where to save')
    parser.add_argument('--resume', default='', help='resume from checkpoint')
    parser.add_argument(
        "--sync-bn",
        dest="sync_bn",
        help="Use sync batch norm",
        action="store_true",
    )
    parser.add_argument(
        "--test-only",
        dest="test_only",
        # default=True,
        help="Only test the model",
        action="store_true",
    )

    # Mixed precision training parameters
    parser.add_argument('--amp', default=True, action='store_true',
                        help='Use AMP training')


    # distributed training parameters
    parser.add_argument('--world-size', default=1, type=int,
                        help='number of distributed processes')
    parser.add_argument('--dist-url', default='env://', help='url used to set up distributed training')

    parser.add_argument('--tb', default=True,  action='store_true',
                        help='Use TensorBoard to record logs')
    parser.add_argument('--T', default=16, type=int, help='simulation steps')
    # parser.add_argument('--adam', default=True, action='store_true',
    #                     help='Use Adam')

    # Optimizer Parameters
    parser.add_argument('--opt', default='adamw', type=str, metavar="OPTIMIZER", help='Optimizer (default: "adamw")')
    parser.add_argument('--opt-eps', default=1e-8, type=float, metavar='EPSILON', help='Optimizer Epsilon (default: 1e-8)')
    parser.add_argument('--opt-betas', default=None, type=float, metavar='BETA', help='Optimizer Betas')
    parser.add_argument('--weight-decay', default=0.06, type=float, help='weight decay')
    parser.add_argument('--momentum', default=0.9, type=float, metavar='M', help='Momentum for SGD. Adam will not use momentum')

    parser.add_argument('--connect_f', default='ADD', type=str, help='element-wise connect function')
    parser.add_argument('--T_train', default=None, type=int)

    #Learning rate scheduler
    parser.add_argument('--sched', default='cosine', type=str, metavar='SCHEDULER',
                        help='LR scheduler (default: "cosine"')
    parser.add_argument('--lr', type=float, default=1e-3, metavar='LR',
                        help='learning rate (default: 5e-4)')
    parser.add_argument('--lr-noise', type=float, nargs='+', default=None, metavar='pct, pct',
                        help='learning rate noise on/off epoch percentages')
    parser.add_argument('--lr-noise-pct', type=float, default=0.67, metavar='PERCENT',
                        help='learning rate noise limit percent (default: 0.67)')
    parser.add_argument('--lr-noise-std', type=float, default=1.0, metavar='STDDEV',
                        help='learning rate noise std-dev (default: 1.0)')
    parser.add_argument('--lr-cycle-mul', type=float, default=1.0, metavar='MULT',
                        help='learning rate cycle len multiplier (default: 1.0)')
    parser.add_argument('--lr-cycle-limit', type=int, default=1, metavar='N',
                        help='learning rate cycle limit')
    parser.add_argument('--warmup-lr', type=float, default=1e-5, metavar='LR',
                        help='warmup learning rate (default: 1e-6)')
    parser.add_argument('--min-lr', type=float, default=1e-5, metavar='LR',
                        help='lower lr bound for cyclic schedulers that hit 0 (1e-5)')
    parser.add_argument('--epochs', type=int, default=192, metavar='N',
                        help='number of epochs to train (default: 2)')
    parser.add_argument('--epoch-repeats', type=float, default=0., metavar='N',
                        help='epoch repeat multiplier (number of times to repeat dataset epoch per train epoch).')
    parser.add_argument('--start-epoch', default=0, type=int, metavar='N',
                        help='manual epoch number (useful on restarts)')
    parser.add_argument('--decay-epochs', type=float, default=20, metavar='N',
                        help='epoch interval to decay LR')
    parser.add_argument('--warmup-epochs', type=int, default=10, metavar='N',
                        help='epochs to warmup LR, if scheduler supports')
    parser.add_argument('--cooldown-epochs', type=int, default=10, metavar='N',
                        help='epochs to cooldown LR at min_lr, after cyclic schedule ends')
    parser.add_argument('--patience-epochs', type=int, default=10, metavar='N',
                        help='patience epochs for Plateau LR scheduler (default: 10')
    parser.add_argument('--decay-rate', '--dr', type=float, default=0.1, metavar='RATE',
                        help='LR decay rate (default: 0.1)')

    # Augmentation & regularization parameters
    parser.add_argument('--smoothing', type=float, default=0.1,
                        help='Label smoothing (default: 0.1)')
    parser.add_argument('--mixup', type=float, default=0.5,
                        help='mixup alpha, mixup enabled if > 0. (default: 0.)')
    parser.add_argument('--cutmix', type=float, default=0.,
                        help='cutmix alpha, cutmix enabled if > 0. (default: 0.)')
    parser.add_argument('--cutmix-minmax', type=float, nargs='+', default=None,
                        help='cutmix min/max ratio, overrides alpha and enables cutmix if set (default: None)')
    parser.add_argument('--mixup-prob', type=float, default=0.5,
                        help='Probability of performing mixup or cutmix when either/both is enabled')
    parser.add_argument('--mixup-switch-prob', type=float, default=0.5,
                        help='Probability of switching to cutmix when both mixup and cutmix enabled')
    parser.add_argument('--mixup-mode', type=str, default='batch',
                        help='How to apply mixup/cutmix params. Per "batch", "pair", or "elem"')
    parser.add_argument('--mixup-off-epoch', default=0, type=int, metavar='N',
                        help='Turn off mixup after this epoch, disabled if 0 (default: 0)')

    parser.add_argument('--lambda-energy', type=float, default=0.02,
                        help='weight for physics energy loss. Higher lambda means more pruning.')
    parser.add_argument('--gumbel-temp-start', type=float, default=1.5,
                        help='initial Gumbel temperature (default: 1.0)')
    parser.add_argument('--gumbel-temp-end', type=float, default=0.1,
                        help='final Gumbel temperature (default: 0.5)')
    parser.add_argument('--pruning-interval', type=int, default=20, metavar='N',
                        help='Number of epochs to accumulate mask before pruning')
    parser.add_argument('--pruning-start-epoch', type=int, default=10, metavar='N',
                        help='Epoch to start the pruning cycle')
    parser.add_argument('--pruning-threshold', type=float, default=0.5, metavar='VAL',
                        help='Threshold for pruning channels based on accumulated mask probability')
    parser.add_argument('--pruning-end-offset', type=int, default=30, metavar='N',
                        help='Stop pruning this many epochs before the end (fine-tuning phase)')

    parser.add_argument('--global-pruning', action='store_true', default=True,
                        help='Enable Global Embedding Pruning (default: True)')
    parser.add_argument('--pe-max-prune-rate', type=float, default=0.02, metavar='VAL',
                    help='Maximum pruning rate per step for Patch Embedding layers (default: 0.4)')
    parser.add_argument('--mlp-max-prune-rate', type=float, default=0.08, metavar='VAL',
                    help='Maximum pruning rate per step for MLP layers (default: 0.5)')
    parser.add_argument('--experiment', default='', type=str, metavar='NAME',
                    help='name of train experiment, name of sub-folder for output')
    args = parser.parse_args()
    return args

def load_pruned_checkpoint(model, checkpoint_path, log_info=True):
    if log_info:
        print(f'Loading pruned model from {checkpoint_path}...')
    try:
        checkpoint = torch.load(checkpoint_path, map_location='cpu')
    except Exception as e:
        print(f"Failed to load checkpoint file {checkpoint_path}: {e}")
        return None, None, None

    if 'model' in checkpoint:
        state_dict = checkpoint['model']
    else:
        state_dict = checkpoint

    for name, module in model.named_modules():
        weight_key = f"{name}.weight"
        if weight_key not in state_dict:
            continue
        ckpt_weight = state_dict[weight_key]

        if isinstance(module, SpikingConv2d):
            ckpt_out_ch = ckpt_weight.shape[0]
            ckpt_in_ch = ckpt_weight.shape[1] * module.groups
            if ckpt_out_ch != module.out_channels or ckpt_in_ch != module.in_channels:
                if log_info:
                    print(f"  [Resize] SpikingConv {name}: ({module.in_channels}, {module.out_channels}) -> ({ckpt_in_ch}, {ckpt_out_ch})")
                module.manual_resize(in_channels=ckpt_in_ch, out_channels=ckpt_out_ch)
        
        elif isinstance(module, nn.Linear):

            ckpt_out = ckpt_weight.shape[0]
            ckpt_in = ckpt_weight.shape[1]
            if ckpt_out != module.out_features or ckpt_in != module.in_features:
                 if log_info:
                    print(f"  [Resize] Linear {name}: ({module.in_features}, {module.out_features}) -> ({ckpt_in}, {ckpt_out})")
                 module.in_features = ckpt_in
                 module.out_features = ckpt_out
                 module.weight = nn.Parameter(torch.empty(ckpt_out, ckpt_in))
                 if module.bias is not None:
                     module.bias = nn.Parameter(torch.empty(ckpt_out))

    model.load_state_dict(state_dict, strict=False)
    if log_info:
        print("Successfully loaded pruned weights into resized model.")

    return checkpoint.get('epoch', None), checkpoint.get('optimizer', None), checkpoint.get('lr_scheduler', None)

def set_model_mask_accumulation(model, enable=True):
    real_model = model.module if hasattr(model, 'module') else model
    for m in real_model.modules():
        if isinstance(m, SpikingConv2d):
            if enable:
                if not m.enable_mask_accumulation:
                    m.start_mask_accumulation()
            else:
                m.enable_mask_accumulation = False


def split_to_train_test_set(train_ratio: float, origin_dataset: torch.utils.data.Dataset, num_classes: int, random_split: bool = False):
    label_idx = []
    for i in range(num_classes):
        label_idx.append([])

    for i, item in enumerate(origin_dataset):
        y = item[1]
        if isinstance(y, np.ndarray) or isinstance(y, torch.Tensor):
            y = y.item()
        label_idx[y].append(i)
    train_idx = []
    test_idx = []
    if random_split:
        for i in range(num_classes):
            np.random.shuffle(label_idx[i])

    for i in range(num_classes):
        pos = math.ceil(label_idx[i].__len__() * train_ratio)
        train_idx.extend(label_idx[i][0: pos])
        test_idx.extend(label_idx[i][pos: label_idx[i].__len__()])

    return torch.utils.data.Subset(origin_dataset, train_idx), torch.utils.data.Subset(origin_dataset, test_idx)



def train_one_epoch(model, criterion, optimizer, data_loader, device, epoch, print_freq, 
                    scaler=None, T_train=None, aug=None, trival_aug=None, mixup_fn=None,
                    regularizer_fn=None, temp_scheduler=None):
    model.train()
    metric_logger = utils.MetricLogger(delimiter="  ")
    metric_logger.add_meter('lr', utils.SmoothedValue(window_size=1, fmt='{value}'))
    metric_logger.add_meter('img/s', utils.SmoothedValue(window_size=10, fmt='{value}'))

    metric_logger.add_meter('reg_loss', utils.SmoothedValue(window_size=10, fmt='{value:.4f}'))
    metric_logger.add_meter('temp', utils.SmoothedValue(window_size=1, fmt='{value:.4f}'))

    header = 'Epoch: [{}]'.format(epoch)

    current_temp = 0.0
    if temp_scheduler:

        current_temp = temp_scheduler.get_temp()

    for image, target in metric_logger.log_every(data_loader, print_freq, header):
        start_time = time.time()
        image, target = image.to(device), target.to(device)
        image = image.float()  # [N, T, C, H, W]
        N,T,C,H,W = image.shape

        if aug != None:
            image = torch.stack([(aug(image[i])) for i in range(N)])

        if trival_aug != None:
            image = torch.stack([(trival_aug(image[i])) for i in range(N)])

        if mixup_fn is not None:

            image, target = mixup_fn(image, target)
            target_for_compu_acc = target.argmax(dim=-1)


        if T_train:
            sec_list = np.random.choice(image.shape[1], T_train, replace=False)
            sec_list.sort()
            image = image[:, sec_list]

        optimizer.zero_grad()

        if scaler is not None:
            with amp.autocast():
                output = model(image)
                loss_cls = criterion(output, target)

                loss_reg = torch.tensor(0.0, device=device)
                if regularizer_fn is not None:
                    loss_reg = regularizer_fn()
                
                loss = loss_cls + loss_reg

            scaler.scale(loss).backward()
            scaler.step(optimizer)
            scaler.update()
        else:
            output = model(image)
            loss_cls = criterion(output, target)

            loss_reg = torch.tensor(0.0, device=device)
            if regularizer_fn is not None:
                loss_reg = regularizer_fn()
                
            loss = loss_cls + loss_reg
            
            loss.backward()
            optimizer.step()

        functional.reset_net(model)
        if mixup_fn is not None:
            acc1, acc5 = utils.accuracy(output, target_for_compu_acc, topk=(1, 5))
        else:
            acc1, acc5 = utils.accuracy(output, target, topk=(1, 5))
        batch_size = image.shape[0]
        loss_s = loss.item()
        if math.isnan(loss_s):
            raise ValueError('loss is Nan')
        acc1_s = acc1.item()
        acc5_s = acc5.item()

        metric_logger.update(loss=loss_s, lr=optimizer.param_groups[0]["lr"])

        metric_logger.meters['reg_loss'].update(loss_reg.item(), n=batch_size)
        metric_logger.meters['temp'].update(current_temp, n=1)

        metric_logger.meters['acc1'].update(acc1_s, n=batch_size)
        metric_logger.meters['acc5'].update(acc5_s, n=batch_size)
        metric_logger.meters['img/s'].update(batch_size / (time.time() - start_time))

    if temp_scheduler is not None:
        temp_scheduler.step()

    # gather the stats from all processes
    metric_logger.synchronize_between_processes()
    return metric_logger.loss.global_avg, metric_logger.acc1.global_avg, metric_logger.acc5.global_avg



def evaluate(model, criterion, data_loader, device, print_freq=100, header='Test:'):
    model.eval()
    metric_logger = utils.MetricLogger(delimiter="  ")
    with torch.no_grad():
        for image, target in metric_logger.log_every(data_loader, print_freq, header):
            image = image.to(device, non_blocking=True)
            target = target.to(device, non_blocking=True)
            image = image.float()
            output = model(image)
            loss = criterion(output, target)
            functional.reset_net(model)

            acc1, acc5 = utils.accuracy(output, target, topk=(1, 5))
            batch_size = image.shape[0]
            metric_logger.update(loss=loss.item())
            metric_logger.meters['acc1'].update(acc1.item(), n=batch_size)
            metric_logger.meters['acc5'].update(acc5.item(), n=batch_size)
    # gather the stats from all processes
    metric_logger.synchronize_between_processes()

    loss, acc1, acc5 = metric_logger.loss.global_avg, metric_logger.acc1.global_avg, metric_logger.acc5.global_avg
    print(f' * Acc@1 = {acc1}, Acc@5 = {acc5}, loss = {loss}')
    return loss, acc1, acc5

def load_data(dataset_dir, distributed, T):
    # Data loading code
    print("Loading data")

    st = time.time()

    dataset_train = dvs128_gesture.DVS128Gesture(root=dataset_dir, train=True, data_type='frame', frames_number=T,
                                                 split_by='number')
    dataset_test = dvs128_gesture.DVS128Gesture(root=dataset_dir, train=False, data_type='frame', frames_number=T,
                                                split_by='number')
    print("Took", time.time() - st)

    print("Creating data loaders")
    if distributed:
        train_sampler = torch.utils.data.distributed.DistributedSampler(dataset_train)
        test_sampler = torch.utils.data.distributed.DistributedSampler(dataset_test)
    else:
        train_sampler = torch.utils.data.RandomSampler(dataset_train)
        test_sampler = torch.utils.data.SequentialSampler(dataset_test)

    return dataset_train, dataset_test, train_sampler, test_sampler

def main(args):

    max_test_acc1 = 0.
    test_acc5_at_max_test_acc1 = 0.


    train_tb_writer = None
    te_tb_writer = None

    utils.init_distributed_mode(args)
    print(args)

    output_dir = os.path.join(args.output_dir, f'{args.model}_b{args.batch_size}_T{args.T}')

    if args.T_train:
        output_dir += f'_Ttrain{args.T_train}'

    if args.weight_decay:
        output_dir += f'_wd{args.weight_decay}'


    if args.opt == 'adamw':
        output_dir += '_adamw'
    else:
        output_dir += '_sgd'

    if args.connect_f:
        output_dir += f'_cnf_{args.connect_f}'

    if not os.path.exists(output_dir):
        utils.mkdir(output_dir)

    output_dir = os.path.join(output_dir, f'lr{args.lr}')
    if not os.path.exists(output_dir):
        utils.mkdir(output_dir)

    device = torch.device(args.device)

    data_path = args.data_path

    dataset_train, dataset_test, train_sampler, test_sampler = load_data(data_path, args.distributed, args.T)

    data_loader = torch.utils.data.DataLoader(
        dataset=dataset_train,
        batch_size=args.batch_size,
        shuffle=True,
        num_workers=args.workers,
        drop_last=True,
        # sampler=train_sampler,
        pin_memory=True)

    data_loader_test = torch.utils.data.DataLoader(
        dataset=dataset_test,
        batch_size=args.batch_size,
        shuffle=False,
        num_workers=args.workers,
        # sampler=test_sampler,
        drop_last=False,
        pin_memory=True)

    pruned_cfg = None
    if args.resume and os.path.exists(args.resume):
        try:
            temp_ckpt = torch.load(args.resume, map_location='cpu')
            if 'structure_config' in temp_ckpt:
                pruned_cfg = temp_ckpt['structure_config']
                print("Found 'structure_config' in checkpoint! Initializing model with pruned structure.")
            del temp_ckpt
        except Exception as e:
            print(f"Error checking checkpoint structure: {e}")

    model = create_model(
        'Spikingformer',
        pretrained=False,
        drop_rate=0.,
        drop_path_rate=0.1,
        drop_block_rate=None,
        pruned_structure_cfg=pruned_cfg
    )

    print("Creating model")
    n_parameters = sum(p.numel() for p in model.parameters() if p.requires_grad)
    print(f"number of params: {n_parameters}")
    model.to(device)

    if args.distributed and args.sync_bn:
        model = torch.nn.SyncBatchNorm.convert_sync_batchnorm(model)
    # criterion_train = LabelSmoothingCrossEntropy()
    criterion_train = SoftTargetCrossEntropy().cuda()
    criterion = nn.CrossEntropyLoss()

    optimizer = create_optimizer(args, model)
    if args.amp:
        scaler = amp.GradScaler()
    else:
        scaler = None

    lr_scheduler, num_epochs = create_scheduler(args, optimizer)
    
    start_epoch = 0
    model_without_ddp = model
    if args.distributed:
        model = torch.nn.parallel.DistributedDataParallel(model, device_ids=[args.gpu])
        model_without_ddp = model.module

    if args.resume:
        resume_epoch, loaded_optim_state, loaded_sched_state = load_pruned_checkpoint(
            model_without_ddp, args.resume
        )
        if loaded_optim_state:
            try:
                optimizer.load_state_dict(loaded_optim_state)
            except Exception as e:
                print(f"Optimizer load failed (expected if pruned): {e}")
        if loaded_sched_state:
            lr_scheduler.load_state_dict(loaded_sched_state)
        
        if resume_epoch is not None:
             args.start_epoch = resume_epoch + 1

    regularizer_criterion = None
    
    print(f"Using Physics-Aware Mode: Minimizing Energy (SOPs). Lambda={args.lambda_energy}")
    regularizer_criterion = EnergyPenaltyTerm(
        model,
        lambda_energy=args.lambda_energy
    )

    temp_scheduler = GumbelTemperatureScheduler(
        model,
        init_temp=args.gumbel_temp_start,
        final_temp=args.gumbel_temp_end,
        total_epochs=args.epochs
    )

    fine_tuning_start_epoch = args.epochs - args.pruning_end_offset
    print(f"Fine-tuning will start at epoch {fine_tuning_start_epoch}.")
    
    if args.test_only:

        evaluate(model, criterion, data_loader_test, device=device, header='Test:')

        return

    if args.tb and utils.is_main_process():
        purge_step_train = args.start_epoch
        purge_step_te = args.start_epoch
        train_tb_writer = SummaryWriter(output_dir + '_logs/train', purge_step=purge_step_train)
        te_tb_writer = SummaryWriter(output_dir + '_logs/te', purge_step=purge_step_te)
        with open(output_dir + '_logs/args.txt', 'w', encoding='utf-8') as args_txt:
            args_txt.write(str(args))

        print(f'purge_step_train={purge_step_train}, purge_step_te={purge_step_te}')


    train_snn_aug = transforms.Compose([
                    transforms.RandomHorizontalFlip(p=0.5)
                    ])
    train_trivalaug = autoaugment.SNNAugmentWide()
    mixup_fn = None
    mixup_active = args.mixup > 0 or args.cutmix > 0. or args.cutmix_minmax is not None
    if mixup_active:
        mixup_args = dict(
            mixup_alpha=args.mixup, cutmix_alpha=args.cutmix, cutmix_minmax=args.cutmix_minmax,
            prob=args.mixup_prob, switch_prob=args.mixup_switch_prob, mode=args.mixup_mode,
            label_smoothing=args.smoothing, num_classes=args.num_classes)
        mixup_fn = Mixup(**mixup_args)

    print("Start training")
    start_time = time.time()

    for epoch in range(args.start_epoch, num_epochs):
        save_max = False
        if args.distributed:
            train_sampler.set_epoch(epoch)
        if epoch >= 75:
            mixup_fn.mixup_enabled = False

        is_accumulating_phase = False
        is_pruning_epoch = False

        if args.pruning_start_epoch <= epoch < (args.epochs - args.pruning_end_offset):
            cycle_index = (epoch - args.pruning_start_epoch) % args.pruning_interval
            is_accumulating_phase = True
            if cycle_index == (args.pruning_interval - 1):
                is_pruning_epoch = True

        set_model_mask_accumulation(model, is_accumulating_phase)

        if epoch == fine_tuning_start_epoch:
            print("---------------------------------------------------------")
            print(f"Epoch {epoch}: Transitioning to FINE-TUNING Phase")
            print("Permanently removing Gating Layers & Cleaning Optimizer...")
            print("---------------------------------------------------------")

            removed_count = 0
            for m in model_without_ddp.modules():
                if isinstance(m, SpikingConv2d):
                    if hasattr(m, 'gating_layer'):
                        del m.gating_layer
                        m.gating_layer = None
                    if hasattr(m, 'running_fr'): del m.running_fr
                    if hasattr(m, 'current_probs'): m.current_probs = None
                    m.static_mode = True
                    removed_count += 1
            print(f"Removed gating from {removed_count} layers.")

            gc.collect()
            torch.cuda.empty_cache()
            
            valid_params = set(model_without_ddp.parameters())

            for group in optimizer.param_groups:
                new_params = [p for p in group['params'] if p in valid_params]
                group['params'] = new_params

            keys_to_remove = [p for p in optimizer.state.keys() if p not in valid_params]
            for p in keys_to_remove:
                del optimizer.state[p]
            print(f"Optimizer cleaned. Removed {len(keys_to_remove)} expired params.")

            if args.distributed:
                 model = torch.nn.parallel.DistributedDataParallel(model_without_ddp, device_ids=[args.gpu])

        current_regularizer = regularizer_criterion
        if epoch >= fine_tuning_start_epoch:
            current_regularizer = None

        train_loss, train_acc1, train_acc5 = train_one_epoch(
            model, criterion_train, optimizer, data_loader, device, epoch,
            args.print_freq, scaler, args.T_train,
            train_snn_aug, train_trivalaug, mixup_fn,
            regularizer_fn=current_regularizer,
            temp_scheduler=temp_scheduler
        )

        if is_pruning_epoch:
            print(f"*** Epoch {epoch}: Executing Physical Pruning ***")
            
            torch.cuda.empty_cache()
            
            if hasattr(model_without_ddp, 'prune_model'):
                stats = model_without_ddp.prune_model(
                    threshold=args.pruning_threshold,
                    optimizer=optimizer,
                    global_pruning=args.global_pruning,
                    pe_max_prune_rate=args.pe_max_prune_rate,
                    mlp_max_prune_rate=args.mlp_max_prune_rate
                )

            gc.collect()
            torch.cuda.empty_cache()

            if args.distributed:
                model = torch.nn.parallel.DistributedDataParallel(model_without_ddp, device_ids=[args.gpu])


        if utils.is_main_process():
            train_tb_writer.add_scalar('train_loss', train_loss, epoch)
            train_tb_writer.add_scalar('train_acc1', train_acc1, epoch)
            train_tb_writer.add_scalar('train_acc5', train_acc5, epoch)
        lr_scheduler.step(epoch + 1)

        test_loss, test_acc1, test_acc5 = evaluate(model, criterion, data_loader_test, device=device, header='Test:')
        if te_tb_writer is not None:
            if utils.is_main_process():

                te_tb_writer.add_scalar('test_loss', test_loss, epoch)
                te_tb_writer.add_scalar('test_acc1', test_acc1, epoch)
                te_tb_writer.add_scalar('test_acc5', test_acc5, epoch)


        if max_test_acc1 < test_acc1:
            max_test_acc1 = test_acc1
            test_acc5_at_max_test_acc1 = test_acc5
            save_max = True


        if output_dir:
            current_structure_cfg = None
            if hasattr(model_without_ddp, 'export_structure_config'):
                current_structure_cfg = model_without_ddp.export_structure_config()

            checkpoint = {
                'model': model_without_ddp.state_dict(),
                'optimizer': optimizer.state_dict(),
                'lr_scheduler': lr_scheduler.state_dict(),
                'epoch': epoch,
                'args': args,
                'max_test_acc1': max_test_acc1,
                'test_acc5_at_max_test_acc1': test_acc5_at_max_test_acc1,
                'structure_config': current_structure_cfg,
            }

            if save_max:
                utils.save_on_master(
                    checkpoint,
                    os.path.join(output_dir, 'checkpoint_max_test_acc1.pth'))
        total_time = time.time() - start_time
        total_time_str = str(datetime.timedelta(seconds=int(total_time)))

        print('Training time {}'.format(total_time_str), 'max_test_acc1', max_test_acc1, 'test_acc5_at_max_test_acc1', test_acc5_at_max_test_acc1)
        print(output_dir)
    if output_dir:
        utils.save_on_master(
            checkpoint,
            os.path.join(output_dir, f'checkpoint_{epoch}.pth'))

    return max_test_acc1




if __name__ == "__main__":
    args = parse_args()
    main(args)
