{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# GeoCert : Binary MNIST Example\n",
    "In this example, we'll train a neural network to distinguish between 1s and 7s and run GeoCert on it"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "# =====================\n",
    "# Imports\n",
    "# =====================\n",
    "import sys\n",
    "sys.path.append('..')\n",
    "sys.path.append('../mister_ed') # library for adversarial examples\n",
    "\n",
    "from geocert import GeoCert\n",
    "from plnn import PLNN\n",
    "import pickle\n",
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.optim as optim\n",
    "from torch.autograd import Variable\n",
    "from torchvision import datasets, transforms\n",
    "import mnist.mnist_loader as  ml \n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Step 1: Train a model\n",
    "What follows is a block to \n",
    "1. Load MNIST 1's and 7's \n",
    "2. Define some boilerplate training code and standard regularizers\n",
    "3. Try to load a pretrained model. If no such model exists, train a model and save it so we can load it next time"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Load valset \n",
    "valset = ml.load_single_digits('val', [1, 7], batch_size=16, shuffle=False)        "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "##################################################################################\n",
    "#                                                                                #\n",
    "#                       Network Model + Data Loading                             #\n",
    "#                                                                                #\n",
    "##################################################################################\n",
    "\n",
    "# Define functions to train and evaluate a network \n",
    "\n",
    "def l1_loss(net):\n",
    "    return sum([_.norm(p=1) for _ in net.parameters() if _.dim() > 1])\n",
    "\n",
    "def l2_loss(net):\n",
    "    return sum([_.norm(p=2) for _ in net.parameters() if _.dim() > 1])\n",
    "  \n",
    "    \n",
    "def train(net, trainset, num_epochs):\n",
    "    opt = optim.Adam(net.parameters(), lr=1e-3, weight_decay=0)\n",
    "    for epoch in range(num_epochs):\n",
    "        err_acc = 0\n",
    "        err_count = 0\n",
    "        for data, labels in trainset:\n",
    "            output = net(Variable(data.view(-1, 784)))\n",
    "            l = nn.CrossEntropyLoss()(output, Variable(labels)).view([1])\n",
    "            l1_scale = torch.Tensor([1e-3]) # L1 regularization for speed\n",
    "            l += l1_scale * l1_loss(net).view([1])\n",
    "            \n",
    "            err_acc += (output.max(1)[1].data != labels).float().mean() \n",
    "            err_count += 1\n",
    "            opt.zero_grad() \n",
    "            (l).backward() \n",
    "            opt.step() \n",
    "        print(\"(%02d) error:\" % epoch, err_acc / err_count)\n",
    "            \n",
    "        \n",
    "def test_acc(net, valset):\n",
    "    err_acc = 0 \n",
    "    err_count = 0 \n",
    "    for data, labels in valset:\n",
    "        n = data.shape[0]\n",
    "        output = net(Variable(data.view(-1, 784)))\n",
    "        err_acc += (output.max(1)[1].data != labels).float().mean() * n\n",
    "        err_count += n\n",
    "        \n",
    "    print(\"Accuracy of: %.03f\" % (1 - (err_acc / err_count).item()))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Training a new network\n",
      "[784, 10, 50, 10, 2]\n",
      "(00) error: tensor(0.0305)\n",
      "(01) error: tensor(0.0076)\n",
      "(02) error: tensor(0.0064)\n",
      "(03) error: tensor(0.0054)\n",
      "(04) error: tensor(0.0052)\n",
      "(05) error: tensor(0.0052)\n",
      "(06) error: tensor(0.0047)\n",
      "(07) error: tensor(0.0043)\n",
      "(08) error: tensor(0.0043)\n",
      "(09) error: tensor(0.0040)\n",
      "Accuracy of: 0.992\n"
     ]
    }
   ],
   "source": [
    "NETWORK_NAME = '17_mnist_small.pkl'\n",
    "MNIST_DIM = 784\n",
    "layer_sizes = [MNIST_DIM, 10, 50, 10, 2]\n",
    "\n",
    "try: \n",
    "    network = pickle.load(open(NETWORK_NAME, 'rb'))\n",
    "    net = network.net\n",
    "    print(\"Loaded pretrained network\")\n",
    "except:\n",
    "    print(\"Training a new network\")\n",
    "    trainset = ml.load_single_digits('train', [1, 7], batch_size=16, shuffle=False)  \n",
    "    network = PLNN([MNIST_DIM, 10, 50, 10, 2])\n",
    "    net = network.net\n",
    "    train(net, trainset, 10)\n",
    "    pickle.dump(network, open(NETWORK_NAME, 'wb'))\n",
    "    \n",
    "test_acc(net, valset)    "
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Step 2: Run GeoCert \n",
    "We'll now demonstrate basic GeoCert usage..."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "# First make the GeoCert object \n",
    "geo = GeoCert(network, hyperbox_bounds=(0.0, 1.0),\n",
    "                      verbose=True, neuron_bounds='ia')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "# And pick an example image \n",
    "example_image = valset[0][0][0]\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Starting upper bound computation\n",
      "Upper bound of 0.48637357354164124 in 5.75 seconds\n",
      "---Initial Polytope---\n",
      "Academic license - for non-commercial use only\n",
      "(p0) Popped: 0.109737  | 0.486374\n",
      "(p0) Popped: 0.119779  | 0.486374\n",
      "(p0) Popped: 0.160164  | 0.486374\n",
      "(p0) Popped: 0.164347  | 0.486374\n",
      "(p0) Popped: 0.164703  | 0.486374\n",
      "(p0) Popped: 0.165997  | 0.486374\n",
      "(p0) Popped: 0.175892  | 0.486374\n",
      "-------------------- DOMAIN UPDATE | L_inf 0.251913 | L_2 5.462510\n",
      "(p0) Popped: 0.180657  | 0.251913\n",
      "(p0) Popped: 0.184537  | 0.251913\n",
      "(p0) Popped: 0.198900  | 0.251913\n",
      "(p0) Popped: 0.203945  | 0.251913\n",
      "(p0) Popped: 0.204625  | 0.251913\n",
      "(p0) Popped: 0.205135  | 0.251913\n",
      "(p0) Popped: 0.212172  | 0.251913\n",
      "(p0) Popped: 0.213016  | 0.251913\n",
      "(p0) Popped: 0.213419  | 0.251913\n",
      "(p0) Popped: 0.213896  | 0.251913\n",
      "(p0) Popped: 0.216059  | 0.251913\n",
      "(p0) Popped: 0.217895  | 0.251913\n",
      "(p0) Popped: 0.226287  | 0.251913\n",
      "(p0) Popped: 0.236035  | 0.251913\n",
      "GeoCert Return Object\n",
      "\tProblem Type: min_dist\n",
      "\tStatus: SUCCESS\n",
      "\tRobustness: 0.2519\n"
     ]
    }
   ],
   "source": [
    "# And then run geocert \n",
    "example_output = geo.run(example_image, compute_upper_bound=True, \n",
    "                             lp_norm='l_inf', problem_type='min_dist')\n",
    "print(example_output)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Decision bound value:  tensor([[0.1439, 0.1439]], grad_fn=<AddmmBackward>)\n"
     ]
    }
   ],
   "source": [
    "# The return object can be interacted upon like ...\n",
    "best_adv_ex = torch.Tensor(example_output.best_ex)\n",
    "print(\"Decision bound value: \", network(best_adv_ex))\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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EYEPAACJCHy67oYbbohFixYN+rof//jHceWVV76hs8qyjP7+/jf0ewDwxmzcuDEuuOCCOPLII2Py5MkxZcqU+MpXvtLo97rpppti+/btXd4QDiydqqqq4V6CPHbu3BkjRoxo7byyLOOKK66ItWvXtnYmAP/yz3/+M6ZMmRLnnHNOXH/99dHX1xfbtm2L73znO3H55Zfv8+/zfz8/Op1ObN68Od72trcN4daQmyf47LOHH344pkyZEpMmTYqTTz45nn322SjLMiZOnBgXXXRR9Pf3x/333x9z586Nb3zjGxERsWXLljjnnHNiwoQJMX369Jg3b17MnTs3IiIWLlwYM2fOjIjdoX7MMcfEpZdeGscdd1xMnDgx1qxZExG7b/qnnXZavP/974+JEyfGeeedF1u3bh2WawCQwQMPPBBHH310HHfccXHNNdfE2LFj4/e//3385je/iTPOOCNOOOGEmDRpUnzzm998fWZPPwMiIhYvXhyjR4+Om266Kfr6+iIiYuTIka/H/Y4dO2LBggXxgQ98IPr7++Pss8+OzZs3R0TE3Llz48ILL4yTTjopjjnmmPjsZz8bERHTp0+P/v7+2LhxY5uXBdIQ+OyTjRs3xnnnnRd33313PPXUU3HJJZfE7Nmzo6qqWL9+fcyZMyfWrl0bZ5111n/MffGLX4xDDjkk1q9fHw899FA89thjez3jueeei/PPPz+efPLJmD9/flx33XUREdHX1xeLFy+ONWvWxLp16+Ktb31r3HHHHUP6fgGy2rhxY1x44YVx//33x5NPPhkTJkyIv/3tb7Fr164499xz4+tf/3o8/vjjsXr16rjrrrvi8ccfH/BnwBNPPBHTpk3b63m33HJLvOUtb4lf/epXsXbt2jj22GPjC1/4wuu//sQTT8SDDz4Yzz33XHzrW9+KiIhf/OIXsXbt2hg3btyQXw/ISOCzT375y1/GscceG8cee2xERHz605+Ol156Kf785z/HkUceGSeffPIe5x555JG44IILotPpxOjRo+Occ87Z6xlHHXVUnHjiiRERMW3atNiwYUNERFRVFbfddltMnjw5Jk2aFA8++KC/kgPQ0OrVq2PSpEkxYcKEiIg4//zz481vfnO8+uqr8cwzz8SnPvWp6O/vjw9+8IOxZcuWePbZZwf8GTCYpUuXxj333BP9/f3R398f3//+9+N3v/vd679+1llnxejRo4fmzcIBqr2/LE1ao0aN2ufXdjqdvf7awQcf/Pr/7uvri507d0bE7n/8++ijj8bKlStjzJgxcfvtt8ejjz7afGEA/p+qquKwww7b4wOUZcuW7XXu+OOPj7vuumvA3/eOO+6Ij33sY3v89To/Q4B94wk++2Tq1Knx9NNPx7p16yIi4gc/+EGMHz8+xo8fP+DcRz7ykbj77rujqqp45ZVX4t5776199ubNm2Ps2LExZsyY2LJlSyxcuLDJWwAgdt/Pn3rqqXj++ecjIuKee+6J1157LQ466KAYM2ZMfO9733v9tS+88EJs2rRpwJ8B5557bvz973+PL33pS7Fr166I2P0v3t5+++0RETFz5sy47bbbYtu2bRERsW3btnjmmWf2ut/o0aPjH//4x5C8dzhQeILPPjn88MNj0aJFMWfOnNi5c2cceuihcd999w36L0DdcMMNcdFFF8XRRx8dY8eOjeOOO672fxlhzpw58cADD8T73ve+OPzww2P69Onxhz/84Y28HYAD1rhx4+K73/1uzJw5Mw466KA49dRTY9SoUTF27NhYvnx5XHHFFXHbbbfFrl27YuzYsbF48eIYP378Hn8GdDqdGDlyZKxcuTIWLFgQRx11VIwaNSo6nU6cd955ERFxzTXXxKuvvhonnnji6/8U95prromJEyfucb/Pfe5zceqpp8bIkSPjpz/9qb+HDw34z2QypHbs2BG7du2Kgw8+OLZu3RqnnXZazJ8/f8C/iw/A0NqyZcvrf+996dKlce2118b69euHeSugWzzBZ0ht3rw5Tj/99Ni1a1ds3749zjzzzDj77LOHey2AA9odd9wRS5YsiV27dsWYMWP26f+cENh/eIIPAACJ+JdsAQAgEYEPAACJCHwAAEhE4AMAQCICHwAAEhH4AACQyP8Cs80YASQpgvgAAAAASUVORK5CYII=\n",
      "text/plain": [
       "<Figure size 960x960 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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LX5/5513To8//njj6/b29nTzzTe/oO6hhx5Kr3zlK7c6/l133ZXGjBlTuE54/vz9l7/8JaW06fUTEWnx4sVp9OjRje+vWbMmDR8+PP32t7/d6jl/48aN6aMf/Wj6xCc+sdX5LrzwwjRt2rTG19OmTUunnXZaSmlTPmhtbU1PPfVUY3vZ1xz56t3Fv1/0aHfffXcMHz48hg8fHhERxx9/fJx++unx6KOPxl577RXjx4/f4v+79dZbY8aMGdHS0hJ9+vSJY445JhYvXrzF2kGDBsX+++8fEREHHHBAfO1rX4uIiJRSzJgxI66//vrYsGFDPPnkk/HWt771JdhL6Hmefy/tHXfcEfvuu29cdtllsXLlyhfUnXjiifHggw/Gv//978Zf6DZu3LjZ14ceemice+65MX78+NiwYUO8613vitmzZ8fRRx8dERErV66Mu+++O1paWuLXv/51XH311XH77bfHDjvsEHfccUdMmTIlFi1aFBERv/vd72LBggXxpje9Kc4+++z48pe/HN/+9rdj/vz5MW3atLjzzjtj9913j7Vr10ZExIMPPhgdHR1x0003Rd++fWPx4sUxbty4WLp0aeywww6N/dhxxx1j9uzZMXHixBg4cGAcf/zx8YMf/CB22223Rs3EiRNjw4YNMXHixLjgggu2+NaD+fPnN35elbVy5cp49tln47WvfW3je3vuuWc8/PDDW6xfs2ZNtLe3x3PPPReTJ0+O8847r3HFdPTo0bFw4cJ46qmnom/fvtu0DnqWu+66K1pbW2Pw4MERETF16tQ49dRT4z//+U8sWrQojj322Ebt6tWr489//nMsW7Zsq+f8Itdcc008+eST8dOf/jQiItavX7/ZX7SOPvro6NOnT4V7SHcn4DepXXbZpXRtS0vLVre94hWvaPy7V69ejT/j//CHP4y5c+fGbbfdFn379o2LLroo5s6d++IXDDQ8f/Lfd999IyLi5JNPjo9//OPbPM6aNWvi1ltvjeXLlze+9/TTTzfevx+x6c/zz/8MuPbaa+OPf/zjZiF51apV8cwzz0TEpl/y3/SmNzX+ffHFF0dExPXXXx/vf//7Y/fdd4+ITW8NiIi48cYbY/HixXHQQQc1xnvZy14WDz/8cOy9996brXWfffaJL3/5y3HAAQfEtGnTNnsbzt///vfYY489Ys2aNXHqqafGmWeeGZdeeukL9veRRx6JQYMGNb5ev359jBkzpvH1Y4891rgBQP/+/eOWW24pfhD/y+677x6PPvpoDBgwIFatWhXHHHNMfP3rX4+zzjorIiJ69+4dr371q+Oxxx4T8HlRUkrRr1+/WLBgwQu2XXfddVv9f6NHj47LLrus03EvvvjiOOSQQ7a4fVsyAz2D9+B3obFjx8af/vSnuO+++yIi4kc/+lG8/vWvj9e//vWd/r93vOMd8b3vfS9SSvH000/Hj3/8422e+4knnoj+/ftH3759Y/Xq1XHFFVe8mF0ASujsl/DOpJQiYtMvDM+/Z3zx4sXx2c9+tlHz3yf2lFJMnTp1s/eYP/7447HjjjtGxNZ/4e9s/oMPPniz8R599NEXhPvn/eEPf4jddtst/vGPf2z2/T322CMiInbeeec47bTTtvoh25122mmzDwb+3//932Zzv+51r2v8+/lwv+uuu0bv3r1j2bJljf+3dOnSxpz/bYcddogBAwZERES/fv3igx/84AvWsm7dusbjBVszduzYWLhwYeOX7SuvvDLWr18fO+ywQ/Tt23ezD8gvXrw4Vq1a1ek5/7jjjot///vfccEFF8Rzzz0XEZs+eHvRRRdFRMTkyZNjxowZjb+urV27tvGXuS3p06dPPPnkky/JvtM9CPhdaLfddourrroqTjzxxGhtbY2ZM2fG7NmzC8PA5z//+Vi9enUMGTIkJk2aFCNGjIhXvepV2zT3iSeeGGvXro23vOUtcdhhh3X6oTdg2xxwwAGxcOHC+Otf/xoREd/97ndj/fr12zzOLrvsEhMmTIgvfelLje899thj8cgjj2yx/t3vfndceeWVjbenbNy4sfHB+s4cccQRceWVV8bjjz8eEZvCw9q1a+PQQw+NW265JRYuXNio/e1vf7vFMebMmRM33XRTLFq0KO6+++6YNWtWRGy6mPB8KNm4cWPMmjUrRo4cucUxWltbN/vrRFlHH310fOtb34qITW9DevTRR7f4FscVK1Y0Pnz7n//8J372s59ttpbly5dHS0tLvPGNb9zmNdCzDBgwIL7zne/E5MmTo62tLf70pz/FLrvsEv379485c+bEz372s2htbW3cMOOZZ57p9Jy/0047xW233RZLliyJQYMGxfDhw2P//fdvvHY+85nPRHt7e+y///7R2toaY8eO3eJfCZ73qU99Kg4++GAfsu3JuvD9/7xI69evT88880xKKaWnn346HXjggelHP/pRF68K+G/XXHNNGjx4cGptbU1nnXXWFj9km1JKX/jCF9IZZ5zR+H//+/Xy5cvT8ccfn4YOHZqGDRuW9t9//8YHZ/97nOddffXVadSoUam1tTUNHjw4fepTn0opbf4B/JRSuu6669L48eMbX19xxRVp+PDhqbW1Ne23335p6dKlKaWUbr755jR27NjGeMcdd9wL9vXvf/97esMb3pAWLVqUUkpp8eLF6Q1veEO6//7705133tkYd999900nnHBCWrly5RYfsxUrVqQ999wzbdiwYYvbt/Yh22XLlqWDDz44DRo0KO27776bfRD4c5/7XJo5c2ZKKaWf/vSnaejQoY21fOxjH0vr1q3b7DE4+eSTtzgH/K///lDrz3/+8zR48OAuXA1sriWl//83YLqNFStWxGGHHRbPPfdcrFu3Lo488sj40pe+9KLfBgDQLE4//fR4+9vf3vgQcZ3GjRsXl112WQwZMqT2uel+pk+fHrNmzYrnnnsu+vbtG5dcckmMGjWqq5cFEREh4APQNFauXBm//OUv44QTTqh13uXLl8ett94aU6ZMqXVegJeCgA8AABnxIVsAAMiIgA8AABkR8AEAICMCPgAAZETABwCAjAj4AACQkf8HFIu9qVn4ceQAAAAASUVORK5CYII=\n",
      "text/plain": [
       "<Figure size 960x960 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# And there's an easy shorthand to display these images: \n",
    "example_output.display_images(include_diffs=False) \n",
    "example_output.display_images(include_diffs=True) # Easy way to show diffs between original and adv. ex\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "---Initial Polytope---\n",
      "(p0) Popped: 0.015449  | 4.000000\n",
      "(p0) Popped: 0.016587  | 4.000000\n",
      "(p0) Popped: 1.286739  | 4.000000\n",
      "(p0) Popped: 1.313097  | 4.000000\n",
      "(p0) Popped: 1.363232  | 4.000000\n",
      "(p0) Popped: 1.365227  | 4.000000\n",
      "(p0) Popped: 1.391826  | 4.000000\n",
      "(p0) Popped: 1.437648  | 4.000000\n",
      "(p0) Popped: 1.963248  | 4.000000\n",
      "(p0) Popped: 2.173057  | 4.000000\n",
      "(p0) Popped: 2.820217  | 4.000000\n",
      "(p0) Popped: 2.820216  | 4.000000\n",
      "(p0) Popped: 2.820221  | 4.000000\n",
      "(p0) Popped: 2.997484  | 4.000000\n",
      "(p0) Popped: 3.003162  | 4.000000\n",
      "(p0) Popped: 3.016613  | 4.000000\n",
      "(p0) Popped: 3.017497  | 4.000000\n",
      "(p0) Popped: 3.024027  | 4.000000\n",
      "(p0) Popped: 3.036220  | 4.000000\n",
      "(p0) Popped: 3.237544  | 4.000000\n",
      "(p0) Popped: 3.833163  | 4.000000\n",
      "GeoCert Return Object\n",
      "\tProblem Type: decision_problem\n",
      "\tStatus: SUCCESS\n",
      "\tRadius 4.00\n",
      "\n"
     ]
    }
   ],
   "source": [
    "# And we can also run the decision problem...\n",
    "example_2 = valset[0][0][1]\n",
    "example_2_output = geo.run(example_2, lp_norm='l_2', problem_type='decision_problem', \n",
    "                               decision_radius=4.0)\n",
    "print(example_2_output)\n",
    "# Recall that 'SUCCESS' status means an adversarial example DOES exist w/in the radius"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "---Initial Polytope---\n",
      "(p0) Popped: 0.055832  | 1.000000\n",
      "(p0) Popped: 0.056154  | 1.000000\n",
      "(p0) Popped: 0.133925  | 1.000000\n",
      "(p0) Popped: 0.133925  | 1.000000\n",
      "(p0) Popped: 0.144106  | 1.000000\n",
      "(p0) Popped: 0.150180  | 1.000000\n",
      "(p0) Popped: 0.150180  | 1.000000\n",
      "(p0) Popped: 0.157653  | 1.000000\n",
      "(p0) Popped: 0.158780  | 1.000000\n",
      "(p0) Popped: 0.170858  | 1.000000\n",
      "(p0) Popped: 0.175962  | 1.000000\n",
      "(p0) Popped: 0.182197  | 1.000000\n",
      "(p0) Popped: 0.182393  | 1.000000\n",
      "(p0) Popped: 0.182714  | 1.000000\n",
      "(p0) Popped: 0.184272  | 1.000000\n",
      "(p0) Popped: 0.185871  | 1.000000\n",
      "(p0) Popped: 0.185899  | 1.000000\n",
      "(p0) Popped: 0.189254  | 1.000000\n",
      "(p0) Popped: 0.189254  | 1.000000\n",
      "(p0) Popped: 0.190690  | 1.000000\n",
      "(p0) Popped: 0.192021  | 1.000000\n",
      "(p0) Popped: 0.192021  | 1.000000\n",
      "(p0) Popped: 0.194611  | 1.000000\n",
      "(p0) Popped: 0.206525  | 1.000000\n",
      "(p0) Popped: 0.212611  | 1.000000\n",
      "(p0) Popped: 0.215649  | 1.000000\n",
      "(p0) Popped: 0.216742  | 1.000000\n",
      "(p0) Popped: 0.218444  | 1.000000\n",
      "(p0) Popped: 0.219127  | 1.000000\n",
      "(p0) Popped: 0.220038  | 1.000000\n",
      "(p0) Popped: 0.220538  | 1.000000\n",
      "(p0) Popped: 0.235952  | 1.000000\n",
      "(p0) Popped: 0.238375  | 1.000000\n",
      "(p0) Popped: 0.248345  | 1.000000\n",
      "(p0) Popped: 0.255292  | 1.000000\n",
      "(p0) Popped: 0.386507  | 1.000000\n",
      "(p0) Popped: 0.388677  | 1.000000\n",
      "(p0) Popped: 0.458700  | 1.000000\n",
      "(p0) Popped: 0.460673  | 1.000000\n",
      "(p0) Popped: 0.466731  | 1.000000\n",
      "(p0) Popped: 0.468369  | 1.000000\n",
      "(p0) Popped: 0.602741  | 1.000000\n",
      "(p0) Popped: 0.604280  | 1.000000\n",
      "(p0) Popped: 0.604815  | 1.000000\n",
      "(p0) Popped: 0.605080  | 1.000000\n",
      "(p0) Popped: 0.605228  | 1.000000\n",
      "(p0) Popped: 0.605373  | 1.000000\n",
      "(p0) Popped: 0.605647  | 1.000000\n",
      "(p0) Popped: 0.606417  | 1.000000\n",
      "(p0) Popped: 0.606728  | 1.000000\n",
      "(p0) Popped: 0.607238  | 1.000000\n",
      "(p0) Popped: 0.607255  | 1.000000\n",
      "(p0) Popped: 0.608136  | 1.000000\n",
      "(p0) Popped: 0.608136  | 1.000000\n",
      "(p0) Popped: 0.608190  | 1.000000\n",
      "(p0) Popped: 0.608257  | 1.000000\n",
      "(p0) Popped: 0.608257  | 1.000000\n",
      "(p0) Popped: 0.608299  | 1.000000\n",
      "(p0) Popped: 0.610950  | 1.000000\n",
      "(p0) Popped: 0.612634  | 1.000000\n",
      "(p0) Popped: 0.613130  | 1.000000\n",
      "(p0) Popped: 0.616051  | 1.000000\n",
      "(p0) Popped: 0.619585  | 1.000000\n",
      "(p0) Popped: 0.620696  | 1.000000\n",
      "(p0) Popped: 0.621311  | 1.000000\n",
      "(p0) Popped: 0.623362  | 1.000000\n",
      "(p0) Popped: 0.624005  | 1.000000\n",
      "(p0) Popped: 0.624332  | 1.000000\n",
      "(p0) Popped: 0.624511  | 1.000000\n",
      "(p0) Popped: 0.624736  | 1.000000\n",
      "(p0) Popped: 0.625007  | 1.000000\n",
      "(p0) Popped: 0.625189  | 1.000000\n",
      "(p0) Popped: 0.625203  | 1.000000\n",
      "(p0) Popped: 0.626383  | 1.000000\n",
      "(p0) Popped: 0.626706  | 1.000000\n",
      "(p0) Popped: 0.626936  | 1.000000\n",
      "(p0) Popped: 0.626936  | 1.000000\n",
      "(p0) Popped: 0.627066  | 1.000000\n",
      "(p0) Popped: 0.627180  | 1.000000\n",
      "(p0) Popped: 0.627180  | 1.000000\n",
      "(p0) Popped: 0.627287  | 1.000000\n",
      "(p0) Popped: 0.630318  | 1.000000\n",
      "(p0) Popped: 0.634123  | 1.000000\n",
      "(p0) Popped: 0.642272  | 1.000000\n",
      "(p0) Popped: 0.643601  | 1.000000\n",
      "(p0) Popped: 0.671803  | 1.000000\n",
      "(p0) Popped: 0.674035  | 1.000000\n",
      "(p0) Popped: 0.682463  | 1.000000\n",
      "(p0) Popped: 0.701311  | 1.000000\n",
      "(p0) Popped: 0.731181  | 1.000000\n",
      "(p0) Popped: 0.745470  | 1.000000\n",
      "(p0) Popped: 0.790644  | 1.000000\n",
      "(p0) Popped: 0.791748  | 1.000000\n",
      "(p0) Popped: 0.813246  | 1.000000\n",
      "(p0) Popped: 0.820690  | 1.000000\n",
      "(p0) Popped: 0.856769  | 1.000000\n",
      "(p0) Popped: 0.857614  | 1.000000\n",
      "(p0) Popped: 0.872617  | 1.000000\n",
      "(p0) Popped: 0.877461  | 1.000000\n",
      "(p0) Popped: 0.954826  | 1.000000\n",
      "(p0) Popped: 0.955598  | 1.000000\n",
      "(p0) Popped: 0.960429  | 1.000000\n",
      "(p0) Popped: 0.961139  | 1.000000\n",
      "(p0) Popped: 0.964271  | 1.000000\n",
      "(p0) Popped: 0.964623  | 1.000000\n",
      "(p0) Popped: 0.965111  | 1.000000\n",
      "(p0) Popped: 0.965309  | 1.000000\n",
      "(p0) Popped: 0.965791  | 1.000000\n",
      "(p0) Popped: 0.967368  | 1.000000\n",
      "(p0) Popped: 0.968550  | 1.000000\n",
      "(p0) Popped: 0.969220  | 1.000000\n",
      "(p0) Popped: 0.969431  | 1.000000\n",
      "(p0) Popped: 0.969970  | 1.000000\n",
      "(p0) Popped: 0.970050  | 1.000000\n",
      "(p0) Popped: 0.970564  | 1.000000\n",
      "(p0) Popped: 0.971228  | 1.000000\n",
      "(p0) Popped: 0.974459  | 1.000000\n",
      "(p0) Popped: 0.975377  | 1.000000\n",
      "(p0) Popped: 0.975886  | 1.000000\n",
      "(p0) Popped: 0.978341  | 1.000000\n",
      "(p0) Popped: 0.978828  | 1.000000\n",
      "(p0) Popped: 0.979336  | 1.000000\n",
      "(p0) Popped: 0.980105  | 1.000000\n",
      "(p0) Popped: 0.980612  | 1.000000\n",
      "(p0) Popped: 0.982145  | 1.000000\n",
      "(p0) Popped: 0.982947  | 1.000000\n",
      "(p0) Popped: 0.983407  | 1.000000\n",
      "(p0) Popped: 0.989221  | 1.000000\n",
      "Warning for adding constraints: zero or small (< 1e-13) coefficients, ignored\n",
      "(p0) Popped: 0.989746  | 1.000000\n",
      "COUNTED 130 LINEAR REGIONS\n"
     ]
    }
   ],
   "source": [
    "# And we can also do things like count the number of linear regions in a ball (like the entire domain...)\n",
    "example_3 = valset[0][0][2]\n",
    "example_3_output = geo.run(example_3, lp_norm='l_inf', problem_type='count_regions', decision_radius=1.0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "print(example_3_output)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "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.6.8"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
