{
  "nbformat": 4,
  "nbformat_minor": 0,
  "metadata": {
    "colab": {
      "provenance": [],
      "machine_shape": "hm",
      "gpuType": "A100"
    },
    "kernelspec": {
      "name": "python3",
      "display_name": "Python 3"
    },
    "language_info": {
      "name": "python"
    },
    "accelerator": "GPU"
  },
  "cells": [
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "WdPysIbDhFsa",
        "outputId": "7258b1dc-41aa-40c8-e3b0-ce0ef1802752"
      },
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "\u001b[?25l   \u001b[90m━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━\u001b[0m \u001b[32m0.0/44.8 kB\u001b[0m \u001b[31m?\u001b[0m eta \u001b[36m-:--:--\u001b[0m\r\u001b[2K   \u001b[90m━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━\u001b[0m \u001b[32m44.8/44.8 kB\u001b[0m \u001b[31m1.8 MB/s\u001b[0m eta \u001b[36m0:00:00\u001b[0m\n",
            "\u001b[?25h  Preparing metadata (setup.py) ... \u001b[?25l\u001b[?25hdone\n",
            "\u001b[2K   \u001b[90m━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━\u001b[0m \u001b[32m363.4/363.4 MB\u001b[0m \u001b[31m3.3 MB/s\u001b[0m eta \u001b[36m0:00:00\u001b[0m\n",
            "\u001b[2K   \u001b[90m━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━\u001b[0m \u001b[32m13.8/13.8 MB\u001b[0m \u001b[31m91.9 MB/s\u001b[0m eta \u001b[36m0:00:00\u001b[0m\n",
            "\u001b[2K   \u001b[90m━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━\u001b[0m \u001b[32m24.6/24.6 MB\u001b[0m \u001b[31m90.0 MB/s\u001b[0m eta \u001b[36m0:00:00\u001b[0m\n",
            "\u001b[2K   \u001b[90m━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━\u001b[0m \u001b[32m883.7/883.7 kB\u001b[0m \u001b[31m59.6 MB/s\u001b[0m eta \u001b[36m0:00:00\u001b[0m\n",
            "\u001b[2K   \u001b[90m━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━\u001b[0m \u001b[32m664.8/664.8 MB\u001b[0m \u001b[31m1.7 MB/s\u001b[0m eta \u001b[36m0:00:00\u001b[0m\n",
            "\u001b[2K   \u001b[90m━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━\u001b[0m \u001b[32m211.5/211.5 MB\u001b[0m \u001b[31m11.5 MB/s\u001b[0m eta \u001b[36m0:00:00\u001b[0m\n",
            "\u001b[2K   \u001b[90m━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━\u001b[0m \u001b[32m56.3/56.3 MB\u001b[0m \u001b[31m42.4 MB/s\u001b[0m eta \u001b[36m0:00:00\u001b[0m\n",
            "\u001b[2K   \u001b[90m━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━\u001b[0m \u001b[32m127.9/127.9 MB\u001b[0m \u001b[31m19.2 MB/s\u001b[0m eta \u001b[36m0:00:00\u001b[0m\n",
            "\u001b[2K   \u001b[90m━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━\u001b[0m \u001b[32m207.5/207.5 MB\u001b[0m \u001b[31m3.3 MB/s\u001b[0m eta \u001b[36m0:00:00\u001b[0m\n",
            "\u001b[2K   \u001b[90m━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━\u001b[0m \u001b[32m21.1/21.1 MB\u001b[0m \u001b[31m110.1 MB/s\u001b[0m eta \u001b[36m0:00:00\u001b[0m\n",
            "\u001b[?25h  Building wheel for clip (setup.py) ... \u001b[?25l\u001b[?25hdone\n"
          ]
        }
      ],
      "source": [
        "!pip install --quiet ftfy regex tqdm\n",
        "!pip install --quiet git+https://github.com/openai/CLIP.git\n",
        "!pip install --quiet pycocotools\n"
      ]
    },
    {
      "cell_type": "markdown",
      "source": [
        "# Download and Prepare the Flickr8k Dataset"
      ],
      "metadata": {
        "id": "OQ5eAnW_lmiK"
      }
    },
    {
      "cell_type": "code",
      "source": [
        "!wget \"https://github.com/awsaf49/flickr-dataset/releases/download/v1.0/flickr8k.zip\"\n",
        "!unzip -q flickr8k.zip -d ./flickr8k\n",
        "!rm flickr8k.zip\n",
        "!echo \"Downloaded Flickr8k dataset successfully.\""
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "OcIsaazVKPio",
        "outputId": "da3189d5-d52e-419b-859a-a3beccb37918"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "--2025-05-13 21:08:32--  https://github.com/awsaf49/flickr-dataset/releases/download/v1.0/flickr8k.zip\n",
            "Resolving github.com (github.com)... 140.82.114.3\n",
            "Connecting to github.com (github.com)|140.82.114.3|:443... connected.\n",
            "HTTP request sent, awaiting response... 302 Found\n",
            "Location: https://objects.githubusercontent.com/github-production-release-asset-2e65be/753516996/d7c62b13-1e50-40ea-8fae-f34a44b1695f?X-Amz-Algorithm=AWS4-HMAC-SHA256&X-Amz-Credential=releaseassetproduction%2F20250513%2Fus-east-1%2Fs3%2Faws4_request&X-Amz-Date=20250513T210832Z&X-Amz-Expires=300&X-Amz-Signature=f11cf73238a69bbd114a29a25881aa49c685c63b16b66d7765e5ee4d3605e129&X-Amz-SignedHeaders=host&response-content-disposition=attachment%3B%20filename%3Dflickr8k.zip&response-content-type=application%2Foctet-stream [following]\n",
            "--2025-05-13 21:08:32--  https://objects.githubusercontent.com/github-production-release-asset-2e65be/753516996/d7c62b13-1e50-40ea-8fae-f34a44b1695f?X-Amz-Algorithm=AWS4-HMAC-SHA256&X-Amz-Credential=releaseassetproduction%2F20250513%2Fus-east-1%2Fs3%2Faws4_request&X-Amz-Date=20250513T210832Z&X-Amz-Expires=300&X-Amz-Signature=f11cf73238a69bbd114a29a25881aa49c685c63b16b66d7765e5ee4d3605e129&X-Amz-SignedHeaders=host&response-content-disposition=attachment%3B%20filename%3Dflickr8k.zip&response-content-type=application%2Foctet-stream\n",
            "Resolving objects.githubusercontent.com (objects.githubusercontent.com)... 185.199.109.133, 185.199.108.133, 185.199.110.133, ...\n",
            "Connecting to objects.githubusercontent.com (objects.githubusercontent.com)|185.199.109.133|:443... connected.\n",
            "HTTP request sent, awaiting response... 200 OK\n",
            "Length: 1112971163 (1.0G) [application/octet-stream]\n",
            "Saving to: ‘flickr8k.zip’\n",
            "\n",
            "flickr8k.zip        100%[===================>]   1.04G  37.0MB/s    in 21s     \n",
            "\n",
            "2025-05-13 21:08:53 (51.2 MB/s) - ‘flickr8k.zip’ saved [1112971163/1112971163]\n",
            "\n",
            "Downloaded Flickr8k dataset successfully.\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "import os, glob, random, torch\n",
        "from PIL import Image\n",
        "from torch.utils.data import DataLoader, Dataset\n",
        "from torchvision import transforms\n",
        "import clip\n",
        "\n",
        "# ─── 0. Set data_root to where you unzipped flickr8k.zip ────────────────\n",
        "data_root = \"./flickr8k\"\n",
        "\n",
        "# ─── 1. Auto‐detect the captions file ────────────────────────────────────\n",
        "#    (search recursively for any .txt under data_root)\n",
        "caption_candidates = glob.glob(os.path.join(data_root, \"**\", \"*.txt\"), recursive=True)\n",
        "assert caption_candidates, f\"No .txt files found under {data_root}\"\n",
        "captions_file = caption_candidates[0]\n",
        "print(\"Using captions file:\", captions_file)\n",
        "\n",
        "# ─── 2. Auto‐detect the images directory ─────────────────────────────────\n",
        "#    (find the first subfolder containing JPGs)\n",
        "images_dir = None\n",
        "for root, dirs, files in os.walk(data_root):\n",
        "    if any(f.lower().endswith((\".jpg\", \".png\")) for f in files):\n",
        "        images_dir = root\n",
        "        break\n",
        "assert images_dir, f\"No image files found under {data_root}\"\n",
        "print(\"Using images dir:\", images_dir)\n",
        "\n",
        "# ─── 3. Hyperparameters ─────────────────────────────────────────────────\n",
        "input_resolution = 224\n",
        "context_length   = 77\n",
        "batch_size       = 64\n",
        "seed             = 42\n",
        "\n",
        "random.seed(seed)\n",
        "torch.manual_seed(seed)\n",
        "\n",
        "# ─── 4. Read & split the captions ────────────────────────────────────────\n",
        "#    Flickr8k.token.txt lines are \"<image_name>#<idx>  <caption>\"\n",
        "# ─── 4. Read & split the captions (fix split on space) ────────────────────\n",
        "# captions.txt lines look like:\n",
        "#   \"1000268201_693b08cb0e.jpg#0 A child in a pink dress is climbing up a set of stairs .\"\n",
        "# ─── 4. Read & split the captions (CSV-style) ───────────────────────────────\n",
        "import csv\n",
        "from sklearn.model_selection import train_test_split\n",
        "\n",
        "# ─── 4. Read & split the captions (CSV parsing + skip header) ──────────────\n",
        "caps = {}\n",
        "with open(captions_file, \"r\") as f:\n",
        "    reader = csv.reader(f)\n",
        "    next(reader, None)           # skip the header row if present\n",
        "    for row in reader:\n",
        "        if len(row) < 2: continue\n",
        "        key, caption = row[0], row[1]\n",
        "        img_name = key.split(\"#\")[0]\n",
        "        caps.setdefault(img_name, []).append(caption)\n",
        "\n",
        "# Now filter out any keys that didn’t get images\n",
        "all_imgs = [img for img in caps.keys()\n",
        "            if os.path.exists(os.path.join(images_dir, img))]\n",
        "# and do your 85/15 split\n",
        "train_imgs, val_imgs = train_test_split(all_imgs,\n",
        "                                        test_size=0.15,\n",
        "                                        random_state=seed)\n",
        "\n",
        "\n",
        "# ─── 5. Transforms ───────────────────────────────────────────────────────\n",
        "train_tf = transforms.Compose([\n",
        "    transforms.Resize((input_resolution, input_resolution)),\n",
        "    transforms.ToTensor(),\n",
        "    transforms.Normalize(\n",
        "        mean=(0.48145466, 0.4578275, 0.40821073),\n",
        "        std=(0.26862954, 0.26130258, 0.27577711)\n",
        "    )\n",
        "])\n",
        "eval_tf = train_tf\n",
        "\n",
        "# ─── 6. Dataset classes ──────────────────────────────────────────────────\n",
        "class Flickr8kTrain(Dataset):\n",
        "    def __init__(self, img_list, caps, transform=None):\n",
        "        self.transform = transform\n",
        "        self.data = [\n",
        "            (os.path.join(images_dir, img), cap)\n",
        "            for img in img_list for cap in caps[img]\n",
        "        ]\n",
        "    def __len__(self): return len(self.data)\n",
        "    def __getitem__(self, i):\n",
        "        img_path, caption = self.data[i]\n",
        "        img = Image.open(img_path).convert(\"RGB\")\n",
        "        if self.transform: img = self.transform(img)\n",
        "        txt = clip.tokenize(caption, context_length=context_length)[0]\n",
        "        return img, txt\n",
        "\n",
        "class Flickr8kEval(Dataset):\n",
        "    def __init__(self, img_list, caps, transform=None):\n",
        "        self.transform = transform\n",
        "        self.data = [\n",
        "            (os.path.join(images_dir, img), caps[img])\n",
        "            for img in img_list\n",
        "        ]\n",
        "    def __len__(self): return len(self.data)\n",
        "    def __getitem__(self, i):\n",
        "        img_path, captions = self.data[i]\n",
        "        img = Image.open(img_path).convert(\"RGB\")\n",
        "        if self.transform: img = self.transform(img)\n",
        "        return img, captions\n",
        "\n",
        "def eval_collate(batch):\n",
        "    imgs, caps = zip(*batch)\n",
        "    return torch.stack(imgs, 0), list(caps)\n",
        "\n",
        "# ─── 7. Build dataloaders ────────────────────────────────────────────────\n",
        "train_ds = Flickr8kTrain(train_imgs, caps, transform=train_tf)\n",
        "val_ds   = Flickr8kEval(  val_imgs,   caps, transform=eval_tf)\n",
        "\n",
        "train_dataloader = DataLoader(\n",
        "    train_ds, batch_size=batch_size, shuffle=True,\n",
        "    num_workers=2, pin_memory=True\n",
        ")\n",
        "val_dataloader = DataLoader(\n",
        "    val_ds, batch_size=batch_size, shuffle=False,\n",
        "    num_workers=2, pin_memory=True, collate_fn=eval_collate\n",
        ")\n",
        "\n",
        "print(f\"Train samples: {len(train_ds)}, Val samples: {len(val_ds)}\")\n"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "WfxmIzvYCRI6",
        "outputId": "aef7170b-cfe8-47ec-f77a-a1c024c70637"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Using captions file: ./flickr8k/captions.txt\n",
            "Using images dir: ./flickr8k/Images\n",
            "Train samples: 34385, Val samples: 1214\n"
          ]
        }
      ]
    },
    {
      "cell_type": "markdown",
      "source": [
        "# Load the Teacher Model: CLIP RN50 Model"
      ],
      "metadata": {
        "id": "FMucybFulqLQ"
      }
    },
    {
      "cell_type": "code",
      "source": [
        "import torch\n",
        "import clip\n",
        "import numpy as np\n",
        "\n",
        "\n",
        "device = 'cuda' if torch.cuda.is_available() else 'cpu'\n",
        "\n",
        "\n",
        "# Load the CLIP model\n",
        "model, preprocess = clip.load(\"RN50\", device)\n",
        "model.eval()\n",
        "\n",
        "input_resolution = model.visual.input_resolution\n",
        "context_length = model.context_length\n",
        "vocab_size = model.vocab_size\n",
        "\n",
        "print(\"Model parameters:\", f\"{np.sum([int(np.prod(p.shape)) for p in model.parameters()]):,}\")\n",
        "print(\"Input resolution:\", input_resolution)\n",
        "print(\"Context length:\", context_length)\n",
        "print(\"Vocab size:\", vocab_size)\n"
      ],
      "metadata": {
        "id": "NtcJ2B3fhLfo",
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "outputId": "aa35ab35-c25f-423f-cbcc-8186f22ff216"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "100%|███████████████████████████████████████| 244M/244M [00:21<00:00, 11.8MiB/s]\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Model parameters: 102,007,137\n",
            "Input resolution: 224\n",
            "Context length: 77\n",
            "Vocab size: 49408\n"
          ]
        }
      ]
    },
    {
      "cell_type": "markdown",
      "source": [
        "# Define the Student Model (ResNet-34)"
      ],
      "metadata": {
        "id": "GifHBhlWlr29"
      }
    },
    {
      "cell_type": "code",
      "source": [
        "import torch.nn as nn\n",
        "import torchvision.models as models\n",
        "import torch.nn.functional as F\n",
        "\n",
        "\n",
        "# Student Image Encoder (ResNet-34)\n",
        "class StudentImageEncoder(nn.Module):\n",
        "    def __init__(self, output_dim):\n",
        "        super(StudentImageEncoder, self).__init__()\n",
        "        self.encoder = models.resnet34(pretrained=True)\n",
        "        self.encoder.fc = nn.Linear(self.encoder.fc.in_features, output_dim)\n",
        "\n",
        "    def forward(self, x):\n",
        "        x = self.encoder(x)\n",
        "        x = x / x.norm(dim=-1, keepdim=True)  # Normalize\n",
        "        return x\n",
        "\n",
        "\n",
        "class StudentTextEncoder(nn.Module):\n",
        "    def __init__(self, vocab_size, context_length, output_dim):\n",
        "        super(StudentTextEncoder, self).__init__()\n",
        "        self.token_embedding = nn.Embedding(vocab_size, output_dim)\n",
        "        self.positional_embedding = nn.Parameter(torch.zeros(context_length, output_dim))\n",
        "        nn.init.normal_(self.positional_embedding, std=0.01)\n",
        "        encoder_layer = nn.TransformerEncoderLayer(d_model=output_dim, nhead=8)\n",
        "        self.transformer = nn.TransformerEncoder(encoder_layer, num_layers=2)\n",
        "        self.ln_final = nn.LayerNorm(output_dim)\n",
        "\n",
        "    def forward(self, x):\n",
        "        # x shape: (batch_size, seq_len)\n",
        "        x = self.token_embedding(x) + self.positional_embedding  # (batch_size, seq_len, output_dim)\n",
        "        x = x.permute(1, 0, 2)  # (seq_len, batch_size, output_dim)\n",
        "        x = self.transformer(x)\n",
        "        x = x.permute(1, 0, 2)  # (batch_size, seq_len, output_dim)\n",
        "        x = self.ln_final(x)\n",
        "        x = x.mean(dim=1)  # Mean pooling over the sequence length\n",
        "        x = x / x.norm(dim=-1, keepdim=True)  # Normalize to unit length\n",
        "        return x  # (batch_size, output_dim)\n",
        "\n"
      ],
      "metadata": {
        "id": "Bu5lXd8ehL2K"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "markdown",
      "source": [
        "# Define the Contrastive Loss Function Using LCL - TE1 - TE2"
      ],
      "metadata": {
        "id": "KijNXkynlvLL"
      }
    },
    {
      "cell_type": "code",
      "source": [
        "def contrastive_loss_with_kl_l2_te(\n",
        "    student_image_features,\n",
        "    student_text_features,\n",
        "    teacher_image_features,\n",
        "    teacher_text_features,\n",
        "    temperature=0.07,\n",
        "    alpha=1.0,  # weight for KL term\n",
        "    beta=100,   # weight for L2 term\n",
        "    gamma=5  # weight for TE rewards\n",
        "):\n",
        "    eps = 1e-6\n",
        "\n",
        "    # Normalize features\n",
        "    student_image_features = student_image_features / student_image_features.norm(dim=-1, keepdim=True)\n",
        "    student_text_features = student_text_features / student_text_features.norm(dim=-1, keepdim=True)\n",
        "    teacher_image_features = teacher_image_features / teacher_image_features.norm(dim=-1, keepdim=True)\n",
        "    teacher_text_features = teacher_text_features / teacher_text_features.norm(dim=-1, keepdim=True)\n",
        "\n",
        "    # Compute student logits\n",
        "    logits_per_image_student = student_image_features @ student_text_features.t() / temperature\n",
        "    logits_per_text_student = logits_per_image_student.t()\n",
        "\n",
        "    # Compute teacher logits (no gradients)\n",
        "    with torch.no_grad():\n",
        "        logits_per_image_teacher = teacher_image_features @ teacher_text_features.t() / temperature\n",
        "        logits_per_text_teacher = logits_per_image_teacher.t()\n",
        "\n",
        "    # Contrastive loss\n",
        "    batch_size = student_image_features.size(0)\n",
        "    labels = torch.arange(batch_size, device=student_image_features.device)\n",
        "    loss_image = F.cross_entropy(logits_per_image_student, labels)\n",
        "    loss_text = F.cross_entropy(logits_per_text_student, labels)\n",
        "    contrastive_loss = (loss_image + loss_text) / 2\n",
        "\n",
        "\n",
        "    # L2 distance loss\n",
        "    #l2_img = F.mse_loss(student_image_features, teacher_image_features)\n",
        "    #l2_txt = F.mse_loss(student_text_features, teacher_text_features)\n",
        "    #l2_loss = (l2_img + l2_txt) / 2\n",
        "\n",
        "\n",
        "\n",
        "    # ---------------------\n",
        "    # Cosine Similarity TE Surrogate\n",
        "    # ---------------------\n",
        "    def cosine_te(student_features, teacher_features):\n",
        "        \"\"\"\n",
        "        Approximates transfer entropy by computing the cosine similarity between\n",
        "        the differences (i.e., directional changes) of consecutive embeddings.\n",
        "        A higher cosine similarity indicates that the student is following the teacher's direction.\n",
        "        \"\"\"\n",
        "        # Compute differences along the batch dimension (assumes batch ordering approximates temporal ordering)\n",
        "        student_diff = student_features[1:] - student_features[:-1]\n",
        "        teacher_diff = teacher_features[1:] - teacher_features[:-1]\n",
        "        cos_sim = F.cosine_similarity(student_diff, teacher_diff, dim=-1, eps=eps)\n",
        "        return cos_sim.mean()\n",
        "\n",
        "    te_img = cosine_te(student_image_features, teacher_image_features)\n",
        "    te_txt = cosine_te(student_text_features, teacher_text_features)\n",
        "    te1 = (te_img + te_txt) / 2\n",
        "\n",
        "\n",
        "\n",
        "    # ---------------------\n",
        "    # Cosine Similarity TE Surrogate on Concatenated Differences\n",
        "    # ---------------------\n",
        "    # Compute differences along the batch dimension (assumes batch ordering approximates temporal ordering)\n",
        "    student_diff_img = student_image_features[1:] - student_image_features[:-1]\n",
        "    teacher_diff_img = teacher_image_features[1:] - teacher_image_features[:-1]\n",
        "    student_diff_txt = student_text_features[1:] - student_text_features[:-1]\n",
        "    teacher_diff_txt = teacher_text_features[1:] - teacher_text_features[:-1]\n",
        "\n",
        "    # Concatenate differences from image and text modalities\n",
        "    student_diff_cat = torch.cat([student_diff_img, student_diff_txt], dim=-1)\n",
        "    teacher_diff_cat = torch.cat([teacher_diff_img, teacher_diff_txt], dim=-1)\n",
        "\n",
        "    # Compute cosine similarity between concatenated difference vectors\n",
        "    te2 = F.cosine_similarity(student_diff_cat, teacher_diff_cat, dim=-1, eps=eps).mean()\n",
        "\n",
        "\n",
        "    # Combine all losses\n",
        "    # Increase synergy by subtracting gamma * synergy\n",
        "    # Maximize redundancy by subtracting epsilon * redundancy\n",
        "    #total_loss = (contrastive_loss + alpha * kl_loss + beta * l2_loss - gamma * te)\n",
        "    total_loss = (contrastive_loss - gamma * te1 - gamma * te2)\n",
        "\n",
        "\n",
        "    return total_loss, te1, te2\n"
      ],
      "metadata": {
        "id": "adWN2SVlXyrb"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "markdown",
      "source": [
        "# Set Up the Training Loop"
      ],
      "metadata": {
        "id": "BijGiufjlw8K"
      }
    },
    {
      "cell_type": "code",
      "source": [
        "# Instantiate student models\n",
        "student_image_encoder = StudentImageEncoder(output_dim=1024).to(device)\n",
        "student_text_encoder = StudentTextEncoder(vocab_size, context_length, output_dim=1024).to(device)\n",
        "\n",
        "# Define optimizer\n",
        "optimizer = torch.optim.Adam(\n",
        "    list(student_image_encoder.parameters()) + list(student_text_encoder.parameters()),\n",
        "    lr=1e-4\n",
        ")\n"
      ],
      "metadata": {
        "id": "UyA-cl0LhM6g",
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "outputId": "baeb1f36-9c8d-4625-ea65-15066d9be51f"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "/usr/local/lib/python3.11/dist-packages/torchvision/models/_utils.py:208: UserWarning: The parameter 'pretrained' is deprecated since 0.13 and may be removed in the future, please use 'weights' instead.\n",
            "  warnings.warn(\n",
            "/usr/local/lib/python3.11/dist-packages/torchvision/models/_utils.py:223: UserWarning: Arguments other than a weight enum or `None` for 'weights' are deprecated since 0.13 and may be removed in the future. The current behavior is equivalent to passing `weights=ResNet34_Weights.IMAGENET1K_V1`. You can also use `weights=ResNet34_Weights.DEFAULT` to get the most up-to-date weights.\n",
            "  warnings.warn(msg)\n",
            "Downloading: \"https://download.pytorch.org/models/resnet34-b627a593.pth\" to /root/.cache/torch/hub/checkpoints/resnet34-b627a593.pth\n",
            "100%|██████████| 83.3M/83.3M [00:00<00:00, 214MB/s]\n",
            "/usr/local/lib/python3.11/dist-packages/torch/nn/modules/transformer.py:385: UserWarning: enable_nested_tensor is True, but self.use_nested_tensor is False because encoder_layer.self_attn.batch_first was not True(use batch_first for better inference performance)\n",
            "  warnings.warn(\n"
          ]
        }
      ]
    },
    {
      "cell_type": "markdown",
      "source": [
        "# Train the Student Model"
      ],
      "metadata": {
        "id": "0isNcb4ClyWa"
      }
    },
    {
      "cell_type": "code",
      "source": [
        "# Training Loop\n",
        "num_epochs = 10  # the number of epochs\n",
        "\n",
        "for epoch in range(num_epochs):\n",
        "    student_image_encoder.train()\n",
        "    student_text_encoder.train()\n",
        "\n",
        "    # Initialize trackers for average metrics\n",
        "    total_loss = 0.0\n",
        "    total_te1 = 0.0\n",
        "    total_te2 = 0.0\n",
        "\n",
        "    total_batches = len(train_dataloader)\n",
        "\n",
        "    for batch_idx, (images, texts) in enumerate(train_dataloader):\n",
        "        images = images.to(device)\n",
        "        texts = texts.to(device)\n",
        "\n",
        "        # Teacher outputs\n",
        "        with torch.no_grad():\n",
        "            teacher_image_features = model.encode_image(images)\n",
        "            teacher_text_features = model.encode_text(texts)\n",
        "\n",
        "        # Student outputs\n",
        "        student_image_features = student_image_encoder(images).to(teacher_image_features.dtype)\n",
        "        student_text_features = student_text_encoder(texts).to(teacher_text_features.dtype)\n",
        "\n",
        "        # Compute Contrastive Loss with detailed returns\n",
        "        loss, te1, te2 = contrastive_loss_with_kl_l2_te(\n",
        "            student_image_features,\n",
        "            student_text_features,\n",
        "            teacher_image_features,\n",
        "            teacher_text_features\n",
        "        )\n",
        "\n",
        "        # Backpropagation\n",
        "        optimizer.zero_grad()\n",
        "        loss.backward()\n",
        "        optimizer.step()\n",
        "\n",
        "        # Update trackers\n",
        "        total_loss += loss.item()\n",
        "        total_te1 += te1.item()\n",
        "        total_te2 += te2.item()\n",
        "\n",
        "        if batch_idx % 100 == 0:\n",
        "            print(\n",
        "                f\"Epoch [{epoch+1}/{num_epochs}], Step [{batch_idx}/{len(train_dataloader)}], \"\n",
        "                f\"Loss: {loss.item():.4f}, \"\n",
        "                f\"TE1: {te1:.4f}, TE2: {te2:.4f}\"\n",
        "            )\n",
        "\n",
        "    # Compute average metrics for the epoch\n",
        "    avg_loss = total_loss / total_batches\n",
        "    avg_te1 = total_te1 / total_batches\n",
        "    avg_te2 = total_te2 / total_batches\n",
        "\n",
        "#    avg_synergy_diff = total_synergy_diff / total_batches\n",
        "\n",
        "    # Print epoch-level metrics\n",
        "    print(f\"Epoch [{epoch+1}/{num_epochs}] Averages -> \"\n",
        "          f\"Loss: {avg_loss:.4f}, \"\n",
        "          f\"TE1: {avg_te1:.4f}, TE2: {avg_te2:.4f}\")\n"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "OIO4zuMbXdbY",
        "outputId": "008dd3f4-201c-4305-d4ff-c662369ef7c5"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Epoch [1/10], Step [0/538], Loss: 4.1523, TE1: 0.0014, TE2: 0.0029\n",
            "Epoch [1/10], Step [100/538], Loss: -2.0039, TE1: 0.4521, TE2: 0.4541\n",
            "Epoch [1/10], Step [200/538], Loss: -3.7598, TE1: 0.5439, TE2: 0.5469\n",
            "Epoch [1/10], Step [300/538], Loss: -4.2930, TE1: 0.5693, TE2: 0.5688\n",
            "Epoch [1/10], Step [400/538], Loss: -4.7812, TE1: 0.6104, TE2: 0.6113\n",
            "Epoch [1/10], Step [500/538], Loss: -4.8984, TE1: 0.6064, TE2: 0.6069\n",
            "Epoch [1/10] Averages -> Loss: -3.2217, TE1: 0.5133, TE2: 0.5142\n",
            "Epoch [2/10], Step [0/538], Loss: -5.0078, TE1: 0.6021, TE2: 0.6021\n",
            "Epoch [2/10], Step [100/538], Loss: -5.0859, TE1: 0.6060, TE2: 0.6064\n",
            "Epoch [2/10], Step [200/538], Loss: -5.5703, TE1: 0.6465, TE2: 0.6479\n",
            "Epoch [2/10], Step [300/538], Loss: -5.7891, TE1: 0.6689, TE2: 0.6689\n",
            "Epoch [2/10], Step [400/538], Loss: -5.5742, TE1: 0.6543, TE2: 0.6562\n",
            "Epoch [2/10], Step [500/538], Loss: -6.2305, TE1: 0.7012, TE2: 0.7017\n",
            "Epoch [2/10] Averages -> Loss: -5.7382, TE1: 0.6561, TE2: 0.6567\n",
            "Epoch [3/10], Step [0/538], Loss: -6.3359, TE1: 0.6943, TE2: 0.6934\n",
            "Epoch [3/10], Step [100/538], Loss: -6.3750, TE1: 0.6904, TE2: 0.6904\n",
            "Epoch [3/10], Step [200/538], Loss: -6.6797, TE1: 0.7319, TE2: 0.7314\n",
            "Epoch [3/10], Step [300/538], Loss: -6.5703, TE1: 0.7129, TE2: 0.7139\n",
            "Epoch [3/10], Step [400/538], Loss: -6.8281, TE1: 0.7344, TE2: 0.7349\n",
            "Epoch [3/10], Step [500/538], Loss: -6.8438, TE1: 0.7368, TE2: 0.7378\n",
            "Epoch [3/10] Averages -> Loss: -6.5254, TE1: 0.7099, TE2: 0.7104\n",
            "Epoch [4/10], Step [0/538], Loss: -7.1094, TE1: 0.7534, TE2: 0.7529\n",
            "Epoch [4/10], Step [100/538], Loss: -6.8672, TE1: 0.7354, TE2: 0.7363\n",
            "Epoch [4/10], Step [200/538], Loss: -6.9414, TE1: 0.7285, TE2: 0.7305\n",
            "Epoch [4/10], Step [300/538], Loss: -6.8594, TE1: 0.7314, TE2: 0.7329\n",
            "Epoch [4/10], Step [400/538], Loss: -7.1016, TE1: 0.7598, TE2: 0.7603\n",
            "Epoch [4/10], Step [500/538], Loss: -7.0859, TE1: 0.7568, TE2: 0.7573\n",
            "Epoch [4/10] Averages -> Loss: -6.9850, TE1: 0.7460, TE2: 0.7466\n",
            "Epoch [5/10], Step [0/538], Loss: -7.0977, TE1: 0.7500, TE2: 0.7510\n",
            "Epoch [5/10], Step [100/538], Loss: -7.2148, TE1: 0.7695, TE2: 0.7710\n",
            "Epoch [5/10], Step [200/538], Loss: -7.3867, TE1: 0.7754, TE2: 0.7754\n",
            "Epoch [5/10], Step [300/538], Loss: -7.2031, TE1: 0.7666, TE2: 0.7681\n",
            "Epoch [5/10], Step [400/538], Loss: -7.1641, TE1: 0.7529, TE2: 0.7539\n",
            "Epoch [5/10], Step [500/538], Loss: -7.3516, TE1: 0.7764, TE2: 0.7773\n",
            "Epoch [5/10] Averages -> Loss: -7.3061, TE1: 0.7719, TE2: 0.7726\n",
            "Epoch [6/10], Step [0/538], Loss: -7.6953, TE1: 0.8027, TE2: 0.8018\n",
            "Epoch [6/10], Step [100/538], Loss: -7.5391, TE1: 0.7939, TE2: 0.7944\n",
            "Epoch [6/10], Step [200/538], Loss: -7.6445, TE1: 0.7944, TE2: 0.7959\n",
            "Epoch [6/10], Step [300/538], Loss: -7.5469, TE1: 0.7915, TE2: 0.7910\n",
            "Epoch [6/10], Step [400/538], Loss: -7.5000, TE1: 0.7876, TE2: 0.7886\n",
            "Epoch [6/10], Step [500/538], Loss: -7.7344, TE1: 0.8027, TE2: 0.8042\n",
            "Epoch [6/10] Averages -> Loss: -7.5498, TE1: 0.7925, TE2: 0.7930\n",
            "Epoch [7/10], Step [0/538], Loss: -7.8281, TE1: 0.8120, TE2: 0.8115\n",
            "Epoch [7/10], Step [100/538], Loss: -7.6875, TE1: 0.8018, TE2: 0.8022\n",
            "Epoch [7/10], Step [200/538], Loss: -7.6016, TE1: 0.8047, TE2: 0.8052\n",
            "Epoch [7/10], Step [300/538], Loss: -7.8516, TE1: 0.8164, TE2: 0.8164\n",
            "Epoch [7/10], Step [400/538], Loss: -7.8477, TE1: 0.8169, TE2: 0.8164\n",
            "Epoch [7/10], Step [500/538], Loss: -7.7227, TE1: 0.8076, TE2: 0.8071\n",
            "Epoch [7/10] Averages -> Loss: -7.7371, TE1: 0.8089, TE2: 0.8093\n",
            "Epoch [8/10], Step [0/538], Loss: -7.9375, TE1: 0.8262, TE2: 0.8267\n",
            "Epoch [8/10], Step [100/538], Loss: -7.8516, TE1: 0.8140, TE2: 0.8154\n",
            "Epoch [8/10], Step [200/538], Loss: -7.8984, TE1: 0.8252, TE2: 0.8252\n",
            "Epoch [8/10], Step [300/538], Loss: -7.9688, TE1: 0.8223, TE2: 0.8228\n",
            "Epoch [8/10], Step [400/538], Loss: -7.9375, TE1: 0.8228, TE2: 0.8228\n",
            "Epoch [8/10], Step [500/538], Loss: -7.8281, TE1: 0.8271, TE2: 0.8267\n",
            "Epoch [8/10] Averages -> Loss: -7.9016, TE1: 0.8226, TE2: 0.8231\n",
            "Epoch [9/10], Step [0/538], Loss: -8.0625, TE1: 0.8311, TE2: 0.8320\n",
            "Epoch [9/10], Step [100/538], Loss: -8.0781, TE1: 0.8340, TE2: 0.8350\n",
            "Epoch [9/10], Step [200/538], Loss: -8.1562, TE1: 0.8408, TE2: 0.8408\n",
            "Epoch [9/10], Step [300/538], Loss: -7.9570, TE1: 0.8296, TE2: 0.8301\n",
            "Epoch [9/10], Step [400/538], Loss: -7.9414, TE1: 0.8232, TE2: 0.8237\n",
            "Epoch [9/10], Step [500/538], Loss: -8.0234, TE1: 0.8369, TE2: 0.8359\n",
            "Epoch [9/10] Averages -> Loss: -8.0253, TE1: 0.8336, TE2: 0.8340\n",
            "Epoch [10/10], Step [0/538], Loss: -8.1484, TE1: 0.8403, TE2: 0.8408\n",
            "Epoch [10/10], Step [100/538], Loss: -8.1484, TE1: 0.8457, TE2: 0.8486\n",
            "Epoch [10/10], Step [200/538], Loss: -8.2031, TE1: 0.8477, TE2: 0.8467\n",
            "Epoch [10/10], Step [300/538], Loss: -8.1016, TE1: 0.8428, TE2: 0.8428\n",
            "Epoch [10/10], Step [400/538], Loss: -8.1406, TE1: 0.8447, TE2: 0.8447\n",
            "Epoch [10/10], Step [500/538], Loss: -8.0859, TE1: 0.8418, TE2: 0.8433\n",
            "Epoch [10/10] Averages -> Loss: -8.1265, TE1: 0.8427, TE2: 0.8431\n"
          ]
        }
      ]
    },
    {
      "cell_type": "markdown",
      "source": [
        "# Evaluate the Trained Student Model"
      ],
      "metadata": {
        "id": "Q_tN8A1Cyzzj"
      }
    },
    {
      "cell_type": "code",
      "source": [
        "import torch\n",
        "from torch.utils.data import DataLoader\n",
        "from torchvision import datasets, transforms\n",
        "from torch.utils.data import Dataset\n",
        "import clip\n",
        "import os\n",
        "import numpy as np\n",
        "\n",
        "device = \"cuda\" if torch.cuda.is_available() else \"cpu\"\n",
        "input_resolution = 224\n",
        "context_length = 77\n",
        "\n",
        "\n",
        "\n",
        "student_image_encoder.eval()\n",
        "student_text_encoder.eval()\n",
        "\n",
        "all_image_features = []\n",
        "all_text_features = []\n",
        "image_to_text_indices = []  # For each image, store which text indices correspond to its captions\n",
        "all_captions_flat = []  # We'll store all captions globally\n",
        "\n",
        "with torch.no_grad():\n",
        "    image_count = 0\n",
        "    text_count = 0\n",
        "    for images, batch_captions in val_dataloader:\n",
        "        # images: (B, C, H, W)\n",
        "        # batch_captions: list of length B, each item is a list of captions for that image\n",
        "\n",
        "        images = images.to(device)\n",
        "\n",
        "        # Encode images\n",
        "        image_feats = student_image_encoder(images)\n",
        "        image_feats = image_feats / image_feats.norm(dim=-1, keepdim=True)\n",
        "        all_image_features.append(image_feats.cpu())\n",
        "\n",
        "        # Flatten captions for this batch\n",
        "        flat_captions = []\n",
        "        image_to_text_map_for_batch = []\n",
        "        for caps in batch_captions:\n",
        "            start_idx = len(flat_captions)\n",
        "            flat_captions.extend(caps)  # add all captions from this image\n",
        "            end_idx = len(flat_captions)\n",
        "            # This image's captions correspond to indices [start_idx+text_count, end_idx+text_count)\n",
        "            image_to_text_map_for_batch.append((start_idx + text_count, end_idx + text_count))\n",
        "\n",
        "        # Tokenize all captions in the batch at once\n",
        "        texts = clip.tokenize(flat_captions, context_length=context_length).to(device)\n",
        "        text_feats = student_text_encoder(texts)\n",
        "        text_feats = text_feats / text_feats.norm(dim=-1, keepdim=True)\n",
        "\n",
        "        # Store text features globally\n",
        "        all_text_features.append(text_feats.cpu())\n",
        "        all_captions_flat.extend(flat_captions)\n",
        "\n",
        "        # Update the global mapping\n",
        "        for (start_idx, end_idx) in image_to_text_map_for_batch:\n",
        "            image_to_text_indices.append(list(range(start_idx, end_idx)))\n",
        "\n",
        "        image_count += images.size(0)\n",
        "        text_count += len(flat_captions)\n",
        "\n",
        "all_image_features = torch.cat(all_image_features, dim=0)  # (N_images, embed_dim)\n",
        "all_text_features = torch.cat(all_text_features, dim=0)    # (N_captions_total, embed_dim)\n",
        "\n",
        "# Compute similarity matrix: shape (N_images, N_captions_total)\n",
        "sim_matrix = all_image_features @ all_text_features.t()\n",
        "\n",
        "def compute_recall_with_multiple_captions(sim_matrix, image_to_text_indices, k=1):\n",
        "    n = sim_matrix.size(0)\n",
        "    successes = 0\n",
        "    for i in range(n):\n",
        "        scores = sim_matrix[i]\n",
        "        sorted_indices = torch.argsort(scores, descending=True)\n",
        "\n",
        "        correct_indices = set(image_to_text_indices[i])\n",
        "        ranks_of_correct = []\n",
        "        for cidx in correct_indices:\n",
        "            pos = (sorted_indices == cidx).nonzero(as_tuple=True)\n",
        "            if len(pos) > 0:\n",
        "                ranks_of_correct.append(pos[0].item())\n",
        "\n",
        "        if len(ranks_of_correct) > 0:\n",
        "            min_rank = min(ranks_of_correct)\n",
        "            if min_rank < k:\n",
        "                successes += 1\n",
        "    recall = successes / n\n",
        "    return recall\n",
        "\n",
        "# Image-to-Text Retrieval\n",
        "r1 = compute_recall_with_multiple_captions(sim_matrix, image_to_text_indices, k=1)\n",
        "r5 = compute_recall_with_multiple_captions(sim_matrix, image_to_text_indices, k=5)\n",
        "r10 = compute_recall_with_multiple_captions(sim_matrix, image_to_text_indices, k=10)\n",
        "\n",
        "print(\"Image-to-Text Retrieval:\")\n",
        "print(f\"Recall@1: {r1*100:.2f}%\")\n",
        "print(f\"Recall@5: {r5*100:.2f}%\")\n",
        "print(f\"Recall@10: {r10*100:.2f}%\")\n",
        "\n",
        "# Text-to-Image Retrieval\n",
        "# Create reverse mapping from text index to image index\n",
        "text_to_image = [None]*all_text_features.size(0)\n",
        "for i, tinds in enumerate(image_to_text_indices):\n",
        "    for t in tinds:\n",
        "        text_to_image[t] = i\n",
        "\n",
        "sim_matrix_t2i = sim_matrix.t()  # (N_captions_total, N_images)\n",
        "\n",
        "def compute_recall_text_to_image(sim_matrix_t2i, text_to_image, k=1):\n",
        "    m = sim_matrix_t2i.size(0)\n",
        "    successes = 0\n",
        "    for j in range(m):\n",
        "        scores = sim_matrix_t2i[j]\n",
        "        sorted_indices = torch.argsort(scores, descending=True)\n",
        "        correct_image = text_to_image[j]\n",
        "        rank = (sorted_indices == correct_image).nonzero(as_tuple=True)[0].item()\n",
        "        if rank < k:\n",
        "            successes += 1\n",
        "    recall = successes / m\n",
        "    return recall\n",
        "\n",
        "r1_t2i = compute_recall_text_to_image(sim_matrix_t2i, text_to_image, k=1)\n",
        "r5_t2i = compute_recall_text_to_image(sim_matrix_t2i, text_to_image, k=5)\n",
        "r10_t2i = compute_recall_text_to_image(sim_matrix_t2i, text_to_image, k=10)\n",
        "\n",
        "print(\"Text-to-Image Retrieval:\")\n",
        "print(f\"Recall@1: {r1_t2i*100:.2f}%\")\n",
        "print(f\"Recall@5: {r5_t2i*100:.2f}%\")\n",
        "print(f\"Recall@10: {r10_t2i*100:.2f}%\")\n"
      ],
      "metadata": {
        "id": "d42i-P_Yrq-Y",
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "outputId": "b1c8ce16-8b8d-4eea-e561-1e95b1bea643"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Image-to-Text Retrieval:\n",
            "Recall@1: 32.29%\n",
            "Recall@5: 62.36%\n",
            "Recall@10: 75.29%\n",
            "Text-to-Image Retrieval:\n",
            "Recall@1: 24.50%\n",
            "Recall@5: 54.56%\n",
            "Recall@10: 68.14%\n"
          ]
        }
      ]
    }
  ]
}