{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 26,
   "id": "32f72bc4-f5db-4fb3-ad70-5eb1b761f818",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "import torch\n",
    "import torchvision\n",
    "from torchvision.datasets import MNIST\n",
    "import torch.nn.functional as F\n",
    "import matplotlib.pyplot as plt\n",
    "import numpy as np"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 223,
   "id": "91772411-0f2b-4551-830c-b73dd56814bf",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "data_dir = \"<root_dir>\" # Change as wish\n",
    "mnist_train = MNIST(root=data_dir, download=True, train=True, transform=None)\n",
    "mnist_test = MNIST(root=data_dir, download=True, train=False, transform=None)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 222,
   "id": "e92ecd8b-ffd2-4382-9d46-5792722ff866",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "def pad_image(image,pad_size,left=True):\n",
    "    if left:\n",
    "        new_image = F.pad(image, (pad_size, 0, pad_size, 0),\"constant\", 0)\n",
    "    else: \n",
    "        new_image = F.pad(image, (0, pad_size, 0, pad_size),\"constant\", 0)\n",
    "    return new_image\n",
    "\n",
    "def random_shift(image, max_shift):\n",
    "\n",
    "    right_shift = np.random.randint(max_shift)\n",
    "    down_shift = np.random.randint(max_shift)\n",
    "\n",
    "    new_image = torch.roll(image, shifts=(down_shift, right_shift), dims=(0, 1))\n",
    "    return new_image\n",
    "\n",
    "def pad_then_shift(image, pad_size, left=True):\n",
    "    return random_shift(pad_image(image, pad_size, left),pad_size/4)\n",
    "\n",
    "def create_m1_samples(data, targets, class_size=5000, pad_size=28):\n",
    "    all_data = []\n",
    "    all_targets = []\n",
    "    for cla_idx in range(10):\n",
    "        cla_length = torch.sum(targets==cla_idx).item()\n",
    "        # Some randomness\n",
    "        chosen_ones = np.random.permutation(cla_length)[:class_size]\n",
    "        cur_cla_data = data[targets==cla_idx][chosen_ones]\n",
    "        for i in range(class_size):\n",
    "            cur_data = pad_then_shift(cur_cla_data[i], pad_size)\n",
    "            all_data.append(cur_data)\n",
    "            \n",
    "            cur_target = torch.zeros(10)\n",
    "            cur_target[cla_idx] = 1.0\n",
    "            all_targets.append(cur_target)\n",
    "    return torch.stack(all_data, dim=0), torch.stack(all_targets, dim=0)\n",
    "\n",
    "def create_m2_samples(data, targets, class_size=5000, pad_size=28):\n",
    "    all_data = []\n",
    "    all_targets = []\n",
    "    for idx_1 in range(10):\n",
    "        for idx_2 in range(idx_1+1, 10):\n",
    "            cla_length_1 = torch.sum(targets==idx_1).item()\n",
    "            cla_length_2 = torch.sum(targets==idx_2).item()\n",
    "            # Some randomness\n",
    "            chosen_ones_1 = np.random.permutation(cla_length_1)[:class_size]\n",
    "            chosen_ones_2 = np.random.permutation(cla_length_2)[:class_size]\n",
    "            cla_data_1 = data[targets==idx_1][chosen_ones_1]\n",
    "            cla_data_2 = data[targets==idx_2][chosen_ones_2]\n",
    "        \n",
    "            for i in range(class_size):\n",
    "                first_left = torch.rand(1) < 0.5\n",
    "                cur_data_1 = pad_then_shift(cla_data_1[i], pad_size, left=first_left)\n",
    "                cur_data_2 = pad_then_shift(cla_data_2[i], pad_size, left=not first_left)\n",
    "                all_data.append(torch.maximum(cur_data_1, cur_data_2))\n",
    "\n",
    "                cur_target = torch.zeros(10)\n",
    "                cur_target[idx_1] = 0.5\n",
    "                cur_target[idx_2] = 0.5\n",
    "                all_targets.append(cur_target)\n",
    "    return torch.stack(all_data, dim=0), torch.stack(all_targets, dim=0)\n",
    "\n",
    "def create_dataset(num_samples, data, targets, pad_size=28):\n",
    "    m1_num, m2_num = num_samples\n",
    "    m1_data, m1_targets = create_m1_samples(data, targets, class_size=m1_num, pad_size=pad_size)\n",
    "    m2_data, m2_targets = create_m2_samples(data, targets, class_size=m2_num, pad_size=pad_size)\n",
    "    \n",
    "    return torch.cat([m1_data, m2_data], dim=0), torch.cat([m1_targets, m2_targets], dim=0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 242,
   "id": "8e0e0497-144f-47ba-8b80-fd109d5daa2d",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "trainset, trainlabels = create_dataset([3100, 200], mnist_train.data / torch.max(mnist_train.data), \n",
    "                                       mnist_train.targets, pad_size=28)\n",
    "testset, testlabels = create_dataset([800, 50], mnist_test.data / torch.max(mnist_test.data), \n",
    "                                       mnist_test.targets, pad_size=28)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 250,
   "id": "e47ccf0a-eb4b-4c0a-87bf-e5d52992fe83",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "import pickle\n",
    "to_save = {\n",
    "            \"train_data\": trainset,\n",
    "            \"train_label\": trainlabels,\n",
    "            \"test_data\": testset,\n",
    "            \"test_label\": testlabels\n",
    "            }\n",
    "    \n",
    "with open(\"<root_dir>/mnist_combine.pkl\", 'wb') as f: \n",
    "    pickle.dump(to_save, f)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 265,
   "id": "1da3f02f-eeeb-4292-abfa-de4f5630276c",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "class MNIST_Combine(torch.utils.data.Dataset):\n",
    "    def __init__(self, root, train=True):\n",
    "        super().__init__()\n",
    "        \n",
    "        self.root=root\n",
    "        with open(root, 'rb') as f:\n",
    "            contents = pickle.load(f)\n",
    "        if train:\n",
    "            self.data = contents['train_data']\n",
    "            self.targets = contents['train_label']\n",
    "        else:\n",
    "            self.data = contents['test_data']\n",
    "            self.targets = contents['test_label']\n",
    "        print(\"data size\", self.data.shape)\n",
    "    \n",
    "    def __len__(self):\n",
    "        return self.targets.shape[0]\n",
    "    \n",
    "    def __getitem__(self, idx):\n",
    "        if \"mnist\" in self.root:\n",
    "            return self.data[idx].unsqueeze(0), self.targets[idx]\n",
    "        else:\n",
    "            return self.data[idx], self.targets[idx]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 269,
   "id": "91ccead8-1962-4cfd-9d6c-550b900b4d34",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "mc_trainset = MNIST_Combine(\"<root>/mnist_combine.pkl\")\n",
    "trainloader = torch.utils.data.DataLoader(\n",
    "            mc_trainset, batch_size=10, shuffle=True, num_workers=1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 207,
   "id": "878b758f-c9fa-438f-b472-bc978a40a91e",
   "metadata": {
    "tags": []
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor([0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.0000, 0.5000,\n",
      "        0.5000])\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "ii = -1\n",
    "plt.imshow(i1[ii])\n",
    "print(i2[ii])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "b9b9c606-3b48-4238-8539-680aca8a84a7",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "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.9.7"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
