{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "1b44defd-95e5-4e17-9af8-8c0ae18cdd1c",
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "from sklearn.linear_model import LogisticRegression as LR\n",
    "import random\n",
    "from kymatio.numpy import Scattering2D\n",
    "import tqdm\n",
    "import warnings\n",
    "warnings.filterwarnings(\"ignore\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "6d0622d5-d3c0-4db4-b216-e2e83a9bdc7a",
   "metadata": {},
   "outputs": [],
   "source": [
    "# The dataset was processed using torchvision transforms}, as \n",
    "#\\texttt{ransform }$= t\\texttt{ransforms.}C\\texttt{ompose(}[ transforms.Resize((40, 40))  transforms.ToTensor()\n",
    "])\n",
    "$"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "d9cd1603-fae5-4433-8a1a-5fcdf9b189b6",
   "metadata": {},
   "outputs": [],
   "source": [
    "Xtrain=np.load(\"datasets/Xtrain.npy\")\n",
    "ytrain=np.load(\"datasets/ytrain.npy\")\n",
    "Xtest=np.load(\"datasets/Xtest.npy\")\n",
    "ytest=np.load(\"datasets/ytest.npy\")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "53652bfa-79e4-4244-b9e4-72f0ea4e41a4",
   "metadata": {},
   "outputs": [],
   "source": [
    "#Preprocessing using the scattering transform \n",
    "H=Xtrain.shape[-2]\n",
    "W=Xtrain.shape[-1]\n",
    "scattering = Scattering2D(shape=(H, W), J=4, L=6)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "054567b8-ae75-4f93-8794-7a9f4ef35cb9",
   "metadata": {},
   "outputs": [],
   "source": [
    "Xtrain=scattering(Xtrain)\n",
    "Xtest=scattering(Xtest)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "8175b303-fa56-40b1-b128-18ae853eaa44",
   "metadata": {},
   "outputs": [],
   "source": [
    "#Preprocessing\n",
    "Xtrain=Xtrain.reshape(len(ytrain),-1)\n",
    "Xtest=Xtest.reshape(len(ytest),-1)\n",
    "mean=Xtrain.mean(axis=0)\n",
    "norm=Xtrain.std()\n",
    "Xtrain=Xtrain -mean \n",
    "Xtrain/=norm\n",
    "Xtest=Xtest - mean\n",
    "Xtest/=norm\n",
    "#Going from 0,1 labels to -1,1\n",
    "ytrain=np.sign(ytrain-.5)\n",
    "ytest=np.sign(ytest-.5)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "d99aa73c-587d-4c8f-b7f8-ad73be7ffb5b",
   "metadata": {},
   "outputs": [],
   "source": [
    "#Shuffling the training set\n",
    "perm = list(range(len(ytrain)))\n",
    "random.shuffle(perm)\n",
    "Xtrain=Xtrain[perm]\n",
    "ytrain=ytrain[perm]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "id": "92716ffc-4d7b-4710-bf6e-7f9ea4f4e726",
   "metadata": {},
   "outputs": [],
   "source": [
    "def balance(X, y): #For a given dataset X and labels y, resamples so that classes are balanced.\n",
    "    \n",
    "    classes = [-1,1]\n",
    "    idx_per_class = {c: np.where(y == c)[0] for c in classes}\n",
    "\n",
    "    N = len(y)\n",
    "    num_classes = 2\n",
    "    per_class= {1: int(N/2), -1: N- int(N/2)}\n",
    "\n",
    "    sampled_indices = []\n",
    "\n",
    "    for c in classes:\n",
    "        idx = idx_per_class[c]\n",
    "        if len(idx) >= per_class[c]:   #if majority class, downsampled without replacement.\n",
    "            chosen = np.random.choice(idx, per_class[c], replace=False)\n",
    "        else:                          #if minority class, upsampled with replacement.\n",
    "            chosen = np.random.choice(idx, per_class[c], replace=True)\n",
    "\n",
    "        sampled_indices.append(chosen)\n",
    "\n",
    "    sampled_indices = np.concatenate(sampled_indices)\n",
    "    \n",
    "\n",
    "    return X[sampled_indices], y[sampled_indices]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "id": "0c0cc532-77e0-4c7d-bbdc-13f371d9a971",
   "metadata": {},
   "outputs": [],
   "source": [
    "γ=.5           #budget\n",
    "λ=.01          #regularization"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "id": "326b56ba-2c9c-4bc8-8bdd-01fa6be2995d",
   "metadata": {},
   "outputs": [],
   "source": [
    "N=len(ytrain)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 35,
   "id": "3143e4c9-1866-4bb9-acd1-528f054446f1",
   "metadata": {},
   "outputs": [],
   "source": [
    "def get_error(ψ, C=1/λ):\n",
    "\n",
    "    n0=int(ψ*N)\n",
    "    ng=int(γ*N)\n",
    "    \n",
    "    idx=np.random.choice(range(len(ytrain)),n0)\n",
    "    idx_=np.array([k for k in range(N) if k not in idx])\n",
    "    X0=Xtrain[idx]\n",
    "    y0=ytrain[idx]\n",
    "\n",
    "    X0, y0=balance(X0,y0)                          #balancing the dataset\n",
    "    lr=LR(C=C, fit_intercept=False)                #Fitting logistic regression \n",
    "    lr.fit(X0, y0, sample_weight=np.ones(n0)/n0)\n",
    "    \n",
    "    Xr=Xtrain[idx_]                                #remaining unlabeled data\n",
    "    yr=ytrain[idx_]                                #remaining labels (assumed unknown)\n",
    "    \n",
    "    w0=(lr.coef_).flatten()    \n",
    "    u=np.abs(Xr@w0)                                #predictions of the base classifier\n",
    "\n",
    "    m=ng-n0                                        #remaining budget\n",
    "    u_sorted = sorted(u)\n",
    "    lower = u_sorted[m-1]                          \n",
    "    upper = u_sorted[m]\n",
    "    κ = (lower + upper) / 2                        #There are m samples with u[i]<\\kappa\n",
    "    \n",
    "\n",
    "    X=np.vstack((X0, Xr[u<κ]))                     #Concatenating X0 with the selected samples\n",
    "    y=np.hstack((y0, yr[u<κ]))\n",
    "    X, y=balance(X,y)                              #Balancing before training\n",
    "    lr=LR(C=C, fit_intercept=False)                #Fitting logistic regression\n",
    "    nf=X.shape[0]\n",
    "    lr.fit(X, y, sample_weight=np.ones(nf)/nf)\n",
    "    ypred=lr.predict(Xtest)                       #Error is evaluated on unbalanced test set.\n",
    "    error=np.mean(ypred!=ytest)\n",
    "\n",
    "    return error\n",
    "\n",
    "    \n",
    "\n",
    "    \n",
    "    \n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 61,
   "id": "2edb7704-c3d1-463e-829a-a06fa45942ae",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████████████████████████████████████████████| 10/10 [02:36<00:00, 15.67s/it]\n"
     ]
    }
   ],
   "source": [
    "psis=np.linspace(.005, .995*γ, 10)\n",
    "errors=[]\n",
    "first=True\n",
    "for ψ in tqdm.tqdm(psis):\n",
    "    running=[]\n",
    "    for i in range(50):\n",
    "        err=get_error(ψ)\n",
    "        running.append(err)\n",
    "    rrunning=np.array(running)\n",
    "    if first:\n",
    "        first=False\n",
    "        errors=running\n",
    "    else:\n",
    "        errors=np.vstack((errors, running))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 85,
   "id": "8906f2d9-9252-4c7a-8592-ef4c8f19a397",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "Text(0, 0.5, '$\\\\mathcal{E}_{\\\\rm gen}$')"
      ]
     },
     "execution_count": 85,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 400x400 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "plt.figure(figsize=(4,4))\n",
    "plt.errorbar(psis,errors.mean(1),errors.std(1), marker=\"o\", capsize=2, c=\"b\")\n",
    "plt.xlabel(r\"$\\psi$\", fontsize=20)\n",
    "plt.ylabel(r\"$\\mathcal{E}_{\\rm gen}$\", fontsize=20)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "6a6218a8-5c38-4071-9a88-271f1cbfed9a",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python [conda env:base] *",
   "language": "python",
   "name": "conda-base-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.12.4"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
