#!/bin/bash

# Define the GPUs to be used
GPUS=(1 2 3 4 5 6 7)

# Function to execute a training task on a specific GPU
run_training() {
    local gpu=$1
    local save_dir=$2
    local unlearn_method=$3
    local class_to_replace=$4
    local num_indexes_to_replace=$5
    local unlearn_epochs=$6
    local unlearn_lr=$7
    local arch=$8
    local additional_params=${9}
    local log_file=${10}

    CUDA_VISIBLE_DEVICES=$gpu nohup python -u main_forget.py \
        --data $DATASETS_BASE_DIR \
        --save_dir $save_dir \
        --unlearn $unlearn_method \
        --class_to_replace $class_to_replace \
        --num_indexes_to_replace $num_indexes_to_replace \
        --unlearn_epochs $unlearn_epochs \
        --unlearn_lr $unlearn_lr \
        --arch $arch \
        $additional_params \
        > $log_file 2>&1 &
}

# Training Dataset directories
DATASETS_BASE_DIR="/nvme/data/3ai/lips/datasets"
TINYIMAGENET_DIR="$DATASETS_BASE_DIR/tiny-imagenet-200"

# Directory for saving the training outputs like Models and Logs
OUTPUT_BASE_DIR="/nvme/data/3ai/lips/outputs"

OUTPUT_LOG_DIR_TRAIN_MU="/nvme/data/3ai/lips/logs/step-01-machine-unlearning"

# Check if the TinyImagenet data directory exists
if [ ! -d "$TINYIMAGENET_DIR" ]; then
    echo "TinyImagenet data directory not found: $TINYIMAGENET_DIR"
    exit 1
fi

if [ ! -d "$OUTPUT_BASE_DIR" ]; then
    mkdir -p $OUTPUT_BASE_DIR
fi

if [ ! -d "$OUTPUT_LOG_DIR_TRAIN_MU" ]; then
    mkdir -p $OUTPUT_LOG_DIR_TRAIN_MU
fi

