{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Visually Descriptive Language Model (VDLM) Demo"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "VDLM is a visual reasoning framework for vector graphics, that operates on text-based visual descriptions---SVG representations and learned intermediate symbolic representations, which enable zero-shot reasoning with an off-the-shelf LLM. \n",
    "In this demo, we show how to use VDLM for performing visual question answering on vector graphics images, with the following steps:\n",
    "- Step 1: The input image is first converted into individual SVG paths with the help of a rule-based Raster-to-SVG algorithm [VTracer](https://github.com/visioncortex/vtracer).\n",
    "- Step 2: The SVG paths are then transformed into text-based Primal Visual Descriptions (PVD) (in JSON) with a finetuned [Mistral-7b model]() **(hidden for anonymous submission)**.\n",
    "- Step 3: The PVD representation is then feed into the an inference-only LLM / LMM, e.g., GPT-4o, along with the question, to perform get the final answer.\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Prerequisites\n",
    "\n",
    "- **Set up the environment**\n",
    "    ```bash\n",
    "    conda env create -f environment.yml\n",
    "    conda activate vdlm\n",
    "    ```\n",
    "    Select the correct kernel for the demo notebook.\n",
    "\n",
    "- **Download the pretrained SVG-to-PVD model**\n",
    "    ```bash\n",
    "    mkdir -p data/ckpts\n",
    "    cd data/ckpts\n",
    "    git lfs install\n",
    "    git clone **(hidden for anonymous submission)**\n",
    "    ```\n",
    "\n",
    "- **Serve the SVG-to-PVD model**\n",
    "\n",
    "    Once the pretrained model is downloaded, return to the root directory, then run the following command to start vllm server:\n",
    "    ```bash\n",
    "    CUDA_VISIBLE_DEVICES=0 ./vllm_serve_model.sh\n",
    "    ```\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Prepare Inputs"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<PIL.PngImagePlugin.PngImageFile image mode=RGBA size=336x336>"
      ]
     },
     "execution_count": 4,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "from inference_perception import *\n",
    "# We prepared some input image examples in `demo_examples/image_inputs`\n",
    "img_path = \"demo_examples/image_inputs/lines_segments.png\"\n",
    "question = \"How many line segments are there in the image? What's the total length of all the line segments in the image?\"\n",
    "\n",
    "# visualize input image\n",
    "Image.open(img_path)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Setup SVG-to-PVD Model Client"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "try automatically find the hosted model id...\n",
      "Using model: data/ckpts/PVD-160k-Mistral-7b\n",
      "generation_configs: {'temperature': 0.0, 'max_tokens': 8192, 'top_p': 1.0}\n"
     ]
    }
   ],
   "source": [
    "openai_api_key = \"EMPTY\"\n",
    "openai_api_base = \"http://localhost:8000/v1\"\n",
    "\n",
    "print(f\"try automatically find the hosted model id...\")\n",
    "model_list_data = requests.get(f\"{openai_api_base}/models\").json()\n",
    "model_id = model_list_data[\"data\"][0][\"id\"]\n",
    "print(f\"Using model: {model_id}\")\n",
    "\n",
    "client = setup_client(openai_api_base=openai_api_base, openai_api_key=openai_api_key)\n",
    "\n",
    "generation_configs = DEFAULT_GENERATION_CONFIGS\n",
    "print(\"generation_configs:\", generation_configs)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Step 1: Image -> SVG"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Path 0 added with per step difference: 1 | overall difference: 0.004912591568490141\n",
      "Path 1 added with per step difference: 0.0032472739912958646 | overall difference: 0.0016653175771942765\n",
      "Path 2 added with per step difference: 0.0016653175771942765 | overall difference: 0.0\n",
      "Overall dif: 0.0 < 0.0005, stopping the addition of new paths.\n",
      "Saved 3 paths out of 3\n",
      "Saved indices [0, 1, 2]\n",
      "\n",
      "-------------------------\n",
      "parsed single svg path 0:\n",
      "svg code:\n",
      "\n",
      "<?xml version=\"1.0\" ?>\n",
      "<svg xmlns=\"http://www.w3.org/2000/svg\" version=\"1.1\" width=\"336\" height=\"336\">\n",
      "<path d=\"M0,0 L336,0 L336,336 L0,336 Z \" fill=\"#FEFEFE\" transform=\"translate(0,0)\"/>\n",
      "</svg>\n",
      "\n"
     ]
    },
    {
     "data": {
      "image/svg+xml": [
       "<svg xmlns=\"http://www.w3.org/2000/svg\" version=\"1.1\" width=\"336\" height=\"336\">\n",
       "<path d=\"M0,0 L336,0 L336,336 L0,336 Z \" fill=\"#FEFEFE\" transform=\"translate(0,0)\"/>\n",
       "</svg>"
      ],
      "text/plain": [
       "<IPython.core.display.SVG object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "-------------------------\n",
      "parsed single svg path 1:\n",
      "svg code:\n",
      "\n",
      "<?xml version=\"1.0\" ?>\n",
      "<svg xmlns=\"http://www.w3.org/2000/svg\" version=\"1.1\" width=\"336\" height=\"336\">\n",
      "<path d=\"M0,0 L6,1 L85,1 L90,0 L93,2 L92,8 L86,9 L85,6 L6,6 L6,8 L0,9 L-2,6 L-1,1 Z \" fill=\"#030303\" transform=\"translate(200,175)\"/>\n",
      "</svg>\n",
      "\n"
     ]
    },
    {
     "data": {
      "image/svg+xml": [
       "<svg xmlns=\"http://www.w3.org/2000/svg\" version=\"1.1\" width=\"336\" height=\"336\">\n",
       "<path d=\"M0,0 L6,1 L85,1 L90,0 L93,2 L92,8 L86,9 L85,6 L6,6 L6,8 L0,9 L-2,6 L-1,1 Z \" fill=\"#030303\" transform=\"translate(200,175)\"/>\n",
       "</svg>"
      ],
      "text/plain": [
       "<IPython.core.display.SVG object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "-------------------------\n",
      "parsed single svg path 2:\n",
      "svg code:\n",
      "\n",
      "<?xml version=\"1.0\" ?>\n",
      "<svg xmlns=\"http://www.w3.org/2000/svg\" version=\"1.1\" width=\"336\" height=\"336\">\n",
      "<path d=\"M0,0 L6,1 L7,6 L6,8 L4,8 L4,39 L7,40 L6,46 L0,47 L-2,44 L-1,39 L-1,8 L-2,2 Z \" fill=\"#060606\" transform=\"translate(58,125)\"/>\n",
      "</svg>\n",
      "\n"
     ]
    },
    {
     "data": {
      "image/svg+xml": [
       "<svg xmlns=\"http://www.w3.org/2000/svg\" version=\"1.1\" width=\"336\" height=\"336\">\n",
       "<path d=\"M0,0 L6,1 L7,6 L6,8 L4,8 L4,39 L7,40 L6,46 L0,47 L-2,44 L-1,39 L-1,8 L-2,2 Z \" fill=\"#060606\" transform=\"translate(58,125)\"/>\n",
       "</svg>"
      ],
      "text/plain": [
       "<IPython.core.display.SVG object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "from IPython.display import SVG, display\n",
    "\n",
    "# SVG conversion configs\n",
    "svg_conversion_method = \"raw_svg_individual_paths_w_rec_check\"\n",
    "if \"geoclidean\" in img_path or \"nlvr\" in img_path:\n",
    "    diff_threshold = 5e-6 # preserving smaller objects during filtering\n",
    "else:\n",
    "    diff_threshold = 5e-4\n",
    "img_2_svg_configs=CONFIGS[\"default\"]\n",
    "\n",
    "# output dir config\n",
    "model_version = model_id.split(\"/\")[-1]\n",
    "img_base_name = os.path.basename(img_path).replace(\".png\", \"\").replace(\".jpg\", \"\")\n",
    "output_dir = f\"demo_examples/perception_output/{model_version}/{img_base_name}\"\n",
    "output_perception_dir = f\"{output_dir}/output_perception\"\n",
    "svg_output_dir=f\"{output_dir}/svg\"\n",
    "os.makedirs(output_dir, exist_ok=True)\n",
    "\n",
    "# save original image\n",
    "img = Image.open(img_path)\n",
    "img.save(f\"{output_dir}/input_img.png\")\n",
    "\n",
    "# Image -> SVG: the SVG paths are stored in {output_dir}/svg\n",
    "svg_strs, svg_paths = img_2_svg_strs(method=svg_conversion_method, img_path=img_path, svg_config=img_2_svg_configs, svg_output_dir=svg_output_dir, topk_paths=30, diff_threshold=diff_threshold)\n",
    "print()\n",
    "\n",
    "# display each filtered single SVG paths\n",
    "for i, svg_path in enumerate(svg_paths):\n",
    "    print(\"-------------------------\")\n",
    "    print(f\"parsed single svg path {i}:\")\n",
    "    print(\"svg code:\\n\")\n",
    "    print(svg_strs[i])\n",
    "    display(SVG(filename=svg_path))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Step 2: SVG -> PVD\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Aggregated PVD perception results:\n",
      "\n",
      "{'object_0': [{'type': 'rectangle', 'vertices': [[0.0, 336.0], [0.0, 0.0], [336.0, 0.0], [336.0, 336.0]], 'color': [254, 254, 254], 'style': 'filled shape'}], 'object_1': [{'type': 'line_segment', 'vertices': [[202, 179], [289, 179]], 'color': [3, 3, 3], 'line_width': 5}], 'object_2': [{'type': 'line_segment', 'vertices': [[60, 167], [60, 129]], 'color': [6, 6, 6], 'line_width': 4}]}\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAxYAAAGXCAYAAAAuzcbzAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjguMywgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/H5lhTAAAACXBIWXMAAA9hAAAPYQGoP6dpAABaC0lEQVR4nO3deXhU9b3H8c8ESAhLEsKSsMsmi7IoKkYUF1BWFcVWEBUswoWCrYLUYl25rbhcFctVqW2vWCt6qxWtC1hkLTVylYpsQoWioBDWJsMaIPndP3xmzJBMMss5c86ceb+eJ88jM2fO+X0Izvd8f/M7Z3zGGCMAAAAAiEOa0wMAAAAAkPxoLAAAAADEjcYCAAAAQNxoLAAAAADEjcYCAAAAQNxoLAAAAADEjcYCAAAAQNxoLAAAAADEjcYCAAAAQNxoLJD0HnroIfl8vpheO2/ePPl8Pn311VfWDqqCr776Sj6fT/PmzbPtGAAAuNXYsWN1xhlnOD0MJACNBRyzceNG3XzzzWrZsqUyMjLUokULjR49Whs3bnR6aI5Yvny5fD6f3njjDaeHAgA1CkzMBH7q1q2rM888U1OmTNGePXskST/5yU/k8/m0devWsPv5xS9+IZ/Pp3Xr1kmSzjjjjOA+09LSlJOTo+7du2vChAlavXp1xOO77LLLQsaXmZmpHj16aPbs2SovL48vvMvs2rVLDz30kNauXZvSY4DzaCzgiDfffFPnnnuulixZottuu03PPfecxo0bp2XLluncc8/VggULIt7Xfffdp2PHjsU0jltuuUXHjh1T27ZtY3o9AKS6mTNn6uWXX9Z///d/66KLLtLzzz+vgoICHT16VKNHj5YkzZ8/P+zrX331VXXv3l09evQIPtarVy+9/PLL+sMf/qBZs2bp8ssv1zvvvKMLL7xQU6dOjXhsrVq10ssvv6yXX35Zs2bNUt26dXXXXXfp/vvvjz2wC+3atUsPP/yw441FuDH89re/1ZYtWxI/KCRcbacHgNSzbds23XLLLWrfvr1Wrlyppk2bBp/76U9/qksuuUS33HKL1q1bp/bt24fdz5EjR1S/fn3Vrl1btWvH9k+5Vq1aqlWrVkyvBQBIgwcP1nnnnSdJuv3229W4cWM99dRTevvttzVq1Ch17NhRr776qh544IFKry0sLNT27dv16KOPhjzesmVL3XzzzSGPPfbYY7rpppv09NNPq1OnTpo0aVKNY8vOzg7Zz8SJE9WlSxfNmTNHM2fOTNn3/6NHj6pevXoJO16dOnUSdiw4i08skHBPPPGEjh49qhdeeCGkqZCkJk2a6De/+Y2OHDmixx9/PPh44DqKTZs26aabblKjRo108cUXhzxX0bFjx/STn/xETZo0UcOGDXXNNdfo22+/lc/n00MPPRTcrqprLM444wwNGzZMq1at0gUXXKC6deuqffv2+sMf/hByjIMHD+ruu+9W9+7d1aBBA2VlZWnw4MH6/PPPLfqb+j7bP//5T918883Kzs5W06ZNdf/998sYo507d+raa69VVlaW8vPz9eSTT4a8/sSJE3rggQfUu3dvZWdnq379+rrkkku0bNmySsc6cOCAbrnlFmVlZSknJ0djxozR559/XuX1IZs3b9YNN9yg3Nxc1a1bV+edd57+8pe/WJYbQPK64oorJEnbt2+XJI0ePVqbN2/WP/7xj0rbzp8/Xz6fT6NGjapxv5mZmXr55ZeVm5urX/3qVzLGRD22unXr6vzzz9ehQ4e0d+/ekOf++Mc/qnfv3srMzFRubq5GjhypnTt3VtrH6tWrNWTIEDVq1Ej169dXjx499Mwzz4Rss3TpUl1yySWqX7++cnJydO211+qLL74I2Sbw/r5161aNHTtWOTk5ys7O1m233aajR4+GbLt48WJdfPHFysnJUYMGDdS5c2fde++9kr5bRnv++edLkm677bbg0q/A+/Zll12ms88+W2vWrFG/fv1Ur1694GtPr4kBZ5xxhsaOHRvyWHFxse666y6dccYZysjIUKtWrXTrrbdq//79NY6hqmssjhw5omnTpql169bKyMhQ586d9V//9V+Vfq8+n09TpkzRW2+9pbPPPlsZGRk666yztGjRokrjhvNoLJBw77zzjs444wxdcsklVT7fr18/nXHGGXrvvfcqPfeDH/xAR48e1SOPPKLx48eHPcbYsWM1Z84cDRkyRI899pgyMzM1dOjQiMe4detW3XDDDbryyiv15JNPqlGjRho7dmzI9R//+te/9NZbb2nYsGF66qmnNH36dK1fv16XXnqpdu3aFfGxInHjjTeqvLxcjz76qPr06aNf/vKXmj17tq688kq1bNlSjz32mDp27Ki7775bK1euDL7O7/frd7/7nS677DI99thjeuihh7Rv3z4NHDgw5OPq8vJyXX311Xr11Vc1ZswY/epXv9Lu3bs1ZsyYSmPZuHGjLrzwQn3xxRf6+c9/rieffFL169fX8OHDo1rCBsCbtm3bJklq3LixJIVdDlVWVqY//elPuuSSS9SmTZuI9t2gQQNdd911+vbbb7Vp06aYxhe4oUZOTk7wsV/96le69dZb1alTJz311FO68847tWTJEvXr10/FxcXB7RYvXqx+/fpp06ZN+ulPf6onn3xSl19+ud59993gNh9++KEGDhyovXv36qGHHtLUqVP10UcfqW/fvlXeKOSHP/yhDh06pFmzZumHP/yh5s2bp4cffjj4/MaNGzVs2DCVlpZq5syZevLJJ3XNNdfo73//uySpa9eumjlzpiRpwoQJwaVf/fr1C+7jwIEDGjx4sHr16qXZs2fr8ssvj+rv7PDhw7rkkks0Z84cXXXVVXrmmWc0ceJEbd68Wd98801EY6jIGKNrrrlGTz/9tAYNGqSnnnpKnTt31vTp06tc6rZq1Sr9+Mc/1siRI/X444/r+PHjGjFihA4cOBBVDiSAARKouLjYSDLXXntttdtdc801RpLx+/3GGGMefPBBI8mMGjWq0raB5wLWrFljJJk777wzZLuxY8caSebBBx8MPvbiiy8aSWb79u3Bx9q2bWskmZUrVwYf27t3r8nIyDDTpk0LPnb8+HFTVlYWcozt27ebjIwMM3PmzJDHJJkXX3yx2szLli0zkszrr79eKduECROCj506dcq0atXK+Hw+8+ijjwYf//e//20yMzPNmDFjQrYtLS0NOc6///1vk5eXZ370ox8FH/vzn/9sJJnZs2cHHysrKzNXXHFFpbH379/fdO/e3Rw/fjz4WHl5ubnoootMp06dqs0IwDsC758ffvih2bdvn9m5c6d57bXXTOPGjU1mZqb55ptvgtuef/75plWrViHvmYsWLTKSzG9+85uQ/bZt29YMHTo07HGffvppI8m8/fbb1Y7v0ksvNV26dDH79u0z+/btM5s3bzbTp083kkL2/9VXX5latWqZX/3qVyGvX79+valdu3bw8VOnTpl27dqZtm3bmn//+98h25aXlwf/u1evXqZZs2bmwIEDwcc+//xzk5aWZm699dbgY4H394rvxcYYc91115nGjRtXyrtv376wWT/55JOwdebSSy81kszcuXMrPXd6TQxo27ZtSC154IEHjCTz5ptvVto2kL26MYwZM8a0bds2+Oe33nrLSDK//OUvQ7a74YYbjM/nM1u3bg0ZY3p6eshjn3/+uZFk5syZU+lYcBafWCChDh06JElq2LBhtdsFnvf7/SGPT5w4scZjBD4e/fGPfxzy+B133BHxOLt16xbyiUrTpk3VuXNn/etf/wo+lpGRobS07/4XKisr04EDB4IfUVf1kX88br/99uB/16pVS+edd56MMRo3blzw8ZycnEpjrFWrltLT0yV996nEwYMHderUKZ133nkhY1y0aJHq1KkT8ilQWlqaJk+eHDKOgwcPaunSpcEZtv3792v//v06cOCABg4cqC+//FLffvutpdkBuNuAAQPUtGlTtW7dWiNHjlSDBg20YMECtWzZMrjNzTffrG+++SbkE9X58+crPT1dP/jBD6I6XoMGDSR9X0+qs3nzZjVt2lRNmzZVly5d9MQTT+iaa64JWd755ptvqry8XD/84Q+D72n79+9Xfn6+OnXqFFw6+tlnn2n79u268847Qz7tkBRcjrt7926tXbtWY8eOVW5ubvD5Hj166Morr9T7779faYyn17VLLrlEBw4cCNa/wLHefvvtmO9mlZGRodtuuy2m10rSn//8Z/Xs2VPXXXddpediud37+++/r1q1auknP/lJyOPTpk2TMUYLFy4MeXzAgAHq0KFD8M89evRQVlZWSL2DO9BYIKECDUNNBSFcA9KuXbsaj/H1118rLS2t0rYdO3aMeJxVfSzfqFEj/fvf/w7+uby8PHgRYUZGhpo0aaKmTZtq3bp1KikpifhYsYwnOztbdevWVZMmTSo9XnGMkvTSSy+pR48eqlu3rho3bqymTZvqvffeCxnj119/rebNm1e6mO/0v7OtW7fKGKP7778/WKwDPw8++KAkVVq3DMDbnn32WS1evFjLli3Tpk2b9K9//UsDBw4M2WbkyJGqVatWcDnU8ePHtWDBAg0ePFiNGjWK6niHDx+WVPMElfTdtQKLFy/WBx98oOeee04tW7bUvn37VLdu3eA2X375pYwx6tSpU6X3tS+++CL4nhZY4nX22WeHPd7XX38tSercuXOl57p27ar9+/fryJEjIY+f/v4e+PsIvJffeOON6tu3r26//Xbl5eVp5MiR+tOf/hRVk9GyZcvgJFMstm3bVm3uaH399ddq0aJFpd9h165dg89XFElNhjtwVygkVHZ2tpo3bx68X3k469atU8uWLZWVlRXyeGZmpp3DCwp3pxBT4aKyRx55RPfff79+9KMf6T//8z+Vm5urtLQ03XnnnZbfI72q8UQyxj/+8Y8aO3ashg8frunTp6tZs2aqVauWZs2aFSyS0QjkuvvuuyudOARE08ABSH4XXHBB8K5Q4TRr1kxXXnml/vznP+vZZ5/VO++8o0OHDgWvv4jGhg0bJEX2XlO/fn0NGDAg+Oe+ffvq3HPP1b333qtf//rXkr57X/P5fFq4cGGV76uBT0jsUtN7eWZmplauXKlly5bpvffe06JFi/S///u/uuKKK/TXv/41ojtbRVs7y8rKotrebpHUO7gDjQUSbtiwYfrtb3+rVatWBe/sVNHf/vY3ffXVV/qP//iPmPbftm1blZeXa/v27erUqVPw8eq+oCkWb7zxhi6//HL9/ve/D3m8uLi40icJTnnjjTfUvn17vfnmmyEfVwc+XQho27atli1bVukWhKf/nQVu/1unTp2QYg0ANRk9erQWLVqkhQsXav78+crKytLVV18d1T4OHz6sBQsWqHXr1sHZ7Wj06NFDN998s37zm9/o7rvvVps2bdShQwcZY9SuXTudeeaZYV8bWIqzYcOGsO9/ge9Equo7GzZv3qwmTZqofv36UY87LS1N/fv3V//+/fXUU0/pkUce0S9+8QstW7ZMAwYMiGk5kvTdrH/Fi9Ol7+4muHv37pDHOnToEGzowolmDG3bttWHH36oQ4cOhXxqsXnz5uDzSE4shULCTZ8+XZmZmfqP//iPSnd0OHjwoCZOnKh69epp+vTpMe0/MJP+3HPPhTw+Z86c2AYcRq1atSrNlrz++uuuusYgMMtTcZyrV69WYWFhyHYDBw7UyZMn9dvf/jb4WHl5uZ599tmQ7Zo1a6bLLrtMv/nNbyoVHknat2+flcMH4CHDhw9XvXr19Nxzz2nhwoW6/vrrQ5Yk1eTYsWO65ZZbdPDgweC3dcfiZz/7mU6ePKmnnnpKknT99derVq1aevjhhyu9pxtjgnXq3HPPVbt27TR79uxKJ+OB1zVv3ly9evXSSy+9FLLNhg0b9Ne//lVDhgyJerwHDx6s9FivXr0kSaWlpZIUbFZOH1dNOnToEHLdiyS98MILlT6xGDFihD7//PMq7/wXyB7NGIYMGaKysjL993//d8jjTz/9tHw+nwYPHhxNDLgIn1gg4Tp16qSXXnpJo0ePVvfu3TVu3Di1a9dOX331lX7/+99r//79evXVV0Mu1IpG7969NWLECM2ePVsHDhzQhRdeqBUrVuif//ynpNguNKvKsGHDNHPmTN1222266KKLtH79er3yyivVfqlfog0bNkxvvvmmrrvuOg0dOlTbt2/X3Llz1a1bt+A6Zem7gn/BBRdo2rRp2rp1q7p06aK//OUvwYJW8e/s2Wef1cUXX6zu3btr/Pjxat++vfbs2aPCwkJ98803ln6PBwDvaNCggYYPHx68zqK6ZVDffvut/vjHP0r67lOKTZs26fXXX1dRUZGmTZsW8yfa0nc35xgyZIh+97vf6f7771eHDh30y1/+UjNmzNBXX32l4cOHq2HDhtq+fbsWLFigCRMm6O6771ZaWpqef/55XX311erVq5duu+02NW/eXJs3b9bGjRv1wQcfSPruu5oGDx6sgoICjRs3TseOHdOcOXOUnZ1d5XdG1GTmzJlauXKlhg4dqrZt22rv3r167rnn1KpVq+Cn/h06dFBOTo7mzp2rhg0bqn79+urTp0+N1yXefvvtmjhxokaMGKErr7xSn3/+uT744INKn7pPnz5db7zxhn7wgx/oRz/6kXr37q2DBw/qL3/5i+bOnauePXtGNYarr75al19+uX7xi1/oq6++Us+ePfXXv/5Vb7/9tu68886Y6z9cwIE7UQHGGGPWrVtnRo0aZZo3b27q1Klj8vPzzahRo8z69esrbRu4LV9Vt9s7/Xazxhhz5MgRM3nyZJObm2saNGhghg8fbrZs2WIkhdyiNdztZqu61eGll15qLr300uCfjx8/bqZNm2aaN29uMjMzTd++fU1hYWGl7ay43ezpuceMGWPq169f5RjPOuus4J/Ly8vNI488Ytq2bWsyMjLMOeecY959991Kt/4zxph9+/aZm266yTRs2NBkZ2ebsWPHmr///e9GknnttddCtt22bZu59dZbTX5+vqlTp45p2bKlGTZsmHnjjTeqzQjAOwLvn5988knEr3nvvfeMJNO8efNKt+sOCNzyW5Lx+XwmKyvLnHXWWWb8+PFm9erVER/r9PfDipYvX17pVqt//vOfzcUXX2zq169v6tevb7p06WImT55stmzZEvLaVatWmSuvvNI0bNjQ1K9f3/To0aPSbU8//PBD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/DJkm4MHD2r06NHKyspSTk6Oxo0bp8OHDycwRc05xo4dW+l3NGjQoJBt3JBDkmbNmqXzzz9fDRs2VLNmzTR8+HBt2bIlZJtI/k3t2LFDQ4cOVb169dSsWTNNnz5dp06dSmSUiLJcdtlllX43EydODNnG6SzPP/+8evToEfxCo4KCAi1cuDD4fLL8Prwg2WoE9cEd76teqRHUB/fVBymFaoRJMq+99ppJT083//M//2M2btxoxo8fb3JycsyePXucHlq1HnzwQXPWWWeZ3bt3B3/27dsXfH7ixImmdevWZsmSJebTTz81F154obnoooscHPH33n//ffOLX/zCvPnmm0aSWbBgQcjzjz76qMnOzjZvvfWW+fzzz80111xj2rVrZ44dOxbcZtCgQaZnz57m448/Nn/7299Mx44dzahRo1yVY8yYMWbQoEEhv6ODBw+GbOOGHMYYM3DgQPPiiy+aDRs2mLVr15ohQ4aYNm3amMOHDwe3qenf1KlTp8zZZ59tBgwYYD777DPz/vvvmyZNmpgZM2a4Lsull15qxo8fH/K7KSkpcVWWv/zlL+a9994z//znP82WLVvMvffea+rUqWM2bNhgjEme30eyS8YaQX1wx/uqV2oE9cF99cGY1KkRSddYXHDBBWby5MnBP5eVlZkWLVqYWbNmOTiqmj344IOmZ8+eVT5XXFxs6tSpY15//fXgY1988YWRZAoLCxM0wsic/mZbXl5u8vPzzRNPPBF8rLi42GRkZJhXX33VGGPMpk2bjCTzySefBLdZuHCh8fl85ttvv03Y2CsKVzSuvfbasK9xY46AvXv3GklmxYoVxpjI/k29//77Ji0tzRQVFQW3ef75501WVpYpLS1NbIAKTs9izHeF46c//WnY17g1S6NGjczvfve7pP59JJtkrBHUB/e9r3qpRlAf3JnFGG/WiKRaCnXixAmtWbNGAwYMCD6WlpamAQMGqLCw0MGRRebLL79UixYt1L59e40ePVo7duyQJK1Zs0YnT54MydWlSxe1adPG9bm2b9+uoqKikLFnZ2erT58+wbEXFhYqJydH5513XnCbAQMGKC0tTatXr074mKuzfPlyNWvWTJ07d9akSZN04MCB4HNuzlFSUiJJys3NlRTZv6nCwkJ1795deXl5wW0GDhwov9+vjRs3JnD0oU7PEvDKK6+oSZMmOvvsszVjxgwdPXo0+JzbspSVlem1117TkSNHVFBQkNS/j2SSzDWC+uC+99WqJGONoD64L4uXa0RtpwcQjf3796usrCzkL1WS8vLytHnzZodGFZk+ffpo3rx56ty5s3bv3q2HH35Yl1xyiTZs2KCioiKlp6crJycn5DV5eXkqKipyZsARCoyvqt9J4LmioiI1a9Ys5PnatWsrNzfXVfkGDRqk66+/Xu3atdO2bdt07733avDgwSosLFStWrVcm6O8vFx33nmn+vbtq7PPPluSIvo3VVRUVOXvLfCcE6rKIkk33XST2rZtqxYtWmjdunW65557tGXLFr355pvB8bohy/r161VQUKDjx4+rQYMGWrBggbp166a1a9cm5e8j2SRrjaA+uO99tSrJWCOoD+7Kkgo1Iqkai2Q2ePDg4H/36NFDffr0Udu2bfWnP/1JmZmZDo4MASNHjgz+d/fu3dWjRw916NBBy5cvV//+/R0cWfUmT56sDRs2aNWqVU4PJW7hskyYMCH43927d1fz5s3Vv39/bdu2TR06dEj0MMPq3Lmz1q5dq5KSEr3xxhsaM2aMVqxY4fSw4HLUh+SQjDWC+uCe+iClRo1IqqVQTZo0Ua1atSpdJb9nzx7l5+c7NKrY5OTk6Mwzz9TWrVuVn5+vEydOqLi4OGSbZMgVGF91v5P8/Hzt3bs35PlTp07p4MGDrs7Xvn17NWnSRFu3bpXkzhxTpkzRu+++q2XLlqlVq1bBxyP5N5Wfn1/l7y3wXKKFy1KVPn36SFLI78YNWdLT09WxY0f17t1bs2bNUs+ePfXMM88k5e8jGXmlRlAf3F8fJPfXCOqDu+qDlBo1Iqkai/T0dPXu3VtLliwJPlZeXq4lS5aooKDAwZFF7/Dhw9q2bZuaN2+u3r17q06dOiG5tmzZoh07drg+V7t27ZSfnx8ydr/fr9WrVwfHXlBQoOLiYq1Zsya4zdKlS1VeXh58A3Cjb775RgcOHFDz5s0luSuHMUZTpkzRggULtHTpUrVr1y7k+Uj+TRUUFGj9+vUhhXDx4sXKyspSt27dEhNENWepytq1ayUp5HfjhiynKy8vV2lpaVL9PpKZV2oE9cH99UFyb42gPqyV5P76IHm0Rjh77Xj0XnvtNZORkWHmzZtnNm3aZCZMmGBycnJCrpJ3o2nTppnly5eb7du3m7///e9mwIABpkmTJmbv3r3GmO9uM9amTRuzdOlS8+mnn5qCggJTUFDg8Ki/c+jQIfPZZ5+Zzz77zEgyTz31lPnss8/M119/bYz57naCOTk55u233zbr1q0z1157bZW3EzznnHPM6tWrzapVq0ynTp0Sfgu+6nIcOnTI3H333aawsNBs377dfPjhh+bcc881nTp1MsePH3dVDmOMmTRpksnOzjbLly8PucXe0aNHg9vU9G8qcOu6q666yqxdu9YsWrTING3aNOG3rqspy9atW83MmTPNp59+arZv327efvtt0759e9OvXz9XZfn5z39uVqxYYbZv327WrVtnfv7znxufz2f++te/GmOS5/eR7JKxRlAf3PG+6pUaQX1wX30wJnVqRNI1FsYYM2fOHNOmTRuTnp5uLrjgAvPxxx87PaQa3XjjjaZ58+YmPT3dtGzZ0tx4441m69atweePHTtmfvzjH5tGjRqZevXqmeuuu87s3r3bwRF/b9myZUZSpZ8xY8YYY767peD9999v8vLyTEZGhunfv7/ZsmVLyD4OHDhgRo0aZRo0aGCysrLMbbfdZg4dOuSaHEePHjVXXXWVadq0qalTp45p27atGT9+fKWTETfkMMZUmUOSefHFF4PbRPJv6quvvjKDBw82mZmZpkmTJmbatGnm5MmTrsqyY8cO069fP5Obm2syMjJMx44dzfTp00PuU+6GLD/60Y9M27ZtTXp6umnatKnp379/sGAYkzy/Dy9IthpBfXDH+6pXagT1wX31wZjUqRE+Y4yx/nMQAAAAAKkkqa6xAAAAAOBONBYAAAAA4kZjAQAAACBuNBYAAAAA4kZjAQAAACBuNBYAAAAA4kZjAQAAACBuNBYAAAAA4kZjAQAAACBuNBYAAAAA4kZjAQAAACBuNBYAAAAA4vb//pVI1f77s5YAAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 800x400 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "==================================\n",
      "Prompt for reasoning:\n",
      "\n",
      "The following JSON contains an approximated perception of a 2d scene. Each object (potentially including the background) is represented by a list of geometric shapes. If the object contain multiple shapes, it is a composite object. The (x, y) coordinates are in pixels, and (0, 0) is the top-left corner of the image.\n",
      "\n",
      "--- perception ---\n",
      "{'object_0': [{'type': 'rectangle', 'vertices': [[0.0, 336.0], [0.0, 0.0], [336.0, 0.0], [336.0, 336.0]], 'color': [254, 254, 254], 'style': 'filled shape'}], 'object_1': [{'type': 'line_segment', 'vertices': [[202, 179], [289, 179]], 'color': [3, 3, 3], 'line_width': 5}], 'object_2': [{'type': 'line_segment', 'vertices': [[60, 167], [60, 129]], 'color': [6, 6, 6], 'line_width': 4}]}\n",
      "------\n",
      "\n",
      "Note that perception can be noisy. Make educated guesses if necessary. \n",
      "Think step by step and answer the following question:\n",
      "How many line segments are there in the image? What's the total length of all the line segments in the image?\n",
      "\n"
     ]
    }
   ],
   "source": [
    "# === SVG-to-PVD generation ===\n",
    "inputs = []\n",
    "responses = []\n",
    "\n",
    "# generate PVD json for each SVG path\n",
    "for i, svg_str in enumerate(svg_strs):\n",
    "    \n",
    "    prompt = get_prompt_general_from_svg_str(svg_str, \"Describe the visual content of the image in a JSON format.\")\n",
    "    system_prompt = SYSTEM_MESSAGES[\"svg_expert\"]\n",
    "    \n",
    "    messages = [\n",
    "        {\n",
    "            \"role\": \"system\",\n",
    "            \"content\": system_prompt\n",
    "        },\n",
    "        {\n",
    "            \"role\": \"user\",\n",
    "            \"content\": prompt\n",
    "        }\n",
    "    ]\n",
    "\n",
    "    query_obj = {\n",
    "        \"model\": model_id,\n",
    "        \"messages\": messages,\n",
    "        **generation_configs\n",
    "    }\n",
    "    chat_response = client.chat.completions.create(**query_obj)\n",
    "    inputs.append(messages)\n",
    "    responses.append(chat_response.choices[0].message.content)\n",
    "\n",
    "# save perception results\n",
    "os.makedirs(output_perception_dir, exist_ok=True)\n",
    "resized_img_size_for_vis = Image.open(img_path).size # do not resize, use the original image size\n",
    "response_dict = {}\n",
    "for i, response in enumerate(responses):\n",
    "    svg_path = svg_paths[i]\n",
    "    svg_basename = os.path.basename(svg_path)\n",
    "    vis_img = visualize_pvd_shape_prediction(response, resized_img_size_for_vis, alpha=1)\n",
    "    vis_img.save(f\"{output_perception_dir}/pred_{svg_basename}\".replace(\".svg\", \".png\"))\n",
    "    response_dict[svg_basename] = {\n",
    "        \"input\": inputs[i],\n",
    "        \"response\": response\n",
    "    }\n",
    "\n",
    "with open(f\"{output_perception_dir}/responses.json\", \"w\") as f:\n",
    "    json.dump(response_dict, f, indent=4)\n",
    "\n",
    "# save visualization of the aggregated PVD perception by rendering the shapes\n",
    "if len(responses) > 1:\n",
    "    img = Image.new(\"RGBA\", resized_img_size_for_vis, (255, 255, 255, 0))\n",
    "    draw = ImageDraw.Draw(img)\n",
    "    for response in responses:\n",
    "        img = visualize_pvd_shape_prediction(response, resized_img_size_for_vis, img, draw, alpha=1)\n",
    "    img.save(f\"{output_perception_dir}/pred_all.png\")\n",
    "\n",
    "# === construct the prompt for reasoning with the aggregated perception results and the task query ===\n",
    "perception_result = get_perception_from_pvd_responses(response_dict)\n",
    "propmt_for_reasoning = DEFAULT_REASONING_PROMPT.format(perception=perception_result, question=question)\n",
    "with open(f\"{output_dir}/prompt_for_reasoning.txt\", \"w\") as f:\n",
    "    f.write(propmt_for_reasoning)\n",
    "\n",
    "\n",
    "# show PVD results\n",
    "print(\"Aggregated PVD perception results:\\n\")\n",
    "print(perception_result)\n",
    "\n",
    "# show rendered PVD reconstruction of the original image\n",
    "import matplotlib.pyplot as plt\n",
    "fig, axs = plt.subplots(1, 2, figsize=(8, 4))\n",
    "axs[0].imshow(plt.imread(img_path))\n",
    "axs[0].set_title(\"Original Image\")\n",
    "\n",
    "axs[1].imshow(plt.imread(f\"{output_perception_dir}/pred_all.png\"))\n",
    "axs[1].set_title(\"PVD Reconstruction\")\n",
    "\n",
    "plt.tight_layout()\n",
    "plt.show()\n",
    "\n",
    "# show prompt for reasoning\n",
    "print(\"==================================\")\n",
    "print(\"Prompt for reasoning:\\n\")\n",
    "print(propmt_for_reasoning)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Step 3: PVD + Question -> Answer"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# The following example shows using GPT-4 to reason about the prompt obtained from last step \n",
    "# that contains the PVD perception results and the task question.\n",
    "\n",
    "from gpt4_v_inference import *\n",
    "\n",
    "# TODO: set your OpenAI API key here\n",
    "os.environ[\"OPENAI_API_KEY\"] = \"your openai api key here\"\n",
    "\n",
    "\n",
    "### VDLM-txt example: reasoning with text-only LLM ###\n",
    "openai_config = {\n",
    "    \"api_type\": \"chat_completion\", \n",
    "    \"model\": \"gpt-4-turbo-preview\",\n",
    "    \"temperature\": 0.0,\n",
    "    \"max_tokens\": 4096,\n",
    "    \"top_p\": 1.0\n",
    "}\n",
    "\n",
    "openai_model = OpenAIWrapper(openai_config)\n",
    "ret, raw_responses = openai_model.run(prompt=propmt_for_reasoning, images=[])\n",
    "\n",
    "print(\"==================================\")\n",
    "print(\"GPT-4 response:\\n\")\n",
    "print(ret['text_outputs'][0]['content'])\n",
    "\n",
    "\n",
    "### VDLM-mm example: reasoning with multimodal LLM ###\n",
    "openai_config = {\n",
    "    \"api_type\": \"chat_completion\", \n",
    "    \"model\": \"gpt-4o-2024-05-13\",\n",
    "    \"temperature\": 0.0,\n",
    "    \"max_tokens\": 4096,\n",
    "    \"top_p\": 1.0\n",
    "}\n",
    "openai_model = OpenAIWrapper(openai_config)\n",
    "ret, raw_responses = openai_model.run(prompt=propmt_for_reasoning, images=[img_path])\n"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "vllm",
   "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
}
