{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "2aa5c5a5",
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "from scipy.special import factorial\n",
    "import scipy.optimize as opt\n",
    "from scipy.optimize import minimize\n",
    "from scipy.stats import poisson\n",
    "from scipy.optimize import differential_evolution\n",
    "import pandas as pd\n",
    "import os"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "cf0618d0",
   "metadata": {},
   "outputs": [],
   "source": [
    "def level_k_choice_probabilities(expected_utilities, gamma, k):\n",
    "    lambda_k = gamma * k  \n",
    "    util = np.max(lambda_k * expected_utilities)  \n",
    "    exp_utilities = np.exp(lambda_k * expected_utilities - util)\n",
    "    return exp_utilities / np.sum(exp_utilities)  \n",
    "\n",
    " \n",
    "def calculate_expected_utilities(payoff_matrix, p_k_1): \n",
    "    return np.dot(payoff_matrix, p_k_1)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "1ca2fbe3",
   "metadata": {},
   "outputs": [],
   "source": [
    "def poisson_weights(tau, level):\n",
    "    levels = np.arange(0, level + 1)\n",
    "    weights = poisson.pmf(levels, tau)\n",
    "    return weights\n",
    "\n",
    "def aggregate_choice_probabilities(payoff_matrix, tau, gamma, level):\n",
    "    num_strategies = payoff_matrix.shape[0]  \n",
    "    overall_probs = np.zeros(num_strategies)  \n",
    "\n",
    "    weights = poisson_weights(tau, level)\n",
    "    level_probs = np.ones(num_strategies) / num_strategies  \n",
    "    overall_probs = weights[0] * level_probs\n",
    "    \n",
    "    for k in range(1, level + 1):\n",
    "        expected_utilities = calculate_expected_utilities(payoff_matrix, level_probs)\n",
    "        level_probs = level_k_choice_probabilities(expected_utilities, gamma, k)\n",
    "        overall_probs += weights[k] * level_probs\n",
    "        \n",
    "    return overall_probs"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "b8f0cc8e",
   "metadata": {},
   "outputs": [],
   "source": [
    "def log_likelihood(params, payoff_matrices, observed_probs_list, level):\n",
    "    tau, gamma = params\n",
    "    log_likelihood_value = 0\n",
    "    for payoff_matrix, observed_probs in zip(payoff_matrices, observed_probs_list):\n",
    "        predicted_probs = aggregate_choice_probabilities(payoff_matrix, tau, gamma, level)\n",
    "        predicted_probs = np.maximum(predicted_probs, 1e-8)\n",
    "        log_likelihood_value += np.sum(observed_probs * np.log(predicted_probs))\n",
    "               \n",
    "    if np.isnan(log_likelihood_value):\n",
    "        return 1e10 \n",
    "\n",
    "    return -log_likelihood_value\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "92a46685",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Define games\n",
    "all_games = {\n",
    "    \"Competitive-Base\": {\n",
    "        \"payoff_matrices\": [\n",
    "            np.array([[10, 0, -5], [-10, 5, 8], [0, 5, -5]]),\n",
    "            np.array([[-10, 5, 8], [10, 0, -5], [0, -5, 5]])\n",
    "        ],\n",
    "\n",
    "    }\n",
    "#and others\n",
    "}\n",
    "init = [(), (), ()]\n",
    "results = []"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "3d78182a",
   "metadata": {},
   "outputs": [],
   "source": [
    "for game_name, game_data in all_games.items():\n",
    "    payoff_matrices = game_data[\"payoff_matrices\"]\n",
    "    for model_name, observed_probs_list in game_data[\"observed_probs\"].items():\n",
    "        best_result = None\n",
    "        best_log_likelihood = float(\"inf\")\n",
    "        for guess in init:\n",
    "            level = 10\n",
    "            result = minimize(\n",
    "                log_likelihood,\n",
    "                guess,\n",
    "                args=(payoff_matrices, observed_probs_list, level),\n",
    "                method='L-BFGS-B',\n",
    "                bounds=[(None, level), (1e-8, None)],\n",
    "                options={'disp': False}\n",
    "            )\n",
    "\n",
    "            if result.success and result.fun < best_log_likelihood:\n",
    "                best_result = result\n",
    "                best_log_likelihood = result.fun\n",
    "\n",
    "        results.append({\n",
    "            \"Game\": game_name,\n",
    "            \"Model\": model_name,\n",
    "            \"Tau\": best_result.x[0] if best_result else None,\n",
    "            \"Gamma\": best_result.x[1] if best_result else None,\n",
    "            \"Log-Likelihood\": -best_log_likelihood if best_result else None\n",
    "        })\n",
    "\n",
    "results_df = pd.DataFrame(results)\n",
    "results_df.to_csv(\"results.csv\", index=False)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "15045ae7",
   "metadata": {},
   "outputs": [],
   "source": [
    "def estimate_parameters_for_game(game_name, payoff_matrices, csv_path, level=10):\n",
    "    \n",
    "    if not os.path.exists(csv_path):\n",
    "        print(f\"[WARNING] CSV for game '{game_name}' not found at '{csv_path}'. Skipping.\")\n",
    "        return None\n",
    "    \n",
    "    data = pd.read_csv(csv_path)\n",
    "    data.rename(columns={data.columns[0]: \"Demographic ID\"}, inplace=True)\n",
    "\n",
    "    data.iloc[:, 1:] = data.iloc[:, 1:] / 100.0\n",
    "    m = payoff_matrices[0].shape[0]  \n",
    "    n = payoff_matrices[1].shape[1]     \n",
    "    results = []\n",
    "    for index, row in data.iterrows():\n",
    "        demographic_id = row.iloc[0]  \n",
    "        observed_probs_player1 = row.iloc[1 : 1 + m].to_numpy()\n",
    "        observed_probs_player2 = row.iloc[1 + m : 1 + m + n].to_numpy()       \n",
    "        observed_probs_list = [observed_probs_player1, observed_probs_player2]\n",
    "        \n",
    "        best_result = None\n",
    "        best_log_likelihood = float(\"inf\")\n",
    "        \n",
    "        for guess in init:\n",
    "            result = minimize(\n",
    "                log_likelihood,\n",
    "                guess,  \n",
    "                args=(payoff_matrices, observed_probs_list, level),\n",
    "                method='L-BFGS-B',\n",
    "                bounds=[(None, level), (1e-8, None)],  \n",
    "                options={'disp': False}\n",
    "            )\n",
    "            if result.success and result.fun < best_log_likelihood:\n",
    "                best_result = result\n",
    "                best_log_likelihood = result.fun\n",
    "\n",
    "        if best_result is not None:\n",
    "            results.append({\n",
    "                \"Model\": demographic_id,\n",
    "                \"Tau\": best_result.x[0],\n",
    "                \"Gamma\": best_result.x[1],\n",
    "                \"Log-Likelihood\": -best_log_likelihood\n",
    "            })\n",
    "        else:\n",
    "            results.append({\n",
    "                \"Model\": demographic_id,\n",
    "                \"Tau\": None,\n",
    "                \"Gamma\": None,\n",
    "                \"Log-Likelihood\": None\n",
    "            })\n",
    "    \n",
    "    results_df = pd.DataFrame(results)\n",
    "    return results_df\n",
    "def main_estimation_pipeline():\n",
    "    all_results = []  \n",
    "    \n",
    "    for game_name, game_info in all_games.items():\n",
    "        csv_file = f\"sr/{game_name}_CoT.csv\"\n",
    "        \n",
    "        payoff_matrices = game_info[\"payoff_matrices\"]\n",
    "        \n",
    "        print(f\"Running estimation for: {game_name}\")\n",
    "        results_df = estimate_parameters_for_game(\n",
    "            game_name=game_name,\n",
    "            payoff_matrices=payoff_matrices,\n",
    "            csv_path=csv_file,\n",
    "            level=10\n",
    "        )\n",
    "        \n",
    "        if results_df is not None:\n",
    "            out_csv = f\"{game_name}_estimations.csv\"\n",
    "            results_df.to_csv(out_csv, index=False)\n",
    "            print(f\"Results saved -> {out_csv}\")\n",
    "            \n",
    "            # Store results for display\n",
    "            all_results.append((game_name, results_df))\n",
    "            \n",
    "        else:\n",
    "            print(f\"Skipping {game_name} (no CSV or no valid results).\")\n",
    "        \n",
    "        print(\"-\" * 50)\n",
    "    print(\"\\nSummary of Estimation Results:\")\n",
    "    for game_name, results_df in all_results:\n",
    "        print(f\"Game: {game_name}\")\n",
    "        print(results_df)\n",
    "        print(\"-\" * 50)\n",
    "\n",
    "if __name__ == \"__main__\":\n",
    "    main_estimation_pipeline()\n"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.9.13"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
