{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "c98385ef-94bf-424e-a767-c918c303a225",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/state/partition1/llgrid/pkg/anaconda/anaconda3-2021b/lib/python3.8/_collections_abc.py:832: MatplotlibDeprecationWarning: Support for setting the 'text.latex.preamble' or 'pgf.preamble' rcParam to a list of strings is deprecated since 3.3 and will be removed two minor releases later; set it to a single string instead.\n",
      "  self[key] = other[key]\n"
     ]
    }
   ],
   "source": [
    "import e2cnn\n",
    "from e2cnn import gspaces\n",
    "from e2cnn import nn as e2nn\n",
    "import torch\n",
    "import torch.multiprocessing\n",
    "from torch import nn\n",
    "import torch.optim as optim\n",
    "from torchvision import datasets, transforms\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "from matplotlib import pyplot as plt\n",
    "from src.nn_training import Group\n",
    "from tqdm.notebook import tqdm\n",
    "import seaborn as sns\n",
    "sns.set_style()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "c49891b6-9d6c-4d30-9fa7-08af8de750ab",
   "metadata": {},
   "outputs": [],
   "source": [
    "transform_comp = transforms.Compose([transforms.ToTensor(),\n",
    "                                     transforms.Resize((28, 28)),\n",
    "                                     transforms.Normalize((0.1307,), (0.3081,))])\n",
    "mnist_data = datasets.MNIST('./data', transform=transform_comp)\n",
    "\n",
    "batch_size = 250\n",
    "data_loader = torch.utils.data.DataLoader(mnist_data,\n",
    "                                          batch_size=batch_size,\n",
    "                                          shuffle=False,\n",
    "                                          num_workers=20)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "55f81741-b1e4-433d-bdb5-c358666a3fa7",
   "metadata": {},
   "outputs": [],
   "source": [
    "for i, batch in enumerate(data_loader):\n",
    "    x, y = batch\n",
    "    break\n",
    "x = x[0:1, ...]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "4e0f28b7-b4e1-4be7-b748-b0f5be6c824b",
   "metadata": {},
   "outputs": [],
   "source": [
    "def get_schatten_norm_sum(fcoeffs, irrep_sizes, l):\n",
    "    \"\"\"\n",
    "    Given the Fourier coefficients of the linearization of our network,\n",
    "    this f-n computes the expression for which we want to reach a scaling\n",
    "    of its stationary point in our main theorems (Thm 4.2 and Thm 4.4).\n",
    "    \"\"\"\n",
    "    \n",
    "    # all we're doing here is getting the 2/l norm of the singular values\n",
    "    # of the block diagonal matrix that has each of the irreps of dim n\n",
    "    # repeated n times along its diagonal\n",
    "    \n",
    "    # our code could also be written more cleanly/concisely as\n",
    "    # block_diag_matrix_irrs = torch.block_diag(*[torch.block_diag(*([flat_irrep.reshape(size, size)] * size)) \n",
    "    #                                                              for size, flat_irrep in zip(irrep_sizes, fcoeffs)])\n",
    "    # sigmas = torch.linalg.svdvals(block_diag_matrix_irrs)\n",
    "    # return torch.linalg.vector_norm(sigmas, ord=(2./3.))\n",
    "    # but that'd be a bit slower because of the SVD on a larger (though sparse) matrix\n",
    "    \n",
    "    sum_of_schatten_norms = 0.\n",
    "    flat_irreps_list = torch.split(fcoeffs, [s ** 2 for s in irrep_sizes])\n",
    "\n",
    "    for flat_irrep, irrep_size in zip(flat_irreps_list, irrep_sizes):\n",
    "        irrep = flat_irrep.reshape(irrep_size, irrep_size)\n",
    "        try:\n",
    "            sigmas = torch.linalg.svdvals(irrep)\n",
    "        except:\n",
    "            return None\n",
    "        sum_of_schatten_norms += irrep_size * torch.linalg.vector_norm(sigmas, ord=(2. / l)) ** (2. / l)\n",
    "    return sum_of_schatten_norms ** (l / 2.)\n",
    "\n",
    "def eval_schatten_norm(net, inps):\n",
    "    group = Group('MNIST')\n",
    "    fourier_m = group.f_mat\n",
    "    net.eval()\n",
    "    sch_norms = []\n",
    "    for i in range(1):\n",
    "        interm, out = net(inps[i:i + 1])\n",
    "        out.backward()\n",
    "        linearization = interm.grad[0].permute([1, 2, 0])\n",
    "        # re-order to match GAP order\n",
    "        linearization = linearization[:, :, [0, 2, 1, 3, 4, 6, 5, 7]].reshape(-1, 1) - 1.\n",
    "        linearization = linearization / torch.linalg.norm(linearization)\n",
    "        linearization = torch.complex(linearization, torch.zeros_like(linearization))\n",
    "        fspace_linearization = torch.conj(fourier_m).T @ linearization\n",
    "        sch_norm = get_schatten_norm_sum(fspace_linearization, group.irrep_sizes, 3)\n",
    "        sch_norms.append(sch_norm)\n",
    "    net.train()\n",
    "    return torch.stack(sch_norms).mean()\n",
    "\n",
    "def exp_loss(y_pred, y):\n",
    "    \"\"\"\n",
    "    Exponential loss for classification.\n",
    "    \"\"\"\n",
    "    return torch.mean(torch.exp(-y_pred * y))\n",
    "\n",
    "def postprocess(batch):\n",
    "    x, y = batch\n",
    "    # classify 0 vs 5, ignore other digits\n",
    "    x = x[(y == 0) + (y == 5)]\n",
    "    y = y[(y == 0) + (y == 5)]\n",
    "    y[y == 5] = -1\n",
    "    y[y == 0] = 1\n",
    "    y = y.reshape(-1, 1)\n",
    "    reduced_bs = len(y)\n",
    "    return x, y"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "69084464-333c-4794-b59b-59fddb552e2e",
   "metadata": {},
   "outputs": [],
   "source": [
    "class Gcnn(torch.nn.Module):\n",
    "    def __init__(self):\n",
    "        super(Gcnn, self).__init__()\n",
    "        self.r2_act = gspaces.FlipRot2dOnR2(N=4)\n",
    "\n",
    "        self.feat_type_in  = e2nn.FieldType(self.r2_act, [self.r2_act.trivial_repr])\n",
    "        self.feat_type_out = e2nn.FieldType(self.r2_act, [self.r2_act.regular_repr])\n",
    "        self.conv1 = e2nn.R2Conv(self.feat_type_in, self.feat_type_out,\n",
    "                                 kernel_size=25, padding=12,\n",
    "                                 padding_mode='circular', bias=False)\n",
    "        self.conv2 = e2nn.R2Conv(self.feat_type_out, self.feat_type_out,\n",
    "                                 kernel_size=15, padding=0,\n",
    "                                 padding_mode='circular', bias=False)\n",
    "        self.conv3 = e2nn.R2Conv(self.feat_type_out, self.feat_type_out,\n",
    "                                 kernel_size=13, padding=0,\n",
    "                                 padding_mode='circular', bias=False)\n",
    "        self.relu = e2nn.ELU(self.feat_type_out)\n",
    "        self.pooling = e2nn.GroupPooling(self.feat_type_out)\n",
    "        self.batchnorm = e2nn.InnerBatchNorm(self.feat_type_out)\n",
    "\n",
    "    def forward(self, x):\n",
    "        bs = x.size(0)\n",
    "        x = e2nn.GeometricTensor(x, self.feat_type_in)\n",
    "        interm = self.conv1(x)\n",
    "        interm.tensor.retain_grad()\n",
    "#         x = self.batchnorm(interm)\n",
    "#         x = self.relu(x)\n",
    "        x = self.relu(interm)\n",
    "        x = self.conv2(x)\n",
    "#         x = self.batchnorm(x)\n",
    "        x = self.relu(x)\n",
    "        x = self.conv3(x)\n",
    "#         x = self.batchnorm(x)\n",
    "#         x = self.pooling(x)\n",
    "\n",
    "        return interm.tensor, torch.mean(x.tensor.reshape(bs, -1), axis=1, keepdim=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "623a3339-a87a-41f9-b657-60e9d4d11086",
   "metadata": {},
   "outputs": [],
   "source": [
    "class Cnn(torch.nn.Module):\n",
    "    def __init__(self):\n",
    "        super(Cnn, self).__init__()\n",
    "        self.r2_act = gspaces.TrivialOnR2()\n",
    "\n",
    "        self.feat_type_in  = e2nn.FieldType(self.r2_act, [self.r2_act.regular_repr])\n",
    "        self.feat_type_out = e2nn.FieldType(self.r2_act, 8 * [self.r2_act.regular_repr])\n",
    "        self.conv1 = e2nn.R2Conv(self.feat_type_in, self.feat_type_out,\n",
    "                                 kernel_size=25, padding=12, \n",
    "                                 padding_mode='circular', bias=False)\n",
    "        self.conv2 = e2nn.R2Conv(self.feat_type_out, self.feat_type_out,\n",
    "                                 kernel_size=15, bias=False)\n",
    "        self.conv3 = e2nn.R2Conv(self.feat_type_out, self.feat_type_out,\n",
    "                                 kernel_size=13, bias=False)\n",
    "        self.relu = e2nn.ELU(self.feat_type_out)\n",
    "        self.pooling = e2nn.GroupPooling(self.feat_type_out, )\n",
    "\n",
    "    def forward(self, x):\n",
    "        bs = x.size(0)\n",
    "        x = e2nn.GeometricTensor(x, self.feat_type_in)\n",
    "        interm = self.conv1(x)\n",
    "        interm.tensor.retain_grad()\n",
    "        x = self.relu(interm)\n",
    "        x = self.conv2(x)\n",
    "        x = self.relu(x)\n",
    "        x = self.conv3(x)\n",
    "\n",
    "        return interm.tensor, torch.mean(x.tensor.reshape(bs, -1), axis=1, keepdim=True)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "7b036d5f-7e9c-4a3d-86d6-a069aab6cb27",
   "metadata": {},
   "source": [
    "# Training"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 65,
   "id": "c7c80a1e-ab91-4f44-b637-7ab09daec33b",
   "metadata": {},
   "outputs": [],
   "source": [
    "epochs = 300\n",
    "num_batches = 1\n",
    "init_lr= 1e-3\n",
    "N = 15"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 66,
   "id": "0740152b-caa4-405a-a161-25e9a30b109b",
   "metadata": {},
   "outputs": [],
   "source": [
    "dataset = [postprocess(batch) for batch in data_loader]"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "27b9868f-b57f-4c65-9462-c5cd3be102df",
   "metadata": {},
   "source": [
    "## G-CNN"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 76,
   "id": "8288c441-6cfc-4d51-bde4-d8f326897a66",
   "metadata": {
    "tags": []
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "CPU times: user 17h 28min 21s, sys: 58min 56s, total: 18h 27min 18s\n",
      "Wall time: 30min 18s\n"
     ]
    }
   ],
   "source": [
    "%%time\n",
    "results_gcnn = []\n",
    "for run_itr in range(N):\n",
    "    net = Gcnn().train()\n",
    "    optimizer = optim.SGD(net.parameters(), lr=init_lr)\n",
    "    loss = torch.Tensor([1.])\n",
    "    acc = torch.Tensor([0.])\n",
    "    for epoch in range(epochs):\n",
    "        net.train()\n",
    "#         itr = tqdm(enumerate(dataset[0:num_batches]))\n",
    "        itr = enumerate(dataset[0:num_batches])\n",
    "        if epoch == 150:\n",
    "            for g in optimizer.param_groups:\n",
    "                g['lr'] /= 5.\n",
    "        for i, batch in itr:\n",
    "            x, y = batch\n",
    "            if epoch % 5 == 0 and i == 0:\n",
    "                sch_norm = eval_schatten_norm(net, x)\n",
    "#                 print(f'Schatten norm at epoch {epoch} is {sch_norm:.1f}; acc: {acc.item()}')\n",
    "                results_gcnn.append([run_itr, epoch, sch_norm.item(), loss.item()])\n",
    "\n",
    "            optimizer.zero_grad()\n",
    "            interm, preds = net(x)\n",
    "#             loss = (torch.abs(torch.sigmoid(preds) - y)).mean()\n",
    "            loss = exp_loss(preds, y)\n",
    "            loss.backward()\n",
    "            optimizer.step()\n",
    "\n",
    "            acc = torch.sum(torch.sign(preds) == y) / len(y)\n",
    "#             itr.set_description(f'loss: {loss.item():.4f}, acc: {acc.item():.3f}')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 77,
   "id": "619984d8-0d5b-4d01-9df5-1e9cc6f638ee",
   "metadata": {},
   "outputs": [],
   "source": [
    "dfs_gcnn = pd.DataFrame(results_gcnn, columns=['N', 'epoch', 'schatten norm', 'loss'])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 95,
   "id": "219f0d1b-6763-49ed-bfa3-2da1d8dcb4e5",
   "metadata": {},
   "outputs": [],
   "source": [
    "dfs_gcnn.to_csv('gcnn_e2.csv', index=False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 90,
   "id": "d7009d4b-e1e3-4460-bbd5-82fc19b73b6f",
   "metadata": {},
   "outputs": [],
   "source": [
    "dfs_gcnn = pd.read_csv('gcnn_e2.csv')"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "ec19fa08-2eec-48ca-81fd-a4db3865c6dc",
   "metadata": {},
   "source": [
    "## CNN"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 80,
   "id": "a4f8aa6e-a041-4e29-b3df-4ecbabaeaca7",
   "metadata": {
    "tags": []
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "CPU times: user 16h 20min 17s, sys: 56min 15s, total: 17h 16min 33s\n",
      "Wall time: 27min 45s\n"
     ]
    }
   ],
   "source": [
    "%%time\n",
    "results_cnn = []\n",
    "for run_itr in range(N):\n",
    "    net = Cnn()\n",
    "    optimizer = optim.SGD(net.parameters(), lr=init_lr)\n",
    "    loss = torch.Tensor([1.])\n",
    "    acc = torch.Tensor([0.])    \n",
    "    for epoch in range(epochs):\n",
    "#         itr = tqdm(enumerate(dataset[0:num_batches]))\n",
    "        itr = enumerate(dataset[0:num_batches])\n",
    "        if epoch == 150:\n",
    "            for g in optimizer.param_groups:\n",
    "                g['lr'] /= 5.\n",
    "        for i, batch in itr:\n",
    "            x, y = batch\n",
    "            if epoch % 5 == 0 and i == 0:\n",
    "                sch_norm = eval_schatten_norm(net, x)\n",
    "#                 print(f'Schatten norm at epoch {epoch} is {sch_norm:.1f}; acc is {acc.item()}')\n",
    "                results_cnn.append([run_itr, epoch, sch_norm.item(), loss.item()])\n",
    "\n",
    "            optimizer.zero_grad()\n",
    "            interm, preds = net(x)\n",
    "            loss = exp_loss(preds, y)\n",
    "            loss.backward()\n",
    "            optimizer.step()\n",
    "            acc = torch.sum(torch.sign(preds - 1.) == y) / len(y)\n",
    "#             itr.set_description(f'loss: {loss.item():.4f}, acc: {acc.item():.3f}')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 81,
   "id": "ead6574d-42f4-4fcd-8138-1fd3b36cb800",
   "metadata": {},
   "outputs": [],
   "source": [
    "dfs_cnn = pd.DataFrame(results_cnn, columns=['N', 'epoch', 'schatten norm', 'loss'])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 94,
   "id": "b4b072b1-2232-4cd2-a52f-a966a30e2e72",
   "metadata": {},
   "outputs": [],
   "source": [
    "dfs_cnn.to_csv('cnn_e2.csv', index=False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 91,
   "id": "c7b1d279-03aa-4c52-a832-93baa8185992",
   "metadata": {},
   "outputs": [],
   "source": [
    "dfs_cnn = pd.read_csv('cnn_e2.csv')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "8909f869-4dbd-4f0f-9c58-35e8e8a25c80",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 102,
   "id": "3d71a502-e61c-4ae1-bf9c-3870646a9747",
   "metadata": {},
   "outputs": [],
   "source": [
    "scaling=1.8"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 103,
   "id": "e6d18e90-39f7-4ab2-9758-1a1c3470e075",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 485.1x299.88 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# sns.lineplot(dfs_gcnn[\"epoch\"], dfs_gcnn[\"schatten_norm\"], label='G-CNN')\n",
    "plt.figure(figsize=(scaling * 2.695, scaling * 1.666))\n",
    "sns.lineplot(data=dfs_gcnn, x=\"epoch\", y=\"schatten norm\", label='G-CNN')\n",
    "sns.lineplot(data=dfs_cnn, x=\"epoch\", y=\"schatten norm\", label='CNN')\n",
    "plt.ylabel(r'$\\displaystyle\\|\\widehat{\\boldsymbol{\\beta}}\\|^{(S)}_{2/L}$')\n",
    "plt.legend()\n",
    "plt.savefig('e2_fourier_norm.pdf')\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "4d0af74c-b965-48e9-a186-4ed2d53a21ac",
   "metadata": {},
   "outputs": [],
   "source": [
    "plt.figure(figsize=(scaling * 2.695, scaling * 1.666))\n",
    "sns.lineplot(data=dfs_gcnn, x=\"epoch\", y=\"loss\", label='G-CNN')\n",
    "sns.lineplot(data=dfs_cnn, x=\"epoch\", y=\"loss\", label='CNN')\n",
    "plt.legend();\n",
    "plt.savefig('e2_loss.pdf')\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "36b9401d-1443-45e4-bda7-d1921e432653",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python [conda env:root] *",
   "language": "python",
   "name": "conda-root-py"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.8.11"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
