#!/bin/bash

# ============================================================================
# DRFUSE BAYESIAN OPTIMIZATION SEARCH - MODIFY THESE VALUES TO CUSTOMIZE THE SEARCH
# ============================================================================

# Fold selection for bayesian search (can modify to include more folds)
SEARCH_FOLDS=(1)  

# Model Configuration
MODEL="drfuse"
TASK="los"  
GPU=0

# Basic Experiment Settings
PRETRAINED=true
USE_DEMOGRAPHICS=false
CROSS_EVAL=""  # Set to "matched_to_full" or "full_to_matched" if needed
MATCHED=false

# Bayesian Optimization Settings
N_CALLS=20                    # Total number of optimization iterations
N_INITIAL_POINTS=5          # Number of random initial points
ACQUISITION_FUNC="gp_hedge"  # Acquisition function: 'LCB', 'EI', 'PI', 'gp_hedge'
N_JOBS=8                     # Number of parallel jobs (-1 for all cores)

# Resume settings
RESUME_FROM_CHECKPOINT=false  # Set to true to resume from previous run
CHECKPOINT_FILE=""           # Path to previous bayesian_optimization_result.pkl (auto-detect if empty)

# Search Space Bounds - Define parameter ranges for Bayesian optimization
# Format: [min_value, max_value] for continuous parameters
# For discrete parameters, we'll use choice spaces in the Python script

# Core training parameters - FIXED LEARNING RATE
LR_FIXED=0.0001              # Fixed learning rate
BATCH_SIZE_CHOICES="16"
EPOCHS_VALUES=(50)
PATIENCE_VALUES=(10)

# Seeds for multiple runs
SEEDS=(42 123 1234)

# Task-specific parameters
INPUT_DIM_VALUES=(498)                # EHR input dimension
NUM_CLASSES_VALUES=(25)                # For phenotype task: 6, for mortality: 1 (auto-adjusted)

# DRFuse-specific encoder parameters
EHR_ENCODER_CHOICES="transformer"    # EHR encoder options
CXR_ENCODER_CHOICES="resnet50"   # CXR encoder options

# Architecture parameters
HIDDEN_SIZE_CHOICES="256"         # Hidden dimension choices
EHR_DROPOUT_FIXED=0.2            # Fixed dropout rate

# EHR Transformer-specific parameters (when ehr_encoder = 'transformer')
EHR_N_HEAD_CHOICES="4"                 # Number of attention heads
EHR_N_LAYERS_DISTINCT_CHOICES="1"     # Distinct layers
EHR_N_LAYERS_FEAT_CHOICES="1"         # Feature layers
EHR_N_LAYERS_SHARED_CHOICES="1"       # Shared layers

# EHR LSTM-specific parameters (when ehr_encoder = 'lstm')
EHR_LSTM_BIDIRECTIONAL_CHOICES="true"
EHR_LSTM_NUM_LAYERS_CHOICES="1"

# Fusion parameters
FUSION_METHOD_CHOICES="concate"          # Only concate supported for now
LOGIT_AVERAGE_CHOICES="true"
ATTN_FUSION_CHOICES="mid"
DISENTANGLE_LOSS_CHOICES="jsd"

# Lambda weight parameters (continuous ranges)
LAMBDA_DISENTANGLE_SHARED_MIN=0.01
LAMBDA_DISENTANGLE_SHARED_MAX=2.0
LAMBDA_DISENTANGLE_EHR_MIN=0.01
LAMBDA_DISENTANGLE_EHR_MAX=2.0
LAMBDA_DISENTANGLE_CXR_MIN=0.01
LAMBDA_DISENTANGLE_CXR_MAX=2.0
LAMBDA_PRED_EHR_MIN=0.01
LAMBDA_PRED_EHR_MAX=2.0
LAMBDA_PRED_CXR_MIN=0.01
LAMBDA_PRED_CXR_MAX=2.0
LAMBDA_PRED_SHARED_MIN=0.01
LAMBDA_PRED_SHARED_MAX=2.0
LAMBDA_ATTN_AUX_MIN=0.01
LAMBDA_ATTN_AUX_MAX=2.0

# ============================================================================
# SCRIPT IMPLEMENTATION - GENERALLY NO NEED TO MODIFY BELOW THIS LINE
# ============================================================================

# Function to generate dynamic results directory
generate_results_dir() {
    local model=$1
    local task=$2
    local use_demographics=$3
    local cross_eval=$4
    local matched=$5
    local pretrained=$6
    
    local demographic_str
    if [ "$use_demographics" = "true" ]; then
        demographic_str="demo"
    else
        demographic_str="no_demo"
    fi
    
    local matched_str
    if [ "$matched" = "true" ]; then
        matched_str="matched"
    else
        matched_str="full"
    fi
    
    local pretrained_str
    if [ "$pretrained" = "true" ]; then
        pretrained_str="pretrained"
    else
        pretrained_str="no_pretrained"
    fi
    
    # Handle cross_eval parameter
    local cross_eval_str
    if [ -n "$cross_eval" ]; then
        cross_eval_str="$cross_eval"
    else
        cross_eval_str="standard"
    fi
    
    # Generate results directory name
    local results_dirname="${model}_${task}-${demographic_str}-${cross_eval_str}-${matched_str}-${pretrained_str}_bayesian_search_results"
    
    echo "${BASE_DIR}/../bayesian_search_experiments_full/${model}/${task}/lightning_logs/${results_dirname}"
}

