{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Details\n",
    "\n",
    "The notebook contains the synthetic data experiment with high noise (sigma^2 = 0.5) and low noise (sigma^2 = 00.5) for HNTF (Cichocki et al.). Due to the non-convexity of the HNCPD task, results will vary run to run, so the results from this notebook will not exactly match those in the paper."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "# loading packages and functions\n",
    "import sys\n",
    "\n",
    "sys.path.append(\"./src\")\n",
    "import torch\n",
    "import numpy as np\n",
    "from matplotlib import pyplot as plt\n",
    "\n",
    "from NNCPD import outer_product, outer_product_np\n",
    "#\n",
    "import torch.nn as nn\n",
    "from torch.autograd import Variable\n",
    "from writer import Writer\n",
    "\n",
    "from sklearn.decomposition import NMF\n",
    "\n",
    "import tensorly as tl\n",
    "from tensorly import unfold as tl_unfold\n",
    "from tensorly.decomposition import parafac, non_negative_parafac"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Generate Data Tensor (Low Noise)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "## set the network parameters\n",
    "torch.set_default_tensor_type(torch.DoubleTensor)\n",
    "\n",
    "\n",
    "n1 = 40\n",
    "n2 = 40\n",
    "n3 = 40\n",
    "\n",
    "r = 12\n",
    "\n",
    "\n",
    "a = 1\n",
    "b = 3\n",
    "\n",
    "X = np.zeros((40,40,40))\n",
    "\n",
    "\n",
    "X[0:10,0:15,0:15] = a * np.ones((10,15,15))\n",
    "X[10:25,15:25,15:30] = a * np.ones((15,10,15))\n",
    "X[25:40,25:40,30:40] = a * np.ones((15,15,10))\n",
    "\n",
    "X[0:5,0:10,0:5] = b\n",
    "X[5:10,10:15,5:15] = b\n",
    "\n",
    "X[10:15,15:18,15:20] = b\n",
    "X[15:20,18:23,20:25] = b\n",
    "X[20:25,23:25,25:30] = b\n",
    "\n",
    "X[25:30,25:35,30:35] = b\n",
    "X[30:40,35:40,35:40] = b\n",
    "\n",
    "#add Gaussian Noise\n",
    "np.random.seed(1)\n",
    "X = X + 0.05 * np.abs(np.random.randn(40, 40, 40))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "r=7\n",
    "factors_tl = non_negative_parafac(X, r)[1]\n",
    "\n",
    "X1 =  factors_tl[0]\n",
    "X2 =  factors_tl[1]\n",
    "X3 =  factors_tl[2]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "#set a consistent vmin and vmax for visualization\n",
    "vmin=0\n",
    "vmax=3"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Reconstruction loss: 12.015365937320105\n",
      "Relative reconstruction loss: 0.09044921085441056\n"
     ]
    },
    {
     "data": {
      "image/png": "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tVlre3C13XBoXk4vY2Mz3DQaRATqPYfK639pbfOkusar63huu5ZtPknrGvdKCgIAAAghoAEAAIQQ0AAAAEIIaAAAACEENAAAgBACGgAAQIjza/YTqsu3XuW8ZwnnR5ridG2ut9azT6KU67/1M6L19i7FduMV1kPDRa0xh6qZ0vG2UObJ9fALGgAAQAgBDQAAIISABgAAEEJAAwAACCGgAQAAhBDQAAAAQpxfs0+mrddbwzkSznfn+rO8Jps0qe5ukzCHklTX/3S6/b+vUVFvDm2SOfRxfkEDAAAIIaABAACEENAAAABCCGgAAAAhBDQAAIAQAhoAAECI82v21Zdm8r5clXG68vd77nyfq3IehrY652p/h8P5+7uEnlXg1etVqd6bpd/TSuuxiu2mKb86unWeXH0tdsIcar12Kq37LGdix3nZ+/tQ6+vccqgNzBOW4Rc0AACAEAIaAABACAENAAAghIAGAAAQQkADAAAIIaABAACEGKcz6nXHcXxvGIZ311sOcAW+ME3Ti60bm0PAAswh4NJm59BZAQ0AAID1+L84AgAAhBDQAAAAQghoAAAAIQQ0AACAEAIaAABACAENAAAghIAGAAAQQkADAAAIIaABAACE+D/lC0PPuyPJjwAAAABJRU5ErkJggg==\n",
      "text/plain": [
       "<Figure size 864x360 with 3 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "approx = outer_product_np(X1, X2, X3)\n",
    "print(\"Reconstruction loss:\", np.linalg.norm(np.ndarray.flatten(X-approx), 2))\n",
    "print(\"Relative reconstruction loss:\", np.linalg.norm(np.ndarray.flatten(X-approx), 2)  / np.linalg.norm(np.ndarray.flatten(X), 2))\n",
    "\n",
    "fig, axs = plt.subplots(1, 3, constrained_layout=True, figsize=(12,5))\n",
    "axs[0].axes.get_xaxis().set_ticks([])\n",
    "axs[0].axes.get_yaxis().set_ticks([])\n",
    "axs[1].axes.get_xaxis().set_ticks([])\n",
    "axs[1].axes.get_yaxis().set_ticks([])\n",
    "axs[2].axes.get_xaxis().set_ticks([])\n",
    "axs[2].axes.get_yaxis().set_ticks([])\n",
    "X_max = np.max(approx,axis=0)\n",
    "axs[0].imshow(X_max, vmin=vmin, vmax=vmax)\n",
    "X_max = np.max(approx,axis=1)\n",
    "axs[1].imshow(X_max, vmin=vmin, vmax=vmax)\n",
    "X_max = np.max(approx,axis=2)\n",
    "axs[2].imshow(X_max, vmin=vmin, vmax=vmax)\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Visualize Original Data Tensor"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 864x360 with 3 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "fig, axs = plt.subplots(1, 3, constrained_layout=True, figsize=(12,5))\n",
    "axs[0].axes.get_xaxis().set_ticks([])\n",
    "axs[0].axes.get_yaxis().set_ticks([])\n",
    "axs[1].axes.get_xaxis().set_ticks([])\n",
    "axs[1].axes.get_yaxis().set_ticks([])\n",
    "axs[2].axes.get_xaxis().set_ticks([])\n",
    "axs[2].axes.get_yaxis().set_ticks([])\n",
    "color = \"viridis\"\n",
    "X_max = np.max(X,axis=0)\n",
    "axs[0].imshow(X_max, cmap=plt.get_cmap(color), vmin=vmin, vmax=vmax)\n",
    "X_max = np.max(X,axis=1)\n",
    "axs[1].imshow(X_max, cmap=plt.get_cmap(color), vmin=vmin, vmax=vmax)\n",
    "X_max = np.max(X,axis=2)\n",
    "axs[2].imshow(X_max, cmap=plt.get_cmap(color), vmin=vmin, vmax=vmax)\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [],
   "source": [
    "def cichocki(X, r):\n",
    "    \"\"\"\n",
    "    Given a 3-mode tensor X and a list of rank r, computes to Chichocki \n",
    "    \"\"\"\n",
    "\n",
    "    factors = non_negative_parafac(X, r[0])[1]\n",
    "    A, D, S = factors[0], factors[1], factors[2]\n",
    "    factorizations = [[A, D, S]]\n",
    "    \n",
    "    for i in range(1, len(r)):\n",
    "        \n",
    "        factors = non_negative_parafac(outer_product_np(np.identity(r[i-1]), D, S), r[i])[1]\n",
    "        A, D, S = factors[0], factors[1], factors[2]\n",
    "        factorizations.append([A,D,S])\n",
    "        \n",
    "    return factorizations  "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [],
   "source": [
    "factorizations = cichocki(X, [7,5,3])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Reconstruction loss: 12.015365937320107\n",
      "Relative reconstruction loss: 0.09044921085441057\n"
     ]
    },
    {
     "data": {
      "image/png": "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tVlre3C13XBoXk4vY2Mz3DQaRATqPYfK639pbfOkusar63huu5ZtPknrGvdKCgIAAAghoAEAAIQQ0AAAAEIIaAAAACEENAAAgBACGgAAQIjza/YTqsu3XuW8ZwnnR5ridG2ut9azT6KU67/1M6L19i7FduMV1kPDRa0xh6qZ0vG2UObJ9fALGgAAQAgBDQAAIISABgAAEEJAAwAACCGgAQAAhBDQAAAAQpxfs0+mrddbwzkSznfn+rO8Jps0qe5ukzCHklTX/3S6/b+vUVFvDm2SOfRxfkEDAAAIIaABAACEENAAAABCCGgAAAAhBDQAAIAQAhoAAECI82v21Zdm8r5clXG68vd77nyfq3IehrY652p/h8P5+7uEnlXg1etVqd6bpd/TSuuxiu2mKb86unWeXH0tdsIcar12Kq37LGdix3nZ+/tQ6+vccqgNzBOW4Rc0AACAEAIaAABACAENAAAghIAGAAAQQkADAAAIIaABAACEGKcz6nXHcXxvGIZ311sOcAW+ME3Ti60bm0PAAswh4NJm59BZAQ0AAID1+L84AgAAhBDQAAAAQghoAAAAIQQ0AACAEAIaAABACAENAAAghIAGAAAQQkADAAAIIaABAACE+D/lC0PPuyPJjwAAAABJRU5ErkJggg==\n",
      "text/plain": [
       "<Figure size 864x360 with 3 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Reconstruction loss: 31.144135499785857\n",
      "Relative reconstruction loss: 0.23444666549429757\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 864x360 with 3 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Reconstruction loss: 71.62928371240724\n",
      "Relative reconstruction loss: 0.5392105591832641\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 864x360 with 3 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "for i in range(len(factorizations)):\n",
    "    \n",
    "    A = factorizations[0][0]\n",
    "    for j in range(1, i+1):\n",
    "        A = A @ factorizations[j][0]\n",
    "        \n",
    "    _, D, S = factorizations[i]\n",
    "    \n",
    "    approx = outer_product_np(A,D,S)\n",
    "    print(\"Reconstruction loss:\", np.linalg.norm(np.ndarray.flatten(X-approx), 2))\n",
    "    print(\"Relative reconstruction loss:\", np.linalg.norm(np.ndarray.flatten(X-approx), 2)  / np.linalg.norm(np.ndarray.flatten(X), 2))\n",
    "\n",
    "    fig, axs = plt.subplots(1, 3, constrained_layout=True, figsize=(12,5))\n",
    "    axs[0].axes.get_xaxis().set_ticks([])\n",
    "    axs[0].axes.get_yaxis().set_ticks([])\n",
    "    axs[1].axes.get_xaxis().set_ticks([])\n",
    "    axs[1].axes.get_yaxis().set_ticks([])\n",
    "    axs[2].axes.get_xaxis().set_ticks([])\n",
    "    axs[2].axes.get_yaxis().set_ticks([])\n",
    "    X_max = np.max(approx,axis=0)\n",
    "    axs[0].imshow(X_max, vmin=vmin, vmax=vmax)\n",
    "    X_max = np.max(approx,axis=1)\n",
    "    axs[1].imshow(X_max, vmin=vmin, vmax=vmax)\n",
    "    X_max = np.max(approx,axis=2)\n",
    "    axs[2].imshow(X_max, vmin=vmin, vmax=vmax)\n",
    "    plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Generate Data Tensor (High Noise) "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [],
   "source": [
    "## set the network parameters\n",
    "torch.set_default_tensor_type(torch.DoubleTensor)\n",
    "\n",
    "\n",
    "n1 = 40\n",
    "n2 = 40\n",
    "n3 = 40\n",
    "\n",
    "r = 12\n",
    "\n",
    "\n",
    "a = 1\n",
    "b = 3\n",
    "\n",
    "X = np.zeros((40,40,40))\n",
    "\n",
    "\n",
    "X[0:10,0:15,0:15] = a * np.ones((10,15,15))\n",
    "X[10:25,15:25,15:30] = a * np.ones((15,10,15))\n",
    "X[25:40,25:40,30:40] = a * np.ones((15,15,10))\n",
    "\n",
    "X[0:5,0:10,0:5] = b\n",
    "X[5:10,10:15,5:15] = b\n",
    "\n",
    "X[10:15,15:18,15:20] = b\n",
    "X[15:20,18:23,20:25] = b\n",
    "X[20:25,23:25,25:30] = b\n",
    "\n",
    "X[25:30,25:35,30:35] = b\n",
    "X[30:40,35:40,35:40] = b\n",
    "\n",
    "#add Gaussian Noise\n",
    "np.random.seed(0)\n",
    "X = X + 0.5 * np.abs(np.random.randn(40, 40, 40))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [],
   "source": [
    "factorizations = cichocki(X, [7,5,3])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Reconstruction loss: 78.02716601846193\n",
      "Relative reconstruction loss: 0.3904184843109785\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 864x360 with 3 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Reconstruction loss: 89.9664425462736\n",
      "Relative reconstruction loss: 0.4501581171031648\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 864x360 with 3 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Reconstruction loss: 115.54421188371417\n",
      "Relative reconstruction loss: 0.578139619525239\n"
     ]
    },
    {
     "data": {
      "image/png": "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zviNlbZwqQ7dcGuRaZryNjPNLQtaW9DckL+gAQAADMICDQAAYBAWaAAAAIOwQAMAABiEBRoAAMAgLNAAAAAGcX3MfpWKnyLZY85lcewUYmVTdaEVc4jpTOYQQTztG9H9zXYk8zEcDKdL9zI+06ofpPpCdVFK003X3fz5oapz3odTNe9/1I2VDf21GgNL2sKh2UfKY1tG2KbraGjF109Tv7PHE67blu4zj7r3K1XZubbeqS40pHes20862yPE7hPeVUk73j5ta9O4tnyucKzq56PF9je1I/jjniThWHivtvpK6rPd+by7tUWoslOuvUXFyltzTFMe//G5VX2h+x2bnmmoM85f6d0Yzld+D3W360oPoDEP+QsaAADAICzQAAAABmGBBgAAMAgLNAAAgEFYoAEAAAzCAg0AAGAQV8Xs794cpw++/q11W3Assrj3Kcs1RcemKM5QLkWGpnKpnZ1zNX3//nvrgzGnOhRLkadVYmjz0toR3GkLhHS+ENm6ezxf5+mut/XDv/6n//it27W1P/pP/ubZf0/bFRzv62NxK4bidj38p+flW+8ejtP7/+Pb5w9254ZTcSG7enw/fuWDur6u1P4kxvsW/x630ei146u/EqpMccdxbg4nLPOtQ5muFG8d2tie26oI6DB3pXb83E//o15DNvCH/tlPnv33+Vhf2+m9cHFhW55qcn74z8+bh+bH43T/6x89q463lqLJq2+oacrj+BD2jel+D/2eD+tjlfQJFbdvqn31V+tjcVuC5jzU/e6pLOmepG77Qsb/KH74Z/76+QN3Yaui1/XDWcI34q4o9/CroUx5BAAAgJuyQAMAABiEBRoAAMAgLNAAAAAGYYEGAAAwCAs0AACAQVwVs7+JFB87imZMPS/UZ/hxn4pk5fRLzXLoRQ9348Uvmuc6Krm7pUQRpx8jmVO89RZSlHPKea4iyFOkdHe7jBABH6OjQ0z6HBqTns9tpe0dujWeL5i2K1g973srozy257rV90vq590x0NmaY5py3HxnTkxbJDS3+0hjZJN5aO2Y/TCPJqmNvDz+ggYAADAICzQAAIBBWKABAAAMwgINAABgEBZoAAAAg7BAAwAAGMT1Mftr54meivpSzGhVZpouxMOmWOnmsc4ad+17OOV05aUbvRru11zF2KZ7HJT1XSqXUn1TlcdwbcU1zMe3a9PIWincqUyM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\n",
      "text/plain": [
       "<Figure size 864x360 with 3 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "for i in range(len(factorizations)):\n",
    "    \n",
    "    A = factorizations[0][0]\n",
    "    for j in range(1, i+1):\n",
    "        A = A @ factorizations[j][0]\n",
    "        \n",
    "    _, D, S = factorizations[i]\n",
    "    \n",
    "    approx = outer_product_np(A,D,S)\n",
    "    print(\"Reconstruction loss:\", np.linalg.norm(np.ndarray.flatten(X-approx), 2))\n",
    "    print(\"Relative reconstruction loss:\", np.linalg.norm(np.ndarray.flatten(X-approx), 2)  / np.linalg.norm(np.ndarray.flatten(X), 2))\n",
    "\n",
    "    fig, axs = plt.subplots(1, 3, constrained_layout=True, figsize=(12,5))\n",
    "    axs[0].axes.get_xaxis().set_ticks([])\n",
    "    axs[0].axes.get_yaxis().set_ticks([])\n",
    "    axs[1].axes.get_xaxis().set_ticks([])\n",
    "    axs[1].axes.get_yaxis().set_ticks([])\n",
    "    axs[2].axes.get_xaxis().set_ticks([])\n",
    "    axs[2].axes.get_yaxis().set_ticks([])\n",
    "    X_max = np.max(approx,axis=0)\n",
    "    axs[0].imshow(X_max, vmin=vmin, vmax=vmax)\n",
    "    X_max = np.max(approx,axis=1)\n",
    "    axs[1].imshow(X_max, vmin=vmin, vmax=vmax)\n",
    "    X_max = np.max(approx,axis=2)\n",
    "    axs[2].imshow(X_max, vmin=vmin, vmax=vmax)\n",
    "    plt.show()"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.6.9"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
