{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import pickle\n",
    "import torch\n",
    "\n",
    "path0= 'checkpoints/lamackl/test_v1'  # input checkpoint dir\n",
    "\n",
    "with open(f'{path0}/history.pkl', 'rb') as f:\n",
    "    history = pickle.load(f)\n",
    "batch=len(history['phi_loss'][0])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "### observing Loss in moving average\n",
    "from matplotlib import pyplot as plt\n",
    "\n",
    "a = []\n",
    "title0= 'phi_loss'  # corresponds to   L_{solve}  in the Algorithm 1.\n",
    "\n",
    "for i in history[title0]:  \n",
    "    if len(i)==batch:\n",
    "        a.extend(i)\n",
    "his_tensor = torch.stack(a).reshape(-1,len(i)).transpose(0,1)\n",
    "his_mean = his_tensor.mean(dim=0)\n",
    "stride=30\n",
    "his_momen =[]\n",
    "for i in range(len(his_mean)-stride):\n",
    "    his_momen.append(his_mean[i:i+stride].mean())\n",
    "plt.plot(list(range(len(his_momen))), his_momen)\n",
    "\n",
    "plt.title(title0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[<matplotlib.lines.Line2D at 0x7fb2260d45b0>]"
      ]
     },
     "execution_count": 12,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAiMAAAGzCAYAAAD9pBdvAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjguMywgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/H5lhTAAAACXBIWXMAAA9hAAAPYQGoP6dpAABtY0lEQVR4nO3dd3hT99k//veRZEm2bMsTL8wwe88wslvIgiclaTNLSps2TZOQp9Ad2jSrydc8bdpfZlNK06bNKNlJ20xKAiRhhOXE7I0NeGAbb2uf3x/S51g2kqw936/r8nVh+9g6CGzfvtdHkmVZBhEREVGMqGJ9A0RERJTaGIwQERFRTDEYISIiophiMEJEREQxxWCEiIiIYorBCBEREcUUgxEiIiKKKQYjREREFFMMRoiIiCimGIwQUUQ999xzkCQJx48f530QkUcMRoiIiCimGIwQERFRTDEYISIiophiMEJEfbz22muQJAkbNmw4532rVq2CJEnYvXs3vvzyS3znO99BRUUF9Ho9iouL8d3vfhfNzc0h38MDDzwASZKwf/9+3HDDDcjOzkZ+fj6WLVsGk8nU51pJknD33XfjxRdfxJgxY6DX6zFjxgxs3Lgx5PsgouhgMEJEfSxcuBCZmZl45ZVXznnfyy+/jAkTJmDixIlYu3Ytjh49iltvvRVPPvkkbrrpJqxZswYLFiyALMthuZcbbrgBJpMJlZWVWLBgAZ544gncfvvt51y3YcMGLF++HLfccgseeughNDc348orr8Tu3bvDch9EFGEyEVE/N998szxo0CDZZrMpb6urq5NVKpX80EMPybIsy93d3ed83D//+U8ZgLxx40blbX/7299kAPKxY8f8fvz7779fBiB/7Wtf6/P2u+66SwYgf/HFF8rbAMgA5O3btytvO3HihKzX6+Vrr702pPsgouhgZoSIznHjjTeisbER69evV9722muvweFw4MYbbwQApKenK+8zmUxoamrCnDlzAAA7d+4My30sXbq0z+v/+7//CwB49913+7x97ty5mDFjhvL6kCFDsGjRInzwwQew2+1huRciihwGI0R0jiuvvBJGoxEvv/yy8raXX34ZU6dOxejRowEALS0tWLZsGYqKipCeno7CwkIMHz4cANDW1haW+xg1alSf10eMGAGVSnXOrpD+1wHA6NGj0d3djTNnzoTlXogochiM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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "#### observing validation metric (score or ppl) \n",
    "a = []\n",
    "#========= control ========\n",
    "title0= 'val_score'\n",
    "# title0= 'val_ppl'\n",
    "#==========================\n",
    "\n",
    "for i in history[title0]:  \n",
    "    a.append(i)\n",
    "plt.title(title0)\n",
    "plt.plot(list(range(len(a))), a)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "######################### Observing Weight ###################\n",
    "### Observe the heat map of the token weights, predicted upon the validation data.\n",
    "\n",
    "from transformers import AutoTokenizer\n",
    "import matplotlib\n",
    "import matplotlib.pyplot as plt\n",
    "import matplotlib.animation\n",
    "import numpy as np\n",
    "import torch\n",
    "from PIL import Image, ImageDraw, ImageFont\n",
    "\n",
    "model_name = \"meta-llama/Llama-2-7b-hf\"\n",
    "\n",
    "tokenizer = AutoTokenizer.from_pretrained(model_name, trust_remote_code=True)\n",
    "tokenizer.pad_token = tokenizer.eos_token   \n",
    "\n",
    "##### control ##############\n",
    "num0 = -1  #  last valiation\n",
    "id0 = 14  # 14th validation data\n",
    "##############################\n",
    "\n",
    "weight_mu = history['val_weight_mu'][num0][id0]\n",
    "tokens = history['val_Data'][num0][id0]\n",
    "weight_mu = weight_mu / torch.mean(weight_mu)\n",
    "decoded_words=[]\n",
    "for token0 in tokens[1:]:\n",
    "\n",
    "    word = tokenizer.decode([token0], skip_special_tokens=True, clean_up_tokenization_spaces=True)\n",
    "    decoded_words.append(word)\n",
    "decoded_words = ['\\s' if x == '' else x for x in decoded_words]\n",
    "\n",
    "colors = []\n",
    "for w, p in zip(decoded_words, weight_mu):\n",
    "        colors.append([w, (int(255*p),150,100)])\n",
    "\n",
    "words = decoded_words#[num]\n",
    "block_width = 50\n",
    "block_height = 20\n",
    "max_blocks_per_row = 20  # Adjust as needed to fit your desired layout\n",
    "\n",
    "# Calculate the image size\n",
    "img_width = block_width * min(len(words), max_blocks_per_row)\n",
    "img_height = block_height * ((len(words) - 1) // max_blocks_per_row + 1)\n",
    "image = Image.new('RGB', (img_width, img_height), (255, 255, 255))\n",
    "draw = ImageDraw.Draw(image)\n",
    "\n",
    "# Set up the font\n",
    "try:\n",
    "    font = ImageFont.truetype(\"arial.ttf\", 16)  # Adjust font size as needed\n",
    "except IOError:\n",
    "    font = ImageFont.load_default()\n",
    "\n",
    "# Draw the color blocks and the words on them\n",
    "for i, (word, color) in enumerate(colors):\n",
    "    row = i // max_blocks_per_row\n",
    "    col = i % max_blocks_per_row\n",
    "    x_position = col * block_width\n",
    "    y_position = row * block_height\n",
    "    \n",
    "    # color = word_to_color(word)\n",
    "    draw.rectangle([x_position, y_position, x_position + block_width, y_position + block_height], fill=color)\n",
    "    \n",
    "    # Adjust text position if necessary\n",
    "    text_x = x_position + 5  # Small padding from the left\n",
    "    text_y = y_position + (block_height - 8) / 2  # Vertically center text\n",
    "    draw.text((text_x, text_y), word, fill=\"black\", font=font)\n",
    "\n",
    "# Display or save the image\n",
    "image.show()  "
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "yb_qlora3.9",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.9.18"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
