{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "cd78aaca",
   "metadata": {},
   "source": [
    "# Supplementary Material - Code"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "c4f2d28b",
   "metadata": {},
   "source": [
    "# DU-Shapley: A Shapley Value Proxy for Efficient Dataset Valuation"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "403f4dc7",
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "import math\n",
    "import random\n",
    "import scipy.special\n",
    "\n",
    "\n",
    "plt.rcParams.update({\n",
    "    \"text.usetex\": True,\n",
    "    \"font.family\": \"sans-serif\",\n",
    "    \"font.sans-serif\": \"Helvetica\",\n",
    "})\n",
    "\n",
    "from matplotlib import rc\n",
    "rc('font',**{'family':'sans-serif','sans-serif':['Helvetica']})\n",
    "rc('text', usetex=True)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "59daed1a",
   "metadata": {},
   "source": [
    "# Distribution of q(S)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "94aedce8",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAiYAAAGnCAYAAACDypymAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjcuMSwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/bCgiHAAAACXBIWXMAAA9hAAAPYQGoP6dpAAAkl0lEQVR4nO3dz3IaV+L28UeTuDRTlUhtnNkk8VSltf2tEKwnVYY7gPgKBPss6GLnrHjhDsBXYNF3QKcqvzWhV7PlZOF4NTZqyLxvjcqV4V04MEb8UbcAcUDfTxU1pjk0pzuM+uH866PxeDwWAACABf606woAAABMEEwAAIA1CCYAAMAaBBMAAGANggkAALAGwQQAAFiDYAIAAKxBMAEAANYgmAAAAGt8mqSwMUa+78t1XRljVCqV5DjO0vJhGOri4kK9Xm9uexAEkqRut6uXL1+u3A8AAHgYEgWTYrE4DRnGGF1cXKjdbi8sOwkwYRjOvRYEgSqViiSp0Wjo2bNnc+EFAAA8PEdx75VjjJkJJpL0+PFjXV1drf6AoyN9/BFhGOrZs2fT9xljdHZ2pn6/L9d173IMAADgQMRuMQmCQKlUamZbKpVSGIZKp9OxPzCdTuvly5fT51EUTfd10/X1ta6vr6fP//Of/2gwGOjJkyc6OjqK/ZkAAGB3xuOxfvvtN3355Zf6059WD2+NHUwmAeKmwWCQqHKSVCgUpv9+9eqVcrncwjEmtVpNP/zwQ+L9AwAA+7x+/Vpff/31yjKJxpgssiywxH2v7/tLx5dUq1V9//330+fD4VB/+9vf9Pr1a52cnNz5cwEAwP0ZjUZ6+vSpPv/881vLxg4mjuPMtY4MBoO1ZtN4nqdOp7N0H8fHxzo+Pp7bfnJyQjABAGDPxBmGEXsdk1wut3B7JpOJX6OPNBoNeZ4n13UVRdFaLS8AAOAwxA4mN2fMGGOUyWSmrR1hGMoYs/C9N0OH7/tKp9PTUHJ5eck6JgAAINkYk3a7Lc/zlM1m1e12Z9YwqdVqymaz0/VJgiBQp9OZea1QKEynHX/McRyVSqV1jwUAAOy52OuY2GA0Gun09FTD4ZAxJgAA7Ikk12/ulQMAAKxBMAEAANYgmAAAAGusvcAaAGA979+/1++//77ragCxffLJJ3r06NFW9k0wAYAdGY1Gevv27cw9wYB9cXx8rC+++GLjk1EIJgCwA6PRSG/evNFnn32mL774Qo8ePeLmpNgL4/FY79+/13A41Js3byRpo+GEYAIAO/D27Vt99tln+vrrrwkk2Dt/+ctf9Pnnn+vXX3/V27dvNxpMGPwKAPfs/fv3ur6+1unpKaEEe+vo6Einp6e6vr7W+/fvN7ZfggkA3LPJQNdtDR4E7svkO7zJwdsEEwDYEVpLsO+28R0mmAAAAGsQTAAAgDUIJgCAvWWM2ek+wjDc6edv4v22YbowAFjoxYtd1yCeXdbT8zw9f/587f0YYxQEgUqlUuL3XlxcqNfr3fmzN3EM69TfRrSYAAD2TqvV0tnZmdLp9K1lG42Gzs/P9fjxYx0dHSmfz8v3/enruVxO/X5fQRAkqoPv+2uFilXH0Gg0lM/np3Ve9Dg/P1+r/rYimAAA9ooxRu12+9YWAt/3lc/n5bqu2u222u22KpWKgiBQsVic6QKp1+vyPC9RPZrN5p1bKZYdQxAEOj8/l+M48jxPvV5P/X5fzWZTjuOo3++r3+/r6upqpqXmLvW3FV05AIC9Uq/XVS6XV5bxfV8XFxf65Zdf5DiOJMl1XeVyOUVRpFarNfee58+fy/M81ev1W+sQRZEkTfed1KJjMMaoWCzqxx9/nGtF8TxPpVJJrusu3WeS+tvsaDwej3ddibhGo5FOT081HA43ftMgALgv//73v/XLL7/om2++0Z///OeFZRhjstzZ2Zn6/f7KMo8fP5brugvHf/i+L8dxlMvlZrZHUaRvvvlGV1dXt9ZhEmzu2mKy6BiMMUqlUnNhp9VqqVwu6+rqamUQSlL/TYnzXZaSXb/pygEA7A3f92ONK4miSGEYLpw1UygU5kKJ9KH1I5VKxRqrsU43zrJjcF13YfBoNptKp9O3ts4kqb/NCCYAACsYY1Qul9VqtdRoNCRprruj0+kon8/fuq9Jd8b5+XmisRe5XO7WKcDGmJVdKrcdR9xjkP4bsOIOso1Tf9sxxgQAsHNBEMjzPP3444/TloF8Pj+3RsdkHMZtKpWK0um0PM9To9FQq9VStVpVpVJZ+b6zszN1u92VZZrN5tIxLnGOI+4xTPYnaWELzyJx6m87WkwAADsVRZHy+bxevnw5010xGAzmLsiTcRhx5HI59Xo9tdttpVIpeZ6nx48fr1yQzHGcWxcsC4JgYVCIexxJjuHVq1dyHCdW91Xc+tuOYAIA2KmLiwu5rjt38Q3DcK7LYzAYrBxr0Wq1pjNmJgqFgvr9vkqlkqIoWtm147ru3Ptv1imTyax1HLcdw8eWhaBlbqv/PiCYAAB2KggCFQqFuW3S4i6MRRfeKIp0dnYmafkU3mazKWn1Eu5RFK0MDbd148Q9jjjhwRijKIoSLeJ2W/33AcEEALAzk4vvzZaRTqezcCZKKpXSYDCY20+r1ZIxZmlrxsdWXegHg8HKga0///zzwm6VJMex7BhumiyqdjPsrHJb/fcBwQQAsDOTsRY3x1z4vr+wtWRZV8W7d+8kSbVabelnNRoNFQqFlQNgoyhaemFftQR9kuOI293SarUWLpbWaDSWvn9V/fcFwQQAsDOTgZ0fd680Gg0ZYxZOqU2n0wtnnVSrVTmOI9/3dX5+Lt/3ZYxRGIbyfV/FYlHv3r1Tu91eWZ9utzvtErrp1atXS1svkhzHsmOQPgQLY4w8z1Mmk5muVDvZXi6X1e/3l3bXrKr/vmC6MABgp9rttur1ugaDwcwYiUXdMvl8fuHgVcdx9Msvv8jzvOm9cFzXleu6yufztwaSiTAM9fLly7ntk3CwqjUi7nEsOoYwDPXs2bO5lpCPx81kMhmVy+WVXTvL6r9PCCYAYKF9WZJ+E1zXnQ5MlT7cF2bZSqcftyDcfN1xnJn9JBWG4dLPvby8vHXtkbjHsegY0un02kvJr6r/PqErBwBgldumyE5WVd20Wq22dMZNu93Wd999l2h/q45jG8ewqv77hGACALDGZAn2VUu2VyoVdTqdjX7uZFbNoiBhjJHjOIlaIm47jk0fw6r67xuCCQDACp7n6dmzZ5I+3OvG9/2lZZvNZuxl3eMol8tLx6Gsmo2zSNzj2OQxrKr/vjkaj8fjXVciriS3TQYAW8W9VTxWC8NQxphE63wsMplGvGxga7FY3NpFfxPHcFv9tynudznJ9ZtgAgD3jGCCQ7GNYEJXDgAAsAbBBAAAWINgAgAArEEwAQAA1iCYAAAAaxBMAACANQgmAADAGgQTAABgDYIJAACwBsEEAABY49NdVwAAsMA/X+y6BvH89cWua7C3jDE7ub+NbXW4iRYTAADumed5iqJo19WQMUatVmvX1ZhBMAEA7FS5XNb5+bmOjo50fn6ucrmsMAynr0dRpGKxqMePH+vx48cqFov3flGPomimno8fP1a5XFYQBAvreXR0pGKxOHMcE61WS2dnZ0qn03Of4XmeGo2GGo2GyuXyRo4zDEOdn58vfC2Xy6nf788cx65xd2EAuGex7sj6wLpyGo2GPM9Tv99f2rVQLBZVrVbnLuj3aXKRr1QqqtfrC8v4vq9ut7vwdWOMyuWyOp3O3Gvn5+d6+fLl9PiMMcrn8+r1enIcJ1E9JyFHkn7++WeFYahVl/vz83P1er1EnyFxd2EAwIHqdDpyHGfleIdUKrXTUCJpGhCePHmytEw6nVY2m134Wr1eV7lcnts+6U75+Phc11U6nVatVrtTPZvNpprNpp4/f35r+efPn0+DzK4RTAAAOxcEgXK53NLXoyhK3GpgoyAIVCgU5ra3221lMpm57dlsVr7vb71epVLJmrEmiYKJMUaNRkO+76vRaNza97WsXyvpfgAAh2syvmHVL/sgCJTP5++rSlvh+/7SFp8gCHR2dja33XVdGWO2fp10HEepVMqKsSaJpgsXi8VpH5QxRhcXF2q32wvL+r4v13UXDvxJsh8AwGGbjLdY1WKybMzGPul0OgvD1arQMWklMsZsvRsrl8spDMOV/x3uQ+wWE2PMzHPXdVcmq0KhsPAkJt0PAOCwBUEg13VXdtUkbTH4eIbM2dnZrd0hjUYj0f7vYtmaIYPBQJJWHv+kzDadnZ2p2+1u/XNuEzuYBEGgVCo1sy2VSi1sEbmP/QAADsNtv9KjKFrYzbHK+fm5stmsfvzxR3meN30scx8XZGPM3PXvNvc51MFxnLnGg12I3ZWz7OQkTXFJ9nN9fa3r6+vp89FolOizAAB2m7SYrxo/smzA6DKTqcelUknSh5ku3333nZ49e6ZisTg3dCAMw8SB4S4Gg8HCVpHJZy+6Pk6ujfdRP9d1rRjzufasnE0dxKL91Go1nZ6eTh9Pnz7dyGcBAOwQZ3xJp9NJtGx6t9udhpIJx3GmYxvPz8+nrfRBEKhYLMaeKjsJCO/evVtaZlWrw6JrXZzZRvexbLwtM59iBxPHceZaNZalv03tp1qtajgcTh+vX79O9FkAAPs5jrP0WmKMWbpq6TKrLuLtdlvPnz+fruBaLBbVbDZjX/gndV01/CAMw4VjLFOp1NJehskKrDdFUXTr+JtNGQwGVtw3J3YwWZZmF8273tR+jo+PdXJyMvMAAByObDa7tOU9iiLV6/W51o/b3DZ7p1KpaDwe6+rqSldXV4lnodTrdQVBsLTey1avXdVVUiwWF04E6XQ6ibqx1jEJQbsWO5jcrKwxRplMZpriwjBc2nz18X+I2/YDAHg4JjM4b86KCcNQnuep2Wxu7bPvet0plUoqFAoqFosz2yf301nWLZROp5cOsp2Er4/DiTFGxpi5oHV2dpZoldZV3U4f63a7iQcZb0OidUza7bY8z1M2m1W3250ZQFSr1ZTNZlWpVCR9OLmTvsPJa5PUt2o/AABt7B40+6DX68nzPBWLxekYjvPz862GknW12235vj9TZ8dxVK/XlwaefD6/MlBMzkMYhtMxMYvuqSMp1jIbk6XvLy8vp5/vuq6KxeLCVqIwDPXy5ctb97tt3MQPAO5Z3Buf4fCcnZ3d6aZ8N7VarcRdXKuEYaharZa4oYCb+AEAsMfK5bI196T5WK1WW3hzwV0gmAAAcE8qlcrS7pm4oija6Lomk3vx7Hop+gmCCQAA96jZbM4NnE2i1WptdKZOuVy2aqwnwQQAgHvkuq6q1eqt9+9ZZjLJZBMajYaazaZVM2MTzcoBAADrS6fTW79bcBybDDmbQosJAACwBsEEAABYg2ACADuyR8tIAQtt4ztMMAGAe/bJJ59Ikt6/f7/jmgDrmXyHJ9/pTSCYAMA9e/TokY6PjzUcDmk1wd4aj8caDoc6Pj7Wo0ePNrZfZuUAwA588cUXevPmjX799Vednp7q0aNHOjo62nW1gFuNx2O9f/9ew+FQ//rXv/TVV19tdP8EEwDYgcn9Qt6+fas3b97suDZAcsfHx/rqq682fu86ggkA7MjJyYlOTk70/v17/f7777uuDhDbJ598stHum48RTABgxx49erS1P/LAvmHwKwAAsAbBBAAAWINgAgAArEEwAQAA1iCYAAAAaxBMAACANQgmAADAGgQTAABgDYIJAACwBsEEAABYg2ACAACsQTABAADWIJgAAABrEEwAAIA1CCYAAMAaBBMAAGANggkAALAGwQQAAFiDYAIAAKxBMAEAANYgmAAAAGsQTAAAgDUIJgAAwBoEEwAAYA2CCQAAsAbBBAAAWINgAgAArEEwAQAA1iCYAAAAaxBMAACANQgmAADAGgQTAABgDYIJAACwBsEEAABYg2ACAACsQTABAADWIJgAAABrfJqksDFGvu/LdV0ZY1QqleQ4TuKyxhgFQaBUKiVjjAqFglzXXfdYAADAnksUTIrFonq9nqQP4eLi4kLtdjtxWd/3ValUpmXL5bKazeadDgAAAByO2F05xpiZ567rKgiCO5V99epVkjoCAIAHInYwmXS9fCyVSikMw8RlU6mUzs/Pp106+Xz+LnUHAAAHJnYwiaJo4fbBYJC47KRL5+zsTO12W4VCYWH56+trjUajmQcAADhcicaYLLIshKwqGwSB6vW6jDEql8uStHCMSa1W0w8//LBuFQEAwJ6I3WLiOM5c68hgMFg4K2dVWWOMut2ucrmcSqWS+v2+Li8v58alSFK1WtVwOJw+Xr9+Hbe6AABgD8UOJrlcbuH2TCaTqGwYhspms9NtruuqWq0ubHk5Pj7WycnJzAMAAByu2MHk5jojxhhlMplpi0kYhtNWj1Vl0+m0ut3uzOvv3r1TOp2+S/0BAMABORqPx+O4hY0xajabymaz6na7qlar02BSLBaVzWan65OsKhsEgcIwnD7P5XKxFlgbjUY6PT3VcDik9QQAgD2R5PqdKJjsGsEEAID9k+T6zb1yAACANQgmAADAGgQTAABgDYIJAACwBsEEAABYg2ACAACsQTABAADWIJgAAABrEEwAAIA1CCYAAMAaBBMAAGANggkAALAGwQQAAFiDYAIAAKxBMAEAANYgmAAAAGsQTAAAgDUIJgAAwBoEEwAAYA2CCQAAsAbBBAAAWINgAgAArEEwAQAA1iCYAAAAaxBMAACANQgmAADAGgQTAABgDYIJAACwBsEEAABYg2ACAACsQTABAADWIJgAAABrEEwAAIA1CCYAAMAaBBMAAGANggkAALAGwQQAAFiDYAIAAKxBMAEAANYgmAAAAGsQTAAAgDUIJgAAwBoEEwAAYA2CCQAAsAbBBAAAWINgAgAArEEwAQAA1iCYAAAAaxBMAACANQgmAADAGgQTAABgjU+TFDbGyPd9ua4rY4xKpZIcx7lT2SAIZIyR67qSpFwud+eDAAAAhyFRMCkWi+r1epI+BI+Liwu12+3EZYMgULvdVrPZlDFG+Xxe/X5/neMAAAAHIHYwMcbMPHddV0EQ3KlsuVyehhbXddXpdGJXGAAAHK7YY0yCIFAqlZrZlkqlFIZhorLGGA0GAzmOozAMFUXRtDvnpuvra41Go5kHAAA4XLGDSRRFC7cPBoNEZcMwVCqVmo4/abVa8n1/YflarabT09Pp4+nTp3GrCwAA9tDas3KWhZBlZQeDgYwxyuVychxHpVJJxWJxYflqtarhcDh9vH79et3qAgAAi8UeY+I4zlzryKRLJknZjx+TspIUhqHS6fTMe46Pj3V8fBy3igAAYM/FbjFZNp03k8kkKrtsPAkAAEDsYHIzUBhjlMlkZlo8JrNxVpV1XVeZTGbaBTRZy+RmawkAAHh4Eq1j0m635Xmestmsut3uzBomtVpN2WxWlUrl1rKT187Pz9Xr9ZguDAAAJElH4/F4vOtKxDUajXR6eqrhcKiTk5NdVwcAAMSQ5PrNvXIAAIA1CCYAAMAaBBMAAGANggkAALAGwQQAAFiDYAIAAKxBMAEAANYgmAAAAGsQTAAAgDUIJgAAwBoEEwAAYI1EN/EDAAAW+OeL7e37r1vcdwwEE+BAvXixX/vd5r63WWfsN75z9iGYfGxbCXTH6RMAgH1BMMHB2Mdf8pj17f+82M6O/yl+IGA3ttnlcqAY/AoAAKxBiwnuHa0PAJLY9t+MrbTU/XPzu3woCCb77oBHZgMAHh6CCQBgc7bwY+nb//nwvz/9Y/P7hn0IJgAAbNhP/7u9fX/79+3t2wYEEyz00/9KP/1j17WwCF1mAHAvCCYf2VbC/ekfDPgEACAOpgsDAABr0GICAMAe2Vbrvi1jVwgm9+Db/3nBnHYAdmFFUliKrhwAAGANWkyAXdvaL9ft7HebA7kn61XsFWZsTW1ziqy0xXspwSoEE9y7bf5xYQEm3LdD7+8H7htdOQAAwBq0mAAx8KsY9+3Dd+7FVvbN9w42I5gAB4ouMwD7iGAC7NC2BwsCwL5hjAkAALAGLSYADt42b0q5l1OcAYsRTLAUawYAh4kuRNiMrhwAAGANWkwAJEZrGoBtocUEAABYgxaTPUdf8Sx+yWMZvhvAfqDFBAAAWINgAgAArEEwAQAA1iCYAAAAaxBMAACANQgmAADAGgQTAABgDdYxuSesNwIAwO1oMQEAANYgmAAAAGsQTAAAgDUSjTExxsj3fbmuK2OMSqWSHMdZq6zneapWq0v3AwAAHo5EwaRYLKrX60n6EDwuLi7UbrfvXDYMQzUaDVWr1bvUHQAAHJjYXTnGmJnnrusqCIK1yhpj5Lpu3CoAAIADFzuYBEGgVCo1sy2VSikMwzuV9X1fhUIhaX0BAMABi92VE0XRwu2DwSBx2SiKYo0pub6+1vX19fT5aDS69T0AAGB/rT0rZ1kIWVX28vJSuVzu1vK1Wk2np6fTx9OnT+9YSwAAsA9iBxPHceZaRwaDwcKWj1VlgyDQd999F+szq9WqhsPh9PH69eu41QUAAHsodjBZ1sKRyWQSl728vFSr1VKr1ZIxRrVabeFYlePjY52cnMw8AADA4Yo9xuTm7BljjDKZzLTFJAxDOY4j13VXlr0ZWsrlssrlMrNzAABAsjEm7XZbnufJ9301m82ZdUlqtZp8349VVvow3qTRaEiS6vX6whYTAADwsByNx+PxrisR12g00unpqYbD4Va6dX7yX2x8nwAA7INv//7HP/76YuP7TnL95l45AADAGgQTAABgDYIJAACwBsEEAABYg2ACAACsQTABAADWIJgAAABrEEwAAIA1CCYAAMAaBBMAAGANggkAALAGwQQAAFiDYAIAAKxBMAEAANYgmAAAAGsQTAAAgDUIJgAAwBoEEwAAYA2CCQAAsAbBBAAAWINgAgAArEEwAQAA1iCYAAAAaxBMAACANQgmAADAGgQTAABgDYIJAACwBsEEAABYg2ACAACsQTABAADWIJgAAABrEEwAAIA1CCYAAMAaBBMAAGANggkAALAGwQQAAFiDYAIAAKxBMAEAANYgmAAAAGsQTAAAgDUIJgAAwBoEEwAAYA2CCQAAsAbBBAAAWINgAgAArEEwAQAA1iCYAAAAaxBMAACANQgmAADAGgQTAABgDYIJAACwxqdJChtj5Pu+XNeVMUalUkmO4yQuG4ahgiCQJHW7Xb18+XLpfgAAwMORKJgUi0X1ej1JH4LHxcWF2u124rJBEKhSqUiSGo2Gnj17Ni0LAAAerthdOcaYmeeu605bPZKUDcNQtVpt+lqhUFAYhnPvAQAAD0/sYBIEgVKp1My2VCqlMAwTlU2n03r58uV0exRF09dvur6+1mg0mnkAAIDDFTuYTALETYPBIHHZQqEw3fbq1SvlcrmFY0xqtZpOT0+nj6dPn8atLgAA2ENrz8pZFkLilI2iSL7vLx2nUq1WNRwOp4/Xr1+vUVMAAGC72INfHceZax0ZDAYLWzrilvU8T51OZ+mMnOPjYx0fH8etIgAA2HOxW0xyudzC7ZlM5k5lG42GPM+T67qKoihRywsAADhMsYOJ67ozz40xymQyM2uTTGbW3FbW932l0+lpKLm8vGQdEwAAkGwdk3a7Lc/zlM1m1e12Z8aG1Go1ZbPZ6foky8oaY1QsFmf26ziOSqXSuscCAAD23NF4PB7vuhJxjUYjnZ6eajgc6uTkZOP7/8l/sfF9AgCwD779+x//+OuLje87yfWbe+UAAABrEEwAAIA1CCYAAMAaBBMAAGANggkAALAGwQQAAFiDYAIAAKxBMAEAANYgmAAAAGsQTAAAgDUIJgAAwBoEEwAAYA2CCQAAsAbBBAAAWINgAgAArEEwAQAA1iCYAAAAaxBMAACANQgmAADAGgQTAABgDYIJAACwBsEEAABYg2ACAACsQTABAADWIJgAAABrEEwAAIA1CCYAAMAaBBMAAGANggkAALAGwQQAAFiDYAIAAKxBMAEAANYgmAAAAGsQTAAAgDUIJgAAwBoEEwAAYA2CCQAAsAbBBAAAWINgAgAArEEwAQAA1iCYAAAAaxBMAACANQgmAADAGgQTAABgDYIJAACwBsEEAABYg2ACAACsQTABAADWIJgAAABrEEwAAIA1Pk1S2Bgj3/fluq6MMSqVSnIcJ3HZJPsBAAAPR6JgUiwW1ev1JH0IFxcXF2q324nLJtkPAAB4OGJ35RhjZp67rqsgCBKXTbIfAADwsMQOJkEQKJVKzWxLpVIKwzBR2ST7AQAAD0vsrpwoihZuHwwGicom2c/19bWur6+nz4fDoSRpNBqtruwd/d//d317IQAADtDotz/+cbz5a+zkuj0ej28tm2iMySLLgkbSsoteq9Vq+uGHH+a2P336NPZnAgCAJP7P1vb822+/6fT0dGWZ2MHEcZy5Vo3BYLBwNs2qskn2U61W9f3330+f/+c//9FgMNCTJ090dHQUt+qxjEYjPX36VK9fv9bJyclG943/4jzfD87z/eA83x/O9f3Y1nkej8f67bff9OWXX95aNnYwyeVyajabc9szmUyisq7rxt7P8fGxjo+PZ7Zte1rxyckJX/p7wHm+H5zn+8F5vj+c6/uxjfN8W0vJROxg4rruzHNjjDKZzDQohGEox3Hkuu7KsjeDxc39AACAhyvRGJN2uy3P85TNZtXtdmfWHqnVaspms6pUKreWXfUaAAB4uI7GcYbIPgDX19eq1WqqVqtz3UfYHM7z/eA83w/O8/3hXN8PG84zwQQAAFiDm/gBAABrEEwAAIA1CCYAAMAaBBMAAGCNtZekt10URarVanry5Ikkqd/vq16vx143Zd33PxTrnidjjJrNpqIokjFGjuOoXq/PrYnz0G36++j7vowx02n++GBT59nzPD158kTv3r2T9GE1a/52zNrk3+h3794piiKVy2Wl0+kt1no/hWGoi4sL9Xq9RO+79+vg+MCl0+lxr9ebPu/3+2PXdcdXV1f38v6HYp3z1O/3x6VSaWZbpVIZSxr3+/1NV3WvbfL7eHV1NXYcZ1yv1zdYw8Ow7nnu9/vjXC43s49KpTIuFAqbrureW/dc3/zbMR6P5879Q3Z1dTUulUrjUqk0TqfT47tc9u/7OnjQXTmtVkuSZpKz67pKp9Oq1Wpbf/9Dse55qtfrqtfrc9scx1GxWNxsZffYpr+Pk/1h1ibOc7FYnPvVHoahUqnUZiu75zbxNzqfz89tr9frC2998hA5jqNms6lms6nnz58nfv8uroMHHUza7fbCe/Bks1n5vr/19z8U656ny8tLXVxczG3P5XIKw3AjdTwEm/w+BkGgXC63qaodlHXP86R7rFAozGzvdDpcLG9Y91z3+311Op1tVA1/2MV18KCDSRAEOjs7m9vuuq6MMYqiaKvvfyjWPU+LvvSYt8nvYxiG9MEvse55bjabfKdjWvdcZ7NZtVoteZ43s71Wq6lcLm+yqg/WLq6DBxtMVp2syYAdY8zW3v9QbOI8dTqdhfdLCsOQwa9/2OT3sdFoMNh1iU2c559//lmu6yoIAjUaDTUaDZXLZVr/btjEuS4UCsrlcmo0Gjo7O1MQBCqXywx+3ZBdXQcPdlbOYDCQpJWjhidltvH+h2Jb5ykIAhljaKb9w6bOMy0lq23iPE9mlkVRNA2AURTpm2++0Y8//sj5/8OmvtOdTkee56nRaCifz6tUKtFNuSG7ug4ebIvJKus2PdGFE88656lcLqtSqfAHJoYk5/nVq1ec0zuKc54nvx5vjjFxHEffffcdg7ljSvKdbrVaiqJIvV5PuVxOrVZL5+fntGhv2TavgwcbTCaj3xedvEnCWzVCft33PxTbOE/FYlG5XG5ups5Dtonz3Gq1VK1WN163Q7KpvxuLWkXOzs4Ym/aRTX2n+/2+ms2m0un0dIBxGIaEwA3Y1XXwYINJnIVfVo1fWPf9D8Wmz1Oj0ZDrusxeuGHd82yMUSqVYnGvW2zz7wZj02Zt4m+H53lzP2BKpZJ6vZ7CMORcr2lX18GDHWMifZhu2u/357ZHUSTXdW896eu+/6HY1HmaTD37+A8NYyL+a53zPBmvc3PMThRFevXqlfr9vvL5/NwU14do3e9zOp1e+Atzso0fNP+1zrmOomjpr/V0Oq1cLkfr1Abs4jp4sC0m0ocugSAI5rZ3Op1Yf4DXff9DsYnzNPl1c3O2yKL9PlTrnOdcLjddZOnjhyQ9f/5czWaT7/Qf1v0+l8tl/fzzz3Pbu92u0uk0P2g+ss65dhxHg8FgafgYDAb8qNmAnVwHt7KerEVc1x13Op3p88lSuovKVSqVO7//oVvnPPf7/XE6nR7X6/WZR6VSGedyua3XfZ+s+32+SVKscg/Nuuc5nU6Pm83m9Hmv1xs7jsMy6Qusc67b7fbCvxH1en3cbrc3X9k9N7nVxzK2XAcPuitHknq9njzPUxiGchxHvV4v0RTUdd//UKxznvL5vIwxC9d54Ff8rE19H8vl8rT/fTKrYTLoGJv7u/HxIl+9Xo9unAXWOdeFQkGu66pcLk9boriJ37zJ9/Dy8lLSh7+5ruvG/v/8fV8Hj8bj8XhrewcAAEjgoMeYAACA/UIwAQAA1iCYAAAAaxBMAACANQgmAADAGgQTAABgDYIJAACwBsEEAABYg2ACAACsQTABAADWIJgAAABrEEwAAIA1CCYAAMAa/x+USq2SrnQakAAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "I = [10,50,500] # number of players\n",
    "# I = [100] # number of players\n",
    "T = 10**4 # number of samples\n",
    "\n",
    "mu = 100\n",
    "sigma = 20\n",
    "\n",
    "for k in range(len(I)):\n",
    "\n",
    "    i = I[k]\n",
    "    \n",
    "    vector_a = np.array([random.uniform(1,10) for l in range(i)])\n",
    "#     vector_a = np.array([np.random.normal(5,0.1) for l in range(i)])\n",
    "\n",
    "    n_gauss = np.array([int(np.floor(random.randint(1,100))) for l in range(i)])\n",
    "#     n_gauss = np.floor(np.random.normal(mu,sigma,size = i))\n",
    "    \n",
    "    X_gauss = np.zeros(T)\n",
    "    U_gauss= np.zeros(T)\n",
    "    \n",
    "    X_num = np.zeros(T)\n",
    "    X_num_sqrt = np.zeros(T)\n",
    "    X_dem = np.zeros(T)\n",
    "\n",
    "    for t in range(T):\n",
    "        K = np.random.randint(0,i)\n",
    "        perm = np.random.permutation(range(i-1))\n",
    "        U_gauss[t] = np.random.uniform(0,1)\n",
    "        \n",
    "        if K != 0:\n",
    "            X_gauss[t] = np.floor(np.sum(vector_a[perm[0:K]]*n_gauss[perm[0:K]])**2 \\\n",
    "                / (np.sum((vector_a[perm[0:K]]**2)*n_gauss[perm[0:K]])))\n",
    "            X_num_sqrt[t] = (np.sum(vector_a[perm[0:K]]*n_gauss[perm[0:K]]))\n",
    "            X_num[t] = X_num_sqrt[t]**2\n",
    "            X_dem[t] = np.sum((vector_a[perm[0:K]]**2)*n_gauss[perm[0:K]])\n",
    "        \n",
    "        if K == 0:\n",
    "            X_gauss[t] = n_gauss[0]\n",
    "            X_num_sqrt[t] = (vector_a[0]*n_gauss[0])\n",
    "            X_num[t] = X_num_sqrt[t]**2\n",
    "            X_dem[t] = (vector_a[0]**2)*n_gauss[0]\n",
    "            \n",
    "    X_gauss = X_gauss/(np.sum(vector_a*n_gauss)**2 / (np.sum((vector_a**2)*n_gauss)))\n",
    "    X_num = X_num/((np.sum(vector_a*n_gauss))**2)\n",
    "    X_num_sqrt = X_num_sqrt/(np.sum(vector_a*n_gauss))\n",
    "    X_dem = X_dem/(np.sum((vector_a**2)*n_gauss))\n",
    "                              \n",
    "       \n",
    "    plt.figure()\n",
    "    weights = np.ones_like(X_gauss)/float(len(X_gauss))\n",
    "    plt.hist(X_gauss,weights=weights,bins=20,alpha=0.5,label='$q(\\mathcal{S})/q(\\mathcal{I})$', color = 'blue')\n",
    "    weights = np.ones_like(U_gauss)/float(len(U_gauss))\n",
    "    plt.hist(U_gauss,weights=weights,bins=20,alpha=0.5,label='$U \\sim \\mathrm{U}(0,1)$', color = 'gold')\n",
    "    plt.ylim(0,0.12)\n",
    "    plt.xticks(size=15)\n",
    "    plt.legend(prop = { \"size\": 15 }, loc = 1)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "1afa1328",
   "metadata": {},
   "source": [
    "# Experiments Section B.2"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "318a1ae0",
   "metadata": {},
   "source": [
    "# Value function and Power set function"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "587440ac",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Worth function\n",
    "def worth(d, k_S):\n",
    "    return d / (d+1-k_S)\n",
    "\n",
    "# Powerset\n",
    "def powerset(s):\n",
    "    S = []\n",
    "    x = len(s)\n",
    "    for i in range(1 << x):\n",
    "        S.append([s[j] for j in range(x) if (i & (1 << j))])\n",
    "    return S\n",
    "\n",
    "def k_S(sigma,eps,data):\n",
    "    gamma = sigma/eps\n",
    "    num = np.sum(gamma*data)**2\n",
    "    dem = np.sum((gamma**2)*data)\n",
    "    return np.floor(num/dem)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "5ab4efd7",
   "metadata": {},
   "source": [
    "# Shapley value exact computation and approximations"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "a64eabc7",
   "metadata": {},
   "outputs": [],
   "source": [
    "def Shapley_exact_brut_force(d,sigma,eps,data):\n",
    "    N = len(data)\n",
    "    Players = np.arange(N).astype(int)\n",
    "    \n",
    "    SV = np.zeros(N)\n",
    "    for i in range(N):\n",
    "        print('Player',i)\n",
    "        Players_aux = np.copy(Players)\n",
    "        Players_aux = np.delete(Players_aux, i)\n",
    "        total_set = powerset(list(Players_aux))\n",
    "        SVi = np.zeros(len(total_set)-1)\n",
    "        for k in range(len(total_set)-1):\n",
    "            s = np.array(total_set[k+1])\n",
    "            s_with_i = np.append(np.copy(s),i).astype(int)\n",
    "            \n",
    "            k_S_with_i = k_S(sigma[s_with_i],eps[s_with_i],data[s_with_i])\n",
    "            k_S_no_i = k_S(sigma[s],eps[s],data[s])\n",
    "            \n",
    "            SVi[k] = (1/scipy.special.binom(np.sum(N)-1,len(s)))*(worth(d,k_S_with_i)\\\n",
    "                                                                - worth(d,k_S_no_i))\n",
    "            \n",
    "        SV[i] = (1/np.sum(N))*np.sum(SVi) + (1/np.sum(N)) * (worth(d,data[i]) - worth(d,0))\n",
    "    \n",
    "    return SV\n",
    "\n",
    "def Shapley_approx_MC(d,sigma,eps,data,T):\n",
    "    N = len(data)\n",
    "    SVi = np.zeros(N)\n",
    "    for t in range(T):\n",
    "        perm = np.random.permutation(N)\n",
    "        for i in range(N):\n",
    "            idx = np.where(perm == i)[0][0]\n",
    "            if idx != 0:\n",
    "                data_coalition_with_i = k_S(sigma[perm[0:idx+1]],eps[perm[0:idx+1]],data[perm[0:idx+1]])\n",
    "                data_coalition_no_i = k_S(sigma[perm[0:idx]],eps[perm[0:idx]],data[perm[0:idx]])\n",
    "                \n",
    "                contribution = worth(d,data_coalition_with_i) - worth(d,data_coalition_no_i) \n",
    "                \n",
    "            else:\n",
    "                contribution = worth(d,data[i]) - worth(d,0)\n",
    "            \n",
    "            SVi[i] = SVi[i] + contribution\n",
    "    \n",
    "    return SVi/T\n",
    "\n",
    "# MC-anti-Shapley\n",
    "def Shapley_approx_MC_anti(d,sigma,eps,data,T):\n",
    "    b = len(data)\n",
    "    SVi = np.zeros(b)\n",
    "    if T == 1:\n",
    "        T = 2\n",
    "    for t in range(int(T/2)):\n",
    "        perm = np.random.permutation(b)\n",
    "        perm_anti = np.flip(perm)\n",
    "        for i in range(b):\n",
    "            idx = np.where(perm==i)[0][0]\n",
    "            if idx != 0:\n",
    "                data_coalition_1_with_i = k_S(sigma[perm[0:idx+1]],eps[perm[0:idx+1]],data[perm[0:idx+1]])\n",
    "                data_coalition_1_without_i = k_S(sigma[perm[0:idx]],eps[perm[0:idx]],data[perm[0:idx]]) \n",
    "                \n",
    "                contribution_1 = worth(d,data_coalition_1_with_i) - worth(d,data_coalition_1_without_i) \n",
    "            else:\n",
    "                contribution_1 = worth(d,data[i]) - worth(d,0)\n",
    "                \n",
    "            idx = np.where(perm_anti==i)[0][0]\n",
    "            if idx != 0:\n",
    "                data_coalition_2_with_i = k_S(sigma[perm_anti[0:idx+1]],eps[perm_anti[0:idx+1]],data[perm_anti[0:idx+1]])\n",
    "                data_coalition_2_without_i = k_S(sigma[perm_anti[0:idx]],eps[perm_anti[0:idx]],data[perm_anti[0:idx]]) \n",
    "                \n",
    "                contribution_2 = worth(d,data_coalition_2_with_i) - worth(d,data_coalition_2_without_i) \n",
    "            else:\n",
    "                contribution_2 = worth(d,data[i]) - worth(d,0)\n",
    "            \n",
    "            SVi[i] = SVi[i] + contribution_1\n",
    "            SVi[i] = SVi[i] + contribution_2\n",
    "        \n",
    "    return SVi/T\n",
    "\n",
    "\n",
    "# Owen-Shapley\n",
    "def Shapley_approx_Owen(d,sigma,eps,data,T):\n",
    "    b = len(data)\n",
    "    SVi = np.zeros(b)\n",
    "    for t in range(T):\n",
    "        for i in range(b):\n",
    "            SVii = 0\n",
    "            q_proba = np.random.uniform()\n",
    "\n",
    "            idx = np.random.binomial(1, q_proba, b-1)\n",
    "            B = idx * list(set(range(b)) - set([i]))\n",
    "            \n",
    "            B_no_i = B[B != 0]\n",
    "            \n",
    "            if len(B_no_i) != 0:\n",
    "                B_with_i = np.append(np.copy(np.array(B_no_i)),i)\n",
    "\n",
    "                data_coalition_with_i = k_S(sigma[B],eps[B],data[B])\n",
    "\n",
    "                data_coalition_without_i = k_S(sigma[B_no_i],eps[B_no_i],data[B_no_i]) \n",
    "\n",
    "                contribution = worth(d,data_coalition_with_i) - worth(d,data_coalition_without_i) \n",
    "            else:\n",
    "                contribution = worth(d,data[i]) - worth(d,0)\n",
    "            \n",
    "            SVi[i] += contribution\n",
    "\n",
    "    return SVi/T\n",
    "\n",
    "def projection_matrix(b):\n",
    "    U =  np.ones([b-1,b])\n",
    "    for i in range(b-1):\n",
    "        U[i,i+1] = -i - 1\n",
    "        U[i,i+2:] = 0\n",
    "        U[i,:] = U[i,:]/np.linalg.norm(U[i,:],2)\n",
    "    return U\n",
    "\n",
    "def orthogonal_permutations(samples,b):\n",
    "    U = projection_matrix(b)\n",
    "    X = np.random.normal(0,1,size = (2*samples,b-1))\n",
    "    Y = np.zeros([2*samples,b])\n",
    "    \n",
    "    for i in range(samples):\n",
    "        for j in range(i+1):\n",
    "            X_tran = np.transpose(X[i,:])\n",
    "            X[i,:] = X[i,:] - (X[j,:]@X_tran)*X[j,:]\n",
    "            \n",
    "        X[i,:] = X[i,:]/np.linalg.norm(X[i,:],2)\n",
    "        Y[2*i,:] = np.argsort(np.transpose(U)@X[i,:])\n",
    "        Y[2*i+1,:] = np.argsort(np.transpose(U)@(-X[i,:]))\n",
    "    \n",
    "    return Y\n",
    "\n",
    "def Shapley_approx_Ortho(d,sigma,eps,data,T):\n",
    "    b = len(data)\n",
    "    SVi = np.zeros(b)\n",
    "    Y = orthogonal_permutations(T,b)\n",
    "    for t in range(T):\n",
    "        perm = Y[t].astype(int)\n",
    "        for i in range(b):\n",
    "            idx = np.where(perm==i)[0][0]\n",
    "            \n",
    "            if idx != 0:\n",
    "                data_coalition_with_i = k_S(sigma[perm[0:idx+1]],eps[perm[0:idx+1]],data[perm[0:idx+1]])\n",
    "                data_coalition_no_i = k_S(sigma[perm[0:idx]],eps[perm[0:idx]],data[perm[0:idx]])\n",
    "                \n",
    "                contribution = worth(d,data_coalition_with_i) - worth(d,data_coalition_no_i) \n",
    "                \n",
    "            else:\n",
    "                contribution = worth(d,data[i]) - worth(d,0)\n",
    "            \n",
    "            SVi[i] = SVi[i] + contribution\n",
    "            \n",
    "    return SVi/T\n",
    "\n",
    "def Shapley_approx_DU(d,sigma,eps,data):\n",
    "    N = len(data)\n",
    "    SV = np.zeros(N)\n",
    "    Players = np.arange(N).astype(int)\n",
    "    \n",
    "    mean_all_players = np.floor(k_S(sigma,eps,data)/(N-1))\n",
    "    \n",
    "    for i in range(N):\n",
    "        Players_aux = np.copy(Players)\n",
    "        Players_aux = np.delete(Players_aux, i)\n",
    "        \n",
    "        mean_no_i = k_S(sigma[Players_aux],eps[Players_aux],data[Players_aux])\n",
    "        mean_no_i = mean_no_i/(N-1)\n",
    "        mean_no_i = np.floor(mean_no_i)\n",
    "        \n",
    "        Sum = 0\n",
    "        for k in range(1,N):\n",
    "            Sum = Sum + worth(d,k*mean_all_players) - worth(d,k*mean_no_i)\n",
    "        \n",
    "        SV[i] = (Sum + worth(d,data[i]) - worth(d,0))/N\n",
    "        \n",
    "    return SV"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "cda2193b",
   "metadata": {},
   "source": [
    "# Running examples\n",
    "\n",
    "## Exact computation and DU-Shapley approx"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "3534f3c4",
   "metadata": {},
   "outputs": [],
   "source": [
    "I = 20\n",
    "d = 10\n",
    "\n",
    "Gamma = 100\n",
    "\n",
    "data = np.sort(np.array([random.randint(20,1000) for i in range(I)]))\n",
    "sigma = np.array([random.randint(1,Gamma) for i in range(I)])\n",
    "eps = np.ones(I)\n",
    "\n",
    "print('Data',data)\n",
    "print('Sigma',sigma)\n",
    "print('Eps',eps)\n",
    "\n",
    "SV = Shapley_exact_brut_force(d,sigma,eps,data)\n",
    "DU_SV = Shapley_approx_DU(d,sigma,eps,data)\n",
    "\n",
    "print('--------------- Efficiency ---------------')\n",
    "print('w(N) - w(0)',worth(d,k_S(sigma,eps,data))-worth(d,0))\n",
    "print('Sum SV',np.sum(SV))\n",
    "print('Sum DU',np.sum(DU_SV))"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "f261e097",
   "metadata": {},
   "source": [
    "## State of the art approx"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "6e28b071",
   "metadata": {},
   "outputs": [],
   "source": [
    "Max_budget = 10*I\n",
    "Space = 5\n",
    "Budgets = np.unique(np.append(np.linspace(np.amax(np.array([1,I/10])),I,Space),np.linspace(2*I,Max_budget,Space)).astype(int))\n",
    "print(Budgets)\n",
    "\n",
    "T = len(Budgets)\n",
    "Rep = 10\n",
    "rep = 25\n",
    "\n",
    "MSE_DU = np.mean((DU_SV - SV)**2)\n",
    "\n",
    "MSE_MC = np.zeros([T,Rep])\n",
    "MSE_anti_MC = np.zeros([T,Rep])\n",
    "MSE_Owen = np.zeros([T,Rep])\n",
    "MSE_Ortho = np.zeros([T,Rep])\n",
    "\n",
    "for t in range(T):\n",
    "    print('Iteration',t+1)\n",
    "    for R in range(Rep):\n",
    "        for r in range(rep):\n",
    "            MSE_Ortho[t,R] = MSE_Ortho[t,R] + np.mean((Shapley_approx_Ortho(d,sigma,eps,data,Budgets[t]) - SV)**2)\n",
    "            MSE_MC[t,R] = MSE_MC[t,R] + np.mean((Shapley_approx_MC(d,sigma,eps,data,Budgets[t]) - SV)**2)\n",
    "            MSE_anti_MC[t,R] = MSE_anti_MC[t,R] + np.mean((Shapley_approx_MC_anti(d,sigma,eps,data,Budgets[t]) - SV)**2)\n",
    "            MSE_Owen[t,R] = MSE_Owen[t,R] + np.mean((Shapley_approx_Owen(d,sigma,eps,data,Budgets[t]) - SV)**2)                \n",
    "        MSE_Ortho[t,R] = MSE_Ortho[t,R]/rep\n",
    "        MSE_MC[t,R] = MSE_MC[t,R]/rep\n",
    "        MSE_anti_MC[t,R] = MSE_anti_MC[t,R]/rep\n",
    "        MSE_Owen[t,R] = MSE_Owen[t,R]/rep\n",
    "print('Finished')"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "9d41c90a",
   "metadata": {},
   "source": [
    "# Figures"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "0d2d10f9",
   "metadata": {},
   "outputs": [],
   "source": [
    "mse_mean_MC = np.mean(MSE_MC,axis=1)\n",
    "mse_std_MC = np.std(MSE_MC,axis=1)\n",
    "\n",
    "mse_mean_MC_anti = np.mean(MSE_anti_MC,axis=1)\n",
    "mse_std_MC_anti = np.std(MSE_anti_MC,axis=1)\n",
    "\n",
    "mse_mean_Owen = np.mean(MSE_Owen,axis=1)\n",
    "mse_std_Owen = np.std(MSE_Owen,axis=1)\n",
    "\n",
    "mse_mean_Ortho = np.mean(MSE_Ortho,axis=1)\n",
    "mse_std_Ortho = np.std(MSE_Ortho,axis=1)\n",
    "\n",
    "plt.figure()\n",
    "plt.axhline(y=np.sum(MSE_DU), color='purple', linestyle='-',label=\"DU-Shapley\")\n",
    "\n",
    "plt.plot(Budgets/I,mse_mean_MC,label=\"MC-Shapley\")\n",
    "plt.fill_between(Budgets/I, mse_mean_MC - mse_std_MC, mse_mean_MC + mse_std_MC, color='C0', alpha=0.2)\n",
    "\n",
    "plt.plot(Budgets/I,mse_mean_MC_anti,label=\"MC-Anti-Shapley\")\n",
    "plt.fill_between(Budgets/I, mse_mean_MC_anti - mse_std_MC_anti, mse_mean_MC_anti + mse_std_MC_anti, color='C1', alpha=0.2)\n",
    "\n",
    "plt.plot(Budgets/I,mse_mean_Owen,label=\"Owen-Shapley\")\n",
    "plt.fill_between(Budgets/I, mse_mean_Owen - mse_std_Owen, mse_mean_Owen + mse_std_Owen, color='C2', alpha=0.2)\n",
    "\n",
    "plt.plot(Budgets/I,mse_mean_Ortho,label=\"Ortho-Shapley\")\n",
    "plt.fill_between(Budgets/I, mse_mean_Ortho - mse_std_Ortho, mse_mean_Ortho + mse_std_Ortho, color='C3', alpha=0.2)\n",
    "plt.axvline(x = 1, color='black', linestyle='-')\n",
    "\n",
    "plt.yscale('log')\n",
    "plt.xscale('log')\n",
    "plt.xticks(size=15)\n",
    "plt.yticks(size=15)\n",
    "plt.grid(True, which=\"both\")\n",
    "plt.ylabel('MSE',size=20)\n",
    "plt.xlabel('Sampling budget w.r.t. DU-Shapley',size=13)\n",
    "plt.legend()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "233177d3",
   "metadata": {},
   "source": [
    "# Experiments Section 4.1"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "49c36ce8",
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "from sklearn import datasets, linear_model\n",
    "from sklearn.metrics import mean_squared_error, r2_score\n",
    "import math, random\n",
    "from matplotlib import pyplot as plt\n",
    "\n",
    "def powerset(s):\n",
    "    S = []\n",
    "    x = len(s)\n",
    "    for i in range(1 << x):\n",
    "        S.append([s[j] for j in range(x) if (i & (1 << j))])\n",
    "    return S\n",
    "\n",
    "#np.random.seed(10)\n",
    "\n",
    "def get_micro_set(d,theta,sigma):\n",
    "    X = np.random.normal(mu,1,size=(d,d))\n",
    "    Y = X.dot(theta) + np.random.normal(0,sigma,size=d)\n",
    "    return (X,Y)\n",
    "\n",
    "\n",
    "def homogeneous_x_generator(n,d):\n",
    "    mu = np.zeros(d)\n",
    "    X = np.random.normal(mu,1,size=(n,d))\n",
    "    return X,mu \n",
    "\n",
    "def get_data_sets(d,theta,b,sigma,n_list,x_generator=heterogeneous_x_generator):\n",
    "    X = []\n",
    "    Y = []\n",
    "    mus = []\n",
    "    for i in range(b):\n",
    "        n = n_list[i]\n",
    "        x, mu = x_generator(n,d)\n",
    "        X.append(x)\n",
    "        Y.append(X[i].dot(theta) + np.random.normal(0,sigma,size=n))\n",
    "        mus.append(mu)\n",
    "    return (X,Y),mus\n",
    "    \n",
    "\n",
    "def get_test_set(d,theta,sigma,n_points=10**4):\n",
    "    mu = np.zeros(d)\n",
    "    n = n_points\n",
    "    X_test = np.random.normal(mu,1,size=(n_points,d))\n",
    "    Y_test = X_test.dot(theta) + np.random.normal(0,sigma,size=n_points)\n",
    "    return (X_test,Y_test)\n",
    "\n",
    "def get_test_prediction(X_train,Y_train,X_test):\n",
    "    model = linear_model.LinearRegression()\n",
    "    model.fit(X_train,Y_train)\n",
    "    pred = model.predict(X_test)\n",
    "    return pred\n",
    "\n",
    "def get_value(X_train,Y_train,X_micro,Y_micro,X_test,Y_test):\n",
    "    pred = get_test_prediction(X_train,Y_train,X_test)\n",
    "    pred_0 = get_test_prediction(X_micro,Y_micro,X_test)\n",
    "    u = mean_squared_error(Y_test, pred_0) - mean_squared_error(Y_test, pred)\n",
    "    return u \n",
    "\n",
    "def get_training_data(I,data_micro,data_player):\n",
    "    X_train  = data_micro[0]\n",
    "    Y_train  = data_micro[1]\n",
    "    X = data_player[0]\n",
    "    Y = data_player[1]\n",
    "    for j in I:\n",
    "        X_train = np.concatenate((X_train,X[j]),axis=0)\n",
    "        Y_train = np.concatenate((Y_train,Y[j]),axis=0)\n",
    "    return (X_train,Y_train)\n",
    "\n",
    "def get_group_utility(I,data_micro,data_player,data_test):\n",
    "    (X_train,Y_train) = get_training_data(I,data_micro,data_player)\n",
    "    X_test,Y_test = data_test\n",
    "    u =  get_value(X_train,Y_train,data_micro[0],data_micro[1],X_test,Y_test)\n",
    "    return u\n",
    "\n",
    "def Shapley_exact(b,utility):\n",
    "    SV = np.zeros(b)\n",
    "    for i in range(b):\n",
    "        total_set = powerset(list(set(range(b)) - set([i])))\n",
    "        SVi = np.zeros(len(total_set)-1)\n",
    "        for k in range(len(total_set)-1):\n",
    "            s = total_set[k+1]\n",
    "            marginal = (utility(list(np.concatenate((s,[i])))) - utility(s))\n",
    "            SVi[k] = (math.factorial(len(s)) * math.factorial(b - len(s) - 1) / math.factorial(b)) * marginal\n",
    "        SV[i] = np.sum(SVi) + (1/b) * (utility([i])- utility([]))\n",
    "    return SV\n",
    "\n",
    "def Shapley_approx_MC(T,R,rep,utility):\n",
    "    SVi = np.zeros((b,T,rep,R))\n",
    "    for k in range(R):\n",
    "        for r in range(rep):\n",
    "            for t in range(T):\n",
    "                perm = np.random.permutation(b)\n",
    "                for i in range(b):\n",
    "                    idx = np.where(perm==i)[0][0]\n",
    "                    if idx != 0:\n",
    "                        a = utility(perm[0:idx+1]) - utility(perm[0:idx]) \n",
    "                    if idx == 0:\n",
    "                        a = utility(perm[0:idx+1])\n",
    "                    SVi[i,t,r,k] = (t * SVi[i,t-1,r,k] + a) / (t+1) \n",
    "    return SVi\n",
    "\n",
    "def Shapley_approx_MC_anti(T,R,rep,utility):\n",
    "    SVi = np.zeros((b,T,rep,R))\n",
    "    for k in range(R):\n",
    "        for r in range(rep):\n",
    "            for t in range(T):\n",
    "                perm = np.random.permutation(b)\n",
    "                perm_anti = np.flip(perm)\n",
    "                for i in range(b):\n",
    "                    \n",
    "                    idx = np.where(perm==i)[0][0]\n",
    "                    if idx != 0:\n",
    "                        a1 = utility(perm[0:idx+1]) - utility(perm[0:idx]) \n",
    "                    if idx == 0:\n",
    "                        a1 = utility(perm[0:idx+1]) \n",
    "                        \n",
    "                    idx = np.where(perm_anti==i)[0][0]\n",
    "                    if idx != 0:\n",
    "                        a2 = utility(perm_anti[0:idx+1]) - utility(perm_anti[0:idx]) \n",
    "                    if idx == 0:\n",
    "                        a2 = utility(perm_anti[0:idx+1]) \n",
    "                        \n",
    "                    SVi[i,t,r,k] = (2*t * SVi[i,t-1,r,k] + a1 + a2) / (2*(t+1))\n",
    "    return SVi\n",
    "\n",
    "def Shapley_approx_Owen(T,R,rep,T_inner,utility):\n",
    "    SVi = np.zeros((b,T*T_inner,rep,R))\n",
    "    for k in range(R):\n",
    "        for r in range(rep):\n",
    "            for i in range(b):\n",
    "                for t in range(T):\n",
    "                    SVii = 0\n",
    "                    for tt in range(T_inner):\n",
    "        \n",
    "                        q_proba = np.random.uniform()\n",
    "        \n",
    "                        idx = np.random.binomial(1, q_proba, b-1)\n",
    "\n",
    "                        B = idx * list(set(range(b)) - set([i]))\n",
    "                        B = B[B != 0]\n",
    "\n",
    "                        SVii += utility(np.concatenate((B,[i]))) - utility(B)\n",
    "                \n",
    "                    SVi[i,t,r,k] = (t * SVi[i,t-1,r,k] + SVii/T_inner) / (t+1)\n",
    "    return SVi\n",
    "\n",
    "def shapley_MC(T,R,rep,b,data_micro,data_player,data_test):\n",
    "    utility = lambda I: get_group_utility(I,data_micro,data_player,data_test)\n",
    "    shapley = Shapley_approx_MC(T,R,rep,utility)\n",
    "    return shapley\n",
    "\n",
    "def shapley_MC_anti(T,R,rep,b,data_micro,data_player,data_test):\n",
    "    utility = lambda I: get_group_utility(I,data_micro,data_player,data_test)\n",
    "    shapley = Shapley_approx_MC_anti(T,R,rep,utility)\n",
    "    return shapley\n",
    "\n",
    "def shapley_Owen(T,R,T_inner,rep,b,data_micro,data_player,data_test):\n",
    "    utility = lambda I: get_group_utility(I,data_micro,data_player,data_test)\n",
    "    shapley = Shapley_approx_Owen(T,R,rep,T_inner,utility)\n",
    "    return shapley\n",
    "\n",
    "def shapley_pure(b,data_micro,data_player,data_test):\n",
    "    utility = lambda I: get_group_utility(I,data_micro,data_player,data_test)\n",
    "    shapley = Shapley_exact(b,utility)\n",
    "    return shapley\n",
    "\n",
    "def _DU_shapley(i,b,data_micro,data_player,data_test):\n",
    "    full_group = set(list(range(b)))\n",
    "    I = list(full_group-set([i]))\n",
    "    (X_train_full,Y_train_full) = get_training_data(I,data_micro,data_player)\n",
    "    _get_value = lambda  x_data,y_data:  get_value(x_data,y_data,data_micro[0],data_micro[1],data_test[0],data_test[1])\n",
    "    ret = 0 \n",
    "    X_i = data_player[0][i]\n",
    "    Y_i = data_player[1][i]\n",
    "    n_competition = b-1\n",
    "    for k in range(n_competition+1):\n",
    "        nb_elem =k * int(len(Y_train_full)/(n_competition))\n",
    "        idx = random.sample(range(len(Y_train_full)), nb_elem)\n",
    "        X_train =  np.concatenate((X_train_full[idx,:],data_micro[0]),axis=0)\n",
    "        Y_train = np.concatenate((Y_train_full[idx] ,data_micro[1]),axis=0)\n",
    "        u = _get_value(X_train,Y_train)\n",
    "        ret -= u\n",
    "        X_train =  np.concatenate((X_train,X_i),axis=0)\n",
    "        Y_train = np.concatenate((Y_train ,Y_i),axis=0)\n",
    "        u = _get_value(X_train,Y_train)\n",
    "        ret += u\n",
    "    return ret/(n_competition+1)\n",
    "\n",
    "def DU_shapley(b,data_micro,data_player,data_test):\n",
    "    return [_DU_shapley(i,b,data_micro,data_player,data_test) for i in range(b)]"
   ]
  }
 ],
 "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.11.3"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
