{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 161,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<torch._C.Generator at 0x76a1b00f48b0>"
      ]
     },
     "execution_count": 161,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "import torch\n",
    "import torch.nn as nn\n",
    "import numpy as np\n",
    "\n",
    "torch.manual_seed(0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 162,
   "metadata": {},
   "outputs": [],
   "source": [
    "class DeepSetNetwork(nn.Module):\n",
    "    def __init__(self, input_dim):\n",
    "        super(DeepSetNetwork, self).__init__()\n",
    "        self.linear1 = nn.Linear(input_dim, input_dim)\n",
    "        self.init_weights()\n",
    "\n",
    "    def init_weights(self):\n",
    "        alpha, gamma = np.random.rand() * 2 - 1, np.random.rand() * 2 - 1\n",
    "        weights = alpha * torch.eye(self.linear1.weight.shape[0]) + gamma * (torch.ones_like(self.linear1.weight))\n",
    "        self.linear1.weight.data = weights\n",
    "        self.linear1.bias.data = torch.zeros_like(self.linear1.bias)\n",
    "\n",
    "    def forward(self, x):\n",
    "        x = self.linear1(x)\n",
    "        x = torch.relu(x)\n",
    "        x = torch.sum(x, dim=1)\n",
    "        return x.view(-1, 1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 163,
   "metadata": {},
   "outputs": [],
   "source": [
    "def sample_from_cube(npoints, ndim, side_length):\n",
    "    vec = np.random.uniform(-side_length, side_length, (npoints, ndim))\n",
    "    return torch.Tensor(vec)\n",
    "\n",
    "def get_jacobians(points, model):\n",
    "    \"\"\"\n",
    "    Compute the Jacobian of the model outputs with respect to its inputs for each input point.\n",
    "    \n",
    "    :param points: A tensor of shape (N, M) containing N points of dimension M.\n",
    "    :param model: A PyTorch model that accepts inputs of shape (N, M) and outputs a tensor of shape (N, 1).\n",
    "    :returns: A tensor of Jacobians of shape (N, M), where each row corresponds to the Jacobian of the model output with respect to the input point.\n",
    "    \"\"\"\n",
    "    points.requires_grad = True\n",
    "    output = model(points)\n",
    "    output.backward(torch.ones_like(output))\n",
    "    jacobians = points.grad\n",
    "    return jacobians\n",
    "\n",
    "def remove_duplicates(array):\n",
    "    return np.unique(array, axis=0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 164,
   "metadata": {},
   "outputs": [],
   "source": [
    "def sort_point_components(tensor):\n",
    "    return torch.sort(tensor, dim=1).values"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 165,
   "metadata": {},
   "outputs": [],
   "source": [
    "import math\n",
    "\n",
    "def estimate_linear_regions(model, radius, n_points, point_dim):\n",
    "    points = sample_from_cube(npoints=n_points, ndim=point_dim, side_length=radius)\n",
    "    jacobians = get_jacobians(points, model).detach().numpy()\n",
    "    unique_jacobians = remove_duplicates(np.around(jacobians, 10))\n",
    "    return unique_jacobians.shape[0]\n",
    "\n",
    "\n",
    "def estimate_linear_regions_using_fundamental_domain(model, radius, n_points, point_dim):\n",
    "    points = sample_from_cube(npoints=n_points, ndim=point_dim, side_length=radius)\n",
    "    points = sort_point_components(points)\n",
    "    jacobians = get_jacobians(points, model).detach().numpy()\n",
    "    unique_jacobians = remove_duplicates(np.around(jacobians, 10))\n",
    "    total = 0\n",
    "    for jacobian in unique_jacobians:\n",
    "        counts = np.unique(jacobian, return_counts=True)[1]\n",
    "        total += math.factorial(point_dim) / np.prod([math.factorial(count) for count in counts])\n",
    "    return total"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 166,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Running experiment with 10 models, 2 input dimension, search radius 5, and 100 points.\n",
      "Model 0\n",
      "Regular: 4, Fundamental domain: 4.0, Ratio: 1.0\n",
      "Model 1\n",
      "Regular: 4, Fundamental domain: 4.0, Ratio: 1.0\n",
      "Model 2\n",
      "Regular: 4, Fundamental domain: 4.0, Ratio: 1.0\n",
      "Model 3\n",
      "Regular: 4, Fundamental domain: 4.0, Ratio: 1.0\n",
      "Model 4\n",
      "Regular: 4, Fundamental domain: 4.0, Ratio: 1.0\n",
      "Model 5\n",
      "Regular: 4, Fundamental domain: 4.0, Ratio: 1.0\n",
      "Model 6\n",
      "Regular: 4, Fundamental domain: 4.0, Ratio: 1.0\n",
      "Model 7\n",
      "Regular: 4, Fundamental domain: 4.0, Ratio: 1.0\n",
      "Model 8\n",
      "Regular: 4, Fundamental domain: 4.0, Ratio: 1.0\n",
      "Model 9\n",
      "Regular: 4, Fundamental domain: 4.0, Ratio: 1.0\n",
      "Average ratio: 1.0\n",
      "Running experiment with 10 models, 3 input dimension, search radius 5, and 1000 points.\n",
      "Model 0\n",
      "Regular: 8, Fundamental domain: 8.0, Ratio: 1.0\n",
      "Model 1\n",
      "Regular: 6, Fundamental domain: 6.0, Ratio: 1.0\n",
      "Model 2\n",
      "Regular: 8, Fundamental domain: 8.0, Ratio: 1.0\n",
      "Model 3\n",
      "Regular: 8, Fundamental domain: 8.0, Ratio: 1.0\n",
      "Model 4\n",
      "Regular: 8, Fundamental domain: 8.0, Ratio: 1.0\n",
      "Model 5\n",
      "Regular: 8, Fundamental domain: 8.0, Ratio: 1.0\n",
      "Model 6\n",
      "Regular: 8, Fundamental domain: 8.0, Ratio: 1.0\n",
      "Model 7\n",
      "Regular: 8, Fundamental domain: 8.0, Ratio: 1.0\n",
      "Model 8\n",
      "Regular: 8, Fundamental domain: 10.0, Ratio: 1.25\n",
      "Model 9\n",
      "Regular: 8, Fundamental domain: 8.0, Ratio: 1.0\n",
      "Average ratio: 1.025\n",
      "Running experiment with 10 models, 4 input dimension, search radius 5, and 10000 points.\n",
      "Model 0\n",
      "Regular: 16, Fundamental domain: 16.0, Ratio: 1.0\n",
      "Model 1\n",
      "Regular: 16, Fundamental domain: 16.0, Ratio: 1.0\n",
      "Model 2\n",
      "Regular: 16, Fundamental domain: 16.0, Ratio: 1.0\n",
      "Model 3\n",
      "Regular: 16, Fundamental domain: 29.0, Ratio: 1.8125\n",
      "Model 4\n",
      "Regular: 16, Fundamental domain: 29.0, Ratio: 1.8125\n",
      "Model 5\n",
      "Regular: 16, Fundamental domain: 19.0, Ratio: 1.1875\n",
      "Model 6\n",
      "Regular: 16, Fundamental domain: 16.0, Ratio: 1.0\n",
      "Model 7\n",
      "Regular: 16, Fundamental domain: 16.0, Ratio: 1.0\n",
      "Model 8\n",
      "Regular: 16, Fundamental domain: 16.0, Ratio: 1.0\n",
      "Model 9\n",
      "Regular: 16, Fundamental domain: 16.0, Ratio: 1.0\n",
      "Average ratio: 1.18125\n",
      "Running experiment with 10 models, 5 input dimension, search radius 5, and 100000 points.\n",
      "Model 0\n",
      "Regular: 32, Fundamental domain: 36.0, Ratio: 1.125\n",
      "Model 1\n",
      "Regular: 32, Fundamental domain: 47.0, Ratio: 1.46875\n",
      "Model 2\n",
      "Regular: 32, Fundamental domain: 32.0, Ratio: 1.0\n",
      "Model 3\n",
      "Regular: 29, Fundamental domain: 12.0, Ratio: 0.41379310344827586\n",
      "Model 4\n",
      "Regular: 32, Fundamental domain: 32.0, Ratio: 1.0\n",
      "Model 5\n",
      "Regular: 32, Fundamental domain: 32.0, Ratio: 1.0\n",
      "Model 6\n",
      "Regular: 32, Fundamental domain: 32.0, Ratio: 1.0\n",
      "Model 7\n",
      "Regular: 32, Fundamental domain: 51.0, Ratio: 1.59375\n",
      "Model 8\n",
      "Regular: 32, Fundamental domain: 77.0, Ratio: 2.40625\n",
      "Model 9\n",
      "Regular: 32, Fundamental domain: 51.0, Ratio: 1.59375\n",
      "Average ratio: 1.2601293103448277\n",
      "Running experiment with 10 models, 6 input dimension, search radius 5, and 1000000 points.\n",
      "Model 0\n",
      "Regular: 64, Fundamental domain: 247.0, Ratio: 3.859375\n",
      "Model 1\n",
      "Regular: 64, Fundamental domain: 64.0, Ratio: 1.0\n",
      "Model 2\n",
      "Regular: 64, Fundamental domain: 147.0, Ratio: 2.296875\n",
      "Model 3\n",
      "Regular: 64, Fundamental domain: 177.0, Ratio: 2.765625\n",
      "Model 4\n",
      "Regular: 64, Fundamental domain: 247.0, Ratio: 3.859375\n",
      "Model 5\n",
      "Regular: 64, Fundamental domain: 182.0, Ratio: 2.84375\n",
      "Model 6\n",
      "Regular: 64, Fundamental domain: 138.0, Ratio: 2.15625\n",
      "Model 7\n",
      "Regular: 64, Fundamental domain: 64.0, Ratio: 1.0\n",
      "Model 8\n",
      "Regular: 64, Fundamental domain: 147.0, Ratio: 2.296875\n",
      "Model 9\n",
      "Regular: 64, Fundamental domain: 64.0, Ratio: 1.0\n",
      "Average ratio: 2.3078125\n"
     ]
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "\n",
    "def run_experiment(num_models, point_dim, search_radius, base_num_points):\n",
    "    num_points_adjusted_for_dimension = base_num_points ** point_dim\n",
    "    print(f\"Running experiment with {num_models} models, {point_dim} input dimension, search radius {search_radius}, and {num_points_adjusted_for_dimension} points.\")\n",
    "    models = [DeepSetNetwork(point_dim) for _ in range(num_models)]\n",
    "    all_ratios = []\n",
    "    for i, model in enumerate(models):\n",
    "        print(f\"Model {i}\")\n",
    "        regular_sampling = estimate_linear_regions(model, search_radius, num_points_adjusted_for_dimension, point_dim)\n",
    "        fundamental_domain_sampling = estimate_linear_regions_using_fundamental_domain(model, search_radius, num_points_adjusted_for_dimension//(math.factorial(point_dim)), point_dim)\n",
    "        ratio = fundamental_domain_sampling / regular_sampling\n",
    "        all_ratios.append(ratio)\n",
    "        print(f\"Regular: {regular_sampling}, Fundamental domain: {fundamental_domain_sampling}, Ratio: {ratio}\")\n",
    "\n",
    "    average_ratio = sum(all_ratios) / num_models\n",
    "    print(f\"Average ratio: {average_ratio}\")\n",
    "\n",
    "    return all_ratios\n",
    "\n",
    "NUM_MODELS = 10\n",
    "point_dim_to_ratios = {}\n",
    "for point_dim in range(2, 7):\n",
    "    point_dim_to_ratios[point_dim] = run_experiment(num_models=NUM_MODELS, point_dim=point_dim, search_radius=5, base_num_points=10)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 167,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Input dimension 2: mean 1.0, variance 0.0\n",
      "Input dimension 3: mean 1.025, variance 0.005625\n",
      "Input dimension 4: mean 1.18125, variance 0.10269531250000001\n",
      "Input dimension 5: mean 1.2601293103448277, variance 0.2584934694560048\n",
      "Input dimension 6: mean 2.3078125, variance 1.0465844726562499\n"
     ]
    },
    {
     "data": {
      "image/png": "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GzZs0JYtW9SqVSunxjz77LOaP3++PvzwQ0VFRVXZz2g0ymisvZd1AQAAz6rWmht3sdlsGj9+vNatW6esrCy1bt3aqXFPP/20nnrqKb3//vuKiYmp4SoBAEB94nK4+dvf/lZpu8FgkL+/v9q2bauePXvK1/fyK6STk5OVlpamt99+W4GBgSoqKpIkBQcH2++8SkpKUsuWLWU2myVJCxYs0MyZM5WWlqbIyEj7mICAAAUEBLh6OgAAwMu4HG6ee+45HTt2TOXl5WrWrJkk6aefflLjxo0VEBCgo0ePqk2bNtq8efNl17b88qqGC1/lIEmrVq3SiBEjJEn5+fny8fFxGHPmzBn94Q9/cBgza9YszZ4929XTAQAAXsblcDNv3jwtX75cf//733XDDTdIkg4cOKCHH35YDz30kOLi4vTHP/5RkydP1tq1ay95LGcesZOVleWw/80337haMgAAuIq4HG6mT5+uN9980x5sJKlt27Z69tlndf/99+vQoUN6+umndf/997u1UAAAAGe4/PqFwsJCnTt37qL2c+fO2de/tGjRQidPnrzy6gAAAFzkcri588479fDDD2v37t32tt27d2vs2LG66667JEl79uxx+s4nAAAAd3I53KxcuVIhISGKjo62P0MmJiZGISEhWrlypaTzdy4tXLjQ7cUCAABcjstrbsLDw5WRkaGvvvpK+/fvlyTdfPPNuvnmm+197rzzTvdVCAAA4IJqP8SvTZs2MhgMuuGGG9SggUefBQgAAGDn8rRUeXm5Ro8ercaNG+uWW25Rfn6+JGn8+PGaP3++2wsEAABwhcvhJjU1VZ999tlFL6uMj4/XmjVr3FocAACAq1yeT1q/fr3WrFmj3/zmNzIYDPb2W265RQcPHnRrcQAAAK5y+crNsWPHFBoaelF7WVmZQ9gBAADwBJfDTUxMjN599137/i+B5u9//7tiY2NdOpbZbFa3bt0UGBio0NBQJSYmat++fZcd98Ybb6hdu3by9/dXp06dtHHjRtdOAgAAeK1qvVuqX79++uKLL3Tu3DktWbJEX3zxhbZt26bs7GyXjpWdna3k5GR169ZN586d07Rp09SnTx998cUXatKkSaVjtm3bpiFDhshsNut//ud/lJaWpsTERO3atUsdO3Z09XQAAICXMdiceXvlfzl06JDMZrM+++wznTp1SrfeequmTJmi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eAeXn1C8atWqmqjDaf/61780atQopaenq3///h6tBYBnXdvYuQe7OdsPgHeo1usXdu7cqddff135+fk6c+aMw2dvvfWW08c5deqUDhw4YN8/fPiw8vLyFBISouuuu06pqak6cuSIXnnlFUnnp6KGDx+uJUuWqHv37ir6v/eRNGrUSMHBwdU5FQD12K2htyqscZiOlh+tdN2NQQaFNQ7TraG3eqA6AJ7i8t1S6enp6tGjh7788kutW7dOZ8+e1eeff66PPvrI5YCxc+dOde3a1f6MmpSUFHXt2tV+W3dhYaHy8/Pt/ZcvX65z584pOTlZERER9m3ixImungYAL+Dr46upt02t9DODzt/VOeW2KSwmBq4yLj/nJioqSg8//LCSk5MVGBiozz77TK1bt9bDDz+siIiIOn9nEs+5AbxPZc+5CW8crim3TeE5N/WItbxc+249/5qFm3flyueCu3KBGn3OzcGDB+1rXfz8/FRWViaDwaDJkydr+fLl1asYAK5A/PXxWj9gvX3/xbtf1Kb7NxFsgKuUy+GmWbNmOnnypCSpZcuW2rt3ryTpxIkTKi8vd291AOCkC6eeosOimYoCrmIuLyju2bOnMjIy1KlTJw0cOFATJ07URx99pIyMDN199901USMAAIDTXA43zz//vE6fPi1Jevzxx9WwYUNt27ZN999/v6ZPn+72AoFac6ZMmtfi/M/Tvpf8mni2HgBAtbgcbi58j5OPj4+mTv31ToWff/7ZPVUBAABUk8trbipjsVi0aNEitW7d2h2HAwAAqDanw43FYlFqaqpiYmLUo0cPrV+/XtL5Jxa3bt1azz33nCZPnlxTdQI1z1rx68/fbnPcBwDUG05PS82cOVMvvfSS4uPjtW3bNg0cOFAjR47U9u3btWjRIg0cOFC+vJ4e9dUX70jv/fXX/df+IAW1kH67QOowwHN1AQBc5nS4eeONN/TKK69owIAB2rt3r6KionTu3Dl99tlnMhgMNVkjULO+eEd6PUn678f3lxaebx/0CgEHAOoRp6elvvvuO0VHn39yZMeOHWU0GjV58mSCzf+psNqUc/AHvZ13RDkHf1CF1aUHP8NTrBXSpim6KNhIv7ZtmsoUFQDUI06Hm4qKCvn5+dn3GzRooICAgCv68i1btighIUEtWrSQwWCwr+O5lKysLN16660yGo1q27atVq9efUU1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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "for k, v in point_dim_to_ratios.items():\n",
    "    mean = sum(v) / len(v)\n",
    "    variance = sum([(x - mean) ** 2 for x in v]) / len(v)\n",
    "    print(f\"Input dimension {k}: mean {mean}, variance {variance}\")\n",
    "    plt.errorbar(k, mean, yerr=variance**0.5, fmt='o')\n",
    "plt.yticks(np.arange(0, 3.2, 0.2))\n",
    "plt.xticks(range(2, 7))\n",
    "plt.xlabel(\"Input dimension\")\n",
    "plt.ylabel(\"Ratio of linear regions\")\n",
    "plt.show()"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "myenv",
   "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.10.12"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
