{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 192,
   "id": "6d002e08",
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import numpy.linalg as la\n",
    "import cvxpy as cp\n",
    "import matplotlib.pyplot as plt\n",
    "import math\n",
    "import random\n",
    "import time\n",
    "\n",
    "import warnings\n",
    "warnings.filterwarnings('ignore')\n",
    "\n",
    "seed=42\n",
    "random.seed(a=seed)\n",
    "np.random.seed(seed=seed)\n",
    "\n",
    "def relu(x):\n",
    "    return np.maximum(0,x)\n",
    "def drelu(x):\n",
    "    return x>=0"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 193,
   "id": "5175c026",
   "metadata": {},
   "outputs": [],
   "source": [
    "n=50 #choose 160 for a challenging case\n",
    "d=3\n",
    "def spiral_xy(i, spiral_num):\n",
    "    \"\"\"\n",
    "    Create the data for a spiral.\n",
    "\n",
    "    Arguments:\n",
    "        i runs from 0 to 96\n",
    "        spiral_num is 1 or -1\n",
    "    \"\"\"\n",
    "    φ = i/16 * math.pi\n",
    "    r = 6.5 * ((104 - i)/104)\n",
    "    x = (r * math.cos(φ) * spiral_num)/13 + 0.5\n",
    "    y = (r * math.sin(φ) * spiral_num)/13 + 0.5\n",
    "    return (x, y)\n",
    "\n",
    "def spiral(spiral_num):\n",
    "    return [spiral_xy(i, spiral_num) for i in range(n//2)]\n",
    "\n",
    "a = spiral(1)\n",
    "b = spiral(-1)\n",
    "X=2*np.concatenate((a,b),axis=0)-1\n",
    "X=np.append(X,np.ones((n,1)),axis=1)\n",
    "y=np.concatenate((np.ones(n//2),-np.ones(n//2)))\n",
    "\n",
    "\n",
    "beta=1e-5\n",
    "P=200\n",
    "\n",
    "np.random.seed(0)\n",
    "dmat=np.empty((n,0))\n",
    "\n",
    "## Finite approximation of all possible sign patterns\n",
    "for i in range(P):\n",
    "    u=np.random.randn(d,1)\n",
    "    dmat=np.append(dmat,drelu(np.dot(X,u)),axis=1)\n",
    "\n",
    "dmat=(np.unique(dmat,axis=1))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 194,
   "id": "ce87eb55",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(50, 141)"
      ]
     },
     "execution_count": 194,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "dmat.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 195,
   "id": "81b281d2",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "2-layer convex program objective value:  0.0007843337681130952\n"
     ]
    }
   ],
   "source": [
    "# Optimal CVX\n",
    "m=dmat.shape[1]\n",
    "Uopt1=cp.Variable((d,m)) # c_i in paper\n",
    "Uopt2=cp.Variable((d,m)) # c_i' in paper\n",
    "\n",
    "## Below we use hinge loss as a performance metric for binary classification\n",
    "yopt1=cp.Parameter((n,1))\n",
    "yopt2=cp.Parameter((n,1))\n",
    "yopt1=cp.sum(cp.multiply(dmat,(X*Uopt1)),axis=1)\n",
    "yopt2=cp.sum(cp.multiply(dmat,(X*Uopt2)),axis=1)\n",
    "#cost=cp.sum(cp.pos(1-cp.multiply(y,yopt1-yopt2)))/n+beta*(cp.mixed_norm(Uopt1.T,2,1)+cp.mixed_norm(Uopt2.T,2,1))\n",
    "cost=cp.sum_squares(y-(yopt1-yopt2))/(2*n)+beta*(cp.mixed_norm(Uopt1.T,2,1)+cp.mixed_norm(Uopt2.T,2,1))\n",
    "\n",
    "constraints=[]\n",
    "constraints+=[cp.multiply((2*dmat-np.ones((n,m))),(X*Uopt1))>=0]\n",
    "constraints+=[cp.multiply((2*dmat-np.ones((n,m))),(X*Uopt2))>=0]\n",
    "prob=cp.Problem(cp.Minimize(cost),constraints)\n",
    "prob.solve(solver=cp.CLARABEL,warm_start=True)\n",
    "cvx_opt=prob.value\n",
    "if prob.status != \"optimal\":\n",
    "    print(\"Convex: Status convex: \",prob.status)\n",
    "print(\"2-layer convex program objective value: \",cvx_opt)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 196,
   "id": "d4bc6873",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Number of neurons: 141\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "# BP\n",
    "sigma=1e-2\n",
    "mu=0.5\n",
    "print(\"Number of neurons:\",m)\n",
    "mbp=2*m\n",
    "ITERS=15000\n",
    "U=sigma*np.random.randn(d,mbp)\n",
    "w=sigma*np.random.randn(mbp,1)\n",
    "yall=y.reshape(-1,1)\n",
    "obj_bp=np.empty((ITERS,1))\n",
    "batch_size=n\n",
    "nb=batch_size\n",
    "for i in range(ITERS):\n",
    "    if i%8000 == 0 and i>0:\n",
    "        mu=mu/5\n",
    "    samp=np.random.choice(n,batch_size)\n",
    "    Xgd=X[samp,:]\n",
    "    ygd=yall[samp,:]\n",
    "    a1=np.dot(Xgd,U)\n",
    "    yest=np.dot(relu(a1),w)\n",
    "    yest_all=np.dot(relu(np.dot(X,U)),w)\n",
    "    obj_bp[i]=np.linalg.norm(yall-yest_all)**2/(2*n)+(beta/2)*(np.linalg.norm(U,'fro')**2+np.linalg.norm(w,'fro')**2)\n",
    "    gradw= relu(a1).T@(yest-ygd)/nb\n",
    "    gradU=np.zeros((d,mbp))\n",
    "  \n",
    "    gradU=Xgd.T@(drelu(a1)*((yest-ygd)@w.T))/nb\n",
    "\n",
    "    U=(1-mu*beta)*U-mu*gradU\n",
    "    w=(1-mu*beta)*w-mu*gradw\n",
    "    \n",
    "    \n",
    "# Plot\n",
    "plt.figure(0)\n",
    "plt.semilogy(obj_bp,label='GD')\n",
    "plt.axhline(cvx_opt,color='k',label='Optimal')\n",
    "plt.ylabel('Objective value')\n",
    "plt.xlabel('Iteration')\n",
    "plt.legend();"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 204,
   "id": "eff7c9b6",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 504x216 with 2 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 504x216 with 2 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "samp=100\n",
    "x1=np.linspace(-1,1,samp).reshape(-1,1)\n",
    "x2=np.linspace(-1,1,samp).reshape(-1,1)\n",
    "\n",
    "Xtest=np.ones((samp**2,d))\n",
    "count=0\n",
    "for i in range(samp):\n",
    "    for j in range(samp):\n",
    "        Xtest[count]=[x1[i, 0],x2[j, 0],1]\n",
    "        count+=1\n",
    "\n",
    "ntest=Xtest.shape[0]\n",
    "\n",
    "Uopt1v=Uopt1.value\n",
    "Uopt2v=Uopt2.value\n",
    "\n",
    "yest_cvx=np.sum(drelu(Xtest@Uopt1v)*(Xtest@Uopt1v)-drelu(Xtest@Uopt2v)*(Xtest@Uopt2v),axis=1)\n",
    "yest_bp=np.dot(relu(np.dot(Xtest,U)),w)\n",
    "\n",
    "\n",
    "pos_cvx=np.where(np.sign(yest_cvx)==1)\n",
    "neg_cvx=np.where(np.sign(yest_cvx)==-1)\n",
    "pos_bp=np.where(np.sign(yest_bp)==1)\n",
    "neg_bp=np.where(np.sign(yest_bp)==-1)\n",
    "pos=np.where(y==1)\n",
    "neg=np.where(y==-1)\n",
    "\n",
    "\n",
    "fig, (ax1, ax2) = plt.subplots(1, 2, figsize=(7, 3))\n",
    "\n",
    "ax1.plot(Xtest[pos_cvx,0],Xtest[pos_cvx,1],'co');\n",
    "ax1.plot(Xtest[neg_cvx,0],Xtest[neg_cvx,1],'mo');\n",
    "ax2.plot(X[pos,0],X[pos,1],'cx');\n",
    "ax2.plot(X[neg,0],X[neg,1],'mx');\n",
    "ax2.set_yticklabels([])\n",
    "#ax1.set_title(\"2-layer Convex\")\n",
    "plt.savefig('convex.pdf', bbox_inches='tight')\n",
    "\n",
    "\n",
    "fig, (ax1, ax2) = plt.subplots(1, 2, figsize=(7, 3))\n",
    "ax1.plot(Xtest[pos_bp,0],Xtest[pos_bp,1],'co');\n",
    "ax1.plot(Xtest[neg_bp,0],Xtest[neg_bp,1],'mo');\n",
    "ax2.plot(X[pos,0],X[pos,1],'cx');\n",
    "ax2.plot(X[neg,0],X[neg,1],'mx');\n",
    "ax2.set_yticklabels([])\n",
    "#ax1.set_title(\"2-layer GD\")\n",
    "plt.savefig('sgd.pdf', bbox_inches='tight')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "a0827602",
   "metadata": {},
   "outputs": [],
   "source": [
    "### NEW ###\n",
    "\n",
    "N = len(y)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 102,
   "id": "4e6b903e",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(3, 141)"
      ]
     },
     "execution_count": 102,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "Uopt2.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 91,
   "id": "7fbbcce8",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0.0004711505532681101"
      ]
     },
     "execution_count": 91,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "beta*(np.sum(np.linalg.norm(Uopt1.value.T,2,axis=0)+np.linalg.norm(Uopt2.value.T,2,axis=0)))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 38,
   "id": "e5f4d565",
   "metadata": {},
   "outputs": [],
   "source": [
    "def sign(a):\n",
    "    return 2 * int(a >= 0) - 1"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 156,
   "id": "fb79a768",
   "metadata": {},
   "outputs": [],
   "source": [
    "# compute analytic center\n",
    "def center(S, R=1, boxinit=False, reg=False, reg_value=2e-4):\n",
    "    \"\"\"S is list of affine inequalities described as tuple of LHS vector/matrix and RHS scalar/vector\"\"\"\n",
    "    s = cp.Variable(2*d*m)\n",
    "    obj = 0 if boxinit else cp.log(R - cp.norm(s))\n",
    "    constraints = []\n",
    "    if reg:\n",
    "        U = cp.reshape(s, (2*d, m), order='F')\n",
    "        #obj += cp.log(reg_value - beta*(cp.mixed_norm(U[:d].T,2,1)+cp.mixed_norm(U[d:].T,2,1)))\n",
    "        constraints = [beta*(cp.mixed_norm(U[:d].T,2,1)+cp.mixed_norm(U[d:].T,2,1)) <= reg_value]\n",
    "    if len(S) > 0:\n",
    "        obj += cp.sum([cp.log(rhs - lhs @ s) for lhs, rhs in S])\n",
    "    prob = cp.Problem(cp.Maximize(obj), constraints)\n",
    "    prob.solve(solver=cp.MOSEK)\n",
    "    return s.value"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 105,
   "id": "94add25d",
   "metadata": {},
   "outputs": [],
   "source": [
    "# cut set\n",
    "def cut(S, x, y, dmat_row):\n",
    "    m = len(dmat_row)\n",
    "    # f = cp.sum(cp.multiply(dmat, X*(Uopt1 - Uopt2)), axis=1)\n",
    "    # f_i = dmat[i, 0] * X[i, :] [I - I] var[:, 0] + dmat[i, 1] * X[i, :] [I - I] var[:, 1] + ...  where var = [Uopt1; Uopt2]\n",
    "    # f_i = dmat[i, :] @ [X[i, :] [I - I] var[:, 0]; X[i, :] [I - I] var[:, 1]; ...]\n",
    "    S.append((-y * dmat_row @ np.kron(np.eye(m), np.concatenate((x, -x)).T), 0))\n",
    "    \n",
    "    relu_constraint = -np.kron(np.diag(2*dmat_row-np.ones(m)), np.kron(np.eye(2), x))\n",
    "    for lhs in relu_constraint:\n",
    "        S.append((lhs, 0))\n",
    "    # potentially remove redundant constraints here"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 106,
   "id": "754b9a43",
   "metadata": {},
   "outputs": [],
   "source": [
    "def pred_point(i, U1v, U2v): # corresponds to <w(theta), x>\n",
    "    y1 = np.sum(np.multiply(dmat[i],(X[i][np.newaxis, :] @ U1v)),axis=1)\n",
    "    y2 = np.sum(np.multiply(dmat[i],(X[i][np.newaxis, :] @ U2v)),axis=1)\n",
    "    return y1 - y2\n",
    "\n",
    "def pred_point_simplified(i, U1v, U2v):\n",
    "    var = np.vstack((U1v, U2v)).flatten(order='F')\n",
    "    return (dmat[i] @ np.kron(np.eye(len(dmat[i])), np.concatenate((X[i], -X[i])).T)) @ var\n",
    "\n",
    "def pred_point_simplified_vec(i, vec):\n",
    "    return (dmat[i] @ np.kron(np.eye(len(dmat[i])), np.concatenate((X[i], -X[i])).T)) @ vec"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 107,
   "id": "395237cc",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([0.99943778])"
      ]
     },
     "execution_count": 107,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# check that simplification is correct\n",
    "\n",
    "pred_point(3, Uopt1v, Uopt2v)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 108,
   "id": "5c23f368",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0.9994377816575323"
      ]
     },
     "execution_count": 108,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "pred_point_simplified(3, Uopt1v, Uopt2v)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 109,
   "id": "24336288",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0.9994377816575323"
      ]
     },
     "execution_count": 109,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "vec = np.vstack((Uopt1v, Uopt2v)).flatten(order='F')\n",
    "pred_point_simplified_vec(3, vec)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 110,
   "id": "8c7cbfab",
   "metadata": {},
   "outputs": [],
   "source": [
    "def constraint(i, U1v, U2v):\n",
    "    return np.vstack((\n",
    "        np.multiply((2*dmat[i]-np.ones((1,m))),(X[i] @ U1v)),\n",
    "        np.multiply((2*dmat[i]-np.ones((1,m))),(X[i] @ U2v))\n",
    "    )).flatten(order='F')\n",
    "\n",
    "def constraint_simplified(i, U1v, U2v):\n",
    "    var = np.vstack((U1v, U2v)).flatten(order='F')\n",
    "    return np.kron(np.diag(2*dmat[i]-np.ones(m)), np.kron(np.eye(2), X[i])) @ var"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 111,
   "id": "7af6f3a0",
   "metadata": {},
   "outputs": [],
   "source": [
    "def in_Ct(c, Ct, eps=1e-3):\n",
    "    for lhs, rhs in Ct:\n",
    "        if lhs @ c > rhs + eps:\n",
    "            return False\n",
    "    return True"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 112,
   "id": "6d7f81fe",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "3"
      ]
     },
     "execution_count": 112,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "d"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 113,
   "id": "a3c685f6",
   "metadata": {},
   "outputs": [],
   "source": [
    "def sample_lattice(S, R=1):\n",
    "    d = 3\n",
    "    l = cp.Variable(2*d*m)\n",
    "    d = np.random.randn(2*d*m)\n",
    "    obj = (d / np.linalg.norm(d)) @ l\n",
    "    prob = cp.Problem(cp.Maximize(obj), [cp.norm(l) <= R] + [lhs @ l <= rhs for lhs, rhs in S])\n",
    "    prob.solve(cp.MOSEK)\n",
    "    return l.value"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 114,
   "id": "a6bf2cf5",
   "metadata": {},
   "outputs": [],
   "source": [
    "R = 1\n",
    "\n",
    "C0_lower = -R*np.ones(2*d*m)\n",
    "C0_upper = R*np.ones(2*d*m)\n",
    "\n",
    "def sample_classifier(Ct, c, maxiter=10**5):\n",
    "    for _ in range(maxiter):\n",
    "        #candidate = np.random.uniform(C0_lower, C0_upper)\n",
    "        candidate = c + np.random.randn(*c.shape)\n",
    "        if in_Ct(candidate, Ct):\n",
    "            return candidate\n",
    "    print(f'Failed to sample after {maxiter} tries.')\n",
    "    return None"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 115,
   "id": "744953ae",
   "metadata": {},
   "outputs": [],
   "source": [
    "def query(C, c, data_skip, M=100):\n",
    "        \n",
    "    mini = np.inf\n",
    "    i_mini = -1\n",
    "    \n",
    "    maxi = -np.inf\n",
    "    i_maxi = -1\n",
    "    \n",
    "    minabs = np.inf\n",
    "    i_minabs = -1\n",
    "    \n",
    "    #theta_s = np.zeros(2*d*m)\n",
    "    #for _ in range(M):\n",
    "    #    #th = sample_classifier(C, c)\n",
    "    #    th = sample_lattice(C)\n",
    "    #    if th is None:\n",
    "    #        return None, None\n",
    "    #    theta_s += (1/M) * th\n",
    "        \n",
    "    #theta_matrix = np.reshape(c, (2*d, m), order='F')\n",
    "    #U1_query=theta_matrix[:d]\n",
    "    #U2_query=theta_matrix[d:]\n",
    "    \n",
    "    for i in range(N): # search in finite data (D implicit) set to re-use dmat then\n",
    "        if i not in data_skip:\n",
    "            pred = pred_point_simplified_vec(i, c)\n",
    "            if pred < mini:\n",
    "                i_mini = 1*i\n",
    "                mini = pred\n",
    "            if pred > maxi:\n",
    "                i_maxi = 1*i\n",
    "                maxi = pred\n",
    "            if abs(pred) < minabs:\n",
    "                i_minabs = 1*i\n",
    "                minabs = abs(pred)\n",
    "    \n",
    "    return i_mini, i_maxi, i_minabs"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 132,
   "id": "42a5cc24",
   "metadata": {},
   "outputs": [],
   "source": [
    "# final convex solve\n",
    "\n",
    "def convex_solve(used):\n",
    "\n",
    "    used_unique = list(set(used))\n",
    "\n",
    "    Uopt1_final=cp.Variable((d,m)) # c_i in paper\n",
    "    Uopt2_final=cp.Variable((d,m)) # c_i' in paper\n",
    "\n",
    "    yopt1_final=cp.sum(cp.multiply(dmat[used_unique],(X[used_unique]*Uopt1_final)),axis=1)\n",
    "    yopt2_final=cp.sum(cp.multiply(dmat[used_unique],(X[used_unique]*Uopt2_final)),axis=1)\n",
    "\n",
    "    cost=cp.sum_squares(y[used_unique]-(yopt1_final-yopt2_final))/(2*n)+beta*(cp.mixed_norm(Uopt1_final.T,2,1)+cp.mixed_norm(Uopt2_final.T,2,1))\n",
    "\n",
    "    constraints=[]\n",
    "    constraints+=[cp.multiply((2*dmat[used_unique]-np.ones((len(used_unique),m))),(X[used_unique]*Uopt1_final))>=0]\n",
    "    constraints+=[cp.multiply((2*dmat[used_unique]-np.ones((len(used_unique),m))),(X[used_unique]*Uopt2_final))>=0]\n",
    "    prob_final=cp.Problem(cp.Minimize(cost),constraints)\n",
    "    prob_final.solve(solver=cp.CLARABEL,warm_start=True)\n",
    "    \n",
    "    return Uopt1_final.value, Uopt2_final.value, beta*(cp.mixed_norm(Uopt1_final.value.T,2,1)+cp.mixed_norm(Uopt2_final.value.T,2,1))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 207,
   "id": "48388dbc",
   "metadata": {},
   "outputs": [],
   "source": [
    "# cutting plane method\n",
    "\n",
    "def cutting_plane(n_points=100, maxit=100, boxinit=False):\n",
    "    \n",
    "    data_tried = []\n",
    "    data_used = []\n",
    "    \n",
    "    Ct = []\n",
    "    if boxinit:\n",
    "        for i, (l, u) in enumerate(zip(C0_lower, C0_upper)):\n",
    "            one_vec = np.zeros(2*d*m)\n",
    "            one_vec[i] = 1\n",
    "            Ct.append((one_vec, u))\n",
    "            Ct.append((-one_vec, -l))\n",
    "    \n",
    "    c = None\n",
    "    did_cut = True\n",
    "    it = 0\n",
    "    while len(data_used) < n_points and it < maxit: # TODO: replace by proper termination criterion\n",
    "        if did_cut:\n",
    "            #if len(data_used) > 0:\n",
    "            #    _, _, reg_value = convex_solve(data_used)\n",
    "            #else:\n",
    "            #    reg_value = 1e-5\n",
    "            c = center(Ct, R=R)\n",
    "            did_cut = False\n",
    "        i_mini, i_maxi, i_minabs = query(Ct, c, data_tried)\n",
    "        if i_mini is None:\n",
    "            return Ct, c, data_used\n",
    "        data_tried += [i_mini, i_maxi]\n",
    "        #if it >= 30:\n",
    "        #    print(X[i_mini], np.sign(pred_point_simplified_vec(i_mini, c)), y[i_mini])\n",
    "        #    print(X[i_maxi], np.sign(pred_point_simplified_vec(i_maxi, c)), y[i_maxi])\n",
    "            #print(X[i_minabs], np.sign(pred_point_simplified_vec(i_minabs, c)), y[i_minabs])\n",
    "        if True: #len(data_used) < 4 * n_points // 5:\n",
    "            if np.sign(pred_point_simplified_vec(i_mini, c)) != y[i_mini]:\n",
    "                print(f'Cutting at iteration {it}')\n",
    "                cut(Ct, X[i_mini], y[i_mini], dmat[i_mini])\n",
    "                data_used.append(i_mini)\n",
    "                did_cut = True\n",
    "            if np.sign(pred_point_simplified_vec(i_maxi, c)) != y[i_maxi]:\n",
    "                print(f'Cutting at iteration {it}')\n",
    "                cut(Ct, X[i_maxi], y[i_maxi], dmat[i_maxi])\n",
    "                data_used.append(i_maxi)\n",
    "                did_cut = True\n",
    "        else:\n",
    "            if np.sign(pred_point_simplified_vec(i_minabs, c)) != y[i_minabs]:\n",
    "                print(f'Cutting at iteration {it}')\n",
    "                cut(Ct, X[i_minabs], y[i_minabs], dmat[i_minabs])\n",
    "                data_used.append(i_minabs)\n",
    "                did_cut = True\n",
    "        it += 1\n",
    "    \n",
    "    return Ct, c, data_used"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 208,
   "id": "19499960",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Cutting at iteration 0\n",
      "Cutting at iteration 0\n",
      "Cutting at iteration 1\n",
      "Cutting at iteration 2\n",
      "Cutting at iteration 3\n",
      "Cutting at iteration 4\n",
      "Cutting at iteration 5\n",
      "Cutting at iteration 5\n",
      "Cutting at iteration 8\n",
      "Cutting at iteration 9\n",
      "Cutting at iteration 12\n",
      "Cutting at iteration 16\n",
      "Cutting at iteration 17\n",
      "Cutting at iteration 22\n",
      "Cutting at iteration 23\n",
      "Cutting at iteration 23\n",
      "size of C: 4528\n",
      "used: [0, 0, 25, 41, 16, 21, 8, 34, 49, 29, 24, 14, 19, 31, 5, 40]\n"
     ]
    }
   ],
   "source": [
    "C, c, used = cutting_plane(15)\n",
    "print(f'size of C: {len(C)}')\n",
    "print(f'used: {used}')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 210,
   "id": "8375755d",
   "metadata": {},
   "outputs": [],
   "source": [
    "def visualize(Uopt1v, Uopt2v, title):\n",
    "\n",
    "    yest_cvx=np.sum(drelu(Xtest@Uopt1v)*(Xtest@Uopt1v)-drelu(Xtest@Uopt2v)*(Xtest@Uopt2v),axis=1)\n",
    "    yest_bp=np.dot(relu(np.dot(Xtest,U)),w)\n",
    "\n",
    "\n",
    "    pos_cvx=np.where(np.sign(yest_cvx)==1)\n",
    "    neg_cvx=np.where(np.sign(yest_cvx)==-1)\n",
    "    pos_bp=np.where(np.sign(yest_bp)==1)\n",
    "    neg_bp=np.where(np.sign(yest_bp)==-1)\n",
    "    pos=np.where(y==1)\n",
    "    neg=np.where(y==-1)\n",
    "\n",
    "\n",
    "    fig, (ax1, ax2) = plt.subplots(1, 2, figsize=(7, 3))\n",
    "\n",
    "    ax1.plot(Xtest[pos_cvx,0],Xtest[pos_cvx,1],'co');\n",
    "    ax1.plot(Xtest[neg_cvx,0],Xtest[neg_cvx,1],'mo');\n",
    "    ax2.plot(X[pos,0],X[pos,1],'cx');\n",
    "    ax2.plot(X[neg,0],X[neg,1],'mx');\n",
    "    ax2.set_yticklabels([])\n",
    "    #ax1.set_title(\"2-layer GD\")\n",
    "    plt.savefig(f'{title}.pdf', bbox_inches='tight', dpi=100)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 211,
   "id": "1dae24eb",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 504x216 with 2 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 504x216 with 2 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "theta_matrix = np.reshape(c, (2*d, m), order='F')\n",
    "visualize(theta_matrix[:d], theta_matrix[d:], 'pre_norefine')\n",
    "\n",
    "Uopt1_final_v, Uopt2_final_v, _ = convex_solve(used)\n",
    "visualize(Uopt1_final_v, Uopt2_final_v, 'post_norefine')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "2988053a",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "cpg_env",
   "language": "python",
   "name": "cpg_env"
  },
  "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.10.2"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