BASE_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
RESULTS_DIR=$(generate_results_dir "$MODEL" "$TASK" "$USE_DEMOGRAPHICS" "$CROSS_EVAL" "$MATCHED" "$PRETRAINED")
LOG_FILE="${RESULTS_DIR}/bayesian_search_$(date +%Y%m%d_%H%M%S).log"

# Create results directory
mkdir -p "$RESULTS_DIR"

# Function to log messages
log() {
    echo "[$(date '+%Y-%m-%d %H:%M:%S')] $1" | tee -a "$LOG_FILE"
}

# Create the Bayesian optimization script
create_bayesian_optimizer() {
cat > "${RESULTS_DIR}/bayesian_optimizer.py" << 'EOF'
import os
import sys
import subprocess
import json
import pandas as pd
import numpy as np
from datetime import datetime
import re

# Check if skopt is available, if not install it
try:
    from skopt import gp_minimize
    from skopt.space import Real, Integer, Categorical
    from skopt.utils import use_named_args
    from skopt.acquisition import gaussian_ei, gaussian_lcb, gaussian_pi
    from skopt import dump, load
    print("scikit-optimize is available")
except ImportError:
    print("Installing scikit-optimize...")
    subprocess.check_call([sys.executable, "-m", "pip", "install", "scikit-optimize"])
    from skopt import gp_minimize
    from skopt.space import Real, Integer, Categorical
    from skopt.utils import use_named_args
    from skopt.acquisition import gaussian_ei, gaussian_lcb, gaussian_pi
    from skopt import dump, load

class BayesianDRFuseOptimizer:
    def __init__(self, config):
        self.config = config
        self.results_dir = config['results_dir']
        self.log_file = config['log_file']
        self.iteration = 0
        self.best_score = -np.inf
        self.best_params = None
        
        # Initialize results tracking
        self.results_data = []
        
        # Resume from checkpoint logic
        self.previous_result = None
        if config.get('resume_from_checkpoint', False):
            checkpoint_file = config.get('checkpoint_file', '')
            if not checkpoint_file:
                # Auto-detect checkpoint file
                checkpoint_file = os.path.join(self.results_dir, "bayesian_optimization_result.pkl")
            
            if os.path.exists(checkpoint_file):
                try:
                    self.previous_result = load(checkpoint_file)
                    self.log(f"Loaded checkpoint from: {checkpoint_file}")
                    self.log(f"Previous optimization had {len(self.previous_result.x_iters)} iterations")
                    
                    # Load previous results data if exists
                    csv_file = os.path.join(self.results_dir, "results_summary.csv")
                    if os.path.exists(csv_file):
                        prev_df = pd.read_csv(csv_file)
                        self.results_data = prev_df.to_dict('records')
                        self.iteration = len(self.results_data)
                        
                        # Find best previous result
                        if len(prev_df) > 0:
                            best_row = prev_df.loc[prev_df['acc'].idxmax()]
                            self.best_score = best_row['acc']
                            
                    self.log(f"Resuming from iteration {self.iteration}, best ACC so far: {self.best_score:.4f}")
                    
                except Exception as e:
                    self.log(f"Failed to load checkpoint: {e}, starting fresh")
                    self.previous_result = None
            else:
                self.log(f"Checkpoint file not found: {checkpoint_file}, starting fresh")
        
        # Define search space for DRFuse - REMOVED EHR_DROPOUT (now fixed)
        self.dimensions = [
            Categorical(config['batch_size_choices'], name='batch_size'),
            Categorical(config['ehr_encoder_choices'], name='ehr_encoder'),
            Categorical(config['cxr_encoder_choices'], name='cxr_encoder'),
            Categorical(config['hidden_size_choices'], name='hidden_size'),
            Categorical(config['ehr_n_head_choices'], name='ehr_n_head'),
            Categorical(config['ehr_n_layers_distinct_choices'], name='ehr_n_layers_distinct'),
            Categorical(config['ehr_n_layers_feat_choices'], name='ehr_n_layers_feat'),
            Categorical(config['ehr_n_layers_shared_choices'], name='ehr_n_layers_shared'),
            Categorical(config['ehr_lstm_bidirectional_choices'], name='ehr_lstm_bidirectional'),
            Categorical(config['ehr_lstm_num_layers_choices'], name='ehr_lstm_num_layers'),
            Categorical(config['logit_average_choices'], name='logit_average'),
            Categorical(config['attn_fusion_choices'], name='attn_fusion'),
            Categorical(config['disentangle_loss_choices'], name='disentangle_loss'),
            Real(config['lambda_disentangle_shared_min'], config['lambda_disentangle_shared_max'], name='lambda_disentangle_shared'),
            Real(config['lambda_disentangle_ehr_min'], config['lambda_disentangle_ehr_max'], name='lambda_disentangle_ehr'),
            Real(config['lambda_disentangle_cxr_min'], config['lambda_disentangle_cxr_max'], name='lambda_disentangle_cxr'),
            Real(config['lambda_pred_ehr_min'], config['lambda_pred_ehr_max'], name='lambda_pred_ehr'),
            Real(config['lambda_pred_cxr_min'], config['lambda_pred_cxr_max'], name='lambda_pred_cxr'),
            Real(config['lambda_pred_shared_min'], config['lambda_pred_shared_max'], name='lambda_pred_shared'),
            Real(config['lambda_attn_aux_min'], config['lambda_attn_aux_max'], name='lambda_attn_aux')
        ]
        
        self.dimension_names = [dim.name for dim in self.dimensions]
        
    def log(self, message):
        timestamp = datetime.now().strftime('%Y-%m-%d %H:%M:%S')
        log_message = f"[{timestamp}] {message}"
        print(log_message)
        with open(self.log_file, 'a') as f:
            f.write(log_message + '\n')
    
    def is_valid_drfuse_config(self, params_dict):
        """Check if DRFuse configuration is valid"""
        # Check if transformer head configuration is valid when using transformer encoder
        if params_dict['ehr_encoder'] == 'transformer':
            hidden_size = params_dict['hidden_size']
            n_head = params_dict['ehr_n_head']
            return hidden_size % n_head == 0
        return True
    
    def extract_metrics_from_log(self, log_file):
        """Extract all performance metrics from experiment log"""
        try:
            with open(log_file, 'r') as f:
                content = f.read()
            
            metrics = {}
            
            # Extract all metrics using regex patterns
            patterns = {
                'ACC': r"overall/ACC:\s*([0-9]+\.[0-9]+)",
                'F1_macro': r"overall/F1_macro:\s*([0-9]+\.[0-9]+)",
                'F1_weighted': r"overall/F1_weighted:\s*([0-9]+\.[0-9]+)",
                'Precision_macro': r"overall/Precision_macro:\s*([0-9]+\.[0-9]+)",
                'Precision_weighted': r"overall/Precision_weighted:\s*([0-9]+\.[0-9]+)",
                'Recall_macro': r"overall/Recall_macro:\s*([0-9]+\.[0-9]+)",
                'Recall_weighted': r"overall/Recall_weighted:\s*([0-9]+\.[0-9]+)"
            }
            
            for metric_name, pattern in patterns.items():
                matches = re.findall(pattern, content)
                if matches:
                    metrics[metric_name] = float(matches[-1])  # Take the last match
                else:
                    metrics[metric_name] = None
                    
            return metrics
        except Exception:
            return {}
    
    def run_experiment_with_seeds(self, params_dict, fold):
        """Run experiment with multiple seeds and return statistics"""
        # Check if configuration is valid for DRFuse
        if not self.is_valid_drfuse_config(params_dict):
            self.log(f"Skipping invalid config: hidden_size={params_dict['hidden_size']}, ehr_n_head={params_dict['ehr_n_head']} (not divisible)")
            return -1.0, -1.0  # Return poor scores for invalid configurations
        
        self.iteration += 1
        
        # Add fixed dropout to params_dict
        params_dict['ehr_dropout'] = self.config['ehr_dropout_fixed']
        
        # Create experiment name
        exp_name = f"bayes_iter{self.iteration}_fold{fold}_lr{self.config['lr_fixed']:.6f}_bs{params_dict['batch_size']}_" \
                   f"ehr_{params_dict['ehr_encoder']}_cxr_{params_dict['cxr_encoder']}_hs{params_dict['hidden_size']}_" \
                   f"edr{params_dict['ehr_dropout']:.3f}_enh{params_dict['ehr_n_head']}_" \
                   f"dist{params_dict['ehr_n_layers_distinct']}_feat{params_dict['ehr_n_layers_feat']}_" \
                   f"shared{params_dict['ehr_n_layers_shared']}_la{params_dict['logit_average']}_" \
                   f"af{params_dict['attn_fusion']}_dl{params_dict['disentangle_loss']}"
        
        self.log(f"Starting Bayesian iteration {self.iteration}: {exp_name}")
        
        # Run experiments for all seeds
        all_metrics = []
        
        for seed in self.config['seeds']:
            seed_exp_name = f"{exp_name}_seed{seed}"
            self.log(f"  Running seed {seed}...")
            
            # Build command for DRFuse with fixed learning rate and dropout
            cmd = [
                "python", "../main.py",
                "--model", self.config['model'],
                "--mode", "train",
                "--task", self.config['task'],
                "--fold", str(fold),
                "--gpu", str(self.config['gpu']),
                "--lr", str(self.config['lr_fixed']),  # Fixed learning rate
                "--batch_size", str(params_dict['batch_size']),
                "--epochs", str(self.config['epochs']),
                "--patience", str(self.config['patience']),
                "--input_dim", str(self.config['input_dim']),
                "--num_classes", str(self.config['num_classes']),
                "--ehr_encoder", params_dict['ehr_encoder'],
                "--cxr_encoder", params_dict['cxr_encoder'],
                "--hidden_size", str(params_dict['hidden_size']),
                "--ehr_dropout", str(params_dict['ehr_dropout']),  # Fixed dropout
                "--ehr_n_head", str(params_dict['ehr_n_head']),
                "--ehr_n_layers_distinct", str(params_dict['ehr_n_layers_distinct']),
                "--ehr_n_layers_feat", str(params_dict['ehr_n_layers_feat']),
                "--ehr_n_layers_shared", str(params_dict['ehr_n_layers_shared']),
                "--ehr_lstm_num_layers", str(params_dict['ehr_lstm_num_layers']),
                "--fusion_method", self.config['fusion_method'],
                "--attn_fusion", params_dict['attn_fusion'],
                "--disentangle_loss", params_dict['disentangle_loss'],
                "--lambda_disentangle_shared", str(params_dict['lambda_disentangle_shared']),
                "--lambda_disentangle_ehr", str(params_dict['lambda_disentangle_ehr']),
                "--lambda_disentangle_cxr", str(params_dict['lambda_disentangle_cxr']),
                "--lambda_pred_ehr", str(params_dict['lambda_pred_ehr']),
                "--lambda_pred_cxr", str(params_dict['lambda_pred_cxr']),
                "--lambda_pred_shared", str(params_dict['lambda_pred_shared']),
                "--lambda_attn_aux", str(params_dict['lambda_attn_aux']),
                "--seed", str(seed),  # Add seed parameter
                "--log_dir", f"../bayesian_search_experiments_full/{self.config['model']}/{self.config['task']}"
            ]
            
            # Add conditional parameters (boolean flags)
            if self.config['pretrained']:
                cmd.append("--pretrained")
                
            if self.config['matched']:
                cmd.append("--matched")

            if self.config['use_demographics']:
                cmd.append("--use_demographics")

            # Handle boolean parameters from optimization
            if params_dict['ehr_lstm_bidirectional'] == 'true':
                cmd.append("--ehr_lstm_bidirectional")

            if params_dict['logit_average'] == 'true':
                cmd.append("--logit_average")

            if self.config['cross_eval']:
                cmd.extend(["--cross_eval", self.config['cross_eval']])
            
            # Create experiment directory for this seed
            seed_exp_dir = os.path.join(self.results_dir, seed_exp_name)
            os.makedirs(seed_exp_dir, exist_ok=True)
            
            # Run experiment
            try:
                with open(os.path.join(seed_exp_dir, "output.log"), "w") as output_file:
                    result = subprocess.run(
                        cmd,
                        cwd=self.config['base_dir'],
                        stdout=output_file,
                        stderr=subprocess.STDOUT,
                        timeout=None  # 1 hour timeout
                    )
                
                # Extract all metrics
                metrics = self.extract_metrics_from_log(os.path.join(seed_exp_dir, "output.log"))
                
                if metrics and any(v is not None for v in metrics.values()):
                    all_metrics.append(metrics)
                    self.log(f"    Seed {seed}: " + " | ".join([f"{k}={v:.4f}" if v is not None else f"{k}=N/A" for k, v in metrics.items()]))
                else:
                    self.log(f"    Seed {seed}: Failed to extract metrics")
                    
            except subprocess.TimeoutExpired:
                self.log(f"    Seed {seed}: Timed out")
            except Exception as e:
                self.log(f"    Seed {seed}: Error - {e}")
        
        # Calculate statistics for all metrics
        if len(all_metrics) > 0:
            # Calculate mean and std for each metric
            metric_stats = {}
            for metric_name in ['ACC', 'F1_macro', 'F1_weighted', 'Precision_macro', 'Precision_weighted', 'Recall_macro', 'Recall_weighted']:
                values = [m.get(metric_name) for m in all_metrics if m.get(metric_name) is not None]
                if values:
                    metric_stats[f'{metric_name}_mean'] = np.mean(values)
                    metric_stats[f'{metric_name}_std'] = np.std(values)
                else:
                    metric_stats[f'{metric_name}_mean'] = None
                    metric_stats[f'{metric_name}_std'] = None
            
            # Log results in the requested format for all metrics
            result_str = f"Iteration {self.iteration} - "
            result_parts = []
            for metric_name in ['ACC', 'F1_macro', 'F1_weighted', 'Precision_macro', 'Precision_weighted', 'Recall_macro', 'Recall_weighted']:
                mean_key = f'{metric_name}_mean'
                std_key = f'{metric_name}_std'
                if metric_stats[mean_key] is not None:
                    result_parts.append(f"{metric_name}: {metric_stats[mean_key]:.4f}±{metric_stats[std_key]:.4f}")
                else:
                    result_parts.append(f"{metric_name}: N/A")
            
            self.log(result_str + " | ".join(result_parts))
            
            # Update best result based on ACC (instead of PRAUC for LoS)
            acc_mean = metric_stats.get('ACC_mean')
            if acc_mean is not None and acc_mean > self.best_score:
                self.best_score = acc_mean
                self.best_params = params_dict.copy()
                self.log(f"New best ACC: {self.best_score:.4f}±{metric_stats.get('ACC_std', 0):.4f}")
            
            # Save result
            result_data = {
                'iteration': self.iteration,
                'experiment_name': exp_name,
                'fold': fold,
                'lr_fixed': self.config['lr_fixed'],
                **params_dict,
                'task': self.config['task'],
                'use_demographics': self.config['use_demographics'],
                'cross_eval': self.config['cross_eval'],
                'pretrained': self.config['pretrained'],
                **metric_stats,  # Include all metric statistics
                'all_metrics': all_metrics  # Include raw metrics from all seeds
            }
            self.results_data.append(result_data)
            
            return acc_mean if acc_mean is not None else -1.0, metric_stats.get('F1_macro_mean', -1.0)
        else:
            self.log(f"Failed to get valid results from any seed in iteration {self.iteration}")
            return -1.0, -1.0
    
    def objective_function(self, params):
        """Objective function for Bayesian optimization"""
        # Convert params list to dict
        params_dict = dict(zip(self.dimension_names, params))
        
        # Validate DRFuse configuration first
        if not self.is_valid_drfuse_config(params_dict):
            # Return very poor score for invalid configurations
            return 1.0  # High value because we minimize
        
        # Run experiments for all folds and average the results
        scores = []
        for fold in self.config['search_folds']:
            score, _ = self.run_experiment_with_seeds(params_dict, fold)
            scores.append(score)
        
        # Return negative score because skopt minimizes
        avg_score = np.mean(scores)
        return -avg_score  # Negative because we want to maximize ACC
    
    def run_optimization(self):
        """Run Bayesian optimization"""
        self.log("Starting Bayesian Optimization for DRFuse")
        self.log(f"Fixed Learning Rate: {self.config['lr_fixed']}")
        self.log(f"Fixed EHR Dropout: {self.config['ehr_dropout_fixed']}")
        self.log(f"Seeds: {self.config['seeds']}")
        self.log(f"Search space: {[dim.name for dim in self.dimensions]}")
        
        if self.previous_result:
            # Continue from previous optimization
            remaining_calls = self.config['n_calls'] - len(self.previous_result.x_iters)
            if remaining_calls <= 0:
                self.log("Previous optimization already completed the requested iterations")
                return self.previous_result
                
            self.log(f"Continuing optimization: {remaining_calls} remaining calls")
            
            # Continue optimization using tell/ask interface
            from skopt import Optimizer
            
            # Create optimizer with same settings
            opt = Optimizer(
                dimensions=self.dimensions,
                acq_func=self.config['acquisition_func'],
                n_initial_points=0,  # No initial points needed
                random_state=42
            )
            
            # Tell the optimizer about previous results
            for x, y in zip(self.previous_result.x_iters, self.previous_result.func_vals):
                opt.tell(x, y)
                
            # Continue optimization
            for i in range(remaining_calls):
                next_x = opt.ask()
                next_y = self.objective_function(next_x)
                opt.tell(next_x, next_y)
                
                # Save checkpoint periodically
                if (i + 1) % 5 == 0:  # Save every 5 iterations
                    optimization_file = os.path.join(self.results_dir, "bayesian_optimization_result.pkl")
                    dump(opt, optimization_file)
                    self.log(f"Checkpoint saved at iteration {len(self.previous_result.x_iters) + i + 1}")
            
            result = opt
            
        else:
            # Fresh start
            self.log(f"Total iterations: {self.config['n_calls']}")
            self.log(f"Initial random points: {self.config['n_initial_points']}")
            
            # Run optimization
            result = gp_minimize(
                func=self.objective_function,
                dimensions=self.dimensions,
                n_calls=self.config['n_calls'],
                n_initial_points=self.config['n_initial_points'],
                acq_func=self.config['acquisition_func'],
                n_jobs=self.config['n_jobs'],
                random_state=42
            )
        
        # Save optimization result
        optimization_file = os.path.join(self.results_dir, "bayesian_optimization_result.pkl")
        dump(result, optimization_file)
        
        # Save all results to CSV
        if self.results_data:
            df = pd.DataFrame(self.results_data)
            csv_file = os.path.join(self.results_dir, "results_summary.csv")
            df.to_csv(csv_file, index=False)
        
        # Final analysis
        self.log("=== BAYESIAN OPTIMIZATION COMPLETED ===")
        self.log(f"Best ACC found: {self.best_score:.4f}")
        self.log(f"Best parameters: {self.best_params}")
        
        # Save best parameters
        best_params_file = os.path.join(self.results_dir, "best_params.txt")
        with open(best_params_file, 'w') as f:
            f.write("DRFuse Bayesian Optimization Best Parameters\n")
            f.write("=" * 50 + "\n")
            f.write(f"Best ACC: {self.best_score:.4f}\n")
            f.write(f"Total iterations: {self.iteration}\n\n")
            f.write("Best Parameters:\n")
            if self.best_params:
                for param, value in self.best_params.items():
                    f.write(f"  {param}: {value}\n")
            f.write(f"Fixed Learning Rate: {self.config['lr_fixed']}\n")
            f.write(f"Fixed EHR Dropout: {self.config['ehr_dropout_fixed']}\n")
            f.write(f"Seeds used: {self.config['seeds']}\n")
        
        # Generate convergence plot
        self.generate_convergence_plot(result)
        
        return result
    
    def generate_convergence_plot(self, result):
        """Generate convergence plot"""
        try:
            import matplotlib.pyplot as plt
            
            plt.figure(figsize=(15, 10), facecolor='white')
            
            # Plot convergence
            scores = [-y for y in result.func_vals]  # Convert back to positive
            best_scores = [max(scores[:i+1]) for i in range(len(scores))]
            
            plt.subplot(2, 3, 1)
            plt.plot(scores, 'bo-', alpha=0.6, label='ACC')
            plt.plot(best_scores, 'r-', linewidth=2, label='Best ACC')
            plt.xlabel('Iteration')
            plt.ylabel('ACC')
            plt.title('Bayesian Optimization Convergence')
            plt.legend()
            plt.grid(True, alpha=0.3)
            
            # Plot parameter exploration
            if hasattr(result, 'x_iters') and len(result.x_iters) > 5:
                param_data = pd.DataFrame(result.x_iters, columns=self.dimension_names)
                
                # Hidden size vs EHR n_head relationship
                plt.subplot(2, 3, 2)
                for i, (hidden_size, n_head) in enumerate(zip(param_data['hidden_size'], param_data['ehr_n_head'])):
                    color = 'green' if hidden_size % n_head == 0 else 'red'
                    plt.scatter(hidden_size, n_head, c=color, alpha=0.6)
                plt.xlabel('hidden_size')
                plt.ylabel('ehr_n_head')
                plt.title('Hidden Size vs EHR N_Head (Green=Valid, Red=Invalid)')
                plt.grid(True, alpha=0.3)
                
                # Weight decay progression
                plt.subplot(2, 3, 3)
                plt.plot(param_data['ehr_dropout'], 'g-', alpha=0.7)
                plt.xlabel('Iteration')
                plt.ylabel('EHR Dropout')
                plt.title('EHR Dropout Exploration')
                plt.grid(True, alpha=0.3)
                
                # Lambda parameters exploration
                plt.subplot(2, 3, 4)
                lambda_cols = [col for col in param_data.columns if 'lambda' in col]
                for col in lambda_cols[:3]:  # Show first 3 lambda parameters
                    plt.plot(param_data[col], alpha=0.7, label=col)
                plt.xlabel('Iteration')
                plt.ylabel('Lambda Values')
                plt.title('Lambda Parameters Exploration')
                plt.legend()
                plt.grid(True, alpha=0.3)
                
                # Encoder combination exploration
                plt.subplot(2, 3, 5)
                ehr_encoders = param_data['ehr_encoder'].unique()
                cxr_encoders = param_data['cxr_encoder'].unique()
                
                for i, ehr_enc in enumerate(ehr_encoders):
                    for j, cxr_enc in enumerate(cxr_encoders):
                        mask = (param_data['ehr_encoder'] == ehr_enc) & (param_data['cxr_encoder'] == cxr_enc)
                        if mask.any():
                            combo_scores = [scores[k] for k in range(len(scores)) if mask.iloc[k]]
                            plt.scatter([f"{ehr_enc}+{cxr_enc}"] * len(combo_scores), combo_scores, 
                                      alpha=0.7, label=f"{ehr_enc}+{cxr_enc}")
                plt.xlabel('Encoder Combination')
                plt.ylabel('ACC')
                plt.title('Performance by Encoder Combination')
                plt.xticks(rotation=45)
                plt.grid(True, alpha=0.3)
                
                # Architecture layers vs performance
                plt.subplot(2, 3, 6)
                total_layers = param_data['ehr_n_layers_distinct'] + param_data['ehr_n_layers_feat'] + param_data['ehr_n_layers_shared']
                plt.scatter(total_layers, param_data['hidden_size'], 
                           c=scores, cmap='viridis', alpha=0.7)
                plt.colorbar(label='ACC')
                plt.xlabel('Total EHR Layers')
                plt.ylabel('Hidden Size')
                plt.title('Architecture vs Performance')
                plt.grid(True, alpha=0.3)
            
            plt.tight_layout()
            plot_file = os.path.join(self.results_dir, "convergence_plot.png")
            plt.savefig(plot_file, dpi=300, bbox_inches='tight')
            plt.close()
            
            self.log(f"Convergence plot saved to: {plot_file}")
            
        except Exception as e:
            self.log(f"Could not generate convergence plot: {e}")

def main():
    # Read configuration from environment variables
    config = {
        'results_dir': os.environ.get('RESULTS_DIR'),
        'log_file': os.environ.get('LOG_FILE'),
        'base_dir': os.environ.get('BASE_DIR'),
        'model': os.environ.get('MODEL'),
        'task': os.environ.get('TASK'),
        'gpu': int(os.environ.get('GPU', 0)),
        'epochs': int(os.environ.get('EPOCHS', 50)),
        'patience': int(os.environ.get('PATIENCE', 10)),
        'input_dim': int(os.environ.get('INPUT_DIM', 498)),
        'num_classes': int(os.environ.get('NUM_CLASSES', 6)),
        'pretrained': os.environ.get('PRETRAINED', 'true').lower() == 'true',
        'matched': os.environ.get('MATCHED', 'true').lower() == 'true',
        'use_demographics': os.environ.get('USE_DEMOGRAPHICS', 'false').lower() == 'true',
        'cross_eval': os.environ.get('CROSS_EVAL', ''),
        'fusion_method': os.environ.get('FUSION_METHOD', 'concate'),
        'search_folds': [int(x) for x in os.environ.get('SEARCH_FOLDS', '1').split(',')],
        
        # Bayesian optimization parameters
        'n_calls': int(os.environ.get('N_CALLS', 20)),  # 改为20
        'n_initial_points': int(os.environ.get('N_INITIAL_POINTS', 5)),  # 改为5
        'acquisition_func': os.environ.get('ACQUISITION_FUNC', 'gp_hedge'),
        'n_jobs': int(os.environ.get('N_JOBS', 1)),
        
        # Resume parameters
        'resume_from_checkpoint': os.environ.get('RESUME_FROM_CHECKPOINT', 'false').lower() == 'true',
        'checkpoint_file': os.environ.get('CHECKPOINT_FILE', ''),
        
        # Search space bounds
        'lr_fixed': float(os.environ.get('LR_FIXED', 0.0001)),
        'seeds': [int(x) for x in os.environ.get('SEEDS', '42,123,1234').split(',')],
        'batch_size_choices': [int(x) for x in os.environ.get('BATCH_SIZE_CHOICES', '16,32').split(',')],
        'ehr_dropout_fixed': float(os.environ.get('EHR_DROPOUT_FIXED', 0.2)),
        
        # DRFuse-specific parameters
        'ehr_encoder_choices': os.environ.get('EHR_ENCODER_CHOICES', 'transformer,lstm').split(','),
        'cxr_encoder_choices': os.environ.get('CXR_ENCODER_CHOICES', 'resnet50,vit_b_16').split(','),
        'hidden_size_choices': [int(x) for x in os.environ.get('HIDDEN_SIZE_CHOICES', '128,256,512').split(',')],
        'ehr_n_head_choices': [int(x) for x in os.environ.get('EHR_N_HEAD_CHOICES', '4,8').split(',')],
        'ehr_n_layers_distinct_choices': [int(x) for x in os.environ.get('EHR_N_LAYERS_DISTINCT_CHOICES', '1,2').split(',')],
        'ehr_n_layers_feat_choices': [int(x) for x in os.environ.get('EHR_N_LAYERS_FEAT_CHOICES', '1,2').split(',')],
        'ehr_n_layers_shared_choices': [int(x) for x in os.environ.get('EHR_N_LAYERS_SHARED_CHOICES', '1,2').split(',')],
        'ehr_lstm_bidirectional_choices': os.environ.get('EHR_LSTM_BIDIRECTIONAL_CHOICES', 'true,false').split(','),
        'ehr_lstm_num_layers_choices': [int(x) for x in os.environ.get('EHR_LSTM_NUM_LAYERS_CHOICES', '1,2,3').split(',')],
        'logit_average_choices': os.environ.get('LOGIT_AVERAGE_CHOICES', 'true,false').split(','),
        'attn_fusion_choices': os.environ.get('ATTN_FUSION_CHOICES', 'mid,late').split(','),
        'disentangle_loss_choices': os.environ.get('DISENTANGLE_LOSS_CHOICES', 'mse,jsd').split(','),
        
        # Lambda parameters
        'lambda_disentangle_shared_min': float(os.environ.get('LAMBDA_DISENTANGLE_SHARED_MIN', 0.1)),
        'lambda_disentangle_shared_max': float(os.environ.get('LAMBDA_DISENTANGLE_SHARED_MAX', 2.0)),
        'lambda_disentangle_ehr_min': float(os.environ.get('LAMBDA_DISENTANGLE_EHR_MIN', 0.1)),
        'lambda_disentangle_ehr_max': float(os.environ.get('LAMBDA_DISENTANGLE_EHR_MAX', 2.0)),
        'lambda_disentangle_cxr_min': float(os.environ.get('LAMBDA_DISENTANGLE_CXR_MIN', 0.1)),
        'lambda_disentangle_cxr_max': float(os.environ.get('LAMBDA_DISENTANGLE_CXR_MAX', 2.0)),
        'lambda_pred_ehr_min': float(os.environ.get('LAMBDA_PRED_EHR_MIN', 0.1)),
        'lambda_pred_ehr_max': float(os.environ.get('LAMBDA_PRED_EHR_MAX', 2.0)),
        'lambda_pred_cxr_min': float(os.environ.get('LAMBDA_PRED_CXR_MIN', 0.1)),
        'lambda_pred_cxr_max': float(os.environ.get('LAMBDA_PRED_CXR_MAX', 2.0)),
        'lambda_pred_shared_min': float(os.environ.get('LAMBDA_PRED_SHARED_MIN', 0.1)),
        'lambda_pred_shared_max': float(os.environ.get('LAMBDA_PRED_SHARED_MAX', 2.0)),
        'lambda_attn_aux_min': float(os.environ.get('LAMBDA_ATTN_AUX_MIN', 0.1)),
        'lambda_attn_aux_max': float(os.environ.get('LAMBDA_ATTN_AUX_MAX', 2.0))
    }
    
    # Auto-adjust num_classes for mortality task
    if config['task'] == 'mortality':
        config['num_classes'] = 1
    
    # Create and run optimizer
    optimizer = BayesianDRFuseOptimizer(config)
    result = optimizer.run_optimization()
    
    print("DRFuse Bayesian optimization completed successfully!")

if __name__ == "__main__":
    main()
EOF
}

