{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Robust classification of digits by polynomials\n",
    "\n",
    "In this notebook, we will demonstrate that the proposed framework is applicable to real-world scrnarios. We will train a Chebyshev polynomial to label a pictures as being one of the digits 0 to 9. We will use a subset of UCI digits dataset that is available in the scikit-learn python package. Each sample is a 8x8 grayscale picture. We will transform the samples to fit inside $x\\in[-1,1]^{64}$.\n",
    "\n",
    "We will use the Chebyshev polynomials $T_\\alpha$ as features. However, it would be computationally intractable to to include all possible polynomial bases in the hypothesis. To see why, consider the count of all polynomial bases where each $x_i$ has a degree of at most one. This set has the same cardinality with a set that contains all possible subsets of $\\{x_i\\}_{i=1}^{64}$, i.e. the power set of $\\{x_i\\}_{i=1}^{64}$. Consequently, we have to decide which polynomial basis would be added to the hypothesis.\n",
    "\n",
    "To do so, we enumerate all the possible walks up to $D$ steps on a 8x8 lattice, and then map each walk to a polynomial. A node in the lattice represents a pixel, and the structure of the lattice represents the structure of a 2D image. A walk with $D$ steps would be mapped to a degree $D$ polynomial. For example, the following walk\n",
    "$$\n",
    "x_{00} \\rightarrow x_{01} \\rightarrow x_{11} \\rightarrow x_{10} \\rightarrow x_{00}\n",
    "$$\n",
    "would be mapped to\n",
    "$$\n",
    "T_2(x_{00})T_1(x_{01})T_1(x_{11})T_1(x_{10}).\n",
    "$$"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import networkx as nx\n",
    "\n",
    "def all_paths(G, max_depth, history=None, depth=1, paths=None, graph_search=False):\n",
    "    if depth > max_depth:\n",
    "        return []\n",
    "\n",
    "    if paths is None:\n",
    "        paths = set()\n",
    "\n",
    "    if history is None:\n",
    "        for v in G:\n",
    "            hist = [v]\n",
    "            paths.add(tuple(hist))\n",
    "            all_paths(G, max_depth, hist, depth+1, paths)\n",
    "    else:\n",
    "        for v in G[history[-1]]:\n",
    "            if graph_search and v in history:\n",
    "                continue\n",
    "            hist = history + [v]\n",
    "            paths.add(tuple(sorted(hist)))\n",
    "            all_paths(G, max_depth, hist, depth+1, paths)\n",
    "\n",
    "    return list(sorted(paths))\n",
    "\n",
    "def get_basis(G, deg):\n",
    "    paths = all_paths(G, deg)\n",
    "\n",
    "    basis = []\n",
    "    for path in paths:\n",
    "        j, d = [], []\n",
    "        for v in set(path):\n",
    "            j.append(v[0]*8+v[1])\n",
    "            d.append(path.count(v))\n",
    "        basis.append((j, d))\n",
    "    return basis\n",
    "\n",
    "G = nx.generators.lattice.grid_2d_graph(8, 8)\n",
    "\n",
    "# We add self-loops to each node so that a node could be repeated in the walk even when the length\n",
    "# of the walk is less than the lenght of the shortest loop in the graph\n",
    "for v in G:\n",
    "    G.add_edge(v, v)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Now that we have choosen our set of features, we can apply the proposed robust framework to train a robust classifier for the digits dataset. First, we will load and preprocess the dataset."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "from sklearn import datasets, metrics, svm\n",
    "from sklearn.model_selection import train_test_split\n",
    "\n",
    "digits = datasets.load_digits()\n",
    "\n",
    "# flatten the images\n",
    "n_samples = len(digits.images)\n",
    "data = digits.images.reshape((n_samples, -1))\n",
    "\n",
    "# Split data into 50% train and 50% test subsets\n",
    "X_train, X_test, y_train, y_test = train_test_split(\n",
    "    data, digits.target, test_size=0.5, shuffle=False)\n",
    "\n",
    "# Scale the pixels into [-1, 1] interval\n",
    "X_train = X_train / 8 - 1\n",
    "X_test = X_test / 8 - 1"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "We will need to compute the tunning matrix for the set of bases."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "from itertools import combinations_with_replacement, repeat\n",
    "\n",
    "def Itv(bi, bj):\n",
    "    \"\"\"\n",
    "    Helper function for computing the inner product between Chebyshev polynomials of the first kind\n",
    "    \"\"\"\n",
    "    return np.where(bi == bj, np.where(bi == 0, 1, 1/2), 0)\n",
    "\n",
    "def Iuv(bi, bj):\n",
    "    \"\"\"\n",
    "    Helper function for computing the inner product between Chebyshev polynomials of the second kind\n",
    "    \"\"\"\n",
    "    return np.where((bi % 2) == (bj % 2), np.where(bi % 2 == 0, 1 + 2 * np.floor(np.minimum(bi, bj) / 2), 2 * np.ceil(np.minimum(bi, bj) / 2)), 0)\n",
    "\n",
    "def Ib(bi, bj):\n",
    "    \"\"\"\n",
    "    Helper function for computing the tuning matrix\n",
    "    \"\"\"\n",
    "    res = 0\n",
    "    N = len(bi)\n",
    "    temp = Itv(bi, bj) ** (1 - np.eye(bi.size))\n",
    "    res = np.sum(bi * bj * Iuv(bi - 1, bj - 1) * np.prod(temp, axis=-1))\n",
    "    return res\n",
    "\n",
    "def tunning_mat(basis):\n",
    "    M = len(basis)\n",
    "    MM = M*(M-1) / 2 + M\n",
    "    Sigma = np.eye(M)\n",
    "\n",
    "    counter = 0\n",
    "    for i, j in combinations_with_replacement(range(M), r=2):\n",
    "        if counter%100 == 0: print('\\rcomputing the tuning matrix %f' % (counter / MM), end='')\n",
    "        counter += 1\n",
    "        bi = np.zeros(64)\n",
    "        bi[basis[i][0]] = basis[i][1]\n",
    "        bj = np.zeros(64)\n",
    "        bj[basis[j][0]] = basis[j][1]\n",
    "        inner = Ib(bi, bj)\n",
    "        Sigma[(i, j), (j, i)] = inner\n",
    "    print()\n",
    "    return Sigma"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Finally, we implement a one-step $\\ell^2$ normalized gradient based attack to see how robust the output of the learning rules are."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "from scipy.misc import derivative\n",
    "\n",
    "def attack_grad(clf, x, y, C):\n",
    "    def obj(r, i):\n",
    "        _x = np.copy(x)\n",
    "        _x[0, i] = r\n",
    "\n",
    "        log_proba = clf(_x)\n",
    "        return log_proba[0, y]\n",
    "    \n",
    "    x = x[None, :]\n",
    "    grad = np.array([derivative(lambda r: obj(r, i), x[0, i]) for i in range(x.shape[1])])\n",
    "    adv_dir = grad / np.linalg.norm(grad, ord=2)\n",
    "\n",
    "    return np.clip(x - C * adv_dir[None, :], -1, 1)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Next, we will conduct an experiment to show that the proposed framework would indeed result in a hypothesis that is more robust against adversarial attacks. To do so, we will train orthonormal and harmonic Chebyshev polynomials of increasing degree, and compare their performance against a blackbox attack. We will use a MLP classifier as the surrogate."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy of MLP classifier on natural samples: 0.94\n",
      "attacking the test set 0.889878\n",
      "Accuracy of MLP classifier on adversarial samples: 0.61\n"
     ]
    }
   ],
   "source": [
    "from sklearn.neural_network import MLPClassifier\n",
    "\n",
    "def f1(clf, X, y):\n",
    "    yhat = clf.predict(X)\n",
    "    return metrics.f1_score(yhat, y, pos_label=None, average='weighted')\n",
    "\n",
    "\n",
    "cls_mlp = MLPClassifier(hidden_layer_sizes=(40, 30, 20), activation='relu', solver='adam', random_state=2)\n",
    "cls_mlp.fit(X_train, y_train)\n",
    "\n",
    "nat_mlp_score = f1(cls_mlp, X_test, y_test)\n",
    "\n",
    "print('Accuracy of MLP classifier on natural samples: %1.2f' % nat_mlp_score)\n",
    "\n",
    "adv_mlp = []\n",
    "counter = 0\n",
    "for x, y in zip(X_test, y_test):\n",
    "    if counter % 100 == 0:\n",
    "        print('\\rattacking the test set %f' % (counter / X_test.shape[0]), end='')\n",
    "    counter += 1\n",
    "    adv_mlp.append(attack_grad(cls_mlp.predict_log_proba, x, y, 1))\n",
    "print('\\rattacking the test set 1.0')\n",
    "adv_mlp = np.vstack(adv_mlp)\n",
    "\n",
    "adv_mlp_score = f1(cls_mlp, adv_mlp, y_test)\n",
    "print('Accuracy of MLP classifier on adversarial samples: %1.2f' % adv_mlp_score)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "computing the Vandermonde matrix 0.99\n",
      "computing the Vandermonde matrix 0.99\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "C:\\Users\\Ramin\\AppData\\Roaming\\Python\\Python38\\site-packages\\sklearn\\svm\\_base.py:976: ConvergenceWarning: Liblinear failed to converge, increase the number of iterations.\n",
      "  warnings.warn(\"Liblinear failed to converge, increase \"\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "computing the Vandermonde matrix 0.99\n",
      "computing the tuning matrix 0.961538\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "C:\\Users\\Ramin\\AppData\\Roaming\\Python\\Python38\\site-packages\\sklearn\\svm\\_base.py:976: ConvergenceWarning: Liblinear failed to converge, increase the number of iterations.\n",
      "  warnings.warn(\"Liblinear failed to converge, increase \"\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "computing the Vandermonde matrix 0.99\n",
      "computing the Vandermonde matrix 0.99\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "C:\\Users\\Ramin\\AppData\\Roaming\\Python\\Python38\\site-packages\\sklearn\\svm\\_base.py:976: ConvergenceWarning: Liblinear failed to converge, increase the number of iterations.\n",
      "  warnings.warn(\"Liblinear failed to converge, increase \"\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "computing the Vandermonde matrix 0.99\n",
      "computing the tuning matrix 0.999308\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "C:\\Users\\Ramin\\AppData\\Roaming\\Python\\Python38\\site-packages\\sklearn\\svm\\_base.py:976: ConvergenceWarning: Liblinear failed to converge, increase the number of iterations.\n",
      "  warnings.warn(\"Liblinear failed to converge, increase \"\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "computing the Vandermonde matrix 0.99\n",
      "computing the Vandermonde matrix 0.99\n",
      "computing the Vandermonde matrix 0.99\n",
      "computing the tuning matrix 0.999970\n",
      "computing the Vandermonde matrix 0.99\n",
      "computing the Vandermonde matrix 0.99\n",
      "computing the Vandermonde matrix 0.99\n",
      "computing the tuning matrix 0.999977\n"
     ]
    }
   ],
   "source": [
    "from numpy.polynomial import chebyshev as ch\n",
    "from itertools import product\n",
    "from scipy.linalg import sqrtm\n",
    "\n",
    "def vander(X, basis, deg, verbose=True):\n",
    "    N = X.shape[0]\n",
    "    M = len(basis)\n",
    "    V = np.ones((N, M))\n",
    "    for n, m in product(range(N), range(M)):\n",
    "        if verbose and n%10 == 0 and m%M == 0: print('\\rcomputing the Vandermonde matrix %1.2f' % (n/N), end='')\n",
    "        v = ch.chebvander(X[n], deg)\n",
    "        V[n, m] = np.prod(v[basis[m]])\n",
    "    print()\n",
    "    return V\n",
    "\n",
    "degrees = [1, 2, 3, 4]\n",
    "\n",
    "nat_orth_score = []\n",
    "nat_sync_score = []\n",
    "\n",
    "adv_orth_score = []\n",
    "adv_sync_score = []\n",
    "\n",
    "for deg in degrees:\n",
    "    basis = get_basis(G, deg)\n",
    "\n",
    "    # Orth\n",
    "    V_train_orth = vander(X_train, basis, deg)\n",
    "    V_test_orth = vander(X_test, basis, deg)\n",
    "\n",
    "    clf_orth = svm.LinearSVC()\n",
    "    clf_orth.fit(V_train_orth, y_train)\n",
    "\n",
    "    nat_orth_score.append(f1(clf_orth, V_test_orth, y_test))\n",
    "    V_orth_adv = vander(adv_mlp, basis, deg)\n",
    "    adv_orth_score.append(f1(clf_orth, V_orth_adv, y_test))\n",
    "    # Orth\n",
    "\n",
    "    # Harm\n",
    "    Sigma = tunning_mat(basis)\n",
    "    Lambda = np.real(sqrtm(np.linalg.inv(Sigma)))\n",
    "\n",
    "    V_train_harm = V_train_orth @ Lambda\n",
    "    V_test_harm = V_test_orth @ Lambda\n",
    "\n",
    "    clf_harm = svm.LinearSVC()\n",
    "    clf_harm.fit(V_train_harm, y_train)\n",
    "\n",
    "    nat_sync_score.append(f1(clf_harm, V_test_harm, y_test))\n",
    "    V_harm_adv = V_orth_adv @ Lambda\n",
    "    adv_sync_score.append(f1(clf_harm, V_harm_adv, y_test))\n",
    "    # Harm"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "In the end, we will plot the accuracy of each classifer on the natural and adversarial examples as a function of the maximum degree of the included polynomials."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "The PostScript backend does not support transparency; partially transparent artists will be rendered opaque.\n",
      "The PostScript backend does not support transparency; partially transparent artists will be rendered opaque.\n",
      "The PostScript backend does not support transparency; partially transparent artists will be rendered opaque.\n",
      "The PostScript backend does not support transparency; partially transparent artists will be rendered opaque.\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 432x288 with 2 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "from matplotlib.ticker import FormatStrFormatter\n",
    "\n",
    "fig, ax = plt.subplots(1, 2)\n",
    "ax[0].yaxis.set_major_formatter(FormatStrFormatter('%.2f'))\n",
    "ax[0].set_title('Natural')\n",
    "ax[0].plot(degrees, [nat_mlp_score] * len(degrees), label='baseline', marker='.')\n",
    "ax[0].plot(degrees, nat_orth_score, label='orth. Cheb.', marker='v')\n",
    "ax[0].plot(degrees, nat_sync_score, label='harm. Cheb.', marker='x')\n",
    "ax[0].set_ylabel(r'avg. f1-score')\n",
    "ax[0].set_xlabel(r'degree')\n",
    "ax[0].legend()\n",
    "\n",
    "ax[1].yaxis.set_major_formatter(FormatStrFormatter('%.2f'))\n",
    "ax[1].set_title('Adversarial')\n",
    "ax[1].plot(degrees, [adv_mlp_score] * len(degrees), label='baseline', marker='.')\n",
    "ax[1].plot(degrees, adv_orth_score, label='orth. Cheb.', marker='v')\n",
    "ax[1].plot(degrees, adv_sync_score, label='harm. Cheb.', marker='x')\n",
    "ax[1].set_xlabel(r'degree')\n",
    "ax[1].legend()\n",
    "\n",
    "fig.savefig('digits.eps', bbox_inches='tight', dpi=100)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "The result suggests that as we include more polynomial bases, the harmonic hypothesis gets more robust. On the other hand, the orthonormal Chebyshev does not show the same pattern, and it seems that including higher degree polynomials would worsen its performance on adversarial examples. Moreover, we can see that the adversarial examples of the MLP classifier transfers to the polynomial hypothesis. This observation is inline with the prediction of our framework in which ANNs and polynomials are different representations of the same normal hypothesis class."
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3.8.5 64-bit",
   "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.8.5"
  },
  "orig_nbformat": 4,
  "vscode": {
   "interpreter": {
    "hash": "e50d3b6b967b1f11efb2cb0ce102d6b35cde92629c83d0f916850b20db4f520f"
   }
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
