{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "import torch\n",
    "import numpy as np\n",
    "import cvxpy as cp\n",
    "import matplotlib.pyplot as plt"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Solve for the convex problem"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "X = np.array([[1,0],[0,1],[1,1]])\n",
    "y = np.array([1,0,0])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "D_list = []\n",
    "D_list.append(np.diag(np.array([0,0,0])))\n",
    "D_list.append(np.diag(np.array([0,1,0])))\n",
    "D_list.append(np.diag(np.array([0,1,1])))\n",
    "D_list.append(np.diag(np.array([1,0,0])))\n",
    "D_list.append(np.diag(np.array([1,0,1])))\n",
    "D_list.append(np.diag(np.array([1,1,1])))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "w_list = []\n",
    "for i in range(len(D_list)):\n",
    "    w_list.append(cp.Variable(2))\n",
    "w_list2 = []\n",
    "for i in range(len(D_list)):\n",
    "    w_list2.append(cp.Variable(2))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "ECOS 2.0.7 - (C) embotech GmbH, Zurich Switzerland, 2012-15. Web: www.embotech.com/ECOS\n",
      "\n",
      "It     pcost       dcost      gap   pres   dres    k/t    mu     step   sigma     IR    |   BT\n",
      " 0  +0.000e+00  -2.913e-01  +7e+01  9e-01  9e-01  1e+00  1e+00    ---    ---    1  1  - |  -  - \n",
      " 1  -1.441e-01  +5.125e-02  +3e+01  1e-01  2e-01  4e-01  6e-01  0.7224  2e-01   1  1  1 |  0  0\n",
      " 2  +1.558e-01  +1.364e-01  +9e+00  3e-02  2e-02  2e-02  2e-01  0.9890  3e-01   1  2  2 |  0  0\n",
      " 3  +1.374e-01  +1.341e-01  +2e+00  4e-03  3e-03  3e-03  4e-02  0.8140  4e-02   2  2  2 |  0  0\n",
      " 4  +1.296e-01  +1.292e-01  +2e-01  4e-04  4e-04  4e-04  5e-03  0.8868  1e-02   2  2  2 |  0  0\n",
      " 5  +1.294e-01  +1.292e-01  +9e-02  1e-04  1e-04  9e-05  2e-03  0.8204  3e-01   2  2  2 |  0  0\n",
      " 6  +1.291e-01  +1.291e-01  +1e-02  1e-05  2e-05  1e-05  2e-04  0.8995  1e-02   2  1  1 |  0  0\n",
      " 7  +1.290e-01  +1.290e-01  +4e-03  6e-06  7e-06  3e-06  9e-05  0.7624  3e-01   3  2  2 |  0  0\n",
      " 8  +1.290e-01  +1.290e-01  +2e-04  3e-07  4e-07  1e-07  5e-06  0.9890  4e-02   3  2  2 |  0  0\n",
      " 9  +1.290e-01  +1.290e-01  +1e-05  1e-08  2e-08  7e-09  2e-07  0.9560  5e-03   4  2  3 |  0  0\n",
      "10  +1.290e-01  +1.290e-01  +2e-07  2e-10  3e-10  1e-10  4e-09  0.9828  2e-04   1  2  2 |  0  0\n",
      "11  +1.290e-01  +1.290e-01  +6e-09  8e-12  1e-11  4e-12  1e-10  0.9676  1e-04   2  1  2 |  0  0\n",
      "\n",
      "OPTIMAL (within feastol=1.0e-11, reltol=4.9e-08, abstol=6.3e-09).\n",
      "Runtime: 0.000796 seconds.\n",
      "\n"
     ]
    }
   ],
   "source": [
    "output = 0\n",
    "for i in range(len(D_list)):\n",
    "    output+=D_list[i]@X@(w_list[i]-w_list2[i])\n",
    "beta = 0.1\n",
    "regularization = 0\n",
    "for i in range(len(D_list)):\n",
    "    regularization+=cp.norm(w_list[i])+cp.norm(w_list2[i])\n",
    "constraints = []\n",
    "I = np.eye(3)\n",
    "for i in range(len(D_list)):\n",
    "    constraints.append((2*D_list[i]-I)@X@w_list[i]>=0)\n",
    "    constraints.append((2*D_list[i]-I)@X@w_list2[i]>=0)\n",
    "p_star = cp.Problem(cp.Minimize(0.5*cp.norm(output-y)**2+beta*regularization),constraints).solve(verbose=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[-7.02817750e-11 -7.03096961e-11]\n",
      "[-8.36335657e-11  2.72694326e-11]\n",
      "[-3.50883040e-11  5.42595458e-11]\n",
      "[ 7.16663179e-08 -7.16950155e-08]\n",
      "[ 0.85869861 -0.79094767]\n",
      "[3.0491535e-10 2.9522568e-11]\n",
      "[-7.02817750e-11 -7.03096961e-11]\n",
      "[-8.44016619e-11  2.89083509e-11]\n",
      "[-1.93247622e-11  2.71290929e-10]\n",
      "[-4.32644317e-13 -6.78790463e-11]\n",
      "[ 3.29983046e-11 -3.90122054e-12]\n",
      "[2.75031024e-11 2.56482080e-10]\n"
     ]
    }
   ],
   "source": [
    "for i in range(len(D_list)):\n",
    "    print(w_list[i].value)\n",
    "for i in range(len(D_list)):\n",
    "    print(w_list2[i].value)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "w_star = w_list[4].value"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0.14130221509211527"
      ]
     },
     "execution_count": 9,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "np.linalg.norm(np.maximum(X@w_star,0)-y)**2+beta*np.linalg.norm(w_star)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Verify the optimal set"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The optimal solution is unqiue.\n"
     ]
    }
   ],
   "source": [
    "constraints_new = constraints.copy()\n",
    "constraints_new.append(0.5*cp.norm(output-y)**2+beta*regularization<=p_star)\n",
    "non_unique=False\n",
    "for i in range(len(w_list)):\n",
    "    for j in range(2):\n",
    "        p_low = cp.Problem(cp.Minimize(w_list[i][j]),constraints_new).solve()\n",
    "        p_up = cp.Problem(cp.Maximize(w_list[i][j]),constraints_new).solve()\n",
    "        if p_up-p_low>1e-8:\n",
    "            non_unique = True\n",
    "if non_unique:\n",
    "    print('The optimal solution is not unqiue.')\n",
    "else:\n",
    "    print('The optimal solution is unqiue.')"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Solve for the nonconvex problem"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [],
   "source": [
    "X_t = torch.Tensor(X)\n",
    "y_t = torch.Tensor(y).reshape([-1,1])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [],
   "source": [
    "class Feedforward(torch.nn.Module):\n",
    "    def __init__(self, input_size, hidden_size, output_size, activation='ReLU'):\n",
    "        super(Feedforward, self).__init__()\n",
    "        self.input_size = input_size\n",
    "        self.hidden_size  = hidden_size\n",
    "        self.output_size = output_size\n",
    "        self.fc1 = torch.nn.Linear(self.input_size, self.hidden_size,bias=False)\n",
    "        if activation=='ReLU':\n",
    "            self.activation = torch.nn.ReLU()\n",
    "        else:\n",
    "            raise Exception('Not implemented')\n",
    "        self.fc2 = torch.nn.Linear(self.hidden_size, self.output_size,bias=False)\n",
    "\n",
    "    \n",
    "    def forward(self, x):\n",
    "        hidden = self.fc1(x)\n",
    "        hidden = self.activation(hidden)\n",
    "        output = self.fc2(hidden)\n",
    "        return output"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "trial: 0 gap: 0.037645319219706985\n",
      "trial: 1 gap: 0.0038139878548398287\n"
     ]
    }
   ],
   "source": [
    "trial = 0\n",
    "while trial<10:\n",
    "    m = 5\n",
    "    torch.manual_seed(trial)\n",
    "    net = Feedforward(2,m,1)\n",
    "    optimizer = torch.optim.SGD(net.parameters(),lr=1e-1,momentum=0,weight_decay=0.5*beta)\n",
    "    iter_num = 1000\n",
    "    loss_fn = torch.nn.MSELoss()\n",
    "    for i in range(iter_num):\n",
    "        optimizer.zero_grad()\n",
    "        output = net(X_t)\n",
    "        loss = loss_fn(output, y_t)\n",
    "        loss.backward()\n",
    "        optimizer.step()\n",
    "    nn_para = list(net.parameters())\n",
    "    loss_all=0.5*loss+0.5*beta*(torch.norm(nn_para[0])**2+torch.norm(nn_para[1])**2)\n",
    "    print('trial: {} gap: {}'.format(trial, np.abs(loss_all.item()-p_star)))\n",
    "    if np.abs(loss_all.item()-p_star)<1e-2:\n",
    "        break\n",
    "    trial = trial+1\n",
    "w_net1 =list(net.parameters())[0]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "trial: 0 gap: 0.003839409235836533\n"
     ]
    }
   ],
   "source": [
    "trial = 0\n",
    "while trial<10:\n",
    "    m = 5\n",
    "    torch.manual_seed(20+trial)\n",
    "    net = Feedforward(2,m,1)\n",
    "    optimizer = torch.optim.SGD(net.parameters(),lr=1e-1,momentum=0,weight_decay=0.5*beta)\n",
    "    iter_num = 1000\n",
    "    loss_fn = torch.nn.MSELoss()\n",
    "    for i in range(iter_num):\n",
    "        optimizer.zero_grad()\n",
    "        output = net(X_t)\n",
    "        loss = loss_fn(output, y_t)\n",
    "        loss.backward()\n",
    "        optimizer.step()\n",
    "    nn_para = list(net.parameters())\n",
    "    loss_all=0.5*loss+0.5*beta*(torch.norm(nn_para[0])**2+torch.norm(nn_para[1])**2)\n",
    "    print('trial: {} gap: {}'.format(trial, np.abs(loss_all.item()-p_star)))\n",
    "    if np.abs(loss_all.item()-p_star)<1e-2:\n",
    "        break\n",
    "    trial = trial+1\n",
    "w_net2 =list(net.parameters())[0]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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+OKLL/Dz88PHx4dJkyaRn5/Pvn378PLyIjs7m8uXL+Pp6cnBgweLxHXt3zwoKAgvLy+GDx9uTPJdXFx4/fXXufvuu1mzZg1hYWEYDAY6derECy+8UKSc6dOn06VLFwICAijui7TIyEj69u2Lr68v9913H6mpqQD069ePZ599lj59+uDu7s6+ffsYNmwYbm5uvPLKK8XWZZlorW/7ARwBHK4+dwCOFLPPPCCJwhm7TgOZwBelKd/X11f/3aFDh27YVqL9q7R+o7nWrzX4/8cbzQu336aTJ09qS0tL/ccff2ittR4xYoReuXKl1lrre+65R8fFxWmttd69e7f29/fXWmsdFBSk16xZYyyjXr16Wmutt27dquvWratPnDihtdZ64cKFety4cVprrWNjY7Wzs7POysrSn332mXZ1ddXp6ek6KytLt2rVSickJJQYY05Oju7cubP+3//+d9vnKSqXzMxMfejQIb1v3z4dGRmpT548qTMyMirk2GX6zAlxm4AIbUJ7VBkfwNvAi1efvwgsuMX+/YDvSlN2ce2jEHfC7bQBh37dqhc+PEgf2rHN5ONHREToTp066YyMDH3p0iXt4eGhf//9d33y5EkN6B07dujc3Fz9yCOP6OnTp+t9+/ZpBwcHfeDAAV1QUKC11rp169Y6LS3NeA134MABnZ+fr7t06aLHjx+vCwoK9Pr16/XQoUO11lpfvHhR5+bmaq213rRpkx42bJjWuvC6bdCgQTfEWNJ1Wlpamu7du7exvZ4/f76eM2dOkZi01nrfvn26b9++WmutX3vtNd2lSxedmZmptdZ68ODBOiQkRGut9fLly40xBgUF6eHDh+v8/HwdExOj27ZtW2z9XX+cspQ9cuRIY73Ur1+/SJ1duwa+5sqVK9rV1VXv3bu3SP2V9br27NmzRc4jMDBQ//rrr/rQoUN68ODBOicnR2ut9eTJk/WKFSu01lq//PLLevr06To4OFi/+eabN5z/9e8TrbUeP368fvvtt41189Zbb2mttU5OTtbOzs767NmzOjc3V/v7++tvvvlGa601oL/44guttdZz5szRU6ZMMdbTmjVrdE5Oju7Ro4c+e/as1lrrr776So8fP15rrXXfvn31jBkztNZav/vuu9rBwUGnpKTo7Oxs7ejoqM+dO3dDzMV95kpqI01dfuBbIAiYf/VneDGJ4kxgJoBS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pUtkZ2czfvx4DAYDnTt3ZuvWrTetrw8++KBI3YWEhDB16lSgcNpbPz8/fHx8mDRpEvn5+ezbtw8vLy+ys7O5fPkynp6eHDx4sEhcJb1Prv0bvP7669x9992sWbOGsLAwDAYDnTp14oUXXihSzvTp0+nSpQsBAQGkpaXdcP6RkZH07dsXX19f7rvvPlJTUwHo168fzz77LH369MHd3Z19+/YxbNgw3NzceOWVV4qtS1H1bD56ka3Km6Zj36XFmLexadedbTFJrP8jGa01GzduJCcnx9xhCiGEEBXOycmJOXPmkJCQwOrVq3FxccHJyQmA6Oho9uzZU3WGJmitK+3D19dX/92hQ4du2HYrh37dqhc+PEgf2rGtzH/7dxEREbpTp046IyNDX7p0SXt4eOjff/9dnzx5UgN6x44dWmutx48fr99++22ttdatW7fWaWlpxjKuvT558qS2tLTUBw4c0Pn5+bpLly56/PjxuqCgQK9fv14PHTpUa631xYsXdW5urtZa602bNulhw4ZprbXeunWrHjRo0A0xfvbZZ9rV1VWnp6frrKws3apVK52QkKDT0tJ07969dUZGhtZa6/nz5+s5c+bcEOO+fft03759tdZav/baa7pLly46MzNTa6314MGDdUhIiNZa6+XLlxtjDAoK0sOHD9f5+fk6JiZGt23bttj6u/44ZSl75MiRxnqpX79+kTr7448/ihzjypUr2tXVVe/du7dI/S1cuFCPGzdOa611bGysdnZ21llZWSXW19mzZ4ucR2BgoP7111/1oUOH9ODBg3VOTo7WWuvJkyfrFStWaK21fvnll/X06dN1cHCwfvPNN284/1u9T9566y2ttdbJycna2dlZnz17Vufm5mp/f3/9zTffaK21BvQXX3yhtdZ6zpw5esqUKcZ6WrNmjc7JydE9evTQZ8+e1Vpr/dVXX+nx48drrbXu27evnjFjhtZa63fffVc7ODjolJQUnZ2drR0dHfW5c+duiPl2PnPCvHrO+0W3fuG7Io9WMzbonvN+0Xv37tWAbtasmX7ppZd0fHy8ucPVWmsNROhK0O5UlUdx7aMQQgjTjBo1SgO6S5cu+tNPP9WXL182d0ha65LbyGrdI7dx8QKWPPYPflz2DgA/Ll3Eksf+wcbFC267zB07dvDQQw9Rr149bG1tGTZsGL/++isAzs7O9OrVC4Ax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\n",
      "text/plain": [
       "<Figure size 1080x216 with 2 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "fig, ax = plt.subplots(1,2,figsize=(15,3))\n",
    "w_net_np = w_net1.detach().numpy()\n",
    "ax_sub = ax[0]\n",
    "for i in range(m):\n",
    "    ax_sub.scatter(w_net_np[i,0],w_net_np[i,1],label='neuron {}'.format(i+1))\n",
    "ax_sub.scatter(w_star[0],w_star[1],marker='*',label='optimal neuron from convex problem')\n",
    "ax_sub.plot([w_star[0],0],[w_star[1],0],'k--')\n",
    "ax_sub.legend(loc = 'lower left')\n",
    "w_net_np = w_net2.detach().numpy()\n",
    "ax_sub = ax[1]\n",
    "for i in range(m):\n",
    "    ax_sub.scatter(w_net_np[i,0],w_net_np[i,1],label='neuron {}'.format(i+1))\n",
    "ax_sub.scatter(w_star[0],w_star[1],marker='*',label='optimal neuron from convex problem')\n",
    "ax_sub.plot([w_star[0],0],[w_star[1],0],'k--')\n",
    "ax_sub.legend(loc = 'upper right')\n",
    "fig.savefig('toy.png')"
   ]
  }
 ],
 "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.8.5"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
