{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "b202c570",
   "metadata": {},
   "source": [
    "# Preview MultiID-2M"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "45c328fe",
   "metadata": {},
   "source": [
    "### Image Downloader\n",
    "\n",
    "You probably need to set the proxies in the following function if needed."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "5beeb7ee",
   "metadata": {},
   "outputs": [],
   "source": [
    "# -*- coding: utf-8 -*-\n",
    "# install requests tqdm\n",
    "%pip install requests tqdm pillow\n",
    "import requests\n",
    "import os\n",
    "from tqdm import tqdm\n",
    "from urllib.parse import urlparse\n",
    "import concurrent.futures\n",
    "\n",
    "def download_image(url, save_dir, save_name=None):\n",
    "    \"\"\"Download an image from URL and save it to directory\"\"\"\n",
    "    print(f\"Downloading {url} to {save_dir}\")\n",
    "\n",
    "    # Set up proxies if your network requires it\n",
    "    proxies = {\n",
    "    }\n",
    "\n",
    "    try:\n",
    "        response = requests.get(url, timeout=10, proxies=proxies)\n",
    "        response.raise_for_status()\n",
    "\n",
    "        # Extract filename from URL or generate one\n",
    "        parsed_url = urlparse(url)\n",
    "        if save_name:\n",
    "            filename = save_name\n",
    "        else:\n",
    "            # Use the last part of the URL path as the filename\n",
    "            # or generate a unique name if not available\n",
    "            filename = os.path.basename(parsed_url.path)\n",
    "            if not filename or '.' not in filename:\n",
    "                filename = f\"image_{hash(url)}.jpg\"\n",
    "        if not filename or '.' not in filename:\n",
    "            filename = f\"image_{hash(url)}.jpg\"\n",
    "\n",
    "        # Save the image\n",
    "        filepath = os.path.join(save_dir, filename)\n",
    "        with open(filepath, 'wb') as f:\n",
    "            f.write(response.content)\n",
    "        return True\n",
    "    except Exception as e:\n",
    "        # Optionally log the specific error\n",
    "        print(f\"Failed to download {url}: {str(e)}\")\n",
    "        return False"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "d66e34e7",
   "metadata": {},
   "source": [
    "### The CSV Reader"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 108,
   "id": "6bb2115f",
   "metadata": {},
   "outputs": [],
   "source": [
    "import csv\n",
    "import random\n",
    "import os\n",
    "import mmap\n",
    "\n",
    "class LargeCSVReader:\n",
    "    def __init__(self, csv_path):\n",
    "        \"\"\"\n",
    "        Initialize with path to a CSV file\n",
    "        \n",
    "        Args:\n",
    "            csv_path (str): Path to the CSV file\n",
    "        \"\"\"\n",
    "        if not os.path.exists(csv_path):\n",
    "            raise FileNotFoundError(f\"CSV file not found: {csv_path}\")\n",
    "            \n",
    "        self.csv_path = csv_path\n",
    "        self.line_offsets = self._index_file()\n",
    "        self.num_rows = len(self.line_offsets) - 1  # Last offset is EOF\n",
    "        \n",
    "        # Read the header\n",
    "        if self.num_rows > 0:\n",
    "            with open(self.csv_path, 'r', newline='') as f:\n",
    "                csv_reader = csv.reader(f)\n",
    "                try:\n",
    "                    self.header = next(csv_reader)\n",
    "                except StopIteration:\n",
    "                    self.header = []\n",
    "        else:\n",
    "            self.header = []\n",
    "    \n",
    "    def _index_file(self):\n",
    "        \"\"\"\n",
    "        Index the file to find the byte offset of each row\n",
    "        \n",
    "        Returns:\n",
    "            list: List of byte offsets for the start of each line\n",
    "        \"\"\"\n",
    "        offsets = [0]  # First line starts at position 0\n",
    "        \n",
    "        with open(self.csv_path, 'rb') as f:\n",
    "            # Use memory mapping for efficient scanning\n",
    "            mm = mmap.mmap(f.fileno(), 0, access=mmap.ACCESS_READ)\n",
    "            \n",
    "            current_pos = 0\n",
    "            while current_pos < mm.size():\n",
    "                newline_pos = mm.find(b'\\n', current_pos)\n",
    "                \n",
    "                if newline_pos == -1:\n",
    "                    # No more newlines, add EOF position\n",
    "                    offsets.append(mm.size())\n",
    "                    break\n",
    "                    \n",
    "                # Add the position after the newline\n",
    "                next_pos = newline_pos + 1\n",
    "                offsets.append(next_pos)\n",
    "                \n",
    "                # Move to the position after the newline\n",
    "                current_pos = next_pos\n",
    "                \n",
    "            mm.close()\n",
    "            \n",
    "        return offsets\n",
    "    \n",
    "    def get_random_row(self, include_header=False):\n",
    "        \"\"\"\n",
    "        Return a random row from the CSV file\n",
    "        \n",
    "        Args:\n",
    "            include_header (bool): Whether to include the header row in selection\n",
    "                \n",
    "        Returns:\n",
    "            list: A random row from the CSV file or None if file is empty\n",
    "        \"\"\"\n",
    "            \n",
    "        random_idx = random.randint(1, 200000)\n",
    "        return self.get_row(random_idx)\n",
    "    \n",
    "    def get_row(self, index):\n",
    "        \"\"\"\n",
    "        Get a specific row by index\n",
    "        \n",
    "        Args:\n",
    "            index (int): Row index (0 is header if present)\n",
    "                \n",
    "        Returns:\n",
    "            list: The requested row or None if index is out of bounds\n",
    "        \"\"\"\n",
    "        if index < 0 or index >= self.num_rows:\n",
    "            return None\n",
    "            \n",
    "        # Get the byte offsets for the selected row\n",
    "        start_pos = self.line_offsets[index]\n",
    "        end_pos = self.line_offsets[index + 1] if index + 1 < len(self.line_offsets) else None\n",
    "        \n",
    "        # Read just that row\n",
    "        with open(self.csv_path, 'r', newline='') as f:\n",
    "            f.seek(start_pos)\n",
    "            if end_pos is not None:\n",
    "                line = f.read(end_pos - start_pos)\n",
    "            else:\n",
    "                line = f.read()\n",
    "                \n",
    "            # Remove trailing newline if present\n",
    "            if line and line[-1] == '\\n':\n",
    "                line = line[:-1]\n",
    "            if line and line[-1] == '\\r':\n",
    "                line = line[:-1]\n",
    "                \n",
    "            return next(csv.reader([line]))\n",
    "    \n",
    "    def __len__(self):\n",
    "        \"\"\"Return the number of rows in the file (including header)\"\"\"\n",
    "        return self.num_rows"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "fa077e27",
   "metadata": {},
   "source": [
    "### Load the CSV"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "95ef1d1e",
   "metadata": {},
   "outputs": [],
   "source": [
    "from IPython.display import Image as IPyImage, display\n",
    "import ast\n",
    "from PIL import Image\n",
    "print(os.path.abspath('.'))\n",
    "csv_path = '.\\\\100ksubset.csv' # if you are using windows\n",
    "# csv_path = './100ksubset.csv' # if you are using linux\n",
    "csv_reader = LargeCSVReader(csv_path)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 89,
   "id": "79a6bed1",
   "metadata": {},
   "outputs": [],
   "source": [
    "def image_crop(img, box):\n",
    "    return img.crop(box)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "e0513078",
   "metadata": {},
   "source": [
    "### Random image display\n",
    "\n",
    "Run this code block to display a random image from the CSV file, along with its corresponding captions and metadata.\n",
    "\n",
    "The identities in the images are anonymized. Each identity is mapped to a unique number.\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 237,
   "id": "0f6fb1bd",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "['id', 'aesthetic_score', 'ram_result_advertisement_logit', 'bboxes', 'ram_result_screenshot_logit', 'ram_result_poster_logit', 'ram_result_print_logit', 'text_box', 'ram_result_portrait_logit', 'caption_en', 'ram_result_photography_logit', 'name', 'ram_result_collage_logit', 'crop', 'url', 'ram_result_cartoon_logit']\n",
      "id: 459677312f8d390bc422343706f47165dc354e31\n",
      "aesthetic_score: 4.09765625\n",
      "bboxes: [[142, 89, 243, 228], [316, 30, 435, 170]]\n",
      "text_box: []\n",
      "caption_en:  two people in the water at night. The person on the left is wearing a white shirt, while the one on the right is in a dark shirt. Both appear to be clinging to something, with the person on the right leaning towards the other. They seem to be in a tense or desperate situation.\n",
      "name: '0924','0817'\n",
      "crop: [0, 0, 545, 307]\n",
      "url: https://c.ndtvimg.com/2023-02/0kcec388_titanic-_625x300_09_February_23.jpg?downsize=545:307\n",
      "Downloading https://c.ndtvimg.com/2023-02/0kcec388_titanic-_625x300_09_February_23.jpg?downsize=545:307 to .\\\n",
      "Image downloaded successfully.\n"
     ]
    },
    {
     "data": {
      "image/jpeg": "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",
      "text/plain": [
       "<IPython.core.display.Image object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "row = csv_reader.get_random_row()\n",
    "print(csv_reader.header)\n",
    "url = None\n",
    "crop = None\n",
    "for i, col in enumerate(csv_reader.header):\n",
    "    if \"ram\" in col.lower():\n",
    "        continue\n",
    "    if \"url\" == col.lower():\n",
    "        url = row[i]\n",
    "    if \"crop\" == col.lower():\n",
    "        crop = row[i]\n",
    "    print(f\"{col}: {row[i]}\")\n",
    "\n",
    "download_success = download_image(url,\".\\\\\", 'tmp.jpg')\n",
    "if download_success:\n",
    "    print(\"Image downloaded successfully.\")\n",
    "    image = Image.open('tmp.jpg')\n",
    "    # The following \"if crop\" would crop out the text area in the image. Comment it out if you want to see the whole image.\n",
    "    if crop:\n",
    "        # If crop coordinates are provided, crop the image\n",
    "        if isinstance(crop, str):\n",
    "            crop = ast.literal_eval(crop)\n",
    "        image = image_crop(image, crop)\n",
    "        # Save the cropped image\n",
    "        image.save('tmp.jpg')\n",
    "    display(IPyImage(filename='tmp.jpg'))\n",
    "else:\n",
    "    print(\"Failed to download image. Click the url to check it on the browser.\")"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "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.13.3"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