# Index to keep track of GPU allocation
GPU_INDEX=0
NUM_GPUS=${#GPUS[@]}

# 1.1.1 Resnet18 Cifar10
run_training ${GPUS[GPU_INDEX % NUM_GPUS]} "$OUTPUT_BASE_DIR/resnet18_cifar10/retrain" retrain 0,1,3,5,6 2250 100 0.1 resnet18 "" "$OUTPUT_LOG_DIR_TRAIN_MU/Resnet18-Cifar10-retrain.log"
((GPU_INDEX++))
run_training ${GPUS[GPU_INDEX % NUM_GPUS]} "$OUTPUT_BASE_DIR/resnet18_cifar10/FT" FT 0,1,3,5,6 2250 100 0.1 resnet18 "--load_ff --resume" "$OUTPUT_LOG_DIR_TRAIN_MU/Resnet18-Cifar10-ft.log"
((GPU_INDEX++))
run_training ${GPUS[GPU_INDEX % NUM_GPUS]} "$OUTPUT_BASE_DIR/resnet18_cifar10/GA" GA 0,1,3,5,6 2250 4 0.0001 resnet18 "--load_ff --resume" "$OUTPUT_LOG_DIR_TRAIN_MU/Resnet18-Cifar10-ga.log"
((GPU_INDEX++))
run_training ${GPUS[GPU_INDEX % NUM_GPUS]} "$OUTPUT_BASE_DIR/resnet18_cifar10/FF" fisher 0,1,3,5,6 2250 100 0.1 resnet18 "--load_ff --resume --alpha 16.5" "$OUTPUT_LOG_DIR_TRAIN_MU/Resnet18-Cifar10-ff.log"
((GPU_INDEX++))
run_training ${GPUS[GPU_INDEX % NUM_GPUS]} "$OUTPUT_BASE_DIR/resnet18_cifar10/IU" wfisher 0,1,3,5,6 2250 100 0.1 resnet18 "--load_ff --resume --alpha 16" "$OUTPUT_LOG_DIR_TRAIN_MU/Resnet18-Cifar10-iu.log"
((GPU_INDEX++))
run_training ${GPUS[GPU_INDEX % NUM_GPUS]} "$OUTPUT_BASE_DIR/resnet18_cifar10/FT_prune" FT_prune 0,1,3,5,6 2250 50 0.01 resnet18 "--load_ff --resume --alpha 0.005" "$OUTPUT_LOG_DIR_TRAIN_MU/Resnet18-Cifar10-ft-prune.log"
((GPU_INDEX++))

# 1.1.2 Resnet18 Cifar100
run_training ${GPUS[GPU_INDEX % NUM_GPUS]} "$OUTPUT_BASE_DIR/resnet18_cifar100/retrain" retrain 11,22,33,44,55 225 100 0.1 resnet18 "--dataset cifar100" "$OUTPUT_LOG_DIR_TRAIN_MU/Resnet18-Cifar100-retrain.log"
((GPU_INDEX++))
run_training ${GPUS[GPU_INDEX % NUM_GPUS]} "$OUTPUT_BASE_DIR/resnet18_cifar100/FT" FT 11,22,33,44,55 225 100 0.1 resnet18 "--dataset cifar100 --load_ff --resume" "$OUTPUT_LOG_DIR_TRAIN_MU/Resnet18-Cifar100-ft.log"
((GPU_INDEX++))
run_training ${GPUS[GPU_INDEX % NUM_GPUS]} "$OUTPUT_BASE_DIR/resnet18_cifar100/GA" GA 11,22,33,44,55 225 4 0.001 resnet18 "--dataset cifar100 --load_ff --resume" "$OUTPUT_LOG_DIR_TRAIN_MU/Resnet18-Cifar100-ga.log"
((GPU_INDEX++))
run_training ${GPUS[GPU_INDEX % NUM_GPUS]} "$OUTPUT_BASE_DIR/resnet18_cifar100/FF" fisher 11,22,33,44,55 225 100 0.1 resnet18 "--dataset cifar100 --load_ff --resume --alpha 20" "$OUTPUT_LOG_DIR_TRAIN_MU/Resnet18-Cifar100-ff.log"
((GPU_INDEX++))
run_training ${GPUS[GPU_INDEX % NUM_GPUS]} "$OUTPUT_BASE_DIR/resnet18_cifar100/IU" wfisher 11,22,33,44,55 225 100 0.1 resnet18 "--dataset cifar100 --load_ff --resume --alpha 160" "$OUTPUT_LOG_DIR_TRAIN_MU/Resnet18-Cifar100-iu.log"
((GPU_INDEX++))
run_training ${GPUS[GPU_INDEX % NUM_GPUS]} "$OUTPUT_BASE_DIR/resnet18_cifar100/FT_prune" FT_prune 11,22,33,44,55 225 50 0.001 resnet18 "--dataset cifar100 --load_ff --resume --alpha 0.005" "$OUTPUT_LOG_DIR_TRAIN_MU/Resnet18-Cifar100-ft-prune.log"
((GPU_INDEX++))

# 1.1.3 Resnet18 TinyImagenet
run_training ${GPUS[GPU_INDEX % NUM_GPUS]} "$OUTPUT_BASE_DIR/resnet18_tinyimg/retrain" retrain 1,51,101,151,198 250 100 0.1 resnet18 "--dataset TinyImagenet --data_dir $TINYIMAGENET_DIR" "$OUTPUT_LOG_DIR_TRAIN_MU/Resnet18-tinyim-retrain.log"
((GPU_INDEX++))
run_training ${GPUS[GPU_INDEX % NUM_GPUS]} "$OUTPUT_BASE_DIR/resnet18_tinyimg/FT" FT 1,51,101,151,198 250 100 0.1 resnet18 "--dataset TinyImagenet --data_dir $TINYIMAGENET_DIR --load_ff --resume" "$OUTPUT_LOG_DIR_TRAIN_MU/Resnet18-tinyim-ft.log"
((GPU_INDEX++))
run_training ${GPUS[GPU_INDEX % NUM_GPUS]} "$OUTPUT_BASE_DIR/resnet18_tinyimg/GA" GA 1,51,101,151,198 250 5 0.00001 resnet18 "--dataset TinyImagenet --data_dir $TINYIMAGENET_DIR --load_ff --resume" "$OUTPUT_LOG_DIR_TRAIN_MU/Resnet18-tinyim-ga.log"
((GPU_INDEX++))
run_training ${GPUS[GPU_INDEX % NUM_GPUS]} "$OUTPUT_BASE_DIR/resnet18_tinyimg/FF" fisher 1,51,101,151,198 250 100 0.1 resnet18 "--dataset TinyImagenet --data_dir $TINYIMAGENET_DIR --load_ff --resume --alpha 20" "$OUTPUT_LOG_DIR_TRAIN_MU/Resnet18-tinyim-ff.log"
((GPU_INDEX++))
run_training ${GPUS[GPU_INDEX % NUM_GPUS]} "$OUTPUT_BASE_DIR/resnet18_tinyimg/IU" wfisher 1,51,101,151,198 250 100 0.1 resnet18 "--dataset TinyImagenet --data_dir $TINYIMAGENET_DIR --load_ff --resume --alpha 160" "$OUTPUT_LOG_DIR_TRAIN_MU/Resnet18-tinyim-iu.log"
((GPU_INDEX++))
run_training ${GPUS[GPU_INDEX % NUM_GPUS]} "$OUTPUT_BASE_DIR/resnet18_tinyimg/FT_prune" FT_prune 1,51,101,151,198 250 50 0.001 resnet18 "--dataset TinyImagenet --data_dir $TINYIMAGENET_DIR --load_ff --resume --alpha 0.001" "$OUTPUT_LOG_DIR_TRAIN_MU/Resnet18-tinyim-ft-prune.log"
((GPU_INDEX++))

# 1.1.4 Resnet18 fashionMNIST
run_training ${GPUS[GPU_INDEX % NUM_GPUS]} "$OUTPUT_BASE_DIR/resnet18_fmnist/retrain" retrain 1,3,5,7,9 2700 100 0.1 resnet18 "--dataset fashionMNIST" "$OUTPUT_LOG_DIR_TRAIN_MU/Resnet18-fsmnist-retrain.log"
((GPU_INDEX++))
run_training ${GPUS[GPU_INDEX % NUM_GPUS]} "$OUTPUT_BASE_DIR/resnet18_fmnist/FT" FT 1,3,5,7,9 2700 100 0.1 resnet18 "--dataset fashionMNIST --load_ff --resume" "$OUTPUT_LOG_DIR_TRAIN_MU/Resnet18-fsmnist-ft.log"
((GPU_INDEX++))
run_training ${GPUS[GPU_INDEX % NUM_GPUS]} "$OUTPUT_BASE_DIR/resnet18_fmnist/GA" GA 1,3,5,7,9 2700 5 0.00001 resnet18 "--dataset fashionMNIST --load_ff --resume" "$OUTPUT_LOG_DIR_TRAIN_MU/Resnet18-fsmnist-ga.log"
((GPU_INDEX++))
run_training ${GPUS[GPU_INDEX % NUM_GPUS]} "$OUTPUT_BASE_DIR/resnet18_fmnist/FF" fisher 1,3,5,7,9 2700 100 0.1 resnet18 "--dataset fashionMNIST --load_ff --resume --alpha 16.5" "$OUTPUT_LOG_DIR_TRAIN_MU/Resnet18-fsmnist-ff.log"
((GPU_INDEX++))
run_training ${GPUS[GPU_INDEX % NUM_GPUS]} "$OUTPUT_BASE_DIR/resnet18_fmnist/IU" wfisher 1,3,5,7,9 2700 100 0.1 resnet18 "--dataset fashionMNIST --load_ff --resume --alpha 100" "$OUTPUT_LOG_DIR_TRAIN_MU/Resnet18-fsmnist-iu.log"
((GPU_INDEX++))
run_training ${GPUS[GPU_INDEX % NUM_GPUS]} "$OUTPUT_BASE_DIR/resnet18_fmnist/FT_prune" FT_prune 1,3,5,7,9 2700 50 0.02 resnet18 "--dataset fashionMNIST --load_ff --resume --alpha 0.03" "$OUTPUT_LOG_DIR_TRAIN_MU/Resnet18-fsmnist-ft-prune.log"
((GPU_INDEX++))

# 1.2.1 vgg16_bn_lth Cifar10
run_training ${GPUS[GPU_INDEX % NUM_GPUS]} "$OUTPUT_BASE_DIR/vgg16_cifar10/retrain" retrain 0,1,3,5,6 2250 100 0.1 vgg16_bn_lth "" "$OUTPUT_LOG_DIR_TRAIN_MU/VGG16-Cifar10-retrain.log"
((GPU_INDEX++))
run_training ${GPUS[GPU_INDEX % NUM_GPUS]} "$OUTPUT_BASE_DIR/vgg16_cifar10/FT" FT 0,1,3,5,6 2250 100 0.1 vgg16_bn_lth "--load_ff --resume" "$OUTPUT_LOG_DIR_TRAIN_MU/VGG16-Cifar10-ft.log"
((GPU_INDEX++))
run_training ${GPUS[GPU_INDEX % NUM_GPUS]} "$OUTPUT_BASE_DIR/vgg16_cifar10/GA" GA 0,1,3,5,6 2250 4 0.0001 vgg16_bn_lth "--load_ff --resume" "$OUTPUT_LOG_DIR_TRAIN_MU/VGG16-Cifar10-ga.log"
((GPU_INDEX++))
run_training ${GPUS[GPU_INDEX % NUM_GPUS]} "$OUTPUT_BASE_DIR/vgg16_cifar10/FF" fisher 0,1,3,5,6 2250 100 0.1 vgg16_bn_lth "--load_ff --resume --alpha 16.5" "$OUTPUT_LOG_DIR_TRAIN_MU/VGG16-Cifar10-ff.log"
((GPU_INDEX++))
run_training ${GPUS[GPU_INDEX % NUM_GPUS]} "$OUTPUT_BASE_DIR/vgg16_cifar10/IU" wfisher 0,1,3,5,6 2250 100 0.1 vgg16_bn_lth "--load_ff --resume --alpha 40" "$OUTPUT_LOG_DIR_TRAIN_MU/VGG16-Cifar10-iu.log"
((GPU_INDEX++))
run_training ${GPUS[GPU_INDEX % NUM_GPUS]} "$OUTPUT_BASE_DIR/vgg16_cifar10/FT_prune" FT_prune 0,1,3,5,6 2250 50 0.01 vgg16_bn_lth "--load_ff --resume --alpha 0.005" "$OUTPUT_LOG_DIR_TRAIN_MU/VGG16-Cifar10-ft-prune.log"
((GPU_INDEX++))

# 1.2.2 vgg16_bn_lth Cifar100
run_training ${GPUS[GPU_INDEX % NUM_GPUS]} "$OUTPUT_BASE_DIR/vgg16_cifar100/retrain" retrain 11,22,33,44,55 225 100 0.1 vgg16_bn_lth "--dataset cifar100" "$OUTPUT_LOG_DIR_TRAIN_MU/VGG16-Cifar100-retrain.log"
((GPU_INDEX++))
run_training ${GPUS[GPU_INDEX % NUM_GPUS]} "$OUTPUT_BASE_DIR/vgg16_cifar100/FT" FT 11,22,33,44,55 225 100 0.1 vgg16_bn_lth "--dataset cifar100 --load_ff --resume" "$OUTPUT_LOG_DIR_TRAIN_MU/VGG16-Cifar100-ft.log"
((GPU_INDEX++))
run_training ${GPUS[GPU_INDEX % NUM_GPUS]} "$OUTPUT_BASE_DIR/vgg16_cifar100/GA" GA 11,22,33,44,55 225 4 0.001 vgg16_bn_lth "--dataset cifar100 --load_ff --resume" "$OUTPUT_LOG_DIR_TRAIN_MU/VGG16-Cifar100-ga.log"
((GPU_INDEX++))
run_training ${GPUS[GPU_INDEX % NUM_GPUS]} "$OUTPUT_BASE_DIR/vgg16_cifar100/FF" fisher 11,22,33,44,55 225 100 0.1 vgg16_bn_lth "--dataset cifar100 --load_ff --resume --alpha 16.5" "$OUTPUT_LOG_DIR_TRAIN_MU/VGG16-Cifar100-ff.log"
((GPU_INDEX++))
run_training ${GPUS[GPU_INDEX % NUM_GPUS]} "$OUTPUT_BASE_DIR/vgg16_cifar100/IU" wfisher 11,22,33,44,55 225 100 0.1 vgg16_bn_lth "--dataset cifar100 --load_ff --resume --alpha 200" "$OUTPUT_LOG_DIR_TRAIN_MU/VGG16-Cifar100-iu.log"
((GPU_INDEX++))
run_training ${GPUS[GPU_INDEX % NUM_GPUS]} "$OUTPUT_BASE_DIR/vgg16_cifar100/FT_prune" FT_prune 11,22,33,44,55 225 50 0.001 vgg16_bn_lth "--dataset cifar100 --load_ff --resume --alpha 0.005" "$OUTPUT_LOG_DIR_TRAIN_MU/VGG16-Cifar100-ft-prune.log"
((GPU_INDEX++))

# 1.2.3 vgg16_bn_lth TinyImagenet
run_training ${GPUS[GPU_INDEX % NUM_GPUS]} "$OUTPUT_BASE_DIR/vgg16_tinyimg/retrain" retrain 1,51,101,151,198 250 100 0.1 vgg16_bn_lth "--dataset TinyImagenet --data_dir $TINYIMAGENET_DIR" "$OUTPUT_LOG_DIR_TRAIN_MU/VGG16-tinyim-retrain.log"
((GPU_INDEX++))
run_training ${GPUS[GPU_INDEX % NUM_GPUS]} "$OUTPUT_BASE_DIR/vgg16_tinyimg/FT" FT 1,51,101,151,198 250 100 0.1 vgg16_bn_lth "--dataset TinyImagenet --data_dir $TINYIMAGENET_DIR --load_ff --resume" "$OUTPUT_LOG_DIR_TRAIN_MU/VGG16-tinyim-ft.log"
((GPU_INDEX++))
run_training ${GPUS[GPU_INDEX % NUM_GPUS]} "$OUTPUT_BASE_DIR/vgg16_tinyimg/GA" GA 1,51,101,151,198 250 4 0.0001 vgg16_bn_lth "--dataset TinyImagenet --data_dir $TINYIMAGENET_DIR --load_ff --resume" "$OUTPUT_LOG_DIR_TRAIN_MU/VGG16-tinyim-ga.log"
((GPU_INDEX++))
run_training ${GPUS[GPU_INDEX % NUM_GPUS]} "$OUTPUT_BASE_DIR/vgg16_tinyimg/FF" fisher 1,51,101,151,198 250 100 0.1 vgg16_bn_lth "--dataset TinyImagenet --data_dir $TINYIMAGENET_DIR --load_ff --resume --alpha 16.5" "$OUTPUT_LOG_DIR_TRAIN_MU/VGG16-tinyim-ff.log"
((GPU_INDEX++))
run_training ${GPUS[GPU_INDEX % NUM_GPUS]} "$OUTPUT_BASE_DIR/vgg16_tinyimg/IU" wfisher 1,51,101,151,198 250 100 0.1 vgg16_bn_lth "--dataset TinyImagenet --data_dir $TINYIMAGENET_DIR --load_ff --resume --alpha 100" "$OUTPUT_LOG_DIR_TRAIN_MU/VGG16-tinyim-iu.log"
((GPU_INDEX++))
run_training ${GPUS[GPU_INDEX % NUM_GPUS]} "$OUTPUT_BASE_DIR/vgg16_tinyimg/FT_prune" FT_prune 1,51,101,151,198 250 30 0.002 vgg16_bn_lth "--dataset TinyImagenet --data_dir $TINYIMAGENET_DIR --load_ff --resume --alpha 0.001" "$OUTPUT_LOG_DIR_TRAIN_MU/VGG16-tinyim-ft-prune.log"
((GPU_INDEX++))

# 1.2.4 vgg16_bn_lth fashionMNIST
run_training ${GPUS[GPU_INDEX % NUM_GPUS]} "$OUTPUT_BASE_DIR/vgg16_fmnist/retrain" retrain 1,3,5,7,9 2700 100 0.1 vgg16_bn_lth "--dataset fashionMNIST" "$OUTPUT_LOG_DIR_TRAIN_MU/VGG16-fashionmn-retrain.log"
((GPU_INDEX++))
run_training ${GPUS[GPU_INDEX % NUM_GPUS]} "$OUTPUT_BASE_DIR/vgg16_fmnist/FT" FT 1,3,5,7,9 2700 100 0.1 vgg16_bn_lth "--dataset fashionMNIST --load_ff --resume" "$OUTPUT_LOG_DIR_TRAIN_MU/VGG16-fashionmn-ft.log"
((GPU_INDEX++))
run_training ${GPUS[GPU_INDEX % NUM_GPUS]} "$OUTPUT_BASE_DIR/vgg16_fmnist/GA" GA 1,3,5,7,9 2700 4 0.0001 vgg16_bn_lth "--dataset fashionMNIST --load_ff --resume" "$OUTPUT_LOG_DIR_TRAIN_MU/VGG16-fashionmn-ga.log"
((GPU_INDEX++))
run_training ${GPUS[GPU_INDEX % NUM_GPUS]} "$OUTPUT_BASE_DIR/vgg16_fmnist/FF" fisher 1,3,5,7,9 2700 100 0.1 vgg16_bn_lth "--dataset fashionMNIST --load_ff --resume --alpha 16.5" "$OUTPUT_LOG_DIR_TRAIN_MU/VGG16-fashionmn-ff.log"
((GPU_INDEX++))
run_training ${GPUS[GPU_INDEX % NUM_GPUS]} "$OUTPUT_BASE_DIR/vgg16_fmnist/IU" wfisher 1,3,5,7,9 2700 100 0.1 vgg16_bn_lth "--dataset fashionMNIST --load_ff --resume --alpha 40" "$OUTPUT_LOG_DIR_TRAIN_MU/VGG16-fashionmn-iu.log"
((GPU_INDEX++))
run_training ${GPUS[GPU_INDEX % NUM_GPUS]} "$OUTPUT_BASE_DIR/vgg16_fmnist/FT_prune" FT_prune 1,3,5,7,9 2700 50 0.005 vgg16_bn_lth "--dataset fashionMNIST --load_ff --resume --alpha 0.02" "$OUTPUT_LOG_DIR_TRAIN_MU/VGG16-fashionmn-ft-prune.log"
((GPU_INDEX++))

echo "All tasks have been started."
