import os
from random import shuffle
import subprocess

import argparse

from config_generator import generate_configurations, generate_string_from_params, generate_name_from_params

def update_script(save_folder):
    """
    Generates a SLURM batch script for job submission based on the provided folder.

    Parameters:
    - save_folder (str): The path to the folder where the script will be saved.

    Returns:
    - str: The file name of the script.
    """
    # Extract the last directory name from the save_folder path to use as the job name
    # Define the full path to the script file
    file_name = save_folder + '/cluster_submit.sh'
    print('file name', file_name)
    
    script_string = """#!/bin/bash\n#SBATCH --job-name=test_timeseries-""" + save_folder.split('/')[-1] + """\n#SBATCH --mail-user=...\n#SBATCH --mail-type=begin,end,fail\n#SBATCH --partition=compute\n#SBATCH --account=...\n#SBATCH --time=8:00:00\n#SBATCH --ntasks=10\n#SBATCH --output=./runs/slurm-%j.out\nsource ~/.bashrc\nmodule purge\nmodule load git\nconda activate environment\necho "running" \necho $1\necho $2\npython -u $1"""
    # Write the script string to the file
    with open(file_name, 'w') as file:
        file.write(script_string)
    
    return file_name

if __name__ == "__main__":
    # Set up argument parsing for the YAML configuration file path
    parser = argparse.ArgumentParser(description="Run tasks from a YAML file.")
    parser.add_argument('yaml_path', type=str, help='Path to the YAML configuration file.')
    args = parser.parse_args()

    # Extract the configuration path from the parsed arguments
    config_path = args.yaml_path

    # Number of CPUs used
    num_cpus = 64
    # Whether to overwrite existing results
    overwrite = False

    # Generate configurations from the YAML file
    results_folder, all_configurations = generate_configurations(config_path)
    # Number of repeats for each configuration
    nb_repeats = all_configurations[0].nb_repeats

    if not os.path.exists(results_folder):
                os.makedirs(results_folder)

    already_there = []
    configurations = []

    # Process each configuration
    for configuration in all_configurations:
        suffix = generate_name_from_params(configuration)
        save_folder = configuration.save_folder + '/' + suffix

        if not os.path.exists(save_folder):
            os.makedirs(save_folder)
        
        current_results_files = [f for f in os.listdir(save_folder) if os.path.isfile(os.path.join(save_folder, f))]
        
        if suffix not in configurations:
            if (overwrite == False) and (suffix + '.dat' in current_results_files):
                already_there.append(suffix)
                pass
            else:
                configurations.append(configuration)

    num_configs = len(configurations) 

    print("number of todo configs ", num_configs)
    print("number of existing configs ", len(already_there))
    print("cpus %s" % num_cpus)

    print("Shuffle configs to create equal computation time chunks ")
    shuffle(configurations)
    if num_configs == 0:
        raise ValueError("No configs to do...")
    
    # Function to split configurations into chunks
    def split(a, n):
        k, m = len(a) // n, len(a) % n
        return [a[i * k + min(i, m):(i + 1) * k + min(i + 1, m)] for i in range(n)]
    
    # Iterate over configurations to set up and submit jobs
    for config in configurations:
        suffix = generate_string_from_params(config)
        save_folder = config.save_folder + '/' + suffix

        print('save folder', save_folder)
        
        runs_folder = save_folder + '/runs'
        if not os.path.exists(runs_folder):
            os.makedirs(runs_folder)

        print('saving runs to', runs_folder)
        # Script to be executed in the SLURM job
        job_list = [(conf, i) for i in range(nb_repeats) for conf in config]
        use_script = 'endogenous_contexts_simulation_study/compute_regime_graphs_timeseries.py'
        # Update the SLURM script with the current configuration
        script_file_name = update_script(save_folder)
    
        print('CONF: ', script_file_name)
        # Construct the sbatch command
        submit_string = ['sbatch', script_file_name, 
                        use_script + " %d %d %s" % (
                        num_cpus, nb_repeats, suffix)] 
        
        process = subprocess.Popen(submit_string) 
        # Execute the sbatch command to submit the job
        output = process.communicate()
