{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "import jax\n",
    "from typing import Any, Callable, Sequence, Optional\n",
    "from jax import numpy as jnp\n",
    "import flax\n",
    "from flax import linen as nn\n",
    "from functools import partial\n",
    "import numpy as np\n",
    "import tensorflow as tf\n",
    "import tensorflow_datasets as tfds\n",
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "##################################\n",
    "# Conic Activation\n",
    "##################################\n",
    "import functools\n",
    "\n",
    "@functools.partial(jax.jit, static_argnames=['channel_axis','variant','eps','num_groups','share_axis'])\n",
    "def colu(input: jnp.ndarray, \n",
    "         channel_axis: int = -1, \n",
    "         variant: str = \"soft\", \n",
    "         eps: float = 1e-7, \n",
    "         num_groups: int = 1, \n",
    "         share_axis: bool = False\n",
    "         ):\n",
    "    \"\"\"project the input x onto the axes dimension\"\"\"\n",
    "    \"\"\"output dimension = S = axes + cone sections = [len=(G or 1)] + G * [len=(S-1)]\"\"\"\n",
    "    if num_groups == 0: # trivial case\n",
    "        return input\n",
    "    num_channels = input.shape[channel_axis]\n",
    "    if (share_axis and num_groups == num_channels - 1) or (not share_axis and num_groups * 2 == num_channels): # pointwise case\n",
    "        return nn.silu(input) if variant == \"soft\" else nn.relu(input)\n",
    "    group_size = (num_channels - 1) // num_groups + 1 if share_axis else num_channels // num_groups\n",
    "        \n",
    "    # y = axes, x = cone sections\n",
    "    if share_axis:\n",
    "        assert (num_channels - 1) % num_groups == 0, \"Channel size must be a multiple of number of cones plus one\"\n",
    "        y, x = jnp.split(input, [1], axis=channel_axis)\n",
    "    else:\n",
    "        assert num_channels % num_groups == 0, \"Channel size must be a multiple of number of cones\"\n",
    "        y, x = jnp.split(input, [num_groups], axis=channel_axis)\n",
    "        group_size = num_channels // num_groups # S = C / G\n",
    "\n",
    "    assert channel_axis < 0, \"channel_axis must be negative\" # Comply with broadcasting on first dimensions\n",
    "    x_old_shape = x.shape\n",
    "    y_old_shape = y.shape\n",
    "    x_shape = x.shape[:channel_axis] + (num_groups, group_size - 1) # NG(S-1)\n",
    "    if share_axis:\n",
    "        y_shape = y.shape[:channel_axis] + (1, 1) # N11\n",
    "    else: \n",
    "        y_shape = y.shape[:channel_axis] + (num_groups, 1) # NG1\n",
    "    if channel_axis < -1:\n",
    "        x_shape += x.shape[(channel_axis+1):] # NGSHW if channel_axis = -3\n",
    "        y_shape += y.shape[(channel_axis+1):] # NG1HW\n",
    "    x = x.reshape(x_shape)\n",
    "    y = y.reshape(y_shape)\n",
    "    xn = jnp.linalg.norm(x,axis=channel_axis,keepdims=True) * nn.relu(y) # NG1HW\n",
    "    y = nn.relu(y)\n",
    "    mask = y / (xn + eps) # NG1HW\n",
    "    if variant == \"sqrt\":\n",
    "        mask = jnp.sqrt(mask)\n",
    "    elif variant == \"log\":\n",
    "        mask = jnp.log(jnp.max(mask,0)+1)\n",
    "    elif variant == \"soft\":\n",
    "        mask = nn.sigmoid(mask - .5)\n",
    "    elif variant == \"hard\":\n",
    "        mask = mask.clip(0,1)\n",
    "    else:\n",
    "        raise NotImplementedError(\"variant must be soft or hard.\")\n",
    "\n",
    "    x = mask * x # NGSHW\n",
    "    x = x.reshape(x_old_shape)\n",
    "    y = y.reshape(y_old_shape)\n",
    "    output = jnp.concatenate([y,x],axis=channel_axis)\n",
    "\n",
    "    return output\n",
    "\n",
    "@functools.partial(jax.jit, static_argnames=['scaling','eps'])\n",
    "def rcolu_(x, scaling=\"constant\",eps=1e-8):\n",
    "    \"\"\"x = w + v, v || e\"\"\"\n",
    "    C = x.shape[-1]\n",
    "    # e = jnp.ones(C) / jnp.sqrt(C)\n",
    "    vn = jnp.sum(x,axis=-1,keepdims=True) / jnp.sqrt(C) # dot(x, e)\n",
    "    v = jnp.repeat(vn,C,axis=-1) / jnp.sqrt(C) # outer(v, e)\n",
    "    w = x - v\n",
    "    wn = jnp.linalg.norm(w, axis=-1, keepdims=True)\n",
    "    m = jnp.maximum(vn, 0.) / (wn + eps)\n",
    "    m = jnp.minimum(m, 1.) \n",
    "    w_ = w * m # project onto cone\n",
    "    x = v + w_\n",
    "    \n",
    "    return x\n",
    "\n",
    "@functools.partial(jax.jit, static_argnames=['dim','num_groups','axis','scaling','eps'])\n",
    "def rcolu(x,\n",
    "          dim=4,\n",
    "          num_groups=None,\n",
    "          scaling='constant',\n",
    "          axis=-1,\n",
    "          eps=1e-7\n",
    "          ):\n",
    "    \"\"\"dim=S, num_groups=S\"\"\"\n",
    "    if len(x.shape) == 0:\n",
    "        return x # edge case\n",
    "    assert (dim is not None) ^ (num_groups is not None)\n",
    "    x = jnp.moveaxis(x, axis, -1)\n",
    "    shape = x.shape\n",
    "    if dim is None:\n",
    "        assert shape[-1] % num_groups == 0\n",
    "        dim = shape[-1] // num_groups\n",
    "    if num_groups is None:\n",
    "        assert shape[-1] % dim == 0\n",
    "        num_groups = shape[-1] // dim\n",
    "    new_shape = x.shape[:-1] + (num_groups, dim)\n",
    "    x = x.reshape(new_shape)\n",
    "    x = rcolu_(x,scaling,eps)\n",
    "    x = x.reshape(shape)\n",
    "    x = jnp.moveaxis(x, -1, axis)\n",
    "    return x\n",
    "\n",
    "# # some test\n",
    "# x = jnp.zeros(6).at[0].set(1)\n",
    "# y = rcolu(x,dim=3)\n",
    "# y.shape\n",
    "# # some assertion \n",
    "# y1 = jnp.sum(y,axis=-1,keepdims=True) / jnp.sqrt(3) # dot(y, e)\n",
    "# jnp.linalg.norm(y) / y1\n",
    "\n",
    "##################################\n",
    "# Hyperparameters\n",
    "##################################\n",
    "dim = 3\n",
    "C = 3"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Rotate Color Hue (Real World Example)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "# vae\n",
    "# import jax\n",
    "# import jax.numpy as jnp\n",
    "# import numpy as np\n",
    "\n",
    "# import flax.linen as nn\n",
    "# import optax\n",
    "# from flax.training import train_state\n",
    "# from flax.training.common_utils import shard, shard_prng_key\n",
    "# import tensorflow as tf\n",
    "# import tensorflow_datasets as tfds\n",
    "# from tqdm import tqdm\n",
    "# # from tensorboardX import SummaryWriter\n",
    "# import os\n",
    "# import logging\n",
    "\n",
    "# tf.get_logger().setLevel(logging.ERROR)\n",
    "# os.environ['TF_CPP_MIN_LOG_LEVEL'] = '3'  # 0 = all messages are logged, 1 = INFO messages are not printed, 2 = INFO and WARNING messages are not printed, 3 = INFO, WARNING, and ERROR messages are not printed\n",
    "\n",
    "# # Define UNet VAE architecture\n",
    "# class Encoder(nn.Module):\n",
    "#     latent_dim: int\n",
    "\n",
    "#     @nn.compact\n",
    "#     def __call__(self, x):\n",
    "#         # First convolutional layer\n",
    "#         x = nn.Conv(features=128, kernel_size=(3, 3), strides=(2, 2))(x)\n",
    "#         x = nn.LayerNorm()(x)\n",
    "#         x = nn.relu(x)\n",
    "#         x = nn.Conv(features=32, kernel_size=(3, 3), strides=(2, 2))(x)\n",
    "\n",
    "#         z_mean, z_log_var = jnp.split(x, [16], -1)\n",
    "        \n",
    "#         return z_mean, z_log_var\n",
    "\n",
    "# class Decoder(nn.Module):\n",
    "#     latent_dim: int\n",
    "\n",
    "#     @nn.compact\n",
    "#     def __call__(self, z):\n",
    "#         x = nn.ConvTranspose(features=128, kernel_size=(3, 3), strides=(2, 2))(z)\n",
    "#         x = nn.LayerNorm()(x)\n",
    "#         x = nn.relu(x)\n",
    "        \n",
    "#         # Second deconvolutional layer\n",
    "#         x = nn.ConvTranspose(features=3, kernel_size=(3, 3), strides=(2, 2))(x)\n",
    "        \n",
    "#         return x\n",
    "\n",
    "# class UNetVAE(nn.Module):\n",
    "#     latent_dim: int = 32\n",
    "\n",
    "#     def setup(self):\n",
    "#         self.encoder = Encoder(latent_dim=self.latent_dim)\n",
    "#         self.decoder = Decoder(latent_dim=self.latent_dim)\n",
    "\n",
    "#     def __call__(self, x):\n",
    "#         z_mean, z_log_var = self.encoder(x)\n",
    "#         epsilon = jax.random.normal(self.make_rng('normal'), z_mean.shape)\n",
    "#         z = z_mean + jnp.exp(0.5 * z_log_var) * epsilon\n",
    "#         x_recon = self.decoder(z)\n",
    "#         return x_recon, z_mean, z_log_var\n",
    "\n",
    "# # Loss function\n",
    "# def loss_fn(params, apply_fn, batch, rng):\n",
    "#     x_recon, z_mean, z_log_var = apply_fn({'params': params}, batch, rngs={'normal': rng})\n",
    "#     recon_loss = jnp.mean(jnp.sum((x_recon - batch) ** 2, axis=(1, 2, 3)))\n",
    "#     kl_loss = -0.5 * jnp.mean(jnp.sum(1 + z_log_var - jnp.square(z_mean) - jnp.exp(z_log_var), axis=-1))\n",
    "#     return recon_loss + kl_loss, recon_loss, kl_loss\n",
    "\n",
    "# # Training step\n",
    "# @jax.pmap\n",
    "# def train_step(state, batch, rng):\n",
    "#     def grad_fn(params):\n",
    "#         loss, recon_loss, kl_loss = loss_fn(params, state.apply_fn, batch, rng)\n",
    "#         return loss, (loss, recon_loss, kl_loss)\n",
    "\n",
    "#     grad_fn = jax.value_and_grad(grad_fn, has_aux=True)\n",
    "#     (_, (loss, recon_loss, kl_loss)), grads = grad_fn(state.params)\n",
    "#     state = state.apply_gradients(grads=grads)\n",
    "#     return state, loss, recon_loss, kl_loss\n",
    "\n",
    "\n",
    "# # Prepare dataset for Fashion MNIST with coloring\n",
    "# def preprocess_fn(data):\n",
    "#     image = tf.image.convert_image_dtype(data['image'], tf.float32)\n",
    "#     image = tf.image.resize(image, [64, 64])  # Resizing to 64x64 if needed\n",
    "#     image = tf.image.grayscale_to_rgb(image)  # Convert grayscale to RGB\n",
    "#     return image\n",
    "\n",
    "# def load_dataset(split, batch_size):\n",
    "#     ds = tfds.load('fashion_mnist', split=split, shuffle_files=True)\n",
    "#     ds = ds.map(preprocess_fn, num_parallel_calls=tf.data.AUTOTUNE)\n",
    "#     ds = ds.batch(batch_size)\n",
    "#     ds = ds.prefetch(tf.data.AUTOTUNE)\n",
    "#     return ds\n",
    "\n",
    "# # Initialize model and optimizer\n",
    "# def create_train_state(rng, learning_rate, latent_dim):\n",
    "#     model = UNetVAE(latent_dim=latent_dim)\n",
    "#     params = model.init(rng, jnp.ones([1, 64, 64, 3]))['params']\n",
    "#     tx = optax.adam(learning_rate)\n",
    "#     return train_state.TrainState.create(apply_fn=model.apply, params=params, tx=tx)\n",
    "\n",
    "# # Main training loop\n",
    "# def train_model(epochs, batch_size, learning_rate, latent_dim):\n",
    "#     rng = jax.random.PRNGKey(0)\n",
    "#     rng, init_rng = jax.random.split(rng)\n",
    "\n",
    "#     state = create_train_state(init_rng, learning_rate, latent_dim)\n",
    "#     state = jax.device_put_replicated(state, jax.local_devices())\n",
    "\n",
    "#     train_ds = load_dataset('train', batch_size)\n",
    "#     val_ds = load_dataset('test', batch_size)\n",
    "\n",
    "#     # Create a SummaryWriter to log data for Tensorboard\n",
    "#     # writer = SummaryWriter(logdir=\"logs/UNetVAE\")\n",
    "\n",
    "#     pbar = tqdm(range(epochs), total=len(train_ds))\n",
    "#     global_step = 0\n",
    "#     for epoch in pbar:\n",
    "#         # Training loop\n",
    "#         train_iter = iter(train_ds)\n",
    "#         val_iter = iter(val_ds)\n",
    "#         for batch in train_iter:\n",
    "#             rng, step_rng = jax.random.split(rng)\n",
    "#             shard_rngs = shard_prng_key(step_rng)\n",
    "#             batch = jax.numpy.array(batch)\n",
    "#             if batch.shape[0] != batch_size:\n",
    "#                 continue\n",
    "#             colors = sample_points_on_circle(rng, batch_size)[:,None,None,:]\n",
    "#             batch = batch * colors\n",
    "#             batch = shard(batch)\n",
    "#             state, loss, recon_loss, kl_loss = train_step(state, batch, shard_rngs)\n",
    "#             global_step += 1\n",
    "            \n",
    "#         pbar.set_description(f'loss={jax.device_get(loss).mean()}')\n",
    "\n",
    "#         pbar.close()\n",
    "\n",
    "#         # Validation loop\n",
    "#         val_losses = []\n",
    "#         for val_batch in val_iter:\n",
    "#             val_batch = jax.numpy.array(val_batch)\n",
    "#             if val_batch.shape[0] % jax.device_count() != 0:\n",
    "#                 continue\n",
    "#             val_batch = shard(val_batch)\n",
    "#             _, val_loss, _, _ = train_step(state, val_batch, shard_rngs)\n",
    "#             val_losses.append(jax.device_get(val_loss).mean())\n",
    "        \n",
    "#         avg_val_loss = np.mean(val_losses)\n",
    "#         # print(f\"Validation Loss after epoch {epoch + 1}: {avg_val_loss}\")\n",
    "\n",
    "#     return jax.device_get(flax.jax_utils.unreplicate(state.params))\n",
    "\n",
    "# # Set parameters and train\n",
    "# params = train_model(epochs=100, batch_size=512, learning_rate=1e-4, latent_dim=32)\n",
    "# jax.tree.map(lambda x:x.shape,params)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "2024-11-27 15:28:43.385771: I tensorflow/core/kernels/data/tf_record_dataset_op.cc:376] The default buffer size is 262144, which is overridden by the user specified `buffer_size` of 8388608\n",
      "2024-11-27 15:28:43.391437: W tensorflow/core/kernels/data/cache_dataset_ops.cc:914] The calling iterator did not fully read the dataset being cached. In order to avoid unexpected truncation of the dataset, the partially cached contents of the dataset  will be discarded. This can happen if you have an input pipeline similar to `dataset.cache().take(k).repeat()`. You should use `dataset.take(k).cache().repeat()` instead.\n",
      "2024-11-27 15:28:43.391843: I tensorflow/core/framework/local_rendezvous.cc:405] Local rendezvous is aborting with status: OUT_OF_RANGE: End of sequence\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 1000x400 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(32, 32, 32, 3) (32,)\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "2024-11-27 15:28:43.616075: W tensorflow/core/kernels/data/cache_dataset_ops.cc:914] The calling iterator did not fully read the dataset being cached. In order to avoid unexpected truncation of the dataset, the partially cached contents of the dataset  will be discarded. This can happen if you have an input pipeline similar to `dataset.cache().take(k).repeat()`. You should use `dataset.take(k).cache().repeat()` instead.\n"
     ]
    }
   ],
   "source": [
    "# color rotation\n",
    "\n",
    "# Load dataset\n",
    "dataset = tfds.load('cifar10', split='train')\n",
    "\n",
    "# Define hue rotation function\n",
    "def rotate_hue(example, delta):\n",
    "    example['image'] = tf.cast(example['image'], tf.float32)\n",
    "    example['image'] = (example['image'] + 0.5) / 255.0 #* 2.0 - 1.0  # Normalize to [-1, 1]\n",
    "    example['image'] = tf.clip_by_value(example['image'], 0.0, 1.0)\n",
    "    rotate_fn = lambda x, d: tf.image.adjust_hue(x, d)\n",
    "    example['rotated_image'] = tf.concat([rotate_fn(example['image'][:16],delta), rotate_fn(example['image'][16:],-delta)], axis=0)\n",
    "    example['rotated_image'] = tf.concat([rotate_fn(example['rotated_image'][:,:16],.5*delta), rotate_fn(example['rotated_image'][:,16:],-.5*delta)], axis=1)\n",
    "    return example\n",
    "\n",
    "# Apply hue rotation\n",
    "delta = .33  # Adjust this value as needed\n",
    "dataset = dataset.map(lambda example: rotate_hue(example, delta),\n",
    "                      num_parallel_calls=tf.data.experimental.AUTOTUNE)\n",
    "\n",
    "# Optional: Visualize the results\n",
    "def show_examples(dataset, num_examples):\n",
    "#     plt.figure(figsize=(10, 4))\n",
    "#     for i, example in enumerate(dataset.take(num_examples)):\n",
    "#         plt.subplot(2, num_examples, i+1)\n",
    "#         plt.imshow(example['image'])\n",
    "#         plt.axis('off')\n",
    "#         plt.subplot(2, num_examples, num_examples+i+1)\n",
    "#         plt.imshow(example['rotated_image'])\n",
    "#         plt.axis('off')\n",
    "#     plt.show()\n",
    "\n",
    "    original_images = []\n",
    "    rotated_images = []\n",
    "\n",
    "    # Iterate through the dataset and collect the required number of examples\n",
    "    for example in dataset.take(num_examples):\n",
    "        # Convert tensors to NumPy arrays\n",
    "        original = example['image'].numpy()\n",
    "        rotated = example['rotated_image'].numpy()\n",
    "        \n",
    "        original_images.append(original)\n",
    "        rotated_images.append(rotated)\n",
    "\n",
    "    # Concatenate original images horizontally\n",
    "    row_original = np.hstack(original_images)\n",
    "\n",
    "    # Concatenate rotated images horizontally\n",
    "    row_rotated = np.hstack(rotated_images)\n",
    "\n",
    "    # Stack the two rows vertically\n",
    "    combined_image = np.vstack((row_original, row_rotated))\n",
    "\n",
    "    # Display the combined image\n",
    "    plt.figure(figsize=(num_examples * 2, 4))  # Adjust width based on num_examples\n",
    "    plt.imshow(combined_image)\n",
    "    plt.axis('off')  # Hide axis\n",
    "    plt.show()\n",
    "\n",
    "show_examples(dataset, 5)\n",
    "\n",
    "# example for training\n",
    "batch_size = 32\n",
    "dataset = dataset.shuffle(buffer_size=1024)\n",
    "dataset = dataset.batch(batch_size)\n",
    "dataset = dataset.prefetch(buffer_size=tf.data.experimental.AUTOTUNE)\n",
    "for example in dataset:\n",
    "    print(example['image'].shape, example['label'].shape)\n",
    "    break\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Train"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "ResNet(\n",
       "    # attributes\n",
       "    filter_list = [16, 32, 64]\n",
       "    N = 3\n",
       "    num_classes = 10\n",
       "    dtype = float32\n",
       "    kernel_init = init\n",
       ")"
      ]
     },
     "execution_count": 5,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "############# Models #############\n",
    "\n",
    "ModuleDef = Any\n",
    "dtypedef = Any\n",
    "ACT = rcolu\n",
    "\n",
    "class ResidualBlock(nn.Module):\n",
    "    # Define collection of datafields here\n",
    "    in_channels: int\n",
    "\n",
    "    # For batchnorm, you can pass it as a ModuleDef\n",
    "    norm: ModuleDef\n",
    "\n",
    "    # dtype for fp16/32 training\n",
    "    dtype: dtypedef = jnp.float32\n",
    "\n",
    "    # define init for conv layers\n",
    "    kernel_init: Callable = nn.initializers.kaiming_normal()\n",
    "\n",
    "    @nn.compact\n",
    "    def __call__(self, x):\n",
    "        residual = x\n",
    "\n",
    "        x = nn.Conv(\n",
    "            kernel_size=(3, 3),\n",
    "            strides=1,\n",
    "            features=self.in_channels,\n",
    "            padding=\"SAME\",\n",
    "            use_bias=False,\n",
    "            kernel_init=self.kernel_init,\n",
    "            dtype=self.dtype,\n",
    "        )(x) \n",
    "        x = self.norm()(x)\n",
    "        x = ACT(x)\n",
    "        x = nn.Conv(\n",
    "            kernel_size=(3, 3),\n",
    "            strides=1,\n",
    "            features=self.in_channels,\n",
    "            padding=\"SAME\",\n",
    "            use_bias=False,\n",
    "            kernel_init=self.kernel_init,\n",
    "            dtype=self.dtype,\n",
    "        )(x) \n",
    "        x = self.norm()(x)\n",
    "\n",
    "        x = x + residual\n",
    "\n",
    "        return ACT(x)\n",
    "\n",
    "\n",
    "class DownSampleResidualBlock(nn.Module):\n",
    "    # Define collection of datafields here\n",
    "    in_channels: int\n",
    "    out_channels: int\n",
    "\n",
    "    # For batchnorm, you can pass it as a ModuleDef\n",
    "    norm: ModuleDef\n",
    "\n",
    "    # dtype for fp16/32 training\n",
    "    dtype: dtypedef = jnp.float32\n",
    "\n",
    "    # define init for conv layers\n",
    "    kernel_init: Callable = nn.initializers.kaiming_normal()\n",
    "\n",
    "    @nn.compact\n",
    "    def __call__(self, x):\n",
    "        residual = x\n",
    "\n",
    "        x = nn.Conv(\n",
    "            kernel_size=(3, 3),\n",
    "            strides=1,\n",
    "            features=self.in_channels,\n",
    "            padding=\"SAME\",\n",
    "            use_bias=False,\n",
    "            kernel_init=self.kernel_init,\n",
    "            dtype=self.dtype,\n",
    "        )(x) \n",
    "        x = self.norm()(x)\n",
    "        x = ACT(x)\n",
    "        x = nn.Conv(\n",
    "            kernel_size=(3, 3),\n",
    "            strides=(2, 2),\n",
    "            features=self.out_channels,\n",
    "            padding=((1, 1), (1, 1)),\n",
    "            use_bias=False,\n",
    "            kernel_init=self.kernel_init,\n",
    "            dtype=self.dtype,\n",
    "        )(x) \n",
    "        x = self.norm()(x)\n",
    "\n",
    "        x = x + self.pad_identity(residual)\n",
    "\n",
    "        return ACT(x)\n",
    "\n",
    "    @nn.nowrap\n",
    "    def pad_identity(self, x):\n",
    "        # Pad identity connection when downsampling\n",
    "        return jnp.pad(\n",
    "            x[:, ::2, ::2, ::],\n",
    "            ((0, 0), (0, 0), (0, 0), (self.out_channels // 4, self.out_channels // 4)),\n",
    "            \"constant\",\n",
    "        )\n",
    "\n",
    "\n",
    "class ResNet(nn.Module):\n",
    "    # Define collection of datafields here\n",
    "    filter_list: Sequence[int]\n",
    "    N: int\n",
    "    num_classes: int\n",
    "\n",
    "    # dtype for fp16/32 training\n",
    "    dtype: dtypedef = jnp.float32\n",
    "\n",
    "    # define init for conv and linear layers\n",
    "    kernel_init: Callable = nn.initializers.kaiming_normal()\n",
    "\n",
    "    # For train/test differences, want to pass “mode switches” to __call__\n",
    "    @nn.compact\n",
    "    def __call__(self, x, train):\n",
    "\n",
    "        norm = partial(\n",
    "            nn.BatchNorm,\n",
    "            use_running_average=not train,\n",
    "            momentum=0.1,\n",
    "            epsilon=1e-5,\n",
    "            dtype=self.dtype,\n",
    "        )\n",
    "        x = nn.Conv(\n",
    "            kernel_size=(3, 3),\n",
    "            strides=1,\n",
    "            features=self.filter_list[0],\n",
    "            padding=\"SAME\",\n",
    "            use_bias=False,\n",
    "            kernel_init=self.kernel_init,\n",
    "            dtype=self.dtype,\n",
    "        )(x) #if not CONV3D else make_conv('3x3',conv3d=True,features=self.filter_list[0],use_bias=False,dtype=self.dtype,)(x)\n",
    "\n",
    "        x = norm()(x)\n",
    "        x = ACT(x)\n",
    "\n",
    "        # First stage\n",
    "        for _ in range(0, self.N - 1):\n",
    "            x = ResidualBlock(\n",
    "                in_channels=self.filter_list[0], norm=norm, dtype=self.dtype\n",
    "            )(x)\n",
    "\n",
    "        x = DownSampleResidualBlock(\n",
    "            in_channels=self.filter_list[0],\n",
    "            out_channels=self.filter_list[1],\n",
    "            norm=norm,\n",
    "            dtype=self.dtype,\n",
    "        )(x)\n",
    "\n",
    "        # Second stage\n",
    "        for _ in range(0, self.N - 1):\n",
    "            x = ResidualBlock(\n",
    "                in_channels=self.filter_list[1], norm=norm, dtype=self.dtype\n",
    "            )(x)\n",
    "\n",
    "        x = DownSampleResidualBlock(\n",
    "            in_channels=self.filter_list[1],\n",
    "            out_channels=self.filter_list[2],\n",
    "            norm=norm,\n",
    "            dtype=self.dtype,\n",
    "        )(x)\n",
    "\n",
    "        # Third stage\n",
    "        for _ in range(0, self.N):\n",
    "            x = ResidualBlock(\n",
    "                in_channels=self.filter_list[2], norm=norm, dtype=self.dtype\n",
    "            )(x)\n",
    "\n",
    "        # Global pooling\n",
    "        x = jnp.mean(x, axis=(1, 2))\n",
    "\n",
    "        x = x.reshape(x.shape[0], -1)\n",
    "        x = nn.Dense(\n",
    "            features=self.num_classes, kernel_init=self.kernel_init, dtype=self.dtype\n",
    "        )(x)\n",
    "\n",
    "        return x\n",
    "\n",
    "\n",
    "def _resnet(layers, N, dtype=jnp.float32, num_classes=10):\n",
    "    model = ResNet(filter_list=layers, N=N, dtype=dtype, num_classes=num_classes)\n",
    "    return model\n",
    "\n",
    "\n",
    "def ResNet20(\n",
    "    dtype=jnp.float32,\n",
    "):\n",
    "    return _resnet(layers=[16, 32, 64], N=3, dtype=dtype, num_classes=10)\n",
    "\n",
    "ResNet20()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "#### Training with COLU Activation ####\n",
      "-- Run 1/1 --\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "f572416bfcdf4d3880906c86fae4c683",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/10000 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "#### Training with RELU Activation ####\n",
      "-- Run 1/1 --\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "50d2f4d635c44f99979757c9967d81bb",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/10000 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 700x400 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import jax\n",
    "import jax.numpy as jnp\n",
    "import flax.linen as nn\n",
    "import optax\n",
    "from flax.training import train_state\n",
    "import tensorflow_datasets as tfds\n",
    "import tensorflow as tf\n",
    "from tqdm.auto import tqdm\n",
    "import random\n",
    "from typing import Any\n",
    "\n",
    "class TrainState(train_state.TrainState):\n",
    "    batch_stats: Any = None\n",
    "\n",
    "def rotate_image(image):\n",
    "\n",
    "    # Generate a random rotation angle between 0 and 2π\n",
    "    angle = tf.random.uniform((), minval=0, maxval=2 * np.pi)\n",
    "    \n",
    "    # Define rotation axis (1,1,1) normalized\n",
    "    axis = tf.constant([1.0, 1.0, 1.0], dtype=tf.float32)\n",
    "    axis = axis / tf.norm(axis)\n",
    "    \n",
    "    # Compute rotation matrix using Rodrigues' rotation formula\n",
    "    cos_angle = tf.cos(angle)\n",
    "    sin_angle = tf.sin(angle)\n",
    "    one_minus_cos = 1.0 - cos_angle\n",
    "    ux, uy, uz = axis[0], axis[1], axis[2]\n",
    "    \n",
    "    # Rotation matrix components\n",
    "    R = tf.stack([\n",
    "        [cos_angle + ux * ux * one_minus_cos,\n",
    "         ux * uy * one_minus_cos - uz * sin_angle,\n",
    "         ux * uz * one_minus_cos + uy * sin_angle],\n",
    "        [uy * ux * one_minus_cos + uz * sin_angle,\n",
    "         cos_angle + uy * uy * one_minus_cos,\n",
    "         uy * uz * one_minus_cos - ux * sin_angle],\n",
    "        [uz * ux * one_minus_cos - uy * sin_angle,\n",
    "         uz * uy * one_minus_cos + ux * sin_angle,\n",
    "         cos_angle + uz * uz * one_minus_cos]\n",
    "    ])\n",
    "    \n",
    "    # Reshape image to (H * W, 3) for matrix multiplication\n",
    "    shape = tf.shape(image)\n",
    "    h, w = shape[0], shape[1]\n",
    "    image_flat = tf.reshape(image, [-1, 3])  # Shape: (H * W, 3)\n",
    "    \n",
    "    # Apply rotation matrix to RGB values\n",
    "    rotated_image_flat = tf.matmul(image_flat, R)\n",
    "    \n",
    "    # Reshape back to original image shape\n",
    "    rotated_image = tf.reshape(rotated_image_flat, [h, w, 3])\n",
    "    return rotated_image\n",
    "    \n",
    "\n",
    "def preprocess_test(example):\n",
    "    \"\"\"pixel-homogeneous rotation\"\"\"\n",
    "    image = example['image']\n",
    "    image = tf.cast(image, tf.float32)\n",
    "    image = (image + 0.5) / 255.0 #* 2.0 - 1.0  # Normalize to [-1, 1]\n",
    "    image = tf.clip_by_value(image, 0.0, 1.0)\n",
    "    # task: hue-biased learning\n",
    "    # delta = tf.cast(example['label'], tf.float32)\n",
    "    delta = tf.random.uniform((),0,1.)\n",
    "    # delta = (res + delta) / 10\n",
    "    # image = tf.image.adjust_hue(image,delta)\n",
    "    # image = tf.image.adjust_saturation(image,3.) # push to color boundary\n",
    "    image = rotate_image(image)\n",
    "    example['image'] = image\n",
    "    return example\n",
    "\n",
    "\n",
    "def preprocess(example):\n",
    "    example = preprocess_test(example)\n",
    "    return example\n",
    "\n",
    "\n",
    "# def rotate_hue(example, delta):\n",
    "#     \"\"\"Going beyond: heterogeneous rotations\"\"\"\n",
    "#     example['image'] = tf.cast(example['image'], tf.float32)\n",
    "#     example['image'] = (example['image'] + 0.5) / 255.0 #* 2.0 - 1.0  # Normalize to [-1, 1]\n",
    "#     example['image'] = tf.clip_by_value(example['image'], 0.0, 1.0)\n",
    "#     rotate_fn = lambda x, d: tf.image.adjust_hue(x, d)\n",
    "#     # # per-pixel rotation (slower and destroys spatial continuity)\n",
    "#     # example['image'] = tf.concat(\n",
    "#     #     [\n",
    "#     #     tf.concat(\n",
    "#     #         [\n",
    "#     #         rotate_fn(example['image'][i:i+1,j:j+1],delta[i,j])\n",
    "#     #         for i in range(32)\n",
    "#     #         ], axis=0)\n",
    "#     #     for j in range(32)\n",
    "#     #     ], axis=1)\n",
    "#     # patch rotation (faster, more general)\n",
    "#     example['image'] = tf.concat(\n",
    "#         [\n",
    "#         tf.concat(\n",
    "#             [\n",
    "#             rotate_fn(example['image'][i:i+16,j:j+16],delta[i,j])\n",
    "#             for i in range(2)\n",
    "#             ], axis=0)\n",
    "#         for j in range(2)\n",
    "#         ], axis=1)\n",
    "#     return example\n",
    "\n",
    "\n",
    "# toy model\n",
    "# class TwoLayerNet(nn.Module):\n",
    "#     act_fn: 'relu'\n",
    "#     @nn.compact\n",
    "#     def __call__(self, x):\n",
    "#         x = x.reshape((x.shape[0], -1))  # Flatten input\n",
    "#         x = nn.Dense(features=512)(x)\n",
    "#         if self.act_fn == 'relu':\n",
    "#             x = nn.relu(x)\n",
    "#         else:\n",
    "#             x = colu(x)\n",
    "            \n",
    "#         x = nn.Dense(features=10)(x)\n",
    "#         return x\n",
    "\n",
    "def data_loader(data, batch_size, shuffle=True):\n",
    "    # infinite iterator\n",
    "    def batch_generator(data):\n",
    "        local_indices = np.arange(len(data['image']))\n",
    "        # Shuffle if requested\n",
    "        if shuffle:\n",
    "            np.random.shuffle(local_indices)\n",
    "        while True:\n",
    "            for start_idx in range(0, len(local_indices), batch_size):\n",
    "                end_idx = start_idx + batch_size\n",
    "                if end_idx > len(local_indices):\n",
    "                    break\n",
    "                batch_idx = local_indices[start_idx:end_idx]\n",
    "                image, label = data['image'][batch_idx], data['label'][batch_idx]\n",
    "                # hue_angle = random.uniform(-1.,1.)\n",
    "                # image = rotate_hue(image, hue_angle) # note: train-time datal processing could be a bottleneck\n",
    "                yield (image,label)\n",
    "    \n",
    "    return batch_generator(data)\n",
    "\n",
    "# optional: dataset saving\n",
    "# def prepare_and_save_data(save_path='preprocessed_data', num_samples=50000):\n",
    "#     \"\"\"Preprocess and save data to files\"\"\"\n",
    "#     if not os.path.exists(save_path):\n",
    "#         os.makedirs(save_path)\n",
    "#     # Load raw data\n",
    "#     ds = tfds.load('cifar10', split='train')\n",
    "#     # Initialize arrays to store preprocessed data\n",
    "#     images = []\n",
    "#     labels = []\n",
    "#     # Process data\n",
    "#     rng = jax.random.PRNGKey(0)\n",
    "#     for i, example in enumerate(tqdm(ds, total=num_samples)):\n",
    "#         if i >= num_samples:\n",
    "#             break\n",
    "#         # Preprocess image\n",
    "#         image = tf.cast(example['image'], tf.float32) / 255.0\n",
    "#         delta = jax.random.uniform(rng, (2,2), minval=-1., maxval=1.)\n",
    "#         processed_image = rotate_hue(example, delta)['image']   \n",
    "#         # Store in lists\n",
    "#         images.append(processed_image.numpy())\n",
    "#         labels.append(example['label'])\n",
    "#     # Convert to arrays\n",
    "#     images = np.array(images)\n",
    "#     labels = np.array(labels)\n",
    "#     # Save using numpy\n",
    "#     np.savez(\n",
    "#         os.path.join(save_path, 'preprocessed_data.npz'),\n",
    "#         images=images,\n",
    "#         labels=labels\n",
    "#     )\n",
    "#     print(f\"Saved preprocessed data to {save_path}\")\n",
    "\n",
    "# def load_preprocessed_data(data_path='preprocessed_data/preprocessed_data.npz', batch_size=128):\n",
    "#     \"\"\"Load preprocessed data and create dataset\"\"\"\n",
    "#     # Load the saved data\n",
    "#     data = jnp.load(data_path,allow_pickle=True)\n",
    "#     images = data['images']\n",
    "#     labels = data['labels']\n",
    "#     # Create TensorFlow dataset\n",
    "#     ds, test_data = data_loader(images, labels, batch_size=32)\n",
    "#     return ds, test_data # iterator, dict(X=... , Y=...)\n",
    "\n",
    "\n",
    "def create_train_state(rng, learning_rate, act_fn='relu'):\n",
    "    global ACT\n",
    "    if act_fn == 'relu':\n",
    "        ACT = nn.relu\n",
    "        model = ResNet20()\n",
    "    else:\n",
    "        ACT = partial(rcolu,dim=4)\n",
    "        model = ResNet20()\n",
    "    variables = model.init(rng, jnp.ones((1, 32, 32, 3)), train=True)\n",
    "    params = variables['params']\n",
    "    batch_stats = variables.get('batch_stats',None)\n",
    "    tx = optax.adamw(learning_rate,weight_decay=.01)\n",
    "    return TrainState.create(\n",
    "        apply_fn=model.apply,\n",
    "        params=params,\n",
    "        tx=tx,\n",
    "        batch_stats=batch_stats,\n",
    "    )\n",
    "\n",
    "\n",
    "@jax.jit\n",
    "def train_step(state, batch):\n",
    "    def loss_fn(params):\n",
    "        images, labels = batch\n",
    "        variables = {'params': params, 'batch_stats': state.batch_stats}\n",
    "        outputs, new_model_state = state.apply_fn(\n",
    "            variables, images, train=True, mutable=['batch_stats']\n",
    "        )\n",
    "        logits = outputs\n",
    "        loss = optax.softmax_cross_entropy_with_integer_labels(\n",
    "            logits=logits, labels=labels).mean()\n",
    "        # loss = optax.squared_error(logits[:,0],labels).mean()\n",
    "        return loss, (logits, new_model_state)\n",
    "    grad_fn = jax.value_and_grad(loss_fn, has_aux=True)\n",
    "    (loss, (logits, new_model_state)), grads = grad_fn(state.params)\n",
    "    state = state.apply_gradients(\n",
    "        grads=grads, batch_stats=new_model_state['batch_stats']\n",
    "    )\n",
    "    accuracy = jnp.mean(jnp.argmax(logits, -1) == batch[1])\n",
    "\n",
    "    return state, loss, accuracy\n",
    "\n",
    "\n",
    "@jax.jit\n",
    "def test_step(state, batch):\n",
    "    images, labels = batch\n",
    "    variables = {'params': state.params, 'batch_stats': state.batch_stats}\n",
    "    logits = state.apply_fn(variables, images, train=False, mutable=False)\n",
    "    loss = optax.softmax_cross_entropy_with_integer_labels(\n",
    "        logits=logits, labels=labels).mean()\n",
    "    # loss = optax.l2_loss(logits[:,0],labels).max()\n",
    "    accuracy = jnp.mean(jnp.argmax(logits, -1) == batch[1])\n",
    "    return loss, accuracy\n",
    "\n",
    "\n",
    "def train(num_steps=100000, act_fn='relu', eval_interval=1000, seed=0):\n",
    "    rng = jax.random.PRNGKey(seed)\n",
    "    random.seed(seed)\n",
    "    state = create_train_state(rng, learning_rate=1e-3, act_fn=act_fn)\n",
    "    \n",
    "    # Initialize lists to store metrics\n",
    "    train_losses = []\n",
    "    train_accuracies = []\n",
    "    test_losses = []\n",
    "    test_accuracies = []\n",
    "    evaluation_steps = []\n",
    "    \n",
    "    # Prepare the training and test datasets\n",
    "    # Note: Loading entire dataset into memory might not be optimal for large datasets\n",
    "    train_ds = tfds.load('cifar10', split='train', as_supervised=False)\n",
    "    train_ds = train_ds.map(preprocess).batch(50000).as_numpy_iterator().next()\n",
    "    train_loader = data_loader(train_ds, batch_size=128)\n",
    "    \n",
    "    test_ds = tfds.load('cifar10', split='test', as_supervised=False)\n",
    "    test_ds = test_ds.map(preprocess_test).batch(10000).as_numpy_iterator().next()\n",
    "    test_loader = data_loader(test_ds, batch_size=10000, shuffle=False)\n",
    "    \n",
    "    pbar = tqdm(range(1, num_steps + 1))\n",
    "    for step in pbar:\n",
    "        batch = next(train_loader)\n",
    "        state, loss, accuracy = train_step(state, batch)\n",
    "        # state, loss = train_step(state, batch)\n",
    "        \n",
    "        # At each evaluation interval, record metrics\n",
    "        if step % eval_interval == 0 or step == 1:\n",
    "            # Append training metrics\n",
    "            train_losses.append(float(loss))\n",
    "            train_accuracies.append(float(accuracy))\n",
    "            \n",
    "            # Evaluate on the test set\n",
    "            test_batch = next(test_loader)\n",
    "            test_loss, test_accuracy = test_step(state, test_batch)\n",
    "            # test_loss = test_step(state, test_batch)\n",
    "            test_losses.append(float(test_loss))\n",
    "            test_accuracies.append(float(test_accuracy))\n",
    "            \n",
    "            # Record the step number\n",
    "            evaluation_steps.append(step)\n",
    "            \n",
    "            pbar.set_description(f'[{act_fn}] Step {step}: '\n",
    "                  f'Train Loss = {loss:.3f}, '\n",
    "                  f'Train Accuracy = {accuracy:.3f}, '\n",
    "                  f'Test Loss = {test_loss:.3f}, '\n",
    "                  f'Test Accuracy = {test_accuracy:.3f}'\n",
    "                  )\n",
    "    \n",
    "    # Return the collected metrics\n",
    "    return {\n",
    "        'steps': evaluation_steps,\n",
    "        'train_losses': train_losses,\n",
    "        'train_accuracies': train_accuracies,\n",
    "        'test_losses': test_losses,\n",
    "        'test_accuracies': test_accuracies,\n",
    "        'activation': act_fn\n",
    "    }\n",
    "\n",
    "# Plotting function with standard deviation\n",
    "def plot_learning_curves(runs_metrics, activation_functions):\n",
    "    plt.figure(figsize=(7, 4))\n",
    "\n",
    "    # Plot Loss\n",
    "    plt.subplot(1, 2, 1)\n",
    "    for act_fn in activation_functions:\n",
    "        metrics = runs_metrics[act_fn]\n",
    "        steps = metrics['steps']  # Corrected: Use the full list of steps\n",
    "        train_losses = np.array([run['train_losses'] for run in metrics['runs']])\n",
    "        test_losses = np.array([run['test_losses'] for run in metrics['runs']])\n",
    "        \n",
    "        # Compute mean and std\n",
    "        train_loss_mean = train_losses.mean(axis=0)\n",
    "        train_loss_std = train_losses.std(axis=0)\n",
    "        test_loss_mean = test_losses.mean(axis=0)\n",
    "        test_loss_std = test_losses.std(axis=0)\n",
    "        \n",
    "        # Plot mean\n",
    "        plt.plot(steps, train_loss_mean, label=f'Train Loss ({act_fn})')\n",
    "        plt.plot(steps, test_loss_mean, label=f'Test Loss ({act_fn})')\n",
    "        \n",
    "        # Plot std shaded region\n",
    "        plt.fill_between(steps, train_loss_mean - train_loss_std, train_loss_mean + train_loss_std, alpha=0.2)\n",
    "        plt.fill_between(steps, test_loss_mean - test_loss_std, test_loss_mean + test_loss_std, alpha=0.2)\n",
    "    \n",
    "    plt.xlabel('Training Steps')\n",
    "    plt.ylabel('Loss')\n",
    "    # plt.title('Training and Test Loss')\n",
    "    plt.legend()\n",
    "    plt.grid(True)\n",
    "\n",
    "    # Plot Accuracy\n",
    "    plt.subplot(1, 2, 2)\n",
    "    for act_fn in activation_functions:\n",
    "        metrics = runs_metrics[act_fn]\n",
    "        steps = metrics['steps']  # Corrected: Use the full list of steps\n",
    "        train_accuracies = np.array([run['train_accuracies'] for run in metrics['runs']])\n",
    "        test_accuracies = np.array([run['test_accuracies'] for run in metrics['runs']])\n",
    "        \n",
    "        # Compute mean and std\n",
    "        train_acc_mean = train_accuracies.mean(axis=0)\n",
    "        train_acc_std = train_accuracies.std(axis=0)\n",
    "        test_acc_mean = test_accuracies.mean(axis=0)\n",
    "        test_acc_std = test_accuracies.std(axis=0)\n",
    "        \n",
    "        # Plot mean\n",
    "        plt.plot(steps, train_acc_mean, label=f'Train Acc ({act_fn})')\n",
    "        plt.plot(steps, test_acc_mean, label=f'Test Acc ({act_fn})')\n",
    "        \n",
    "        # Plot std shaded region\n",
    "        plt.fill_between(steps, train_acc_mean - train_acc_std, train_acc_mean + train_acc_std, alpha=0.2)\n",
    "        plt.fill_between(steps, test_acc_mean - test_acc_std, test_acc_mean + test_acc_std, alpha=0.2)\n",
    "    \n",
    "    plt.xlabel('Training Steps')\n",
    "    plt.ylabel('Accuracy')\n",
    "    # plt.title('Training and Test Accuracy')\n",
    "    plt.legend()\n",
    "    plt.grid(True)\n",
    "\n",
    "    plt.tight_layout()\n",
    "    plt.show()\n",
    "\n",
    "if __name__ == '__main__':\n",
    "    # Define activation functions to compare\n",
    "    activation_functions = ['colu', 'relu']\n",
    "    \n",
    "    # Number of runs per activation function\n",
    "    num_runs = 1  # You can increase this number for more robust statistics\n",
    "    \n",
    "    # Number of training steps and evaluation interval\n",
    "    num_steps = 10000\n",
    "    eval_interval = 1000\n",
    "    \n",
    "    # Dictionary to store metrics for each activation function\n",
    "    runs_metrics = {act_fn: {'runs': [], 'steps': []} for act_fn in activation_functions}\n",
    "    \n",
    "    for act_fn in activation_functions:\n",
    "        print(f'#### Training with {act_fn.upper()} Activation ####')\n",
    "        for run in range(1, num_runs + 1):\n",
    "            print(f'-- Run {run}/{num_runs} --')\n",
    "            seed = run  # Different seed for each run\n",
    "            metrics = train(num_steps=num_steps, act_fn=act_fn, eval_interval=eval_interval, seed=seed)\n",
    "            runs_metrics[act_fn]['runs'].append({\n",
    "                'train_losses': metrics['train_losses'],\n",
    "                'train_accuracies': metrics['train_accuracies'],\n",
    "                'test_losses': metrics['test_losses'],\n",
    "                'test_accuracies': metrics['test_accuracies'],\n",
    "            })\n",
    "            if run == 1:\n",
    "                runs_metrics[act_fn]['steps'] = metrics['steps']  # Assuming steps are the same across runs\n",
    "    \n",
    "    # Plot the learning curves comparison with standard deviation\n",
    "    plot_learning_curves(runs_metrics, activation_functions)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "2024-11-26 13:43:59.813163: W tensorflow/core/kernels/data/cache_dataset_ops.cc:914] The calling iterator did not fully read the dataset being cached. In order to avoid unexpected truncation of the dataset, the partially cached contents of the dataset  will be discarded. This can happen if you have an input pipeline similar to `dataset.cache().take(k).repeat()`. You should use `dataset.take(k).cache().repeat()` instead.\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# sanity check of dataloader\n",
    "def preprocess(example,train=True):\n",
    "    image = example['image']\n",
    "    image = tf.cast(image, tf.float32)\n",
    "    image = (image + 0.5) / 255.0 #* 2.0 - 1.0  # Normalize to [-1, 1]\n",
    "    image = tf.clip_by_value(image, 0.0, 1.0)\n",
    "    # task: hue-biased learning\n",
    "    # delta = tf.cast(example['label'], tf.float32)\n",
    "    # push to color boundary\n",
    "    delta = tf.random.uniform((),0,1.)\n",
    "    # delta = (res + delta) / 10\n",
    "    image = tf.image.adjust_hue(image,delta)\n",
    "    image = tf.image.adjust_saturation(image,3.)\n",
    "    example['image'] = image\n",
    "    return example\n",
    "\n",
    "\n",
    "train_ds = tfds.load('cifar10', split='train', as_supervised=False)\n",
    "train_ds = train_ds.map(preprocess)#.batch(50000).as_numpy_iterator().next()\n",
    "# train_loader = data_loader(train_ds, batch_size=128)\n",
    "# train_ds['label'].shape\n",
    "for example in train_ds.take(1):\n",
    "    plt.imshow(example['image'])\n",
    "    plt.axis('off')\n",
    "    break\n",
    "# show_examples(train_ds,5)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# import jax\n",
    "# import jax.numpy as jnp\n",
    "# import flax.linen as nn\n",
    "# import optax\n",
    "# from flax.training import train_state\n",
    "# import tensorflow_datasets as tfds\n",
    "# import tensorflow as tf\n",
    "# from tqdm.auto import tqdm\n",
    "\n",
    "# # Define the model\n",
    "# class TwoLayerNet(nn.Module):\n",
    "#     @nn.compact\n",
    "#     def __call__(self, x):\n",
    "#         x = x.reshape((x.shape[0], -1))  # Flatten input\n",
    "#         x = nn.Dense(features=512)(x)\n",
    "#         x = nn.relu(x)\n",
    "#         x = nn.Dense(features=10)(x)\n",
    "#         return x\n",
    "\n",
    "# # Create data pipeline\n",
    "# def prepare_data():\n",
    "#     def preprocess(example):\n",
    "#         # rng = jax.random.PRNGKey(0)\n",
    "#         # delta = jax.random.uniform(rng, (32,32), minval=-1., maxval=1.)\n",
    "#         example['image'] = rotate_hue(example['image'],.1)\n",
    "#         example['image'] = tf.cast(example['image'], tf.float32) / 255.0\n",
    "#         return example['image'], example['label']\n",
    "\n",
    "#     ds = tfds.load('cifar10', split='train')\n",
    "#     ds = ds.map(preprocess)\n",
    "#     ds = ds.batch(128).prefetch(2)\n",
    "#     return ds\n",
    "\n",
    "# # Create training state\n",
    "# def create_train_state(rng, learning_rate):\n",
    "#     model = TwoLayerNet()\n",
    "#     params = model.init(rng, jnp.ones((1, 32, 32, 3)))['params']\n",
    "#     tx = optax.adam(learning_rate)\n",
    "#     return train_state.TrainState.create(\n",
    "#         apply_fn=model.apply,\n",
    "#         params=params,\n",
    "#         tx=tx,\n",
    "#     )\n",
    "\n",
    "# # Training step\n",
    "# @jax.jit\n",
    "# def train_step(state, batch):\n",
    "#     def loss_fn(params):\n",
    "#         images, labels = batch\n",
    "#         logits = state.apply_fn({'params': params}, images)\n",
    "#         loss = optax.softmax_cross_entropy_with_integer_labels(\n",
    "#             logits=logits, labels=labels).mean()\n",
    "#         return loss, logits\n",
    "\n",
    "#     grad_fn = jax.value_and_grad(loss_fn, has_aux=True)\n",
    "#     (loss, logits), grads = grad_fn(state.params)\n",
    "#     state = state.apply_gradients(grads=grads)\n",
    "    \n",
    "#     accuracy = jnp.mean(jnp.argmax(logits, -1) == batch[1])\n",
    "#     return state, loss, accuracy\n",
    "\n",
    "# # Training loop\n",
    "# def train(num_epochs=10):\n",
    "#     rng = jax.random.PRNGKey(0)\n",
    "#     state = create_train_state(rng, learning_rate=1e-3)\n",
    "    \n",
    "#     ds = prepare_data()\n",
    "    \n",
    "#     for epoch in range(num_epochs):\n",
    "#         losses = []\n",
    "#         accuracies = []\n",
    "        \n",
    "#         # pbar = tqdm(ds, desc=f'Epoch {epoch + 1}')\n",
    "#         for batch in ds:\n",
    "#             batch = (jnp.array(batch[0]), jnp.array(batch[1]))\n",
    "#             state, loss, accuracy = train_step(state, batch)\n",
    "#             losses.append(loss)\n",
    "#             accuracies.append(accuracy)\n",
    "            \n",
    "#         avg_loss = jnp.mean(jnp.array(losses))\n",
    "#         avg_acc = jnp.mean(jnp.array(accuracies))\n",
    "#         print(f'Epoch {epoch + 1}: loss = {avg_loss:.3f}, accuracy = {avg_acc:.3f}')\n",
    "#         # pbar.close()\n",
    "\n",
    "# if __name__ == '__main__':\n",
    "#     train()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# import jax\n",
    "# import jax.numpy as jnp\n",
    "# import flax.linen as nn\n",
    "# import optax\n",
    "# from flax.training import train_state\n",
    "# import tensorflow_datasets as tfds\n",
    "# import tensorflow as tf\n",
    "# from flax.training.common_utils import shard, shard_prng_key\n",
    "# from tqdm.auto import tqdm\n",
    "\n",
    "# print('Number of devices:', jax.device_count())\n",
    "\n",
    "# # Define the model\n",
    "# class TwoLayerNet(nn.Module):\n",
    "#     @nn.compact\n",
    "#     def __call__(self, x):\n",
    "#         x = x.reshape((x.shape[0], -1))  # Flatten input\n",
    "#         x = nn.Dense(features=512)(x)\n",
    "#         x = nn.relu(x)\n",
    "#         x = nn.Dense(features=10)(x)\n",
    "#         return x\n",
    "\n",
    "# # Create data pipeline\n",
    "# def prepare_data(batch_size_per_device):\n",
    "#     def preprocess(example):\n",
    "#         image = tf.cast(example['image'], tf.float32) / 255.0\n",
    "#         label = example['label']\n",
    "#         return image, label\n",
    "\n",
    "#     # Adjust batch size for number of devices\n",
    "#     total_batch_size = batch_size_per_device * jax.device_count()\n",
    "    \n",
    "#     ds = tfds.load('cifar10', split='train')\n",
    "#     ds = ds.map(preprocess)\n",
    "#     ds = ds.batch(total_batch_size, drop_remainder=True)\n",
    "#     ds = ds.prefetch(2)\n",
    "#     return ds\n",
    "\n",
    "# # Create training state\n",
    "# def create_train_state(rng, learning_rate):\n",
    "#     model = TwoLayerNet()\n",
    "#     params = model.init(rng, jnp.ones((1, 32, 32, 3)))['params']\n",
    "#     tx = optax.adam(learning_rate)\n",
    "#     return train_state.TrainState.create(\n",
    "#         apply_fn=model.apply,\n",
    "#         params=params,\n",
    "#         tx=tx,\n",
    "#     )\n",
    "\n",
    "# # Training step (now with pmap)\n",
    "# @jax.pmap\n",
    "# def train_step(state, batch, rng):\n",
    "#     def loss_fn(params):\n",
    "#         images, labels = batch\n",
    "#         logits = state.apply_fn({'params': params}, images)\n",
    "#         loss = optax.softmax_cross_entropy_with_integer_labels(\n",
    "#             logits=logits, labels=labels).mean()\n",
    "#         return loss, logits\n",
    "\n",
    "#     grad_fn = jax.value_and_grad(loss_fn, has_aux=True)\n",
    "#     (loss, logits), grads = grad_fn(state.params)\n",
    "#     state = state.apply_gradients(grads=grads)\n",
    "    \n",
    "#     accuracy = jnp.mean(jnp.argmax(logits, -1) == batch[1])\n",
    "#     return state, loss, accuracy\n",
    "\n",
    "# # Training loop\n",
    "# def train(num_epochs=10, batch_size_per_device=32):\n",
    "#     # Initialize RNG and state\n",
    "#     rng = jax.random.PRNGKey(0)\n",
    "#     rng, init_rng = jax.random.split(rng)\n",
    "    \n",
    "#     # Create and replicate the training state across devices\n",
    "#     state = create_train_state(init_rng, learning_rate=1e-3)\n",
    "#     state = flax.jax_utils.replicate(state)\n",
    "    \n",
    "#     ds = prepare_data(batch_size_per_device)\n",
    "    \n",
    "#     for epoch in range(num_epochs):\n",
    "#         losses = []\n",
    "#         accuracies = []\n",
    "        \n",
    "#         for batch in tqdm(ds, desc=f'Epoch {epoch + 1}'):\n",
    "#             # Generate a new RNG key for each training step\n",
    "#             rng, step_rng = jax.random.split(rng)\n",
    "            \n",
    "#             # Shard the data and RNG key across devices\n",
    "#             batch = (jnp.array(batch[0]), jnp.array(batch[1]))\n",
    "#             sharded_rng = shard_prng_key(step_rng)\n",
    "#             batch = shard(batch)\n",
    "            \n",
    "#             # Perform parallel training step\n",
    "#             state, loss, accuracy = train_step(state, batch, sharded_rng)\n",
    "            \n",
    "#             # Collect metrics from all devices\n",
    "#             losses.append(jax.device_get(loss))\n",
    "#             accuracies.append(jax.device_get(accuracy))\n",
    "            \n",
    "#         # Calculate average metrics across all devices\n",
    "#         avg_loss = jnp.mean(jnp.array(losses))\n",
    "#         avg_acc = jnp.mean(jnp.array(accuracies))\n",
    "#         print(f'Epoch {epoch + 1}: loss = {avg_loss:.3f}, accuracy = {avg_acc:.3f}')\n",
    "\n",
    "#     return state\n",
    "\n",
    "# if __name__ == '__main__':\n",
    "#     train(batch_size_per_device=32)"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "base",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.12.7"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
