{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import jax\n",
    "import jax.numpy as jnp\n",
    "from configs.colu import get_config\n",
    "import utils as vae_utils\n",
    "import train\n",
    "import models\n",
    "import input_pipeline\n",
    "import tensorflow_datasets as tfds\n",
    "import matplotlib.pyplot as plt\n",
    "from PIL import Image\n",
    "import numpy as np\n",
    "import glob\n",
    "def mat2pil(mat):\n",
    "    arr = np.array(jnp.clip(mat * 255.0 + 0.5, 0, 255).astype(jnp.uint8))\n",
    "    return Image.fromarray(arr)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "\n",
    "# # enable float64\n",
    "# jax.config.update(\"jax_enable_x64\", True)\n",
    "\n",
    "# # script usage: pass --config=A/config.py\n",
    "# from absl import flags\n",
    "# FLAGS = flags.FLAGS\n",
    "# from ml_collections import config_flags\n",
    "# config_flags.DEFINE_config_file(\n",
    "#     'config',\n",
    "#     None,\n",
    "#     'File path to the training hyperparameter configuration.',\n",
    "#     lock_config=True,\n",
    "# )\n",
    "# config = FLAGS.config\n",
    "# # basic usage\n",
    "# import ml_collections\n",
    "# config = ml_collections.ConfigDict()\n",
    "\n",
    "# key, latents, dataset\n",
    "config = get_config()\n",
    "rng = jax.random.PRNGKey(0)\n",
    "rng, z_key, eval_rng = jax.random.split(rng, 3)\n",
    "z = jax.random.normal(z_key, (64, 20))\n",
    "ds_builder = tfds.builder(config.dataset_name)\n",
    "ds_builder.download_and_prepare()\n",
    "test_ds = input_pipeline.build_test_set(ds_builder)\n",
    "\n",
    "# load a vanilla model\n",
    "config.latents = 500\n",
    "config.num_groups = 100\n",
    "config.share_axis = False\n",
    "config.variant = \"soft\"\n",
    "params = vae_utils.load_model(f'results/colu-group100.msgpack')\n",
    "jax.tree_util.tree_map(lambda x: x.shape,params)\n",
    "# params = jax.tree_map(lambda x: x[0],params) # if you forgot to unreplicate before saving\n",
    "model = models.model(config.latents, config.dataset_name, config.act_fn,num_groups=config.num_groups,variant=config.variant,share_axis=config.share_axis)\n",
    "# the above model reproduces correlated axes\n",
    "\n",
    "# covariance of weights\n",
    "w = params['encoder']['fc1']['kernel']\n",
    "ww = w.T.dot(w)\n",
    "# import matplotlib.pyplot as plt\n",
    "# plt.axis('off')\n",
    "# plt.imshow(ww)\n",
    "display(mat2pil(ww).resize((200,200)))\n",
    "# modify evaluation function to track hidden layers\n",
    "def eval_f(params, raw_images, z, z_rng, model, config):\n",
    "  (recon_x, mean, logvar), intermediates = model.apply({'params': params}, raw_images, z_rng, mutable=[\"intermediates\"],capture_intermediates=True)\n",
    "  return recon_x, mean, logvar, intermediates\n",
    "recon_x, mean, logvar, intermediates = eval_f(params, test_ds, z, eval_rng, model, config)\n",
    "# covariance of intermediate states\n",
    "# intermediates = jax.tree_map(lambda x: x[0],intermediates) # note: this does not filter out tuple, bc tuple is regarded a pytree branch\n",
    "xx = intermediates['intermediates']['encoder']['fc1']['__call__'][0]\n",
    "xxx = jnp.cov(xx.T)\n",
    "# plt.imshow(xxx)\n",
    "# plt.axis('off')\n",
    "xxx[50:55,50:55],xxx[-5:,-5:]\n",
    "display(mat2pil(xxx).resize((200,200)))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [],
   "source": [
    "# load a vintage model\n",
    "config = get_config()\n",
    "config.latents = 2401\n",
    "config.num_groups = 800\n",
    "config.share_axis = True\n",
    "config.variant = \"hard\"\n",
    "params = vae_utils.load_model(f'results/colu_latents2401_group800.msgpack')\n",
    "params = jax.tree_map(lambda x: x[0],params) # if you forgot to unreplicate before saving\n",
    "jax.tree_util.tree_map(lambda x: x.shape,params)\n",
    "model = models.model(config.latents, config.dataset_name, config.act_fn,num_groups=config.num_groups,variant=config.variant,share_axis=config.share_axis)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAPoAAAD7CAYAAABDsImYAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjcuNSwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/xnp5ZAAAACXBIWXMAAA9hAAAPYQGoP6dpAAAE/0lEQVR4nO3dv4vXdRzA8bvMIGqQKCilwSRBS6olIqMpAoeKfgkOgbSYUSYqXFFQQVOoWVRTS2QmlITSH9AWEg0RpQgRREO/lLAgKPDbHsYd7+v09Pl4zJ8Xr+8NT97Ti5ueTCaTKeCidsn5/gHAwhM6BAgdAoQOAUKHAKFDgNAhQOgQIHQIuHSuH5758cahBasOPjE0B8zNd9t3zvqNFx0ChA4BQocAoUOA0CFA6BAgdAgQOgQIHQKEDgFChwChQ4DQIWDO12ujV2hLrvtzaG5qampq+XuXDc1d8c1PwzuPb1s+PAuLlRcdAoQOAUKHAKFDgNAhQOgQIHQIEDoECB0ChA4BQocAoUOA0CFA6BAw5zPVUV/d/c7w7Jq/tgxOXj28c/rk8CgsWl50CBA6BAgdAoQOAUKHAKFDgNAhQOgQIHQIEDoECB0ChA4BQoeABb9eu3n/tuHZgxvfGJqb2frk8M7v7x0ehUXLiw4BQocAoUOA0CFA6BAgdAgQOgQIHQKEDgFChwChQ4DQIUDoECB0CFjwM9UzSyfDs5s+fnpscB6npp88tHdo7sF3dw3vvPWe40Nznx9dPbyTFi86BAgdAoQOAUKHAKFDgNAhQOgQIHQIEDoECB0ChA4BQocAoUPAgl+vXWgeP/bY0Ny1R/8e3nn6wFVjg1uGVxLjRYcAoUOA0CFA6BAgdAgQOgQIHQKEDgFChwChQ4DQIUDoECB0CBA6BDhT/Z+8+tbbw7Pbnxv8Z5IwR150CBA6BAgdAoQOAUKHAKFDgNAhQOgQIHQIEDoECB0ChA4BQocA12v/8vOxa4bmNh2bxwXa7WNjX27cNzS37si2sYVTU1PLVpwemjv97bLhncyfFx0ChA4BQocAoUOA0CFA6BAgdAgQOgQIHQKEDgFChwChQ4DQIcD12gXslk+3Ds2tefb48M5fHr5pbHDdZHgn8+dFhwChQ4DQIUDoECB0CBA6BAgdAoQOAUKHAKFDgNAhQOgQIHQIEDoEOFO9gF35xeVDc6fuXzu88/2Xdg/NbTi0c3gn8+dFhwChQ4DQIUDoECB0CBA6BAgdAoQOAUKHAKFDgNAhQOgQIHQIcL12Afvj+jODc+M7R6/Qlv4+/qasWP/D0Nz+1QeGd9710a7h2cXIiw4BQocAoUOA0CFA6BAgdAgQOgQIHQKEDgFChwChQ4DQIUDoECB0CHCmyjlxw2tfD8+eOrFmaO7O9TuGd15sL+DF9vcAZyF0CBA6BAgdAoQOAUKHAKFDgNAhQOgQIHQIEDoECB0ChA4Brtc4J07et3Z49vWX3xyae+bFp4Z3/nrbZHh2MfKiQ4DQIUDoECB0CBA6BAgdAoQOAUKHAKFDgNAhQOgQIHQIEDoECB0CpieTyZzu8Vbu27PQvwUWjZkNR4bm9hx6YHjn8498ODS3efVns37jRYcAoUOA0CFA6BAgdAgQOgQIHQKEDgFChwChQ4DQIUDoECB0CPBPFuEsdh8eu0Jb+cLsl2T/5ZUljw7NbZ6Z/RsvOgQIHQKEDgFChwChQ4DQIUDoECB0CBA6BAgdAoQOAUKHAKFDgNAhwJkqnMWqD34bmjux947hnTMbDg9O7pj1Cy86BAgdAoQOAUKHAKFDgNAhQOgQIHQIEDoECB0ChA4BQocAoUPA9GQymZzvHwEsLC86BAgdAoQOAUKHAKFDgNAhQOgQIHQIEDoE/AONH263dn4+8QAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 300x300 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# different symmetry groups\n",
    "def copy_pytree(pytree):\n",
    "  return jax.tree_map(jnp.array, pytree)\n",
    "\n",
    "def prune(params0, p):\n",
    "    \"\"\"align params\"\"\"\n",
    "    params = copy_pytree(params0)\n",
    "    params['encoder']['fc1']['kernel'] = jnp.dot(params['encoder']['fc1']['kernel'],p)\n",
    "    params['encoder']['fc1']['bias'] = jnp.dot(params['encoder']['fc1']['bias'],p)\n",
    "    params['encoder']['fc2_logvar']['kernel'] = jnp.dot(p.T,params['encoder']['fc2_logvar']['kernel'])\n",
    "    params['encoder']['fc2_mean']['kernel'] = jnp.dot(p.T,params['encoder']['fc2_mean']['kernel'])\n",
    "    params['decoder']['fc1']['kernel'] = jnp.dot(params['decoder']['fc1']['kernel'],p)\n",
    "    params['decoder']['fc1']['bias'] = jnp.dot(params['decoder']['fc1']['bias'],p)\n",
    "    params['decoder']['fc2']['kernel'] = jnp.dot(p.T,params['decoder']['fc2']['kernel'])\n",
    "    return params\n",
    "\n",
    "def permutation(rng, C):\n",
    "    \"\"\"pointwise\"\"\"\n",
    "    rng, key = jax.random.split(rng, 2)\n",
    "    o = jax.random.permutation(key, C)\n",
    "    p = jnp.eye(C)[o]\n",
    "    return rng, p\n",
    "\n",
    "def orthogonal(rng, C):\n",
    "    \"\"\"ball\"\"\"\n",
    "    rng, key = jax.random.split(rng, 2)\n",
    "    p = jax.random.orthogonal(key,C)\n",
    "    return rng, p\n",
    "\n",
    "def orthogonal(rng, C,G=100):\n",
    "    \"\"\"mix cones' sections\"\"\"\n",
    "    rng, key = jax.random.split(rng, 2)\n",
    "    p = jnp.eye(C)\n",
    "    p = p.at[G:,G:].set(jax.random.orthogonal(key,C-G))\n",
    "    return rng, p\n",
    "\n",
    "def orthogonal(rng, C, G=100):\n",
    "    \"\"\"group-wise\"\"\"\n",
    "    # S(G)✖️O(C-1)^G\n",
    "    p = jnp.eye(C)\n",
    "    S = C // G\n",
    "    S_1 = S - 1\n",
    "    # rng, key = jax.random.split(rng, 2)\n",
    "    # o = jax.random.orthogonal(key,G)\n",
    "    rng, o = permutation(rng, G)\n",
    "    p = p.at[:G,:G].set(o)\n",
    "    rng, key = jax.random.split(rng, 2)\n",
    "    o1 = jax.random.orthogonal(key,S_1)\n",
    "    p1 = jnp.kron(o,o1)\n",
    "    p = p.at[G:,G:].set(p1)\n",
    "    return rng, p\n",
    "\n",
    "def orthogonal(rng, C, G=100):\n",
    "    \"\"\"w/ axis sharing, group-wise\"\"\"\n",
    "    # S(G)✖️O(C-1)^G\n",
    "    p = jnp.eye(C)\n",
    "    S_1 = (C - 1) // G\n",
    "    rng, key = jax.random.split(rng, 2)\n",
    "    o1 = jax.random.orthogonal(key,S_1)\n",
    "    p1 = jnp.kron(jnp.eye(G),o1)\n",
    "    p = p.at[1:,1:].set(p1)\n",
    "    return rng, p\n",
    "\n",
    "def orthogonal(rng, C, G=100):\n",
    "    \"\"\"w/ axis sharing, not group-wise, no group-switching\"\"\"\n",
    "    p = jnp.eye(C)\n",
    "    S_1 = (C - 1) // G\n",
    "    for i in range(G):\n",
    "        rng, key = jax.random.split(rng, 2)\n",
    "        p = p.at[(1+S_1*i):(1+S_1*(i+1)),(1+S_1*i):(1+S_1*(i+1))].set(jax.random.orthogonal(key,S_1))\n",
    "    return rng, p\n",
    "\n",
    "# visualize groupwise orthogonal without switching groups\n",
    "rng, p = orthogonal(rng, C=16, G=5)\n",
    "# # visualize p in matplotlib\n",
    "plt.figure(figsize=(3,3))\n",
    "plt.imshow(p)\n",
    "plt.axis('off')\n",
    "plt.show()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 29,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "before:\n"
     ]
    },
    {
     "data": {
      "image/jpeg": "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",
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAPIAAAA+CAIAAACeINE8AAALCElEQVR4Ae2aTaiO3RfGX/yTEgMZ+EjIiKSIAUlSiglKIokkX0WmmChkIJIYmGBGRmImJjKhfGVAMmDERERK8vF/F8t7vcvaX2vvfd/n0LufwdPa17rWb617nfs85znPOX/91R5tA20DbQNtA20DbQNtA20DbQNtA20DbQNtA20DfW3g/z8efdEbN38D/8sv6bGCbg+iDxkyhJ455mas9Ni4Ai3nrMAMRClG/Z33GV8EXYJl+KFxykBmsXQKEPMA6mic6gfm+/cJAmOh3VY2mJHf7dgY1XJbGCccFBsuJNL9++tizQM9apYFiHeSYrIXW0zzziZbdEjuHCuB8kI6nJmxaNQ5mfiAUxznl78JkT3kpgpiGhE0OS6J8mgng+aWFDNdlFTK5pQEjiOTu2ajEmLy2usn9/KlWN9CXWkSmP0mhMblB3VK0tU0oSMBOaWA6hgqT+pdcVQjjK304iOAPDA/QyzGhgqZXMmvLA/N5upoZPlqZrxag8stsXqluwPFlcryOJyzqgUdLauJk8GsR3kbEZZbVPIxJ3VhJgOl7h0gIubWkr/yKiLDeFPW2xpXouaD7qVbRN61xWn0hEaSjSoXHWphnNBrAxMblgN7SwpECadyNLWj3BIwOYUjM12/vZfrVHDXwEr6tpZjKahMhRrEdUWgo2oRL09miSaZfExWxQ1yZglUveIQS1Y2svjjHhpPGcCXK1IedUQJ64qpjuRhv6srbPKo+ib9Ge+t3eFcJdlPGVwCXQA/lNN4pFrpVEdjStriseRzLJV4rcyiyrsQ6ewwRi8EcTiGZJuxKs60ZFVfS0nitgbRew2cpZQ3a2lPHtQiMBZ6bRJCMT+kkxQccXVQkoGlxOLxNpKzkYGPSvQWRsTKci+ZmH1gvb0gZjVN3NaAqoC+csVfPIWiIyZGQGIZv6zKHalSKRtDVfERzyprnDBexXdn3GNsBBvR8IBYHBCqoNZ0W/PFM50nRieZglgZ4OYuuKQ+5sHlyHnQiKfFzDDbA4m1V3Xi5Na4FiOTqriQ/XxkReqgeUVk+wjSvzJSV+9YubvImh4dKchqhMJIOwJabIogS3gkNZhrUATvUVaRQR25xCt6aQWiuooIQe3NncpVIrReU4lXa/s19zRlrwPY4ckvmDTYsd6lUTkT+Jk8P4SforfEIsoJpT+kSw9ijARFBRg16VSF3iNmy6WZXq1Vy9weqrzXI82GXUQaGW0gKD+1YIWeyWPpCJQKmKBEOqqOrsGogOM2Khvb5RgnGUhb4tWaRqHL4Ef9WLxHekYQZ1LfuMGbRRUauTbjAG4hFBA4YB2tYasJOqFhPAQ1Iw1kLQYu2EP6tuYrqekhdwEOBd5xSaeHaioJlliSQbMUZnlkFxR6RWSzAp68QyC6f1/xP0uG+NsGBaOmb2sJ7XDFjJJwd6017WQtdeFGHHDstosrEshO5pCOR5yQlS0b0m3Bs2FgF0sGt6oTpYbszpk1UvqS0KBmSswEGhQ36KQRY+PtyhqBSeUUl0Hcq3YVbtQJHzPLLp2QJdCN0Te3FwqJmVv7vcQdRSqgF6AlxxsDzllqwUq3vVQX2avbRt5rLBbl2J3MKYE0VSfM5NWhaW47LsytwjyDeVtjiAEIivc7ALOFWlR+aUPYAdMHceex23oQxxqw1bdGvW7gT//O7HU5Dd420DbQNtA20DbQNtA20DbQNtA20DbQNtA20DbQNtA20DbQNtA20NsGDh8+vHjx4t7wDfznb2Dy5MlTp06lj/Hx2Llz5+98WZizjyHHjx/fB/bPZQ4fPpwW/sfMj5vDG9RcBoCnT5+u4Xhrr169Cr7XUCzOmDGDycUEVfjly5eeRlWNej12uxPPqKNGjbp3796rV6+o08GDBz2OHAkbl8HLly/pmIP5xStRHP+S7uIgW3TB+8mQ2PXr19eTjx8/LpkUjxkzph7rJezfv//WrVveVL3IV1HP8RPUjujo99lUepshgbaimGvevHkSiDhWk59btWpV5+QdO3aAyUH+XJ4KxcTRYy2SAFRBESxYBHjQUZwAmoJx48YRh5ViIAjEqYGgVk7oxvSzeNq0aTDXBBJew5G1YJI4cuRIOspsWQzmixcvmAClki85obhsZm8Vtxg7dqw3WyguWbKEubNmzZIIFqWSFWMdWVUhM30iASAC+j2D/WvWrHn48CHpofIsHfysqohZAS9cuNDJqArLAxw5csSrR8ZTKZQjoG8birdv385O1jdv3qwKy47oUlburwpBHz16xCl/mUENkQ2lfguACKSPxWfPnkmxIAYc3zMFEFkCIER6k0oijmXB58+fmeyWT5o0iVNz5851s16F5+Eq+ew1jx49mv3ebK7I7XKrYv4TJ06EoCE9hvs1xwR+ph+7ly5d+jWfd5I0xBLhFaXBEgNCwezZs+lz6/fv31NsqQ15wISBFRzLAhcrObktQLMUsoc+0pEdy+LXr19bOmbAGTds2DC3hlOPHz92U3aFIep59erVdoJ0Kg4d1VsxaZCF9lgS3Hjr1q12FJzgQKHg+vXrpEulIGbyoUOHvLXevl4niR8/foSfPloJ2aTOfqmUxV1xfnan24uIp06dUtPwp+KdNDtw4ADB6eNCemYgnlVTyxG1FOzevdstWbBgATxu1qKgPBRYINKzdOlSRt29e1fqCxcuJF0qBTGT6UMbb+2GDRvIQH8C82aVyCh+Vinv8du3b2Revny5N2sX0ddeknCGrgGdKEggitKhvkkYBnv69GnIvGvXLthCnpCOQgSfPn1CzMGbN29C5V4d5Sor9T179tC75HXr1ilP8siQkO3y5ctkCGVdnWkfPnxwU6ysXLmSPbdv3+aAnpctWxbyJ/UpU6YwZ/78+Umz1cBQ6Z4wYQLGpUCmOoyPHTvGXXKZxtlgy+LTL5rxQmTpvaCdjKoRI0ZwFd9t0GVAt4udTE768IfKQyVMDmVd/evXr1xy48YNlZVDqlg5s45AZVUlzIB6g0RxXZo75jLknJFaen+Sy0cJFXrJc+bMQXevISSiKhls2bIlBInojPUaIimvn8TkkGQ4f/68soVoSZ05Fy9eTDrzDCtWrFAj8jGPku8u7oJpI392mThxYi4f2Js3b3qvhg3v3r2jD0a8hpAIMgJ2qmOoPK7v27ePOV5bJOX1k4ip3ECVSINKGY8gGP3ZtqFDh6IHBdn1ouDcuXPiFAy5XTAdThjnZJv9Fxpgr1y5oppv3LgRWZWqOZ45c4axNRCqDUH4ryeULeOjkH5XCRG4NZwhW0ivLA9h/9XRoHhEZm3atMnyTzZo9+8EORH9exAT6EMGbx0M3qxXPHv2LKYKBfS/KN7aMhFdyspRFeKwHvrhg/KaINTayORyoznbdv/+/cr5ZMskKmmQtFBMnxgyZ+3aterjrZkzZ5a1QJUbhMao0blLDYFrMa1EeUVp6CSu6VJTmx4edArSboNDAiWTfrhfu3YNWQMpYQEKAb3rRcxBAuGkVXkZxKH6BYZv27bNnzar/MeHgZycR6PfbYr38/z5c651P3IxX3fYOH36dKwj7MrL4A0AyG6QRwy7XbJUwnWJzIMHD44ePZowVad51E7+VUheNeLqAROAkydPcq+Ez5fud0h6BS2ezDet1jA9Au3o4gw4grdv33YB7pfB03bVY+/evbh8elfZFTbCoT8eUccnT55EPKEUXk9DhiqdF7Fo0aIqSiv+T26Av5F+x5uHbus7d+78J78o7aJrN0A3Ty2i1bcNtA20DbQNtA20DbQNtA20DbQNDO4G/gZ9jIqHtqfbKQAAAABJRU5ErkJggg==",
      "text/plain": [
       "<PIL.Image.Image image mode=RGB size=242x62>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "after:\n"
     ]
    },
    {
     "data": {
      "image/jpeg": "/9j/4AAQSkZJRgABAQAAAQABAAD/2wBDAAgGBgcGBQgHBwcJCQgKDBQNDAsLDBkSEw8UHRofHh0aHBwgJC4nICIsIxwcKDcpLDAxNDQ0Hyc5PTgyPC4zNDL/2wBDAQkJCQwLDBgNDRgyIRwhMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjL/wAARCAA+APIDASIAAhEBAxEB/8QAHwAAAQUBAQEBAQEAAAAAAAAAAAECAwQFBgcICQoL/8QAtRAAAgEDAwIEAwUFBAQAAAF9AQIDAAQRBRIhMUEGE1FhByJxFDKBkaEII0KxwRVS0fAkM2JyggkKFhcYGRolJicoKSo0NTY3ODk6Q0RFRkdISUpTVFVWV1hZWmNkZWZnaGlqc3R1dnd4eXqDhIWGh4iJipKTlJWWl5iZmqKjpKWmp6ipqrKztLW2t7i5usLDxMXGx8jJytLT1NXW19jZ2uHi4+Tl5ufo6erx8vP09fb3+Pn6/8QAHwEAAwEBAQEBAQEBAQAAAAAAAAECAwQFBgcICQoL/8QAtREAAgECBAQDBAcFBAQAAQJ3AAECAxEEBSExBhJBUQdhcRMiMoEIFEKRobHBCSMzUvAVYnLRChYkNOEl8RcYGRomJygpKjU2Nzg5OkNERUZHSElKU1RVVldYWVpjZGVmZ2hpanN0dXZ3eHl6goOEhYaHiImKkpOUlZaXmJmaoqOkpaanqKmqsrO0tba3uLm6wsPExcbHyMnK0tPU1dbX2Nna4uPk5ebn6Onq8vP09fb3+Pn6/9oADAMBAAIRAxEAPwD5/oorvPCvgSxuPDUvi/xXqMmn+HY3eGOOAf6VeSgDasIZdpBORnnlGBAALKAcHRXv/wAJfEejS+Ib290zSNK8L6FY2h+2vcu88kryOoX/AEp8CNcouEJ5O7CsTuTwCgAq5pWlX2uapb6ZpltJc3lw+yKJOrH+QAGSSeAAScAVTr3z4F6esHg7xHqWn6vaQaxcoLWAyXTBbeVsrD5kRTbkuV2t8+clQAQwYA83T4S+PJLe7nHhq7CWjukgZkDMVGTsUtmQehQEN2zXF12njC68f6Gj+HfFep6lsu0juWtri++0B1VmCnIZsDcD8ueSqkjhSOLoAK0JdC1iCza8l0q+jtVijnaZ7dwgjkJEblsY2sQQD0OOK6D4XeHk8TfEXSLGZoPs6S/aJkmCsJEj+cpsYjfuxtIGcAk4IBr6D8VfFfw14HuLayhWS4Fu7Wn2LT5LcxxqhiyWAbdGUBdVQ7clXBAG1gAfJlFekfF7wKnhnXE1jSB53h7Vv39vPEq+TE7ZbylK8bcYZeBlTgZ2k15vQAV0lr4G1q58EX3i8xRw6TaOiB5SQ05Zwh8sY5CsQCTgdQMkECPwX4TvPGvii10WzfyvNy805QusMajLMQPwAzgFioyM5r3vxp8OvE134Ts/BvhNLS38O2bxBvtVywuLlyxZ3JXKmIGTeVIB3RnauAgIB8yUVc1XSr7Q9UuNM1O2ktry3fZLE/VT/IgjBBHBBBGQap0AFWLqwvLHyPtlpPb+fEs8PnRlPMjb7rrnqpwcEcGvTPCEGm+C/hvfeNdRitJdav3Nv4fjmVXaNkbDToDuGVbnLKMeWBn94M0/i1qlxry+FdYuzYyXF7pXmyS21wJDkyyHyyoxtVM7Rlc53qXkKE0Aeb0UUUAbGn+FtZ1TQ7/W7WzzplhgXF1JKkaKT/CCxG5uR8q5PzKMfMM49fSfjT4kw/Cuw0Pw34Vs7S7g/sxZoribBjZWcBJMxkBywWUtgDJdWB6g8J8W30nXfDnhfxjZwQWt7q/2n7RGJpZnk2uOrMoB2ElT0xlVXciggA8noor1jRvhn4b0zwMnibx7rM9l9siFxYWNjLH50seBjAYHczbk4GAoILEZO0A8nor3TSPjX4N8Opdto3gKOzniRYLOSJo1e4i3c+dJt3KcKp/5aZPU965/4k3eg6x4D8M69Y+GrTQru/uLkQw2ITY8MZCOZSEQ794G0DIALZOTgAHldFFFAFyw0q+1NLx7G2knFlbm6uNnJSIMql8dSAWGcdBkngEinXsFjdXHw0+DM80NxBbeI/EssTxeTdjz4bIqSkmznGcSAEYI80fMGXA8foAKKKKACvdPhfdaB4u8HWfhXxTo93NaaW91dRXYnaC0jiXa7vIwlXc6tLjhTtV16Asa8LrsNQ8ded8OrDwbpunfYbWKU3F7P5+57yTr8wVVG0Hs24/JHz8uSAanxE+JDeIDJoPh+KPTfC0DlY7O1VUjuCsjsJSAilQ2VbYeAQD15rzuiigDrPBPw81rx690ujvaKLV41ma4kKhA6uQ3AJIzHjjJyy8YyRz+pabNplwsMrRyBkDJLES0b8lW2NjDgMrIWXKkqcEjkmm6tqWjXDXGl6hd2M7IUaS1maJiuQcEqQcZAOPYV1g+MHj4WcNqPEc/lxbNpMUZc7CCNz7dzdBnJO7kHIJyAdh4/wBNuPDfwS8N6Tr9jv15ruQCS5uhM9rGpICx4kOFKCEbVBQdwGKmvF6sX99canqNzf3knmXV1K80z7QNzsSWOBwMknpVegDuPhP4s07wb45h1HVUn+ySRNbvJC7DytxHzso/1ijHKnPXcAWUCtD4k/D/AMQWvia71u2tf7S0zWbuS6tbrTg06Yll/dqxA4Y70x2bcApPOPN67jwd8WPFPgmz+w2E8FzYDcUtLyMukbMQSVIIYdDxnb8zHGTmgDrPHX/CTL8BfCcOt2sdskF6YGjnRkuMIrrBlCqhQEDg5ySBG2fmYDxutzxF4w1/xY8T67qUl4YnZ49yKoQsqK20KAACI14HGcnqSTh0AfRfwB03TbnwVrVxpjR2niRne1e9YrM8KlQY3WMgbU3ZO053GM5OAAtiwh1Lwvqn9q+O/iLJcRaXZBrbSobxraW7aPcG3RsUMo3qwVjnzflJOPlPzpY395pl5HeWF3PaXUedk0Ehjdcgg4YcjIJH40Xl/eahKJb27nuZBuw80hcjcxduT6szMfUsT1NAHSfEfxbN4z8a32otNHLaRO1vYlIyg+zqzbDg85OSxzzlj0GAOToooA9U1z4s6Hq3g628Nw+ArSGzs3V7WOTUJWjjYZBJEYRmJDPyW5LZOTXSfEXxtP4f8L+HtKg0aCyvrzRLm1ls7tJfMsbOUoiID5mC2IsZbLZjBKpuK14PXQeKPGGo+Lf7N/tCCxi/s+0W0i+yWyxZVe5x/wCgjCjnaoycgHP0UUUAe6fGvQ77xXqXhrXNGto79L23WyeewufPt1l83aq+ZsVQC8pUOzDceNq7eeb+Jmgy+D/Bfg7w9Je4umimu7+xQooWZiMOwXJZgGaIOSQRF8uPmzzegfEvxh4X0tdM0fWpILNXLrE0McoQnrt3qSBnnA4ySepNcvPPNdXEtxcSyTTyuXkkkYszsTkkk8kk85oAjr0yy+K1jceDrLw94q8I2mvJYoIba4Nx9nkjiG3aoKoSD8oBKkbgACDyT5nRQB6ovxd0jSnsrrwz8PtG0zULNDFHczSGY+WVAbO0IS5wPnZicFh/Ea6/4y6lD4j+DHhbxA62kl3cXEJaSAAiNnhcyopySBvXBGeqDPIr58rrNa8cTat4D0LwnFYx2lppjvJIUkLC4kJO18MCVI3PkBiCXPAAAAByddJ4D8KTeM/GNhoyCQQO++6kTP7uFeXOcEA4+UEjG5lB61zdamh+I9X8NXFzcaNfSWc9xbtbSSRgbvLYgkAkZU5UHcMEY4IoA7T456tcan8UL2G4tvIXT4o7WIHG5kx5m44YjkyEjp8pXIBzXm9STzzXVxLcXEsk08rl5JJGLM7E5JJPJJPOajoAKKKKACiiu30K30fwx4ZtfFOq2NtrN3ezTQWGnTP+5iMYUmaXafmwzKPKbGVbOeRgAzdD+H3izxJYvfaTod1cWqqWExARXwSDsLEbyCpGFzzxXNV7hoE/iPR9Dufil4y1Ce4mijkXQrK8kPzzzDb5ix7lCptz8qjlQWAAVSfD6ACum8E+BtU8earPp+lSW0UkEBneS5ZlTG4LjKq3J3dPY1zNeneB/FvhfTvh1q/hfVp9QsLzWLkq+oWMCsYodsYXzDkMyZ8zKDOVZwMFqAI9c+F+jeH/AA/fXV54/wBFfV7QMH023xIxkDbfLBDbt3TOU4Oc4AJrzWuu8X+DbfRLGz1vRdUXV/D9/LJHb3SwvG0bJ1SQMBtbrj+8FJwBXI0AKqlmCqCWJwAOpr0H/hS/jCPwpd+ILyC2sobaBrhra4lInaNU3k7QCAcZ+ViGyCCBWn+z5p8V78TDPIzhrKxlnjC4wWJWPByOmJG6YOcc4yDrfGH4rXd14jh0zw3fanpx0mWaK6ljmaITy7gMbVPKrsPLddx4HcA8Uor1n4x6Jpl9baV4/wDDqj+y9YHlzgAKVuBnkjP3mCuDgYzGSSS1eTUAFd2Ph2sHwlk8bX2prBLLMqWVmVAMy79h5Jzu4dgAD8qZ78U/hx4GuPHnimKwxPHp0X7y9uYlH7pOcAE8bmI2jr3OCFNe5+KPD3gvxvDpuiQePrO3sbS2S107TbS+hkUTgFEdssWkONqhc564ILGgD5dorZ8U+GdR8I+ILnSNThKSxHKPj5Zo8na6nupx+HIOCCKxqAL2kaPqOvanDp2lWct3dzNtSKMZP1J6ADuTgDqTWp4w8F6p4Iv7Sy1drb7Rc2y3OyGTf5YJI2tx1BU9Mj0Jr0G2hk+G/wAJrO6skYeLPFpC21xAp82C1O07UIzyQV6bWzKO8Yqn4u+HXinULHwn/Zvhu92xaCiTIVUFJVeR3DAY2sfMBAPzHOOWDYAPKKKfNDLbzSQzRvHLGxR0dSGVgcEEHoRTKAO+8L/B/wAUeLfDDa9pv2JYCXEMU8rJJPt/ufKV5bKjLDkHOBzXE31heaXeSWd/aT2l1HjfDPGUdcgEZU8jIIP416h8Z/EjweKNM8P6JeS2+laFa25tFgmOFk2BkkV+pIQoAcnGCQeTVPx9qr+Ifhr4L1rVGkk1qRru3e4aBQbiKNxglwRwuQACOSXOR/EAeZ0UV7hpuleGPA3wu0b4gWmlHXNYcKkcsu9YLafdL+8ZP9hsJnoTGhXaTuoA8s0nwX4n12OCXTNA1G5gnJEc6W7eU2CQf3hG3qCOvUYrKvrC80y8ks7+0ntLqPG+GeMxuuQCMqeRkEH8a9Wt/G/xe+IUtwuhrcJay5jP2CBYYYiqgkCduVboeXz83HBxVT4w6RrFlpvg6712SV9RfTPs9wZZEkbejFuXADMcSAHIwOPmcksQDyyiiigDoPDHhO88TxaxcQv5NrpWnzX08xjLD5FJWMY43MRxkjgMecYPP16j4pibwD8LtN8K7BHrOvEajquVw8cIP7mE5QHqMkZyrI45DV5dQAUUUUAFe5/CHVr/AMKeDrvxRr3iCe38MQO0FrpgAdrmbk4jDDgZZuEIywJYgKc+GVoXeualfaPp+k3F0z2Gn+Z9lh2gCPe25uQMnJ9c+1AGx468dap481w39+fKt48ra2iNlLdD2HqxwMt3x2AAHL0UUAbPhTTtO1bxFbafquoQ6da3AdDeTDKQttO1mGV74HUAEgnIBB7Rvgb4nur1DolzpmraTNuMOqW92nlEBmGGGSd3y8hQwBOMnBx5lRQB6J4vu9F8P+BbHwNpeox6vdi9GqX9/buGt0lMWwRRED5wFIy3qPUlU87oooA3fB/ivUPBfiODWdO2tJGrI8UhYJKjDBVgpGR0I91B7V6TqXw3T4m3c/irwdqmlg3qLc3mmSzt5ttO4PmKTtJ5dWIJCg5JHy4rxmigD2Hx9ocfgP4S6d4R1DU7a71qfVDqBghLERRbGTK5A4zj7wGSzYB2kjx6iigD134S+MPDdn4a1vwf4juH06HVSdt/GOSHURsjNg7cDkEjby+SO5Nqfwy+H2oQ6l4WN54l1eNnEX219sFqwBAk4jXeckYAyOCcqQpPkVFAF7WNYv8AX9Vn1TVLlrm9uCDJKwALYAA4AAGAAOPSqNFFAHaQfFnxxa6NFpNvr0kNnFbi2jWOCJWSMLtADhdwIH8Wc985rrvij4u1u38M+CLCDV76N59DjubuVLxle581VGJAMbgNrYJJzubPcnx2igAooooA901DwM3xY8LeF9Y8O6np8d1YaXHYahDdS7BEY/u4REbbkmQ84G3YQOtcT8SdY0RrXQfCnh+X7VZeH4ZInvhnbczSFTIyDJ+XcpIOcfMcfKATwNFABXWeFfiR4o8GWUtlo19HFaSu8jQvAjjzGQJvyRnIwpAzjKjIIyDydFAHYaz8U/G2urGl34hu0RFK7LUi3DAgA7vLC7s475xk4xmuz+IHiK7vvgX4LtdXuJbrVLy4e589irZji3oNxzkttkj55Jw2Tnr45Vu+1O91L7OLu4eVbaFYIUPCxoowAoHA9T6kknJJNAFSu0+FXhmLxT8QdOs55rVIIHF3LFcKGFwsZBMQU8MW7jsu484weLooA6X4geJW8W+ONU1cSb7d5jHbYDACFflTgnIJADEepPArmqKKACiiigD/2Q==",
      "image/png": "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",
      "text/plain": [
       "<PIL.Image.Image image mode=RGB size=242x62>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# rotation maintains model\n",
    "# symmetry group = perm(G) * O(S-1)^G\n",
    "def orthogonal(rng, C, G=100):\n",
    "    \"\"\"w/ axis sharing, not group-wise, switch groups\"\"\"\n",
    "    p = jnp.eye(C)\n",
    "    # rotation within cones\n",
    "    S_1 = (C - 1) // G\n",
    "    for i in range(G):\n",
    "        rng, key = jax.random.split(rng, 2)\n",
    "        p = p.at[(1+S_1*i):(1+S_1*(i+1)),(1+S_1*i):(1+S_1*(i+1))].set(jax.random.orthogonal(key,S_1))\n",
    "    \n",
    "    # permutation across cones\n",
    "    rng, key = jax.random.split(rng, 2)\n",
    "    o = jax.random.permutation(key, G)\n",
    "    o = jnp.eye(G)[o]\n",
    "    e = jnp.eye(S_1)\n",
    "    p1 = jnp.eye(C).at[1:,1:].set(jnp.kron(o,e))\n",
    "    \n",
    "    return rng, p1.dot(p)\n",
    "\n",
    "rng, p = orthogonal(rng, C=config.latents, G=config.num_groups)\n",
    "\n",
    "\n",
    "metrics, comparison, sample = train.eval_f(params, test_ds, z, eval_rng, model, config)\n",
    "print(\"before:\")\n",
    "display(vae_utils.save_image(comparison, f'results/reconstruction.png', nrow=8))\n",
    "params1 = prune(params,p)\n",
    "metrics, comparison, sample = train.eval_f(params1, test_ds, z, eval_rng, model, config)\n",
    "print(\"after:\")\n",
    "display(vae_utils.save_image(comparison, f'results/reconstruction.png', nrow=8))\n",
    "# print(\"sample:\")\n",
    "# display(vae_utils.save_image(sample, f'results/sample.png', nrow=8))\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Visualize Functions"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 300x200 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 300x200 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# Plotting parameters\n",
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "import glob\n",
    "def pretty_plt():\n",
    "    plt.rcParams.update({\n",
    "        \"text.usetex\": True,\n",
    "        \"font.family\": \"serif\",\n",
    "        \"font.sans-serif\": [\"Computer Modern Roman\"]})\n",
    "    plt.rcParams['axes.spines.right'] = False\n",
    "    plt.rcParams['axes.spines.top'] = False\n",
    "    plt.rcParams['axes.spines.left'] = True\n",
    "    plt.rcParams['axes.spines.bottom'] = True\n",
    "    plt.rcParams['axes.grid'] = True\n",
    "    plt.rcParams['grid.alpha'] = 0.5\n",
    "    plt.rcParams['font.size'] = 17\n",
    "    plt.rcParams['legend.framealpha'] = 0.7\n",
    "    plt.rcParams['xtick.labelsize'] = 14\n",
    "    plt.rcParams['ytick.labelsize'] = 14\n",
    "    # plt.rcParams['xaxis.labellocation'] = 'center'\n",
    "    # plt.rcParams['yaxis.labellocation'] = 'top'\n",
    "    plt.rcParams['legend.fontsize'] = 'small'\n",
    "\n",
    "pretty_plt()\n",
    "x = np.linspace(-2,3,500)\n",
    "y1 = x.clip(0,1)\n",
    "y2 = 1 / (1 + np.exp(-4*x+2))\n",
    "y3 = 1 / (1 + np.exp(-x+.5))\n",
    "plt.figure(figsize=(3,2))\n",
    "plt.plot(x,y1,label=\"Hard Projection\")\n",
    "plt.plot(x,y2,label=\"Firm Projection\")\n",
    "plt.plot(x,y3,label=\"Soft Projection\")\n",
    "plt.xlabel(\"Input\")\n",
    "plt.ylabel(\"Weighting\")\n",
    "plt.legend(loc='center')\n",
    "plt.savefig('results/softprojection.pdf',bbox_inches='tight')\n",
    "plt.show()\n",
    "x = np.linspace(-2,3,500)\n",
    "y1 = x > 0\n",
    "y2 = 1 / (1 + np.exp(-x))\n",
    "plt.figure(figsize=(3,2))\n",
    "plt.plot(x,y1,label=\"ReLU Projection\")\n",
    "plt.plot(x,y2,label=\"Sigmoid Projection\")\n",
    "plt.xlabel(\"Input\")\n",
    "plt.ylabel(\"Weighting\")\n",
    "plt.legend(loc='center')\n",
    "plt.savefig('results/actmask.pdf',bbox_inches='tight')\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Visualization of Learning"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 20/20 [04:53<00:00, 14.70s/it]\n"
     ]
    }
   ],
   "source": [
    "import yaml\n",
    "import os\n",
    "import ml_collections\n",
    "from tqdm import tqdm\n",
    "import numpy as np\n",
    "import glob\n",
    "from ml_collections import ConfigDict\n",
    "import base64\n",
    "\n",
    "# For possible conversion errors:\n",
    "def numpy_scalar_constructor(loader, node):\n",
    "    return None\n",
    "\n",
    "def numpy_dtype_constructor(loader, node):\n",
    "    return None\n",
    "\n",
    "def config_dict_constructor(loader, node):\n",
    "    fields = loader.construct_mapping(node, deep=True)['_fields']\n",
    "    config = ConfigDict()\n",
    "    for key, value in fields.items():\n",
    "        config[key] = value\n",
    "    return config\n",
    "\n",
    "# Register constructors with YAML tags\n",
    "yaml.add_constructor('tag:yaml.org,2002:python/object/apply:numpy.core.multiarray.scalar', numpy_scalar_constructor)\n",
    "yaml.add_constructor('tag:yaml.org,2002:python/object/apply:numpy.dtype', numpy_dtype_constructor)\n",
    "yaml.add_constructor('tag:yaml.org,2002:python/object:ml_collections.config_dict.config_dict.ConfigDict', config_dict_constructor)\n",
    "\n",
    "\n",
    "def process_tfevents(root_dir,precomputed=False):\n",
    "    dirs = sorted(glob.glob(f\"{root_dir}/*/*/\"))\n",
    "    first_configs = []\n",
    "    for dir in tqdm(dirs):\n",
    "        subdirs = glob.glob(os.path.join(dir, '*/')) # different seeds\n",
    "        min_eval_losses = []\n",
    "        min_train_losses = []\n",
    "        configs = []\n",
    "        for subdir in subdirs:\n",
    "            yaml_path = os.path.join(subdir, 'config.yaml')\n",
    "            with open(yaml_path, 'r') as f:\n",
    "                yaml_data = yaml.load(f, Loader=yaml.FullLoader)\n",
    "            config = ml_collections.ConfigDict(yaml_data)\n",
    "            if not precomputed:\n",
    "                tfevent_path = glob.glob(os.path.join(subdir, 'events.out.tfevents.*'))[0] # assume only one file\n",
    "                df = load_tfevents(tfevent_path)\n",
    "                min_eval_loss = df['eval/loss'].min().item()\n",
    "                min_train_loss = df['train/loss'].min().item()\n",
    "                min_eval_losses.append(min_eval_loss)\n",
    "                min_train_losses.append(min_train_loss)\n",
    "                config.min_eval_loss = min_eval_loss\n",
    "                config.min_train_loss = min_train_loss\n",
    "            configs.append(config)\n",
    "\n",
    "        if not precomputed:\n",
    "            min_eval_loss_mean, min_eval_loss_std = np.mean(min_eval_losses).item(), np.std(min_eval_losses).item()\n",
    "            min_train_loss_mean, min_train_loss_std = np.mean(min_train_losses).item(), np.std(min_train_losses).item()\n",
    "            for config, subdir in zip(configs,subdirs):\n",
    "                yaml_path = os.path.join(subdir, 'config.yaml')\n",
    "                config.min_eval_loss_mean = min_eval_loss_mean\n",
    "                config.min_eval_loss_std = min_eval_loss_std\n",
    "                config.min_train_loss_mean = min_train_loss_mean\n",
    "                config.min_train_loss_std = min_train_loss_std\n",
    "                with open(yaml_path, 'w') as f:\n",
    "                    # f.write(config.to_yaml())\n",
    "                    yaml.dump(config._fields, f)\n",
    "        first_configs.append(configs[0])\n",
    "    return first_configs\n",
    "\n",
    "root_dir = 'log/'\n",
    "configs = process_tfevents(root_dir,precomputed=False) # set precomputed to False to recompute"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 36/36 [00:00<00:00, 59.97it/s]\n"
     ]
    }
   ],
   "source": [
    "\"\"\"\n",
    "config.act_fn\n",
    "config.batch_size\n",
    "config.dataset_name\n",
    "config.latents\n",
    "config.learning_rate\n",
    "config.model_name\n",
    "config.num_epochs\n",
    "config.num_groups\n",
    "config.seed\n",
    "config.share_axis\n",
    "config.variant\n",
    "\"\"\"\n",
    "\n",
    "def config_to_latex(configs,share=True,variant=\"soft\"):\n",
    "    latex_table = \"\\\\begin{tabular}{cccc}\\n\\\\toprule\\n \\\\textbf{Width} $C$ & \\\\textbf{Group} $G$ & \\\\textbf{Train Loss} & \\\\textbf{Eval Loss} \\\\\\\\\\n\\\\midrule\\n\"\n",
    "    configs = sorted(configs, key=lambda x: x.num_groups)\n",
    "    i=0\n",
    "    for config in configs:\n",
    "        if config.share_axis and config.variant == \"soft\":\n",
    "            i += 1\n",
    "            row_color = \"\\\\rowcolor{gray!25}\" if i % 2 == 0 else \"\"\n",
    "            latex_table += f\"{row_color} ${config.latents}$ & ${config.num_groups}$ & $({config.min_train_loss_mean/100:.4f}\\\\pm{config.min_train_loss_std/100:.4f})\\\\times 10^2$ & $({config.min_eval_loss_mean/100:.4f}\\\\pm{config.min_eval_loss_std/100:.4f})\\\\times 10^2$ \\\\\\\\\\n\"\n",
    "    latex_table += \"\\\\bottomrule\\n\\\\end{tabular}\\n\"\n",
    "    share_name = \"share\" if share else \"noshare\"\n",
    "    with open(f\"results/vae_{share_name}_{variant}.tex\", \"w\") as f:\n",
    "        f.write(latex_table)\n",
    "\n",
    "root_dir = 'log/'\n",
    "configs = process_tfevents(root_dir,precomputed=True)\n",
    "configs\n",
    "config_to_latex(configs,share=True,variant=\"soft\")\n",
    "config_to_latex(configs,share=False,variant=\"soft\")\n",
    "config_to_latex(configs,share=True,variant=\"hard\")\n",
    "config_to_latex(configs,share=False,variant=\"hard\")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 64,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "1 soft \\cellcolor{white}\n",
      "2 soft \\cellcolor{white}\n",
      "3 soft \\cellcolor{gray!25}\n",
      "4 soft \\cellcolor{gray!25}\n",
      "5 soft \\cellcolor{white}\n",
      "6 soft \\cellcolor{white}\n",
      "7 soft \\cellcolor{gray!25}\n",
      "8 soft \\cellcolor{gray!25}\n",
      "9 soft \\cellcolor{white}\n",
      "10 soft \\cellcolor{white}\n",
      "11 soft \\cellcolor{gray!25}\n",
      "12 soft \\cellcolor{gray!25}\n",
      "13 soft \\cellcolor{white}\n",
      "14 soft \\cellcolor{white}\n",
      "15 soft \\cellcolor{gray!25}\n",
      "16 soft \\cellcolor{gray!25}\n",
      "17 soft \\cellcolor{white}\n",
      "18 soft \\cellcolor{white}\n"
     ]
    }
   ],
   "source": [
    "def config_to_latex_share_or_not(configs):\n",
    "    latex_table = \"\\\\begin{tabular}{ccccc}\\n\\\\toprule\\n \\\\textbf{Width} $C$ & \\\\textbf{Group} $G$ & \\\\textbf{Share Axis?} & \\\\textbf{Train Loss} & \\\\textbf{Eval Loss} \\\\\\\\\\n\\\\midrule\\n\"\n",
    "    configs = sorted(configs, key=lambda x: x.num_groups)\n",
    "    share = []\n",
    "    noshare = []\n",
    "    i=0\n",
    "    for config in configs:\n",
    "        if config.share_axis and config.variant == \"soft\":\n",
    "            cell_color = \"\\\\cellcolor{gray!25}\" if i % 4 >=2 else \"\\\\cellcolor{white}\"\n",
    "            i+=1\n",
    "            row_color = \"\"\n",
    "            print(i,config.variant,cell_color)\n",
    "            share.append(f\"{row_color}${config.latents}$&{cell_color}& Yes & $({config.min_train_loss_mean/100:.4f}\\\\pm{config.min_train_loss_std/100:.4f})\\\\times 10^2$ & $({config.min_eval_loss_mean/100:.4f}\\\\pm{config.min_eval_loss_std/100:.4f})\\\\times 10^2$ \\\\\\\\\\n\")\n",
    "        if config.share_axis == False and config.variant == \"soft\":\n",
    "            cell_color = \"\\\\cellcolor{gray!25}\" if i % 4 >=2 else \"\\\\cellcolor{white}\"\n",
    "            i+=1\n",
    "            row_color = \"\\\\rowcolor{gray!25}\"\n",
    "            print(i,config.variant,cell_color)\n",
    "            noshare.append(f\"{row_color}${config.latents}$ & \\\\multirow{{-2}}{{*}}{{{cell_color}${config.num_groups}$}} & No & $({config.min_train_loss_mean/100:.4f}\\\\pm{config.min_train_loss_std/100:.4f})\\\\times 10^2$ & $({config.min_eval_loss_mean/100:.4f}\\\\pm{config.min_eval_loss_std/100:.4f})\\\\times 10^2$ \\\\\\\\\\n\")\n",
    "    for s, ns in zip(share,noshare):\n",
    "        latex_table += s\n",
    "        latex_table += ns\n",
    "    latex_table += \"\\\\bottomrule\\n\\\\end{tabular}\\n\"\n",
    "    with open(f\"results/share_or_not.tex\", \"w\") as f:\n",
    "        f.write(latex_table)\n",
    "\n",
    "config_to_latex_share_or_not(configs)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 62,
   "metadata": {},
   "outputs": [],
   "source": [
    "def config_to_latex_soft_or_hard(configs):\n",
    "    latex_table = \"\\\\begin{tabular}{ccccc}\\n\\\\toprule\\n \\\\textbf{Width} $C$ & \\\\textbf{Group} $G$ & \\\\textbf{Projection Type} & \\\\textbf{Train Loss} & \\\\textbf{Eval Loss} \\\\\\\\\\n\\\\midrule\\n\"\n",
    "    configs = sorted(configs, key=lambda x: x.num_groups)\n",
    "    S = []\n",
    "    NS = []\n",
    "    i=0\n",
    "    for config in configs:\n",
    "        if config.share_axis and config.variant == \"soft\":\n",
    "            cell_color = \"\\\\cellcolor{gray!25}\" if i % 4 >=2 else \"\\\\cellcolor{white}\"\n",
    "            i+=1\n",
    "            row_color = \"\"\n",
    "            # print(i,config.variant,cell_color)\n",
    "            S.append(f\"{row_color}{cell_color}&{cell_color}&  Soft & $({config.min_train_loss_mean/100:.4f}\\\\pm{config.min_train_loss_std/100:.4f})\\\\times 10^2$ & $({config.min_eval_loss_mean/100:.4f}\\\\pm{config.min_eval_loss_std/100:.4f})\\\\times 10^2$ \\\\\\\\\\n\")\n",
    "        if config.share_axis and config.variant == \"hard\":\n",
    "            cell_color = \"\\\\cellcolor{gray!25}\" if i % 4 >=2 else \"\\\\cellcolor{white}\"\n",
    "            i+=1\n",
    "            row_color = \"\\\\rowcolor{gray!25}\"\n",
    "            # print(i,config.variant,cell_color)\n",
    "            NS.append(f\"{row_color}\\\\multirow{{-2}}{{*}}{{{cell_color}${config.latents}$}} & \\\\multirow{{-2}}{{*}}{{{cell_color}${config.num_groups}$}} &  Hard & $({config.min_train_loss_mean/100:.4f}\\\\pm{config.min_train_loss_std/100:.4f})\\\\times 10^2$ & $({config.min_eval_loss_mean/100:.4f}\\\\pm{config.min_eval_loss_std/100:.4f})\\\\times 10^2$ \\\\\\\\\\n\")\n",
    "    for s, ns in zip(S,NS):\n",
    "        latex_table += s\n",
    "        latex_table += ns\n",
    "    latex_table += \"\\\\bottomrule\\n\\\\end{tabular}\\n\"\n",
    "    with open(f\"results/soft_or_hard.tex\", \"w\") as f:\n",
    "        f.write(latex_table)\n",
    "\n",
    "config_to_latex_soft_or_hard(configs)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [
    {
     "ename": "NameError",
     "evalue": "name 'plt' is not defined",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mNameError\u001b[0m                                 Traceback (most recent call last)",
      "Cell \u001b[0;32mIn[1], line 49\u001b[0m\n\u001b[1;32m     45\u001b[0m     plt\u001b[38;5;241m.\u001b[39mfill_between(data[\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mstep\u001b[39m\u001b[38;5;124m'\u001b[39m],lower,upper,alpha\u001b[38;5;241m=\u001b[39m\u001b[38;5;241m0.3\u001b[39m,color\u001b[38;5;241m=\u001b[39m\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mgray\u001b[39m\u001b[38;5;124m'\u001b[39m)\n\u001b[1;32m     47\u001b[0m     plt\u001b[38;5;241m.\u001b[39mlegend(loc\u001b[38;5;241m=\u001b[39m\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mupper right\u001b[39m\u001b[38;5;124m'\u001b[39m)\n\u001b[0;32m---> 49\u001b[0m \u001b[43mplt\u001b[49m\u001b[38;5;241m.\u001b[39mrcParams[\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mlegend.fontsize\u001b[39m\u001b[38;5;124m'\u001b[39m] \u001b[38;5;241m=\u001b[39m \u001b[38;5;124m'\u001b[39m\u001b[38;5;124mxx-small\u001b[39m\u001b[38;5;124m'\u001b[39m\n\u001b[1;32m     50\u001b[0m root_dir \u001b[38;5;241m=\u001b[39m \u001b[38;5;124m'\u001b[39m\u001b[38;5;124mlog\u001b[39m\u001b[38;5;124m'\u001b[39m\n\u001b[1;32m     51\u001b[0m plot_tfevents(root_dir,name1\u001b[38;5;241m=\u001b[39m\u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mReLU\u001b[39m\u001b[38;5;124m\"\u001b[39m)\n",
      "\u001b[0;31mNameError\u001b[0m: name 'plt' is not defined"
     ]
    }
   ],
   "source": [
    "import re\n",
    "from tensorboard.backend.event_processing.event_accumulator import EventAccumulator\n",
    "DEFAULT_SIZE_GUIDANCE = {\n",
    "    \"compressedHistograms\": 1,\n",
    "    \"images\": 1,\n",
    "    \"scalars\": 0,  # 0 means load all\n",
    "    \"histograms\": 1,\n",
    "}\n",
    "def load_tfevents(path):\n",
    "    event_acc = EventAccumulator(path, DEFAULT_SIZE_GUIDANCE)\n",
    "    event_acc.Reload()\n",
    "    tags = event_acc.Tags()[\"scalars\"]\n",
    "    runlog_data = pd.DataFrame({\"metric\": [], \"value\": [], \"step\": []})\n",
    "    for tag in tags:\n",
    "        event_list = event_acc.Scalars(tag)\n",
    "        values = list(map(lambda x: x.value, event_list))\n",
    "        step = list(map(lambda x: x.step, event_list))\n",
    "        r = {\"metric\": [tag] * len(step), \"value\": values, \"step\": step}\n",
    "        r = pd.DataFrame(r)\n",
    "        runlog_data = pd.concat([runlog_data, r])\n",
    "    return runlog_data\n",
    "    \n",
    "def plot_tfevents(root_dir,name0 = \"Identity\",name1 = \"SiLU\"):\n",
    "    # load all tfevents and index by group size\n",
    "    filenames = sorted(glob.glob(f\"{root_dir}/*/*/*/*events*\"))\n",
    "    dfs = {}\n",
    "    for path in filenames:\n",
    "        df = load_tfevents(path)\n",
    "        match = re.search(r'group(\\d+)_', path)\n",
    "        num_groups = int(match.group(1)) if match else 0\n",
    "        dfs[num_groups] = df\n",
    "    \n",
    "    # plot\n",
    "    plt.figure(figsize=(4,3))\n",
    "    for name, df in sorted(dfs.items()):\n",
    "        data = df.loc[df['metric'] == 'eval/loss']\n",
    "        if name == 0:\n",
    "            name = name0\n",
    "            upper = data['value']\n",
    "        if name == 2400:\n",
    "            name = name1\n",
    "            lower = data['value']\n",
    "        plt.plot(data['step'], data['value'], '-',label=name)\n",
    "\n",
    "    plt.fill_between(data['step'],lower,upper,alpha=0.3,color='gray')\n",
    "\n",
    "    plt.legend(loc='upper right')\n",
    "\n",
    "plt.rcParams['legend.fontsize'] = 'xx-small'\n",
    "root_dir = 'log'\n",
    "plot_tfevents(root_dir,name1=\"ReLU\")\n",
    "plt.xlabel('Epoch')\n",
    "plt.ylabel('Loss')\n",
    "plt.ylim(95,135)\n",
    "# plt.savefig('results/relu-share-axis.pdf',bbox_inches='tight')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 400x300 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "root_dir = 'log/C2401-soft-shareaxis'\n",
    "plot_tfevents(root_dir,name1=\"SiLU\")\n",
    "plt.xlabel('Epoch')\n",
    "plt.ylabel('Loss')\n",
    "plt.ylim(95,135)\n",
    "plt.savefig('results/silu-share-axis.pdf',bbox_inches='tight')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "import pandas as pd\n",
    "import re\n",
    "import glob\n",
    "from tensorboard.backend.event_processing.event_accumulator import EventAccumulator\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "DEFAULT_SIZE_GUIDANCE = {\n",
    "    \"compressedHistograms\": 1,\n",
    "    \"images\": 1,\n",
    "    \"scalars\": 0,  # 0 means load all\n",
    "    \"histograms\": 1,\n",
    "}\n",
    "\n",
    "\n",
    "def plot_tfevents(root_dir, name0=\"Identity\", name1=\"SiLU\"):\n",
    "    # Load all tfevents and index by group size\n",
    "    filenames = sorted(glob.glob(f\"{root_dir}/*/*events*\"))\n",
    "    dfs = {}\n",
    "    min_losses = {}\n",
    "    \n",
    "    for path in filenames:\n",
    "        df = load_tfevents(path)\n",
    "        match = re.search(r'group(\\d+)_', path)\n",
    "        num_groups = int(match.group(1)) if match else 0\n",
    "        dfs[num_groups] = df\n",
    "\n",
    "        # Extract the minimum loss value for each group\n",
    "        min_loss = df[df['metric'] == 'eval/loss']['value'].min()\n",
    "        min_losses[num_groups] = min_loss\n",
    "    \n",
    "    # Generate LaTeX table\n",
    "    latex_table = \"\\\\begin{table}[h]\\n\\\\centering\\n\\\\begin{tabular}{|c|c|}\\n\\\\hline\\nGroup & L2 Loss \\\\\\\\\\n\\\\hline\\n\"\n",
    "    for group, min_loss in sorted(min_losses.items()):\n",
    "        latex_table += f\"{group} & {min_loss:.4f} & {min_bce:.4f}\\\\\\\\\\n\\\\hline\\n\"\n",
    "    latex_table += \"\\\\end{tabular}\\n\\\\caption{Minimum Loss Values}\\n\\\\end{table}\"\n",
    "    \n",
    "    with open(\"min_losses_table.tex\", \"w\") as f:\n",
    "        f.write(latex_table)\n",
    "    \n",
    "\n",
    "# plt.rcParams['legend.fontsize'] = 'xx-small'\n",
    "plot_tfevents(root_dir, name1=\"SiLU\")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "metric   eval/bce   eval/kld   eval/loss  train/grad_norm  train/loss\n",
      "step                                                                 \n",
      "0       97.605545  27.388542  124.994087        83.450333  560.351013\n",
      "1       92.622581  28.739359  121.361938       122.658203  559.804626\n",
      "2       90.350151  28.768597  119.118744       158.466446  507.403259\n",
      "3       88.574249  28.288853  116.863098       329.735779  424.912323\n",
      "4       87.401154  27.884964  115.286118       317.862701  328.711823\n"
     ]
    }
   ],
   "source": [
    "def load_tfevents(path):\n",
    "    event_acc = EventAccumulator(path, DEFAULT_SIZE_GUIDANCE)\n",
    "    event_acc.Reload()\n",
    "    tags = event_acc.Tags()[\"scalars\"]\n",
    "    \n",
    "    records = []\n",
    "    for tag in tags:\n",
    "        event_list = event_acc.Scalars(tag)\n",
    "        for event in event_list:\n",
    "            records.append((event.step, tag, event.value))\n",
    "    \n",
    "    df = pd.DataFrame.from_records(records, columns=['step', 'metric', 'value'])\n",
    "    return df.pivot(index='step', columns='metric', values='value')\n",
    "\n",
    "\n",
    "print(df.head())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [
    {
     "ename": "UnboundLocalError",
     "evalue": "local variable 'data' referenced before assignment",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mUnboundLocalError\u001b[0m                         Traceback (most recent call last)",
      "Cell \u001b[0;32mIn[11], line 68\u001b[0m\n\u001b[1;32m     65\u001b[0m     plt\u001b[38;5;241m.\u001b[39msavefig(\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mresults/relu-share-axis.pdf\u001b[39m\u001b[38;5;124m'\u001b[39m, bbox_inches\u001b[38;5;241m=\u001b[39m\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mtight\u001b[39m\u001b[38;5;124m'\u001b[39m)\n\u001b[1;32m     67\u001b[0m root_dir \u001b[38;5;241m=\u001b[39m \u001b[38;5;124m'\u001b[39m\u001b[38;5;124mlog-epoch2000/colu_shareaxis_True_soft/C2401_group800/05_15_06:04\u001b[39m\u001b[38;5;124m'\u001b[39m\n\u001b[0;32m---> 68\u001b[0m \u001b[43mplot_tfevents\u001b[49m\u001b[43m(\u001b[49m\u001b[43mroot_dir\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mname1\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;124;43m\"\u001b[39;49m\u001b[38;5;124;43mReLU\u001b[39;49m\u001b[38;5;124;43m\"\u001b[39;49m\u001b[43m)\u001b[49m\n",
      "Cell \u001b[0;32mIn[11], line 60\u001b[0m, in \u001b[0;36mplot_tfevents\u001b[0;34m(root_dir, name0, name1)\u001b[0m\n\u001b[1;32m     57\u001b[0m         lower \u001b[38;5;241m=\u001b[39m data\n\u001b[1;32m     58\u001b[0m     plt\u001b[38;5;241m.\u001b[39mplot(data\u001b[38;5;241m.\u001b[39mindex, data\u001b[38;5;241m.\u001b[39mvalues, \u001b[38;5;124m'\u001b[39m\u001b[38;5;124m-\u001b[39m\u001b[38;5;124m'\u001b[39m, label\u001b[38;5;241m=\u001b[39mname)\n\u001b[0;32m---> 60\u001b[0m plt\u001b[38;5;241m.\u001b[39mfill_between(\u001b[43mdata\u001b[49m\u001b[38;5;241m.\u001b[39mindex, lower, upper, alpha\u001b[38;5;241m=\u001b[39m\u001b[38;5;241m0.3\u001b[39m, color\u001b[38;5;241m=\u001b[39m\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mgray\u001b[39m\u001b[38;5;124m'\u001b[39m)\n\u001b[1;32m     61\u001b[0m plt\u001b[38;5;241m.\u001b[39mlegend(loc\u001b[38;5;241m=\u001b[39m\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mupper right\u001b[39m\u001b[38;5;124m'\u001b[39m)\n\u001b[1;32m     62\u001b[0m plt\u001b[38;5;241m.\u001b[39mxlabel(\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mEpoch\u001b[39m\u001b[38;5;124m'\u001b[39m)\n",
      "\u001b[0;31mUnboundLocalError\u001b[0m: local variable 'data' referenced before assignment"
     ]
    },
    {
     "data": {
      "text/plain": [
       "<Figure size 400x300 with 0 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import pandas as pd\n",
    "import re\n",
    "import glob\n",
    "from tensorboard.backend.event_processing.event_accumulator import EventAccumulator\n",
    "\n",
    "DEFAULT_SIZE_GUIDANCE = {\n",
    "    \"compressedHistograms\": 1,\n",
    "    \"images\": 1,\n",
    "    \"scalars\": 0,  # 0 means load all\n",
    "    \"histograms\": 1,\n",
    "}\n",
    "\n",
    "def load_tfevents(path):\n",
    "    event_acc = EventAccumulator(path, DEFAULT_SIZE_GUIDANCE)\n",
    "    event_acc.Reload()\n",
    "    tags = event_acc.Tags()[\"scalars\"]\n",
    "    \n",
    "    records = []\n",
    "    for tag in tags:\n",
    "        event_list = event_acc.Scalars(tag)\n",
    "        for event in event_list:\n",
    "            records.append((event.step, tag, event.value))\n",
    "    \n",
    "    df = pd.DataFrame.from_records(records, columns=['step', 'metric', 'value'])\n",
    "    return df.pivot(index='step', columns='metric', values='value')\n",
    "\n",
    "def plot_tfevents(root_dir, name0=\"Identity\", name1=\"SiLU\"):\n",
    "    filenames = sorted(glob.glob(f\"{root_dir}/*/*events*\"))\n",
    "    dfs = {}\n",
    "    min_losses = {}\n",
    "    \n",
    "    for path in filenames:\n",
    "        df = load_tfevents(path)\n",
    "        match = re.search(r'group(\\d+)_', path)\n",
    "        num_groups = int(match.group(1)) if match else 0\n",
    "        dfs[num_groups] = df\n",
    "\n",
    "        min_loss = df['eval/loss'].min()\n",
    "        min_losses[num_groups] = min_loss\n",
    "    \n",
    "    \n",
    "    plt.figure(figsize=(4, 3))\n",
    "    for name, df in sorted(dfs.items()):\n",
    "        data = df['eval/loss']\n",
    "        if name == 0:\n",
    "            name = name0\n",
    "            upper = data\n",
    "        if name == 2400:\n",
    "            name = name1\n",
    "            lower = data\n",
    "        plt.plot(data.index, data.values, '-', label=name)\n",
    "    \n",
    "    plt.fill_between(data.index, lower, upper, alpha=0.3, color='gray')\n",
    "    plt.legend(loc='upper right')\n",
    "    plt.xlabel('Epoch')\n",
    "    plt.ylabel('Loss')\n",
    "    plt.ylim(95, 135)\n",
    "    plt.savefig('results/relu-share-axis.pdf', bbox_inches='tight')\n",
    "\n",
    "root_dir = 'log-epoch2000/colu_shareaxis_True_soft/C2401_group800/05_15_06:04'\n",
    "plot_tfevents(root_dir, name1=\"ReLU\")\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Optimal orthogonal matrices \n",
    "Note: [It's not an SOCP](https://www.cvxpy.org/examples/basic/socp.html)\n",
    "\n",
    "\\begin{split}\\begin{array}{ll}\n",
    "\\textrm{minimize}_{x\\in\\mathbb R^n}   & \\langle C, x\\rangle_F \\\\\n",
    "\\textrm{subject to} & x^\\top x=I,\n",
    "\\end{array}\\end{split}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "'0.1781394338607788±0.038357503358607686'"
      ]
     },
     "execution_count": 15,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "import jax.numpy as jnp\n",
    "import pymanopt\n",
    "import jax\n",
    "\n",
    "key = jax.random.PRNGKey(0)\n",
    "S_1 = 3\n",
    "A = jax.random.normal(key,(S_1, S_1))\n",
    "\n",
    "def optimize(A):\n",
    "    # Define the manifold\n",
    "    manifold = pymanopt.manifolds.Stiefel(*A.shape)\n",
    "    # Define the objective function\n",
    "    @pymanopt.function.jax(manifold)\n",
    "    def objective(X):\n",
    "        return -jnp.linalg.norm(A * X,\"fro\")  # We use negative sign to demonstrate minimization\n",
    "    # Create the problem instance on the manifold with the objective function\n",
    "    problem = pymanopt.Problem(manifold=manifold, cost=objective)\n",
    "    # Select a solver\n",
    "    solver = pymanopt.optimizers.ConjugateGradient(verbosity=0,min_gradient_norm=1e-4)\n",
    "    # Solve the problem\n",
    "    return solver.run(problem)\n",
    "\n",
    "# test it \n",
    "data = []\n",
    "for i in range(100):\n",
    "    result = optimize(A)\n",
    "    data.append(result.time)\n",
    "data=jnp.array(data)\n",
    "f\"time per cone alignment: {jnp.mean(data):.4f}±{jnp.std(data):.4f}\""
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "from concurrent.futures import ThreadPoolExecutor\n",
    "from tqdm import tqdm\n",
    "def align(xxx, C, G=100, parallel=True):\n",
    "    \"\"\"trivial alignment: test the method by aligning the model towards itself\"\"\"\n",
    "    \"\"\"w/ axis sharing, not group-wise\"\"\"\n",
    "    p = jnp.eye(C)\n",
    "    S_1 = (C - 1) // G\n",
    "    blocks = [xxx[(1+S_1*i):(1+S_1*(i+1)),(1+S_1*i):(1+S_1*(i+1))] for i in range(G)]\n",
    "    if parallel:\n",
    "        with ThreadPoolExecutor(max_workers=G) as executor:\n",
    "            results = list(tqdm(executor.map(optimize, blocks), total=len(blocks)))\n",
    "        for i, result in enumerate(results):\n",
    "            p = p.at[(1+S_1*i):(1+S_1*(i+1)),(1+S_1*i):(1+S_1*(i+1))].set(jax.device_put(result.point))\n",
    "    else:\n",
    "        for i, block in tqdm(enumerate(blocks), total=len(blocks)):\n",
    "            result = optimize(block)\n",
    "            p = p.at[(1+S_1*i):(1+S_1*(i+1)),(1+S_1*i):(1+S_1*(i+1))].set(jax.device_put(result.point))\n",
    "    return p\n",
    "\n",
    "p = align(xxx,C=config.latents, G=config.num_groups,parallel=False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "after:\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/evergreen/.local/lib/python3.8/site-packages/jax/_src/ops/scatter.py:92: FutureWarning: scatter inputs have incompatible types: cannot safely cast value from dtype=float64 to dtype=float32. In future JAX releases this will result in an error.\n",
      "  warnings.warn(\"scatter inputs have incompatible types: cannot safely cast \"\n"
     ]
    },
    {
     "data": {
      "image/jpeg": "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",
      "image/png": "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",
      "text/plain": [
       "<PIL.Image.Image image mode=RGB size=242x62>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "before:\n"
     ]
    },
    {
     "data": {
      "image/jpeg": "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",
      "image/png": "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",
      "text/plain": [
       "<PIL.Image.Image image mode=RGB size=242x62>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "params1 = prune(params,p)\n",
    "print(\"after:\")\n",
    "metrics, comparison, sample = train.eval_f(params1, test_ds, z, eval_rng, model, config)\n",
    "display(vae_utils.save_image(comparison, f'results/reconstruction.png', nrow=8))\n",
    "\n",
    "print(\"before:\")\n",
    "metrics, comparison, sample = train.eval_f(params, test_ds, z, eval_rng, model, config)\n",
    "display(vae_utils.save_image(comparison, f'results/reconstruction.png', nrow=8))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## To be continued in the next notebook"
   ]
  }
 ],
 "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.11.9"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