# Main execution
main() {
    log "Starting DRFuse Bayesian Optimization Search"
    log "Configuration: MODEL=$MODEL, TASK=$TASK, USE_DEMOGRAPHICS=$USE_DEMOGRAPHICS, CROSS_EVAL=$CROSS_EVAL, PRETRAINED=$PRETRAINED"
    log "Results will be saved to: $RESULTS_DIR"
    log "Log file: $LOG_FILE"
    log "Total optimization calls: $N_CALLS"
    log "Initial random points: $N_INITIAL_POINTS"
    log "Acquisition function: $ACQUISITION_FUNC"
    
    # Create the Bayesian optimizer Python script
    create_bayesian_optimizer
    
    # Set environment variables for the Python script
    export RESULTS_DIR="$RESULTS_DIR"
    export LOG_FILE="$LOG_FILE"
    export BASE_DIR="$BASE_DIR"
    export MODEL="$MODEL"
    export TASK="$TASK"
    export GPU="$GPU"
    export EPOCHS="${EPOCHS_VALUES[0]}"
    export PATIENCE="${PATIENCE_VALUES[0]}"
    export INPUT_DIM="${INPUT_DIM_VALUES[0]}"
    export NUM_CLASSES="${NUM_CLASSES_VALUES[0]}"
    export PRETRAINED="$PRETRAINED"
    export MATCHED="$MATCHED"
    export USE_DEMOGRAPHICS="$USE_DEMOGRAPHICS"
    export CROSS_EVAL="$CROSS_EVAL"
    export SEARCH_FOLDS=$(IFS=,; echo "${SEARCH_FOLDS[*]}")
    export FUSION_METHOD="$FUSION_METHOD_CHOICES"
    
    # Bayesian optimization parameters
    export N_CALLS="$N_CALLS"
    export N_INITIAL_POINTS="$N_INITIAL_POINTS"
    export ACQUISITION_FUNC="$ACQUISITION_FUNC"
    export N_JOBS="$N_JOBS"
    
    # Resume parameters
    export RESUME_FROM_CHECKPOINT="$RESUME_FROM_CHECKPOINT"
    export CHECKPOINT_FILE="$CHECKPOINT_FILE"
    
    # Search space bounds
    export LR_FIXED="$LR_FIXED"
    export SEEDS=$(IFS=,; echo "${SEEDS[*]}")
    export BATCH_SIZE_CHOICES="$BATCH_SIZE_CHOICES"
    export EHR_DROPOUT_FIXED="$EHR_DROPOUT_FIXED"
    
    # DRFuse-specific parameters
    export EHR_ENCODER_CHOICES="$EHR_ENCODER_CHOICES"
    export CXR_ENCODER_CHOICES="$CXR_ENCODER_CHOICES"
    export HIDDEN_SIZE_CHOICES="$HIDDEN_SIZE_CHOICES"
    export EHR_N_HEAD_CHOICES="$EHR_N_HEAD_CHOICES"
    export EHR_N_LAYERS_DISTINCT_CHOICES="$EHR_N_LAYERS_DISTINCT_CHOICES"
    export EHR_N_LAYERS_FEAT_CHOICES="$EHR_N_LAYERS_FEAT_CHOICES"
    export EHR_N_LAYERS_SHARED_CHOICES="$EHR_N_LAYERS_SHARED_CHOICES"
    export EHR_LSTM_BIDIRECTIONAL_CHOICES="$EHR_LSTM_BIDIRECTIONAL_CHOICES"
    export EHR_LSTM_NUM_LAYERS_CHOICES="$EHR_LSTM_NUM_LAYERS_CHOICES"
    export LOGIT_AVERAGE_CHOICES="$LOGIT_AVERAGE_CHOICES"
    export ATTN_FUSION_CHOICES="$ATTN_FUSION_CHOICES"
    export DISENTANGLE_LOSS_CHOICES="$DISENTANGLE_LOSS_CHOICES"
    
    # Lambda parameters
    export LAMBDA_DISENTANGLE_SHARED_MIN="$LAMBDA_DISENTANGLE_SHARED_MIN"
    export LAMBDA_DISENTANGLE_SHARED_MAX="$LAMBDA_DISENTANGLE_SHARED_MAX"
    export LAMBDA_DISENTANGLE_EHR_MIN="$LAMBDA_DISENTANGLE_EHR_MIN"
    export LAMBDA_DISENTANGLE_EHR_MAX="$LAMBDA_DISENTANGLE_EHR_MAX"
    export LAMBDA_DISENTANGLE_CXR_MIN="$LAMBDA_DISENTANGLE_CXR_MIN"
    export LAMBDA_DISENTANGLE_CXR_MAX="$LAMBDA_DISENTANGLE_CXR_MAX"
    export LAMBDA_PRED_EHR_MIN="$LAMBDA_PRED_EHR_MIN"
    export LAMBDA_PRED_EHR_MAX="$LAMBDA_PRED_EHR_MAX"
    export LAMBDA_PRED_CXR_MIN="$LAMBDA_PRED_CXR_MIN"
    export LAMBDA_PRED_CXR_MAX="$LAMBDA_PRED_CXR_MAX"
    export LAMBDA_PRED_SHARED_MIN="$LAMBDA_PRED_SHARED_MIN"
    export LAMBDA_PRED_SHARED_MAX="$LAMBDA_PRED_SHARED_MAX"
    export LAMBDA_ATTN_AUX_MIN="$LAMBDA_ATTN_AUX_MIN"
    export LAMBDA_ATTN_AUX_MAX="$LAMBDA_ATTN_AUX_MAX"
    
    log "Starting Python Bayesian optimizer..."
    
    # Run the Bayesian optimizer
    cd "$BASE_DIR"
    python3 "${RESULTS_DIR}/bayesian_optimizer.py"
    
    if [ $? -eq 0 ]; then
        log "DRFuse Bayesian optimization completed successfully!"
        log "Results saved to: $RESULTS_DIR"
        log "Best parameters in: $RESULTS_DIR/best_params.txt"
        log "Full results in: $RESULTS_DIR/results_summary.csv"
        log "Optimization object saved in: $RESULTS_DIR/bayesian_optimization_result.pkl"
    else
        log "DRFuse Bayesian optimization failed!"
        exit 1
    fi
}

# Handle script interruption
cleanup() {
    log "DRFuse Bayesian search interrupted by user"
    exit 1
}

trap cleanup SIGINT SIGTERM

# Run main function
main "$@"
