{
  "cells": [
    {
      "cell_type": "markdown",
      "metadata": {
        "id": "pEAjCLI8QCjU"
      },
      "source": [
        "##### Copyright 2020 Google LLC.\n",
        "\n",
        "Licensed under the Apache License, Version 2.0 (the \"License\");\n",
        "you may not use this file except in compliance with the License.\n",
        "You may obtain a copy of the License at\n",
        "\n",
        "https://www.apache.org/licenses/LICENSE-2.0\n",
        "\n",
        "Unless required by applicable law or agreed to in writing, software\n",
        "distributed under the License is distributed on an \"AS IS\" BASIS,\n",
        "WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n",
        "See the License for the specific language governing permissions and\n",
        "limitations under the License."
      ]
    },
    {
      "cell_type": "markdown",
      "metadata": {
        "id": "2lCKaZSM2Ac0"
      },
      "source": [
        "## RandBits MNIST using Contrastive Learning."
      ]
    },
    {
      "cell_type": "markdown",
      "metadata": {
        "id": "PZrwIvTl05P6"
      },
      "source": [
        "This notebook trains an unsupervised model on the RandBits MNIST dataset using contrastive learning, similar to the results shown in Figure 5(a) in ***Intriguing Properties of Contrastive Losses***."
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "MY8KIuqHHHf_",
        "outputId": "58ee9fc7-ea85-442d-e443-47f619e01fda"
      },
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "Collecting snfpy\n",
            "  Downloading snfpy-0.2.2-py3-none-any.whl.metadata (8.8 kB)\n",
            "Requirement already satisfied: numpy>=1.14 in /usr/local/lib/python3.10/dist-packages (from snfpy) (1.26.4)\n",
            "Requirement already satisfied: scikit-learn in /usr/local/lib/python3.10/dist-packages (from snfpy) (1.5.2)\n",
            "Requirement already satisfied: scipy in /usr/local/lib/python3.10/dist-packages (from snfpy) (1.13.1)\n",
            "Requirement already satisfied: joblib>=1.2.0 in /usr/local/lib/python3.10/dist-packages (from scikit-learn->snfpy) (1.4.2)\n",
            "Requirement already satisfied: threadpoolctl>=3.1.0 in /usr/local/lib/python3.10/dist-packages (from scikit-learn->snfpy) (3.5.0)\n",
            "Downloading snfpy-0.2.2-py3-none-any.whl (550 kB)\n",
            "\u001b[?25l   \u001b[90m━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━\u001b[0m \u001b[32m0.0/550.6 kB\u001b[0m \u001b[31m?\u001b[0m eta \u001b[36m-:--:--\u001b[0m\r\u001b[2K   \u001b[91m━━━━━━━━━━━━━━\u001b[0m\u001b[90m╺\u001b[0m\u001b[90m━━━━━━━━━━━━━━━━━━━━━━━━━\u001b[0m \u001b[32m194.6/550.6 kB\u001b[0m \u001b[31m5.6 MB/s\u001b[0m eta \u001b[36m0:00:01\u001b[0m\r\u001b[2K   \u001b[90m━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━\u001b[0m \u001b[32m550.6/550.6 kB\u001b[0m \u001b[31m7.9 MB/s\u001b[0m eta \u001b[36m0:00:00\u001b[0m\n",
            "\u001b[?25hInstalling collected packages: snfpy\n",
            "Successfully installed snfpy-0.2.2\n"
          ]
        }
      ],
      "source": [
        "!pip install snfpy"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "PIGt3mgJEKDB"
      },
      "outputs": [],
      "source": [
        "import os\n",
        "os.environ[\"CUDA_VISIBLE_DEVICES\"] = \"-1\"\n",
        "\n",
        "#@title Imports.\n",
        "import tensorflow.compat.v2 as tf\n",
        "tf.enable_v2_behavior()\n",
        "\n",
        "\n",
        "import snf.compute\n"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "cellView": "form",
        "id": "0amaY7x4wgGr"
      },
      "outputs": [],
      "source": [
        "import tensorflow_datasets as tfds\n",
        "import matplotlib.pyplot as plt\n",
        "import seaborn as sns\n",
        "import math\n",
        "import pandas as pd"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "dAa3IRTov5Bk"
      },
      "outputs": [],
      "source": [
        "#@title Data preprocessing.\n",
        "def random_crop_and_resize(image):\n",
        "  sample_distorted_bounding_box = tf.image.sample_distorted_bounding_box(\n",
        "      image_size=tf.shape(image),\n",
        "      bounding_boxes=tf.constant(\n",
        "          [0.0, 0.0, 1.0, 1.0], dtype=tf.float32, shape=[1, 1, 4]),\n",
        "      min_object_covered=0.1,\n",
        "      aspect_ratio_range=(3. / 4, 4. / 3.),\n",
        "      area_range=(0.5, 1.0),\n",
        "      max_attempts=100,\n",
        "      use_image_if_no_bounding_boxes=True)\n",
        "  bbox_begin, bbox_size, _ = sample_distorted_bounding_box\n",
        "\n",
        "  # Crop the image to the specified bounding box.\n",
        "  offset_y, offset_x, _ = tf.unstack(bbox_begin)\n",
        "  target_height, target_width, _ = tf.unstack(bbox_size)\n",
        "  image = tf.image.crop_to_bounding_box(\n",
        "      image, offset_y, offset_x, target_height, target_width)\n",
        "\n",
        "  return tf.image.resize(image, [28, 28], method=tf.image.ResizeMethod.BILINEAR)\n",
        "\n",
        "def hash_image_to_bits(image, extra_channel_bits):\n",
        "  hash = tf.compat.v1.strings.to_hash_bucket_fast(\n",
        "      tf.image.encode_jpeg(image),\n",
        "      num_buckets=2**extra_channel_bits)\n",
        "  bits = tf.cast(\n",
        "      tf.math.mod(\n",
        "            tf.bitwise.right_shift(tf.expand_dims([hash], 1),\n",
        "            tf.range(extra_channel_bits, dtype=tf.int64)), 2),\n",
        "      tf.float32)\n",
        "  return bits\n",
        "\n",
        "def pack_extra_channels(image, bits):\n",
        "  extra_channel_bits = tf.shape(bits)[-1]\n",
        "  bits = tf.broadcast_to(bits, [28, 28, extra_channel_bits])\n",
        "  return tf.concat([image, tf.cast(bits, tf.float32)], axis=-1)\n",
        "\n",
        "def get_process_fns(extra_channel_bits):\n",
        "  def preprocess_train_fn(image, label):\n",
        "    bits = hash_image_to_bits(image, extra_channel_bits)\n",
        "    image = tf.image.convert_image_dtype(image, dtype=tf.float32)\n",
        "    label = tf.cast(label, tf.int32)\n",
        "\n",
        "    image_a = random_crop_and_resize(image)\n",
        "    image_b = random_crop_and_resize(image)\n",
        "\n",
        "    # Pack extra channels.\n",
        "    if extra_channel_bits > 0:\n",
        "      image_a = pack_extra_channels(image_a, bits)\n",
        "      image_b = pack_extra_channels(image_b, bits)\n",
        "\n",
        "    image = tf.stack([image_a, image_b], axis=0)  # [2, h, w, c]\n",
        "    return (image, label)\n",
        "\n",
        "  def preprocess_eval_fn(image, label):\n",
        "    bits = hash_image_to_bits(image, extra_channel_bits)\n",
        "    image = tf.image.convert_image_dtype(image, dtype=tf.float32)\n",
        "    label = tf.cast(label, tf.int32)\n",
        "    if extra_channel_bits > 0:\n",
        "      image = pack_extra_channels(image, bits)\n",
        "    return (image, label)\n",
        "\n",
        "  return preprocess_train_fn, preprocess_eval_fn"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "IjaSKhqb1qFV"
      },
      "outputs": [],
      "source": [
        "#@title Objective functions.\n",
        "cls_loss_object = tf.keras.losses.CategoricalCrossentropy(\n",
        "    from_logits=True,\n",
        "    reduction=tf.keras.losses.Reduction.NONE)\n",
        "\n",
        "def get_cls_loss(labels, outputs):\n",
        "  return tf.reduce_mean(cls_loss_object(labels, outputs))\n",
        "\n",
        "def get_contrastive_loss(z1, z2, nt_xent_temp):   # [batch_size, dim]\n",
        "  batch_size = tf.shape(z1)[0]\n",
        "  dim = tf.shape(z1)[1]\n",
        "\n",
        "  z1 = tf.math.l2_normalize(z1, -1)\n",
        "  z2 = tf.math.l2_normalize(z2, -1)\n",
        "\n",
        "  sim = tf.matmul(z1, z2, transpose_b=True)  # [batch_size, batch_size]\n",
        "  sim /= nt_xent_temp\n",
        "\n",
        "  labels = tf.eye(batch_size)\n",
        "\n",
        "  loss = (\n",
        "      get_cls_loss(labels, sim) +\n",
        "      get_cls_loss(labels, tf.transpose(sim))\n",
        "  )\n",
        "  return tf.reduce_mean(loss), sim"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "m6Wh8OnyEKDC"
      },
      "outputs": [],
      "source": [
        "import numpy as np\n",
        "from scipy.linalg import eigh\n",
        "\n",
        "\n",
        "# #@title Objective functions.\n",
        "# cls_loss_object = tf.keras.losses.CategoricalCrossentropy(\n",
        "#     from_logits=True,\n",
        "#     reduction=tf.keras.losses.Reduction.NONE)\n",
        "\n",
        "cls_loss_object = tf.keras.losses.CategoricalCrossentropy(\n",
        "    from_logits=True,\n",
        "    reduction=tf.keras.losses.Reduction.NONE,\n",
        "    label_smoothing=0.0 # Ensure this is a float.\n",
        ")\n",
        "\n",
        "def get_cls_loss(labels, outputs):\n",
        "  return tf.reduce_mean(cls_loss_object(labels, outputs))\n",
        "\n",
        "\n",
        "def repmat(A, N, M):\n",
        "    # A is the input matrix/tensor\n",
        "    # N is the number of times A is repeated along the rows\n",
        "    # M is the number of times A is repeated along the columns\n",
        "    expanded_A = tf.expand_dims(A, 0)  # Add a new dimension for tiling\n",
        "    tiled_A = tf.tile(expanded_A, [N, M, 1])  # Tile in the new dimension\n",
        "    final_shape = [N * tf.shape(A)[0], M * tf.shape(A)[1]]\n",
        "    return tf.reshape(tiled_A, final_shape)\n",
        "\n",
        "\n",
        "def get_nlfda_loss(z1, z2, penal_para, margin):   # [batch_size, dim]\n",
        "    # print(\"the shape of z1\")\n",
        "    # print(z1.shape)\n",
        "    batch_size = tf.shape(z1)[0]\n",
        "    n=2*batch_size\n",
        "    # print(\"batch size\")\n",
        "    # print(batch_size)\n",
        "    dim = tf.shape(z1)[1]\n",
        "\n",
        "    z1 = tf.math.l2_normalize(z1, -1)\n",
        "    z2 = tf.math.l2_normalize(z2, -1)\n",
        "\n",
        "\n",
        "    z=tf.concat([z1, z2], axis=0) # dimension: (2*batch_size,dim)\n",
        "\n",
        "    Sb = tf.zeros((dim, dim), dtype=tf.float32)  # Between-class scatter matrix\n",
        "    Sw = tf.zeros((dim, dim), dtype=tf.float32)  # Within-class scatter matrix\n",
        "\n",
        "    zmean=tf.zeros((batch_size, z.shape[1]), dtype=z.dtype)\n",
        "\n",
        "    for i in tf.range(batch_size):\n",
        "        zc = tf.gather(z, indices=[i, i + batch_size], axis=0) #Kz[:, (i,i+batch_size)]\n",
        "\n",
        "        # 2. Sum the squared elements row-wise\n",
        "        col_sums_squared = tf.reduce_sum(tf.square(zc), axis=1)\n",
        "\n",
        "        # 4. Transpose to get a row matrix\n",
        "        zc2 = tf.reshape(col_sums_squared, (1, -1))  # Explicitly reshape to a row matrix if needed.\n",
        "\n",
        "        mean_zc = tf.reduce_mean(zc, axis=0)\n",
        "\n",
        "        updates = tf.reshape(mean_zc, [1, -1])\n",
        "\n",
        "\n",
        "        #indices = tf.constant([[int(i)]])\n",
        "        indices = tf.reshape(i, [1, 1])  #tf.constant([[i]], dtype=tf.int32)\n",
        "\n",
        "        zmean = tf.tensor_scatter_nd_update(zmean, indices, updates)\n",
        "\n",
        "\n",
        "        rep_zc2 = repmat(zc2, 2, 1)\n",
        "\n",
        "        rep_zc2_T = repmat(tf.transpose(zc2), 1, 2)\n",
        "\n",
        "        distance2 = rep_zc2 + rep_zc2_T - 2 * tf.matmul(zc,tf.transpose(zc))\n",
        "\n",
        "\n",
        "        A = tf.exp(-distance2) #get_affinity_matrix(distance2, 1, 2)\n",
        "\n",
        "\n",
        "        zc1 = tf.reduce_sum(zc, axis=0, keepdims=True)\n",
        "\n",
        "\n",
        "        colSums_A = tf.reshape(tf.reduce_sum(A, axis=0), [-1, 1])  # Sum columns and ensure it's a row vector\n",
        "        replicated_colSums_A = tf.tile(colSums_A, [1, tf.shape(zc)[1]])  # Replicate across rows to match Kc's row count\n",
        "\n",
        "        # Perform the matrix operations as described\n",
        "        # Note: In TensorFlow, matrix multiplication is done using tf.matmul or the '@' operator\n",
        "        Z = tf.matmul(tf.transpose(zc), replicated_colSums_A * zc) - tf.matmul(tf.matmul(tf.transpose(zc), A), zc)\n",
        "\n",
        "\n",
        "        Sb += (Z / tf.cast(n, tf.float32)) + tf.transpose(zc) @ zc * (1 - 2.0 / tf.cast(n, tf.float32)) + tf.transpose(zc1) @ zc1 / tf.cast(n, tf.float32)\n",
        "        Sw += Z / 2.0\n",
        "\n",
        "    z1 = tf.reduce_sum(z, axis=0, keepdims=True)\n",
        "\n",
        "    Sb = Sb -  tf.transpose(z1) @ z1 / tf.cast(n, tf.float32) - Sw\n",
        "\n",
        "    Sb = (Sb + tf.transpose(Sb)) / 2.0  # Final between-class scatter matrix\n",
        "    Sw = (Sw + tf.transpose(Sw)) / 2.0  # Final within-class scatter matrix\n",
        "\n",
        "\n",
        "    eye_mat = penal_para * tf.eye(tf.shape(Sw)[0], dtype=Sb.dtype)\n",
        "\n",
        "    B = Sw + eye_mat  # Make sure this is positive definite\n",
        "\n",
        "\n",
        "    temp = tf.linalg.pinv(B) @ Sb\n",
        "\n",
        "\n",
        "    temp=(temp + tf.transpose(temp)) / 2.0\n",
        "\n",
        "    evals, evecs = tf.linalg.eig(temp)\n",
        "\n",
        "    evals_real = tf.math.real(evals)\n",
        "    evecs_real = tf.math.real(evecs)\n",
        "\n",
        "    # print(\"n eigenvalues before\")\n",
        "    # # print(evals_real.shape)\n",
        "    # # print(tf.cast(1.5, tf.float32))\n",
        "\n",
        "    # print(\"the eigenvalues\")\n",
        "    # tf.print(evals_real)\n",
        "\n",
        "    # print(\"the largest eigenvalue\")\n",
        "    # tf.print(tf.reduce_max(evals_real))\n",
        "    # # print(tf.reduce_max(evals_real).numpy())\n",
        "\n",
        "    # print(\"the smallest eigenvalue\")\n",
        "    # tf.print(tf.reduce_min(evals_real))\n",
        "    # # print(tf.print(tf.reduce_min(evals_real)))\n",
        "\n",
        "\n",
        "    sorted_indices = tf.argsort(evals_real)\n",
        "    evals_sorted = tf.gather(evals_real, sorted_indices)\n",
        "    evecs_sorted = tf.gather(evecs_real, axis=1, indices=sorted_indices)\n",
        "\n",
        "    n_components = 128 - 1\n",
        "    evals = evals_sorted[-n_components:]\n",
        "    # print('evals', evals.shape, evals)\n",
        "\n",
        "    # print(\"number of components, n eigenvalues after\")\n",
        "    # print(evals.shape)\n",
        "\n",
        "    # print(\"the eigenvalues\")\n",
        "    # tf.print(evals)\n",
        "\n",
        "    # print(\"the largest eigenvalue\")\n",
        "    # tf.print(tf.reduce_max(evals))\n",
        "\n",
        "    # print(\"the smallest eigenvalue\")\n",
        "    # tf.print(tf.reduce_min(evals))\n",
        "\n",
        "    # Calculate loss\n",
        "\n",
        "    # threshold = tf.reduce_min(evals) + margin  # default of margin is margin = 0.01\n",
        "    # n_eig = tf.reduce_sum(tf.cast(evals < threshold, tf.float32))\n",
        "\n",
        "\n",
        "    # loss = -tf.reduce_mean(evals[:tf.cast(n_eig, tf.int32)])\n",
        "\n",
        "    loss = -tf.reduce_mean(evals)\n",
        "\n",
        "    coef = tf.matmul(tf.matmul(zmean, evecs_sorted), tf.transpose(evecs_sorted))  # CxD\n",
        "    intercept = -0.5 * tf.linalg.diag_part(tf.matmul(zmean, tf.transpose(coef)))  # C\n",
        "\n",
        "\n",
        "    logit = tf.matmul(z, coef, transpose_b=True) + intercept\n",
        "\n",
        "    ypred=tf.argmax(logit, axis=1)\n",
        "\n",
        "    labels1 = tf.range(batch_size)\n",
        "    labels2 = tf.range(batch_size)\n",
        "\n",
        "    labels= tf.concat([labels1, labels2], axis=0)\n",
        "\n",
        "    correct_predictions = tf.equal(ypred, tf.cast(labels, tf.int64))\n",
        "    accuracy = tf.reduce_mean(tf.cast(correct_predictions, tf.float32))\n",
        "\n",
        "\n",
        "    return tf.reduce_mean(loss), accuracy #tf.reduce_mean(loss), sim\n"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "0usbo7eUseR8"
      },
      "outputs": [],
      "source": [
        "#@title Model.\n",
        "def dense_bn_relu(units):\n",
        "  return tf.keras.Sequential([\n",
        "      tf.keras.layers.Dense(\n",
        "          units, use_bias=False,\n",
        "          kernel_regularizer=tf.keras.regularizers.l2(1e-4)),\n",
        "      tf.keras.layers.BatchNormalization(center=True, scale=True),\n",
        "      tf.keras.layers.ReLU()\n",
        "  ])\n",
        "\n",
        "def conv2d_bn_relu(filters, kernel_size, strides):\n",
        "  return tf.keras.Sequential([\n",
        "      tf.keras.layers.Conv2D(\n",
        "          filters, kernel_size, strides, use_bias=False,\n",
        "          kernel_regularizer=tf.keras.regularizers.l2(1e-4)),\n",
        "      tf.keras.layers.BatchNormalization(center=True, scale=True),\n",
        "      tf.keras.layers.ReLU()\n",
        "  ])\n",
        "\n",
        "class ConvN(tf.keras.Model):\n",
        "\n",
        "  def __init__(self, width_multiplier):\n",
        "    super(ConvN, self).__init__()\n",
        "    self.num_classes = 10\n",
        "    self.latent_dim = 256 * width_multiplier\n",
        "    self.proj_dim = self.latent_dim // 2\n",
        "\n",
        "    self.enc = tf.keras.Sequential([\n",
        "        conv2d_bn_relu(32 * width_multiplier, 3, 2),\n",
        "        conv2d_bn_relu(64 * width_multiplier, 3, 2),\n",
        "        conv2d_bn_relu(64 * width_multiplier, 3, 2),\n",
        "        tf.keras.layers.Flatten(),\n",
        "        dense_bn_relu(self.latent_dim)\n",
        "    ])\n",
        "\n",
        "    self.proj = tf.keras.Sequential([\n",
        "        dense_bn_relu(self.latent_dim * 2),\n",
        "        tf.keras.layers.Dense(\n",
        "            self.proj_dim, use_bias=False, activation=None,\n",
        "            kernel_regularizer=tf.keras.regularizers.l2(1e-4)),\n",
        "    ])\n",
        "\n",
        "    self.classifier = tf.keras.layers.Dense(self.num_classes)\n",
        "\n",
        "  def call(self, inputs, training=False):\n",
        "    y = self.enc(inputs, training=training)\n",
        "    z = self.proj(y, training=training)\n",
        "    pred = self.classifier(tf.stop_gradient(y))\n",
        "    return y, z, pred"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "u9F2LOzm2zy4"
      },
      "outputs": [],
      "source": [
        "#@title Define train_and_eval() for contrastive learning.\n",
        "def train_and_eval(\n",
        "    batch_size=128,\n",
        "    width_multiplier=1,\n",
        "    extra_channel_bits=10,\n",
        "    nt_xent_temp=0.1,\n",
        "    margin=0.01,\n",
        "    learning_rate=0.001,\n",
        "    epochs=10,\n",
        "    log_summary_every_n_steps=100,\n",
        "    eval_every_n_steps=100,\n",
        "    print_output=False):\n",
        "  strategy = tf.distribute.MirroredStrategy()\n",
        "  #strategy = tf.distribute.get_strategy()\n",
        "\n",
        "  # Load dataset.\n",
        "  builder = tfds.builder('mnist')\n",
        "  builder.download_and_prepare()\n",
        "\n",
        "  preprocess_train_fn, preprocess_eval_fn = get_process_fns(extra_channel_bits)\n",
        "  train_dataset = builder.as_dataset(split='train', as_supervised=True)\n",
        "  train_dataset = train_dataset.repeat().map(preprocess_train_fn)\n",
        "  train_dataset = train_dataset.batch(batch_size)\n",
        "\n",
        "  test_dataset = builder.as_dataset(split='test', as_supervised=True)\n",
        "  test_dataset = test_dataset.map(preprocess_eval_fn)\n",
        "  test_dataset = test_dataset.batch(batch_size, drop_remainder=False)\n",
        "  train_iter = iter(train_dataset)\n",
        "\n",
        "  total_steps = int(60000 * epochs / batch_size)\n",
        "  steps_per_epoch_test = math.ceil(10000 / batch_size)\n",
        "\n",
        "  with strategy.scope():\n",
        "\n",
        "    # Model and optimizer.\n",
        "    model = ConvN(width_multiplier)\n",
        "    global_step = tf.Variable(\n",
        "        1, trainable=False, name=\"global_step\", dtype=tf.int64)\n",
        "\n",
        "    lr_schedule = tf.keras.optimizers.schedules.PolynomialDecay(\n",
        "        initial_learning_rate=learning_rate,\n",
        "        decay_steps=total_steps,\n",
        "        end_learning_rate=0.)\n",
        "    optimizer = tf.keras.optimizers.Adam(lr_schedule)\n",
        "\n",
        "    # Define metrics.\n",
        "    loss_metrics = [\n",
        "        \"train_classification_loss\",\n",
        "        \"train_contrastive_loss\",\n",
        "        \"train_total_loss\",\n",
        "    ]\n",
        "    acc_metrics = [\"train_accuracy\", \"eval_accuracy\",\n",
        "                  \"train_contrastive_accuracy\"]\n",
        "    metric_list = {s: tf.keras.metrics.Mean(name=s) for s in loss_metrics}\n",
        "    metric_list.update({\n",
        "        s: tf.keras.metrics.SparseCategoricalAccuracy(name=s)\n",
        "        for s in acc_metrics\n",
        "    })\n",
        "\n",
        "    # Step functions.\n",
        "    @tf.function\n",
        "    def train_step(iterator):\n",
        "      def step_fn(inputs):\n",
        "        images, labels = inputs\n",
        "        labels_one_hot = tf.one_hot(labels, depth=10)\n",
        "        images_a, images_b = tf.unstack(images, num=2, axis=1)\n",
        "        with tf.GradientTape() as tape:\n",
        "          _, za, pred_a = model(images_a, training=True)\n",
        "          _, zb, pred_b = model(images_b, training=True)\n",
        "          # print(\"starting loss calculation\")\n",
        "          contrastive_loss, contrastive_sim = (\n",
        "              get_nlfda_loss(za, zb, nt_xent_temp, margin))\n",
        "          # print(\"loss value\")\n",
        "          # print(contrastive_loss)\n",
        "          # print(\"accuracy value\")\n",
        "          # print(contrastive_sim)\n",
        "          # print(\"start classification loss\")\n",
        "          classifier_loss = get_cls_loss(labels_one_hot, pred_a)\n",
        "          # print(\"classifier loss output\")\n",
        "          # print(classifier_loss)\n",
        "          # print(\"strtaing sum of the model loss\")\n",
        "          wd_loss = sum(model.losses)\n",
        "          # print(\"sum of the model loss\")\n",
        "          # print(wd_loss)\n",
        "          loss = classifier_loss + wd_loss + contrastive_loss\n",
        "\n",
        "          batch_size = tf.shape(images)[0]\n",
        "          metric_list[\"train_contrastive_loss\"].update_state(contrastive_loss)\n",
        "          metric_list[\"train_classification_loss\"].update_state(classifier_loss)\n",
        "          metric_list[\"train_accuracy\"].update_state(labels, pred_a)\n",
        "          # print(\"metric train_contrastive_accuracy\")\n",
        "          # # metric_list[\"train_contrastive_accuracy\"].update_state(\n",
        "          # #     tf.range(batch_size), contrastive_sim)\n",
        "          # print(\"metric train_total_loss\")\n",
        "          metric_list[\"train_total_loss\"].update_state(loss)\n",
        "\n",
        "        gradients = tape.gradient(loss, model.trainable_variables)\n",
        "        optimizer.apply_gradients(zip(gradients, model.trainable_variables))\n",
        "\n",
        "      strategy.run(step_fn, args=(next(iterator),))\n",
        "      global_step.assign_add(1)\n",
        "\n",
        "    @tf.function\n",
        "    def eval_step(iterator):\n",
        "      def step_fn(inputs):\n",
        "        images, labels = inputs\n",
        "        _, _, predictions = model(images, training=False)\n",
        "        metric_list[\"eval_accuracy\"].update_state(labels, predictions)\n",
        "      strategy.run(step_fn, args=(next(iterator),))\n",
        "\n",
        "    # Train and eval loop.\n",
        "    steps = []\n",
        "    eval_accuracies = []\n",
        "    while global_step.numpy() <= total_steps:\n",
        "      train_step(train_iter)\n",
        "      step = global_step.numpy()\n",
        "\n",
        "      if step % log_summary_every_n_steps == 0:\n",
        "        log_msg = \"Steps: {}\".format(step)\n",
        "        for m in loss_metrics + acc_metrics:\n",
        "          if m.startswith(\"train\"):\n",
        "            log_msg += \", {}: {}\".format(m, metric_list[m].result())\n",
        "            metric_list[m].reset_state()\n",
        "        # if print_output:\n",
        "        #   print(log_msg)\n",
        "\n",
        "      if (step % eval_every_n_steps == 0) or (step == total_steps):\n",
        "        eval_iter = iter(test_dataset)\n",
        "\n",
        "        for m in loss_metrics + acc_metrics:\n",
        "          if m.startswith(\"eval\"):\n",
        "            metric_list[m].reset_state()\n",
        "\n",
        "        for _ in range(steps_per_epoch_test):\n",
        "          eval_step(eval_iter)\n",
        "        if print_output:\n",
        "          print(\"Steps: {}, Test accuracy: {}\".format(\n",
        "              step, metric_list[\"eval_accuracy\"].result()))\n",
        "        steps.append(step)\n",
        "        eval_accuracies.append(metric_list[\"eval_accuracy\"].result())\n",
        "\n",
        "    return steps, eval_accuracies, model"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "background_save": true,
          "base_uri": "https://localhost:8080/",
          "height": 176,
          "referenced_widgets": [
            "3b9779a6c3ae47f2910c8f99194e926a",
            "6aa020e1beab49a7987340f81ffd729d",
            "fde27da0a32141fd94503a8d95bb8a0e",
            "20be5f98f8c64a46afffa237f89d1c0c",
            "1a6de75a8abb41f4820c3045bc486305",
            "bbcd1f390b954afdb330062c7c1544a8",
            "044b54e3ace24f53a42b49928de753e1",
            "480f2cc6035d40b2b26cc55476affdb7",
            "42e966b59cc548d5998761cd11332cf0",
            "e605013095724402b16292b268bcfd5e",
            "23884966533a493893cbd5c2cf074b16"
          ]
        },
        "id": "I3an389J6gCq",
        "outputId": "e3ddd4dc-2224-4610-b82e-8d7743b4a672"
      },
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "Downloading and preparing dataset 11.06 MiB (download: 11.06 MiB, generated: 21.00 MiB, total: 32.06 MiB) to /root/tensorflow_datasets/mnist/3.0.1...\n"
          ]
        },
        {
          "data": {
            "application/vnd.jupyter.widget-view+json": {
              "model_id": "3b9779a6c3ae47f2910c8f99194e926a",
              "version_major": 2,
              "version_minor": 0
            },
            "text/plain": [
              "Dl Completed...:   0%|          | 0/5 [00:00<?, ? file/s]"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "Dataset mnist downloaded and prepared to /root/tensorflow_datasets/mnist/3.0.1. Subsequent calls will reuse this data.\n"
          ]
        },
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "WARNING:tensorflow:You are casting an input of type complex64 to an incompatible dtype float32.  This will discard the imaginary part and may not be what you intended.\n",
            "WARNING:tensorflow:You are casting an input of type complex64 to an incompatible dtype float32.  This will discard the imaginary part and may not be what you intended.\n"
          ]
        },
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "Steps: 1000, Test accuracy: 0.11349999904632568\n",
            "Steps: 2000, Test accuracy: 0.11349999904632568\n"
          ]
        },
        {
          "ename": "InvalidArgumentError",
          "evalue": "Graph execution error:\n\nDetected at node Eig defined at (most recent call last):\n  File \"/usr/lib/python3.10/threading.py\", line 973, in _bootstrap\n\n  File \"/usr/lib/python3.10/threading.py\", line 1016, in _bootstrap_inner\n\n  File \"/tmp/__autograph_generated_filesr52idbc.py\", line 25, in step_fn\n\n  File \"<ipython-input-6-eb4a137876b2>\", line 112, in get_nlfda_loss\n\nEigen decomposition was not successful. The input might not be valid.\n\t [[{{node Eig}}]] [Op:__inference_train_step_7053]",
          "output_type": "error",
          "traceback": [
            "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
            "\u001b[0;31mInvalidArgumentError\u001b[0m                      Traceback (most recent call last)",
            "\u001b[0;32m<ipython-input-9-082b8f4bdf3e>\u001b[0m in \u001b[0;36m<cell line: 6>\u001b[0;34m()\u001b[0m\n\u001b[1;32m      6\u001b[0m \u001b[0;32mfor\u001b[0m \u001b[0mmargin\u001b[0m \u001b[0;32min\u001b[0m \u001b[0;34m[\u001b[0m\u001b[0;36m0.06\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m:\u001b[0m \u001b[0;31m#[0.005,0.01,0.05]:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m      7\u001b[0m   \u001b[0;32mfor\u001b[0m \u001b[0mbits\u001b[0m \u001b[0;32min\u001b[0m \u001b[0;34m[\u001b[0m\u001b[0;36m1\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;31m#[0, 5, 10, 15, 20]:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m----> 8\u001b[0;31m     steps, accuracies, model = train_and_eval(\n\u001b[0m\u001b[1;32m      9\u001b[0m         \u001b[0mbatch_size\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0;36m128\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     10\u001b[0m         \u001b[0mwidth_multiplier\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0;36m1\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
            "\u001b[0;32m<ipython-input-8-f60bb1875e58>\u001b[0m in \u001b[0;36mtrain_and_eval\u001b[0;34m(batch_size, width_multiplier, extra_channel_bits, nt_xent_temp, margin, learning_rate, epochs, log_summary_every_n_steps, eval_every_n_steps, print_output)\u001b[0m\n\u001b[1;32m    113\u001b[0m     \u001b[0meval_accuracies\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0;34m[\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    114\u001b[0m     \u001b[0;32mwhile\u001b[0m \u001b[0mglobal_step\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mnumpy\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m \u001b[0;34m<=\u001b[0m \u001b[0mtotal_steps\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 115\u001b[0;31m       \u001b[0mtrain_step\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mtrain_iter\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    116\u001b[0m       \u001b[0mstep\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mglobal_step\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mnumpy\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    117\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n",
            "\u001b[0;32m/usr/local/lib/python3.10/dist-packages/tensorflow/python/util/traceback_utils.py\u001b[0m in \u001b[0;36merror_handler\u001b[0;34m(*args, **kwargs)\u001b[0m\n\u001b[1;32m    151\u001b[0m     \u001b[0;32mexcept\u001b[0m \u001b[0mException\u001b[0m \u001b[0;32mas\u001b[0m \u001b[0me\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    152\u001b[0m       \u001b[0mfiltered_tb\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0m_process_traceback_frames\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0me\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0m__traceback__\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 153\u001b[0;31m       \u001b[0;32mraise\u001b[0m \u001b[0me\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mwith_traceback\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mfiltered_tb\u001b[0m\u001b[0;34m)\u001b[0m \u001b[0;32mfrom\u001b[0m \u001b[0;32mNone\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    154\u001b[0m     \u001b[0;32mfinally\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    155\u001b[0m       \u001b[0;32mdel\u001b[0m \u001b[0mfiltered_tb\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
            "\u001b[0;32m/usr/local/lib/python3.10/dist-packages/tensorflow/python/eager/execute.py\u001b[0m in \u001b[0;36mquick_execute\u001b[0;34m(op_name, num_outputs, inputs, attrs, ctx, name)\u001b[0m\n\u001b[1;32m     51\u001b[0m   \u001b[0;32mtry\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     52\u001b[0m     \u001b[0mctx\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mensure_initialized\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m---> 53\u001b[0;31m     tensors = pywrap_tfe.TFE_Py_Execute(ctx._handle, device_name, op_name,\n\u001b[0m\u001b[1;32m     54\u001b[0m                                         inputs, attrs, num_outputs)\n\u001b[1;32m     55\u001b[0m   \u001b[0;32mexcept\u001b[0m \u001b[0mcore\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0m_NotOkStatusException\u001b[0m \u001b[0;32mas\u001b[0m \u001b[0me\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
            "\u001b[0;31mInvalidArgumentError\u001b[0m: Graph execution error:\n\nDetected at node Eig defined at (most recent call last):\n  File \"/usr/lib/python3.10/threading.py\", line 973, in _bootstrap\n\n  File \"/usr/lib/python3.10/threading.py\", line 1016, in _bootstrap_inner\n\n  File \"/tmp/__autograph_generated_filesr52idbc.py\", line 25, in step_fn\n\n  File \"<ipython-input-6-eb4a137876b2>\", line 112, in get_nlfda_loss\n\nEigen decomposition was not successful. The input might not be valid.\n\t [[{{node Eig}}]] [Op:__inference_train_step_7053]"
          ]
        }
      ],
      "source": [
        "# The actual training. This cell takes a long time to run, especially on a CPU.\n",
        "rows = []\n",
        "cols = ['extra_channel_bits', 'nt_xent_temp', 'eval_accuracy']\n",
        "temp=1\n",
        "\n",
        "for margin in [0.06]: #[0.005,0.01,0.05]:\n",
        "  for bits in [1]:#[0, 5, 10, 15, 20]:\n",
        "    steps, accuracies, model = train_and_eval(\n",
        "        batch_size=128,\n",
        "        width_multiplier=1,\n",
        "        extra_channel_bits=bits,\n",
        "        nt_xent_temp=temp, # [1,5,10]\n",
        "        margin=margin,\n",
        "        learning_rate=0.005,\n",
        "        epochs=10,\n",
        "        eval_every_n_steps=1000,\n",
        "        print_output=True)\n",
        "    rows.append([bits, margin, accuracies[-1].numpy()])\n",
        "    print(\"extra_channel_bits={}, nt_xent_temp={}, eval_accuracy={}\".format(\n",
        "        bits, margin, accuracies[-1]))\n",
        "plot_df = pd.DataFrame.from_records(rows, columns=cols)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "DT5FZD28QZOV",
        "outputId": "a9ee75e0-5dc2-466b-c90f-8b90c41fbf76"
      },
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "training size\n",
            "(1000, 28, 28, 2)\n",
            "noise level 0\n",
            "Average embedding difference: tf.Tensor(0.0, shape=(), dtype=float32)\n",
            "training size\n",
            "(1000, 28, 28, 2)\n",
            "noise level 0.1\n",
            "Average embedding difference: tf.Tensor(0.0, shape=(), dtype=float32)\n",
            "training size\n",
            "(1000, 28, 28, 2)\n",
            "noise level 0.5\n",
            "Average embedding difference: tf.Tensor(3.4793887, shape=(), dtype=float32)\n",
            "training size\n",
            "(1000, 28, 28, 2)\n",
            "noise level 1\n",
            "Average embedding difference: tf.Tensor(6.4514728, shape=(), dtype=float32)\n",
            "training size\n",
            "(1000, 28, 28, 2)\n",
            "noise level 2\n",
            "Average embedding difference: tf.Tensor(8.312645, shape=(), dtype=float32)\n",
            "training size\n",
            "(1000, 28, 28, 2)\n",
            "noise level 5\n",
            "Average embedding difference: tf.Tensor(9.433197, shape=(), dtype=float32)\n",
            "training size\n",
            "(1000, 28, 28, 2)\n",
            "noise level 10\n",
            "Average embedding difference: tf.Tensor(9.813637, shape=(), dtype=float32)\n"
          ]
        }
      ],
      "source": [
        "import random\n",
        "import numpy as np\n",
        "import scipy.ndimage  # Importing scipy for Gaussian filter\n",
        "\n",
        "\n",
        "# Set random seeds for reproducibility\n",
        "random_seed = 123\n",
        "tf.random.set_seed(random_seed)  # Set TensorFlow's random seed\n",
        "np.random.seed(random_seed)  # Set NumPy's random seed\n",
        "random.seed(random_seed)  # Set Python's random seed\n",
        "\n",
        "batch_size=1000\n",
        "extra_channel_bits=1\n",
        "\n",
        "builder = tfds.builder('mnist')\n",
        "builder.download_and_prepare()\n",
        "\n",
        "preprocess_train_fn, preprocess_eval_fn = get_process_fns(extra_channel_bits)\n",
        "train_dataset = builder.as_dataset(split='train', as_supervised=True)\n",
        "train_dataset = train_dataset.repeat().map(preprocess_train_fn)\n",
        "train_dataset = train_dataset.batch(batch_size)\n",
        "\n",
        "test_dataset = builder.as_dataset(split='test', as_supervised=True)\n",
        "test_dataset = test_dataset.map(preprocess_eval_fn)\n",
        "test_dataset = test_dataset.batch(batch_size, drop_remainder=False)\n",
        "train_iter = iter(train_dataset)\n",
        "\n",
        "test_dataset = builder.as_dataset(split='test', as_supervised=True)\n",
        "test_dataset = test_dataset.map(preprocess_eval_fn)\n",
        "test_dataset = test_dataset.batch(batch_size, drop_remainder=False)\n",
        "test_iter = iter(test_dataset)\n",
        "\n",
        "def randomize_binary_channels_with_noise(images, extra_channel_bits, noise_mean=0.0, noise_variance=1.0):\n",
        "    # Extract the extra binary channels (last 'extra_channel_bits' channels)\n",
        "    extra_channels = images[:, :, :, -extra_channel_bits:]\n",
        "\n",
        "    # Add Gaussian noise to the extra channels\n",
        "    noise = tf.random.normal(shape=extra_channels.shape, mean=noise_mean, stddev=noise_variance)\n",
        "    noisy_channels = extra_channels + noise\n",
        "\n",
        "    # Apply threshold to bring the noisy channels back to binary (0 or 1)\n",
        "    randomized_channels = tf.cast(noisy_channels > 0.5, tf.float32)\n",
        "\n",
        "    # Replace the original extra channels with the randomized ones\n",
        "    randomized_images = tf.concat([images[:, :, :, :-extra_channel_bits], randomized_channels], axis=-1)\n",
        "\n",
        "    return randomized_images\n",
        "\n",
        "noise_level=[0, 0.1,0.5,1,2,5,10]\n",
        "\n",
        "for noise in noise_level:\n",
        "\n",
        "    images, labels = test_iter.next()\n",
        "\n",
        "    original_embeddings, z, predictions = model(images, training=False)\n",
        "\n",
        "    print(\"training size\")\n",
        "    print(images.shape)\n",
        "    # print(\"first channel\")\n",
        "    # print(images[1,:,:,0])\n",
        "    # print(\"first extra channel\")\n",
        "    # print(images[1,:,:,1])\n",
        "    # print(\"second extra channel\")\n",
        "    # print(images[1,:,:,2])\n",
        "    # print(\"third extra channel\")\n",
        "    # print(images[1,:,:,3])\n",
        "    # print(\"fourth extra channel\")\n",
        "    # print(images[1,:,:,4])\n",
        "    # print(\"fifth extra channel\")\n",
        "    # print(images[1,:,:,5])\n",
        "\n",
        "\n",
        "    labels_one_hot = tf.one_hot(labels, depth=10)\n",
        "\n",
        "    # images_a, images_b = tf.unstack(images, num=2, axis=1)\n",
        "    # print(images_a.shape)\n",
        "\n",
        "\n",
        "    # Step 2: Randomize added channels (last n channels)\n",
        "    images = tf.cast(images, tf.float32)  # Convert to float\n",
        "\n",
        "    if noise==0:\n",
        "        randomized_images = tf.identity(images)\n",
        "\n",
        "    randomized_images = randomize_binary_channels_with_noise(images, extra_channel_bits, noise_variance=noise)\n",
        "    # print(\"randomized images shape\")\n",
        "    # print(randomized_images.shape)\n",
        "\n",
        "    # Step 3: Get embeddings for images with randomized channels\n",
        "    randomized_embeddings, z, predictions = model(randomized_images, training=False)\n",
        "\n",
        "    # print(\"shape of the embeddings\")\n",
        "    # print(randomized_embeddings[0].shape)\n",
        "\n",
        "    # print(\"shape of the original embeddings\")\n",
        "    # print(original_embeddings[0].shape)\n",
        "\n",
        "    # Step 4: Measure the embedding difference (Euclidean distance)\n",
        "    embedding_diff = tf.norm(original_embeddings - randomized_embeddings, axis=1)\n",
        "    print(\"noise level\", noise)\n",
        "    print(\"Average embedding difference:\", tf.reduce_mean(embedding_diff))\n",
        "\n",
        "\n",
        "\n",
        "\n"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "zyaIxp1GP449"
      },
      "outputs": [],
      "source": []
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 352
        },
        "id": "svvRfGG76zxz",
        "outputId": "da318fd2-71ff-4254-eb90-58822d4b3b3a"
      },
      "outputs": [
        {
          "data": {
            "text/plain": [
              "<Axes: xlabel='extra_channel_bits', ylabel='eval_accuracy'>"
            ]
          },
          "execution_count": 45,
          "metadata": {},
          "output_type": "execute_result"
        },
        {
          "data": {
            "image/png": "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H8/48eO56qqreOqpp6RQijqRQilaLRcXF5588kmmTZuG2WzmyiuvJDc3l7/++gtXV1e8vLxYsmQJcXFxhIWF8dRTTzF+/Hh27dqFh4cHgYGBKIrCTz/9xA033ICjo2ONZ01du3alvLycd999l5tuuqnKRT51MX36dPr06cPDDz/MQw89hMFgYM2aNdx55511uijlcl+/rrp168a9997LuHHjeOutt+jfvz+ZmZnExsbSt29fbrzxxks+poODA+PHj2fOnDnk5eXx2GOPcdddd9XY7BoUFMTmzZs5evQozs7OeHp6MmvWLMLDw+nVqxelpaX89NNP9OjR43Lfrmgl5KpX0aq99NJLvPDCC8TExNCjRw9GjRrFypUrCQoKYuLEicyaNYuwsDAAZs+eja+vLw899BAA7du3Z/bs2Tz77LP4+voyZcqUGl+nX79+zJ07l9dff53evXvz2WefERMTU+ec3bp14/fff2fnzp1EREQQFRXFjz/+iJ1d3f7WvdzXvxRLly5l3LhxPPHEE3Tv3p3Ro0ezdetWOnbsWK/jde3aldtuu40bbriBESNG0LdvX957770at3/yySfR6/X07NkTb29vUlNTMRgMTJ8+nb59+3L11Vej1+v58ssv6/sWRSujqH/vgBBCCCGElZxRCiGEELWQQimEDbj++utxdnau9vHqq69qHa/R9OrVq8b3/dlnn2kdTwhAml6FsAknTpyguLi42nWenp54eno2caKmkZKSUmX4u/P5+vri4uLSxImEuJAUSiGEEKIW0vQqhBBC1EIKpRBCCFELKZRCCCFELaRQCiGEELWQQimEEELUQgqlEEIIUQsplEIIIUQtpFAKIYQQtfh/oNf+AXMOZwEAAAAASUVORK5CYII=",
            "text/plain": [
              "<Figure size 500x500 with 1 Axes>"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        }
      ],
      "source": [
        "plt.figure(figsize=(5, 5))\n",
        "sns.lineplot(\n",
        "    x='extra_channel_bits', y='eval_accuracy', hue='nt_xent_temp',\n",
        "    data=plot_df, palette=sns.color_palette(\"mako_r\", 3))"
      ]
    }
  ],
  "metadata": {
    "accelerator": "TPU",
    "colab": {
      "gpuType": "V28",
      "machine_shape": "hm",
      "provenance": []
    },
    "kernelspec": {
      "display_name": "Python 3",
      "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.10.13"
    },
    "vscode": {
      "interpreter": {
        "hash": "de1f89d7cc53317a9682d88f1fbdbed246fd8feac45ef548d8aa92ddcf105c90"
      }
    },
    "widgets": {
      "application/vnd.jupyter.widget-state+json": {
        "044b54e3ace24f53a42b49928de753e1": {
          "model_module": "@jupyter-widgets/controls",
          "model_module_version": "1.5.0",
          "model_name": "DescriptionStyleModel",
          "state": {
            "_model_module": "@jupyter-widgets/controls",
            "_model_module_version": "1.5.0",
            "_model_name": "DescriptionStyleModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/base",
            "_view_module_version": "1.2.0",
            "_view_name": "StyleView",
            "description_width": ""
          }
        },
        "1a6de75a8abb41f4820c3045bc486305": {
          "model_module": "@jupyter-widgets/base",
          "model_module_version": "1.2.0",
          "model_name": "LayoutModel",
          "state": {
            "_model_module": "@jupyter-widgets/base",
            "_model_module_version": "1.2.0",
            "_model_name": "LayoutModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/base",
            "_view_module_version": "1.2.0",
            "_view_name": "LayoutView",
            "align_content": null,
            "align_items": null,
            "align_self": null,
            "border": null,
            "bottom": null,
            "display": null,
            "flex": null,
            "flex_flow": null,
            "grid_area": null,
            "grid_auto_columns": null,
            "grid_auto_flow": null,
            "grid_auto_rows": null,
            "grid_column": null,
            "grid_gap": null,
            "grid_row": null,
            "grid_template_areas": null,
            "grid_template_columns": null,
            "grid_template_rows": null,
            "height": null,
            "justify_content": null,
            "justify_items": null,
            "left": null,
            "margin": null,
            "max_height": null,
            "max_width": null,
            "min_height": null,
            "min_width": null,
            "object_fit": null,
            "object_position": null,
            "order": null,
            "overflow": null,
            "overflow_x": null,
            "overflow_y": null,
            "padding": null,
            "right": null,
            "top": null,
            "visibility": null,
            "width": null
          }
        },
        "20be5f98f8c64a46afffa237f89d1c0c": {
          "model_module": "@jupyter-widgets/controls",
          "model_module_version": "1.5.0",
          "model_name": "HTMLModel",
          "state": {
            "_dom_classes": [],
            "_model_module": "@jupyter-widgets/controls",
            "_model_module_version": "1.5.0",
            "_model_name": "HTMLModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/controls",
            "_view_module_version": "1.5.0",
            "_view_name": "HTMLView",
            "description": "",
            "description_tooltip": null,
            "layout": "IPY_MODEL_e605013095724402b16292b268bcfd5e",
            "placeholder": "​",
            "style": "IPY_MODEL_23884966533a493893cbd5c2cf074b16",
            "value": " 5/5 [00:00&lt;00:00, 15.22 file/s]"
          }
        },
        "23884966533a493893cbd5c2cf074b16": {
          "model_module": "@jupyter-widgets/controls",
          "model_module_version": "1.5.0",
          "model_name": "DescriptionStyleModel",
          "state": {
            "_model_module": "@jupyter-widgets/controls",
            "_model_module_version": "1.5.0",
            "_model_name": "DescriptionStyleModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/base",
            "_view_module_version": "1.2.0",
            "_view_name": "StyleView",
            "description_width": ""
          }
        },
        "3b9779a6c3ae47f2910c8f99194e926a": {
          "model_module": "@jupyter-widgets/controls",
          "model_module_version": "1.5.0",
          "model_name": "HBoxModel",
          "state": {
            "_dom_classes": [],
            "_model_module": "@jupyter-widgets/controls",
            "_model_module_version": "1.5.0",
            "_model_name": "HBoxModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/controls",
            "_view_module_version": "1.5.0",
            "_view_name": "HBoxView",
            "box_style": "",
            "children": [
              "IPY_MODEL_6aa020e1beab49a7987340f81ffd729d",
              "IPY_MODEL_fde27da0a32141fd94503a8d95bb8a0e",
              "IPY_MODEL_20be5f98f8c64a46afffa237f89d1c0c"
            ],
            "layout": "IPY_MODEL_1a6de75a8abb41f4820c3045bc486305"
          }
        },
        "42e966b59cc548d5998761cd11332cf0": {
          "model_module": "@jupyter-widgets/controls",
          "model_module_version": "1.5.0",
          "model_name": "ProgressStyleModel",
          "state": {
            "_model_module": "@jupyter-widgets/controls",
            "_model_module_version": "1.5.0",
            "_model_name": "ProgressStyleModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/base",
            "_view_module_version": "1.2.0",
            "_view_name": "StyleView",
            "bar_color": null,
            "description_width": ""
          }
        },
        "480f2cc6035d40b2b26cc55476affdb7": {
          "model_module": "@jupyter-widgets/base",
          "model_module_version": "1.2.0",
          "model_name": "LayoutModel",
          "state": {
            "_model_module": "@jupyter-widgets/base",
            "_model_module_version": "1.2.0",
            "_model_name": "LayoutModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/base",
            "_view_module_version": "1.2.0",
            "_view_name": "LayoutView",
            "align_content": null,
            "align_items": null,
            "align_self": null,
            "border": null,
            "bottom": null,
            "display": null,
            "flex": null,
            "flex_flow": null,
            "grid_area": null,
            "grid_auto_columns": null,
            "grid_auto_flow": null,
            "grid_auto_rows": null,
            "grid_column": null,
            "grid_gap": null,
            "grid_row": null,
            "grid_template_areas": null,
            "grid_template_columns": null,
            "grid_template_rows": null,
            "height": null,
            "justify_content": null,
            "justify_items": null,
            "left": null,
            "margin": null,
            "max_height": null,
            "max_width": null,
            "min_height": null,
            "min_width": null,
            "object_fit": null,
            "object_position": null,
            "order": null,
            "overflow": null,
            "overflow_x": null,
            "overflow_y": null,
            "padding": null,
            "right": null,
            "top": null,
            "visibility": null,
            "width": null
          }
        },
        "6aa020e1beab49a7987340f81ffd729d": {
          "model_module": "@jupyter-widgets/controls",
          "model_module_version": "1.5.0",
          "model_name": "HTMLModel",
          "state": {
            "_dom_classes": [],
            "_model_module": "@jupyter-widgets/controls",
            "_model_module_version": "1.5.0",
            "_model_name": "HTMLModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/controls",
            "_view_module_version": "1.5.0",
            "_view_name": "HTMLView",
            "description": "",
            "description_tooltip": null,
            "layout": "IPY_MODEL_bbcd1f390b954afdb330062c7c1544a8",
            "placeholder": "​",
            "style": "IPY_MODEL_044b54e3ace24f53a42b49928de753e1",
            "value": "Dl Completed...: 100%"
          }
        },
        "bbcd1f390b954afdb330062c7c1544a8": {
          "model_module": "@jupyter-widgets/base",
          "model_module_version": "1.2.0",
          "model_name": "LayoutModel",
          "state": {
            "_model_module": "@jupyter-widgets/base",
            "_model_module_version": "1.2.0",
            "_model_name": "LayoutModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/base",
            "_view_module_version": "1.2.0",
            "_view_name": "LayoutView",
            "align_content": null,
            "align_items": null,
            "align_self": null,
            "border": null,
            "bottom": null,
            "display": null,
            "flex": null,
            "flex_flow": null,
            "grid_area": null,
            "grid_auto_columns": null,
            "grid_auto_flow": null,
            "grid_auto_rows": null,
            "grid_column": null,
            "grid_gap": null,
            "grid_row": null,
            "grid_template_areas": null,
            "grid_template_columns": null,
            "grid_template_rows": null,
            "height": null,
            "justify_content": null,
            "justify_items": null,
            "left": null,
            "margin": null,
            "max_height": null,
            "max_width": null,
            "min_height": null,
            "min_width": null,
            "object_fit": null,
            "object_position": null,
            "order": null,
            "overflow": null,
            "overflow_x": null,
            "overflow_y": null,
            "padding": null,
            "right": null,
            "top": null,
            "visibility": null,
            "width": null
          }
        },
        "e605013095724402b16292b268bcfd5e": {
          "model_module": "@jupyter-widgets/base",
          "model_module_version": "1.2.0",
          "model_name": "LayoutModel",
          "state": {
            "_model_module": "@jupyter-widgets/base",
            "_model_module_version": "1.2.0",
            "_model_name": "LayoutModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/base",
            "_view_module_version": "1.2.0",
            "_view_name": "LayoutView",
            "align_content": null,
            "align_items": null,
            "align_self": null,
            "border": null,
            "bottom": null,
            "display": null,
            "flex": null,
            "flex_flow": null,
            "grid_area": null,
            "grid_auto_columns": null,
            "grid_auto_flow": null,
            "grid_auto_rows": null,
            "grid_column": null,
            "grid_gap": null,
            "grid_row": null,
            "grid_template_areas": null,
            "grid_template_columns": null,
            "grid_template_rows": null,
            "height": null,
            "justify_content": null,
            "justify_items": null,
            "left": null,
            "margin": null,
            "max_height": null,
            "max_width": null,
            "min_height": null,
            "min_width": null,
            "object_fit": null,
            "object_position": null,
            "order": null,
            "overflow": null,
            "overflow_x": null,
            "overflow_y": null,
            "padding": null,
            "right": null,
            "top": null,
            "visibility": null,
            "width": null
          }
        },
        "fde27da0a32141fd94503a8d95bb8a0e": {
          "model_module": "@jupyter-widgets/controls",
          "model_module_version": "1.5.0",
          "model_name": "FloatProgressModel",
          "state": {
            "_dom_classes": [],
            "_model_module": "@jupyter-widgets/controls",
            "_model_module_version": "1.5.0",
            "_model_name": "FloatProgressModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/controls",
            "_view_module_version": "1.5.0",
            "_view_name": "ProgressView",
            "bar_style": "success",
            "description": "",
            "description_tooltip": null,
            "layout": "IPY_MODEL_480f2cc6035d40b2b26cc55476affdb7",
            "max": 5,
            "min": 0,
            "orientation": "horizontal",
            "style": "IPY_MODEL_42e966b59cc548d5998761cd11332cf0",
            "value": 5
          }
        }
      }
    }
  },
  "nbformat": 4,
  "nbformat_minor": 0
}