# Contract Theory Experiments

This repository provides an experimental framework for simulating and analyzing **principal–agent contract models** under various settings. It includes implementations of bilevel optimization solvers, reproducible experiment harnesses, and utilities for aggregating and visualizing results.

The workflow supports:

  - **Local runs** for small sweeps and debugging
  - **SLURM jobs** for large-scale experiments
  - **Splitting and merging traces** when jobs exceed runtime limits

-----

## 🖥️ Running Experiments

### Running Locally

Local runs are useful for debugging and testing parameters before scaling up. Each experiment requires a `SETTING`, a `PARAM` to sweep, `GRID` values, and `OUTER_STEPS` for iterations.

**Example**

```bash
SETTINGS=imperfect_signal \
PARAM=sigma \
GRID=0.1 \
OUTER_STEPS=5000 \
python -u logic/experiment.py
```

This generates a trace file at the folder mentioned in the helper function.

-----

### Running on SLURM

For large sweeps or high iteration counts, use SLURM jobs to leverage GPUs. The example below shows a typical submission script.

**Example (σ = 0.01,0.1,0.2,0.4,0.6,0.8,1 in `insurance_prevention`)**

```bash
#!/bin/bash
#SBATCH --job-name=sigma_insurance_prevention_2-5M
#SBATCH --time=72:00:00
#SBATCH --nodes=1
#SBATCH --ntasks=1
#SBATCH --cpus-per-task=8
#SBATCH --mem=64G
#SBATCH --partition=long
#SBATCH --output=/scratch/user/aaryabookseller/Research/archive/script_outputs/insurance_prevention/sigma/25M_updated_sigma_insurance_prevention.out
#SBATCH --error=/scratch/user/aaryabookseller/Research/archive/script_outputs/insurance_prevention/sigma/25M_updated_sigma_insurance_prevention.err
#SBATCH --chdir=/scratch/user/aaryabookseller/Research/archive

module load GCC/13.2.0
module load texlive/20230313
source activate_venv simclr_venv

settings=insurance_prevention PARAM=sigma GRID=0.01,0.1,0.2,0.4,0.6,0.8,1 OUTER_STEPS=2500000 \
python -u logic/experiment.py
```

Submit with `sbatch name_of_your_slurm_file.sbatch`. Results are logged in both SLURM's output files and the trace CSVs.

-----

### ⏱️ Handling Long Runs

If a given variable takes very long to run, split the jobs such that you run n separate jobs for n different values of the given variable. This way you will finish all experiments quicker as they run in parallel. You can later use the _plot_traces.py_ file to combine the different traces from corresponding values into one plot.

**Example (σ = 0.1 in `imperfect_signal`)**

```bash
#!/bin/bash
#SBATCH --job-name=imperfect_sigma01
#SBATCH --time=24:00:00
#SBATCH --nodes=1
#SBATCH --ntasks=4
#SBATCH --cpus-per-task=6
#SBATCH --mem=48G
#SBATCH --partition=gpu
#SBATCH --gres=gpu:t4:4
#SBATCH --output=/scratch/user/aaryabookseller/Research/archive/script_outputs/imperfect_signal/sigma/s01_%j.out
#SBATCH --error=/scratch/user/aaryabookseller/Research/archive/script_outputs/imperfect_signal/sigma/s01_%j.err
#SBATCH --chdir=/scratch/user/aaryabookseller/Research/archive

set -euo pipefail

echo "[$(date)] SLURM job $SLURM_JOB_ID on $(hostname)"
module load GCC/13.2.0
module load texlive/20230313
module load CUDA/12.1
source activate_venv simclr_venv

export OMP_NUM_THREADS=${SLURM_CPUS_PER_TASK}

SETTINGS=imperfect_signal \
PARAM=sigma \
GRID=0.1 \
OUTER_STEPS=1000000 \
python -u logic/experiment.py
```

**Example of partial traces**

```bash
archive/csv/imperfect_signal/sigma_rel/sigma=0p1/trace.csv
archive/csv/imperfect_signal/sigma_rel/sigma=00p1/trace.csv
...
```

-----

## 📊 Postprocessing and Analysis

Use `utils/plot_traces.py` to merge traces and visualize results.

**Example (accumulate traces for σ = 0.1)**

```bash
python -u utils/plot_traces.py \
  --setting insurance_prevention \
  --param sigma \
  --grid 0.01,0.1,0.2,0.4,0.6,0.8,1
```

This script merges partial traces, produces a consolidated trace, and generates plots for utilities, transfers, actions, and errors. Plots are saved in the directory specified in _plot_traces.py_

-----

## ✅ Recommended Workflow

1.  **Prototype locally** with small iterations (5k–20k).
2.  **Scale up on SLURM** for large sweeps (100k–1M+).
3.  **Split jobs** if runtime limits are exceeded.
4.  **Accumulate traces** using `plot_traces.py`.
5.  **Inspect results** in the CSVs (`archive/csv/`) and plots (`archive/plots/`).

-----

## 🔒 Reproducibility

All experiment runs are **deterministic** given fixed random seeds, ensuring results are reproducible. Each CSV contains **full metadata** (e.g., `param`, `setting`) for comprehensive documentation. The entire set of results can be re-generated by simply rerunning experiments with the same parameters.

-----

## 📂 Repository Layout

```text
.
├── logic/              # Core scripts for running experiments
│   ├── experiment.py   # Main script to run experiments
│   ├── optimization.py # Optimization algorithm
│   ├── settings/       # Experiment-specific configuration files
│   │   ├── cara_logistic_signal.py
│   │   ├── cara_poisson_meanexp.py
│   │   ├── crra_logistic_signal.py
│   │   ├── hara_normal.py
│   │   ├── imperfect_signal.py
│   │   ├── insurance_prevention.py
│   │   ├── logistic_signal.py
│   │   ├── logisticU_laplace.py
│   │   ├── milgrom_holstrom.py
│   │   ├── multitask_separable.py
│   │   ├── relative_performance.py
│   │   ├── sqrtw_logistic_signal.py
│   │   └── two_signals.py
├── metrics/            # Scripts for evaluating results
│   └── optimal_solution.py # Defines optimal solutions for comparison
├── plots/              # Generated plots and visualizations
└── utils/              # Helper scripts for data processing and plotting
    ├── combine_csvs.py
    ├── csv_utils.py
    ├── plotting.py
    └── plot_traces.py
```