# nohup python -u run_pisde_gm_ablations.py > run_pisde_gm_ablations.log 2>&1 &

import subprocess
from queue import Queue
from threading import Thread
import time
import os
import sys

def run_jobs(
    job_list, 
    gpus, 
    logger, 
    output_dir
):
    
    job_queue = Queue()
    for job in job_list:
        job_queue.put(job)

    print(f"Total jobs to run: {job_queue.qsize()}")

    def worker(gpu_id):

        while not job_queue.empty():
            
            try:
                config_dict = job_queue.get_nowait()
            except Queue.Empty:
                break

            try:                
                
                # 2. Construct Hydra Command
                # We build a list of "key=value" strings
                overrides = [f"{k}={v}" for k, v in config_dict.items()]
                cmd = [sys.executable, "-u", "/home/fran/work_fran/sampling/experiments/main.py"] \
                    + overrides \
                    + [f"logger={logger}"]
                
                task_name = config_dict["task"]
                sampler_name = config_dict["sampler"]
                timestamp = int(time.time())
                log_filename = f"{output_dir}/{sampler_name}_{task_name}_gpu{gpu_id}_{timestamp}.log"

                current_env = os.environ.copy()
                current_env["CUDA_VISIBLE_DEVICES"] = str(gpu_id)
                current_env["CUDA_LAUNCH_BLOCKING"] = "1"
                current_env["CUDA_DEVICE_ORDER"] = "PCI_BUS_ID"

                print(f"[GPU {gpu_id}] Remaining jobs: {job_queue.qsize()}")
                print(f"[GPU {gpu_id}] Starting job with config:\n{config_dict}")
                
                # 3. Run Process
                # We pass the GPU ID via env var
                with open(log_filename, "w") as log_file:
                    subprocess.run(
                        cmd, 
                        env=current_env,
                        check=False, # Continue even if the process fails
                        stdout=log_file,  # Redirects print() to file
                        stderr=subprocess.STDOUT # Redirects errors to the same file
                    )
            except Exception as e:
                print(f"[GPU {gpu_id}] Error with config {config_dict}: {e}")
            finally:
                job_queue.task_done()
    
    # Launch Threads
    threads = []
    for gpu in gpus:
        t = Thread(target=worker, args=(gpu,))
        t.start()
        threads.append(t)
        # Stagger starts slightly to prevent race conditions on file creation
        time.sleep(5)