{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Fully Steerable 3D Spherical Neurons"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "%load_ext autoreload\n",
    "%autoreload 2\n",
    "%matplotlib inline"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "SEED = 239\n",
    "\n",
    "import os \n",
    "\n",
    "import numpy as np\n",
    "np.random.seed(SEED)\n",
    "np.set_printoptions(precision=4)\n",
    "\n",
    "import torch\n",
    "torch.manual_seed(SEED)\n",
    "if torch.cuda.is_available():\n",
    "    torch.backends.cudnn.deterministic = True\n",
    "    torch.backends.cudnn.benchmark = False\n",
    "import torch.nn as nn\n",
    "import torch.nn.functional as F \n",
    "import torch.optim as optim\n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "import matplotlib.collections as mcol\n",
    "import matplotlib.transforms as mtransforms\n",
    "\n",
    "from matplotlib import cm\n",
    "from mpl_toolkits.mplot3d import Axes3D\n",
    "\n",
    "from scipy.spatial.transform import Rotation as R\n",
    "from scipy.special import softmax\n",
    "\n",
    "from sklearn.metrics import confusion_matrix\n",
    "\n",
    "from tqdm import tqdm\n",
    "\n",
    "from models import PointCMLP, SteerableModel\n",
    "\n",
    "from utils import (plot_confusion_matrix,\n",
    "                   get_tetris_data, plot_shapes, \n",
    "                   visualize_skeleton, get_edges,\n",
    "                   score, build_mlgp, \n",
    "                   construct_filter_banks, unembed_points, \n",
    "                   transform_parameters, build_steerable_model,\n",
    "                   random_axis_angle, torch_rotation_matrix, entropy)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "('1.8.1+cu102', device(type='cuda'))"
      ]
     },
     "execution_count": 3,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "device = torch.device(\"cuda\" if torch.cuda.is_available() else \"cpu\")\n",
    "torch.__version__, device"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Creating a steerable spherical classifer\n",
    "in the following steps:\n",
    "\n",
    "**Step 1**. Train the ancestor MLGP.\n",
    "\n",
    "**Step 2**. Transform the hidden unit parameters into filter banks.\n",
    "\n",
    "**Step 3**. Fix the learned parameters and add the interpolation coefficients $v^k$ as learnable parameters to fulfill the steerability constraint $\\rightarrow$ Steerable spherical classifier."
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Step 1"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### Load the UTKinect skeleton data"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "data_path = 'utkinect_skeletons_data/'\n",
    "\n",
    "action_to_label = {\n",
    "    'sitDown': 0,\n",
    "    'throw': 1,\n",
    "    'carry': 2,\n",
    "    'push': 3,\n",
    "    'waveHands': 4,\n",
    "    'walk': 5,\n",
    "    'clapHands': 6,\n",
    "    'pull': 7,\n",
    "    'pickUp': 8,\n",
    "    'standUp': 9\n",
    "}\n",
    "\n",
    "Xtrain =  np.load(os.path.join(data_path, 'Xtrain.npy'))\n",
    "Ytrain =  np.load(os.path.join(data_path, 'Ytrain.npy'))\n",
    "Xval =  np.load(os.path.join(data_path, 'Xval.npy'))\n",
    "Yval =  np.load(os.path.join(data_path, 'Yval.npy'))\n",
    "Xtest  =  np.load(os.path.join(data_path, 'Xtest.npy'))\n",
    "Ytest  =  np.load(os.path.join(data_path, 'Ytest.npy'))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "# shuffle the data:\n",
    "idcs = np.arange(len(Xtrain))\n",
    "np.random.shuffle(idcs)\n",
    "Xtrain = Xtrain[idcs]\n",
    "Ytrain = Ytrain[idcs]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "Text(0.5, 0.92, 'skeleton for action \"walk\" (5)')"
      ]
     },
     "execution_count": 6,
     "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": [
    "# sanity check:\n",
    "idx = np.random.choice(len(Xtrain))\n",
    "skeleton = Xtrain[idx]\n",
    "edges = get_edges(skeleton)\n",
    "visualize_skeleton(skeleton.T, edges, size=100, show_grid=True, azim=0)\n",
    "plt.title(f'skeleton for action \"{list(action_to_label.keys())[np.argmax(list(action_to_label.values()) == Ytrain[idx])]}\" ({Ytrain[idx]})')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Xtrain.shape, Ytrain.shape: torch.Size([2295, 20, 3]) torch.Size([2295])\n",
      "Xval.shape, Yval.shape: torch.Size([670, 20, 3]) torch.Size([670])\n",
      "Xtest.shape, Ytest.shape: torch.Size([3062, 20, 3]) torch.Size([3062])\n"
     ]
    }
   ],
   "source": [
    "Xtrain = torch.from_numpy(Xtrain).float().to(device)\n",
    "Ytrain = torch.from_numpy(Ytrain).to(device)\n",
    "\n",
    "Xval = torch.from_numpy(Xval).float().to(device)\n",
    "Yval = torch.from_numpy(Yval).to(device)\n",
    "\n",
    "Xtest = torch.from_numpy(Xtest).float().to(device)\n",
    "Ytest = torch.from_numpy(Ytest).to(device)\n",
    "\n",
    "print('Xtrain.shape, Ytrain.shape:', Xtrain.shape, Ytrain.shape)\n",
    "print('Xval.shape, Yval.shape:',     Xval.shape, Yval.shape)\n",
    "print('Xtest.shape, Ytest.shape:',   Xtest.shape, Ytest.shape)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### Train the MLGP on the shapes in the canonical orientation"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "N_GEOMETRIC_NEURONS = 12\n",
    "OUTPUT_DIM = 10 # len(set(Ytrain.numpy()))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "model: MLGP\n",
      "PointCMLP(\n",
      "  (hidden_layers): ModuleList(\n",
      "    (0): Linear(in_features=100, out_features=12, bias=False)\n",
      "  )\n",
      "  (out_layer): Linear(in_features=14, out_features=10, bias=False)\n",
      ")\n",
      "total number of trainable parameters: 1340\n",
      "\n",
      "epoch: 0, loss: 2.313,  acc:  0.156, val_loss: 2.302,  val_acc:  0.152\n",
      "epoch: 500, loss: 0.839,  acc:  0.712, val_loss: 0.865,  val_acc:  0.696\n",
      "epoch: 1000, loss: 0.580,  acc:  0.782, val_loss: 0.623,  val_acc:  0.754\n",
      "epoch: 1500, loss: 0.460,  acc:  0.831, val_loss: 0.519,  val_acc:  0.810\n",
      "epoch: 2000, loss: 0.382,  acc:  0.866, val_loss: 0.448,  val_acc:  0.848\n",
      "epoch: 2500, loss: 0.319,  acc:  0.892, val_loss: 0.389,  val_acc:  0.867\n",
      "epoch: 3000, loss: 0.266,  acc:  0.913, val_loss: 0.340,  val_acc:  0.881\n",
      "epoch: 3500, loss: 0.222,  acc:  0.928, val_loss: 0.306,  val_acc:  0.897\n",
      "epoch: 4000, loss: 0.186,  acc:  0.941, val_loss: 0.282,  val_acc:  0.904\n",
      "epoch: 4500, loss: 0.157,  acc:  0.952, val_loss: 0.265,  val_acc:  0.904\n",
      "epoch: 5000, loss: 0.133,  acc:  0.961, val_loss: 0.257,  val_acc:  0.907\n",
      "epoch: 5500, loss: 0.112,  acc:  0.969, val_loss: 0.255,  val_acc:  0.918\n",
      "epoch: 6000, loss: 0.095,  acc:  0.973, val_loss: 0.255,  val_acc:  0.913\n",
      "epoch: 6500, loss: 0.080,  acc:  0.979, val_loss: 0.257,  val_acc:  0.919\n",
      "epoch: 7000, loss: 0.068,  acc:  0.983, val_loss: 0.261,  val_acc:  0.919\n",
      "epoch: 7500, loss: 0.058,  acc:  0.986, val_loss: 0.267,  val_acc:  0.921\n",
      "epoch: 8000, loss: 0.049,  acc:  0.990, val_loss: 0.276,  val_acc:  0.924\n",
      "epoch: 8500, loss: 0.042,  acc:  0.991, val_loss: 0.289,  val_acc:  0.925\n",
      "epoch: 9000, loss: 0.036,  acc:  0.994, val_loss: 0.306,  val_acc:  0.928\n",
      "epoch: 9500, loss: 0.031,  acc:  0.996, val_loss: 0.327,  val_acc:  0.924\n",
      "epoch: 9999, loss: 0.026,  acc:  0.996, val_loss: 0.327,  val_acc:  0.924\n"
     ]
    }
   ],
   "source": [
    "# set the seed here:\n",
    "torch.manual_seed(SEED)\n",
    "\n",
    "# instantiate the model:\n",
    "model = build_mlgp(input_shape=Xtrain.shape[1:], output_dim=OUTPUT_DIM, hidden_layer_sizes=[N_GEOMETRIC_NEURONS], bias=False)\n",
    "\n",
    "print(model)\n",
    "print('total number of trainable parameters:', sum([np.prod(p.size()) for p in filter(lambda p: p.requires_grad, model.parameters())]))\n",
    "print()\n",
    "\n",
    "model = model.float().to(device)\n",
    "\n",
    "# define the loss and optimizer:\n",
    "criterion = nn.CrossEntropyLoss()\n",
    "optimizer = optim.Adam(model.parameters(), lr=1e-3)\n",
    "\n",
    "epochs = 10000\n",
    "\n",
    "# train the model:\n",
    "for i in range(epochs): \n",
    "        y_pred = model(Xtrain)\n",
    "        loss = criterion(y_pred, Ytrain)\n",
    "        optimizer.zero_grad()\n",
    "        loss.backward()\n",
    "        optimizer.step()\n",
    "        acc = score(y_pred.detach(), Ytrain)\n",
    "\n",
    "        if i % 500 == 0:\n",
    "            y_val_pred = model(Xval)\n",
    "            val_loss = criterion(y_val_pred, Yval)\n",
    "            val_acc = score(y_val_pred.detach(), Yval)\n",
    "            \n",
    "            print('epoch: %d, loss: %.3f,  acc:  %.3f, val_loss: %.3f,  val_acc:  %.3f' % (i, loss.item(), acc, val_loss.item(), val_acc))\n",
    "\n",
    "print('epoch: %d, loss: %.3f,  acc:  %.3f, val_loss: %.3f,  val_acc:  %.3f' % (i, loss.item(), acc, val_loss.item(), val_acc))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### Test the model on the test shape data"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "test_acc:  0.92880\n"
     ]
    }
   ],
   "source": [
    "y_test_pred = model(Xtest)\n",
    "\n",
    "test_acc = score(y_test_pred.detach(), Ytest)\n",
    "print('test_acc:  %.5f' % test_acc)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Confusion matrix, without normalization\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 2 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "targets = Ytest.cpu().numpy()\n",
    "predictions = torch.argmax(y_test_pred.detach(), axis=1).cpu().numpy()\n",
    "cm = confusion_matrix(targets, predictions)\n",
    "plot_confusion_matrix(cm, classes=list(action_to_label.keys()))#, normalize=True)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Step 2"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "The main function in this step is `construct_filter_banks`:\n",
    "- takes in *non-normalized* (raw) spheres $\\tilde{S}_k \\in \\mathbb{R}^5$, i.e., parameters, extracted from the trained ancestor MLGP model;\n",
    "- computes the initial rotations $R_O^k$, i.e., from the original sphere centers $c_k$ to (1,1,1);\n",
    "- computes the tetrahedron rotations $R_{Ti}$, i.e., rotations transforming (1, 1, 1) into the other three vertices;\n",
    "- constructs the filter banks for each sphere as discussed in Section 4.1.2 in the paper\n",
    "\n",
    "\\begin{equation*}\n",
    "\tB(\\tilde{S}_k) = \n",
    "\t\\begin{bmatrix}\n",
    "\t\tR_O^{k\\top} R_{T0} R_O^{k} \\tilde{S}_k \\\\\n",
    "\t\tR_O^{k\\top} R_{T1} R_O^{k} \\tilde{S}_k \\\\\n",
    "\t\tR_O^{k\\top} R_{T2} R_O^{k} \\tilde{S}_k \\\\ \n",
    "\t\tR_O^{k\\top} R_{T3} R_O^{k} \\tilde{S}_k \\\\\n",
    "\t\\end{bmatrix} ~;\n",
    "\\end{equation*}\n",
    "\n",
    "- returns an array of $R_O^k$ (optionally) and an array of the constructed $B(\\tilde{S}_k)$.\n",
    "\n",
    "To demonstrate how it works, we first extract the ancestor model hidden layer (i.e., geometric neurons) spheres, and then take one of them to form a filter bank:"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "hidden_spheres_numpy.shape: (12, 20, 5)\n",
      "\n",
      "S_tilde_k =  [-2.5476  0.6094  0.0804 -0.1381 -0.3713]\n",
      "\n",
      "R_O^k =\n",
      " [[[ 0.4189 -0.6476 -0.6365]\n",
      "  [ 0.7434  0.6471 -0.1692]\n",
      "  [ 0.5214 -0.4023  0.7525]]] \n",
      "\n",
      "B(S_tilde_k) =\n",
      " [[-2.5476  0.6094  0.0804 -0.1381 -0.3713]\n",
      " [ 1.2799  1.3504  1.8456 -0.1381 -0.3713]\n",
      " [ 0.298  -2.5677  0.4316 -0.1381 -0.3713]\n",
      " [ 0.9696  0.6079 -2.3576 -0.1381 -0.3713]]\n"
     ]
    }
   ],
   "source": [
    "# extract the spheres from the ancestor model:\n",
    "original_state_dict = model.state_dict()\n",
    "\n",
    "# get the geometric neuron spheres:\n",
    "hidden_name = 'hidden_layers.0.weight' \n",
    "hidden_spheres = original_state_dict[hidden_name]            # (n_geometric_neurons x N_points*5)\n",
    "\n",
    "# each sphere is a parameter vector of length 5;\n",
    "# each geometric neuron contains a number of spheres corresponding to the number of input points\n",
    "# in the point set\n",
    "\n",
    "# reshape to (n_geometric_neurons x N_points x 5):\n",
    "hidden_spheres_numpy = hidden_spheres.detach().cpu().numpy().reshape(len(hidden_spheres), -1, 5)\n",
    "print('hidden_spheres_numpy.shape:', hidden_spheres_numpy.shape) \n",
    "\n",
    "\n",
    "# e.g., select the third sphere from the second geometric neuron:\n",
    "one_sphere = hidden_spheres_numpy[1,2,:]\n",
    "print('\\nS_tilde_k = ', one_sphere)\n",
    "\n",
    "# construct a filter bank for this sphere:\n",
    "init_rotation, filter_bank = construct_filter_banks(one_sphere, return_init_rotations=True)\n",
    "print('\\nR_O^k =\\n', init_rotation, '\\n\\nB(S_tilde_k) =\\n', filter_bank)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### Visualize\n",
    "\n",
    "We normalize, i.e., *unembed*, the resulting spheres to get their Euclidean $\\mathbb{R}^3$ representation.\n",
    "\n",
    "By unembedding the 5-vectors, we will get the first three elements that represent the Euclidean coordinates of the sphere center:"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "the four sphere centers:\n",
      " [[ 6.8618 -1.6415 -0.2167]\n",
      " [-3.4475 -3.6371 -4.9711]\n",
      " [-0.8027  6.9159 -1.1624]\n",
      " [-2.6116 -1.6373  6.3501]]\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 504x504 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "# the centers of the filter bank spheres:\n",
    "centers = unembed_points(filter_bank)\n",
    "\n",
    "print('\\nthe four sphere centers:\\n', centers)\n",
    "fig = plt.figure(1, figsize=(7,7))\n",
    "ax = fig.add_subplot(111, projection='3d')         \n",
    "ax.scatter(centers[:, 0], centers[:, 1], centers[:, 2], s=100)\n",
    "plt.title('A regular tetrahedron formed by the centers of the $B(S_k)$ spheres')\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "**Step 2** is wrapped into the function `transform_parameters`: \n",
    "- takes in the trained ancestor MLGP model;\n",
    "\n",
    "- transforms its parameters --- uses the hidden layer spheres to create the filter banks and keeps the output (classification) layer the same;\n",
    "\n",
    "- returns the initial rotations $R_O^k$, the filter banks $B(\\tilde{S}_k)$, and the ancestor model output layer parameters."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [],
   "source": [
    "transformed_parameters = transform_parameters(model) # used in the experiments further down"
   ]
  },
  {
   "cell_type": "raw",
   "metadata": {},
   "source": []
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Step 3\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "**Step 3** is wrapped into the `build_steerable_model` function: \n",
    "- creates a steerable model with learnable interpolation coefficients $v^k$ according to the constraint (13) in the paper;\n",
    "\n",
    "   - since $v^k$ depends on the (possibly unknown) rotation $R_B$ applied to the input (see (12) in the paper), we will learn the parameters of this rotation instead, i.e., three axis-angle parameters, if they are unknown (as discussed in the paper in Section 5.2);\n",
    "\n",
    "- the rest of the model parameters are set to be the ones obtained in Step 2 (the filter banks and the unchanged output layer) and are fixed (**not** updated)."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "SteerableModel(\n",
      "  (hidden_layers): ModuleList(\n",
      "    (0): Linear(in_features=400, out_features=12, bias=False)\n",
      "  )\n",
      "  (out_layer): Linear(in_features=14, out_features=10, bias=False)\n",
      ")\n",
      "total number of trainable parameters: 3\n",
      "\n",
      "init_axis_angle:\n",
      " [ 0.2288  0.0725 -0.2987]\n",
      "\n",
      "steerable_model.axis_angle:\n",
      " Parameter containing:\n",
      "tensor([ 0.2288,  0.0725, -0.2987], device='cuda:0', dtype=torch.float64,\n",
      "       requires_grad=True)\n"
     ]
    }
   ],
   "source": [
    "# choose initial model parameters:\n",
    "init_axis_angle = random_axis_angle()\n",
    "\n",
    "# use the initial parameters and the transformed ancestor model parameters obtained in Step 2\n",
    "# to build a steerable spherical classifier:\n",
    "steerable_model = build_steerable_model(input_shape=Xtrain.shape[1:], \n",
    "                                        output_dim=OUTPUT_DIM,\n",
    "                                        hidden_layer_sizes=[N_GEOMETRIC_NEURONS],\n",
    "                                        init_axis_angle=init_axis_angle,\n",
    "                                        transformed_parameters=transformed_parameters,\n",
    "                                        print_hidden_layer_output=False).to(device)\n",
    "print(steerable_model)\n",
    "print('total number of trainable parameters:', \\\n",
    "      sum([np.prod(p.size()) for p in filter(lambda p: p.requires_grad, steerable_model.parameters())]))\n",
    "\n",
    "print('\\ninit_axis_angle:\\n', init_axis_angle)\n",
    "print('\\nsteerable_model.axis_angle:\\n', steerable_model.axis_angle)"
   ]
  },
  {
   "cell_type": "raw",
   "metadata": {},
   "source": []
  },
  {
   "cell_type": "raw",
   "metadata": {},
   "source": []
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Paper experiments"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### Get the ancestor MLGP activations\n",
    "(used in the experiments further down)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "metadata": {},
   "outputs": [],
   "source": [
    "activation = {}\n",
    "def get_activation(name):\n",
    "    def hook(model, input, output):\n",
    "        activation[name] = output.detach()\n",
    "    return hook\n",
    "\n",
    "model.hidden_layers[0].register_forward_hook(get_activation('hidden_layer'))\n",
    "output = model(Xtest)\n",
    "\n",
    "# the ground truth hidden activations -- the ancestor MLGP hidden layer output:\n",
    "gt_hidden_activations = activation['hidden_layer'].detach().cpu().numpy()\n",
    "\n",
    "# the ground truth output activations -- the ancestor MLGP model output: \n",
    "gt_outs = output.detach().cpu().numpy()\n",
    "\n",
    "# gt_hidden_activations, gt_outs"
   ]
  },
  {
   "cell_type": "raw",
   "metadata": {},
   "source": []
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### The known rotation experiment (Section 5.3)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "metadata": {
    "scrolled": false
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "distortion: 0.0\n",
      "\n",
      "ancestor_acc:     0.2515 +/- 0.2309\n",
      "acc:              0.9288 +/- 0.0000\n",
      "\n",
      "ancestor L1 dist: 52.5823 +/- 30.5734\n",
      "L1 dist:          0.0000 +/- 0.0000\n",
      "\n",
      "\n",
      "\n",
      "distortion: 0.005\n",
      "\n",
      "ancestor_acc:     0.2461 +/- 0.2208\n",
      "acc:              0.9238 +/- 0.0021\n",
      "\n",
      "ancestor L1 dist: 51.8241 +/- 29.3599\n",
      "L1 dist:          0.5296 +/- 0.0023\n",
      "\n",
      "\n",
      "\n",
      "distortion: 0.01\n",
      "\n",
      "ancestor_acc:     0.2435 +/- 0.2072\n",
      "acc:              0.9110 +/- 0.0031\n",
      "\n",
      "ancestor L1 dist: 50.4696 +/- 28.4650\n",
      "L1 dist:          1.0595 +/- 0.0045\n",
      "\n",
      "\n",
      "\n",
      "distortion: 0.02\n",
      "\n",
      "ancestor_acc:     0.2349 +/- 0.2057\n",
      "acc:              0.8714 +/- 0.0047\n",
      "\n",
      "ancestor L1 dist: 52.6936 +/- 29.5572\n",
      "L1 dist:          2.1188 +/- 0.0094\n",
      "\n",
      "\n",
      "\n",
      "distortion: 0.03\n",
      "\n",
      "ancestor_acc:     0.2431 +/- 0.2029\n",
      "acc:              0.8225 +/- 0.0056\n",
      "\n",
      "ancestor L1 dist: 51.0154 +/- 29.4376\n",
      "L1 dist:          3.1791 +/- 0.0141\n",
      "\n",
      "\n",
      "\n",
      "distortion: 0.05\n",
      "\n",
      "ancestor_acc:     0.2275 +/- 0.1752\n",
      "acc:              0.7196 +/- 0.0070\n",
      "\n",
      "ancestor L1 dist: 51.2326 +/- 29.0233\n",
      "L1 dist:          5.3041 +/- 0.0221\n",
      "\n",
      "\n"
     ]
    }
   ],
   "source": [
    "np.random.seed(SEED)\n",
    "torch.manual_seed(SEED)\n",
    "\n",
    "n_trials = 1000\n",
    "\n",
    "# the parameter of additive uniform noise to apply to the rotated shapes:\n",
    "distortions = [0.0, 0.005, 0.01, 0.02, 0.03, 0.05]\n",
    "\n",
    "\n",
    "init_axis_angles = []\n",
    "accs = dict()          # classification accuracies for the perturbed rotated shapes\n",
    "dists = dict()         # L1 distances to the ground truth hidden activations\n",
    "\n",
    "# the same for the ancestor:\n",
    "ancestor_accs = dict() \n",
    "ancestor_dists = dict()\n",
    "\n",
    "\n",
    "for distortion in distortions:\n",
    "    accs[distortion] = []  \n",
    "    dists[distortion] = [] \n",
    "    ancestor_accs[distortion] = [] \n",
    "    ancestor_dists[distortion] = [] \n",
    "    \n",
    "    print('\\ndistortion:', distortion)\n",
    "    \n",
    "    for n in range(n_trials):       \n",
    "        # construct a random ground truth rotation:\n",
    "        init_axis_angle = random_axis_angle()\n",
    "        init_axis_angles.append(init_axis_angle)\n",
    "        gt_rotation = torch_rotation_matrix(init_axis_angle).float().to(device)\n",
    "\n",
    "        # rotate the shapes with the ground truth:\n",
    "        test_data = Xtest.reshape(-1, 3) @ gt_rotation.T\n",
    "        test_data = test_data.reshape(Xtest.shape)\n",
    "        test_label = Ytest\n",
    "\n",
    "        # add uniform noise to the transformed shapes:\n",
    "        noise = distortion * (2 * torch.rand(test_data.shape).to(device) - 1)\n",
    "        test_data += noise\n",
    "\n",
    "        # construct the steerable model with the initial axis-angle parameters:\n",
    "        steerable_model = build_steerable_model(input_shape=test_data.shape[1:], \n",
    "                                                output_dim=OUTPUT_DIM,\n",
    "                                                hidden_layer_sizes=[N_GEOMETRIC_NEURONS],\n",
    "                                                init_axis_angle=init_axis_angle,\n",
    "                                                transformed_parameters=transformed_parameters,\n",
    "                                                print_hidden_layer_output=False).float().to(device)\n",
    "\n",
    "        # get the model output:\n",
    "        output = steerable_model(test_data)\n",
    "        ancestor_output = model(test_data)\n",
    "        \n",
    "        # compute the model accuracy for the perturbed rotated shapes:\n",
    "        acc = score(output.detach(), test_label)\n",
    "        accs[distortion].append(acc)\n",
    "        \n",
    "        ancestor_acc = score(ancestor_output.detach(), test_label)\n",
    "        ancestor_accs[distortion].append(ancestor_acc)\n",
    "        \n",
    "        # compute the L1 distance between the hidden activations:\n",
    "        hidden_activations = steerable_model.hidden_layer_activations.cpu().numpy()\n",
    "        dist = np.linalg.norm(hidden_activations - gt_hidden_activations, ord=1, axis=1)\n",
    "        dist = np.mean(dist)\n",
    "        dists[distortion].append(dist)\n",
    "        \n",
    "        ancestor_hidden_activations = model.hidden_layer_activations.cpu().numpy()\n",
    "        ancestor_dist = np.linalg.norm(ancestor_hidden_activations - gt_hidden_activations, ord=1, axis=1)\n",
    "        ancestor_dist = np.mean(ancestor_dist)\n",
    "        ancestor_dists[distortion].append(ancestor_dist)\n",
    "\n",
    "        # if n % 10 == 0:\n",
    "            # print('\\nexperiment #%d/%d' % (n+1, n_trials))\n",
    "            # print('\\nadditive_uniform_noise:\\n', noise)\n",
    "            # print('\\ngt_rotation:\\n', gt_rotation)\n",
    "            # print('\\nacc:  %.3f' % acc)\n",
    "            \n",
    "    print()\n",
    "    print('ancestor_acc:     %.4f +/- %.4f' % (np.mean(ancestor_accs[distortion]), np.std(ancestor_accs[distortion])))\n",
    "    print('acc:              %.4f +/- %.4f' % (np.mean(accs[distortion]), np.std(accs[distortion])))\n",
    "    print()\n",
    "    print('ancestor L1 dist: %.4f +/- %.4f' % (np.mean(ancestor_dists[distortion]), np.std(ancestor_dists[distortion])))\n",
    "    print('L1 dist:          %.4f +/- %.4f' % (np.mean(dists[distortion]), np.std(dists[distortion])))\n",
    "    print(end='\\n\\n')"
   ]
  },
  {
   "cell_type": "raw",
   "metadata": {},
   "source": []
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### The adaptive rotation experiment (Section 5.4)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "metadata": {
    "scrolled": false
   },
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      "  0%|          | 0/1000 [00:00<?, ?it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "\n",
      "\n",
      "distortion = 0 +/- 10 degrees\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 1000/1000 [1:01:31<00:00,  3.69s/it]\n",
      "  0%|          | 0/1000 [00:00<?, ?it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "mean_init_acc: 0.7680, mean_final_acc: 0.7690\n",
      "\n",
      "mean_init_dist: 0.1828 +/- 0.3311, mean_final_dist: 0.1958 +/- 0.3627\n",
      "\n",
      "\n",
      "\n",
      "distortion = 5 +/- 10 degrees\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 1000/1000 [1:01:47<00:00,  3.71s/it]\n",
      "  0%|          | 0/1000 [00:00<?, ?it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "mean_init_acc: 0.8000, mean_final_acc: 0.8030\n",
      "\n",
      "mean_init_dist: 0.1719 +/- 0.3303, mean_final_dist: 0.1791 +/- 0.3534\n",
      "\n",
      "\n",
      "\n",
      "distortion = 10 +/- 10 degrees\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 1000/1000 [1:01:07<00:00,  3.67s/it]\n",
      "  0%|          | 0/1000 [00:00<?, ?it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "mean_init_acc: 0.7390, mean_final_acc: 0.7410\n",
      "\n",
      "mean_init_dist: 0.2308 +/- 0.3780, mean_final_dist: 0.2383 +/- 0.4000\n",
      "\n",
      "\n",
      "\n",
      "distortion = 15 +/- 10 degrees\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 1000/1000 [1:05:52<00:00,  3.95s/it]\n",
      "  0%|          | 0/1000 [00:00<?, ?it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "mean_init_acc: 0.6880, mean_final_acc: 0.6890\n",
      "\n",
      "mean_init_dist: 0.2901 +/- 0.4087, mean_final_dist: 0.2933 +/- 0.4320\n",
      "\n",
      "\n",
      "\n",
      "distortion = 30 +/- 10 degrees\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 1000/1000 [1:04:16<00:00,  3.86s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "mean_init_acc: 0.4460, mean_final_acc: 0.4460\n",
      "\n",
      "mean_init_dist: 0.5429 +/- 0.4625, mean_final_dist: 0.5461 +/- 0.4816\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "np.random.seed(SEED)\n",
    "torch.manual_seed(SEED)\n",
    "\n",
    "_, N, D = Xtest.shape\n",
    "\n",
    "n_trials = 1000\n",
    "epochs = 100\n",
    "\n",
    "# print_period = 50 # for online optimization\n",
    "\n",
    "\n",
    "# distortion rotation angle parameters (in degrees): \n",
    "distortion_std = 10\n",
    "distortion_means = [0, 5, 10, 15, 30]\n",
    "# distortion_means = [5, 10]\n",
    "\n",
    "all_results = dict()\n",
    "\n",
    "for distortion_mean in distortion_means:\n",
    "    print('\\n\\n\\ndistortion = %d +/- %d degrees' % (distortion_mean, distortion_std))\n",
    "    \n",
    "    gt_axis_angles = []\n",
    "    init_axis_angles = []\n",
    "    gt_labels = []\n",
    "    \n",
    "    init_outputs = []\n",
    "    init_predictions = []\n",
    "    init_losses = []\n",
    "    init_accs = []\n",
    "    init_dists = []\n",
    "\n",
    "    final_outputs = []\n",
    "    final_predictions = []\n",
    "    final_losses = []\n",
    "    final_accs = []\n",
    "    final_dists = []\n",
    "    \n",
    "    optimized_axis_angles = []\n",
    "    \n",
    "    \n",
    "    for n in tqdm(range(n_trials)):\n",
    "        \n",
    "        # 1) Randomly transform the test data:\n",
    "        gt_axis_angle = random_axis_angle()\n",
    "        gt_axis_angles.append(gt_axis_angle)\n",
    "        gt_rotation = torch_rotation_matrix(gt_axis_angle).float().to(device)\n",
    "        # print('\\ngt_rotation:\\n', gt_rotation)\n",
    "        \n",
    "        # select a sample:\n",
    "        shape_idx = np.random.choice(len(Xtest))\n",
    "        gt_labels.append(Ytest[shape_idx])\n",
    "#         print('\\nshape_label:', Ytest[shape_idx])\n",
    "        \n",
    "        test_data = Xtest[shape_idx:shape_idx+1].reshape(-1, 3) @ gt_rotation.T\n",
    "        test_data = test_data.reshape(1, N, D)              \n",
    "        test_label = Ytest[shape_idx:shape_idx+1]  \n",
    "\n",
    "        # 2) Initialize steerable model parameters with distorted ground truth axis-angle \n",
    "        # create a \"distortion\" axis-angle:\n",
    "        distortion_angle = np.radians(distortion_std)*np.random.randn() + np.radians(distortion_mean)\n",
    "#         print('\\ndistortion_angle:', np.degrees(distortion_angle))\n",
    "        distortion_axis_angle = random_axis_angle(angle=distortion_angle)\n",
    "        distortion_matrix = torch_rotation_matrix(distortion_axis_angle).float()\n",
    "\n",
    "        # by multiplying distortion_matrix with the gt_rotation matrix, \n",
    "        # we can control the rotation angle randomness:\n",
    "        distorted_rotation =  distortion_matrix @ gt_rotation.cpu()   \n",
    "        init_r = R.from_matrix(distorted_rotation.cpu().numpy())\n",
    "        init_axis_angle = init_r.as_rotvec()\n",
    "\n",
    "        init_axis_angles.append(init_axis_angle)\n",
    "\n",
    "        # construct the steerable model with init_axis_angle:\n",
    "        steerable_model = build_steerable_model(input_shape=test_data.shape[1:], \n",
    "                                                output_dim=OUTPUT_DIM,\n",
    "                                                hidden_layer_sizes=[N_GEOMETRIC_NEURONS],\n",
    "                                                init_axis_angle=init_axis_angle,\n",
    "                                                transformed_parameters=transformed_parameters,\n",
    "                                                print_hidden_layer_output=False).float().to(device)\n",
    "        \n",
    "        \n",
    "        # get initial model output, hidden_activations, acc and loss:\n",
    "        output = steerable_model(test_data)\n",
    "        init_output = output.detach().cpu().numpy()\n",
    "        init_outputs.append(init_output)\n",
    "    \n",
    "        init_dist = np.linalg.norm(softmax(init_output) - softmax(gt_outs[shape_idx]), ord=1)\n",
    "        init_dists.append(init_dist)\n",
    "\n",
    "        init_acc = score(output.detach(), test_label)\n",
    "        init_accs.append(init_acc)\n",
    "        \n",
    "        init_loss = entropy(output, is_logits=True).item()\n",
    "        init_losses.append(init_loss)\n",
    "\n",
    "#         print('\\ninit_acc:  %.3f' % init_acc)\n",
    "\n",
    "        # collect the initial predictions:\n",
    "        init_prediction = torch.argmax(output.detach(), axis=1)\n",
    "        init_predictions.append(init_prediction.cpu().numpy())\n",
    "#         print('initial prediction:', init_prediction.cpu().numpy())\n",
    "#         print()\n",
    "\n",
    "\n",
    "        # 3) Otimize the entropy loss wrt the axis-angle parameters of the steerable model:\n",
    "        optimizer = optim.Adam(steerable_model.parameters(), lr=1e-2)\n",
    "\n",
    "        for i in range(epochs):          \n",
    "            # compute the entropy loss:\n",
    "            output = steerable_model(test_data)\n",
    "            loss = entropy(output, is_logits=True)\n",
    "\n",
    "            # backpropagate:\n",
    "            optimizer.zero_grad()\n",
    "            loss.backward()\n",
    "            optimizer.step()        \n",
    "\n",
    "#             if i % print_period == 0:\n",
    "#                 print('epoch: %d,  loss: %.3f' % (i, loss.item())) \n",
    "\n",
    "                \n",
    "        # store the final data:                \n",
    "        final_losses.append(loss.item())\n",
    "        final_output = output.detach().cpu().numpy()\n",
    "        final_outputs.append(final_output)\n",
    "        \n",
    "        final_dist = np.linalg.norm(softmax(final_output) - softmax(gt_outs[shape_idx]), ord=1)\n",
    "        final_dists.append(final_dist)\n",
    "\n",
    "        final_acc = score(output.detach(), test_label)\n",
    "        final_accs.append(final_acc)\n",
    "\n",
    "        final_prediction = torch.argmax(output.detach(), axis=1)\n",
    "        final_predictions.append(final_prediction.cpu().numpy())\n",
    "        \n",
    "        optimized_axis_angle = steerable_model.axis_angle.detach().cpu().numpy()\n",
    "        optimized_axis_angles.append(optimized_axis_angle)\n",
    "\n",
    "#         print('epoch: %d,  loss: %.3f' % (i, loss.item()))  \n",
    "#         print('\\nfinal_acc: %.3f' % final_acc)\n",
    "#         print('final prediction:', final_prediction.cpu().numpy()) \n",
    "#         print('\\ngt_axis_angle:\\n', gt_axis_angle)\n",
    "#         print('\\ninit_axis_angle:\\n', init_axis_angle)\n",
    "#         print('optimized_axis_angle:\\n', optimized_axis_angle) \n",
    "        \n",
    "        \n",
    "        \n",
    "    all_results[distortion_mean] = {        \n",
    "        'gt_axis_angles': gt_axis_angles,\n",
    "        'gt_labels': gt_labels,\n",
    "\n",
    "        'init_axis_angles': init_axis_angles,\n",
    "        'init_outputs': init_outputs,\n",
    "        'init_predictions': init_predictions,\n",
    "        'init_losses': init_losses,\n",
    "        'init_accs': init_accs,\n",
    "        'init_dists': init_dists,\n",
    "        \n",
    "        'final_outputs': final_outputs,\n",
    "        'final_predictions': final_predictions,\n",
    "        'final_losses': final_losses,\n",
    "        'final_accs': final_accs,\n",
    "        'final_dists': final_dists,\n",
    "        \n",
    "        'optimized_axis_angles': optimized_axis_angles,\n",
    "    }\n",
    "    \n",
    "    \n",
    "    print('\\nmean_init_acc: %.4f, mean_final_acc: %.4f' \n",
    "          % (np.mean(all_results[distortion_mean]['init_accs']), np.mean(all_results[distortion_mean]['final_accs'])))\n",
    "    \n",
    "    print('\\nmean_init_dist: %.4f +/- %.4f, mean_final_dist: %.4f +/- %.4f'  \n",
    "          % (np.mean(all_results[distortion_mean]['init_dists']), np.std(all_results[distortion_mean]['init_dists']), \n",
    "             np.mean(all_results[distortion_mean]['final_dists']), np.std(all_results[distortion_mean]['final_dists'])))"
   ]
  }
 ],
 "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.7.10"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
