{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "cd7a270c",
   "metadata": {},
   "outputs": [],
   "source": [
    "import scipy\n",
    "import numpy as np\n",
    "import networkx as nx\n",
    "import ot\n",
    "import time\n",
    "import multiprocessing as mp\n",
    "from tqdm import tqdm, trange\n",
    "import torch\n",
    "import cvxpy as cp\n",
    "import torch.nn as nn\n",
    "from torch.optim import Adam, SGD\n",
    "import matplotlib.pyplot as plt\n",
    "from dataset import *\n",
    "from models import *\n",
    "import h5py"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "dfd60d9c",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<mpl_toolkits.mplot3d.art3d.Path3DCollection at 0x7fe15016bf10>"
      ]
     },
     "execution_count": 2,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "# Load testing data\n",
    "\n",
    "filename = '/data/sam/modelnet/data/modelnet40_ply_hdf5_2048/ply_data_train1.h5'\n",
    "dataset = None\n",
    "labels = None\n",
    "with h5py.File(filename,'r') as h5f: \n",
    "    dataset = h5f['data'][:]\n",
    "    labels = h5f['label'][:]\n",
    "    \n",
    "dataset = dataset - np.expand_dims(np.mean(dataset, axis=0), 0)  # center\n",
    "dist = np.max(np.sqrt(np.sum(dataset ** 2, axis=1)), 0)\n",
    "dataset = dataset / dist  # scale\n",
    "\n",
    "f = open('/data/sam/modelnet/data/shape_names.txt')\n",
    "lbl_names = f.readlines()\n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "fig = plt.figure()\n",
    "ax = fig.add_subplot(projection='3d')\n",
    "ax.scatter(dataset[208][:, 0], dataset[208][:, 1], dataset[208][:, 2], s=3)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "35d648c9",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 200/200 [00:00<00:00, 666.75it/s]\n"
     ]
    }
   ],
   "source": [
    "#Prepare test dataset\n",
    "import time\n",
    "P_test = []\n",
    "Q_test = []\n",
    "dists_test = []\n",
    "nmin = 10\n",
    "nmax = 200\n",
    "all_points = dataset[1040]\n",
    "raw_emd = []\n",
    "shtime = []\n",
    "shs = []\n",
    "for i in trange(200):\n",
    "    psz = np.random.randint(low=nmin, high=nmax)\n",
    "    qsz = np.random.randint(low=nmin, high=nmax)\n",
    "    \n",
    "    P = all_points[np.random.randint(low=0, high=2048, size=psz)]\n",
    "    Q = all_points[np.random.randint(low=0, high=2048, size=qsz)]\n",
    "    mat = ot.dist(P, Q, metric='euclidean')\n",
    "    p = (1/psz) * np.ones(psz)\n",
    "    q = (1/qsz) * np.ones(qsz)\n",
    "    dist = ot.emd2(p, q, mat)\n",
    "    P_test.append(torch.tensor(P, dtype=torch.float32))\n",
    "    Q_test.append( torch.tensor(Q, dtype=torch.float32))\n",
    "    dists_test.append(dist)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "3bc49ecd",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 30/30 [01:56<00:00,  3.88s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "average 0.8135583639144898 std dev 0.03709801515880551\n",
      "average 24.534481939259614 std dev 1.8343001813076107\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "shtime = []\n",
    "shs = []\n",
    "P_test =[]\n",
    "Q_test = []\n",
    "dists_test = []\n",
    "for i in trange(30):\n",
    "    psz = 5000\n",
    "    qsz = 5000\n",
    "    \n",
    "    P = all_points[np.random.randint(low=0, high=2048, size=psz)]\n",
    "    Q = all_points[np.random.randint(low=0, high=2048, size=qsz)]\n",
    "    mat = ot.dist(P, Q, metric='euclidean')\n",
    "    p = (1/psz) * np.ones(psz)\n",
    "    q = (1/qsz) * np.ones(qsz)\n",
    "    \n",
    "    start = time.time()\n",
    "    sh = ot.sinkhorn2(p, q, mat, reg=3.0)\n",
    "    end = time.time()\n",
    "    dist = ot.emd2(p, q, mat)\n",
    "    shs.append(abs(sh-dist)/dist)\n",
    "    \n",
    "    shtime.append(end - start)\n",
    "    P_test.append(torch.tensor(P, dtype=torch.float32))\n",
    "    Q_test.append(torch.tensor(Q, dtype=torch.float32))\n",
    "    dists_test.append(dist)\n",
    "print(\"average\", np.mean(shtime), \"std dev\", np.std(shtime))\n",
    "print(\"average\", np.mean(shs), \"std dev\", np.std(shs))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 92,
   "id": "27d9d7bf",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "average 4.322709083557129 std dev 0.8316484791383189\n"
     ]
    }
   ],
   "source": [
    "print(\"average\", np.mean(shtime), \"std dev\", np.std(shtime))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 82,
   "id": "80f8c77a",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "24.19717745413585 1.9470141536861683\n"
     ]
    }
   ],
   "source": [
    "shs = np.array(shs)\n",
    "emds = np.array(dists_test)\n",
    "errs = abs(shs - emds)/emds\n",
    "print(np.mean(errs), np.std(errs))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "6a7dbf61",
   "metadata": {},
   "outputs": [],
   "source": [
    "load_train_dataset = np.load('/data/sam/modelnet/data/train-datasets/item-2047-pairs-30.npz',allow_pickle=True)\n",
    "Ps = load_train_dataset['P']\n",
    "Qs = load_train_dataset['Q']\n",
    "dists = load_train_dataset['dists']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "d99fe30d",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<mpl_toolkits.mplot3d.art3d.Path3DCollection at 0x7f1db54766d0>"
      ]
     },
     "execution_count": 7,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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d911F/feey/33nsv3/3ud+OOsdvtfO1rX+PgwYNIksSWLVvYtWsXRqORz33uc7ztbW8jGAyyY8cOnn76aa655pr5HPKMkZ2dzXPPPae64J/5zGf453/+Z4aHh/n973/PZz/7WVwuF9deey27d+9Ou8glkQ6scOxbW1sxmUy43W7Wr18/bZup+UCyPbxer2fRokUsWrRITQX29PQwNjY2oZ+BEIKf/exnvPbaa+zZs2dO1Ypmw7ZL5b1nI+Z1T79ixQpefvllFi1axMDAAJdffjktLS1xxzz00EOqNBbAbbfdxuWXX87NN98cd9ztt9/O2rVrZ11RdSZhtVpVTYChoSGuvvpqVRNgphF7p9NJc3MzBQUFjI6OYjabKSsrm1BkM19QymOXLl2aUotuJRU4MjKCw+HgkUceQafT0d7ezt69e+ecvzAbbbvm5mbe//73q4G8HTt20NraetYH8ua9a61CCCkvL2doaGjCMansm5T6+dtvv30+hzvvKC4u5qMf/Sgf/ehHcTqdPP7443zjG9+gu7tb1QRIh5PvcDhoaWlh8+bNmM3mpEU2Ss/4+dD6S9fgYaLH8tJLL/HHP/4RnU7HBz/4QR566KE5HatOp+O+++5j586dKttuzZo1cWy7j370o3zoQx+itrZWZdsBrFmzhptuuonVq1ej0+n4yU9+siB9FuYbs17pr7jiCgYHByf8/lvf+ha33HILTqdT/V1BQcGEff0PfvAD/H4/d999NwDf+MY3MJvNfO5znwOiD9b111/Pzp07ueOOO6a9obMRo6OjqiZAa2srb3/727nhhhvYsmXLpBOA3W5XxTcmWx1j6cA6nU4tCJqLRgszMfhEPPzww/z617/miSeewGKx0NHRcUYq/94EOLtW+j//+c+T/q2srIyBgQHVvU+WwqmoqODll19WX/f29nL55Zerrz/+8Y9TV1d3zho8RHX8b775Zm6++Wa8Xi9PPfUUP/vZzzh27BiXXXYZN9xwAxdccIG6yig679M1ocjKymL58uUsX75cVQdubGxEkqRZ0YHnwuAfe+wxHnjgAZ544glVIuw8NfgFx7zu6e+8806KiorUQJ7dbud73/te3DF2u50tW7Zw+PBhADZv3syhQ4coLCzk7rvv5sSJEzz66KPnMgVzUvj9fp577jn27NnD4cOHufjiiykuLsblcvGtb31rxm5wIh1YmQBSoQOHw2GOHj3KkiVLZmzwjz/+OPfddx9PPPHEnKQW7XY7733ve+ns7KSqqorf/e53SXsdPvDAA3zzm98E4O6771Yltr70pS/x4IMP4nA45r3acBKcO+Qcm83GTTfdRHd3N8uWLeN3v/sdhYWFHDx4kJ/+9Kf8z//8DwC//OUv+fa3vw1Ev4CPfOQj9Pb2smTJElauXKm6o5/+9KfnRCnlbEQwGOTLX/4y//d//0dRURFbt27lhhtu4NJLL51VN59EOrAiXpGMWKQYfGVlZdKefqng6aef5gc/+AFPPvnknLEFP//5z1NYWKguLg6HI2mWaOvWrXFZokOHDlFQUMC+fftYtmwZdXV154XRI4SY6udNC5vNJq644gpRW1srrrjiCmG325Med//994va2lpRW1sr7r//fvX3X/ziF0VlZaXIyspaqCHPCj6fT9xyyy1ibGxMBINB8ec//1ncdtttYvXq1eJDH/qQ2LNnj7Db7cLj8cz4x+l0ira2NvHaa6+JF154QTQ0NIiBgQHhdruFy+USr7zyimhvb5/x+ffu3Su2bdsmRkZG5vSzqa+vF/39/UIIIfr7+0V9ff2EY37729+Kj3/84+rrj3/84+K3v/1t3DFn8FmYzg7n9OespeGeA7P7nCASifC3v/2NPXv28PLLL7N27Vp2797NFVdcMavqvUQ6cCgUoqKiYsZNR1555RW+/OUv8+STT854WzAZ8vPz1YCxEIKCgoK4ADJMHzCGKNfifFjpz1ra0969e9UA4C233MLll18+weifffZZrrzyStWNvPLKK3nmmWe4+eabectb3rLQQ54XaLVaLrvsMi677DJkWWb//v08+uijfPvb36auro53vvOdXHXVVWnXASh04KKiIo4cOUJRURF+v599+/alXWf/6quv8qUvfYknnnhixgY/VZYoFpIknbWKNguFs9bo54oDcC5Bo9Fw4YUXcuGFFyLLMocPH+bRRx/lBz/4AcuWLWP37t1cc801KdN0Y/fwymedqA48Xffd/fv38/nPf57HH398Vr355jtLdD7hTW30mdl95tBoNGzdupWtW7fyne98h2PHjrFnzx7e8Y53UFZWxu7du7nuuusmDaZFIpGkijtT0YETG1oeOnSIz372s/zxj3+Mm3znGop+3V133cUDDzzA7t27Jxyzc+dOvvjFL6o8keeee06tuDvvMM2m/02LcyB4c0Ygy7Jobm4WX/va18S2bdvElVdeKe677z7R2dmpBtxGR0fFX/7yF9HW1pZykM7tdov+/n5x9OhR8bWvfU1ceumlYvny5eLIkSPzfk9Wq1W8/e1vF7W1tWLHjh3CZrMJIYR4/fXXxUc/+lH1uP/93/8VNTU1oqamRvzyl79Uf3/nnXeKiooKIUmSqKioEF/96lfnfcwJyATyUsFsOQAKzmDw5oxDCEFbW5uqCWA2m9m5cyfPPPMMP/zhD6mvr5/ReY8dO8ZnPvMZtm7dyoEDB/jUpz7FrbfeOreDP7eQSdmlgnNgdn9TQfEAamtrxfbt28Wll14qvvvd74qWlhbhdrtTXvEPHTok1q1bJ5qamtRzB4PBM3hnZwUWdKU/a41+vvD000+L+vp6UVNTI77zne9M+Lvf7xc33XSTqKmpEdu3bxcdHR3q37797W+LmpoaUV9fL5555pkFHPXc4Ne//rV44IEHhCzLoqenR/zHf/yHuOyyy8SFF14ovvWtb4mmpqYpJ4CjR4+KdevWiaNHj57pWznbkDH6M4VwOCyqq6tFe3u7CAQCYv369aK5uTnumJ/85CfitttuE0II8dBDD4mbbrpJCCFEc3OzWL9+vfD7/eL06dOiurpahMPhBb+HuYYsy2JgYED85Cc/ETt27BDbtm0T99xzjzhy5EjcBKDc/6FDh+bs2rMhYHk8HnHttdeKFStWiNWrV4svfOELczaueUDG6M8UXnvtNXHVVVepr7/97W+Lb3/723HHXHXVVeK1114TQggRCoVEUVGRkGV5wrGxx51LGBkZEb/4xS/E1VdfLbZs2SLuvvtu8ac//Uls2LBB7Nu3b06vdeedd6re1ne+8x3x+c9/fsIxNptNLF++XNhsNmG328Xy5ctVZuKLL74ohBAiEAiISy65RDz11FNzOr45xIIa/flXxTIFMnpq06O4uJiPfexjPP300zz//PPU1tbyuc99jrvvvpsLLrhgTq+1d+9etSjmlltu4Y9//OOEY2IJWAUFBSoBy2Kx8La3vQ2ItjXbvHkzvb29czq+sxUZo89gxigoKOCWW27h5MmT3HjjjXN+/rkWYdmxY8ecj/FsxJuanLPQyOipLTzmm4AVDoe5+eab+cxnPpOp1x9HxuhjENu9tKKigocffpjf/va3ccco7K8LL7wwrnvprl27eP/738+//Mu/0N/fT2trK9u3bz9Dd3L2ICPCcgYwzab/vMOTTz4p6urqRHV1tfjmN78phBDiy1/+sti7d68QIlrieuONN4qamhqxbds20d7err73m9/8pqiurhb19fVv5qDRWYPPfe5zcYG8O++8c8IxNptNVFVVCbvdLux2u6iqqlI5G1/60pfEu971LhGJRBZ03DNAJnp/LmOmPACr1Souv/xykZWVJT71qU8t8KjPDGZDwOrp6RGAWLlypdiwYYPYsGGD+MUvfnFG7iMFLKjRn7U03LMRs+mr5vF4OHLkCE1NTTQ1NXHfffedwTvJYI6RaWt1ruLAgQPU1tZSXV2NwWDgfe97H3v37o07JjZNdeONN/LCCy8ghCArK4tLLrlkwfraZ3DuImP0C4jZ8AAyyGCukDH6DBYcs+lxCHD11VezYcMG1qxZwyc+8QkikchCDf2cQMboFxDp8ACAOB7AuQSlx2Frays7duxQW0LHQulxuH//fg4cOMDXvvY1dXL43e9+R0NDA01NTYyMjPDoo48u9C2c1cgY/QIilgcQDAZ5+OGH2bVrV9wxCg8AiOMBnEuYDb0Wos1BIDopBoPBc+7zmW9kyDkLiNn0VQOoqqpidHSUYDDIH//4R5577rm4yP/Zgrmg1+7cuZMDBw5wzTXXzAsF+FxGxugXGNdeey3XXntt3O++/vWvq/83mUyTuqudnZ3Tnv+ZZ57h9ttvJxKJ8LGPfYy77ror7u+BQIAPf/jDHDp0iKKiIh555BGqqqp4/vnnueuuuwgGgxgMBr7//e/z9re/Pf0bHMd802ufffZZ/H4/H/jAB3jxxRe58sorZzzW8w4LTQzI/MzfD6AF2oFqwAA0AKsTjvkk8NPx/78PeGT8/5uAxeP/Xwv0zeM4W4BF4/9fBLQkOeZm4Gcxr38G3JzkuA8D953pz/5s+sns6c8tbAfahBCnhRBB4GEgURp2N6CEwvcAOyRJkoQQR4QQ/eO/bwbMkiTNvr1tcjwO3DL+/1uAvUmOeRa4SpKkAkmSCoCrgGclScqWJGkRgCRJOuA64OQ8jfOcRMbozy1UAD0xr3vHf5f0GCFEGHABiemBdwOHhRCBeRrnvcCVkiS1AleMv0aSpK2SJP3P+NjswDeA18d/vj7+uyzgcUmSGoGjwDDw03ka5zmJzJ4+gzhIkrQG+C7RlXVeIISwAROK24UQB4GPxbz+JfDLhGOGgG3zNbbzAZmV/txCHxDbVaJy/HdJjxl3j/MA2/jrSuAPwIeFEO3zPtoMzggyRn9u4XWgTpKk5ZIkGYgG6h5POCZ2P30j8KIQQkiSlA88CdwlhHh1oQacwcIjY/TnEMb36J8mGgQ7AfxOCNEsSdLXJUlSWED/CxRJktQG/Aug5PQ+DdQCX5Ek6ej4z0TVigzOekxXWptBBhmcY8is9BlkcJ4hY/QZZHCeIWP0GWRwniFj9BlkcJ4hY/QZZHCeIWP0GWRwniFj9BlkcJ4hY/QZZHCe4f8DZ/dWD3gZ+moAAAAASUVORK5CYII=\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "\n",
    "fig = plt.figure()\n",
    "ax = fig.add_subplot(projection='3d')\n",
    "ax.scatter(P_test[0][:, 0], P_test[0][:, 1], P_test[0][:, 2])\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "b769b02d",
   "metadata": {},
   "outputs": [
    {
     "ename": "NameError",
     "evalue": "name 'Ps' is not defined",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mNameError\u001b[0m                                 Traceback (most recent call last)",
      "\u001b[0;32m<ipython-input-16-5e2a67d9ecab>\u001b[0m in \u001b[0;36m<module>\u001b[0;34m\u001b[0m\n\u001b[0;32m----> 1\u001b[0;31m \u001b[0mP_test\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mPs\u001b[0m\u001b[0;34m[\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;36m10\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mtolist\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m      2\u001b[0m \u001b[0mQ_test\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mQs\u001b[0m\u001b[0;34m[\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;36m10\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mtolist\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m      3\u001b[0m \u001b[0mdists_test\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mdists\u001b[0m\u001b[0;34m[\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;36m10\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;31mNameError\u001b[0m: name 'Ps' is not defined"
     ]
    }
   ],
   "source": [
    "P_test = Ps[:10].tolist()\n",
    "Q_test = Qs[:10].tolist()\n",
    "dists_test = dists[:10]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "2e8bbe7d",
   "metadata": {},
   "outputs": [],
   "source": [
    "# load non-final mlp models\n",
    "from train import *\n",
    "device = torch.device('cpu')\n",
    "in_dim = 3\n",
    "h_out = [20, 40, 60, 80]\n",
    "h_layers = [4, 6, 8]\n",
    "g_out = [1]\n",
    "g_layers = [1]\n",
    "models = []\n",
    "# (h_out, h_layers, g_out, g_layers)\n",
    "model_params = []\n",
    "for h in h_out:\n",
    "    for hl in h_layers:\n",
    "        for g in g_out:\n",
    "            for gl in g_layers:\n",
    "                model_params.append((h, hl, g, gl))\n",
    "                model_name = generate_name(in_dim, h, g, hl, gl, False, 'sigmoid')\n",
    "                PATH = '/data/sam/modelnet/models/siamese/1040/l2-{modelname}.pt'.format(modelname=model_name)\n",
    "                model = PointNetLearner(in_dim, h, g, num_h_layers=hl, \n",
    "                            num_g_layers=gl, final_mlp=False, activation='sigmoid')\n",
    "                model.load_state_dict(torch.load(PATH, map_location=device))\n",
    "                models.append(model)\n",
    " "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "e3497967",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "average 0.0006694185733795166 std 0.00018177421803932517\n",
      "average 0.0008451986312866211 std 0.00021643655808066496\n",
      "average 0.001092357635498047 std 0.0005520482242373716\n",
      "average 0.0007860660552978516 std 0.0002542233585496329\n",
      "average 0.001025512218475342 std 0.0003527588188312912\n",
      "average 0.001287081241607666 std 0.0005660571980324222\n",
      "average 0.000930091142654419 std 0.0009615860582381357\n",
      "average 0.001106271743774414 std 0.0002721490661376299\n",
      "average 0.0014520180225372314 std 0.00047259590631152585\n",
      "average 0.0007025980949401855 std 0.0002863868105240546\n",
      "average 0.0009335589408874512 std 0.00024123847052719995\n",
      "average 0.0012712597846984864 std 0.0004386428684069183\n"
     ]
    }
   ],
   "source": [
    "for model in models:\n",
    "    timing = []\n",
    "    for i in range(len(P_test)):\n",
    "        P = P_test[i].to(device)\n",
    "        Q = Q_test[i].to(device)\n",
    "        start = time.time()\n",
    "        pvec = model.get_embedding(P)\n",
    "        qvec = model.get_embedding(Q)\n",
    "        torch.norm(pvec-qvec)\n",
    "        end = time.time()\n",
    "        timing.append(end - start)\n",
    "    print(\"average\", np.mean(timing), \"std\", np.std(timing))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "aeec3f49",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 17%|█▋        | 2/12 [00:00<00:01,  7.36it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 0 Model parameters: (20, 4, 1, 1) Average error: 1312.9035469154426 std. dev: 874.4389084393688\n",
      "Model index: 1 Model parameters: (20, 6, 1, 1) Average error: 1295.4341726963573 std. dev: 862.7219594605077\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 33%|███▎      | 4/12 [00:00<00:01,  6.51it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 2 Model parameters: (20, 8, 1, 1) Average error: 1643.9623264904994 std. dev: 1094.7789899096672\n",
      "Model index: 3 Model parameters: (40, 4, 1, 1) Average error: 2423.2798212325647 std. dev: 1613.0156113317253\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 42%|████▏     | 5/12 [00:00<00:01,  6.04it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 4 Model parameters: (40, 6, 1, 1) Average error: 3138.464889802036 std. dev: 2089.3436981503783\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 58%|█████▊    | 7/12 [00:01<00:00,  5.48it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 5 Model parameters: (40, 8, 1, 1) Average error: 2748.874828248315 std. dev: 1830.0410079931794\n",
      "Model index: 6 Model parameters: (60, 4, 1, 1) Average error: 4047.331404933536 std. dev: 2694.0522998005035\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 67%|██████▋   | 8/12 [00:01<00:00,  5.26it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 7 Model parameters: (60, 6, 1, 1) Average error: 3771.7224485078314 std. dev: 2510.7823049956146\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 83%|████████▎ | 10/12 [00:01<00:00,  5.24it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 8 Model parameters: (60, 8, 1, 1) Average error: 4111.9979938610495 std. dev: 2737.3536293808856\n",
      "Model index: 9 Model parameters: (80, 4, 1, 1) Average error: 4493.819516578201 std. dev: 2989.8136995661616\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 92%|█████████▏| 11/12 [00:02<00:00,  5.17it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 10 Model parameters: (80, 6, 1, 1) Average error: 5207.884838015494 std. dev: 3466.545908565981\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 12/12 [00:02<00:00,  5.29it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 11 Model parameters: (80, 8, 1, 1) Average error: 5648.918974087292 std. dev: 3759.616432544752\n",
      "Lowest AVERAGE ERROR: 1\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "all_results = []\n",
    "average_errors = []\n",
    "for k in trange(len(models)):\n",
    "    model = models[k]\n",
    "    results = []\n",
    "    model.eval()\n",
    "    timing = []\n",
    "    for i in range(len(P_test)):\n",
    "        P = P_test[i].to(device)\n",
    "        Q = Q_test[i].to(device)\n",
    "        dist = dists_test[i]\n",
    "        pvec = model.get_embedding(P).detach().numpy()\n",
    "        qvec = model.get_embedding(Q).detach().numpy()\n",
    "        #print(np.linalg.norm(pvec - qvec, ord=1), dist)\n",
    "        results.append(abs(np.linalg.norm(pvec - qvec, ord=2)- dist)/dist)\n",
    "    all_results.append(results)\n",
    "    avg_error = np.average(results)\n",
    "    std = np.std(results)\n",
    "    average_errors.append(avg_error)\n",
    "    print(\"Model index:\", k ,\"Model parameters:\", model_params[k], \"Average error:\", avg_error, \"std. dev:\", std )\n",
    "print(\"Lowest AVERAGE ERROR:\", np.argmin(average_errors))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "9d2cd70e",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.collections.PathCollection at 0x7fe14df5daf0>"
      ]
     },
     "execution_count": 7,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "# Get scatter plot of distances\n",
    "best_index = 1\n",
    "best_result = all_results[best_index]\n",
    "plt.scatter(dists_test, best_result)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "bb6a601c",
   "metadata": {},
   "outputs": [],
   "source": [
    "# load final mlp models\n",
    "from train import *\n",
    "device = torch.device('cpu')\n",
    "in_dim = 3\n",
    "h_out = [20, 40, 60, 80]\n",
    "h_layers = [6, 8, 10]\n",
    "g_out = [10, 20, 30]\n",
    "g_layers = [2, 3]\n",
    "models = []\n",
    "# (h_out, h_layers, g_out, g_layers)\n",
    "model_params = []\n",
    "for h in h_out:\n",
    "    for hl in h_layers:\n",
    "        for g in g_out:\n",
    "            for gl in g_layers:\n",
    "                model_params.append((h, hl, g, gl))\n",
    "                model_name = generate_name(in_dim, h, g, hl, gl, True, 'sigmoid')\n",
    "                PATH = '/data/sam/modelnet/models/mlp/1040/{modelname}'.format(modelname=model_name)\n",
    "                model = PointNetLearner(in_dim, h, g, num_h_layers=hl, \n",
    "                            num_g_layers=gl, final_mlp=False, activation='sigmoid')\n",
    "                model.load_state_dict(torch.load(PATH, map_location=device))\n",
    "                models.append(model)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 96,
   "id": "cb336e13",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "average 0.09888561248779297 std 0.02563246031593887\n"
     ]
    }
   ],
   "source": [
    "timing = []\n",
    "for i in range(len(P_test)):\n",
    "    P = P_test[i].to(device)\n",
    "    Q = Q_test[i].to(device)\n",
    "    start = time.time()\n",
    "    estimate = model.get_full_result(P, Q)\n",
    "    end = time.time()\n",
    "    timing.append(end - start)\n",
    "print(\"average\", np.mean(timing), \"std\", np.std(timing))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "1fa0f3d0",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  3%|▎         | 2/72 [00:00<00:11,  6.04it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 0 Model parameters: (20, 6, 10, 2) Avg. error: 114.80745 std: 28.735064\n",
      "Model index: 1 Model parameters: (20, 6, 10, 3) Avg. error: 132.62729 std: 33.154636\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  6%|▌         | 4/72 [00:00<00:10,  6.30it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 2 Model parameters: (20, 6, 20, 2) Avg. error: 86.697 std: 21.903437\n",
      "Model index: 3 Model parameters: (20, 6, 20, 3) Avg. error: 97.03904 std: 24.274458\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  8%|▊         | 6/72 [00:00<00:10,  6.44it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 4 Model parameters: (20, 6, 30, 2) Avg. error: 85.9708 std: 21.585787\n",
      "Model index: 5 Model parameters: (20, 6, 30, 3) Avg. error: 103.93363 std: 25.97901\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 10%|▉         | 7/72 [00:01<00:10,  6.03it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 6 Model parameters: (20, 8, 10, 2) Avg. error: 115.29189 std: 28.810036\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 12%|█▎        | 9/72 [00:01<00:11,  5.52it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 7 Model parameters: (20, 8, 10, 3) Avg. error: 106.67257 std: 26.721926\n",
      "Model index: 8 Model parameters: (20, 8, 20, 2) Avg. error: 133.70795 std: 33.41798\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 15%|█▌        | 11/72 [00:01<00:11,  5.36it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 9 Model parameters: (20, 8, 20, 3) Avg. error: 89.93863 std: 22.513388\n",
      "Model index: 10 Model parameters: (20, 8, 30, 2) Avg. error: 79.99986 std: 20.072449\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 17%|█▋        | 12/72 [00:02<00:11,  5.04it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 11 Model parameters: (20, 8, 30, 3) Avg. error: 95.13563 std: 23.828493\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 18%|█▊        | 13/72 [00:02<00:12,  4.85it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 12 Model parameters: (20, 10, 10, 2) Avg. error: 80.407394 std: 20.195768\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 19%|█▉        | 14/72 [00:02<00:12,  4.83it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 13 Model parameters: (20, 10, 10, 3) Avg. error: 94.581894 std: 23.698448\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 21%|██        | 15/72 [00:02<00:11,  4.80it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 14 Model parameters: (20, 10, 20, 2) Avg. error: 87.95533 std: 21.161625\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 22%|██▏       | 16/72 [00:03<00:12,  4.47it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 15 Model parameters: (20, 10, 20, 3) Avg. error: 93.39752 std: 23.411129\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 24%|██▎       | 17/72 [00:03<00:12,  4.57it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 16 Model parameters: (20, 10, 30, 2) Avg. error: 95.36086 std: 23.476776\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 26%|██▋       | 19/72 [00:03<00:10,  4.83it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 17 Model parameters: (20, 10, 30, 3) Avg. error: 95.53795 std: 23.123384\n",
      "Model index: 18 Model parameters: (40, 6, 10, 2) Avg. error: 95.89502 std: 24.038311\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 28%|██▊       | 20/72 [00:03<00:10,  5.08it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 19 Model parameters: (40, 6, 10, 3) Avg. error: 101.64788 std: 25.465334\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 29%|██▉       | 21/72 [00:04<00:10,  4.87it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 20 Model parameters: (40, 6, 20, 2) Avg. error: 97.083275 std: 24.312544\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 32%|███▏      | 23/72 [00:04<00:09,  5.04it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 21 Model parameters: (40, 6, 20, 3) Avg. error: 122.39652 std: 30.612726\n",
      "Model index: 22 Model parameters: (40, 6, 30, 2) Avg. error: 138.68138 std: 34.653194\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 33%|███▎      | 24/72 [00:04<00:09,  5.16it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 23 Model parameters: (40, 6, 30, 3) Avg. error: 95.89923 std: 24.050539\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 35%|███▍      | 25/72 [00:04<00:09,  4.87it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 24 Model parameters: (40, 8, 10, 2) Avg. error: 123.3827 std: 30.857618\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 36%|███▌      | 26/72 [00:05<00:09,  4.72it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 25 Model parameters: (40, 8, 10, 3) Avg. error: 109.28987 std: 27.361391\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 38%|███▊      | 27/72 [00:05<00:09,  4.62it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 26 Model parameters: (40, 8, 20, 2) Avg. error: 103.428154 std: 25.907875\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 39%|███▉      | 28/72 [00:05<00:09,  4.57it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 27 Model parameters: (40, 8, 20, 3) Avg. error: 91.415955 std: 22.927008\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 40%|████      | 29/72 [00:05<00:09,  4.58it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 28 Model parameters: (40, 8, 30, 2) Avg. error: 129.94746 std: 32.496086\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 42%|████▏     | 30/72 [00:05<00:09,  4.55it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 29 Model parameters: (40, 8, 30, 3) Avg. error: 113.34894 std: 28.145622\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 43%|████▎     | 31/72 [00:06<00:09,  4.32it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 30 Model parameters: (40, 10, 10, 2) Avg. error: 113.411194 std: 28.384035\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 44%|████▍     | 32/72 [00:06<00:09,  4.11it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 31 Model parameters: (40, 10, 10, 3) Avg. error: 91.23796 std: 22.88286\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 46%|████▌     | 33/72 [00:06<00:09,  4.00it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 32 Model parameters: (40, 10, 20, 2) Avg. error: 99.648125 std: 24.971245\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 47%|████▋     | 34/72 [00:07<00:09,  3.92it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 33 Model parameters: (40, 10, 20, 3) Avg. error: 107.04618 std: 26.804771\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 49%|████▊     | 35/72 [00:07<00:09,  3.87it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 34 Model parameters: (40, 10, 30, 2) Avg. error: 105.39194 std: 26.382523\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 50%|█████     | 36/72 [00:07<00:09,  3.81it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 35 Model parameters: (40, 10, 30, 3) Avg. error: 95.786125 std: 24.013128\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 51%|█████▏    | 37/72 [00:07<00:08,  4.01it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 36 Model parameters: (60, 6, 10, 2) Avg. error: 100.46644 std: 25.172415\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 53%|█████▎    | 38/72 [00:07<00:08,  4.20it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 37 Model parameters: (60, 6, 10, 3) Avg. error: 107.12298 std: 26.82381\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 54%|█████▍    | 39/72 [00:08<00:07,  4.26it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 38 Model parameters: (60, 6, 20, 2) Avg. error: 116.09946 std: 29.050749\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 56%|█████▌    | 40/72 [00:08<00:07,  4.37it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 39 Model parameters: (60, 6, 20, 3) Avg. error: 95.17199 std: 23.858944\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 57%|█████▋    | 41/72 [00:08<00:06,  4.52it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 40 Model parameters: (60, 6, 30, 2) Avg. error: 87.36244 std: 21.921648\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 58%|█████▊    | 42/72 [00:08<00:06,  4.60it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 41 Model parameters: (60, 6, 30, 3) Avg. error: 97.132835 std: 24.333027\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 60%|█████▉    | 43/72 [00:09<00:06,  4.43it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 42 Model parameters: (60, 8, 10, 2) Avg. error: 118.944 std: 29.756445\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 61%|██████    | 44/72 [00:09<00:07,  3.96it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 43 Model parameters: (60, 8, 10, 3) Avg. error: 92.87919 std: 23.290123\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 62%|██████▎   | 45/72 [00:09<00:06,  3.92it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 44 Model parameters: (60, 8, 20, 2) Avg. error: 86.17198 std: 21.626171\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 64%|██████▍   | 46/72 [00:09<00:06,  3.86it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 45 Model parameters: (60, 8, 20, 3) Avg. error: 98.94232 std: 24.794302\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 65%|██████▌   | 47/72 [00:10<00:06,  3.90it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 46 Model parameters: (60, 8, 30, 2) Avg. error: 98.337006 std: 24.639782\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 67%|██████▋   | 48/72 [00:10<00:06,  3.87it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 47 Model parameters: (60, 8, 30, 3) Avg. error: 99.38616 std: 24.904415\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 68%|██████▊   | 49/72 [00:10<00:06,  3.71it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 48 Model parameters: (60, 10, 10, 2) Avg. error: 98.260155 std: 24.625069\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 69%|██████▉   | 50/72 [00:11<00:06,  3.52it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 49 Model parameters: (60, 10, 10, 3) Avg. error: 89.66974 std: 22.493904\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 71%|███████   | 51/72 [00:11<00:06,  3.46it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 50 Model parameters: (60, 10, 20, 2) Avg. error: 138.61201 std: 34.636017\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 72%|███████▏  | 52/72 [00:11<00:05,  3.40it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 51 Model parameters: (60, 10, 20, 3) Avg. error: 112.97326 std: 28.275185\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 74%|███████▎  | 53/72 [00:11<00:05,  3.32it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 52 Model parameters: (60, 10, 30, 2) Avg. error: 106.82963 std: 26.731457\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 76%|███████▋  | 55/72 [00:12<00:04,  3.67it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 53 Model parameters: (60, 10, 30, 3) Avg. error: 95.00542 std: 23.819464\n",
      "Model index: 54 Model parameters: (80, 6, 10, 2) Avg. error: 107.76524 std: 26.983147\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 78%|███████▊  | 56/72 [00:12<00:04,  3.92it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 55 Model parameters: (80, 6, 10, 3) Avg. error: 110.54507 std: 27.672783\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 79%|███████▉  | 57/72 [00:12<00:03,  4.16it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 56 Model parameters: (80, 6, 20, 2) Avg. error: 95.56986 std: 23.957521\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 81%|████████  | 58/72 [00:13<00:03,  4.34it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 57 Model parameters: (80, 6, 20, 3) Avg. error: 74.80466 std: 18.531816\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 82%|████████▏ | 59/72 [00:13<00:02,  4.35it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 58 Model parameters: (80, 6, 30, 2) Avg. error: 100.470566 std: 25.173437\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 83%|████████▎ | 60/72 [00:13<00:02,  4.42it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 59 Model parameters: (80, 6, 30, 3) Avg. error: 99.50318 std: 24.933947\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 85%|████████▍ | 61/72 [00:13<00:02,  4.29it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 60 Model parameters: (80, 8, 10, 2) Avg. error: 90.036896 std: 22.584827\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 86%|████████▌ | 62/72 [00:14<00:02,  4.16it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 61 Model parameters: (80, 8, 10, 3) Avg. error: 94.8699 std: 23.783953\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 88%|████████▊ | 63/72 [00:14<00:02,  4.08it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 62 Model parameters: (80, 8, 20, 2) Avg. error: 88.604744 std: 22.229692\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 89%|████████▉ | 64/72 [00:14<00:01,  4.00it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 63 Model parameters: (80, 8, 20, 3) Avg. error: 115.536514 std: 28.911093\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 90%|█████████ | 65/72 [00:14<00:01,  3.97it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 64 Model parameters: (80, 8, 30, 2) Avg. error: 94.4817 std: 23.686369\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 92%|█████████▏| 66/72 [00:15<00:01,  3.93it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 65 Model parameters: (80, 8, 30, 3) Avg. error: 94.3226 std: 23.647524\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 93%|█████████▎| 67/72 [00:15<00:01,  3.60it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 66 Model parameters: (80, 10, 10, 2) Avg. error: 79.24252 std: 19.907057\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 94%|█████████▍| 68/72 [00:15<00:01,  3.47it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 67 Model parameters: (80, 10, 10, 3) Avg. error: 96.65371 std: 24.22673\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 96%|█████████▌| 69/72 [00:16<00:00,  3.36it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 68 Model parameters: (80, 10, 20, 2) Avg. error: 115.80783 std: 28.978403\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 97%|█████████▋| 70/72 [00:16<00:00,  3.25it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 69 Model parameters: (80, 10, 20, 3) Avg. error: 94.117676 std: 23.597376\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 99%|█████████▊| 71/72 [00:16<00:00,  2.95it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 70 Model parameters: (80, 10, 30, 2) Avg. error: 112.463554 std: 28.148733\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 72/72 [00:17<00:00,  4.17it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 71 Model parameters: (80, 10, 30, 3) Avg. error: 99.658356 std: 24.971943\n",
      "Lowest AVERAGE ERROR: 57\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "all_results = []\n",
    "average_errors = []\n",
    "result_dictionary = {}\n",
    "for params in model_params:\n",
    "    hwidth = params[0]\n",
    "    hdepth = params[1]\n",
    "    result_dictionary[(hwidth, hdepth)] = []\n",
    "for k in trange(len(models)):\n",
    "    model = models[k]\n",
    "    results = []\n",
    "    model.eval()\n",
    "    for i in range(len(P_test)):\n",
    "        P = P_test[i].to(device)\n",
    "        Q = Q_test[i].to(device)\n",
    "        dist = dists_test[i]\n",
    "        estimate = model.get_full_result(P, Q).detach().cpu().numpy()\n",
    "        results.append(abs((1-estimate)- dist)/dist)\n",
    "    all_results.append(results)\n",
    "    avg_error = np.average(results)\n",
    "    std = np.std(results)\n",
    "    average_errors.append(avg_error)\n",
    "    result_dictionary[(model_params[k][0], model_params[k][1])].append((model_params[k][2], model_params[k][3], avg_error, std))\n",
    "    print(\"Model index:\", k ,\"Model parameters:\", model_params[k], \"Avg. error:\", avg_error, \"std:\", std)\n",
    "print(\"Lowest AVERAGE ERROR:\", np.argmin(average_errors))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "d8a43f56",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(200, 1)\n",
      "(200,)\n",
      "[[ 1.         -0.93315493]\n",
      " [-0.93315493  1.        ]]\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "# Get scatter plot of distances\n",
    "best_index = 57\n",
    "best_result = np.array(all_results[best_index])\n",
    "dists_test = np.array(dists_test)\n",
    "print(best_result.shape)\n",
    "print(dists_test.shape)\n",
    "plt.scatter(dists_test, best_result)\n",
    "r = np.corrcoef(dists_test,best_result[:, 0])\n",
    "print(r)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 99,
   "id": "f2b368e4",
   "metadata": {},
   "outputs": [],
   "source": [
    "# transform dataset for mlp\n",
    "def transform_pcd_for_mlp(Ps, Qs, max_points=200):\n",
    "    num_pcds = len(Ps)\n",
    "    dim = Ps[0].shape[1]\n",
    "    dataset = []\n",
    "    szs = []\n",
    "    #for i in range(len(Ps)):\n",
    "    #    szs.append(len(Ps[i]))\n",
    "    #print(np.max(szs))\n",
    "    \n",
    "    for i in range(num_pcds):\n",
    "        vec = np.zeros((2 * max_points, dim + 1))\n",
    "        P = Ps[i]\n",
    "        Q = Qs[i]\n",
    "        vec[:len(P), : dim] = P\n",
    "        vec[:len(P), dim] = 1\n",
    "\n",
    "        vec[:len(Q), : dim] = Q\n",
    "        vec[:len(Q), dim] = 1\n",
    "        dataset.append(vec)\n",
    "\n",
    "    return dataset\n",
    "vecs = transform_pcd_for_mlp(P_test, Q_test, max_points=5000)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 97,
   "id": "07813e73",
   "metadata": {},
   "outputs": [],
   "source": [
    "hidden_dim = [ 80]\n",
    "layers = [10]\n",
    "in_dim = 2*200*4\n",
    "models = []\n",
    "for h in hidden_dim:\n",
    "    for l in layers:\n",
    "        PATH = '/data/sam/modelnet/models/vanilla/1040/hidden-dim-{dim}-layers-{layer}'.format(dim=h, layer=l)\n",
    "        model = MLP(in_dim, h, l, activation='sigmoid')\n",
    "        model.load_state_dict(torch.load(PATH, map_location=device))\n",
    "        models.append(model)\n",
    "        "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 100,
   "id": "7a80f74e",
   "metadata": {},
   "outputs": [
    {
     "ename": "RuntimeError",
     "evalue": "mat1 and mat2 shapes cannot be multiplied (1x40000 and 1600x80)",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mRuntimeError\u001b[0m                              Traceback (most recent call last)",
      "\u001b[0;32m<ipython-input-100-3ccedf8ac8ec>\u001b[0m in \u001b[0;36m<module>\u001b[0;34m\u001b[0m\n\u001b[1;32m      4\u001b[0m     \u001b[0minput_vec\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mtorch\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mtensor\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mv\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mdtype\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0mtorch\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mfloat32\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mto\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mdevice\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m      5\u001b[0m     \u001b[0mstart\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mtime\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mtime\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m----> 6\u001b[0;31m     \u001b[0mout\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mmodel\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mnetwork\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0minput_vec\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m      7\u001b[0m     \u001b[0mend\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mtime\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mtime\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m      8\u001b[0m     \u001b[0mtiming\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mappend\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mend\u001b[0m \u001b[0;34m-\u001b[0m \u001b[0mstart\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m~/anaconda3/envs/mmetric_net/lib/python3.8/site-packages/torch/nn/modules/module.py\u001b[0m in \u001b[0;36m_call_impl\u001b[0;34m(self, *input, **kwargs)\u001b[0m\n\u001b[1;32m    887\u001b[0m             \u001b[0mresult\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0m_slow_forward\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m*\u001b[0m\u001b[0minput\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m**\u001b[0m\u001b[0mkwargs\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    888\u001b[0m         \u001b[0;32melse\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 889\u001b[0;31m             \u001b[0mresult\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mforward\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m*\u001b[0m\u001b[0minput\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m**\u001b[0m\u001b[0mkwargs\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    890\u001b[0m         for hook in itertools.chain(\n\u001b[1;32m    891\u001b[0m                 \u001b[0m_global_forward_hooks\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mvalues\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m~/anaconda3/envs/mmetric_net/lib/python3.8/site-packages/torch/nn/modules/container.py\u001b[0m in \u001b[0;36mforward\u001b[0;34m(self, input)\u001b[0m\n\u001b[1;32m    117\u001b[0m     \u001b[0;32mdef\u001b[0m \u001b[0mforward\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mself\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0minput\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    118\u001b[0m         \u001b[0;32mfor\u001b[0m \u001b[0mmodule\u001b[0m \u001b[0;32min\u001b[0m \u001b[0mself\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 119\u001b[0;31m             \u001b[0minput\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mmodule\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0minput\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    120\u001b[0m         \u001b[0;32mreturn\u001b[0m \u001b[0minput\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    121\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m~/anaconda3/envs/mmetric_net/lib/python3.8/site-packages/torch/nn/modules/module.py\u001b[0m in \u001b[0;36m_call_impl\u001b[0;34m(self, *input, **kwargs)\u001b[0m\n\u001b[1;32m    887\u001b[0m             \u001b[0mresult\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0m_slow_forward\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m*\u001b[0m\u001b[0minput\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m**\u001b[0m\u001b[0mkwargs\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    888\u001b[0m         \u001b[0;32melse\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 889\u001b[0;31m             \u001b[0mresult\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mforward\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m*\u001b[0m\u001b[0minput\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m**\u001b[0m\u001b[0mkwargs\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    890\u001b[0m         for hook in itertools.chain(\n\u001b[1;32m    891\u001b[0m                 \u001b[0m_global_forward_hooks\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mvalues\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m~/anaconda3/envs/mmetric_net/lib/python3.8/site-packages/torch/nn/modules/linear.py\u001b[0m in \u001b[0;36mforward\u001b[0;34m(self, input)\u001b[0m\n\u001b[1;32m     92\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     93\u001b[0m     \u001b[0;32mdef\u001b[0m \u001b[0mforward\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mself\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0minput\u001b[0m\u001b[0;34m:\u001b[0m \u001b[0mTensor\u001b[0m\u001b[0;34m)\u001b[0m \u001b[0;34m->\u001b[0m \u001b[0mTensor\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m---> 94\u001b[0;31m         \u001b[0;32mreturn\u001b[0m \u001b[0mF\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mlinear\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0minput\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mweight\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mbias\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m     95\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     96\u001b[0m     \u001b[0;32mdef\u001b[0m \u001b[0mextra_repr\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mself\u001b[0m\u001b[0;34m)\u001b[0m \u001b[0;34m->\u001b[0m \u001b[0mstr\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m~/anaconda3/envs/mmetric_net/lib/python3.8/site-packages/torch/nn/functional.py\u001b[0m in \u001b[0;36mlinear\u001b[0;34m(input, weight, bias)\u001b[0m\n\u001b[1;32m   1751\u001b[0m     \u001b[0;32mif\u001b[0m \u001b[0mhas_torch_function_variadic\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0minput\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mweight\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   1752\u001b[0m         \u001b[0;32mreturn\u001b[0m \u001b[0mhandle_torch_function\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mlinear\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m(\u001b[0m\u001b[0minput\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mweight\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0minput\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mweight\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mbias\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0mbias\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m-> 1753\u001b[0;31m     \u001b[0;32mreturn\u001b[0m \u001b[0mtorch\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0m_C\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0m_nn\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mlinear\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0minput\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mweight\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mbias\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m   1754\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   1755\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;31mRuntimeError\u001b[0m: mat1 and mat2 shapes cannot be multiplied (1x40000 and 1600x80)"
     ]
    }
   ],
   "source": [
    "timing = []\n",
    "for i in range(len(vecs)):\n",
    "    v = vecs[i].flatten()\n",
    "    input_vec = torch.tensor(v, dtype=torch.float32).to(device)\n",
    "    start = time.time()\n",
    "    out = model.network(input_vec)\n",
    "    end = time.time()\n",
    "    timing.append(end - start)\n",
    "print(np.mean(timing), np.std(timing))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 68,
   "id": "b7a61c8b",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 33%|███▎      | 4/12 [00:00<00:00, 16.51it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 0 Average error: 1.8346152 std. dev: 0.7003131\n",
      "Model index: 1 Average error: 2.3012414 std. dev: 0.8155965\n",
      "Model index: 2 Average error: 2.7071521 std. dev: 0.91587996\n",
      "Model index: 3 Average error: 1.0472242 std. dev: 0.5035044\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 67%|██████▋   | 8/12 [00:00<00:00, 16.38it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 4 Average error: 0.47676608 std. dev: 0.30767322\n",
      "Model index: 5 Average error: 0.9032716 std. dev: 0.46476486\n",
      "Model index: 6 Average error: 0.696738 std. dev: 0.39753202\n",
      "Model index: 7 Average error: 0.5957972 std. dev: 0.3592815\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 83%|████████▎ | 10/12 [00:00<00:00, 16.03it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 8 Average error: 0.72668356 std. dev: 0.4084772\n",
      "Model index: 9 Average error: 0.3816113 std. dev: 0.25858563\n",
      "Model index: 10 Average error: 0.41587842 std. dev: 0.27717087\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 12/12 [00:00<00:00, 15.61it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model index: 11 Average error: 0.28651744 std. dev: 0.19426703\n",
      "Lowest AVERAGE ERROR: 11\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "all_results = []\n",
    "average_errors = []\n",
    "for k in trange(len(models)):\n",
    "    model = models[k]\n",
    "    results = []\n",
    "    model.eval()\n",
    "    for i in range(len(vecs)):\n",
    "        v = vecs[i].flatten()\n",
    "        input_vec = torch.tensor(v, dtype=torch.float32).to(device)\n",
    "        out = model.network(input_vec).detach().numpy()\n",
    "        #print(np.linalg.norm(pvec - qvec, ord=1), dist)\n",
    "        dist = dists_test[i]\n",
    "        results.append(abs(out- dist)/dist)\n",
    "    all_results.append(results)\n",
    "    avg_error = np.average(results)\n",
    "    std = np.std(results)\n",
    "    average_errors.append(avg_error)\n",
    "    print(\"Model index:\", k, \"Average error:\", avg_error, \"std. dev:\", std )\n",
    "print(\"Lowest AVERAGE ERROR:\", np.argmin(average_errors))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "152b5093",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "local-venv",
   "language": "python",
   "name": "local-venv"
  },
  "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.8.10"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
