{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "import random\n",
    "import math\n",
    "import ray\n",
    "from scipy.optimize import minimize\n",
    "import cvxpy as cp\n",
    "from scipy.stats import norm\n",
    "from scipy.integrate import quad"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "ray.init()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "ray.shutdown()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "class Mechanisms():\n",
    "    def __init__(self, K, N_Exp, N_c):\n",
    "        self.K = K\n",
    "        # self.c = np.random.uniform(0, 1, K)\n",
    "        self.mu = np.random.uniform(0, 0.5, (K, K))\n",
    "        self.gamma = np.random.uniform(0.8, 0.9, (K, K))\n",
    "        self.d = np.random.uniform(2, 5, K)\n",
    "        self.rho = np.random.uniform(1, 1.5, K)\n",
    "        self.a = np.zeros(K)\n",
    "        # self.b = np.ones(K) * 10\n",
    "        self.b = np.array([10,1])\n",
    "        self.N= N_Exp\n",
    "        self.N_c = N_c\n",
    "\n",
    "# c0 normal c1 uniform\n",
    "\n",
    "    def PDF(self, x):\n",
    "        # return 1\n",
    "        return norm.pdf(x, 0, 1) / (norm.cdf(10, 0, 1) - norm.cdf(0, 0, 1))\n",
    "\n",
    "    def CDF(self, x):\n",
    "        # return x\n",
    "        return (norm.cdf(x, 0, 1) - norm.cdf(0, 0, 1)) / (norm.cdf(10, 0, 1) - norm.cdf(0, 0, 1))\n",
    "\n",
    "    def expectation(self):\n",
    "        def f(x):\n",
    "            return x * self.PDF(x)\n",
    "        mean = quad(f, 0, 10)[0]\n",
    "        # return [mean] * self.K\n",
    "        # return [0.5] * self.K\n",
    "        return [mean, 0.5]\n",
    "\n",
    "    def Solve_Y(self, v, c):        \n",
    "        def objective(y):\n",
    "            return np.dot(v, y)\n",
    "        \n",
    "        def constraint(y, i):\n",
    "            return y[i] ** (1 / self.rho[i]) + np.sum(self.mu[i, :] * y ** (self.gamma[i, :] / self.rho[i])) - self.d[i]\n",
    "        \n",
    "        y0 = np.ones_like(v)\n",
    "        constraints = [{'type': 'ineq', 'fun': lambda y, i=i: constraint(y, i)} for i in range(len(v))]\n",
    "        bounds = [(0, None) for _ in range(len(v))]\n",
    "        result = minimize(objective, y0, method='SLSQP', bounds=bounds, constraints=constraints)\n",
    "        \n",
    "        y_optimal = result.x\n",
    "        min_value = result.fun\n",
    "        return y_optimal, min_value\n",
    "\n",
    "    def Solve_P(self, y_optimal, v, c):\n",
    "        p = np.zeros(self.K)\n",
    "        for i in range(self.K):\n",
    "            p[i] = y_optimal[i] * c[i]\n",
    "            integral = []\n",
    "            for n in range(self.N):\n",
    "                tilde_c_i = np.random.uniform(c[i], self.b[i])\n",
    "                tilde_c = np.copy(c)\n",
    "                tilde_c[i] = tilde_c_i\n",
    "                tilde_v = np.copy(v)\n",
    "                if v[0]< c[0] + 0.5 * self.CDF(c[0]) / self.PDF(c[0]):\n",
    "                    tilde_v[i] = tilde_c_i\n",
    "                else:\n",
    "                    if i==0:\n",
    "                        tilde_v[i] = tilde_c_i + self.CDF(tilde_c_i) / self.PDF(tilde_c_i)\n",
    "                    else:\n",
    "                        tilde_v[i] = tilde_c_i + tilde_c_i / 1\n",
    "                integral.append((self.b[i]-c[i]) * self.Solve_Y(tilde_v, tilde_c)[0][i])\n",
    "            p[i] += np.mean(integral)\n",
    "        return p\n",
    "    \n",
    "    def Mech(self):\n",
    "        Rev1, Rev2, Rev3, Rev4, Rev5, Rev6=[], [], [], [], [], []\n",
    "        SW1, SW2, SW3, SW4, SW5, SW6=[], [], [], [], [], []\n",
    "        for _ in range(self.N_c):\n",
    "            # c = np.random.uniform(0, 1, self.K)\n",
    "\n",
    "            c = []\n",
    "            while len(c) < 1:\n",
    "                sample = np.random.normal(0, 1)  \n",
    "                if 0 <= sample <= self.b[0]:           \n",
    "                    c.append(sample)\n",
    "            c.append(np.random.uniform(0, 1))\n",
    "            c = np.array(c)\n",
    "\n",
    "            v = [0 for i in range(self.K)]\n",
    "            for i in range(self.K):\n",
    "                if i==0:\n",
    "                    v[i] = c[i] + self.CDF(c[i]) / self.PDF(c[i])\n",
    "                else:\n",
    "                    v[i] = c[i] + c[i] / 1\n",
    "            y_optimal1, _ = self.Solve_Y(v, c)\n",
    "            p1 = self.Solve_P(y_optimal1, v, c)\n",
    "            Rev1.append(-sum(p1))\n",
    "            SW1.append(- np.dot(c, y_optimal1))\n",
    "\n",
    "            y_optimal2, _ = self.Solve_Y(c, c)\n",
    "            p2 = self.Solve_P(y_optimal2, c, c)\n",
    "            Rev2.append(-sum(p2))\n",
    "            SW2.append(-np.dot(c, y_optimal2))\n",
    "\n",
    "            SW4.append(-np.dot(c, y_optimal2))\n",
    "            y_optimal3, _ = self.Solve_Y(self.b, c)\n",
    "            Rev4.append(-np.dot(self.b, y_optimal3))\n",
    "\n",
    "            Rev5.append(-np.dot(self.b, y_optimal3))\n",
    "            SW5.append(-np.dot(c, y_optimal3))\n",
    "\n",
    "            y_optimal4, _ = self.Solve_Y(self.expectation(), c)\n",
    "            Rev6.append(-np.dot(self.b, y_optimal4))\n",
    "            SW6.append(-np.dot(c, y_optimal4))\n",
    "\n",
    "        return Rev1, SW1, Rev2, SW2, Rev4, SW4, Rev5, SW5, Rev6, SW6"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "task = Mechanisms(2, N_Exp=100, N_c=1000)\n",
    "Rev1, SW1, Rev2, SW2, Rev4, SW4, Rev5, SW5, Rev6, SW6 = task.Mech()\n",
    "# c1 and c2 follow the same distribution, so M5 = M6\n",
    "print(np.mean(Rev1) - np.mean(Rev2))\n",
    "print(np.mean(Rev2) - np.mean(Rev4))\n",
    "print(np.mean(Rev4) - np.mean(Rev6))\n",
    "print(np.mean(SW2) - np.mean(SW1))\n",
    "print(np.mean(SW2) - np.mean(SW6))\n",
    "print(np.mean(SW6) - np.mean(SW5))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# for i in range(100):\n",
    "#     print(Rev1[i], Rev2[i], Rev4[i], Rev5[i], Rev6[i])\n",
    "\n",
    "print(np.mean(Rev1), np.mean(Rev2), np.mean(Rev4), np.mean(Rev5), np.mean(Rev6))\n",
    "plt.plot([np.mean(Rev1), np.mean(Rev2), np.mean(Rev4), np.mean(Rev5), np.mean(Rev6)], label='Revenue')\n",
    "plt.savefig('revenue_average.png')\n",
    "\n",
    "# here b is proportional to E[c] so Rev6 is the same as Rev5\n",
    "print(np.mean(Rev5) - np.mean(Rev6))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "regrets = [Rev1, Rev2, Rev4, Rev5, Rev6]\n",
    "\n",
    "\n",
    "algorithm_names = ['Mechanism 1', 'Mechanism 2', 'Mechanism 4', 'Mechanism 5', 'Mechanism 6']\n",
    "\n",
    "\n",
    "plt.figure()\n",
    "box = plt.boxplot(\n",
    "    regrets,\n",
    "    labels=algorithm_names,\n",
    "    patch_artist=True,  \n",
    "    showmeans=True,     \n",
    "    meanline=True,      \n",
    "    boxprops=dict(facecolor='white', color='black'),  \n",
    "    whiskerprops=dict(color='black'),                \n",
    "    capprops=dict(color='black'),                   \n",
    "    flierprops=dict(marker='o', color='black'),  \n",
    "    medianprops=dict(color='orange'),   \n",
    "    meanprops=dict(color='red')         \n",
    ")\n",
    "\n",
    "# plt.title('Box Plot with Custom Style', fontsize=14)\n",
    "plt.xlabel('Mechanisms')\n",
    "plt.ylabel('Revenue')\n",
    "\n",
    "plt.tight_layout()\n",
    "\n",
    "plt.savefig('Rev_boxplot.png')\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "regrets = [SW1, SW2, SW4, SW5, SW6]\n",
    "\n",
    "\n",
    "algorithm_names = ['Mechanism 1', 'Mechanism 2', 'Mechanism 4', 'Mechanism 5', 'Mechanism 6']\n",
    "\n",
    "plt.figure()\n",
    "box = plt.boxplot(\n",
    "    regrets,\n",
    "    labels=algorithm_names,\n",
    "    patch_artist=True,  \n",
    "    showmeans=True,     \n",
    "    meanline=True,      \n",
    "    boxprops=dict(facecolor='white', color='black'), \n",
    "    whiskerprops=dict(color='black'),               \n",
    "    capprops=dict(color='black'),                    \n",
    "    flierprops=dict(marker='o', color='black'),  \n",
    "    medianprops=dict(color='orange'),  \n",
    "    meanprops=dict(color='red')         \n",
    ")\n",
    "\n",
    "\n",
    "# plt.title('Box Plot with Custom Style', fontsize=14)\n",
    "plt.xlabel('Mechanisms')\n",
    "plt.ylabel('Social Welfare')\n",
    "\n",
    "plt.tight_layout()\n",
    "\n",
    "plt.savefig('SW_boxplot.png')\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "plt.figure()\n",
    "plt.plot(Rev1[:100], label='Rev1')\n",
    "plt.plot(Rev2[:100], label='Rev2')\n",
    "plt.plot(Rev4[:100], label='Rev4 and Rev5')\n",
    "# plt.plot(Rev5[:10], label='Rev5')\n",
    "plt.plot(Rev6[:100], label='Rev6')\n",
    "plt.xlabel('Trials')\n",
    "plt.ylabel('Revenue')\n",
    "plt.legend()\n",
    "plt.savefig('revenue_all.png')\n",
    "plt.show()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "plt.figure()\n",
    "plt.plot(SW1[:100], label='SW1')\n",
    "plt.plot(SW2[:100], label='SW2')\n",
    "# plt.plot(SW4[:10], label='SW4')\n",
    "# plt.plot(SW5[:100], label='SW5')\n",
    "# plt.plot(SW6[:100], label='SW6')\n",
    "plt.xlabel('Trials')\n",
    "plt.ylabel('Social Welfare')\n",
    "plt.legend()\n",
    "plt.savefig('SW.png')\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "plt.plot([np.mean(SW1), np.mean(SW2), np.mean(SW4), np.mean(SW5), np.mean(SW6)], label='SW')\n",
    "plt.savefig('SW_average.png')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "print(np.mean(SW1), np.mean(SW2), np.mean(SW4), np.mean(SW5), np.mean(SW6))\n",
    "# similarly, c+F/f=2c\n",
    "print(np.mean(SW2) - np.mean(SW1))\n",
    "print(np.mean(SW6) - np.mean(SW5))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "print(np.var(Rev1), np.var(Rev2))\n",
    "print(np.var(SW1), np.var(SW2))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "Rev4[:190]"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "base",
   "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.11.7"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
