import numpy as np
import os
import sys
import pickle
from joblib import Parallel, delayed

# To run this script: python -m experiments.experiment_3

"""
Experiment 3: The Incentivizability Region

This experiment focuses solely on the  of the mechanism.

We sweep the Gaming Cost (c_minus) to empirically demonstrate the bound where 
c^- is too low for any improvement incentivization. 

Hypothesis:
    The Max Reachable Attribute will be 0 when c_minus is below the critical threshold:
    Critical Point = (1 - beta * gamma) * c_plus

Output:
    - Saves a dictionary of results to: organized/data/exp_3/exp_3_data.pkl

"""

sys.path.append(os.path.abspath(os.path.join(os.path.dirname(__file__), '..')))
from functions.greedy_search import find_optimal_thresholds_greedy

def solve_single_c_minus(cm, base_params):
    """
    Solves for Max Attribute given a specific gaming cost (cm).
    """
    p = base_params.copy()
    p['c_minus'] = cm
    
    mus = find_optimal_thresholds_greedy(
        params=p, 
        target_M=None, 
        max_L=100, 
        search_cap=250.0
    )
    # Return the last threshold found (Max Incentivizable Attribute)
    return mus[-1]

def run_cliff_sweep():
    # Base Parameters
    base_params = {
        'beta': 0.8,
        'gamma': 0.9,
        'c_plus': 1.0,
        'c_minus': 0.7,   
        'r': 1.0,       
        'delta': 0.0,    
        'mu_list': []
    }
    
    print("\n Running Sweep: (c_minus)")
    
    # Sweep from 0.1 to 1.2 
    c_minus_vals = np.linspace(0.1, 0.99, 50)
    
    # Parallel Execution
    max_attr_cminus = Parallel(n_jobs=-1, verbose=5)(
        delayed(solve_single_c_minus)(cm, base_params) for cm in c_minus_vals
    )

    return {
        'c_minus_sweep': {
            'x_vals': c_minus_vals,
            'y_vals': np.array(max_attr_cminus)
        },

        'params': {
            'beta': base_params['beta'],
            'gamma': base_params['gamma'],
            'c_plus': base_params['c_plus']
        }
    }

def save_data(results):
    data_dir = os.path.join('data', 'exp_3')
    if not os.path.exists(data_dir): os.makedirs(data_dir)
    
    file_path = os.path.join(data_dir, 'exp_3_data.pkl')
    with open(file_path, 'wb') as f:
        pickle.dump(results, f)
    print(f"Data saved to {file_path}")

if __name__ == "__main__":
    results = run_cliff_sweep()
    save_data(results)